From ce45868973620a1fed4505e0fc9c2f9a928bdff0 Mon Sep 17 00:00:00 2001
From: Jonathan Chang <me@jonathanchang.org>
Date: Sat, 9 Jan 2021 13:34:17 +1100
Subject: [PATCH] Improve documentation and add type signatures.

---
 Library/Homebrew/formula.rb         | 180 ++++++++++++++++++----------
 Library/Homebrew/utils/inreplace.rb |  17 ++-
 docs/Bottles.md                     |   5 -
 3 files changed, 128 insertions(+), 74 deletions(-)

diff --git a/Library/Homebrew/formula.rb b/Library/Homebrew/formula.rb
index 347a211271..37c08fb136 100644
--- a/Library/Homebrew/formula.rb
+++ b/Library/Homebrew/formula.rb
@@ -67,15 +67,6 @@ class Formula
   extend Predicable
 
   # @!method inreplace(paths, before = nil, after = nil)
-  # Actually implemented in {Utils::Inreplace.inreplace}.
-  # Sometimes we have to change a bit before we install. Mostly we
-  # prefer a patch but if you need the `prefix` of this formula in the
-  # patch you have to resort to `inreplace`, because in the patch
-  # you don't have access to any variable defined by the formula. Only
-  # `HOMEBREW_PREFIX` is available in the embedded patch.
-  #
-  # `inreplace` supports regular expressions:
-  # <pre>inreplace "somefile.cfg", /look[for]what?/, "replace by #{bin}/tool"</pre>
   # @see Utils::Inreplace.inreplace
 
   # The name of this {Formula}.
@@ -590,6 +581,7 @@ class Formula
   end
 
   # {PkgVersion} of the linked keg for the formula.
+  sig { returns(T.nilable(PkgVersion)) }
   def linked_version
     return unless linked?
 
@@ -668,9 +660,12 @@ class Formula
 
   # The directory where the formula's binaries should be installed.
   # This is not symlinked into `HOMEBREW_PREFIX`.
-  # It is also commonly used to install files that we do not wish to be
+  # It is commonly used to install files that we do not wish to be
   # symlinked into `HOMEBREW_PREFIX` from one of the other directories and
   # instead manually create symlinks or wrapper scripts into e.g. {#bin}.
+  # <pre>libexec.install "foo.jar"
+  # bin.write_jar_script libexec/"foo.jar", "foo"
+  # </pre>
   def libexec
     prefix/"libexec"
   end
@@ -1104,14 +1099,9 @@ class Formula
   # @private
   delegate keg_only_reason: :"self.class"
 
-  # sometimes the formula cleaner breaks things
-  # skip cleaning paths in a formula with a class method like this:
-  #   skip_clean "bin/foo", "lib/bar"
-  # keep .la files with:
-  #   skip_clean :la
+  # @see .skip_clean
   # @private
   sig { params(path: Pathname).returns(T::Boolean) }
-
   def skip_clean?(path)
     return true if path.extname == ".la" && self.class.skip_clean_paths.include?(:la)
 
@@ -1119,11 +1109,7 @@ class Formula
     self.class.skip_clean_paths.include? to_check
   end
 
-  # Sometimes we accidentally install files outside prefix. After we fix that,
-  # users will get nasty link conflict error. So we create an allowlist here to
-  # allow overwriting certain files. e.g.
-  #   link_overwrite "bin/foo", "lib/bar"
-  #   link_overwrite "share/man/man1/baz-*"
+  # @see .link_overwrite
   # @private
   def link_overwrite?(path)
     # Don't overwrite files not created by Homebrew.
@@ -1163,36 +1149,42 @@ class Formula
   # Defaults to false.
   # @!method deprecated?
   # @return [Boolean]
+  # @see .deprecate!
   delegate deprecated?: :"self.class"
 
   # The date that this {Formula} was or becomes deprecated.
   # Returns `nil` if no date is specified.
   # @!method deprecation_date
   # @return Date
+  # @see .deprecate!
   delegate deprecation_date: :"self.class"
 
   # The reason this {Formula} is deprecated.
   # Returns `nil` if no reason is specified or the formula is not deprecated.
   # @!method deprecation_reason
   # @return [String, Symbol]
+  # @see .deprecate!
   delegate deprecation_reason: :"self.class"
 
   # Whether this {Formula} is disabled (i.e. cannot be installed).
   # Defaults to false.
   # @!method disabled?
   # @return [Boolean]
+  # @see .disable!
   delegate disabled?: :"self.class"
 
   # The date that this {Formula} was or becomes disabled.
   # Returns `nil` if no date is specified.
   # @!method disable_date
   # @return Date
