307 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			307 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: false # rubocop:todo Sorbet/TrueSigil
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "exceptions"
 | |
| require "extend/ENV"
 | |
| require "utils"
 | |
| require "PATH"
 | |
| 
 | |
| module Homebrew
 | |
|   module Bundle
 | |
|     module Commands
 | |
|       module Exec
 | |
|         # Homebrew's global environment variables that we don't want to leak into
 | |
|         # the `brew bundle exec` environment.
 | |
|         HOMEBREW_ENV_CLEANUP = %w[
 | |
|           HOMEBREW_HELP_MESSAGE
 | |
|           HOMEBREW_API_DEFAULT_DOMAIN
 | |
|           HOMEBREW_BOTTLE_DEFAULT_DOMAIN
 | |
|           HOMEBREW_BREW_DEFAULT_GIT_REMOTE
 | |
|           HOMEBREW_CORE_DEFAULT_GIT_REMOTE
 | |
|           HOMEBREW_DEFAULT_CACHE
 | |
|           HOMEBREW_DEFAULT_LOGS
 | |
|           HOMEBREW_DEFAULT_TEMP
 | |
|           HOMEBREW_REQUIRED_RUBY_VERSION
 | |
|           HOMEBREW_PRODUCT
 | |
|           HOMEBREW_SYSTEM
 | |
|           HOMEBREW_PROCESSOR
 | |
|           HOMEBREW_PHYSICAL_PROCESSOR
 | |
|           HOMEBREW_BREWED_CURL_PATH
 | |
|           HOMEBREW_USER_AGENT_CURL
 | |
|           HOMEBREW_USER_AGENT
 | |
|           HOMEBREW_GENERIC_DEFAULT_PREFIX
 | |
|           HOMEBREW_GENERIC_DEFAULT_REPOSITORY
 | |
|           HOMEBREW_DEFAULT_PREFIX
 | |
|           HOMEBREW_DEFAULT_REPOSITORY
 | |
|           HOMEBREW_AUTO_UPDATE_COMMAND
 | |
|           HOMEBREW_BREW_GIT_REMOTE
 | |
|           HOMEBREW_COMMAND_DEPTH
 | |
|           HOMEBREW_CORE_GIT_REMOTE
 | |
|           HOMEBREW_MACOS_VERSION_NUMERIC
 | |
|           HOMEBREW_MINIMUM_GIT_VERSION
 | |
|           HOMEBREW_MACOS_NEWEST_UNSUPPORTED
 | |
|           HOMEBREW_MACOS_OLDEST_SUPPORTED
 | |
|           HOMEBREW_MACOS_OLDEST_ALLOWED
 | |
|           HOMEBREW_GITHUB_PACKAGES_AUTH
 | |
|         ].freeze
 | |
| 
 | |
|         PATH_LIKE_ENV_REGEX = /.+#{File::PATH_SEPARATOR}/
 | |
| 
 | |
|         def self.run(*args, global: false, file: nil, subcommand: "", services: false)
 | |
|           # Cleanup Homebrew's global environment
 | |
|           HOMEBREW_ENV_CLEANUP.each { |key| ENV.delete(key) }
 | |
| 
 | |
|           # Store the old environment so we can check if things were already set
 | |
|           # before we start mutating it.
 | |
|           old_env = ENV.to_h
 | |
| 
 | |
|           # Setup Homebrew's ENV extensions
 | |
|           ENV.activate_extensions!
 | |
|           raise UsageError, "No command to execute was specified!" if args.blank?
 | |
| 
 | |
|           command = args.first
 | |
| 
 | |
|           require "bundle/brewfile"
 | |
|           @dsl = Brewfile.read(global:, file:)
 | |
| 
 | |
|           require "formula"
 | |
|           require "formulary"
 | |
| 
 | |
|           ENV.deps = @dsl.entries.filter_map do |entry|
 | |
|             next if entry.type != :brew
 | |
| 
 | |
|             Formulary.factory(entry.name)
 | |
|           end
 | |
| 
 | |
|           # Allow setting all dependencies to be keg-only
 | |
|           # (i.e. should be explicitly in HOMEBREW_*PATHs ahead of HOMEBREW_PREFIX)
 | |
|           ENV.keg_only_deps = if ENV["HOMEBREW_BUNDLE_EXEC_ALL_KEG_ONLY_DEPS"].present?
 | |
|             ENV.delete("HOMEBREW_BUNDLE_EXEC_ALL_KEG_ONLY_DEPS")
 | |
|             ENV.deps
 | |
|           else
 | |
|             ENV.deps.select(&:keg_only?)
 | |
|           end
 | |
|           ENV.setup_build_environment
 | |
| 
 | |
|           # Enable compiler flag filtering
 | |
|           ENV.refurbish_args
 | |
| 
 | |
|           # Set up `nodenv`, `pyenv` and `rbenv` if present.
 | |
