Skip to content

Commit

Permalink
Add support for Homebrew wrappers
Browse files Browse the repository at this point in the history
Allow the ability for a system administrator to use
`HOMEBREW_BREW_WRAPPER` and `HOMEBREW_FORCE_BREW_WRAPPER` variables to
enforce the usage of a particular `brew` command for non-trivial (e.g.
`brew --prefix` is considered trivial, it doesn't need to write to the
prefix) Homebrew commands.

This also introduces a `HOMEBREW_ORIGINAL_BREW_FILE` variable for some
internal usage; `HOMEBREW_BREW_FILE` was being used internally for
both "how should we shell out to Homebrew" and "what should we use
to check permissions on Homebrew". `HOMEBREW_ORIGINAL_BREW_FILE` is
now used just for the latter case.

Inspired by conversation in
Homebrew/homebrew-bundle#1551 which suggested
this was worth fixing in wider than just `brew bundle`.
  • Loading branch information
MikeMcQuaid committed Jan 7, 2025
1 parent d87d336 commit 5a921f9
Show file tree
Hide file tree
Showing 11 changed files with 98 additions and 18 deletions.
50 changes: 43 additions & 7 deletions Library/Homebrew/brew.sh
Original file line number Diff line number Diff line change
Expand Up @@ -127,12 +127,6 @@ case "$1" in
homebrew-shellenv "$1"
exit 0
;;
setup-ruby)
source "${HOMEBREW_LIBRARY}/Homebrew/cmd/setup-ruby.sh"
shift
homebrew-setup-ruby "$1"
exit 0
;;
esac

source "${HOMEBREW_LIBRARY}/Homebrew/help.sh"
Expand Down Expand Up @@ -184,8 +178,50 @@ case "$@" in
;;
esac

source "${HOMEBREW_LIBRARY}/Homebrew/utils/helpers.sh"
if [[ -n "${HOMEBREW_FORCE_BREW_WRAPPER}" ]]
then
if [[ -z "${HOMEBREW_BREW_WRAPPER:-}" ]]
then
odie <<EOS
HOMEBREW_FORCE_BREW_WRAPPER was set to
${HOMEBREW_FORCE_BREW_WRAPPER}
but HOMEBREW_BREW_WRAPPER was unset. This indicates that you are running
${HOMEBREW_BREW_FILE}
directly but should instead run
${HOMEBREW_FORCE_BREW_WRAPPER}
EOS
elif [[ "${HOMEBREW_FORCE_BREW_WRAPPER:-}" != "${HOMEBREW_BREW_WRAPPER:-}" ]]
then
odie <<EOS
HOMEBREW_FORCE_BREW_WRAPPER was set to
${HOMEBREW_FORCE_BREW_WRAPPER}
but HOMEBREW_BREW_WRAPPER was set to
${HOMEBREW_BREW_WRAPPER}
This indicates that you are running
${HOMEBREW_BREW_FILE}
directly but should instead run:
${HOMEBREW_FORCE_BREW_WRAPPER}
EOS
fi
fi

# commands that take a single or no arguments and need to write to HOMEBREW_PREFIX.
# HOMEBREW_LIBRARY set by bin/brew
# shellcheck disable=SC2154
# doesn't need a default case as other arguments handled elsewhere.
# shellcheck disable=SC2249
case "$1" in
setup-ruby)
source "${HOMEBREW_LIBRARY}/Homebrew/cmd/setup-ruby.sh"
shift
homebrew-setup-ruby "$1"
exit 0
;;
esac

#####
##### Next, define all helper functions.
##### Next, define all other helper functions.
#####
source "${HOMEBREW_LIBRARY}/Homebrew/utils/helpers.sh"

