/** * Copyright (c) OpenLens Authors. All rights reserved. * Licensed under MIT License. See LICENSE in root directory for more information. */ import "./pod-details-list.scss"; import React from "react"; import kebabCase from "lodash/kebabCase"; import { reaction } from "mobx"; import { disposeOnUnmount, observer } from "mobx-react"; import { podStore } from "./legacy-store"; import type { Pod } from "../../../common/k8s-api/endpoints"; import { autoBind, bytesToUnits, cssNames, interval, prevDefault } from "../../utils"; import { LineProgress } from "../line-progress"; import type { KubeObject } from "../../../common/k8s-api/kube-object"; import { Table, TableCell, TableHead, TableRow } from "../table"; import { Spinner } from "../spinner"; import { DrawerTitle } from "../drawer"; import { KubeObjectStatusIcon } from "../kube-object-status-icon"; import { showDetails } from "../kube-detail-params"; enum sortBy { name = "name", node = "node", namespace = "namespace", cpu = "cpu", memory = "memory", } export interface PodDetailsListProps { pods: Pod[]; owner: KubeObject; maxCpu?: number; maxMemory?: number; } @observer export class PodDetailsList extends React.Component { constructor(props: PodDetailsListProps) { super(props); autoBind(this); } private metricsWatcher = interval(120, () => { podStore.loadKubeMetrics(this.props.owner.getNs()); }); componentDidMount() { this.metricsWatcher.start(true); disposeOnUnmount(this, [ reaction(() => this.props.owner, () => this.metricsWatcher.restart(true)), ]); } componentWillUnmount() { this.metricsWatcher.stop(); } renderCpuUsage(id: string, usage: number) { const { maxCpu } = this.props; const value = usage.toFixed(3); if (!maxCpu) { if (parseFloat(value) === 0) return 0; return value; } const tooltip = (

{`CPU: ${Math.ceil(usage * 100 / maxCpu)}%`}
{usage.toFixed(3)}

); return ( ); } renderMemoryUsage(id: string, usage: number) { const { maxMemory } = this.props; if (!maxMemory) return usage ? bytesToUnits(usage) : 0; const tooltip = (

{`Memory: ${Math.ceil(usage * 100 / maxMemory)}%`}
{bytesToUnits(usage, { precision: 3 })}

); return ( ); } getTableRow(uid: string) { const { pods } = this.props; const pod = pods.find(pod => pod.getId() == uid); if (!pod) { return; } const metrics = podStore.getPodKubeMetrics(pod); return ( showDetails(pod.selfLink, false))} > {pod.getName()} {pod.getNodeName()} {pod.getNs()} {`${pod.getRunningContainers().length} / ${pod.getContainers().length}`} {this.renderCpuUsage(`cpu-${pod.getId()}`, metrics.cpu)} {this.renderMemoryUsage(`memory-${pod.getId()}`, metrics.memory)} {pod.getStatusMessage()} ); } render() { const { pods } = this.props; if (!podStore.isLoaded) { return (
); } if (!pods.length) { return null; } const virtual = pods.length > 20; return (
Pods pod.getName(), [sortBy.node]: pod => pod.getNodeName(), [sortBy.namespace]: pod => pod.getNs(), [sortBy.cpu]: pod => podStore.getPodKubeMetrics(pod).cpu, [sortBy.memory]: pod => podStore.getPodKubeMetrics(pod).memory, }} sortByDefault={{ sortBy: sortBy.cpu, orderBy: "desc" }} sortSyncWithUrl={false} getTableRow={this.getTableRow} renderRow={( virtual ? undefined : (pod => this.getTableRow(pod.getId())) )} className="box grow" > Name Node Namespace Ready CPU Memory Status
); } }