|           env_formulae = %w[nodenv pyenv rbenv]
 | |
|           ENV.deps.each do |dep|
 | |
|             dep_name = dep.name
 | |
|             next unless env_formulae.include?(dep_name)
 | |
| 
 | |
|             dep_root = ENV.fetch("HOMEBREW_#{dep_name.upcase}_ROOT", "#{Dir.home}/.#{dep_name}")
 | |
|             ENV.prepend_path "PATH", Pathname.new(dep_root)/"shims"
 | |
|           end
 | |
| 
 | |
|           # Setup pkg-config, if present, to help locate packages
 | |
|           # Only need this on Linux as Homebrew provides a shim on macOS
 | |
|           # TODO: use extend/OS here
 | |
|           # rubocop:todo Homebrew/MoveToExtendOS
 | |
|           if OS.linux? && (pkgconf = Formulary.factory("pkgconf")) && pkgconf.any_version_installed?
 | |
|             ENV.prepend_path "PATH", pkgconf.opt_bin.to_s
 | |
|           end
 | |
|           # rubocop:enable Homebrew/MoveToExtendOS
 | |
| 
 | |
|           # For commands which aren't either absolute or relative
 | |
|           # Add the command directory to PATH, since it may get blown away by superenv
 | |
|           if command.exclude?("/") && (which_command = which(command)).present?
 | |
|             ENV.prepend_path "PATH", which_command.dirname.to_s
 | |
|           end
 | |
| 
 | |
|           # Replace the formula versions from the environment variables
 | |
|           ENV.deps.each do |formula|
 | |
|             formula_name = formula.name
 | |
|             formula_version = Bundle.formula_versions_from_env(formula_name)
 | |
|             next unless formula_version
 | |
| 
 | |
|             ENV.each do |key, value|
 | |
