Commit cd89e30c authored by Andrew Fontaine's avatar Andrew Fontaine

Rework Users with ID for Feature Flags

Now defines list of feature flags per scope, instead of per flag.

This allows different scopes to enable feature flags for different
users!
parent 123bcddf
---
title: Make User IDs work per scope in Feature Flags
merge_request: 19399
author:
type: added
<script> <script>
import Vue from 'vue'; import Vue from 'vue';
import _ from 'underscore'; import _ from 'underscore';
import { GlButton, GlBadge, GlTooltip, GlTooltipDirective } from '@gitlab/ui'; import {
GlButton,
GlBadge,
GlTooltip,
GlTooltipDirective,
GlFormTextarea,
GlFormCheckbox,
} from '@gitlab/ui';
import { s__, sprintf } from '~/locale'; import { s__, sprintf } from '~/locale';
import featureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin'; import featureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
import ToggleButton from '~/vue_shared/components/toggle_button.vue'; import ToggleButton from '~/vue_shared/components/toggle_button.vue';
...@@ -10,6 +17,7 @@ import EnvironmentsDropdown from './environments_dropdown.vue'; ...@@ -10,6 +17,7 @@ import EnvironmentsDropdown from './environments_dropdown.vue';
import { import {
ROLLOUT_STRATEGY_ALL_USERS, ROLLOUT_STRATEGY_ALL_USERS,
ROLLOUT_STRATEGY_PERCENT_ROLLOUT, ROLLOUT_STRATEGY_PERCENT_ROLLOUT,
ROLLOUT_STRATEGY_USER_ID,
ALL_ENVIRONMENTS_NAME, ALL_ENVIRONMENTS_NAME,
INTERNAL_ID_PREFIX, INTERNAL_ID_PREFIX,
} from '../constants'; } from '../constants';
...@@ -20,6 +28,8 @@ export default { ...@@ -20,6 +28,8 @@ export default {
components: { components: {
GlButton, GlButton,
GlBadge, GlBadge,
GlFormCheckbox,
GlFormTextarea,
GlTooltip, GlTooltip,
ToggleButton, ToggleButton,
Icon, Icon,
...@@ -77,6 +87,7 @@ export default { ...@@ -77,6 +87,7 @@ export default {
ROLLOUT_STRATEGY_ALL_USERS, ROLLOUT_STRATEGY_ALL_USERS,
ROLLOUT_STRATEGY_PERCENT_ROLLOUT, ROLLOUT_STRATEGY_PERCENT_ROLLOUT,
ROLLOUT_STRATEGY_USER_ID,
// Matches numbers 0 through 100 // Matches numbers 0 through 100
rolloutPercentageRegex: /^[0-9]$|^[1-9][0-9]$|^100$/, rolloutPercentageRegex: /^[0-9]$|^[1-9][0-9]$|^100$/,
...@@ -102,11 +113,6 @@ export default { ...@@ -102,11 +113,6 @@ export default {
permissionsFlag() { permissionsFlag() {
return this.glFeatures.featureFlagPermissions; return this.glFeatures.featureFlagPermissions;
}, },
userIds() {
const scope = this.formScopes.find(s => Array.isArray(s.rolloutUserIds)) || {};
return scope.rolloutUserIds || [];
},
}, },
methods: { methods: {
isAllEnvironment(name) { isAllEnvironment(name) {
...@@ -154,18 +160,9 @@ export default { ...@@ -154,18 +160,9 @@ export default {
scopes: this.formScopes, scopes: this.formScopes,
}); });
}, },
updateUserIds(userIds) {
this.formScopes = this.formScopes.map(s => ({
...s,
rolloutUserIds: userIds,
}));
},
canUpdateScope(scope) { canUpdateScope(scope) {
return !this.permissionsFlag || scope.canUpdate; return !this.permissionsFlag || scope.canUpdate;
}, },
isRolloutPercentageInvalid: _.memoize(function isRolloutPercentageInvalid(percentage) { isRolloutPercentageInvalid: _.memoize(function isRolloutPercentageInvalid(percentage) {
return !this.$options.rolloutPercentageRegex.test(percentage); return !this.$options.rolloutPercentageRegex.test(percentage);
}), }),
...@@ -187,6 +184,17 @@ export default { ...@@ -187,6 +184,17 @@ export default {
rolloutPercentageId(index) { rolloutPercentageId(index) {
return `rollout-percentage-${index}`; return `rollout-percentage-${index}`;
}, },
rolloutUserId(index) {
return `rollout-user-id-${index}`;
},
shouldDisplayIncludeUserIds(scope) {
return ![ROLLOUT_STRATEGY_ALL_USERS, ROLLOUT_STRATEGY_USER_ID].includes(
scope.rolloutStrategy,
);
},
shouldDisplayUserIds(scope) {
return scope.rolloutStrategy === ROLLOUT_STRATEGY_USER_ID || scope.shouldIncludeUserIds;
},
}, },
}; };
</script> </script>
...@@ -241,7 +249,7 @@ export default { ...@@ -241,7 +249,7 @@ export default {
<div <div
v-for="(scope, index) in filteredScopes" v-for="(scope, index) in filteredScopes"
:key="scope.id" :key="scope.id"
class="gl-responsive-table-row" class="gl-responsive-table-row align-items-start"
role="row" role="row"
> >
<div class="table-section section-30" role="gridcell"> <div class="table-section section-30" role="gridcell">
...@@ -251,7 +259,7 @@ export default { ...@@ -251,7 +259,7 @@ export default {
<div <div
class="table-mobile-content js-feature-flag-status d-flex align-items-center justify-content-start" class="table-mobile-content js-feature-flag-status d-flex align-items-center justify-content-start"
> >
<p v-if="isAllEnvironment(scope.environmentScope)" class="js-scope-all pl-3"> <p v-if="isAllEnvironment(scope.environmentScope)" class="js-scope-all pl-3 mb-0">
{{ $options.allEnvironmentsText }} {{ $options.allEnvironmentsText }}
</p> </p>
...@@ -285,7 +293,7 @@ export default { ...@@ -285,7 +293,7 @@ export default {
</div> </div>
</div> </div>
<div class="table-section section-40" role="gridcell"> <div class="table-section section-40 align-items-start" role="gridcell">
<div class="table-mobile-header" role="rowheader"> <div class="table-mobile-header" role="rowheader">
{{ s__('FeatureFlags|Rollout Strategy') }} {{ s__('FeatureFlags|Rollout Strategy') }}
</div> </div>
...@@ -300,12 +308,15 @@ export default { ...@@ -300,12 +308,15 @@ export default {
:disabled="!scope.active" :disabled="!scope.active"
class="form-control select-control w-100 js-rollout-strategy" class="form-control select-control w-100 js-rollout-strategy"
> >
<option :value="$options.ROLLOUT_STRATEGY_ALL_USERS">{{ <option :value="$options.ROLLOUT_STRATEGY_ALL_USERS">
s__('FeatureFlags|All users') {{ s__('FeatureFlags|All users') }}
}}</option> </option>
<option :value="$options.ROLLOUT_STRATEGY_PERCENT_ROLLOUT">{{ <option :value="$options.ROLLOUT_STRATEGY_PERCENT_ROLLOUT">
s__('FeatureFlags|Percent rollout (logged in users)') {{ s__('FeatureFlags|Percent rollout (logged in users)') }}
}}</option> </option>
<option :value="$options.ROLLOUT_STRATEGY_USER_ID">
{{ s__('FeatureFlags|User IDs') }}
</option>
</select> </select>
<i aria-hidden="true" data-hidden="true" class="fa fa-chevron-down"></i> <i aria-hidden="true" data-hidden="true" class="fa fa-chevron-down"></i>
</div> </div>
...@@ -342,6 +353,24 @@ export default { ...@@ -342,6 +353,24 @@ export default {
</gl-tooltip> </gl-tooltip>
<span class="ml-1">%</span> <span class="ml-1">%</span>
</div> </div>
<div class="d-flex flex-column align-items-start mt-2 w-100">
<gl-form-checkbox
v-if="shouldDisplayIncludeUserIds(scope)"
v-model="scope.shouldIncludeUserIds"
>
{{ s__('FeatureFlags|Include additional user IDs') }}
</gl-form-checkbox>
<template v-if="shouldDisplayUserIds(scope)">
<label :for="rolloutUserId(index)" class="mb-2">
{{ s__('FeatureFlags|User IDs') }}
</label>
<gl-form-textarea
:id="rolloutUserId(index)"
v-model="scope.rolloutUserIds"
class="w-100"
/>
</template>
</div>
</div> </div>
</div> </div>
...@@ -413,7 +442,6 @@ export default { ...@@ -413,7 +442,6 @@ export default {
</div> </div>
</div> </div>
</fieldset> </fieldset>
<user-with-id :value="userIds" @input="updateUserIds" />
<div class="form-actions"> <div class="form-actions">
<gl-button <gl-button
......
...@@ -29,7 +29,10 @@ export const mapToScopesViewModel = scopesFromRails => ...@@ -29,7 +29,10 @@ export const mapToScopesViewModel = scopesFromRails =>
strat => strat.name === ROLLOUT_STRATEGY_USER_ID, strat => strat.name === ROLLOUT_STRATEGY_USER_ID,
); );
const rolloutUserIds = (fetchUserIdParams(userStrategy) || '').split(',').filter(id => id); const rolloutUserIds = (fetchUserIdParams(userStrategy) || '')
.split(',')
.filter(id => id)
.join(', ');
return { return {
id: s.id, id: s.id,
...@@ -43,6 +46,7 @@ export const mapToScopesViewModel = scopesFromRails => ...@@ -43,6 +46,7 @@ export const mapToScopesViewModel = scopesFromRails =>
// eslint-disable-next-line no-underscore-dangle // eslint-disable-next-line no-underscore-dangle
shouldBeDestroyed: Boolean(s._destroy), shouldBeDestroyed: Boolean(s._destroy),
shouldIncludeUserIds: rolloutUserIds.length > 0,
}; };
}); });
/** /**
...@@ -59,8 +63,8 @@ export const mapFromScopesViewModel = params => { ...@@ -59,8 +63,8 @@ export const mapFromScopesViewModel = params => {
} }
const userIdParameters = {}; const userIdParameters = {};
if (Array.isArray(s.rolloutUserIds) && s.rolloutUserIds.length > 0) { if (s.shouldIncludeUserIds || s.rolloutStrategy === ROLLOUT_STRATEGY_USER_ID) {
userIdParameters.userIds = s.rolloutUserIds.join(','); userIdParameters.userIds = (s.rolloutUserIds || '').replace(/, /g, ',');
} }
// Strip out any internal IDs // Strip out any internal IDs
...@@ -113,7 +117,7 @@ export const createNewEnvironmentScope = (overrides = {}, featureFlagPermissions ...@@ -113,7 +117,7 @@ export const createNewEnvironmentScope = (overrides = {}, featureFlagPermissions
id: _.uniqueId(INTERNAL_ID_PREFIX), id: _.uniqueId(INTERNAL_ID_PREFIX),
rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS, rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS,
rolloutPercentage: DEFAULT_PERCENT_ROLLOUT, rolloutPercentage: DEFAULT_PERCENT_ROLLOUT,
rolloutUserIds: [], rolloutUserIds: '',
}; };
const newScope = { const newScope = {
......
...@@ -51,8 +51,9 @@ describe('feature flags helpers spec', () => { ...@@ -51,8 +51,9 @@ describe('feature flags helpers spec', () => {
protected: true, protected: true,
rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT, rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT,
rolloutPercentage: '56', rolloutPercentage: '56',
rolloutUserIds: ['123', '234'], rolloutUserIds: '123, 234',
shouldBeDestroyed: true, shouldBeDestroyed: true,
shouldIncludeUserIds: true,
}, },
]; ];
...@@ -102,7 +103,8 @@ describe('feature flags helpers spec', () => { ...@@ -102,7 +103,8 @@ describe('feature flags helpers spec', () => {
shouldBeDestroyed: true, shouldBeDestroyed: true,
rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT, rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT,
rolloutPercentage: '48', rolloutPercentage: '48',
rolloutUserIds: ['123', '234'], rolloutUserIds: '123, 234',
shouldIncludeUserIds: true,
}, },
], ],
}; };
...@@ -179,7 +181,7 @@ describe('feature flags helpers spec', () => { ...@@ -179,7 +181,7 @@ describe('feature flags helpers spec', () => {
id: expect.stringContaining(INTERNAL_ID_PREFIX), id: expect.stringContaining(INTERNAL_ID_PREFIX),
rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS, rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS,
rolloutPercentage: DEFAULT_PERCENT_ROLLOUT, rolloutPercentage: DEFAULT_PERCENT_ROLLOUT,
rolloutUserIds: [], rolloutUserIds: '',
}; };
const actual = createNewEnvironmentScope(); const actual = createNewEnvironmentScope();
...@@ -199,7 +201,7 @@ describe('feature flags helpers spec', () => { ...@@ -199,7 +201,7 @@ describe('feature flags helpers spec', () => {
id: expect.stringContaining(INTERNAL_ID_PREFIX), id: expect.stringContaining(INTERNAL_ID_PREFIX),
rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS, rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS,
rolloutPercentage: DEFAULT_PERCENT_ROLLOUT, rolloutPercentage: DEFAULT_PERCENT_ROLLOUT,
rolloutUserIds: [], rolloutUserIds: '',
}; };
const actual = createNewEnvironmentScope(overrides); const actual = createNewEnvironmentScope(overrides);
......
import _ from 'underscore'; import _ from 'underscore';
import { createLocalVue, mount } from '@vue/test-utils'; import { createLocalVue, mount } from '@vue/test-utils';
import { GlFormCheckbox, GlFormTextarea } from '@gitlab/ui';
import Form from 'ee/feature_flags/components/form.vue'; import Form from 'ee/feature_flags/components/form.vue';
import ToggleButton from '~/vue_shared/components/toggle_button.vue'; import ToggleButton from '~/vue_shared/components/toggle_button.vue';
import EnvironmentsDropdown from 'ee/feature_flags/components/environments_dropdown.vue'; import EnvironmentsDropdown from 'ee/feature_flags/components/environments_dropdown.vue';
...@@ -101,6 +102,8 @@ describe('feature flag form', () => { ...@@ -101,6 +102,8 @@ describe('feature flag form', () => {
protected: false, protected: false,
rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT, rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT,
rolloutPercentage: '54', rolloutPercentage: '54',
rolloutUserIds: '123',
shouldIncludeUserIds: true,
}, },
{ {
id: 2, id: 2,
...@@ -110,6 +113,8 @@ describe('feature flag form', () => { ...@@ -110,6 +113,8 @@ describe('feature flag form', () => {
protected: true, protected: true,
rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT, rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT,
rolloutPercentage: '54', rolloutPercentage: '54',
rolloutUserIds: '123',
shouldIncludeUserIds: true,
}, },
], ],
}); });
...@@ -124,6 +129,16 @@ describe('feature flag form', () => { ...@@ -124,6 +129,16 @@ describe('feature flag form', () => {
expect(wrapper.find('.js-add-new-scope').exists()).toEqual(true); expect(wrapper.find('.js-add-new-scope').exists()).toEqual(true);
}); });
it('renders the user id checkbox', () => {
expect(wrapper.find(GlFormCheckbox).exists()).toBe(true);
});
it('renders the user id text area', () => {
expect(wrapper.find(GlFormTextarea).exists()).toBe(true);
expect(wrapper.find(GlFormTextarea).vm.value).toBe('123');
});
describe('update scope', () => { describe('update scope', () => {
describe('on click on toggle', () => { describe('on click on toggle', () => {
it('should update the scope', () => { it('should update the scope', () => {
...@@ -247,7 +262,7 @@ describe('feature flag form', () => { ...@@ -247,7 +262,7 @@ describe('feature flag form', () => {
active: false, active: false,
rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS, rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS,
rolloutPercentage: DEFAULT_PERCENT_ROLLOUT, rolloutPercentage: DEFAULT_PERCENT_ROLLOUT,
rolloutUserIds: [], rolloutUserIds: '',
}, },
], ],
}); });
...@@ -301,7 +316,7 @@ describe('feature flag form', () => { ...@@ -301,7 +316,7 @@ describe('feature flag form', () => {
protected: true, protected: true,
rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT, rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT,
rolloutPercentage: '55', rolloutPercentage: '55',
rolloutUserIds: [], rolloutUserIds: '',
}, },
{ {
id: jasmine.any(String), id: jasmine.any(String),
...@@ -311,7 +326,7 @@ describe('feature flag form', () => { ...@@ -311,7 +326,7 @@ describe('feature flag form', () => {
protected: false, protected: false,
rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS, rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS,
rolloutPercentage: DEFAULT_PERCENT_ROLLOUT, rolloutPercentage: DEFAULT_PERCENT_ROLLOUT,
rolloutUserIds: [], rolloutUserIds: '',
}, },
{ {
id: jasmine.any(String), id: jasmine.any(String),
...@@ -321,7 +336,7 @@ describe('feature flag form', () => { ...@@ -321,7 +336,7 @@ describe('feature flag form', () => {
protected: false, protected: false,
rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT, rolloutStrategy: ROLLOUT_STRATEGY_PERCENT_ROLLOUT,
rolloutPercentage: DEFAULT_PERCENT_ROLLOUT, rolloutPercentage: DEFAULT_PERCENT_ROLLOUT,
rolloutUserIds: [], rolloutUserIds: '',
}, },
]); ]);
}) })
...@@ -330,87 +345,4 @@ describe('feature flag form', () => { ...@@ -330,87 +345,4 @@ describe('feature flag form', () => {
}); });
}); });
}); });
describe('updateUserIds', () => {
beforeEach(() => {
factory({
...requiredProps,
name: 'feature_flag_1',
description: 'this is a feature flag',
scopes: [
{
environment_scope: 'production',
can_update: true,
protected: true,
active: false,
},
{
environment_scope: 'staging',
can_update: true,
protected: true,
active: false,
},
],
});
});
it('should set the user ids on all scopes', () => {
wrapper.vm.updateUserIds(['123', '456']);
wrapper.vm.formScopes.forEach(s => {
expect(s.rolloutUserIds).toEqual(['123', '456']);
});
});
});
describe('userIds', () => {
it('should get the user ids from the first scope with them', () => {
factory({
...requiredProps,
name: 'feature_flag_1',
description: 'this is a feature flag',
scopes: [
{
environment_scope: 'production',
can_update: true,
protected: true,
active: false,
rolloutUserIds: ['123', '456'],
},
{
environment_scope: 'staging',
can_update: true,
protected: true,
active: false,
rolloutUserIds: ['123', '456'],
},
],
});
expect(wrapper.vm.userIds).toEqual(['123', '456']);
});
it('should return an empty array if there are no user IDs set', () => {
factory({
...requiredProps,
name: 'feature_flag_1',
description: 'this is a feature flag',
scopes: [
{
environment_scope: 'production',
can_update: true,
protected: true,
active: false,
},
{
environment_scope: 'staging',
can_update: true,
protected: true,
active: false,
},
],
});
expect(wrapper.vm.userIds).toEqual([]);
});
});
}); });
...@@ -66,7 +66,7 @@ describe('New feature flag form', () => { ...@@ -66,7 +66,7 @@ describe('New feature flag form', () => {
active: true, active: true,
rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS, rolloutStrategy: ROLLOUT_STRATEGY_ALL_USERS,
rolloutPercentage: DEFAULT_PERCENT_ROLLOUT, rolloutPercentage: DEFAULT_PERCENT_ROLLOUT,
rolloutUserIds: [], rolloutUserIds: '',
}, },
]); ]);
......
...@@ -7222,6 +7222,9 @@ msgstr "" ...@@ -7222,6 +7222,9 @@ msgstr ""
msgid "FeatureFlags|Inactive flag for %{scope}" msgid "FeatureFlags|Inactive flag for %{scope}"
msgstr "" msgstr ""
msgid "FeatureFlags|Include additional user IDs"
msgstr ""
msgid "FeatureFlags|Install a %{docs_link_anchored_start}compatible client library%{docs_link_anchored_end} and specify the API URL, application name, and instance ID during the configuration setup. %{docs_link_start}More Information%{docs_link_end}" msgid "FeatureFlags|Install a %{docs_link_anchored_start}compatible client library%{docs_link_anchored_end} and specify the API URL, application name, and instance ID during the configuration setup. %{docs_link_start}More Information%{docs_link_end}"
msgstr "" msgstr ""
......
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