2020-07-28 20:06:15 +00:00
|
|
|
import { field, logger } from "@coder/logger"
|
2020-10-16 17:43:49 +00:00
|
|
|
import { Express } from "express"
|
2020-07-28 20:06:15 +00:00
|
|
|
import * as fs from "fs"
|
|
|
|
import * as path from "path"
|
|
|
|
import * as util from "util"
|
|
|
|
import { Args } from "./cli"
|
2020-09-30 20:22:54 +00:00
|
|
|
import { paths } from "./util"
|
2020-07-28 20:06:15 +00:00
|
|
|
|
|
|
|
/* eslint-disable @typescript-eslint/no-var-requires */
|
|
|
|
|
2020-10-16 17:43:49 +00:00
|
|
|
export type Activate = (app: Express, args: Args) => void
|
2020-07-28 20:06:15 +00:00
|
|
|
|
2020-09-30 20:22:54 +00:00
|
|
|
/**
|
|
|
|
* Plugins must implement this interface.
|
|
|
|
*/
|
2020-07-28 20:06:15 +00:00
|
|
|
export interface Plugin {
|
|
|
|
activate: Activate
|
|
|
|
}
|
|
|
|
|
2020-08-05 17:54:58 +00:00
|
|
|
/**
|
|
|
|
* Intercept imports so we can inject code-server when the plugin tries to
|
|
|
|
* import it.
|
|
|
|
*/
|
2020-07-28 20:06:15 +00:00
|
|
|
const originalLoad = require("module")._load
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
require("module")._load = function (request: string, parent: object, isMain: boolean): any {
|
|
|
|
return originalLoad.apply(this, [request.replace(/^code-server/, path.resolve(__dirname, "../..")), parent, isMain])
|
|
|
|
}
|
|
|
|
|
2020-09-30 20:22:54 +00:00
|
|
|
/**
|
|
|
|
* Load a plugin and run its activation function.
|
|
|
|
*/
|
2020-10-16 17:43:49 +00:00
|
|
|
const loadPlugin = async (pluginPath: string, app: Express, args: Args): Promise<void> => {
|
2020-07-28 20:06:15 +00:00
|
|
|
try {
|
|
|
|
const plugin: Plugin = require(pluginPath)
|
2020-10-16 17:43:49 +00:00
|
|
|
plugin.activate(app, args)
|
2020-09-30 20:43:43 +00:00
|
|
|
|
2020-10-07 17:54:40 +00:00
|
|
|
const packageJson = require(path.join(pluginPath, "package.json"))
|
2020-09-30 20:43:43 +00:00
|
|
|
logger.debug(
|
|
|
|
"Loaded plugin",
|
2020-10-07 17:54:40 +00:00
|
|
|
field("name", packageJson.name || path.basename(pluginPath)),
|
|
|
|
field("path", pluginPath),
|
|
|
|
field("version", packageJson.version || "n/a"),
|
2020-09-30 20:43:43 +00:00
|
|
|
)
|
2020-07-28 20:06:15 +00:00
|
|
|
} catch (error) {
|
2020-10-07 17:25:23 +00:00
|
|
|
logger.error(error.message)
|
2020-07-28 20:06:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-30 20:22:54 +00:00
|
|
|
/**
|
|
|
|
* Load all plugins in the specified directory.
|
|
|
|
*/
|
2020-10-16 17:43:49 +00:00
|
|
|
const _loadPlugins = async (pluginDir: string, app: Express, args: Args): Promise<void> => {
|
2020-07-28 20:06:15 +00:00
|
|
|
try {
|
2020-09-30 20:22:54 +00:00
|
|
|
const files = await util.promisify(fs.readdir)(pluginDir, {
|
|
|
|
withFileTypes: true,
|
|
|
|
})
|
2020-10-16 17:43:49 +00:00
|
|
|
await Promise.all(files.map((file) => loadPlugin(path.join(pluginDir, file.name), app, args)))
|
2020-07-28 20:06:15 +00:00
|
|
|
} catch (error) {
|
|
|
|
if (error.code !== "ENOENT") {
|
|
|
|
logger.warn(error.message)
|
|
|
|
}
|
|
|
|
}
|
2020-09-30 20:22:54 +00:00
|
|
|
}
|
2020-07-29 20:02:14 +00:00
|
|
|
|
2020-09-30 20:22:54 +00:00
|
|
|
/**
|
2020-10-07 17:13:12 +00:00
|
|
|
* Load all plugins from the `plugins` directory, directories specified by
|
|
|
|
* `CS_PLUGIN_PATH` (colon-separated), and individual plugins specified by
|
|
|
|
* `CS_PLUGIN` (also colon-separated).
|
2020-09-30 20:22:54 +00:00
|
|
|
*/
|
2020-10-16 17:43:49 +00:00
|
|
|
export const loadPlugins = async (app: Express, args: Args): Promise<void> => {
|
2020-10-08 21:18:00 +00:00
|
|
|
const pluginPath = process.env.CS_PLUGIN_PATH || `${path.join(paths.data, "plugins")}:/usr/share/code-server/plugins`
|
2020-10-07 17:13:12 +00:00
|
|
|
const plugin = process.env.CS_PLUGIN || ""
|
2020-09-30 20:22:54 +00:00
|
|
|
await Promise.all([
|
|
|
|
// Built-in plugins.
|
2020-10-16 17:43:49 +00:00
|
|
|
_loadPlugins(path.resolve(__dirname, "../../plugins"), app, args),
|
2020-09-30 20:22:54 +00:00
|
|
|
// User-added plugins.
|
2020-10-11 06:14:43 +00:00
|
|
|
...pluginPath
|
|
|
|
.split(":")
|
|
|
|
.filter((p) => !!p)
|
2020-10-16 17:43:49 +00:00
|
|
|
.map((dir) => _loadPlugins(path.resolve(dir), app, args)),
|
2020-10-07 17:13:12 +00:00
|
|
|
// Individual plugins so you don't have to symlink or move them into a
|
|
|
|
// directory specifically for plugins. This lets you load plugins that are
|
|
|
|
// on the same level as other directories that are not plugins (if you tried
|
|
|
|
// to use CS_PLUGIN_PATH code-server would try to load those other
|
|
|
|
// directories as plugins). Intended for development.
|
2020-10-11 06:14:43 +00:00
|
|
|
...plugin
|
|
|
|
.split(":")
|
|
|
|
.filter((p) => !!p)
|
2020-10-16 17:43:49 +00:00
|
|
|
.map((dir) => loadPlugin(path.resolve(dir), app, args)),
|
2020-09-30 20:22:54 +00:00
|
|
|
])
|
2020-07-28 20:06:15 +00:00
|
|
|
}
|