2023-03-06 09:49:53 -08:00
|
|
|
# typed: true
|
2019-04-19 15:38:03 +09:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2020-09-25 21:13:26 +02:00
|
|
|
require "cask/config"
|
2020-09-27 22:53:01 +02:00
|
|
|
require "cask/cmd"
|
|
|
|
require "cask/cmd/install"
|
2017-03-18 17:02:08 +02:00
|
|
|
require "missing_formula"
|
2014-06-16 13:36:17 -05:00
|
|
|
require "formula_installer"
|
2016-07-29 20:31:32 -06:00
|
|
|
require "development_tools"
|
2018-06-05 23:19:18 -04:00
|
|
|
require "install"
|
2019-01-03 16:23:44 +00:00
|
|
|
require "cleanup"
|
2019-04-17 18:25:08 +09:00
|
|
|
require "cli/parser"
|
2020-07-02 12:53:52 +01:00
|
|
|
require "upgrade"
|
2010-09-11 20:22:54 +01:00
|
|
|
|
2014-06-18 22:41:47 -05:00
|
|
|
module Homebrew
|
2020-10-20 12:03:48 +02:00
|
|
|
extend T::Sig
|
|
|
|
|
2020-07-25 21:33:48 +02:00
|
|
|
module_function
|
|
|
|
|
2020-10-20 12:03:48 +02:00
|
|
|
sig { returns(CLI::Parser) }
|
2019-01-26 16:44:03 +05:30
|
|
|
def install_args
|
2023-02-21 00:25:02 +00:00
|
|
|
Homebrew::CLI::Parser.new do
|
2021-01-15 15:04:02 -05:00
|
|
|
description <<~EOS
|
2020-09-28 01:31:47 +02:00
|
|
|
Install a <formula> or <cask>. Additional options specific to a <formula> may be
|
|
|
|
appended to the command.
|
2019-01-26 16:44:03 +05:30
|
|
|
|
2021-11-25 09:10:59 +00:00
|
|
|
Unless `HOMEBREW_NO_INSTALLED_DEPENDENTS_CHECK` is set, `brew upgrade` or `brew reinstall` will be run for
|
|
|
|
outdated dependents and dependents with broken linkage, respectively.
|
|
|
|
|
2020-09-28 01:31:47 +02:00
|
|
|
Unless `HOMEBREW_NO_INSTALL_CLEANUP` is set, `brew cleanup` will then be run for
|
|
|
|
the installed formulae or, every 30 days, for all formulae.
|
2021-11-25 09:10:59 +00:00
|
|
|
|
2023-02-10 23:15:40 -05:00
|
|
|
Unless `HOMEBREW_NO_INSTALL_UPGRADE` is set, `brew install` <formula> will upgrade <formula> if it
|
2021-11-25 09:10:59 +00:00
|
|
|
is already installed but outdated.
|
2019-01-26 16:44:03 +05:30
|
|
|
EOS
|
2020-07-30 18:40:10 +02:00
|
|
|
switch "-d", "--debug",
|
2020-10-08 11:18:24 +02:00
|
|
|
description: "If brewing fails, open an interactive debugging session with access to IRB " \
|
2019-08-06 14:22:24 -04:00
|
|
|
"or a shell inside the temporary build directory."
|
2020-07-27 03:59:52 +02:00
|
|
|
switch "-f", "--force",
|
2020-10-08 11:18:24 +02:00
|
|
|
description: "Install formulae without checking for previously installed keg-only or " \
|
2022-06-28 10:09:59 +01:00
|
|
|
"non-migrated versions. When installing casks, overwrite existing files " \
|
2021-01-07 21:23:06 +00:00
|
|
|
"(binaries and symlinks are excluded, unless originally from the same cask)."
|
2020-07-30 18:40:10 +02:00
|
|
|
switch "-v", "--verbose",
|
2019-04-30 08:44:35 +01:00
|
|
|
description: "Print the verification and postinstall steps."
|
2022-08-15 21:48:57 +09:00
|
|
|
switch "-n", "--dry-run",
|
|
|
|
description: "Show what would be installed, but do not actually install anything."
|
2020-10-08 11:18:24 +02:00
|
|
|
[
|
|
|
|
[:switch, "--formula", "--formulae", {
|
|
|
|
description: "Treat all named arguments as formulae.",
|
|
|
|
}],
|
|
|
|
[:flag, "--env=", {
|
2021-04-09 09:30:36 +01:00
|
|
|
description: "Disabled other than for internal Homebrew use.",
|
2021-12-17 00:17:15 +00:00
|
|
|
hidden: true,
|
2020-10-08 11:18:24 +02:00
|
|
|
}],
|
|
|
|
[:switch, "--ignore-dependencies", {
|
|
|
|
description: "An unsupported Homebrew development flag to skip installing any dependencies of any kind. " \
|
|
|
|
"If the dependencies are not already present, the formula will have issues. If you're not " \
|
|
|
|
"developing Homebrew, consider adjusting your PATH rather than using this flag.",
|
|
|
|
}],
|
|
|
|
[:switch, "--only-dependencies", {
|
|
|
|
description: "Install the dependencies with specified options but do not install the " \
|
|
|
|
"formula itself.",
|
|
|
|
}],
|
|
|
|
[:flag, "--cc=", {
|
|
|
|
description: "Attempt to compile using the specified <compiler>, which should be the name of the " \
|
|
|
|
"compiler's executable, e.g. `gcc-7` for GCC 7. In order to use LLVM's clang, specify " \
|
|
|
|
"`llvm_clang`. To use the Apple-provided clang, specify `clang`. This option will only " \
|
|
|
|
"accept compilers that are provided by Homebrew or bundled with macOS. Please do not " \
|
|
|
|
"file issues if you encounter errors while using this option.",
|
|
|
|
}],
|
|
|
|
[:switch, "-s", "--build-from-source", {
|
|
|
|
description: "Compile <formula> from source even if a bottle is provided. " \
|
|
|
|
"Dependencies will still be installed from bottles if they are available.",
|
|
|
|
}],
|
|
|
|
[:switch, "--force-bottle", {
|
|
|
|
description: "Install from a bottle if it exists for the current or newest version of " \
|
|
|
|
"macOS, even if it would not normally be used for installation.",
|
|
|
|
}],
|
|
|
|
[:switch, "--include-test", {
|
|
|
|
description: "Install testing dependencies required to run `brew test` <formula>.",
|
|
|
|
}],
|
|
|
|
[:switch, "--HEAD", {
|
2021-07-06 23:48:42 +05:30
|
|
|
description: "If <formula> defines it, install the HEAD version, aka. main, trunk, unstable, master.",
|
2020-10-08 11:18:24 +02:00
|
|
|
}],
|
|
|
|
[:switch, "--fetch-HEAD", {
|
|
|
|
description: "Fetch the upstream repository to detect if the HEAD installation of the " \
|
|
|
|
"formula is outdated. Otherwise, the repository's HEAD will only be checked for " \
|
|
|
|
"updates when a new stable or development version has been released.",
|
|
|
|
}],
|
|
|
|
[:switch, "--keep-tmp", {
|
|
|
|
description: "Retain the temporary files created during installation.",
|
|
|
|
}],
|
2022-07-26 12:15:53 +01:00
|
|
|
[:switch, "--debug-symbols", {
|
|
|
|
depends_on: "--build-from-source",
|
2023-02-10 23:15:40 -05:00
|
|
|
description: "Generate debug symbols on build. Source will be retained in a cache directory.",
|
2022-07-26 12:15:53 +01:00
|
|
|
}],
|
2020-10-08 11:18:24 +02:00
|
|
|
[:switch, "--build-bottle", {
|
|
|
|
description: "Prepare the formula for eventual bottling during installation, skipping any " \
|
|
|
|
"post-install steps.",
|
|
|
|
}],
|
2023-03-23 15:22:27 +00:00
|
|
|
[:switch, "--skip-post-install", {
|
|
|
|
description: "Install but skip any post-install steps.",
|
|
|
|
}],
|
2020-10-08 11:18:24 +02:00
|
|
|
[:flag, "--bottle-arch=", {
|
|
|
|
depends_on: "--build-bottle",
|
|
|
|
description: "Optimise bottles for the specified architecture rather than the oldest " \
|
|
|
|
"architecture supported by the version of macOS the bottles are built on.",
|
|
|
|
}],
|
|
|
|
[:switch, "--display-times", {
|
|
|
|
env: :display_install_times,
|
2021-09-02 08:42:48 -07:00
|
|
|
description: "Print install times for each package at the end of the run.",
|
2020-10-08 11:18:24 +02:00
|
|
|
}],
|
|
|
|
[:switch, "-i", "--interactive", {
|
|
|
|
description: "Download and patch <formula>, then open a shell. This allows the user to " \
|
|
|
|
"run `./configure --help` and otherwise determine how to turn the software " \
|
|
|
|
"package into a Homebrew package.",
|
|
|
|
}],
|
|
|
|
[:switch, "-g", "--git", {
|
|
|
|
description: "Create a Git repository, useful for creating patches to the software.",
|
|
|
|
}],
|
2022-01-10 10:05:44 -06:00
|
|
|
[:switch, "--overwrite", {
|
|
|
|
description: "Delete files that already exist in the prefix while linking.",
|
|
|
|
}],
|
2022-10-08 01:08:15 +01:00
|
|
|
].each do |args|
|
|
|
|
options = args.pop
|
2020-10-03 02:45:32 +02:00
|
|
|
send(*args, **options)
|
|
|
|
conflicts "--cask", args.last
|
2020-09-25 21:13:26 +02:00
|
|
|
end
|
2020-10-03 02:45:32 +02:00
|
|
|
formula_options
|
2020-10-08 11:18:24 +02:00
|
|
|
[
|
|
|
|
[:switch, "--cask", "--casks", { description: "Treat all named arguments as casks." }],
|
2022-10-08 01:08:15 +01:00
|
|
|
*Cask::Cmd::AbstractCommand::OPTIONS.map(&:dup),
|
|
|
|
*Cask::Cmd::Install::OPTIONS.map(&:dup),
|
|
|
|
].each do |args|
|
|
|
|
options = args.pop
|
2020-10-03 02:45:32 +02:00
|
|
|
send(*args, **options)
|
|
|
|
conflicts "--formula", args.last
|
2020-09-25 21:13:26 +02:00
|
|
|
end
|
2020-10-03 02:45:32 +02:00
|
|
|
cask_options
|
2020-09-25 21:13:26 +02:00
|
|
|
|
2020-10-08 11:18:24 +02:00
|
|
|
conflicts "--ignore-dependencies", "--only-dependencies"
|
|
|
|
conflicts "--build-from-source", "--build-bottle", "--force-bottle"
|
2022-10-21 23:28:51 -04:00
|
|
|
conflicts "--adopt", "--force"
|
2021-01-10 14:26:40 -05:00
|
|
|
|
|
|
|
named_args [:formula, :cask], min: 1
|
2014-02-11 21:25:26 -08:00
|
|
|
end
|
2019-01-26 16:44:03 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def install
|
2020-07-25 03:48:33 +02:00
|
|
|
args = install_args.parse
|
2019-12-15 23:50:01 +05:30
|
|
|
|
2020-12-28 09:45:06 +00:00
|
|
|
if args.env.present?
|
2021-04-09 09:30:36 +01:00
|
|
|
# Can't use `replacement: false` because `install_args` are used by
|
|
|
|
# `build.rb`. Instead, `hide_from_man_page` and don't do anything with
|
|
|
|
# this argument here.
|
|
|
|
odisabled "brew install --env", "`env :std` in specific formula files"
|
2020-12-28 09:45:06 +00:00
|
|
|
end
|
|
|
|
|
2020-10-07 12:12:10 +02:00
|
|
|
args.named.each do |name|
|
|
|
|
next if File.exist?(name)
|
2021-10-28 01:25:27 -04:00
|
|
|
next unless name =~ HOMEBREW_TAP_FORMULA_REGEX
|
2020-09-25 21:13:26 +02:00
|
|
|
|
2020-10-07 12:12:10 +02:00
|
|
|
tap = Tap.fetch(Regexp.last_match(1), Regexp.last_match(2))
|
2023-02-24 13:25:18 +00:00
|
|
|
next if (tap.core_tap? || tap == "homebrew/cask") && !EnvConfig.no_install_from_api?
|
2021-10-28 01:25:27 -04:00
|
|
|
|
2020-10-07 12:12:10 +02:00
|
|
|
tap.install unless tap.installed?
|
|
|
|
end
|
2019-07-25 21:20:49 +01:00
|
|
|
|
2019-04-02 08:18:30 +01:00
|
|
|
if args.ignore_dependencies?
|
|
|
|
opoo <<~EOS
|
2021-01-24 21:40:41 -05:00
|
|
|
#{Tty.bold}`--ignore-dependencies` is an unsupported Homebrew developer flag!#{Tty.reset}
|
2019-04-02 08:18:30 +01:00
|
|
|
Adjust your PATH to put any preferred versions of applications earlier in the
|
|
|
|
PATH rather than using this unsupported flag!
|
|
|
|
|
|
|
|
EOS
|
|
|
|
end
|
|
|
|
|
2021-01-20 12:13:22 -05:00
|
|
|
begin
|
2022-06-14 17:19:29 -04:00
|
|
|
formulae, casks = args.named.to_formulae_and_casks
|
2021-01-20 12:13:22 -05:00
|
|
|
.partition { |formula_or_cask| formula_or_cask.is_a?(Formula) }
|
|
|
|
rescue FormulaOrCaskUnavailableError, Cask::CaskUnavailableError => e
|
|
|
|
retry if Tap.install_default_cask_tap_if_necessary(force: args.cask?)
|
|
|
|
|
|
|
|
raise e
|
|
|
|
end
|
2014-10-31 18:33:39 -05:00
|
|
|
|
2020-09-25 21:13:26 +02:00
|
|
|
if casks.any?
|
|
|
|
Cask::Cmd::Install.install_casks(
|
|
|
|
*casks,
|
|
|
|
binaries: args.binaries?,
|
|
|
|
verbose: args.verbose?,
|
|
|
|
force: args.force?,
|
2022-10-21 23:28:51 -04:00
|
|
|
adopt: args.adopt?,
|
2020-09-25 21:13:26 +02:00
|
|
|
require_sha: args.require_sha?,
|
2021-01-10 20:19:31 +01:00
|
|
|
skip_cask_deps: args.skip_cask_deps?,
|
2020-09-25 21:13:26 +02:00
|
|
|
quarantine: args.quarantine?,
|
2021-09-10 23:11:41 +09:00
|
|
|
quiet: args.quiet?,
|
2022-08-15 21:48:57 +09:00
|
|
|
dry_run: args.dry_run?,
|
2020-09-25 21:13:26 +02:00
|
|
|
)
|
2019-01-29 08:30:22 +00:00
|
|
|
end
|
2014-11-28 15:02:42 +00:00
|
|
|
|
2019-01-29 08:30:22 +00:00
|
|
|
# if the user's flags will prevent bottle only-installations when no
|
|
|
|
# developer tools are available, we need to stop them early on
|
2023-02-17 14:33:53 +00:00
|
|
|
build_flags = []
|
2021-03-22 16:11:27 +00:00
|
|
|
unless DevelopmentTools.installed?
|
|
|
|
build_flags << "--HEAD" if args.HEAD?
|
|
|
|
build_flags << "--build-bottle" if args.build_bottle?
|
|
|
|
build_flags << "--build-from-source" if args.build_from_source?
|
|
|
|
|
|
|
|
raise BuildFlagsError.new(build_flags, bottled: formulae.all?(&:bottled?)) if build_flags.present?
|
|
|
|
end
|
2014-06-16 13:38:14 -05:00
|
|
|
|
2023-02-17 14:33:53 +00:00
|
|
|
if build_flags.present? && !Homebrew::EnvConfig.developer?
|
|
|
|
opoo "building from source is not supported!"
|
|
|
|
puts "You're on your own. Failures are expected so don't create any issues, please!"
|
|
|
|
end
|
|
|
|
|
2021-06-10 10:50:05 -04:00
|
|
|
installed_formulae = formulae.select do |f|
|
|
|
|
Install.install_formula?(
|
|
|
|
f,
|
|
|
|
head: args.HEAD?,
|
|
|
|
fetch_head: args.fetch_HEAD?,
|
|
|
|
only_dependencies: args.only_dependencies?,
|
|
|
|
force: args.force?,
|
|
|
|
quiet: args.quiet?,
|
|
|
|
)
|
2019-01-29 08:30:22 +00:00
|
|
|
end
|
2014-06-16 13:38:14 -05:00
|
|
|
|
2020-09-25 21:13:26 +02:00
|
|
|
return if installed_formulae.empty?
|
2018-09-17 02:45:00 +02:00
|
|
|
|
2020-07-27 11:37:07 +02:00
|
|
|
Install.perform_preinstall_checks(cc: args.cc)
|
2014-03-15 10:40:18 -05:00
|
|
|
|
2021-09-01 22:27:34 -07:00
|
|
|
Install.install_formulae(
|
|
|
|
installed_formulae,
|
|
|
|
build_bottle: args.build_bottle?,
|
|
|
|
force_bottle: args.force_bottle?,
|
|
|
|
bottle_arch: args.bottle_arch,
|
|
|
|
ignore_deps: args.ignore_dependencies?,
|
|
|
|
only_deps: args.only_dependencies?,
|
|
|
|
include_test_formulae: args.include_test_formulae,
|
|
|
|
build_from_source_formulae: args.build_from_source_formulae,
|
|
|
|
cc: args.cc,
|
|
|
|
git: args.git?,
|
|
|
|
interactive: args.interactive?,
|
|
|
|
keep_tmp: args.keep_tmp?,
|
2022-07-26 12:15:53 +01:00
|
|
|
debug_symbols: args.debug_symbols?,
|
2021-09-01 22:27:34 -07:00
|
|
|
force: args.force?,
|
2022-01-10 10:05:44 -06:00
|
|
|
overwrite: args.overwrite?,
|
2021-09-01 22:27:34 -07:00
|
|
|
debug: args.debug?,
|
|
|
|
quiet: args.quiet?,
|
|
|
|
verbose: args.verbose?,
|
2022-08-16 20:59:01 +09:00
|
|
|
dry_run: args.dry_run?,
|
2023-03-30 21:08:40 -04:00
|
|
|
skip_post_install: args.skip_post_install?,
|
2021-09-01 22:27:34 -07:00
|
|
|
)
|
2020-07-02 12:53:52 +01:00
|
|
|
|
2021-03-18 14:46:48 +00:00
|
|
|
Upgrade.check_installed_dependents(
|
|
|
|
installed_formulae,
|
|
|
|
flags: args.flags_only,
|
|
|
|
installed_on_request: args.named.present?,
|
|
|
|
force_bottle: args.force_bottle?,
|
|
|
|
build_from_source_formulae: args.build_from_source_formulae,
|
|
|
|
interactive: args.interactive?,
|
|
|
|
keep_tmp: args.keep_tmp?,
|
2022-07-26 12:15:53 +01:00
|
|
|
debug_symbols: args.debug_symbols?,
|
2021-03-18 14:46:48 +00:00
|
|
|
force: args.force?,
|
|
|
|
debug: args.debug?,
|
|
|
|
quiet: args.quiet?,
|
|
|
|
verbose: args.verbose?,
|
2022-08-16 20:59:01 +09:00
|
|
|
dry_run: args.dry_run?,
|
2021-03-18 14:46:48 +00:00
|
|
|
)
|
2020-07-02 12:53:52 +01:00
|
|
|
|
2022-08-16 20:59:01 +09:00
|
|
|
Cleanup.periodic_clean!(dry_run: args.dry_run?)
|
2022-07-20 08:24:55 -07:00
|
|
|
|
2020-07-25 21:33:48 +02:00
|
|
|
Homebrew.messages.display_messages(display_times: args.display_times?)
|
2019-01-29 08:30:22 +00:00
|
|
|
rescue FormulaUnreadableError, FormulaClassUnavailableError,
|
2019-01-29 09:38:20 +00:00
|
|
|
TapFormulaUnreadableError, TapFormulaClassUnavailableError => e
|
2019-01-29 08:30:22 +00:00
|
|
|
# Need to rescue before `FormulaUnavailableError` (superclass of this)
|
|
|
|
# is handled, as searching for a formula doesn't make sense here (the
|
|
|
|
# formula was found, but there's a problem with its implementation).
|
2020-06-25 22:17:45 +02:00
|
|
|
$stderr.puts e.backtrace if Homebrew::EnvConfig.developer?
|
2019-01-29 08:30:22 +00:00
|
|
|
ofail e.message
|
2022-12-10 12:59:06 -08:00
|
|
|
rescue FormulaOrCaskUnavailableError, Cask::CaskUnavailableError => e
|
2023-01-20 19:53:00 -08:00
|
|
|
Homebrew.failed = true
|
|
|
|
|
2022-12-10 12:59:06 -08:00
|
|
|
# formula name or cask token
|
2023-03-06 09:49:53 -08:00
|
|
|
name = case e
|
|
|
|
when FormulaOrCaskUnavailableError then e.name
|
|
|
|
when Cask::CaskUnavailableError then e.token
|
|
|
|
else T.absurd(e)
|
|
|
|
end
|
2022-12-10 12:59:06 -08:00
|
|
|
|
|
|
|
if name == "updog"
|
2019-01-29 08:30:22 +00:00
|
|
|
ofail "What's updog?"
|
|
|
|
return
|
|
|
|
end
|
2016-12-25 18:54:08 -05:00
|
|
|
|
2021-07-12 12:07:10 +01:00
|
|
|
opoo e
|
2022-12-12 19:51:18 -08:00
|
|
|
|
|
|
|
reason = MissingFormula.reason(name, silent: true)
|
|
|
|
if !args.cask? && reason
|
|
|
|
$stderr.puts reason
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2022-12-13 20:31:33 -08:00
|
|
|
# We don't seem to get good search results when the tap is specified
|
|
|
|
# so we might as well return early.
|
2022-12-12 19:51:18 -08:00
|
|
|
return if name.include?("/")
|
|
|
|
|
2022-12-17 11:03:18 -08:00
|
|
|
require "search"
|
2022-12-31 15:52:05 -08:00
|
|
|
|
|
|
|
package_types = []
|
|
|
|
package_types << "formulae" unless args.cask?
|
|
|
|
package_types << "casks" unless args.formula?
|
|
|
|
|
|
|
|
ohai "Searching for similarly named #{package_types.join(" and ")}..."
|
2017-03-22 21:55:29 +00:00
|
|
|
|
2022-12-10 12:59:06 -08:00
|
|
|
# Don't treat formula/cask name as a regex
|
|
|
|
query = string_or_regex = name
|
2022-12-17 11:03:18 -08:00
|
|
|
all_formulae, all_casks = Search.search_names(query, string_or_regex, args)
|
2022-12-12 19:51:18 -08:00
|
|
|
|
2022-12-13 20:38:00 -08:00
|
|
|
if all_formulae.any?
|
2022-12-12 19:51:18 -08:00
|
|
|
ohai "Formulae", Formatter.columns(all_formulae)
|
|
|
|
first_formula = all_formulae.first.to_s
|
|
|
|
puts <<~EOS
|
|
|
|
|
|
|
|
To install #{first_formula}, run:
|
|
|
|
brew install #{first_formula}
|
|
|
|
EOS
|
2013-09-05 12:24:21 -07:00
|
|
|
end
|
2022-12-13 20:38:00 -08:00
|
|
|
puts if all_formulae.any? && all_casks.any?
|
|
|
|
if all_casks.any?
|
2022-12-12 19:51:18 -08:00
|
|
|
ohai "Casks", Formatter.columns(all_casks)
|
|
|
|
first_cask = all_casks.first.to_s
|
|
|
|
puts <<~EOS
|
|
|
|
|
|
|
|
To install #{first_cask}, run:
|
|
|
|
brew install --cask #{first_cask}
|
|
|
|
EOS
|
|
|
|
end
|
2022-12-13 20:38:00 -08:00
|
|
|
return if all_formulae.any? || all_casks.any?
|
2022-12-12 19:51:18 -08:00
|
|
|
|
2022-12-31 15:52:05 -08:00
|
|
|
odie "No #{package_types.join(" or ")} found for #{name}."
|
2010-09-11 20:22:54 +01:00
|
|
|
end
|
|
|
|
end
|