mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
Treat default weblinks same as any other weblinks (#3385)
- Use seenWeblink and check for non-arrays - Spread out link validation Signed-off-by: Sebastian Malton <sebastian@malton.name>
This commit is contained in:
parent
0204eef848
commit
da53aa0d79
@ -3,7 +3,7 @@
|
|||||||
"productName": "OpenLens",
|
"productName": "OpenLens",
|
||||||
"description": "OpenLens - Open Source IDE for Kubernetes",
|
"description": "OpenLens - Open Source IDE for Kubernetes",
|
||||||
"homepage": "https://github.com/lensapp/lens",
|
"homepage": "https://github.com/lensapp/lens",
|
||||||
"version": "5.1.3",
|
"version": "5.1.4",
|
||||||
"main": "static/build/main.js",
|
"main": "static/build/main.js",
|
||||||
"copyright": "© 2021 OpenLens Authors",
|
"copyright": "© 2021 OpenLens Authors",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
|
|||||||
@ -24,8 +24,10 @@ import { catalogCategoryRegistry } from "../catalog/catalog-category-registry";
|
|||||||
import { productName } from "../vars";
|
import { productName } from "../vars";
|
||||||
import { WeblinkStore } from "../weblink-store";
|
import { WeblinkStore } from "../weblink-store";
|
||||||
|
|
||||||
|
export type WebLinkStatusPhase = "available" | "unavailable";
|
||||||
|
|
||||||
export interface WebLinkStatus extends CatalogEntityStatus {
|
export interface WebLinkStatus extends CatalogEntityStatus {
|
||||||
phase: "available" | "unavailable";
|
phase: WebLinkStatusPhase;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type WebLinkSpec = {
|
export type WebLinkSpec = {
|
||||||
|
|||||||
@ -21,7 +21,7 @@
|
|||||||
|
|
||||||
import { action, comparer, observable, makeObservable } from "mobx";
|
import { action, comparer, observable, makeObservable } from "mobx";
|
||||||
import { BaseStore } from "./base-store";
|
import { BaseStore } from "./base-store";
|
||||||
import migrations from "../migrations/hotbar-store";
|
import migrations from "../migrations/weblinks-store";
|
||||||
import * as uuid from "uuid";
|
import * as uuid from "uuid";
|
||||||
import { toJS } from "./utils";
|
import { toJS } from "./utils";
|
||||||
|
|
||||||
|
|||||||
@ -19,43 +19,19 @@
|
|||||||
* 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 { observable, reaction } from "mobx";
|
import { computed, observable, reaction } from "mobx";
|
||||||
import { WeblinkStore } from "../../common/weblink-store";
|
import { WeblinkStore } from "../../common/weblink-store";
|
||||||
import { WebLink } from "../../common/catalog-entities";
|
import { WebLink } from "../../common/catalog-entities";
|
||||||
import { catalogEntityRegistry } from "../catalog";
|
import { catalogEntityRegistry } from "../catalog";
|
||||||
import got from "got";
|
import got from "got";
|
||||||
import logger from "../logger";
|
import type { Disposer } from "../../common/utils";
|
||||||
import { docsUrl, slackUrl } from "../../common/vars";
|
import { random } from "lodash";
|
||||||
|
|
||||||
const defaultLinks = [
|
|
||||||
{ title: "Lens Website", url: "https://k8slens.dev" },
|
|
||||||
{ title: "Lens Documentation", url: docsUrl },
|
|
||||||
{ title: "Lens Community Slack", url: slackUrl },
|
|
||||||
{ title: "Kubernetes Documentation", url: "https://kubernetes.io/docs/home/" },
|
|
||||||
{ title: "Lens on Twitter", url: "https://twitter.com/k8slens" },
|
|
||||||
{ title: "Lens Official Blog", url: "https://medium.com/k8slens" }
|
|
||||||
].map((link) => (
|
|
||||||
new WebLink({
|
|
||||||
metadata: {
|
|
||||||
uid: link.url,
|
|
||||||
name: link.title,
|
|
||||||
source: "app",
|
|
||||||
labels: {}
|
|
||||||
},
|
|
||||||
spec: {
|
|
||||||
url: link.url
|
|
||||||
},
|
|
||||||
status: {
|
|
||||||
phase: "available",
|
|
||||||
active: true
|
|
||||||
}
|
|
||||||
})
|
|
||||||
));
|
|
||||||
|
|
||||||
async function validateLink(link: WebLink) {
|
async function validateLink(link: WebLink) {
|
||||||
try {
|
try {
|
||||||
const response = await got.get(link.spec.url, {
|
const response = await got.get(link.spec.url, {
|
||||||
throwHttpErrors: false
|
throwHttpErrors: false,
|
||||||
|
timeout: 20_000,
|
||||||
});
|
});
|
||||||
|
|
||||||
if (response.statusCode >= 200 && response.statusCode < 500) {
|
if (response.statusCode >= 200 && response.statusCode < 500) {
|
||||||
@ -63,7 +39,7 @@ async function validateLink(link: WebLink) {
|
|||||||
} else {
|
} else {
|
||||||
link.status.phase = "unavailable";
|
link.status.phase = "unavailable";
|
||||||
}
|
}
|
||||||
} catch(error) {
|
} catch {
|
||||||
link.status.phase = "unavailable";
|
link.status.phase = "unavailable";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -71,32 +47,60 @@ async function validateLink(link: WebLink) {
|
|||||||
|
|
||||||
export function syncWeblinks() {
|
export function syncWeblinks() {
|
||||||
const weblinkStore = WeblinkStore.getInstance();
|
const weblinkStore = WeblinkStore.getInstance();
|
||||||
const weblinks = observable.array(defaultLinks);
|
const webLinkEntities = observable.map<string, [WebLink, Disposer]>();
|
||||||
|
|
||||||
|
function periodicallyCheckLink(link: WebLink): Disposer {
|
||||||
|
validateLink(link);
|
||||||
|
|
||||||
|
let interval: NodeJS.Timeout;
|
||||||
|
const timeout = setTimeout(() => {
|
||||||
|
interval = setInterval(() => validateLink(link), 60 * 60 * 1000); // every 60 minutes
|
||||||
|
}, random(0, 5 * 60 * 1000, false)); // spread out over 5 minutes
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
clearTimeout(timeout);
|
||||||
|
clearInterval(interval);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
reaction(() => weblinkStore.weblinks, (links) => {
|
reaction(() => weblinkStore.weblinks, (links) => {
|
||||||
weblinks.replace(links.map((data) => new WebLink({
|
const seenWeblinks = new Set<string>();
|
||||||
metadata: {
|
|
||||||
uid: data.id,
|
|
||||||
name: data.name,
|
|
||||||
source: "local",
|
|
||||||
labels: {}
|
|
||||||
},
|
|
||||||
spec: {
|
|
||||||
url: data.url
|
|
||||||
},
|
|
||||||
status: {
|
|
||||||
phase: "available",
|
|
||||||
active: true
|
|
||||||
}
|
|
||||||
})));
|
|
||||||
weblinks.push(...defaultLinks);
|
|
||||||
|
|
||||||
for (const link of weblinks) {
|
for (const weblinkData of links) {
|
||||||
validateLink(link).catch((error) => {
|
seenWeblinks.add(weblinkData.id);
|
||||||
logger.error(`failed to validate link ${link.spec.url}: %s`, error);
|
|
||||||
});
|
if (!webLinkEntities.has(weblinkData.id)) {
|
||||||
|
const link = new WebLink({
|
||||||
|
metadata: {
|
||||||
|
uid: weblinkData.id,
|
||||||
|
name: weblinkData.name,
|
||||||
|
source: "local",
|
||||||
|
labels: {}
|
||||||
|
},
|
||||||
|
spec: {
|
||||||
|
url: weblinkData.url
|
||||||
|
},
|
||||||
|
status: {
|
||||||
|
phase: "available",
|
||||||
|
active: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
webLinkEntities.set(weblinkData.id, [
|
||||||
|
link,
|
||||||
|
periodicallyCheckLink(link),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stop checking and remove weblinks that are no longer in the store
|
||||||
|
for (const [weblinkId, [, disposer]] of webLinkEntities) {
|
||||||
|
if (!seenWeblinks.has(weblinkId)) {
|
||||||
|
disposer();
|
||||||
|
webLinkEntities.delete(weblinkId);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}, {fireImmediately: true});
|
}, {fireImmediately: true});
|
||||||
|
|
||||||
catalogEntityRegistry.addObservableSource("weblinks", weblinks);
|
catalogEntityRegistry.addComputedSource("weblinks", computed(() => Array.from(webLinkEntities.values(), ([link]) => link)));
|
||||||
}
|
}
|
||||||
|
|||||||
43
src/migrations/weblinks-store/5.1.4.ts
Normal file
43
src/migrations/weblinks-store/5.1.4.ts
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
/**
|
||||||
|
* 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 { docsUrl, slackUrl } from "../../common/vars";
|
||||||
|
import type { WeblinkData } from "../../common/weblink-store";
|
||||||
|
import type { MigrationDeclaration } from "../helpers";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
version: "5.1.4",
|
||||||
|
run(store) {
|
||||||
|
const weblinksRaw: any = store.get("weblinks");
|
||||||
|
const weblinks = (Array.isArray(weblinksRaw) ? weblinksRaw : []) as WeblinkData[];
|
||||||
|
|
||||||
|
weblinks.push(
|
||||||
|
{ id: "https://k8slens.dev", name: "Lens Website", url: "https://k8slens.dev" },
|
||||||
|
{ id: docsUrl, name: "Lens Documentation", url: docsUrl },
|
||||||
|
{ id: slackUrl, name: "Lens Community Slack", url: slackUrl },
|
||||||
|
{ id: "https://kubernetes.io/docs/home/", name: "Kubernetes Documentation", url: "https://kubernetes.io/docs/home/" },
|
||||||
|
{ id: "https://twitter.com/k8slens", name: "Lens on Twitter", url: "https://twitter.com/k8slens" },
|
||||||
|
{ id: "https://medium.com/k8slens", name: "Lens Official Blog", url: "https://medium.com/k8slens" }
|
||||||
|
);
|
||||||
|
|
||||||
|
store.set("weblinks", weblinks);
|
||||||
|
}
|
||||||
|
} as MigrationDeclaration;
|
||||||
28
src/migrations/weblinks-store/index.ts
Normal file
28
src/migrations/weblinks-store/index.ts
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
/**
|
||||||
|
* 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 { joinMigrations } from "../helpers";
|
||||||
|
|
||||||
|
import version514 from "./5.1.4";
|
||||||
|
|
||||||
|
export default joinMigrations(
|
||||||
|
version514,
|
||||||
|
);
|
||||||
Loading…
Reference in New Issue
Block a user