Commit 680decd0 authored by GitLab Bot's avatar GitLab Bot

Automatic merge of gitlab-org/gitlab master

parents 1a57f833 fa60193b
...@@ -10,7 +10,7 @@ module Resolvers ...@@ -10,7 +10,7 @@ module Resolvers
argument :sort, Types::IssueSortEnum, argument :sort, Types::IssueSortEnum,
description: 'Sort issues by this criteria', description: 'Sort issues by this criteria',
required: false, required: false,
default_value: 'created_desc' default_value: :created_desc
type Types::IssueType.connection_type, null: true type Types::IssueType.connection_type, null: true
......
...@@ -52,7 +52,7 @@ module Resolvers ...@@ -52,7 +52,7 @@ module Resolvers
argument :sort, Types::MergeRequestSortEnum, argument :sort, Types::MergeRequestSortEnum,
description: 'Sort merge requests by this criteria', description: 'Sort merge requests by this criteria',
required: false, required: false,
default_value: 'created_desc' default_value: :created_desc
def self.single def self.single
::Resolvers::MergeRequestResolver ::Resolvers::MergeRequestResolver
......
...@@ -17,7 +17,7 @@ module Resolvers ...@@ -17,7 +17,7 @@ module Resolvers
argument :sort, Types::SortEnum, argument :sort, Types::SortEnum,
description: 'Sort users by this criteria', description: 'Sort users by this criteria',
required: false, required: false,
default_value: 'created_desc' default_value: :created_desc
argument :search, GraphQL::STRING_TYPE, argument :search, GraphQL::STRING_TYPE,
required: false, required: false,
......
...@@ -7,10 +7,10 @@ module Types ...@@ -7,10 +7,10 @@ module Types
# Deprecated, as we prefer uppercase enums # Deprecated, as we prefer uppercase enums
# https://gitlab.com/groups/gitlab-org/-/epics/1838 # https://gitlab.com/groups/gitlab-org/-/epics/1838
value 'updated_desc', 'Updated at descending order', deprecated: { reason: 'Use UPDATED_DESC', milestone: '13.5' } value 'updated_desc', 'Updated at descending order', value: :updated_desc, deprecated: { reason: 'Use UPDATED_DESC', milestone: '13.5' }
value 'updated_asc', 'Updated at ascending order', deprecated: { reason: 'Use UPDATED_ASC', milestone: '13.5' } value 'updated_asc', 'Updated at ascending order', value: :updated_asc, deprecated: { reason: 'Use UPDATED_ASC', milestone: '13.5' }
value 'created_desc', 'Created at descending order', deprecated: { reason: 'Use CREATED_DESC', milestone: '13.5' } value 'created_desc', 'Created at descending order', value: :created_desc, deprecated: { reason: 'Use CREATED_DESC', milestone: '13.5' }
value 'created_asc', 'Created at ascending order', deprecated: { reason: 'Use CREATED_ASC', milestone: '13.5' } value 'created_asc', 'Created at ascending order', value: :created_asc, deprecated: { reason: 'Use CREATED_ASC', milestone: '13.5' }
value 'UPDATED_DESC', 'Updated at descending order', value: :updated_desc value 'UPDATED_DESC', 'Updated at descending order', value: :updated_desc
value 'UPDATED_ASC', 'Updated at ascending order', value: :updated_asc value 'UPDATED_ASC', 'Updated at ascending order', value: :updated_asc
......
---
title: Improve logging on feature flag modification
merge_request: 48417
author:
type: other
...@@ -665,6 +665,31 @@ installations from source. ...@@ -665,6 +665,31 @@ installations from source.
It logs the progress of the export process. It logs the progress of the export process.
## `features_json.log`
> [Introduced](https://gitlab.com/gitlab-org/gitlab-foss/-/issues/59587) in GitLab 13.7.
This file's location depends on how you installed GitLab:
- For Omnibus GitLab packages: `/var/log/gitlab/gitlab-rails/features_json.log`
- For installations from source: `/home/git/gitlab/log/features_json.log`
The modification events from [Feature flags in development of GitLab](../development/feature_flags/index.md)
are recorded in this file. For example:
```json
{"severity":"INFO","time":"2020-11-24T02:30:59.860Z","correlation_id":null,"key":"cd_auto_rollback","action":"enable","extra.thing":"true"}
{"severity":"INFO","time":"2020-11-24T02:31:29.108Z","correlation_id":null,"key":"cd_auto_rollback","action":"enable","extra.thing":"true"}
{"severity":"INFO","time":"2020-11-24T02:31:29.129Z","correlation_id":null,"key":"cd_auto_rollback","action":"disable","extra.thing":"false"}
{"severity":"INFO","time":"2020-11-24T02:31:29.177Z","correlation_id":null,"key":"cd_auto_rollback","action":"enable","extra.thing":"Project:1"}
{"severity":"INFO","time":"2020-11-24T02:31:29.183Z","correlation_id":null,"key":"cd_auto_rollback","action":"disable","extra.thing":"Project:1"}
{"severity":"INFO","time":"2020-11-24T02:31:29.188Z","correlation_id":null,"key":"cd_auto_rollback","action":"enable_percentage_of_time","extra.percentage":"50"}
{"severity":"INFO","time":"2020-11-24T02:31:29.193Z","correlation_id":null,"key":"cd_auto_rollback","action":"disable_percentage_of_time"}
{"severity":"INFO","time":"2020-11-24T02:31:29.198Z","correlation_id":null,"key":"cd_auto_rollback","action":"enable_percentage_of_actors","extra.percentage":"50"}
{"severity":"INFO","time":"2020-11-24T02:31:29.203Z","correlation_id":null,"key":"cd_auto_rollback","action":"disable_percentage_of_actors"}
{"severity":"INFO","time":"2020-11-24T02:31:29.329Z","correlation_id":null,"key":"cd_auto_rollback","action":"remove"}
```
## `auth.log` ## `auth.log`
> Introduced in GitLab 12.0. > Introduced in GitLab 12.0.
......
...@@ -228,8 +228,10 @@ existing gates (e.g. `--group=gitlab-org`) in the above processes. ...@@ -228,8 +228,10 @@ existing gates (e.g. `--group=gitlab-org`) in the above processes.
### Feature flag change logging ### Feature flag change logging
Any feature flag change that affects GitLab.com (production) will #### Chatops level
automatically be logged in an issue.
Any feature flag change that affects GitLab.com (production) via [Chatops](https://gitlab.com/gitlab-com/chatops)
is automatically logged in an issue.
The issue is created in the The issue is created in the
[gl-infra/feature-flag-log](https://gitlab.com/gitlab-com/gl-infra/feature-flag-log/-/issues?scope=all&utf8=%E2%9C%93&state=closed) [gl-infra/feature-flag-log](https://gitlab.com/gitlab-com/gl-infra/feature-flag-log/-/issues?scope=all&utf8=%E2%9C%93&state=closed)
...@@ -243,6 +245,12 @@ marker to make the change even more visible. ...@@ -243,6 +245,12 @@ marker to make the change even more visible.
Changes to the issue format can be submitted in the Changes to the issue format can be submitted in the
[Chatops project](https://gitlab.com/gitlab-com/chatops). [Chatops project](https://gitlab.com/gitlab-com/chatops).
#### Instance level
Any feature flag change that affects any GitLab instance is automatically logged in
[features_json.log](../../administration/logs.md#features_jsonlog).
You can search the change history in [Kibana](https://about.gitlab.com/handbook/support/workflows/kibana.html).
## Cleaning up ## Cleaning up
A feature flag should be removed as soon as it is no longer needed. Each additional A feature flag should be removed as soon as it is no longer needed. Each additional
......
...@@ -294,28 +294,30 @@ The shared example requires certain `let` variables and methods to be set up: ...@@ -294,28 +294,30 @@ The shared example requires certain `let` variables and methods to be set up:
```ruby ```ruby
describe 'sorting and pagination' do describe 'sorting and pagination' do
let(:sort_project) { create(:project, :public) } let_it_be(:sort_project) { create(:project, :public) }
let(:data_path) { [:project, :issues] } let(:data_path) { [:project, :issues] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for( :project, { full_path: sort_project.full_path },
'project', query_nodes(:issues, :id, include_pagination_info: true, args: params))
{ 'fullPath' => sort_project.full_path },
query_graphql_field('issues', params, "#{page_info} edges { node { id } }")
) )
end end
def pagination_results_data(data) def pagination_results_data(nodes)
data.map { |issue| issue.dig('node', 'iid').to_i } nodes.map { |issue| issue['iid'].to_i }
end end
context 'when sorting by weight' do context 'when sorting by weight' do
... let_it_be(:issues) { make_some_issues_with_weights }
context 'when ascending' do context 'when ascending' do
let(:ordered_issues) { issues.sort_by(&:weight) }
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'WEIGHT_ASC' } let(:sort_param) { :WEIGHT_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [weight_issue3.iid, weight_issue5.iid, weight_issue1.iid, weight_issue4.iid, weight_issue2.iid] } let(:expected_results) { ordered_issues.map(&:iid) }
end end
end end
end
``` ```
...@@ -245,8 +245,8 @@ group. ...@@ -245,8 +245,8 @@ group.
| Browse group | ✓ | ✓ | ✓ | ✓ | ✓ | | Browse group | ✓ | ✓ | ✓ | ✓ | ✓ |
| View group wiki pages **(PREMIUM)** | ✓ (6) | ✓ | ✓ | ✓ | ✓ | | View group wiki pages **(PREMIUM)** | ✓ (6) | ✓ | ✓ | ✓ | ✓ |
| View Insights charts **(ULTIMATE)** | ✓ | ✓ | ✓ | ✓ | ✓ | | View Insights charts **(ULTIMATE)** | ✓ | ✓ | ✓ | ✓ | ✓ |
| View group epic **(ULTIMATE)** | ✓ | ✓ | ✓ | ✓ | ✓ | | View group epic **(PREMIUM)** | ✓ | ✓ | ✓ | ✓ | ✓ |
| Create/edit group epic **(ULTIMATE)** | | ✓ | ✓ | ✓ | ✓ | | Create/edit group epic **(PREMIUM)** | | ✓ | ✓ | ✓ | ✓ |
| Manage group labels | | ✓ | ✓ | ✓ | ✓ | | Manage group labels | | ✓ | ✓ | ✓ | ✓ |
| See a container registry | | ✓ | ✓ | ✓ | ✓ | | See a container registry | | ✓ | ✓ | ✓ | ✓ |
| Pull [packages](packages/index.md) | | ✓ | ✓ | ✓ | ✓ | | Pull [packages](packages/index.md) | | ✓ | ✓ | ✓ | ✓ |
...@@ -270,7 +270,7 @@ group. ...@@ -270,7 +270,7 @@ group.
| Create/Delete group deploy tokens | | | | | ✓ | | Create/Delete group deploy tokens | | | | | ✓ |
| Manage group members | | | | | ✓ | | Manage group members | | | | | ✓ |
| Delete group | | | | | ✓ | | Delete group | | | | | ✓ |
| Delete group epic **(ULTIMATE)** | | | | | ✓ | | Delete group epic **(PREMIUM)** | | | | | ✓ |
| Edit SAML SSO Billing **(SILVER ONLY)** | ✓ | ✓ | ✓ | ✓ | ✓ (4) | | Edit SAML SSO Billing **(SILVER ONLY)** | ✓ | ✓ | ✓ | ✓ | ✓ (4) |
| View group Audit Events | | | | | ✓ | | View group Audit Events | | | | | ✓ |
| Disable notification emails | | | | | ✓ | | Disable notification emails | | | | | ✓ |
......
...@@ -79,7 +79,7 @@ module RequirementsManagement ...@@ -79,7 +79,7 @@ module RequirementsManagement
def sort(items) def sort(items)
sorts = RequirementsManagement::Requirement.simple_sorts.keys sorts = RequirementsManagement::Requirement.simple_sorts.keys
sort = sorts.include?(params[:sort]) ? params[:sort] : 'id_desc' sort = sorts.include?(params[:sort]&.to_s) ? params[:sort] : 'id_desc'
items.order_by(sort) items.order_by(sort)
end end
......
...@@ -44,44 +44,4 @@ RSpec.describe Feature, stub_feature_flags: false do ...@@ -44,44 +44,4 @@ RSpec.describe Feature, stub_feature_flags: false do
end end
end end
end end
describe '.enable_group' do
context 'when running on a Geo primary node' do
before do
stub_primary_node
end
it 'does not create a Geo::CacheInvalidationEvent if there are no Geo secondary nodes' do
allow(Gitlab::Geo).to receive(:secondary_nodes) { [] }
expect { described_class.enable_group(:foo, :bar) }.not_to change(Geo::CacheInvalidationEvent, :count)
end
it 'creates a Geo::CacheInvalidationEvent' do
allow(Gitlab::Geo).to receive(:secondary_nodes) { [double] }
expect { described_class.enable_group(:foo, :bar) }.to change(Geo::CacheInvalidationEvent, :count).by(1)
end
end
end
describe '.disable_group' do
context 'when running on a Geo primary node' do
before do
stub_primary_node
end
it 'does not create a Geo::CacheInvalidationEvent if there are no Geo secondary nodes' do
allow(Gitlab::Geo).to receive(:secondary_nodes) { [] }
expect { described_class.disable_group(:foo, :bar) }.not_to change(Geo::CacheInvalidationEvent, :count)
end
it 'creates a Geo::CacheInvalidationEvent' do
allow(Gitlab::Geo).to receive(:secondary_nodes) { [double] }
expect { described_class.disable_group(:foo, :bar) }.to change(Geo::CacheInvalidationEvent, :count).by(1)
end
end
end
end end
...@@ -64,28 +64,22 @@ RSpec.describe 'get board lists' do ...@@ -64,28 +64,22 @@ RSpec.describe 'get board lists' do
end end
describe 'sorting and pagination' do describe 'sorting and pagination' do
let(:data_path) { [board_parent_type, :boards, :edges, 0, :node, :lists] } let(:data_path) { [board_parent_type, :boards, :nodes, 0, :lists] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for(
board_parent_type, board_parent_type,
{ 'fullPath' => board_parent.full_path }, { 'fullPath' => board_parent.full_path },
<<~BOARDS <<~BOARDS
boards(first: 1) { boards(first: 1) {
edges { nodes {
node { #{query_nodes(:lists, :id, args: params, include_pagination_info: true)}
#{query_graphql_field('lists', params, "#{page_info} edges { node { id } }")}
}
} }
} }
BOARDS BOARDS
) )
end end
def pagination_results_data(data)
data.map { |list| list.dig('node', 'id') }
end
context 'when using default sorting' do context 'when using default sorting' do
let!(:milestone_list) { create(:milestone_list, board: board, milestone: milestone, position: 10) } let!(:milestone_list) { create(:milestone_list, board: board, milestone: milestone, position: 10) }
let!(:milestone_list2) { create(:milestone_list, board: board, milestone: milestone2, position: 2) } let!(:milestone_list2) { create(:milestone_list, board: board, milestone: milestone2, position: 2) }
...@@ -98,7 +92,7 @@ RSpec.describe 'get board lists' do ...@@ -98,7 +92,7 @@ RSpec.describe 'get board lists' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { } let(:sort_param) { }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { lists.map { |list| list.to_global_id.to_s } } let(:expected_results) { lists.map { |list| global_id_of(list) } }
end end
end end
end end
......
...@@ -359,35 +359,33 @@ RSpec.describe 'getting group information' do ...@@ -359,35 +359,33 @@ RSpec.describe 'getting group information' do
let(:data_path) { [:group, :codeCoverageActivities] } let(:data_path) { [:group, :codeCoverageActivities] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for(
'group', :group, { full_path: group.full_path },
{ 'fullPath' => group.full_path },
<<~QUERY <<~QUERY
codeCoverageActivities(startDate: "#{start_date}" #{params}) { codeCoverageActivities(startDate: "#{start_date}" #{params}) {
#{page_info} edges { #{page_info}
node { nodes { averageCoverage }
averageCoverage
}
}
} }
QUERY QUERY
) )
end end
def pagination_results_data(data)
data.map { |coverage| coverage.dig('node', 'averageCoverage') }
end
context 'when default sorting' do context 'when default sorting' do
let!(:coverage_1) { create(:ci_daily_build_group_report_result, project: project_1) } let_it_be(:cov_1) { create(:ci_daily_build_group_report_result, project: project_1, coverage: 77.0) }
let!(:coverage_2) { create(:ci_daily_build_group_report_result, project: project_2, coverage: 88.8, date: 1.week.ago) } let_it_be(:cov_2) { create(:ci_daily_build_group_report_result, project: project_2, coverage: 88.8, date: 1.week.ago) }
let(:start_date) { 1.week.ago.to_date.to_s } let_it_be(:cov_3) { create(:ci_daily_build_group_report_result, project: project_1, coverage: 66.6, date: 2.weeks.ago) }
let_it_be(:cov_4) { create(:ci_daily_build_group_report_result, project: project_2, coverage: 99.9, date: 3.weeks.ago) }
let_it_be(:cov_5) { create(:ci_daily_build_group_report_result, project: project_1, coverage: 44.4, date: 4.weeks.ago) }
let_it_be(:cov_6) { create(:ci_daily_build_group_report_result, project: project_1, coverage: 100.0, date: 6.weeks.ago) }
let(:start_date) { 5.weeks.ago.to_date.to_s }
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:node_path) { ['averageCoverage'] }
let(:sort_param) { } let(:sort_param) { }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [88.8, 77.0] } let(:expected_results) { [cov_1, cov_2, cov_3, cov_4, cov_5].reverse.map(&:coverage) }
end end
end end
end end
......
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe 'Namespace.projects' do
include GraphqlHelpers
describe 'sorting and pagination' do
let_it_be(:ns) { create(:group) }
let_it_be(:current_user) { create(:user) }
let!(:project_1) { create(:project, namespace: ns, name: 'Project', path: 'project') }
let!(:project_2) { create(:project, namespace: ns, name: 'Test Project', path: 'test-project') }
let!(:project_3) { create(:project, namespace: ns, name: 'Test', path: 'test') }
let!(:project_4) { create(:project, namespace: ns, name: 'Test Project Other', path: 'other-test-project') }
let(:data_path) { [:namespace, :projects] }
let(:ns_args) { graphql_args(full_path: ns.full_path) }
let(:project_args) { graphql_args(include_subgroups: true, search: 'test') }
before do
ns.add_owner(current_user)
end
def pagination_query(params)
graphql_query_for(:namespace, ns_args,
query_nodes(:projects, :name, include_pagination_info: true, args: params + project_args))
end
context 'when sorting by STORAGE' do
before do
project_4.statistics.update!(lfs_objects_size: 1, repository_size: 4.gigabytes)
project_2.statistics.update!(lfs_objects_size: 1, repository_size: 2.gigabytes)
project_3.statistics.update!(lfs_objects_size: 2, repository_size: 1.gigabytes)
end
it_behaves_like 'sorted paginated query' do
let(:node_path) { %w[name] }
let(:sort_param) { :STORAGE }
let(:first_param) { 2 }
let(:expected_results) { [project_4, project_2, project_3].map(&:name) }
end
end
end
end
...@@ -11,18 +11,12 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -11,18 +11,12 @@ RSpec.describe 'getting an issue list for a project' do
let(:sort_project) { create(:project, :public) } let(:sort_project) { create(:project, :public) }
let(:data_path) { [:project, :issues] } let(:data_path) { [:project, :issues] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for(:project, { full_path: sort_project.full_path },
'project', query_nodes(:issues, :iid, args: params, include_pagination_info: true)
{ 'fullPath' => sort_project.full_path },
query_graphql_field('issues', params, "#{page_info} edges { node { iid weight} }")
) )
end end
def pagination_results_data(data)
data.map { |issue| issue.dig('node', 'iid').to_i }
end
context 'when sorting by weight' do context 'when sorting by weight' do
let!(:weight_issue1) { create(:issue, project: sort_project, weight: 5) } let!(:weight_issue1) { create(:issue, project: sort_project, weight: 5) }
let!(:weight_issue2) { create(:issue, project: sort_project, weight: nil) } let!(:weight_issue2) { create(:issue, project: sort_project, weight: nil) }
...@@ -32,17 +26,19 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -32,17 +26,19 @@ RSpec.describe 'getting an issue list for a project' do
context 'when ascending' do context 'when ascending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'WEIGHT_ASC' } let(:node_path) { %w[iid] }
let(:sort_param) { :WEIGHT_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [weight_issue3.iid, weight_issue5.iid, weight_issue1.iid, weight_issue4.iid, weight_issue2.iid] } let(:expected_results) { [weight_issue3, weight_issue5, weight_issue1, weight_issue4, weight_issue2].map { |i| i.iid.to_s } }
end end
end end
context 'when descending' do context 'when descending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'WEIGHT_DESC' } let(:node_path) { %w[iid] }
let(:sort_param) { :WEIGHT_DESC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [weight_issue1.iid, weight_issue5.iid, weight_issue3.iid, weight_issue4.iid, weight_issue2.iid] } let(:expected_results) { [weight_issue1, weight_issue5, weight_issue3, weight_issue4, weight_issue2].map { |i| i.iid.to_s } }
end end
end end
end end
......
...@@ -161,16 +161,12 @@ RSpec.describe 'getting a requirement list for a project' do ...@@ -161,16 +161,12 @@ RSpec.describe 'getting a requirement list for a project' do
describe 'sorting and pagination' do describe 'sorting and pagination' do
let_it_be(:data_path) { [:project, :requirements] } let_it_be(:data_path) { [:project, :requirements] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( nested_internal_id_query(:project, sort_project, :requirements, params)
'project',
{ 'fullPath' => sort_project.full_path },
query_graphql_field('requirements', params, "#{page_info} edges { node { iid createdAt} }")
)
end end
def pagination_results_data(data) def pagination_results_data(data)
data.map { |issue| issue.dig('node', 'iid').to_i } data.map { |issue| issue.dig('iid').to_i }
end end
context 'when sorting by created_at' do context 'when sorting by created_at' do
...@@ -183,7 +179,7 @@ RSpec.describe 'getting a requirement list for a project' do ...@@ -183,7 +179,7 @@ RSpec.describe 'getting a requirement list for a project' do
context 'when ascending' do context 'when ascending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'created_asc' } let(:sort_param) { :CREATED_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [requirement4.iid, requirement3.iid, requirement5.iid, requirement1.iid, requirement2.iid] } let(:expected_results) { [requirement4.iid, requirement3.iid, requirement5.iid, requirement1.iid, requirement2.iid] }
end end
...@@ -191,7 +187,7 @@ RSpec.describe 'getting a requirement list for a project' do ...@@ -191,7 +187,7 @@ RSpec.describe 'getting a requirement list for a project' do
context 'when descending' do context 'when descending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'created_desc' } let(:sort_param) { :CREATED_DESC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [requirement2.iid, requirement1.iid, requirement5.iid, requirement3.iid, requirement4.iid] } let(:expected_results) { [requirement2.iid, requirement1.iid, requirement5.iid, requirement3.iid, requirement4.iid] }
end end
......
...@@ -63,27 +63,29 @@ RSpec.describe 'getting test reports of a requirement' do ...@@ -63,27 +63,29 @@ RSpec.describe 'getting test reports of a requirement' do
let_it_be(:data_path) { [:project, :requirement, :testReports] } let_it_be(:data_path) { [:project, :requirement, :testReports] }
let_it_be(:test_report_3) { create(:test_report, requirement: requirement, created_at: 4.days.ago) } let_it_be(:test_report_3) { create(:test_report, requirement: requirement, created_at: 4.days.ago) }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for(:project, { full_path: project.full_path },
'project', "requirement { testReports(#{params}) { #{page_info} nodes { id } } }"
{ 'fullPath' => project.full_path },
"requirement { testReports(#{params}) { #{page_info} edges { node { id } } } }"
) )
end end
def pagination_results_data(data) let(:in_creation_order) do
data.map { |test_report| test_report.dig('node', 'id') } [test_report_3, test_report_2, test_report_1]
end end
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'created_asc' } let(:sort_param) { :CREATED_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) do let(:expected_results) do
[ in_creation_order.map { |r| global_id_of(r) }
test_report_3.to_global_id.to_s, end
test_report_2.to_global_id.to_s, end
test_report_1.to_global_id.to_s
] it_behaves_like 'sorted paginated query' do
let(:sort_param) { :CREATED_DESC }
let(:first_param) { 2 }
let(:expected_results) do
in_creation_order.reverse.map { |r| global_id_of(r) }
end end
end end
end end
......
...@@ -87,40 +87,39 @@ class Feature ...@@ -87,40 +87,39 @@ class Feature
end end
def enable(key, thing = true) def enable(key, thing = true)
log(key: key, action: __method__, thing: thing)
get(key).enable(thing) get(key).enable(thing)
end end
def disable(key, thing = false) def disable(key, thing = false)
log(key: key, action: __method__, thing: thing)
get(key).disable(thing) get(key).disable(thing)
end end
def enable_group(key, group)
get(key).enable_group(group)
end
def disable_group(key, group)
get(key).disable_group(group)
end
def enable_percentage_of_time(key, percentage) def enable_percentage_of_time(key, percentage)
log(key: key, action: __method__, percentage: percentage)
get(key).enable_percentage_of_time(percentage) get(key).enable_percentage_of_time(percentage)
end end
def disable_percentage_of_time(key) def disable_percentage_of_time(key)
log(key: key, action: __method__)
get(key).disable_percentage_of_time get(key).disable_percentage_of_time
end end
def enable_percentage_of_actors(key, percentage) def enable_percentage_of_actors(key, percentage)
log(key: key, action: __method__, percentage: percentage)
get(key).enable_percentage_of_actors(percentage) get(key).enable_percentage_of_actors(percentage)
end end
def disable_percentage_of_actors(key) def disable_percentage_of_actors(key)
log(key: key, action: __method__)
get(key).disable_percentage_of_actors get(key).disable_percentage_of_actors
end end
def remove(key) def remove(key)
return unless persisted_name?(key) return unless persisted_name?(key)
log(key: key, action: __method__)
get(key).remove get(key).remove
end end
...@@ -145,6 +144,10 @@ class Feature ...@@ -145,6 +144,10 @@ class Feature
Feature::Definition.register_hot_reloader! Feature::Definition.register_hot_reloader!
end end
def logger
@logger ||= Feature::Logger.build
end
private private
def flipper def flipper
...@@ -192,6 +195,14 @@ class Feature ...@@ -192,6 +195,14 @@ class Feature
def l2_cache_backend def l2_cache_backend
Rails.cache Rails.cache
end end
def log(key:, action:, **extra)
extra ||= {}
extra = extra.transform_keys { |k| "extra.#{k}" }
extra = extra.transform_values { |v| v.respond_to?(:flipper_id) ? v.flipper_id : v }
extra = extra.transform_values(&:to_s)
logger.info(key: key, action: action, **extra)
end
end end
class Target class Target
......
# frozen_string_literal: true
class Feature
class Logger < ::Gitlab::JsonLogger
def self.file_name_noext
'features_json'
end
end
end
...@@ -300,7 +300,119 @@ RSpec.describe Feature, stub_feature_flags: false do ...@@ -300,7 +300,119 @@ RSpec.describe Feature, stub_feature_flags: false do
end end
end end
shared_examples_for 'logging' do
let(:expected_action) { }
let(:expected_extra) { }
it 'logs the event' do
expect(Feature.logger).to receive(:info).with(key: key, action: expected_action, **expected_extra)
subject
end
end
describe '.enable' do
subject { described_class.enable(key, thing) }
let(:key) { :awesome_feature }
let(:thing) { true }
it_behaves_like 'logging' do
let(:expected_action) { :enable }
let(:expected_extra) { { "extra.thing" => "true" } }
end
context 'when thing is an actor' do
let(:thing) { create(:project) }
it_behaves_like 'logging' do
let(:expected_action) { :enable }
let(:expected_extra) { { "extra.thing" => "#{thing.flipper_id}" } }
end
end
end
describe '.disable' do
subject { described_class.disable(key, thing) }
let(:key) { :awesome_feature }
let(:thing) { false }
it_behaves_like 'logging' do
let(:expected_action) { :disable }
let(:expected_extra) { { "extra.thing" => "false" } }
end
context 'when thing is an actor' do
let(:thing) { create(:project) }
it_behaves_like 'logging' do
let(:expected_action) { :disable }
let(:expected_extra) { { "extra.thing" => "#{thing.flipper_id}" } }
end
end
end
describe '.enable_percentage_of_time' do
subject { described_class.enable_percentage_of_time(key, percentage) }
let(:key) { :awesome_feature }
let(:percentage) { 50 }
it_behaves_like 'logging' do
let(:expected_action) { :enable_percentage_of_time }
let(:expected_extra) { { "extra.percentage" => "#{percentage}" } }
end
end
describe '.disable_percentage_of_time' do
subject { described_class.disable_percentage_of_time(key) }
let(:key) { :awesome_feature }
it_behaves_like 'logging' do
let(:expected_action) { :disable_percentage_of_time }
let(:expected_extra) { {} }
end
end
describe '.enable_percentage_of_actors' do
subject { described_class.enable_percentage_of_actors(key, percentage) }
let(:key) { :awesome_feature }
let(:percentage) { 50 }
it_behaves_like 'logging' do
let(:expected_action) { :enable_percentage_of_actors }
let(:expected_extra) { { "extra.percentage" => "#{percentage}" } }
end
end
describe '.disable_percentage_of_actors' do
subject { described_class.disable_percentage_of_actors(key) }
let(:key) { :awesome_feature }
it_behaves_like 'logging' do
let(:expected_action) { :disable_percentage_of_actors }
let(:expected_extra) { {} }
end
end
describe '.remove' do describe '.remove' do
subject { described_class.remove(key) }
let(:key) { :awesome_feature }
before do
described_class.enable(key)
end
it_behaves_like 'logging' do
let(:expected_action) { :remove }
let(:expected_extra) { {} }
end
context 'for a non-persisted feature' do context 'for a non-persisted feature' do
it 'returns nil' do it 'returns nil' do
expect(described_class.remove(:non_persisted_feature_flag)).to be_nil expect(described_class.remove(:non_persisted_feature_flag)).to be_nil
......
...@@ -66,28 +66,22 @@ RSpec.describe 'get board lists' do ...@@ -66,28 +66,22 @@ RSpec.describe 'get board lists' do
describe 'sorting and pagination' do describe 'sorting and pagination' do
let_it_be(:current_user) { user } let_it_be(:current_user) { user }
let(:data_path) { [board_parent_type, :boards, :edges, 0, :node, :lists] } let(:data_path) { [board_parent_type, :boards, :nodes, 0, :lists] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for(
board_parent_type, board_parent_type,
{ 'fullPath' => board_parent.full_path }, { 'fullPath' => board_parent.full_path },
<<~BOARDS <<~BOARDS
boards(first: 1) { boards(first: 1) {
edges { nodes {
node { #{query_graphql_field(:lists, params, "#{page_info} nodes { id }")}
#{query_graphql_field('lists', params, "#{page_info} edges { node { id } }")}
}
} }
} }
BOARDS BOARDS
) )
end end
def pagination_results_data(data)
data.map { |list| list.dig('node', 'id') }
end
context 'when using default sorting' do context 'when using default sorting' do
let!(:label_list) { create(:list, board: board, label: label, position: 10) } let!(:label_list) { create(:list, board: board, label: label, position: 10) }
let!(:label_list2) { create(:list, board: board, label: label2, position: 2) } let!(:label_list2) { create(:list, board: board, label: label2, position: 2) }
...@@ -99,7 +93,7 @@ RSpec.describe 'get board lists' do ...@@ -99,7 +93,7 @@ RSpec.describe 'get board lists' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { } let(:sort_param) { }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { lists.map { |list| list.to_global_id.to_s } } let(:expected_results) { lists.map { |list| global_id_of(list) } }
end end
end end
end end
......
...@@ -80,38 +80,34 @@ RSpec.describe 'getting projects' do ...@@ -80,38 +80,34 @@ RSpec.describe 'getting projects' do
end end
describe 'sorting and pagination' do describe 'sorting and pagination' do
let_it_be(:ns) { create(:group) }
let_it_be(:current_user) { create(:user) }
let_it_be(:project_1) { create(:project, name: 'Project', path: 'project', namespace: ns) }
let_it_be(:project_2) { create(:project, name: 'Test Project', path: 'test-project', namespace: ns) }
let_it_be(:project_3) { create(:project, name: 'Test', path: 'test', namespace: ns) }
let_it_be(:project_4) { create(:project, name: 'Test Project Other', path: 'other-test-project', namespace: ns) }
let(:data_path) { [:namespace, :projects] } let(:data_path) { [:namespace, :projects] }
def pagination_query(params, page_info) let(:ns_args) { { full_path: ns.full_path } }
graphql_query_for( let(:search) { 'test' }
'namespace',
{ 'fullPath' => subject.full_path }, before do
<<~QUERY ns.add_owner(current_user)
projects(includeSubgroups: #{include_subgroups}, search: "#{search}", #{params}) {
#{page_info} edges {
node {
#{all_graphql_fields_for('Project')}
}
}
}
QUERY
)
end end
def pagination_results_data(data) def pagination_query(params)
data.map { |project| project.dig('node', 'name') } arguments = params.merge(include_subgroups: include_subgroups, search: search)
graphql_query_for(:namespace, ns_args, query_graphql_field(:projects, arguments, <<~GQL))
#{page_info}
nodes { name }
GQL
end end
context 'when sorting by similarity' do context 'when sorting by similarity' do
let!(:project_1) { create(:project, name: 'Project', path: 'project', namespace: subject) }
let!(:project_2) { create(:project, name: 'Test Project', path: 'test-project', namespace: subject) }
let!(:project_3) { create(:project, name: 'Test', path: 'test', namespace: subject) }
let!(:project_4) { create(:project, name: 'Test Project Other', path: 'other-test-project', namespace: subject) }
let(:search) { 'test' }
let(:current_user) { user }
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'SIMILARITY' } let(:node_path) { %w[name] }
let(:sort_param) { :SIMILARITY }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [project_3.name, project_2.name, project_4.name] } let(:expected_results) { [project_3.name, project_2.name, project_4.name] }
end end
......
...@@ -142,16 +142,14 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -142,16 +142,14 @@ RSpec.describe 'getting an issue list for a project' do
describe 'sorting and pagination' do describe 'sorting and pagination' do
let_it_be(:data_path) { [:project, :issues] } let_it_be(:data_path) { [:project, :issues] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for(:project, { full_path: sort_project.full_path },
'project', query_graphql_field(:issues, params, "#{page_info} nodes { iid }")
{ 'fullPath' => sort_project.full_path },
query_graphql_field('issues', params, "#{page_info} edges { node { iid dueDate} }")
) )
end end
def pagination_results_data(data) def pagination_results_data(data)
data.map { |issue| issue.dig('node', 'iid').to_i } data.map { |issue| issue.dig('iid').to_i }
end end
context 'when sorting by due date' do context 'when sorting by due date' do
...@@ -164,7 +162,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -164,7 +162,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when ascending' do context 'when ascending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'DUE_DATE_ASC' } let(:sort_param) { :DUE_DATE_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [due_issue3.iid, due_issue5.iid, due_issue1.iid, due_issue4.iid, due_issue2.iid] } let(:expected_results) { [due_issue3.iid, due_issue5.iid, due_issue1.iid, due_issue4.iid, due_issue2.iid] }
end end
...@@ -172,7 +170,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -172,7 +170,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when descending' do context 'when descending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'DUE_DATE_DESC' } let(:sort_param) { :DUE_DATE_DESC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [due_issue1.iid, due_issue5.iid, due_issue3.iid, due_issue4.iid, due_issue2.iid] } let(:expected_results) { [due_issue1.iid, due_issue5.iid, due_issue3.iid, due_issue4.iid, due_issue2.iid] }
end end
...@@ -189,7 +187,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -189,7 +187,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when ascending' do context 'when ascending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'RELATIVE_POSITION_ASC' } let(:sort_param) { :RELATIVE_POSITION_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [relative_issue5.iid, relative_issue3.iid, relative_issue1.iid, relative_issue4.iid, relative_issue2.iid] } let(:expected_results) { [relative_issue5.iid, relative_issue3.iid, relative_issue1.iid, relative_issue4.iid, relative_issue2.iid] }
end end
...@@ -209,7 +207,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -209,7 +207,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when ascending' do context 'when ascending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'PRIORITY_ASC' } let(:sort_param) { :PRIORITY_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [priority_issue3.iid, priority_issue1.iid, priority_issue2.iid, priority_issue4.iid] } let(:expected_results) { [priority_issue3.iid, priority_issue1.iid, priority_issue2.iid, priority_issue4.iid] }
end end
...@@ -217,7 +215,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -217,7 +215,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when descending' do context 'when descending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'PRIORITY_DESC' } let(:sort_param) { :PRIORITY_DESC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [priority_issue1.iid, priority_issue3.iid, priority_issue2.iid, priority_issue4.iid] } let(:expected_results) { [priority_issue1.iid, priority_issue3.iid, priority_issue2.iid, priority_issue4.iid] }
end end
...@@ -236,7 +234,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -236,7 +234,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when ascending' do context 'when ascending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'LABEL_PRIORITY_ASC' } let(:sort_param) { :LABEL_PRIORITY_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [label_issue3.iid, label_issue1.iid, label_issue2.iid, label_issue4.iid] } let(:expected_results) { [label_issue3.iid, label_issue1.iid, label_issue2.iid, label_issue4.iid] }
end end
...@@ -244,7 +242,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -244,7 +242,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when descending' do context 'when descending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'LABEL_PRIORITY_DESC' } let(:sort_param) { :LABEL_PRIORITY_DESC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [label_issue2.iid, label_issue3.iid, label_issue1.iid, label_issue4.iid] } let(:expected_results) { [label_issue2.iid, label_issue3.iid, label_issue1.iid, label_issue4.iid] }
end end
...@@ -261,7 +259,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -261,7 +259,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when ascending' do context 'when ascending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'MILESTONE_DUE_ASC' } let(:sort_param) { :MILESTONE_DUE_ASC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [milestone_issue2.iid, milestone_issue3.iid, milestone_issue1.iid] } let(:expected_results) { [milestone_issue2.iid, milestone_issue3.iid, milestone_issue1.iid] }
end end
...@@ -269,7 +267,7 @@ RSpec.describe 'getting an issue list for a project' do ...@@ -269,7 +267,7 @@ RSpec.describe 'getting an issue list for a project' do
context 'when descending' do context 'when descending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'MILESTONE_DUE_DESC' } let(:sort_param) { :MILESTONE_DUE_DESC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) { [milestone_issue3.iid, milestone_issue2.iid, milestone_issue1.iid] } let(:expected_results) { [milestone_issue3.iid, milestone_issue2.iid, milestone_issue1.iid] }
end end
......
...@@ -259,29 +259,19 @@ RSpec.describe 'getting merge request listings nested in a project' do ...@@ -259,29 +259,19 @@ RSpec.describe 'getting merge request listings nested in a project' do
describe 'sorting and pagination' do describe 'sorting and pagination' do
let(:data_path) { [:project, :mergeRequests] } let(:data_path) { [:project, :mergeRequests] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for(:project, { full_path: project.full_path },
:project,
{ full_path: project.full_path },
<<~QUERY <<~QUERY
mergeRequests(#{params}) { mergeRequests(#{params}) {
#{page_info} edges { #{page_info} nodes { id }
node {
id
}
}
} }
QUERY QUERY
) )
end end
def pagination_results_data(data)
data.map { |project| project.dig('node', 'id') }
end
context 'when sorting by merged_at DESC' do context 'when sorting by merged_at DESC' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'MERGED_AT_DESC' } let(:sort_param) { :MERGED_AT_DESC }
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) do let(:expected_results) do
...@@ -291,7 +281,7 @@ RSpec.describe 'getting merge request listings nested in a project' do ...@@ -291,7 +281,7 @@ RSpec.describe 'getting merge request listings nested in a project' do
merge_request_c, merge_request_c,
merge_request_e, merge_request_e,
merge_request_a merge_request_a
].map(&:to_gid).map(&:to_s) ].map { |mr| global_id_of(mr) }
end end
before do before do
...@@ -304,33 +294,6 @@ RSpec.describe 'getting merge request listings nested in a project' do ...@@ -304,33 +294,6 @@ RSpec.describe 'getting merge request listings nested in a project' do
merge_request_b.metrics.update!(merged_at: 1.day.ago) merge_request_b.metrics.update!(merged_at: 1.day.ago)
end end
context 'when paginating backwards' do
let(:params) { 'first: 2, sort: MERGED_AT_DESC' }
let(:page_info) { 'pageInfo { startCursor endCursor }' }
before do
post_graphql(pagination_query(params, page_info), current_user: current_user)
end
it 'paginates backwards correctly' do
# first page
first_page_response_data = graphql_dig_at(Gitlab::Json.parse(response.body), :data, *data_path, :edges)
end_cursor = graphql_dig_at(Gitlab::Json.parse(response.body), :data, :project, :mergeRequests, :pageInfo, :endCursor)
# second page
params = "first: 2, after: \"#{end_cursor}\", sort: MERGED_AT_DESC"
post_graphql(pagination_query(params, page_info), current_user: current_user)
start_cursor = graphql_dig_at(Gitlab::Json.parse(response.body), :data, :project, :mergeRequests, :pageInfo, :start_cursor)
# going back to the first page
params = "last: 2, before: \"#{start_cursor}\", sort: MERGED_AT_DESC"
post_graphql(pagination_query(params, page_info), current_user: current_user)
backward_paginated_response_data = graphql_dig_at(Gitlab::Json.parse(response.body), :data, *data_path, :edges)
expect(first_page_response_data).to eq(backward_paginated_response_data)
end
end
end end
end end
end end
......
...@@ -59,20 +59,16 @@ RSpec.describe 'Users' do ...@@ -59,20 +59,16 @@ RSpec.describe 'Users' do
describe 'sorting and pagination' do describe 'sorting and pagination' do
let_it_be(:data_path) { [:users] } let_it_be(:data_path) { [:users] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for("users", params, "#{page_info} edges { node { id } }") graphql_query_for(:users, params, "#{page_info} nodes { id }")
end
def pagination_results_data(data)
data.map { |user| user.dig('node', 'id') }
end end
context 'when sorting by created_at' do context 'when sorting by created_at' do
let_it_be(:ascending_users) { [user3, user2, user1, current_user].map(&:to_global_id).map(&:to_s) } let_it_be(:ascending_users) { [user3, user2, user1, current_user].map { |u| global_id_of(u) } }
context 'when ascending' do context 'when ascending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'created_asc' } let(:sort_param) { :CREATED_ASC }
let(:first_param) { 1 } let(:first_param) { 1 }
let(:expected_results) { ascending_users } let(:expected_results) { ascending_users }
end end
...@@ -80,7 +76,7 @@ RSpec.describe 'Users' do ...@@ -80,7 +76,7 @@ RSpec.describe 'Users' do
context 'when descending' do context 'when descending' do
it_behaves_like 'sorted paginated query' do it_behaves_like 'sorted paginated query' do
let(:sort_param) { 'created_desc' } let(:sort_param) { :CREATED_DESC }
let(:first_param) { 1 } let(:first_param) { 1 }
let(:expected_results) { ascending_users.reverse } let(:expected_results) { ascending_users.reverse }
end end
......
...@@ -16,80 +16,111 @@ ...@@ -16,80 +16,111 @@
# #
# Example: # Example:
# describe 'sorting and pagination' do # describe 'sorting and pagination' do
# let(:sort_project) { create(:project, :public) } # let_it_be(:sort_project) { create(:project, :public) }
# let(:data_path) { [:project, :issues] } # let(:data_path) { [:project, :issues] }
# #
# def pagination_query(params, page_info) # def pagination_query(arguments)
# graphql_query_for( # graphql_query_for(:project, { full_path: sort_project.full_path },
# 'project', # query_nodes(:issues, :iid, include_pagination_info: true, args: arguments)
# { 'fullPath' => sort_project.full_path },
# query_graphql_field('issues', params, "#{page_info} edges { node { id } }")
# ) # )
# end # end
# #
# def pagination_results_data(data) # # A method transforming nodes to data to match against
# data.map { |issue| issue.dig('node', 'iid').to_i } # # default: the identity function
# def pagination_results_data(issues)
# issues.map { |issue| issue['iid].to_i }
# end # end
# #
# context 'when sorting by weight' do # context 'when sorting by weight' do
# ... # let_it_be(:issues) { make_some_issues_with_weights }
#
# context 'when ascending' do # context 'when ascending' do
# let(:ordered_issues) { issues.sort_by(&:weight) }
#
# it_behaves_like 'sorted paginated query' do # it_behaves_like 'sorted paginated query' do
# let(:sort_param) { 'WEIGHT_ASC' } # let(:sort_param) { :WEIGHT_ASC }
# let(:first_param) { 2 } # let(:first_param) { 2 }
# let(:expected_results) { [weight_issue3.iid, weight_issue5.iid, weight_issue1.iid, weight_issue4.iid, weight_issue2.iid] } # let(:expected_results) { ordered_issues.map(&:iid) }
# end # end
# end # end
# #
RSpec.shared_examples 'sorted paginated query' do RSpec.shared_examples 'sorted paginated query' do
# Provided as a convenience when constructing queries using string concatenation
let(:page_info) { 'pageInfo { startCursor endCursor }' }
# Convenience for using default implementation of pagination_results_data
let(:node_path) { ['id'] }
it_behaves_like 'requires variables' do it_behaves_like 'requires variables' do
let(:required_variables) { [:sort_param, :first_param, :expected_results, :data_path, :current_user] } let(:required_variables) { [:sort_param, :first_param, :expected_results, :data_path, :current_user] }
end end
describe do describe do
let(:sort_argument) { "sort: #{sort_param}" if sort_param.present? } let(:sort_argument) { graphql_args(sort: sort_param) }
let(:first_argument) { "first: #{first_param}" if first_param.present? }
let(:params) { sort_argument } let(:params) { sort_argument }
let(:start_cursor) { graphql_data_at(*data_path, :pageInfo, :startCursor) }
let(:end_cursor) { graphql_data_at(*data_path, :pageInfo, :endCursor) }
let(:sorted_edges) { graphql_data_at(*data_path, :edges) }
let(:page_info) { "pageInfo { startCursor endCursor }" }
def pagination_query(params, page_info) # Convenience helper for the large number of queries defined as a projection
raise('pagination_query(params, page_info) must be defined in the test, see example in comment') unless defined?(super) # from some root value indexed by full_path to a collection of objects with IID
def nested_internal_id_query(root_field, parent, field, args, selection: :iid)
graphql_query_for(root_field, { full_path: parent.full_path },
query_nodes(field, selection, args: args, include_pagination_info: true)
)
end
def pagination_query(params)
raise('pagination_query(params) must be defined in the test, see example in comment') unless defined?(super)
super super
end end
def pagination_results_data(data) def pagination_results_data(nodes)
raise('pagination_results_data(data) must be defined in the test, see example in comment') unless defined?(super) if defined?(super)
super(nodes)
else
nodes.map { |n| n.dig(*node_path) }
end
end
def results
nodes = graphql_dig_at(graphql_data(fresh_response_data), *data_path, :nodes)
pagination_results_data(nodes)
end
def end_cursor
graphql_dig_at(graphql_data(fresh_response_data), *data_path, :page_info, :end_cursor)
end
super(data) def start_cursor
graphql_dig_at(graphql_data(fresh_response_data), *data_path, :page_info, :start_cursor)
end end
let(:query) { pagination_query(params) }
before do before do
post_graphql(pagination_query(params, page_info), current_user: current_user) post_graphql(query, current_user: current_user)
end end
context 'when sorting' do context 'when sorting' do
it 'sorts correctly' do it 'sorts correctly' do
expect(pagination_results_data(sorted_edges)).to eq expected_results expect(results).to eq expected_results
end end
context 'when paginating' do context 'when paginating' do
let(:params) { [sort_argument, first_argument].compact.join(',') } let(:params) { sort_argument.merge(first: first_param) }
let(:first_page) { expected_results.first(first_param) }
let(:rest) { expected_results.drop(first_param) }
it 'paginates correctly' do it 'paginates correctly' do
expect(pagination_results_data(sorted_edges)).to eq expected_results.first(first_param) expect(results).to eq first_page
cursored_query = pagination_query([sort_argument, "after: \"#{end_cursor}\""].compact.join(','), page_info) fwds = pagination_query(sort_argument.merge(after: end_cursor))
post_graphql(cursored_query, current_user: current_user) post_graphql(fwds, current_user: current_user)
expect(response).to have_gitlab_http_status(:ok) expect(results).to eq rest
response_data = graphql_dig_at(Gitlab::Json.parse(response.body), :data, *data_path, :edges) bwds = pagination_query(sort_argument.merge(before: start_cursor))
post_graphql(bwds, current_user: current_user)
expect(pagination_results_data(response_data)).to eq expected_results.drop(first_param) expect(results).to eq first_page
end end
end end
end end
......
...@@ -47,18 +47,12 @@ RSpec.shared_examples 'group and project boards query' do ...@@ -47,18 +47,12 @@ RSpec.shared_examples 'group and project boards query' do
describe 'sorting and pagination' do describe 'sorting and pagination' do
let(:data_path) { [board_parent_type, :boards] } let(:data_path) { [board_parent_type, :boards] }
def pagination_query(params, page_info) def pagination_query(params)
graphql_query_for( graphql_query_for(board_parent_type, { full_path: board_parent.full_path },
board_parent_type, query_nodes(:boards, :id, include_pagination_info: true, args: params)
{ 'fullPath' => board_parent.full_path },
query_graphql_field('boards', params, "#{page_info} edges { node { id } }")
) )
end end
def pagination_results_data(data)
data.map { |board| board.dig('node', 'id') }
end
context 'when using default sorting' do context 'when using default sorting' do
let!(:board_B) { create(:board, resource_parent: board_parent, name: 'B') } let!(:board_B) { create(:board, resource_parent: board_parent, name: 'B') }
let!(:board_C) { create(:board, resource_parent: board_parent, name: 'C') } let!(:board_C) { create(:board, resource_parent: board_parent, name: 'C') }
...@@ -72,9 +66,9 @@ RSpec.shared_examples 'group and project boards query' do ...@@ -72,9 +66,9 @@ RSpec.shared_examples 'group and project boards query' do
let(:first_param) { 2 } let(:first_param) { 2 }
let(:expected_results) do let(:expected_results) do
if board_parent.multiple_issue_boards_available? if board_parent.multiple_issue_boards_available?
boards.map { |board| board.to_global_id.to_s } boards.map { |board| global_id_of(board) }
else else
[boards.first.to_global_id.to_s] [global_id_of(boards.first)]
end end
end end
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment