Commit fc7595c7 authored by Phil Hughes's avatar Phil Hughes

Allow approval rules to be reset to project defaults

This adds a button that forces a reset of the approval rules
to match what is set in the projects settings.

https://gitlab.com/gitlab-org/gitlab/-/issues/212754
parent 4bd9f816
<script> <script>
import { mapState, mapActions } from 'vuex'; import { mapState, mapActions } from 'vuex';
import { GlLoadingIcon, GlDeprecatedButton } from '@gitlab/ui'; import { GlLoadingIcon, GlButton } from '@gitlab/ui';
import showToast from '~/vue_shared/plugins/global_toast';
import { __ } from '~/locale';
import ModalRuleCreate from './modal_rule_create.vue'; import ModalRuleCreate from './modal_rule_create.vue';
import ModalRuleRemove from './modal_rule_remove.vue'; import ModalRuleRemove from './modal_rule_remove.vue';
...@@ -8,7 +10,7 @@ export default { ...@@ -8,7 +10,7 @@ export default {
components: { components: {
ModalRuleCreate, ModalRuleCreate,
ModalRuleRemove, ModalRuleRemove,
GlDeprecatedButton, GlButton,
GlLoadingIcon, GlLoadingIcon,
}, },
props: { props: {
...@@ -21,7 +23,9 @@ export default { ...@@ -21,7 +23,9 @@ export default {
computed: { computed: {
...mapState({ ...mapState({
settings: 'settings', settings: 'settings',
isLoading: state => state.approvals.isLoading,
hasLoaded: state => state.approvals.hasLoaded, hasLoaded: state => state.approvals.hasLoaded,
targetBranch: state => state.approvals.targetBranch,
}), }),
createModalId() { createModalId() {
return `${this.settings.prefix}-approvals-create-modal`; return `${this.settings.prefix}-approvals-create-modal`;
...@@ -29,19 +33,28 @@ export default { ...@@ -29,19 +33,28 @@ export default {
removeModalId() { removeModalId() {
return `${this.settings.prefix}-approvals-remove-modal`; return `${this.settings.prefix}-approvals-remove-modal`;
}, },
targetBranch() {
if (this.settings.prefix === 'mr-edit' && !this.settings.mrSettingsPath) {
return this.settings.mrCreateTargetBranch;
}
return null;
},
}, },
mounted() { mounted() {
return this.fetchRules(this.targetBranch); return this.fetchRules({ targetBranch: this.targetBranch });
}, },
methods: { methods: {
...mapActions(['fetchRules']), ...mapActions(['fetchRules', 'undoRulesChange']),
...mapActions({ openCreateModal: 'createModal/open' }), ...mapActions({ openCreateModal: 'createModal/open' }),
resetToProjectDefaults() {
const { targetBranch } = this;
return this.fetchRules({ targetBranch, resetToDefault: true }).then(() => {
const toast = showToast(__('Approval rules reset to project defaults'), {
action: {
text: __('Undo'),
onClick: () => {
this.undoRulesChange();
toast.goAway(0);
},
},
});
});
},
}, },
}; };
</script> </script>
...@@ -54,14 +67,27 @@ export default { ...@@ -54,14 +67,27 @@ export default {
<slot name="rules"></slot> <slot name="rules"></slot>
</div> </div>
<div v-if="settings.canEdit && settings.allowMultiRule" class="border-bottom py-3 px-2"> <div v-if="settings.canEdit && settings.allowMultiRule" class="border-bottom py-3 px-2">
<div v-if="settings.allowMultiRule" class="d-flex"> <div class="d-flex">
<gl-deprecated-button <gl-button
class="ml-auto btn-info btn-inverted" v-if="targetBranch"
:disabled="isLoading"
class="gl-ml-auto"
data-testid="reset-to-defaults"
@click="resetToProjectDefaults"
>
{{ __('Reset to project defaults') }}
</gl-button>
<gl-button
:class="{ 'gl-ml-3': targetBranch, 'gl-ml-auto': !targetBranch }"
:disabled="isLoading"
category="secondary"
variant="info"
data-qa-selector="add_approvers_button" data-qa-selector="add_approvers_button"
data-testid="add-approval-rule"
@click="openCreateModal(null)" @click="openCreateModal(null)"
> >
{{ __('Add approval rule') }} {{ __('Add approval rule') }}
</gl-deprecated-button> </gl-button>
</div> </div>
</div> </div>
<slot name="footer"></slot> <slot name="footer"></slot>
......
...@@ -18,15 +18,11 @@ export default { ...@@ -18,15 +18,11 @@ export default {
EmptyRule, EmptyRule,
RuleInput, RuleInput,
}, },
data() {
return {
targetBranch: '',
};
},
computed: { computed: {
...mapState(['settings']), ...mapState(['settings']),
...mapState({ ...mapState({
rules: state => state.approvals.rules, rules: state => state.approvals.rules,
targetBranch: state => state.approvals.targetBranch,
}), }),
hasNamedRule() { hasNamedRule() {
if (this.settings.allowMultiRule) { if (this.settings.allowMultiRule) {
...@@ -60,14 +56,18 @@ export default { ...@@ -60,14 +56,18 @@ export default {
}, },
immediate: true, immediate: true,
}, },
targetBranch() {
this.fetchRules({ targetBranch: this.targetBranch });
},
}, },
mounted() { mounted() {
if (this.isEditPath) { if (this.isEditPath) {
this.mergeRequestTargetBranchElement = document.querySelector('#merge_request_target_branch'); this.mergeRequestTargetBranchElement = document.querySelector('#merge_request_target_branch');
const targetBranch = this.mergeRequestTargetBranchElement?.value;
this.targetBranch = this.mergeRequestTargetBranchElement?.value; this.setTargetBranch(targetBranch);
if (this.targetBranch) { if (targetBranch) {
targetBranchMutationObserver = new MutationObserver(this.onTargetBranchMutation); targetBranchMutationObserver = new MutationObserver(this.onTargetBranchMutation);
targetBranchMutationObserver.observe(this.mergeRequestTargetBranchElement, { targetBranchMutationObserver.observe(this.mergeRequestTargetBranchElement, {
attributes: true, attributes: true,
...@@ -85,13 +85,12 @@ export default { ...@@ -85,13 +85,12 @@ export default {
} }
}, },
methods: { methods: {
...mapActions(['setEmptyRule', 'addEmptyRule', 'fetchRules']), ...mapActions(['setEmptyRule', 'addEmptyRule', 'fetchRules', 'setTargetBranch']),
onTargetBranchMutation() { onTargetBranchMutation() {
const selectedTargetBranchValue = this.mergeRequestTargetBranchElement.value; const selectedTargetBranchValue = this.mergeRequestTargetBranchElement.value;
if (this.targetBranch !== selectedTargetBranchValue) { if (this.targetBranch !== selectedTargetBranchValue) {
this.targetBranch = selectedTargetBranchValue; this.setTargetBranch(selectedTargetBranchValue);
this.fetchRules(this.targetBranch);
} }
}, },
indicatorText(rule) { indicatorText(rule) {
......
...@@ -11,11 +11,13 @@ const INPUT_DELETE = 'merge_request[approval_rules_attributes][][_destroy]'; ...@@ -11,11 +11,13 @@ const INPUT_DELETE = 'merge_request[approval_rules_attributes][][_destroy]';
const INPUT_REMOVE_HIDDEN_GROUPS = const INPUT_REMOVE_HIDDEN_GROUPS =
'merge_request[approval_rules_attributes][][remove_hidden_groups]'; 'merge_request[approval_rules_attributes][][remove_hidden_groups]';
const INPUT_FALLBACK_APPROVALS_REQUIRED = 'merge_request[approvals_before_merge]'; const INPUT_FALLBACK_APPROVALS_REQUIRED = 'merge_request[approvals_before_merge]';
const INPUT_RESET_TO_DEFAULTS_ID = 'merge_request[reset_approval_rules_to_defaults]';
export default { export default {
computed: { computed: {
...mapState(['settings']), ...mapState(['settings']),
...mapState({ ...mapState({
resetToDefault: state => state.approvals.resetToDefault,
rules: state => state.approvals.rules, rules: state => state.approvals.rules,
rulesToDelete: state => state.approvals.rulesToDelete, rulesToDelete: state => state.approvals.rulesToDelete,
fallbackApprovalsRequired: state => state.approvals.fallbackApprovalsRequired, fallbackApprovalsRequired: state => state.approvals.fallbackApprovalsRequired,
...@@ -30,11 +32,18 @@ export default { ...@@ -30,11 +32,18 @@ export default {
INPUT_DELETE, INPUT_DELETE,
INPUT_REMOVE_HIDDEN_GROUPS, INPUT_REMOVE_HIDDEN_GROUPS,
INPUT_FALLBACK_APPROVALS_REQUIRED, INPUT_FALLBACK_APPROVALS_REQUIRED,
INPUT_RESET_TO_DEFAULTS_ID,
}; };
</script> </script>
<template> <template>
<div v-if="settings.canEdit"> <div v-if="settings.canEdit">
<input
v-if="resetToDefault"
type="hidden"
value="true"
:name="$options.INPUT_RESET_TO_DEFAULTS_ID"
/>
<div v-for="id in rulesToDelete" :key="id"> <div v-for="id in rulesToDelete" :key="id">
<input :value="id" :name="$options.INPUT_ID" type="hidden" /> <input :value="id" :name="$options.INPUT_ID" type="hidden" />
<input :value="1" :name="$options.INPUT_DELETE" type="hidden" /> <input :value="1" :name="$options.INPUT_DELETE" type="hidden" />
......
...@@ -12,18 +12,19 @@ export default function mountApprovalInput(el) { ...@@ -12,18 +12,19 @@ export default function mountApprovalInput(el) {
return null; return null;
} }
const mrCreateTargetBranch = el.dataset.mrSettingsPath const targetBranch =
? null document.querySelector('#js-target-branch-title')?.textContent ||
: document.querySelector('#js-target-branch-title')?.textContent; document.querySelector('#merge_request_target_branch')?.value;
const store = createStore(mrEditModule(), { const store = createStore(mrEditModule(), {
...el.dataset, ...el.dataset,
prefix: 'mr-edit', prefix: 'mr-edit',
canEdit: parseBoolean(el.dataset.canEdit), canEdit: parseBoolean(el.dataset.canEdit),
allowMultiRule: parseBoolean(el.dataset.allowMultiRule), allowMultiRule: parseBoolean(el.dataset.allowMultiRule),
mrCreateTargetBranch,
}); });
store.dispatch('setTargetBranch', targetBranch);
return new Vue({ return new Vue({
el, el,
store, store,
......
export const SET_LOADING = 'SET_LOADING'; export const SET_LOADING = 'SET_LOADING';
export const SET_APPROVAL_SETTINGS = 'SET_APPROVAL_SETTINGS'; export const SET_APPROVAL_SETTINGS = 'SET_APPROVAL_SETTINGS';
export const ADD_EMPTY_RULE = 'ADD_EMPTY_RULE'; export const ADD_EMPTY_RULE = 'ADD_EMPTY_RULE';
export const SET_RESET_TO_DEFAULT = 'SET_RESET_TO_DEFAULT';
export const UNDO_RULES = 'UNDO_RULES';
...@@ -8,7 +8,7 @@ export default { ...@@ -8,7 +8,7 @@ export default {
[types.SET_APPROVAL_SETTINGS](state, settings) { [types.SET_APPROVAL_SETTINGS](state, settings) {
state.hasLoaded = true; state.hasLoaded = true;
state.rules = settings.rules; state.rules = settings.rules;
state.initialRules = settings.rules; state.initialRules = [...settings.rules];
state.fallbackApprovalsRequired = settings.fallbackApprovalsRequired; state.fallbackApprovalsRequired = settings.fallbackApprovalsRequired;
state.minFallbackApprovalsRequired = settings.minFallbackApprovalsRequired; state.minFallbackApprovalsRequired = settings.minFallbackApprovalsRequired;
}, },
...@@ -26,4 +26,12 @@ export default { ...@@ -26,4 +26,12 @@ export default {
isNew: true, isNew: true,
}); });
}, },
[types.SET_RESET_TO_DEFAULT](state, resetToDefault) {
state.resetToDefault = resetToDefault;
state.oldRules = [...state.rules];
},
[types.UNDO_RULES](state) {
state.resetToDefault = false;
state.rules = [...state.oldRules];
},
}; };
...@@ -5,4 +5,6 @@ export default () => ({ ...@@ -5,4 +5,6 @@ export default () => ({
fallbackApprovalsRequired: 0, fallbackApprovalsRequired: 0,
minFallbackApprovalsRequired: 0, minFallbackApprovalsRequired: 0,
initialRules: [], initialRules: [],
oldRules: [],
resetToDefault: false,
}); });
...@@ -52,8 +52,9 @@ export const requestRules = ({ commit }) => { ...@@ -52,8 +52,9 @@ export const requestRules = ({ commit }) => {
commit(types.SET_LOADING, true); commit(types.SET_LOADING, true);
}; };
export const receiveRulesSuccess = ({ commit }, settings) => { export const receiveRulesSuccess = ({ commit }, { resetToDefault, settings }) => {
commit(types.SET_LOADING, false); commit(types.SET_LOADING, false);
commit(types.SET_RESET_TO_DEFAULT, resetToDefault);
commit(types.SET_APPROVAL_SETTINGS, settings); commit(types.SET_APPROVAL_SETTINGS, settings);
}; };
...@@ -61,11 +62,14 @@ export const receiveRulesError = () => { ...@@ -61,11 +62,14 @@ export const receiveRulesError = () => {
createFlash(__('An error occurred fetching the approval rules.')); createFlash(__('An error occurred fetching the approval rules.'));
}; };
export const fetchRules = ({ rootState, dispatch }, targetBranch = '') => { export const fetchRules = (
{ rootState, dispatch },
{ targetBranch = '', resetToDefault = false },
) => {
dispatch('requestRules'); dispatch('requestRules');
const { mrSettingsPath, projectSettingsPath } = rootState.settings; const { mrSettingsPath, projectSettingsPath } = rootState.settings;
const path = mrSettingsPath || projectSettingsPath; const path = resetToDefault ? projectSettingsPath : mrSettingsPath || projectSettingsPath;
const params = targetBranch const params = targetBranch
? { ? {
...@@ -82,7 +86,7 @@ export const fetchRules = ({ rootState, dispatch }, targetBranch = '') => { ...@@ -82,7 +86,7 @@ export const fetchRules = ({ rootState, dispatch }, targetBranch = '') => {
...settings, ...settings,
rules: settings.rules.map(x => (x.id ? x : seedNewRule(x))), rules: settings.rules.map(x => (x.id ? x : seedNewRule(x))),
})) }))
.then(settings => dispatch('receiveRulesSuccess', settings)) .then(settings => dispatch('receiveRulesSuccess', { settings, resetToDefault }))
.catch(() => dispatch('receiveRulesError')); .catch(() => dispatch('receiveRulesError'));
}; };
...@@ -148,4 +152,9 @@ export const addEmptyRule = ({ commit }) => { ...@@ -148,4 +152,9 @@ export const addEmptyRule = ({ commit }) => {
commit(types.ADD_EMPTY_RULE); commit(types.ADD_EMPTY_RULE);
}; };
export const setTargetBranch = ({ commit }, targetBranch) =>
commit(types.SET_TARGET_BRANCH, targetBranch);
export const undoRulesChange = ({ commit }) => commit(types.UNDO_RULES);
export default () => {}; export default () => {};
...@@ -7,3 +7,4 @@ export const SET_FALLBACK_RULE = 'SET_FALLBACK_RULE'; ...@@ -7,3 +7,4 @@ export const SET_FALLBACK_RULE = 'SET_FALLBACK_RULE';
export const POST_REGULAR_RULE = 'POST_REGULAR_RULE'; export const POST_REGULAR_RULE = 'POST_REGULAR_RULE';
export const DELETE_ANY_RULE = 'DELETE_ANY_RULE'; export const DELETE_ANY_RULE = 'DELETE_ANY_RULE';
export const SET_EMPTY_RULE = 'SET_EMPTY_RULE'; export const SET_EMPTY_RULE = 'SET_EMPTY_RULE';
export const SET_TARGET_BRANCH = 'SET_TARGET_BRANCH';
...@@ -85,4 +85,7 @@ export default { ...@@ -85,4 +85,7 @@ export default {
]; ];
} }
}, },
[types.SET_TARGET_BRANCH](state, targetBranch) {
state.targetBranch = targetBranch;
},
}; };
...@@ -3,4 +3,5 @@ import baseState from '../base/state'; ...@@ -3,4 +3,5 @@ import baseState from '../base/state';
export default () => ({ export default () => ({
...baseState(), ...baseState(),
rulesToDelete: [], rulesToDelete: [],
targetBranch: '',
}); });
---
title: Allow approval rules to be reset to project defaults
merge_request: 32657
author:
type: added
# frozen_string_literal: true
require 'spec_helper'
describe 'Merge Requests > User resets approvers', :js do
include FeatureApprovalHelper
let(:project) { create(:project, :repository) }
let(:user) { project.creator }
let(:first_user) { create(:user) }
let(:project_approvers) { create_list(:user, 3) }
let(:merge_request) do
create(:merge_request, approval_users: [first_user], title: 'Bugfix1', source_project: project)
end
let!(:rule) { create(:approval_project_rule, project: project, users: project_approvers, approvals_required: 1 )}
before do
stub_licensed_features(multiple_approval_rules: true)
project_approvers.each do |approver|
project.add_developer(approver)
end
merge_request.approvals.create!(user: first_user)
project.add_developer(user)
sign_in(user)
visit edit_project_merge_request_path(project, merge_request)
wait_for_requests
end
it 'resets approvers for merge requests' do
expect_avatar(find('.js-members'), first_user)
click_button 'Reset to project defaults'
wait_for_requests
expect_avatar(find('.js-members'), project_approvers)
click_button 'Save changes'
wait_for_requests
expect(page).to have_content 'Requires approval'
end
end
import { shallowMount, createLocalVue } from '@vue/test-utils'; import { shallowMount, createLocalVue } from '@vue/test-utils';
import Vuex from 'vuex'; import Vuex from 'vuex';
import { GlLoadingIcon, GlDeprecatedButton } from '@gitlab/ui'; import { GlLoadingIcon } from '@gitlab/ui';
import showToast from '~/vue_shared/plugins/global_toast';
import App from 'ee/approvals/components/app.vue'; import App from 'ee/approvals/components/app.vue';
import ModalRuleCreate from 'ee/approvals/components/modal_rule_create.vue'; import ModalRuleCreate from 'ee/approvals/components/modal_rule_create.vue';
import ModalRuleRemove from 'ee/approvals/components/modal_rule_remove.vue'; import ModalRuleRemove from 'ee/approvals/components/modal_rule_remove.vue';
import { createStoreOptions } from 'ee/approvals/stores'; import { createStoreOptions } from 'ee/approvals/stores';
import settingsModule from 'ee/approvals/stores/modules/project_settings'; import settingsModule from 'ee/approvals/stores/modules/project_settings';
jest.mock('~/vue_shared/plugins/global_toast');
const localVue = createLocalVue(); const localVue = createLocalVue();
localVue.use(Vuex); localVue.use(Vuex);
...@@ -18,6 +21,7 @@ describe('EE Approvals App', () => { ...@@ -18,6 +21,7 @@ describe('EE Approvals App', () => {
let wrapper; let wrapper;
let slots; let slots;
const targetBranchName = 'development';
const factory = () => { const factory = () => {
wrapper = shallowMount(localVue.extend(App), { wrapper = shallowMount(localVue.extend(App), {
localVue, localVue,
...@@ -25,7 +29,8 @@ describe('EE Approvals App', () => { ...@@ -25,7 +29,8 @@ describe('EE Approvals App', () => {
store: new Vuex.Store(store), store: new Vuex.Store(store),
}); });
}; };
const findAddButton = () => wrapper.find(GlDeprecatedButton); const findAddButton = () => wrapper.find('[data-testid="add-approval-rule"]');
const findResetButton = () => wrapper.find('[data-testid="reset-to-defaults"]');
const findLoadingIcon = () => wrapper.find(GlLoadingIcon); const findLoadingIcon = () => wrapper.find(GlLoadingIcon);
const findRules = () => wrapper.find(`.${TEST_RULES_CLASS}`); const findRules = () => wrapper.find(`.${TEST_RULES_CLASS}`);
...@@ -40,20 +45,16 @@ describe('EE Approvals App', () => { ...@@ -40,20 +45,16 @@ describe('EE Approvals App', () => {
}); });
store.modules.approvals.actions = { store.modules.approvals.actions = {
fetchRules: jest.fn(), fetchRules: jest.fn().mockResolvedValue(),
}; };
store.modules.approvals.state.targetBranch = targetBranchName;
jest.spyOn(store.modules.approvals.actions, 'fetchRules'); jest.spyOn(store.modules.approvals.actions, 'fetchRules');
jest.spyOn(store.modules.createModal.actions, 'open'); jest.spyOn(store.modules.createModal.actions, 'open');
}); });
describe('targetBranch', () => { describe('targetBranch', () => {
const targetBranchName = 'development';
beforeEach(() => {
store.state.settings.mrCreateTargetBranch = targetBranchName;
});
it('passes the target branch name in fetchRules for MR create path', () => { it('passes the target branch name in fetchRules for MR create path', () => {
store.state.settings.prefix = 'mr-edit'; store.state.settings.prefix = 'mr-edit';
store.state.settings.mrSettingsPath = null; store.state.settings.mrSettingsPath = null;
...@@ -61,30 +62,31 @@ describe('EE Approvals App', () => { ...@@ -61,30 +62,31 @@ describe('EE Approvals App', () => {
expect(store.modules.approvals.actions.fetchRules).toHaveBeenCalledWith( expect(store.modules.approvals.actions.fetchRules).toHaveBeenCalledWith(
expect.anything(), expect.anything(),
targetBranchName, { targetBranch: targetBranchName },
undefined, undefined,
); );
}); });
it('does not pass the target branch name in fetchRules for MR edit path', () => { it('passes the target branch name in fetchRules for MR edit path', () => {
store.state.settings.prefix = 'mr-edit'; store.state.settings.prefix = 'mr-edit';
store.state.settings.mrSettingsPath = 'some/path'; store.state.settings.mrSettingsPath = 'some/path';
factory(); factory();
expect(store.modules.approvals.actions.fetchRules).toHaveBeenCalledWith( expect(store.modules.approvals.actions.fetchRules).toHaveBeenCalledWith(
expect.anything(), expect.anything(),
null, { targetBranch: targetBranchName },
undefined, undefined,
); );
}); });
it('does not pass the target branch name in fetchRules for project settings path', () => { it('does not pass the target branch name in fetchRules for project settings path', () => {
store.state.settings.prefix = 'project-settings'; store.state.settings.prefix = 'project-settings';
store.modules.approvals.state.targetBranch = null;
factory(); factory();
expect(store.modules.approvals.actions.fetchRules).toHaveBeenCalledWith( expect(store.modules.approvals.actions.fetchRules).toHaveBeenCalledWith(
expect.anything(), expect.anything(),
null, { targetBranch: null },
undefined, undefined,
); );
}); });
...@@ -203,4 +205,36 @@ describe('EE Approvals App', () => { ...@@ -203,4 +205,36 @@ describe('EE Approvals App', () => {
expect(findAddButton().exists()).toBe(false); expect(findAddButton().exists()).toBe(false);
}); });
}); });
describe('when resetting to project defaults', () => {
const targetBranch = 'development';
beforeEach(() => {
store.state.settings.targetBranch = targetBranch;
store.state.settings.prefix = 'mr-edit';
store.state.settings.allowMultiRule = true;
store.modules.approvals.state.hasLoaded = true;
store.modules.approvals.state.rules = [{ id: 1 }];
});
it('calls fetchRules to reset to defaults', () => {
factory();
findResetButton().vm.$emit('click');
return wrapper.vm.$nextTick().then(() => {
expect(store.modules.approvals.actions.fetchRules).toHaveBeenLastCalledWith(
expect.anything(),
{ targetBranch, resetToDefault: true },
undefined,
);
expect(showToast).toHaveBeenCalledWith('Approval rules reset to project defaults', {
action: {
text: 'Undo',
onClick: expect.anything(),
},
});
});
});
});
}); });
...@@ -40,12 +40,6 @@ describe('EE Approvals MRRules', () => { ...@@ -40,12 +40,6 @@ describe('EE Approvals MRRules', () => {
.find(UserAvatarList) .find(UserAvatarList)
.props('items'); .props('items');
const findRuleControls = () => wrapper.find('td.js-controls').find(RuleControls); const findRuleControls = () => wrapper.find('td.js-controls').find(RuleControls);
const setTargetBranchInputValue = () => {
const value = 'new value';
const element = document.querySelector('#merge_request_target_branch');
element.value = value;
return value;
};
const callTargetBranchHandler = MutationObserverSpy => { const callTargetBranchHandler = MutationObserverSpy => {
const onTargetBranchMutationHandler = MutationObserverSpy.mock.calls[0][0]; const onTargetBranchMutationHandler = MutationObserverSpy.mock.calls[0][0];
return onTargetBranchMutationHandler(); return onTargetBranchMutationHandler();
...@@ -61,6 +55,7 @@ describe('EE Approvals MRRules', () => { ...@@ -61,6 +55,7 @@ describe('EE Approvals MRRules', () => {
store.modules.approvals.state = { store.modules.approvals.state = {
hasLoaded: true, hasLoaded: true,
rules: [], rules: [],
targetBranch: 'master',
}; };
store.modules.approvals.actions.putRule = jest.fn(); store.modules.approvals.actions.putRule = jest.fn();
}); });
...@@ -87,6 +82,7 @@ describe('EE Approvals MRRules', () => { ...@@ -87,6 +82,7 @@ describe('EE Approvals MRRules', () => {
fetchRules: jest.fn(), fetchRules: jest.fn(),
addEmptyRule: jest.fn(), addEmptyRule: jest.fn(),
setEmptyRule: jest.fn(), setEmptyRule: jest.fn(),
setTargetBranch: jest.fn(),
}; };
store.state.settings.mrSettingsPath = 'some/path'; store.state.settings.mrSettingsPath = 'some/path';
store.state.settings.eligibleApproversDocsPath = 'some/path'; store.state.settings.eligibleApproversDocsPath = 'some/path';
...@@ -97,25 +93,20 @@ describe('EE Approvals MRRules', () => { ...@@ -97,25 +93,20 @@ describe('EE Approvals MRRules', () => {
targetBranchInputElement.parentNode.removeChild(targetBranchInputElement); targetBranchInputElement.parentNode.removeChild(targetBranchInputElement);
}); });
it('sets the target branch data to be the same value as the target branch dropdown', () => {
factory();
expect(wrapper.vm.targetBranch).toBe(initialTargetBranch);
});
it('updates the target branch data when the target branch dropdown is changed', () => { it('updates the target branch data when the target branch dropdown is changed', () => {
factory(); factory();
const newValue = setTargetBranchInputValue();
callTargetBranchHandler(global.MutationObserver); callTargetBranchHandler(global.MutationObserver);
expect(wrapper.vm.targetBranch).toBe(newValue); expect(store.modules.approvals.actions.setTargetBranch).toHaveBeenCalled();
}); });
it('re-fetches rules when target branch has changed', () => { it('re-fetches rules when target branch has changed', () => {
factory(); factory();
setTargetBranchInputValue(); store.modules.approvals.state.targetBranch = 'master123';
callTargetBranchHandler(global.MutationObserver);
return wrapper.vm.$nextTick().then(() => {
expect(store.modules.approvals.actions.fetchRules).toHaveBeenCalled(); expect(store.modules.approvals.actions.fetchRules).toHaveBeenCalled();
}); });
});
it('disconnects MutationObserver when component gets destroyed', () => { it('disconnects MutationObserver when component gets destroyed', () => {
const mockDisconnect = jest.fn(); const mockDisconnect = jest.fn();
......
...@@ -35,4 +35,28 @@ describe('EE approvals base module mutations', () => { ...@@ -35,4 +35,28 @@ describe('EE approvals base module mutations', () => {
expect(state).toEqual(expect.objectContaining(settings)); expect(state).toEqual(expect.objectContaining(settings));
}); });
}); });
describe(types.SET_RESET_TO_DEFAULT, () => {
it('resets rules', () => {
state.rules = ['test'];
mutations[types.SET_RESET_TO_DEFAULT](state, true);
expect(state.resetToDefault).toBe(true);
expect(state.oldRules).toEqual(['test']);
});
});
describe(types.UNDO_RULES, () => {
it('undos rules', () => {
const oldRules = ['old'];
state.rules = ['new'];
state.oldRules = oldRules;
mutations[types.UNDO_RULES](state, true);
expect(state.resetToDefault).toBe(false);
expect(state.rules).toEqual(oldRules);
});
});
}); });
import testAction from 'helpers/vuex_action_helper';
import * as types from 'ee/approvals/stores/modules/mr_edit/mutation_types';
import * as actions from 'ee/approvals/stores/modules/mr_edit/actions';
describe('Approval MR edit module actions', () => {
describe('setTargetBranch', () => {
it('commits SET_TARGET_BRANCH', done => {
testAction(
actions.setTargetBranch,
'master',
{},
[{ type: types.SET_TARGET_BRANCH, payload: 'master' }],
[],
done,
);
});
});
describe('undoRulesChange', () => {
it('commits UNDO_RULES', done => {
testAction(actions.undoRulesChange, null, {}, [{ type: types.UNDO_RULES }], [], done);
});
});
});
...@@ -2626,6 +2626,9 @@ msgstr "" ...@@ -2626,6 +2626,9 @@ msgstr ""
msgid "Approval rules" msgid "Approval rules"
msgstr "" msgstr ""
msgid "Approval rules reset to project defaults"
msgstr ""
msgid "ApprovalRuleRemove|%d member" msgid "ApprovalRuleRemove|%d member"
msgid_plural "ApprovalRuleRemove|%d members" msgid_plural "ApprovalRuleRemove|%d members"
msgstr[0] "" msgstr[0] ""
...@@ -18803,6 +18806,9 @@ msgstr "" ...@@ -18803,6 +18806,9 @@ msgstr ""
msgid "Reset template" msgid "Reset template"
msgstr "" msgstr ""
msgid "Reset to project defaults"
msgstr ""
msgid "Resetting the authorization key for this project will require updating the authorization key in every alert source it is enabled in." msgid "Resetting the authorization key for this project will require updating the authorization key in every alert source it is enabled in."
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