5c19962930
* Avoid spawning code-server with --reuse-window and --new-window These flags mean the user explicitly wants to open in an existing instance so if the socket is down it should error rather than try to spawn code-server normally. * Set session socket into environment variable While I was at it I added a CLI flag to override the default. I also swapped the default to --user-data-dir. The value is set on an environment variable so it can be used by the extension host similar to VSCODE_IPC_HOOK_CLI. * Add e2e test for opening files externally
119 lines
3.4 KiB
TypeScript
119 lines
3.4 KiB
TypeScript
import { logger } from "@coder/logger"
|
|
import * as express from "express"
|
|
import * as fs from "fs"
|
|
import * as path from "path"
|
|
import { HttpCode } from "../../../src/common/http"
|
|
import { AuthType } from "../../../src/node/cli"
|
|
import { codeServer, PluginAPI } from "../../../src/node/plugin"
|
|
import * as apps from "../../../src/node/routes/apps"
|
|
import * as httpserver from "../../utils/httpserver"
|
|
const fsp = fs.promises
|
|
|
|
// Jest overrides `require` so our usual override doesn't work.
|
|
jest.mock("code-server", () => codeServer, { virtual: true })
|
|
|
|
/**
|
|
* Use $LOG_LEVEL=debug to see debug logs.
|
|
*/
|
|
describe("plugin", () => {
|
|
let papi: PluginAPI
|
|
let s: httpserver.HttpServer
|
|
|
|
beforeAll(async () => {
|
|
// Only include the test plugin to avoid contaminating results with other
|
|
// plugins that might be on the filesystem.
|
|
papi = new PluginAPI(logger, `${path.resolve(__dirname, "test-plugin")}:meow`, "")
|
|
await papi.loadPlugins(false)
|
|
|
|
const app = express.default()
|
|
const wsApp = express.default()
|
|
|
|
const common: express.RequestHandler = (req, _, next) => {
|
|
// Routes might use these arguments.
|
|
req.args = {
|
|
_: [],
|
|
auth: AuthType.None,
|
|
host: "localhost",
|
|
port: 8080,
|
|
"proxy-domain": [],
|
|
config: "~/.config/code-server/config.yaml",
|
|
verbose: false,
|
|
"disable-file-downloads": false,
|
|
usingEnvPassword: false,
|
|
usingEnvHashedPassword: false,
|
|
"extensions-dir": "",
|
|
"user-data-dir": "",
|
|
"session-socket": "",
|
|
}
|
|
next()
|
|
}
|
|
|
|
app.use(common)
|
|
wsApp.use(common)
|
|
|
|
papi.mount(app, wsApp)
|
|
app.use("/api/applications", apps.router(papi))
|
|
|
|
s = new httpserver.HttpServer()
|
|
await s.listen(app)
|
|
s.listenUpgrade(wsApp)
|
|
})
|
|
|
|
afterAll(async () => {
|
|
await s.dispose()
|
|
})
|
|
|
|
it("/api/applications", async () => {
|
|
const resp = await s.fetch("/api/applications")
|
|
expect(resp.status).toBe(200)
|
|
const body = await resp.json()
|
|
logger.debug(`${JSON.stringify(body)}`)
|
|
expect(body).toStrictEqual([
|
|
{
|
|
name: "Test App",
|
|
version: "4.0.1",
|
|
|
|
description: "This app does XYZ.",
|
|
iconPath: "/test-plugin/test-app/icon.svg",
|
|
homepageURL: "https://example.com",
|
|
path: "/test-plugin/test-app",
|
|
|
|
plugin: {
|
|
name: "test-plugin",
|
|
version: "1.0.0",
|
|
modulePath: path.join(__dirname, "test-plugin"),
|
|
|
|
displayName: "Test Plugin",
|
|
description: "Plugin used in code-server tests.",
|
|
routerPath: "/test-plugin",
|
|
homepageURL: "https://example.com",
|
|
},
|
|
},
|
|
])
|
|
})
|
|
|
|
it("/test-plugin/test-app", async () => {
|
|
const indexHTML = await fsp.readFile(path.join(__dirname, "test-plugin/public/index.html"), {
|
|
encoding: "utf8",
|
|
})
|
|
const resp = await s.fetch("/test-plugin/test-app")
|
|
expect(resp.status).toBe(200)
|
|
const body = await resp.text()
|
|
expect(body).toBe(indexHTML)
|
|
})
|
|
|
|
it("/test-plugin/test-app (websocket)", async () => {
|
|
const ws = s.ws("/test-plugin/test-app")
|
|
const message = await new Promise((resolve) => {
|
|
ws.once("message", (message) => resolve(message))
|
|
})
|
|
ws.terminate()
|
|
expect(message).toBe("hello")
|
|
})
|
|
|
|
it("/test-plugin/error", async () => {
|
|
const resp = await s.fetch("/test-plugin/error")
|
|
expect(resp.status).toBe(HttpCode.LargePayload)
|
|
})
|
|
})
|