mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* Convert runMany and runManySync to use injectManyWithMeta Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fixup type errors due to new Runnable requirements Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add documentation for verifyRunnablesAreDAG Signed-off-by: Sebastian Malton <sebastian@malton.name> * Simplify convertToWithIdWith Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move all utility functions to separate package Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move testing utilities to separate package Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move run-many and run-many-sync to separate package Signed-off-by: Sebastian Malton <sebastian@malton.name> * Replace all internal uses of utilities with new packages Signed-off-by: Sebastian Malton <sebastian@malton.name> * Use new @k8slens/run-many package in core Signed-off-by: Sebastian Malton <sebastian@malton.name> * Add dep to open-lens Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fixup type errors Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fixup uses of @k8slens/test-utils Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fixup getGlobalOverride Signed-off-by: Sebastian Malton <sebastian@malton.name> * Move tests to new package too Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix type errors Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fixup uses of AsyncResult and autoBind Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fixup remaining import issues Signed-off-by: Sebastian Malton <sebastian@malton.name> * Finial fixups to fix build Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix lint Signed-off-by: Sebastian Malton <sebastian@malton.name> * Revert moving "testUsingFakeTime" to separate package - This fixes tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix integration tests Signed-off-by: Sebastian Malton <sebastian@malton.name> * Fix unit test failing due to spelling fix Signed-off-by: Sebastian Malton <sebastian@malton.name> --------- Signed-off-by: Sebastian Malton <sebastian@malton.name>
95 lines
3.5 KiB
TypeScript
95 lines
3.5 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
import type { DiContainerForInjection, InjectionInstanceWithMeta } from "@ogre-tools/injectable";
|
|
import { getOrInsertSetFor, isDefined } from "@k8slens/utilities";
|
|
import * as uuid from "uuid";
|
|
import assert from "assert";
|
|
import type { Runnable, RunnableSync, RunnableSyncWithId, RunnableWithId } from "./types";
|
|
|
|
const computedNextEdge = (traversed: string[], graph: Map<string, Set<string>>, currentId: string, seenIds: Set<string>) => {
|
|
seenIds.add(currentId);
|
|
const currentNode = graph.get(currentId);
|
|
|
|
assert(currentNode, `Runnable graph does not contain node with id="${currentId}"`);
|
|
|
|
for (const nextId of currentNode.values()) {
|
|
if (traversed.includes(nextId)) {
|
|
throw new Error(`Cycle in runnable graph: "${traversed.join(`" -> "`)}" -> "${nextId}"`);
|
|
}
|
|
|
|
computedNextEdge([...traversed, nextId], graph, nextId, seenIds);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Verifies that the graph produces by `runnables`'s `runAfter`'s is acyclic. Namely that it
|
|
* produces a Directed Acyclic Graph.
|
|
* @param tokenId The ID of the injectionToken that was `injectManyWithMeta()`-ed. Used for error messages
|
|
* @param runnables The list of runnables to check.
|
|
*/
|
|
export function verifyRunnablesAreDAG<Param>(tokenId: string, runnables: RunnableWithId<Param>[]): void;
|
|
export function verifyRunnablesAreDAG<Param>(tokenId: string, runnables: RunnableSyncWithId<Param>[]): void;
|
|
|
|
export function verifyRunnablesAreDAG<Param>(tokenId: string, runnables: (RunnableWithId<Param>[]) | (RunnableSyncWithId<Param>[])): void {
|
|
const rootId = uuid.v4();
|
|
const runnableGraph = new Map<string, Set<string>>();
|
|
const seenIds = new Set<string>();
|
|
const addRunnableId = getOrInsertSetFor(runnableGraph);
|
|
|
|
// Build the Directed graph
|
|
for (const runnable of runnables) {
|
|
addRunnableId(runnable.id);
|
|
|
|
if (runnable.runAfter.length === 0) {
|
|
addRunnableId(rootId).add(runnable.id);
|
|
} else {
|
|
for (const parentRunnable of runnable.runAfter) {
|
|
addRunnableId(parentRunnable.id).add(runnable.id);
|
|
}
|
|
}
|
|
}
|
|
|
|
addRunnableId(rootId);
|
|
|
|
// Do a DFS to find any cycles
|
|
computedNextEdge([], runnableGraph, rootId, seenIds);
|
|
|
|
for (const id of runnableGraph.keys()) {
|
|
if (!seenIds.has(id)) {
|
|
const runnable = runnables.find(runnable => runnable.id === id);
|
|
|
|
if (!runnable) {
|
|
throw new Error(`Runnable "${id}" is not part of the injection token "${tokenId}"`);
|
|
}
|
|
|
|
const runAfters = [runnable.runAfter]
|
|
.flat()
|
|
.filter(isDefined)
|
|
.map(runnable => runnable.id)
|
|
.join('", "');
|
|
|
|
throw new Error(`Runnable "${id}" is unreachable for injection token "${tokenId}": run afters "${runAfters}" are a part of different injection tokens.`);
|
|
}
|
|
}
|
|
}
|
|
|
|
export interface ConvertToWithId {
|
|
<Param>(src: InjectionInstanceWithMeta<Runnable<Param>>): RunnableWithId<Param>;
|
|
<Param>(src: InjectionInstanceWithMeta<RunnableSync<Param>>): RunnableSyncWithId<Param>;
|
|
}
|
|
|
|
export const convertToWithIdWith = (di: DiContainerForInjection) => {
|
|
const convert = <Param>(meta: { id: string }, instance: Runnable<Param>): RunnableWithId<Param> => ({
|
|
id: meta.id,
|
|
run: instance.run,
|
|
runAfter: [instance.runAfter]
|
|
.flat()
|
|
.filter(isDefined)
|
|
.map((injectable) => convert(injectable, di.inject(injectable))),
|
|
});
|
|
|
|
return ((src) => convert(src.meta, src.instance)) as ConvertToWithId;
|
|
};
|