Commit 50c0e91c authored by mo khan's avatar mo khan Committed by Mayra Cabrera

Disable editing of License-Check name

* Allow editing during rule creation
* Add Changelog entry
* Updates the backend to prevent the `License-Check` rule name from
being edited.
parent aaa9dc01
...@@ -8,6 +8,8 @@ import ApproversSelect from './approvers_select.vue'; ...@@ -8,6 +8,8 @@ import ApproversSelect from './approvers_select.vue';
import { TYPE_USER, TYPE_GROUP, TYPE_HIDDEN_GROUPS } from '../constants'; import { TYPE_USER, TYPE_GROUP, TYPE_HIDDEN_GROUPS } from '../constants';
const DEFAULT_NAME = 'Default'; const DEFAULT_NAME = 'Default';
const DEFAULT_NAME_FOR_LICENSE_REPORT = 'License-Check';
const READONLY_NAMES = [DEFAULT_NAME_FOR_LICENSE_REPORT];
export default { export default {
components: { components: {
...@@ -103,6 +105,12 @@ export default { ...@@ -103,6 +105,12 @@ export default {
this.settings.allowMultiRule && this.isFallback && !this.name && !this.approvers.length this.settings.allowMultiRule && this.isFallback && !this.name && !this.approvers.length
); );
}, },
isPersisted() {
return this.initRule && this.initRule.id;
},
isNameDisabled() {
return this.isPersisted && READONLY_NAMES.includes(this.name);
},
removeHiddenGroups() { removeHiddenGroups() {
return this.containsHiddenGroups && !this.approversByType[TYPE_HIDDEN_GROUPS]; return this.containsHiddenGroups && !this.approversByType[TYPE_HIDDEN_GROUPS];
}, },
...@@ -226,6 +234,7 @@ export default { ...@@ -226,6 +234,7 @@ export default {
<input <input
v-model="name" v-model="name"
:class="{ 'is-invalid': validation.name }" :class="{ 'is-invalid': validation.name }"
:disabled="isNameDisabled"
class="form-control" class="form-control"
name="name" name="name"
type="text" type="text"
......
...@@ -12,6 +12,7 @@ class ApprovalProjectRule < ApplicationRecord ...@@ -12,6 +12,7 @@ class ApprovalProjectRule < ApplicationRecord
} }
alias_method :code_owner, :code_owner? alias_method :code_owner, :code_owner?
validate :validate_default_license_report_name, on: :update, if: :report_approver?
validates :name, uniqueness: { scope: :project_id } validates :name, uniqueness: { scope: :project_id }
...@@ -46,4 +47,11 @@ class ApprovalProjectRule < ApplicationRecord ...@@ -46,4 +47,11 @@ class ApprovalProjectRule < ApplicationRecord
report_type: report_type report_type: report_type
) )
end end
def validate_default_license_report_name
return unless name_changed?
return unless name_was == ApprovalRuleLike::DEFAULT_NAME_FOR_LICENSE_REPORT
errors.add(:name, _("cannot be modified"))
end
end end
---
title: Disable editing of the 'License-Check' approval rule name
merge_request: 16149
author:
type: added
import { shallowMount, createLocalVue } from '@vue/test-utils';
import Vuex from 'vuex';
import { createStoreOptions } from 'ee/approvals/stores';
import projectSettingsModule from 'ee/approvals/stores/modules/project_settings';
import RuleForm from 'ee/approvals/components/rule_form.vue';
const localVue = createLocalVue();
localVue.use(Vuex);
describe('RuleForm', () => {
const exampleRule = {
id: 1,
name: 'Example',
approvalsRequired: 0,
users: [],
groups: [],
};
let wrapper;
let store;
const findNameInput = () => wrapper.find('input[name=name]');
const createStore = () => createStoreOptions(projectSettingsModule(), { projectId: '7' });
const createComponent = (vuexStore, props = {}) =>
shallowMount(localVue.extend(RuleForm), {
propsData: {
...props,
},
store: new Vuex.Store(vuexStore),
localVue,
sync: false,
});
beforeEach(() => {
store = createStore();
});
afterEach(() => {
if (wrapper) wrapper.destroy();
});
describe('isNameDisabled', () => {
beforeEach(() => {
store.state.settings.allowMultiRule = true;
});
describe('When creating the License-Check rule', () => {
beforeEach(() => {
wrapper = createComponent(store, {
initRule: { ...exampleRule, ...{ id: null, name: 'License-Check' } },
});
});
it('does not disable the name text field', () => {
expect(findNameInput().attributes('disabled')).toBe(undefined);
});
});
describe('When creating any other rule', () => {
beforeEach(() => {
wrapper = createComponent(store, {
initRule: { ...exampleRule, ...{ id: null, name: 'QA' } },
});
});
it('does not disable the name text field', () => {
expect(findNameInput().attributes('disabled')).toBe(undefined);
});
});
describe('When editing the License-Check rule', () => {
beforeEach(() => {
wrapper = createComponent(store, {
initRule: { ...exampleRule, ...{ id: 1, name: 'License-Check' } },
});
});
it('disables the name text field', () => {
expect(findNameInput().attributes('disabled')).toBe('disabled');
});
});
describe('When editing any other rule', () => {
beforeEach(() => {
wrapper = createComponent(store, {
initRule: { ...exampleRule, ...{ id: 1, name: 'QA' } },
});
});
it('does not disable the name text field', () => {
expect(findNameInput().attributes('disabled')).toBe(undefined);
});
});
});
});
...@@ -88,4 +88,39 @@ describe ApprovalProjectRule do ...@@ -88,4 +88,39 @@ describe ApprovalProjectRule do
end end
end end
end end
describe "validation" do
let(:project_approval_rule) { create(:approval_project_rule) }
let(:license_compliance_rule) { create(:approval_project_rule, :license_management) }
let(:vulnerability_check_rule) { create(:approval_project_rule, :security) }
context "when creating a new rule" do
specify { expect(project_approval_rule).to be_valid }
specify { expect(license_compliance_rule).to be_valid }
specify { expect(vulnerability_check_rule).to be_valid }
end
context "when attempting to edit the name of the rule" do
subject { project_approval_rule }
before do
subject.name = SecureRandom.uuid
end
specify { expect(subject).to be_valid }
context "with a `License-Check` rule" do
subject { license_compliance_rule }
specify { expect(subject).not_to be_valid }
specify { expect { subject.valid? }.to change { subject.errors[:name].present? } }
end
context "with a `Vulnerability-Check` rule" do
subject { vulnerability_check_rule }
specify { expect(subject).to be_valid }
end
end
end
end end
...@@ -18148,6 +18148,9 @@ msgstr "" ...@@ -18148,6 +18148,9 @@ msgstr ""
msgid "cannot be in the same project" msgid "cannot be in the same project"
msgstr "" msgstr ""
msgid "cannot be modified"
msgstr ""
msgid "cannot block others" msgid "cannot block others"
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