168 lines
5.4 KiB
Ruby
168 lines
5.4 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'spec_helper'
|
|
|
|
RSpec.describe 'getting a collection of projects', feature_category: :source_code_management do
|
|
include GraphqlHelpers
|
|
|
|
let_it_be(:current_user) { create(:user) }
|
|
let_it_be(:group) { create(:group, name: 'public-group', developers: current_user) }
|
|
let_it_be(:projects) { create_list(:project, 5, :public, group: group) }
|
|
let_it_be(:other_project) { create(:project, :public, group: group) }
|
|
let_it_be(:archived_project) { create(:project, :archived, group: group) }
|
|
|
|
let(:filters) { {} }
|
|
|
|
let(:query) do
|
|
graphql_query_for(
|
|
:projects,
|
|
filters,
|
|
"nodes {#{all_graphql_fields_for('Project', max_depth: 1, excluded: ['productAnalyticsState'])} }"
|
|
)
|
|
end
|
|
|
|
context 'when archived argument is ONLY' do
|
|
let(:filters) { { archived: :ONLY } }
|
|
|
|
it 'returns only archived projects' do
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
expect(graphql_data_at(:projects, :nodes))
|
|
.to contain_exactly(a_graphql_entity_for(archived_project))
|
|
end
|
|
end
|
|
|
|
context 'when archived argument is INCLUDE' do
|
|
let(:filters) { { archived: :INCLUDE } }
|
|
|
|
it 'returns archived and non-archived projects' do
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
expect(graphql_data_at(:projects, :nodes))
|
|
.to contain_exactly(
|
|
*projects.map { |project| a_graphql_entity_for(project) },
|
|
a_graphql_entity_for(other_project),
|
|
a_graphql_entity_for(archived_project)
|
|
)
|
|
end
|
|
end
|
|
|
|
context 'when archived argument is EXCLUDE' do
|
|
let(:filters) { { archived: :EXCLUDE } }
|
|
|
|
it 'returns only non-archived projects' do
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
expect(graphql_data_at(:projects, :nodes))
|
|
.to contain_exactly(
|
|
*projects.map { |project| a_graphql_entity_for(project) },
|
|
a_graphql_entity_for(other_project)
|
|
)
|
|
end
|
|
end
|
|
|
|
describe 'min_access_level' do
|
|
let_it_be(:project_with_owner_access) { create(:project, :private) }
|
|
|
|
before_all do
|
|
project_with_owner_access.add_owner(current_user)
|
|
end
|
|
|
|
context 'when min_access_level is OWNER' do
|
|
let(:filters) { { min_access_level: :OWNER } }
|
|
|
|
it 'returns only projects user has owner access to' do
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
expect(graphql_data_at(:projects, :nodes))
|
|
.to contain_exactly(a_graphql_entity_for(project_with_owner_access))
|
|
end
|
|
end
|
|
|
|
context 'when min_access_level is DEVELOPER' do
|
|
let(:filters) { { min_access_level: :DEVELOPER } }
|
|
|
|
it 'returns only projects user has developer or higher access to' do
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
expect(graphql_data_at(:projects, :nodes))
|
|
.to contain_exactly(
|
|
*projects.map { |project| a_graphql_entity_for(project) },
|
|
a_graphql_entity_for(other_project),
|
|
a_graphql_entity_for(project_with_owner_access)
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when providing full_paths filter' do
|
|
let(:project_full_paths) { projects.map(&:full_path) }
|
|
let(:filters) { { full_paths: project_full_paths } }
|
|
|
|
let(:single_project_query) do
|
|
graphql_query_for(
|
|
:projects,
|
|
{ full_paths: [project_full_paths.first] },
|
|
"nodes {#{all_graphql_fields_for('Project', max_depth: 1, excluded: ['productAnalyticsState'])} }"
|
|
)
|
|
end
|
|
|
|
it_behaves_like 'a working graphql query that returns data' do
|
|
before do
|
|
post_graphql(query, current_user: current_user)
|
|
end
|
|
end
|
|
|
|
it 'avoids N+1 queries', :use_sql_query_cache, :clean_gitlab_redis_cache do
|
|
post_graphql(single_project_query, current_user: current_user)
|
|
|
|
control = ActiveRecord::QueryRecorder.new do
|
|
post_graphql(single_project_query, current_user: current_user)
|
|
end
|
|
|
|
# There is an N+1 query for max_member_access_for_user_ids
|
|
# There is an N+1 query for duo_features_enabled cascading setting
|
|
# https://gitlab.com/gitlab-org/gitlab/-/issues/442164
|
|
expect do
|
|
post_graphql(query, current_user: current_user)
|
|
end.not_to exceed_all_query_limit(control).with_threshold(17)
|
|
end
|
|
|
|
it 'returns the expected projects' do
|
|
post_graphql(query, current_user: current_user)
|
|
returned_full_paths = graphql_data_at(:projects, :nodes).pluck('fullPath')
|
|
|
|
expect(returned_full_paths).to match_array(project_full_paths)
|
|
end
|
|
|
|
context 'when users provides more than 50 full_paths' do
|
|
let(:filters) { { full_paths: Array.new(51) { other_project.full_path } } }
|
|
|
|
it 'returns an error' do
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
expect(graphql_errors).to contain_exactly(
|
|
hash_including('message' => _('You cannot provide more than 50 full_paths'))
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when providing the programming_language_name argument' do
|
|
let_it_be(:project) { projects.first }
|
|
let_it_be(:ruby) { create(:programming_language, name: 'Ruby') }
|
|
let_it_be(:repository_language) do
|
|
create(:repository_language, project: project, programming_language: ruby, share: 1)
|
|
end
|
|
|
|
let(:filters) { { programming_language_name: 'ruby' } }
|
|
|
|
it 'returns the expected projects' do
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
expect(graphql_data_at(:projects, :nodes))
|
|
.to contain_exactly(a_graphql_entity_for(project))
|
|
end
|
|
end
|
|
end
|