# frozen_string_literal: true require 'spec_helper' RSpec.describe 'getting project information', feature_category: :projects do include GraphqlHelpers let_it_be(:group) { create(:group) } let_it_be(:project, reload: true) { create(:project, :repository, group: group) } let_it_be(:current_user) { create(:user) } let(:project_fields) { all_graphql_fields_for('project'.to_s.classify, max_depth: 1) } let(:query) do graphql_query_for(:project, { full_path: project.full_path }, project_fields) end context 'when the user has full access to the project' do before do project.add_maintainer(current_user) end it 'includes the project', :use_clean_rails_memory_store_caching, :request_store do post_graphql(query, current_user: current_user) expect(graphql_data['project']).not_to be_nil end end context 'when the user has access to the project', :use_clean_rails_memory_store_caching, :request_store do before_all do project.add_developer(current_user) end it 'includes the project' do post_graphql(query, current_user: current_user) expect(graphql_data['project']).not_to be_nil end it_behaves_like 'a working graphql query' do before do post_graphql(query, current_user: current_user) end end context 'when there are pipelines present' do let(:project_fields) { query_nodes(:pipelines) } before do create(:ci_pipeline, project: project) end it 'is included in the pipelines connection' do post_graphql(query, current_user: current_user) expect(graphql_data_at(:project, :pipelines, :nodes)).to contain_exactly(a_kind_of(Hash)) end end context 'topics' do it 'includes empty topics array if no topics set' do post_graphql(query, current_user: current_user) expect(graphql_data_at(:project, :topics)).to match([]) end it 'includes topics array' do project.update!(topic_list: 'topic1, topic2, topic3') post_graphql(query, current_user: current_user) expect(graphql_data_at(:project, :topics)).to match(%w[topic1 topic2 topic3]) end end it 'includes inherited members in project_members' do group_member = create(:group_member, group: group) project_member = create(:project_member, project: project) member_query = <<~GQL query { project(fullPath: "#{project.full_path}") { projectMembers { nodes { id user { username } ... on ProjectMember { project { id } } ... on GroupMember { group { id } } } } } } GQL post_graphql(member_query, current_user: current_user) member_ids = graphql_data.dig('project', 'projectMembers', 'nodes') expect(member_ids).to include( a_hash_including( 'id' => group_member.to_global_id.to_s, 'group' => { 'id' => group.to_global_id.to_s } ) ) expect(member_ids).to include( a_hash_including( 'id' => project_member.to_global_id.to_s, 'project' => { 'id' => project.to_global_id.to_s } ) ) end end describe 'is_catalog_resource' do before do project.add_owner(current_user) end let(:catalog_resource_query) do <<~GRAPHQL { project(fullPath: "#{project.full_path}") { isCatalogResource } } GRAPHQL end context 'when the project is not a catalog resource' do it 'is false' do post_graphql(catalog_resource_query, current_user: current_user) expect(graphql_data.dig('project', 'isCatalogResource')).to be(false) end end context 'when the project is a catalog resource' do before do create(:catalog_resource, project: project) end it 'is true' do post_graphql(catalog_resource_query, current_user: current_user) expect(graphql_data.dig('project', 'isCatalogResource')).to be(true) end end context 'for N+1 queries with isCatalogResource' do let_it_be(:project1) { create(:project, group: group) } let_it_be(:project2) { create(:project, group: group) } it 'avoids N+1 database queries' do pending('See: https://gitlab.com/gitlab-org/gitlab/-/issues/403634') ctx = { current_user: current_user } baseline_query = graphql_query_for(:project, { full_path: project1.full_path }, 'isCatalogResource') query = <<~GQL query { a: #{query_graphql_field(:project, { full_path: project1.full_path }, 'isCatalogResource')} b: #{query_graphql_field(:project, { full_path: project2.full_path }, 'isCatalogResource')} } GQL control = ActiveRecord::QueryRecorder.new do run_with_clean_state(baseline_query, context: ctx) end expect { run_with_clean_state(query, context: ctx) }.not_to exceed_query_limit(control) end end end context 'when the user has reporter access to the project' do let(:statistics_query) do <<~GRAPHQL { project(fullPath: "#{project.full_path}") { statistics { wikiSize } } } GRAPHQL end before do project.add_reporter(current_user) create(:project_statistics, project: project, wiki_size: 100) end it 'allows fetching project statistics' do post_graphql(statistics_query, current_user: current_user) expect(graphql_data.dig('project', 'statistics')).to include('wikiSize' => 100.0) end end context 'when the user has guest access' do context 'when the project has public pipelines' do before do pipeline = create(:ci_pipeline, project: project) create(:ci_build, project: project, pipeline: pipeline, name: 'a test job') project.add_guest(current_user) end it 'shows all jobs' do query = <<~GQL query { project(fullPath: "#{project.full_path}") { jobs { nodes { name stage { name } } } } } GQL post_graphql(query, current_user: current_user) expect(graphql_data_at(:project, :jobs, :nodes)).to contain_exactly({ 'name' => 'a test job', 'stage' => { 'name' => 'test' } }) end end end context 'when the user does not have access to the project' do it 'returns an empty field' do post_graphql(query, current_user: current_user) expect(graphql_data['project']).to be_nil end it_behaves_like 'a working graphql query' do before do post_graphql(query, current_user: current_user) end end end context 'with timelog categories' do let_it_be(:timelog_category) do create(:timelog_category, namespace: project.project_namespace, name: 'TimelogCategoryTest') end let(:project_fields) do <<~GQL timelogCategories { nodes { #{all_graphql_fields_for('TimeTrackingTimelogCategory')} } } GQL end context 'when user is guest and the project is public' do before do project.update!(visibility_level: Gitlab::VisibilityLevel::PUBLIC) end it 'includes empty timelog categories array' do post_graphql(query, current_user: current_user) expect(graphql_data_at(:project, :timelogCategories, :nodes)).to match([]) end end context 'when user has reporter role' do before do project.add_reporter(current_user) end it 'returns the timelog category with all its fields' do post_graphql(query, current_user: current_user) expect(graphql_data_at(:project, :timelogCategories, :nodes)) .to contain_exactly(a_graphql_entity_for(timelog_category)) end context 'when timelog_categories flag is disabled' do before do stub_feature_flags(timelog_categories: false) end it 'returns no timelog categories' do post_graphql(query, current_user: current_user) expect(graphql_data_at(:project, :timelogCategories)).to be_nil end end end context 'for N+1 queries' do let!(:project1) { create(:project) } let!(:project2) { create(:project) } before do project1.add_reporter(current_user) project2.add_reporter(current_user) end it 'avoids N+1 database queries' do pending('See: https://gitlab.com/gitlab-org/gitlab/-/issues/369396') ctx = { current_user: current_user } baseline_query = <<~GQL query { a: project(fullPath: "#{project1.full_path}") { ... p } } fragment p on Project { timelogCategories { nodes { name } } } GQL query = <<~GQL query { a: project(fullPath: "#{project1.full_path}") { ... p } b: project(fullPath: "#{project2.full_path}") { ... p } } fragment p on Project { timelogCategories { nodes { name } } } GQL control = ActiveRecord::QueryRecorder.new do run_with_clean_state(baseline_query, context: ctx) end expect { run_with_clean_state(query, context: ctx) }.not_to exceed_query_limit(control) end end end end