Fix access removal of added item
Previously, when removing an added item (of any type) nested inside of another added type, the removal has no effect. This is because, in the case of an add, there is no corresponding remove object, as there is in the case of a modified section. Example: Modified access rules with a new ref and rule-2 was removed before saving. { ..., refs/for/new: { added: true, permissions: { permission-1: { added: true, rules { rule-1: { added: true, action: 'ALLOW" }, rule-2: { added: true, removed: true, action: 'ALLOW" } } } } } } This change fixes the problem by actually removing the item completely if it was added in the first place, as opposed to keeping a placeholder that can be undone, which makes sense as both ways allow restoration of the initial state of the access object. The new access JSON for this example is as follows: There is no reason for rule-2 to be in here at all because the server does not have to remove it (it did not exist before). { ..., refs/for/new: { added: true, permissions: { permission-1: { added: true, rules { rule-1: { added: true, action: 'ALLOW" }, } } } } } Bug: Issue 8795 Change-Id: Ief7ebd18f0ad207358b68f504ffe2b0b322340f9
This commit is contained in:
parent
712f0ea28e
commit
7c57cf9104
@ -122,7 +122,8 @@ limitations under the License.
|
|||||||
labels="[[labels]]"
|
labels="[[labels]]"
|
||||||
section="[[section.id]]"
|
section="[[section.id]]"
|
||||||
editing="[[editing]]"
|
editing="[[editing]]"
|
||||||
groups="[[groups]]">
|
groups="[[groups]]"
|
||||||
|
on-added-permission-removed="_handleAddedPermissionRemoved">
|
||||||
</gr-permission>
|
</gr-permission>
|
||||||
</template>
|
</template>
|
||||||
<div id="addPermission">
|
<div id="addPermission">
|
||||||
|
@ -23,6 +23,11 @@
|
|||||||
* @event access-modified
|
* @event access-modified
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fired when a section that was previously added was removed.
|
||||||
|
* @event added-section-removed
|
||||||
|
*/
|
||||||
|
|
||||||
const GLOBAL_NAME = 'GLOBAL_CAPABILITIES';
|
const GLOBAL_NAME = 'GLOBAL_CAPABILITIES';
|
||||||
|
|
||||||
// The name that gets automatically input when a new reference is added.
|
// The name that gets automatically input when a new reference is added.
|
||||||
@ -130,6 +135,12 @@
|
|||||||
return section.id === 'GLOBAL_CAPABILITIES' ? 'hide' : '';
|
return section.id === 'GLOBAL_CAPABILITIES' ? 'hide' : '';
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_handleAddedPermissionRemoved(e) {
|
||||||
|
const index = e.model.index;
|
||||||
|
this._permissions = this._permissions.slice(0, index).concat(
|
||||||
|
this._permissions.slice(index + 1, this._permissions.length));
|
||||||
|
},
|
||||||
|
|
||||||
_computeLabelOptions(labels) {
|
_computeLabelOptions(labels) {
|
||||||
const labelOptions = [];
|
const labelOptions = [];
|
||||||
for (const labelName of Object.keys(labels)) {
|
for (const labelName of Object.keys(labels)) {
|
||||||
@ -184,6 +195,10 @@
|
|||||||
},
|
},
|
||||||
|
|
||||||
_handleRemoveReference() {
|
_handleRemoveReference() {
|
||||||
|
if (this.section.value.added) {
|
||||||
|
this.dispatchEvent(new CustomEvent('added-section-removed',
|
||||||
|
{bubbles: true}));
|
||||||
|
}
|
||||||
this._deleted = true;
|
this._deleted = true;
|
||||||
this.section.value.deleted = true;
|
this.section.value.deleted = true;
|
||||||
this.dispatchEvent(new CustomEvent('access-modified', {bubbles: true}));
|
this.dispatchEvent(new CustomEvent('access-modified', {bubbles: true}));
|
||||||
|
@ -507,6 +507,23 @@ limitations under the License.
|
|||||||
assert.isFalse(element._deleted);
|
assert.isFalse(element._deleted);
|
||||||
assert.isNotOk(element.section.value.deleted);
|
assert.isNotOk(element.section.value.deleted);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('removing an added permission', () => {
|
||||||
|
element.editing = true;
|
||||||
|
assert.equal(element._permissions.length, 1);
|
||||||
|
element.$$('gr-permission').fire('added-permission-removed');
|
||||||
|
flushAsynchronousOperations();
|
||||||
|
assert.equal(element._permissions.length, 0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('remove an added section', () => {
|
||||||
|
const removeStub = sandbox.stub();
|
||||||
|
element.addEventListener('added-section-removed', removeStub);
|
||||||
|
element.editing = true;
|
||||||
|
element.section.value.added = true;
|
||||||
|
MockInteractions.tap(element.$.deleteBtn);
|
||||||
|
assert.isTrue(removeStub.called);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -115,7 +115,8 @@ limitations under the License.
|
|||||||
group-name="[[_computeGroupName(groups, rule.id)]]"
|
group-name="[[_computeGroupName(groups, rule.id)]]"
|
||||||
permission="[[permission.id]]"
|
permission="[[permission.id]]"
|
||||||
rule="{{rule}}"
|
rule="{{rule}}"
|
||||||
section="[[section]]"></gr-rule-editor>
|
section="[[section]]"
|
||||||
|
on-added-rule-removed="_handleAddedRuleRemoved"></gr-rule-editor>
|
||||||
</template>
|
</template>
|
||||||
<div id="addRule">
|
<div id="addRule">
|
||||||
<gr-autocomplete
|
<gr-autocomplete
|
||||||
|
@ -25,6 +25,11 @@
|
|||||||
* @event access-modified
|
* @event access-modified
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fired when a permission that was previously added was removed.
|
||||||
|
* @event added-permission-removed
|
||||||
|
*/
|
||||||
|
|
||||||
Polymer({
|
Polymer({
|
||||||
is: 'gr-permission',
|
is: 'gr-permission',
|
||||||
|
|
||||||
@ -117,6 +122,12 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_handleAddedRuleRemoved(e) {
|
||||||
|
const index = e.model.index;
|
||||||
|
this._rules = this._rules.slice(0, index)
|
||||||
|
.concat(this._rules.slice(index + 1, this._rules.length));
|
||||||
|
},
|
||||||
|
|
||||||
_handleValueChange() {
|
_handleValueChange() {
|
||||||
this.permission.value.modified = true;
|
this.permission.value.modified = true;
|
||||||
// Allows overall access page to know a change has been made.
|
// Allows overall access page to know a change has been made.
|
||||||
@ -124,6 +135,10 @@
|
|||||||
},
|
},
|
||||||
|
|
||||||
_handleRemovePermission() {
|
_handleRemovePermission() {
|
||||||
|
if (this.permission.value.added) {
|
||||||
|
this.dispatchEvent(new CustomEvent('added-permission-removed',
|
||||||
|
{bubbles: true}));
|
||||||
|
}
|
||||||
this._deleted = true;
|
this._deleted = true;
|
||||||
this.permission.value.deleted = true;
|
this.permission.value.deleted = true;
|
||||||
this.dispatchEvent(new CustomEvent('access-modified', {bubbles: true}));
|
this.dispatchEvent(new CustomEvent('access-modified', {bubbles: true}));
|
||||||
|
@ -327,11 +327,36 @@ limitations under the License.
|
|||||||
assert.equal(Object.keys(element.permission.value.rules).length, 2);
|
assert.equal(Object.keys(element.permission.value.rules).length, 2);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('removing an added rule', () => {
|
||||||
|
element.name = 'Priority';
|
||||||
|
element.section = 'refs/*';
|
||||||
|
element.groups = {};
|
||||||
|
element.$.groupAutocomplete.text = 'new group name';
|
||||||
|
assert.equal(element._rules.length, 2);
|
||||||
|
element.$$('gr-rule-editor').fire('added-rule-removed');
|
||||||
|
flushAsynchronousOperations();
|
||||||
|
assert.equal(element._rules.length, 1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('removing an added permission', () => {
|
||||||
|
const removeStub = sandbox.stub();
|
||||||
|
element.addEventListener('added-permission-removed', removeStub);
|
||||||
|
element.editing = true;
|
||||||
|
element.name = 'Priority';
|
||||||
|
element.section = 'refs/*';
|
||||||
|
element.permission.value.added = true;
|
||||||
|
MockInteractions.tap(element.$.removeBtn);
|
||||||
|
assert.isTrue(removeStub.called);
|
||||||
|
});
|
||||||
|
|
||||||
test('removing the permission', () => {
|
test('removing the permission', () => {
|
||||||
element.editing = true;
|
element.editing = true;
|
||||||
element.name = 'Priority';
|
element.name = 'Priority';
|
||||||
element.section = 'refs/*';
|
element.section = 'refs/*';
|
||||||
|
|
||||||
|
const removeStub = sandbox.stub();
|
||||||
|
element.addEventListener('added-permission-removed', removeStub);
|
||||||
|
|
||||||
assert.isFalse(element.$.permission.classList.contains('deleted'));
|
assert.isFalse(element.$.permission.classList.contains('deleted'));
|
||||||
assert.isFalse(element._deleted);
|
assert.isFalse(element._deleted);
|
||||||
MockInteractions.tap(element.$.removeBtn);
|
MockInteractions.tap(element.$.removeBtn);
|
||||||
@ -340,6 +365,7 @@ limitations under the License.
|
|||||||
MockInteractions.tap(element.$.undoRemoveBtn);
|
MockInteractions.tap(element.$.undoRemoveBtn);
|
||||||
assert.isFalse(element.$.permission.classList.contains('deleted'));
|
assert.isFalse(element.$.permission.classList.contains('deleted'));
|
||||||
assert.isFalse(element._deleted);
|
assert.isFalse(element._deleted);
|
||||||
|
assert.isFalse(removeStub.called);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('modify a permission', () => {
|
test('modify a permission', () => {
|
||||||
|
@ -111,7 +111,8 @@ limitations under the License.
|
|||||||
section="{{section}}"
|
section="{{section}}"
|
||||||
labels="[[_labels]]"
|
labels="[[_labels]]"
|
||||||
editing="[[_editing]]"
|
editing="[[_editing]]"
|
||||||
groups="[[_groups]]"></gr-access-section>
|
groups="[[_groups]]"
|
||||||
|
on-added-section-removed="_handleAddedSectionRemoved"></gr-access-section>
|
||||||
</template>
|
</template>
|
||||||
<div class="referenceContainer">
|
<div class="referenceContainer">
|
||||||
<gr-button id="addReferenceBtn"
|
<gr-button id="addReferenceBtn"
|
||||||
|
@ -243,6 +243,12 @@
|
|||||||
return inheritsFrom ? 'show' : '';
|
return inheritsFrom ? 'show' : '';
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_handleAddedSectionRemoved(e) {
|
||||||
|
const index = e.model.index;
|
||||||
|
this._sections = this._sections.slice(0, index)
|
||||||
|
.concat(this._sections.slice(index + 1, this._sections.length));
|
||||||
|
},
|
||||||
|
|
||||||
_handleEditingChanged(editing, editingOld) {
|
_handleEditingChanged(editing, editingOld) {
|
||||||
// Ignore when editing gets set initially.
|
// Ignore when editing gets set initially.
|
||||||
if (!editingOld || editing) { return; }
|
if (!editingOld || editing) { return; }
|
||||||
|
@ -301,6 +301,14 @@ limitations under the License.
|
|||||||
flushAsynchronousOperations();
|
flushAsynchronousOperations();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('removing an added section', () => {
|
||||||
|
element.editing = true;
|
||||||
|
assert.equal(element._sections.length, 1);
|
||||||
|
element.$$('gr-access-section').fire('added-section-removed');
|
||||||
|
flushAsynchronousOperations();
|
||||||
|
assert.equal(element._sections.length, 0);
|
||||||
|
});
|
||||||
|
|
||||||
test('button visibility for non admin', () => {
|
test('button visibility for non admin', () => {
|
||||||
assert.equal(getComputedStyle(element.$.saveBtn).display, 'none');
|
assert.equal(getComputedStyle(element.$.saveBtn).display, 'none');
|
||||||
assert.equal(getComputedStyle(element.$.editBtn).display, 'none');
|
assert.equal(getComputedStyle(element.$.editBtn).display, 'none');
|
||||||
|
@ -23,6 +23,11 @@
|
|||||||
* @event access-modified
|
* @event access-modified
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fired when a rule that was previously added was removed.
|
||||||
|
* @event added-rule-removed
|
||||||
|
*/
|
||||||
|
|
||||||
const PRIORITY_OPTIONS = [
|
const PRIORITY_OPTIONS = [
|
||||||
'BATCH',
|
'BATCH',
|
||||||
'INTERACTIVE',
|
'INTERACTIVE',
|
||||||
@ -202,6 +207,10 @@
|
|||||||
},
|
},
|
||||||
|
|
||||||
_handleRemoveRule() {
|
_handleRemoveRule() {
|
||||||
|
if (this.rule.value.added) {
|
||||||
|
this.dispatchEvent(new CustomEvent('added-rule-removed',
|
||||||
|
{bubbles: true}));
|
||||||
|
}
|
||||||
this._deleted = true;
|
this._deleted = true;
|
||||||
this.rule.value.deleted = true;
|
this.rule.value.deleted = true;
|
||||||
this.dispatchEvent(new CustomEvent('access-modified', {bubbles: true}));
|
this.dispatchEvent(new CustomEvent('access-modified', {bubbles: true}));
|
||||||
|
@ -319,6 +319,7 @@ limitations under the License.
|
|||||||
element.section = 'refs/*';
|
element.section = 'refs/*';
|
||||||
element._setupValues(element.rule);
|
element._setupValues(element.rule);
|
||||||
flushAsynchronousOperations();
|
flushAsynchronousOperations();
|
||||||
|
element.rule.value.added = true;
|
||||||
});
|
});
|
||||||
|
|
||||||
test('_ruleValues and _originalRuleValues are set correctly', () => {
|
test('_ruleValues and _originalRuleValues are set correctly', () => {
|
||||||
@ -328,9 +329,9 @@ limitations under the License.
|
|||||||
const expectedRuleValue = {
|
const expectedRuleValue = {
|
||||||
action: 'ALLOW',
|
action: 'ALLOW',
|
||||||
force: false,
|
force: false,
|
||||||
|
added: true,
|
||||||
};
|
};
|
||||||
assert.deepEqual(element.rule.value, expectedRuleValue);
|
assert.deepEqual(element.rule.value, expectedRuleValue);
|
||||||
assert.deepEqual(element._originalRuleValues, expectedRuleValue);
|
|
||||||
test('values are set correctly', () => {
|
test('values are set correctly', () => {
|
||||||
assert.equal(element.$.action.bindValue, expectedRuleValue.action);
|
assert.equal(element.$.action.bindValue, expectedRuleValue.action);
|
||||||
assert.equal(element.$.force.bindValue, expectedRuleValue.action);
|
assert.equal(element.$.force.bindValue, expectedRuleValue.action);
|
||||||
@ -346,6 +347,15 @@ limitations under the License.
|
|||||||
// The original value should now differ from the rule values.
|
// The original value should now differ from the rule values.
|
||||||
assert.notDeepEqual(element._originalRuleValues, element.rule.value);
|
assert.notDeepEqual(element._originalRuleValues, element.rule.value);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('remove value', () => {
|
||||||
|
element.editing = true;
|
||||||
|
const removeStub = sandbox.stub();
|
||||||
|
element.addEventListener('added-rule-removed', removeStub);
|
||||||
|
MockInteractions.tap(element.$.removeBtn);
|
||||||
|
flushAsynchronousOperations();
|
||||||
|
assert.isTrue(removeStub.called);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
suite('already existing rule with labels', () => {
|
suite('already existing rule with labels', () => {
|
||||||
@ -389,10 +399,13 @@ limitations under the License.
|
|||||||
});
|
});
|
||||||
|
|
||||||
test('modify value', () => {
|
test('modify value', () => {
|
||||||
|
const removeStub = sandbox.stub();
|
||||||
|
element.addEventListener('added-rule-removed', removeStub);
|
||||||
assert.isNotOk(element.rule.value.modified);
|
assert.isNotOk(element.rule.value.modified);
|
||||||
Polymer.dom(element.root).querySelector('#labelMin').bindValue = 1;
|
Polymer.dom(element.root).querySelector('#labelMin').bindValue = 1;
|
||||||
flushAsynchronousOperations();
|
flushAsynchronousOperations();
|
||||||
assert.isTrue(element.rule.value.modified);
|
assert.isTrue(element.rule.value.modified);
|
||||||
|
assert.isFalse(removeStub.called);
|
||||||
|
|
||||||
// The original value should now differ from the rule values.
|
// The original value should now differ from the rule values.
|
||||||
assert.notDeepEqual(element._originalRuleValues, element.rule.value);
|
assert.notDeepEqual(element._originalRuleValues, element.rule.value);
|
||||||
@ -417,6 +430,7 @@ limitations under the License.
|
|||||||
element.section = 'refs/*';
|
element.section = 'refs/*';
|
||||||
element._setupValues(element.rule);
|
element._setupValues(element.rule);
|
||||||
flushAsynchronousOperations();
|
flushAsynchronousOperations();
|
||||||
|
element.rule.value.added = true;
|
||||||
});
|
});
|
||||||
|
|
||||||
test('_ruleValues and _originalRuleValues are set correctly', () => {
|
test('_ruleValues and _originalRuleValues are set correctly', () => {
|
||||||
@ -429,9 +443,9 @@ limitations under the License.
|
|||||||
max: element.label.values[element.label.values.length - 1].value,
|
max: element.label.values[element.label.values.length - 1].value,
|
||||||
min: element.label.values[0].value,
|
min: element.label.values[0].value,
|
||||||
action: 'ALLOW',
|
action: 'ALLOW',
|
||||||
|
added: true,
|
||||||
};
|
};
|
||||||
assert.deepEqual(element.rule.value, expectedRuleValue);
|
assert.deepEqual(element.rule.value, expectedRuleValue);
|
||||||
assert.deepEqual(element._originalRuleValues, expectedRuleValue);
|
|
||||||
test('values are set correctly', () => {
|
test('values are set correctly', () => {
|
||||||
assert.equal(
|
assert.equal(
|
||||||
element.$.action.bindValue,
|
element.$.action.bindValue,
|
||||||
@ -507,6 +521,7 @@ limitations under the License.
|
|||||||
element.section = 'refs/*';
|
element.section = 'refs/*';
|
||||||
element._setupValues(element.rule);
|
element._setupValues(element.rule);
|
||||||
flushAsynchronousOperations();
|
flushAsynchronousOperations();
|
||||||
|
element.rule.value.added = true;
|
||||||
});
|
});
|
||||||
|
|
||||||
test('_ruleValues and _originalRuleValues are set correctly', () => {
|
test('_ruleValues and _originalRuleValues are set correctly', () => {
|
||||||
@ -516,9 +531,9 @@ limitations under the License.
|
|||||||
const expectedRuleValue = {
|
const expectedRuleValue = {
|
||||||
action: 'ALLOW',
|
action: 'ALLOW',
|
||||||
force: false,
|
force: false,
|
||||||
|
added: true,
|
||||||
};
|
};
|
||||||
assert.deepEqual(element.rule.value, expectedRuleValue);
|
assert.deepEqual(element.rule.value, expectedRuleValue);
|
||||||
assert.deepEqual(element._originalRuleValues, expectedRuleValue);
|
|
||||||
test('values are set correctly', () => {
|
test('values are set correctly', () => {
|
||||||
assert.equal(element.$.action.bindValue, expectedRuleValue.action);
|
assert.equal(element.$.action.bindValue, expectedRuleValue.action);
|
||||||
assert.equal(element.$.force.bindValue, expectedRuleValue.action);
|
assert.equal(element.$.force.bindValue, expectedRuleValue.action);
|
||||||
@ -576,44 +591,5 @@ limitations under the License.
|
|||||||
assert.notDeepEqual(element._originalRuleValues, element.rule.value);
|
assert.notDeepEqual(element._originalRuleValues, element.rule.value);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
suite('new edit rule', () => {
|
|
||||||
setup(() => {
|
|
||||||
element.group = 'Group Name';
|
|
||||||
element.permission = 'editTopicName';
|
|
||||||
element.rule = {
|
|
||||||
id: '123',
|
|
||||||
};
|
|
||||||
element.section = 'refs/*';
|
|
||||||
element._setupValues(element.rule);
|
|
||||||
flushAsynchronousOperations();
|
|
||||||
});
|
|
||||||
|
|
||||||
test('_ruleValues and _originalRuleValues are set correctly', () => {
|
|
||||||
// Since the element does not already have default values, they should
|
|
||||||
// be set. The original values should be set to those too.
|
|
||||||
assert.isNotOk(element.rule.value.modified);
|
|
||||||
const expectedRuleValue = {
|
|
||||||
action: 'ALLOW',
|
|
||||||
force: false,
|
|
||||||
};
|
|
||||||
assert.deepEqual(element.rule.value, expectedRuleValue);
|
|
||||||
assert.deepEqual(element._originalRuleValues, expectedRuleValue);
|
|
||||||
test('values are set correctly', () => {
|
|
||||||
assert.equal(element.$.action.bindValue, expectedRuleValue.action);
|
|
||||||
assert.equal(element.$.force.bindValue, expectedRuleValue.action);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
test('modify value', () => {
|
|
||||||
assert.isNotOk(element.rule.value.modified);
|
|
||||||
element.$.force.bindValue = true;
|
|
||||||
flushAsynchronousOperations();
|
|
||||||
assert.isTrue(element.rule.value.modified);
|
|
||||||
|
|
||||||
// The original value should now differ from the rule values.
|
|
||||||
assert.notDeepEqual(element._originalRuleValues, element.rule.value);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user