1639 lines
52 KiB
Ruby
1639 lines
52 KiB
Ruby
|
# frozen_string_literal: true
|
||
|
|
||
|
require 'spec_helper'
|
||
|
|
||
|
RSpec.describe Projects::FeatureFlagsController do
|
||
|
include Gitlab::Routing
|
||
|
include FeatureFlagHelpers
|
||
|
|
||
|
let_it_be(:project) { create(:project) }
|
||
|
let_it_be(:developer) { create(:user) }
|
||
|
let_it_be(:reporter) { create(:user) }
|
||
|
let(:user) { developer }
|
||
|
|
||
|
before_all do
|
||
|
project.add_developer(developer)
|
||
|
project.add_reporter(reporter)
|
||
|
end
|
||
|
|
||
|
before do
|
||
|
sign_in(user)
|
||
|
end
|
||
|
|
||
|
describe 'GET index' do
|
||
|
render_views
|
||
|
|
||
|
subject { get(:index, params: view_params) }
|
||
|
|
||
|
context 'when there is no feature flags' do
|
||
|
it 'responds with success' do
|
||
|
is_expected.to have_gitlab_http_status(:ok)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'for a list of feature flags' do
|
||
|
let!(:feature_flags) { create_list(:operations_feature_flag, 50, project: project) }
|
||
|
|
||
|
it 'responds with success' do
|
||
|
is_expected.to have_gitlab_http_status(:ok)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when the user is a reporter' do
|
||
|
let(:user) { reporter }
|
||
|
|
||
|
it 'responds with not found' do
|
||
|
is_expected.to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
describe 'GET #index.json' do
|
||
|
subject { get(:index, params: view_params, format: :json) }
|
||
|
|
||
|
let!(:feature_flag_active) do
|
||
|
create(:operations_feature_flag, project: project, active: true, name: 'feature_flag_a')
|
||
|
end
|
||
|
|
||
|
let!(:feature_flag_inactive) do
|
||
|
create(:operations_feature_flag, project: project, active: false, name: 'feature_flag_b')
|
||
|
end
|
||
|
|
||
|
it 'returns all feature flags as json response' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['feature_flags'].count).to eq(2)
|
||
|
expect(json_response['feature_flags'].first['name']).to eq(feature_flag_active.name)
|
||
|
expect(json_response['feature_flags'].second['name']).to eq(feature_flag_inactive.name)
|
||
|
end
|
||
|
|
||
|
it 'returns CRUD paths' do
|
||
|
subject
|
||
|
|
||
|
expected_edit_path = edit_project_feature_flag_path(project, feature_flag_active)
|
||
|
expected_update_path = project_feature_flag_path(project, feature_flag_active)
|
||
|
expected_destroy_path = project_feature_flag_path(project, feature_flag_active)
|
||
|
|
||
|
feature_flag_json = json_response['feature_flags'].first
|
||
|
|
||
|
expect(feature_flag_json['edit_path']).to eq(expected_edit_path)
|
||
|
expect(feature_flag_json['update_path']).to eq(expected_update_path)
|
||
|
expect(feature_flag_json['destroy_path']).to eq(expected_destroy_path)
|
||
|
end
|
||
|
|
||
|
it 'returns the summary of feature flags' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['count']['all']).to eq(2)
|
||
|
expect(json_response['count']['enabled']).to eq(1)
|
||
|
expect(json_response['count']['disabled']).to eq(1)
|
||
|
end
|
||
|
|
||
|
it 'matches json schema' do
|
||
|
is_expected.to match_response_schema('feature_flags')
|
||
|
end
|
||
|
|
||
|
it 'returns false for active when the feature flag is inactive even if it has an active scope' do
|
||
|
create(:operations_feature_flag_scope,
|
||
|
feature_flag: feature_flag_inactive,
|
||
|
environment_scope: 'production',
|
||
|
active: true)
|
||
|
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
feature_flag_json = json_response['feature_flags'].second
|
||
|
|
||
|
expect(feature_flag_json['active']).to eq(false)
|
||
|
end
|
||
|
|
||
|
it 'returns the feature flag iid' do
|
||
|
subject
|
||
|
|
||
|
feature_flag_json = json_response['feature_flags'].first
|
||
|
|
||
|
expect(feature_flag_json['iid']).to eq(feature_flag_active.iid)
|
||
|
end
|
||
|
|
||
|
context 'when scope is specified' do
|
||
|
let(:view_params) do
|
||
|
{ namespace_id: project.namespace, project_id: project, scope: scope }
|
||
|
end
|
||
|
|
||
|
context 'when all feature flags are requested' do
|
||
|
let(:scope) { 'all' }
|
||
|
|
||
|
it 'returns all feature flags' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['feature_flags'].count).to eq(2)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when enabled feature flags are requested' do
|
||
|
let(:scope) { 'enabled' }
|
||
|
|
||
|
it 'returns enabled feature flags' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['feature_flags'].count).to eq(1)
|
||
|
expect(json_response['feature_flags'].first['active']).to be_truthy
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when disabled feature flags are requested' do
|
||
|
let(:scope) { 'disabled' }
|
||
|
|
||
|
it 'returns disabled feature flags' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['feature_flags'].count).to eq(1)
|
||
|
expect(json_response['feature_flags'].first['active']).to be_falsy
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when feature flags have additional scopes' do
|
||
|
let!(:feature_flag_active_scope) do
|
||
|
create(:operations_feature_flag_scope,
|
||
|
feature_flag: feature_flag_active,
|
||
|
environment_scope: 'production',
|
||
|
active: false)
|
||
|
end
|
||
|
|
||
|
let!(:feature_flag_inactive_scope) do
|
||
|
create(:operations_feature_flag_scope,
|
||
|
feature_flag: feature_flag_inactive,
|
||
|
environment_scope: 'staging',
|
||
|
active: false)
|
||
|
end
|
||
|
|
||
|
it 'returns a correct summary' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['count']['all']).to eq(2)
|
||
|
expect(json_response['count']['enabled']).to eq(1)
|
||
|
expect(json_response['count']['disabled']).to eq(1)
|
||
|
end
|
||
|
|
||
|
it 'recognizes feature flag 1 as active' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['feature_flags'].first['active']).to be_truthy
|
||
|
end
|
||
|
|
||
|
it 'recognizes feature flag 2 as inactive' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['feature_flags'].second['active']).to be_falsy
|
||
|
end
|
||
|
|
||
|
it 'has ordered scopes' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['feature_flags'][0]['scopes'][0]['id'])
|
||
|
.to be < json_response['feature_flags'][0]['scopes'][1]['id']
|
||
|
expect(json_response['feature_flags'][1]['scopes'][0]['id'])
|
||
|
.to be < json_response['feature_flags'][1]['scopes'][1]['id']
|
||
|
end
|
||
|
|
||
|
it 'does not have N+1 problem' do
|
||
|
recorded = ActiveRecord::QueryRecorder.new { subject }
|
||
|
|
||
|
related_count = recorded.log
|
||
|
.count { |query| query.include?('operations_feature_flag') }
|
||
|
|
||
|
expect(related_count).to be_within(5).of(2)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'with version 1 and 2 feature flags' do
|
||
|
let!(:new_version_feature_flag) do
|
||
|
create(:operations_feature_flag, :new_version_flag, project: project, name: 'feature_flag_c')
|
||
|
end
|
||
|
|
||
|
it 'returns all feature flags as json response' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['feature_flags'].count).to eq(3)
|
||
|
end
|
||
|
|
||
|
it 'returns only version 1 flags when new version flags are disabled' do
|
||
|
stub_feature_flags(feature_flags_new_version: false)
|
||
|
|
||
|
subject
|
||
|
|
||
|
expected = [feature_flag_active.name, feature_flag_inactive.name].sort
|
||
|
expect(json_response['feature_flags'].map { |f| f['name'] }.sort).to eq(expected)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
describe 'GET new' do
|
||
|
render_views
|
||
|
|
||
|
subject { get(:new, params: view_params) }
|
||
|
|
||
|
it 'renders the form' do
|
||
|
is_expected.to have_gitlab_http_status(:ok)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
describe 'GET #show.json' do
|
||
|
subject { get(:show, params: params, format: :json) }
|
||
|
|
||
|
let!(:feature_flag) do
|
||
|
create(:operations_feature_flag, project: project)
|
||
|
end
|
||
|
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns the feature flag as json response' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['name']).to eq(feature_flag.name)
|
||
|
expect(json_response['active']).to eq(feature_flag.active)
|
||
|
expect(json_response['version']).to eq('legacy_flag')
|
||
|
end
|
||
|
|
||
|
it 'matches json schema' do
|
||
|
is_expected.to match_response_schema('feature_flag')
|
||
|
end
|
||
|
|
||
|
it 'routes based on iid' do
|
||
|
other_project = create(:project)
|
||
|
other_project.add_developer(user)
|
||
|
other_feature_flag = create(:operations_feature_flag, project: other_project,
|
||
|
name: 'other_flag')
|
||
|
params = {
|
||
|
namespace_id: other_project.namespace,
|
||
|
project_id: other_project,
|
||
|
iid: other_feature_flag.iid
|
||
|
}
|
||
|
|
||
|
get(:show, params: params, format: :json)
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['name']).to eq(other_feature_flag.name)
|
||
|
end
|
||
|
|
||
|
it 'routes based on iid when new version flags are disabled' do
|
||
|
stub_feature_flags(feature_flags_new_version: false)
|
||
|
other_project = create(:project)
|
||
|
other_project.add_developer(user)
|
||
|
other_feature_flag = create(:operations_feature_flag, project: other_project,
|
||
|
name: 'other_flag')
|
||
|
params = {
|
||
|
namespace_id: other_project.namespace,
|
||
|
project_id: other_project,
|
||
|
iid: other_feature_flag.iid
|
||
|
}
|
||
|
|
||
|
get(:show, params: params, format: :json)
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['name']).to eq(other_feature_flag.name)
|
||
|
end
|
||
|
|
||
|
context 'when feature flag is not found' do
|
||
|
let!(:feature_flag) { }
|
||
|
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: 1
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns 404' do
|
||
|
is_expected.to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when user is reporter' do
|
||
|
let(:user) { reporter }
|
||
|
|
||
|
it 'returns 404' do
|
||
|
is_expected.to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when feature flags have additional scopes' do
|
||
|
context 'when there is at least one active scope' do
|
||
|
let!(:feature_flag) do
|
||
|
create(:operations_feature_flag, project: project, active: false)
|
||
|
end
|
||
|
|
||
|
let!(:feature_flag_scope_production) do
|
||
|
create(:operations_feature_flag_scope,
|
||
|
feature_flag: feature_flag,
|
||
|
environment_scope: 'review/*',
|
||
|
active: true)
|
||
|
end
|
||
|
|
||
|
it 'returns false for active' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['active']).to eq(false)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when all scopes are inactive' do
|
||
|
let!(:feature_flag) do
|
||
|
create(:operations_feature_flag, project: project, active: false)
|
||
|
end
|
||
|
|
||
|
let!(:feature_flag_scope_production) do
|
||
|
create(:operations_feature_flag_scope,
|
||
|
feature_flag: feature_flag,
|
||
|
environment_scope: 'production',
|
||
|
active: false)
|
||
|
end
|
||
|
|
||
|
it 'recognizes the feature flag as inactive' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['active']).to be_falsy
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'with a version 2 feature flag' do
|
||
|
let!(:new_version_feature_flag) do
|
||
|
create(:operations_feature_flag, :new_version_flag, project: project)
|
||
|
end
|
||
|
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: new_version_feature_flag.iid
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns the feature flag' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['name']).to eq(new_version_feature_flag.name)
|
||
|
expect(json_response['active']).to eq(new_version_feature_flag.active)
|
||
|
expect(json_response['version']).to eq('new_version_flag')
|
||
|
end
|
||
|
|
||
|
it 'returns a 404 when new version flags are disabled' do
|
||
|
stub_feature_flags(feature_flags_new_version: false)
|
||
|
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
|
||
|
it 'returns strategies ordered by id' do
|
||
|
first_strategy = create(:operations_strategy, feature_flag: new_version_feature_flag)
|
||
|
second_strategy = create(:operations_strategy, feature_flag: new_version_feature_flag)
|
||
|
|
||
|
subject
|
||
|
|
||
|
expect(json_response['strategies'].map { |s| s['id'] }).to eq([first_strategy.id, second_strategy.id])
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
describe 'POST create.json' do
|
||
|
subject { post(:create, params: params, format: :json) }
|
||
|
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns 200' do
|
||
|
is_expected.to have_gitlab_http_status(:ok)
|
||
|
end
|
||
|
|
||
|
it 'creates a new feature flag' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['name']).to eq('my_feature_flag')
|
||
|
expect(json_response['active']).to be_truthy
|
||
|
end
|
||
|
|
||
|
it 'creates a default scope' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['scopes'].count).to eq(1)
|
||
|
expect(json_response['scopes'].first['environment_scope']).to eq('*')
|
||
|
expect(json_response['scopes'].first['active']).to be_truthy
|
||
|
end
|
||
|
|
||
|
it 'matches json schema' do
|
||
|
is_expected.to match_response_schema('feature_flag')
|
||
|
end
|
||
|
|
||
|
context 'when the same named feature flag has already existed' do
|
||
|
before do
|
||
|
create(:operations_feature_flag, name: 'my_feature_flag', project: project)
|
||
|
end
|
||
|
|
||
|
it 'returns 400' do
|
||
|
is_expected.to have_gitlab_http_status(:bad_request)
|
||
|
end
|
||
|
|
||
|
it 'returns an error message' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['message']).to include('Name has already been taken')
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'without the active parameter' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag'
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'creates a flag with active set to true' do
|
||
|
expect { subject }.to change { Operations::FeatureFlag.count }.by(1)
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(response).to match_response_schema('feature_flag')
|
||
|
expect(json_response['active']).to eq(true)
|
||
|
expect(Operations::FeatureFlag.last.active).to eq(true)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when user is reporter' do
|
||
|
let(:user) { reporter }
|
||
|
|
||
|
it 'returns 404' do
|
||
|
is_expected.to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creates additional scope' do
|
||
|
let(:params) do
|
||
|
view_params.merge({
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
scopes_attributes: [{ environment_scope: '*', active: true },
|
||
|
{ environment_scope: 'production', active: false }]
|
||
|
}
|
||
|
})
|
||
|
end
|
||
|
|
||
|
it 'creates feature flag scopes successfully' do
|
||
|
expect { subject }.to change { Operations::FeatureFlagScope.count }.by(2)
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
end
|
||
|
|
||
|
it 'creates feature flag scopes in a correct order' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['scopes'].first['environment_scope']).to eq('*')
|
||
|
expect(json_response['scopes'].second['environment_scope']).to eq('production')
|
||
|
end
|
||
|
|
||
|
context 'when default scope is not placed first' do
|
||
|
let(:params) do
|
||
|
view_params.merge({
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
scopes_attributes: [{ environment_scope: 'production', active: false },
|
||
|
{ environment_scope: '*', active: true }]
|
||
|
}
|
||
|
})
|
||
|
end
|
||
|
|
||
|
it 'returns 400' do
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:bad_request)
|
||
|
expect(json_response['message'])
|
||
|
.to include('Default scope has to be the first element')
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creates additional scope with a percentage rollout' do
|
||
|
it 'creates a strategy for the scope' do
|
||
|
params = view_params.merge({
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
scopes_attributes: [{ environment_scope: '*', active: true },
|
||
|
{ environment_scope: 'production', active: false,
|
||
|
strategies: [{ name: 'gradualRolloutUserId',
|
||
|
parameters: { groupId: 'default', percentage: '42' } }] }]
|
||
|
}
|
||
|
})
|
||
|
|
||
|
post(:create, params: params, format: :json)
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
production_strategies_json = json_response['scopes'].second['strategies']
|
||
|
expect(production_strategies_json).to eq([{
|
||
|
'name' => 'gradualRolloutUserId',
|
||
|
'parameters' => { "groupId" => "default", "percentage" => "42" }
|
||
|
}])
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creates additional scope with a userWithId strategy' do
|
||
|
it 'creates a strategy for the scope' do
|
||
|
params = view_params.merge({
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
scopes_attributes: [{ environment_scope: '*', active: true },
|
||
|
{ environment_scope: 'production', active: false,
|
||
|
strategies: [{ name: 'userWithId',
|
||
|
parameters: { userIds: '123,4,6722' } }] }]
|
||
|
}
|
||
|
})
|
||
|
|
||
|
post(:create, params: params, format: :json)
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
production_strategies_json = json_response['scopes'].second['strategies']
|
||
|
expect(production_strategies_json).to eq([{
|
||
|
'name' => 'userWithId',
|
||
|
'parameters' => { "userIds" => "123,4,6722" }
|
||
|
}])
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creates an additional scope without a strategy' do
|
||
|
it 'creates a default strategy' do
|
||
|
params = view_params.merge({
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
scopes_attributes: [{ environment_scope: '*', active: true }]
|
||
|
}
|
||
|
})
|
||
|
|
||
|
post(:create, params: params, format: :json)
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
default_strategies_json = json_response['scopes'].first['strategies']
|
||
|
expect(default_strategies_json).to eq([{ "name" => "default", "parameters" => {} }])
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creating a version 2 feature flag' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
version: 'new_version_flag'
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'creates a new feature flag' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['name']).to eq('my_feature_flag')
|
||
|
expect(json_response['active']).to be_truthy
|
||
|
expect(json_response['version']).to eq('new_version_flag')
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creating a version 2 feature flag with strategies and scopes' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
version: 'new_version_flag',
|
||
|
strategies_attributes: [{
|
||
|
name: 'userWithId',
|
||
|
parameters: { userIds: 'user1' },
|
||
|
scopes_attributes: [{ environment_scope: '*' }]
|
||
|
}]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'creates a new feature flag with the strategies and scopes' do
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['name']).to eq('my_feature_flag')
|
||
|
expect(json_response['active']).to eq(true)
|
||
|
expect(json_response['strategies'].count).to eq(1)
|
||
|
|
||
|
strategy_json = json_response['strategies'].first
|
||
|
expect(strategy_json).to have_key('id')
|
||
|
expect(strategy_json['name']).to eq('userWithId')
|
||
|
expect(strategy_json['parameters']).to eq({ 'userIds' => 'user1' })
|
||
|
expect(strategy_json['scopes'].count).to eq(1)
|
||
|
|
||
|
scope_json = strategy_json['scopes'].first
|
||
|
expect(scope_json).to have_key('id')
|
||
|
expect(scope_json['environment_scope']).to eq('*')
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creating a version 2 feature flag with a gradualRolloutUserId strategy' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
version: 'new_version_flag',
|
||
|
strategies_attributes: [{
|
||
|
name: 'gradualRolloutUserId',
|
||
|
parameters: { groupId: 'default', percentage: '15' },
|
||
|
scopes_attributes: [{ environment_scope: 'production' }]
|
||
|
}]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'creates the new strategy' do
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
|
||
|
strategy_json = json_response['strategies'].first
|
||
|
expect(strategy_json['name']).to eq('gradualRolloutUserId')
|
||
|
expect(strategy_json['parameters']).to eq({ 'groupId' => 'default', 'percentage' => '15' })
|
||
|
expect(strategy_json['scopes'].count).to eq(1)
|
||
|
|
||
|
scope_json = strategy_json['scopes'].first
|
||
|
expect(scope_json['environment_scope']).to eq('production')
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creating a version 2 feature flag with a flexibleRollout strategy' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
version: 'new_version_flag',
|
||
|
strategies_attributes: [{
|
||
|
name: 'flexibleRollout',
|
||
|
parameters: { groupId: 'default', rollout: '15', stickiness: 'DEFAULT' },
|
||
|
scopes_attributes: [{ environment_scope: 'production' }]
|
||
|
}]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'creates the new strategy' do
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
|
||
|
strategy_json = json_response['strategies'].first
|
||
|
expect(strategy_json['name']).to eq('flexibleRollout')
|
||
|
expect(strategy_json['parameters']).to eq({ 'groupId' => 'default', 'rollout' => '15', 'stickiness' => 'DEFAULT' })
|
||
|
expect(strategy_json['scopes'].count).to eq(1)
|
||
|
|
||
|
scope_json = strategy_json['scopes'].first
|
||
|
expect(scope_json['environment_scope']).to eq('production')
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when creating a version 2 feature flag with a gitlabUserList strategy' do
|
||
|
let!(:user_list) do
|
||
|
create(:operations_feature_flag_user_list, project: project,
|
||
|
name: 'My List', user_xids: 'user1,user2')
|
||
|
end
|
||
|
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
version: 'new_version_flag',
|
||
|
strategies_attributes: [{
|
||
|
name: 'gitlabUserList',
|
||
|
parameters: {},
|
||
|
user_list_id: user_list.id,
|
||
|
scopes_attributes: [{ environment_scope: 'production' }]
|
||
|
}]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'creates the new strategy' do
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies']).to match([a_hash_including({
|
||
|
'name' => 'gitlabUserList',
|
||
|
'parameters' => {},
|
||
|
'user_list' => {
|
||
|
'id' => user_list.id,
|
||
|
'iid' => user_list.iid,
|
||
|
'name' => 'My List',
|
||
|
'user_xids' => 'user1,user2'
|
||
|
},
|
||
|
'scopes' => [a_hash_including({
|
||
|
'environment_scope' => 'production'
|
||
|
})]
|
||
|
})])
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when version parameter is invalid' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
version: 'bad_version'
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns a 400' do
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:bad_request)
|
||
|
expect(json_response).to eq({ 'message' => 'Version is invalid' })
|
||
|
expect(Operations::FeatureFlag.count).to eq(0)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when version 2 flags are disabled' do
|
||
|
context 'and attempting to create a version 2 flag' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true,
|
||
|
version: 'new_version_flag'
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns a 400' do
|
||
|
stub_feature_flags(feature_flags_new_version: false)
|
||
|
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:bad_request)
|
||
|
expect(Operations::FeatureFlag.count).to eq(0)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'and attempting to create a version 1 flag' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
operations_feature_flag: {
|
||
|
name: 'my_feature_flag',
|
||
|
active: true
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'creates the flag' do
|
||
|
stub_feature_flags(feature_flags_new_version: false)
|
||
|
|
||
|
subject
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(Operations::FeatureFlag.count).to eq(1)
|
||
|
expect(json_response['version']).to eq('legacy_flag')
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
describe 'DELETE destroy.json' do
|
||
|
subject { delete(:destroy, params: params, format: :json) }
|
||
|
|
||
|
let!(:feature_flag) { create(:operations_feature_flag, project: project) }
|
||
|
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns 200' do
|
||
|
is_expected.to have_gitlab_http_status(:ok)
|
||
|
end
|
||
|
|
||
|
it 'deletes one feature flag' do
|
||
|
expect { subject }.to change { Operations::FeatureFlag.count }.by(-1)
|
||
|
end
|
||
|
|
||
|
it 'destroys the default scope' do
|
||
|
expect { subject }.to change { Operations::FeatureFlagScope.count }.by(-1)
|
||
|
end
|
||
|
|
||
|
it 'matches json schema' do
|
||
|
is_expected.to match_response_schema('feature_flag')
|
||
|
end
|
||
|
|
||
|
context 'when user is reporter' do
|
||
|
let(:user) { reporter }
|
||
|
|
||
|
it 'returns 404' do
|
||
|
is_expected.to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when the feature flag does not exist' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: 0
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns not found' do
|
||
|
is_expected.to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when there is an additional scope' do
|
||
|
let!(:scope) { create_scope(feature_flag, 'production', false) }
|
||
|
|
||
|
it 'destroys the default scope and production scope' do
|
||
|
expect { subject }.to change { Operations::FeatureFlagScope.count }.by(-2)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'with a version 2 flag' do
|
||
|
let!(:new_version_flag) { create(:operations_feature_flag, :new_version_flag, project: project) }
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: new_version_flag.iid
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'deletes the flag' do
|
||
|
expect { subject }.to change { Operations::FeatureFlag.count }.by(-1)
|
||
|
end
|
||
|
|
||
|
context 'when new version flags are disabled' do
|
||
|
it 'returns a 404' do
|
||
|
stub_feature_flags(feature_flags_new_version: false)
|
||
|
|
||
|
expect { subject }.not_to change { Operations::FeatureFlag.count }
|
||
|
expect(response).to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
describe 'PUT update.json' do
|
||
|
def put_request(feature_flag, feature_flag_params)
|
||
|
params = {
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: feature_flag_params
|
||
|
}
|
||
|
|
||
|
put(:update, params: params, format: :json, as: :json)
|
||
|
end
|
||
|
|
||
|
before do
|
||
|
stub_feature_flags(
|
||
|
feature_flags_legacy_read_only: false,
|
||
|
feature_flags_legacy_read_only_override: false
|
||
|
)
|
||
|
end
|
||
|
|
||
|
subject { put(:update, params: params, format: :json) }
|
||
|
|
||
|
let!(:feature_flag) do
|
||
|
create(:operations_feature_flag,
|
||
|
:legacy_flag,
|
||
|
name: 'ci_live_trace',
|
||
|
active: true,
|
||
|
project: project)
|
||
|
end
|
||
|
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: {
|
||
|
name: 'ci_new_live_trace'
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns 200' do
|
||
|
is_expected.to have_gitlab_http_status(:ok)
|
||
|
end
|
||
|
|
||
|
it 'updates the name of the feature flag name' do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['name']).to eq('ci_new_live_trace')
|
||
|
end
|
||
|
|
||
|
it 'matches json schema' do
|
||
|
is_expected.to match_response_schema('feature_flag')
|
||
|
end
|
||
|
|
||
|
context 'when updates active' do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: {
|
||
|
active: false
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'updates active from true to false' do
|
||
|
expect { subject }
|
||
|
.to change { feature_flag.reload.active }.from(true).to(false)
|
||
|
end
|
||
|
|
||
|
it "does not change default scope's active" do
|
||
|
expect { subject }
|
||
|
.not_to change { feature_flag.default_scope.reload.active }.from(true)
|
||
|
end
|
||
|
|
||
|
it 'updates active from false to true when an inactive feature flag has an active scope' do
|
||
|
feature_flag = create(:operations_feature_flag, project: project, name: 'my_flag', active: false)
|
||
|
create(:operations_feature_flag_scope, feature_flag: feature_flag, environment_scope: 'production', active: true)
|
||
|
|
||
|
put_request(feature_flag, { active: true })
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(response).to match_response_schema('feature_flag')
|
||
|
expect(json_response['active']).to eq(true)
|
||
|
expect(feature_flag.reload.active).to eq(true)
|
||
|
expect(feature_flag.default_scope.reload.active).to eq(false)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when user is reporter' do
|
||
|
let(:user) { reporter }
|
||
|
|
||
|
it 'returns 404' do
|
||
|
is_expected.to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context "when creates an additional scope for production environment" do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: {
|
||
|
scopes_attributes: [{ environment_scope: 'production', active: false }]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'creates a production scope' do
|
||
|
expect { subject }.to change { feature_flag.reload.scopes.count }.by(1)
|
||
|
|
||
|
expect(json_response['scopes'].last['environment_scope']).to eq('production')
|
||
|
expect(json_response['scopes'].last['active']).to be_falsy
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context "when creates a default scope" do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: {
|
||
|
scopes_attributes: [{ environment_scope: '*', active: false }]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns 400' do
|
||
|
is_expected.to have_gitlab_http_status(:bad_request)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context "when updates a default scope's active value" do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: {
|
||
|
scopes_attributes: [
|
||
|
{
|
||
|
id: feature_flag.default_scope.id,
|
||
|
environment_scope: '*',
|
||
|
active: false
|
||
|
}
|
||
|
]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it "updates successfully" do
|
||
|
subject
|
||
|
|
||
|
expect(json_response['scopes'].first['environment_scope']).to eq('*')
|
||
|
expect(json_response['scopes'].first['active']).to be_falsy
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context "when changes default scope's spec" do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: {
|
||
|
scopes_attributes: [
|
||
|
{
|
||
|
id: feature_flag.default_scope.id,
|
||
|
environment_scope: 'review/*'
|
||
|
}
|
||
|
]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'returns 400' do
|
||
|
is_expected.to have_gitlab_http_status(:bad_request)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context "when destroys the default scope" do
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: {
|
||
|
scopes_attributes: [
|
||
|
{
|
||
|
id: feature_flag.default_scope.id,
|
||
|
_destroy: 1
|
||
|
}
|
||
|
]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'raises an error' do
|
||
|
expect { subject }.to raise_error(ActiveRecord::ReadOnlyRecord)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context "when destroys a production scope" do
|
||
|
let!(:production_scope) { create_scope(feature_flag, 'production', true) }
|
||
|
let(:params) do
|
||
|
{
|
||
|
namespace_id: project.namespace,
|
||
|
project_id: project,
|
||
|
iid: feature_flag.iid,
|
||
|
operations_feature_flag: {
|
||
|
scopes_attributes: [
|
||
|
{
|
||
|
id: production_scope.id,
|
||
|
_destroy: 1
|
||
|
}
|
||
|
]
|
||
|
}
|
||
|
}
|
||
|
end
|
||
|
|
||
|
it 'destroys successfully' do
|
||
|
subject
|
||
|
|
||
|
scopes = json_response['scopes']
|
||
|
expect(scopes.any? { |scope| scope['environment_scope'] == 'production' })
|
||
|
.to be_falsy
|
||
|
end
|
||
|
end
|
||
|
|
||
|
describe "updating the strategy" do
|
||
|
it 'creates a default strategy' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
strategies: [{ name: 'default', parameters: {} }]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
scope_json = json_response['scopes'].find do |s|
|
||
|
s['environment_scope'] == 'production'
|
||
|
end
|
||
|
expect(scope_json['strategies']).to eq([{
|
||
|
"name" => "default",
|
||
|
"parameters" => {}
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'creates a gradualRolloutUserId strategy' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
strategies: [{ name: 'gradualRolloutUserId',
|
||
|
parameters: { groupId: 'default', percentage: "70" } }]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
scope_json = json_response['scopes'].find do |s|
|
||
|
s['environment_scope'] == 'production'
|
||
|
end
|
||
|
expect(scope_json['strategies']).to eq([{
|
||
|
"name" => "gradualRolloutUserId",
|
||
|
"parameters" => {
|
||
|
"groupId" => "default",
|
||
|
"percentage" => "70"
|
||
|
}
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'creates a userWithId strategy' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [{ name: 'default', parameters: {} }])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
strategies: [{ name: 'userWithId', parameters: { userIds: 'sam,fred' } }]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
scope_json = json_response['scopes'].find do |s|
|
||
|
s['environment_scope'] == 'production'
|
||
|
end
|
||
|
expect(scope_json['strategies']).to eq([{
|
||
|
"name" => "userWithId",
|
||
|
"parameters" => { "userIds" => "sam,fred" }
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'updates an existing strategy' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [{ name: 'default', parameters: {} }])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
strategies: [{ name: 'gradualRolloutUserId',
|
||
|
parameters: { groupId: 'default', percentage: "50" } }]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
scope_json = json_response['scopes'].find do |s|
|
||
|
s['environment_scope'] == 'production'
|
||
|
end
|
||
|
expect(scope_json['strategies']).to eq([{
|
||
|
"name" => "gradualRolloutUserId",
|
||
|
"parameters" => {
|
||
|
"groupId" => "default",
|
||
|
"percentage" => "50"
|
||
|
}
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'clears an existing strategy' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [{ name: 'default', parameters: {} }])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
strategies: []
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
scope_json = json_response['scopes'].find do |s|
|
||
|
s['environment_scope'] == 'production'
|
||
|
end
|
||
|
expect(scope_json['strategies']).to eq([])
|
||
|
end
|
||
|
|
||
|
it 'accepts multiple strategies' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [{ name: 'default', parameters: {} }])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
strategies: [
|
||
|
{ name: 'gradualRolloutUserId', parameters: { groupId: 'mygroup', percentage: '55' } },
|
||
|
{ name: 'userWithId', parameters: { userIds: 'joe' } }
|
||
|
]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
scope_json = json_response['scopes'].find do |s|
|
||
|
s['environment_scope'] == 'production'
|
||
|
end
|
||
|
expect(scope_json['strategies'].length).to eq(2)
|
||
|
expect(scope_json['strategies']).to include({
|
||
|
"name" => "gradualRolloutUserId",
|
||
|
"parameters" => { "groupId" => "mygroup", "percentage" => "55" }
|
||
|
})
|
||
|
expect(scope_json['strategies']).to include({
|
||
|
"name" => "userWithId",
|
||
|
"parameters" => { "userIds" => "joe" }
|
||
|
})
|
||
|
end
|
||
|
|
||
|
it 'does not modify strategies when there is no strategies key in the params' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [{ name: 'default', parameters: {} }])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{ id: scope.id }])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
scope_json = json_response['scopes'].find do |s|
|
||
|
s['environment_scope'] == 'production'
|
||
|
end
|
||
|
expect(scope_json['strategies']).to eq([{
|
||
|
"name" => "default",
|
||
|
"parameters" => {}
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'leaves an existing strategy when there are no strategies in the params' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [{ name: 'gradualRolloutUserId',
|
||
|
parameters: { groupId: 'default', percentage: '10' } }])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{ id: scope.id }])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
scope_json = json_response['scopes'].find do |s|
|
||
|
s['environment_scope'] == 'production'
|
||
|
end
|
||
|
expect(scope_json['strategies']).to eq([{
|
||
|
"name" => "gradualRolloutUserId",
|
||
|
"parameters" => { "groupId" => "default", "percentage" => "10" }
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'does not accept extra parameters in the strategy params' do
|
||
|
scope = create_scope(feature_flag, 'production', true, [{ name: 'default', parameters: {} }])
|
||
|
|
||
|
put_request(feature_flag, scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
strategies: [{ name: 'userWithId', parameters: { userIds: 'joe', groupId: 'default' } }]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:bad_request)
|
||
|
expect(json_response['message']).to eq(["Scopes strategies parameters are invalid"])
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when legacy feature flags are set to be read only' do
|
||
|
it 'does not update the flag' do
|
||
|
stub_feature_flags(feature_flags_legacy_read_only: true)
|
||
|
|
||
|
put_request(feature_flag, name: 'ci_new_live_trace')
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:bad_request)
|
||
|
expect(json_response['message']).to eq(["Legacy feature flags are read-only"])
|
||
|
end
|
||
|
|
||
|
it 'updates the flag if the legacy read-only override is enabled for a particular project' do
|
||
|
stub_feature_flags(
|
||
|
feature_flags_legacy_read_only: true,
|
||
|
feature_flags_legacy_read_only_override: project
|
||
|
)
|
||
|
|
||
|
put_request(feature_flag, name: 'ci_new_live_trace')
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['name']).to eq('ci_new_live_trace')
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'with a version 2 feature flag' do
|
||
|
let!(:new_version_flag) do
|
||
|
create(:operations_feature_flag,
|
||
|
:new_version_flag,
|
||
|
name: 'new-feature',
|
||
|
active: true,
|
||
|
project: project)
|
||
|
end
|
||
|
|
||
|
it 'creates a new strategy and scope' do
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
name: 'userWithId',
|
||
|
parameters: { userIds: 'user1' },
|
||
|
scopes_attributes: [{
|
||
|
environment_scope: 'production'
|
||
|
}]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies'].count).to eq(1)
|
||
|
strategy_json = json_response['strategies'].first
|
||
|
expect(strategy_json['name']).to eq('userWithId')
|
||
|
expect(strategy_json['parameters']).to eq({
|
||
|
'userIds' => 'user1'
|
||
|
})
|
||
|
expect(strategy_json['scopes'].count).to eq(1)
|
||
|
scope_json = strategy_json['scopes'].first
|
||
|
expect(scope_json['environment_scope']).to eq('production')
|
||
|
end
|
||
|
|
||
|
it 'creates a gradualRolloutUserId strategy' do
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
name: 'gradualRolloutUserId',
|
||
|
parameters: { groupId: 'default', percentage: '30' }
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies'].count).to eq(1)
|
||
|
strategy_json = json_response['strategies'].first
|
||
|
expect(strategy_json['name']).to eq('gradualRolloutUserId')
|
||
|
expect(strategy_json['parameters']).to eq({
|
||
|
'groupId' => 'default',
|
||
|
'percentage' => '30'
|
||
|
})
|
||
|
expect(strategy_json['scopes']).to eq([])
|
||
|
end
|
||
|
|
||
|
it 'creates a flexibleRollout strategy' do
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
name: 'flexibleRollout',
|
||
|
parameters: { groupId: 'default', rollout: '30', stickiness: 'DEFAULT' }
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies'].count).to eq(1)
|
||
|
strategy_json = json_response['strategies'].first
|
||
|
expect(strategy_json['name']).to eq('flexibleRollout')
|
||
|
expect(strategy_json['parameters']).to eq({
|
||
|
'groupId' => 'default',
|
||
|
'rollout' => '30',
|
||
|
'stickiness' => 'DEFAULT'
|
||
|
})
|
||
|
expect(strategy_json['scopes']).to eq([])
|
||
|
end
|
||
|
|
||
|
it 'creates a gitlabUserList strategy' do
|
||
|
user_list = create(:operations_feature_flag_user_list, project: project, name: 'My List', user_xids: 'user1,user2')
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
name: 'gitlabUserList',
|
||
|
parameters: {},
|
||
|
user_list_id: user_list.id
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies']).to match([a_hash_including({
|
||
|
'id' => an_instance_of(Integer),
|
||
|
'name' => 'gitlabUserList',
|
||
|
'parameters' => {},
|
||
|
'user_list' => {
|
||
|
'id' => user_list.id,
|
||
|
'iid' => user_list.iid,
|
||
|
'name' => 'My List',
|
||
|
'user_xids' => 'user1,user2'
|
||
|
},
|
||
|
'scopes' => []
|
||
|
})])
|
||
|
end
|
||
|
|
||
|
it 'supports switching the associated user list for an existing gitlabUserList strategy' do
|
||
|
user_list = create(:operations_feature_flag_user_list, project: project, name: 'My List', user_xids: 'user1,user2')
|
||
|
strategy = create(:operations_strategy, feature_flag: new_version_flag, name: 'gitlabUserList', parameters: {}, user_list: user_list)
|
||
|
other_user_list = create(:operations_feature_flag_user_list, project: project, name: 'Other List', user_xids: 'user3')
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy.id,
|
||
|
user_list_id: other_user_list.id
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies']).to eq([{
|
||
|
'id' => strategy.id,
|
||
|
'name' => 'gitlabUserList',
|
||
|
'parameters' => {},
|
||
|
'user_list' => {
|
||
|
'id' => other_user_list.id,
|
||
|
'iid' => other_user_list.iid,
|
||
|
'name' => 'Other List',
|
||
|
'user_xids' => 'user3'
|
||
|
},
|
||
|
'scopes' => []
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'automatically dissociates the user list when switching the type of an existing gitlabUserList strategy' do
|
||
|
user_list = create(:operations_feature_flag_user_list, project: project, name: 'My List', user_xids: 'user1,user2')
|
||
|
strategy = create(:operations_strategy, feature_flag: new_version_flag, name: 'gitlabUserList', parameters: {}, user_list: user_list)
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy.id,
|
||
|
name: 'gradualRolloutUserId',
|
||
|
parameters: {
|
||
|
groupId: 'default',
|
||
|
percentage: '25'
|
||
|
}
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies']).to eq([{
|
||
|
'id' => strategy.id,
|
||
|
'name' => 'gradualRolloutUserId',
|
||
|
'parameters' => {
|
||
|
'groupId' => 'default',
|
||
|
'percentage' => '25'
|
||
|
},
|
||
|
'scopes' => []
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'does not delete a user list when deleting a gitlabUserList strategy' do
|
||
|
user_list = create(:operations_feature_flag_user_list, project: project, name: 'My List', user_xids: 'user1,user2')
|
||
|
strategy = create(:operations_strategy, feature_flag: new_version_flag, name: 'gitlabUserList', parameters: {}, user_list: user_list)
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy.id,
|
||
|
_destroy: true
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies']).to eq([])
|
||
|
expect(::Operations::FeatureFlags::Strategy.count).to eq(0)
|
||
|
expect(::Operations::FeatureFlags::StrategyUserList.count).to eq(0)
|
||
|
expect(::Operations::FeatureFlags::UserList.first).to eq(user_list)
|
||
|
end
|
||
|
|
||
|
it 'returns not found when trying to create a gitlabUserList strategy with an invalid user list id' do
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
name: 'gitlabUserList',
|
||
|
parameters: {},
|
||
|
user_list_id: 1
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:not_found)
|
||
|
end
|
||
|
|
||
|
it 'returns not found when trying to update a gitlabUserList strategy with a user list from another project' do
|
||
|
user_list = create(:operations_feature_flag_user_list, project: project, name: 'My List', user_xids: 'user1,user2')
|
||
|
strategy = create(:operations_strategy, feature_flag: new_version_flag, name: 'gitlabUserList', parameters: {}, user_list: user_list)
|
||
|
other_project = create(:project)
|
||
|
other_user_list = create(:operations_feature_flag_user_list, project: other_project, name: 'Other List', user_xids: 'some,one')
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy.id,
|
||
|
user_list_id: other_user_list.id
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:not_found)
|
||
|
expect(strategy.reload.user_list).to eq(user_list)
|
||
|
end
|
||
|
|
||
|
it 'allows setting multiple gitlabUserList strategies to the same user list' do
|
||
|
user_list_a = create(:operations_feature_flag_user_list, project: project, name: 'My List A', user_xids: 'user1,user2')
|
||
|
user_list_b = create(:operations_feature_flag_user_list, project: project, name: 'My List B', user_xids: 'user3,user4')
|
||
|
strategy_a = create(:operations_strategy, feature_flag: new_version_flag, name: 'gitlabUserList', parameters: {}, user_list: user_list_a)
|
||
|
strategy_b = create(:operations_strategy, feature_flag: new_version_flag, name: 'gitlabUserList', parameters: {}, user_list: user_list_a)
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy_a.id,
|
||
|
user_list_id: user_list_b.id
|
||
|
}, {
|
||
|
id: strategy_b.id,
|
||
|
user_list_id: user_list_b.id
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(strategy_a.reload.user_list).to eq(user_list_b)
|
||
|
expect(strategy_b.reload.user_list).to eq(user_list_b)
|
||
|
end
|
||
|
|
||
|
it 'updates an existing strategy' do
|
||
|
strategy = create(:operations_strategy, feature_flag: new_version_flag, name: 'default', parameters: {})
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy.id,
|
||
|
name: 'userWithId',
|
||
|
parameters: { userIds: 'user2,user3' }
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies']).to eq([{
|
||
|
'id' => strategy.id,
|
||
|
'name' => 'userWithId',
|
||
|
'parameters' => { 'userIds' => 'user2,user3' },
|
||
|
'scopes' => []
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'updates an existing scope' do
|
||
|
strategy = create(:operations_strategy, feature_flag: new_version_flag, name: 'default', parameters: {})
|
||
|
scope = create(:operations_scope, strategy: strategy, environment_scope: 'staging')
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy.id,
|
||
|
scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
environment_scope: 'sandbox'
|
||
|
}]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies'].first['scopes']).to eq([{
|
||
|
'id' => scope.id,
|
||
|
'environment_scope' => 'sandbox'
|
||
|
}])
|
||
|
end
|
||
|
|
||
|
it 'deletes an existing strategy' do
|
||
|
strategy = create(:operations_strategy, feature_flag: new_version_flag, name: 'default', parameters: {})
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy.id,
|
||
|
_destroy: true
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies']).to eq([])
|
||
|
end
|
||
|
|
||
|
it 'deletes an existing scope' do
|
||
|
strategy = create(:operations_strategy, feature_flag: new_version_flag, name: 'default', parameters: {})
|
||
|
scope = create(:operations_scope, strategy: strategy, environment_scope: 'staging')
|
||
|
|
||
|
put_request(new_version_flag, strategies_attributes: [{
|
||
|
id: strategy.id,
|
||
|
scopes_attributes: [{
|
||
|
id: scope.id,
|
||
|
_destroy: true
|
||
|
}]
|
||
|
}])
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(json_response['strategies'].first['scopes']).to eq([])
|
||
|
end
|
||
|
|
||
|
it 'does not update the flag if version 2 flags are disabled' do
|
||
|
stub_feature_flags(feature_flags_new_version: false)
|
||
|
|
||
|
put_request(new_version_flag, { name: 'some-other-name' })
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:not_found)
|
||
|
expect(new_version_flag.reload.name).to eq('new-feature')
|
||
|
end
|
||
|
|
||
|
it 'updates the flag when legacy feature flags are set to be read only' do
|
||
|
stub_feature_flags(feature_flags_legacy_read_only: true)
|
||
|
|
||
|
put_request(new_version_flag, name: 'some-other-name')
|
||
|
|
||
|
expect(response).to have_gitlab_http_status(:ok)
|
||
|
expect(new_version_flag.reload.name).to eq('some-other-name')
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
private
|
||
|
|
||
|
def view_params
|
||
|
{ namespace_id: project.namespace, project_id: project }
|
||
|
end
|
||
|
end
|