4d9fa38a5e
* ui: add parameterized dispatch interface This commit adds a new interface for dispatching parameteried jobs, if the user has the right permissions. The UI can be accessed by viewing a parameterized job and clicking on the "Dispatch Job" button located in the "Job Launches" section. * fix failing lint test * clean up dispatch and remove meta This commit cleans up a few things that had typos and inconsistent naming. In line with this, the custom `meta` view was removed in favor of using the included `AttributesTable`. * ui: encode dispatch job payload and start adding tests * ui: remove unused test imports * ui: redesign job dispatch form * ui: initial acceptance tests for dispatch job * ui: generate parameterized job children with correct id format * ui: fix job dispatch breadcrumb link * ui: refactor job dispatch component into glimmer component and add form validation * ui: remove unused CSS class * ui: align job dispatch button * ui: handle namespace-specific requests on job dispatch * ui: rename payloadMissing to payloadHasError * ui: don't re-fetch job spec on dispatch job * ui: keep overview tab selected on job dispatch page * ui: fix task and task-group linting * ui: URL encode job id on dispatch job tests * ui: fix error when job meta is null * ui: handle job dispatch from adapter * ui: add more tests for dispatch job page * ui: add "job dispatch" capability check * ui: update job dispatch from code review Co-authored-by: Luiz Aoqui <luiz@hashicorp.com>
138 lines
3.2 KiB
JavaScript
138 lines
3.2 KiB
JavaScript
import Component from '@glimmer/component';
|
|
import { tracked } from '@glimmer/tracking';
|
|
import { inject as service } from '@ember/service';
|
|
import { action } from '@ember/object';
|
|
import { A } from '@ember/array';
|
|
import { task } from 'ember-concurrency';
|
|
import { noCase } from 'no-case';
|
|
import { titleCase } from 'title-case';
|
|
import messageFromAdapterError from 'nomad-ui/utils/message-from-adapter-error';
|
|
|
|
class MetaField {
|
|
@tracked value;
|
|
@tracked error;
|
|
|
|
name;
|
|
required;
|
|
title;
|
|
|
|
constructor(meta) {
|
|
this.name = meta.name;
|
|
this.required = meta.required;
|
|
this.title = meta.title;
|
|
this.value = meta.value;
|
|
this.error = meta.error;
|
|
}
|
|
|
|
validate() {
|
|
this.error = '';
|
|
|
|
if (this.required && !this.value) {
|
|
this.error = `Missing required meta parameter "${this.name}".`;
|
|
}
|
|
}
|
|
}
|
|
|
|
export default class JobDispatch extends Component {
|
|
@service router;
|
|
@service config;
|
|
|
|
@tracked metaFields = [];
|
|
@tracked payload = '';
|
|
@tracked payloadHasError = false;
|
|
|
|
errors = A([]);
|
|
|
|
constructor() {
|
|
super(...arguments);
|
|
|
|
// Helper for mapping the params into a useable form.
|
|
const mapper = (values = [], required) =>
|
|
values.map(
|
|
x =>
|
|
new MetaField({
|
|
name: x,
|
|
required,
|
|
title: titleCase(noCase(x)),
|
|
value: this.args.job.meta ? this.args.job.meta[x] : '',
|
|
})
|
|
);
|
|
|
|
// Fetch the different types of parameters.
|
|
const required = mapper(this.args.job.parameterizedDetails.MetaRequired, true);
|
|
const optional = mapper(this.args.job.parameterizedDetails.MetaOptional, false);
|
|
|
|
// Merge them, required before optional.
|
|
this.metaFields = required.concat(optional);
|
|
}
|
|
|
|
get hasPayload() {
|
|
return this.args.job.parameterizedDetails.Payload !== 'forbidden';
|
|
}
|
|
|
|
get payloadRequired() {
|
|
return this.args.job.parameterizedDetails.Payload === 'required';
|
|
}
|
|
|
|
@action
|
|
dispatch() {
|
|
this.validateForm();
|
|
if (this.errors.length > 0) {
|
|
this.scrollToError();
|
|
return;
|
|
}
|
|
|
|
this.onDispatched.perform();
|
|
}
|
|
|
|
@action
|
|
cancel() {
|
|
this.router.transitionTo('jobs.job');
|
|
}
|
|
|
|
@task({ drop: true }) *onDispatched() {
|
|
// Try to create the dispatch.
|
|
try {
|
|
let paramValues = {};
|
|
this.metaFields.forEach(m => (paramValues[m.name] = m.value));
|
|
const dispatch = yield this.args.job.dispatch(paramValues, this.payload);
|
|
|
|
// Navigate to the newly created instance.
|
|
this.router.transitionTo('jobs.job', dispatch.DispatchedJobID);
|
|
} catch (err) {
|
|
const error = messageFromAdapterError(err) || 'Could not dispatch job';
|
|
this.errors.pushObject(error);
|
|
this.scrollToError();
|
|
}
|
|
}
|
|
|
|
scrollToError() {
|
|
if (!this.config.isTest) {
|
|
window.scrollTo(0, 0);
|
|
}
|
|
}
|
|
|
|
resetErrors() {
|
|
this.payloadHasError = false;
|
|
this.errors.clear();
|
|
}
|
|
|
|
validateForm() {
|
|
this.resetErrors();
|
|
|
|
// Make sure that we have all of the meta fields that we need.
|
|
this.metaFields.forEach(f => {
|
|
f.validate();
|
|
if (f.error) {
|
|
this.errors.pushObject(f.error);
|
|
}
|
|
});
|
|
|
|
// Validate payload.
|
|
if (this.payloadRequired && !this.payload) {
|
|
this.errors.pushObject('Missing required payload.');
|
|
this.payloadHasError = true;
|
|
}
|
|
}
|
|
}
|