mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* Making apiBase injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert all of Helm functions to be DI Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make PortForward's use of apiBase fully injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert all metric requests to be injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace resource applier with injectables Signed-off-by: Sebastian Malton <sebastian@malton.name> * Switch KubeJsonApi.forCluster to be injectable but do not use Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert the rest of shell sessions to be DI-ed - This is a prerequesit for using the new createKubeJsonApiForClusterInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Use new createKubeJsonApiForClusterInjectable for openNodeShellSession Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make KubeconfigDialog injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove jest-fetch-mock and make fetch injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests with new global override Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add new injectable for create KubeJsonApi and JsonApi instances Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix showing-details-for-helm-release behavioural tests - Remove HelmChartStore in favour of all injectables - Create a model for UpgradeChartDockTab Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix show details and updating helm releases tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix residual typing issues related to metrics Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix crash on load due to circular dependency Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix create resource tab not working Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove legacy apiBase global Signed-off-by: Sebastian Malton <sebastian@malton.name> * Introduce and use isDebuggingInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Introduce and use windowLocationInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove global legacy apiKube Signed-off-by: Sebastian Malton <sebastian@malton.name> * Improve injectable filenames compared to the injectables inside Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove modifying input in requestActivePortForwardInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Introduce and use get(Milli)SecondsFromUnixEpochInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Switch to non-reactive way of gettting possible helm release versions Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix typo Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix bug in KubeApi constructor Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert all KubeApi related tests to use asyncFn Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix unit tests after introducing new injectables that have side effects Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix bad rebase causing tests to fail Signed-off-by: Sebastian Malton <sebastian@malton.name> * Improve expects for multiple field values Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix crash will looking up api refs Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix breaking change on KubeApi.list Signed-off-by: Sebastian Malton <sebastian@malton.name> * Better fix for formatting urls Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove injectable for time since we should just use useMockTime Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add happy path behavioural tests for upgrade chart tab Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove debug message Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix showing-details-for-helm-release tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix installing-helm-chart-from-new-tab tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests relating to hosted cluster id Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots to recent changes in master Co-authored-by: Janne Savolainen <janne.savolainen@live.fi> Signed-off-by: Iku-turso <mikko.aspiala@gmail.com> * Reupdated upgrade chart new tab test snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix flakiness in unit test when using <Animated> Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix flakiness and improve tests for DeleteClusterDialog Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix kubeconfig-sync tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix <Extensions> tests by removing mockFs and making everything injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix build issues Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix getElectronAppPathInjectable override not returning absolute paths - Also fixes the listing-active-helm-repos-in-prefs tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace all uses of getAbsolutePath with joinPaths as it is more correct and less confusing Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix opening application window tests by making override properly absolute Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots relating no longer using getAbsolutePath Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix and add behavioural tests for RenderDelay Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix extension discovery tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix test flakiness because of path side effects, propagate uses to as many places Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix extension-discovery tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add global override to fix some tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Rewrite and fix implementation of KubeconfigManager and its tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests by global override pathExists Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix unit tests failing on windows by using injectable verions of path functions Signed-off-by: Sebastian Malton <sebastian@malton.name> * Attempt to fix test timeout by using runInAction Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots after rebase Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots after rebase Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests after rebase Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix setupIpcMainHandlers usage Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> Signed-off-by: Sebastian Malton <sebastian@malton.name> Signed-off-by: Iku-turso <mikko.aspiala@gmail.com> Co-authored-by: Iku-turso <mikko.aspiala@gmail.com>
173 lines
5.6 KiB
TypeScript
173 lines
5.6 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import { v4 as uuid } from "uuid";
|
|
import { Watch, CoreV1Api } from "@kubernetes/client-node";
|
|
import type { KubeConfig } from "@kubernetes/client-node";
|
|
import type { ShellSessionArgs, ShellSessionDependencies } from "../shell-session";
|
|
import { ShellOpenError, ShellSession } from "../shell-session";
|
|
import { get, once } from "lodash";
|
|
import { Node, NodeApi } from "../../../common/k8s-api/endpoints";
|
|
import { TerminalChannels } from "../../../common/terminal/channels";
|
|
import type { CreateKubeJsonApiForCluster } from "../../../common/k8s-api/create-kube-json-api-for-cluster.injectable";
|
|
|
|
export interface NodeShellSessionArgs extends ShellSessionArgs {
|
|
nodeName: string;
|
|
}
|
|
|
|
export interface NodeShellSessionDependencies extends ShellSessionDependencies {
|
|
createKubeJsonApiForCluster: CreateKubeJsonApiForCluster;
|
|
}
|
|
|
|
export class NodeShellSession extends ShellSession {
|
|
ShellType = "node-shell";
|
|
|
|
protected readonly podName = `node-shell-${uuid()}`;
|
|
protected readonly nodeName: string;
|
|
protected readonly cwd: string | undefined = undefined;
|
|
|
|
constructor(protected readonly dependencies: NodeShellSessionDependencies, { nodeName, ...args }: NodeShellSessionArgs) {
|
|
super(dependencies, args);
|
|
this.nodeName = nodeName;
|
|
}
|
|
|
|
public async open() {
|
|
const kc = await this.cluster.getProxyKubeconfig();
|
|
const coreApi = kc.makeApiClient(CoreV1Api);
|
|
const shell = await this.kubectl.getPath();
|
|
|
|
const cleanup = once(() => {
|
|
coreApi
|
|
.deleteNamespacedPod(this.podName, "kube-system")
|
|
.catch(error => this.dependencies.logger.warn(`[NODE-SHELL]: failed to remove pod shell`, error));
|
|
});
|
|
|
|
this.websocket.once("close", cleanup);
|
|
|
|
try {
|
|
await this.createNodeShellPod(coreApi);
|
|
await this.waitForRunningPod(kc);
|
|
} catch (error) {
|
|
cleanup();
|
|
|
|
this.send({
|
|
type: TerminalChannels.STDOUT,
|
|
data: `Error occurred: ${get(error, "response.body.message", error ? String(error) : "unknown error")}`,
|
|
});
|
|
|
|
throw new ShellOpenError(
|
|
"failed to create node pod",
|
|
error instanceof Error
|
|
? { cause: error }
|
|
: undefined,
|
|
);
|
|
}
|
|
|
|
const env = await this.getCachedShellEnv();
|
|
const args = ["exec", "-i", "-t", "-n", "kube-system", this.podName, "--"];
|
|
const nodeApi = new NodeApi({
|
|
objectConstructor: Node,
|
|
request: this.dependencies.createKubeJsonApiForCluster(this.cluster.id),
|
|
});
|
|
const node = await nodeApi.get({ name: this.nodeName });
|
|
|
|
if (!node) {
|
|
throw new Error(`No node with name=${this.nodeName} found`);
|
|
}
|
|
|
|
const nodeOs = node.getOperatingSystem();
|
|
|
|
switch (nodeOs) {
|
|
default:
|
|
this.dependencies.logger.warn(`[NODE-SHELL-SESSION]: could not determine node OS, falling back with assumption of linux`);
|
|
// fallthrough
|
|
case "linux":
|
|
args.push("sh", "-c", "((clear && bash) || (clear && ash) || (clear && sh))");
|
|
break;
|
|
case "windows":
|
|
args.push("powershell");
|
|
break;
|
|
}
|
|
|
|
await this.openShellProcess(shell, args, env);
|
|
}
|
|
|
|
protected createNodeShellPod(coreApi: CoreV1Api) {
|
|
const imagePullSecrets = this.cluster.imagePullSecret
|
|
? [{
|
|
name: this.cluster.imagePullSecret,
|
|
}]
|
|
: undefined;
|
|
|
|
return coreApi
|
|
.createNamespacedPod("kube-system", {
|
|
metadata: {
|
|
name: this.podName,
|
|
namespace: "kube-system",
|
|
},
|
|
spec: {
|
|
nodeName: this.nodeName,
|
|
restartPolicy: "Never",
|
|
terminationGracePeriodSeconds: 0,
|
|
hostPID: true,
|
|
hostIPC: true,
|
|
hostNetwork: true,
|
|
tolerations: [{
|
|
operator: "Exists",
|
|
}],
|
|
priorityClassName: "system-node-critical",
|
|
containers: [{
|
|
name: "shell",
|
|
image: this.cluster.nodeShellImage,
|
|
securityContext: {
|
|
privileged: true,
|
|
},
|
|
command: ["nsenter"],
|
|
args: ["-t", "1", "-m", "-u", "-i", "-n", "sleep", "14000"],
|
|
}],
|
|
imagePullSecrets,
|
|
},
|
|
});
|
|
}
|
|
|
|
protected waitForRunningPod(kc: KubeConfig): Promise<void> {
|
|
this.dependencies.logger.debug(`[NODE-SHELL]: waiting for ${this.podName} to be running`);
|
|
|
|
return new Promise((resolve, reject) => {
|
|
new Watch(kc)
|
|
.watch(`/api/v1/namespaces/kube-system/pods`,
|
|
{},
|
|
// callback is called for each received object.
|
|
(type, { metadata: { name }, status }) => {
|
|
if (name === this.podName) {
|
|
switch (status.phase) {
|
|
case "Running":
|
|
return resolve();
|
|
case "Failed":
|
|
return reject(`Failed to be created: ${status.message || "unknown error"}`);
|
|
}
|
|
}
|
|
},
|
|
// done callback is called if the watch terminates normally
|
|
(err) => {
|
|
this.dependencies.logger.error(`[NODE-SHELL]: ${this.podName} was not created in time`);
|
|
reject(err);
|
|
},
|
|
)
|
|
.then(req => {
|
|
setTimeout(() => {
|
|
this.dependencies.logger.error(`[NODE-SHELL]: aborting wait for ${this.podName}, timing out`);
|
|
req.abort();
|
|
reject("Pod creation timed out");
|
|
}, 2 * 60 * 1000); // 2 * 60 * 1000
|
|
})
|
|
.catch(error => {
|
|
this.dependencies.logger.error(`[NODE-SHELL]: waiting for ${this.podName} failed: ${error}`);
|
|
reject(error);
|
|
});
|
|
});
|
|
}
|
|
}
|