Files
nomad/ui/app/controllers/clients/client/index.js
Phil Renaud cabe05705c [ui] Disconnected Clients: "Unknown" allocations in the UI (#12544)
* Unknown status for allocations accounted for

* Canary string removed

* Test cleanup

* Generate unknown in mirage

* aacidentally oovervoowled

* Update ui/app/components/allocation-status-bar.js

Co-authored-by: Derek Strickland <1111455+DerekStrickland@users.noreply.github.com>

* Disconnected state on job status in client

* Renaming Disconnected to Unknown in the job-status-in-client

* Unknown accounted for on job rows filtering and testsfix

* Adding lostAllocs as a computed dependency

* Unknown client status within acceptance test

* Swatches updated and PR comments addressed

* Unknown and disconnected added to test fixtures

Co-authored-by: Derek Strickland <1111455+DerekStrickland@users.noreply.github.com>
2022-04-22 11:25:02 -04:00

261 lines
6.6 KiB
JavaScript

/* eslint-disable ember/no-observers */
/* eslint-disable ember/no-incorrect-calls-with-inline-anonymous-functions */
import { alias } from '@ember/object/computed';
import Controller from '@ember/controller';
import { action, computed } from '@ember/object';
import { observes } from '@ember-decorators/object';
import { scheduleOnce } from '@ember/runloop';
import { task } from 'ember-concurrency';
import intersection from 'lodash.intersection';
import Sortable from 'nomad-ui/mixins/sortable';
import Searchable from 'nomad-ui/mixins/searchable';
import messageFromAdapterError from 'nomad-ui/utils/message-from-adapter-error';
import {
serialize,
deserializedQueryParam as selection,
} from 'nomad-ui/utils/qp-serialize';
import classic from 'ember-classic-decorator';
@classic
export default class ClientController extends Controller.extend(
Sortable,
Searchable
) {
queryParams = [
{
currentPage: 'page',
},
{
searchTerm: 'search',
},
{
sortProperty: 'sort',
},
{
sortDescending: 'desc',
},
{
onlyPreemptions: 'preemptions',
},
{
qpNamespace: 'namespace',
},
{
qpJob: 'job',
},
{
qpStatus: 'status',
},
];
// Set in the route
flagAsDraining = false;
qpNamespace = '';
qpJob = '';
qpStatus = '';
currentPage = 1;
pageSize = 8;
sortProperty = 'modifyIndex';
sortDescending = true;
@computed()
get searchProps() {
return ['shortId', 'name'];
}
onlyPreemptions = false;
@computed('model.allocations.[]', 'preemptions.[]', 'onlyPreemptions')
get visibleAllocations() {
return this.onlyPreemptions ? this.preemptions : this.model.allocations;
}
@computed(
'visibleAllocations.[]',
'selectionNamespace',
'selectionJob',
'selectionStatus'
)
get filteredAllocations() {
const { selectionNamespace, selectionJob, selectionStatus } = this;
return this.visibleAllocations.filter((alloc) => {
if (
selectionNamespace.length &&
!selectionNamespace.includes(alloc.get('namespace'))
) {
return false;
}
if (
selectionJob.length &&
!selectionJob.includes(alloc.get('plainJobId'))
) {
return false;
}
if (
selectionStatus.length &&
!selectionStatus.includes(alloc.clientStatus)
) {
return false;
}
return true;
});
}
@alias('filteredAllocations') listToSort;
@alias('listSorted') listToSearch;
@alias('listSearched') sortedAllocations;
@selection('qpNamespace') selectionNamespace;
@selection('qpJob') selectionJob;
@selection('qpStatus') selectionStatus;
eligibilityError = null;
stopDrainError = null;
drainError = null;
showDrainNotification = false;
showDrainUpdateNotification = false;
showDrainStoppedNotification = false;
@computed('model.allocations.@each.wasPreempted')
get preemptions() {
return this.model.allocations.filterBy('wasPreempted');
}
@computed('model.events.@each.time')
get sortedEvents() {
return this.get('model.events').sortBy('time').reverse();
}
@computed('model.drivers.@each.name')
get sortedDrivers() {
return this.get('model.drivers').sortBy('name');
}
@computed('model.hostVolumes.@each.name')
get sortedHostVolumes() {
return this.model.hostVolumes.sortBy('name');
}
@(task(function* (value) {
try {
yield value ? this.model.setEligible() : this.model.setIneligible();
} catch (err) {
const error = messageFromAdapterError(err) || 'Could not set eligibility';
this.set('eligibilityError', error);
}
}).drop())
setEligibility;
@(task(function* () {
try {
this.set('flagAsDraining', false);
yield this.model.cancelDrain();
this.set('showDrainStoppedNotification', true);
} catch (err) {
this.set('flagAsDraining', true);
const error = messageFromAdapterError(err) || 'Could not stop drain';
this.set('stopDrainError', error);
}
}).drop())
stopDrain;
@(task(function* () {
try {
yield this.model.forceDrain({
IgnoreSystemJobs: this.model.drainStrategy.ignoreSystemJobs,
});
} catch (err) {
const error = messageFromAdapterError(err) || 'Could not force drain';
this.set('drainError', error);
}
}).drop())
forceDrain;
@observes('model.isDraining')
triggerDrainNotification() {
if (!this.model.isDraining && this.flagAsDraining) {
this.set('showDrainNotification', true);
}
this.set('flagAsDraining', this.model.isDraining);
}
@action
gotoAllocation(allocation) {
this.transitionToRoute('allocations.allocation', allocation);
}
@action
setPreemptionFilter(value) {
this.set('onlyPreemptions', value);
}
@action
drainNotify(isUpdating) {
this.set('showDrainUpdateNotification', isUpdating);
}
@action
setDrainError(err) {
const error = messageFromAdapterError(err) || 'Could not run drain';
this.set('drainError', error);
}
get optionsAllocationStatus() {
return [
{ key: 'pending', label: 'Pending' },
{ key: 'running', label: 'Running' },
{ key: 'complete', label: 'Complete' },
{ key: 'failed', label: 'Failed' },
{ key: 'lost', label: 'Lost' },
{ key: 'unknown', label: 'Unknown' },
];
}
@computed('model.allocations.[]', 'selectionJob', 'selectionNamespace')
get optionsJob() {
// Only show options for jobs in the selected namespaces, if any.
const ns = this.selectionNamespace;
const jobs = Array.from(
new Set(
this.model.allocations
.filter((a) => ns.length === 0 || ns.includes(a.namespace))
.mapBy('plainJobId')
)
).compact();
// Update query param when the list of jobs changes.
scheduleOnce('actions', () => {
// eslint-disable-next-line ember/no-side-effects
this.set('qpJob', serialize(intersection(jobs, this.selectionJob)));
});
return jobs.sort().map((job) => ({ key: job, label: job }));
}
@computed('model.allocations.[]', 'selectionNamespace')
get optionsNamespace() {
const ns = Array.from(
new Set(this.model.allocations.mapBy('namespace'))
).compact();
// Update query param when the list of namespaces changes.
scheduleOnce('actions', () => {
// eslint-disable-next-line ember/no-side-effects
this.set(
'qpNamespace',
serialize(intersection(ns, this.selectionNamespace))
);
});
return ns.sort().map((n) => ({ key: n, label: n }));
}
setFacetQueryParam(queryParam, selection) {
this.set(queryParam, serialize(selection));
}
}