Expand Down
7 changes: 7 additions & 0 deletions Library/Homebrew/env_config.rb
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,9 @@ module EnvConfig
description: "Use this URL as the Homebrew/brew `git`(1) remote.",
default: HOMEBREW_BREW_DEFAULT_GIT_REMOTE,
},
HOMEBREW_BREW_WRAPPER: {
description: "If set, use wrapper to call `brew` rather than auto-detecting it.",
},
HOMEBREW_BROWSER: {
description: "Use this as the browser when opening project homepages.",
default_text: "`$BROWSER` or the OS's default browser.",
Expand Down Expand Up @@ -242,6 +245,10 @@ module EnvConfig
"Automatically set if the system version of `git` is too old.",
boolean: true,
},
HOMEBREW_FORCE_BREW_WRAPPER: {
description: "If set, require `HOMEBREW_BREW_WRAPPER` to be set to the same value as " \
"`HOMEBREW_FORCE_BREW_WRAPPER` for non-trivial `brew` commands.",
},
HOMEBREW_FORCE_VENDOR_RUBY: {
description: "If set, always use Homebrew's vendored, relocatable Ruby version even if the system version " \
"of Ruby is new enough.",
Expand Down
2 changes: 1 addition & 1 deletion Library/Homebrew/formula.rb
Original file line number Diff line number Diff line change
Expand Up @@ -1499,7 +1499,7 @@ def skip_clean?(path)
# @see .link_overwrite
def link_overwrite?(path)
# Don't overwrite files not created by Homebrew.
return false if path.stat.uid != HOMEBREW_BREW_FILE.stat.uid
return false if path.stat.uid != HOMEBREW_ORIGINAL_BREW_FILE.stat.uid

# Don't overwrite files belong to other keg except when that
# keg's formula is deleted.
Expand Down
2 changes: 1 addition & 1 deletion Library/Homebrew/global.rb
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,7 @@ def running_as_root?
end

def owner_uid
@owner_uid ||= HOMEBREW_BREW_FILE.stat.uid
@owner_uid ||= HOMEBREW_ORIGINAL_BREW_FILE.stat.uid

Check warning on line 103 in Library/Homebrew/global.rb

View check run for this annotation

Codecov / codecov/patch

Library/Homebrew/global.rb#L103

Added line #L103 was not covered by tests
end

def running_as_root_but_not_owned_by_root?
Expand Down
4 changes: 2 additions & 2 deletions Library/Homebrew/mktemp.rb
Original file line number Diff line number Diff line change
Expand Up @@ -67,8 +67,8 @@ def run(chdir: true, &_block)
# Reference from `man 2 open`
# > When a new file is created, it is given the group of the directory which
# contains it.
group_id = if HOMEBREW_BREW_FILE.grpowned?
HOMEBREW_BREW_FILE.stat.gid
group_id = if HOMEBREW_ORIGINAL_BREW_FILE.grpowned?
HOMEBREW_ORIGINAL_BREW_FILE.stat.gid
else
Process.gid
end
Expand Down
2 changes: 1 addition & 1 deletion Library/Homebrew/sandbox.rb
Original file line number Diff line number Diff line change
Expand Up @@ -103,7 +103,7 @@ def allow_write_log(formula)

sig { void }
def deny_write_homebrew_repository
deny_write path: HOMEBREW_BREW_FILE
deny_write path: HOMEBREW_ORIGINAL_BREW_FILE

Check warning on line 106 in Library/Homebrew/sandbox.rb

View check run for this annotation

Codecov / codecov/patch

Library/Homebrew/sandbox.rb#L106

Added line #L106 was not covered by tests
if HOMEBREW_PREFIX.to_s == HOMEBREW_REPOSITORY.to_s
deny_write_path HOMEBREW_LIBRARY
deny_write_path HOMEBREW_REPOSITORY/".git"
Expand Down
6 changes: 6 additions & 0 deletions Library/Homebrew/sorbet/rbi/dsl/homebrew/env_config.rbi

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 5 additions & 0 deletions Library/Homebrew/startup/config.rb
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,11 @@
raise "HOMEBREW_BREW_FILE was not exported! Please call bin/brew directly!" unless ENV["HOMEBREW_BREW_FILE"]

# Path to `bin/brew` main executable in `HOMEBREW_PREFIX`
# Used for e.g. permissions checks.
HOMEBREW_ORIGINAL_BREW_FILE = Pathname(ENV.fetch("HOMEBREW_ORIGINAL_BREW_FILE")).freeze

Check warning on line 8 in Library/Homebrew/startup/config.rb

View check run for this annotation

Codecov / codecov/patch

Library/Homebrew/startup/config.rb#L8

Added line #L8 was not covered by tests

# Path to the executable that should be used to run `brew`.
# This may be HOMEBREW_ORIGINAL_BREW_FILE or HOMEBREW_BREW_WRAPPER.
HOMEBREW_BREW_FILE = Pathname(ENV.fetch("HOMEBREW_BREW_FILE")).freeze

# Where we link under
Expand Down
2 changes: 1 addition & 1 deletion Library/Homebrew/style.rb
Original file line number Diff line number Diff line change
Expand Up @@ -304,7 +304,7 @@ def self.json_result!(result)

def self.shell_scripts
[
HOMEBREW_BREW_FILE,
HOMEBREW_ORIGINAL_BREW_FILE,

Check warning on line 307 in Library/Homebrew/style.rb

View check run for this annotation

Codecov / codecov/patch

Library/Homebrew/style.rb#L307

Added line #L307 was not covered by tests
HOMEBREW_REPOSITORY/"completions/bash/brew",
HOMEBREW_REPOSITORY/"Dockerfile",
*HOMEBREW_REPOSITORY.glob(".devcontainer/**/*.sh"),
Expand Down
1 change: 1 addition & 0 deletions Library/Homebrew/test/support/lib/startup/config.rb
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@

raise "HOMEBREW_BREW_FILE was not exported! Please call bin/brew directly!" unless ENV["HOMEBREW_BREW_FILE"]

HOMEBREW_ORIGINAL_BREW_FILE = Pathname.new(ENV.fetch("HOMEBREW_ORIGINAL_BREW_FILE")).freeze
HOMEBREW_BREW_FILE = Pathname.new(ENV.fetch("HOMEBREW_BREW_FILE")).freeze

TEST_TMPDIR = ENV.fetch("HOMEBREW_TEST_TMPDIR") do |k|
Expand Down
35 changes: 30 additions & 5 deletions bin/brew
Original file line number Diff line number Diff line change
Expand Up @@ -116,6 +116,23 @@ unset BREW_FILE_DIRECTORY
# keg_relocate.rb, formula_cellar_checks.rb, and test/global_spec.rb need to change.
HOMEBREW_LIBRARY="${HOMEBREW_REPOSITORY}/Library"

# Use HOMEBREW_BREW_WRAPPER if set.
export HOMEBREW_ORIGINAL_BREW_FILE="${HOMEBREW_BREW_FILE}"
if [[ -n "${HOMEBREW_BREW_WRAPPER:-}" ]]
then
HOMEBREW_BREW_FILE="${HOMEBREW_BREW_WRAPPER}"
fi

# These variables are exported in this file and are not allowed to be overridden by the user.
BIN_BREW_EXPORTED_VARS=(
HOMEBREW_BREW_FILE
HOMEBREW_PREFIX
HOMEBREW_REPOSITORY
HOMEBREW_LIBRARY
HOMEBREW_USER_CONFIG_HOME
HOMEBREW_ORIGINAL_BREW_FILE
)

# Load Homebrew's variable configuration files from disk.
export_homebrew_env_file() {
local env_file
Expand All @@ -126,6 +143,15 @@ export_homebrew_env_file() {
do
# only load HOMEBREW_* lines
[[ "${line}" = "HOMEBREW_"* ]] || continue

# forbid overriding variables that are set in this file
local invalid_variable
for VAR in "${BIN_BREW_EXPORTED_VARS[@]}"
do
[[ "${line}" = "${VAR}"* ]] && invalid_variable="${VAR}"
done
[[ -n "${invalid_variable}" ]] && continue

export "${line?}"
done <"${env_file}"
}
Expand Down Expand Up @@ -212,11 +238,10 @@ done

unset VAR VAR_NEW MANPAGE_VARS USED_BY_HOMEBREW_VARS

export HOMEBREW_BREW_FILE
export HOMEBREW_PREFIX
export HOMEBREW_REPOSITORY
export HOMEBREW_LIBRARY
export HOMEBREW_USER_CONFIG_HOME
for VAR in "${BIN_BREW_EXPORTED_VARS[@]}"
do
export "${VAR?}"
done

# set from user environment
# shellcheck disable=SC2154
Expand Down

0 comments on commit 5a921f9

Please sign in to comment.