debian-mirror-gitlab/app/controllers/projects/repositories_controller.rb

131 lines
3.7 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
class Projects::RepositoriesController < Projects::ApplicationController
2018-05-09 12:01:36 +05:30
include ExtractsPath
2019-12-04 20:38:33 +05:30
include StaticObjectExternalStorage
2020-04-08 14:13:33 +05:30
include Gitlab::RateLimitHelpers
2020-03-28 13:19:24 +05:30
include HotlinkInterceptor
2019-12-04 20:38:33 +05:30
prepend_before_action(only: [:archive]) { authenticate_sessionless_user!(:archive) }
2018-05-09 12:01:36 +05:30
2021-01-29 00:20:46 +05:30
skip_before_action :default_cache_headers, only: :archive
2014-09-02 18:07:02 +05:30
# Authorize
2015-09-11 14:41:01 +05:30
before_action :require_non_empty_project, except: :create
2020-04-08 14:13:33 +05:30
before_action :archive_rate_limit!, only: :archive
2020-03-28 13:19:24 +05:30
before_action :intercept_hotlinking!, only: :archive
2018-05-09 12:01:36 +05:30
before_action :assign_archive_vars, only: :archive
2019-10-12 21:52:04 +05:30
before_action :assign_append_sha, only: :archive
2015-09-11 14:41:01 +05:30
before_action :authorize_download_code!
before_action :authorize_admin_project!, only: :create
2019-12-04 20:38:33 +05:30
before_action :redirect_to_external_storage, only: :archive, if: :static_objects_external_storage_enabled?
2014-09-02 18:07:02 +05:30
2021-01-03 14:25:43 +05:30
feature_category :source_code_management
2015-04-26 12:48:37 +05:30
def create
@project.create_repository
redirect_to project_path(@project)
2014-09-02 18:07:02 +05:30
end
def archive
2020-03-13 15:44:24 +05:30
return render_404 if html_request?
2019-10-12 21:52:04 +05:30
set_cache_headers
return if archive_not_modified?
2018-05-09 12:01:36 +05:30
2019-10-12 21:52:04 +05:30
send_git_archive @repository, **repo_params
2021-06-08 01:23:25 +05:30
rescue StandardError => ex
2015-10-24 18:46:33 +05:30
logger.error("#{self.class.name}: #{ex}")
2018-11-18 11:00:15 +05:30
git_not_found!
2014-09-02 18:07:02 +05:30
end
2018-05-09 12:01:36 +05:30
2019-10-12 21:52:04 +05:30
private
2020-04-08 14:13:33 +05:30
def archive_rate_limit!
if archive_rate_limit_reached?(current_user, @project)
render plain: ::Gitlab::RateLimitHelpers::ARCHIVE_RATE_LIMIT_REACHED_MESSAGE, status: :too_many_requests
end
end
2019-10-12 21:52:04 +05:30
def repo_params
@repo_params ||= { ref: @ref, path: params[:path], format: params[:format], append_sha: @append_sha }
end
def set_cache_headers
2021-01-08 16:13:35 +05:30
expires_in cache_max_age(archive_metadata['CommitId']), public: Guest.can?(:download_code, project)
2019-10-12 21:52:04 +05:30
fresh_when(etag: archive_metadata['ArchivePath'])
end
def archive_not_modified?
# Check response freshness (Last-Modified and ETag)
# against request If-Modified-Since and If-None-Match conditions.
request.fresh?(response)
end
def archive_metadata
@archive_metadata ||= @repository.archive_metadata(
@ref,
'', # Where archives are stored isn't really important for ETag purposes
repo_params[:format],
path: repo_params[:path],
append_sha: @append_sha
)
end
def cache_max_age(commit_id)
if @ref == commit_id
# This is a link to an archive by a commit SHA. That means that the archive
# is immutable. The only reason to invalidate the cache is if the commit
# was deleted or if the user lost access to the repository.
Repository::ARCHIVE_CACHE_TIME_IMMUTABLE
else
# A branch or tag points at this archive. That means that the expected archive
# content may change over time.
Repository::ARCHIVE_CACHE_TIME
end
end
def assign_append_sha
@append_sha = params[:append_sha]
if @ref
shortname = "#{@project.path}-#{@ref.tr('/', '-')}"
@append_sha = false if @filename == shortname
end
end
2018-05-09 12:01:36 +05:30
def assign_archive_vars
if params[:id]
2020-03-13 15:44:24 +05:30
@ref, @filename = extract_ref_and_filename(params[:id])
2018-05-09 12:01:36 +05:30
else
@ref = params[:ref]
@filename = nil
end
rescue InvalidPathError
render_404
end
2020-03-13 15:44:24 +05:30
# path can be of the form:
# master
# master/first.zip
# master/first/second.tar.gz
# master/first/second/third.zip
#
# In the archive case, we know that the last value is always the filename, so we
# do a greedy match to extract the ref. This avoid having to pull all ref names
# from Redis.
def extract_ref_and_filename(id)
path = id.strip
2021-09-30 23:02:18 +05:30
data = path.match(%r{(.*)/(.*)})
2020-03-13 15:44:24 +05:30
if data
[data[1], data[2]]
else
[path, nil]
end
end
2014-09-02 18:07:02 +05:30
end
2019-12-04 20:38:33 +05:30
2021-06-08 01:23:25 +05:30
Projects::RepositoriesController.prepend_mod_with('Projects::RepositoriesController')