1196 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			1196 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| require "json"
 | |
| require "rexml/document"
 | |
| require "time"
 | |
| 
 | |
| class AbstractDownloadStrategy
 | |
|   extend Forwardable
 | |
|   include FileUtils
 | |
| 
 | |
|   attr_reader :meta, :name, :version, :resource
 | |
|   attr_reader :shutup
 | |
| 
 | |
|   def initialize(name, resource)
 | |
|     @name = name
 | |
|     @resource = resource
 | |
|     @url = resource.url
 | |
|     @version = resource.version
 | |
|     @meta = resource.specs
 | |
|   end
 | |
| 
 | |
|   # Download and cache the resource as {#cached_location}.
 | |
|   def fetch
 | |
|   end
 | |
| 
 | |
|   # Suppress output
 | |
|   def shutup!
 | |
|     @shutup = true
 | |
|   end
 | |
| 
 | |
|   def puts(*args)
 | |
|     super(*args) unless shutup
 | |
|   end
 | |
| 
 | |
|   def ohai(*args)
 | |
|     super(*args) unless shutup
 | |
|   end
 | |
| 
 | |
|   # Unpack {#cached_location} into the current working directory, and possibly
 | |
|   # chdir into the newly-unpacked directory.
 | |
|   # Unlike {Resource#stage}, this does not take a block.
 | |
|   def stage
 | |
|   end
 | |
| 
 | |
|   # @!attribute [r] cached_location
 | |
|   # The path to the cached file or directory associated with the resource.
 | |
|   def cached_location
 | |
|   end
 | |
| 
 | |
|   # @!attribute [r]
 | |
|   # return most recent modified time for all files in the current working directory after stage.
 | |
|   def source_modified_time
 | |
|     Pathname.pwd.to_enum(:find).select(&:file?).map(&:mtime).max
 | |
|   end
 | |
| 
 | |
|   # Remove {#cached_location} and any other files associated with the resource
 | |
|   # from the cache.
 | |
|   def clear_cache
 | |
|     rm_rf(cached_location)
 | |
|   end
 | |
| 
 | |
|   def expand_safe_system_args(args)
 | |
|     args = args.dup
 | |
|     args.each_with_index do |arg, ii|
 | |
|       next unless arg.is_a? Hash
 | |
|       if ARGV.verbose?
 | |
|         args.delete_at ii
 | |
|       else
 | |
|         args[ii] = arg[:quiet_flag]
 | |
|       end
 | |
|       return args
 | |
|     end
 | |
|     # 2 as default because commands are eg. svn up, git pull
 | |
|     args.insert(2, "-q") unless ARGV.verbose?
 | |
|     args
 | |
|   end
 | |
| 
 | |
|   def safe_system(*args)
 | |
|     if @shutup
 | |
|       quiet_system(*args) || raise(ErrorDuringExecution.new(args.shift, *args))
 | |
|     else
 | |
|       super(*args)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def quiet_safe_system(*args)
 | |
|     safe_system(*expand_safe_system_args(args))
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def xzpath
 | |
|     "#{HOMEBREW_PREFIX}/opt/xz/bin/xz"
 | |
|   end
 | |
| 
 | |
|   def lzippath
 | |
|     "#{HOMEBREW_PREFIX}/opt/lzip/bin/lzip"
 | |
|   end
 | |
| 
 | |
|   def lhapath
 | |
|     "#{HOMEBREW_PREFIX}/opt/lha/bin/lha"
 | |
|   end
 | |
| 
 | |
|   def cvspath
 | |
|     @cvspath ||= %W[
 | |
|       /usr/bin/cvs
 | |
|       #{HOMEBREW_PREFIX}/bin/cvs
 | |
|       #{HOMEBREW_PREFIX}/opt/cvs/bin/cvs
 | |
|       #{which("cvs")}
 | |
|     ].find { |p| File.executable? p }
 | |
|   end
 | |
| 
 | |
|   def hgpath
 | |
|     @hgpath ||= %W[
 | |
|       #{which("hg")}
 | |
|       #{HOMEBREW_PREFIX}/bin/hg
 | |
|       #{HOMEBREW_PREFIX}/opt/mercurial/bin/hg
 | |
|     ].find { |p| File.executable? p }
 | |
|   end
 | |
| 
 | |
|   def bzrpath
 | |
|     @bzrpath ||= %W[
 | |
|       #{which("bzr")}
 | |
|       #{HOMEBREW_PREFIX}/bin/bzr
 | |
|       #{HOMEBREW_PREFIX}/opt/bazaar/bin/bzr
 | |
|     ].find { |p| File.executable? p }
 | |
|   end
 | |
| 
 | |
|   def fossilpath
 | |
|     @fossilpath ||= %W[
 | |
|       #{which("fossil")}
 | |
|       #{HOMEBREW_PREFIX}/bin/fossil
 | |
|       #{HOMEBREW_PREFIX}/opt/fossil/bin/fossil
 | |
|     ].find { |p| File.executable? p }
 | |
|   end
 | |
| end
 | |
| 
 | |
| class VCSDownloadStrategy < AbstractDownloadStrategy
 | |
|   REF_TYPES = [:tag, :branch, :revisions, :revision].freeze
 | |
| 
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     @ref_type, @ref = extract_ref(meta)
 | |
|     @revision = meta[:revision]
 | |
|     @clone = HOMEBREW_CACHE/cache_filename
 | |
|   end
 | |
| 
 | |
|   def fetch
 | |
|     ohai "Cloning #{@url}"
 | |
| 
 | |
|     if cached_location.exist? && repo_valid?
 | |
|       puts "Updating #{cached_location}"
 | |
|       update
 | |
|     elsif cached_location.exist?
 | |
|       puts "Removing invalid repository from cache"
 | |
|       clear_cache
 | |
|       clone_repo
 | |
|     else
 | |
|       clone_repo
 | |
|     end
 | |
| 
 | |
