/** * Copyright (c) OpenLens Authors. All rights reserved. * Licensed under MIT License. See LICENSE in root directory for more information. */ import "./pod-details-container.scss"; import React from "react"; import type { Container, PodContainerStatus, Pod } from "../../../common/k8s-api/endpoints"; import { DrawerItem } from "../drawer"; import { cssNames, isDefined } from "../../utils"; import { StatusBrick } from "../status-brick"; import { Badge } from "../badge"; import { ContainerEnvironment } from "./pod-container-env"; import { PodContainerPort } from "./pod-container-port"; import { ResourceMetrics } from "../resource-metrics"; import type { MetricData } from "../../../common/k8s-api/endpoints/metrics.api"; import { ContainerCharts } from "./container-charts"; import { LocaleDate } from "../locale-date"; import { ClusterMetricsResourceType } from "../../../common/cluster-types"; import type { PortForwardStore } from "../../port-forward"; import { disposeOnUnmount, observer } from "mobx-react"; import { withInjectables } from "@ogre-tools/injectable-react"; import portForwardStoreInjectable from "../../port-forward/port-forward-store/port-forward-store.injectable"; import type { GetActiveClusterEntity } from "../../api/catalog/entity/get-active-cluster-entity.injectable"; import getActiveClusterEntityInjectable from "../../api/catalog/entity/get-active-cluster-entity.injectable"; export interface PodDetailsContainerProps { pod: Pod; container: Container; metrics?: Partial>; } interface Dependencies { portForwardStore: PortForwardStore; getActiveClusterEntity: GetActiveClusterEntity; } @observer class NonInjectedPodDetailsContainer extends React.Component { componentDidMount() { disposeOnUnmount(this, [ this.props.portForwardStore.watch(), ]); } renderStatus(state: string, status: PodContainerStatus | null | undefined) { const { ready = false, state: containerState = {}} = status ?? {}; const { terminated } = containerState; return ( {state} {ready ? ", ready" : ""} {terminated ? ` - ${terminated.reason} (exit code: ${terminated.exitCode})` : ""} ); } renderLastState(lastState: string, status: PodContainerStatus | null | undefined) { const { lastState: lastContainerState = {}} = status ?? {}; const { terminated } = lastContainerState; if (lastState === "terminated" && terminated) { return ( {lastState}
Reason: {`Reason: ${terminated.reason} - exit code: ${terminated.exitCode}`}
{"Started at: "} {}
{"Finished at: "} {}
); } return null; } render() { const { pod, container, metrics, getActiveClusterEntity } = this.props; if (!pod || !container) return null; const { name, image, imagePullPolicy, ports, volumeMounts, command, args } = container; const status = pod.getContainerStatuses().find(status => status.name === container.name); const state = status ? Object.keys(status?.state ?? {})[0] : ""; const lastState = status ? Object.keys(status?.lastState ?? {})[0] : ""; const ready = status ? status.ready : ""; const imageId = status? status.imageID : ""; const liveness = pod.getLivenessProbe(container); const readiness = pod.getReadinessProbe(container); const startup = pod.getStartupProbe(container); const isInitContainer = !!pod.getInitContainers().find(c => c.name == name); const isMetricHidden = getActiveClusterEntity()?.isMetricHidden(ClusterMetricsResourceType.Container); return (
{name}
{(!isMetricHidden && !isInitContainer && metrics) && ( )} {status && ( {this.renderStatus(state, status)} )} {lastState && ( {this.renderLastState(lastState, status)} )} {imagePullPolicy && imagePullPolicy !== "IfNotPresent" && ( {imagePullPolicy} )} {ports && ports.length > 0 && ( { ports .filter(isDefined) .map((port) => ( )) } )} {} {volumeMounts && volumeMounts.length > 0 && ( { volumeMounts.map(mount => { const { name, mountPath, readOnly } = mount; return ( {mountPath} {`from ${name} (${readOnly ? "ro" : "rw"})`} ); }) } )} {liveness.length > 0 && ( { liveness.map((value, index) => ( )) } )} {readiness.length > 0 && ( { readiness.map((value, index) => ( )) } )} {startup.length > 0 && ( { startup.map((value, index) => ( )) } )} {command && ( {command.join(" ")} )} {args && ( {args.join(" ")} )}
); } } export const PodDetailsContainer = withInjectables(NonInjectedPodDetailsContainer, { getProps: (di, props) => ({ ...props, portForwardStore: di.inject(portForwardStoreInjectable), getActiveClusterEntity: di.inject(getActiveClusterEntityInjectable), }), });