# frozen_string_literal: true require 'spec_helper' RSpec.describe Resolvers::NamespaceProjectsResolver, feature_category: :groups_and_projects do include GraphqlHelpers let_it_be(:current_user) { create(:user) } let_it_be(:group) { create(:group) } let(:ctx) { { current_user: current_user } } let(:args) { default_args } let(:default_args) do { include_subgroups: false, include_archived: true, not_aimed_for_deletion: false, search: nil, sort: nil, ids: nil, with_issues_enabled: nil, with_merge_requests_enabled: nil, with_namespace_domain_pages: nil } end describe '#resolve' do subject { resolve(described_class, obj: namespace, args: args, ctx: ctx, arg_style: :internal) } let_it_be(:user_namespaced_projects) { create_list(:project, 3, namespace: current_user.namespace) } let_it_be(:group_namespaced_projects) do [ create(:project, name: 'Project', path: 'project', namespace: group), create(:project, :archived, name: 'Test Project', path: 'test-project', namespace: group), create(:project, name: 'Test', path: 'test', namespace: group, marked_for_deletion_at: 1.day.ago, pending_delete: true) ] end let_it_be(:nested_group_projects) do create(:group, parent: group).then do |nested_group| [ create(:project, group: nested_group), create(:project, group: nested_group, marked_for_deletion_at: 1.day.ago, pending_delete: true) ] end end before_all do group_namespaced_projects.each { |p| p.add_developer(current_user) } nested_group_projects.each { |p| p.add_developer(current_user) } end context "with a group" do let(:namespace) { group } let(:expected_projects) { group_namespaced_projects } it { is_expected.to contain_exactly(*expected_projects) } context 'when include_subgroups is true' do let(:args) { default_args.merge(include_subgroups: true) } let(:expected_projects) { group_namespaced_projects + nested_group_projects } it { is_expected.to contain_exactly(*expected_projects) } end context 'when not_aimed_for_deletion is true' do let(:args) { default_args.merge(not_aimed_for_deletion: true, include_subgroups: true) } let(:expected_projects) { group_namespaced_projects.first(2) << nested_group_projects.first } it { is_expected.to contain_exactly(*expected_projects) } end context 'when include_archived is false' do let(:args) { default_args.merge(include_archived: false) } let(:expected_projects) { group_namespaced_projects - [group_namespaced_projects.second] } it { is_expected.to contain_exactly(*expected_projects) } end context 'search and similarity sorting' do let(:project_names) { subject.map { |project| project['name'] } } let(:args) { default_args.merge(sort: :similarity, search: 'test') } it 'returns projects ordered by similarity to the search input' do expect(project_names.first).to eq('Test') expect(project_names.second).to eq('Test Project') end it 'filters out result that do not match the search input' do expect(project_names).not_to include('Project') end context 'when `search` parameter is not given' do let(:args) { default_args.merge(sort: :similarity, search: nil) } it 'returns all projects' do is_expected.to match_array(group_namespaced_projects) end end context 'when only search term is given' do let(:args) { default_args.merge(sort: nil, search: 'test') } it 'filters out result that do not match the search input, and applies default similarity sort' do expect(project_names.first).to eq('Test') expect(project_names.second).to eq('Test Project') end end end context 'path sorting' do let(:project_paths) { subject.map { |project| project['path'] } } let(:args) { default_args.merge(sort: :path_asc) } it 'returns projects sorted by path' do expect(project_paths.first).to eq('project') expect(project_paths.second).to eq('test') expect(project_paths.third).to eq('test-project') end context 'when sorting by path in descending order' do let(:args) { default_args.merge(sort: :path_desc) } it 'returns projects sorted by path in descending order' do expect(project_paths.first).to eq('test-project') expect(project_paths.second).to eq('test') expect(project_paths.third).to eq('project') end end end context 'full path sorting' do let_it_be(:parent_group) { create(:group) } let_it_be(:nested_group_1) { create(:group, parent: parent_group, path: 'alpha') } let_it_be(:nested_group_2) { create(:group, parent: parent_group, path: 'beta') } let_it_be(:deeply_nested_group) { create(:group, parent: nested_group_1, path: 'gamma') } let_it_be(:projects_with_various_paths) do [ create(:project, path: 'zebra', namespace: parent_group), create(:project, path: 'apple', namespace: nested_group_1), create(:project, path: 'banana', namespace: nested_group_2), create(:project, path: 'cherry', namespace: deeply_nested_group) ] end let(:namespace) { parent_group } let(:args) { default_args.merge(include_subgroups: true, sort: :full_path_asc) } let(:project_full_paths) { subject.map(&:full_path) } before_all do projects_with_various_paths.each { |p| p.add_developer(current_user) } end it 'returns projects sorted by full path in ascending order' do expect(project_full_paths).to eq([ "#{parent_group.path}/alpha/apple", "#{parent_group.path}/alpha/gamma/cherry", "#{parent_group.path}/beta/banana", "#{parent_group.path}/zebra" ]) end context 'when sorting by full path in descending order' do let(:args) { default_args.merge(include_subgroups: true, sort: :full_path_desc) } it 'returns projects sorted by full path in descending order' do expect(project_full_paths).to eq([ "#{parent_group.path}/zebra", "#{parent_group.path}/beta/banana", "#{parent_group.path}/alpha/gamma/cherry", "#{parent_group.path}/alpha/apple" ]) end end end context 'ids filtering' do let(:args) { default_args.merge(include_subgroups: false) } context 'when ids is provided' do let(:args) { super().merge(ids: [group_namespaced_projects.last.to_global_id.to_s]) } it { is_expected.to contain_exactly(group_namespaced_projects.last) } end context 'when ids is nil' do let(:args) { super().merge(ids: nil) } it { is_expected.to contain_exactly(*group_namespaced_projects) } end end context 'with_namespace_domain_pages' do before do group_namespaced_projects[0...-1].each do |project| project.project_setting.update!(pages_unique_domain_enabled: false) end group_namespaced_projects.last.project_setting.update!( pages_unique_domain_enabled: true, pages_unique_domain: 'foo123.example.com' ) end let(:args) { default_args.merge(with_namespace_domain_pages: true) } let(:expected_projects) { group_namespaced_projects[0...-1] } it { is_expected.to contain_exactly(*expected_projects) } end context 'archived_only argument' do context 'when archived_only is true' do let(:args) { default_args.merge(archived_only: true) } let(:expected_projects) { group_namespaced_projects.second } it { is_expected.to contain_exactly(*expected_projects) } end context 'when archived_only is false' do let(:args) { default_args.merge(archived_only: false) } let(:expected_projects) { group_namespaced_projects } it { is_expected.to contain_exactly(*expected_projects) } end context 'when archived_only is not specified' do let(:args) { default_args.merge(archived_only: nil) } let(:expected_projects) { group_namespaced_projects } it { is_expected.to contain_exactly(*expected_projects) } end end end context 'with an user namespace' do let(:namespace) { current_user.namespace } it { is_expected.to contain_exactly(*user_namespaced_projects) } end context "when passing a non existent, batch loaded namespace" do let(:namespace) do BatchLoader::GraphQL.for("non-existent-path").batch do |_fake_paths, loader, _| loader.call("non-existent-path", nil) end end it { is_expected.to be_empty } end end it 'has an high complexity regardless of arguments' do field = Types::BaseField.new(name: 'test', type: GraphQL::Types::String.connection_type, resolver_class: described_class, null: false, max_page_size: 100) expect(field.complexity.call({}, {}, 1)).to eq 24 expect(field.complexity.call({}, { include_subgroups: true }, 1)).to eq 24 end end