4a80bcb42c
* Add trace log level * Use active eval to implement spdlog * Split server/client active eval interfaces Since all properties are *not* valid on both sides * +200% fire resistance * Implement exec using active evaluations * Fully implement child process streams * Watch impl, move child_process back to explicitly adding events Automatically forwarding all events might be the right move, but wanna think/discuss it a bit more because it didn't come out very cleanly. * Would you like some args with that callback? * Implement the rest of child_process using active evals * Rampant memory leaks Emit "kill" to active evaluations when client disconnects in order to kill processes. Most likely won't be the final solution. * Resolve some minor issues with output panel * Implement node-pty with active evals * Provide clearTimeout to vm sandbox * Implement socket with active evals * Extract some callback logic Also remove some eval interfaces, need to re-think those. * Implement net.Server and remainder of net.Socket using active evals * Implement dispose for active evaluations * Use trace for express requests * Handle sending buffers through evaluation events * Make event logging a bit more clear * Fix some errors due to us not actually instantiating until connect/listen * is this a commit message? * We can just create the evaluator in the ctor Not sure what I was thinking. * memory leak for you, memory leak for everyone * it's a ternary now * Don't dispose automatically on close or error The code may or may not be disposable at that point. * Handle parsing buffers on the client side as well * Remove unused protobuf * Remove TypedValue * Remove unused forkProvider and test * Improve dispose pattern for active evals * Socket calls close after error; no need to bind both * Improve comment * Comment is no longer wishy washy due to explicit boolean * Simplify check for sendHandle and options * Replace _require with __non_webpack_require__ Webpack will then replace this with `require` which we then provide to the vm sandbox. * Provide path.parse * Prevent original-fs from loading * Start with a pid of -1 vscode immediately checks the PID to see if the debug process launch correctly, but of course we don't get the pid synchronously. * Pass arguments to bootstrap-fork * Fully implement streams Was causing errors because internally the stream would set this.writing to true and it would never become false, so subsequent messages would never send. * Fix serializing errors and streams emitting errors multiple times * Was emitting close to data * Fix missing path for spawned processes * Move evaluation onDispose call Now it's accurate and runs when the active evaluation has actually disposed. * Fix promisifying fs.exists * Fix some active eval callback issues * Patch existsSync in debug adapter
371 lines
9.5 KiB
TypeScript
371 lines
9.5 KiB
TypeScript
/// <reference path="../../../../lib/vscode/src/typings/electron.d.ts" />
|
|
import { EventEmitter } from "events";
|
|
import * as fs from "fs";
|
|
import { logger, field } from "@coder/logger";
|
|
import { IKey, Dialog as DialogBox } from "./dialog";
|
|
import { clipboard } from "./clipboard";
|
|
import { client } from "./client";
|
|
|
|
declare var __non_webpack_require__: typeof require;
|
|
|
|
// tslint:disable-next-line no-any
|
|
(global as any).getOpenUrls = (): string[] => {
|
|
return [];
|
|
};
|
|
|
|
// This is required to make the fill load in Node without erroring.
|
|
if (typeof document === "undefined") {
|
|
// tslint:disable-next-line no-any
|
|
(global as any).document = {} as any;
|
|
}
|
|
|
|
const oldCreateElement: <K extends keyof HTMLElementTagNameMap>(
|
|
tagName: K, options?: ElementCreationOptions,
|
|
) => HTMLElementTagNameMap[K] = document.createElement;
|
|
|
|
const newCreateElement = <K extends keyof HTMLElementTagNameMap>(tagName: K): HTMLElementTagNameMap[K] => {
|
|
const createElement = <K extends keyof HTMLElementTagNameMap>(tagName: K): HTMLElementTagNameMap[K] => {
|
|
return oldCreateElement.call(document, tagName);
|
|
};
|
|
|
|
if (tagName === "webview") {
|
|
const view = createElement("iframe") as HTMLIFrameElement;
|
|
view.style.border = "0px";
|
|
const frameID = Math.random().toString();
|
|
view.addEventListener("error", (event) => {
|
|
logger.error("iframe error", field("event", event));
|
|
});
|
|
window.addEventListener("message", (event) => {
|
|
if (!event.data || !event.data.id) {
|
|
return;
|
|
}
|
|
if (event.data.id !== frameID) {
|
|
return;
|
|
}
|
|
const e = new CustomEvent("ipc-message");
|
|
(e as any).channel = event.data.channel; // tslint:disable-line no-any
|
|
(e as any).args = event.data.data; // tslint:disable-line no-any
|
|
view.dispatchEvent(e);
|
|
});
|
|
view.sandbox.add("allow-same-origin", "allow-scripts", "allow-popups", "allow-forms");
|
|
Object.defineProperty(view, "preload", {
|
|
set: (url: string): void => {
|
|
view.onload = (): void => {
|
|
if (view.contentDocument) {
|
|
view.contentDocument.body.id = frameID;
|
|
view.contentDocument.body.parentElement!.style.overflow = "hidden";
|
|
const script = createElement("script");
|
|
script.src = url;
|
|
view.contentDocument.head.appendChild(script);
|
|
}
|
|
};
|
|
},
|
|
});
|
|
(view as any).getWebContents = (): void => undefined; // tslint:disable-line no-any
|
|
(view as any).send = (channel: string, ...args: any[]): void => { // tslint:disable-line no-any
|
|
if (args[0] && typeof args[0] === "object" && args[0].contents) {
|
|
// TODO
|
|
// args[0].contents = (args[0].contents as string).replace(/"(file:\/\/[^"]*)"/g, (m) => `"${getFetchUrl(m)}"`);
|
|
// args[0].contents = (args[0].contents as string).replace(/"vscode-resource:([^"]*)"/g, (m) => `"${getFetchUrl(m)}"`);
|
|
}
|
|
if (view.contentWindow) {
|
|
view.contentWindow.postMessage({
|
|
channel,
|
|
data: args,
|
|
id: frameID,
|
|
}, "*");
|
|
}
|
|
};
|
|
|
|
return view;
|
|
}
|
|
|
|
return createElement(tagName);
|
|
};
|
|
|
|
document.createElement = newCreateElement;
|
|
|
|
class Clipboard {
|
|
public has(): boolean {
|
|
return false;
|
|
}
|
|
|
|
public writeText(value: string): Promise<void> {
|
|
return clipboard.writeText(value);
|
|
}
|
|
}
|
|
|
|
class Shell {
|
|
public async moveItemToTrash(path: string): Promise<void> {
|
|
await client.evaluate((path) => {
|
|
const trash = __non_webpack_require__("trash") as typeof import("trash");
|
|
|
|
return trash(path);
|
|
}, path);
|
|
}
|
|
}
|
|
|
|
class App extends EventEmitter {
|
|
public isAccessibilitySupportEnabled(): boolean {
|
|
return false;
|
|
}
|
|
|
|
public setAsDefaultProtocolClient(): void {
|
|
throw new Error("not implemented");
|
|
}
|
|
}
|
|
|
|
class Dialog {
|
|
public showSaveDialog(_: void, options: Electron.SaveDialogOptions, callback: (filename: string | undefined) => void): void {
|
|
const defaultPath = options.defaultPath || "/untitled";
|
|
const fileIndex = defaultPath.lastIndexOf("/");
|
|
const extensionIndex = defaultPath.lastIndexOf(".");
|
|
const saveDialogOptions = {
|
|
buttons: ["Cancel", "Save"],
|
|
detail: "Enter a path for this file",
|
|
input: {
|
|
value: defaultPath,
|
|
selection: {
|
|
start: fileIndex === -1 ? 0 : fileIndex + 1,
|
|
end: extensionIndex === -1 ? defaultPath.length : extensionIndex,
|
|
},
|
|
},
|
|
message: "Save file",
|
|
};
|
|
|
|
const dialog = new DialogBox(saveDialogOptions);
|
|
dialog.onAction((action) => {
|
|
if (action.key !== IKey.Enter && action.buttonIndex !== 1) {
|
|
dialog.hide();
|
|
|
|
return callback(undefined);
|
|
}
|
|
|
|
const inputValue = dialog.inputValue || "";
|
|
const filePath = inputValue.replace(/\/+$/, "");
|
|
const split = filePath.split("/");
|
|
const fileName = split.pop();
|
|
const parentName = split.pop() || "/";
|
|
if (fileName === "") {
|
|
dialog.error = "You must enter a file name.";
|
|
|
|
return;
|
|
}
|
|
|
|
fs.stat(filePath, (error, stats) => {
|
|
if (error && error.code === "ENOENT") {
|
|
dialog.hide();
|
|
callback(filePath);
|
|
} else if (error) {
|
|
dialog.error = error.message;
|
|
} else if (stats.isDirectory()) {
|
|
dialog.error = `A directory named "${fileName}" already exists.`;
|
|
} else {
|
|
dialog.error = undefined;
|
|
|
|
const confirmDialog = new DialogBox({
|
|
message: `A file named "${fileName}" already exists. Do you want to replace it?`,
|
|
detail: `The file already exists in "${parentName}". Replacing it will overwrite its contents.`,
|
|
buttons: ["Cancel", "Replace"],
|
|
});
|
|
|
|
confirmDialog.onAction((action) => {
|
|
if (action.buttonIndex === 1) {
|
|
confirmDialog.hide();
|
|
|
|
return callback(filePath);
|
|
}
|
|
|
|
confirmDialog.hide();
|
|
dialog.show();
|
|
});
|
|
|
|
dialog.hide();
|
|
confirmDialog.show();
|
|
}
|
|
});
|
|
});
|
|
dialog.show();
|
|
}
|
|
|
|
public showOpenDialog(): void {
|
|
throw new Error("not implemented");
|
|
}
|
|
|
|
public showMessageBox(_: void, options: Electron.MessageBoxOptions, callback: (button: number | undefined, checked: boolean) => void): void {
|
|
const dialog = new DialogBox(options);
|
|
dialog.onAction((action) => {
|
|
dialog.hide();
|
|
callback(action.buttonIndex, false);
|
|
});
|
|
dialog.show();
|
|
}
|
|
}
|
|
|
|
class WebFrame {
|
|
public getZoomFactor(): number {
|
|
return 1;
|
|
}
|
|
|
|
public getZoomLevel(): number {
|
|
return 1;
|
|
}
|
|
|
|
public setZoomLevel(): void {
|
|
// Nothing.
|
|
}
|
|
}
|
|
|
|
class Screen {
|
|
public getAllDisplays(): [] {
|
|
return [];
|
|
}
|
|
}
|
|
|
|
class WebRequest extends EventEmitter {
|
|
public onBeforeRequest(): void {
|
|
throw new Error("not implemented");
|
|
}
|
|
|
|
public onBeforeSendHeaders(): void {
|
|
throw new Error("not implemented");
|
|
}
|
|
|
|
public onHeadersReceived(): void {
|
|
throw new Error("not implemented");
|
|
}
|
|
}
|
|
|
|
class Session extends EventEmitter {
|
|
public webRequest = new WebRequest();
|
|
|
|
public resolveProxy(url: string, callback: (proxy: string) => void): void {
|
|
// TODO: not sure what this actually does.
|
|
callback(url);
|
|
}
|
|
}
|
|
|
|
class WebContents extends EventEmitter {
|
|
public session = new Session();
|
|
}
|
|
|
|
class BrowserWindow extends EventEmitter {
|
|
public webContents = new WebContents();
|
|
private representedFilename: string = "";
|
|
|
|
public static getFocusedWindow(): undefined {
|
|
return undefined;
|
|
}
|
|
|
|
public focus(): void {
|
|
window.focus();
|
|
}
|
|
|
|
public show(): void {
|
|
window.focus();
|
|
}
|
|
|
|
public reload(): void {
|
|
location.reload();
|
|
}
|
|
|
|
public isMaximized(): boolean {
|
|
return false;
|
|
}
|
|
|
|
public setFullScreen(fullscreen: boolean): void {
|
|
if (fullscreen) {
|
|
document.documentElement.requestFullscreen();
|
|
} else {
|
|
document.exitFullscreen();
|
|
}
|
|
}
|
|
|
|
public isFullScreen(): boolean {
|
|
return document.fullscreenEnabled;
|
|
}
|
|
|
|
public isFocused(): boolean {
|
|
return document.hasFocus();
|
|
}
|
|
|
|
public setMenuBarVisibility(): void {
|
|
throw new Error("not implemented");
|
|
}
|
|
|
|
public setAutoHideMenuBar(): void {
|
|
throw new Error("not implemented");
|
|
}
|
|
|
|
public setRepresentedFilename(filename: string): void {
|
|
this.representedFilename = filename;
|
|
}
|
|
|
|
public getRepresentedFilename(): string {
|
|
return this.representedFilename;
|
|
}
|
|
|
|
public setTitle(value: string): void {
|
|
document.title = value;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* We won't be able to do a 1 to 1 fill because things like moveItemToTrash for
|
|
* example returns a boolean while we need a promise.
|
|
*/
|
|
class ElectronFill {
|
|
public readonly shell = new Shell();
|
|
public readonly clipboard = new Clipboard();
|
|
public readonly app = new App();
|
|
public readonly dialog = new Dialog();
|
|
public readonly webFrame = new WebFrame();
|
|
public readonly screen = new Screen();
|
|
|
|
private readonly rendererToMainEmitter = new EventEmitter();
|
|
private readonly mainToRendererEmitter = new EventEmitter();
|
|
|
|
public get BrowserWindow(): typeof BrowserWindow {
|
|
return BrowserWindow;
|
|
}
|
|
|
|
// tslint:disable no-any
|
|
public get ipcRenderer(): object {
|
|
return {
|
|
send: (str: string, ...args: any[]): void => {
|
|
this.rendererToMainEmitter.emit(str, {
|
|
sender: module.exports.ipcMain,
|
|
}, ...args);
|
|
},
|
|
on: (str: string, listener: (...args: any[]) => void): void => {
|
|
this.mainToRendererEmitter.on(str, listener);
|
|
},
|
|
once: (str: string, listener: (...args: any[]) => void): void => {
|
|
this.mainToRendererEmitter.once(str, listener);
|
|
},
|
|
removeListener: (str: string, listener: (...args: any[]) => void): void => {
|
|
this.mainToRendererEmitter.removeListener(str, listener);
|
|
},
|
|
};
|
|
}
|
|
|
|
public get ipcMain(): object {
|
|
return {
|
|
send: (str: string, ...args: any[]): void => {
|
|
this.mainToRendererEmitter.emit(str, {
|
|
sender: module.exports.ipcRenderer,
|
|
}, ...args);
|
|
},
|
|
on: (str: string, listener: (...args: any[]) => void): void => {
|
|
this.rendererToMainEmitter.on(str, listener);
|
|
},
|
|
once: (str: string, listener: (...args: any[]) => void): void => {
|
|
this.rendererToMainEmitter.once(str, listener);
|
|
},
|
|
};
|
|
}
|
|
// tslint:enable no-any
|
|
}
|
|
|
|
module.exports = new ElectronFill();
|