/** * Copyright (c) OpenLens Authors. All rights reserved. * Licensed under MIT License. See LICENSE in root directory for more information. */ import "./pod-container-env.scss"; import React, { useEffect, useState } from "react"; import { observer } from "mobx-react"; import type { IPodContainer, Secret } from "../../../common/k8s-api/endpoints"; import { DrawerItem } from "../drawer"; import { autorun } from "mobx"; import { secretsStore } from "../+config-secrets/secrets.store"; import { configMapsStore } from "../+config-maps/config-maps.store"; import { Icon } from "../icon"; import { base64, cssNames, iter } from "../../utils"; import _ from "lodash"; export interface ContainerEnvironmentProps { container: IPodContainer; namespace: string; } export const ContainerEnvironment = observer((props: ContainerEnvironmentProps) => { const { container: { env, envFrom }, namespace } = props; useEffect( () => autorun(() => { for (const { valueFrom } of env ?? []) { if (valueFrom?.configMapKeyRef) { configMapsStore.load({ name: valueFrom.configMapKeyRef.name, namespace }); } } for (const { configMapRef, secretRef } of envFrom ?? []) { if (secretRef?.name) { secretsStore.load({ name: secretRef.name, namespace }); } if (configMapRef?.name) { configMapsStore.load({ name: configMapRef.name, namespace }); } } }), []); const renderEnv = () => { const orderedEnv = _.sortBy(env, "name"); return orderedEnv.map(variable => { const { name, value, valueFrom } = variable; let secretValue = null; if (value) { secretValue = value; } if (valueFrom) { const { fieldRef, secretKeyRef, configMapKeyRef } = valueFrom; if (fieldRef) { const { apiVersion, fieldPath } = fieldRef; secretValue = `fieldRef(${apiVersion}:${fieldPath})`; } if (secretKeyRef) { secretValue = ( ); } if (configMapKeyRef) { const { name, key } = configMapKeyRef; const configMap = configMapsStore.getByName(name, namespace); secretValue = configMap ? configMap.data[key] : `configMapKeyRef(${name}${key})`; } } return (
{name}: {secretValue}
); }); }; const renderEnvFrom = () => { return Array.from(iter.filterFlatMap(envFrom, vars => { if (vars.configMapRef?.name) { return renderEnvFromConfigMap(vars.configMapRef.name); } if (vars.secretRef?.name) { return renderEnvFromSecret(vars.secretRef.name); } return null; })); }; const renderEnvFromConfigMap = (configMapName: string) => { const configMap = configMapsStore.getByName(configMapName, namespace); if (!configMap) return null; return Object.entries(configMap.data).map(([name, value]) => (
{name}: {value}
)); }; const renderEnvFromSecret = (secretName: string) => { const secret = secretsStore.getByName(secretName, namespace); if (!secret) return null; return Object.keys(secret.data).map(key => { const secretKeyRef = { name: secret.getName(), key, }; const value = ( ); return (
{key}: {value}
); }); }; return ( {env && renderEnv()} {envFrom && renderEnvFrom()} ); }); export interface SecretKeyProps { reference: { name: string; key: string; }; namespace: string; } const SecretKey = (props: SecretKeyProps) => { const { reference: { name, key }, namespace } = props; const [loading, setLoading] = useState(false); const [secret, setSecret] = useState(); const showKey = async (evt: React.MouseEvent) => { evt.preventDefault(); setLoading(true); const secret = await secretsStore.load({ name, namespace }); setLoading(false); setSecret(secret); }; if (secret?.data?.[key]) { return <>{base64.decode(secret.data[key])}; } return ( <> secretKeyRef({name}.{key})  ); };