#!/bin/bash # Options # # -V, --verbose # Enable verbose output for the installer # # -f, -y, --force, --yes # Skip the confirmation prompt during installation # # -p, --platform # Override the platform identified by the installer # # -b, --bin-dir # Override the bin installation directory # # -a, --arch # Override the architecture identified by the installer # # -B, --base-url # Override the base URL used for downloading releases set -euo pipefail printf "\n" BOLD="$(tput bold 2>/dev/null || echo '')" GREY="$(tput setaf 0 2>/dev/null || echo '')" UNDERLINE="$(tput smul 2>/dev/null || echo '')" RED="$(tput setaf 1 2>/dev/null || echo '')" GREEN="$(tput setaf 2 2>/dev/null || echo '')" YELLOW="$(tput setaf 3 2>/dev/null || echo '')" BLUE="$(tput setaf 4 2>/dev/null || echo '')" MAGENTA="$(tput setaf 5 2>/dev/null || echo '')" NO_COLOR="$(tput sgr0 2>/dev/null || echo '')" info() { printf "%s\n" "${BOLD}${GREY}>${NO_COLOR} $*" } warn() { printf "%s\n" "${YELLOW}! $*${NO_COLOR}" } error() { printf "%s\n" "${RED}x $*${NO_COLOR}" >&2 } complete() { printf "%s\n" "${GREEN}✓${NO_COLOR} $*" } # Gets path to a temporary file, even if get_tmpfile() { local suffix suffix="$1" if hash mktemp; then printf "%s.%s" "$(mktemp)" "${suffix}" else # No really good options here--let's pick a default + hope printf "/tmp/starship.%s" "${suffix}" fi } # Test if a location is writeable by trying to write to it. Windows does not let # you test writeability other than by writing: https://stackoverflow.com/q/1999988 test_writeable() { local path path="${1:-}/test.txt" if touch "${path}" 2>/dev/null; then rm "${path}" return 0 else return 1 fi } fetch() { local command if hash curl 2>/dev/null; then set +e command="curl --silent --fail --location $1" curl --silent --fail --location "$1" rc=$? set -e else if hash wget 2>/dev/null; then set +e command="wget -O- -q $1" wget -O- -q "$1" rc=$? set -e else error "No HTTP download program (curl, wget) found…" exit 1 fi fi if [ $rc -ne 0 ]; then printf "\n" >&2 error "Command failed (exit code $rc): ${BLUE}${command}${NO_COLOR}" printf "\n" >&2 info "This is likely due to Starship not yet supporting your configuration." >&2 info "If you would like to see a build for your configuration," >&2 info "please create an issue requesting a build for ${MAGENTA}${ARCH}-${PLATFORM}${NO_COLOR}:" >&2 info "${BOLD}${UNDERLINE}https://github.com/starship/starship/issues/new/${NO_COLOR}\n" >&2 exit $rc fi } fetch_and_unpack() { local sudo local tmpfile sudo="$1" # I'd like to separate this into a fetch() and unpack() function, but I can't # figure out how to get bash functions to read STDIN/STDOUT from pipes if [ "${EXT}" = "tar.gz" ]; then fetch "${URL}" | ${sudo} tar xz"${VERBOSE}"f - -C "${BIN_DIR}" elif [ "${EXT}" = "zip" ]; then # According to https://unix.stackexchange.com/q/2690, zip files cannot be read # through a pipe. We'll have to do our own file-based setup. tmpfile="$(get_tmpfile "${EXT}")" fetch "${URL}" >"${tmpfile}" ${sudo} unzip "${tmpfile}" -d "${BIN_DIR}" rm "${tmpfile}" else error "Unknown package extension." info "This almost certainly results from a bug in this script--please file a" info "bug report at https://github.com/starship/starship/issues" exit 1 fi } elevate_priv() { if ! hash sudo 2>/dev/null; then error 'Could not find the command "sudo", needed to get permissions for install.' info "If you are on Windows, please run your shell as an administrator, then" info "rerun this script. Otherwise, please run this script as root, or install" info "sudo." exit 1 fi if ! sudo -v; then error "Superuser not granted, aborting installation" exit 1 fi } install() { local msg local sudo if test_writeable "${BIN_DIR}"; then sudo="" msg="Installing Starship, please wait…" else warn "Escalated permissions are required to install to ${BIN_DIR}" elevate_priv sudo="sudo" msg="Installing Starship as root, please wait…" fi info "$msg" fetch_and_unpack "${sudo}" } # Currently supporting: # - win (Git Bash) # - darwin # - linux # - linux_musl (Alpine) detect_platform() { local platform platform="$(uname -s | tr '[:upper:]' '[:lower:]')" # check for MUSL if [ "${platform}" = "linux" ]; then if ldd /bin/sh | grep -i musl >/dev/null; then platform=unknown-linux-musl fi fi # mingw is Git-Bash if echo "${platform}" | grep -i mingw >/dev/null; then platform=pc-windows-msvc fi if [ "${platform}" = "linux" ]; then platform=unknown-linux-gnu fi if [ "${platform}" = "darwin" ]; then platform=apple-darwin fi echo "${platform}" } # Currently supporting: # - x86_64 # - i386 detect_arch() { local arch arch="$(uname -m | tr '[:upper:]' '[:lower:]')" # `uname -m` in some cases mis-reports 32-bit OS as 64-bit, so double check if [ "${arch}" = "x64" ] && [ "$(getconf LONG_BIT)" -eq 32 ]; then arch=i386 fi echo "${arch}" } confirm() { if [ -z "${FORCE-}" ]; then printf "%s " "${MAGENTA}?${NO_COLOR} $* ${BOLD}[y/N]${NO_COLOR}" set +e read -r yn