open-nomad/ui/app/controllers/allocations/allocation/index.js
2022-09-08 13:54:50 -04:00

191 lines
4.8 KiB
JavaScript

/* eslint-disable ember/no-observers */
import Controller from '@ember/controller';
import { inject as service } from '@ember/service';
import { action, computed } from '@ember/object';
import { observes } from '@ember-decorators/object';
import { computed as overridable } from 'ember-overridable-computed';
import { alias } from '@ember/object/computed';
import { task } from 'ember-concurrency';
import Sortable from 'nomad-ui/mixins/sortable';
import { lazyClick } from 'nomad-ui/helpers/lazy-click';
import { watchRecord } from 'nomad-ui/utils/properties/watch';
import messageForError from 'nomad-ui/utils/message-from-adapter-error';
import classic from 'ember-classic-decorator';
import { union } from '@ember/object/computed';
import { tracked } from '@glimmer/tracking';
@classic
export default class IndexController extends Controller.extend(Sortable) {
@service token;
@service store;
queryParams = [
{
sortProperty: 'sort',
},
{
sortDescending: 'desc',
},
{
activeServiceID: 'service',
},
];
sortProperty = 'name';
sortDescending = false;
@alias('model.states') listToSort;
@alias('listSorted') sortedStates;
// Set in the route
preempter = null;
@overridable(function () {
// { title, description }
return null;
})
error;
@computed('model.allocatedResources.ports.@each.label')
get ports() {
return (this.get('model.allocatedResources.ports') || []).sortBy('label');
}
@computed('model.states.@each.task')
get tasks() {
return this.get('model.states').mapBy('task') || [];
}
@computed('tasks.@each.services')
get taskServices() {
return this.get('tasks')
.map((t) => ((t && t.services) || []).toArray())
.flat()
.compact();
}
@computed('model.taskGroup.services.@each.name')
get groupServices() {
return (this.get('model.taskGroup.services') || []).sortBy('name');
}
@union('taskServices', 'groupServices') services;
@computed('model.{healthChecks,id}', 'services')
get servicesWithHealthChecks() {
return this.services.map((service) => {
if (this.model.healthChecks) {
const healthChecks = Object.values(this.model.healthChecks)?.filter(
(check) => {
const refPrefix =
check.Task || check.Group.split('.')[1].split('[')[0];
const currentServiceName = `${refPrefix}-${check.Service}`;
return currentServiceName === service.refID;
}
);
healthChecks.forEach((check) => {
service.healthChecks.pushObject(check);
});
}
// Contextualize healthchecks for the allocation we're in
service.healthChecks = service.healthChecks.filterBy(
'Alloc',
this.model.id
);
return service;
});
}
onDismiss() {
this.set('error', null);
}
@watchRecord('allocation') watchNext;
@observes('model.nextAllocation.clientStatus')
observeWatchNext() {
const nextAllocation = this.model.nextAllocation;
if (nextAllocation && nextAllocation.content) {
this.watchNext.perform(nextAllocation);
} else {
this.watchNext.cancelAll();
}
}
@task(function* () {
try {
yield this.model.stop();
// Eagerly update the allocation clientStatus to avoid flickering
this.model.set('clientStatus', 'complete');
} catch (err) {
this.set('error', {
title: 'Could Not Stop Allocation',
description: messageForError(err, 'manage allocation lifecycle'),
});
}
})
stopAllocation;
@task(function* () {
try {
yield this.model.restart();
} catch (err) {
this.set('error', {
title: 'Could Not Restart Allocation',
description: messageForError(err, 'manage allocation lifecycle'),
});
}
})
restartAllocation;
@task(function* () {
try {
yield this.model.restartAll();
} catch (err) {
this.set('error', {
title: 'Could Not Restart All Tasks',
description: messageForError(err, 'manage allocation lifecycle'),
});
console.error(err);
}
})
restartAll;
@action
gotoTask(allocation, task) {
this.transitionToRoute('allocations.allocation.task', task);
}
@action
taskClick(allocation, task, event) {
lazyClick([() => this.send('gotoTask', allocation, task), event]);
}
//#region Services
@tracked activeServiceID = null;
@action handleServiceClick(service) {
this.set('activeServiceID', service.refID);
}
@computed('activeServiceID', 'services')
get activeService() {
return this.services.findBy('refID', this.activeServiceID);
}
@action closeSidebar() {
this.set('activeServiceID', null);
}
keyCommands = [
{
label: 'Close Evaluations Sidebar',
pattern: ['Escape'],
action: () => this.closeSidebar(),
},
];
//#endregion Services
}