open-nomad/ui/app/controllers/topology.js

147 lines
4.4 KiB
JavaScript
Raw Normal View History

import Controller from '@ember/controller';
import { computed, action } from '@ember/object';
import { alias } from '@ember/object/computed';
import { inject as service } from '@ember/service';
import { tracked } from '@glimmer/tracking';
import classic from 'ember-classic-decorator';
import { reduceBytes, reduceHertz } from 'nomad-ui/utils/units';
const sumAggregator = (sum, value) => sum + (value || 0);
const formatter = new Intl.NumberFormat(window.navigator.locale || 'en', {
maximumFractionDigits: 2,
});
@classic
export default class TopologyControllers extends Controller {
@service userSettings;
@alias('userSettings.showTopoVizPollingNotice') showPollingNotice;
@tracked filteredNodes = null;
@computed('model.nodes.@each.datacenter')
get datacenters() {
return Array.from(new Set(this.model.nodes.mapBy('datacenter'))).compact();
}
@computed('model.allocations.@each.isScheduled')
get scheduledAllocations() {
return this.model.allocations.filterBy('isScheduled');
}
@computed('model.nodes.@each.resources')
get totalMemory() {
const mibs = this.model.nodes.mapBy('resources.memory').reduce(sumAggregator, 0);
return mibs * 1024 * 1024;
}
@computed('model.nodes.@each.resources')
get totalCPU() {
return this.model.nodes.mapBy('resources.cpu').reduce((sum, cpu) => sum + (cpu || 0), 0);
}
@computed('totalMemory')
get totalMemoryFormatted() {
return formatter.format(reduceBytes(this.totalMemory)[0]);
}
@computed('totalMemory')
get totalMemoryUnits() {
return reduceBytes(this.totalMemory)[1];
}
@computed('totalCPU')
get totalCPUFormatted() {
return formatter.format(reduceHertz(this.totalCPU, null, 'MHz')[0]);
}
@computed('totalCPU')
get totalCPUUnits() {
return reduceHertz(this.totalCPU, null, 'MHz')[1];
}
@computed('scheduledAllocations.@each.allocatedResources')
get totalReservedMemory() {
const mibs = this.scheduledAllocations
.mapBy('allocatedResources.memory')
.reduce(sumAggregator, 0);
return mibs * 1024 * 1024;
}
@computed('scheduledAllocations.@each.allocatedResources')
get totalReservedCPU() {
return this.scheduledAllocations.mapBy('allocatedResources.cpu').reduce(sumAggregator, 0);
}
@computed('totalMemory', 'totalReservedMemory')
get reservedMemoryPercent() {
2020-09-11 19:15:41 +00:00
if (!this.totalReservedMemory || !this.totalMemory) return 0;
return this.totalReservedMemory / this.totalMemory;
}
@computed('totalCPU', 'totalReservedCPU')
get reservedCPUPercent() {
2020-09-11 19:15:41 +00:00
if (!this.totalReservedCPU || !this.totalCPU) return 0;
return this.totalReservedCPU / this.totalCPU;
}
@computed('activeAllocation.taskGroupName', 'scheduledAllocations.@each.{job,taskGroupName}')
get siblingAllocations() {
if (!this.activeAllocation) return [];
const taskGroup = this.activeAllocation.taskGroupName;
const jobId = this.activeAllocation.belongsTo('job').id();
return this.scheduledAllocations.filter(allocation => {
return allocation.taskGroupName === taskGroup && allocation.belongsTo('job').id() === jobId;
});
}
@computed('activeNode')
get nodeUtilization() {
const node = this.activeNode;
const [formattedMemory, memoryUnits] = reduceBytes(node.memory * 1024 * 1024);
const totalReservedMemory = node.allocations.mapBy('memory').reduce(sumAggregator, 0);
const totalReservedCPU = node.allocations.mapBy('cpu').reduce(sumAggregator, 0);
return {
totalMemoryFormatted: formattedMemory.toFixed(2),
totalMemoryUnits: memoryUnits,
totalMemory: node.memory * 1024 * 1024,
totalReservedMemory: totalReservedMemory * 1024 * 1024,
reservedMemoryPercent: totalReservedMemory / node.memory,
totalCPU: node.cpu,
totalReservedCPU,
reservedCPUPercent: totalReservedCPU / node.cpu,
};
}
@computed('siblingAllocations.@each.node')
get uniqueActiveAllocationNodes() {
return this.siblingAllocations.mapBy('node.id').uniq();
}
@action
async setAllocation(allocation) {
if (allocation) {
await allocation.reload();
await allocation.job.reload();
}
this.set('activeAllocation', allocation);
}
@action
setNode(node) {
this.set('activeNode', node);
}
@action
handleTopoVizDataError(errors) {
const filteredNodesError = errors.findBy('type', 'filtered-nodes');
if (filteredNodesError) {
this.filteredNodes = filteredNodesError.context;
}
}
}