2022-11-29 22:26:34 -08:00
|
|
|
#!/usr/bin/env zx
|
|
|
|
|
|
|
|
const execSync = require('child_process').execSync
|
|
|
|
|
|
|
|
$.log = (entry) => {
|
|
|
|
if (entry.kind === 'cmd' && entry.cmd.substring(0, 4) === 'logg') {
|
|
|
|
//execSync(entry.cmd, {stdio: 'inherit', shell: true})
|
|
|
|
} else {
|
|
|
|
log(entry)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let installData
|
|
|
|
const installOrders = {};
|
|
|
|
const installOrdersPre = [];
|
|
|
|
const installOrdersPost = [];
|
|
|
|
const osType = await OSType();
|
|
|
|
let osID = osType
|
|
|
|
if(osType === 'linux') {
|
|
|
|
osID = await realeaseID()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Download the installation map
|
|
|
|
async function downloadInstallData() {
|
|
|
|
const response = await fetch(
|
2022-11-30 01:43:30 -08:00
|
|
|
"https://gitlab.com/megabyte-labs/misc/dotfiles/-/raw/master/.local/share/chezmoi/software.yml"
|
2022-11-29 22:26:34 -08:00
|
|
|
);
|
|
|
|
if (response.ok) {
|
|
|
|
const text = await response.text()
|
|
|
|
return YAML.parse(text)
|
|
|
|
} else {
|
|
|
|
await $`logg error 'Failed to download the installation map'`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Creates the installOrders object which maps package managers to arrays of packages to install
|
|
|
|
async function generateInstallOrders() {
|
|
|
|
const packagesToInstall = process.argv.slice(3);
|
|
|
|
const installerPreference = await OSTypeInstallerKey()
|
|
|
|
await $`logg info 'Installer preference category detected as ${installerPreference}'`
|
|
|
|
const preferenceOrder = installData.installerPreference[installerPreference];
|
|
|
|
await $`logg info 'Preference order acquired:'`
|
|
|
|
console.log(preferenceOrder)
|
|
|
|
const softwarePackages = installData.softwarePackages;
|
|
|
|
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 {
|
|
|
|
await $`logg warn 'The package \`${pkg}\` was not found in the installation map'`
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
for (let preference of preferenceOrder) {
|
|
|
|
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] = [];
|
|
|
|
}
|
|
|
|
await $`logg info 'Found a match for the package \`${pkg}\` (${packageKey} via ${scopedPreference})'`
|
|
|
|
const newPackages = packages[scopedPreference];
|
|
|
|
installOrders[preference] = installOrders[preference].concat(
|
|
|
|
typeof newPackages === "string" ? [newPackages] : newPackages
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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 {
|
|
|
|
const macOS = await $`test -d /Applications && test -d Library`
|
|
|
|
if (macOS.exitCode === 0) {
|
|
|
|
return 'darwin'
|
|
|
|
} else {
|
|
|
|
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) {
|
|
|
|
if (packageManager === 'appimage') {
|
|
|
|
} else if (packageManager === 'ansible') {
|
|
|
|
await $`logg info 'Ensuring temporary passwordless sudo privileges used by Ansible are removed'`
|
|
|
|
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 === 'crew') {
|
|
|
|
} else if (packageManager === 'dnf') {
|
|
|
|
} else if (packageManager === 'flatpak') {
|
|
|
|
} 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) {
|
|
|
|
if (packageManager === 'appimage') {
|
|
|
|
} else if (packageManager === 'ansible') {
|
|
|
|
await $`logg info 'Temporarily enabling passwordless sudo for Ansible role installations'`
|
|
|
|
await $`sudo echo "$(whoami) ALL=(ALL:ALL) NOPASSWD: ALL # TEMPORARY FOR ANSIBLE INSTALL" > /etc/sudoers`
|
|
|
|
} else if (packageManager === 'apk') {
|
|
|
|
} else if (packageManager === 'apt') {
|
|
|
|
await $`sudo apt-get update`
|
|
|
|
} 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') {
|
|
|
|
await $`sudo pacman -Syu`
|
|
|
|
} else if (packageManager === 'pipx') {
|
|
|
|
} else if (packageManager === 'pkg') {
|
|
|
|
} else if (packageManager === 'port') {
|
|
|
|
} else if (packageManager === 'scoop') {
|
|
|
|
} else if (packageManager === 'crew') {
|
|
|
|
} else if (packageManager === 'dnf') {
|
|
|
|
} else if (packageManager === 'flatpak') {
|
|
|
|
} else if (packageManager === 'snap') {
|
|
|
|
} else if (packageManager === 'whalebrew') {
|
|
|
|
} else if (packageManager === 'winget') {
|
|
|
|
} else if (packageManager === 'yay') {
|
|
|
|
} else if (packageManager === 'zypper') {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function ensureInstalled(bin, callback) {
|
|
|
|
const installed = which.sync(bin, { nothrow: true })
|
|
|
|
if (installed) {
|
|
|
|
await $`logg info '\`${bin}\` is available'`
|
|
|
|
} else {
|
|
|
|
await $`logg warn '\`${bin}\` is not installed!'`
|
|
|
|
if (callback) {
|
2022-12-02 06:21:47 -08:00
|
|
|
await callback
|
2022-11-29 22:26:34 -08:00
|
|
|
} else {
|
|
|
|
await $`logg error 'There does not appear to be an installation method available for \`${bin}\`'`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the package manager is available
|
|
|
|
let packageManagerInstalled = {};
|
|
|
|
async function ensurePackageManager(packageManager) {
|
|
|
|
await $`logg info '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') {
|
|
|
|
$`
|
|
|
|
if [ ! -f "$HOME/.cache/megabytelabs/ansible-installed" ] || ! command -v ansible > /dev/null; then
|
|
|
|
pipx install ansible
|
|
|
|
pipx inject ansible PyObjC PyObjC-core docker lxml netaddr pexpect python-vagrant pywinrm requests-credssp watchdog
|
|
|
|
touch "$HOME/.cache/megabytelabs/ansible-installed"
|
|
|
|
else
|
|
|
|
logg info '\`ansible\` and its supporting packages appear to be installed'
|
|
|
|
fi
|
|
|
|
`
|
|
|
|
} else if (packageManager === 'apk') {
|
|
|
|
await ensureInstalled('apk', false)
|
|
|
|
} else if (packageManager === 'apt') {
|
|
|
|
await ensureInstalled('apt', false)
|
|
|
|
} else if (packageManager === 'basher') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('basher', $`
|
2022-11-29 22:26:34 -08:00
|
|
|
# TODO
|
|
|
|
echo "Bash script that installs basher here"
|
|
|
|
`)
|
|
|
|
} else if (packageManager === 'binary') {
|
|
|
|
} else if (packageManager === 'bpkg') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('bpkg', $`
|
2022-11-29 22:26:34 -08:00
|
|
|
# TODO
|
|
|
|
echo "Bash script that installs bpkg here"
|
|
|
|
`)
|
|
|
|
} else if (packageManager === 'brew' || packageManager === 'cask') {
|
2022-12-02 06:30:38 -08:00
|
|
|
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
|
|
|
|
logg info '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
|
|
|
|
logg warn '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
|
2022-11-29 22:26:34 -08:00
|
|
|
fi
|
|
|
|
fi
|
2022-12-02 06:30:38 -08:00
|
|
|
`)
|
|
|
|
}
|
2022-11-29 22:26:34 -08:00
|
|
|
} else if (packageManager === 'cargo') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('cargo', $`
|
2022-11-29 22:26:34 -08:00
|
|
|
# TODO Bash script that installs cargo
|
|
|
|
`)
|
|
|
|
} else if (packageManager === 'choco') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('choco', $`
|
2022-11-29 22:26:34 -08:00
|
|
|
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') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('crew', $`
|
2022-11-29 22:26:34 -08:00
|
|
|
# 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) {
|
|
|
|
await $`logg info '\`dnf\` is available'`
|
|
|
|
} else if (yum) {
|
|
|
|
await $`logg info '\`yum\` is available'`
|
|
|
|
} else {
|
|
|
|
await $`logg error 'Both \`dnf\` and \`yum\` are not available'`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'flatpak') {
|
|
|
|
const flatpak = which.sync('flatpak', { nothrow: true })
|
|
|
|
if (flatpak) {
|
|
|
|
await $`logg info '\`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 {
|
|
|
|
await $`logg error '\`flatpak\` failed to install!'`
|
|
|
|
}
|
|
|
|
await $`logg info '\`flatpak\` was installed. It may require a reboot to function correctly.'`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'gem') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('gem', $`brew install ruby`)
|
2022-11-29 22:26:34 -08:00
|
|
|
} else if (packageManager === 'go') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('gem', $`brew install go`)
|
2022-11-29 22:26:34 -08:00
|
|
|
} else if (packageManager === 'nix') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('nix', $`
|
2022-11-29 22:26:34 -08:00
|
|
|
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) {
|
|
|
|
await $`logg info '\`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') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('pipx', $`brew install pipx && pipx ensurepath`)
|
2022-11-29 22:26:34 -08:00
|
|
|
} else if (packageManager === 'pkg') {
|
|
|
|
await ensureInstalled('pkg', false)
|
|
|
|
} else if (packageManager === 'port') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('port', $`
|
2022-12-02 06:11:26 -08:00
|
|
|
echo -n "TODO - script that installs port on macOS here"
|
2022-11-29 22:26:34 -08:00
|
|
|
`)
|
|
|
|
} else if (packageManager === 'scoop') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('scoop', $`
|
2022-11-29 22:26:34 -08:00
|
|
|
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') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('whalebrew', $`brew install whalebrew`)
|
2022-11-29 22:26:34 -08:00
|
|
|
} else if (packageManager === 'winget') {
|
2022-12-02 06:21:47 -08:00
|
|
|
await ensureInstalled('winget', $`
|
2022-11-29 22:26:34 -08:00
|
|
|
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) {
|
|
|
|
try {
|
|
|
|
if (packageManager === 'appimage') {
|
|
|
|
} else if (packageManager === 'ansible') {
|
|
|
|
for (const role of packages) {
|
|
|
|
await $`ansible localhost -m include_role -a name=${role}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'apk') {
|
|
|
|
await $`sudo apk add ${packages}`
|
|
|
|
} else if (packageManager === 'apt') {
|
|
|
|
await $`sudo apt-get install -y ${packages}`
|
|
|
|
} else if (packageManager === 'basher') {
|
|
|
|
} else if (packageManager === 'binary') {
|
|
|
|
} else if (packageManager === 'brew') {
|
|
|
|
await $`brew install ${packages}`
|
|
|
|
} else if (packageManager === 'cask') {
|
|
|
|
await $`brew install --cask ${packages}`
|
|
|
|
} else if (packageManager === 'cargo') {
|
|
|
|
for (const cargoPkg of packages) {
|
|
|
|
await $`cargo install ${cargoPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'choco') {
|
|
|
|
await $`choco install -y ${packages}`
|
|
|
|
} else if (packageManager === 'crew') {
|
|
|
|
} else if (packageManager === 'dnf') {
|
|
|
|
const dnf = which.sync('dnf', { nothrow: true })
|
|
|
|
const yum = which.sync('yum', { nothrow: true })
|
|
|
|
if (dnf) {
|
|
|
|
await $`dnf install -y ${packages}`
|
|
|
|
} else if (yum) {
|
|
|
|
await $`yum install -y ${packages}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'flatpak') {
|
|
|
|
for (const flatPkg of packages) {
|
|
|
|
await $`sudo flatpak install flathub ${flatPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'gem') {
|
|
|
|
for (const gem of packages) {
|
|
|
|
await $`gem install ${gem}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'go') {
|
|
|
|
for (const goPkg of packages) {
|
|
|
|
await $`go install ${goPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'nix') {
|
|
|
|
} else if (packageManager === 'npm') {
|
|
|
|
for (const npmPkg of packages) {
|
|
|
|
await $`volta install ${npmPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'pacman') {
|
|
|
|
await $`sudo pacman -Sy --noconfirm --needed ${packages}`
|
|
|
|
} else if (packageManager === 'pipx') {
|
|
|
|
for (const pipxPkg of packages) {
|
|
|
|
await $`pipx install ${pipxPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'pkg') {
|
|
|
|
} else if (packageManager === 'port') {
|
|
|
|
for (const portPkg of packages) {
|
|
|
|
await $`sudo port install ${portPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'scoop') {
|
|
|
|
for (const scoopPkg of packages) {
|
|
|
|
await $`scoop install ${scoopPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'snap') {
|
|
|
|
for (const snapPkg of packages) {
|
|
|
|
await $`sudo snap install -y ${snapPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'whalebrew') {
|
|
|
|
for (const whalePkg of packages) {
|
|
|
|
await $`whalebrew install ${snapPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'winget') {
|
|
|
|
} else if (packageManager === 'yay') {
|
|
|
|
for (const yayPkg of packages) {
|
|
|
|
await $`yay -Sy --noconfirm --needed ${yayPkg}`
|
|
|
|
}
|
|
|
|
} else if (packageManager === 'zypper') {
|
|
|
|
await $`sudo zypper install -y ${zypPkg}`
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
await $`logg error 'Failed to install via \`${packageManager}\`'`
|
|
|
|
await $`logg info 'Proceeding with the installation..'`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// main process
|
|
|
|
async function main() {
|
|
|
|
await $`logg info 'Fetching the latest version of the installation map'`
|
|
|
|
installData = await downloadInstallData();
|
|
|
|
await $`logg info 'Calculating the install orders'`
|
|
|
|
await generateInstallOrders();
|
|
|
|
await $`logg info 'Ensuring any package managers that will be used are installed / configured'`
|
|
|
|
const packageManagers = Object.keys(installOrders);
|
|
|
|
for (const packageManager of packageManagers) {
|
|
|
|
await ensurePackageManager(packageManager);
|
|
|
|
}
|
|
|
|
await $`logg info 'The install orders were generated:'`
|
|
|
|
console.log(installOrders)
|
|
|
|
await $`logg info 'Running package manager pre-installation steps'`
|
|
|
|
for (const packageManager of packageManagers) {
|
|
|
|
await beforeInstall(packageManager);
|
|
|
|
}
|
|
|
|
await $`logg info 'Running package-specific pre-installation steps'`
|
|
|
|
for (const script of installOrdersPre) {
|
|
|
|
await $`${script}`;
|
|
|
|
}
|
|
|
|
await $`logg info 'Installing the packages'`
|
|
|
|
for (const packageManager of packageManagers) {
|
|
|
|
const asyncOrders = [];
|
|
|
|
asyncOrders.push(
|
|
|
|
Promise.resolve(
|
|
|
|
installPackageList(packageManager, installOrders[packageManager])
|
|
|
|
)
|
|
|
|
);
|
|
|
|
await Promise.all(asyncOrders);
|
|
|
|
}
|
|
|
|
await $`logg info 'Running package-specific post-installation steps'`
|
|
|
|
for (const script of installOrdersPost) {
|
|
|
|
await $`${script}`;
|
|
|
|
}
|
|
|
|
await $`logg info 'Running package manager post-installation steps'`
|
|
|
|
for (const packageManager of packageManagers) {
|
|
|
|
await afterInstall(packageManager);
|
|
|
|
}
|
|
|
|
await $`logg success 'Done!'`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start the main process
|
|
|
|
await main();
|