X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/cdd94fae8d40ad6dcee3de5d36de2d89921fd12d..6790e24febd872dcf98b1356ad1d06fc69fc2915:/services/api/app/controllers/application_controller.rb diff --git a/services/api/app/controllers/application_controller.rb b/services/api/app/controllers/application_controller.rb index fe5598e0dc..4a1a7d46db 100644 --- a/services/api/app/controllers/application_controller.rb +++ b/services/api/app/controllers/application_controller.rb @@ -20,12 +20,11 @@ class ApplicationController < ActionController::Base include LoadParam include RecordFilters - ERROR_ACTIONS = [:render_error, :render_not_found] - - respond_to :json protect_from_forgery + ERROR_ACTIONS = [:render_error, :render_not_found] + before_filter :respond_with_json_by_default before_filter :remote_ip before_filter :load_read_auths @@ -46,6 +45,17 @@ class ApplicationController < ActionController::Base attr_accessor :resource_attrs + begin + rescue_from(Exception, + ArvadosModel::PermissionDeniedError, + :with => :render_error) + rescue_from(ActiveRecord::RecordNotFound, + ActionController::RoutingError, + ActionController::UnknownController, + AbstractController::ActionNotFound, + :with => :render_not_found) + end + def index @objects.uniq!(&:id) if @select.nil? or @select.include? "id" if params[:eager] and params[:eager] != '0' and params[:eager] != 0 and params[:eager] != '' @@ -85,21 +95,6 @@ class ApplicationController < ActionController::Base end end - begin - rescue_from Exception, - :with => :render_error - rescue_from ActiveRecord::RecordNotFound, - :with => :render_not_found - rescue_from ActionController::RoutingError, - :with => :render_not_found - rescue_from ActionController::UnknownController, - :with => :render_not_found - rescue_from AbstractController::ActionNotFound, - :with => :render_not_found - rescue_from ArvadosModel::PermissionDeniedError, - :with => :render_error - end - def render_404_if_no_object render_not_found "Object not found" if !@object end @@ -111,20 +106,34 @@ class ApplicationController < ActionController::Base end if @object and @object.errors and @object.errors.full_messages and not @object.errors.full_messages.empty? errors = @object.errors.full_messages + logger.error errors.inspect else errors = [e.inspect] end status = e.respond_to?(:http_status) ? e.http_status : 422 - render json: { errors: errors }, status: status + send_error(*errors, status: status) end def render_not_found(e=ActionController::RoutingError.new("Path not found")) logger.error e.inspect - render json: { errors: ["Path not found"] }, status: 404 + send_error("Path not found", status: 404) end protected + def send_error(*args) + if args.last.is_a? Hash + err = args.pop + else + err = {} + end + err[:errors] ||= args + err[:error_token] = [Time.now.utc.to_i, "%08x" % rand(16 ** 8)].join("+") + status = err.delete(:status) || 422 + logger.error "Error #{err[:error_token]}: #{status}" + render json: err, status: status + end + def find_objects_for_index @objects ||= model_class.readable_by(*@read_users) apply_where_limit_order_params @@ -247,12 +256,8 @@ class ApplicationController < ActionController::Base def require_login if not current_user respond_to do |format| - format.json { - render :json => { errors: ['Not logged in'] }.to_json, status: 401 - } - format.html { - redirect_to '/auth/joshid' - } + format.json { send_error("Not logged in", status: 401) } + format.html { redirect_to '/auth/joshid' } end false end @@ -260,14 +265,14 @@ class ApplicationController < ActionController::Base def admin_required unless current_user and current_user.is_admin - render :json => { errors: ['Forbidden'] }.to_json, status: 403 + send_error("Forbidden", status: 403) end end def require_auth_scope if @read_auths.empty? if require_login != false - render :json => { errors: ['Forbidden'] }.to_json, status: 403 + send_error("Forbidden", status: 403) end false end @@ -347,6 +352,14 @@ class ApplicationController < ActionController::Base accept_param_as_json :reader_tokens, Array def render_list + if @select + # This information helps clients understand what they're seeing + # (Workbench always expects it), but they can't select it explicitly + # because it's not an SQL column. Always add it. + # I believe this is safe because clients can always deduce what they're + # looking at by the returned UUID anyway. + @select |= ["kind"] + end @object_list = { :kind => "arvados##{(@response_resource_name || resource_name).camelize(:lower)}List", :etag => "",