]> git.openstreetmap.org Git - rails.git/blob - app/controllers/changesets_controller.rb
Merge branch 'pull/5206'
[rails.git] / app / controllers / changesets_controller.rb
1 # The ChangesetController is the RESTful interface to Changeset objects
2
3 class ChangesetsController < ApplicationController
4   include UserMethods
5
6   layout "site"
7
8   before_action :authorize_web
9   before_action :set_locale
10   before_action -> { check_database_readable(:need_api => true) }, :only => [:index, :feed, :show]
11   before_action :require_oauth, :only => :show
12   before_action :check_database_writable, :only => [:subscribe, :unsubscribe]
13
14   authorize_resource
15
16   around_action :web_timeout
17
18   ##
19   # list non-empty changesets in reverse chronological order
20   def index
21     param! :max_id, Integer, :min => 1
22
23     @params = params.permit(:display_name, :bbox, :friends, :nearby, :max_id, :list)
24
25     if request.format == :atom && @params[:max_id]
26       redirect_to url_for(@params.merge(:max_id => nil)), :status => :moved_permanently
27       return
28     end
29
30     if @params[:display_name]
31       user = User.find_by(:display_name => @params[:display_name])
32       if !user || !user.active?
33         render_unknown_user @params[:display_name]
34         return
35       end
36     end
37
38     if (@params[:friends] || @params[:nearby]) && !current_user
39       require_user
40       return
41     end
42
43     if request.format == :html && !@params[:list]
44       require_oauth
45       render :action => :history, :layout => map_layout
46     else
47       changesets = conditions_nonempty(Changeset.all)
48
49       if @params[:display_name]
50         changesets = if user.data_public? || user == current_user
51                        changesets.where(:user => user)
52                      else
53                        changesets.where("false")
54                      end
55       elsif @params[:bbox]
56         changesets = conditions_bbox(changesets, BoundingBox.from_bbox_params(params))
57       elsif @params[:friends] && current_user
58         changesets = changesets.where(:user => current_user.friends.identifiable)
59       elsif @params[:nearby] && current_user
60         changesets = changesets.where(:user => current_user.nearby)
61       end
62
63       changesets = changesets.where(:changesets => { :id => ..@params[:max_id] }) if @params[:max_id]
64
65       @changesets = changesets.order("changesets.id DESC").limit(20).preload(:user, :changeset_tags, :comments)
66
67       render :action => :index, :layout => false
68     end
69   end
70
71   ##
72   # list edits as an atom feed
73   def feed
74     index
75   end
76
77   def show
78     @type = "changeset"
79     @changeset = Changeset.find(params[:id])
80     case turbo_frame_request_id
81     when "changeset_nodes"
82       @node_pages, @nodes = paginate(:old_nodes, :conditions => { :changeset_id => @changeset.id }, :per_page => 20, :parameter => "node_page")
83       render :partial => "elements", :locals => { :type => "node", :elements => @nodes, :pages => @node_pages }
84     when "changeset_ways"
85       @way_pages, @ways = paginate(:old_ways, :conditions => { :changeset_id => @changeset.id }, :per_page => 20, :parameter => "way_page")
86       render :partial => "elements", :locals => { :type => "way", :elements => @ways, :pages => @way_pages }
87     when "changeset_relations"
88       @relation_pages, @relations = paginate(:old_relations, :conditions => { :changeset_id => @changeset.id }, :per_page => 20, :parameter => "relation_page")
89       render :partial => "elements", :locals => { :type => "relation", :elements => @relations, :pages => @relation_pages }
90     else
91       @comments = if current_user&.moderator?
92                     @changeset.comments.unscope(:where => :visible).includes(:author)
93                   else
94                     @changeset.comments.includes(:author)
95                   end
96       @node_pages, @nodes = paginate(:old_nodes, :conditions => { :changeset_id => @changeset.id }, :per_page => 20, :parameter => "node_page")
97       @way_pages, @ways = paginate(:old_ways, :conditions => { :changeset_id => @changeset.id }, :per_page => 20, :parameter => "way_page")
98       @relation_pages, @relations = paginate(:old_relations, :conditions => { :changeset_id => @changeset.id }, :per_page => 20, :parameter => "relation_page")
99       if @changeset.user.active? && @changeset.user.data_public?
100         changesets = conditions_nonempty(@changeset.user.changesets)
101         @next_by_user = changesets.where("id > ?", @changeset.id).reorder(:id => :asc).first
102         @prev_by_user = changesets.where(:id => ...@changeset.id).reorder(:id => :desc).first
103       end
104       render :layout => map_layout
105     end
106   rescue ActiveRecord::RecordNotFound
107     render :template => "browse/not_found", :status => :not_found, :layout => map_layout
108   end
109
110   ##
111   # subscribe to a changeset
112   def subscribe
113     @changeset = Changeset.find(params[:id])
114
115     if request.post?
116       @changeset.subscribe(current_user) unless @changeset.subscribed?(current_user)
117
118       redirect_to changeset_path(@changeset)
119     end
120   rescue ActiveRecord::RecordNotFound
121     render :action => "no_such_entry", :status => :not_found
122   end
123
124   ##
125   # unsubscribe from a changeset
126   def unsubscribe
127     @changeset = Changeset.find(params[:id])
128
129     if request.post?
130       @changeset.unsubscribe(current_user)
131
132       redirect_to changeset_path(@changeset)
133     end
134   rescue ActiveRecord::RecordNotFound
135     render :action => "no_such_entry", :status => :not_found
136   end
137
138   private
139
140   #------------------------------------------------------------
141   # utility functions below.
142   #------------------------------------------------------------
143
144   ##
145   # if a bounding box was specified do some sanity checks.
146   # restrict changesets to those enclosed by a bounding box
147   def conditions_bbox(changesets, bbox)
148     if bbox
149       bbox.check_boundaries
150       bbox = bbox.to_scaled
151
152       changesets.where("min_lon < ? and max_lon > ? and min_lat < ? and max_lat > ?",
153                        bbox.max_lon.to_i, bbox.min_lon.to_i,
154                        bbox.max_lat.to_i, bbox.min_lat.to_i)
155     else
156       changesets
157     end
158   end
159
160   ##
161   # eliminate empty changesets (where the bbox has not been set)
162   # this should be applied to all changeset list displays
163   def conditions_nonempty(changesets)
164     changesets.where("num_changes > 0")
165   end
166 end