+  # @see .disable!
   delegate disable_date: :"self.class"
 
   # The reason this {Formula} is disabled.
   # Returns `nil` if no reason is specified or the formula is not disabled.
   # @!method disable_reason
   # @return [String, Symbol]
+  # @see .disable!
   delegate disable_reason: :"self.class"
 
   sig { returns(T::Boolean) }
@@ -1406,21 +1398,22 @@ class Formula
   end
 
   # Standard parameters for configure builds.
+  sig { returns(T::Array[String]) }
   def std_configure_args
     ["--disable-debug", "--disable-dependency-tracking", "--prefix=#{prefix}", "--libdir=#{lib}"]
   end
 
   # Standard parameters for cargo builds.
+  sig { returns(T::Array[T.any(String, Pathname)]) }
   def std_cargo_args
     ["--locked", "--root", prefix, "--path", "."]
   end
 
   # Standard parameters for CMake builds.
+  #
   # Setting `CMAKE_FIND_FRAMEWORK` to "LAST" tells CMake to search for our
   # libraries before trying to utilize Frameworks, many of which will be from
   # 3rd party installs.
-  # Note that there isn't a std_autotools variant because autotools is a lot
-  # less consistent and the standard parameters are more memorable.
   sig { returns(T::Array[String]) }
   def std_cmake_args
     args = %W[
@@ -1445,6 +1438,7 @@ class Formula
   end
 
   # Standard parameters for Go builds.
+  sig { returns(T::Array[T.any(String, Pathname)]) }
   def std_go_args
     ["-trimpath", "-o", bin/name]
   end
@@ -1469,6 +1463,20 @@ class Formula
     ["--prefix=#{prefix}", "--libdir=#{lib}", "--buildtype=release", "--wrap-mode=nofallback"]
   end
 
+  # Shared library names according to platform conventions.
+  #
+  # Optionally specify a `version` to restrict the shared library to a specific
+  # version. The special string "*" matches any version.
+  #
+  # If `name` is specified as "*", match any shared library of any version.
+  #
+  # <pre>
+  # shared_library("foo")      #=> foo.dylib
+  # shared_library("foo", 1)   #=> foo.1.dylib
+  # shared_library("foo", "*") #=> foo.2.dylib, foo.1.dylib, foo.dylib
+  # shared_library("*")        #=> foo.dylib, bar.dylib
+  # </pre>
+  sig { params(name: String, version: T.nilable(T.any(String, Integer))).returns(String) }
   def shared_library(name, version = nil)
     return "*.dylib" if name == "*" && (version.blank? || version == "*")
 
@@ -1675,6 +1683,8 @@ class Formula
     end
   end
 
+  # Returns the {PkgVersion} for this formula if it is installed.
+  # If not, return `nil`.
   def any_installed_version
     any_installed_keg&.version
   end
@@ -1981,7 +1991,8 @@ class Formula
   # has to be opened first.
   # <pre>system "./bootstrap.sh", "--arg1", "--prefix=#{prefix}"</pre>
   #
-  # For CMake we have some necessary defaults in {#std_cmake_args}:
+  # For CMake and other build systems we have some necessary defaults in e.g.
+  # {#std_cmake_args}:
   # <pre>system "cmake", ".", *std_cmake_args</pre>
   #
   # If the arguments given to `configure` (or `make` or `cmake`) are depending
@@ -2435,7 +2446,6 @@ class Formula
     #
     # <pre>bottle do
     #   root_url "https://example.com" # Optional root to calculate bottle URLs.
-    #   prefix "/opt/homebrew" # Optional HOMEBREW_PREFIX in which the bottles were built.
     #   cellar "/opt/homebrew/Cellar" # Optional HOMEBREW_CELLAR in which the bottles were built.
     #   rebuild 1 # Marks the old bottle as outdated without bumping the version/revision of the formula.
     #   sha256 "ef65c759c5097a36323fa9c77756468649e8d1980a3a4e05695c05e39568967c" => :catalina
@@ -2443,16 +2453,12 @@ class Formula
     #   sha256 "91dd0caca9bd3f38c439d5a7b6f68440c4274945615fae035ff0a369264b8a2f" => :high_sierra
     # end</pre>
     #
-    # Only formulae where the upstream URL breaks or moves frequently, require compiling
-    # or have a reasonable amount of patches/resources should be bottled.
-    # Formulae which do not meet the above requirements should not be bottled.
+    # Homebrew maintainers aim to bottle all formulae that require compilation.
     #
-    # Formulae which should not be bottled and can be installed without any compile
-    # required should be tagged with:
+    # Formulae that can be installed without compilation should be tagged with:
     # <pre>bottle :unneeded</pre>
     #
-    # Otherwise formulae which do not meet the above requirements and should not
-    # be bottled should be tagged with:
+    # Formulae which should not be bottled should be tagged with:
     # <pre>bottle :disable, "reasons"</pre>
     def bottle(*args, &block)
       stable.bottle(*args, &block)
@@ -2533,32 +2539,41 @@ class Formula
     end
 
     # The dependencies for this formula. Use strings for the names of other
-    # formulae. Homebrew provides some :special dependencies for stuff that
-    # requires certain extra handling (often changing some ENV vars or
-    # deciding if to use the system provided version or not).
-    # <pre># `:build` means this dep is only needed during build.
-    # depends_on "cmake" => :build</pre>
-    # <pre># `:recommended` dependencies are built by default.
-    # # But a `--without-...` option is generated to opt-out.
-    # depends_on "readline" => :recommended</pre>
-    # <pre># `:optional` dependencies are NOT built by default unless the
-    # # auto-generated `--with-...` option is passed.
-    # depends_on "glib" => :optional</pre>
-    # <pre># If you need to specify that another formula has to be built with/out
-    # # certain options (note, no `--` needed before the option):
-    # depends_on "zeromq" => "with-pgm"
+    # formulae. Homebrew provides some `:special` {Requirement}s for stuff
+    # that needs extra handling (often changing some ENV vars or
+    # deciding whether to use the system provided version).
+    #
+    # `:build` means this dependency is only needed during build.
+    # <pre>depends_on "cmake" => :build</pre>
+    #
+    # `:test` means this dependency is only needed during testing.
+    # <pre>depends_on "node" => :test</pre>
+    #
+    # `:recommended` dependencies are built by default.
+    # But a `--without-...` option is generated to opt-out.
+    # <pre>depends_on "readline" => :recommended</pre>
+    #
+    # `:optional` dependencies are NOT built by default unless the
+    # auto-generated `--with-...` option is passed.
+    # <pre>depends_on "glib" => :optional</pre>
+    #
+    # If you need to specify that another formula has to be built with/out
+    # certain options (note, no `--` needed before the option):
+    # <pre>depends_on "zeromq" => "with-pgm"
     # depends_on "qt" => ["with-qtdbus", "developer"] # Multiple options.</pre>
-    # <pre># Optional and enforce that "boost" is built with `--with-c++11`.
-    # depends_on "boost" => [:optional, "with-c++11"]</pre>
-    # <pre># If a dependency is only needed in certain cases:
-    # depends_on "sqlite" if MacOS.version >= :catalina
+    #
+    # Optional and enforce that "boost" is built with `--with-c++11`.
+    # <pre>depends_on "boost" => [:optional, "with-c++11"]</pre>
+    #
+    # If a dependency is only needed in certain cases:
+    # <pre>depends_on "sqlite" if MacOS.version >= :catalina
     # depends_on xcode: :build # If the formula really needs full Xcode to compile.
     # depends_on macos: :mojave # Needs at least macOS Mojave (10.14) to run.
-    # <pre># It is possible to only depend on something if
-    # # `build.with?` or `build.without? "another_formula"`:
-    # depends_on "postgresql" if build.without? "sqlite"</pre>
-    # <pre># Require Python if `--with-python` is passed to `brew install example`:
-    # depends_on "python" => :optional</pre>
+    # </pre>
+    #
+    # It is possible to only depend on something if
+    # `build.with?` or `build.without? "another_formula"`:
+    # <pre>depends_on "postgresql" if build.without? "sqlite"</pre>
     def depends_on(dep)
       specs.each { |spec| spec.depends_on(dep) }
     end
@@ -2659,6 +2674,13 @@ class Formula
       names.each { |name| conflicts << FormulaConflict.new(name, opts[:because]) }
     end
 
+    # Skip cleaning paths in a formula.
+    #
+    # Sometimes the formula {Cleaner cleaner} breaks things.
+    # Preserve cleaned paths with:
+    # <pre>skip_clean "bin/foo", "lib/bar"</pre>
+    # Keep .la files with:
+    # <pre>skip_clean :la</pre>
     def skip_clean(*paths)
       paths.flatten!
       # Specifying :all is deprecated and will become an error
@@ -2670,13 +2692,16 @@ class Formula
       @skip_clean_paths ||= Set.new
     end
 
-    # Software that will not be symlinked into the `brew --prefix` will only
-    # live in its Cellar. Other formulae can depend on it and then brew will
-    # add the necessary includes and libs (etc.) during the brewing of that
-    # other formula. But generally, keg-only formulae are not in your PATH
-    # and not seen by compilers if you build your own software outside of
-    # Homebrew. This way, we don't shadow software provided by macOS.
+    # Software that will not be symlinked into the `brew --prefix` and will
+    # only live in its Cellar. Other formulae can depend on it and Homebrew
+    # will add the necessary includes, libraries, and other paths while
+    # building that other formula.
+    #
+    # Keg-only formulae are not in your PATH and are not seen by compilers
+    # if you build your own software outside of Homebrew. This way, we
+    # don't shadow software provided by macOS.
     # <pre>keg_only :provided_by_macos</pre>
+    # <pre>keg_only :versioned_formulae</pre>
     # <pre>keg_only "because I want it so"</pre>
     def keg_only(reason, explanation = "")
       @keg_only_reason = KegOnlyReason.new(reason, explanation)
@@ -2712,6 +2737,15 @@ class Formula
       specs.each { |spec| spec.fails_with(compiler, &block) }
     end
 
+    # Marks the {Formula} as needing a certain standard, so Homebrew
+    # will fall back to other compilers if the default compiler
+    # does not implement that standard.
+    #
+    # We generally prefer to {.depends_on} a desired compiler and to
+    # explicitly use that compiler in a formula's {#install} block,
+    # rather than implicitly finding a suitable compiler with `needs`.
+    #
+    # @see #fails_with
     def needs(*standards)
       specs.each { |spec| spec.needs(*standards) }
     end
@@ -2749,9 +2783,9 @@ class Formula
     end
 
     # @!attribute [w] livecheck
-    # Livecheck can be used to check for newer versions of the software.
+    # {Livecheck} can be used to check for newer versions of the software.
     # This method evaluates the DSL specified in the livecheck block of the
-    # {Formula} (if it exists) and sets the instance variables of a Livecheck
+    # {Formula} (if it exists) and sets the instance variables of a {Livecheck}
     # object accordingly. This is used by `brew livecheck` to check for newer
     # versions of the software.
     #
@@ -2790,6 +2824,8 @@ class Formula
     # will not be deprecated.
     # <pre>deprecate! date: "2020-08-27", because: :unmaintained</pre>
     # <pre>deprecate! date: "2020-08-27", because: "has been replaced by foo"</pre>
+    # @see https://docs.brew.sh/Deprecating-Disabling-and-Removing-Formulae
+    # @see DeprecateDisable::DEPRECATE_DISABLE_REASONS
     def deprecate!(date: nil, because: nil)
       odeprecated "`deprecate!` without a reason", "`deprecate! because: \"reason\"`" if because.blank?
       odeprecated "`deprecate!` without a date", "`deprecate! date: \"#{Date.today}\"`" if date.blank?
@@ -2805,6 +2841,7 @@ class Formula
     # Whether this {Formula} is deprecated (i.e. warns on installation).
     # Defaults to false.
     # @return [Boolean]
+    # @see .deprecate!
     def deprecated?
       @deprecated == true
     end
@@ -2812,18 +2849,22 @@ class Formula
     # The date that this {Formula} was or becomes deprecated.
     # Returns `nil` if no date is specified.
     # @return Date
+    # @see .deprecate!
     attr_reader :deprecation_date
 
     # The reason for deprecation of a {Formula}.
     # @return [nil] if no reason was provided or the formula is not deprecated.
     # @return [String, Symbol]
+    # @see .deprecate!
     attr_reader :deprecation_reason
 
-    # Disables a {Formula}  (on a given date, if provided) so it cannot be
+    # Disables a {Formula} (on a given date, if provided) so it cannot be
     # installed. If the date has not yet passed the formula
     # will be deprecated instead of disabled.
     # <pre>disable! date: "2020-08-27", because: :does_not_build</pre>
     # <pre>disable! date: "2020-08-27", because: "has been replaced by foo"</pre>
+    # @see https://docs.brew.sh/Deprecating-Disabling-and-Removing-Formulae
+    # @see DeprecateDisable::DEPRECATE_DISABLE_REASONS
     def disable!(date: nil, because: nil)
       odeprecated "`disable!` without a reason", "`disable! because: \"reason\"`" if because.blank?
       odeprecated "`disable!` without a date", "`disable! date: \"#{Date.today}\"`" if date.blank?
@@ -2843,6 +2884,7 @@ class Formula
     # Whether this {Formula} is disabled (i.e. cannot be installed).
     # Defaults to false.
     # @return [Boolean]
+    # @see .disable!
     def disabled?
       @disabled == true
     end
@@ -2850,14 +2892,21 @@ class Formula
     # The date that this {Formula} was or becomes disabled.
     # Returns `nil` if no date is specified.
     # @return Date
+    # @see .disable!
     attr_reader :disable_date
 
     # The reason this {Formula} is disabled.
     # Returns `nil` if no reason was provided or the formula is not disabled.
     # @return [String, Symbol]
+    # @see .disable!
     attr_reader :disable_reason
 
-    # @private
+    # Permit overwriting certain files while linking.
+    #
+    # Sometimes we accidentally install files outside prefix. Once we fix that,
+    # users will get a link conflict error. Overwrite those files with:
+    # <pre>link_overwrite "bin/foo", "lib/bar"</pre>
+    # <pre>link_overwrite "share/man/man1/baz-*"</pre>
     def link_overwrite(*paths)
       paths.flatten!
       link_overwrite_paths.merge(paths)
@@ -2868,6 +2917,7 @@ class Formula
       @link_overwrite_paths ||= Set.new
     end
 
+    # Permit links to certain libraries that don't exist. Available on Linux only.
     def ignore_missing_libraries(*)
       raise FormulaSpecificationError, "#{__method__} is available on Linux only"
     end
diff --git a/Library/Homebrew/utils/inreplace.rb b/Library/Homebrew/utils/inreplace.rb
index 951b55148c..95a19aeaa6 100644
--- a/Library/Homebrew/utils/inreplace.rb
+++ b/Library/Homebrew/utils/inreplace.rb
@@ -23,14 +23,22 @@ module Utils
     module_function
 
     # Sometimes we have to change a bit before we install. Mostly we
-    # prefer a patch but if you need the `prefix` of this formula in the
-    # patch you have to resort to `inreplace`, because in the patch
-    # you don't have access to any var defined by the formula. Only
-    # `HOMEBREW_PREFIX` is available in the embedded patch.
+    # prefer a patch, but if you need the {Formula#prefix prefix} of
+    # this formula in the patch you have to resort to `inreplace`,
+    # because in the patch you don't have access to any variables
+    # defined by the formula, as only `HOMEBREW_PREFIX` is available
+    # in the {DATAPatch embedded patch}.
     #
     # `inreplace` supports regular expressions:
     # <pre>inreplace "somefile.cfg", /look[for]what?/, "replace by #{bin}/tool"</pre>
     #
+    # `inreplace` supports blocks:
+    # <pre>inreplace "Makefile" do |s|
+    #   s.gsub! "/usr/local", HOMEBREW_PREFIX.to_s
+    # end
+    # </pre>
+    #
+    # @see StringInreplaceExtension
     # @api public
     sig do
       params(
@@ -65,6 +73,7 @@ module Utils
       raise Error, errors unless errors.empty?
     end
 
+    # @api private
     def inreplace_pairs(path, replacement_pairs, read_only_run: false, silent: false)
       str = File.open(path, "rb", &:read) || ""
       contents = StringInreplaceExtension.new(str)
diff --git a/docs/Bottles.md b/docs/Bottles.md
index 4b1d4fa332..c7de6e026a 100644
--- a/docs/Bottles.md
+++ b/docs/Bottles.md
@@ -33,7 +33,6 @@ A full example:
 ```ruby
 bottle do
   root_url "https://example.com"
-  prefix "/opt/homebrew"
   cellar "/opt/homebrew/Cellar"
   rebuild 4
   sha256 "4921af80137af9cc3d38fd17c9120da882448a090b0a8a3a19af3199b415bfca" => :sierra
@@ -50,10 +49,6 @@ By default this is omitted and the Homebrew default bottle URL root is used. Thi
 Optionally contains the value of `HOMEBREW_CELLAR` in which the bottles were built.
 Most compiled software contains references to its compiled location so cannot be simply relocated anywhere on disk. If this value is `:any` or `:any_skip_relocation` this means that the bottle can be safely installed in any Cellar as it did not contain any references to its installation Cellar. This can be omitted if a bottle is compiled (as all default Homebrew ones are) for the default `HOMEBREW_CELLAR` of `/usr/local/Cellar`.
 
-### Prefix (`prefix`)
-Optionally contains the value of `HOMEBREW_PREFIX` in which the bottles were built.
-See description of `cellar`. When `cellar` is `:any` or `:any_skip_relocation` the prefix should be omitted.
-
 ### Rebuild version (`rebuild`)
 Optionally contains the rebuild version of the bottle.
 Sometimes bottles may need be updated without bumping the version of the formula, e.g. a new patch was applied. In that case the rebuild will have a value of 1 or more.
-- 
GitLab