mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
Ignore extensions' clusters when adding to kubeConfigSync (#3313)
This commit is contained in:
parent
621f5ba0b3
commit
9b2031159d
@ -19,10 +19,6 @@
|
|||||||
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Console } from "console";
|
|
||||||
|
|
||||||
console = new Console(process.stdout, process.stderr);
|
|
||||||
|
|
||||||
import mockFs from "mock-fs";
|
import mockFs from "mock-fs";
|
||||||
|
|
||||||
jest.mock("electron", () => {
|
jest.mock("electron", () => {
|
||||||
@ -37,27 +33,27 @@ jest.mock("electron", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
import { UserStore } from "../user-store";
|
import { UserStore } from "../user-store";
|
||||||
|
import { Console } from "console";
|
||||||
import { SemVer } from "semver";
|
import { SemVer } from "semver";
|
||||||
import electron from "electron";
|
import electron from "electron";
|
||||||
import { stdout, stderr } from "process";
|
import { stdout, stderr } from "process";
|
||||||
import { beforeEachWrapped } from "../../../integration/helpers/utils";
|
import { beforeEachWrapped } from "../../../integration/helpers/utils";
|
||||||
import { ThemeStore } from "../../renderer/theme.store";
|
import { ThemeStore } from "../../renderer/theme.store";
|
||||||
|
import type { ClusterStoreModel } from "../cluster-store";
|
||||||
|
|
||||||
console = new Console(stdout, stderr);
|
console = new Console(stdout, stderr);
|
||||||
|
|
||||||
describe("user store tests", () => {
|
describe("user store tests", () => {
|
||||||
describe("for an empty config", () => {
|
describe("for an empty config", () => {
|
||||||
beforeEachWrapped(() => {
|
beforeEachWrapped(() => {
|
||||||
UserStore.resetInstance();
|
|
||||||
mockFs({ tmp: { "config.json": "{}", "kube_config": "{}" } });
|
mockFs({ tmp: { "config.json": "{}", "kube_config": "{}" } });
|
||||||
|
|
||||||
(UserStore.createInstance() as any).refreshNewContexts = jest.fn(() => Promise.resolve());
|
(UserStore.createInstance() as any).refreshNewContexts = jest.fn(() => Promise.resolve());
|
||||||
|
|
||||||
UserStore.getInstance();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
mockFs.restore();
|
mockFs.restore();
|
||||||
|
UserStore.resetInstance();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("allows setting and retrieving lastSeenAppVersion", () => {
|
it("allows setting and retrieving lastSeenAppVersion", () => {
|
||||||
@ -99,14 +95,31 @@ describe("user store tests", () => {
|
|||||||
|
|
||||||
describe("migrations", () => {
|
describe("migrations", () => {
|
||||||
beforeEachWrapped(() => {
|
beforeEachWrapped(() => {
|
||||||
UserStore.resetInstance();
|
|
||||||
mockFs({
|
mockFs({
|
||||||
"tmp": {
|
"tmp": {
|
||||||
"config.json": JSON.stringify({
|
"config.json": JSON.stringify({
|
||||||
user: { username: "foobar" },
|
user: { username: "foobar" },
|
||||||
preferences: { colorTheme: "light" },
|
preferences: { colorTheme: "light" },
|
||||||
lastSeenAppVersion: "1.2.3"
|
lastSeenAppVersion: "1.2.3"
|
||||||
})
|
}),
|
||||||
|
"lens-cluster-store.json": JSON.stringify({
|
||||||
|
clusters: [
|
||||||
|
{
|
||||||
|
id: "foobar",
|
||||||
|
kubeConfigPath: "tmp/extension_data/foo/bar",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "barfoo",
|
||||||
|
kubeConfigPath: "some/other/path",
|
||||||
|
},
|
||||||
|
]
|
||||||
|
} as ClusterStoreModel),
|
||||||
|
"extension_data": {},
|
||||||
|
},
|
||||||
|
"some": {
|
||||||
|
"other": {
|
||||||
|
"path": "is file",
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -114,6 +127,7 @@ describe("user store tests", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
|
UserStore.resetInstance();
|
||||||
mockFs.restore();
|
mockFs.restore();
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -122,5 +136,12 @@ describe("user store tests", () => {
|
|||||||
|
|
||||||
expect(us.lastSeenAppVersion).toBe("0.0.0");
|
expect(us.lastSeenAppVersion).toBe("0.0.0");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it.only("skips clusters for adding to kube-sync with files under extension_data/", () => {
|
||||||
|
const us = UserStore.getInstance();
|
||||||
|
|
||||||
|
expect(us.syncKubeconfigEntries.has("tmp/extension_data/foo/bar")).toBe(false);
|
||||||
|
expect(us.syncKubeconfigEntries.has("some/other/path")).toBe(true);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
130
src/common/utils/__tests__/paths.test.ts
Normal file
130
src/common/utils/__tests__/paths.test.ts
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2021 OpenLens Authors
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
* this software and associated documentation files (the "Software"), to deal in
|
||||||
|
* the Software without restriction, including without limitation the rights to
|
||||||
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||||||
|
* the Software, and to permit persons to whom the Software is furnished to do so,
|
||||||
|
* subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||||||
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||||||
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { describeIf } from "../../../../integration/helpers/utils";
|
||||||
|
import { isWindows } from "../../vars";
|
||||||
|
import { isLogicalChildPath } from "../paths";
|
||||||
|
|
||||||
|
describe("isLogicalChildPath", () => {
|
||||||
|
describeIf(isWindows)("windows tests", () => {
|
||||||
|
it.each([
|
||||||
|
{
|
||||||
|
parentPath: "C:\\Foo",
|
||||||
|
testPath: "C:\\Foo\\Bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "C:\\Foo",
|
||||||
|
testPath: "C:\\Bar",
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "C:\\Foo",
|
||||||
|
testPath: "C:/Bar",
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "C:\\Foo",
|
||||||
|
testPath: "C:/Foo/Bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "C:\\Foo",
|
||||||
|
testPath: "D:\\Foo\\Bar",
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
])("test %#", (testData) => {
|
||||||
|
expect(isLogicalChildPath(testData.parentPath, testData.testPath)).toBe(testData.expected);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describeIf(!isWindows)("posix tests", () => {
|
||||||
|
it.each([
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo",
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/bar",
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foobar",
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo/bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo/../bar",
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo/./bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo/.bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo/..bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo/...bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo/..\\.bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/bar/../foo",
|
||||||
|
testPath: "/foo/bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "/foo/\\bar",
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
parentPath: "/foo",
|
||||||
|
testPath: "./bar",
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
])("test %#", (testData) => {
|
||||||
|
expect(isLogicalChildPath(testData.parentPath, testData.testPath)).toBe(testData.expected);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -33,3 +33,35 @@ function resolveTilde(filePath: string) {
|
|||||||
export function resolvePath(filePath: string): string {
|
export function resolvePath(filePath: string): string {
|
||||||
return path.resolve(resolveTilde(filePath));
|
return path.resolve(resolveTilde(filePath));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if `testPath` represents a potential filesystem entry that would be
|
||||||
|
* logically "within" the `parentPath` directory.
|
||||||
|
*
|
||||||
|
* This function will return `true` in the above case, and `false` otherwise.
|
||||||
|
* It will return `false` if the two paths are the same (after resolving them).
|
||||||
|
*
|
||||||
|
* The function makes no FS calls and is platform dependant. Meaning that the
|
||||||
|
* results are only guaranteed to be correct for the platform you are running
|
||||||
|
* on.
|
||||||
|
* @param parentPath The known path of a directory
|
||||||
|
* @param testPath The path that is to be tested
|
||||||
|
*/
|
||||||
|
export function isLogicalChildPath(parentPath: string, testPath: string): boolean {
|
||||||
|
parentPath = path.resolve(parentPath);
|
||||||
|
testPath = path.resolve(testPath);
|
||||||
|
|
||||||
|
if (parentPath === testPath) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (testPath.length >= parentPath.length) {
|
||||||
|
if (testPath === parentPath) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
testPath = path.dirname(testPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|||||||
@ -20,19 +20,21 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { app } from "electron";
|
import { app } from "electron";
|
||||||
import { existsSync, readJsonSync } from "fs-extra";
|
import { existsSync, readFileSync } from "fs";
|
||||||
import path from "path";
|
import path from "path";
|
||||||
import os from "os";
|
import os from "os";
|
||||||
import { ClusterStore, ClusterStoreModel } from "../../common/cluster-store";
|
import { ClusterStore, ClusterStoreModel } from "../../common/cluster-store";
|
||||||
import type { KubeconfigSyncEntry, UserPreferencesModel } from "../../common/user-store";
|
import type { KubeconfigSyncEntry, UserPreferencesModel } from "../../common/user-store";
|
||||||
import { MigrationDeclaration, migrationLog } from "../helpers";
|
import { MigrationDeclaration, migrationLog } from "../helpers";
|
||||||
|
import { isLogicalChildPath } from "../../common/utils";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
version: "5.0.3-beta.1",
|
version: "5.0.3-beta.1",
|
||||||
run(store) {
|
run(store) {
|
||||||
try {
|
try {
|
||||||
const { syncKubeconfigEntries = [], ...preferences }: UserPreferencesModel = store.get("preferences") ?? {};
|
const { syncKubeconfigEntries = [], ...preferences }: UserPreferencesModel = store.get("preferences") ?? {};
|
||||||
const { clusters = [] }: ClusterStoreModel = readJsonSync(path.resolve(app.getPath("userData"), "lens-cluster-store.json")) ?? {};
|
const { clusters = [] }: ClusterStoreModel = JSON.parse(readFileSync(path.resolve(app.getPath("userData"), "lens-cluster-store.json"), "utf-8")) ?? {};
|
||||||
|
const extensionDataDir = path.resolve(app.getPath("userData"), "extension_data");
|
||||||
const syncPaths = new Set(syncKubeconfigEntries.map(s => s.filePath));
|
const syncPaths = new Set(syncKubeconfigEntries.map(s => s.filePath));
|
||||||
|
|
||||||
syncPaths.add(path.join(os.homedir(), ".kube"));
|
syncPaths.add(path.join(os.homedir(), ".kube"));
|
||||||
@ -50,6 +52,11 @@ export default {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isLogicalChildPath(extensionDataDir, cluster.kubeConfigPath)) {
|
||||||
|
migrationLog(`Skipping ${cluster.id} because kubeConfigPath is placed under an extension_data folder`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (!existsSync(cluster.kubeConfigPath)) {
|
if (!existsSync(cluster.kubeConfigPath)) {
|
||||||
migrationLog(`Skipping ${cluster.id} because kubeConfigPath no longer exists`);
|
migrationLog(`Skipping ${cluster.id} because kubeConfigPath no longer exists`);
|
||||||
continue;
|
continue;
|
||||||
@ -64,8 +71,6 @@ export default {
|
|||||||
migrationLog("Final list of synced paths", updatedSyncEntries);
|
migrationLog("Final list of synced paths", updatedSyncEntries);
|
||||||
store.set("preferences", { ...preferences, syncKubeconfigEntries: updatedSyncEntries });
|
store.set("preferences", { ...preferences, syncKubeconfigEntries: updatedSyncEntries });
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.log(error);
|
|
||||||
|
|
||||||
if (error.code !== "ENOENT") {
|
if (error.code !== "ENOENT") {
|
||||||
// ignore files being missing
|
// ignore files being missing
|
||||||
throw error;
|
throw error;
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user