2019-12-21 20:55:43 +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 Resolvers::IssuesResolver do
|
2019-02-15 15:39:39 +05:30
|
|
|
include GraphqlHelpers
|
|
|
|
|
|
|
|
let(:current_user) { create(:user) }
|
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
let_it_be(:group) { create(:group) }
|
|
|
|
let_it_be(:project) { create(:project, group: group) }
|
|
|
|
let_it_be(:other_project) { create(:project, group: group) }
|
|
|
|
|
|
|
|
let_it_be(:milestone) { create(:milestone, project: project) }
|
|
|
|
let_it_be(:assignee) { create(:user) }
|
|
|
|
let_it_be(:issue1) { create(:issue, project: project, state: :opened, created_at: 3.hours.ago, updated_at: 3.hours.ago, milestone: milestone) }
|
|
|
|
let_it_be(:issue2) { create(:issue, project: project, state: :closed, title: 'foo', created_at: 1.hour.ago, updated_at: 1.hour.ago, closed_at: 1.hour.ago, assignees: [assignee]) }
|
|
|
|
let_it_be(:issue3) { create(:issue, project: other_project, state: :closed, title: 'foo', created_at: 1.hour.ago, updated_at: 1.hour.ago, closed_at: 1.hour.ago, assignees: [assignee]) }
|
|
|
|
let_it_be(:issue4) { create(:issue) }
|
|
|
|
let_it_be(:label1) { create(:label, project: project) }
|
|
|
|
let_it_be(:label2) { create(:label, project: project) }
|
2019-07-31 22:56:46 +05:30
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
context "with a project" do
|
2019-07-31 22:56:46 +05:30
|
|
|
before do
|
|
|
|
project.add_developer(current_user)
|
|
|
|
create(:label_link, label: label1, target: issue1)
|
|
|
|
create(:label_link, label: label1, target: issue2)
|
|
|
|
create(:label_link, label: label2, target: issue2)
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
describe '#resolve' do
|
|
|
|
it 'finds all issues' do
|
|
|
|
expect(resolve_issues).to contain_exactly(issue1, issue2)
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
it 'filters by state' do
|
|
|
|
expect(resolve_issues(state: 'opened')).to contain_exactly(issue1)
|
|
|
|
expect(resolve_issues(state: 'closed')).to contain_exactly(issue2)
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
it 'filters by milestone' do
|
|
|
|
expect(resolve_issues(milestone_title: milestone.title)).to contain_exactly(issue1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by assignee_username' do
|
|
|
|
expect(resolve_issues(assignee_username: assignee.username)).to contain_exactly(issue2)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by assignee_id' do
|
|
|
|
expect(resolve_issues(assignee_id: assignee.id)).to contain_exactly(issue2)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by any assignee' do
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(resolve_issues(assignee_id: IssuableFinder::Params::FILTER_ANY)).to contain_exactly(issue2)
|
2020-04-08 14:13:33 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by no assignee' do
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(resolve_issues(assignee_id: IssuableFinder::Params::FILTER_NONE)).to contain_exactly(issue1)
|
2020-04-08 14:13:33 +05:30
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
it 'filters by labels' do
|
|
|
|
expect(resolve_issues(label_name: [label1.title])).to contain_exactly(issue1, issue2)
|
|
|
|
expect(resolve_issues(label_name: [label1.title, label2.title])).to contain_exactly(issue2)
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
describe 'filters by created_at' do
|
|
|
|
it 'filters by created_before' do
|
|
|
|
expect(resolve_issues(created_before: 2.hours.ago)).to contain_exactly(issue1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by created_after' do
|
|
|
|
expect(resolve_issues(created_after: 2.hours.ago)).to contain_exactly(issue2)
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
describe 'filters by updated_at' do
|
|
|
|
it 'filters by updated_before' do
|
|
|
|
expect(resolve_issues(updated_before: 2.hours.ago)).to contain_exactly(issue1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by updated_after' do
|
|
|
|
expect(resolve_issues(updated_after: 2.hours.ago)).to contain_exactly(issue2)
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
describe 'filters by closed_at' do
|
|
|
|
let!(:issue3) { create(:issue, project: project, state: :closed, closed_at: 3.hours.ago) }
|
|
|
|
|
|
|
|
it 'filters by closed_before' do
|
|
|
|
expect(resolve_issues(closed_before: 2.hours.ago)).to contain_exactly(issue3)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by closed_after' do
|
|
|
|
expect(resolve_issues(closed_after: 2.hours.ago)).to contain_exactly(issue2)
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
2020-01-01 13:55:28 +05:30
|
|
|
context 'when searching issues' do
|
|
|
|
it 'returns correct issues' do
|
|
|
|
expect(resolve_issues(search: 'foo')).to contain_exactly(issue2)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'uses project search optimization' do
|
2020-07-28 23:09:34 +05:30
|
|
|
expected_arguments = a_hash_including(
|
2020-01-01 13:55:28 +05:30
|
|
|
search: 'foo',
|
2020-07-28 23:09:34 +05:30
|
|
|
attempt_project_search_optimizations: true
|
|
|
|
)
|
2020-01-01 13:55:28 +05:30
|
|
|
expect(IssuesFinder).to receive(:new).with(anything, expected_arguments).and_call_original
|
|
|
|
|
|
|
|
resolve_issues(search: 'foo')
|
|
|
|
end
|
2019-07-31 22:56:46 +05:30
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
describe 'sorting' do
|
|
|
|
context 'when sorting by created' do
|
|
|
|
it 'sorts issues ascending' do
|
|
|
|
expect(resolve_issues(sort: 'created_asc')).to eq [issue1, issue2]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sorts issues descending' do
|
|
|
|
expect(resolve_issues(sort: 'created_desc')).to eq [issue2, issue1]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when sorting by due date' do
|
2020-05-24 23:13:21 +05:30
|
|
|
let_it_be(:project) { create(:project) }
|
|
|
|
let_it_be(:due_issue1) { create(:issue, project: project, due_date: 3.days.from_now) }
|
|
|
|
let_it_be(:due_issue2) { create(:issue, project: project, due_date: nil) }
|
|
|
|
let_it_be(:due_issue3) { create(:issue, project: project, due_date: 2.days.ago) }
|
|
|
|
let_it_be(:due_issue4) { create(:issue, project: project, due_date: nil) }
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
it 'sorts issues ascending' do
|
|
|
|
expect(resolve_issues(sort: :due_date_asc)).to eq [due_issue3, due_issue1, due_issue4, due_issue2]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sorts issues descending' do
|
|
|
|
expect(resolve_issues(sort: :due_date_desc)).to eq [due_issue1, due_issue3, due_issue4, due_issue2]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when sorting by relative position' do
|
2020-05-24 23:13:21 +05:30
|
|
|
let_it_be(:project) { create(:project) }
|
|
|
|
let_it_be(:relative_issue1) { create(:issue, project: project, relative_position: 2000) }
|
|
|
|
let_it_be(:relative_issue2) { create(:issue, project: project, relative_position: nil) }
|
|
|
|
let_it_be(:relative_issue3) { create(:issue, project: project, relative_position: 1000) }
|
|
|
|
let_it_be(:relative_issue4) { create(:issue, project: project, relative_position: nil) }
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
it 'sorts issues ascending' do
|
|
|
|
expect(resolve_issues(sort: :relative_position_asc)).to eq [relative_issue3, relative_issue1, relative_issue4, relative_issue2]
|
|
|
|
end
|
|
|
|
end
|
2020-05-24 23:13:21 +05:30
|
|
|
|
|
|
|
context 'when sorting by priority' do
|
|
|
|
let_it_be(:project) { create(:project) }
|
|
|
|
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(:priority_label1) { create(:label, project: project, priority: 1) }
|
|
|
|
let_it_be(:priority_label2) { create(:label, project: project, priority: 5) }
|
|
|
|
let_it_be(:priority_issue1) { create(:issue, project: project, labels: [priority_label1], milestone: late_milestone) }
|
|
|
|
let_it_be(:priority_issue2) { create(:issue, project: project, labels: [priority_label2]) }
|
|
|
|
let_it_be(:priority_issue3) { create(:issue, project: project, milestone: early_milestone) }
|
|
|
|
let_it_be(:priority_issue4) { create(:issue, project: project) }
|
|
|
|
|
|
|
|
it 'sorts issues ascending' do
|
|
|
|
expect(resolve_issues(sort: :priority_asc).items).to eq([priority_issue3, priority_issue1, priority_issue2, priority_issue4])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sorts issues descending' do
|
|
|
|
expect(resolve_issues(sort: :priority_desc).items).to eq([priority_issue1, priority_issue3, priority_issue2, priority_issue4])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when sorting by label priority' do
|
|
|
|
let_it_be(:project) { create(:project) }
|
|
|
|
let_it_be(:label1) { create(:label, project: project, priority: 1) }
|
|
|
|
let_it_be(:label2) { create(:label, project: project, priority: 5) }
|
|
|
|
let_it_be(:label3) { create(:label, project: project, priority: 10) }
|
|
|
|
let_it_be(:label_issue1) { create(:issue, project: project, labels: [label1]) }
|
|
|
|
let_it_be(:label_issue2) { create(:issue, project: project, labels: [label2]) }
|
|
|
|
let_it_be(:label_issue3) { create(:issue, project: project, labels: [label1, label3]) }
|
|
|
|
let_it_be(:label_issue4) { create(:issue, project: project) }
|
|
|
|
|
|
|
|
it 'sorts issues ascending' do
|
|
|
|
expect(resolve_issues(sort: :label_priority_asc).items).to eq([label_issue3, label_issue1, label_issue2, label_issue4])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sorts issues descending' do
|
|
|
|
expect(resolve_issues(sort: :label_priority_desc).items).to eq([label_issue2, label_issue3, label_issue1, label_issue4])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when sorting by milestone due date' do
|
|
|
|
let_it_be(:project) { create(:project) }
|
|
|
|
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(:milestone_issue1) { create(:issue, project: project) }
|
|
|
|
let_it_be(:milestone_issue2) { create(:issue, project: project, milestone: early_milestone) }
|
|
|
|
let_it_be(:milestone_issue3) { create(:issue, project: project, milestone: late_milestone) }
|
|
|
|
|
|
|
|
it 'sorts issues ascending' do
|
|
|
|
expect(resolve_issues(sort: :milestone_due_asc).items).to eq([milestone_issue2, milestone_issue3, milestone_issue1])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sorts issues descending' do
|
|
|
|
expect(resolve_issues(sort: :milestone_due_desc).items).to eq([milestone_issue3, milestone_issue2, milestone_issue1])
|
|
|
|
end
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
it 'returns issues user can see' do
|
|
|
|
project.add_guest(current_user)
|
|
|
|
|
|
|
|
create(:issue, confidential: true)
|
|
|
|
|
|
|
|
expect(resolve_issues).to contain_exactly(issue1, issue2)
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
2019-02-15 15:39:39 +05:30
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it 'finds a specific issue with iid', :request_store do
|
|
|
|
result = batch_sync(max_queries: 2) { resolve_issues(iid: issue1.iid) }
|
|
|
|
|
|
|
|
expect(result).to contain_exactly(issue1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'batches queries that only include IIDs', :request_store do
|
|
|
|
result = batch_sync(max_queries: 2) do
|
|
|
|
resolve_issues(iid: issue1.iid) + resolve_issues(iids: issue2.iid)
|
|
|
|
end
|
|
|
|
|
|
|
|
expect(result).to contain_exactly(issue1, issue2)
|
2019-07-31 22:56:46 +05:30
|
|
|
end
|
2019-02-15 15:39:39 +05:30
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it 'finds a specific issue with iids', :request_store do
|
|
|
|
result = batch_sync(max_queries: 2) do
|
|
|
|
resolve_issues(iids: [issue1.iid])
|
|
|
|
end
|
|
|
|
|
|
|
|
expect(result).to contain_exactly(issue1)
|
2019-07-31 22:56:46 +05:30
|
|
|
end
|
2019-02-15 15:39:39 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
it 'finds multiple issues with iids' do
|
2020-07-28 23:09:34 +05:30
|
|
|
create(:issue, project: project, author: current_user)
|
|
|
|
|
|
|
|
expect(batch_sync { resolve_issues(iids: [issue1.iid, issue2.iid]) })
|
2019-07-31 22:56:46 +05:30
|
|
|
.to contain_exactly(issue1, issue2)
|
|
|
|
end
|
2019-02-15 15:39:39 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
it 'finds only the issues within the project we are looking at' do
|
|
|
|
another_project = create(:project)
|
|
|
|
iids = [issue1, issue2].map(&:iid)
|
2019-02-15 15:39:39 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
iids.each do |iid|
|
|
|
|
create(:issue, project: another_project, iid: iid)
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
expect(batch_sync { resolve_issues(iids: iids) }).to contain_exactly(issue1, issue2)
|
2019-07-31 22:56:46 +05:30
|
|
|
end
|
2019-02-15 15:39:39 +05:30
|
|
|
end
|
2019-07-31 22:56:46 +05:30
|
|
|
end
|
2019-03-02 22:35:43 +05:30
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
context "with a group" do
|
|
|
|
before do
|
|
|
|
group.add_developer(current_user)
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#resolve' do
|
|
|
|
it 'finds all group issues' do
|
|
|
|
result = resolve(described_class, obj: group, ctx: { current_user: current_user })
|
|
|
|
|
|
|
|
expect(result).to contain_exactly(issue1, issue2, issue3)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
context "when passing a non existent, batch loaded project" do
|
|
|
|
let(:project) do
|
2019-12-04 20:38:33 +05:30
|
|
|
BatchLoader::GraphQL.for("non-existent-path").batch do |_fake_paths, loader, _|
|
2019-07-31 22:56:46 +05:30
|
|
|
loader.call("non-existent-path", nil)
|
|
|
|
end
|
2019-03-02 22:35:43 +05:30
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
it "returns nil without breaking" do
|
|
|
|
expect(resolve_issues(iids: ["don't", "break"])).to be_empty
|
2019-03-02 22:35:43 +05:30
|
|
|
end
|
2019-07-31 22:56:46 +05:30
|
|
|
end
|
2019-03-02 22:35:43 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
it 'increases field complexity based on arguments' do
|
2019-09-04 21:01:54 +05:30
|
|
|
field = Types::BaseField.new(name: 'test', type: GraphQL::STRING_TYPE.connection_type, resolver_class: described_class, null: false, max_page_size: 100)
|
2019-03-02 22:35:43 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
expect(field.to_graphql.complexity.call({}, {}, 1)).to eq 4
|
|
|
|
expect(field.to_graphql.complexity.call({}, { labelName: 'foo' }, 1)).to eq 8
|
2019-02-15 15:39:39 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def resolve_issues(args = {}, context = { current_user: current_user })
|
|
|
|
resolve(described_class, obj: project, args: args, ctx: context)
|
|
|
|
end
|
|
|
|
end
|