mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* Option to install an extension from filesystem/url #1227 -- part 1 (UI) Signed-off-by: Roman <ixrock@gmail.com> * DropFileInput: common component to handle droped files (replaced also in add-cluster-page) Signed-off-by: Roman <ixrock@gmail.com> * fix: install via url-string on input.submit Signed-off-by: Roman <ixrock@gmail.com> * ui tweaks & minor fixes Signed-off-by: Roman <ixrock@gmail.com> * more ui/ux tweaks & fixes Signed-off-by: Roman <ixrock@gmail.com> * layout fixes Signed-off-by: Roman <ixrock@gmail.com> * component renaming: `copy-to-click` => `copy-to-clipboard` => `clipboard` Signed-off-by: Roman <ixrock@gmail.com> * reworks -- part 1 Signed-off-by: Roman <ixrock@gmail.com> * fix downloading file, added common/utils/downloadFile Signed-off-by: Roman <ixrock@gmail.com> * confirm before install, unpack tar first steps Signed-off-by: Roman <ixrock@gmail.com> * installation flow, extracting .tgz Signed-off-by: Roman <ixrock@gmail.com> * clean up, fix lint issues Signed-off-by: Roman <ixrock@gmail.com> * update .azure-pipelines.yml Signed-off-by: Roman <ixrock@gmail.com> * fixes & refactoring Signed-off-by: Roman <ixrock@gmail.com> * fix lint harder :/ Signed-off-by: Roman <ixrock@gmail.com> * fix validation Signed-off-by: Roman <ixrock@gmail.com> * fix validation harder Signed-off-by: Roman <ixrock@gmail.com> * responding to comments, fixed package validation Signed-off-by: Roman <ixrock@gmail.com> * common/utils/tar.ts: reject with Error-type Signed-off-by: Roman <ixrock@gmail.com> * fix: unit-tests Signed-off-by: Roman <ixrock@gmail.com>
117 lines
3.0 KiB
TypeScript
117 lines
3.0 KiB
TypeScript
import type { InstalledExtension } from "./extension-discovery";
|
|
import { action, observable, reaction } from "mobx";
|
|
import { filesystemProvisionerStore } from "../main/extension-filesystem";
|
|
import logger from "../main/logger";
|
|
|
|
export type LensExtensionId = string; // path to manifest (package.json)
|
|
export type LensExtensionConstructor = new (...args: ConstructorParameters<typeof LensExtension>) => LensExtension;
|
|
|
|
export interface LensExtensionManifest {
|
|
name: string;
|
|
version: string;
|
|
description?: string;
|
|
main?: string; // path to %ext/dist/main.js
|
|
renderer?: string; // path to %ext/dist/renderer.js
|
|
lens?: object; // fixme: add more required fields for validation
|
|
}
|
|
|
|
export class LensExtension {
|
|
readonly manifest: LensExtensionManifest;
|
|
readonly manifestPath: string;
|
|
readonly isBundled: boolean;
|
|
|
|
@observable private isEnabled = false;
|
|
|
|
constructor({ manifest, manifestPath, isBundled }: InstalledExtension) {
|
|
this.manifest = manifest;
|
|
this.manifestPath = manifestPath;
|
|
this.isBundled = !!isBundled;
|
|
}
|
|
|
|
get id(): LensExtensionId {
|
|
// This is the symlinked path under node_modules
|
|
return this.manifestPath;
|
|
}
|
|
|
|
get name() {
|
|
return this.manifest.name;
|
|
}
|
|
|
|
get version() {
|
|
return this.manifest.version;
|
|
}
|
|
|
|
/**
|
|
* getExtensionFileFolder returns the path to an already created folder. This
|
|
* folder is for the sole use of this extension.
|
|
*
|
|
* Note: there is no security done on this folder, only obfiscation of the
|
|
* folder name.
|
|
*/
|
|
async getExtensionFileFolder(): Promise<string> {
|
|
return filesystemProvisionerStore.requestDirectory(this.id);
|
|
}
|
|
|
|
get description() {
|
|
return this.manifest.description;
|
|
}
|
|
|
|
@action
|
|
async enable() {
|
|
if (this.isEnabled) return;
|
|
this.isEnabled = true;
|
|
this.onActivate();
|
|
logger.info(`[EXTENSION]: enabled ${this.name}@${this.version}`);
|
|
}
|
|
|
|
@action
|
|
async disable() {
|
|
if (!this.isEnabled) return;
|
|
this.isEnabled = false;
|
|
this.onDeactivate();
|
|
logger.info(`[EXTENSION]: disabled ${this.name}@${this.version}`);
|
|
}
|
|
|
|
toggle(enable?: boolean) {
|
|
if (typeof enable === "boolean") {
|
|
enable ? this.enable() : this.disable();
|
|
} else {
|
|
this.isEnabled ? this.disable() : this.enable();
|
|
}
|
|
}
|
|
|
|
async whenEnabled(handlers: () => Promise<Function[]>) {
|
|
const disposers: Function[] = [];
|
|
const unregisterHandlers = () => {
|
|
disposers.forEach(unregister => unregister());
|
|
disposers.length = 0;
|
|
};
|
|
const cancelReaction = reaction(() => this.isEnabled, async (isEnabled) => {
|
|
if (isEnabled) {
|
|
const handlerDisposers = await handlers();
|
|
disposers.push(...handlerDisposers);
|
|
} else {
|
|
unregisterHandlers();
|
|
}
|
|
}, {
|
|
fireImmediately: true
|
|
});
|
|
return () => {
|
|
unregisterHandlers();
|
|
cancelReaction();
|
|
};
|
|
}
|
|
|
|
protected onActivate() {
|
|
// mock
|
|
}
|
|
|
|
protected onDeactivate() {
|
|
// mock
|
|
}
|
|
}
|
|
|
|
export function sanitizeExtensionName(name: string) {
|
|
return name.replace("@", "").replace("/", "--");
|
|
}
|