1341 lines
41 KiB
Text
1341 lines
41 KiB
Text
#!/usr/bin/env zx
|
||
|
||
const execSync = require("child_process").execSync;
|
||
|
||
// Log symbols
|
||
const figuresDefault = {
|
||
bullet: "●",
|
||
circle: "◯",
|
||
cross: "✖",
|
||
lozenge: "◆",
|
||
play: "▶",
|
||
pointer: "❯",
|
||
square: "◼",
|
||
star: "★",
|
||
tick: "✔",
|
||
};
|
||
|
||
const figuresFallback = {
|
||
bullet: "■",
|
||
circle: "□",
|
||
cross: "×",
|
||
lozenge: "♦",
|
||
play: "►",
|
||
pointer: ">",
|
||
square: "■",
|
||
star: "✶",
|
||
tick: "√",
|
||
};
|
||
|
||
function isUnicodeSupported() {
|
||
if (process.platform !== "win32") {
|
||
// Linux console (kernel)
|
||
return process.env.TERM !== "linux";
|
||
}
|
||
|
||
return (
|
||
Boolean(process.env.CI) ||
|
||
// Windows Terminal
|
||
Boolean(process.env.WT_SESSION) ||
|
||
// ConEmu and cmder
|
||
process.env.ConEmuTask === "{cmd::Cmder}" ||
|
||
process.env.TERM_PROGRAM === "vscode" ||
|
||
process.env.TERM === "xterm-256color" ||
|
||
process.env.TERM === "alacritty"
|
||
);
|
||
}
|
||
|
||
const figures = isUnicodeSupported() ? figuresDefault : figuresFallback;
|
||
|
||
function log(type, label, msg) {
|
||
let icon, message;
|
||
if (type === "info") {
|
||
icon = chalk.cyanBright(figures.pointer);
|
||
message = chalk.gray.bold(msg);
|
||
} else if (type === "star") {
|
||
icon = chalk.yellowBright(figures.star);
|
||
message = chalk.bold(msg);
|
||
} else if (type === "success") {
|
||
icon = chalk.greenBright(figures.play);
|
||
message = chalk.bold(msg);
|
||
} else if (type === "warn") {
|
||
icon = `${chalk.yellowBright(
|
||
figures.lozenge
|
||
)} ${chalk.bold.black.bgYellowBright(" WARNING ")}`;
|
||
message = chalk.yellowBright(msg);
|
||
} else if (type === "error") {
|
||
icon = `${chalk.redBright(figures.cross)} ${chalk.black.bold.bgRedBright(
|
||
" ERROR "
|
||
)}`;
|
||
message = chalk.redBright(msg);
|
||
}
|
||
const outputMessage = `${icon} ${chalk.bold(label)} ${message}`;
|
||
console.log(outputMessage);
|
||
}
|
||
|
||
let installData;
|
||
const installOrders = {};
|
||
const installOrdersPre = [];
|
||
const installOrdersPost = [];
|
||
let osType, osID;
|
||
|
||
// Download the installation map
|
||
async function downloadInstallData() {
|
||
const response = await fetch(
|
||
"https://gitlab.com/megabyte-labs/misc/dotfiles/-/raw/master/.local/share/chezmoi/software.yml"
|
||
);
|
||
if (response.ok) {
|
||
const text = await response.text();
|
||
return YAML.parse(text);
|
||
} else {
|
||
log("error", "Catalog Download", `Failed to download the installation map`);
|
||
log("info", "Catalog Download", `Falling back to local version of software.yml`);
|
||
const text = fs.readFileSync(process.env.HOME + "/.local/share/chezmoi/software.yml");
|
||
return YAML.parse(text);
|
||
}
|
||
}
|
||
|
||
// Creates the installOrders object which maps package managers to arrays of packages to install
|
||
async function generateInstallOrders() {
|
||
const logStage = "Install Orders";
|
||
const packagesToInstall = process.argv.slice(3);
|
||
const installerPreference = await OSTypeInstallerKey();
|
||
log(
|
||
"info",
|
||
logStage,
|
||
`Installer preference category detected as ${installerPreference}`
|
||
);
|
||
const preferenceOrder = installData.installerPreference[installerPreference];
|
||
log("info", logStage, `Preference order acquired:`);
|
||
console.log(preferenceOrder);
|
||
const softwarePackages = installData.softwarePackages;
|
||
pkgFor: for (let pkg of packagesToInstall) {
|
||
let packageKey;
|
||
if (softwarePackages[pkg + ":" + osID]) {
|
||
packageKey = pkg + ":" + osID;
|
||
} else if (softwarePackages[pkg + ":" + osType]) {
|
||
packageKey = pkg + ":" + osType;
|
||
} else if (softwarePackages[pkg]) {
|
||
packageKey = pkg;
|
||
} else {
|
||
log(
|
||
"warn",
|
||
logStage,
|
||
`The package \`${pkg}\` was not found in the installation map`
|
||
);
|
||
continue;
|
||
}
|
||
for (let preference of preferenceOrder) {
|
||
let currentSelector,
|
||
doubleScoped,
|
||
scopedPkgManager,
|
||
scopedSystem,
|
||
normalCheck;
|
||
if (
|
||
softwarePackages[packageKey][preference + ":" + osID] ||
|
||
softwarePackages[packageKey][preference + ":" + osType] ||
|
||
softwarePackages[packageKey][preference]
|
||
) {
|
||
// Handle the _when attribute
|
||
currentSelector = "when";
|
||
doubleScoped =
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference + ":" + osID
|
||
] ||
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + osID + ":" + preference
|
||
] ||
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference + ":" + osType
|
||
] ||
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + osType + ":" + preference
|
||
];
|
||
scopedPkgManager =
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference
|
||
];
|
||
scopedSystem =
|
||
softwarePackages[packageKey]["_" + currentSelector + ":" + osID] ||
|
||
softwarePackages[packageKey]["_" + currentSelector + ":" + osType];
|
||
normalCheck = softwarePackages[packageKey]["_" + currentSelector];
|
||
if (doubleScoped) {
|
||
try {
|
||
await $doubleScoped;
|
||
} catch (e) {
|
||
let pref;
|
||
if (
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference + ":" + osID
|
||
]
|
||
) {
|
||
pref = preference + ":" + osID;
|
||
} else if (
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference + ":" + osType
|
||
]
|
||
) {
|
||
pref = preference + ":" + osType;
|
||
} else if (
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + osID + ":" + preference
|
||
]
|
||
) {
|
||
pref = osID + ":" + preference;
|
||
} else if (
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + osType + ":" + preference
|
||
]
|
||
) {
|
||
pref = osType + ":" + preference;
|
||
}
|
||
log(
|
||
"info",
|
||
"Skipping Package",
|
||
`${pkg} is being skipped because of the _when:${pref} condition`
|
||
);
|
||
continue pkgFor;
|
||
}
|
||
} else if (scopedPkgManager) {
|
||
try {
|
||
await $scopedPkgManager;
|
||
} catch (e) {
|
||
const pref = preference;
|
||
log(
|
||
"info",
|
||
"Skipping Package",
|
||
`${pkg} is being skipped because of the _when:${pref} condition`
|
||
);
|
||
continue pkgFor;
|
||
}
|
||
} else if (scopedSystem) {
|
||
try {
|
||
await $scopedSystem;
|
||
} catch (e) {
|
||
let pref;
|
||
if (
|
||
softwarePackages[packageKey]["_" + currentSelector + ":" + osID]
|
||
) {
|
||
pref = osID;
|
||
} else if (
|
||
softwarePackages[packageKey]["_" + currentSelector + ":" + osType]
|
||
) {
|
||
pref = osType;
|
||
}
|
||
log(
|
||
"info",
|
||
"Skipping Package",
|
||
`${pkg} is being skipped because of the _when:${pref} condition`
|
||
);
|
||
continue pkgFor;
|
||
}
|
||
} else if (normalCheck) {
|
||
try {
|
||
await $(normalCheck);
|
||
} catch (e) {
|
||
log(
|
||
"info",
|
||
"Skipping Package",
|
||
`${pkg} is being skipped because of the _when condition`
|
||
);
|
||
continue pkgFor;
|
||
}
|
||
}
|
||
|
||
// Handle the _bin attribute
|
||
currentSelector = "bin";
|
||
doubleScoped =
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference + ":" + osID
|
||
] ||
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + osID + ":" + preference
|
||
] ||
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference + ":" + osType
|
||
] ||
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + osType + ":" + preference
|
||
];
|
||
scopedPkgManager =
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference
|
||
];
|
||
scopedSystem =
|
||
softwarePackages[packageKey]["_" + currentSelector + ":" + osID] ||
|
||
softwarePackages[packageKey]["_" + currentSelector + ":" + osType];
|
||
normalCheck = softwarePackages[packageKey]["_" + currentSelector];
|
||
if (doubleScoped) {
|
||
const bin = which.sync(doubleScoped, { nothrow: true });
|
||
if (bin) {
|
||
let pref;
|
||
if (
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference + ":" + osID
|
||
]
|
||
) {
|
||
pref = preference + ":" + osID;
|
||
} else if (
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + preference + ":" + osType
|
||
]
|
||
) {
|
||
pref = preference + ":" + osType;
|
||
} else if (
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + osID + ":" + preference
|
||
]
|
||
) {
|
||
pref = osID + ":" + preference;
|
||
} else if (
|
||
softwarePackages[packageKey][
|
||
"_" + currentSelector + ":" + osType + ":" + preference
|
||
]
|
||
) {
|
||
pref = osType + ":" + preference;
|
||
}
|
||
log(
|
||
"info",
|
||
"Skipping Package",
|
||
`${pkg} is being skipped because of the _bin:${pref} condition`
|
||
);
|
||
log("info", "Skipping Package", `${bin} is already in the PATH`);
|
||
continue pkgFor;
|
||
}
|
||
} else if (scopedPkgManager) {
|
||
const bin = which.sync(scopedPkgManager, { nothrow: true });
|
||
if (bin) {
|
||
const pref = preference;
|
||
log(
|
||
"info",
|
||
"Skipping Package",
|
||
`${pkg} is being skipped because of the _bin:${pref} condition`
|
||
);
|
||
log("info", "Skipping Package", `${bin} is already in the PATH`);
|
||
continue pkgFor;
|
||
}
|
||
} else if (scopedSystem) {
|
||
const bin = which.sync(scopedSystem, { nothrow: true });
|
||
if (bin) {
|
||
let pref;
|
||
if (
|
||
softwarePackages[packageKey]["_" + currentSelector + ":" + osID]
|
||
) {
|
||
pref = osID;
|
||
} else if (
|
||
softwarePackages[packageKey]["_" + currentSelector + ":" + osType]
|
||
) {
|
||
pref = osType;
|
||
}
|
||
log(
|
||
"info",
|
||
"Skipping Package",
|
||
`${pkg} is being skipped because of the _bin:${pref} condition`
|
||
);
|
||
log("info", "Skipping Package", `${bin} is already in the PATH`);
|
||
continue pkgFor;
|
||
}
|
||
} else if (normalCheck) {
|
||
const bin = which.sync(normalCheck, { nothrow: true });
|
||
if (bin) {
|
||
log(
|
||
"info",
|
||
"Skipping Package",
|
||
`${pkg} is being skipped because of the _bin condition`
|
||
);
|
||
log("info", "Skipping Package", `${bin} is already in the PATH`);
|
||
continue pkgFor;
|
||
}
|
||
}
|
||
if (softwarePackages[packageKey][preference + ":" + osID]) {
|
||
await updateInstallMaps(
|
||
preference,
|
||
softwarePackages[packageKey],
|
||
preference + ":" + osID,
|
||
pkg,
|
||
packageKey
|
||
);
|
||
break;
|
||
} else if (softwarePackages[packageKey][preference + ":" + osType]) {
|
||
await updateInstallMaps(
|
||
preference,
|
||
softwarePackages[packageKey],
|
||
preference + ":" + osType,
|
||
pkg,
|
||
packageKey
|
||
);
|
||
break;
|
||
} else if (softwarePackages[packageKey][preference]) {
|
||
await updateInstallMaps(
|
||
preference,
|
||
softwarePackages[packageKey],
|
||
preference,
|
||
pkg,
|
||
packageKey
|
||
);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return installOrders;
|
||
}
|
||
|
||
// Update install, pre-hook, and post-hook objects
|
||
async function updateInstallMaps(
|
||
preference,
|
||
packages,
|
||
scopedPreference,
|
||
pkg,
|
||
packageKey
|
||
) {
|
||
const preHook = getHook(packages, "pre", scopedPreference, preference);
|
||
if (preHook) {
|
||
installOrdersPre.concat(typeof preHook === "string" ? [preHook] : preHook);
|
||
}
|
||
const postHook = getHook(packages, "post", scopedPreference, preference);
|
||
if (postHook) {
|
||
installOrdersPost.concat(
|
||
typeof postHook === "string" ? [postHook] : postHook
|
||
);
|
||
}
|
||
if (!installOrders[preference]) {
|
||
installOrders[preference] = [];
|
||
}
|
||
log(
|
||
"info",
|
||
"Install Orders",
|
||
`Found a match for the package \`${pkg}\` (${packageKey} via ${scopedPreference})`
|
||
);
|
||
const newPackages = packages[scopedPreference];
|
||
const newPkgs = typeof newPackages === "string" ? [newPackages] : newPackages;
|
||
installOrders[preference] = installOrders[preference].concat(newPkgs);
|
||
}
|
||
|
||
// Get pre / post install hooks
|
||
function getHook(packages, hook, scopedPreference, preference) {
|
||
const hookLabel = "_" + hook + ":";
|
||
if (packages[hookLabel + scopedPreference]) {
|
||
return packages[hookLabel + scopedPreference];
|
||
} else if (packages[hookLabel + preference]) {
|
||
return packages[hookLabel + preference];
|
||
} else if (packages[hookLabel + osID]) {
|
||
return packages;
|
||
} else if (packages[hookLabel + osType]) {
|
||
return packages[hookLabel + osType];
|
||
} else if (packages["_" + hook]) {
|
||
return packages["_" + hook];
|
||
}
|
||
}
|
||
|
||
// Acquire OS type installer key (for the installerPreference data key)
|
||
async function OSTypeInstallerKey() {
|
||
const apt = which.sync("apt-get", { nothrow: true });
|
||
const dnf = which.sync("dnf", { nothrow: true });
|
||
const freebsd = which.sync("pkg", { nothrow: true });
|
||
const pacman = which.sync("pacman", { nothrow: true });
|
||
const yum = which.sync("yum", { nothrow: true });
|
||
const zypper = which.sync("zypper", { nothrow: true });
|
||
if (apt) {
|
||
return "apt";
|
||
} else if (dnf || yum) {
|
||
return "dnf";
|
||
} else if (pacman) {
|
||
return "pacman";
|
||
} else if (zypper) {
|
||
return "zypper";
|
||
} else if (freebsd) {
|
||
return "freebsd";
|
||
} else {
|
||
try {
|
||
await $`test -d /Applications && test -d /Library`;
|
||
return "darwin";
|
||
} catch (e) {
|
||
return "windows";
|
||
}
|
||
}
|
||
}
|
||
|
||
// Acquire OS type
|
||
async function OSType() {
|
||
try {
|
||
await $`test -d /Applications && test -d /Library`;
|
||
return "darwin";
|
||
} catch (e) {
|
||
try {
|
||
await $`test -f /etc/os-release`;
|
||
return "linux";
|
||
} catch (e) {
|
||
return "windows";
|
||
}
|
||
}
|
||
}
|
||
|
||
// Acquire release ID (for Linux)
|
||
async function releaseID() {
|
||
const ID = await $`
|
||
if [ -f /etc/os-release ]; then
|
||
. /etc/os-release
|
||
echo -n $ID
|
||
fi
|
||
`;
|
||
return ID.stdout;
|
||
}
|
||
|
||
// Post-install hook
|
||
async function afterInstall(packageManager) {
|
||
const logStage = "Post-Install Package Manager";
|
||
if (packageManager === "appimage") {
|
||
} else if (packageManager === "ansible") {
|
||
log(
|
||
"info",
|
||
logStage,
|
||
`Ensuring temporary passwordless sudo privileges used by Ansible are removed`
|
||
);
|
||
const gsed = which.sync("gsed", { nothrow: true });
|
||
if (gsed) {
|
||
await $`sudo gsed -i '/# TEMPORARY FOR ANSIBLE INSTALL/d' /etc/sudoers`;
|
||
} else {
|
||
await $`sudo sed -i '/# TEMPORARY FOR ANSIBLE INSTALL/d' /etc/sudoers`;
|
||
}
|
||
} else if (packageManager === "apk") {
|
||
} else if (packageManager === "apt") {
|
||
} else if (packageManager === "basher") {
|
||
} else if (packageManager === "binary") {
|
||
} else if (packageManager === "brew" || packageManager === "cask") {
|
||
} else if (packageManager === "cargo") {
|
||
} else if (packageManager === "choco") {
|
||
} else if (packageManager === "crew") {
|
||
} else if (packageManager === "dnf") {
|
||
} else if (packageManager === "flatpak") {
|
||
} else if (packageManager === "gem") {
|
||
} else if (packageManager === "go") {
|
||
} else if (packageManager === "nix") {
|
||
} else if (packageManager === "npm") {
|
||
} else if (packageManager === "pacman") {
|
||
} else if (packageManager === "pipx") {
|
||
} else if (packageManager === "pkg") {
|
||
} else if (packageManager === "port") {
|
||
} else if (packageManager === "scoop") {
|
||
} else if (packageManager === "snap") {
|
||
} else if (packageManager === "whalebrew") {
|
||
} else if (packageManager === "winget") {
|
||
} else if (packageManager === "yay") {
|
||
} else if (packageManager === "zypper") {
|
||
}
|
||
}
|
||
|
||
// Pre-install hook
|
||
async function beforeInstall(packageManager) {
|
||
const logStage = "Pre-Install Package Manager";
|
||
if (packageManager === "appimage") {
|
||
} else if (packageManager === "ansible") {
|
||
log(
|
||
"info",
|
||
logStage,
|
||
`Temporarily enabling passwordless sudo for Ansible role installations`
|
||
);
|
||
await $`echo "$(whoami) ALL=(ALL:ALL) NOPASSWD: ALL # TEMPORARY FOR ANSIBLE INSTALL" | sudo tee -a`;
|
||
} else if (packageManager === "apk") {
|
||
await $`sudo apk update`;
|
||
} else if (packageManager === "apt") {
|
||
await $`sudo apt-get update`;
|
||
} else if (packageManager === "basher") {
|
||
} else if (packageManager === "binary") {
|
||
} else if (packageManager === "brew" || packageManager === "cask") {
|
||
await $`brew update`;
|
||
} else if (packageManager === "cargo") {
|
||
} else if (packageManager === "choco") {
|
||
} else if (packageManager === "crew") {
|
||
await $`crew update`;
|
||
} else if (packageManager === "dnf") {
|
||
const dnf = which.sync("dnf", { nothrow: true });
|
||
const yum = which.sync("yum", { nothrow: true });
|
||
if (dnf) {
|
||
await $`dnf check-update`;
|
||
} else if (yum) {
|
||
await $`yum check-update`;
|
||
}
|
||
} else if (packageManager === "flatpak") {
|
||
await $`sudo flatpak update`;
|
||
} else if (packageManager === "gem") {
|
||
} else if (packageManager === "go") {
|
||
} else if (packageManager === "nix") {
|
||
await $`nix-channel --update`;
|
||
} else if (packageManager === "npm") {
|
||
} else if (packageManager === "pacman") {
|
||
await $`sudo pacman -Syu`;
|
||
} else if (packageManager === "pipx") {
|
||
} else if (packageManager === "pkg") {
|
||
await $`sudo pkg upgrade`;
|
||
} else if (packageManager === "port") {
|
||
const port = which.sync("port", { nothrow: true });
|
||
if (port) {
|
||
await $`sudo port sync`;
|
||
} else {
|
||
log(
|
||
"error",
|
||
"Port Not Installed",
|
||
"Skipping sudo port sync step because port is not installed"
|
||
);
|
||
}
|
||
} else if (packageManager === "scoop") {
|
||
await $`scoop update`;
|
||
} else if (packageManager === "snap") {
|
||
await $`sudo snap refresh`;
|
||
} else if (packageManager === "whalebrew") {
|
||
if (osType === "darwin") {
|
||
const docker = which.sync("docker", { nothrow: true });
|
||
if (!docker) {
|
||
await $`brew install --cask docker`;
|
||
}
|
||
try {
|
||
await $`docker run --rm hello-world`;
|
||
} catch (e) {
|
||
log(
|
||
"warn",
|
||
logStage,
|
||
`The command \`docker run --rm hello-world\` failed`
|
||
);
|
||
try {
|
||
log(
|
||
"info",
|
||
logStage,
|
||
"Attempting to open `/Applications/Docker.app` (Docker Desktop for macOS). This should take about 30 seconds."
|
||
);
|
||
const promises = [
|
||
$`test -d /Applications/Docker.app`,
|
||
$`open /Applications/Docker.app`,
|
||
];
|
||
await Promise.all(promises);
|
||
const gum = which.sync("gum", { nothrow: true });
|
||
if (gum) {
|
||
execSync(
|
||
'gum spin --spinner dot --title "Waiting for Docker Desktop to start up.." -- sleep 30',
|
||
{ stdio: "inherit", shell: true }
|
||
);
|
||
} else {
|
||
await $`sleep 30`;
|
||
}
|
||
} catch (e) {
|
||
log("warn", logStage, `Docker Desktop appears to not be installed!`);
|
||
}
|
||
}
|
||
}
|
||
} else if (packageManager === "winget") {
|
||
await $`winget source update`;
|
||
} else if (packageManager === "yay") {
|
||
} else if (packageManager === "zypper") {
|
||
await $`sudo zypper update`;
|
||
}
|
||
}
|
||
|
||
async function ensureInstalled(bin, callback) {
|
||
const logStage = "Package Manager Install";
|
||
const installed = which.sync(bin, { nothrow: true });
|
||
if (installed) {
|
||
log("info", logStage, `\`${bin}\` is available`);
|
||
} else {
|
||
log("warn", logStage, `\`${bin}\` is not installed!`);
|
||
if (callback) {
|
||
await callback;
|
||
} else {
|
||
log(
|
||
"error",
|
||
logStage,
|
||
`There does not appear to be an installation method available for \`${bin}\``
|
||
);
|
||
}
|
||
}
|
||
}
|
||
|
||
async function ensurePackageManagerAnsible() {
|
||
await $`pipx install ansible`;
|
||
await $`pipx inject ansible PyObjC PyObjC-core docker lxml netaddr pexpect python-vagrant pywinrm requests-credssp watchdog`;
|
||
await $`mkdir -p "$HOME/.cache/megabyte-labs"`;
|
||
await $`touch "$HOME/.cache/megabyte-labs/ansible-installed"`;
|
||
log(
|
||
"info",
|
||
"Package Manager Install",
|
||
`Ansible and its supporting packages are now installed via pipx`
|
||
);
|
||
}
|
||
|
||
// Ensure the package manager is available
|
||
let packageManagerInstalled = {};
|
||
async function ensurePackageManager(packageManager) {
|
||
const logStage = "Package Manager Install";
|
||
log("info", logStage, `Ensuring \`${packageManager}\` is set up`);
|
||
if (packageManagerInstalled[packageManager]) {
|
||
return;
|
||
} else {
|
||
packageManagerInstalled[packageManager] = true;
|
||
}
|
||
if (packageManager === "ansible") {
|
||
await ensurePackageManager("pipx");
|
||
}
|
||
if (
|
||
packageManager === "gem" ||
|
||
packageManager === "go" ||
|
||
packageManager === "npm" ||
|
||
packageManager === "pipx" ||
|
||
packageManager === "whalebrew"
|
||
) {
|
||
await ensurePackageManager("brew");
|
||
}
|
||
if (packageManager === "appimage") {
|
||
} else if (packageManager === "ansible") {
|
||
try {
|
||
await $`test -f "$HOME/.cache/megabyte-labs/ansible-installed"`;
|
||
const ansible = which.sync("ansible", { nothrow: true });
|
||
if (ansible) {
|
||
log(
|
||
"info",
|
||
logStage,
|
||
`\`ansible\` and its supporting packages appear to be installed`
|
||
);
|
||
} else {
|
||
await ensurePackageManagerAnsible();
|
||
}
|
||
} catch (e) {
|
||
await ensurePackageManagerAnsible();
|
||
}
|
||
} else if (packageManager === "apk") {
|
||
await ensureInstalled("apk", false);
|
||
} else if (packageManager === "apt") {
|
||
await ensureInstalled("apt", false);
|
||
} else if (packageManager === "basher") {
|
||
await ensureInstalled(
|
||
"basher",
|
||
$`
|
||
# TODO
|
||
echo "Bash script that installs basher here"
|
||
`
|
||
);
|
||
} else if (packageManager === "binary") {
|
||
} else if (packageManager === "bpkg") {
|
||
await ensureInstalled(
|
||
"bpkg",
|
||
$`
|
||
# TODO
|
||
echo "Bash script that installs bpkg here"
|
||
`
|
||
);
|
||
} else if (packageManager === "brew" || packageManager === "cask") {
|
||
const brew = which.sync("brew", { nothrow: true });
|
||
if (!brew) {
|
||
await ensureInstalled(
|
||
"brew",
|
||
$`
|
||
if command -v sudo > /dev/null && sudo -n true; then
|
||
echo | bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
|
||
else
|
||
log('info', logStage, 'Homebrew is not installed. Password may be required.')
|
||
bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)" || BREW_EXIT_CODE="$?"
|
||
if [ -n "$BREW_EXIT_CODE" ]; then
|
||
if command -v brew > /dev/null; then
|
||
log('warn', logStage, 'Homebrew was installed but part of the installation failed. Attempting to fix..')
|
||
BREW_DIRS="share/man share/doc share/zsh/site-functions etc/bash_completion.d"
|
||
for BREW_DIR in $BREW_DIRS; do
|
||
if [ -d "$(brew --prefix)/$BREW_DIR" ]; then
|
||
sudo chown -R "$(whoami)" "$(brew --prefix)/$BREW_DIR"
|
||
fi
|
||
done
|
||
brew update --force --quiet
|
||
fi
|
||
fi
|
||
fi
|
||
`
|
||
);
|
||
}
|
||
} else if (packageManager === "cargo") {
|
||
await ensureInstalled(
|
||
"cargo",
|
||
$`
|
||
# TODO Bash script that installs cargo
|
||
`
|
||
);
|
||
} else if (packageManager === "choco") {
|
||
await ensureInstalled(
|
||
"choco",
|
||
$`
|
||
powershell "Set-ExecutionPolicy Bypass -Scope Process -Force; [System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072; iex ((New-Object System.Net.WebClient).DownloadString('https://community.chocolatey.org/install.ps1'))"
|
||
`
|
||
);
|
||
} else if (packageManager === "crew") {
|
||
await ensureInstalled(
|
||
"crew",
|
||
$`
|
||
# TODO Bash script that installs crew here
|
||
# Source: https://github.com/chromebrew/chromebrew
|
||
curl -Ls git.io/vddgY | bash
|
||
`
|
||
);
|
||
} else if (packageManager === "dnf") {
|
||
const dnf = which.sync("dnf", { nothrow: true });
|
||
const yum = which.sync("yum", { nothrow: true });
|
||
if (dnf) {
|
||
log("info", logStage, `\`dnf\` is available`);
|
||
} else if (yum) {
|
||
log("info", logStage, `\`yum\` is available`);
|
||
} else {
|
||
log("error", logStage, `Both \`dnf\` and \`yum\` are not available`);
|
||
}
|
||
} else if (packageManager === "flatpak") {
|
||
const flatpak = which.sync("flatpak", { nothrow: true });
|
||
if (flatpak) {
|
||
log("info", logStage, `\`flatpak\` is available`);
|
||
} else {
|
||
const apk = which.sync("apk", { nothrow: true });
|
||
const apt = which.sync("apt", { nothrow: true });
|
||
const dnf = which.sync("dnf", { nothrow: true });
|
||
const yum = which.sync("yum", { nothrow: true });
|
||
const pacman = which.sync("pacman", { nothrow: true });
|
||
const zypper = which.sync("zypper", { nothrow: true });
|
||
if (apk) {
|
||
$`sudo apk add flatpak`;
|
||
} else if (apt) {
|
||
$`
|
||
sudo apt install -y flatpak
|
||
if [ -f /usr/bin/gnome-shell ]; then
|
||
sudo apt install -y gnome-software-plugin-flatpak
|
||
fi
|
||
if [ -f /usr/bin/plasmashell ]; then
|
||
sudo apt install -y plasmashell
|
||
fi
|
||
`;
|
||
} else if (dnf) {
|
||
await $`sudo dnf install -y flatpak`;
|
||
} else if (yum) {
|
||
await $`sudo yum install -y flatpak`;
|
||
} else if (pacman) {
|
||
await $`sudo pacman -Sy flatpak`;
|
||
} else if (zypper) {
|
||
await $`sudo zypper install -y flatpak`;
|
||
}
|
||
const flatpakPost = which.sync("flatpak", { nothrow: true });
|
||
if (flatpakPost) {
|
||
await $`flatpak remote-add --if-not-exists flathub https://flathub.org/repo/flathub.flatpakrepo`;
|
||
} else {
|
||
log("error", logStage, `\`flatpak\` failed to install!`);
|
||
}
|
||
log(
|
||
"info",
|
||
logStage,
|
||
`\`flatpak\` was installed. It may require a reboot to function correctly.`
|
||
);
|
||
}
|
||
} else if (packageManager === "gem") {
|
||
await ensureInstalled("gem", $`brew install ruby`);
|
||
} else if (packageManager === "go") {
|
||
await ensureInstalled("gem", $`brew install go`);
|
||
} else if (packageManager === "nix") {
|
||
await ensureInstalled(
|
||
"nix",
|
||
$`
|
||
if [ -d /Applications ] && [ -d /Library ]; then
|
||
sh <(curl -L https://nixos.org/nix/install)
|
||
else
|
||
sh <(curl -L https://nixos.org/nix/install) --daemon
|
||
fi
|
||
`
|
||
);
|
||
} else if (packageManager === "npm") {
|
||
const npm = which("npm", { nothrow: true });
|
||
const node = which("node", { nothrow: true });
|
||
const volta = which("volta", { nothrow: true });
|
||
if (npm && node && volta) {
|
||
log("info", logStage, `\`npm\`, \`node\`, and \`volta\` are available`);
|
||
} else {
|
||
if (!volta) {
|
||
await $`brew install volta`;
|
||
}
|
||
await $`
|
||
if [ -z "$VOLTA_HOME" ]; then
|
||
volta setup
|
||
fi
|
||
export PATH="$VOLTA_HOME/bin:$PATH"
|
||
volta install node
|
||
`;
|
||
}
|
||
} else if (packageManager === "pacman") {
|
||
await ensureInstalled("pacman", false);
|
||
} else if (packageManager === "pipx") {
|
||
await ensureInstalled("pipx", $`brew install pipx && pipx ensurepath`);
|
||
} else if (packageManager === "pkg") {
|
||
await ensureInstalled("pkg", false);
|
||
} else if (packageManager === "port") {
|
||
await ensureInstalled(
|
||
"port",
|
||
$`
|
||
echo -n "TODO - script that installs port on macOS here"
|
||
`
|
||
);
|
||
} else if (packageManager === "scoop") {
|
||
await ensureInstalled(
|
||
"scoop",
|
||
$`
|
||
powershell 'Set-ExecutionPolicy RemoteSigned -Scope CurrentUser'
|
||
powershell 'irm get.scoop.sh | iex
|
||
`
|
||
);
|
||
} else if (packageManager === "snap") {
|
||
const apk = which.sync("apk", { nothrow: true });
|
||
const apt = which.sync("apt-get", { nothrow: true });
|
||
const dnf = which.sync("dnf", { nothrow: true });
|
||
const yum = which.sync("yum", { nothrow: true });
|
||
const pacman = which.sync("pacman", { nothrow: true });
|
||
const zypper = which.sync("zypper", { nothrow: true });
|
||
if (apt) {
|
||
await $`
|
||
if [ -f /etc/apt/preferences.d/nosnap.pref ]; then
|
||
sudo mv /etc/apt/preferences.d/nosnap.pref /etc/apt/nosnap.pref.bak
|
||
fi
|
||
sudo apt install -y snapd
|
||
`;
|
||
// TODO Following may be required on Kali -> https://snapcraft.io/docs/installing-snap-on-kali
|
||
// systemctl enable --now snapd apparmor
|
||
} else if (dnf) {
|
||
await $`
|
||
sudo dnf install -y snapd
|
||
if [ ! -d /snap ]; then
|
||
sudo ln -s /var/lib/snapd/snap /snap
|
||
fi
|
||
`;
|
||
} else if (yum) {
|
||
await $`
|
||
sudo yum install -y snapd
|
||
sudo systemctl enable --now snapd.socket
|
||
if [ ! -d /snap ]; then
|
||
sudo ln -s /var/lib/snapd/snap /snap
|
||
fi
|
||
`;
|
||
} else if (pacman) {
|
||
await $`
|
||
if [ -f /etc/arch-release ]; then
|
||
sudo git clone https://aur.archlinux.org/snapd.git /usr/local/src/snapd
|
||
cd /usr/local/src/snapd
|
||
sudo makepkg -si
|
||
else
|
||
sudo pacman -S snapd
|
||
sudo systemctl enable --now snapd.socket
|
||
if [ ! -d /snap ]; then
|
||
sudo ln -s /var/lib/snapd/snap /snap
|
||
fi
|
||
fi
|
||
`;
|
||
} else if (zypper) {
|
||
// TODO See https://snapcraft.io/docs/installing-snap-on-opensuse
|
||
await $`
|
||
echo "TODO - Bash script that installs snap w/ zypper"
|
||
`;
|
||
}
|
||
const snap = which.sync("snap", { nothrow: true });
|
||
if (snap) {
|
||
$`sudo snap install core`;
|
||
}
|
||
} else if (packageManager === "whalebrew") {
|
||
await ensureInstalled("whalebrew", $`brew install whalebrew`);
|
||
} else if (packageManager === "winget") {
|
||
await ensureInstalled(
|
||
"winget",
|
||
$`
|
||
echo "TODO - Script that installs winget here"
|
||
`
|
||
);
|
||
} else if (packageManager === "yay") {
|
||
const yay = which.sync("yay", { nothrow: true });
|
||
await $`sudo pacman -S --needed base-devel git`;
|
||
await $`
|
||
if [ -d /usr/local/src ]; then
|
||
git clone https://aur.archlinux.org/yay.git /usr/local/src/yay
|
||
cd /usr/local/src/yay
|
||
makepkg -si
|
||
fi
|
||
`;
|
||
} else if (packageManager === "zypper") {
|
||
await ensureInstalled("zypper", false);
|
||
}
|
||
}
|
||
|
||
// Installs a list of packages via the specified package manager
|
||
async function installPackageList(packageManager, packages) {
|
||
const logStage = "Package Install";
|
||
try {
|
||
if (packageManager === "appimage") {
|
||
} else if (packageManager === "ansible") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`ansible localhost -m setup -m include_role -a name=${pkg} -e ansible_user=${process.env.USER} -e include_homebrew_install=False`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Ansible Role Failure",
|
||
`There was an error installing ${pkg} with Ansible`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "apk") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`sudo apk add ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"APK Install Failure",
|
||
`There was an error installing ${pkg} with apk`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "apt") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`sudo apt-get install -y ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"apt-get Failure",
|
||
`There was an error installing ${pkg} with apt-get`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "basher") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`basher install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Basher Failure",
|
||
`There was an error installing ${pkg} with basher`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "binary") {
|
||
} else if (packageManager === "brew") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`brew install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Homebrew Failure",
|
||
`There was an error installing ${pkg} with brew`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "cask") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`brew install --cask ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Homebrew Cask Failure",
|
||
`There was an error installing ${pkg} with Homebrew Cask`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "cargo") {
|
||
for (const pkg of packages) {
|
||
try {
|
||
await $`cargo install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Cargo Failure",
|
||
`There was an error installing ${pkg} with Cargo`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "choco") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`choco install -y ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Chocolatey Failure",
|
||
`There was an error installing ${pkg} with Chocolatey`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "crew") {
|
||
} else if (packageManager === "dnf") {
|
||
const dnf = which.sync("dnf", { nothrow: true });
|
||
const yum = which.sync("yum", { nothrow: true });
|
||
if (dnf) {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`sudo dnf install -y ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"dnf Failure",
|
||
`There was an error installing ${pkg} with dnf`
|
||
);
|
||
}
|
||
}
|
||
} else if (yum) {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`sudo yum install -y ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"yum Failure",
|
||
`There was an error installing ${pkg} with yum`
|
||
);
|
||
}
|
||
}
|
||
}
|
||
} else if (packageManager === "flatpak") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`sudo flatpak install flathub ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Flatpak Failure",
|
||
`There was an error installing ${pkg} with flatpak`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "gem") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`gem install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Gem Failure",
|
||
`There was an error installing ${pkg} with gem`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "go") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`go install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Go Failure",
|
||
`There was an error installing ${pkg} with go`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "nix") {
|
||
} else if (packageManager === "npm") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`volta install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Volta Failure",
|
||
`There was an error installing ${pkg} with volta`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "pacman") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`sudo pacman -Sy --noconfirm --needed ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Pacman Failure",
|
||
`There was an error installing ${pkg} with pacman`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "pipx") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`pipx install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"PIPX Failure",
|
||
`There was an error installing ${pkg} with pipx`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "pkg") {
|
||
} else if (packageManager === "port") {
|
||
const port = which.sync("port", { nothrow: true });
|
||
if (port) {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`sudo port install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Port Failure",
|
||
`There was an error installing ${pkg} with port`
|
||
);
|
||
}
|
||
}
|
||
} else {
|
||
log(
|
||
"error",
|
||
"Port Not Installed",
|
||
`Unable to install with port because it is not installed. Skipping installation of ${packages}`
|
||
);
|
||
}
|
||
} else if (packageManager === "scoop") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`scoop install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Scoop Failure",
|
||
`There was an error installing ${pkg} with scoop`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "snap") {
|
||
for (let pkg of packages) {
|
||
// TODO _snapClassic
|
||
try {
|
||
await $`sudo snap install -y ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Snap Failure",
|
||
`There was an error installing ${pkg} with snap`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "whalebrew") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`whalebrew install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Whalebrew Failure",
|
||
`There was an error installing ${pkg} with whalebrew`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "winget") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`winget install ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Winget Failure",
|
||
`There was an error installing ${pkg} with winget`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "yay") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`yay -Sy --noconfirm --needed ${pkg}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Yay Failure",
|
||
`There was an error installing ${pkg} with yay`
|
||
);
|
||
}
|
||
}
|
||
} else if (packageManager === "zypper") {
|
||
for (let pkg of packages) {
|
||
try {
|
||
await $`sudo zypper install -y ${packages}`;
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
"Zypper Failure",
|
||
`There was an error installing ${pkg} with zypper`
|
||
);
|
||
}
|
||
}
|
||
}
|
||
} catch (e) {
|
||
log(
|
||
"error",
|
||
logStage,
|
||
`Possibly encountered an error while installing via \`${packageManager}\``
|
||
);
|
||
log("info", logStage, `Proceeding with the installation..`);
|
||
}
|
||
}
|
||
|
||
// main process
|
||
async function main() {
|
||
osType = await OSType();
|
||
osID = osType;
|
||
if (osType === "linux") {
|
||
osID = await releaseID();
|
||
}
|
||
log(
|
||
"info",
|
||
"Catalog Download",
|
||
`Fetching the latest version of the installation map`
|
||
);
|
||
installData = await downloadInstallData();
|
||
log("info", "Install Orders", `Calculating the install orders`);
|
||
await generateInstallOrders();
|
||
log(
|
||
"info",
|
||
"Ensure Package Manager Installed",
|
||
`Ensuring any package managers that will be used are installed / configured`
|
||
);
|
||
const packageManagers = Object.keys(installOrders);
|
||
for (const packageManager of packageManagers) {
|
||
await ensurePackageManager(packageManager);
|
||
}
|
||
log("info", "Install Orders", `The install orders were generated:`);
|
||
console.log(installOrders);
|
||
log(
|
||
"info",
|
||
"Package Manager Pre-Install",
|
||
`Running package manager pre-installation steps`
|
||
);
|
||
for (const packageManager of packageManagers) {
|
||
await beforeInstall(packageManager);
|
||
}
|
||
log(
|
||
"info",
|
||
"Package Pre-Install",
|
||
`Running package-specific pre-installation steps`
|
||
);
|
||
for (const script of installOrdersPre) {
|
||
await $`${script}`;
|
||
}
|
||
log("info", "Package Install", `Installing the packages`);
|
||
for (const packageManager of packageManagers) {
|
||
const asyncOrders = [];
|
||
asyncOrders.push(
|
||
installPackageList(packageManager, installOrders[packageManager])
|
||
);
|
||
await Promise.all(asyncOrders);
|
||
}
|
||
log(
|
||
"info",
|
||
"Package Post-Install",
|
||
`Running package-specific post-installation steps`
|
||
);
|
||
for (const script of installOrdersPost) {
|
||
await $`${script}`;
|
||
}
|
||
log(
|
||
"info",
|
||
"Package Manager Post-Install",
|
||
`Running package manager post-installation steps`
|
||
);
|
||
for (const packageManager of packageManagers) {
|
||
await afterInstall(packageManager);
|
||
}
|
||
log("success", "Installation Complete", `Done!`);
|
||
}
|
||
|
||
// Start the main process
|
||
await main();
|