From 8115411d95cfb59a50b50b312010bd5e55edd06f Mon Sep 17 00:00:00 2001 From: neo Date: Wed, 9 Mar 2022 18:52:32 +0800 Subject: [PATCH] =?UTF-8?q?feat:=20=E6=9B=B4=E6=96=B0=20install-origin.sh?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- install-origin.sh | 270 ++++++++++++++++++++++++++-------------------- 1 file changed, 154 insertions(+), 116 deletions(-) diff --git a/install-origin.sh b/install-origin.sh index fe6624a..e3f23c0 100755 --- a/install-origin.sh +++ b/install-origin.sh @@ -1,4 +1,9 @@ #!/bin/bash + +# We don't need return codes for "$(command)", only stdout is needed. +# Allow `[[ -n "$(command)" ]]`, `func "$(command)"`, pipes, etc. +# shellcheck disable=SC2312 + set -u abort() { @@ -6,16 +11,87 @@ abort() { exit 1 } +# Fail fast with a concise message when not using bash +# Single brackets are needed here for POSIX compatibility +# shellcheck disable=SC2292 if [ -z "${BASH_VERSION:-}" ] then abort "Bash is required to interpret this script." fi +# Check if script is run with force-interactive mode in CI +if [[ -n "${CI-}" && -n "${INTERACTIVE-}" ]] +then + abort "Cannot run force-interactive mode in CI." +fi + +# Check if both `INTERACTIVE` and `NONINTERACTIVE` are set +# Always use single-quoted strings with `exp` expressions +# shellcheck disable=SC2016 +if [[ -n "${INTERACTIVE-}" && -n "${NONINTERACTIVE-}" ]] +then + abort 'Both `$INTERACTIVE` and `$NONINTERACTIVE` are set. Please unset at least one variable and try again.' +fi + +# string formatters +if [[ -t 1 ]] +then + tty_escape() { printf "\033[%sm" "$1"; } +else + tty_escape() { :; } +fi +tty_mkbold() { tty_escape "1;$1"; } +tty_underline="$(tty_escape "4;39")" +tty_blue="$(tty_mkbold 34)" +tty_red="$(tty_mkbold 31)" +tty_bold="$(tty_mkbold 39)" +tty_reset="$(tty_escape 0)" + +shell_join() { + local arg + printf "%s" "$1" + shift + for arg in "$@" + do + printf " " + printf "%s" "${arg// /\ }" + done +} + +chomp() { + printf "%s" "${1/"$'\n'"/}" +} + +ohai() { + printf "${tty_blue}==>${tty_bold} %s${tty_reset}\n" "$(shell_join "$@")" +} + +warn() { + printf "${tty_red}Warning${tty_reset}: %s\n" "$(chomp "$1")" +} + # Check if script is run non-interactively (e.g. CI) # If it is run non-interactively we should not prompt for passwords. -if [[ ! -t 0 || -n "${CI-}" ]] +# Always use single-quoted strings with `exp` expressions +# shellcheck disable=SC2016 +if [[ -z "${NONINTERACTIVE-}" ]] then - NONINTERACTIVE=1 + if [[ -n "${CI-}" ]] + then + warn 'Running in non-interactive mode because `$CI` is set.' + NONINTERACTIVE=1 + elif [[ ! -t 0 ]] + then + if [[ -z "${INTERACTIVE-}" ]] + then + warn 'Running in non-interactive mode because `stdin` is not a TTY.' + NONINTERACTIVE=1 + else + warn 'Running in interactive mode despite `stdin` not being a TTY because `$INTERACTIVE` is set.' + fi + fi +else + ohai 'Running in non-interactive mode because `$NONINTERACTIVE` is set.' fi # First check OS. @@ -47,12 +123,12 @@ then fi HOMEBREW_CACHE="${HOME}/Library/Caches/Homebrew" - STAT_FLAG="-f" + STAT_PRINTF=("stat" "-f") PERMISSION_FORMAT="%A" - CHOWN="/usr/sbin/chown" - CHGRP="/usr/bin/chgrp" + CHOWN=("/usr/sbin/chown") + CHGRP=("/usr/bin/chgrp") GROUP="admin" - TOUCH="/usr/bin/touch" + TOUCH=("/usr/bin/touch") else UNAME_MACHINE="$(uname -m)" @@ -61,13 +137,15 @@ else HOMEBREW_PREFIX_DEFAULT="/home/linuxbrew/.linuxbrew" HOMEBREW_CACHE="${HOME}/.cache/Homebrew" - STAT_FLAG="--printf" + STAT_PRINTF=("stat" "--printf") PERMISSION_FORMAT="%a" - CHOWN="/bin/chown" - CHGRP="/bin/chgrp" + CHOWN=("/bin/chown") + CHGRP=("/bin/chgrp") GROUP="$(id -gn)" - TOUCH="/bin/touch" + TOUCH=("/bin/touch") fi +CHMOD=("/bin/chmod") +MKDIR=("/bin/mkdir" "-p") HOMEBREW_BREW_DEFAULT_GIT_REMOTE="https://github.com/Homebrew/brew" HOMEBREW_CORE_DEFAULT_GIT_REMOTE="https://github.com/Homebrew/homebrew-core" @@ -100,20 +178,6 @@ REQUIRED_GIT_VERSION=2.7.0 # HOMEBREW_MINIMUM_GIT_VERSION in brew.sh in Homebr export HOMEBREW_NO_ANALYTICS_THIS_RUN=1 export HOMEBREW_NO_ANALYTICS_MESSAGE_OUTPUT=1 -# string formatters -if [[ -t 1 ]] -then - tty_escape() { printf "\033[%sm" "$1"; } -else - tty_escape() { :; } -fi -tty_mkbold() { tty_escape "1;$1"; } -tty_underline="$(tty_escape "4;39")" -tty_blue="$(tty_mkbold 34)" -tty_red="$(tty_mkbold 31)" -tty_bold="$(tty_mkbold 39)" -tty_reset="$(tty_escape 0)" - unset HAVE_SUDO_ACCESS # unset this from the environment have_sudo_access() { @@ -122,29 +186,22 @@ have_sudo_access() { return 1 fi - local -a args + local -a SUDO=("/usr/bin/sudo") if [[ -n "${SUDO_ASKPASS-}" ]] then - args=("-A") + SUDO+=("-A") elif [[ -n "${NONINTERACTIVE-}" ]] then - args=("-n") + SUDO+=("-n") fi if [[ -z "${HAVE_SUDO_ACCESS-}" ]] then - if [[ -n "${args[*]-}" ]] - then - SUDO="/usr/bin/sudo ${args[*]}" - else - SUDO="/usr/bin/sudo" - fi if [[ -n "${NONINTERACTIVE-}" ]] then - # Don't add quotes around ${SUDO} here - ${SUDO} -l mkdir &>/dev/null + "${SUDO[@]}" -l mkdir &>/dev/null else - ${SUDO} -v && ${SUDO} -l mkdir &>/dev/null + "${SUDO[@]}" -v && "${SUDO[@]}" -l mkdir &>/dev/null fi HAVE_SUDO_ACCESS="$?" fi @@ -157,29 +214,6 @@ have_sudo_access() { return "${HAVE_SUDO_ACCESS}" } -shell_join() { - local arg - printf "%s" "$1" - shift - for arg in "$@" - do - printf " " - printf "%s" "${arg// /\ }" - done -} - -chomp() { - printf "%s" "${1/"$'\n'"/}" -} - -ohai() { - printf "${tty_blue}==>${tty_bold} %s${tty_reset}\n" "$(shell_join "$@")" -} - -warn() { - printf "${tty_red}Warning${tty_reset}: %s\n" "$(chomp "$1")" -} - execute() { if ! "$@" then @@ -222,7 +256,7 @@ ring_bell() { wait_for_user() { local c echo - echo "Press RETURN to continue or any other key to abort" + echo "Press ${tty_bold}RETURN${tty_reset} to continue or any other key to abort:" getc c # we test for \r and \n because some stuff does \r instead if ! [[ "${c}" == $'\r' || "${c}" == $'\n' ]] @@ -264,7 +298,7 @@ should_install_command_line_tools() { } get_permission() { - stat "${STAT_FLAG}" "${PERMISSION_FORMAT}" "$1" + "${STAT_PRINTF[@]}" "${PERMISSION_FORMAT}" "$1" } user_only_chmod() { @@ -276,7 +310,7 @@ exists_but_not_writable() { } get_owner() { - stat "${STAT_FLAG}" "%u" "$1" + "${STAT_PRINTF[@]}" "%u" "$1" } file_not_owned() { @@ -284,14 +318,14 @@ file_not_owned() { } get_group() { - stat "${STAT_FLAG}" "%g" "$1" + "${STAT_PRINTF[@]}" "%g" "$1" } file_not_grpowned() { [[ " $(id -G "${USER}") " != *" $(get_group "$1") "* ]] } -# Please sync with 'test_ruby()' in 'Library/Homebrew/utils/ruby.sh' from Homebrew/brew repository. +# Please sync with 'test_ruby()' in 'Library/Homebrew/utils/ruby.sh' from the Homebrew/brew repository. test_ruby() { if [[ ! -x "$1" ]] then @@ -326,7 +360,7 @@ test_git() { version_ge "$(major_minor "${git_version_output##* }")" "$(major_minor "${REQUIRED_GIT_VERSION}")" } -# Search given executable in PATH (remove dependency for `which` command) +# Search for the given executable in PATH (avoids a dependency on the `which` command) which() { # Alias to Bash built-in command `type -P` type -P "$@" @@ -365,13 +399,13 @@ outdated_glibc() { if [[ -n "${HOMEBREW_ON_LINUX-}" ]] && no_usable_ruby && outdated_glibc then abort "$( - cat <<-EOFABORT + cat </dev/null then - ohai "Select the Homebrew installation directory" + ohai "Select a Homebrew installation directory:" echo "- ${tty_bold}Enter your password${tty_reset} to install to ${tty_underline}${HOMEBREW_PREFIX_DEFAULT}${tty_reset} (${tty_bold}recommended${tty_reset})" echo "- ${tty_bold}Press Control-D${tty_reset} to install to ${tty_underline}${HOME}/.linuxbrew${tty_reset}" echo "- ${tty_bold}Press Control-C${tty_reset} to cancel installation" @@ -507,7 +541,7 @@ if [[ -d "${HOMEBREW_PREFIX}" && ! -x "${HOMEBREW_PREFIX}" ]] then abort "$( cat </dev/null || return execute "git" "init" "-q" @@ -922,7 +956,7 @@ echo "$( cat <