|     version.update_commit(last_commit) if head?
 | |
| 
 | |
|     return unless @ref_type == :tag
 | |
|     return unless @revision && current_revision
 | |
|     return if current_revision == @revision
 | |
|     raise <<-EOS.undent
 | |
|       #{@ref} tag should be #{@revision}
 | |
|       but is actually #{current_revision}
 | |
|     EOS
 | |
|   end
 | |
| 
 | |
|   def fetch_last_commit
 | |
|     fetch
 | |
|     last_commit
 | |
|   end
 | |
| 
 | |
|   def commit_outdated?(commit)
 | |
|     @last_commit ||= fetch_last_commit
 | |
|     commit != @last_commit
 | |
|   end
 | |
| 
 | |
|   def cached_location
 | |
|     @clone
 | |
|   end
 | |
| 
 | |
|   delegate head?: :version
 | |
| 
 | |
|   # Return last commit's unique identifier for the repository.
 | |
|   # Return most recent modified timestamp unless overridden.
 | |
|   def last_commit
 | |
|     source_modified_time.to_i.to_s
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def cache_tag
 | |
|     "__UNKNOWN__"
 | |
|   end
 | |
| 
 | |
|   def cache_filename
 | |
|     "#{name}--#{cache_tag}"
 | |
|   end
 | |
| 
 | |
|   def repo_valid?
 | |
|     true
 | |
|   end
 | |
| 
 | |
|   def clone_repo
 | |
|   end
 | |
| 
 | |
|   def update
 | |
|   end
 | |
| 
 | |
|   def current_revision
 | |
|   end
 | |
| 
 | |
|   def extract_ref(specs)
 | |
|     key = REF_TYPES.find { |type| specs.key?(type) }
 | |
|     [key, specs[key]]
 | |
|   end
 | |
| end
 | |
| 
 | |
| class AbstractFileDownloadStrategy < AbstractDownloadStrategy
 | |
|   def stage
 | |
|     case type = cached_location.compression_type
 | |
|     when :zip
 | |
|       with_system_path { quiet_safe_system "unzip", "-qq", cached_location }
 | |
|       chdir
 | |
|     when :gzip_only
 | |
|       with_system_path { buffered_write("gunzip") }
 | |
|     when :bzip2_only
 | |
|       with_system_path { buffered_write("bunzip2") }
 | |
|     when :gzip, :bzip2, :xz, :compress, :tar
 | |
|       tar_flags = "x"
 | |
|       if type == :gzip
 | |
|         tar_flags << "z"
 | |
|       elsif type == :bzip2
 | |
|         tar_flags << "j"
 | |
|       elsif type == :xz
 | |
|         tar_flags << "J"
 | |
|       end
 | |
|       tar_flags << "f"
 | |
|       with_system_path do
 | |
|         if type == :xz && DependencyCollector.tar_needs_xz_dependency?
 | |
|           pipe_to_tar(xzpath)
 | |
|         else
 | |
|           safe_system "tar", tar_flags, cached_location
 | |
|         end
 | |
|       end
 | |
|       chdir
 | |
|     when :lzip
 | |
|       with_system_path { pipe_to_tar(lzippath) }
 | |
|       chdir
 | |
|     when :lha
 | |
|       safe_system lhapath, "x", cached_location
 | |
|     when :xar
 | |
|       safe_system "/usr/bin/xar", "-xf", cached_location
 | |
|     when :rar
 | |
|       quiet_safe_system "unrar", "x", "-inul", cached_location
 | |
|     when :p7zip
 | |
|       safe_system "7zr", "x", cached_location
 | |
|     else
 | |
|       cp cached_location, basename_without_params, preserve: true
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def chdir
 | |
|     entries = Dir["*"]
 | |
|     case entries.length
 | |
|     when 0 then raise "Empty archive"
 | |
|     when 1 then begin
 | |
|         Dir.chdir entries.first
 | |
|       rescue
 | |
|         nil
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def pipe_to_tar(tool)
 | |
|     Utils.popen_read(tool, "-dc", cached_location.to_s) do |rd|
 | |
|       Utils.popen_write("tar", "xf", "-") do |wr|
 | |
|         buf = ""
 | |
|         wr.write(buf) while rd.read(16384, buf)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # gunzip and bunzip2 write the output file in the same directory as the input
 | |
|   # file regardless of the current working directory, so we need to write it to
 | |
|   # the correct location ourselves.
 | |
|   def buffered_write(tool)
 | |
|     target = File.basename(basename_without_params, cached_location.extname)
 | |
| 
 | |
|     Utils.popen_read(tool, "-f", cached_location.to_s, "-c") do |pipe|
 | |
|       File.open(target, "wb") do |f|
 | |
|         buf = ""
 | |
|         f.write(buf) while pipe.read(16384, buf)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def basename_without_params
 | |
|     # Strip any ?thing=wad out of .c?thing=wad style extensions
 | |
|     File.basename(@url)[/[^?]+/]
 | |
|   end
 | |
| 
 | |
|   def ext
 | |
|     # We need a Pathname because we've monkeypatched extname to support double
 | |
|     # extensions (e.g. tar.gz).
 | |
|     # We can't use basename_without_params, because given a URL like
 | |
|     #   https://example.com/download.php?file=foo-1.0.tar.gz
 | |
|     # the extension we want is ".tar.gz", not ".php".
 | |
|     Pathname.new(@url).extname[/[^?]+/]
 | |
|   end
 | |
| end
 | |
| 
 | |
| class CurlDownloadStrategy < AbstractFileDownloadStrategy
 | |
|   attr_reader :mirrors, :tarball_path, :temporary_path
 | |
| 
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     @mirrors = resource.mirrors.dup
 | |
|     @tarball_path = HOMEBREW_CACHE/"#{name}-#{version}#{ext}"
 | |
|     @temporary_path = Pathname.new("#{cached_location}.incomplete")
 | |
|   end
 | |
| 
 | |
|   def fetch
 | |
