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

162 lines
5.8 KiB
Ruby
Raw Normal View History

2021-03-11 19:13:27 +05:30
# frozen_string_literal: true
###
# API endpoints for the RubyGem package registry
module API
class RubygemPackages < ::API::Base
include ::API::Helpers::Authentication
helpers ::API::Helpers::PackagesHelpers
feature_category :package_registry
2022-07-16 23:28:13 +05:30
urgency :low
2021-03-11 19:13:27 +05:30
# The Marshal version can be found by "#{Marshal::MAJOR_VERSION}.#{Marshal::MINOR_VERSION}"
# Updating the version should require a GitLab API version change.
MARSHAL_VERSION = '4.8'
2021-04-17 20:07:23 +05:30
PACKAGE_FILENAME = 'package.gem'
2021-03-11 19:13:27 +05:30
FILE_NAME_REQUIREMENTS = {
file_name: API::NO_SLASH_URL_PART_REGEX
}.freeze
content_type :binary, 'application/octet-stream'
authenticate_with do |accept|
accept.token_types(:personal_access_token, :deploy_token, :job_token)
.sent_through(:http_token)
end
before do
require_packages_enabled!
2021-04-17 20:07:23 +05:30
authenticate_non_get!
2021-03-11 19:13:27 +05:30
not_found! unless Feature.enabled?(:rubygem_packages, user_project)
end
params do
requires :id, type: String, desc: 'The ID or full path of a project'
end
resource :projects, requirements: API::NAMESPACE_OR_PROJECT_REQUIREMENTS do
namespace ':id/packages/rubygems' do
desc 'Download the spec index file' do
detail 'This feature was introduced in GitLab 13.9'
end
params do
requires :file_name, type: String, desc: 'Spec file name'
end
get ":file_name", requirements: FILE_NAME_REQUIREMENTS do
# To be implemented in https://gitlab.com/gitlab-org/gitlab/-/issues/299267
not_found!
end
desc 'Download the gemspec file' do
detail 'This feature was introduced in GitLab 13.9'
end
params do
requires :file_name, type: String, desc: 'Gemspec file name'
end
get "quick/Marshal.#{MARSHAL_VERSION}/:file_name", requirements: FILE_NAME_REQUIREMENTS do
# To be implemented in https://gitlab.com/gitlab-org/gitlab/-/issues/299284
not_found!
end
desc 'Download the .gem package' do
detail 'This feature was introduced in GitLab 13.9'
end
params do
requires :file_name, type: String, desc: 'Package file name'
end
get "gems/:file_name", requirements: FILE_NAME_REQUIREMENTS do
2022-10-11 01:57:18 +05:30
authorize_read_package!(user_project)
2021-04-17 20:07:23 +05:30
2022-03-02 08:16:31 +05:30
package_files = ::Packages::PackageFile
.for_rubygem_with_file_name(user_project, params[:file_name])
2022-04-04 11:22:00 +05:30
package_file = package_files.installable.last!
2021-04-17 20:07:23 +05:30
2021-09-04 01:27:46 +05:30
track_package_event('pull_package', :rubygems, project: user_project, namespace: user_project.namespace)
2021-04-17 20:07:23 +05:30
2022-10-11 01:57:18 +05:30
present_package_file!(package_file)
2021-03-11 19:13:27 +05:30
end
namespace 'api/v1' do
desc 'Authorize a gem upload from workhorse' do
detail 'This feature was introduced in GitLab 13.9'
end
post 'gems/authorize' do
2021-04-17 20:07:23 +05:30
authorize_workhorse!(
subject: user_project,
has_length: false,
maximum_size: user_project.actual_limits.rubygems_max_file_size
)
2021-03-11 19:13:27 +05:30
end
desc 'Upload a gem' do
detail 'This feature was introduced in GitLab 13.9'
end
2021-04-17 20:07:23 +05:30
params do
requires :file, type: ::API::Validations::Types::WorkhorseFile, desc: 'The package file to be published (generated by Multipart middleware)'
end
2021-03-11 19:13:27 +05:30
post 'gems' do
2021-04-17 20:07:23 +05:30
authorize_upload!(user_project)
bad_request!('File is too large') if user_project.actual_limits.exceeded?(:rubygems_max_file_size, params[:file].size)
2021-09-04 01:27:46 +05:30
track_package_event('push_package', :rubygems, user: current_user, project: user_project, namespace: user_project.namespace)
2021-04-17 20:07:23 +05:30
2021-04-29 21:17:54 +05:30
package_file = nil
2021-10-27 15:23:28 +05:30
ApplicationRecord.transaction do
2021-04-17 20:07:23 +05:30
package = ::Packages::CreateTemporaryPackageService.new(
user_project, current_user, declared_params.merge(build: current_authenticated_job)
).execute(:rubygems, name: ::Packages::Rubygems::TEMPORARY_PACKAGE_NAME)
file_params = {
2022-08-27 11:52:29 +05:30
file: params[:file],
2021-04-17 20:07:23 +05:30
file_name: PACKAGE_FILENAME
}
2021-04-29 21:17:54 +05:30
package_file = ::Packages::CreatePackageFileService.new(
2021-04-17 20:07:23 +05:30
package, file_params.merge(build: current_authenticated_job)
).execute
end
2021-04-29 21:17:54 +05:30
if package_file
::Packages::Rubygems::ExtractionWorker.perform_async(package_file.id) # rubocop:disable CodeReuse/Worker
created!
else
bad_request!('Package creation failed')
end
2021-04-17 20:07:23 +05:30
rescue ObjectStorage::RemoteStoreError => e
Gitlab::ErrorTracking.track_exception(e, extra: { file_name: params[:file_name], project_id: user_project.id })
forbidden!
2021-03-11 19:13:27 +05:30
end
desc 'Fetch a list of dependencies' do
detail 'This feature was introduced in GitLab 13.9'
end
params do
2021-04-17 20:07:23 +05:30
optional :gems, type: Array[String], coerce_with: ::API::Validations::Types::CommaSeparatedToArray.coerce, desc: 'Comma delimited gem names'
2021-03-11 19:13:27 +05:30
end
get 'dependencies' do
2021-04-17 20:07:23 +05:30
authorize_read_package!
if params[:gems].blank?
status :ok
else
results = params[:gems].map do |gem_name|
service_result = Packages::Rubygems::DependencyResolverService.new(user_project, current_user, gem_name: gem_name).execute
render_api_error!(service_result.message, service_result.http_status) if service_result.error?
service_result.payload
end
content_type 'application/octet-stream'
Marshal.dump(results.flatten)
end
2021-03-11 19:13:27 +05:30
end
end
end
end
end
end