thelounge/server/command-line/install.ts

121 lines
3.5 KiB
TypeScript
Raw Normal View History

/* eslint-disable @typescript-eslint/restrict-template-expressions */
import log from "../log";
import colors from "chalk";
import semver from "semver";
import Helper from "../helper";
import Config from "../config";
import Utils from "./utils";
import {Command} from "commander";
import {FullMetadata} from "package-json";
2017-09-18 15:57:24 +00:00
type CustomMetadata = FullMetadata & {
thelounge: {
supports: string;
};
};
2017-09-18 15:57:24 +00:00
const program = new Command("install");
2017-09-18 15:57:24 +00:00
program
.argument(
"<package>",
"package to install. Use `file:$path_to_package_dir` to install a local package"
)
2017-09-18 15:57:24 +00:00
.description("Install a theme or a package")
.on("--help", Utils.extraHelp)
.action(async function (packageName: string) {
const fs = await import("fs");
const fspromises = fs.promises;
const path = await import("path");
const packageJson = await import("package-json");
2017-09-18 15:57:24 +00:00
if (!fs.existsSync(Config.getConfigPath())) {
log.error(`${Config.getConfigPath()} does not exist.`);
2017-09-18 15:57:24 +00:00
return;
}
log.info("Retrieving information about the package...");
// TODO: type
let readFile: any = null;
let isLocalFile = false;
if (packageName.startsWith("file:")) {
isLocalFile = true;
// our yarn invocation sets $HOME to the cachedir, so we must expand ~ now
// else the path will be invalid when npm expands it.
packageName = expandTildeInLocalPath(packageName);
readFile = fspromises
.readFile(path.join(packageName.substring("file:".length), "package.json"), "utf-8")
.then((data) => JSON.parse(data) as typeof packageJson);
} else {
const split = packageName.split("@");
packageName = split[0];
const packageVersion = split[1] || "latest";
readFile = packageJson.default(packageName, {
fullMetadata: true,
version: packageVersion,
});
}
if (!readFile) {
// no-op, error should've been thrown before this point
return;
}
readFile
.then((json: CustomMetadata) => {
const humanVersion = isLocalFile ? packageName : `${json.name} v${json.version}`;
2019-07-17 09:33:59 +00:00
if (!("thelounge" in json)) {
log.error(`${colors.red(humanVersion)} does not have The Lounge metadata.`);
2017-09-18 15:57:24 +00:00
2019-07-17 09:33:59 +00:00
process.exit(1);
}
2017-09-18 15:57:24 +00:00
2020-08-21 11:26:35 +00:00
if (
json.thelounge.supports &&
!semver.satisfies(Helper.getVersionNumber(), json.thelounge.supports)
) {
log.error(
`${colors.red(
humanVersion
2020-08-21 11:26:35 +00:00
)} does not support The Lounge v${Helper.getVersionNumber()}. Supported version(s): ${
json.thelounge.supports
}`
);
process.exit(2);
}
log.info(`Installing ${colors.green(humanVersion)}...`);
const yarnVersion = isLocalFile ? packageName : `${json.name}@${json.version}`;
return Utils.executeYarnCommand("add", "--exact", yarnVersion)
2019-07-17 09:33:59 +00:00
.then(() => {
log.info(`${colors.green(humanVersion)} has been successfully installed.`);
if (isLocalFile) {
// yarn v1 is buggy if a local filepath is used and doesn't update
// the lockfile properly. We need to run an install in that case
// even though that's supposed to be done by the add subcommand
return Utils.executeYarnCommand("install").catch((err) => {
throw `Failed to update lockfile after package install ${err}`;
});
}
2019-07-17 09:33:59 +00:00
})
.catch((code) => {
throw `Failed to install ${colors.red(humanVersion)}. Exit code: ${code}`;
2019-07-17 09:33:59 +00:00
});
})
.catch((e) => {
log.error(`${e}`);
process.exit(1);
2017-09-18 15:57:24 +00:00
});
});
function expandTildeInLocalPath(packageName: string): string {
const path = packageName.substring("file:".length);
return "file:" + Helper.expandHome(path);
}
export default program;