api06: simplify diff uploading.
[rails.git] / app / controllers / changeset_controller.rb
1 # The ChangesetController is the RESTful interface to Changeset objects
2
3 class ChangesetController < ApplicationController
4   require 'xml/libxml'
5
6   before_filter :authorize, :only => [:create, :update, :delete, :upload]
7   before_filter :check_write_availability, :only => [:create, :update, :delete, :upload]
8   before_filter :check_read_availability, :except => [:create, :update, :delete, :upload]
9   after_filter :compress_output
10
11   # Create a changeset from XML.
12   def create
13     if request.put?
14       cs = Changeset.from_xml(request.raw_post, true)
15
16       if cs
17         cs.user_id = @user.id
18         cs.save_with_tags!
19         render :text => cs.id.to_s, :content_type => "text/plain"
20       else
21         render :nothing => true, :status => :bad_request
22       end
23     else
24       render :nothing => true, :status => :method_not_allowed
25     end
26   end
27
28   def create_prim(ids, prim, nd)
29     prim.version = 0
30     prim.user_id = @user.id
31     prim.visible = true
32     prim.save_with_history!
33
34     ids[nd['id'].to_i] = prim.id
35   end
36
37   def fix_way(w, node_ids)
38     w.nds = w.instance_eval { @nds }.
39       map { |nd| node_ids[nd] || nd }
40     return w
41   end
42
43   def fix_rel(r, ids)
44     r.members = r.instance_eval { @members }.
45       map { |memb| [memb[0], ids[memb[0]][memb[1].to_i] || memb[1], memb[2]] }
46     return r
47   end
48   
49   def read
50     begin
51       changeset = Changeset.find(params[:id])
52       render :text => changeset.to_xml.to_s, :content_type => "text/xml"
53     rescue ActiveRecord::RecordNotFound
54       render :nothing => true, :status => :not_found
55     end
56   end
57   
58   def close 
59     begin
60       if not request.put?
61         render :nothing => true, :status => :method_not_allowed
62         return
63       end
64       changeset = Changeset.find(params[:id])
65       changeset.open = false
66       changeset.save
67       render :nothing => true
68     rescue ActiveRecord::RecordNotFound
69       render :nothing => true, :status => :not_found
70     end
71   end
72
73   def upload
74     if not request.put?
75       render :nothing => true, :status => :method_not_allowed
76       return
77     end
78
79     p = XML::Reader.new request.raw_post
80
81     node_ids, way_ids, rel_ids = {}, {}, {}
82     ids = {"node"=>node_ids, "way"=>way_ids, "relation"=>rel_ids}
83
84     models = {"node"=>Node, "way"=>Way, "relation"=>Relation}
85
86     res = XML::Document.new
87     res.encoding = 'UTF-8'
88     root = XML::Node.new 'osm'
89     root['version'] = '0.6'
90     root['creator'] = 'OpenStreetMap.org'
91     res.root = root
92
93     root << XML::Node.new_comment(" Warning: this is a 0.6 result document, " +
94       "not a normal OSM file. ")
95
96     Changeset.transaction do
97       while p.read == 1
98         puts p.name
99         break if p.node_type == 15 # end element
100         next unless p.node_type == 1 # element
101
102         case p.name
103         when 'create':
104           while p.read == 1
105             puts "#{p.name} #{p.node_type}"
106             break if p.node_type == 15 # end element
107             next unless p.node_type == 1 # element
108
109             model = models[p.name]
110             next if model.nil?
111
112             elem = XML::Node.new p.name
113             nd = p.expand; p.next
114             osm = model.from_xml_node(nd, true)
115             elem['old_id'] = nd['id']
116
117             case nd.name
118             when 'way':
119               fix_way(osm, node_ids)
120               raise OSM::APIPreconditionFailedError.new if !osm.preconditions_ok?
121             when 'relation':
122               fix_rel(osm, ids)
123               raise OSM::APIPreconditionFailedError.new if !osm.preconditions_ok?
124             end
125
126             create_prim ids[nd.name], osm, nd
127             elem['new_id'] = osm.id.to_s
128             elem['new_version'] = osm.version.to_s
129             root << elem
130           end
131         when 'modify':
132           while p.read == 1
133             break if p.node_type == 15 # end element
134             next unless p.node_type == 1 # element
135
136             model = models[p.name]
137             next if model.nil?
138
139             elem = XML::Node.new p.name
140             new_osm = model.from_xml_node(p.expand); p.next
141             osm = model.find(new_osm.id)
142             osm.update_from new_osm, @user
143             elem['old_id'] = elem['new_id'] = osm.id.to_s
144             elem['new_version'] = osm.version.to_s
145             root << elem
146           end
147         when 'delete':
148           while p.read == 1
149             break if p.node_type == 15 # end element
150             next unless p.node_type == 1 # element
151
152             model = models[p.name]
153             next if model.nil?
154
155             elem = XML::Node.new p.name
156             osm = model.find(p.expand['id']); p.next
157             osm.delete_with_history(@user)
158             elem['old_id'] = elem['new_id'] = osm.id.to_s
159             elem['new_version'] = osm.version.to_s
160             root << elem
161           end
162         end
163       end
164     end
165
166     render :text => res.to_s, :content_type => "text/xml"
167
168   rescue OSM::APIError => ex
169     render ex.render_opts
170   end
171 end