UNPKG

code-server

Version:

Run VS Code on a remote server.

237 lines • 10.8 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || (function () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; Object.defineProperty(exports, "__esModule", { value: true }); exports.ensureVSCodeLoaded = exports.wsRouter = exports.router = void 0; exports.dispose = dispose; const logger_1 = require("@coder/logger"); const crypto = __importStar(require("crypto")); const express = __importStar(require("express")); const fs_1 = require("fs"); const os = __importStar(require("os")); const path = __importStar(require("path")); const util_1 = require("../../common/util"); const cli_1 = require("../cli"); const constants_1 = require("../constants"); const http_1 = require("../http"); const socket_1 = require("../socket"); const util_2 = require("../util"); const wsRouter_1 = require("../wsRouter"); exports.router = express.Router(); exports.wsRouter = (0, wsRouter_1.Router)(); /** * Load then create the VS Code server. */ function loadVSCode(req) { return __awaiter(this, void 0, void 0, function* () { // Since server-main.js is an ES module, we have to use `import`. However, // tsc will transpile this to `require` unless we change our module type, // which will also require that we switch to ESM, since a hybrid approach // breaks importing `rotating-file-stream` for some reason. To work around // this, use `eval` for now, but we should consider switching to ESM. let modPath = path.join(constants_1.vsRootPath, "out/server-main.js"); if (os.platform() === "win32") { // On Windows, absolute paths of ESM modules must be a valid file URI. modPath = "file:///" + modPath.replace(/\\/g, "/"); } const mod = (yield eval(`import("${modPath}")`)); const serverModule = yield mod.loadCodeWithNls(); return serverModule.createServer(null, Object.assign(Object.assign({}, (yield (0, cli_1.toCodeArgs)(req.args))), { "accept-server-license-terms": true, // This seems to be used to make the connection token flags optional (when // set to 1.63) but we have always included them. compatibility: "1.64", "without-connection-token": true })); }); } // To prevent loading the module more than once at a time. We also have the // resolved value so you do not need to `await` everywhere. let vscodeServerPromise; // The resolved value from the dynamically loaded VS Code server. Do not use // without first calling and awaiting `ensureCodeServerLoaded`. let vscodeServer; /** * Ensure the VS Code server is loaded. */ const ensureVSCodeLoaded = (req, _, next) => __awaiter(void 0, void 0, void 0, function* () { if (vscodeServer) { return next(); } if (!vscodeServerPromise) { vscodeServerPromise = loadVSCode(req); } try { vscodeServer = yield vscodeServerPromise; } catch (error) { vscodeServerPromise = undefined; // Unset so we can try again. (0, util_1.logError)(logger_1.logger, "CodeServerRouteWrapper", error); if (constants_1.isDevMode) { return next(new Error((error instanceof Error ? error.message : error) + " (Have you applied the patches? If so, VS Code may still be compiling)")); } return next(error); } return next(); }); exports.ensureVSCodeLoaded = ensureVSCodeLoaded; exports.router.get("/", exports.ensureVSCodeLoaded, (req, res, next) => __awaiter(void 0, void 0, void 0, function* () { const isAuthenticated = yield (0, http_1.authenticated)(req); const NO_FOLDER_OR_WORKSPACE_QUERY = !req.query.folder && !req.query.workspace; // Ew means the workspace was closed so clear the last folder/workspace. const FOLDER_OR_WORKSPACE_WAS_CLOSED = req.query.ew; if (!isAuthenticated) { const to = (0, http_1.self)(req); return (0, http_1.redirect)(req, res, "login", { to: to !== "/" ? to : undefined, }); } if (NO_FOLDER_OR_WORKSPACE_QUERY && !FOLDER_OR_WORKSPACE_WAS_CLOSED) { const settings = yield req.settings.read(); const lastOpened = settings.query || {}; // This flag disables the last opened behavior const IGNORE_LAST_OPENED = req.args["ignore-last-opened"]; const HAS_LAST_OPENED_FOLDER_OR_WORKSPACE = lastOpened.folder || lastOpened.workspace; const HAS_FOLDER_OR_WORKSPACE_FROM_CLI = req.args._.length > 0; const to = (0, http_1.self)(req); let folder = undefined; let workspace = undefined; // Redirect to the last folder/workspace if nothing else is opened. if (HAS_LAST_OPENED_FOLDER_OR_WORKSPACE && !IGNORE_LAST_OPENED) { folder = lastOpened.folder; workspace = lastOpened.workspace; } else if (HAS_FOLDER_OR_WORKSPACE_FROM_CLI) { const lastEntry = path.resolve(req.args._[req.args._.length - 1]); const entryIsFile = yield (0, util_2.isFile)(lastEntry); const IS_WORKSPACE_FILE = entryIsFile && path.extname(lastEntry) === ".code-workspace"; if (IS_WORKSPACE_FILE) { workspace = lastEntry; } else if (!entryIsFile) { folder = lastEntry; } } if (folder || workspace) { return (0, http_1.redirect)(req, res, to, { folder, workspace, }); } } // Store the query parameters so we can use them on the next load. This // also allows users to create functionality around query parameters. yield req.settings.write({ query: req.query }); next(); })); exports.router.get("/manifest.json", (req, res) => __awaiter(void 0, void 0, void 0, function* () { res.writeHead(200, { "Content-Type": "application/manifest+json" }); res.end((0, http_1.replaceTemplates)(req, JSON.stringify({ name: req.args["app-name"], short_name: req.args["app-name"], start_url: ".", display: "fullscreen", display_override: ["window-controls-overlay"], description: "Run Code on a remote server.", icons: [192, 512] .map((size) => [ { src: `{{BASE}}/_static/src/browser/media/pwa-icon-${size}.png`, type: "image/png", sizes: `${size}x${size}`, purpose: "any", }, { src: `{{BASE}}/_static/src/browser/media/pwa-icon-maskable-${size}.png`, type: "image/png", sizes: `${size}x${size}`, purpose: "maskable", }, ]) .flat(), }, null, 2))); })); let mintKeyPromise; exports.router.post("/mint-key", (req, res) => __awaiter(void 0, void 0, void 0, function* () { if (!mintKeyPromise) { mintKeyPromise = new Promise((resolve) => __awaiter(void 0, void 0, void 0, function* () { const keyPath = path.join(req.args["user-data-dir"], "serve-web-key-half"); logger_1.logger.debug(`Reading server web key half from ${keyPath}`); try { resolve(yield fs_1.promises.readFile(keyPath)); return; } catch (error) { if (error.code !== "ENOENT") { (0, util_1.logError)(logger_1.logger, `read ${keyPath}`, error); } } // VS Code wants 256 bits. const key = crypto.randomBytes(32); try { yield fs_1.promises.writeFile(keyPath, key); } catch (error) { (0, util_1.logError)(logger_1.logger, `write ${keyPath}`, error); } resolve(key); })); } const key = yield mintKeyPromise; res.end(key); })); exports.router.all(/.*/, http_1.ensureAuthenticated, exports.ensureVSCodeLoaded, (req, res) => __awaiter(void 0, void 0, void 0, function* () { vscodeServer.handleRequest(req, res); })); const socketProxyProvider = new socket_1.SocketProxyProvider(); exports.wsRouter.ws(/.*/, http_1.ensureOrigin, http_1.ensureAuthenticated, exports.ensureVSCodeLoaded, (req) => __awaiter(void 0, void 0, void 0, function* () { const wrappedSocket = yield socketProxyProvider.createProxy(req.ws); // This should actually accept a duplex stream but it seems Code has not // been updated to match the Node 16 types so cast for now. There does not // appear to be any code specific to sockets so this should be fine. vscodeServer.handleUpgrade(req, wrappedSocket); req.ws.resume(); })); function dispose() { vscodeServer === null || vscodeServer === void 0 ? void 0 : vscodeServer.dispose(); socketProxyProvider.stop(); } //# sourceMappingURL=vscode.js.map