diff options
author | GitLab Bot <gitlab-bot@gitlab.com> | 2022-12-20 17:22:11 +0300 |
---|---|---|
committer | GitLab Bot <gitlab-bot@gitlab.com> | 2022-12-20 17:22:11 +0300 |
commit | 0c872e02b2c822e3397515ec324051ff540f0cd5 (patch) | |
tree | ce2fb6ce7030e4dad0f4118d21ab6453e5938cdd /spec/requests/api/graphql/project/issues_spec.rb | |
parent | f7e05a6853b12f02911494c4b3fe53d9540d74fc (diff) |
Add latest changes from gitlab-org/gitlab@15-7-stable-eev15.7.0-rc42
Diffstat (limited to 'spec/requests/api/graphql/project/issues_spec.rb')
-rw-r--r-- | spec/requests/api/graphql/project/issues_spec.rb | 698 |
1 files changed, 86 insertions, 612 deletions
diff --git a/spec/requests/api/graphql/project/issues_spec.rb b/spec/requests/api/graphql/project/issues_spec.rb index 214165cb171..ec5e3c6f0de 100644 --- a/spec/requests/api/graphql/project/issues_spec.rb +++ b/spec/requests/api/graphql/project/issues_spec.rb @@ -2,44 +2,92 @@ require 'spec_helper' -RSpec.describe 'getting an issue list for a project' do +RSpec.describe 'getting an issue list for a project', feature_category: :team_planning do include GraphqlHelpers let_it_be(:group) { create(:group) } let_it_be(:project) { create(:project, :repository, :public, group: group) } let_it_be(:current_user) { create(:user) } 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(:milestone1) { create(:milestone, project: project, due_date: 10.days.from_now) } + let_it_be(:milestone2) { create(:milestone, project: project, due_date: 20.days.from_now) } + let_it_be(:milestone3) { create(:milestone, project: project, due_date: 30.days.from_now) } + let_it_be(:milestone4) { create(:milestone, project: project, due_date: 40.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] } + let_it_be(:issue_a) do + create( + :issue, + project: project, + discussion_locked: true, + labels: [priority3], + relative_position: 1000, + milestone: milestone4 + ) + end - let(:issue_a_gid) { issue_a.to_global_id.to_s } - let(:issue_b_gid) { issue_b.to_global_id.to_s } - let(:issues_data) { graphql_data['project']['issues']['nodes'] } - let(:issue_filter_params) { {} } + let_it_be(:issue_b) do + create( + :issue, + :with_alert, + project: project, + title: 'title matching issue i', + due_date: 3.days.ago, + relative_position: 3000, + labels: [priority2, priority3], + milestone: milestone1 + ) + end - let(:fields) do - <<~QUERY - nodes { - #{all_graphql_fields_for('issues'.classify)} - } - QUERY + let_it_be(:issue_c) do + create( + :issue, + project: project, + labels: [priority1], + milestone: milestone2, + due_date: 1.day.ago, + relative_position: nil + ) + end + + let_it_be(:issue_d) do + create(:issue, + project: project, + labels: [priority2], + due_date: 3.days.from_now, + relative_position: 5000, + milestone: milestone3 + ) + end + + let_it_be(:issue_e) do + create( + :issue, + :confidential, + project: project, + due_date: 1.day.from_now, + relative_position: nil + ) end + let_it_be(:issues, reload: true) { [issue_a, issue_b, issue_c, issue_d, issue_e] } + + let(:issue_nodes_path) { %w[project issues nodes] } + let(:issue_filter_params) { {} } + # 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) } + let_it_be(:external_user) { create(:user) } + + let(:public_projects) { [project] } + + before_all do + group.add_developer(current_user) + end # filters let(:expected_negated_assignee_issues) { [issue_b, issue_c, issue_d, issue_e] } @@ -50,24 +98,31 @@ RSpec.describe 'getting an issue list for a project' do let(:unlocked_discussion_issues) { [issue_b, issue_c, issue_d, issue_e] } let(:search_title_term) { 'matching issue' } let(:title_search_issue) { issue_b } + let(:confidential_issues) { [issue_e] } + let(:non_confidential_issues) { [issue_a, issue_b, issue_c, issue_d] } + let(:public_non_confidential_issues) { non_confidential_issues } # 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] } + let(:expected_priority_sorted_asc) { [issue_b, issue_c, issue_d, issue_a, issue_e] } + let(:expected_priority_sorted_desc) { [issue_a, issue_d, issue_c, issue_b, issue_e] } + let(:expected_due_date_sorted_desc) { [issue_d, issue_e, issue_c, issue_b, issue_a] } + let(:expected_due_date_sorted_asc) { [issue_b, issue_c, issue_e, issue_d, issue_a] } + let(:expected_relative_position_sorted_asc) { [issue_a, issue_b, issue_d, issue_c, issue_e] } + let(:expected_label_priority_sorted_asc) { [issue_c, issue_d, issue_b, issue_a, issue_e] } + let(:expected_label_priority_sorted_desc) { [issue_a, issue_d, issue_b, issue_c, issue_e] } + let(:expected_milestone_sorted_asc) { [issue_b, issue_c, issue_d, issue_a, issue_e] } + let(:expected_milestone_sorted_desc) { [issue_a, issue_d, issue_c, issue_b, issue_e] } + + # N+1 queries + let(:same_project_issue1) { issue_a } + let(:same_project_issue2) { 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) end def pagination_query(params) @@ -77,591 +132,10 @@ RSpec.describe 'getting an issue list for a project' do query_graphql_field(:issues, params, "#{page_info} nodes { id }") ) end - end - - context 'when limiting the number of results' do - let(:query) do - <<~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 } - end - - it_behaves_like 'a working graphql query' do - before do - 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 - - context 'when no limit is provided' do - 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 - end - end - - it 'is expected to check permissions on the first issue only' do - 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) - - post_graphql(query, current_user: current_user, variables: variables) - end - end - - 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) - - expect(issues_data).to eq([]) - end - end - - context 'when there is a confidential issue' do - let_it_be(:confidential_issue) do - create(:issue, :confidential, project: project) - end - - let(:confidential_issue_gid) { confidential_issue.to_global_id.to_s } - - context 'when the user cannot see confidential issues' do - it 'returns issues without confidential issues' do - post_graphql(query, current_user: current_user) - - expect(issues_data.size).to eq(5) - - issues_data.each do |issue| - expect(issue['confidential']).to eq(false) - end - end - - 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) - - expect(issue_ids).to match_array(issues.map { |i| i.to_gid.to_s }) - end - end - end - - context 'when the user can see confidential issues' do - before do - project.add_developer(current_user) - end - - it 'returns issues with confidential issues' do - post_graphql(query, current_user: current_user) - - expect(issues_data.size).to eq(6) - - confidentials = issues_data.map do |issue| - issue['confidential'] - end - - expect(confidentials).to contain_exactly(true, false, false, false, false, false) - end - - 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) - - expect(issue_ids).to contain_exactly(confidential_issue_gid) - 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) - - expect(issue_ids).to match_array([issue_a, issue_b, issue_c, issue_d, issue_e].map { |i| i.to_gid.to_s }) - end - end - end - end - - describe 'sorting and pagination' do - let_it_be(:sort_project) { create(:project, :public) } - let_it_be(:data_path) { [:project, :issues] } - - def pagination_query(params) - graphql_query_for( - :project, - { full_path: sort_project.full_path }, - query_graphql_field(:issues, params, "#{page_info} nodes { iid }") - ) - end - def pagination_results_data(data) - data.map { |issue| issue['iid'].to_i } + def post_query(request_user = current_user) + post_graphql(query, current_user: request_user) end - - # rubocop:disable RSpec/MultipleMemoizedHelpers - 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) } - - context 'when ascending' do - it_behaves_like 'sorted paginated query' do - let(:sort_param) { :DUE_DATE_ASC } - let(:first_param) { 2 } - let(:all_records) { [due_issue3.iid, due_issue5.iid, due_issue1.iid, due_issue4.iid, due_issue2.iid] } - end - end - - context 'when descending' do - it_behaves_like 'sorted paginated query' do - let(:sort_param) { :DUE_DATE_DESC } - let(:first_param) { 2 } - let(:all_records) { [due_issue1.iid, due_issue5.iid, due_issue3.iid, due_issue4.iid, due_issue2.iid] } - end - end - end - - 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) } - - context 'when ascending' do - it_behaves_like 'sorted paginated query', is_reversible: true do - let(:sort_param) { :RELATIVE_POSITION_ASC } - let(:first_param) { 2 } - let(:all_records) do - [ - relative_issue5.iid, relative_issue3.iid, relative_issue1.iid, - relative_issue2.iid, relative_issue4.iid - ] - end - end - end - end - - 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) } - - context 'when ascending' do - it_behaves_like 'sorted paginated query' do - 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] } - end - end - - context 'when descending' do - it_behaves_like 'sorted paginated query' do - 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] } - end - end - end - # rubocop:enable RSpec/MultipleMemoizedHelpers - - 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) } - - context 'when ascending' do - it_behaves_like 'sorted paginated query' do - let(:sort_param) { :MILESTONE_DUE_ASC } - let(:first_param) { 2 } - let(:all_records) { [milestone_issue2.iid, milestone_issue3.iid, milestone_issue1.iid] } - end - end - - context 'when descending' do - it_behaves_like 'sorted paginated query' do - let(:sort_param) { :MILESTONE_DUE_DESC } - let(:first_param) { 2 } - let(:all_records) { [milestone_issue3.iid, milestone_issue2.iid, milestone_issue1.iid] } - end - end - end - end - - context 'when fetching alert management alert' do - let(:fields) do - <<~QUERY - nodes { - iid - alertManagementAlert { - title - } - alertManagementAlerts { - nodes { - title - } - } - } - 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) } - - create(:alert_management_alert, :with_incident, project: project) - - 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) - - alert_titles = issues_data.map { |issue| issue.dig('alertManagementAlert', 'title') } - expected_titles = issues.map { |issue| issue.alert_management_alert&.title } - - expect(alert_titles).to contain_exactly(*expected_titles) - end - - 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 - end - - 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 - create(:issue_customer_relations_contact, :for_issue, issue: issue_a) - - control = ActiveRecord::QueryRecorder.new(skip_cached: false) { clean_state_query } - - create(:issue_customer_relations_contact, :for_issue, issue: issue_a) - - expect { clean_state_query }.not_to exceed_all_query_limit(control) - end - end - - context 'when fetching labels' do - let(:fields) do - <<~QUERY - nodes { - id - labels { - nodes { - id - } - } - } - 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) - response_data.map do |node| - node['labels']['nodes'].map { |u| u['id'] } - 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) } - expect(issues_data.count).to eq(5) - 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 - issues_data = Gitlab::Json.parse(response.body)['data']['project']['issues']['nodes'] - expect(issues_data.count).to eq(6) - expect(response_label_ids(issues_data)).to match_array(labels_as_global_ids(new_issues)) - end - end - - context 'when fetching assignees' do - let(:fields) do - <<~QUERY - nodes { - id - assignees { - nodes { - id - } - } - } - 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) - response_data.map do |node| - node['assignees']['nodes'].map { |node| node['id'] } - 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) } - expect(issues_data.count).to eq(5) - 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 - issues_data = Gitlab::Json.parse(response.body)['data']['project']['issues']['nodes'] - expect(issues_data.count).to eq(6) - expect(response_assignee_ids(issues_data)).to match_array(assignees_as_global_ids(new_issues)) - end - end - - 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 - nodes { - id - escalationStatus - } - 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) - - statuses = issues_data.map { |issue| issue['escalationStatus'] } - - expect(statuses).to contain_exactly(escalation_status.status_name.upcase.to_s, nil, nil, nil, nil) - 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 - - 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 }, - query_graphql_field( - :issues, search_params, - query_graphql_field(:nodes, nil, requested_fields) - ) - ) - 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 - - 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 - - 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 - - include_examples 'N+1 query check' - end - - 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 - - include_examples 'N+1 query check' - end - - 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 - end - - def issue_ids - graphql_dig_at(issues_data, :id) end def query(params = issue_filter_params) |