| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  | # typed: true | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 18:25:08 +09:00
										 |  |  | require "cli/parser" | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  | require "utils/git" | 
					
						
							|  |  |  | require "formulary" | 
					
						
							| 
									
										
										
										
											2020-08-18 00:23:23 +01:00
										 |  |  | require "software_spec" | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  | require "tap" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module Homebrew | 
					
						
							| 
									
										
										
										
											2024-01-18 22:18:42 +00:00
										 |  |  |   BOTTLE_BLOCK_REGEX = /  bottle (?:do.+?end|:[a-z]+)\n\n/m | 
					
						
							| 
									
										
										
										
											2021-11-18 21:53:32 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-20 12:03:48 +02:00
										 |  |  |   sig { returns(CLI::Parser) } | 
					
						
							| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  |   def self.extract_args | 
					
						
							| 
									
										
										
										
											2018-10-03 20:16:05 +05:30
										 |  |  |     Homebrew::CLI::Parser.new do | 
					
						
							| 
									
										
										
										
											2023-02-10 23:15:40 -05:00
										 |  |  |       usage_banner "`extract` [`--version=`] [`--force`] <formula> <tap>" | 
					
						
							| 
									
										
										
										
											2021-01-15 15:04:02 -05:00
										 |  |  |       description <<~EOS | 
					
						
							| 
									
										
										
										
											2018-10-08 22:49:03 -04:00
										 |  |  |         Look through repository history to find the most recent version of <formula> and | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:02 -05:00
										 |  |  |         create a copy in <tap>. Specifically, the command will create the new | 
					
						
							|  |  |  |         formula file at <tap>`/Formula/`<formula>`@`<version>`.rb`. If the tap is not | 
					
						
							| 
									
										
										
										
											2019-03-19 14:07:50 +08:00
										 |  |  |         installed yet, attempt to install/clone the tap before continuing. To extract | 
					
						
							| 
									
										
										
										
											2019-08-20 00:04:14 -04:00
										 |  |  |         a formula from a tap that is not `homebrew/core` use its fully-qualified form of | 
					
						
							|  |  |  |         <user>`/`<repo>`/`<formula>. | 
					
						
							| 
									
										
										
										
											2018-10-03 20:16:05 +05:30
										 |  |  |       EOS | 
					
						
							| 
									
										
										
										
											2019-08-06 13:23:19 -04:00
										 |  |  |       flag   "--version=", | 
					
						
							| 
									
										
										
										
											2019-08-20 00:04:14 -04:00
										 |  |  |              description: "Extract the specified <version> of <formula> instead of the most recent." | 
					
						
							| 
									
										
										
										
											2020-07-27 03:59:52 +02:00
										 |  |  |       switch "-f", "--force", | 
					
						
							|  |  |  |              description: "Overwrite the destination formula if it already exists." | 
					
						
							| 
									
										
										
										
											2020-07-30 18:40:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-22 16:53:46 +01:00
										 |  |  |       named_args [:formula, :tap], number: 2, without_api: true | 
					
						
							| 
									
										
										
										
											2018-07-29 21:02:36 -04:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-10-03 20:16:05 +05:30
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  |   def self.extract | 
					
						
							| 
									
										
										
										
											2020-07-30 18:40:10 +02:00
										 |  |  |     args = extract_args.parse | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-01 03:13:56 +01:00
										 |  |  |     if (tap_with_name = args.named.first&.then { Tap.with_formula_name(_1) }) | 
					
						
							|  |  |  |       source_tap, name = tap_with_name | 
					
						
							| 
									
										
										
										
											2020-11-09 20:09:16 +11:00
										 |  |  |     else | 
					
						
							|  |  |  |       name = args.named.first.downcase | 
					
						
							|  |  |  |       source_tap = CoreTap.instance | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2023-06-22 16:53:46 +01:00
										 |  |  |     raise TapFormulaUnavailableError.new(source_tap, name) unless source_tap.installed? | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-04 17:28:24 +00:00
										 |  |  |     destination_tap = Tap.fetch(args.named.second) | 
					
						
							| 
									
										
										
										
											2020-04-05 15:44:50 +01:00
										 |  |  |     unless Homebrew::EnvConfig.developer? | 
					
						
							| 
									
										
										
										
											2020-02-27 10:36:10 +00:00
										 |  |  |       odie "Cannot extract formula to homebrew/core!" if destination_tap.core_tap? | 
					
						
							| 
									
										
										
										
											2023-07-13 19:45:28 +01:00
										 |  |  |       odie "Cannot extract formula to homebrew/cask!" if destination_tap.core_cask_tap? | 
					
						
							| 
									
										
										
										
											2020-02-27 10:36:10 +00:00
										 |  |  |       odie "Cannot extract formula to the same tap!" if destination_tap == source_tap | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-07-30 18:41:45 -04:00
										 |  |  |     destination_tap.install unless destination_tap.installed? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |     repo = source_tap.path | 
					
						
							| 
									
										
										
										
											2019-03-19 14:07:50 +08:00
										 |  |  |     pattern = if source_tap.core_tap? | 
					
						
							| 
									
										
										
										
											2023-08-04 16:21:31 +01:00
										 |  |  |       [source_tap.new_formula_path(name), repo/"Formula/#{name}.rb"].uniq | 
					
						
							| 
									
										
										
										
											2019-03-19 14:07:50 +08:00
										 |  |  |     else | 
					
						
							|  |  |  |       # A formula can technically live in the root directory of a tap or in any of its subdirectories | 
					
						
							|  |  |  |       [repo/"#{name}.rb", repo/"**/#{name}.rb"] | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-10 14:59:09 -04:00
										 |  |  |     if args.version | 
					
						
							| 
									
										
										
										
											2018-08-25 13:40:57 -04:00
										 |  |  |       ohai "Searching repository history" | 
					
						
							| 
									
										
										
										
											2018-08-02 00:21:09 -04:00
										 |  |  |       version = args.version | 
					
						
							| 
									
										
										
										
											2019-12-01 22:26:03 +08:00
										 |  |  |       version_segments = Gem::Version.new(version).segments if Gem::Version.correct?(version) | 
					
						
							| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  |       rev = T.let(nil, T.nilable(String)) | 
					
						
							|  |  |  |       test_formula = T.let(nil, T.nilable(Formula)) | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |       result = "" | 
					
						
							| 
									
										
										
										
											2018-08-02 00:21:09 -04:00
										 |  |  |       loop do | 
					
						
							| 
									
										
										
										
											2019-12-01 22:26:03 +08:00
										 |  |  |         rev = rev.nil? ? "HEAD" : "#{rev}~1" | 
					
						
							| 
									
										
										
										
											2020-08-23 06:32:26 +02:00
										 |  |  |         rev, (path,) = Utils::Git.last_revision_commit_of_files(repo, pattern, before_commit: rev) | 
					
						
							| 
									
										
										
										
											2020-04-14 17:50:54 +05:30
										 |  |  |         if rev.nil? && source_tap.shallow? | 
					
						
							|  |  |  |           odie <<~EOS | 
					
						
							|  |  |  |             Could not find #{name} but #{source_tap} is a shallow clone! | 
					
						
							|  |  |  |             Try again after running: | 
					
						
							|  |  |  |               git -C "#{source_tap.path}" fetch --unshallow | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         elsif rev.nil? | 
					
						
							|  |  |  |           odie "Could not find #{name}! The formula or version may not have existed." | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |         file = repo/path | 
					
						
							| 
									
										
										
										
											2020-08-23 06:32:26 +02:00
										 |  |  |         result = Utils::Git.last_revision_of_file(repo, file, before_commit: rev) | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |         if result.empty? | 
					
						
							| 
									
										
										
										
											2019-12-01 22:26:03 +08:00
										 |  |  |           odebug "Skipping revision #{rev} - file is empty at this revision" | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |           next | 
					
						
							| 
									
										
										
										
											2018-08-02 00:21:09 -04:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-02 00:21:09 -04:00
										 |  |  |         test_formula = formula_at_revision(repo, name, file, rev) | 
					
						
							|  |  |  |         break if test_formula.nil? || test_formula.version == version | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-01 22:26:03 +08:00
										 |  |  |         if version_segments && Gem::Version.correct?(test_formula.version) | 
					
						
							|  |  |  |           test_formula_version_segments = Gem::Version.new(test_formula.version).segments | 
					
						
							|  |  |  |           if version_segments.length < test_formula_version_segments.length | 
					
						
							| 
									
										
										
										
											2020-06-18 11:19:04 -04:00
										 |  |  |             odebug "Apply semantic versioning with #{test_formula_version_segments}" | 
					
						
							| 
									
										
										
										
											2019-12-01 22:26:03 +08:00
										 |  |  |             break if version_segments == test_formula_version_segments.first(version_segments.length) | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         odebug "Trying #{test_formula.version} from revision #{rev} against desired #{version}" | 
					
						
							| 
									
										
										
										
											2018-08-02 00:21:09 -04:00
										 |  |  |       end | 
					
						
							|  |  |  |       odie "Could not find #{name}! The formula or version may not have existed." if test_formula.nil? | 
					
						
							| 
									
										
										
										
											2018-08-10 14:59:09 -04:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2019-03-19 14:07:50 +08:00
										 |  |  |       # Search in the root directory of <repo> as well as recursively in all of its subdirectories | 
					
						
							| 
									
										
										
										
											2024-02-22 23:29:55 +00:00
										 |  |  |       files = Dir[repo/"{,**/}"].filter_map do |dir| | 
					
						
							| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  |         Pathname.glob("#{dir}/#{name}.rb").find(&:file?) | 
					
						
							| 
									
										
										
										
											2024-02-22 23:29:55 +00:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if files.empty? | 
					
						
							|  |  |  |         ohai "Searching repository history" | 
					
						
							| 
									
										
										
										
											2020-08-23 06:32:26 +02:00
										 |  |  |         rev, (path,) = Utils::Git.last_revision_commit_of_files(repo, pattern) | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |         odie "Could not find #{name}! The formula or version may not have existed." if rev.nil? | 
					
						
							|  |  |  |         file = repo/path | 
					
						
							| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  |         version = T.must(formula_at_revision(repo, name, file, rev)).version | 
					
						
							| 
									
										
										
										
											2020-08-23 06:32:26 +02:00
										 |  |  |         result = Utils::Git.last_revision_of_file(repo, file) | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  |         file = files.fetch(0).realpath | 
					
						
							|  |  |  |         rev = T.let("HEAD", T.nilable(String)) | 
					
						
							| 
									
										
										
										
											2019-03-17 18:39:47 +08:00
										 |  |  |         version = Formulary.factory(file).version | 
					
						
							|  |  |  |         result = File.read(file) | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-08-02 00:21:09 -04:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 09:20:45 -04:00
										 |  |  |     # The class name has to be renamed to match the new filename, | 
					
						
							|  |  |  |     # e.g. Foo version 1.2.3 becomes FooAT123 and resides in Foo@1.2.3.rb. | 
					
						
							| 
									
										
										
										
											2019-03-15 11:27:39 +08:00
										 |  |  |     class_name = Formulary.class_s(name) | 
					
						
							| 
									
										
										
										
											2020-02-21 17:26:24 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Remove any existing version suffixes, as a new one will be added later | 
					
						
							|  |  |  |     name.sub!(/\b@(.*)\z\b/i, "") | 
					
						
							| 
									
										
										
										
											2019-03-15 11:27:39 +08:00
										 |  |  |     versioned_name = Formulary.class_s("#{name}@#{version}") | 
					
						
							| 
									
										
										
										
											2020-07-02 11:57:11 +01:00
										 |  |  |     result.sub!("class #{class_name} < Formula", "class #{versioned_name} < Formula") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Remove bottle blocks, they won't work. | 
					
						
							| 
									
										
										
										
											2021-11-18 21:53:32 -08:00
										 |  |  |     result.sub!(BOTTLE_BLOCK_REGEX, "") | 
					
						
							| 
									
										
										
										
											2018-08-02 00:21:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 23:20:15 -07:00
										 |  |  |     path = destination_tap.path/"Formula/#{name}@#{version.to_s.downcase}.rb" | 
					
						
							| 
									
										
										
										
											2018-07-30 18:41:45 -04:00
										 |  |  |     if path.exist? | 
					
						
							| 
									
										
										
										
											2020-03-04 17:28:24 +00:00
										 |  |  |       unless args.force? | 
					
						
							| 
									
										
										
										
											2018-07-30 18:41:45 -04:00
										 |  |  |         odie <<~EOS | 
					
						
							|  |  |  |           Destination formula already exists: #{path} | 
					
						
							|  |  |  |           To overwrite it and continue anyways, run: | 
					
						
							| 
									
										
										
										
											2018-10-08 22:49:03 -04:00
										 |  |  |             brew extract --force --version=#{version} #{name} #{destination_tap.name} | 
					
						
							| 
									
										
										
										
											2018-07-30 18:41:45 -04:00
										 |  |  |         EOS | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2019-12-01 22:26:03 +08:00
										 |  |  |       odebug "Overwriting existing formula at #{path}" | 
					
						
							| 
									
										
										
										
											2018-07-30 18:41:45 -04:00
										 |  |  |       path.delete | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2020-07-06 15:30:57 -04:00
										 |  |  |     ohai "Writing formula for #{name} from revision #{rev} to:", path | 
					
						
							| 
									
										
										
										
											2021-05-03 13:24:32 +01:00
										 |  |  |     path.dirname.mkpath | 
					
						
							| 
									
										
										
										
											2018-07-30 18:41:45 -04:00
										 |  |  |     path.write result | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  |   sig { params(repo: Pathname, name: String, file: Pathname, rev: String).returns(T.nilable(Formula)) } | 
					
						
							|  |  |  |   def self.formula_at_revision(repo, name, file, rev) | 
					
						
							| 
									
										
										
										
											2018-08-10 14:59:09 -04:00
										 |  |  |     return if rev.empty? | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-23 06:32:26 +02:00
										 |  |  |     contents = Utils::Git.last_revision_of_file(repo, file, before_commit: rev) | 
					
						
							| 
									
										
										
										
											2018-08-16 12:52:54 -04:00
										 |  |  |     contents.gsub!("@url=", "url ") | 
					
						
							|  |  |  |     contents.gsub!("require 'brewkit'", "require 'formula'") | 
					
						
							| 
									
										
										
										
											2021-11-18 21:53:32 -08:00
										 |  |  |     contents.sub!(BOTTLE_BLOCK_REGEX, "") | 
					
						
							| 
									
										
										
										
											2021-07-17 13:45:59 +08:00
										 |  |  |     with_monkey_patch { Formulary.from_contents(name, file, contents, ignore_errors: true) } | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2024-01-26 11:36:08 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   private_class_method def self.with_monkey_patch | 
					
						
							|  |  |  |     # Since `method_defined?` is not a supported type guard, the use of `alias_method` below is not typesafe: | 
					
						
							|  |  |  |     BottleSpecification.class_eval do | 
					
						
							|  |  |  |       T.unsafe(self).alias_method :old_method_missing, :method_missing if method_defined?(:method_missing) | 
					
						
							|  |  |  |       define_method(:method_missing) do |*| | 
					
						
							|  |  |  |         # do nothing | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Module.class_eval do | 
					
						
							|  |  |  |       T.unsafe(self).alias_method :old_method_missing, :method_missing if method_defined?(:method_missing) | 
					
						
							|  |  |  |       define_method(:method_missing) do |*| | 
					
						
							|  |  |  |         # do nothing | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Resource.class_eval do | 
					
						
							|  |  |  |       T.unsafe(self).alias_method :old_method_missing, :method_missing if method_defined?(:method_missing) | 
					
						
							|  |  |  |       define_method(:method_missing) do |*| | 
					
						
							|  |  |  |         # do nothing | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DependencyCollector.class_eval do | 
					
						
							|  |  |  |       T.unsafe(self).alias_method :old_parse_symbol_spec, :parse_symbol_spec if method_defined?(:parse_symbol_spec) | 
					
						
							|  |  |  |       define_method(:parse_symbol_spec) do |*| | 
					
						
							|  |  |  |         # do nothing | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     yield | 
					
						
							|  |  |  |   ensure | 
					
						
							|  |  |  |     BottleSpecification.class_eval do | 
					
						
							|  |  |  |       if method_defined?(:old_method_missing) | 
					
						
							|  |  |  |         T.unsafe(self).alias_method :method_missing, :old_method_missing | 
					
						
							|  |  |  |         undef :old_method_missing | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Module.class_eval do | 
					
						
							|  |  |  |       if method_defined?(:old_method_missing) | 
					
						
							|  |  |  |         T.unsafe(self).alias_method :method_missing, :old_method_missing | 
					
						
							|  |  |  |         undef :old_method_missing | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Resource.class_eval do | 
					
						
							|  |  |  |       if method_defined?(:old_method_missing) | 
					
						
							|  |  |  |         T.unsafe(self).alias_method :method_missing, :old_method_missing | 
					
						
							|  |  |  |         undef :old_method_missing | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DependencyCollector.class_eval do | 
					
						
							|  |  |  |       if method_defined?(:old_parse_symbol_spec) | 
					
						
							|  |  |  |         T.unsafe(self).alias_method :parse_symbol_spec, :old_parse_symbol_spec | 
					
						
							|  |  |  |         undef :old_parse_symbol_spec | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2018-07-29 20:51:57 -04:00
										 |  |  | end |