some changes suggested by SteveC / yet untested
[rails.git] / app / controllers / relation_controller.rb
1 class RelationController < ApplicationController
2   require 'xml/libxml'
3
4   before_filter :authorize, :only => [:create, :update, :delete]
5   before_filter :check_availability, :only => [:create, :update, :delete]
6
7   after_filter :compress_output
8
9   def create
10     if request.put?
11       relation = Relation.from_xml(request.raw_post, true)
12
13       if relation
14         if !relation.preconditions_ok?
15           render :nothing => true, :status => :precondition_failed
16         else
17           relation.user_id = @user.id
18
19           if relation.save_with_history
20             render :text => relation.id.to_s, :content_type => "text/plain"
21           else
22             render :text => "save error", :status => :internal_server_error
23           end
24         end
25       else
26         render :nothing => true, :status => :bad_request
27       end
28     else
29       render :nothing => true, :status => :method_not_allowed
30     end
31   end
32
33   def read
34     begin
35       relation = Relation.find(params[:id])
36
37       if relation.visible
38         render :text => relation.to_xml.to_s, :content_type => "text/xml"
39       else
40         render :nothing => true, :status => :gone
41       end
42     rescue ActiveRecord::RecordNotFound
43       render :nothing => true, :status => :not_found
44     rescue
45       render :nothing => true, :status => :internal_server_error
46     end
47   end
48
49   def update
50     begin
51       relation = Relation.find(params[:id])
52
53       if relation.visible
54         new_relation = Relation.from_xml(request.raw_post)
55
56         if new_relation and new_relation.id == relation.id
57           if !new_relation.preconditions_ok?
58             render :nothing => true, :status => :precondition_failed
59           else
60             relation.user_id = @user.id
61             relation.tags = new_relation.tags
62             relation.members = new_relation.members
63             relation.visible = true
64
65             if relation.save_with_history
66               render :nothing => true
67             else
68               render :nothing => true, :status => :internal_server_error
69             end
70           end
71         else
72           render :nothing => true, :status => :bad_request
73         end
74       else
75         render :nothing => true, :status => :gone
76       end
77     rescue ActiveRecord::RecordNotFound
78       render :nothing => true, :status => :not_found
79     rescue
80       render :nothing => true, :status => :internal_server_error
81     end
82   end
83
84   def delete
85 #XXX check if member somewhere!
86     begin
87       relation = Relation.find(params[:id])
88
89       if relation.visible
90         if RelationMember.find(:first, :joins => "INNER JOIN current_relations ON current_relations.id=current_relation_members.id", :conditions => [ "visible = 1 AND member_type='relation' and member_id=?", params[:id]])
91           render :nothing => true, :status => :precondition_failed
92         else
93           relation.user_id = @user.id
94           relation.tags = []
95           relation.members = []
96           relation.visible = false
97
98           if relation.save_with_history
99             render :nothing => true
100           else
101             render :nothing => true, :status => :internal_server_error
102           end
103         end
104       else
105         render :nothing => true, :status => :gone
106       end
107     rescue ActiveRecord::RecordNotFound
108       render :nothing => true, :status => :not_found
109     rescue
110       render :nothing => true, :status => :internal_server_error
111     end
112   end
113
114   def full
115     begin
116       relation = Relation.find(params[:id])
117
118       if relation.visible
119         # In future, we might want to do all the data fetch in one step
120         seg_ids = relation.segs + [-1]
121         segments = Segment.find_by_sql "select * from current_segments where visible = 1 and id IN (#{seg_ids.join(',')})"
122
123         node_ids = segments.collect {|segment| segment.node_a }
124         node_ids += segments.collect {|segment| segment.node_b }
125         node_ids += [-1]
126         nodes = Node.find(node_ids, :conditions => "visible = TRUE")
127
128         # Render
129         doc = OSM::API.new.get_xml_doc
130         nodes.each do |node|
131           doc.root << node.to_xml_node()
132         end
133         segments.each do |segment|
134           doc.root << segment.to_xml_node()
135         end
136         doc.root << relation.to_xml_node()
137
138         render :text => doc.to_s, :content_type => "text/xml"
139       else
140         render :nothing => true, :status => :gone
141       end
142     rescue ActiveRecord::RecordNotFound
143       render :nothing => true, :status => :not_found
144     rescue
145       render :nothing => true, :status => :internal_server_error
146     end
147   end
148
149   def relations
150     ids = params['relations'].split(',').collect { |w| w.to_i }
151
152     if ids.length > 0
153       doc = OSM::API.new.get_xml_doc
154
155       Relation.find(ids).each do |relation|
156         doc.root << relation.to_xml_node
157       end
158
159       render :text => doc.to_s, :content_type => "text/xml"
160     else
161       render :nothing => true, :status => :bad_request
162     end
163   end
164
165   def relations_for_object(objtype)
166     relationids = RelationMember.find(:all, :conditions => ['member_type=? and member_id=?', objtype, params[:id]]).collect { |ws| ws.id }.uniq
167
168     if relationids.length > 0
169       doc = OSM::API.new.get_xml_doc
170
171       Relation.find(relationids).each do |relation|
172         doc.root << relation.to_xml_node
173       end
174
175       render :text => doc.to_s, :content_type => "text/xml"
176     else
177       render :nothing => true, :status => :bad_request
178     end
179   end
180 end