|               opt = %r{/opt/#{formula_name}([/:$])}
 | |
|               next unless value.match(opt)
 | |
| 
 | |
|               cellar = "/Cellar/#{formula_name}/#{formula_version}\\1"
 | |
| 
 | |
|               # Look for PATH-like environment variables
 | |
|               ENV[key] = if key.include?("PATH") && value.match?(PATH_LIKE_ENV_REGEX)
 | |
|                 rejected_opts = []
 | |
|                 path = PATH.new(ENV.fetch("PATH"))
 | |
|                            .reject do |path_value|
 | |
|                   rejected_opts << path_value if path_value.match?(opt)
 | |
|                 end
 | |
|                 rejected_opts.each do |path_value|
 | |
|                   path.prepend(path_value.gsub(opt, cellar))
 | |
|                 end
 | |
|                 path.to_s
 | |
|               else
 | |
|                 value.gsub(opt, cellar)
 | |
|               end
 | |
|             end
 | |
|           end
 | |
| 
 | |
|           # Ensure brew bundle sh/env commands have access to other tools in the PATH
 | |
|           if ["sh", "env"].include?(subcommand) && (homebrew_path = ENV.fetch("HOMEBREW_PATH", nil))
 | |
|             ENV.append_path "PATH", homebrew_path
 | |
|           end
 | |
| 
 | |
|           # For commands which aren't either absolute or relative
 | |
|           raise "command was not found in your PATH: #{command}" if command.exclude?("/") && which(command).nil?
 | |
| 
 | |
|           if subcommand == "env"
 | |
|             ENV.sort.each do |key, value|
 | |
|               # No need to export empty values.
 | |
|               next if value.blank?
 | |
| 
 | |
|               # Skip exporting non-Homebrew things that were already set in the old environment.
 | |
|               next if !key.start_with?("HOMEBREW_") && old_env.key?(key) && old_env[key] == value
 | |
| 
 | |
|               puts "export #{key}=\"#{Utils::Shell.sh_quote(value)}\""
 | |
|             end
 | |
|             return
 | |
|           end
 | |
| 
 | |
|           if services
 | |
|             require "bundle/brew_services"
 | |
| 
 | |
|             exit_code = 0
 | |
|             run_services(@dsl.entries) do
 | |
|               Kernel.system(*args)
 | |
|               exit_code = $CHILD_STATUS.exitstatus
 | |
|             end
 | |
|             exit!(exit_code)
 | |
|           else
 | |
|             exec(*args)
 | |
|           end
 | |
|         end
 | |
| 
 | |
|         sig {
 | |
|           params(
 | |
|             entries: T::Array[Homebrew::Bundle::Dsl::Entry],
 | |
|             _block:  T.proc.params(
 | |
|               info:                 T::Hash[String, T.anything],
 | |
|               service_file:         Pathname,
 | |
|               conflicting_services: T::Array[T::Hash[String, T.anything]],
 | |
|             ).void,
 | |
|           ).void
 | |
|         }
 | |
|         private_class_method def self.map_service_info(entries, &_block)
 | |
|           entries_formulae = entries.filter_map do |entry|
 | |
|             next if entry.type != :brew
 | |
| 
 | |
|             formula = Formula[entry.name]
 | |
|             next unless formula.any_version_installed?
 | |
| 
 | |
|             [entry, formula]
 | |
|           end.to_h
 | |
| 
 | |
|           conflicts = entries_formulae.to_h do |entry, formula|
 | |
|             [
 | |
|               entry,
 | |
|               (
 | |
|                 formula.versioned_formulae_names +
 | |
|                   formula.conflicts.map(&:name) +
 | |
|                   Array(entry.options[:conflicts_with])
 | |
|               ).uniq,
 | |
|             ]
 | |
|           end
 | |
| 
 | |
|           # The formula + everything that could possible conflict with the service
 | |
|           names_to_query = entries_formulae.flat_map do |entry, formula|
 | |
|             [
 | |
|               formula.name,
 | |
|               *conflicts.fetch(entry),
 | |
|             ]
 | |
|           end
 | |
| 
 | |
|           # We parse from a command invocation so that brew wrappers can invoke special actions
 | |
|           # for the elevated nature of `brew services`
 | |
|           services_info = JSON.parse(
 | |
|             Utils.safe_popen_read(HOMEBREW_BREW_FILE, "services", "info", "--json", *names_to_query),
 | |
|           )
 | |
| 
 | |
|           entries_formulae.filter_map do |entry, formula|
 | |
|             service_file = Bundle::BrewServices.versioned_service_file(entry.name)
 | |
| 
 | |
|             unless service_file&.file?
 | |
|               prefix = formula.any_installed_prefix
 | |
|               next if prefix.nil?
 | |
| 
 | |
|               service_file = if Homebrew::Services::System.launchctl?
 | |
|                 prefix/"#{formula.plist_name}.plist"
 | |
|               else
 | |
|                 prefix/"#{formula.service_name}.service"
 | |
|               end
 | |
|             end
 | |
| 
 | |
|             next unless service_file.file?
 | |
| 
 | |
|             info = services_info.find { |candidate| candidate["name"] == formula.name }
 | |
|             conflicting_services = services_info.select do |candidate|
 | |
|               next unless candidate["running"]
 | |
| 
 | |
|               conflicts.fetch(entry).include?(candidate["name"])
 | |
|             end
 | |
| 
 | |
|             raise "Failed to get service info for #{entry.name}" if info.nil?
 | |
| 
 | |
|             yield info, service_file, conflicting_services
 | |
|           end
 | |
|         end
 | |
| 
 | |
|         sig { params(entries: T::Array[Homebrew::Bundle::Dsl::Entry], _block: T.nilable(T.proc.void)).void }
 | |
|         private_class_method def self.run_services(entries, &_block)
 | |
|           services_to_restart = []
 | |
| 
 | |
|           map_service_info(entries) do |info, service_file, conflicting_services|
 | |
|             if info["running"] && !Bundle::BrewServices.stop(info["name"], keep: true)
 | |
|               opoo "Failed to stop #{info["name"]} service"
 | |
|             end
 | |
| 
 | |
|             conflicting_services.each do |conflict|
 | |
|               if Bundle::BrewServices.stop(conflict["name"], keep: true)
 | |
|                 services_to_restart << conflict["name"] if conflict["registered"]
 | |
|               else
 | |
|                 opoo "Failed to stop #{conflict["name"]} service"
 | |
|               end
 | |
|             end
 | |
| 
 | |
|             unless Bundle::BrewServices.run(info["name"], file: service_file)
 | |
|               opoo "Failed to start #{info["name"]} service"
 | |
|             end
 | |
|           end
 | |
| 
 | |
|           return unless block_given?
 | |
| 
 | |
|           begin
 | |
|             yield
 | |
|           ensure
 | |
|             # Do a full re-evaluation of services instead state has changed
 | |
|             stop_services(entries)
 | |
| 
 | |
|             services_to_restart.each do |service|
 | |
|               next if Bundle::BrewServices.run(service)
 | |
| 
 | |
|               opoo "Failed to restart #{service} service"
 | |
|             end
 | |
|           end
 | |
|         end
 | |
| 
 | |
|         sig { params(entries: T::Array[Homebrew::Bundle::Dsl::Entry]).void }
 | |
|         private_class_method def self.stop_services(entries)
 | |
|           map_service_info(entries) do |info, _, _|
 | |
|             next unless info["loaded"]
 | |
| 
 | |
|             # Try avoid services not started by `brew bundle services`
 | |
|             next if Homebrew::Services::System.launchctl? && info["registered"]
 | |
| 
 | |
|             if info["running"] && !Bundle::BrewServices.stop(info["name"], keep: true)
 | |
|               opoo "Failed to stop #{info["name"]} service"
 | |
|             end
 | |
|           end
 | |
|         end
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 | 
