debian-mirror-gitlab/spec/requests/api/graphql/packages/package_spec.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

375 lines
12 KiB
Ruby
Raw Normal View History

2021-03-11 19:13:27 +05:30
# frozen_string_literal: true
require 'spec_helper'
2023-03-04 22:38:38 +05:30
RSpec.describe 'package details', feature_category: :package_registry do
2021-03-11 19:13:27 +05:30
include GraphqlHelpers
2022-03-02 08:16:31 +05:30
let_it_be_with_reload(:group) { create(:group) }
let_it_be_with_reload(:project) { create(:project, group: group) }
2022-10-11 01:57:18 +05:30
let_it_be_with_reload(:composer_package) { create(:composer_package, :last_downloaded_at, project: project) }
2022-03-02 08:16:31 +05:30
let_it_be(:user) { create(:user) }
2021-03-11 19:13:27 +05:30
let_it_be(:composer_json) { { name: 'name', type: 'type', license: 'license', version: 1 } }
let_it_be(:composer_metadatum) do
# we are forced to manually create the metadatum, without using the factory to force the sha to be a string
# and avoid an error where gitaly can't find the repository
2021-04-29 21:17:54 +05:30
create(:composer_metadatum, package: composer_package, target_sha: 'foo_sha', composer_json: composer_json)
2021-03-11 19:13:27 +05:30
end
let(:depth) { 3 }
2021-04-29 21:17:54 +05:30
let(:excluded) { %w[metadata apiFuzzingCiConfiguration pipeline packageFiles] }
let(:metadata) { query_graphql_fragment('ComposerMetadata') }
2022-08-27 11:52:29 +05:30
let(:package_files) { all_graphql_fields_for('PackageFile') }
2021-06-08 01:23:25 +05:30
let(:package_global_id) { global_id_of(composer_package) }
let(:package_details) { graphql_data_at(:package) }
2021-03-11 19:13:27 +05:30
let(:query) do
graphql_query_for(:package, { id: package_global_id }, <<~FIELDS)
2021-04-29 21:17:54 +05:30
#{all_graphql_fields_for('PackageDetailsType', max_depth: depth, excluded: excluded)}
2021-03-11 19:13:27 +05:30
metadata {
2021-04-29 21:17:54 +05:30
#{metadata}
}
packageFiles {
nodes {
#{package_files}
}
2021-03-11 19:13:27 +05:30
}
FIELDS
end
subject { post_graphql(query, current_user: user) }
2022-03-02 08:16:31 +05:30
context 'with unauthorized user' do
2021-03-11 19:13:27 +05:30
before do
2022-03-02 08:16:31 +05:30
project.update!(visibility_level: Gitlab::VisibilityLevel::PRIVATE)
2023-04-23 21:23:45 +05:30
project.add_guest(user)
2021-03-11 19:13:27 +05:30
end
2022-03-02 08:16:31 +05:30
it 'returns no packages' do
subject
expect(graphql_data_at(:package)).to be_nil
2021-03-11 19:13:27 +05:30
end
2023-04-23 21:23:45 +05:30
context 'with access to package registry for everyone' do
before do
project.project_feature.update!(package_registry_access_level: ProjectFeature::PUBLIC)
subject
end
it_behaves_like 'a working graphql query' do
it 'matches the JSON schema' do
expect(package_details).to match_schema('graphql/packages/package_details')
end
end
it '`public_package` returns true' do
expect(graphql_data_at(:package, :public_package)).to eq(true)
end
end
end
context 'when project is public' do
let_it_be(:public_project) { create(:project, :public, group: group) }
let_it_be(:composer_package) { create(:composer_package, project: public_project) }
let(:package_global_id) { global_id_of(composer_package) }
before do
subject
end
it_behaves_like 'a working graphql query' do
before do
subject
end
it 'matches the JSON schema' do
expect(package_details).to match_schema('graphql/packages/package_details')
end
end
it '`public_package` returns true' do
expect(graphql_data_at(:package, :public_package)).to eq(true)
end
2021-04-29 21:17:54 +05:30
end
2022-03-02 08:16:31 +05:30
context 'with authorized user' do
before do
project.add_developer(user)
end
2021-03-11 19:13:27 +05:30
2022-03-02 08:16:31 +05:30
it_behaves_like 'a working graphql query' do
before do
subject
end
2021-03-11 19:13:27 +05:30
2022-03-02 08:16:31 +05:30
it 'matches the JSON schema' do
expect(package_details).to match_schema('graphql/packages/package_details')
end
2021-03-11 19:13:27 +05:30
end
2022-10-11 01:57:18 +05:30
context 'with package without last_downloaded_at' do
before do
composer_package.update!(last_downloaded_at: nil)
subject
end
it 'matches the JSON schema' do
expect(package_details).to match_schema('graphql/packages/package_details')
end
end
2022-03-02 08:16:31 +05:30
context 'with package files pending destruction' do
let_it_be(:package_file) { create(:package_file, package: composer_package) }
let_it_be(:package_file_pending_destruction) { create(:package_file, :pending_destruction, package: composer_package) }
2021-04-29 21:17:54 +05:30
2022-03-02 08:16:31 +05:30
let(:package_file_ids) { graphql_data_at(:package, :package_files, :nodes).map { |node| node["id"] } }
2021-04-29 21:17:54 +05:30
2022-03-02 08:16:31 +05:30
it 'does not return them' do
subject
2021-04-29 21:17:54 +05:30
2022-03-02 08:16:31 +05:30
expect(package_file_ids).to contain_exactly(package_file.to_global_id.to_s)
end
2022-01-26 12:08:38 +05:30
end
2022-03-02 08:16:31 +05:30
context 'with a batched query' do
let_it_be(:conan_package) { create(:conan_package, project: project) }
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
let(:batch_query) do
<<~QUERY
{
a: package(id: "#{global_id_of(composer_package)}") { name }
b: package(id: "#{global_id_of(conan_package)}") { name }
}
QUERY
end
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
let(:a_packages_names) { graphql_data_at(:a, :packages, :nodes, :name) }
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
it 'returns an error for the second package and data for the first' do
post_graphql(batch_query, current_user: user)
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
expect(graphql_data_at(:a, :name)).to eq(composer_package.name)
2022-01-26 12:08:38 +05:30
2022-10-11 01:57:18 +05:30
expect_graphql_errors_to_include [/"package" field can be requested only for 1 Query\(s\) at a time./]
2022-03-02 08:16:31 +05:30
expect(graphql_data_at(:b)).to be(nil)
end
2022-01-26 12:08:38 +05:30
end
2023-04-23 21:23:45 +05:30
context 'versions field', :aggregate_failures do
let_it_be(:composer_package2) { create(:composer_package, project: project, name: composer_package.name) }
let_it_be(:composer_package3) { create(:composer_package, :error, project: project, name: composer_package.name) }
let_it_be(:pending_destruction) { create(:composer_package, :pending_destruction, project: project, name: composer_package.name) }
def run_query
versions_nodes = <<~QUERY
nodes { id }
QUERY
query = graphql_query_for(:package, { id: package_global_id }, query_graphql_field("versions", {}, versions_nodes))
post_graphql(query, current_user: user)
end
it 'returns other versions' do
run_query
versions_ids = graphql_data.dig('package', 'versions', 'nodes').pluck('id')
expected_ids = [composer_package2, composer_package3].map(&:to_gid).map(&:to_s)
expect(versions_ids).to contain_exactly(*expected_ids)
end
end
2022-03-02 08:16:31 +05:30
context 'pipelines field', :aggregate_failures do
let(:pipelines) { create_list(:ci_pipeline, 6, project: project) }
let(:pipeline_gids) { pipelines.sort_by(&:id).map(&:to_gid).map(&:to_s).reverse }
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
before do
2022-04-04 11:22:00 +05:30
pipelines.each do |pipeline|
create(:package_build_info, package: composer_package, pipeline: pipeline)
end
2022-03-02 08:16:31 +05:30
end
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
def run_query(args)
pipelines_nodes = <<~QUERY
nodes {
id
}
pageInfo {
startCursor
endCursor
}
QUERY
query = graphql_query_for(:package, { id: package_global_id }, query_graphql_field("pipelines", args, pipelines_nodes))
post_graphql(query, current_user: user)
end
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
it 'loads the second page with pagination first correctly' do
run_query(first: 2)
pipeline_ids = graphql_data.dig('package', 'pipelines', 'nodes').pluck('id')
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
expect(pipeline_ids).to eq(pipeline_gids[0..1])
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
cursor = graphql_data.dig('package', 'pipelines', 'pageInfo', 'endCursor')
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
run_query(first: 2, after: cursor)
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
pipeline_ids = graphql_data.dig('package', 'pipelines', 'nodes').pluck('id')
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
expect(pipeline_ids).to eq(pipeline_gids[2..3])
end
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
it 'loads the second page with pagination last correctly' do
run_query(last: 2)
pipeline_ids = graphql_data.dig('package', 'pipelines', 'nodes').pluck('id')
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
expect(pipeline_ids).to eq(pipeline_gids[4..5])
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
cursor = graphql_data.dig('package', 'pipelines', 'pageInfo', 'startCursor')
run_query(last: 2, before: cursor)
pipeline_ids = graphql_data.dig('package', 'pipelines', 'nodes').pluck('id')
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
expect(pipeline_ids).to eq(pipeline_gids[2..3])
end
end
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
context 'package managers paths' do
2022-01-26 12:08:38 +05:30
before do
2022-03-02 08:16:31 +05:30
subject
2022-01-26 12:08:38 +05:30
end
2022-03-02 08:16:31 +05:30
it 'returns npm_url correctly' do
expect(graphql_data_at(:package, :npm_url)).to eq("http://localhost/api/v4/projects/#{project.id}/packages/npm")
end
it 'returns maven_url correctly' do
expect(graphql_data_at(:package, :maven_url)).to eq("http://localhost/api/v4/projects/#{project.id}/packages/maven")
end
it 'returns conan_url correctly' do
expect(graphql_data_at(:package, :conan_url)).to eq("http://localhost/api/v4/projects/#{project.id}/packages/conan")
end
it 'returns nuget_url correctly' do
expect(graphql_data_at(:package, :nuget_url)).to eq("http://localhost/api/v4/projects/#{project.id}/packages/nuget/index.json")
end
it 'returns pypi_url correctly' do
expect(graphql_data_at(:package, :pypi_url)).to eq("http://__token__:<your_personal_token>@localhost/api/v4/projects/#{project.id}/packages/pypi/simple")
end
it 'returns pypi_setup_url correctly' do
expect(graphql_data_at(:package, :pypi_setup_url)).to eq("http://localhost/api/v4/projects/#{project.id}/packages/pypi")
end
it 'returns composer_url correctly' do
expect(graphql_data_at(:package, :composer_url)).to eq("http://localhost/api/v4/group/#{group.id}/-/packages/composer/packages.json")
end
2022-01-26 12:08:38 +05:30
2022-03-02 08:16:31 +05:30
it 'returns composer_config_repository_url correctly' do
expect(graphql_data_at(:package, :composer_config_repository_url)).to eq("localhost/#{group.id}")
2022-01-26 12:08:38 +05:30
end
2023-05-27 22:25:52 +05:30
context 'with access to package registry for everyone' do
before do
project.project_feature.update!(package_registry_access_level: ProjectFeature::PUBLIC)
subject
end
it 'returns pypi_url correctly' do
expect(graphql_data_at(:package, :pypi_url)).to eq("http://__token__:<your_personal_token>@localhost/api/v4/projects/#{project.id}/packages/pypi/simple")
end
end
context 'when project is public' do
let_it_be(:public_project) { create(:project, :public, group: group) }
let_it_be(:composer_package) { create(:composer_package, project: public_project) }
let(:package_global_id) { global_id_of(composer_package) }
before do
subject
end
it 'returns pypi_url correctly' do
expect(graphql_data_at(:package, :pypi_url)).to eq("http://localhost/api/v4/projects/#{public_project.id}/packages/pypi/simple")
end
end
2022-01-26 12:08:38 +05:30
end
2023-01-13 00:05:48 +05:30
context 'web_path' do
before do
subject
end
it 'returns web_path correctly' do
expect(graphql_data_at(:package, :_links, :web_path)).to eq("/#{project.full_path}/-/packages/#{composer_package.id}")
end
context 'with terraform module' do
let_it_be(:terraform_package) { create(:terraform_module_package, project: project) }
let(:package_global_id) { global_id_of(terraform_package) }
it 'returns web_path correctly' do
expect(graphql_data_at(:package, :_links, :web_path)).to eq("/#{project.full_path}/-/infrastructure_registry/#{terraform_package.id}")
end
end
end
2023-03-04 22:38:38 +05:30
2023-04-23 21:23:45 +05:30
context 'public_package' do
context 'when project is private' do
let_it_be(:private_project) { create(:project, :private, group: group) }
let_it_be(:composer_package) { create(:composer_package, project: private_project) }
let(:package_global_id) { global_id_of(composer_package) }
before do
private_project.add_developer(user)
end
it 'returns false' do
subject
expect(graphql_data_at(:package, :public_package)).to eq(false)
end
context 'with access to package registry for everyone' do
before do
private_project.project_feature.update!(package_registry_access_level: ProjectFeature::PUBLIC)
subject
end
it 'returns true' do
expect(graphql_data_at(:package, :public_package)).to eq(true)
end
end
end
context 'when project is public' do
let_it_be(:public_project) { create(:project, :public, group: group) }
let_it_be(:composer_package) { create(:composer_package, project: public_project) }
let(:package_global_id) { global_id_of(composer_package) }
before do
subject
end
it 'returns true' do
expect(graphql_data_at(:package, :public_package)).to eq(true)
end
end
end
2023-03-04 22:38:38 +05:30
context 'with package that has no default status' do
before do
composer_package.update!(status: :error)
subject
end
it "does not return package's details" do
expect(package_details).to be_nil
end
end
2022-01-26 12:08:38 +05:30
end
2021-03-11 19:13:27 +05:30
end