2020-05-10 04:15:29 -04:00
|
|
|
import { field, Level, logger } from "@coder/logger"
|
2021-03-15 16:15:24 -05:00
|
|
|
import { promises as fs } from "fs"
|
2020-05-10 01:19:32 -04:00
|
|
|
import yaml from "js-yaml"
|
2020-05-14 06:08:37 -04:00
|
|
|
import * as os from "os"
|
2020-02-06 13:11:38 -06:00
|
|
|
import * as path from "path"
|
2022-03-02 15:36:38 -07:00
|
|
|
import { canConnect, generateCertificate, generatePassword, humanPath, paths, isNodeJSErrnoException } from "./util"
|
2021-10-29 16:03:57 -07:00
|
|
|
|
|
|
|
const DEFAULT_SOCKET_PATH = path.join(os.tmpdir(), "vscode-ipc")
|
2020-02-06 12:29:19 -06:00
|
|
|
|
2021-05-04 16:46:08 -05:00
|
|
|
export enum Feature {
|
2021-09-29 23:14:56 -04:00
|
|
|
// No current experimental features!
|
|
|
|
Placeholder = "placeholder",
|
2021-05-04 16:46:08 -05:00
|
|
|
}
|
|
|
|
|
2020-10-15 17:00:21 -05:00
|
|
|
export enum AuthType {
|
|
|
|
Password = "password",
|
|
|
|
None = "none",
|
|
|
|
}
|
|
|
|
|
2020-02-06 18:26:07 -06:00
|
|
|
export class Optional<T> {
|
|
|
|
public constructor(public readonly value?: T) {}
|
|
|
|
}
|
|
|
|
|
2020-02-19 11:06:32 -06:00
|
|
|
export enum LogLevel {
|
|
|
|
Trace = "trace",
|
|
|
|
Debug = "debug",
|
|
|
|
Info = "info",
|
|
|
|
Warn = "warn",
|
|
|
|
Error = "error",
|
|
|
|
}
|
|
|
|
|
2020-02-06 18:26:07 -06:00
|
|
|
export class OptionalString extends Optional<string> {}
|
|
|
|
|
2022-03-22 15:07:14 -05:00
|
|
|
/**
|
|
|
|
* Code flags provided by the user.
|
|
|
|
*/
|
|
|
|
export interface UserProvidedCodeArgs {
|
|
|
|
"disable-telemetry"?: boolean
|
|
|
|
force?: boolean
|
|
|
|
"user-data-dir"?: string
|
|
|
|
"enable-proposed-api"?: string[]
|
|
|
|
"extensions-dir"?: string
|
|
|
|
"builtin-extensions-dir"?: string
|
|
|
|
"install-extension"?: string[]
|
|
|
|
"uninstall-extension"?: string[]
|
|
|
|
"list-extensions"?: boolean
|
|
|
|
"locate-extension"?: string[]
|
|
|
|
"show-versions"?: boolean
|
|
|
|
category?: string
|
|
|
|
"github-auth"?: string
|
|
|
|
"disable-update-check"?: boolean
|
2022-04-13 09:39:05 -07:00
|
|
|
"disable-file-downloads"?: boolean
|
2022-03-22 15:07:14 -05:00
|
|
|
}
|
|
|
|
|
2021-11-10 00:28:31 -05:00
|
|
|
/**
|
|
|
|
* Arguments that the user explicitly provided on the command line. All
|
|
|
|
* arguments must be optional.
|
|
|
|
*
|
|
|
|
* For arguments with defaults see DefaultedArgs.
|
|
|
|
*/
|
2022-03-22 15:07:14 -05:00
|
|
|
export interface UserProvidedArgs extends UserProvidedCodeArgs {
|
2020-10-15 16:17:04 -05:00
|
|
|
config?: string
|
|
|
|
auth?: AuthType
|
|
|
|
password?: string
|
2020-12-18 12:20:38 -05:00
|
|
|
"hashed-password"?: string
|
2020-10-15 16:17:04 -05:00
|
|
|
cert?: OptionalString
|
2020-11-12 11:52:02 -06:00
|
|
|
"cert-host"?: string
|
2020-10-15 16:17:04 -05:00
|
|
|
"cert-key"?: string
|
2021-05-04 16:46:08 -05:00
|
|
|
enable?: string[]
|
2020-10-15 16:17:04 -05:00
|
|
|
help?: boolean
|
|
|
|
host?: string
|
2022-01-04 12:37:11 -06:00
|
|
|
locale?: string
|
2021-11-10 00:28:31 -05:00
|
|
|
port?: number
|
2020-10-15 16:17:04 -05:00
|
|
|
json?: boolean
|
2020-02-19 11:06:32 -06:00
|
|
|
log?: LogLevel
|
2020-10-15 16:17:04 -05:00
|
|
|
open?: boolean
|
|
|
|
"bind-addr"?: string
|
|
|
|
socket?: string
|
2022-03-04 00:54:35 +08:00
|
|
|
"socket-mode"?: string
|
2020-10-15 16:17:04 -05:00
|
|
|
version?: boolean
|
|
|
|
"proxy-domain"?: string[]
|
|
|
|
"reuse-window"?: boolean
|
|
|
|
"new-window"?: boolean
|
2021-11-10 00:28:31 -05:00
|
|
|
"ignore-last-opened"?: boolean
|
2020-10-15 16:17:04 -05:00
|
|
|
link?: OptionalString
|
2021-11-10 00:28:31 -05:00
|
|
|
verbose?: boolean
|
|
|
|
/* Positional arguments. */
|
|
|
|
_?: string[]
|
2020-02-06 18:26:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
interface Option<T> {
|
|
|
|
type: T
|
|
|
|
/**
|
|
|
|
* Short flag for the option.
|
|
|
|
*/
|
|
|
|
short?: string
|
|
|
|
/**
|
|
|
|
* Whether the option is a path and should be resolved.
|
|
|
|
*/
|
|
|
|
path?: boolean
|
|
|
|
/**
|
|
|
|
* Description of the option. Leave blank to hide the option.
|
|
|
|
*/
|
|
|
|
description?: string
|
2020-10-09 07:38:38 -04:00
|
|
|
|
|
|
|
/**
|
2021-11-29 12:03:33 -08:00
|
|
|
* If marked as deprecated, the option is marked as deprecated in help.
|
2020-10-09 07:38:38 -04:00
|
|
|
*/
|
2021-11-29 12:03:33 -08:00
|
|
|
deprecated?: boolean
|
2020-02-06 18:26:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
type OptionType<T> = T extends boolean
|
|
|
|
? "boolean"
|
|
|
|
: T extends OptionalString
|
|
|
|
? typeof OptionalString
|
2020-02-19 11:06:32 -06:00
|
|
|
: T extends LogLevel
|
|
|
|
? typeof LogLevel
|
2020-02-06 18:26:07 -06:00
|
|
|
: T extends AuthType
|
|
|
|
? typeof AuthType
|
|
|
|
: T extends number
|
|
|
|
? "number"
|
|
|
|
: T extends string
|
|
|
|
? "string"
|
|
|
|
: T extends string[]
|
|
|
|
? "string[]"
|
|
|
|
: "unknown"
|
|
|
|
|
2022-03-11 13:27:19 -07:00
|
|
|
export type Options<T> = {
|
2020-02-06 18:26:07 -06:00
|
|
|
[P in keyof T]: Option<OptionType<T[P]>>
|
2020-02-06 12:29:19 -06:00
|
|
|
}
|
|
|
|
|
2022-03-11 13:27:19 -07:00
|
|
|
export const options: Options<Required<UserProvidedArgs>> = {
|
2020-02-06 18:26:07 -06:00
|
|
|
auth: { type: AuthType, description: "The type of authentication to use." },
|
2020-05-10 04:15:29 -04:00
|
|
|
password: {
|
|
|
|
type: "string",
|
|
|
|
description: "The password for password authentication (can only be passed in via $PASSWORD or the config file).",
|
|
|
|
},
|
2020-12-18 12:20:38 -05:00
|
|
|
"hashed-password": {
|
2020-12-08 14:54:17 -06:00
|
|
|
type: "string",
|
|
|
|
description:
|
2021-06-02 14:18:54 -07:00
|
|
|
"The password hashed with argon2 for password authentication (can only be passed in via $HASHED_PASSWORD or the config file). \n" +
|
2020-12-08 14:54:17 -06:00
|
|
|
"Takes precedence over 'password'.",
|
|
|
|
},
|
2020-02-06 18:26:07 -06:00
|
|
|
cert: {
|
|
|
|
type: OptionalString,
|
|
|
|
path: true,
|
2020-10-30 05:26:40 -04:00
|
|
|
description: "Path to certificate. A self signed certificate is generated if none is provided.",
|
|
|
|
},
|
|
|
|
"cert-host": {
|
|
|
|
type: "string",
|
|
|
|
description: "Hostname to use when generating a self signed certificate.",
|
2020-02-06 18:26:07 -06:00
|
|
|
},
|
|
|
|
"cert-key": { type: "string", path: true, description: "Path to certificate key when using non-generated cert." },
|
2020-02-18 12:24:12 -06:00
|
|
|
"disable-telemetry": { type: "boolean", description: "Disable telemetry." },
|
2020-11-25 11:37:40 -05:00
|
|
|
"disable-update-check": {
|
|
|
|
type: "boolean",
|
|
|
|
description:
|
|
|
|
"Disable update check. Without this flag, code-server checks every 6 hours against the latest github release and \n" +
|
|
|
|
"then notifies you once every week that a new release is available.",
|
|
|
|
},
|
2022-04-13 09:39:05 -07:00
|
|
|
"disable-file-downloads": {
|
|
|
|
type: "boolean",
|
2022-05-06 16:08:25 -07:00
|
|
|
description:
|
|
|
|
"Disable file downloads from Code. This can also be set with CS_DISABLE_FILE_DOWNLOADS set to 'true' or '1'.",
|
2022-04-13 09:39:05 -07:00
|
|
|
},
|
2021-05-04 16:46:08 -05:00
|
|
|
// --enable can be used to enable experimental features. These features
|
|
|
|
// provide no guarantees.
|
|
|
|
enable: { type: "string[]" },
|
2020-02-06 18:26:07 -06:00
|
|
|
help: { type: "boolean", short: "h", description: "Show this output." },
|
|
|
|
json: { type: "boolean" },
|
2022-01-04 12:37:11 -06:00
|
|
|
locale: { type: "string" }, // The preferred way to set the locale is via the UI.
|
2020-02-19 11:06:32 -06:00
|
|
|
open: { type: "boolean", description: "Open in browser on startup. Does not work remotely." },
|
2020-04-27 09:22:52 -04:00
|
|
|
|
2020-05-10 04:15:29 -04:00
|
|
|
"bind-addr": {
|
|
|
|
type: "string",
|
|
|
|
description: "Address to bind to in host:port. You can also use $PORT to override the port.",
|
|
|
|
},
|
2020-04-27 09:22:52 -04:00
|
|
|
|
2020-05-10 02:35:15 -04:00
|
|
|
config: {
|
|
|
|
type: "string",
|
2020-05-10 04:15:29 -04:00
|
|
|
description: "Path to yaml config file. Every flag maps directly to a key in the config file.",
|
2020-05-10 02:35:15 -04:00
|
|
|
},
|
2020-05-10 01:19:32 -04:00
|
|
|
|
2020-04-27 09:22:52 -04:00
|
|
|
// These two have been deprecated by bindAddr.
|
|
|
|
host: { type: "string", description: "" },
|
|
|
|
port: { type: "number", description: "" },
|
|
|
|
|
2020-04-28 18:29:25 -04:00
|
|
|
socket: { type: "string", path: true, description: "Path to a socket (bind-addr will be ignored)." },
|
2022-03-04 00:54:35 +08:00
|
|
|
"socket-mode": { type: "string", description: "File mode of the socket." },
|
2020-02-06 18:26:07 -06:00
|
|
|
version: { type: "boolean", short: "v", description: "Display version information." },
|
|
|
|
_: { type: "string[]" },
|
|
|
|
|
|
|
|
"user-data-dir": { type: "string", path: true, description: "Path to the user data directory." },
|
|
|
|
"extensions-dir": { type: "string", path: true, description: "Path to the extensions directory." },
|
|
|
|
"builtin-extensions-dir": { type: "string", path: true },
|
2020-03-13 13:21:46 -05:00
|
|
|
"list-extensions": { type: "boolean", description: "List installed VS Code extensions." },
|
|
|
|
force: { type: "boolean", description: "Avoid prompts when installing VS Code extensions." },
|
2021-10-28 13:27:17 -07:00
|
|
|
"locate-extension": { type: "string[]" },
|
2021-11-10 00:28:31 -05:00
|
|
|
category: { type: "string" },
|
2020-08-13 18:08:35 -05:00
|
|
|
"install-extension": {
|
|
|
|
type: "string[]",
|
2020-09-09 00:06:28 -04:00
|
|
|
description:
|
|
|
|
"Install or update a VS Code extension by id or vsix. The identifier of an extension is `${publisher}.${name}`.\n" +
|
|
|
|
"To install a specific version provide `@${version}`. For example: 'vscode.csharp@1.2.3'.",
|
2020-08-13 18:08:35 -05:00
|
|
|
},
|
2020-08-26 14:18:40 -05:00
|
|
|
"enable-proposed-api": {
|
|
|
|
type: "string[]",
|
|
|
|
description:
|
|
|
|
"Enable proposed API features for extensions. Can receive one or more extension IDs to enable individually.",
|
|
|
|
},
|
2020-03-13 13:21:46 -05:00
|
|
|
"uninstall-extension": { type: "string[]", description: "Uninstall a VS Code extension by id." },
|
|
|
|
"show-versions": { type: "boolean", description: "Show VS Code extension versions." },
|
2022-03-02 14:02:51 -06:00
|
|
|
"github-auth": {
|
|
|
|
type: "string",
|
|
|
|
description: "GitHub authentication token (can only be passed in via $GITHUB_TOKEN or the config file).",
|
|
|
|
},
|
2020-03-23 12:08:50 -05:00
|
|
|
"proxy-domain": { type: "string[]", description: "Domain used for proxying ports." },
|
2020-12-15 11:22:58 -05:00
|
|
|
"ignore-last-opened": {
|
|
|
|
type: "boolean",
|
|
|
|
short: "e",
|
2020-12-15 11:14:21 -06:00
|
|
|
description: "Ignore the last opened directory or workspace in favor of an empty window.",
|
2020-12-15 11:22:58 -05:00
|
|
|
},
|
2020-08-27 11:06:21 -07:00
|
|
|
"new-window": {
|
|
|
|
type: "boolean",
|
|
|
|
short: "n",
|
2020-09-15 12:47:33 -05:00
|
|
|
description: "Force to open a new window.",
|
2020-08-27 11:06:21 -07:00
|
|
|
},
|
|
|
|
"reuse-window": {
|
|
|
|
type: "boolean",
|
|
|
|
short: "r",
|
2020-09-15 12:47:33 -05:00
|
|
|
description: "Force to open a file or folder in an already opened window.",
|
2020-08-27 11:06:21 -07:00
|
|
|
},
|
|
|
|
|
2020-02-19 11:06:32 -06:00
|
|
|
log: { type: LogLevel },
|
2020-02-06 18:26:07 -06:00
|
|
|
verbose: { type: "boolean", short: "vvv", description: "Enable verbose logging." },
|
2020-09-08 19:39:17 -04:00
|
|
|
|
2020-10-09 12:57:20 -04:00
|
|
|
link: {
|
2020-10-07 15:54:41 -04:00
|
|
|
type: OptionalString,
|
2020-09-08 20:30:31 -04:00
|
|
|
description: `
|
2021-03-16 22:59:30 -04:00
|
|
|
Securely bind code-server via our cloud service with the passed name. You'll get a URL like
|
2022-02-01 22:15:19 +05:30
|
|
|
https://hostname-username.coder.co at which you can easily access your code-server instance.
|
2020-10-06 21:10:46 -04:00
|
|
|
Authorization is done via GitHub.
|
|
|
|
`,
|
2021-11-29 12:03:33 -08:00
|
|
|
deprecated: true,
|
2020-09-08 20:30:31 -04:00
|
|
|
},
|
2020-02-06 18:26:07 -06:00
|
|
|
}
|
|
|
|
|
2022-03-11 13:27:19 -07:00
|
|
|
export const optionDescriptions = (opts: Partial<Options<Required<UserProvidedArgs>>> = options): string[] => {
|
|
|
|
const entries = Object.entries(opts).filter(([, v]) => !!v.description)
|
2020-02-06 18:26:07 -06:00
|
|
|
const widths = entries.reduce(
|
|
|
|
(prev, [k, v]) => ({
|
|
|
|
long: k.length > prev.long ? k.length : prev.long,
|
|
|
|
short: v.short && v.short.length > prev.short ? v.short.length : prev.short,
|
|
|
|
}),
|
2020-02-14 19:46:00 -05:00
|
|
|
{ short: 0, long: 0 },
|
2020-02-06 18:26:07 -06:00
|
|
|
)
|
2020-12-08 18:38:20 -05:00
|
|
|
return entries.map(([k, v]) => {
|
|
|
|
const help = `${" ".repeat(widths.short - (v.short ? v.short.length : 0))}${v.short ? `-${v.short}` : " "} --${k} `
|
|
|
|
return (
|
|
|
|
help +
|
|
|
|
v.description
|
|
|
|
?.trim()
|
|
|
|
.split(/\n/)
|
|
|
|
.map((line, i) => {
|
|
|
|
line = line.trim()
|
|
|
|
if (i === 0) {
|
2021-11-29 12:03:33 -08:00
|
|
|
return " ".repeat(widths.long - k.length) + (v.deprecated ? "(deprecated) " : "") + line
|
2020-12-08 18:38:20 -05:00
|
|
|
}
|
|
|
|
return " ".repeat(widths.long + widths.short + 6) + line
|
|
|
|
})
|
|
|
|
.join("\n") +
|
|
|
|
(typeof v.type === "object" ? ` [${Object.values(v.type).join(", ")}]` : "")
|
|
|
|
)
|
|
|
|
})
|
2020-02-06 18:26:07 -06:00
|
|
|
}
|
|
|
|
|
2021-06-03 16:30:33 -07:00
|
|
|
export function splitOnFirstEquals(str: string): string[] {
|
|
|
|
// we use regex instead of "=" to ensure we split at the first
|
|
|
|
// "=" and return the following substring with it
|
|
|
|
// important for the hashed-password which looks like this
|
|
|
|
// $argon2i$v=19$m=4096,t=3,p=1$0qR/o+0t00hsbJFQCKSfdQ$oFcM4rL6o+B7oxpuA4qlXubypbBPsf+8L531U7P9HYY
|
|
|
|
// 2 means return two items
|
|
|
|
// Source: https://stackoverflow.com/a/4607799/3015595
|
2021-06-03 16:37:46 -07:00
|
|
|
// We use the ? to say the the substr after the = is optional
|
|
|
|
const split = str.split(/=(.+)?/, 2)
|
2021-06-03 16:30:33 -07:00
|
|
|
|
|
|
|
return split
|
|
|
|
}
|
|
|
|
|
2021-11-10 00:28:31 -05:00
|
|
|
/**
|
|
|
|
* Parse arguments into UserProvidedArgs. This should not go beyond checking
|
|
|
|
* that arguments are valid types and have values when required.
|
|
|
|
*/
|
2020-05-19 00:39:57 -04:00
|
|
|
export const parse = (
|
2020-05-10 04:15:29 -04:00
|
|
|
argv: string[],
|
|
|
|
opts?: {
|
2021-01-14 10:53:58 -05:00
|
|
|
configFile?: string
|
2020-05-10 04:15:29 -04:00
|
|
|
},
|
2021-11-10 00:28:31 -05:00
|
|
|
): UserProvidedArgs => {
|
2020-05-10 04:15:29 -04:00
|
|
|
const error = (msg: string): Error => {
|
|
|
|
if (opts?.configFile) {
|
|
|
|
msg = `error reading ${opts.configFile}: ${msg}`
|
|
|
|
}
|
2021-06-07 15:45:11 -07:00
|
|
|
|
2020-05-10 04:15:29 -04:00
|
|
|
return new Error(msg)
|
|
|
|
}
|
|
|
|
|
2021-11-10 00:28:31 -05:00
|
|
|
const args: UserProvidedArgs = {}
|
2020-02-06 18:26:07 -06:00
|
|
|
let ended = false
|
|
|
|
|
|
|
|
for (let i = 0; i < argv.length; ++i) {
|
|
|
|
const arg = argv[i]
|
|
|
|
|
|
|
|
// -- signals the end of option parsing.
|
2020-08-04 15:08:45 -05:00
|
|
|
if (!ended && arg === "--") {
|
2020-02-06 18:26:07 -06:00
|
|
|
ended = true
|
|
|
|
continue
|
|
|
|
}
|
2020-02-06 13:11:38 -06:00
|
|
|
|
2020-02-06 18:26:07 -06:00
|
|
|
// Options start with a dash and require a value if non-boolean.
|
|
|
|
if (!ended && arg.startsWith("-")) {
|
2021-11-10 00:28:31 -05:00
|
|
|
let key: keyof UserProvidedArgs | undefined
|
2020-02-19 10:54:23 -06:00
|
|
|
let value: string | undefined
|
2020-02-06 18:26:07 -06:00
|
|
|
if (arg.startsWith("--")) {
|
2021-06-03 16:37:46 -07:00
|
|
|
const split = splitOnFirstEquals(arg.replace(/^--/, ""))
|
2021-11-10 00:28:31 -05:00
|
|
|
key = split[0] as keyof UserProvidedArgs
|
2020-02-19 10:54:23 -06:00
|
|
|
value = split[1]
|
2021-06-07 15:45:11 -07:00
|
|
|
} else {
|
|
|
|
const short = arg.replace(/^-/, "")
|
|
|
|
const pair = Object.entries(options).find(([, v]) => v.short === short)
|
|
|
|
if (pair) {
|
2021-11-10 00:28:31 -05:00
|
|
|
key = pair[0] as keyof UserProvidedArgs
|
2021-06-07 15:45:11 -07:00
|
|
|
}
|
|
|
|
}
|
2020-02-06 18:26:07 -06:00
|
|
|
|
|
|
|
if (!key || !options[key]) {
|
2020-05-10 04:15:29 -04:00
|
|
|
throw error(`Unknown option ${arg}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (key === "password" && !opts?.configFile) {
|
|
|
|
throw new Error("--password can only be set in the config file or passed in via $PASSWORD")
|
2020-02-06 18:26:07 -06:00
|
|
|
}
|
|
|
|
|
2020-12-18 12:20:38 -05:00
|
|
|
if (key === "hashed-password" && !opts?.configFile) {
|
|
|
|
throw new Error("--hashed-password can only be set in the config file or passed in via $HASHED_PASSWORD")
|
2020-12-08 14:54:17 -06:00
|
|
|
}
|
|
|
|
|
2022-03-02 14:02:51 -06:00
|
|
|
if (key === "github-auth" && !opts?.configFile) {
|
|
|
|
throw new Error("--github-auth can only be set in the config file or passed in via $GITHUB_TOKEN")
|
|
|
|
}
|
|
|
|
|
2020-02-06 18:26:07 -06:00
|
|
|
const option = options[key]
|
|
|
|
if (option.type === "boolean") {
|
|
|
|
;(args[key] as boolean) = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-02-19 10:54:23 -06:00
|
|
|
// Might already have a value if it was the --long=value format.
|
|
|
|
if (typeof value === "undefined") {
|
|
|
|
// A value is only valid if it doesn't look like an option.
|
|
|
|
value = argv[i + 1] && !argv[i + 1].startsWith("-") ? argv[++i] : undefined
|
|
|
|
}
|
|
|
|
|
2020-02-06 18:26:07 -06:00
|
|
|
if (!value && option.type === OptionalString) {
|
|
|
|
;(args[key] as OptionalString) = new OptionalString(value)
|
|
|
|
continue
|
|
|
|
} else if (!value) {
|
2020-05-10 04:15:29 -04:00
|
|
|
throw error(`--${key} requires a value`)
|
|
|
|
}
|
|
|
|
|
2020-08-04 15:08:45 -05:00
|
|
|
if (option.type === OptionalString && value === "false") {
|
2020-05-10 04:15:29 -04:00
|
|
|
continue
|
2020-02-06 18:26:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (option.path) {
|
|
|
|
value = path.resolve(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (option.type) {
|
|
|
|
case "string":
|
|
|
|
;(args[key] as string) = value
|
|
|
|
break
|
|
|
|
case "string[]":
|
|
|
|
if (!args[key]) {
|
|
|
|
;(args[key] as string[]) = []
|
|
|
|
}
|
|
|
|
;(args[key] as string[]).push(value)
|
|
|
|
break
|
|
|
|
case "number":
|
|
|
|
;(args[key] as number) = parseInt(value, 10)
|
|
|
|
if (isNaN(args[key] as number)) {
|
2020-05-10 04:15:29 -04:00
|
|
|
throw error(`--${key} must be a number`)
|
2020-02-06 18:26:07 -06:00
|
|
|
}
|
|
|
|
break
|
|
|
|
case OptionalString:
|
|
|
|
;(args[key] as OptionalString) = new OptionalString(value)
|
|
|
|
break
|
|
|
|
default: {
|
2020-04-28 16:39:01 -05:00
|
|
|
if (!Object.values(option.type).includes(value)) {
|
2020-05-10 04:15:29 -04:00
|
|
|
throw error(`--${key} valid values: [${Object.values(option.type).join(", ")}]`)
|
2020-02-06 18:26:07 -06:00
|
|
|
}
|
|
|
|
;(args[key] as string) = value
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Everything else goes into _.
|
2021-11-10 00:28:31 -05:00
|
|
|
if (typeof args._ === "undefined") {
|
|
|
|
args._ = []
|
|
|
|
}
|
|
|
|
|
2020-02-06 18:26:07 -06:00
|
|
|
args._.push(arg)
|
2020-02-06 13:11:38 -06:00
|
|
|
}
|
|
|
|
|
2020-10-15 16:17:04 -05:00
|
|
|
// If a cert was provided a key must also be provided.
|
|
|
|
if (args.cert && args.cert.value && !args["cert-key"]) {
|
|
|
|
throw new Error("--cert-key is missing")
|
|
|
|
}
|
|
|
|
|
2021-12-10 12:01:35 -06:00
|
|
|
logger.debug(() => [
|
|
|
|
`parsed ${opts?.configFile ? "config" : "command line"}`,
|
2022-03-02 14:02:51 -06:00
|
|
|
field("args", {
|
|
|
|
...args,
|
|
|
|
password: args.password ? "<redacted>" : undefined,
|
|
|
|
"hashed-password": args["hashed-password"] ? "<redacted>" : undefined,
|
|
|
|
"github-auth": args["github-auth"] ? "<redacted>" : undefined,
|
|
|
|
}),
|
2021-12-10 12:01:35 -06:00
|
|
|
])
|
2020-02-06 18:26:07 -06:00
|
|
|
|
2020-09-15 13:43:07 -05:00
|
|
|
return args
|
|
|
|
}
|
|
|
|
|
2021-11-10 00:28:31 -05:00
|
|
|
/**
|
|
|
|
* User-provided arguments with defaults. The distinction between user-provided
|
|
|
|
* args and defaulted args exists so we can tell the difference between end
|
|
|
|
* values and what the user actually provided on the command line.
|
|
|
|
*/
|
2020-10-15 16:17:04 -05:00
|
|
|
export interface DefaultedArgs extends ConfigArgs {
|
|
|
|
auth: AuthType
|
|
|
|
cert?: {
|
|
|
|
value: string
|
|
|
|
}
|
|
|
|
host: string
|
|
|
|
port: number
|
|
|
|
"proxy-domain": string[]
|
|
|
|
verbose: boolean
|
|
|
|
usingEnvPassword: boolean
|
2020-12-08 14:54:17 -06:00
|
|
|
usingEnvHashedPassword: boolean
|
2020-10-15 16:17:04 -05:00
|
|
|
"extensions-dir": string
|
|
|
|
"user-data-dir": string
|
2021-11-10 00:28:31 -05:00
|
|
|
/* Positional arguments. */
|
2022-03-02 15:36:38 -07:00
|
|
|
_: string[]
|
2020-10-15 16:17:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Take CLI and config arguments (optional) and return a single set of arguments
|
|
|
|
* with the defaults set. Arguments from the CLI are prioritized over config
|
|
|
|
* arguments.
|
|
|
|
*/
|
2021-11-10 00:28:31 -05:00
|
|
|
export async function setDefaults(cliArgs: UserProvidedArgs, configArgs?: ConfigArgs): Promise<DefaultedArgs> {
|
2020-10-15 16:17:04 -05:00
|
|
|
const args = Object.assign({}, configArgs || {}, cliArgs)
|
2020-09-15 13:43:07 -05:00
|
|
|
|
|
|
|
if (!args["user-data-dir"]) {
|
|
|
|
args["user-data-dir"] = paths.data
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!args["extensions-dir"]) {
|
|
|
|
args["extensions-dir"] = path.join(args["user-data-dir"], "extensions")
|
|
|
|
}
|
|
|
|
|
2020-04-28 16:39:01 -05:00
|
|
|
// --verbose takes priority over --log and --log takes priority over the
|
|
|
|
// environment variable.
|
|
|
|
if (args.verbose) {
|
|
|
|
args.log = LogLevel.Trace
|
|
|
|
} else if (
|
|
|
|
!args.log &&
|
|
|
|
process.env.LOG_LEVEL &&
|
|
|
|
Object.values(LogLevel).includes(process.env.LOG_LEVEL as LogLevel)
|
|
|
|
) {
|
2020-02-19 11:06:32 -06:00
|
|
|
args.log = process.env.LOG_LEVEL as LogLevel
|
2020-02-06 12:29:19 -06:00
|
|
|
}
|
2020-02-06 18:26:07 -06:00
|
|
|
|
2020-04-28 16:39:01 -05:00
|
|
|
// Sync --log, --verbose, the environment variable, and logger level.
|
|
|
|
if (args.log) {
|
|
|
|
process.env.LOG_LEVEL = args.log
|
|
|
|
}
|
2020-02-06 18:26:07 -06:00
|
|
|
switch (args.log) {
|
2020-02-19 11:06:32 -06:00
|
|
|
case LogLevel.Trace:
|
2020-02-06 18:26:07 -06:00
|
|
|
logger.level = Level.Trace
|
2020-04-28 16:39:01 -05:00
|
|
|
args.verbose = true
|
2020-02-06 18:26:07 -06:00
|
|
|
break
|
2020-02-19 11:06:32 -06:00
|
|
|
case LogLevel.Debug:
|
2020-02-06 18:26:07 -06:00
|
|
|
logger.level = Level.Debug
|
2020-05-19 00:39:57 -04:00
|
|
|
args.verbose = false
|
2020-02-06 18:26:07 -06:00
|
|
|
break
|
2020-02-19 11:06:32 -06:00
|
|
|
case LogLevel.Info:
|
2020-02-06 18:26:07 -06:00
|
|
|
logger.level = Level.Info
|
2020-05-19 00:39:57 -04:00
|
|
|
args.verbose = false
|
2020-02-06 18:26:07 -06:00
|
|
|
break
|
2020-02-19 11:06:32 -06:00
|
|
|
case LogLevel.Warn:
|
2020-02-06 18:26:07 -06:00
|
|
|
logger.level = Level.Warning
|
2020-05-19 00:39:57 -04:00
|
|
|
args.verbose = false
|
2020-02-06 18:26:07 -06:00
|
|
|
break
|
2020-02-19 11:06:32 -06:00
|
|
|
case LogLevel.Error:
|
2020-02-06 18:26:07 -06:00
|
|
|
logger.level = Level.Error
|
2020-05-19 00:39:57 -04:00
|
|
|
args.verbose = false
|
2020-02-06 18:26:07 -06:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2020-10-15 16:17:04 -05:00
|
|
|
// Default to using a password.
|
|
|
|
if (!args.auth) {
|
|
|
|
args.auth = AuthType.Password
|
|
|
|
}
|
|
|
|
|
2021-11-10 00:28:31 -05:00
|
|
|
const addr = bindAddrFromAllSources(configArgs || {}, cliArgs)
|
2020-11-03 14:26:25 -06:00
|
|
|
args.host = addr.host
|
|
|
|
args.port = addr.port
|
2020-10-15 16:17:04 -05:00
|
|
|
|
|
|
|
// If we're being exposed to the cloud, we listen on a random address and
|
|
|
|
// disable auth.
|
|
|
|
if (args.link) {
|
|
|
|
args.host = "localhost"
|
|
|
|
args.port = 0
|
|
|
|
args.socket = undefined
|
2022-03-04 00:54:35 +08:00
|
|
|
args["socket-mode"] = undefined
|
2020-10-15 16:17:04 -05:00
|
|
|
args.cert = undefined
|
2020-11-03 14:30:34 -06:00
|
|
|
args.auth = AuthType.None
|
2020-10-15 16:17:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (args.cert && !args.cert.value) {
|
2020-11-12 11:52:02 -06:00
|
|
|
const { cert, certKey } = await generateCertificate(args["cert-host"] || "localhost")
|
2020-10-15 16:17:04 -05:00
|
|
|
args.cert = {
|
|
|
|
value: cert,
|
|
|
|
}
|
|
|
|
args["cert-key"] = certKey
|
|
|
|
}
|
|
|
|
|
2020-12-08 14:54:17 -06:00
|
|
|
let usingEnvPassword = !!process.env.PASSWORD
|
2020-10-15 16:17:04 -05:00
|
|
|
if (process.env.PASSWORD) {
|
|
|
|
args.password = process.env.PASSWORD
|
|
|
|
}
|
|
|
|
|
2022-04-26 15:09:53 -05:00
|
|
|
if (process.env.CS_DISABLE_FILE_DOWNLOADS?.match(/^(1|true)$/)) {
|
2022-04-13 09:39:05 -07:00
|
|
|
args["disable-file-downloads"] = true
|
|
|
|
}
|
|
|
|
|
2020-12-08 14:54:17 -06:00
|
|
|
const usingEnvHashedPassword = !!process.env.HASHED_PASSWORD
|
|
|
|
if (process.env.HASHED_PASSWORD) {
|
2020-12-18 12:20:38 -05:00
|
|
|
args["hashed-password"] = process.env.HASHED_PASSWORD
|
2020-12-08 14:54:17 -06:00
|
|
|
usingEnvPassword = false
|
|
|
|
}
|
|
|
|
|
2022-03-02 14:02:51 -06:00
|
|
|
if (process.env.GITHUB_TOKEN) {
|
|
|
|
args["github-auth"] = process.env.GITHUB_TOKEN
|
|
|
|
}
|
|
|
|
|
2020-12-08 14:54:17 -06:00
|
|
|
// Ensure they're not readable by child processes.
|
2020-10-15 16:17:04 -05:00
|
|
|
delete process.env.PASSWORD
|
2020-12-08 14:54:17 -06:00
|
|
|
delete process.env.HASHED_PASSWORD
|
2022-03-02 14:02:51 -06:00
|
|
|
delete process.env.GITHUB_TOKEN
|
2020-10-15 16:17:04 -05:00
|
|
|
|
|
|
|
// Filter duplicate proxy domains and remove any leading `*.`.
|
|
|
|
const proxyDomains = new Set((args["proxy-domain"] || []).map((d) => d.replace(/^\*\./, "")))
|
|
|
|
args["proxy-domain"] = Array.from(proxyDomains)
|
|
|
|
|
2021-11-10 00:28:31 -05:00
|
|
|
if (typeof args._ === "undefined") {
|
|
|
|
args._ = []
|
|
|
|
}
|
|
|
|
|
2020-10-15 16:17:04 -05:00
|
|
|
return {
|
|
|
|
...args,
|
|
|
|
usingEnvPassword,
|
2020-12-08 14:54:17 -06:00
|
|
|
usingEnvHashedPassword,
|
2020-10-15 16:17:04 -05:00
|
|
|
} as DefaultedArgs // TODO: Technically no guarantee this is fulfilled.
|
2020-05-19 00:39:57 -04:00
|
|
|
}
|
|
|
|
|
2021-09-28 15:38:59 -07:00
|
|
|
/**
|
|
|
|
* Helper function to return the default config file.
|
|
|
|
*
|
2021-09-28 15:51:00 -07:00
|
|
|
* @param {string} password - Password passed in (usually from generatePassword())
|
2021-09-28 15:38:59 -07:00
|
|
|
* @returns The default config file:
|
|
|
|
*
|
|
|
|
* - bind-addr: 127.0.0.1:8080
|
|
|
|
* - auth: password
|
2021-09-28 15:51:00 -07:00
|
|
|
* - password: <password>
|
2021-09-28 15:38:59 -07:00
|
|
|
* - cert: false
|
|
|
|
*/
|
2021-09-28 15:51:00 -07:00
|
|
|
export function defaultConfigFile(password: string): string {
|
2020-05-10 04:15:29 -04:00
|
|
|
return `bind-addr: 127.0.0.1:8080
|
2020-05-10 01:35:42 -04:00
|
|
|
auth: password
|
2021-09-28 15:51:00 -07:00
|
|
|
password: ${password}
|
2020-05-10 04:15:29 -04:00
|
|
|
cert: false
|
|
|
|
`
|
|
|
|
}
|
|
|
|
|
2021-11-10 00:28:31 -05:00
|
|
|
interface ConfigArgs extends UserProvidedArgs {
|
2020-10-15 16:17:04 -05:00
|
|
|
config: string
|
|
|
|
}
|
|
|
|
|
2020-05-10 04:15:29 -04:00
|
|
|
/**
|
|
|
|
* Reads the code-server yaml config file and returns it as Args.
|
|
|
|
*
|
|
|
|
* @param configPath Read the config from configPath instead of $CODE_SERVER_CONFIG or the default.
|
|
|
|
*/
|
2020-10-15 16:17:04 -05:00
|
|
|
export async function readConfigFile(configPath?: string): Promise<ConfigArgs> {
|
2020-05-10 04:15:29 -04:00
|
|
|
if (!configPath) {
|
|
|
|
configPath = process.env.CODE_SERVER_CONFIG
|
|
|
|
if (!configPath) {
|
|
|
|
configPath = path.join(paths.config, "config.yaml")
|
|
|
|
}
|
|
|
|
}
|
2020-05-10 01:19:32 -04:00
|
|
|
|
2021-03-15 16:15:24 -05:00
|
|
|
await fs.mkdir(path.dirname(configPath), { recursive: true })
|
|
|
|
|
|
|
|
try {
|
2021-09-28 15:51:00 -07:00
|
|
|
const generatedPassword = await generatePassword()
|
|
|
|
await fs.writeFile(configPath, defaultConfigFile(generatedPassword), {
|
2021-03-15 16:15:24 -05:00
|
|
|
flag: "wx", // wx means to fail if the path exists.
|
|
|
|
})
|
2021-11-15 19:40:34 +00:00
|
|
|
logger.info(`Wrote default config file to ${humanPath(os.homedir(), configPath)}`)
|
2021-09-29 23:14:56 -04:00
|
|
|
} catch (error: any) {
|
2021-03-15 16:15:24 -05:00
|
|
|
// EEXIST is fine; we don't want to overwrite existing configurations.
|
|
|
|
if (error.code !== "EEXIST") {
|
|
|
|
throw error
|
|
|
|
}
|
2020-05-10 01:19:32 -04:00
|
|
|
}
|
|
|
|
|
2021-03-15 16:15:24 -05:00
|
|
|
const configFile = await fs.readFile(configPath, "utf8")
|
|
|
|
return parseConfigFile(configFile, configPath)
|
2021-01-14 10:53:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* parseConfigFile parses configFile into ConfigArgs.
|
|
|
|
* configPath is used as the filename in error messages
|
|
|
|
*/
|
|
|
|
export function parseConfigFile(configFile: string, configPath: string): ConfigArgs {
|
2021-01-14 10:55:19 -05:00
|
|
|
if (!configFile) {
|
2021-11-10 00:28:31 -05:00
|
|
|
return { config: configPath }
|
2021-01-14 10:53:58 -05:00
|
|
|
}
|
|
|
|
|
2021-03-18 16:30:10 -05:00
|
|
|
const config = yaml.load(configFile, {
|
2020-05-10 04:15:29 -04:00
|
|
|
filename: configPath,
|
2020-05-10 01:19:32 -04:00
|
|
|
})
|
2020-08-26 14:21:37 -04:00
|
|
|
if (!config || typeof config === "string") {
|
|
|
|
throw new Error(`invalid config: ${config}`)
|
|
|
|
}
|
2020-05-10 01:19:32 -04:00
|
|
|
|
|
|
|
// We convert the config file into a set of flags.
|
|
|
|
// This is a temporary measure until we add a proper CLI library.
|
2020-05-10 01:35:42 -04:00
|
|
|
const configFileArgv = Object.entries(config).map(([optName, opt]) => {
|
2020-05-10 04:15:29 -04:00
|
|
|
if (opt === true) {
|
2020-05-10 01:35:42 -04:00
|
|
|
return `--${optName}`
|
|
|
|
}
|
|
|
|
return `--${optName}=${opt}`
|
|
|
|
})
|
2020-05-19 00:39:57 -04:00
|
|
|
const args = parse(configFileArgv, {
|
2020-05-10 04:15:29 -04:00
|
|
|
configFile: configPath,
|
|
|
|
})
|
|
|
|
return {
|
|
|
|
...args,
|
|
|
|
config: configPath,
|
|
|
|
}
|
|
|
|
}
|
2020-05-10 01:19:32 -04:00
|
|
|
|
2020-11-03 14:26:25 -06:00
|
|
|
function parseBindAddr(bindAddr: string): Addr {
|
2020-05-10 04:15:29 -04:00
|
|
|
const u = new URL(`http://${bindAddr}`)
|
2020-11-03 14:26:25 -06:00
|
|
|
return {
|
|
|
|
host: u.hostname,
|
|
|
|
// With the http scheme 80 will be dropped so assume it's 80 if missing.
|
|
|
|
// This means --bind-addr <addr> without a port will default to 80 as well
|
|
|
|
// and not the code-server default.
|
|
|
|
port: u.port ? parseInt(u.port, 10) : 80,
|
|
|
|
}
|
2020-05-10 01:19:32 -04:00
|
|
|
}
|
|
|
|
|
2020-05-10 04:15:29 -04:00
|
|
|
interface Addr {
|
|
|
|
host: string
|
|
|
|
port: number
|
|
|
|
}
|
|
|
|
|
2021-09-21 10:41:33 -07:00
|
|
|
/**
|
|
|
|
* This function creates the bind address
|
|
|
|
* using the CLI args.
|
|
|
|
*/
|
2021-11-10 00:28:31 -05:00
|
|
|
export function bindAddrFromArgs(addr: Addr, args: UserProvidedArgs): Addr {
|
2020-05-10 04:15:29 -04:00
|
|
|
addr = { ...addr }
|
|
|
|
if (args["bind-addr"]) {
|
2020-11-03 14:26:25 -06:00
|
|
|
addr = parseBindAddr(args["bind-addr"])
|
2020-05-10 01:19:32 -04:00
|
|
|
}
|
2020-05-10 04:15:29 -04:00
|
|
|
if (args.host) {
|
|
|
|
addr.host = args.host
|
2020-05-10 01:19:32 -04:00
|
|
|
}
|
2020-05-14 21:57:10 -04:00
|
|
|
|
|
|
|
if (process.env.PORT) {
|
|
|
|
addr.port = parseInt(process.env.PORT, 10)
|
|
|
|
}
|
2020-05-10 04:15:29 -04:00
|
|
|
if (args.port !== undefined) {
|
|
|
|
addr.port = args.port
|
|
|
|
}
|
|
|
|
return addr
|
|
|
|
}
|
|
|
|
|
2021-11-10 00:28:31 -05:00
|
|
|
function bindAddrFromAllSources(...argsConfig: UserProvidedArgs[]): Addr {
|
2020-05-10 04:15:29 -04:00
|
|
|
let addr: Addr = {
|
|
|
|
host: "localhost",
|
|
|
|
port: 8080,
|
|
|
|
}
|
|
|
|
|
2020-11-03 14:26:25 -06:00
|
|
|
for (const args of argsConfig) {
|
|
|
|
addr = bindAddrFromArgs(addr, args)
|
|
|
|
}
|
2020-05-10 04:15:29 -04:00
|
|
|
|
2020-11-03 14:26:25 -06:00
|
|
|
return addr
|
2020-05-10 01:19:32 -04:00
|
|
|
}
|
2020-05-14 06:08:37 -04:00
|
|
|
|
2021-10-29 16:03:57 -07:00
|
|
|
/**
|
|
|
|
* Reads the socketPath based on path passed in.
|
|
|
|
*
|
|
|
|
* The one usually passed in is the DEFAULT_SOCKET_PATH.
|
|
|
|
*
|
|
|
|
* If it can't read the path, it throws an error and returns undefined.
|
|
|
|
*/
|
|
|
|
export async function readSocketPath(path: string): Promise<string | undefined> {
|
|
|
|
try {
|
|
|
|
return await fs.readFile(path, "utf8")
|
|
|
|
} catch (error) {
|
|
|
|
// If it doesn't exist, we don't care.
|
|
|
|
// But if it fails for some reason, we should throw.
|
|
|
|
// We want to surface that to the user.
|
|
|
|
if (!isNodeJSErrnoException(error) || error.code !== "ENOENT") {
|
|
|
|
throw error
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
|
2020-09-14 15:57:58 -05:00
|
|
|
/**
|
|
|
|
* Determine if it looks like the user is trying to open a file or folder in an
|
|
|
|
* existing instance. The arguments here should be the arguments the user
|
2021-11-10 00:28:31 -05:00
|
|
|
* explicitly passed on the command line, *NOT DEFAULTS* or the configuration.
|
2020-09-14 15:57:58 -05:00
|
|
|
*/
|
2021-11-10 00:28:31 -05:00
|
|
|
export const shouldOpenInExistingInstance = async (args: UserProvidedArgs): Promise<string | undefined> => {
|
2020-09-14 15:57:58 -05:00
|
|
|
// Always use the existing instance if we're running from VS Code's terminal.
|
|
|
|
if (process.env.VSCODE_IPC_HOOK_CLI) {
|
2021-11-10 00:28:31 -05:00
|
|
|
logger.debug("Found VSCODE_IPC_HOOK_CLI")
|
2020-09-14 15:57:58 -05:00
|
|
|
return process.env.VSCODE_IPC_HOOK_CLI
|
|
|
|
}
|
|
|
|
|
2020-09-15 16:51:43 -05:00
|
|
|
// If these flags are set then assume the user is trying to open in an
|
|
|
|
// existing instance since these flags have no effect otherwise.
|
|
|
|
const openInFlagCount = ["reuse-window", "new-window"].reduce((prev, cur) => {
|
2021-11-10 00:28:31 -05:00
|
|
|
return args[cur as keyof UserProvidedArgs] ? prev + 1 : prev
|
2020-09-15 16:51:43 -05:00
|
|
|
}, 0)
|
|
|
|
if (openInFlagCount > 0) {
|
2021-11-10 00:28:31 -05:00
|
|
|
logger.debug("Found --reuse-window or --new-window")
|
2021-10-29 16:03:57 -07:00
|
|
|
return readSocketPath(DEFAULT_SOCKET_PATH)
|
2020-09-15 16:51:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// It's possible the user is trying to spawn another instance of code-server.
|
2021-11-10 00:28:31 -05:00
|
|
|
// 1. Check if any unrelated flags are set (this should only run when
|
|
|
|
// code-server is invoked exactly like this: `code-server my-file`).
|
|
|
|
// 2. That a file or directory was passed.
|
|
|
|
// 3. That the socket is active.
|
|
|
|
if (Object.keys(args).length === 1 && typeof args._ !== "undefined" && args._.length > 0) {
|
2021-10-29 16:03:57 -07:00
|
|
|
const socketPath = await readSocketPath(DEFAULT_SOCKET_PATH)
|
2020-09-15 16:51:43 -05:00
|
|
|
if (socketPath && (await canConnect(socketPath))) {
|
2021-11-10 00:28:31 -05:00
|
|
|
logger.debug("Found existing code-server socket")
|
2020-09-15 16:51:43 -05:00
|
|
|
return socketPath
|
|
|
|
}
|
|
|
|
}
|
2020-09-14 15:57:58 -05:00
|
|
|
|
|
|
|
return undefined
|
|
|
|
}
|
2021-12-10 12:01:35 -06:00
|
|
|
|
2022-03-22 15:07:14 -05:00
|
|
|
/**
|
|
|
|
* Arguments for running Code's server.
|
|
|
|
*
|
|
|
|
* A subset of ../../lib/vscode/src/vs/server/node/serverEnvironmentService.ts:90
|
|
|
|
*/
|
|
|
|
export interface CodeArgs extends UserProvidedCodeArgs {
|
|
|
|
"accept-server-license-terms"?: boolean
|
|
|
|
"connection-token"?: string
|
|
|
|
help: boolean
|
|
|
|
port?: string
|
|
|
|
version: boolean
|
|
|
|
"without-connection-token"?: boolean
|
|
|
|
"without-browser-env-var"?: boolean
|
|
|
|
compatibility: string
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Types for ../../lib/vscode/src/vs/server/node/server.main.ts:65.
|
|
|
|
*/
|
|
|
|
export type SpawnCodeCli = (args: CodeArgs) => Promise<void>
|
|
|
|
|
2021-12-10 12:01:35 -06:00
|
|
|
/**
|
|
|
|
* Convert our arguments to VS Code server arguments.
|
|
|
|
*/
|
2022-03-22 15:07:14 -05:00
|
|
|
export const toCodeArgs = async (args: DefaultedArgs): Promise<CodeArgs> => {
|
2021-12-10 12:01:35 -06:00
|
|
|
return {
|
|
|
|
...args,
|
|
|
|
"accept-server-license-terms": true,
|
2022-03-22 15:07:14 -05:00
|
|
|
// 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",
|
2021-12-10 12:01:35 -06:00
|
|
|
/** Type casting. */
|
|
|
|
help: !!args.help,
|
|
|
|
version: !!args.version,
|
|
|
|
port: args.port?.toString(),
|
|
|
|
}
|
|
|
|
}
|