/** * Copyright (c) 2021 OpenLens Authors * * Permission is hereby granted, free of charge, to any person obtaining a copy of * this software and associated documentation files (the "Software"), to deal in * the Software without restriction, including without limitation the rights to * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of * the Software, and to permit persons to whom the Software is furnished to do so, * subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ import "./node-details.scss"; import React from "react"; import upperFirst from "lodash/upperFirst"; import kebabCase from "lodash/kebabCase"; import { disposeOnUnmount, observer } from "mobx-react"; import { DrawerItem, DrawerItemLabels } from "../drawer"; import { Badge } from "../badge"; import { ResourceMetrics } from "../resource-metrics"; import { podsStore } from "../+workloads-pods/pods.store"; import type { KubeObjectDetailsProps } from "../kube-object-details"; import { formatNodeTaint, getMetricsByNodeNames, IClusterMetrics, Node } from "../../../common/k8s-api/endpoints"; import { NodeCharts } from "./node-charts"; import { makeObservable, observable, reaction } from "mobx"; import { PodDetailsList } from "../+workloads-pods/pod-details-list"; import { KubeObjectMeta } from "../kube-object-meta"; import { getActiveClusterEntity } from "../../api/catalog-entity-registry"; import { ClusterMetricsResourceType } from "../../../common/cluster-types"; import { NodeDetailsResources } from "./node-details-resources"; import { DrawerTitle } from "../drawer/drawer-title"; import { boundMethod } from "../../utils"; import logger from "../../../common/logger"; import { kubeWatchApi } from "../../../common/k8s-api/kube-watch-api"; interface Props extends KubeObjectDetailsProps { } @observer export class NodeDetails extends React.Component { @observable metrics: Partial; constructor(props: Props) { super(props); makeObservable(this); } componentDidMount() { disposeOnUnmount(this, [ reaction(() => this.props.object.getName(), () => { this.metrics = null; }), kubeWatchApi.subscribeStores([ podsStore, ]), ]); } @boundMethod async loadMetrics() { const { object: node } = this.props; this.metrics = await getMetricsByNodeNames([node.getName()]); } render() { const { object: node } = this.props; if (!node) { return null; } if (!(node instanceof Node)) { logger.error("[NodeDetails]: passed object that is not an instanceof Node", node); return null; } const { status } = node; const { nodeInfo, addresses } = status; const conditions = node.getActiveConditions(); const taints = node.getTaints(); const childPods = podsStore.getPodsByNode(node.getName()); const { metrics } = this; const metricTabs = [ "CPU", "Memory", "Disk", "Pods", ]; const isMetricHidden = getActiveClusterEntity()?.isMetricHidden(ClusterMetricsResourceType.Node); return (
{!isMetricHidden && podsStore.isLoaded && ( )} {addresses && { addresses.map(({ type, address }) => (

{type}: {address}

)) }
} {nodeInfo.operatingSystem} ({nodeInfo.architecture}) {nodeInfo.osImage} {nodeInfo.kernelVersion} {nodeInfo.containerRuntimeVersion} {nodeInfo.kubeletVersion} {taints.length > 0 && ( {taints.map(taint => )} )} {conditions && { conditions.map(condition => { const { type } = condition; return (
{upperFirst(key)}
{value}
, ), }} /> ); }) }
}
); } }