2019-12-26 22:10:19 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
RSpec.describe 'getting an issue list for a project' do
|
2019-02-15 15:39:39 +05:30
|
|
|
include GraphqlHelpers
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
let_it_be(:group) { create(:group) }
|
|
|
|
let_it_be(:project) { create(:project, :repository, :public, group: group) }
|
2020-11-24 15:15:51 +05:30
|
|
|
let_it_be(:current_user) { create(:user) }
|
2023-01-13 00:05:48 +05:30
|
|
|
let_it_be(:another_user) { create(:user).tap { |u| group.add_reporter(u) } }
|
|
|
|
let_it_be(:early_milestone) { create(:milestone, project: project, due_date: 10.days.from_now) }
|
|
|
|
let_it_be(:late_milestone) { create(:milestone, project: project, due_date: 30.days.from_now) }
|
|
|
|
let_it_be(:priority1) { create(:label, project: project, priority: 1) }
|
|
|
|
let_it_be(:priority2) { create(:label, project: project, priority: 5) }
|
|
|
|
let_it_be(:priority3) { create(:label, project: project, priority: 10) }
|
|
|
|
|
|
|
|
let_it_be(:issue_a, reload: true) { create(:issue, project: project, discussion_locked: true, labels: [priority3]) }
|
|
|
|
let_it_be(:issue_b, reload: true) { create(:issue, :with_alert, project: project, title: 'title matching issue i') }
|
|
|
|
let_it_be(:issue_c) { create(:issue, project: project, labels: [priority1], milestone: late_milestone) }
|
|
|
|
let_it_be(:issue_d) { create(:issue, project: project, labels: [priority2]) }
|
|
|
|
let_it_be(:issue_e) { create(:issue, project: project, milestone: early_milestone) }
|
|
|
|
let_it_be(:issues, reload: true) { [issue_a, issue_b, issue_c, issue_d, issue_e] }
|
2020-10-24 23:57:45 +05:30
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:issue_a_gid) { issue_a.to_global_id.to_s }
|
|
|
|
let(:issue_b_gid) { issue_b.to_global_id.to_s }
|
2023-01-13 00:05:48 +05:30
|
|
|
let(:issues_data) { graphql_data['project']['issues']['nodes'] }
|
2021-04-29 21:17:54 +05:30
|
|
|
let(:issue_filter_params) { {} }
|
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
let(:fields) do
|
|
|
|
<<~QUERY
|
2023-01-13 00:05:48 +05:30
|
|
|
nodes {
|
|
|
|
#{all_graphql_fields_for('issues'.classify)}
|
2019-02-15 15:39:39 +05:30
|
|
|
}
|
|
|
|
QUERY
|
|
|
|
end
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
# All new specs should be added to the shared example if the change also
|
|
|
|
# affects the `issues` query at the root level of the API.
|
|
|
|
# Shared example also used in spec/requests/api/graphql/issues_spec.rb
|
|
|
|
it_behaves_like 'graphql issue list request spec' do
|
|
|
|
subject(:post_query) { post_graphql(query, current_user: current_user) }
|
|
|
|
|
|
|
|
# filters
|
|
|
|
let(:expected_negated_assignee_issues) { [issue_b, issue_c, issue_d, issue_e] }
|
|
|
|
let(:expected_unioned_assignee_issues) { [issue_a, issue_b] }
|
|
|
|
let(:voted_issues) { [issue_a] }
|
|
|
|
let(:no_award_issues) { [issue_b, issue_c, issue_d, issue_e] }
|
|
|
|
let(:locked_discussion_issues) { [issue_a] }
|
|
|
|
let(:unlocked_discussion_issues) { [issue_b, issue_c, issue_d, issue_e] }
|
|
|
|
let(:search_title_term) { 'matching issue' }
|
|
|
|
let(:title_search_issue) { issue_b }
|
|
|
|
|
|
|
|
# sorting
|
|
|
|
let(:data_path) { [:project, :issues] }
|
|
|
|
let(:expected_severity_sorted_asc) { [issue_c, issue_a, issue_b, issue_e, issue_d] }
|
|
|
|
let(:expected_priority_sorted_asc) { [issue_e, issue_c, issue_d, issue_a, issue_b] }
|
|
|
|
let(:expected_priority_sorted_desc) { [issue_c, issue_e, issue_a, issue_d, issue_b] }
|
|
|
|
|
|
|
|
before_all do
|
|
|
|
issue_a.assignee_ids = current_user.id
|
|
|
|
issue_b.assignee_ids = another_user.id
|
|
|
|
|
|
|
|
create(:award_emoji, :upvote, user: current_user, awardable: issue_a)
|
|
|
|
|
|
|
|
# severity sorting
|
|
|
|
create(:issuable_severity, issue: issue_a, severity: :unknown)
|
|
|
|
create(:issuable_severity, issue: issue_b, severity: :low)
|
|
|
|
create(:issuable_severity, issue: issue_d, severity: :critical)
|
|
|
|
create(:issuable_severity, issue: issue_e, severity: :high)
|
2021-11-11 11:23:49 +05:30
|
|
|
end
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
def pagination_query(params)
|
|
|
|
graphql_query_for(
|
|
|
|
:project,
|
|
|
|
{ full_path: project.full_path },
|
|
|
|
query_graphql_field(:issues, params, "#{page_info} nodes { id }")
|
|
|
|
)
|
2022-10-11 01:57:18 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-07 11:18:12 +05:30
|
|
|
context 'when limiting the number of results' do
|
|
|
|
let(:query) do
|
2021-01-03 14:25:43 +05:30
|
|
|
<<~GQL
|
|
|
|
query($path: ID!, $n: Int) {
|
|
|
|
project(fullPath: $path) {
|
|
|
|
issues(first: $n) { #{fields} }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
GQL
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:issue_limit) { 1 }
|
|
|
|
let(:variables) do
|
|
|
|
{ path: project.full_path, n: issue_limit }
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'a working graphql query' do
|
|
|
|
before do
|
2021-01-03 14:25:43 +05:30
|
|
|
post_graphql(query, current_user: current_user, variables: variables)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'only returns N issues' do
|
|
|
|
expect(issues_data.size).to eq(issue_limit)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
context 'when no limit is provided' do
|
2021-01-03 14:25:43 +05:30
|
|
|
let(:issue_limit) { nil }
|
|
|
|
|
|
|
|
it 'returns all issues' do
|
|
|
|
post_graphql(query, current_user: current_user, variables: variables)
|
|
|
|
|
|
|
|
expect(issues_data.size).to be > 1
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
it 'is expected to check permissions on the first issue only' do
|
2019-07-07 11:18:12 +05:30
|
|
|
allow(Ability).to receive(:allowed?).and_call_original
|
|
|
|
# Newest first, we only want to see the newest checked
|
|
|
|
expect(Ability).not_to receive(:allowed?).with(current_user, :read_issue, issues.first)
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
post_graphql(query, current_user: current_user, variables: variables)
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
context 'when the user does not have access to the issue' do
|
|
|
|
it 'returns nil' do
|
|
|
|
project.project_feature.update!(issues_access_level: ProjectFeature::PRIVATE)
|
|
|
|
|
|
|
|
post_graphql(query)
|
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
expect(issues_data).to eq([])
|
2019-02-15 15:39:39 +05:30
|
|
|
end
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
|
|
|
context 'when there is a confidential issue' do
|
2020-11-24 15:15:51 +05:30
|
|
|
let_it_be(:confidential_issue) do
|
2019-07-07 11:18:12 +05:30
|
|
|
create(:issue, :confidential, project: project)
|
|
|
|
end
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:confidential_issue_gid) { confidential_issue.to_global_id.to_s }
|
|
|
|
|
2019-07-07 11:18:12 +05:30
|
|
|
context 'when the user cannot see confidential issues' do
|
|
|
|
it 'returns issues without confidential issues' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(issues_data.size).to eq(5)
|
2019-07-07 11:18:12 +05:30
|
|
|
|
|
|
|
issues_data.each do |issue|
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(issue['confidential']).to eq(false)
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
context 'filtering for confidential issues' do
|
|
|
|
let(:issue_filter_params) { { confidential: true } }
|
|
|
|
|
|
|
|
it 'returns no issues' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
|
|
|
expect(issues_data.size).to eq(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'filtering for non-confidential issues' do
|
|
|
|
let(:issue_filter_params) { { confidential: false } }
|
|
|
|
|
|
|
|
it 'returns correctly filtered issues' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(issue_ids).to match_array(issues.map { |i| i.to_gid.to_s })
|
2021-11-18 22:05:49 +05:30
|
|
|
end
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the user can see confidential issues' do
|
2021-11-18 22:05:49 +05:30
|
|
|
before do
|
2019-07-07 11:18:12 +05:30
|
|
|
project.add_developer(current_user)
|
2021-11-18 22:05:49 +05:30
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
it 'returns issues with confidential issues' do
|
2019-07-07 11:18:12 +05:30
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(issues_data.size).to eq(6)
|
2019-07-07 11:18:12 +05:30
|
|
|
|
|
|
|
confidentials = issues_data.map do |issue|
|
2023-01-13 00:05:48 +05:30
|
|
|
issue['confidential']
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(confidentials).to contain_exactly(true, false, false, false, false, false)
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
context 'filtering for confidential issues' do
|
|
|
|
let(:issue_filter_params) { { confidential: true } }
|
|
|
|
|
|
|
|
it 'returns correctly filtered issues' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(issue_ids).to contain_exactly(confidential_issue_gid)
|
2021-11-18 22:05:49 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'filtering for non-confidential issues' do
|
|
|
|
let(:issue_filter_params) { { confidential: false } }
|
|
|
|
|
|
|
|
it 'returns correctly filtered issues' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(issue_ids).to match_array([issue_a, issue_b, issue_c, issue_d, issue_e].map { |i| i.to_gid.to_s })
|
2021-11-18 22:05:49 +05:30
|
|
|
end
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
describe 'sorting and pagination' do
|
2022-08-13 15:12:31 +05:30
|
|
|
let_it_be(:sort_project) { create(:project, :public) }
|
2020-05-24 23:13:21 +05:30
|
|
|
let_it_be(:data_path) { [:project, :issues] }
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
def pagination_query(params)
|
2021-04-29 21:17:54 +05:30
|
|
|
graphql_query_for(
|
|
|
|
:project,
|
|
|
|
{ full_path: sort_project.full_path },
|
2021-02-22 17:27:13 +05:30
|
|
|
query_graphql_field(:issues, params, "#{page_info} nodes { iid }")
|
2020-05-24 23:13:21 +05:30
|
|
|
)
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
def pagination_results_data(data)
|
2021-04-29 21:17:54 +05:30
|
|
|
data.map { |issue| issue['iid'].to_i }
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
# rubocop:disable RSpec/MultipleMemoizedHelpers
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when sorting by due date' do
|
|
|
|
let_it_be(:due_issue1) { create(:issue, project: sort_project, due_date: 3.days.from_now) }
|
|
|
|
let_it_be(:due_issue2) { create(:issue, project: sort_project, due_date: nil) }
|
|
|
|
let_it_be(:due_issue3) { create(:issue, project: sort_project, due_date: 2.days.ago) }
|
|
|
|
let_it_be(:due_issue4) { create(:issue, project: sort_project, due_date: nil) }
|
|
|
|
let_it_be(:due_issue5) { create(:issue, project: sort_project, due_date: 1.day.ago) }
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
context 'when ascending' do
|
2020-05-24 23:13:21 +05:30
|
|
|
it_behaves_like 'sorted paginated query' do
|
2021-02-22 17:27:13 +05:30
|
|
|
let(:sort_param) { :DUE_DATE_ASC }
|
2020-05-24 23:13:21 +05:30
|
|
|
let(:first_param) { 2 }
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:all_records) { [due_issue3.iid, due_issue5.iid, due_issue1.iid, due_issue4.iid, due_issue2.iid] }
|
2019-12-26 22:10:19 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when descending' do
|
2020-05-24 23:13:21 +05:30
|
|
|
it_behaves_like 'sorted paginated query' do
|
2021-02-22 17:27:13 +05:30
|
|
|
let(:sort_param) { :DUE_DATE_DESC }
|
2020-05-24 23:13:21 +05:30
|
|
|
let(:first_param) { 2 }
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:all_records) { [due_issue1.iid, due_issue5.iid, due_issue3.iid, due_issue4.iid, due_issue2.iid] }
|
2019-12-26 22:10:19 +05:30
|
|
|
end
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when sorting by relative position' do
|
|
|
|
let_it_be(:relative_issue1) { create(:issue, project: sort_project, relative_position: 2000) }
|
|
|
|
let_it_be(:relative_issue2) { create(:issue, project: sort_project, relative_position: nil) }
|
|
|
|
let_it_be(:relative_issue3) { create(:issue, project: sort_project, relative_position: 1000) }
|
|
|
|
let_it_be(:relative_issue4) { create(:issue, project: sort_project, relative_position: nil) }
|
|
|
|
let_it_be(:relative_issue5) { create(:issue, project: sort_project, relative_position: 500) }
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when ascending' do
|
2022-10-11 01:57:18 +05:30
|
|
|
it_behaves_like 'sorted paginated query', is_reversible: true do
|
2021-02-22 17:27:13 +05:30
|
|
|
let(:sort_param) { :RELATIVE_POSITION_ASC }
|
2020-05-24 23:13:21 +05:30
|
|
|
let(:first_param) { 2 }
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:all_records) do
|
2021-04-29 21:17:54 +05:30
|
|
|
[
|
|
|
|
relative_issue5.iid, relative_issue3.iid, relative_issue1.iid,
|
2021-11-18 22:05:49 +05:30
|
|
|
relative_issue2.iid, relative_issue4.iid
|
2021-04-29 21:17:54 +05:30
|
|
|
]
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when sorting by label priority' do
|
|
|
|
let_it_be(:label1) { create(:label, project: sort_project, priority: 1) }
|
|
|
|
let_it_be(:label2) { create(:label, project: sort_project, priority: 5) }
|
|
|
|
let_it_be(:label3) { create(:label, project: sort_project, priority: 10) }
|
|
|
|
let_it_be(:label_issue1) { create(:issue, project: sort_project, labels: [label1]) }
|
|
|
|
let_it_be(:label_issue2) { create(:issue, project: sort_project, labels: [label2]) }
|
|
|
|
let_it_be(:label_issue3) { create(:issue, project: sort_project, labels: [label1, label3]) }
|
|
|
|
let_it_be(:label_issue4) { create(:issue, project: sort_project) }
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
context 'when ascending' do
|
2020-05-24 23:13:21 +05:30
|
|
|
it_behaves_like 'sorted paginated query' do
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:sort_param) { :LABEL_PRIORITY_ASC }
|
|
|
|
let(:first_param) { 2 }
|
|
|
|
let(:all_records) { [label_issue3.iid, label_issue1.iid, label_issue2.iid, label_issue4.iid] }
|
2019-12-26 22:10:19 +05:30
|
|
|
end
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when descending' do
|
|
|
|
it_behaves_like 'sorted paginated query' do
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:sort_param) { :LABEL_PRIORITY_DESC }
|
|
|
|
let(:first_param) { 2 }
|
|
|
|
let(:all_records) { [label_issue2.iid, label_issue3.iid, label_issue1.iid, label_issue4.iid] }
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2023-01-13 00:05:48 +05:30
|
|
|
# rubocop:enable RSpec/MultipleMemoizedHelpers
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when sorting by milestone due date' do
|
|
|
|
let_it_be(:early_milestone) { create(:milestone, project: sort_project, due_date: 10.days.from_now) }
|
|
|
|
let_it_be(:late_milestone) { create(:milestone, project: sort_project, due_date: 30.days.from_now) }
|
|
|
|
let_it_be(:milestone_issue1) { create(:issue, project: sort_project) }
|
|
|
|
let_it_be(:milestone_issue2) { create(:issue, project: sort_project, milestone: early_milestone) }
|
|
|
|
let_it_be(:milestone_issue3) { create(:issue, project: sort_project, milestone: late_milestone) }
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when ascending' do
|
|
|
|
it_behaves_like 'sorted paginated query' do
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:sort_param) { :MILESTONE_DUE_ASC }
|
|
|
|
let(:first_param) { 2 }
|
|
|
|
let(:all_records) { [milestone_issue2.iid, milestone_issue3.iid, milestone_issue1.iid] }
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when descending' do
|
|
|
|
it_behaves_like 'sorted paginated query' do
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:sort_param) { :MILESTONE_DUE_DESC }
|
|
|
|
let(:first_param) { 2 }
|
|
|
|
let(:all_records) { [milestone_issue3.iid, milestone_issue2.iid, milestone_issue1.iid] }
|
2019-12-26 22:10:19 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
context 'when fetching alert management alert' do
|
2020-11-24 15:15:51 +05:30
|
|
|
let(:fields) do
|
|
|
|
<<~QUERY
|
2023-01-13 00:05:48 +05:30
|
|
|
nodes {
|
2020-11-24 15:15:51 +05:30
|
|
|
iid
|
|
|
|
alertManagementAlert {
|
|
|
|
title
|
|
|
|
}
|
2023-01-13 00:05:48 +05:30
|
|
|
alertManagementAlerts {
|
|
|
|
nodes {
|
|
|
|
title
|
|
|
|
}
|
|
|
|
}
|
2020-11-24 15:15:51 +05:30
|
|
|
}
|
|
|
|
QUERY
|
|
|
|
end
|
|
|
|
|
|
|
|
# Alerts need to have developer permission and above
|
|
|
|
before do
|
|
|
|
project.add_developer(current_user)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'avoids N+1 queries' do
|
|
|
|
control = ActiveRecord::QueryRecorder.new { post_graphql(query, current_user: current_user) }
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
create(:alert_management_alert, :with_incident, project: project)
|
2020-11-24 15:15:51 +05:30
|
|
|
|
|
|
|
expect { post_graphql(query, current_user: current_user) }.not_to exceed_query_limit(control)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the alert data' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
alert_titles = issues_data.map { |issue| issue.dig('alertManagementAlert', 'title') }
|
2020-11-24 15:15:51 +05:30
|
|
|
expected_titles = issues.map { |issue| issue.alert_management_alert&.title }
|
|
|
|
|
|
|
|
expect(alert_titles).to contain_exactly(*expected_titles)
|
|
|
|
end
|
2023-01-13 00:05:48 +05:30
|
|
|
|
|
|
|
it 'returns the alerts data' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
|
|
|
alert_titles = issues_data.map { |issue| issue.dig('alertManagementAlerts', 'nodes') }
|
|
|
|
expected_titles = issues.map do |issue|
|
|
|
|
issue.alert_management_alerts.map { |alert| { 'title' => alert.title } }
|
|
|
|
end
|
|
|
|
|
|
|
|
expect(alert_titles).to contain_exactly(*expected_titles)
|
|
|
|
end
|
2020-11-24 15:15:51 +05:30
|
|
|
end
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
context 'when fetching customer_relations_contacts' do
|
|
|
|
let(:fields) do
|
|
|
|
<<~QUERY
|
|
|
|
nodes {
|
|
|
|
id
|
|
|
|
customerRelationsContacts {
|
|
|
|
nodes {
|
|
|
|
firstName
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
QUERY
|
|
|
|
end
|
|
|
|
|
|
|
|
def clean_state_query
|
|
|
|
run_with_clean_state(query, context: { current_user: current_user })
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'avoids N+1 queries' do
|
2021-12-11 22:18:48 +05:30
|
|
|
create(:issue_customer_relations_contact, :for_issue, issue: issue_a)
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
control = ActiveRecord::QueryRecorder.new(skip_cached: false) { clean_state_query }
|
|
|
|
|
2021-12-11 22:18:48 +05:30
|
|
|
create(:issue_customer_relations_contact, :for_issue, issue: issue_a)
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
expect { clean_state_query }.not_to exceed_all_query_limit(control)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
context 'when fetching labels' do
|
2020-11-24 15:15:51 +05:30
|
|
|
let(:fields) do
|
|
|
|
<<~QUERY
|
2023-01-13 00:05:48 +05:30
|
|
|
nodes {
|
|
|
|
id
|
|
|
|
labels {
|
|
|
|
nodes {
|
|
|
|
id
|
2020-11-24 15:15:51 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
QUERY
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
issues.each do |issue|
|
|
|
|
# create a label for each issue we have to properly test N+1
|
|
|
|
label = create(:label, project: project)
|
|
|
|
issue.update!(labels: [label])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def response_label_ids(response_data)
|
2023-01-13 00:05:48 +05:30
|
|
|
response_data.map do |node|
|
|
|
|
node['labels']['nodes'].map { |u| u['id'] }
|
2020-11-24 15:15:51 +05:30
|
|
|
end.flatten
|
|
|
|
end
|
|
|
|
|
|
|
|
def labels_as_global_ids(issues)
|
|
|
|
issues.map(&:labels).flatten.map(&:to_global_id).map(&:to_s)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'avoids N+1 queries', :aggregate_failures do
|
|
|
|
control = ActiveRecord::QueryRecorder.new { post_graphql(query, current_user: current_user) }
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(issues_data.count).to eq(5)
|
2020-11-24 15:15:51 +05:30
|
|
|
expect(response_label_ids(issues_data)).to match_array(labels_as_global_ids(issues))
|
|
|
|
|
|
|
|
new_issues = issues + [create(:issue, project: project, labels: [create(:label, project: project)])]
|
|
|
|
|
|
|
|
expect { post_graphql(query, current_user: current_user) }.not_to exceed_query_limit(control)
|
|
|
|
# graphql_data is memoized (see spec/support/helpers/graphql_helpers.rb)
|
|
|
|
# so we have to parse the body ourselves the second time
|
2023-01-13 00:05:48 +05:30
|
|
|
issues_data = Gitlab::Json.parse(response.body)['data']['project']['issues']['nodes']
|
|
|
|
expect(issues_data.count).to eq(6)
|
2020-11-24 15:15:51 +05:30
|
|
|
expect(response_label_ids(issues_data)).to match_array(labels_as_global_ids(new_issues))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
context 'when fetching assignees' do
|
2020-11-24 15:15:51 +05:30
|
|
|
let(:fields) do
|
|
|
|
<<~QUERY
|
2023-01-13 00:05:48 +05:30
|
|
|
nodes {
|
|
|
|
id
|
|
|
|
assignees {
|
|
|
|
nodes {
|
|
|
|
id
|
2020-11-24 15:15:51 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
QUERY
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
issues.each do |issue|
|
|
|
|
# create an assignee for each issue we have to properly test N+1
|
|
|
|
assignee = create(:user)
|
|
|
|
issue.update!(assignees: [assignee])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def response_assignee_ids(response_data)
|
2023-01-13 00:05:48 +05:30
|
|
|
response_data.map do |node|
|
|
|
|
node['assignees']['nodes'].map { |node| node['id'] }
|
2020-11-24 15:15:51 +05:30
|
|
|
end.flatten
|
|
|
|
end
|
|
|
|
|
|
|
|
def assignees_as_global_ids(issues)
|
|
|
|
issues.map(&:assignees).flatten.map(&:to_global_id).map(&:to_s)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'avoids N+1 queries', :aggregate_failures do
|
|
|
|
control = ActiveRecord::QueryRecorder.new { post_graphql(query, current_user: current_user) }
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(issues_data.count).to eq(5)
|
2020-11-24 15:15:51 +05:30
|
|
|
expect(response_assignee_ids(issues_data)).to match_array(assignees_as_global_ids(issues))
|
|
|
|
|
|
|
|
new_issues = issues + [create(:issue, project: project, assignees: [create(:user)])]
|
|
|
|
|
|
|
|
expect { post_graphql(query, current_user: current_user) }.not_to exceed_query_limit(control)
|
|
|
|
# graphql_data is memoized (see spec/support/helpers/graphql_helpers.rb)
|
|
|
|
# so we have to parse the body ourselves the second time
|
2023-01-13 00:05:48 +05:30
|
|
|
issues_data = Gitlab::Json.parse(response.body)['data']['project']['issues']['nodes']
|
|
|
|
expect(issues_data.count).to eq(6)
|
2020-11-24 15:15:51 +05:30
|
|
|
expect(response_assignee_ids(issues_data)).to match_array(assignees_as_global_ids(new_issues))
|
2019-12-26 22:10:19 +05:30
|
|
|
end
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
|
2022-03-02 08:16:31 +05:30
|
|
|
context 'when fetching escalation status' do
|
|
|
|
let_it_be(:escalation_status) { create(:incident_management_issuable_escalation_status, issue: issue_a) }
|
|
|
|
|
|
|
|
let(:statuses) { issue_data.to_h { |issue| [issue['iid'], issue['escalationStatus']] } }
|
|
|
|
let(:fields) do
|
|
|
|
<<~QUERY
|
2023-01-13 00:05:48 +05:30
|
|
|
nodes {
|
|
|
|
id
|
|
|
|
escalationStatus
|
2022-03-02 08:16:31 +05:30
|
|
|
}
|
|
|
|
QUERY
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
issue_a.update!(issue_type: Issue.issue_types[:incident])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the escalation status values' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
statuses = issues_data.map { |issue| issue['escalationStatus'] }
|
2022-03-02 08:16:31 +05:30
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
expect(statuses).to contain_exactly(escalation_status.status_name.upcase.to_s, nil, nil, nil, nil)
|
2022-03-02 08:16:31 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'avoids N+1 queries', :aggregate_failures do
|
|
|
|
base_count = ActiveRecord::QueryRecorder.new { run_with_clean_state(query, context: { current_user: current_user }) }
|
|
|
|
|
|
|
|
new_incident = create(:incident, project: project)
|
|
|
|
create(:incident_management_issuable_escalation_status, issue: new_incident)
|
|
|
|
|
|
|
|
expect { run_with_clean_state(query, context: { current_user: current_user }) }.not_to exceed_query_limit(base_count)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
describe 'N+1 query checks' do
|
|
|
|
let(:extra_iid_for_second_query) { issue_b.iid.to_s }
|
|
|
|
let(:search_params) { { iids: [issue_a.iid.to_s] } }
|
|
|
|
|
|
|
|
def execute_query
|
|
|
|
query = graphql_query_for(
|
|
|
|
:project,
|
|
|
|
{ full_path: project.full_path },
|
2021-04-29 21:17:54 +05:30
|
|
|
query_graphql_field(
|
|
|
|
:issues, search_params,
|
2021-01-29 00:20:46 +05:30
|
|
|
query_graphql_field(:nodes, nil, requested_fields)
|
2021-04-29 21:17:54 +05:30
|
|
|
)
|
2021-01-29 00:20:46 +05:30
|
|
|
)
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when requesting `user_notes_count`' do
|
|
|
|
let(:requested_fields) { [:user_notes_count] }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create_list(:note_on_issue, 2, noteable: issue_a, project: project)
|
|
|
|
create(:note_on_issue, noteable: issue_b, project: project)
|
|
|
|
end
|
|
|
|
|
|
|
|
include_examples 'N+1 query check'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when requesting `user_discussions_count`' do
|
|
|
|
let(:requested_fields) { [:user_discussions_count] }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create_list(:note_on_issue, 2, noteable: issue_a, project: project)
|
|
|
|
create(:note_on_issue, noteable: issue_b, project: project)
|
|
|
|
end
|
|
|
|
|
|
|
|
include_examples 'N+1 query check'
|
|
|
|
end
|
2021-04-29 21:17:54 +05:30
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
context 'when requesting `merge_requests_count`' do
|
|
|
|
let(:requested_fields) { [:merge_requests_count] }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create_list(:merge_requests_closing_issues, 2, issue: issue_a)
|
|
|
|
create_list(:merge_requests_closing_issues, 3, issue: issue_b)
|
|
|
|
end
|
|
|
|
|
|
|
|
include_examples 'N+1 query check'
|
|
|
|
end
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
context 'when requesting `timelogs`' do
|
|
|
|
let(:requested_fields) { 'timelogs { nodes { timeSpent } }' }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create_list(:issue_timelog, 2, issue: issue_a)
|
|
|
|
create(:issue_timelog, issue: issue_b)
|
|
|
|
end
|
|
|
|
|
2022-07-23 23:45:48 +05:30
|
|
|
include_examples 'N+1 query check'
|
|
|
|
end
|
2022-11-25 23:54:43 +05:30
|
|
|
|
2022-07-23 23:45:48 +05:30
|
|
|
context 'when requesting `closed_as_duplicate_of`' do
|
|
|
|
let(:requested_fields) { 'closedAsDuplicateOf { id }' }
|
|
|
|
let(:issue_a_dup) { create(:issue, project: project) }
|
|
|
|
let(:issue_b_dup) { create(:issue, project: project) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
issue_a.update!(duplicated_to_id: issue_a_dup)
|
|
|
|
issue_b.update!(duplicated_to_id: issue_a_dup)
|
|
|
|
end
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
include_examples 'N+1 query check'
|
|
|
|
end
|
2022-11-25 23:54:43 +05:30
|
|
|
|
|
|
|
context 'when award emoji votes' do
|
|
|
|
let(:requested_fields) { [:upvotes, :downvotes] }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create_list(:award_emoji, 2, name: 'thumbsup', awardable: issue_a)
|
|
|
|
create_list(:award_emoji, 2, name: 'thumbsdown', awardable: issue_b)
|
|
|
|
end
|
|
|
|
|
|
|
|
include_examples 'N+1 query check'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when requesting participants' do
|
|
|
|
let_it_be(:issue_c) { create(:issue, project: project) }
|
|
|
|
|
|
|
|
let(:search_params) { { iids: [issue_a.iid.to_s, issue_c.iid.to_s] } }
|
|
|
|
let(:requested_fields) { 'participants { nodes { name } }' }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create(:award_emoji, :upvote, awardable: issue_a)
|
|
|
|
create(:award_emoji, :upvote, awardable: issue_b)
|
|
|
|
create(:award_emoji, :upvote, awardable: issue_c)
|
|
|
|
|
|
|
|
note_with_emoji_a = create(:note_on_issue, noteable: issue_a, project: project)
|
|
|
|
note_with_emoji_b = create(:note_on_issue, noteable: issue_b, project: project)
|
|
|
|
note_with_emoji_c = create(:note_on_issue, noteable: issue_c, project: project)
|
|
|
|
|
|
|
|
create(:award_emoji, :upvote, awardable: note_with_emoji_a)
|
|
|
|
create(:award_emoji, :upvote, awardable: note_with_emoji_b)
|
|
|
|
create(:award_emoji, :upvote, awardable: note_with_emoji_c)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Executes 3 extra queries to fetch participant_attrs
|
|
|
|
include_examples 'N+1 query check', threshold: 3
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when requesting labels' do
|
|
|
|
let(:requested_fields) { ['labels { nodes { id } }'] }
|
|
|
|
|
|
|
|
before do
|
|
|
|
project_labels = create_list(:label, 2, project: project)
|
|
|
|
group_labels = create_list(:group_label, 2, group: group)
|
|
|
|
|
|
|
|
issue_a.update!(labels: [project_labels.first, group_labels.first].flatten)
|
|
|
|
issue_b.update!(labels: [project_labels, group_labels].flatten)
|
|
|
|
end
|
|
|
|
|
|
|
|
include_examples 'N+1 query check', skip_cached: false
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
2023-01-13 00:05:48 +05:30
|
|
|
def issue_ids
|
|
|
|
graphql_dig_at(issues_data, :id)
|
2021-11-18 22:05:49 +05:30
|
|
|
end
|
2022-10-11 01:57:18 +05:30
|
|
|
|
|
|
|
def query(params = issue_filter_params)
|
|
|
|
graphql_query_for(
|
|
|
|
'project',
|
|
|
|
{ 'fullPath' => project.full_path },
|
|
|
|
query_graphql_field('issues', params, fields)
|
|
|
|
)
|
|
|
|
end
|
2019-02-15 15:39:39 +05:30
|
|
|
end
|