2021-01-03 14:25:43 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'spec_helper'
|
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
RSpec.describe 'getting milestone listings nested in a project', feature_category: :team_planning do
|
2021-01-03 14:25:43 +05:30
|
|
|
include GraphqlHelpers
|
|
|
|
|
|
|
|
let_it_be(:today) { Time.now.utc.to_date }
|
|
|
|
let_it_be(:project) { create(:project, :repository, :public) }
|
|
|
|
let_it_be(:current_user) { create(:user) }
|
|
|
|
|
|
|
|
let_it_be(:no_dates) { create(:milestone, project: project, title: 'no dates') }
|
|
|
|
let_it_be(:no_end) { create(:milestone, project: project, title: 'no end', start_date: today - 10.days) }
|
|
|
|
let_it_be(:no_start) { create(:milestone, project: project, title: 'no start', due_date: today - 5.days) }
|
|
|
|
let_it_be(:fully_past) { create(:milestone, project: project, title: 'past', start_date: today - 10.days, due_date: today - 5.days) }
|
|
|
|
let_it_be(:covers_today) { create(:milestone, project: project, title: 'present', start_date: today - 5.days, due_date: today + 5.days) }
|
|
|
|
let_it_be(:fully_future) { create(:milestone, project: project, title: 'future', start_date: today + 5.days, due_date: today + 10.days) }
|
|
|
|
let_it_be(:closed) { create(:milestone, :closed, project: project) }
|
|
|
|
|
|
|
|
let(:results) { graphql_data_at(:project, :milestones, :nodes) }
|
|
|
|
|
|
|
|
let(:search_params) { nil }
|
|
|
|
|
|
|
|
def query_milestones(fields)
|
|
|
|
graphql_query_for(
|
|
|
|
:project,
|
|
|
|
{ full_path: project.full_path },
|
2022-11-25 23:54:43 +05:30
|
|
|
query_graphql_field(:milestones, search_params,
|
|
|
|
[
|
|
|
|
query_graphql_field(:nodes, nil, %i[id title])
|
|
|
|
])
|
2021-01-03 14:25:43 +05:30
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def result_list(expected)
|
|
|
|
expected.map do |milestone|
|
2022-07-16 23:28:13 +05:30
|
|
|
a_graphql_entity_for(milestone)
|
2021-01-03 14:25:43 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:query) do
|
|
|
|
query_milestones(all_graphql_fields_for('Milestone', max_depth: 1))
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:all_milestones) do
|
|
|
|
[no_dates, no_end, no_start, fully_past, fully_future, covers_today, closed]
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'a working graphql query' do
|
|
|
|
before do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
shared_examples 'searching with parameters' do
|
|
|
|
it 'finds the right milestones' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
|
|
|
expect(results).to match_array(result_list(expected))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-07-23 23:45:48 +05:30
|
|
|
context 'the user does not have access' do
|
|
|
|
let_it_be(:project) { create(:project) }
|
|
|
|
let_it_be(:milestones) { create_list(:milestone, 2, project: project) }
|
|
|
|
|
|
|
|
it 'is nil' do
|
|
|
|
post_graphql(query, current_user: current_user)
|
|
|
|
|
|
|
|
expect(graphql_data_at(:project)).to be_nil
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'the user has access' do
|
|
|
|
let(:expected) { milestones }
|
|
|
|
|
|
|
|
before do
|
|
|
|
project.add_guest(current_user)
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
context 'there are no search params' do
|
|
|
|
let(:search_params) { nil }
|
|
|
|
let(:expected) { all_milestones }
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'the search params do not match anything' do
|
|
|
|
let(:search_params) { { title: 'wibble' } }
|
|
|
|
let(:expected) { [] }
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'searching by state:closed' do
|
|
|
|
let(:search_params) { { state: :closed } }
|
|
|
|
let(:expected) { [closed] }
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'searching by state:active' do
|
|
|
|
let(:search_params) { { state: :active } }
|
|
|
|
let(:expected) { all_milestones - [closed] }
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'searching by title' do
|
|
|
|
let(:search_params) { { title: 'no start' } }
|
|
|
|
let(:expected) { [no_start] }
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'searching by search_title' do
|
|
|
|
let(:search_params) { { search_title: 'no' } }
|
|
|
|
let(:expected) { [no_dates, no_start, no_end] }
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'searching by containing_date' do
|
|
|
|
let(:search_params) { { containing_date: (today - 7.days).iso8601 } }
|
|
|
|
let(:expected) { [no_start, no_end, fully_past] }
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'searching by containing_date = today' do
|
|
|
|
let(:search_params) { { containing_date: today.iso8601 } }
|
|
|
|
let(:expected) { [no_end, covers_today] }
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'using timeframe argument' do
|
|
|
|
let(:expected) { [no_end, fully_future] }
|
|
|
|
let(:search_params) do
|
|
|
|
{
|
|
|
|
timeframe: {
|
|
|
|
start: (today + 6.days).iso8601,
|
|
|
|
end: (today + 7.days).iso8601
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'searching with parameters'
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'timeframe validations' do
|
|
|
|
let(:vars) do
|
|
|
|
{
|
|
|
|
path: project.full_path,
|
|
|
|
start: (today + 6.days).iso8601,
|
|
|
|
end: (today + 7.days).iso8601
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'a working graphql query' do
|
|
|
|
before do
|
|
|
|
query = <<~GQL
|
|
|
|
query($path: ID!, $start: Date!, $end: Date!) {
|
|
|
|
project(fullPath: $path) {
|
|
|
|
milestones(timeframe: { start: $start, end: $end }) {
|
|
|
|
nodes { id }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
GQL
|
|
|
|
|
|
|
|
post_graphql(query, current_user: current_user, variables: vars)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is invalid to invert the timeframe arguments' do
|
|
|
|
query = <<~GQL
|
|
|
|
query($path: ID!, $start: Date!, $end: Date!) {
|
|
|
|
project(fullPath: $path) {
|
|
|
|
milestones(timeframe: { start: $end, end: $start }) {
|
|
|
|
nodes { id }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
GQL
|
|
|
|
|
|
|
|
post_graphql(query, current_user: current_user, variables: vars)
|
|
|
|
|
|
|
|
expect(graphql_errors).to contain_exactly(a_hash_including('message' => include('start must be before end')))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|