mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
explicit namespaces for their resources Signed-off-by: Michael Pearson <mipearson@gmail.com>
234 lines
5.5 KiB
TypeScript
234 lines
5.5 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import tempy from "tempy";
|
|
import fse from "fs-extra";
|
|
import * as yaml from "js-yaml";
|
|
import { toCamelCase } from "../../common/utils/camelCase";
|
|
import { execFile } from "child_process";
|
|
import { execHelm } from "./exec";
|
|
import assert from "assert";
|
|
import type { JsonObject, JsonValue } from "type-fest";
|
|
import { isObject, json } from "../../common/utils";
|
|
|
|
export async function listReleases(pathToKubeconfig: string, namespace?: string): Promise<Record<string, any>[]> {
|
|
const args = [
|
|
"ls",
|
|
"--all",
|
|
"--output", "json",
|
|
];
|
|
|
|
if (namespace) {
|
|
args.push("-n", namespace);
|
|
} else {
|
|
args.push("--all-namespaces");
|
|
}
|
|
|
|
args.push("--kubeconfig", pathToKubeconfig);
|
|
|
|
const output = json.parse(await execHelm(args));
|
|
|
|
if (!Array.isArray(output) || output.length == 0) {
|
|
return [];
|
|
}
|
|
|
|
return output.filter(isObject).map(toCamelCase);
|
|
}
|
|
|
|
|
|
export async function installChart(chart: string, values: JsonValue, name: string | undefined = "", namespace: string, version: string, kubeconfigPath: string) {
|
|
const valuesFilePath = tempy.file({ name: "values.yaml" });
|
|
|
|
await fse.writeFile(valuesFilePath, yaml.dump(values));
|
|
|
|
const args = ["install"];
|
|
|
|
if (name) {
|
|
args.push(name);
|
|
}
|
|
|
|
args.push(
|
|
chart,
|
|
"--version", version,
|
|
"--values", valuesFilePath,
|
|
"--namespace", namespace,
|
|
"--kubeconfig", kubeconfigPath,
|
|
);
|
|
|
|
if (!name) {
|
|
args.push("--generate-name");
|
|
}
|
|
|
|
try {
|
|
const output = await execHelm(args);
|
|
const releaseName = output.split("\n")[0].split(" ")[1].trim();
|
|
|
|
return {
|
|
log: output,
|
|
release: {
|
|
name: releaseName,
|
|
namespace,
|
|
},
|
|
};
|
|
} finally {
|
|
await fse.unlink(valuesFilePath);
|
|
}
|
|
}
|
|
|
|
export async function upgradeRelease(name: string, chart: string, values: any, namespace: string, version: string, kubeconfigPath: string, kubectlPath: string) {
|
|
const valuesFilePath = tempy.file({ name: "values.yaml" });
|
|
|
|
await fse.writeFile(valuesFilePath, yaml.dump(values));
|
|
|
|
const args = [
|
|
"upgrade",
|
|
name,
|
|
chart,
|
|
"--version", version,
|
|
"--values", valuesFilePath,
|
|
"--namespace", namespace,
|
|
"--kubeconfig", kubeconfigPath,
|
|
];
|
|
|
|
try {
|
|
const output = await execHelm(args);
|
|
|
|
return {
|
|
log: output,
|
|
release: await getRelease(name, namespace, kubeconfigPath, kubectlPath),
|
|
};
|
|
} finally {
|
|
await fse.unlink(valuesFilePath);
|
|
}
|
|
}
|
|
|
|
export async function getRelease(name: string, namespace: string, kubeconfigPath: string, kubectlPath: string) {
|
|
const args = [
|
|
"status",
|
|
name,
|
|
"--namespace", namespace,
|
|
"--kubeconfig", kubeconfigPath,
|
|
"--output", "json",
|
|
];
|
|
|
|
const release = json.parse(await execHelm(args, {
|
|
maxBuffer: 32 * 1024 * 1024 * 1024, // 32 MiB
|
|
}));
|
|
|
|
if (!isObject(release) || Array.isArray(release)) {
|
|
return undefined;
|
|
}
|
|
|
|
release.resources = await getResources(name, namespace, kubeconfigPath, kubectlPath);
|
|
|
|
return release;
|
|
}
|
|
|
|
export async function deleteRelease(name: string, namespace: string, kubeconfigPath: string) {
|
|
return execHelm([
|
|
"delete",
|
|
name,
|
|
"--namespace", namespace,
|
|
"--kubeconfig", kubeconfigPath,
|
|
]);
|
|
}
|
|
|
|
interface GetValuesOptions {
|
|
namespace: string;
|
|
all?: boolean;
|
|
kubeconfigPath: string;
|
|
}
|
|
|
|
export async function getValues(name: string, { namespace, all = false, kubeconfigPath }: GetValuesOptions) {
|
|
const args = [
|
|
"get",
|
|
"values",
|
|
name,
|
|
];
|
|
|
|
if (all) {
|
|
args.push("--all");
|
|
}
|
|
|
|
args.push(
|
|
"--output", "yaml",
|
|
"--namespace", namespace,
|
|
"--kubeconfig", kubeconfigPath,
|
|
);
|
|
|
|
return execHelm(args);
|
|
}
|
|
|
|
export async function getHistory(name: string, namespace: string, kubeconfigPath: string) {
|
|
return json.parse(await execHelm([
|
|
"history",
|
|
name,
|
|
"--output", "json",
|
|
"--namespace", namespace,
|
|
"--kubeconfig", kubeconfigPath,
|
|
]));
|
|
}
|
|
|
|
export async function rollback(name: string, namespace: string, revision: number, kubeconfigPath: string) {
|
|
await execHelm([
|
|
"rollback",
|
|
name,
|
|
`${revision}`,
|
|
"--namespace", namespace,
|
|
"--kubeconfig", kubeconfigPath,
|
|
]);
|
|
}
|
|
|
|
async function getResources(name: string, namespace: string, kubeconfigPath: string, kubectlPath: string) {
|
|
const helmArgs = [
|
|
"get",
|
|
"manifest",
|
|
name,
|
|
"--namespace", namespace,
|
|
"--kubeconfig", kubeconfigPath,
|
|
];
|
|
const kubectlArgs = [
|
|
"get",
|
|
"--kubeconfig", kubeconfigPath,
|
|
"-f", "-",
|
|
"--output", "json",
|
|
"--namespace", namespace,
|
|
];
|
|
|
|
try {
|
|
const helmOutput = await execHelm(helmArgs);
|
|
|
|
return new Promise<JsonObject[]>((resolve, reject) => {
|
|
let stdout = "";
|
|
let stderr = "";
|
|
const kubectl = execFile(kubectlPath, kubectlArgs);
|
|
|
|
kubectl
|
|
.on("exit", (code, signal) => {
|
|
if (typeof code === "number") {
|
|
if (code === 0) {
|
|
const output = json.parse(stdout) as { items: JsonObject[] };
|
|
|
|
resolve(output.items);
|
|
} else {
|
|
reject(stderr);
|
|
}
|
|
} else {
|
|
reject(new Error(`Kubectl exited with signal ${signal}`));
|
|
}
|
|
})
|
|
.on("error", reject);
|
|
|
|
assert(kubectl.stderr && kubectl.stdout && kubectl.stdin, "For some reason the IO streams are undefined");
|
|
|
|
kubectl.stderr.on("data", output => stderr += output);
|
|
kubectl.stdout.on("data", output => stdout += output);
|
|
kubectl.stdin.end(helmOutput);
|
|
});
|
|
} catch {
|
|
return [];
|
|
}
|
|
}
|