mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
158 lines
3.4 KiB
Ruby
158 lines
3.4 KiB
Ruby
# typed: true # rubocop:todo Sorbet/StrictSigil
|
|
# frozen_string_literal: true
|
|
|
|
require "resource"
|
|
require "erb"
|
|
|
|
# Helper module for creating patches.
|
|
module Patch
|
|
def self.create(strip, src, &block)
|
|
case strip
|
|
when :DATA
|
|
DATAPatch.new(:p1)
|
|
when String
|
|
StringPatch.new(:p1, strip)
|
|
when Symbol
|
|
case src
|
|
when :DATA
|
|
DATAPatch.new(strip)
|
|
when String
|
|
StringPatch.new(strip, src)
|
|
else
|
|
ExternalPatch.new(strip, &block)
|
|
end
|
|
when nil
|
|
raise ArgumentError, "nil value for strip"
|
|
else
|
|
raise ArgumentError, "Unexpected value #{strip.inspect} for strip"
|
|
end
|
|
end
|
|
end
|
|
|
|
# An abstract class representing a patch embedded into a formula.
|
|
class EmbeddedPatch
|
|
attr_writer :owner
|
|
attr_reader :strip
|
|
|
|
def initialize(strip)
|
|
@strip = strip
|
|
end
|
|
|
|
sig { returns(T::Boolean) }
|
|
def external?
|
|
false
|
|
end
|
|
|
|
def contents; end
|
|
|
|
def apply
|
|
data = contents.gsub("HOMEBREW_PREFIX", HOMEBREW_PREFIX)
|
|
args = %W[-g 0 -f -#{strip}]
|
|
Utils.safe_popen_write("patch", *args) { |p| p.write(data) }
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def inspect
|
|
"#<#{self.class.name}: #{strip.inspect}>"
|
|
end
|
|
end
|
|
|
|
# A patch at the `__END__` of a formula file.
|
|
class DATAPatch < EmbeddedPatch
|
|
attr_accessor :path
|
|
|
|
def initialize(strip)
|
|
super
|
|
@path = nil
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def contents
|
|
data = +""
|
|
path.open("rb") do |f|
|
|
loop do
|
|
line = f.gets
|
|
break if line.nil? || /^__END__$/.match?(line)
|
|
end
|
|
while (line = f.gets)
|
|
data << line
|
|
end
|
|
end
|
|
data.freeze
|
|
end
|
|
end
|
|
|
|
# A string containing a patch.
|
|
class StringPatch < EmbeddedPatch
|
|
def initialize(strip, str)
|
|
super(strip)
|
|
@str = str
|
|
end
|
|
|
|
def contents
|
|
@str
|
|
end
|
|
end
|
|
|
|
# A string containing a patch.
|
|
class ExternalPatch
|
|
extend Forwardable
|
|
|
|
attr_reader :resource, :strip
|
|
|
|
def_delegators :resource,
|
|
:url, :fetch, :patch_files, :verify_download_integrity,
|
|
:cached_download, :downloaded?, :clear_cache
|
|
|
|
def initialize(strip, &block)
|
|
@strip = strip
|
|
@resource = Resource::Patch.new(&block)
|
|
end
|
|
|
|
sig { returns(T::Boolean) }
|
|
def external?
|
|
true
|
|
end
|
|
|
|
def owner=(owner)
|
|
resource.owner = owner
|
|
resource.version(resource.checksum&.hexdigest || ERB::Util.url_encode(resource.url))
|
|
end
|
|
|
|
def apply
|
|
base_dir = Pathname.pwd
|
|
resource.unpack do
|
|
patch_dir = Pathname.pwd
|
|
if patch_files.empty?
|
|
children = patch_dir.children
|
|
if children.length != 1 || !children.fetch(0).file?
|
|
raise MissingApplyError, <<~EOS
|
|
There should be exactly one patch file in the staging directory unless
|
|
the "apply" method was used one or more times in the patch-do block.
|
|
EOS
|
|
end
|
|
|
|
patch_files << children.fetch(0).basename
|
|
end
|
|
dir = base_dir
|
|
dir /= resource.directory if resource.directory.present?
|
|
dir.cd do
|
|
patch_files.each do |patch_file|
|
|
ohai "Applying #{patch_file}"
|
|
patch_file = patch_dir/patch_file
|
|
safe_system "patch", "-g", "0", "-f", "-#{strip}", "-i", patch_file
|
|
end
|
|
end
|
|
end
|
|
rescue ErrorDuringExecution => e
|
|
f = resource.owner.owner
|
|
cmd, *args = e.cmd
|
|
raise BuildError.new(f, cmd, args, ENV.to_hash)
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def inspect
|
|
"#<#{self.class.name}: #{strip.inspect} #{url.inspect}>"
|
|
end
|
|
end
|