mirror of
https://gitlab.com/gitlab-org/gitlab-foss.git
synced 2025-07-25 16:03:48 +00:00
260 lines
9.4 KiB
Ruby
260 lines
9.4 KiB
Ruby
# 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
|