mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* Remove Singleton from BaseStore to remove global shared state Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove more usages of Singleton Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace use of legacy global execHelm with injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove last use of legacy global execHelm Signed-off-by: Sebastian Malton <sebastian@malton.name> * Extract BaseStore deps into constructor argument Signed-off-by: Sebastian Malton <sebastian@malton.name> * Introduce method to make store migrations injectable - Use it for ClusterStore Signed-off-by: Sebastian Malton <sebastian@malton.name> * Switch HotbarStore to injectable migrations Signed-off-by: Sebastian Malton <sebastian@malton.name> * Switch UserStore to injectable migrations Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move migration utils into common/utils/ Signed-off-by: Sebastian Malton <sebastian@malton.name> * Switch WeblinkStore to injectable migrations Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove dead code Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix type error in base-store tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove tests that reference lastSeenVersion - That value is not used anywhere in code Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove usage of legacy global .getInstance Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove usage of legacy global ClusterStore.getInstance Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add simple migrations dependency for stores without any preexisting migrations Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix messed up import Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add typing to transient injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Cleanup formatting Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix typing in tests to satisfy requirement to have cacheFile Signed-off-by: Sebastian Malton <sebastian@malton.name> * More consistent use of BaseStore.displayName Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add catching of error while starting main application Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move initializing sentry to runnable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove unneeded appPathsInjectionToken - Only had once impl, which was in common anyway Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add support for multiple "runAfter" runnables - Needed so that several dependencies can be declared Signed-off-by: Sebastian Malton <sebastian@malton.name> * Use multiple runAfter support to fix crash on renderer Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove traces Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add global override to fix tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix base store tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix runManyFor tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix hotbar store tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix user store tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add global override for getConfigurationFileModel to fix tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove overrides for configuration stores - Now that there is an override for getConfiguration Signed-off-by: Sebastian Malton <sebastian@malton.name> * Overhaul FS fakes with full in-memory filesystem - This increases our confidence in fs related logic Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove use of global shared Electron.App Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add fake access support Signed-off-by: Sebastian Malton <sebastian@malton.name> * Handle copy as part of fake FS Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add ensureDir/Sync support to fake FS Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix type error Signed-off-by: Sebastian Malton <sebastian@malton.name> * Use pathExistsSync instead of fsInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add createReadStream to fake FS Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add stat to fake FS Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove dead code Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix test failures due to incomplete overrides Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fully injectable-ize BaseStore so that ApplicationBuilder tests work Signed-off-by: Sebastian Malton <sebastian@malton.name> * Consolidate more bootstrapping into startFrame Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move initializing CatalogCategories to runnable in bootstrap Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert contextMenuOpen initializers into runnables Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert navigateForExtension init to runnable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make cluster state sync fully injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move init hotbar store into runnables Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make LensTheme fully injectable and runnable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Cleanup old code from missed from previous commit Signed-off-by: Sebastian Malton <sebastian@malton.name> * Manually split out terminal color names and fully type LensTheme Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix old imports Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove unnecessart awaits Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove dead code Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fully cherry pick injectablizing custom monaco themes Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix duplicate mock warning Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix incorrectly fully cherry picking new runnable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Complete cherry-pick of current cluster injcetablization Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix override file name Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix injecting before app paths are set up Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix injecting before app paths are set up Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix ordering of runnable and order of injection Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert all renderer runnables to late-inject style - To help fix issues around injection time Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix react-beautiful-dnd mocks Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update and fix WriteJson(Sync) to fix error in tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix HotbarStore.load being called twice is being buggy Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update listing-active-helm-repositories-in-preferences snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix sidebar-and-tab-navigation-tests - Move enabling extensions in tests to a proper location - Fix flushing promises Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove props from dnd mock to make snapshot diffs smaller Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix import Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests by overriding things that are no longer overriden by default - NOTE: They are overridden when using ApplicationBuilder Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix hotbar store tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix cluster store tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix extension-loader tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix extension-discovery tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix cluster-role-dialog tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix user store tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix kubeconfig sync tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix sidebar and tab tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove unused code Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix pick paths import type error and simplify signature Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix type error in legacy ipc registration Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove another use of legacy requestOpenPathPicker Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace use of legacy global PathPicker.Pick Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix usage in light of changed prop names Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix catalog tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix more type errors Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix test flakiness by removing side effects from userStore preferences Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix loading Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix type error Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix crash Signed-off-by: Sebastian Malton <sebastian@malton.name> * Cherry pick updated startFrameInjectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add tests to verify runMany behaviour in new possible incorrect configuration Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix init ordering during start frame Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix cluster state sync Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update snapshots after removing side-effects Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add override for technical test Signed-off-by: Sebastian Malton <sebastian@malton.name> * Correctly mark currentlyInClusterFrame as causedSideEffects Signed-off-by: Sebastian Malton <sebastian@malton.name> * Better formatting Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix behaviour regression Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add better logging Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix BaseStore sync Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Update last snapshot Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add global override for randomBytes Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make startMainApplication not an injection time side effect Signed-off-by: Sebastian Malton <sebastian@malton.name> * Choose better names for start-frame runnable tokens Signed-off-by: Sebastian Malton <sebastian@malton.name> * Remove duplication of code in RunManyFor Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add unit tests and fix handling empty runAfter array Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace use of mobx from runManyFor with custom barrier Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add missing test Signed-off-by: Sebastian Malton <sebastian@malton.name> Signed-off-by: Sebastian Malton <sebastian@malton.name>
168 lines
5.4 KiB
TypeScript
168 lines
5.4 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import type { Cluster } from "../../common/cluster/cluster";
|
|
import * as yaml from "js-yaml";
|
|
import tempy from "tempy";
|
|
import type { Patch } from "rfc6902";
|
|
import type { KubernetesObject } from "@kubernetes/client-node";
|
|
import type { EmitAppEvent } from "../../common/app-event-bus/emit-event.injectable";
|
|
import type { Logger } from "../../common/logger";
|
|
import type { WriteFile } from "../../common/fs/write-file.injectable";
|
|
import type { RemovePath } from "../../common/fs/remove.injectable";
|
|
import type { ExecFile } from "../../common/fs/exec-file.injectable";
|
|
import type { JoinPaths } from "../../common/path/join-paths.injectable";
|
|
import type { AsyncResult } from "../../common/utils/async-result";
|
|
|
|
export interface ResourceApplierDependencies {
|
|
emitAppEvent: EmitAppEvent;
|
|
writeFile: WriteFile;
|
|
deleteFile: RemovePath;
|
|
execFile: ExecFile;
|
|
joinPaths: JoinPaths;
|
|
readonly logger: Logger;
|
|
}
|
|
|
|
export class ResourceApplier {
|
|
constructor(protected readonly dependencies: ResourceApplierDependencies, protected readonly cluster: Cluster) {}
|
|
|
|
/**
|
|
* Patch a kube resource's manifest, throwing any error that occurs.
|
|
* @param name The name of the kube resource
|
|
* @param kind The kind of the kube resource
|
|
* @param patch The list of JSON operations
|
|
* @param ns The optional namespace of the kube resource
|
|
*/
|
|
async patch(name: string, kind: string, patch: Patch, ns?: string): Promise<string> {
|
|
this.dependencies.emitAppEvent({ name: "resource", action: "patch" });
|
|
|
|
const kubectl = await this.cluster.ensureKubectl();
|
|
const kubectlPath = await kubectl.getPath();
|
|
const proxyKubeconfigPath = await this.cluster.getProxyKubeconfigPath();
|
|
const args = [
|
|
"--kubeconfig", proxyKubeconfigPath,
|
|
"patch",
|
|
kind,
|
|
name,
|
|
];
|
|
|
|
if (ns) {
|
|
args.push("--namespace", ns);
|
|
}
|
|
|
|
args.push(
|
|
"--type", "json",
|
|
"--patch", JSON.stringify(patch),
|
|
"-o", "json",
|
|
);
|
|
|
|
const result = await this.dependencies.execFile(kubectlPath, args);
|
|
|
|
if (result.callWasSuccessful) {
|
|
return result.response;
|
|
}
|
|
|
|
throw result.error.stderr || result.error.message;
|
|
}
|
|
|
|
async create(resource: string): Promise<AsyncResult<string, string>> {
|
|
this.dependencies.emitAppEvent({ name: "resource", action: "apply" });
|
|
|
|
return this.kubectlApply(this.sanitizeObject(resource));
|
|
}
|
|
|
|
protected async kubectlApply(content: string): Promise<AsyncResult<string, string>> {
|
|
const kubectl = await this.cluster.ensureKubectl();
|
|
const kubectlPath = await kubectl.getPath();
|
|
const proxyKubeconfigPath = await this.cluster.getProxyKubeconfigPath();
|
|
const fileName = tempy.file({ name: "resource.yaml" });
|
|
const args = [
|
|
"apply",
|
|
"--kubeconfig", proxyKubeconfigPath,
|
|
"-o", "json",
|
|
"-f", fileName,
|
|
];
|
|
|
|
this.dependencies.logger.debug(`shooting manifests with ${kubectlPath}`, { args });
|
|
|
|
const execEnv = { ...process.env };
|
|
const httpsProxy = this.cluster.preferences?.httpsProxy;
|
|
|
|
if (httpsProxy) {
|
|
execEnv.HTTPS_PROXY = httpsProxy;
|
|
}
|
|
|
|
try {
|
|
await this.dependencies.writeFile(fileName, content);
|
|
|
|
const result = await this.dependencies.execFile(kubectlPath, args);
|
|
|
|
if (result.callWasSuccessful) {
|
|
return result;
|
|
}
|
|
|
|
return {
|
|
callWasSuccessful: false,
|
|
error: result.error.stderr || result.error.message,
|
|
};
|
|
} finally {
|
|
await this.dependencies.deleteFile(fileName);
|
|
}
|
|
}
|
|
|
|
public async kubectlApplyAll(resources: string[], extraArgs = ["-o", "json"]): Promise<AsyncResult<string, string>> {
|
|
return this.kubectlCmdAll("apply", resources, extraArgs);
|
|
}
|
|
|
|
public async kubectlDeleteAll(resources: string[], extraArgs?: string[]): Promise<AsyncResult<string, string>> {
|
|
return this.kubectlCmdAll("delete", resources, extraArgs);
|
|
}
|
|
|
|
protected async kubectlCmdAll(subCmd: string, resources: string[], parentArgs: string[] = []): Promise<AsyncResult<string, string>> {
|
|
const kubectl = await this.cluster.ensureKubectl();
|
|
const kubectlPath = await kubectl.getPath();
|
|
const proxyKubeconfigPath = await this.cluster.getProxyKubeconfigPath();
|
|
const tmpDir = tempy.directory();
|
|
|
|
await Promise.all(resources.map((resource, index) => this.dependencies.writeFile(
|
|
this.dependencies.joinPaths(tmpDir, `${index}.yaml`),
|
|
resource,
|
|
)));
|
|
|
|
const args = [
|
|
subCmd,
|
|
"--kubeconfig", proxyKubeconfigPath,
|
|
...parentArgs,
|
|
"-f", tmpDir,
|
|
];
|
|
|
|
this.dependencies.logger.info(`[RESOURCE-APPLIER] running kubectl`, { args });
|
|
const result = await this.dependencies.execFile(kubectlPath, args);
|
|
|
|
if (result.callWasSuccessful) {
|
|
return result;
|
|
}
|
|
|
|
this.dependencies.logger.error(`[RESOURCE-APPLIER] kubectl errored: ${result.error.message}`);
|
|
|
|
const splitError = result.error.stderr.split(`.yaml": `);
|
|
|
|
return {
|
|
callWasSuccessful: false,
|
|
error: splitError[1] || result.error.message,
|
|
};
|
|
}
|
|
|
|
protected sanitizeObject(resource: string) {
|
|
const res = yaml.load(resource) as Partial<KubernetesObject> & { status?: object };
|
|
|
|
delete res.status;
|
|
delete res.metadata?.resourceVersion;
|
|
delete res.metadata?.annotations?.["kubectl.kubernetes.io/last-applied-configuration"];
|
|
|
|
return yaml.dump(res);
|
|
}
|
|
}
|