require "cli/parser" require "utils/git" require "formulary" require "tap" def with_monkey_patch BottleSpecification.class_eval do alias_method :old_method_missing, :method_missing if method_defined?(:method_missing) define_method(:method_missing) { |*| } end Module.class_eval do alias_method :old_method_missing, :method_missing if method_defined?(:method_missing) define_method(:method_missing) { |*| } end Resource.class_eval do alias_method :old_method_missing, :method_missing if method_defined?(:method_missing) define_method(:method_missing) { |*| } end DependencyCollector.class_eval do alias_method :old_parse_symbol_spec, :parse_symbol_spec if method_defined?(:parse_symbol_spec) define_method(:parse_symbol_spec) { |*| } end if defined?(DependencyCollector::Compat) DependencyCollector::Compat.class_eval do alias_method :old_parse_string_spec, :parse_string_spec if method_defined?(:parse_string_spec) define_method(:parse_string_spec) { |*| } end end yield ensure BottleSpecification.class_eval do if method_defined?(:old_method_missing) alias_method :method_missing, :old_method_missing undef :old_method_missing end end Module.class_eval do if method_defined?(:old_method_missing) alias_method :method_missing, :old_method_missing undef :old_method_missing end end Resource.class_eval do if method_defined?(:old_method_missing) alias_method :method_missing, :old_method_missing undef :old_method_missing end end DependencyCollector.class_eval do if method_defined?(:old_parse_symbol_spec) alias_method :parse_symbol_spec, :old_parse_symbol_spec undef :old_parse_symbol_spec end end if defined?(DependencyCollector::Compat) DependencyCollector::Compat.class_eval do if method_defined?(:old_parse_string_spec) alias_method :parse_string_spec, :old_parse_string_spec undef :old_parse_string_spec end end end end module Homebrew module_function def extract_args Homebrew::CLI::Parser.new do usage_banner <<~EOS `extract` [] Look through repository history to find the most recent version of and create a copy in `/Formula/``@``.rb`. If the tap is not installed yet, attempt to install/clone the tap before continuing. To extract a from a tap that is not homebrew/core use //. EOS flag "--version=", description: "Extract the provided of instead of the most recent." switch :force switch :debug end end def extract extract_args.parse # Expect exactly two named arguments: formula and tap raise UsageError if ARGV.named.length != 2 if ARGV.named.first !~ HOMEBREW_TAP_FORMULA_REGEX name = ARGV.named.first.downcase source_tap = CoreTap.instance else name = Regexp.last_match(3).downcase source_tap = Tap.fetch(Regexp.last_match(1), Regexp.last_match(2)) raise TapFormulaUnavailableError.new(source_tap, name) unless source_tap.installed? end destination_tap = Tap.fetch(ARGV.named.second) odie "Cannot extract formula to homebrew/core!" if destination_tap.core_tap? odie "Cannot extract formula to the same tap!" if destination_tap == source_tap destination_tap.install unless destination_tap.installed? repo = source_tap.path pattern = if source_tap.core_tap? [repo/"Formula/#{name}.rb"] 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 if args.version ohai "Searching repository history" version = args.version rev = "HEAD" test_formula = nil result = "" loop do rev, (path,) = Git.last_revision_commit_of_files(repo, pattern, before_commit: "#{rev}~1") odie "Could not find #{name}! The formula or version may not have existed." if rev.nil? file = repo/path result = Git.last_revision_of_file(repo, file, before_commit: rev) if result.empty? ohai "Skipping revision #{rev} - file is empty at this revision" if ARGV.debug? next end test_formula = formula_at_revision(repo, name, file, rev) break if test_formula.nil? || test_formula.version == version ohai "Trying #{test_formula.version} from revision #{rev} against desired #{version}" if ARGV.debug? end odie "Could not find #{name}! The formula or version may not have existed." if test_formula.nil? else # Search in the root directory of as well as recursively in all of its subdirectories files = Dir[repo/"{,**/}"].map do |dir| Pathname.glob(["#{dir}/#{name}.rb"]).find(&:file?) end.compact if files.empty? ohai "Searching repository history" rev, (path,) = Git.last_revision_commit_of_files(repo, pattern) odie "Could not find #{name}! The formula or version may not have existed." if rev.nil? file = repo/path version = formula_at_revision(repo, name, file, rev).version result = Git.last_revision_of_file(repo, file) else file = files.first.realpath rev = "HEAD" version = Formulary.factory(file).version result = File.read(file) end end # 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. class_name = Formulary.class_s(name) versioned_name = Formulary.class_s("#{name}@#{version}") result.gsub!("class #{class_name} < Formula", "class #{versioned_name} < Formula") path = destination_tap.path/"Formula/#{name}@#{version}.rb" if path.exist? unless ARGV.force? odie <<~EOS Destination formula already exists: #{path} To overwrite it and continue anyways, run: brew extract --force --version=#{version} #{name} #{destination_tap.name} EOS end ohai "Overwriting existing formula at #{path}" if ARGV.debug? path.delete end ohai "Writing formula for #{name} from revision #{rev} to #{path}" path.write result end # @private def formula_at_revision(repo, name, file, rev) return if rev.empty? contents = Git.last_revision_of_file(repo, file, before_commit: rev) contents.gsub!("@url=", "url ") contents.gsub!("require 'brewkit'", "require 'formula'") with_monkey_patch { Formulary.from_contents(name, file, contents) } end end