2021-01-29 00:20:46 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
# NPM Package Manager Client API
|
|
|
|
#
|
|
|
|
# These API endpoints are not consumed directly by users, so there is no documentation for the
|
|
|
|
# individual endpoints. They are called by the NPM package manager client when users run commands
|
|
|
|
# like `npm install` or `npm publish`. The usage of the GitLab NPM registry is documented here:
|
|
|
|
# https://docs.gitlab.com/ee/user/packages/npm_registry/
|
|
|
|
#
|
|
|
|
# Technical debt: https://gitlab.com/gitlab-org/gitlab/issues/35798
|
|
|
|
#
|
|
|
|
# Caution: This Concern has to be included at the end of the API class
|
|
|
|
# The last route of this Concern has a globbing wildcard that will match all urls.
|
|
|
|
# As such, routes declared after the last route of this Concern will not match any url.
|
|
|
|
module API
|
|
|
|
module Concerns
|
|
|
|
module Packages
|
|
|
|
module NpmEndpoints
|
|
|
|
extend ActiveSupport::Concern
|
|
|
|
|
|
|
|
included do
|
|
|
|
helpers ::API::Helpers::Packages::DependencyProxyHelpers
|
|
|
|
|
|
|
|
before do
|
|
|
|
require_packages_enabled!
|
|
|
|
authenticate_non_get!
|
|
|
|
end
|
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
helpers do
|
2023-07-07 10:43:13 +05:30
|
|
|
params :package_name do
|
|
|
|
requires :package_name, type: String, file_path: true, desc: 'Package name',
|
|
|
|
documentation: { example: 'mypackage' }
|
|
|
|
end
|
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
def redirect_or_present_audit_report
|
|
|
|
redirect_registry_request(
|
|
|
|
forward_to_registry: true,
|
|
|
|
package_type: :npm,
|
|
|
|
path: options[:path][0],
|
|
|
|
body: Gitlab::Json.dump(request.POST),
|
|
|
|
target: project_or_nil,
|
|
|
|
method: route.request_method
|
|
|
|
) do
|
|
|
|
authorize_read_package!(project)
|
|
|
|
|
|
|
|
status :ok
|
|
|
|
present []
|
|
|
|
end
|
|
|
|
end
|
2023-06-20 00:43:36 +05:30
|
|
|
|
|
|
|
def generate_metadata_service(packages)
|
|
|
|
::Packages::Npm::GenerateMetadataService.new(params[:package_name], packages)
|
|
|
|
end
|
2023-03-04 22:38:38 +05:30
|
|
|
end
|
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
params do
|
|
|
|
requires :package_name, type: String, desc: 'Package name'
|
|
|
|
end
|
|
|
|
namespace '-/package/*package_name' do
|
|
|
|
desc 'Get all tags for a given an NPM package' do
|
|
|
|
detail 'This feature was introduced in GitLab 12.7'
|
2023-03-04 22:38:38 +05:30
|
|
|
success [
|
|
|
|
{ code: 200, model: ::API::Entities::NpmPackageTag }
|
|
|
|
]
|
|
|
|
failure [
|
|
|
|
{ code: 400, message: 'Bad Request' },
|
|
|
|
{ code: 403, message: 'Forbidden' },
|
|
|
|
{ code: 404, message: 'Not Found' }
|
|
|
|
]
|
|
|
|
tags %w[npm_packages]
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
2023-05-27 22:25:52 +05:30
|
|
|
route_setting :authentication, job_token_allowed: true, deploy_token_allowed: true
|
2021-01-29 00:20:46 +05:30
|
|
|
get 'dist-tags', format: false, requirements: ::API::Helpers::Packages::Npm::NPM_ENDPOINT_REQUIREMENTS do
|
|
|
|
package_name = params[:package_name]
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
bad_request_missing_attribute!('Package Name') if package_name.blank?
|
2021-01-29 00:20:46 +05:30
|
|
|
|
|
|
|
authorize_read_package!(project)
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
packages = ::Packages::Npm::PackageFinder.new(package_name, project: project)
|
|
|
|
.execute
|
2021-01-29 00:20:46 +05:30
|
|
|
|
|
|
|
not_found! if packages.empty?
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
track_package_event(:list_tags, :npm, project: project, namespace: project.namespace)
|
|
|
|
|
|
|
|
metadata = generate_metadata_service(packages).execute(only_dist_tags: true)
|
|
|
|
present ::Packages::Npm::PackagePresenter.new(metadata),
|
2021-01-29 00:20:46 +05:30
|
|
|
with: ::API::Entities::NpmPackageTag
|
|
|
|
end
|
|
|
|
|
|
|
|
params do
|
|
|
|
requires :tag, type: String, desc: "Package dist-tag"
|
|
|
|
end
|
|
|
|
namespace 'dist-tags/:tag', requirements: ::API::Helpers::Packages::Npm::NPM_ENDPOINT_REQUIREMENTS do
|
|
|
|
desc 'Create or Update the given tag for the given NPM package and version' do
|
|
|
|
detail 'This feature was introduced in GitLab 12.7'
|
2023-03-04 22:38:38 +05:30
|
|
|
success code: 204
|
|
|
|
failure [
|
|
|
|
{ code: 400, message: 'Bad Request' },
|
|
|
|
{ code: 401, message: 'Unauthorized' },
|
|
|
|
{ code: 403, message: 'Forbidden' },
|
|
|
|
{ code: 404, message: 'Not Found' }
|
|
|
|
]
|
|
|
|
tags %w[npm_packages]
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
2023-05-27 22:25:52 +05:30
|
|
|
route_setting :authentication, job_token_allowed: true, deploy_token_allowed: true
|
2021-01-29 00:20:46 +05:30
|
|
|
put format: false do
|
|
|
|
package_name = params[:package_name]
|
|
|
|
version = env['api.request.body']
|
|
|
|
tag = params[:tag]
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
bad_request_missing_attribute!('Package Name') if package_name.blank?
|
|
|
|
bad_request_missing_attribute!('Version') if version.blank?
|
|
|
|
bad_request_missing_attribute!('Tag') if tag.blank?
|
2021-01-29 00:20:46 +05:30
|
|
|
|
|
|
|
authorize_create_package!(project)
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
package = ::Packages::Npm::PackageFinder.new(package_name, project: project)
|
|
|
|
.find_by_version(version)
|
2021-01-29 00:20:46 +05:30
|
|
|
not_found!('Package') unless package
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
track_package_event(:create_tag, :npm, project: project, namespace: project.namespace)
|
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
::Packages::Npm::CreateTagService.new(package, tag).execute
|
|
|
|
|
|
|
|
no_content!
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'Deletes the given tag' do
|
|
|
|
detail 'This feature was introduced in GitLab 12.7'
|
2023-03-04 22:38:38 +05:30
|
|
|
success code: 204
|
|
|
|
failure [
|
|
|
|
{ code: 400, message: 'Bad Request' },
|
|
|
|
{ code: 401, message: 'Unauthorized' },
|
|
|
|
{ code: 403, message: 'Forbidden' },
|
|
|
|
{ code: 404, message: 'Not Found' }
|
|
|
|
]
|
|
|
|
tags %w[npm_packages]
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
2023-05-27 22:25:52 +05:30
|
|
|
route_setting :authentication, job_token_allowed: true, deploy_token_allowed: true
|
2021-01-29 00:20:46 +05:30
|
|
|
delete format: false do
|
|
|
|
package_name = params[:package_name]
|
|
|
|
tag = params[:tag]
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
bad_request_missing_attribute!('Package Name') if package_name.blank?
|
|
|
|
bad_request_missing_attribute!('Tag') if tag.blank?
|
2021-01-29 00:20:46 +05:30
|
|
|
|
|
|
|
authorize_destroy_package!(project)
|
|
|
|
|
|
|
|
package_tag = ::Packages::TagsFinder
|
|
|
|
.new(project, package_name, package_type: :npm)
|
|
|
|
.find_by_name(tag)
|
|
|
|
|
|
|
|
not_found!('Package tag') unless package_tag
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
track_package_event(:delete_tag, :npm, project: project, namespace: project.namespace)
|
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
::Packages::RemoveTagService.new(package_tag).execute
|
|
|
|
|
|
|
|
no_content!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'NPM registry metadata endpoint' do
|
|
|
|
detail 'This feature was introduced in GitLab 11.8'
|
2023-03-04 22:38:38 +05:30
|
|
|
success [
|
|
|
|
{ code: 200, model: ::API::Entities::NpmPackage, message: 'Ok' },
|
|
|
|
{ code: 302, message: 'Found (redirect)' }
|
|
|
|
]
|
|
|
|
failure [
|
|
|
|
{ code: 400, message: 'Bad Request' },
|
|
|
|
{ code: 403, message: 'Forbidden' },
|
|
|
|
{ code: 404, message: 'Not Found' }
|
|
|
|
]
|
|
|
|
tags %w[npm_packages]
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
params do
|
2023-07-07 10:43:13 +05:30
|
|
|
use :package_name
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
route_setting :authentication, job_token_allowed: true, deploy_token_allowed: true
|
|
|
|
get '*package_name', format: false, requirements: ::API::Helpers::Packages::Npm::NPM_ENDPOINT_REQUIREMENTS do
|
|
|
|
package_name = params[:package_name]
|
2023-05-27 22:25:52 +05:30
|
|
|
packages =
|
|
|
|
if Feature.enabled?(:npm_allow_packages_in_multiple_projects)
|
|
|
|
finder_for_endpoint_scope(package_name).execute
|
|
|
|
else
|
|
|
|
::Packages::Npm::PackageFinder.new(package_name, project: project_or_nil)
|
|
|
|
.execute
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
|
|
|
|
redirect_request = project_or_nil.blank? || packages.empty?
|
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
redirect_registry_request(
|
|
|
|
forward_to_registry: redirect_request,
|
|
|
|
package_type: :npm,
|
|
|
|
target: project_or_nil,
|
|
|
|
package_name: package_name
|
|
|
|
) do
|
2021-01-29 00:20:46 +05:30
|
|
|
authorize_read_package!(project)
|
|
|
|
|
|
|
|
not_found!('Packages') if packages.empty?
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
present ::Packages::Npm::PackagePresenter.new(generate_metadata_service(packages).execute),
|
2021-01-29 00:20:46 +05:30
|
|
|
with: ::API::Entities::NpmPackage
|
|
|
|
end
|
|
|
|
end
|
2023-03-04 22:38:38 +05:30
|
|
|
|
|
|
|
desc 'NPM registry bulk advisory endpoint' do
|
|
|
|
detail 'This feature was introduced in GitLab 15.6'
|
|
|
|
success [
|
|
|
|
{ code: 200, message: 'Ok' },
|
|
|
|
{ code: 307, message: 'Temporary Redirect' }
|
|
|
|
]
|
|
|
|
failure [
|
|
|
|
{ code: 401, message: 'Unauthorized' },
|
|
|
|
{ code: 403, message: 'Forbidden' },
|
|
|
|
{ code: 404, message: 'Not Found' }
|
|
|
|
]
|
|
|
|
is_array true
|
|
|
|
tags %w[npm_packages]
|
|
|
|
end
|
|
|
|
route_setting :authentication, job_token_allowed: true, deploy_token_allowed: true
|
|
|
|
post '-/npm/v1/security/advisories/bulk' do
|
|
|
|
redirect_or_present_audit_report
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'NPM registry quick audit endpoint' do
|
|
|
|
detail 'This feature was introduced in GitLab 15.6'
|
|
|
|
success [
|
|
|
|
{ code: 200, message: 'Ok' },
|
|
|
|
{ code: 307, message: 'Temporary Redirect' }
|
|
|
|
]
|
|
|
|
failure [
|
|
|
|
{ code: 401, message: 'Unauthorized' },
|
|
|
|
{ code: 403, message: 'Forbidden' },
|
|
|
|
{ code: 404, message: 'Not Found' }
|
|
|
|
]
|
|
|
|
is_array true
|
|
|
|
tags %w[npm_packages]
|
|
|
|
end
|
|
|
|
route_setting :authentication, job_token_allowed: true, deploy_token_allowed: true
|
|
|
|
post '-/npm/v1/security/audits/quick' do
|
|
|
|
redirect_or_present_audit_report
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|