mirror of https://github.com/portainer/portainer
157 lines
5.3 KiB
JavaScript
157 lines
5.3 KiB
JavaScript
import angular from 'angular';
|
|
import _ from 'lodash-es';
|
|
import filesizeParser from 'filesize-parser';
|
|
import KubernetesResourceReservationHelper from 'Kubernetes/helpers/resourceReservationHelper';
|
|
import { KubernetesResourceReservation } from 'Kubernetes/models/resource-reservation/models';
|
|
|
|
class KubernetesClusterController {
|
|
/* @ngInject */
|
|
constructor(
|
|
$async,
|
|
$state,
|
|
Authentication,
|
|
Notifications,
|
|
LocalStorage,
|
|
KubernetesNodeService,
|
|
KubernetesMetricsService,
|
|
KubernetesApplicationService,
|
|
KubernetesEndpointService
|
|
) {
|
|
this.$async = $async;
|
|
this.$state = $state;
|
|
this.Authentication = Authentication;
|
|
this.Notifications = Notifications;
|
|
this.LocalStorage = LocalStorage;
|
|
this.KubernetesNodeService = KubernetesNodeService;
|
|
this.KubernetesMetricsService = KubernetesMetricsService;
|
|
this.KubernetesApplicationService = KubernetesApplicationService;
|
|
this.KubernetesEndpointService = KubernetesEndpointService;
|
|
|
|
this.onInit = this.onInit.bind(this);
|
|
this.getNodes = this.getNodes.bind(this);
|
|
this.getNodesAsync = this.getNodesAsync.bind(this);
|
|
this.getApplicationsAsync = this.getApplicationsAsync.bind(this);
|
|
this.getEndpointsAsync = this.getEndpointsAsync.bind(this);
|
|
this.hasResourceUsageAccess = this.hasResourceUsageAccess.bind(this);
|
|
}
|
|
|
|
async getEndpointsAsync() {
|
|
try {
|
|
const endpoints = await this.KubernetesEndpointService.get();
|
|
const systemEndpoints = _.filter(endpoints, { Namespace: 'kube-system' });
|
|
this.systemEndpoints = _.filter(systemEndpoints, (ep) => ep.HolderIdentity);
|
|
|
|
const kubernetesEndpoint = _.find(endpoints, { Name: 'kubernetes' });
|
|
if (kubernetesEndpoint && kubernetesEndpoint.Subsets) {
|
|
const ips = _.flatten(_.map(kubernetesEndpoint.Subsets, 'Ips'));
|
|
_.forEach(this.nodes, (node) => {
|
|
node.Api = _.includes(ips, node.IPAddress);
|
|
});
|
|
}
|
|
} catch (err) {
|
|
this.Notifications.error('Failure', err, 'Unable to retrieve environments');
|
|
}
|
|
}
|
|
|
|
getEndpoints() {
|
|
return this.$async(this.getEndpointsAsync);
|
|
}
|
|
|
|
async getNodesAsync() {
|
|
try {
|
|
const nodes = await this.KubernetesNodeService.get();
|
|
_.forEach(nodes, (node) => (node.Memory = filesizeParser(node.Memory)));
|
|
this.nodes = nodes;
|
|
this.CPULimit = _.reduce(this.nodes, (acc, node) => node.CPU + acc, 0);
|
|
this.MemoryLimit = _.reduce(this.nodes, (acc, node) => KubernetesResourceReservationHelper.megaBytesValue(node.Memory) + acc, 0);
|
|
} catch (err) {
|
|
this.Notifications.error('Failure', err, 'Unable to retrieve nodes');
|
|
}
|
|
}
|
|
|
|
getNodes() {
|
|
return this.$async(this.getNodesAsync);
|
|
}
|
|
|
|
async getApplicationsAsync() {
|
|
try {
|
|
this.state.applicationsLoading = true;
|
|
this.applications = await this.KubernetesApplicationService.get();
|
|
const nodeNames = _.map(this.nodes, (node) => node.Name);
|
|
this.resourceReservation = _.reduce(
|
|
this.applications,
|
|
(acc, app) => {
|
|
app.Pods = _.filter(app.Pods, (pod) => nodeNames.includes(pod.Node));
|
|
const resourceReservation = KubernetesResourceReservationHelper.computeResourceReservation(app.Pods);
|
|
acc.CPU += resourceReservation.CPU;
|
|
acc.Memory += resourceReservation.Memory;
|
|
return acc;
|
|
},
|
|
new KubernetesResourceReservation()
|
|
);
|
|
this.resourceReservation.Memory = KubernetesResourceReservationHelper.megaBytesValue(this.resourceReservation.Memory);
|
|
|
|
if (this.hasResourceUsageAccess()) {
|
|
await this.getResourceUsage(this.endpoint.Id);
|
|
}
|
|
} catch (err) {
|
|
this.Notifications.error('Failure', err, 'Unable to retrieve applications');
|
|
} finally {
|
|
this.state.applicationsLoading = false;
|
|
}
|
|
}
|
|
|
|
getApplications() {
|
|
return this.$async(this.getApplicationsAsync);
|
|
}
|
|
|
|
async getResourceUsage(endpointId) {
|
|
try {
|
|
const nodeMetrics = await this.KubernetesMetricsService.getNodes(endpointId);
|
|
const resourceUsageList = nodeMetrics.items.map((i) => i.usage);
|
|
const clusterResourceUsage = resourceUsageList.reduce((total, u) => {
|
|
total.CPU += KubernetesResourceReservationHelper.parseCPU(u.cpu);
|
|
total.Memory += KubernetesResourceReservationHelper.megaBytesValue(u.memory);
|
|
return total;
|
|
}, new KubernetesResourceReservation());
|
|
this.resourceUsage = clusterResourceUsage;
|
|
} catch (err) {
|
|
this.Notifications.error('Failure', err, 'Unable to retrieve cluster resource usage');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check if resource usage stats can be displayed
|
|
* @returns {boolean}
|
|
*/
|
|
hasResourceUsageAccess() {
|
|
return this.isAdmin && this.state.useServerMetrics;
|
|
}
|
|
|
|
async onInit() {
|
|
this.isAdmin = this.Authentication.isAdmin();
|
|
const useServerMetrics = this.endpoint.Kubernetes.Configuration.UseServerMetrics;
|
|
|
|
this.state = {
|
|
applicationsLoading: true,
|
|
viewReady: false,
|
|
useServerMetrics,
|
|
};
|
|
|
|
await this.getNodes();
|
|
if (this.isAdmin) {
|
|
await this.getEndpoints();
|
|
await this.getApplications();
|
|
}
|
|
|
|
this.state.viewReady = true;
|
|
}
|
|
|
|
$onInit() {
|
|
return this.$async(this.onInit);
|
|
}
|
|
}
|
|
|
|
export default KubernetesClusterController;
|
|
angular.module('portainer.kubernetes').controller('KubernetesClusterController', KubernetesClusterController);
|