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, KubernetesComponentStatusService, 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.KubernetesComponentStatusService = KubernetesComponentStatusService; 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.getComponentStatus = this.getComponentStatus.bind(this); this.getComponentStatusAsync = this.getComponentStatusAsync.bind(this); this.getEndpointsAsync = this.getEndpointsAsync.bind(this); this.hasResourceUsageAccess = this.hasResourceUsageAccess.bind(this); } async getComponentStatusAsync() { try { this.componentStatuses = await this.KubernetesComponentStatusService.get(); this.hasUnhealthyComponentStatus = _.find(this.componentStatuses, { Healthy: false }) ? true : false; } catch (err) { this.Notifications.error('Failure', err, 'Unable to retrieve cluster component statuses'); } } getComponentStatus() { return this.$async(this.getComponentStatusAsync); } 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 endpoints'); } } 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', 'Unable to retrieve applications', err); } 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', 'Unable to retrieve cluster resource usage', err); } } /** * 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, hasUnhealthyComponentStatus: false, useServerMetrics, }; await this.getNodes(); if (this.isAdmin) { await this.getEndpoints(); await this.getComponentStatus(); await this.getApplications(); } this.state.viewReady = true; } $onInit() { return this.$async(this.onInit); } } export default KubernetesClusterController; angular.module('portainer.kubernetes').controller('KubernetesClusterController', KubernetesClusterController);