1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00

Make KubeconfigDialog injectable

Signed-off-by: Sebastian Malton <sebastian@malton.name>
This commit is contained in:
Sebastian Malton 2022-08-10 09:37:25 -04:00
parent 6cbb87705f
commit 5a2a9248e8
7 changed files with 168 additions and 125 deletions

View File

@ -9,6 +9,7 @@ import type { Cluster } from "../../../common/cluster/cluster";
import type { V1Secret } from "@kubernetes/client-node"; import type { V1Secret } from "@kubernetes/client-node";
import { CoreV1Api } from "@kubernetes/client-node"; import { CoreV1Api } from "@kubernetes/client-node";
import { clusterRoute } from "../../router/route"; import { clusterRoute } from "../../router/route";
import { dump } from "js-yaml";
const getServiceAccountRouteInjectable = getRouteInjectable({ const getServiceAccountRouteInjectable = getRouteInjectable({
id: "get-service-account-route", id: "get-service-account-route",
@ -50,43 +51,15 @@ const getServiceAccountRouteInjectable = getRouteInjectable({
export default getServiceAccountRouteInjectable; export default getServiceAccountRouteInjectable;
interface ServiceAccountKubeConfig { function generateKubeConfig(username: string, secret: V1Secret, cluster: Cluster): string | undefined {
apiVersion: string;
kind: string;
clusters: {
name: string;
cluster: {
server: string;
"certificate-authority-data": string;
};
}[];
users: {
name: string;
user: {
token: string;
};
}[];
contexts: {
name: string;
context: {
user: string;
cluster: string;
namespace: string | undefined;
};
}[];
"current-context": string;
}
function generateKubeConfig(username: string, secret: V1Secret, cluster: Cluster): ServiceAccountKubeConfig | undefined {
if (!secret.data || !secret.metadata) { if (!secret.data || !secret.metadata) {
return undefined; return undefined;
} }
const { token, "ca.crt": caCrt } = secret.data; const { token, "ca.crt": caCrt } = secret.data;
const tokenData = Buffer.from(token, "base64"); const tokenData = Buffer.from(token, "base64");
return { return dump({
"apiVersion": "v1", "apiVersion": "v1",
"kind": "Config", "kind": "Config",
"clusters": [ "clusters": [
@ -117,5 +90,5 @@ function generateKubeConfig(username: string, secret: V1Secret, cluster: Cluster
}, },
], ],
"current-context": cluster.contextName, "current-context": cluster.contextName,
}; });
} }

View File

@ -7,20 +7,33 @@ import React from "react";
import type { KubeObjectMenuProps } from "../../kube-object-menu"; import type { KubeObjectMenuProps } from "../../kube-object-menu";
import type { ServiceAccount } from "../../../../common/k8s-api/endpoints"; import type { ServiceAccount } from "../../../../common/k8s-api/endpoints";
import { MenuItem } from "../../menu"; import { MenuItem } from "../../menu";
import { openServiceAccountKubeConfig } from "../../kubeconfig-dialog";
import { Icon } from "../../icon"; import { Icon } from "../../icon";
import type { OpenServiceAccountKubeConfigDialog } from "../../kubeconfig-dialog/open-service-account-kube-config-dialog.injectable";
import { withInjectables } from "@ogre-tools/injectable-react";
import openServiceAccountKubeConfigDialogInjectable from "../../kubeconfig-dialog/open-service-account-kube-config-dialog.injectable";
export function ServiceAccountMenu(props: KubeObjectMenuProps<ServiceAccount>) { interface Dependencies {
const { object, toolbar } = props; openServiceAccountKubeConfigDialog: OpenServiceAccountKubeConfigDialog;
}
function NonInjectedServiceAccountMenu(props: KubeObjectMenuProps<ServiceAccount> & Dependencies) {
const { object, toolbar, openServiceAccountKubeConfigDialog } = props;
return ( return (
<MenuItem onClick={() => openServiceAccountKubeConfig(object)}> <MenuItem onClick={() => openServiceAccountKubeConfigDialog(object)}>
<Icon <Icon
material="insert_drive_file" material="insert_drive_file"
tooltip="Kubeconfig File" tooltip="Kubeconfig File"
interactive={toolbar} interactive={toolbar}
/> />
<span className="title">Kubeconfig</span> <span className="title">Kubeconfig</span>
</MenuItem> </MenuItem>
); );
} }
export const ServiceAccountMenu = withInjectables<Dependencies, KubeObjectMenuProps<ServiceAccount>>(NonInjectedServiceAccountMenu, {
getProps: (di, props) => ({
...props,
openServiceAccountKubeConfigDialog: di.inject(openServiceAccountKubeConfigDialogInjectable),
}),
});

View File

@ -23,7 +23,7 @@ import hotbarStoreInjectable from "../../../common/hotbars/store.injectable";
import type { DeleteClusterDialogState } from "./state.injectable"; import type { DeleteClusterDialogState } from "./state.injectable";
import deleteClusterDialogStateInjectable from "./state.injectable"; import deleteClusterDialogStateInjectable from "./state.injectable";
export interface Dependencies { interface Dependencies {
state: IObservableValue<DeleteClusterDialogState | undefined>; state: IObservableValue<DeleteClusterDialogState | undefined>;
hotbarStore: HotbarStore; hotbarStore: HotbarStore;
} }

View File

@ -5,121 +5,95 @@
import styles from "./kubeconfig-dialog.module.scss"; import styles from "./kubeconfig-dialog.module.scss";
import React from "react"; import React from "react";
import { makeObservable, observable } from "mobx"; import type { IObservableValue } from "mobx";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import yaml from "js-yaml";
import type { ServiceAccount } from "../../../common/k8s-api/endpoints";
import { cssNames, saveFileDialog } from "../../utils"; import { cssNames, saveFileDialog } from "../../utils";
import { Button } from "../button"; import { Button } from "../button";
import type { DialogProps } from "../dialog"; import type { DialogProps } from "../dialog";
import { Dialog } from "../dialog"; import { Dialog } from "../dialog";
import { Icon } from "../icon"; import { Icon } from "../icon";
import { Notifications } from "../notifications"; import type { ShowNotification } from "../notifications";
import { Wizard, WizardStep } from "../wizard"; import { Wizard, WizardStep } from "../wizard";
import { apiBase } from "../../api";
import { MonacoEditor } from "../monaco-editor"; import { MonacoEditor } from "../monaco-editor";
import { clipboard } from "electron"; import { clipboard } from "electron";
import { withInjectables } from "@ogre-tools/injectable-react";
import showSuccessNotificationInjectable from "../notifications/show-success-notification.injectable";
import kubeconfigDialogStateInjectable from "./state.injectable";
export interface KubeconfigDialogData { export interface KubeconfigDialogData {
title?: React.ReactNode; title?: React.ReactNode;
loader: () => Promise<any>; config: string;
} }
export interface KubeConfigDialogProps extends Partial<DialogProps> { export interface KubeConfigDialogProps extends Partial<DialogProps> {
} }
const dialogState = observable.box<KubeconfigDialogData | undefined>(); interface Dependencies {
state: IObservableValue<KubeconfigDialogData | undefined>;
showSuccessNotification: ShowNotification;
}
@observer @observer
export class KubeConfigDialog extends React.Component<KubeConfigDialogProps> { class NonInjectedKubeConfigDialog extends React.Component<KubeConfigDialogProps & Dependencies> {
@observable config = ""; // parsed kubeconfig in yaml format constructor(props: KubeConfigDialogProps & Dependencies) {
constructor(props: KubeConfigDialogProps) {
super(props); super(props);
makeObservable(this);
}
static open(data: KubeconfigDialogData) {
dialogState.set(data);
}
static close() {
dialogState.set(undefined);
} }
close = () => { close = () => {
KubeConfigDialog.close(); this.props.state.set(undefined);
}; };
onOpen = (data: KubeconfigDialogData) => { copyToClipboard = (config: string) => {
this.loadConfig(data); clipboard.writeText(config);
this.props.showSuccessNotification("Config copied to clipboard");
}; };
async loadConfig(data: KubeconfigDialogData) { download = (config: string) => {
const config = await data.loader().catch(err => { saveFileDialog("config", config, "text/yaml");
Notifications.error(err);
this.close();
});
this.config = config ? yaml.dump(config) : "";
}
copyToClipboard = () => {
clipboard.writeText(this.config);
Notifications.ok("Config copied to clipboard");
}; };
download = () => { renderContents = (data: KubeconfigDialogData) => (
saveFileDialog("config", this.config, "text/yaml"); <Wizard header={<h5>{ data.title || "Kubeconfig File" }</h5>}>
}; <WizardStep
customButtons={ (
renderContents(data: KubeconfigDialogData) { <div className="actions flex gaps">
const yamlConfig = this.config; <Button plain onClick={() => this.copyToClipboard(data.config)}>
<Icon material="assignment" />
return ( {" Copy to clipboard"}
<Wizard header={<h5>{data.title || "Kubeconfig File"}</h5>}> </Button>
<WizardStep <Button plain onClick={() => this.download(data.config)}>
customButtons={( <Icon material="cloud_download" />
<div className="actions flex gaps"> {" Download file"}
<Button plain onClick={this.copyToClipboard}> </Button>
<Icon material="assignment"/> <Button
{" Copy to clipboard"} plain
</Button> className="box right"
<Button plain onClick={this.download}> onClick={this.close}
<Icon material="cloud_download"/> >
{" Download file"} Close
</Button> </Button>
<Button </div>
plain ) }
className="box right" prev={this.close}
onClick={this.close} >
> <MonacoEditor
Close readOnly
</Button> className={styles.editor}
</div> value={data.config}
)} />
prev={this.close} </WizardStep>
> </Wizard>
<MonacoEditor );
readOnly
className={styles.editor}
value={yamlConfig}
/>
</WizardStep>
</Wizard>
);
}
render() { render() {
const { className, ...dialogProps } = this.props; const { className, state, ...dialogProps } = this.props;
const data = dialogState.get(); const data = state.get();
return ( return (
<Dialog <Dialog
{...dialogProps} {...dialogProps}
className={cssNames(styles.KubeConfigDialog, className)} className={cssNames(styles.KubeConfigDialog, className)}
isOpen={Boolean(data)} isOpen={!!data}
onOpen={data && (() => this.onOpen(data))}
close={this.close} close={this.close}
> >
{data && this.renderContents(data)} {data && this.renderContents(data)}
@ -128,12 +102,10 @@ export class KubeConfigDialog extends React.Component<KubeConfigDialogProps> {
} }
} }
export function openServiceAccountKubeConfig(account: ServiceAccount) { export const KubeConfigDialog = withInjectables<Dependencies, KubeConfigDialogProps>(NonInjectedKubeConfigDialog, {
const accountName = account.getName(); getProps: (di, props) => ({
const namespace = account.getNs(); ...props,
showSuccessNotification: di.inject(showSuccessNotificationInjectable),
KubeConfigDialog.open({ state: di.inject(kubeconfigDialogStateInjectable),
title: `${accountName} kubeconfig`, }),
loader: () => apiBase.get(`/kubeconfig/service-account/${namespace}/${accountName}`), });
});
}

View File

@ -0,0 +1,31 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import type { ServiceAccount } from "../../../common/k8s-api/endpoints";
import { urlBuilderFor } from "../../../common/utils/buildUrl";
import apiBaseInjectable from "../../k8s/api-base.injectable";
import openKubeconfigDialogInjectable from "./open.injectable";
export type OpenServiceAccountKubeConfigDialog = (account: ServiceAccount) => void;
const serviceAccountConfigEndpoint = urlBuilderFor("/kubeconfig/service-account/:namespace/:name");
const openServiceAccountKubeConfigDialogInjectable = getInjectable({
id: "open-service-account-kube-config-dialog",
instantiate: (di): OpenServiceAccountKubeConfigDialog => {
const apiBase = di.inject(apiBaseInjectable);
const openKubeconfigDialog = di.inject(openKubeconfigDialogInjectable);
return (account) => openKubeconfigDialog({
title: `${account.getName()} kubeconfig`,
loader: () => apiBase.get(serviceAccountConfigEndpoint.compile({
name: account.getName(),
namespace: account.getNs(),
})),
});
},
});
export default openServiceAccountKubeConfigDialogInjectable;

View File

@ -0,0 +1,40 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import type React from "react";
import loggerInjectable from "../../../common/logger.injectable";
import showCheckedErrorNotificationInjectable from "../notifications/show-checked-error.injectable";
import kubeconfigDialogStateInjectable from "./state.injectable";
export interface OpenKubeconfigDialogArgs {
title?: React.ReactNode;
loader: () => Promise<string>;
}
export type OpenKubeconfigDialog = (openArgs: OpenKubeconfigDialogArgs) => void;
const openKubeconfigDialogInjectable = getInjectable({
id: "open-kubeconfig-dialog",
instantiate: (di): OpenKubeconfigDialog => {
const state = di.inject(kubeconfigDialogStateInjectable);
const showCheckedErrorNotification = di.inject(showCheckedErrorNotificationInjectable);
const logger = di.inject(loggerInjectable);
return ({ title, loader }) => {
(async () => {
try {
const config = await loader();
state.set({ title, config });
} catch (error) {
showCheckedErrorNotification(error, "Failed to retrive config for dialog");
logger.warn("[KUBEOCONFIG-DIALOG]: failed to retrived config for dialog", error);
}
})();
};
},
});
export default openKubeconfigDialogInjectable;

View File

@ -0,0 +1,14 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import { observable } from "mobx";
import type { KubeconfigDialogData } from "./kubeconfig-dialog";
const kubeconfigDialogStateInjectable = getInjectable({
id: "kubeconfig-dialog-state",
instantiate: () => observable.box<KubeconfigDialogData>(),
});
export default kubeconfigDialogStateInjectable;