open-vault/ui/app/components/secret-edit.js

355 lines
9.3 KiB
JavaScript
Raw Normal View History

import { or } from '@ember/object/computed';
import { isBlank, isNone } from '@ember/utils';
import { inject as service } from '@ember/service';
import Component from '@ember/component';
2018-10-17 04:23:29 +00:00
import { computed, set } from '@ember/object';
import { alias } from '@ember/object/computed';
import { task, waitForEvent } from 'ember-concurrency';
2018-04-03 14:16:57 +00:00
import FocusOnInsertMixin from 'vault/mixins/focus-on-insert';
import keys from 'vault/lib/keycodes';
import KVObject from 'vault/lib/kv-object';
2018-10-16 21:08:31 +00:00
import { maybeQueryRecord } from 'vault/macros/maybe-query-record';
2018-04-03 14:16:57 +00:00
const LIST_ROUTE = 'vault.cluster.secrets.backend.list';
const LIST_ROOT_ROUTE = 'vault.cluster.secrets.backend.list-root';
const SHOW_ROUTE = 'vault.cluster.secrets.backend.show';
export default Component.extend(FocusOnInsertMixin, {
wizard: service(),
router: service(),
store: service(),
2018-04-03 14:16:57 +00:00
// a key model
key: null,
model: null,
2018-04-03 14:16:57 +00:00
// a value to pre-fill the key input - this is populated by the corresponding
// 'initialKey' queryParam
initialKey: null,
// set in the route's setupController hook
mode: null,
secretData: null,
// called with a bool indicating if there's been a change in the secretData
onDataChange() {},
onRefresh() {},
onToggleAdvancedEdit() {},
2018-04-03 14:16:57 +00:00
// did user request advanced mode
preferAdvancedEdit: false,
// use a named action here so we don't have to pass one in
// this will bubble to the route
toggleAdvancedEdit: 'toggleAdvancedEdit',
error: null,
2018-04-03 14:16:57 +00:00
codemirrorString: null,
hasLintError: false,
isV2: false,
2018-04-03 14:16:57 +00:00
init() {
this._super(...arguments);
let secrets = this.model.secretData;
if (!secrets && this.model.selectedVersion) {
this.set('isV2', true);
secrets = this.model.belongsTo('selectedVersion').value().secretData;
}
2018-04-03 14:16:57 +00:00
const data = KVObject.create({ content: [] }).fromJSON(secrets);
this.set('secretData', data);
this.set('codemirrorString', data.toJSONString());
if (data.isAdvanced()) {
this.set('preferAdvancedEdit', true);
}
this.checkRows();
2018-10-08 18:20:55 +00:00
if (this.wizard.featureState === 'details' && this.mode === 'create') {
let engine = this.model.backend.includes('kv') ? 'kv' : this.model.backend;
this.wizard.transitionFeatureMachine('details', 'CONTINUE', engine);
2018-08-28 05:03:55 +00:00
}
if (this.mode === 'edit') {
2018-04-03 14:16:57 +00:00
this.send('addRow');
}
},
waitForKeyUp: task(function*() {
while (true) {
let event = yield waitForEvent(document.body, 'keyup');
this.onEscape(event);
}
})
.on('didInsertElement')
.cancelOn('willDestroyElement'),
partialName: computed('mode', function() {
2018-10-08 18:20:55 +00:00
return `partials/secret-form-${this.mode}`;
2018-04-03 14:16:57 +00:00
}),
2018-10-16 21:08:31 +00:00
updatePath: maybeQueryRecord(
'capabilities',
context => {
if (context.mode === 'create') {
2018-10-16 21:08:31 +00:00
return;
}
2018-10-18 18:03:05 +00:00
let backend = context.isV2 ? context.get('model.engine.id') : context.model.backend;
let id = context.model.id;
let path = context.isV2 ? `${backend}/data/${id}` : `${backend}/${id}`;
return {
id: path,
};
},
'isV2',
'model',
'model.id',
'mode'
),
canDelete: alias('updatePath.canDelete'),
canEdit: alias('updatePath.canUpdate'),
2018-10-16 21:08:31 +00:00
v2UpdatePath: maybeQueryRecord(
'capabilities',
context => {
if (context.mode === 'create' || context.isV2 === false) {
2018-10-16 21:08:31 +00:00
return;
}
2018-10-18 18:03:05 +00:00
let backend = context.get('model.engine.id');
let id = context.model.id;
return {
id: `${backend}/metadata/${id}`,
};
},
'isV2',
'model',
'model.id',
'mode'
),
2018-10-16 21:08:31 +00:00
canEditV2Secret: alias('v2UpdatePath.canUpdate'),
2018-10-18 18:03:05 +00:00
deleteVersionPath: maybeQueryRecord(
'capabilities',
context => {
let backend = context.get('model.engine.id');
let id = context.model.id;
return {
id: `${backend}/delete/${id}`,
};
},
'model.id'
),
canDeleteVersion: alias('deleteVersionPath.canUpdate'),
destroyVersionPath: maybeQueryRecord(
'capabilities',
context => {
let backend = context.get('model.engine.id');
let id = context.model.id;
return {
id: `${backend}/destroy/${id}`,
};
},
'model.id'
),
canDestroyVersion: alias('destroyVersionPath.canUpdate'),
undeleteVersionPath: maybeQueryRecord(
'capabilities',
context => {
let backend = context.get('model.engine.id');
let id = context.model.id;
return {
id: `${backend}/undelete/${id}`,
};
},
'model.id'
),
canUndeleteVersion: alias('undeleteVersionPath.canUpdate'),
isFetchingVersionCapabilities: or(
2018-10-18 20:11:26 +00:00
'deleteVersionPath.isPending',
'destroyVersionPath.isPending',
'undeleteVersionPath.isPending'
2018-10-18 18:03:05 +00:00
),
requestInFlight: or('model.isLoading', 'model.isReloading', 'model.isSaving'),
2018-04-03 14:16:57 +00:00
buttonDisabled: or(
2018-04-03 14:16:57 +00:00
'requestInFlight',
'model.isFolder',
'model.isError',
'model.flagsIsInvalid',
'hasLintError',
'error'
2018-04-03 14:16:57 +00:00
),
modelForData: computed('isV2', 'model', function() {
return this.isV2 ? this.model.belongsTo('selectedVersion').value() : this.model;
}),
2018-04-03 14:16:57 +00:00
basicModeDisabled: computed('secretDataIsAdvanced', 'showAdvancedMode', function() {
return this.secretDataIsAdvanced || this.showAdvancedMode === false;
2018-04-03 14:16:57 +00:00
}),
secretDataAsJSON: computed('secretData', 'secretData.[]', function() {
return this.secretData.toJSON();
2018-04-03 14:16:57 +00:00
}),
secretDataIsAdvanced: computed('secretData', 'secretData.[]', function() {
return this.secretData.isAdvanced();
2018-04-03 14:16:57 +00:00
}),
showAdvancedMode: computed('preferAdvancedEdit', 'secretDataIsAdvanced', 'lastChange', function() {
return this.secretDataIsAdvanced || this.preferAdvancedEdit;
2018-04-03 14:16:57 +00:00
}),
transitionToRoute() {
this.router.transitionTo(...arguments);
2018-04-03 14:16:57 +00:00
},
onEscape(e) {
if (e.keyCode !== keys.ESC || this.mode !== 'show') {
2018-04-03 14:16:57 +00:00
return;
}
const parentKey = this.model.parentKey;
2018-04-03 14:16:57 +00:00
if (parentKey) {
this.transitionToRoute(LIST_ROUTE, parentKey);
} else {
this.transitionToRoute(LIST_ROOT_ROUTE);
}
},
// successCallback is called in the context of the component
persistKey(successCallback) {
let secret = this.model;
let model = this.modelForData;
let isV2 = this.isV2;
let key = model.get('path') || model.id;
if (key.startsWith('/')) {
key = key.replace(/^\/+/g, '');
model.set(model.pathAttr, key);
2018-04-03 14:16:57 +00:00
}
return model.save().then(() => {
if (!model.isError) {
if (isV2 && Object.keys(secret.changedAttributes()).length) {
secret.set('id', key);
// save secret metadata
secret
.save()
.then(() => {
this.saveComplete(successCallback, key);
})
.catch(e => {
this.set(e, e.errors.join(' '));
});
} else {
this.saveComplete(successCallback, key);
2018-08-28 05:03:55 +00:00
}
2018-04-03 14:16:57 +00:00
}
});
},
saveComplete(callback, key) {
if (this.wizard.featureState === 'secret') {
this.wizard.transitionFeatureMachine('secret', 'CONTINUE');
}
callback(key);
},
2018-04-03 14:16:57 +00:00
checkRows() {
if (this.secretData.length === 0) {
2018-04-03 14:16:57 +00:00
this.send('addRow');
}
},
actions: {
//submit on shift + enter
handleKeyDown(e) {
2018-04-03 14:16:57 +00:00
e.stopPropagation();
if (!(e.keyCode === keys.ENTER && e.metaKey)) {
return;
}
let $form = this.element.querySelector('form');
2018-04-03 14:16:57 +00:00
if ($form.length) {
$form.submit();
}
},
handleChange() {
this.set('codemirrorString', this.secretData.toJSONString(true));
2018-10-17 04:23:29 +00:00
set(this.modelForData, 'secretData', this.secretData.toJSON());
2018-04-03 14:16:57 +00:00
},
createOrUpdateKey(type, event) {
event.preventDefault();
2018-10-16 21:08:31 +00:00
let model = this.modelForData;
2018-04-03 14:16:57 +00:00
// prevent from submitting if there's no key
// maybe do something fancier later
if (type === 'create' && isBlank(model.get('path') || model.id)) {
2018-04-03 14:16:57 +00:00
return;
}
this.persistKey(key => {
this.transitionToRoute(SHOW_ROUTE, key);
});
2018-04-03 14:16:57 +00:00
},
deleteKey() {
this.model.destroyRecord().then(() => {
2018-04-03 14:16:57 +00:00
this.transitionToRoute(LIST_ROOT_ROUTE);
});
},
deleteVersion(deleteType = 'destroy') {
let id = this.modelForData.id;
return this.store.adapterFor('secret-v2-version').v2DeleteOperation(this.store, id, deleteType);
},
2018-04-03 14:16:57 +00:00
refresh() {
this.onRefresh();
2018-04-03 14:16:57 +00:00
},
addRow() {
const data = this.secretData;
if (isNone(data.findBy('name', ''))) {
2018-04-03 14:16:57 +00:00
data.pushObject({ name: '', value: '' });
this.send('handleChange');
2018-04-03 14:16:57 +00:00
}
this.checkRows();
},
deleteRow(name) {
const data = this.secretData;
2018-04-03 14:16:57 +00:00
const item = data.findBy('name', name);
if (isBlank(item.name)) {
2018-04-03 14:16:57 +00:00
return;
}
data.removeObject(item);
this.checkRows();
this.send('handleChange');
2018-04-03 14:16:57 +00:00
},
toggleAdvanced(bool) {
this.onToggleAdvancedEdit(bool);
2018-04-03 14:16:57 +00:00
},
codemirrorUpdated(val, codemirror) {
this.set('error', null);
2018-04-03 14:16:57 +00:00
codemirror.performLint();
const noErrors = codemirror.state.lint.marked.length === 0;
if (noErrors) {
try {
this.secretData.fromJSONString(val);
} catch (e) {
this.set('error', e.message);
}
2018-04-03 14:16:57 +00:00
}
this.set('hasLintError', !noErrors);
this.set('codemirrorString', val);
},
formatJSON() {
this.set('codemirrorString', this.secretData.toJSONString(true));
2018-04-03 14:16:57 +00:00
},
},
});