|     ohai "Downloading #{@url}"
 | |
| 
 | |
|     if cached_location.exist?
 | |
|       puts "Already downloaded: #{cached_location}"
 | |
|     else
 | |
|       had_incomplete_download = temporary_path.exist?
 | |
|       begin
 | |
|         _fetch
 | |
|       rescue ErrorDuringExecution
 | |
|         # 33 == range not supported
 | |
|         # try wiping the incomplete download and retrying once
 | |
|         unless $CHILD_STATUS.exitstatus == 33 && had_incomplete_download
 | |
|           raise CurlDownloadStrategyError, @url
 | |
|         end
 | |
| 
 | |
|         ohai "Trying a full download"
 | |
|         temporary_path.unlink
 | |
|         had_incomplete_download = false
 | |
|         retry
 | |
|       end
 | |
|       ignore_interrupts { temporary_path.rename(cached_location) }
 | |
|     end
 | |
|   rescue CurlDownloadStrategyError
 | |
|     raise if mirrors.empty?
 | |
|     puts "Trying a mirror..."
 | |
|     @url = mirrors.shift
 | |
|     retry
 | |
|   end
 | |
| 
 | |
|   def cached_location
 | |
|     tarball_path
 | |
|   end
 | |
| 
 | |
|   def clear_cache
 | |
|     super
 | |
|     rm_rf(temporary_path)
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   # Private method, can be overridden if needed.
 | |
|   def _fetch
 | |
|     url = @url
 | |
| 
 | |
|     if ENV["HOMEBREW_ARTIFACT_DOMAIN"]
 | |
