/** * Copyright (c) OpenLens Authors. All rights reserved. * Licensed under MIT License. See LICENSE in root directory for more information. */ import "./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 "./details-resources"; import { DrawerTitle } from "../drawer/drawer-title"; import { boundMethod, Disposer } from "../../utils"; import logger from "../../../common/logger"; import type { KubeObjectStore } from "../../../common/k8s-api/kube-object.store"; import type { KubeObject } from "../../../common/k8s-api/kube-object"; import { withInjectables } from "@ogre-tools/injectable-react"; import kubeWatchApiInjectable from "../../kube-watch-api/kube-watch-api.injectable"; export interface NodeDetailsProps extends KubeObjectDetailsProps { } interface Dependencies { subscribeStores: (stores: KubeObjectStore[]) => Disposer; } @observer class NonInjectedNodeDetails extends React.Component { @observable metrics: Partial; constructor(props: NodeDetailsProps & Dependencies) { super(props); makeObservable(this); } componentDidMount() { disposeOnUnmount(this, [ reaction(() => this.props.object.getName(), () => { this.metrics = null; }), this.props.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}
, ), }} /> ); }) }
}
); } } export const NodeDetails = withInjectables( NonInjectedNodeDetails, { getProps: (di, props) => ({ subscribeStores: di.inject(kubeWatchApiInjectable).subscribeStores, ...props, }), }, );