mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* Turn on strict mode in tsconfig.json - Add route, clusterRoute, and payloadValidatedClusterRoute helper functions to improve types with backend routes - Turn on the following new lints: - react/jsx-first-prop-new-line - react/jsx-wrap-multilines - react/jsx-one-expression-per-line - react/jsx-max-props-per-line - react/jsx-indent - react/jsx-indent-props Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix build Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace KubeObject scope strings with enum Signed-off-by: Sebastian Malton <sebastian@malton.name> * Revert package.json version changes Signed-off-by: Sebastian Malton <sebastian@malton.name> * revert move hostedCluster(Id) Signed-off-by: Sebastian Malton <sebastian@malton.name> * change some type param names to be not single letters Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove copy-extension-themes Signed-off-by: Sebastian Malton <sebastian@malton.name> * add new make clean action Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix build to not use webpack for generating types Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix kube-object-menu.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix select.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix catalog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * revert move fileNameMigration to index Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix ref logic error Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix log-resource-selector.test.tsx tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix dock-store.test.ts test by overriding createStorage to not touch file system Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix cluster.test.ts tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix kube=api.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fixed hotbar-store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix kubeconfig-manager.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix cluster-role-bindings/__tests__/dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix role-bindings/__tests__/dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix pods.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix delete-cluster-dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix daemonset.store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix replicaset.store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix statefulsets/dialog/dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix replicasets/scale-dialog/dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix deployments.store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix deployments/scale/dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix cronjob.store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix stateful-set.api.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix deployment.api.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix api-manager.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix statefulset.store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix job.store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix pods.store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix scroll-spy.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix hotbar-remove-command.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix catalog-entity-registry.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix welcome.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix verify-that-all-routes-have-route-component.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix pod-tolerations.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * better fix for previous 3 fixes, plus also select.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix kube-object-menu.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix app-paths.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix dock-tabs.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix isReactNode typing Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix sub-title.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix drawer.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix list-layout.tsx and header.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix error-boundary.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix upgrade-chart/store.ts and dock-tab.store.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix install-chart/store.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix edit-resource/store.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix create-resource/store.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix namespace-select.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix namespace-select-filter.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix crd-list.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix wrong types for extensions Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix circular dependency Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix circular dependency on catalogCategoryRegistry Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix api-kube Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix type errors, most <Select /> errors Signed-off-by: Sebastian Malton <sebastian@malton.name> * fixing more type errors Signed-off-by: Sebastian Malton <sebastian@malton.name> * some more fixing type errors Signed-off-by: Sebastian Malton <sebastian@malton.name> * convert all KubeApis to injectable with legacy global backups Signed-off-by: Sebastian Malton <sebastian@malton.name> * factor out into a common file all the exports Signed-off-by: Sebastian Malton <sebastian@malton.name> * convert all KubeObjectStores to injectable with legacy global backups Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix lint Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove unused legacy KubeApi globals Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix bad previous commit Signed-off-by: Sebastian Malton <sebastian@malton.name> * more crash fixing Signed-off-by: Sebastian Malton <sebastian@malton.name> * try and fix behavioural tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix sidebar-and-tab-navigation-for-core.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix sidebar-and-tab-navigation-for-extensions.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-using-application-menu.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix catalog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make ThemeStore non-singleton and fix navigation-to-terminal-preferences.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * extensions.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix catalog-entity-registry.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-using-application-menu.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix log-resource-selector.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix dock-tabs.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix delete-cluster-dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-to-kubernetes-preferences.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-to-editor-preferences.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-to-proxy-preferences.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-using-application-menu.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-to-application-preferences.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix dock-store.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix select.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix role-bindings/__tests__/dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix hotbar-remove-command.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix cluster-role-bindings/__tests__/dialog.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-to-extension-specific-preferences.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-to-telemetry-preferences.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix closing-preferences.test.tsx Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-to-editor-preferences.test.ts Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix navigation-to-proxy-preferences.test.ts - Fix other type errors too Signed-off-by: Sebastian Malton <sebastian@malton.name> * final tweaks Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add more tsconfig files, fix bug in <Catalog> - Make all of history, navigation injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix type errors Signed-off-by: Sebastian Malton <sebastian@malton.name> * Convert all of kube-details-params/ and navigate/ to injectable - This fixes a runtime error that was encountered during testing Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix runtime errors on renderer - remove all static uses of `createPageParam` (and then removed the legacy global) - Made LensRendererExtension and LensMainExtension just used dependencies and not the getLegacyDi - Fixed circular dep in extension-loader Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move registerStore calls to after injectMany Signed-off-by: Sebastian Malton <sebastian@malton.name> * replace all the rest of the legacy uses of apiManager Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix stack overflow and cycles in DI Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix NamespaceSelectFilter not opening Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix WizardStep and AddNamespaceDialog Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix KubeApi's not being registered Signed-off-by: Sebastian Malton <sebastian@malton.name> * cleanup WindowManager Signed-off-by: Sebastian Malton <sebastian@malton.name> * Proper fix for Wizard, fix NamespaceStore.subscribe Signed-off-by: Sebastian Malton <sebastian@malton.name> * Rewrite withTooltip to be more type correct - Fixes mobx related "too many recursive actions" error - Change all the uses of withTooltips to be functional components Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add e2e test to cover kube api registration Signed-off-by: Sebastian Malton <sebastian@malton.name> * cleanup internal-commands Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove cast in <Animate> Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix command-palette e2e test Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix type error after rebase Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix test name Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix lint Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix code to help CodeQL scanner Signed-off-by: Sebastian Malton <sebastian@malton.name> * update intree extension lock files Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix build-extensions picking wrong @types/react Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix tests from rebase Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix type error Signed-off-by: Sebastian Malton <sebastian@malton.name> * Make KubeconfigSyncManager more injectable Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix crash in test mode for Dialog Signed-off-by: Sebastian Malton <sebastian@malton.name> * make Select snapshots deterministic Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix new type error Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix kube-object.store.test.ts typing Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix merge build issues Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix snapshots after merge Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix lint after merge Signed-off-by: Sebastian Malton <sebastian@malton.name> * reexport BaseKubeJsonApiObjectMetadata Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix typo in terminalSpawningPool Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove duplicate license header Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix typo to waitUntilDefined Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove iter use from getLegacyGlobalDiForExtensionApi Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove complex createStorage override Signed-off-by: Sebastian Malton <sebastian@malton.name> * override logger with mocks only when needed for tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove specialized overrideStore flags for getDiForUnitTesting Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove unnecessary | undefined types from the exactOptionalFieldTypes experiment Signed-off-by: Sebastian Malton <sebastian@malton.name> * use more descriptive name for local test mocks Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove unnecessary addition to 'make clean' target Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove oddity of KubeObjectStore.getById(undefined) being allowed Signed-off-by: Sebastian Malton <sebastian@malton.name> * rename KubeObject.getDescriptor in favour of name without fundemental JS meaning Signed-off-by: Sebastian Malton <sebastian@malton.name> * Simplify legacyRegisterApi when working in behaviour unit tests - Don't emit within main environment as there should be no auto registering there Signed-off-by: Sebastian Malton <sebastian@malton.name> * change confusing variable name in ReactiveDuration Signed-off-by: Sebastian Malton <sebastian@malton.name> * make visitor pattern more explicit for Entity contextMenuOpen Signed-off-by: Sebastian Malton <sebastian@malton.name> * toggleDetails -> toggleKubeDetailsPane is more specific Signed-off-by: Sebastian Malton <sebastian@malton.name> * remove outdated comment Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix bug where LensExtension dependencies are not set Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix tests from the revert of react 18 Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix more tests from merge Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix typings with new is-compatible-extension tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * more type fixing Signed-off-by: Sebastian Malton <sebastian@malton.name> * Revert in-tree extension versions Signed-off-by: Sebastian Malton <sebastian@malton.name> * Improve name of guarding injectable for stores and apis - New name better implies that it is just a guard state and does not do anything Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add helper for <Select>.isMulti for storing in a Set<Value> Signed-off-by: Sebastian Malton <sebastian@malton.name> * fix is-compatible-extension.test.ts types Signed-off-by: Sebastian Malton <sebastian@malton.name>
382 lines
13 KiB
TypeScript
382 lines
13 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import type { IComputedValue, ObservableMap } from "mobx";
|
|
import { action, observable, computed, runInAction, makeObservable, observe } from "mobx";
|
|
import type { CatalogEntity } from "../../../common/catalog";
|
|
import type { FSWatcher } from "chokidar";
|
|
import { watch } from "chokidar";
|
|
import type { Stats } from "fs";
|
|
import fs from "fs";
|
|
import path from "path";
|
|
import type stream from "stream";
|
|
import type { Disposer } from "../../../common/utils";
|
|
import { bytesToUnits, getOrInsertWith, iter, noop } from "../../../common/utils";
|
|
import logger from "../../logger";
|
|
import type { KubeConfig } from "@kubernetes/client-node";
|
|
import { loadConfigFromString, splitConfig } from "../../../common/kube-helpers";
|
|
import { catalogEntityFromCluster, ClusterManager } from "../../cluster-manager";
|
|
import { UserStore } from "../../../common/user-store";
|
|
import { ClusterStore } from "../../../common/cluster-store/cluster-store";
|
|
import { createHash } from "crypto";
|
|
import { homedir } from "os";
|
|
import globToRegExp from "glob-to-regexp";
|
|
import { inspect } from "util";
|
|
import type { ClusterModel, UpdateClusterModel } from "../../../common/cluster-types";
|
|
import type { Cluster } from "../../../common/cluster/cluster";
|
|
import type { CatalogEntityRegistry } from "../../catalog/entity-registry";
|
|
|
|
const logPrefix = "[KUBECONFIG-SYNC]:";
|
|
|
|
/**
|
|
* This is the list of globs of which files are ignored when under a folder sync
|
|
*/
|
|
const ignoreGlobs = [
|
|
"*.lock", // kubectl lock files
|
|
"*.swp", // vim swap files
|
|
".DS_Store", // macOS specific
|
|
].map(rawGlob => ({
|
|
rawGlob,
|
|
matcher: globToRegExp(rawGlob),
|
|
}));
|
|
|
|
/**
|
|
* This should be much larger than any kubeconfig text file
|
|
*
|
|
* Even if you have a cert-file, key-file, and client-cert files that is only
|
|
* 12kb of extra data (at 4096 bytes each) which allows for around 150 entries.
|
|
*/
|
|
const folderSyncMaxAllowedFileReadSize = 2 * 1024 * 1024; // 2 MiB
|
|
const fileSyncMaxAllowedFileReadSize = 16 * folderSyncMaxAllowedFileReadSize; // 32 MiB
|
|
|
|
interface Dependencies {
|
|
readonly directoryForKubeConfigs: string;
|
|
readonly entityRegistry: CatalogEntityRegistry;
|
|
createCluster: (model: ClusterModel) => Cluster;
|
|
}
|
|
|
|
const kubeConfigSyncName = "lens:kube-sync";
|
|
|
|
export class KubeconfigSyncManager {
|
|
protected readonly sources = observable.map<string, [IComputedValue<CatalogEntity[]>, Disposer]>();
|
|
protected syncing = false;
|
|
protected syncListDisposer?: Disposer;
|
|
|
|
constructor(protected readonly dependencies: Dependencies) {
|
|
makeObservable(this);
|
|
}
|
|
|
|
@action
|
|
startSync(): void {
|
|
if (this.syncing) {
|
|
return;
|
|
}
|
|
|
|
this.syncing = true;
|
|
|
|
logger.info(`${logPrefix} starting requested syncs`);
|
|
|
|
this.dependencies.entityRegistry.addComputedSource(kubeConfigSyncName, computed(() => (
|
|
Array.from(iter.flatMap(
|
|
this.sources.values(),
|
|
([entities]) => entities.get(),
|
|
))
|
|
)));
|
|
|
|
// This must be done so that c&p-ed clusters are visible
|
|
this.startNewSync(this.dependencies.directoryForKubeConfigs);
|
|
|
|
for (const filePath of UserStore.getInstance().syncKubeconfigEntries.keys()) {
|
|
this.startNewSync(filePath);
|
|
}
|
|
|
|
this.syncListDisposer = observe(UserStore.getInstance().syncKubeconfigEntries, change => {
|
|
switch (change.type) {
|
|
case "add":
|
|
this.startNewSync(change.name);
|
|
break;
|
|
case "delete":
|
|
this.stopOldSync(change.name);
|
|
break;
|
|
}
|
|
});
|
|
}
|
|
|
|
@action
|
|
stopSync() {
|
|
this.syncListDisposer?.();
|
|
|
|
for (const filePath of this.sources.keys()) {
|
|
this.stopOldSync(filePath);
|
|
}
|
|
|
|
this.dependencies.entityRegistry.removeSource(kubeConfigSyncName);
|
|
this.syncing = false;
|
|
}
|
|
|
|
@action
|
|
protected startNewSync(filePath: string): void {
|
|
if (this.sources.has(filePath)) {
|
|
// don't start a new sync if we already have one
|
|
return void logger.debug(`${logPrefix} already syncing file/folder`, { filePath });
|
|
}
|
|
|
|
this.sources.set(
|
|
filePath,
|
|
watchFileChanges(filePath, this.dependencies),
|
|
);
|
|
|
|
logger.info(`${logPrefix} starting sync of file/folder`, { filePath });
|
|
logger.debug(`${logPrefix} ${this.sources.size} files/folders watched`, { files: Array.from(this.sources.keys()) });
|
|
}
|
|
|
|
@action
|
|
protected stopOldSync(filePath: string): void {
|
|
if (!this.sources.delete(filePath)) {
|
|
// already stopped
|
|
return void logger.debug(`${logPrefix} no syncing file/folder to stop`, { filePath });
|
|
}
|
|
|
|
logger.info(`${logPrefix} stopping sync of file/folder`, { filePath });
|
|
logger.debug(`${logPrefix} ${this.sources.size} files/folders watched`, { files: Array.from(this.sources.keys()) });
|
|
}
|
|
}
|
|
|
|
// exported for testing
|
|
export function configToModels(rootConfig: KubeConfig, filePath: string): UpdateClusterModel[] {
|
|
const validConfigs = [];
|
|
|
|
for (const { config, error } of splitConfig(rootConfig)) {
|
|
if (error) {
|
|
logger.debug(`${logPrefix} context failed validation: ${error}`, { context: config.currentContext, filePath });
|
|
} else {
|
|
validConfigs.push({
|
|
kubeConfigPath: filePath,
|
|
contextName: config.currentContext,
|
|
});
|
|
}
|
|
}
|
|
|
|
return validConfigs;
|
|
}
|
|
|
|
type RootSourceValue = [Cluster, CatalogEntity];
|
|
type RootSource = ObservableMap<string, RootSourceValue>;
|
|
|
|
// exported for testing
|
|
export const computeDiff = ({ directoryForKubeConfigs, createCluster }: Pick<Dependencies, "createCluster" | "directoryForKubeConfigs">) => (contents: string, source: RootSource, filePath: string): void => {
|
|
runInAction(() => {
|
|
try {
|
|
const { config, error } = loadConfigFromString(contents);
|
|
|
|
if (error) {
|
|
logger.warn(`${logPrefix} encountered errors while loading config: ${error.message}`, { filePath, details: error.details });
|
|
}
|
|
|
|
const rawModels = configToModels(config, filePath);
|
|
const models = new Map(rawModels.map(m => [m.contextName, m]));
|
|
|
|
logger.debug(`${logPrefix} File now has ${models.size} entries`, { filePath });
|
|
|
|
for (const [contextName, value] of source) {
|
|
const model = models.get(contextName);
|
|
|
|
// remove and disconnect clusters that were removed from the config
|
|
if (!model) {
|
|
// remove from the deleting set, so that if a new context of the same name is added, it isn't marked as deleting
|
|
ClusterManager.getInstance().deleting.delete(value[0].id);
|
|
|
|
value[0].disconnect();
|
|
source.delete(contextName);
|
|
logger.debug(`${logPrefix} Removed old cluster from sync`, { filePath, contextName });
|
|
continue;
|
|
}
|
|
|
|
// TODO: For the update check we need to make sure that the config itself hasn't changed.
|
|
// Probably should make it so that cluster keeps a copy of the config in its memory and
|
|
// diff against that
|
|
|
|
// or update the model and mark it as not needed to be added
|
|
value[0].updateModel(model);
|
|
models.delete(contextName);
|
|
logger.debug(`${logPrefix} Updated old cluster from sync`, { filePath, contextName });
|
|
}
|
|
|
|
for (const [contextName, model] of models) {
|
|
// add new clusters to the source
|
|
try {
|
|
const clusterId = createHash("md5").update(`${filePath}:${contextName}`).digest("hex");
|
|
|
|
const cluster = ClusterStore.getInstance().getById(clusterId) || createCluster({ ...model, id: clusterId });
|
|
|
|
if (!cluster.apiUrl) {
|
|
throw new Error("Cluster constructor failed, see above error");
|
|
}
|
|
|
|
const entity = catalogEntityFromCluster(cluster);
|
|
|
|
if (!filePath.startsWith(directoryForKubeConfigs)) {
|
|
entity.metadata.labels.file = filePath.replace(homedir(), "~");
|
|
}
|
|
source.set(contextName, [cluster, entity]);
|
|
|
|
logger.debug(`${logPrefix} Added new cluster from sync`, { filePath, contextName });
|
|
} catch (error) {
|
|
logger.warn(`${logPrefix} Failed to create cluster from model: ${error}`, { filePath, contextName });
|
|
}
|
|
}
|
|
} catch (error) {
|
|
logger.warn(`${logPrefix} Failed to compute diff: ${error}`, { filePath });
|
|
source.clear(); // clear source if we have failed so as to not show outdated information
|
|
}
|
|
});
|
|
};
|
|
|
|
interface DiffChangedConfigArgs {
|
|
filePath: string;
|
|
source: RootSource;
|
|
stats: fs.Stats;
|
|
maxAllowedFileReadSize: number;
|
|
}
|
|
|
|
const diffChangedConfigFor = (dependencies: Dependencies) => ({ filePath, source, stats, maxAllowedFileReadSize }: DiffChangedConfigArgs): Disposer => {
|
|
logger.debug(`${logPrefix} file changed`, { filePath });
|
|
|
|
if (stats.size >= maxAllowedFileReadSize) {
|
|
logger.warn(`${logPrefix} skipping ${filePath}: size=${bytesToUnits(stats.size)} is larger than maxSize=${bytesToUnits(maxAllowedFileReadSize)}`);
|
|
source.clear();
|
|
|
|
return noop;
|
|
}
|
|
|
|
// TODO: replace with an AbortController with fs.readFile when we upgrade to Node 16 (after it comes out)
|
|
const fileReader = fs.createReadStream(filePath, {
|
|
mode: fs.constants.O_RDONLY,
|
|
});
|
|
const readStream: stream.Readable = fileReader;
|
|
const decoder = new TextDecoder("utf-8", { fatal: true });
|
|
let fileString = "";
|
|
let closed = false;
|
|
|
|
const cleanup = () => {
|
|
closed = true;
|
|
fileReader.close(); // This may not close the stream.
|
|
// Artificially marking end-of-stream, as if the underlying resource had
|
|
// indicated end-of-file by itself, allows the stream to close.
|
|
// This does not cancel pending read operations, and if there is such an
|
|
// operation, the process may still not be able to exit successfully
|
|
// until it finishes.
|
|
fileReader.push(null);
|
|
fileReader.read(0);
|
|
readStream.removeAllListeners();
|
|
};
|
|
|
|
readStream
|
|
.on("data", (chunk: Buffer) => {
|
|
try {
|
|
fileString += decoder.decode(chunk, { stream: true });
|
|
} catch (error) {
|
|
logger.warn(`${logPrefix} skipping ${filePath}: ${error}`);
|
|
source.clear();
|
|
cleanup();
|
|
}
|
|
})
|
|
.on("close", () => cleanup())
|
|
.on("error", error => {
|
|
cleanup();
|
|
logger.warn(`${logPrefix} failed to read file: ${error}`, { filePath });
|
|
})
|
|
.on("end", () => {
|
|
if (!closed) {
|
|
computeDiff(dependencies)(fileString, source, filePath);
|
|
}
|
|
});
|
|
|
|
return cleanup;
|
|
};
|
|
|
|
const watchFileChanges = (filePath: string, dependencies: Dependencies): [IComputedValue<CatalogEntity[]>, Disposer] => {
|
|
const rootSource = observable.map<string, ObservableMap<string, RootSourceValue>>();
|
|
const derivedSource = computed(() => Array.from(iter.flatMap(rootSource.values(), from => iter.map(from.values(), child => child[1]))));
|
|
|
|
let watcher: FSWatcher;
|
|
|
|
(async () => {
|
|
try {
|
|
const stat = await fs.promises.stat(filePath);
|
|
const isFolderSync = stat.isDirectory();
|
|
const cleanupFns = new Map<string, Disposer>();
|
|
const maxAllowedFileReadSize = isFolderSync
|
|
? folderSyncMaxAllowedFileReadSize
|
|
: fileSyncMaxAllowedFileReadSize;
|
|
|
|
watcher = watch(filePath, {
|
|
followSymlinks: true,
|
|
depth: isFolderSync ? 0 : 1, // DIRs works with 0 but files need 1 (bug: https://github.com/paulmillr/chokidar/issues/1095)
|
|
disableGlobbing: true,
|
|
ignorePermissionErrors: true,
|
|
usePolling: false,
|
|
awaitWriteFinish: {
|
|
pollInterval: 100,
|
|
stabilityThreshold: 1000,
|
|
},
|
|
atomic: 150, // for "atomic writes"
|
|
alwaysStat: true,
|
|
});
|
|
|
|
const diffChangedConfig = diffChangedConfigFor(dependencies);
|
|
|
|
watcher
|
|
.on("change", (childFilePath, stats: Stats): void => {
|
|
const cleanup = cleanupFns.get(childFilePath);
|
|
|
|
if (!cleanup) {
|
|
// file was previously ignored, do nothing
|
|
return void logger.debug(`${logPrefix} ${inspect(childFilePath)} that should have been previously ignored has changed. Doing nothing`);
|
|
}
|
|
|
|
cleanup();
|
|
cleanupFns.set(childFilePath, diffChangedConfig({
|
|
filePath: childFilePath,
|
|
source: getOrInsertWith(rootSource, childFilePath, observable.map),
|
|
stats,
|
|
maxAllowedFileReadSize,
|
|
}));
|
|
})
|
|
.on("add", (childFilePath, stats: Stats): void => {
|
|
if (isFolderSync) {
|
|
const fileName = path.basename(childFilePath);
|
|
|
|
for (const ignoreGlob of ignoreGlobs) {
|
|
if (ignoreGlob.matcher.test(fileName)) {
|
|
return void logger.info(`${logPrefix} ignoring ${inspect(childFilePath)} due to ignore glob: ${ignoreGlob.rawGlob}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
cleanupFns.set(childFilePath, diffChangedConfig({
|
|
filePath: childFilePath,
|
|
source: getOrInsertWith(rootSource, childFilePath, observable.map),
|
|
stats,
|
|
maxAllowedFileReadSize,
|
|
}));
|
|
})
|
|
.on("unlink", (childFilePath) => {
|
|
cleanupFns.get(childFilePath)?.();
|
|
cleanupFns.delete(childFilePath);
|
|
rootSource.delete(childFilePath);
|
|
})
|
|
.on("error", error => logger.error(`${logPrefix} watching file/folder failed: ${error}`, { filePath }));
|
|
} catch (error) {
|
|
console.log((error as { stack: unknown }).stack);
|
|
logger.warn(`${logPrefix} failed to start watching changes: ${error}`);
|
|
}
|
|
})();
|
|
|
|
return [derivedSource, () => {
|
|
watcher?.close();
|
|
}];
|
|
};
|