debian-mirror-gitlab/lib/api/files.rb

281 lines
11 KiB
Ruby
Raw Permalink Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
module API
2021-01-03 14:25:43 +05:30
class Files < ::API::Base
2019-02-15 15:39:39 +05:30
include APIGuard
FILE_ENDPOINT_REQUIREMENTS = API::NAMESPACE_OR_PROJECT_REQUIREMENTS.merge(file_path: API::NO_SLASH_URL_PART_REGEX)
2018-03-17 18:26:18 +05:30
2017-09-10 17:25:29 +05:30
# Prevents returning plain/text responses for files with .txt extension
after_validation { content_type "application/json" }
2021-01-29 00:20:46 +05:30
feature_category :source_code_management
2018-11-08 19:23:39 +05:30
helpers ::API::Helpers::HeadersHelpers
2015-09-25 12:07:36 +05:30
helpers do
def commit_params(attrs)
{
file_path: attrs[:file_path],
2017-09-10 17:25:29 +05:30
start_branch: attrs[:start_branch] || attrs[:branch],
2017-08-17 22:00:37 +05:30
branch_name: attrs[:branch],
2015-09-25 12:07:36 +05:30
commit_message: attrs[:commit_message],
file_content: attrs[:content],
2016-09-29 09:46:39 +05:30
file_content_encoding: attrs[:encoding],
author_email: attrs[:author_email],
2017-09-10 17:25:29 +05:30
author_name: attrs[:author_name],
2022-06-21 17:19:12 +05:30
last_commit_sha: attrs[:last_commit_id],
execute_filemode: attrs[:execute_filemode]
2015-09-25 12:07:36 +05:30
}
end
2017-08-17 22:00:37 +05:30
def assign_file_vars!
2023-06-20 00:43:36 +05:30
authorize_read_code!
2017-08-17 22:00:37 +05:30
@commit = user_project.commit(params[:ref])
not_found!('Commit') unless @commit
@repo = user_project.repository
2021-11-11 11:23:49 +05:30
@blob = @repo.blob_at(@commit.sha, params[:file_path], limit: Gitlab::Git::Blob::LFS_POINTER_MAX_SIZE)
2017-08-17 22:00:37 +05:30
not_found!('File') unless @blob
end
2015-09-25 12:07:36 +05:30
def commit_response(attrs)
{
file_path: attrs[:file_path],
2017-08-17 22:00:37 +05:30
branch: attrs[:branch]
2015-09-25 12:07:36 +05:30
}
end
2021-11-11 11:23:49 +05:30
def content_sha
2023-05-27 22:25:52 +05:30
cache_client.fetch("blob_content_sha256:#{user_project.full_path}:#{@blob.id}") do
2021-11-11 11:23:49 +05:30
@blob.load_all_data!
Digest::SHA256.hexdigest(@blob.data)
end
end
2023-05-27 22:25:52 +05:30
def cache_client
Gitlab::Cache::Client.build_with_metadata(
cache_identifier: 'API::Files#content_sha',
feature_category: :source_code_management,
backing_resource: :gitaly
)
end
2022-07-16 23:28:13 +05:30
def fetch_blame_range(blame_params)
return if blame_params[:range].blank?
range = Range.new(blame_params[:range][:start], blame_params[:range][:end])
render_api_error!('range[start] must be less than or equal to range[end]', 400) if range.begin > range.end
range
end
2018-11-08 19:23:39 +05:30
def blob_data
{
file_name: @blob.name,
file_path: @blob.path,
size: @blob.size,
encoding: "base64",
2021-11-11 11:23:49 +05:30
content_sha256: content_sha,
2018-11-08 19:23:39 +05:30
ref: params[:ref],
blob_id: @blob.id,
commit_id: @commit.id,
2022-06-21 17:19:12 +05:30
last_commit_id: @repo.last_commit_id_for_path(@commit.sha, params[:file_path], literal_pathspec: true),
execute_filemode: @blob.executable?
2018-11-08 19:23:39 +05:30
}
end
2017-08-17 22:00:37 +05:30
params :simple_file_params do
2023-01-13 00:05:48 +05:30
requires :file_path, type: String, file_path: true,
desc: 'The url encoded path to the file.', documentation: { example: 'lib%2Fclass%2Erb' }
requires :branch, type: String,
desc: 'Name of the branch to commit into. To create a new branch, also provide `start_branch`.', allow_blank: false,
documentation: { example: 'main' }
requires :commit_message, type: String,
allow_blank: false, desc: 'Commit message', documentation: { example: 'Initial commit' }
optional :start_branch, type: String,
desc: 'Name of the branch to start the new commit from', documentation: { example: 'main' }
optional :author_email, type: String,
desc: 'The email of the author', documentation: { example: 'johndoe@example.com' }
optional :author_name, type: String,
desc: 'The name of the author', documentation: { example: 'John Doe' }
2017-08-17 22:00:37 +05:30
end
2014-09-02 18:07:02 +05:30
2017-08-17 22:00:37 +05:30
params :extended_file_params do
use :simple_file_params
2023-01-13 00:05:48 +05:30
requires :content, type: String, desc: 'File content', documentation: { example: 'file content' }
optional :encoding, type: String, values: %w[base64 text], default: 'text', desc: 'File encoding'
optional :last_commit_id, type: String,
desc: 'Last known commit id for this file',
documentation: { example: '2695effb5807a22ff3d138d593fd856244e155e7' }
2022-06-21 17:19:12 +05:30
optional :execute_filemode, type: Boolean, desc: 'Enable / Disable the executable flag on the file path'
2017-08-17 22:00:37 +05:30
end
end
2014-09-02 18:07:02 +05:30
2017-08-17 22:00:37 +05:30
params do
2023-01-13 00:05:48 +05:30
requires :id, type: String, desc: 'The project ID', documentation: { example: 'gitlab-org/gitlab' }
2017-08-17 22:00:37 +05:30
end
2018-03-17 18:26:18 +05:30
resource :projects, requirements: FILE_ENDPOINT_REQUIREMENTS do
2019-02-15 15:39:39 +05:30
allow_access_with_scope :read_repository, if: -> (request) { request.get? || request.head? }
2019-10-12 21:52:04 +05:30
desc 'Get blame file metadata from repository'
params do
2023-01-13 00:05:48 +05:30
requires :file_path, type: String, file_path: true,
desc: 'The url encoded path to the file.', documentation: { example: 'lib%2Fclass%2Erb' }
requires :ref, type: String,
desc: 'The name of branch, tag or commit', allow_blank: false, documentation: { example: 'main' }
2019-10-12 21:52:04 +05:30
end
head ":id/repository/files/:file_path/blame", requirements: FILE_ENDPOINT_REQUIREMENTS do
assign_file_vars!
set_http_headers(blob_data)
end
desc 'Get blame file from the repository'
params do
2023-01-13 00:05:48 +05:30
requires :file_path, type: String, file_path: true,
desc: 'The url encoded path to the file.', documentation: { example: 'lib%2Fclass%2Erb' }
requires :ref, type: String,
desc: 'The name of branch, tag or commit', allow_blank: false, documentation: { example: 'main' }
2022-07-16 23:28:13 +05:30
optional :range, type: Hash do
2023-01-13 00:05:48 +05:30
requires :start, type: Integer,
desc: 'The first line of the range to blame', allow_blank: false, values: ->(v) { v > 0 }
requires :end, type: Integer,
desc: 'The last line of the range to blame', allow_blank: false, values: ->(v) { v > 0 }
2022-07-16 23:28:13 +05:30
end
2019-10-12 21:52:04 +05:30
end
get ":id/repository/files/:file_path/blame", requirements: FILE_ENDPOINT_REQUIREMENTS do
2022-07-16 23:28:13 +05:30
blame_params = declared_params(include_missing: false)
2019-10-12 21:52:04 +05:30
assign_file_vars!
set_http_headers(blob_data)
2022-07-16 23:28:13 +05:30
blame_ranges = Gitlab::Blame.new(@blob, @commit, range: fetch_blame_range(blame_params)).groups(highlight: false)
2019-10-12 21:52:04 +05:30
present blame_ranges, with: Entities::BlameRange
end
2023-01-13 00:05:48 +05:30
desc 'Get raw file contents from the repository' do
success File
end
2017-08-17 22:00:37 +05:30
params do
2023-01-13 00:05:48 +05:30
requires :file_path, type: String, file_path: true,
desc: 'The url encoded path to the file.', documentation: { example: 'lib%2Fclass%2Erb' }
optional :ref, type: String,
desc: 'The name of branch, tag or commit', allow_blank: false, documentation: { example: 'main' }
2023-03-04 22:38:38 +05:30
optional :lfs, type: Boolean,
desc: 'Retrieve binary data for a file that is an lfs pointer',
default: false
2017-08-17 22:00:37 +05:30
end
2021-12-11 22:18:48 +05:30
get ":id/repository/files/:file_path/raw", requirements: FILE_ENDPOINT_REQUIREMENTS, urgency: :low do
2017-08-17 22:00:37 +05:30
assign_file_vars!
2014-09-02 18:07:02 +05:30
2023-03-04 22:38:38 +05:30
if params[:lfs] && @blob.stored_externally?
lfs_object = LfsObject.find_by_oid(@blob.lfs_oid)
not_found! unless lfs_object&.project_allowed_access?(@project)
present_carrierwave_file!(lfs_object.file)
else
no_cache_headers
set_http_headers(blob_data)
2018-11-08 19:23:39 +05:30
2023-03-04 22:38:38 +05:30
send_git_blob @repo, @blob
end
2017-08-17 22:00:37 +05:30
end
2014-09-02 18:07:02 +05:30
2018-11-08 19:23:39 +05:30
desc 'Get file metadata from repository'
params do
2023-01-13 00:05:48 +05:30
requires :file_path, type: String, file_path: true,
desc: 'The url encoded path to the file.', documentation: { example: 'lib%2Fclass%2Erb' }
requires :ref, type: String,
desc: 'The name of branch, tag or commit', allow_blank: false, documentation: { example: 'main' }
2018-11-08 19:23:39 +05:30
end
2021-12-11 22:18:48 +05:30
head ":id/repository/files/:file_path", requirements: FILE_ENDPOINT_REQUIREMENTS, urgency: :low do
2018-11-08 19:23:39 +05:30
assign_file_vars!
set_http_headers(blob_data)
end
2017-08-17 22:00:37 +05:30
desc 'Get a file from the repository'
params do
2023-01-13 00:05:48 +05:30
requires :file_path, type: String, file_path: true,
desc: 'The url encoded path to the file.', documentation: { example: 'lib%2Fclass%2Erb' }
requires :ref, type: String,
desc: 'The name of branch, tag or commit', allow_blank: false, documentation: { example: 'main' }
2017-08-17 22:00:37 +05:30
end
2018-03-17 18:26:18 +05:30
get ":id/repository/files/:file_path", requirements: FILE_ENDPOINT_REQUIREMENTS do
2017-08-17 22:00:37 +05:30
assign_file_vars!
2014-09-02 18:07:02 +05:30
2021-11-11 11:23:49 +05:30
@blob.load_all_data!
2018-11-08 19:23:39 +05:30
data = blob_data
set_http_headers(data)
data.merge(content: Base64.strict_encode64(@blob.data))
2014-09-02 18:07:02 +05:30
end
2017-08-17 22:00:37 +05:30
desc 'Create new file in repository'
params do
use :extended_file_params
end
2021-12-11 22:18:48 +05:30
post ":id/repository/files/:file_path", requirements: FILE_ENDPOINT_REQUIREMENTS, urgency: :low do
2014-09-02 18:07:02 +05:30
authorize! :push_code, user_project
2017-08-17 22:00:37 +05:30
file_params = declared_params(include_missing: false)
result = ::Files::CreateService.new(user_project, current_user, commit_params(file_params)).execute
2014-09-02 18:07:02 +05:30
if result[:status] == :success
status(201)
2017-08-17 22:00:37 +05:30
commit_response(file_params)
2014-09-02 18:07:02 +05:30
else
2015-04-26 12:48:37 +05:30
render_api_error!(result[:message], 400)
2014-09-02 18:07:02 +05:30
end
end
2017-08-17 22:00:37 +05:30
desc 'Update existing file in repository'
params do
use :extended_file_params
end
2021-12-11 22:18:48 +05:30
put ":id/repository/files/:file_path", requirements: FILE_ENDPOINT_REQUIREMENTS, urgency: :low do
2014-09-02 18:07:02 +05:30
authorize! :push_code, user_project
2017-08-17 22:00:37 +05:30
file_params = declared_params(include_missing: false)
2017-09-10 17:25:29 +05:30
begin
result = ::Files::UpdateService.new(user_project, current_user, commit_params(file_params)).execute
rescue ::Files::UpdateService::FileChangedError => e
render_api_error!(e.message, 400)
end
2014-09-02 18:07:02 +05:30
if result[:status] == :success
status(200)
2017-08-17 22:00:37 +05:30
commit_response(file_params)
2014-09-02 18:07:02 +05:30
else
2015-04-26 12:48:37 +05:30
http_status = result[:http_status] || 400
render_api_error!(result[:message], http_status)
2014-09-02 18:07:02 +05:30
end
end
2017-08-17 22:00:37 +05:30
desc 'Delete an existing file in repository'
params do
use :simple_file_params
end
2018-03-17 18:26:18 +05:30
delete ":id/repository/files/:file_path", requirements: FILE_ENDPOINT_REQUIREMENTS do
2014-09-02 18:07:02 +05:30
authorize! :push_code, user_project
2017-08-17 22:00:37 +05:30
file_params = declared_params(include_missing: false)
result = ::Files::DeleteService.new(user_project, current_user, commit_params(file_params)).execute
2014-09-02 18:07:02 +05:30
2017-08-17 22:00:37 +05:30
if result[:status] != :success
2015-04-26 12:48:37 +05:30
render_api_error!(result[:message], 400)
2014-09-02 18:07:02 +05:30
end
end
end
end
end