mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
- If a Cluster is trying to be connected to but there is no associated entity then don't try and connect Signed-off-by: Sebastian Malton <sebastian@malton.name>
262 lines
8.1 KiB
TypeScript
262 lines
8.1 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import { computed, observable, makeObservable, action } from "mobx";
|
|
import { ipcRendererOn } from "../../../../common/ipc";
|
|
import type { CatalogCategory, CatalogEntity, CatalogEntityData, CatalogCategoryRegistry, CatalogEntityKindData } from "../../../../common/catalog";
|
|
import "../../../../common/catalog-entities";
|
|
import { iter } from "../../../utils";
|
|
import type { Disposer } from "../../../utils";
|
|
import { once } from "lodash";
|
|
import logger from "../../../../common/logger";
|
|
import { CatalogRunEvent } from "../../../../common/catalog/catalog-run-event";
|
|
import { ipcRenderer } from "electron";
|
|
import { catalogInitChannel, catalogItemsChannel, catalogEntityRunListener } from "../../../../common/ipc/catalog";
|
|
import { isMainFrame } from "process";
|
|
import type { Navigate } from "../../../navigation/navigate.injectable";
|
|
|
|
export type EntityFilter = (entity: CatalogEntity) => any;
|
|
export type CatalogEntityOnBeforeRun = (event: CatalogRunEvent) => void | Promise<void>;
|
|
|
|
interface Dependencies {
|
|
navigate: Navigate;
|
|
readonly categoryRegistry: CatalogCategoryRegistry;
|
|
}
|
|
|
|
export class CatalogEntityRegistry {
|
|
protected readonly activeEntityId = observable.box<string | undefined>(undefined);
|
|
protected readonly _entities = observable.map<string, CatalogEntity>([], { deep: true });
|
|
protected readonly filters = observable.set<EntityFilter>([], {
|
|
deep: false,
|
|
});
|
|
protected readonly onBeforeRunHooks = observable.set<CatalogEntityOnBeforeRun>([], {
|
|
deep: false,
|
|
});
|
|
|
|
/**
|
|
* Buffer for keeping entities that don't yet have CatalogCategory synced
|
|
*/
|
|
protected readonly rawEntities: (CatalogEntityData & CatalogEntityKindData)[] = [];
|
|
|
|
constructor(protected readonly dependencies: Dependencies) {
|
|
makeObservable(this);
|
|
}
|
|
|
|
protected getActiveEntityById() {
|
|
const activeEntityId = this.activeEntityId.get();
|
|
|
|
if (!activeEntityId) {
|
|
return undefined;
|
|
}
|
|
|
|
return this._entities.get(activeEntityId);
|
|
}
|
|
|
|
get activeEntity(): CatalogEntity | undefined {
|
|
const entity = this.getActiveEntityById();
|
|
|
|
// If the entity was not found but there are rawEntities to be processed,
|
|
// try to process them and return the entity.
|
|
// This might happen if an extension registered a new Catalog category.
|
|
if (this.activeEntityId.get() && !entity && this.rawEntities.length > 0) {
|
|
this.processRawEntities();
|
|
|
|
return this.getActiveEntityById();
|
|
}
|
|
|
|
return entity;
|
|
}
|
|
|
|
set activeEntity(raw: CatalogEntity | string | undefined) {
|
|
if (raw) {
|
|
const id = typeof raw === "string"
|
|
? raw
|
|
: raw.getId();
|
|
|
|
this.activeEntityId.set(id);
|
|
} else {
|
|
this.activeEntityId.set(undefined);
|
|
}
|
|
}
|
|
|
|
init() {
|
|
ipcRendererOn(catalogItemsChannel, (event, items: (CatalogEntityData & CatalogEntityKindData)[]) => {
|
|
this.updateItems(items);
|
|
});
|
|
|
|
// Make sure that we get items ASAP and not the next time one of them changes
|
|
ipcRenderer.send(catalogInitChannel);
|
|
|
|
if (isMainFrame) {
|
|
ipcRendererOn(catalogEntityRunListener, (event, id: string) => {
|
|
const entity = this.getById(id);
|
|
|
|
if (entity) {
|
|
this.onRun(entity);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
@action updateItems(items: (CatalogEntityData & CatalogEntityKindData)[]) {
|
|
this.rawEntities.length = 0;
|
|
|
|
const newIds = new Set(items.map((item) => item.metadata.uid));
|
|
|
|
for (const uid of this._entities.keys()) {
|
|
if (!newIds.has(uid)) {
|
|
this._entities.delete(uid);
|
|
}
|
|
}
|
|
|
|
for (const item of items) {
|
|
this.updateItem(item);
|
|
}
|
|
}
|
|
|
|
@action protected updateItem(item: (CatalogEntityData & CatalogEntityKindData)) {
|
|
const existing = this._entities.get(item.metadata.uid);
|
|
|
|
if (!existing) {
|
|
const entity = this.dependencies.categoryRegistry.getEntityForData(item);
|
|
|
|
if (entity) {
|
|
this._entities.set(entity.getId(), entity);
|
|
} else {
|
|
this.rawEntities.push(item);
|
|
}
|
|
} else {
|
|
existing.metadata = item.metadata;
|
|
existing.spec = item.spec;
|
|
existing.status = item.status;
|
|
}
|
|
}
|
|
|
|
protected processRawEntities() {
|
|
const items = [...this.rawEntities];
|
|
|
|
this.rawEntities.length = 0;
|
|
|
|
for (const item of items) {
|
|
this.updateItem(item);
|
|
}
|
|
}
|
|
|
|
readonly items = computed(() => {
|
|
this.processRawEntities();
|
|
|
|
return Array.from(this._entities.values());
|
|
});
|
|
|
|
@computed get filteredItems() {
|
|
return Array.from(
|
|
iter.reduce(
|
|
this.filters,
|
|
iter.filter,
|
|
this.items.get().values(),
|
|
),
|
|
);
|
|
}
|
|
|
|
@computed get entities(): Map<string, CatalogEntity> {
|
|
return new Map(
|
|
this.items.get().map(entity => [entity.getId(), entity]),
|
|
);
|
|
}
|
|
|
|
@computed get filteredEntities(): Map<string, CatalogEntity> {
|
|
return new Map(
|
|
this.filteredItems.map(entity => [entity.getId(), entity]),
|
|
);
|
|
}
|
|
|
|
getById(id: string) {
|
|
return this.entities.get(id);
|
|
}
|
|
|
|
getItemsForApiKind<T extends CatalogEntity>(apiVersion: string, kind: string, { filtered = false } = {}): T[] {
|
|
const byApiKind = (item: CatalogEntity) => item.apiVersion === apiVersion && item.kind === kind;
|
|
const entities = filtered ? this.filteredItems : this.items.get();
|
|
|
|
return entities.filter(byApiKind) as T[];
|
|
}
|
|
|
|
getItemsForCategory<T extends CatalogEntity>(category: CatalogCategory, { filtered = false } = {}): T[] {
|
|
const supportedVersions = new Set(category.spec.versions.map((v) => `${category.spec.group}/${v.name}`));
|
|
const byApiVersionKind = (item: CatalogEntity) => supportedVersions.has(item.apiVersion) && item.kind === category.spec.names.kind;
|
|
const entities = filtered ? this.filteredItems : this.items.get();
|
|
|
|
return entities.filter(byApiVersionKind) as T[];
|
|
}
|
|
|
|
/**
|
|
* Add a new filter to the set of item filters
|
|
* @param fn The function that should return a truthy value if that entity should be sent currently "active"
|
|
* @returns A function to remove that filter
|
|
*/
|
|
@action addCatalogFilter(fn: EntityFilter): Disposer {
|
|
this.filters.add(fn);
|
|
|
|
return once(() => void this.filters.delete(fn));
|
|
}
|
|
|
|
/**
|
|
* Add a onBeforeRun hook. If `onBeforeRun` was previously added then it will not be added again
|
|
* @param onBeforeRun The function that should return a boolean if the onRun of catalog entity should be triggered.
|
|
* @returns A function to remove that hook
|
|
*/
|
|
@action addOnBeforeRun(onBeforeRun: CatalogEntityOnBeforeRun): Disposer {
|
|
this.onBeforeRunHooks.add(onBeforeRun);
|
|
|
|
return once(() => void this.onBeforeRunHooks.delete(onBeforeRun));
|
|
}
|
|
|
|
/**
|
|
* Runs all the registered `onBeforeRun` hooks, short circuiting on the first event that's preventDefaulted
|
|
* @param entity The entity to run the hooks on
|
|
* @returns Whether the entities `onRun` method should be executed
|
|
*/
|
|
async onBeforeRun(entity: CatalogEntity): Promise<boolean> {
|
|
logger.debug(`[CATALOG-ENTITY-REGISTRY]: run onBeforeRun on ${entity.getId()}`);
|
|
|
|
const runEvent = new CatalogRunEvent({ target: entity });
|
|
|
|
for (const onBeforeRun of this.onBeforeRunHooks) {
|
|
try {
|
|
await onBeforeRun(runEvent);
|
|
} catch (error) {
|
|
logger.warn(`[CATALOG-ENTITY-REGISTRY]: entity ${entity.getId()} onBeforeRun threw an error`, error);
|
|
}
|
|
|
|
if (runEvent.defaultPrevented) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Perform the onBeforeRun check and, if successful, then proceed to call `entity`'s onRun method
|
|
* @param entity The instance to invoke the hooks and then execute the onRun
|
|
*/
|
|
onRun(entity: CatalogEntity): void {
|
|
this.onBeforeRun(entity)
|
|
.then(doOnRun => {
|
|
if (doOnRun) {
|
|
return entity.onRun?.({
|
|
navigate: this.dependencies.navigate,
|
|
setCommandPaletteContext: (entity) => {
|
|
this.activeEntity = entity;
|
|
},
|
|
});
|
|
} else {
|
|
logger.debug(`onBeforeRun for ${entity.getId()} returned false`);
|
|
}
|
|
})
|
|
.catch(error => logger.error(`[CATALOG-ENTITY-REGISTRY]: entity ${entity.getId()} onRun threw an error`, error));
|
|
}
|
|
}
|