2019-03-12 20:45:44 +00:00
|
|
|
import { mkdirp } from "fs-extra";
|
2019-01-15 18:36:09 +00:00
|
|
|
import * as os from "os";
|
2019-01-26 00:18:21 +00:00
|
|
|
import { logger, field } from "@coder/logger";
|
2019-03-05 03:26:17 +00:00
|
|
|
import { Pong, ClientMessage, WorkingInitMessage, ServerMessage } from "../proto";
|
2019-01-30 00:48:02 +00:00
|
|
|
import { evaluate, ActiveEvaluation } from "./evaluate";
|
2019-02-22 21:56:29 +00:00
|
|
|
import { ForkProvider } from "../common/helpers";
|
2019-01-11 19:33:44 +00:00
|
|
|
import { ReadWriteConnection } from "../common/connection";
|
|
|
|
|
2019-01-15 18:36:09 +00:00
|
|
|
export interface ServerOptions {
|
|
|
|
readonly workingDirectory: string;
|
|
|
|
readonly dataDirectory: string;
|
2019-03-12 16:12:50 +00:00
|
|
|
readonly cacheDirectory: string;
|
2019-02-05 17:15:20 +00:00
|
|
|
readonly builtInExtensionsDirectory: string;
|
2019-02-22 21:56:29 +00:00
|
|
|
readonly fork?: ForkProvider;
|
2019-01-15 18:36:09 +00:00
|
|
|
}
|
|
|
|
|
2019-01-11 19:33:44 +00:00
|
|
|
export class Server {
|
2019-02-06 17:53:23 +00:00
|
|
|
private readonly evals = new Map<number, ActiveEvaluation>();
|
2019-01-24 00:00:38 +00:00
|
|
|
|
2019-01-11 19:33:44 +00:00
|
|
|
public constructor(
|
|
|
|
private readonly connection: ReadWriteConnection,
|
2019-01-18 23:08:44 +00:00
|
|
|
private readonly options?: ServerOptions,
|
2019-01-11 19:33:44 +00:00
|
|
|
) {
|
|
|
|
connection.onMessage((data) => {
|
|
|
|
try {
|
|
|
|
this.handleMessage(ClientMessage.deserializeBinary(data));
|
|
|
|
} catch (ex) {
|
2019-02-05 17:15:20 +00:00
|
|
|
logger.error("Failed to handle client message", field("length", data.byteLength), field("exception", {
|
|
|
|
message: ex.message,
|
|
|
|
stack: ex.stack,
|
|
|
|
}));
|
2019-01-11 19:33:44 +00:00
|
|
|
}
|
|
|
|
});
|
2019-01-30 00:23:30 +00:00
|
|
|
connection.onClose(() => {
|
2019-02-19 16:17:03 +00:00
|
|
|
this.evals.forEach((e) => e.dispose());
|
2019-01-30 00:23:30 +00:00
|
|
|
});
|
2019-01-15 18:36:09 +00:00
|
|
|
|
2019-02-19 16:17:03 +00:00
|
|
|
if (!this.options) {
|
2019-01-15 18:36:09 +00:00
|
|
|
logger.warn("No server options provided. InitMessage will not be sent.");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-03-12 16:12:50 +00:00
|
|
|
Promise.all([
|
2019-03-12 20:45:44 +00:00
|
|
|
mkdirp(this.options.cacheDirectory),
|
|
|
|
mkdirp(this.options.dataDirectory),
|
|
|
|
mkdirp(this.options.workingDirectory),
|
2019-03-12 16:12:50 +00:00
|
|
|
]).catch((error) => {
|
2019-01-18 21:46:40 +00:00
|
|
|
logger.error(error.message, field("error", error));
|
|
|
|
});
|
|
|
|
|
|
|
|
const initMsg = new WorkingInitMessage();
|
2019-02-19 16:17:03 +00:00
|
|
|
initMsg.setDataDirectory(this.options.dataDirectory);
|
|
|
|
initMsg.setWorkingDirectory(this.options.workingDirectory);
|
|
|
|
initMsg.setBuiltinExtensionsDir(this.options.builtInExtensionsDirectory);
|
2019-01-15 18:36:09 +00:00
|
|
|
initMsg.setHomeDirectory(os.homedir());
|
|
|
|
initMsg.setTmpDirectory(os.tmpdir());
|
|
|
|
const platform = os.platform();
|
2019-01-18 21:46:40 +00:00
|
|
|
let operatingSystem: WorkingInitMessage.OperatingSystem;
|
2019-01-15 18:36:09 +00:00
|
|
|
switch (platform) {
|
|
|
|
case "win32":
|
2019-01-18 21:46:40 +00:00
|
|
|
operatingSystem = WorkingInitMessage.OperatingSystem.WINDOWS;
|
2019-01-15 18:36:09 +00:00
|
|
|
break;
|
|
|
|
case "linux":
|
2019-01-18 21:46:40 +00:00
|
|
|
operatingSystem = WorkingInitMessage.OperatingSystem.LINUX;
|
2019-01-15 18:36:09 +00:00
|
|
|
break;
|
|
|
|
case "darwin":
|
2019-01-18 21:46:40 +00:00
|
|
|
operatingSystem = WorkingInitMessage.OperatingSystem.MAC;
|
2019-01-15 18:36:09 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error(`unrecognized platform "${platform}"`);
|
|
|
|
}
|
|
|
|
initMsg.setOperatingSystem(operatingSystem);
|
2019-02-27 18:43:00 +00:00
|
|
|
initMsg.setShell(os.userInfo().shell || global.process.env.SHELL);
|
2019-01-15 18:36:09 +00:00
|
|
|
const srvMsg = new ServerMessage();
|
|
|
|
srvMsg.setInit(initMsg);
|
|
|
|
connection.send(srvMsg.serializeBinary());
|
2019-01-11 19:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private handleMessage(message: ClientMessage): void {
|
|
|
|
if (message.hasNewEval()) {
|
2019-01-26 00:18:21 +00:00
|
|
|
const evalMessage = message.getNewEval()!;
|
2019-02-19 16:17:03 +00:00
|
|
|
logger.trace(() => [
|
2019-01-30 00:23:30 +00:00
|
|
|
"EvalMessage",
|
|
|
|
field("id", evalMessage.getId()),
|
|
|
|
field("args", evalMessage.getArgsList()),
|
|
|
|
field("function", evalMessage.getFunction()),
|
|
|
|
]);
|
2019-01-30 00:48:02 +00:00
|
|
|
const resp = evaluate(this.connection, evalMessage, () => {
|
|
|
|
this.evals.delete(evalMessage.getId());
|
2019-02-19 16:17:03 +00:00
|
|
|
logger.trace(() => [
|
|
|
|
`dispose ${evalMessage.getId()}, ${this.evals.size} left`,
|
|
|
|
]);
|
2019-02-22 21:56:29 +00:00
|
|
|
}, this.options ? this.options.fork : undefined);
|
2019-01-30 00:48:02 +00:00
|
|
|
if (resp) {
|
|
|
|
this.evals.set(evalMessage.getId(), resp);
|
|
|
|
}
|
|
|
|
} else if (message.hasEvalEvent()) {
|
2019-01-30 21:40:01 +00:00
|
|
|
const evalEventMessage = message.getEvalEvent()!;
|
|
|
|
const e = this.evals.get(evalEventMessage.getId());
|
2019-01-30 00:48:02 +00:00
|
|
|
if (!e) {
|
|
|
|
return;
|
|
|
|
}
|
2019-01-30 21:40:01 +00:00
|
|
|
e.onEvent(evalEventMessage);
|
2019-03-05 03:26:17 +00:00
|
|
|
} else if (message.hasPing()) {
|
|
|
|
logger.trace("ping");
|
|
|
|
const srvMsg = new ServerMessage();
|
|
|
|
srvMsg.setPong(new Pong());
|
|
|
|
this.connection.send(srvMsg.serializeBinary());
|
2019-01-26 00:18:21 +00:00
|
|
|
} else {
|
2019-02-19 16:17:03 +00:00
|
|
|
throw new Error("unknown message type");
|
2019-01-11 19:33:44 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-12 19:44:29 +00:00
|
|
|
}
|