Welcome to mirror list, hosted at ThFree Co, Russian Federation.

gitlab.com/gitlab-org/gitlab-foss.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'lib/api/helpers.rb')
-rw-r--r--lib/api/helpers.rb49
1 files changed, 45 insertions, 4 deletions
diff --git a/lib/api/helpers.rb b/lib/api/helpers.rb
index b616f1b35b3..b7f21bd6c22 100644
--- a/lib/api/helpers.rb
+++ b/lib/api/helpers.rb
@@ -147,7 +147,7 @@ module API
if id.is_a?(Integer) || id =~ INTEGER_ID_REGEX
projects.find_by(id: id)
elsif id.include?("/")
- projects.find_by_full_path(id)
+ projects.find_by_full_path(id, follow_redirects: Feature.enabled?(:api_redirect_moved_projects))
end
end
# rubocop: enable CodeReuse/ActiveRecord
@@ -157,10 +157,23 @@ module API
return forbidden! unless authorized_project_scope?(project)
- return project if can?(current_user, :read_project, project)
- return unauthorized! if authenticate_non_public?
+ unless can?(current_user, read_project_ability, project)
+ return unauthorized! if authenticate_non_public?
+
+ return not_found!('Project')
+ end
- not_found!('Project')
+ if project_moved?(id, project)
+ return not_allowed!('Non GET methods are not allowed for moved projects') unless request.get?
+
+ return redirect!(url_with_project_id(project))
+ end
+
+ project
+ end
+
+ def read_project_ability
+ :read_project
end
def authorized_project_scope?(project)
@@ -438,6 +451,13 @@ module API
order_options
end
+ # An error is raised to interrupt user's request and redirect them to the right route.
+ # The error! helper behaves similarly, but it cannot be used because it formats the
+ # response message.
+ def redirect!(location_url)
+ raise ::API::API::MovedPermanentlyError, location_url
+ end
+
# error helpers
def forbidden!(reason = nil)
@@ -737,6 +757,9 @@ module API
@initial_current_user = Gitlab::Auth::UniqueIpsLimiter.limit_user! { find_current_user! }
rescue Gitlab::Auth::UnauthorizedError
unauthorized!
+
+ # Explicitly return `nil`, otherwise an instance of `Rack::Response` is returned when reporting an error
+ nil
end
end
# rubocop:enable Gitlab/ModuleWithInstanceVariables
@@ -837,6 +860,24 @@ module API
def sanitize_id_param(id)
id.present? ? id.to_i : nil
end
+
+ def project_moved?(id, project)
+ return false unless Feature.enabled?(:api_redirect_moved_projects)
+ return false unless id.is_a?(String) && id.include?('/')
+ return false if project.blank? || id == project.full_path
+ return false unless params[:id] == id
+
+ true
+ end
+
+ def url_with_project_id(project)
+ new_params = params.merge(id: project.id.to_s).transform_values { |v| v.is_a?(String) ? CGI.escape(v) : v }
+ new_path = GrapePathHelpers::DecoratedRoute.new(route).path_segments_with_values(new_params).join('/')
+
+ Rack::Request.new(env).tap do |r|
+ r.path_info = "/#{new_path}"
+ end.url
+ end
end
end