Commit 08ece98d authored by Markus Koller's avatar Markus Koller

Merge branch 'remove-feature_flags_legacy_read_only-feature-flag' into 'master'

Remove `feature_flags_legacy_read_only` feature flag

See merge request gitlab-org/gitlab!48441
parents 6904392b f6ab8658
...@@ -30,9 +30,6 @@ export default { ...@@ -30,9 +30,6 @@ export default {
}; };
}, },
translations: { translations: {
legacyFlagAlert: s__(
'FeatureFlags|GitLab is moving to a new way of managing feature flags, and in 13.4, this feature flag will become read-only. Please create a new feature flag.',
),
legacyReadOnlyFlagAlert: s__( legacyReadOnlyFlagAlert: s__(
'FeatureFlags|GitLab is moving to a new way of managing feature flags. This feature flag is read-only, and it will be removed in 14.0. Please create a new feature flag.', 'FeatureFlags|GitLab is moving to a new way of managing feature flags. This feature flag is read-only, and it will be removed in 14.0. Please create a new feature flag.',
), ),
...@@ -59,18 +56,6 @@ export default { ...@@ -59,18 +56,6 @@ export default {
deprecated() { deprecated() {
return this.version === LEGACY_FLAG; return this.version === LEGACY_FLAG;
}, },
deprecatedAndEditable() {
return this.deprecated && !this.hasLegacyReadOnlyFlags;
},
deprecatedAndReadOnly() {
return this.deprecated && this.hasLegacyReadOnlyFlags;
},
hasLegacyReadOnlyFlags() {
return (
this.glFeatures.featureFlagsLegacyReadOnly &&
!this.glFeatures.featureFlagsLegacyReadOnlyOverride
);
},
}, },
created() { created() {
return this.fetchFeatureFlag(); return this.fetchFeatureFlag();
...@@ -91,12 +76,9 @@ export default { ...@@ -91,12 +76,9 @@ export default {
<gl-loading-icon v-if="isLoading" size="xl" class="gl-mt-7" /> <gl-loading-icon v-if="isLoading" size="xl" class="gl-mt-7" />
<template v-else-if="!isLoading && !hasError"> <template v-else-if="!isLoading && !hasError">
<gl-alert v-if="deprecatedAndEditable" variant="warning" :dismissible="false" class="gl-my-5"> <gl-alert v-if="deprecated" variant="warning" :dismissible="false" class="gl-my-5">{{
{{ $options.translations.legacyFlagAlert }} $options.translations.legacyReadOnlyFlagAlert
</gl-alert> }}</gl-alert>
<gl-alert v-if="deprecatedAndReadOnly" variant="warning" :dismissible="false" class="gl-my-5">
{{ $options.translations.legacyReadOnlyFlagAlert }}
</gl-alert>
<div class="gl-display-flex gl-align-items-center gl-mb-4 gl-mt-4"> <div class="gl-display-flex gl-align-items-center gl-mb-4 gl-mt-4">
<gl-toggle <gl-toggle
:value="active" :value="active"
......
...@@ -31,19 +31,12 @@ export default { ...@@ -31,19 +31,12 @@ export default {
}; };
}, },
translations: { translations: {
legacyFlagAlert: s__('FeatureFlags|Flag becomes read only soon'),
legacyFlagReadOnlyAlert: s__('FeatureFlags|Flag is read-only'), legacyFlagReadOnlyAlert: s__('FeatureFlags|Flag is read-only'),
}, },
computed: { computed: {
permissions() { permissions() {
return this.glFeatures.featureFlagPermissions; return this.glFeatures.featureFlagPermissions;
}, },
isLegacyReadOnlyFlagsEnabled() {
return (
this.glFeatures.featureFlagsLegacyReadOnly &&
!this.glFeatures.featureFlagsLegacyReadOnlyOverride
);
},
modalTitle() { modalTitle() {
return sprintf(s__('FeatureFlags|Delete %{name}?'), { return sprintf(s__('FeatureFlags|Delete %{name}?'), {
name: this.deleteFeatureFlagName, name: this.deleteFeatureFlagName,
...@@ -57,18 +50,13 @@ export default { ...@@ -57,18 +50,13 @@ export default {
modalId() { modalId() {
return 'delete-feature-flag'; return 'delete-feature-flag';
}, },
legacyFlagToolTipText() {
const { legacyFlagReadOnlyAlert, legacyFlagAlert } = this.$options.translations;
return this.isLegacyReadOnlyFlagsEnabled ? legacyFlagReadOnlyAlert : legacyFlagAlert;
},
}, },
methods: { methods: {
isLegacyFlag(flag) { isLegacyFlag(flag) {
return flag.version !== NEW_VERSION_FLAG; return flag.version !== NEW_VERSION_FLAG;
}, },
statusToggleDisabled(flag) { statusToggleDisabled(flag) {
return this.isLegacyReadOnlyFlagsEnabled && flag.version === LEGACY_FLAG; return flag.version === LEGACY_FLAG;
}, },
scopeTooltipText(scope) { scopeTooltipText(scope) {
return !scope.active return !scope.active
...@@ -123,9 +111,7 @@ export default { ...@@ -123,9 +111,7 @@ export default {
<template> <template>
<div class="table-holder js-feature-flag-table"> <div class="table-holder js-feature-flag-table">
<div class="gl-responsive-table-row table-row-header" role="row"> <div class="gl-responsive-table-row table-row-header" role="row">
<div class="table-section section-10"> <div class="table-section section-10">{{ s__('FeatureFlags|ID') }}</div>
{{ s__('FeatureFlags|ID') }}
</div>
<div class="table-section section-10" role="columnheader"> <div class="table-section section-10" role="columnheader">
{{ s__('FeatureFlags|Status') }} {{ s__('FeatureFlags|Status') }}
</div> </div>
...@@ -161,9 +147,8 @@ export default { ...@@ -161,9 +147,8 @@ export default {
v-else-if="featureFlag.active" v-else-if="featureFlag.active"
variant="success" variant="success"
data-testid="feature-flag-status-badge" data-testid="feature-flag-status-badge"
>{{ s__('FeatureFlags|Active') }}</gl-badge
> >
{{ s__('FeatureFlags|Active') }}
</gl-badge>
<gl-badge v-else variant="danger">{{ s__('FeatureFlags|Inactive') }}</gl-badge> <gl-badge v-else variant="danger">{{ s__('FeatureFlags|Inactive') }}</gl-badge>
</div> </div>
</div> </div>
...@@ -179,7 +164,7 @@ export default { ...@@ -179,7 +164,7 @@ export default {
</div> </div>
<gl-icon <gl-icon
v-if="isLegacyFlag(featureFlag)" v-if="isLegacyFlag(featureFlag)"
v-gl-tooltip.hover="legacyFlagToolTipText" v-gl-tooltip.hover="$options.translations.legacyFlagReadOnlyAlert"
class="gl-ml-3" class="gl-ml-3"
name="information-o" name="information-o"
/> />
...@@ -205,9 +190,8 @@ export default { ...@@ -205,9 +190,8 @@ export default {
:variant="badgeVariant(scope)" :variant="badgeVariant(scope)"
:data-qa-selector="`feature-flag-scope-${badgeVariant(scope)}-badge`" :data-qa-selector="`feature-flag-scope-${badgeVariant(scope)}-badge`"
class="gl-mr-3 gl-mt-2" class="gl-mr-3 gl-mt-2"
>{{ badgeText(scope) }}</gl-badge
> >
{{ badgeText(scope) }}
</gl-badge>
</template> </template>
<template v-else> <template v-else>
<gl-badge <gl-badge
...@@ -216,9 +200,8 @@ export default { ...@@ -216,9 +200,8 @@ export default {
data-testid="strategy-badge" data-testid="strategy-badge"
variant="info" variant="info"
class="gl-mr-3 gl-mt-2" class="gl-mr-3 gl-mt-2"
>{{ strategyBadgeText(strategy) }}</gl-badge
> >
{{ strategyBadgeText(strategy) }}
</gl-badge>
</template> </template>
</div> </div>
</div> </div>
......
...@@ -143,11 +143,7 @@ export default { ...@@ -143,11 +143,7 @@ export default {
return this.featureFlagIssuesEndpoint.length > 0; return this.featureFlagIssuesEndpoint.length > 0;
}, },
readOnly() { readOnly() {
return ( return this.version === LEGACY_FLAG;
this.glFeatures.featureFlagsLegacyReadOnly &&
!this.glFeatures.featureFlagsLegacyReadOnlyOverride &&
this.version === LEGACY_FLAG
);
}, },
}, },
methods: { methods: {
......
...@@ -10,12 +10,10 @@ class Projects::FeatureFlagsController < Projects::ApplicationController ...@@ -10,12 +10,10 @@ class Projects::FeatureFlagsController < Projects::ApplicationController
before_action :feature_flag, only: [:edit, :update, :destroy] before_action :feature_flag, only: [:edit, :update, :destroy]
before_action :ensure_legacy_flags_writable!, only: [:update] before_action :ensure_flag_writable!, only: [:update]
before_action do before_action do
push_frontend_feature_flag(:feature_flag_permissions) push_frontend_feature_flag(:feature_flag_permissions)
push_frontend_feature_flag(:feature_flags_legacy_read_only, project, default_enabled: true)
push_frontend_feature_flag(:feature_flags_legacy_read_only_override, project)
end end
feature_category :feature_flags feature_category :feature_flags
...@@ -103,10 +101,8 @@ class Projects::FeatureFlagsController < Projects::ApplicationController ...@@ -103,10 +101,8 @@ class Projects::FeatureFlagsController < Projects::ApplicationController
@feature_flag ||= @noteable = project.operations_feature_flags.find_by_iid!(params[:iid]) @feature_flag ||= @noteable = project.operations_feature_flags.find_by_iid!(params[:iid])
end end
def ensure_legacy_flags_writable! def ensure_flag_writable!
if ::Feature.enabled?(:feature_flags_legacy_read_only, project, default_enabled: true) && if feature_flag.legacy_flag?
::Feature.disabled?(:feature_flags_legacy_read_only_override, project) &&
feature_flag.legacy_flag?
render_error_json(['Legacy feature flags are read-only']) render_error_json(['Legacy feature flags are read-only'])
end end
end end
......
---
name: feature_flags_legacy_read_only
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/38353
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/240985
milestone: '13.3'
type: development
group: group::progressive delivery
default_enabled: true
---
name: feature_flags_legacy_read_only_override
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/40431
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/240985
milestone: '13.4'
type: development
group: group::progressive delivery
default_enabled: false
...@@ -16,66 +16,6 @@ RSpec.describe 'User sees feature flag list', :js do ...@@ -16,66 +16,6 @@ RSpec.describe 'User sees feature flag list', :js do
sign_in(user) sign_in(user)
end end
context 'with legacy feature flags' do
before do
create_flag(project, 'ci_live_trace', false).tap do |feature_flag|
create_scope(feature_flag, 'review/*', true)
end
create_flag(project, 'drop_legacy_artifacts', false)
create_flag(project, 'mr_train', true).tap do |feature_flag|
create_scope(feature_flag, 'production', false)
end
stub_feature_flags(feature_flags_legacy_read_only_override: false)
end
context 'when legacy feature flags are not read-only' do
before do
stub_feature_flags(feature_flags_legacy_read_only: false)
end
it 'user updates the status toggle' do
visit(project_feature_flags_path(project))
within_feature_flag_row(1) do
status_toggle_button.click
expect_status_toggle_button_to_be_checked
end
visit(project_audit_events_path(project))
expect(page).to(
have_text('Updated feature flag ci_live_trace. Updated active from "false" to "true".')
)
end
end
context 'when legacy feature flags are read-only but the override is active for a project' do
before do
stub_feature_flags(
feature_flags_legacy_read_only: true,
feature_flags_legacy_read_only_override: project
)
end
it 'user updates the status toggle' do
visit(project_feature_flags_path(project))
within_feature_flag_row(1) do
status_toggle_button.click
expect_status_toggle_button_to_be_checked
end
visit(project_audit_events_path(project))
expect(page).to(
have_text('Updated feature flag ci_live_trace. Updated active from "false" to "true".')
)
end
end
end
context 'with new version flags' do context 'with new version flags' do
before do before do
create(:operations_feature_flag, :new_version_flag, project: project, create(:operations_feature_flag, :new_version_flag, project: project,
......
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe 'User updates feature flag', :js do
include FeatureFlagHelpers
let_it_be(:user) { create(:user) }
let_it_be(:project) { create(:project, namespace: user.namespace) }
before_all do
project.add_developer(user)
end
before do
stub_feature_flags(
feature_flag_permissions: false,
feature_flags_legacy_read_only_override: false
)
sign_in(user)
end
context 'with a legacy feature flag' do
let!(:feature_flag) do
create_flag(project, 'ci_live_trace', true,
description: 'For live trace feature')
end
let!(:scope) { create_scope(feature_flag, 'review/*', true) }
context 'when legacy flags are editable' do
before do
stub_feature_flags(feature_flags_legacy_read_only: false)
visit(edit_project_feature_flag_path(project, feature_flag))
end
context 'when user updates the status of a scope' do
before do
within_scope_row(2) do
within_status { find('.project-feature-toggle').click }
end
click_button 'Save changes'
expect(page).to have_current_path(project_feature_flags_path(project))
end
it 'records audit event' do
visit(project_audit_events_path(project))
expect(page).to(
have_text("Updated feature flag ci_live_trace. Updated rule review/* active state from true to false.")
)
end
end
context 'when user adds a new scope' do
before do
within_scope_row(3) do
within_environment_spec do
find('.js-env-search > input').set('production')
find('.js-create-button').click
end
end
click_button 'Save changes'
expect(page).to have_current_path(project_feature_flags_path(project))
end
it 'records audit event' do
visit(project_audit_events_path(project))
expect(page).to have_text "Updated feature flag ci_live_trace"
end
end
context 'when user deletes a scope' do
before do
within_scope_row(2) do
within_delete { find('.js-delete-scope').click }
end
click_button 'Save changes'
expect(page).to have_current_path(project_feature_flags_path(project))
end
it 'records audit event' do
visit(project_audit_events_path(project))
expect(page).to have_text "Updated feature flag ci_live_trace"
end
end
end
end
end
...@@ -12113,9 +12113,6 @@ msgstr "" ...@@ -12113,9 +12113,6 @@ msgstr ""
msgid "FeatureFlags|Feature flags limit reached (%{featureFlagsLimit}). Delete one or more feature flags before adding new ones." msgid "FeatureFlags|Feature flags limit reached (%{featureFlagsLimit}). Delete one or more feature flags before adding new ones."
msgstr "" msgstr ""
msgid "FeatureFlags|Flag becomes read only soon"
msgstr ""
msgid "FeatureFlags|Flag is read-only" msgid "FeatureFlags|Flag is read-only"
msgstr "" msgstr ""
...@@ -12125,9 +12122,6 @@ msgstr "" ...@@ -12125,9 +12122,6 @@ msgstr ""
msgid "FeatureFlags|Get started with user lists" msgid "FeatureFlags|Get started with user lists"
msgstr "" msgstr ""
msgid "FeatureFlags|GitLab is moving to a new way of managing feature flags, and in 13.4, this feature flag will become read-only. Please create a new feature flag."
msgstr ""
msgid "FeatureFlags|GitLab is moving to a new way of managing feature flags. This feature flag is read-only, and it will be removed in 14.0. Please create a new feature flag." msgid "FeatureFlags|GitLab is moving to a new way of managing feature flags. This feature flag is read-only, and it will be removed in 14.0. Please create a new feature flag."
msgstr "" msgstr ""
......
...@@ -845,413 +845,64 @@ RSpec.describe Projects::FeatureFlagsController do ...@@ -845,413 +845,64 @@ RSpec.describe Projects::FeatureFlagsController do
put(:update, params: params, format: :json, as: :json) put(:update, params: params, format: :json, as: :json)
end end
before do context 'with a legacy feature flag' do
stub_feature_flags( subject { put(:update, params: params, format: :json) }
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!(:feature_flag) do
let(:params) do create(:operations_feature_flag,
{ :legacy_flag,
namespace_id: project.namespace, name: 'ci_live_trace',
project_id: project, active: true,
iid: feature_flag.iid, project: project)
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
end
context "when destroys a production scope" do
let!(:production_scope) { create_scope(feature_flag, 'production', true) }
let(:params) do let(:params) do
{ {
namespace_id: project.namespace, namespace_id: project.namespace,
project_id: project, project_id: project,
iid: feature_flag.iid, iid: feature_flag.iid,
operations_feature_flag: { operations_feature_flag: {
scopes_attributes: [ name: 'ci_new_live_trace'
{
id: production_scope.id,
_destroy: 1
}
]
} }
} }
end end
it 'destroys successfully' do context 'when user is reporter' do
subject let(:user) { reporter }
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) it 'returns 404' do
scope_json = json_response['scopes'].find do |s| is_expected.to have_gitlab_http_status(:not_found)
s['environment_scope'] == 'production'
end end
expect(scope_json['strategies']).to eq([{
"name" => "default",
"parameters" => {}
}])
end end
it 'creates a gradualRolloutUserId strategy' do context "when changing default scope's spec" do
scope = create_scope(feature_flag, 'production', true, []) let(:params) do
{
put_request(feature_flag, scopes_attributes: [{ namespace_id: project.namespace,
id: scope.id, project_id: project,
strategies: [{ name: 'gradualRolloutUserId', iid: feature_flag.iid,
parameters: { groupId: 'default', percentage: "70" } }] operations_feature_flag: {
}]) scopes_attributes: [
{
expect(response).to have_gitlab_http_status(:ok) id: feature_flag.default_scope.id,
scope_json = json_response['scopes'].find do |s| environment_scope: 'review/*'
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 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 }]) it 'returns 400' do
is_expected.to have_gitlab_http_status(:bad_request)
expect(response).to have_gitlab_http_status(:ok)
scope_json = json_response['scopes'].find do |s|
s['environment_scope'] == 'production'
end end
expect(scope_json['strategies']).to eq([{
"name" => "gradualRolloutUserId",
"parameters" => { "groupId" => "default", "percentage" => "10" }
}])
end end
it 'does not accept extra parameters in the strategy params' do it 'does not update a legacy feature flag' 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') put_request(feature_flag, name: 'ci_new_live_trace')
expect(response).to have_gitlab_http_status(:bad_request) expect(response).to have_gitlab_http_status(:bad_request)
expect(json_response['message']).to eq(["Legacy feature flags are read-only"]) expect(json_response['message']).to eq(["Legacy feature flags are read-only"])
end 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 end
context 'with a version 2 feature flag' do context 'with a version 2 feature flag' do
...@@ -1517,15 +1168,6 @@ RSpec.describe Projects::FeatureFlagsController do ...@@ -1517,15 +1168,6 @@ RSpec.describe Projects::FeatureFlagsController do
expect(response).to have_gitlab_http_status(:ok) expect(response).to have_gitlab_http_status(:ok)
expect(json_response['strategies'].first['scopes']).to eq([]) expect(json_response['strategies'].first['scopes']).to eq([])
end 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
end end
......
...@@ -25,7 +25,6 @@ RSpec.describe 'User sees feature flag list', :js do ...@@ -25,7 +25,6 @@ RSpec.describe 'User sees feature flag list', :js do
create_flag(project, 'mr_train', true).tap do |feature_flag| create_flag(project, 'mr_train', true).tap do |feature_flag|
create_scope(feature_flag, 'production', false) create_scope(feature_flag, 'production', false)
end end
stub_feature_flags(feature_flags_legacy_read_only_override: false)
end end
it 'user sees the first flag' do it 'user sees the first flag' do
...@@ -79,41 +78,6 @@ RSpec.describe 'User sees feature flag list', :js do ...@@ -79,41 +78,6 @@ RSpec.describe 'User sees feature flag list', :js do
expect_status_toggle_button_to_be_disabled expect_status_toggle_button_to_be_disabled
end end
end end
context 'when legacy feature flags are not read-only' do
before do
stub_feature_flags(feature_flags_legacy_read_only: false)
end
it 'user updates the status toggle' do
visit(project_feature_flags_path(project))
within_feature_flag_row(1) do
status_toggle_button.click
expect_status_toggle_button_to_be_checked
end
end
end
context 'when legacy feature flags are read-only but the override is active for a project' do
before do
stub_feature_flags(
feature_flags_legacy_read_only: true,
feature_flags_legacy_read_only_override: project
)
end
it 'user updates the status toggle' do
visit(project_feature_flags_path(project))
within_feature_flag_row(1) do
status_toggle_button.click
expect_status_toggle_button_to_be_checked
end
end
end
end end
context 'with new version flags' do context 'with new version flags' do
......
...@@ -14,8 +14,7 @@ RSpec.describe 'User updates feature flag', :js do ...@@ -14,8 +14,7 @@ RSpec.describe 'User updates feature flag', :js do
before do before do
stub_feature_flags( stub_feature_flags(
feature_flag_permissions: false, feature_flag_permissions: false
feature_flags_legacy_read_only_override: false
) )
sign_in(user) sign_in(user)
end end
...@@ -79,117 +78,11 @@ RSpec.describe 'User updates feature flag', :js do ...@@ -79,117 +78,11 @@ RSpec.describe 'User updates feature flag', :js do
let!(:scope) { create_scope(feature_flag, 'review/*', true) } let!(:scope) { create_scope(feature_flag, 'review/*', true) }
context 'when legacy flags are editable' do it 'the user cannot edit the flag' do
before do visit(edit_project_feature_flag_path(project, feature_flag))
stub_feature_flags(feature_flags_legacy_read_only: false)
visit(edit_project_feature_flag_path(project, feature_flag))
end
it 'user sees persisted default scope' do
within_scope_row(1) do
within_environment_spec do
expect(page).to have_content('* (All Environments)')
end
within_status do
expect(find('.project-feature-toggle')['aria-label'])
.to eq('Toggle Status: ON')
end
end
end
context 'when user updates the status of a scope' do
before do
within_scope_row(2) do
within_status { find('.project-feature-toggle').click }
end
click_button 'Save changes'
expect(page).to have_current_path(project_feature_flags_path(project))
end
it 'shows the updated feature flag' do
within_feature_flag_row(1) do
expect(page.find('.feature-flag-name')).to have_content('ci_live_trace')
expect_status_toggle_button_to_be_checked
within_feature_flag_scopes do
expect(page.find('.badge:nth-child(1)')).to have_content('*')
expect(page.find('.badge:nth-child(1)')['class']).to include('badge-info')
expect(page.find('.badge:nth-child(2)')).to have_content('review/*')
expect(page.find('.badge:nth-child(2)')['class']).to include('badge-muted')
end
end
end
end
context 'when user adds a new scope' do
before do
within_scope_row(3) do
within_environment_spec do
find('.js-env-search > input').set('production')
find('.js-create-button').click
end
end
click_button 'Save changes'
expect(page).to have_current_path(project_feature_flags_path(project))
end
it 'shows the newly created scope' do
within_feature_flag_row(1) do
within_feature_flag_scopes do
expect(page.find('.badge:nth-child(3)')).to have_content('production')
expect(page.find('.badge:nth-child(3)')['class']).to include('badge-muted')
end
end
end
end
context 'when user deletes a scope' do
before do
within_scope_row(2) do
within_delete { find('.js-delete-scope').click }
end
click_button 'Save changes'
expect(page).to have_current_path(project_feature_flags_path(project))
end
it 'shows the updated feature flag' do
within_feature_flag_row(1) do
within_feature_flag_scopes do
expect(page).to have_css('.badge:nth-child(1)')
expect(page).not_to have_css('.badge:nth-child(2)')
end
end
end
end
end
context 'when legacy flags are read-only' do
it 'the user cannot edit the flag' do
visit(edit_project_feature_flag_path(project, feature_flag))
expect(page).to have_text 'This feature flag is read-only, and it will be removed in 14.0.'
expect(page).to have_css('button.js-ff-submit.disabled')
end
end
context 'when legacy flags are read-only, but the override is active for one project' do
it 'the user can edit the flag' do
stub_feature_flags(feature_flags_legacy_read_only_override: project)
visit(edit_project_feature_flag_path(project, feature_flag))
status_toggle_button.click
click_button 'Save changes'
expect(page).to have_current_path(project_feature_flags_path(project)) expect(page).to have_text 'This feature flag is read-only, and it will be removed in 14.0.'
within_feature_flag_row(1) do expect(page).to have_css('button.js-ff-submit.disabled')
expect_status_toggle_button_not_to_be_checked
end
end
end end
end end
end end
...@@ -120,13 +120,11 @@ describe('Feature flag table', () => { ...@@ -120,13 +120,11 @@ describe('Feature flag table', () => {
describe('when active and with an update toggle', () => { describe('when active and with an update toggle', () => {
let toggle; let toggle;
let spy;
beforeEach(() => { beforeEach(() => {
props.featureFlags[0].update_path = props.featureFlags[0].destroy_path; props.featureFlags[0].update_path = props.featureFlags[0].destroy_path;
createWrapper(props); createWrapper(props);
toggle = wrapper.find(GlToggle); toggle = wrapper.find(GlToggle);
spy = mockTracking('_category_', toggle.element, jest.spyOn);
}); });
it('should have a toggle', () => { it('should have a toggle', () => {
...@@ -142,14 +140,6 @@ describe('Feature flag table', () => { ...@@ -142,14 +140,6 @@ describe('Feature flag table', () => {
expect(wrapper.emitted('toggle-flag')).toEqual([[flag]]); expect(wrapper.emitted('toggle-flag')).toEqual([[flag]]);
}); });
}); });
it('should track a click', () => {
toggle.trigger('click');
expect(spy).toHaveBeenCalledWith('_category_', 'click_button', {
label: 'feature_flag_toggle',
});
});
}); });
describe('with an active scope and a percentage rollout strategy', () => { describe('with an active scope and a percentage rollout strategy', () => {
...@@ -180,6 +170,8 @@ describe('Feature flag table', () => { ...@@ -180,6 +170,8 @@ describe('Feature flag table', () => {
}); });
describe('with a new version flag', () => { describe('with a new version flag', () => {
let toggle;
let spy;
let badges; let badges;
beforeEach(() => { beforeEach(() => {
...@@ -194,6 +186,7 @@ describe('Feature flag table', () => { ...@@ -194,6 +186,7 @@ describe('Feature flag table', () => {
description: 'flag description', description: 'flag description',
destroy_path: 'destroy/path', destroy_path: 'destroy/path',
edit_path: 'edit/path', edit_path: 'edit/path',
update_path: 'update/path',
version: NEW_VERSION_FLAG, version: NEW_VERSION_FLAG,
scopes: [], scopes: [],
strategies: [ strategies: [
...@@ -226,6 +219,8 @@ describe('Feature flag table', () => { ...@@ -226,6 +219,8 @@ describe('Feature flag table', () => {
provide: { csrfToken: 'fakeToken', glFeatures: { featureFlagsNewVersion: true } }, provide: { csrfToken: 'fakeToken', glFeatures: { featureFlagsNewVersion: true } },
}); });
toggle = wrapper.find(GlToggle);
spy = mockTracking('_category_', toggle.element, jest.spyOn);
badges = wrapper.findAll('[data-testid="strategy-badge"]'); badges = wrapper.findAll('[data-testid="strategy-badge"]');
}); });
...@@ -254,6 +249,14 @@ describe('Feature flag table', () => { ...@@ -254,6 +249,14 @@ describe('Feature flag table', () => {
it('shows the name of a user list for user list', () => { it('shows the name of a user list for user list', () => {
expect(badges.at(3).text()).toContain('User List - test list'); expect(badges.at(3).text()).toContain('User List - test list');
}); });
it('tracks a click', () => {
toggle.trigger('click');
expect(spy).toHaveBeenCalledWith('_category_', 'click_button', {
label: 'feature_flag_toggle',
});
});
}); });
it('renders a feature flag without an iid', () => { it('renders a feature flag without an iid', () => {
......
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