# frozen_string_literal: true require 'spec_helper' RSpec.describe 'getting container repositories in a project' do using RSpec::Parameterized::TableSyntax include GraphqlHelpers let_it_be_with_reload(:project) { create(:project, :private) } let_it_be(:container_repository) { create(:container_repository, project: project) } let_it_be(:container_repositories_delete_scheduled) { create_list(:container_repository, 2, :status_delete_scheduled, project: project) } let_it_be(:container_repositories_delete_failed) { create_list(:container_repository, 2, :status_delete_failed, project: project) } let_it_be(:container_repositories) { [container_repository, container_repositories_delete_scheduled, container_repositories_delete_failed].flatten } let_it_be(:container_expiration_policy) { project.container_expiration_policy } let(:excluded_fields) { %w[pipeline jobs] } let(:container_repositories_fields) do <<~GQL edges { node { #{all_graphql_fields_for('container_repositories'.classify, excluded: excluded_fields)} } } GQL end let(:fields) do <<~GQL #{query_graphql_field('container_repositories', {}, container_repositories_fields)} containerRepositoriesCount GQL end let(:query) do graphql_query_for( 'project', { 'fullPath' => project.full_path }, fields ) end let(:user) { project.owner } let(:variables) { {} } let(:container_repositories_response) { graphql_data.dig('project', 'containerRepositories', 'edges') } let(:container_repositories_count_response) { graphql_data.dig('project', 'containerRepositoriesCount') } before do stub_container_registry_config(enabled: true) container_repositories.each do |repository| stub_container_registry_tags(repository: repository.path, tags: %w(tag1 tag2 tag3), with_manifest: false) end end subject { post_graphql(query, current_user: user, variables: variables) } it_behaves_like 'a working graphql query' do before do subject end it 'matches the JSON schema' do expect(container_repositories_response).to match_schema('graphql/container_repositories') end end context 'with different permissions' do let_it_be(:user) { create(:user) } where(:project_visibility, :role, :access_granted, :can_delete) do :private | :maintainer | true | true :private | :developer | true | true :private | :reporter | true | false :private | :guest | false | false :private | :anonymous | false | false :public | :maintainer | true | true :public | :developer | true | true :public | :reporter | true | false :public | :guest | true | false :public | :anonymous | true | false end with_them do before do project.update!(visibility_level: Gitlab::VisibilityLevel.const_get(project_visibility.to_s.upcase, false)) project.add_user(user, role) unless role == :anonymous end it 'return the proper response' do subject if access_granted expect(container_repositories_response.size).to eq(container_repositories.size) container_repositories_response.each do |repository_response| expect(repository_response.dig('node', 'canDelete')).to eq(can_delete) end else expect(container_repositories_response).to eq(nil) end end end end context 'limiting the number of repositories' do let(:limit) { 1 } let(:variables) do { path: project.full_path, n: limit } end let(:query) do <<~GQL query($path: ID!, $n: Int) { project(fullPath: $path) { containerRepositories(first: $n) { #{container_repositories_fields} } } } GQL end it 'only returns N repositories' do subject expect(container_repositories_response.size).to eq(limit) end end context 'filter by name' do let_it_be(:container_repository) { create(:container_repository, name: 'fooBar', project: project) } let(:name) { 'ooba' } let(:query) do <<~GQL query($path: ID!, $name: String) { project(fullPath: $path) { containerRepositories(name: $name) { #{container_repositories_fields} } } } GQL end let(:variables) do { path: project.full_path, name: name } end before do stub_container_registry_tags(repository: container_repository.path, tags: %w(tag4 tag5 tag6), with_manifest: false) end it 'returns the searched container repository' do subject expect(container_repositories_response.size).to eq(1) expect(container_repositories_response.first.dig('node', 'id')).to eq(container_repository.to_global_id.to_s) end end it_behaves_like 'handling graphql network errors with the container registry' it_behaves_like 'not hitting graphql network errors with the container registry' do let(:excluded_fields) { %w[pipeline jobs tags tagsCount] } end it 'returns the total count of container repositories' do subject expect(container_repositories_count_response).to eq(container_repositories.size) end describe 'sorting and pagination' do let_it_be(:data_path) { [:project, :container_repositories] } let_it_be(:sort_project) { create(:project, :public) } let_it_be(:current_user) { create(:user) } let_it_be(:container_repository1) { create(:container_repository, name: 'b', project: sort_project) } let_it_be(:container_repository2) { create(:container_repository, name: 'a', project: sort_project) } let_it_be(:container_repository3) { create(:container_repository, name: 'd', project: sort_project) } let_it_be(:container_repository4) { create(:container_repository, name: 'c', project: sort_project) } let_it_be(:container_repository5) { create(:container_repository, name: 'e', project: sort_project) } before do stub_container_registry_tags(repository: container_repository1.path, tags: %w(tag1 tag1 tag3), with_manifest: false) stub_container_registry_tags(repository: container_repository2.path, tags: %w(tag4 tag5 tag6), with_manifest: false) stub_container_registry_tags(repository: container_repository3.path, tags: %w(tag7 tag8), with_manifest: false) stub_container_registry_tags(repository: container_repository4.path, tags: %w(tag9), with_manifest: false) stub_container_registry_tags(repository: container_repository5.path, tags: %w(tag10 tag11), with_manifest: false) end def pagination_query(params) graphql_query_for(:project, { full_path: sort_project.full_path }, query_nodes(:container_repositories, :name, include_pagination_info: true, args: params) ) end def pagination_results_data(data) data.map { |container_repository| container_repository.dig('name') } end context 'when sorting by name' do context 'when ascending' do it_behaves_like 'sorted paginated query' do let(:sort_param) { :NAME_ASC } let(:first_param) { 2 } let(:all_records) { [container_repository2.name, container_repository1.name, container_repository4.name, container_repository3.name, container_repository5.name] } end end context 'when descending' do it_behaves_like 'sorted paginated query' do let(:sort_param) { :NAME_DESC } let(:first_param) { 2 } let(:all_records) { [container_repository5.name, container_repository3.name, container_repository4.name, container_repository1.name, container_repository2.name] } end end end end end