]> git.openstreetmap.org Git - rails.git/blob - app/controllers/api/traces_controller.rb
Merge remote-tracking branch 'upstream/pull/4063'
[rails.git] / app / controllers / api / traces_controller.rb
1 module Api
2   class TracesController < ApiController
3     before_action :check_database_readable, :except => [:show, :data]
4     before_action :check_database_writable, :only => [:create, :update, :destroy]
5     before_action :authorize_web
6     before_action :set_locale
7     before_action :authorize
8
9     authorize_resource
10
11     before_action :check_api_readable, :only => [:show, :data]
12     before_action :check_api_writable, :only => [:create, :update, :destroy]
13     before_action :offline_error, :only => [:create, :destroy, :data]
14     around_action :api_call_handle_error
15
16     def show
17       @trace = Trace.visible.find(params[:id])
18
19       head :forbidden unless @trace.public? || @trace.user == current_user
20     end
21
22     def create
23       tags = params[:tags] || ""
24       description = params[:description] || ""
25       visibility = params[:visibility]
26
27       if visibility.nil?
28         visibility = if params[:public]&.to_i&.nonzero?
29                        "public"
30                      else
31                        "private"
32                      end
33       end
34
35       if params[:file].respond_to?(:read)
36         trace = do_create(params[:file], tags, description, visibility)
37
38         if trace.id
39           TraceImporterJob.perform_later(trace)
40           render :plain => trace.id.to_s
41         elsif trace.valid?
42           head :internal_server_error
43         else
44           head :bad_request
45         end
46       else
47         head :bad_request
48       end
49     end
50
51     def update
52       trace = Trace.visible.find(params[:id])
53
54       if trace.user == current_user
55         trace.update_from_xml(request.raw_post)
56         trace.save!
57
58         head :ok
59       else
60         head :forbidden
61       end
62     end
63
64     def destroy
65       trace = Trace.visible.find(params[:id])
66
67       if trace.user == current_user
68         trace.visible = false
69         trace.save!
70         TraceDestroyerJob.perform_later(trace)
71
72         head :ok
73       else
74         head :forbidden
75       end
76     end
77
78     def data
79       trace = Trace.visible.find(params[:id])
80
81       if trace.public? || trace.user == current_user
82         if request.format == Mime[:xml]
83           send_data(trace.xml_file.read, :filename => "#{trace.id}.xml", :type => request.format.to_s, :disposition => "attachment")
84         elsif request.format == Mime[:gpx]
85           send_data(trace.xml_file.read, :filename => "#{trace.id}.gpx", :type => request.format.to_s, :disposition => "attachment")
86         elsif trace.file.attached?
87           redirect_to rails_blob_path(trace.file, :disposition => "attachment")
88         else
89           send_file(trace.trace_name, :filename => "#{trace.id}#{trace.extension_name}", :type => trace.mime_type, :disposition => "attachment")
90         end
91       else
92         head :forbidden
93       end
94     end
95
96     private
97
98     def do_create(file, tags, description, visibility)
99       # Sanitise the user's filename
100       name = file.original_filename.gsub(/[^a-zA-Z0-9.]/, "_")
101
102       # Create the trace object, falsely marked as already
103       # inserted to stop the import daemon trying to load it
104       trace = Trace.new(
105         :name => name,
106         :tagstring => tags,
107         :description => description,
108         :visibility => visibility,
109         :inserted => false,
110         :user => current_user,
111         :timestamp => Time.now.utc,
112         :file => file
113       )
114
115       # Save the trace object
116       trace.save!
117
118       # Finally save the user's preferred privacy level
119       if pref = current_user.preferences.where(:k => "gps.trace.visibility").first
120         pref.v = visibility
121         pref.save
122       else
123         current_user.preferences.create(:k => "gps.trace.visibility", :v => visibility)
124       end
125
126       trace
127     end
128
129     def offline_error
130       report_error "GPX files offline for maintenance", :service_unavailable if Settings.status == "gpx_offline"
131     end
132   end
133 end