|       url = url.sub(%r{^((ht|f)tps?://)?}, ENV["HOMEBREW_ARTIFACT_DOMAIN"].chomp("/") + "/")
 | |
|       ohai "Downloading from #{url}"
 | |
|     end
 | |
| 
 | |
|     urls = actual_urls(url)
 | |
|     unless urls.empty?
 | |
|       ohai "Downloading from #{urls.last}"
 | |
|       if !ENV["HOMEBREW_NO_INSECURE_REDIRECT"].nil? && url.start_with?("https://") &&
 | |
|          urls.any? { |u| !u.start_with? "https://" }
 | |
|         puts "HTTPS to HTTP redirect detected & HOMEBREW_NO_INSECURE_REDIRECT is set."
 | |
|         raise CurlDownloadStrategyError, url
 | |
|       end
 | |
|       url = urls.last
 | |
|     end
 | |
| 
 | |
|     curl url, "-C", downloaded_size, "-o", temporary_path
 | |
|   end
 | |
| 
 | |
|   # Curl options to be always passed to curl,
 | |
|   # with raw head calls (`curl -I`) or with actual `fetch`.
 | |
|   def _curl_opts
 | |
|     copts = []
 | |
|     copts << "--user" << meta.fetch(:user) if meta.key?(:user)
 | |
|     copts
 | |
|   end
 | |
| 
 | |
|   def actual_urls(url)
 | |
|     urls = []
 | |
|     curl_args = _curl_opts << "-I" << "-L" << url
 | |
|     Utils.popen_read("curl", *curl_args).scan(/^Location: (.+)$/).map do |m|
 | |
|       urls << URI.join(urls.last || url, m.first.chomp).to_s
 | |
|     end
 | |
|     urls
 | |
|   end
 | |
| 
 | |
|   def downloaded_size
 | |
|     temporary_path.size? || 0
 | |
|   end
 | |
| 
 | |
|   def curl(*args)
 | |
|     args.concat _curl_opts
 | |
|     args << "--connect-timeout" << "5" unless mirrors.empty?
 | |
|     super
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Detect and download from Apache Mirror
 | |
| class CurlApacheMirrorDownloadStrategy < CurlDownloadStrategy
 | |
|   def apache_mirrors
 | |
|     rd, wr = IO.pipe
 | |
|     buf = ""
 | |
| 
 | |
|     pid = fork do
 | |
|       ENV.delete "HOMEBREW_CURL_VERBOSE"
 | |
|       rd.close
 | |
|       $stdout.reopen(wr)
 | |
|       $stderr.reopen(wr)
 | |
|       curl "#{@url}&asjson=1"
 | |
|     end
 | |
|     wr.close
 | |
| 
 | |
|     rd.readline if ARGV.verbose? # Remove Homebrew output
 | |
|     buf << rd.read until rd.eof?
 | |
|     rd.close
 | |
|     Process.wait(pid)
 | |
|     buf
 | |
|   end
 | |
| 
 | |
|   def _fetch
 | |
|     return super if @tried_apache_mirror
 | |
|     @tried_apache_mirror = true
 | |
| 
 | |
|     mirrors = JSON.parse(apache_mirrors)
 | |
|     path_info = mirrors.fetch("path_info")
 | |
|     @url = mirrors.fetch("preferred") + path_info
 | |
|     @mirrors |= %W[https://archive.apache.org/dist/#{path_info}]
 | |
| 
 | |
|     ohai "Best Mirror #{@url}"
 | |
|     super
 | |
|   rescue IndexError, JSON::ParserError
 | |
|     raise CurlDownloadStrategyError, "Couldn't determine mirror, try again later."
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Download via an HTTP POST.
 | |
| # Query parameters on the URL are converted into POST parameters
 | |
| class CurlPostDownloadStrategy < CurlDownloadStrategy
 | |
|   def _fetch
 | |
|     base_url, data = @url.split("?")
 | |
|     curl base_url, "-d", data, "-C", downloaded_size, "-o", temporary_path
 | |
|   end
 | |
| end
 | |
| 
 | |
| # Use this strategy to download but not unzip a file.
 | |
| # Useful for installing jars.
 | |
| class NoUnzipCurlDownloadStrategy < CurlDownloadStrategy
 | |
|   def stage
 | |
|     cp cached_location, basename_without_params, preserve: true
 | |
|   end
 | |
| end
 | |
| 
 | |
| # This strategy extracts our binary packages.
 | |
| class CurlBottleDownloadStrategy < CurlDownloadStrategy
 | |
|   def stage
 | |
|     ohai "Pouring #{cached_location.basename}"
 | |
|     super
 | |
|   end
 | |
| end
 | |
| 
 | |
| # This strategy extracts local binary packages.
 | |
| class LocalBottleDownloadStrategy < AbstractFileDownloadStrategy
 | |
|   attr_reader :cached_location
 | |
| 
 | |
|   def initialize(path)
 | |
|     @cached_location = path
 | |
|   end
 | |
| 
 | |
|   def stage
 | |
|     ohai "Pouring #{cached_location.basename}"
 | |
|     super
 | |
|   end
 | |
| end
 | |
| 
 | |
| # S3DownloadStrategy downloads tarballs from AWS S3.
 | |
| # To use it, add ":using => S3DownloadStrategy" to the URL section of your
 | |
| # formula.  This download strategy uses AWS access tokens (in the
 | |
| # environment variables AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY)
 | |
| # to sign the request.  This strategy is good in a corporate setting,
 | |
| # because it lets you use a private S3 bucket as a repo for internal
 | |
| # distribution.  (It will work for public buckets as well.)
 | |
| class S3DownloadStrategy < CurlDownloadStrategy
 | |
|   def _fetch
 | |
|     # Put the aws gem requirement here (vs top of file) so it's only
 | |
|     # a dependency of S3 users, not all Homebrew users
 | |
|     require "rubygems"
 | |
|     begin
 | |
|       require "aws-sdk-v1"
 | |
|     rescue LoadError
 | |
|       onoe "Install the aws-sdk gem into the gem repo used by brew."
 | |
|       raise
 | |
|     end
 | |
| 
 | |
|     if @url !~ %r{^https?://([^.].*)\.s3\.amazonaws\.com/(.+)$}
 | |
|       raise "Bad S3 URL: " + @url
 | |
|     end
 | |
|     bucket = Regexp.last_match(1)
 | |
|     key = Regexp.last_match(2)
 | |
| 
 | |
|     ENV["AWS_ACCESS_KEY_ID"] = ENV["HOMEBREW_AWS_ACCESS_KEY_ID"]
 | |
|     ENV["AWS_SECRET_ACCESS_KEY"] = ENV["HOMEBREW_AWS_SECRET_ACCESS_KEY"]
 | |
| 
 | |
|     obj = AWS::S3.new.buckets[bucket].objects[key]
 | |
|     begin
 | |
|       s3url = obj.url_for(:get)
 | |
|     rescue AWS::Errors::MissingCredentialsError
 | |
|       ohai "AWS credentials missing, trying public URL instead."
 | |
|       s3url = obj.public_url
 | |
|     end
 | |
| 
 | |
|     curl s3url, "-C", downloaded_size, "-o", temporary_path
 | |
|   end
 | |
| end
 | |
| 
 | |
| # GitHubPrivateRepositoryDownloadStrategy downloads contents from GitHub
 | |
| # Private Repository. To use it, add
 | |
| # ":using => GitHubPrivateRepositoryDownloadStrategy" to the URL section of
 | |
| # your formula. This download strategy uses GitHub access tokens (in the
 | |
| # environment variables HOMEBREW_GITHUB_API_TOKEN) to sign the request.  This
 | |
| # strategy is suitable for corporate use just like S3DownloadStrategy, because
 | |
| # it lets you use a private GitHub repository for internal distribution.  It
 | |
| # works with public one, but in that case simply use CurlDownloadStrategy.
 | |
| class GitHubPrivateRepositoryDownloadStrategy < CurlDownloadStrategy
 | |
|   require "utils/formatter"
 | |
|   require "utils/github"
 | |
| 
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     parse_url_pattern
 | |
|     set_github_token
 | |
|   end
 | |
| 
 | |
|   def parse_url_pattern
 | |
|     url_pattern = %r{https://github.com/([^/]+)/([^/]+)/(\S+)}
 | |
|     unless @url =~ url_pattern
 | |
|       raise CurlDownloadStrategyError, "Invalid url pattern for GitHub Repository."
 | |
|     end
 | |
| 
 | |
|     _, @owner, @repo, @filepath = *@url.match(url_pattern)
 | |
|   end
 | |
| 
 | |
|   def download_url
 | |
|     "https://#{@github_token}@github.com/#{@owner}/#{@repo}/#{@filepath}"
 | |
|   end
 | |
| 
 | |
|   def _fetch
 | |
|     curl download_url, "-C", downloaded_size, "-o", temporary_path
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def set_github_token
 | |
|     @github_token = ENV["HOMEBREW_GITHUB_API_TOKEN"]
 | |
|     unless @github_token
 | |
|       raise CurlDownloadStrategyError, "Environmental variable HOMEBREW_GITHUB_API_TOKEN is required."
 | |
|     end
 | |
|     validate_github_repository_access!
 | |
|   end
 | |
| 
 | |
|   def validate_github_repository_access!
 | |
|     # Test access to the repository
 | |
|     GitHub.repository(@owner, @repo)
 | |
|   rescue GitHub::HTTPNotFoundError
 | |
|     # We only handle HTTPNotFoundError here,
 | |
|     # becase AuthenticationFailedError is handled within util/github.
 | |
|     message = <<-EOS.undent
 | |
|         HOMEBREW_GITHUB_API_TOKEN can not access the repository: #{@owner}/#{@repo}
 | |
|         This token may not have permission to access the repository or the url of formula may be incorrect.
 | |
|     EOS
 | |
|     raise CurlDownloadStrategyError, message
 | |
|   end
 | |
| end
 | |
| 
 | |
| # GitHubPrivateRepositoryReleaseDownloadStrategy downloads tarballs from GitHub
 | |
| # Release assets. To use it, add
 | |
| # ":using => GitHubPrivateRepositoryReleaseDownloadStrategy" to the URL section
 | |
| # of your formula. This download strategy uses GitHub access tokens (in the
 | |
| # environment variables HOMEBREW_GITHUB_API_TOKEN) to sign the request.
 | |
| class GitHubPrivateRepositoryReleaseDownloadStrategy < GitHubPrivateRepositoryDownloadStrategy
 | |
|   def parse_url_pattern
 | |
|     url_pattern = %r{https://github.com/([^/]+)/([^/]+)/releases/download/([^/]+)/(\S+)}
 | |
|     unless @url =~ url_pattern
 | |
|       raise CurlDownloadStrategyError, "Invalid url pattern for GitHub Release."
 | |
|     end
 | |
| 
 | |
|     _, @owner, @repo, @tag, @filename = *@url.match(url_pattern)
 | |
|   end
 | |
| 
 | |
|   def download_url
 | |
|     "https://#{@github_token}@api.github.com/repos/#{@owner}/#{@repo}/releases/assets/#{asset_id}"
 | |
|   end
 | |
| 
 | |
|   def _fetch
 | |
|     # HTTP request header `Accept: application/octet-stream` is required.
 | |
|     # Without this, the GitHub API will respond with metadata, not binary.
 | |
|     curl download_url, "-C", downloaded_size, "-o", temporary_path, "-H", "Accept: application/octet-stream"
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def asset_id
 | |
|     @asset_id ||= resolve_asset_id
 | |
|   end
 | |
| 
 | |
|   def resolve_asset_id
 | |
|     release_metadata = fetch_release_metadata
 | |
|     assets = release_metadata["assets"].select { |a| a["name"] == @filename }
 | |
|     raise CurlDownloadStrategyError, "Asset file not found." if assets.empty?
 | |
| 
 | |
|     assets.first["id"]
 | |
|   end
 | |
| 
 | |
|   def fetch_release_metadata
 | |
|     release_url = "https://api.github.com/repos/#{@owner}/#{@repo}/releases/tags/#{@tag}"
 | |
|     GitHub.open(release_url)
 | |
|   end
 | |
| end
 | |
| 
 | |
| class SubversionDownloadStrategy < VCSDownloadStrategy
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     @url = @url.sub("svn+http://", "")
 | |
|   end
 | |
| 
 | |
|   def fetch
 | |
|     clear_cache unless @url.chomp("/") == repo_url || quiet_system("svn", "switch", @url, cached_location)
 | |
|     super
 | |
|   end
 | |
| 
 | |
|   def stage
 | |
|     super
 | |
|     quiet_safe_system "svn", "export", "--force", cached_location, Dir.pwd
 | |
|   end
 | |
| 
 | |
|   def source_modified_time
 | |
|     xml = REXML::Document.new(Utils.popen_read("svn", "info", "--xml", cached_location.to_s))
 | |
|     Time.parse REXML::XPath.first(xml, "//date/text()").to_s
 | |
|   end
 | |
| 
 | |
|   def last_commit
 | |
|     Utils.popen_read("svn", "info", "--show-item", "revision", cached_location.to_s).strip
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def repo_url
 | |
|     Utils.popen_read("svn", "info", cached_location.to_s).strip[/^URL: (.+)$/, 1]
 | |
|   end
 | |
| 
 | |
|   def externals
 | |
|     Utils.popen_read("svn", "propget", "svn:externals", @url).chomp.each_line do |line|
 | |
|       name, url = line.split(/\s+/)
 | |
|       yield name, url
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def fetch_repo(target, url, revision = nil, ignore_externals = false)
 | |
|     # Use "svn up" when the repository already exists locally.
 | |
|     # This saves on bandwidth and will have a similar effect to verifying the
 | |
|     # cache as it will make any changes to get the right revision.
 | |
|     svncommand = target.directory? ? "up" : "checkout"
 | |
|     args = ["svn", svncommand]
 | |
|     args << url unless target.directory?
 | |
|     args << target
 | |
|     if revision
 | |
|       ohai "Checking out #{@ref}"
 | |
|       args << "-r" << revision
 | |
|     end
 | |
|     args << "--ignore-externals" if ignore_externals
 | |
|     quiet_safe_system(*args)
 | |
|   end
 | |
| 
 | |
|   def cache_tag
 | |
|     head? ? "svn-HEAD" : "svn"
 | |
|   end
 | |
| 
 | |
|   def repo_valid?
 | |
|     (cached_location/".svn").directory?
 | |
|   end
 | |
| 
 | |
|   def clone_repo
 | |
|     case @ref_type
 | |
|     when :revision
 | |
|       fetch_repo cached_location, @url, @ref
 | |
|     when :revisions
 | |
|       # nil is OK for main_revision, as fetch_repo will then get latest
 | |
|       main_revision = @ref[:trunk]
 | |
|       fetch_repo cached_location, @url, main_revision, true
 | |
| 
 | |
|       externals do |external_name, external_url|
 | |
|         fetch_repo cached_location/external_name, external_url, @ref[external_name], true
 | |
|       end
 | |
|     else
 | |
|       fetch_repo cached_location, @url
 | |
|     end
 | |
|   end
 | |
|   alias update clone_repo
 | |
| end
 | |
| 
 | |
| class GitDownloadStrategy < VCSDownloadStrategy
 | |
|   SHALLOW_CLONE_WHITELIST = [
 | |
|     %r{git://},
 | |
|     %r{https://github\.com},
 | |
|     %r{http://git\.sv\.gnu\.org},
 | |
|     %r{http://llvm\.org},
 | |
|   ].freeze
 | |
| 
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     @ref_type ||= :branch
 | |
|     @ref ||= "master"
 | |
|     @shallow = meta.fetch(:shallow) { true }
 | |
|   end
 | |
| 
 | |
|   def stage
 | |
|     super
 | |
|     cp_r File.join(cached_location, "."), Dir.pwd, preserve: true
 | |
|   end
 | |
| 
 | |
|   def source_modified_time
 | |
|     Time.parse Utils.popen_read("git", "--git-dir", git_dir, "show", "-s", "--format=%cD")
 | |
|   end
 | |
| 
 | |
|   def last_commit
 | |
|     Utils.popen_read("git", "--git-dir", git_dir, "rev-parse", "--short=7", "HEAD").chomp
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def cache_tag
 | |
|     "git"
 | |
|   end
 | |
| 
 | |
|   def cache_version
 | |
|     0
 | |
|   end
 | |
| 
 | |
|   def update
 | |
|     cached_location.cd do
 | |
|       config_repo
 | |
|       update_repo
 | |
|       checkout
 | |
|       reset
 | |
|       update_submodules if submodules?
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def shallow_clone?
 | |
|     @shallow && support_depth?
 | |
|   end
 | |
| 
 | |
|   def shallow_dir?
 | |
|     (git_dir/"shallow").exist?
 | |
|   end
 | |
| 
 | |
|   def support_depth?
 | |
|     @ref_type != :revision && SHALLOW_CLONE_WHITELIST.any? { |regex| @url =~ regex }
 | |
|   end
 | |
| 
 | |
|   def git_dir
 | |
|     cached_location/".git"
 | |
|   end
 | |
| 
 | |
|   def ref?
 | |
|     quiet_system "git", "--git-dir", git_dir, "rev-parse", "-q", "--verify", "#{@ref}^{commit}"
 | |
|   end
 | |
| 
 | |
|   def current_revision
 | |
|     Utils.popen_read("git", "--git-dir", git_dir, "rev-parse", "-q", "--verify", "HEAD").strip
 | |
|   end
 | |
| 
 | |
|   def repo_valid?
 | |
|     quiet_system "git", "--git-dir", git_dir, "status", "-s"
 | |
|   end
 | |
| 
 | |
|   def submodules?
 | |
|     (cached_location/".gitmodules").exist?
 | |
|   end
 | |
| 
 | |
|   def clone_args
 | |
|     args = %w[clone]
 | |
|     args << "--depth" << "1" if shallow_clone?
 | |
| 
 | |
|     case @ref_type
 | |
|     when :branch, :tag
 | |
|       args << "--branch" << @ref
 | |
|     end
 | |
| 
 | |
|     args << @url << cached_location
 | |
|   end
 | |
| 
 | |
|   def refspec
 | |
|     case @ref_type
 | |
|     when :branch then "+refs/heads/#{@ref}:refs/remotes/origin/#{@ref}"
 | |
|     when :tag    then "+refs/tags/#{@ref}:refs/tags/#{@ref}"
 | |
|     else              "+refs/heads/master:refs/remotes/origin/master"
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def config_repo
 | |
|     safe_system "git", "config", "remote.origin.url", @url
 | |
|     safe_system "git", "config", "remote.origin.fetch", refspec
 | |
|   end
 | |
| 
 | |
|   def update_repo
 | |
|     return unless @ref_type == :branch || !ref?
 | |
| 
 | |
|     if !shallow_clone? && shallow_dir?
 | |
|       quiet_safe_system "git", "fetch", "origin", "--unshallow"
 | |
|     else
 | |
|       quiet_safe_system "git", "fetch", "origin"
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def clone_repo
 | |
|     safe_system "git", *clone_args
 | |
|     cached_location.cd do
 | |
|       safe_system "git", "config", "homebrew.cacheversion", cache_version
 | |
|       checkout
 | |
|       update_submodules if submodules?
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def checkout
 | |
|     ohai "Checking out #{@ref_type} #{@ref}" if @ref_type && @ref
 | |
|     quiet_safe_system "git", "checkout", "-f", @ref, "--"
 | |
|   end
 | |
| 
 | |
|   def reset_args
 | |
|     ref = case @ref_type
 | |
|     when :branch
 | |
|       "origin/#{@ref}"
 | |
|     when :revision, :tag
 | |
|       @ref
 | |
|     end
 | |
| 
 | |
|     %W[reset --hard #{ref}]
 | |
|   end
 | |
| 
 | |
|   def reset
 | |
|     quiet_safe_system "git", *reset_args
 | |
|   end
 | |
| 
 | |
|   def update_submodules
 | |
|     quiet_safe_system "git", "submodule", "foreach", "--recursive", "git submodule sync"
 | |
|     quiet_safe_system "git", "submodule", "update", "--init", "--recursive"
 | |
|     fix_absolute_submodule_gitdir_references!
 | |
|   end
 | |
| 
 | |
|   def fix_absolute_submodule_gitdir_references!
 | |
|     # When checking out Git repositories with recursive submodules, some Git
 | |
|     # versions create `.git` files with absolute instead of relative `gitdir:`
 | |
|     # pointers. This works for the cached location, but breaks various Git
 | |
|     # operations once the affected Git resource is staged, i.e. recursively
 | |
|     # copied to a new location. (This bug was introduced in Git 2.7.0 and fixed
 | |
|     # in 2.8.3. Clones created with affected version remain broken.)
 | |
|     # See https://github.com/Homebrew/homebrew-core/pull/1520 for an example.
 | |
|     submodule_dirs = Utils.popen_read(
 | |
|       "git", "submodule", "--quiet", "foreach", "--recursive", "pwd"
 | |
|     )
 | |
|     submodule_dirs.lines.map(&:chomp).each do |submodule_dir|
 | |
|       work_dir = Pathname.new(submodule_dir)
 | |
| 
 | |
|       # Only check and fix if `.git` is a regular file, not a directory.
 | |
|       dot_git = work_dir/".git"
 | |
|       next unless dot_git.file?
 | |
| 
 | |
|       git_dir = dot_git.read.chomp[/^gitdir: (.*)$/, 1]
 | |
|       if git_dir.nil?
 | |
|         onoe "Failed to parse '#{dot_git}'." if ARGV.homebrew_developer?
 | |
|         next
 | |
|       end
 | |
| 
 | |
|       # Only attempt to fix absolute paths.
 | |
|       next unless git_dir.start_with?("/")
 | |
| 
 | |
|       # Make the `gitdir:` reference relative to the working directory.
 | |
|       relative_git_dir = Pathname.new(git_dir).relative_path_from(work_dir)
 | |
|       dot_git.atomic_write("gitdir: #{relative_git_dir}\n")
 | |
|     end
 | |
|   end
 | |
| end
 | |
| 
 | |
| class GitHubGitDownloadStrategy < GitDownloadStrategy
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
| 
 | |
|     return unless %r{^https?://github\.com/(?<user>[^/]+)/(?<repo>[^/]+)\.git$} =~ @url
 | |
|     @user = user
 | |
|     @repo = repo
 | |
|   end
 | |
| 
 | |
|   def github_last_commit
 | |
|     return if ENV["HOMEBREW_NO_GITHUB_API"]
 | |
| 
 | |
|     output, _, status = curl_output "-H", "Accept: application/vnd.github.v3.sha", \
 | |
|       "-I", "https://api.github.com/repos/#{@user}/#{@repo}/commits/#{@ref}"
 | |
| 
 | |
|     commit = output[/^ETag: \"(\h+)\"/, 1] if status.success?
 | |
|     version.update_commit(commit) if commit
 | |
|     commit
 | |
|   end
 | |
| 
 | |
|   def multiple_short_commits_exist?(commit)
 | |
|     return if ENV["HOMEBREW_NO_GITHUB_API"]
 | |
|     output, _, status = curl_output "-H", "Accept: application/vnd.github.v3.sha", \
 | |
|       "-I", "https://api.github.com/repos/#{@user}/#{@repo}/commits/#{commit}"
 | |
| 
 | |
|     !(status.success? && output && output[/^Status: (200)/, 1] == "200")
 | |
|   end
 | |
| 
 | |
|   def commit_outdated?(commit)
 | |
|     @last_commit ||= github_last_commit
 | |
|     if !@last_commit
 | |
|       super
 | |
|     else
 | |
|       return true unless commit
 | |
|       return true unless @last_commit.start_with?(commit)
 | |
|       if multiple_short_commits_exist?(commit)
 | |
|         true
 | |
|       else
 | |
|         version.update_commit(commit)
 | |
|         false
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 | |
| 
 | |
| class CVSDownloadStrategy < VCSDownloadStrategy
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     @url = @url.sub(%r{^cvs://}, "")
 | |
| 
 | |
|     if meta.key?(:module)
 | |
|       @module = meta.fetch(:module)
 | |
|     elsif @url !~ %r{:[^/]+$}
 | |
|       @module = name
 | |
|     else
 | |
|       @module, @url = split_url(@url)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def source_modified_time
 | |
|     # Filter CVS's files because the timestamp for each of them is the moment
 | |
|     # of clone.
 | |
|     max_mtime = Time.at(0)
 | |
|     cached_location.find do |f|
 | |
|       Find.prune if f.directory? && f.basename.to_s == "CVS"
 | |
|       next unless f.file?
 | |
|       mtime = f.mtime
 | |
|       max_mtime = mtime if mtime > max_mtime
 | |
|     end
 | |
|     max_mtime
 | |
|   end
 | |
| 
 | |
|   def stage
 | |
|     cp_r File.join(cached_location, "."), Dir.pwd, preserve: true
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def cache_tag
 | |
|     "cvs"
 | |
|   end
 | |
| 
 | |
|   def repo_valid?
 | |
|     (cached_location/"CVS").directory?
 | |
|   end
 | |
| 
 | |
|   def clone_repo
 | |
|     HOMEBREW_CACHE.cd do
 | |
|       # Login is only needed (and allowed) with pserver; skip for anoncvs.
 | |
|       quiet_safe_system cvspath, { quiet_flag: "-Q" }, "-d", @url, "login" if @url.include? "pserver"
 | |
|       quiet_safe_system cvspath, { quiet_flag: "-Q" }, "-d", @url, "checkout", "-d", cache_filename, @module
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def update
 | |
|     cached_location.cd { quiet_safe_system cvspath, { quiet_flag: "-Q" }, "up" }
 | |
|   end
 | |
| 
 | |
|   def split_url(in_url)
 | |
|     parts = in_url.split(/:/)
 | |
|     mod = parts.pop
 | |
|     url = parts.join(":")
 | |
|     [mod, url]
 | |
|   end
 | |
| end
 | |
| 
 | |
| class MercurialDownloadStrategy < VCSDownloadStrategy
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     @url = @url.sub(%r{^hg://}, "")
 | |
|   end
 | |
| 
 | |
|   def stage
 | |
|     super
 | |
| 
 | |
|     dst = Dir.getwd
 | |
|     cached_location.cd do
 | |
|       if @ref_type && @ref
 | |
|         ohai "Checking out #{@ref_type} #{@ref}" if @ref_type && @ref
 | |
|         safe_system hgpath, "archive", "--subrepos", "-y", "-r", @ref, "-t", "files", dst
 | |
|       else
 | |
|         safe_system hgpath, "archive", "--subrepos", "-y", "-t", "files", dst
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def source_modified_time
 | |
|     Time.parse Utils.popen_read("hg", "tip", "--template", "{date|isodate}", "-R", cached_location.to_s)
 | |
|   end
 | |
| 
 | |
|   def last_commit
 | |
|     Utils.popen_read("hg", "parent", "--template", "{node|short}", "-R", cached_location.to_s)
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def cache_tag
 | |
|     "hg"
 | |
|   end
 | |
| 
 | |
|   def repo_valid?
 | |
|     (cached_location/".hg").directory?
 | |
|   end
 | |
| 
 | |
|   def clone_repo
 | |
|     safe_system hgpath, "clone", @url, cached_location
 | |
|   end
 | |
| 
 | |
|   def update
 | |
|     cached_location.cd { quiet_safe_system hgpath, "pull", "--update" }
 | |
|   end
 | |
| end
 | |
| 
 | |
| class BazaarDownloadStrategy < VCSDownloadStrategy
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     @url = @url.sub(%r{^bzr://}, "")
 | |
|   end
 | |
| 
 | |
|   def stage
 | |
|     # The export command doesn't work on checkouts
 | |
|     # See https://bugs.launchpad.net/bzr/+bug/897511
 | |
|     cp_r File.join(cached_location, "."), Dir.pwd, preserve: true
 | |
|     rm_r ".bzr"
 | |
|   end
 | |
| 
 | |
|   def source_modified_time
 | |
|     Time.parse Utils.popen_read("bzr", "log", "-l", "1", "--timezone=utc", cached_location.to_s)[/^timestamp: (.+)$/, 1]
 | |
|   end
 | |
| 
 | |
|   def last_commit
 | |
|     Utils.popen_read("bzr", "revno", cached_location.to_s).chomp
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def cache_tag
 | |
|     "bzr"
 | |
|   end
 | |
| 
 | |
|   def repo_valid?
 | |
|     (cached_location/".bzr").directory?
 | |
|   end
 | |
| 
 | |
|   def clone_repo
 | |
|     # "lightweight" means history-less
 | |
|     safe_system bzrpath, "checkout", "--lightweight", @url, cached_location
 | |
|   end
 | |
| 
 | |
|   def update
 | |
|     cached_location.cd { quiet_safe_system bzrpath, "update" }
 | |
|   end
 | |
| end
 | |
| 
 | |
| class FossilDownloadStrategy < VCSDownloadStrategy
 | |
|   def initialize(name, resource)
 | |
|     super
 | |
|     @url = @url.sub(%r{^fossil://}, "")
 | |
|   end
 | |
| 
 | |
|   def stage
 | |
|     super
 | |
|     args = [fossilpath, "open", cached_location]
 | |
|     args << @ref if @ref_type && @ref
 | |
|     safe_system(*args)
 | |
|   end
 | |
| 
 | |
|   def source_modified_time
 | |
|     Time.parse Utils.popen_read("fossil", "info", "tip", "-R", cached_location.to_s)[/^uuid: +\h+ (.+)$/, 1]
 | |
|   end
 | |
| 
 | |
|   def last_commit
 | |
|     Utils.popen_read("fossil", "info", "tip", "-R", cached_location.to_s)[/^uuid: +(\h+) .+$/, 1]
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def cache_tag
 | |
|     "fossil"
 | |
|   end
 | |
| 
 | |
|   def clone_repo
 | |
|     safe_system fossilpath, "clone", @url, cached_location
 | |
|   end
 | |
| 
 | |
|   def update
 | |
|     safe_system fossilpath, "pull", "-R", cached_location
 | |
|   end
 | |
| end
 | |
| 
 | |
| class DownloadStrategyDetector
 | |
|   def self.detect(url, strategy = nil)
 | |
|     if strategy.nil?
 | |
|       detect_from_url(url)
 | |
|     elsif strategy.is_a?(Class) && strategy < AbstractDownloadStrategy
 | |
|       strategy
 | |
|     elsif strategy.is_a?(Symbol)
 | |
|       detect_from_symbol(strategy)
 | |
|     else
 | |
|       raise TypeError,
 | |
|         "Unknown download strategy specification #{strategy.inspect}"
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def self.detect_from_url(url)
 | |
|     case url
 | |
|     when %r{^https?://github\.com/[^/]+/[^/]+\.git$}
 | |
|       GitHubGitDownloadStrategy
 | |
|     when %r{^https?://.+\.git$}, %r{^git://}
 | |
|       GitDownloadStrategy
 | |
|     when %r{^https?://www\.apache\.org/dyn/closer\.cgi}, %r{^https?://www\.apache\.org/dyn/closer\.lua}
 | |
|       CurlApacheMirrorDownloadStrategy
 | |
|     when %r{^https?://(.+?\.)?googlecode\.com/svn}, %r{^https?://svn\.}, %r{^svn://}, %r{^https?://(.+?\.)?sourceforge\.net/svnroot/}
 | |
|       SubversionDownloadStrategy
 | |
|     when %r{^cvs://}
 | |
|       CVSDownloadStrategy
 | |
|     when %r{^https?://(.+?\.)?googlecode\.com/hg}
 | |
|       MercurialDownloadStrategy
 | |
|     when %r{^hg://}
 | |
|       MercurialDownloadStrategy
 | |
|     when %r{^bzr://}
 | |
|       BazaarDownloadStrategy
 | |
|     when %r{^fossil://}
 | |
|       FossilDownloadStrategy
 | |
|     when %r{^http://svn\.apache\.org/repos/}, %r{^svn\+http://}
 | |
|       SubversionDownloadStrategy
 | |
|     when %r{^https?://(.+?\.)?sourceforge\.net/hgweb/}
 | |
|       MercurialDownloadStrategy
 | |
|     else
 | |
|       CurlDownloadStrategy
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def self.detect_from_symbol(symbol)
 | |
|     case symbol
 | |
|     when :hg      then MercurialDownloadStrategy
 | |
|     when :nounzip then NoUnzipCurlDownloadStrategy
 | |
|     when :git     then GitDownloadStrategy
 | |
|     when :bzr     then BazaarDownloadStrategy
 | |
|     when :svn     then SubversionDownloadStrategy
 | |
|     when :curl    then CurlDownloadStrategy
 | |
|     when :ssl3    then CurlSSL3DownloadStrategy
 | |
|     when :cvs     then CVSDownloadStrategy
 | |
|     when :post    then CurlPostDownloadStrategy
 | |
|     when :fossil  then FossilDownloadStrategy
 | |
|     else
 | |
|       raise "Unknown download strategy #{symbol} was requested."
 | |
|     end
 | |
|   end
 | |
| end
 | 
