285 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			285 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: false
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| # Helper functions for updating PyPI resources.
 | |
| #
 | |
| # @api private
 | |
| module PyPI
 | |
|   extend T::Sig
 | |
| 
 | |
|   module_function
 | |
| 
 | |
|   PYTHONHOSTED_URL_PREFIX = "https://files.pythonhosted.org/packages/"
 | |
|   private_constant :PYTHONHOSTED_URL_PREFIX
 | |
| 
 | |
|   @pipgrip_installed = nil
 | |
| 
 | |
|   # PyPI Package
 | |
|   #
 | |
|   # @api private
 | |
|   class Package
 | |
|     extend T::Sig
 | |
| 
 | |
|     attr_accessor :name
 | |
|     attr_accessor :extras
 | |
|     attr_accessor :version
 | |
| 
 | |
|     sig { params(package_string: String, is_url: T::Boolean).void }
 | |
|     def initialize(package_string, is_url: false)
 | |
|       @pypi_info = nil
 | |
| 
 | |
|       if is_url
 | |
|         match = if package_string.start_with?(PYTHONHOSTED_URL_PREFIX)
 | |
|           File.basename(package_string).match(/^(.+)-([a-z\d.]+?)(?:.tar.gz|.zip)$/)
 | |
|         end
 | |
|         raise ArgumentError, "package should be a valid PyPI url" if match.blank?
 | |
| 
 | |
|         @name = match[1]
 | |
|         @version = match[2]
 | |
|         return
 | |
|       end
 | |
| 
 | |
|       @name = package_string
 | |
|       @name, @version = @name.split("==") if @name.include? "=="
 | |
| 
 | |
|       return unless match = @name.match(/^(.*?)\[(.+)\]$/)
 | |
| 
 | |
|       @name = match[1]
 | |
|       @extras = match[2].split ","
 | |
|     end
 | |
| 
 | |
|     # Get name, URL, SHA-256 checksum, and latest version for a given PyPI package.
 | |
|     sig { params(version: T.nilable(T.any(String, Version))).returns(T.nilable(T::Array[String])) }
 | |
|     def pypi_info(version: nil)
 | |
|       return @pypi_info if @pypi_info.present? && version.blank?
 | |
| 
 | |
|       version ||= @version
 | |
|       metadata_url = if version.present?
 | |
|         "https://pypi.org/pypi/#{@name}/#{version}/json"
 | |
|       else
 | |
|         "https://pypi.org/pypi/#{@name}/json"
 | |
|       end
 | |
|       out, _, status = curl_output metadata_url, "--location"
 | |
| 
 | |
|       return unless status.success?
 | |
| 
 | |
|       begin
 | |
|         json = JSON.parse out
 | |
|       rescue JSON::ParserError
 | |
|         return
 | |
|       end
 | |
| 
 | |
|       sdist = json["urls"].find { |url| url["packagetype"] == "sdist" }
 | |
|       return json["info"]["name"] if sdist.nil?
 | |
| 
 | |
|       @pypi_info = [json["info"]["name"], sdist["url"], sdist["digests"]["sha256"], json["info"]["version"]]
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def valid_pypi_package?
 | |
|       info = pypi_info
 | |
|       info.present? && info.is_a?(Array)
 | |
|     end
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def to_s
 | |
|       out = @name
 | |
|       out += "[#{@extras.join(",")}]" if @extras.present?
 | |
|       out += "==#{@version}" if @version.present?
 | |
|       out
 | |
|     end
 | |
| 
 | |
|     sig { params(other: Package).returns(T::Boolean) }
 | |
|     def same_package?(other)
 | |
|       @name.tr("_", "-").casecmp(other.name.tr("_", "-")).zero?
 | |
|     end
 | |
| 
 | |
|     # Compare only names so we can use .include? on a Package array
 | |
|     sig { params(other: Package).returns(T::Boolean) }
 | |
|     def ==(other)
 | |
|       same_package?(other)
 | |
|     end
 | |
| 
 | |
|     sig { params(other: Package).returns(T.nilable(Integer)) }
 | |
|     def <=>(other)
 | |
|       @name <=> other.name
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   sig { params(url: String, version: T.any(String, Version)).returns(T.nilable(String)) }
 | |
|   def update_pypi_url(url, version)
 | |
|     package = Package.new url, is_url: true
 | |
| 
 | |
|     return unless package.valid_pypi_package?
 | |
| 
 | |
|     _, url = package.pypi_info(version: version)
 | |
|     url
 | |
|   rescue ArgumentError
 | |
|     nil
 | |
|   end
 | |
| 
 | |
|   # Return true if resources were checked (even if no change).
 | |
|   sig {
 | |
|     params(
 | |
|       formula:                  Formula,
 | |
|       version:                  T.nilable(String),
 | |
|       package_name:             T.nilable(String),
 | |
|       extra_packages:           T.nilable(T::Array[String]),
 | |
|       exclude_packages:         T.nilable(T::Array[String]),
 | |
|       print_only:               T.nilable(T::Boolean),
 | |
|       silent:                   T.nilable(T::Boolean),
 | |
|       ignore_non_pypi_packages: T.nilable(T::Boolean),
 | |
|     ).returns(T.nilable(T::Boolean))
 | |
|   }
 | |
|   def update_python_resources!(formula, version: nil, package_name: nil, extra_packages: nil, exclude_packages: nil,
 | |
|                                print_only: false, silent: false, ignore_non_pypi_packages: false)
 | |
| 
 | |
|     auto_update_list = formula.tap&.pypi_formula_mappings
 | |
|     if auto_update_list.present? && auto_update_list.key?(formula.full_name) &&
 | |
|        package_name.blank? && extra_packages.blank? && exclude_packages.blank?
 | |
| 
 | |
|       list_entry = auto_update_list[formula.full_name]
 | |
|       case list_entry
 | |
|       when false
 | |
|         unless print_only
 | |
|           odie "The resources for \"#{formula.name}\" need special attention. Please update them manually."
 | |
|         end
 | |
|       when String
 | |
|         package_name = list_entry
 | |
|       when Hash
 | |
|         package_name = list_entry["package_name"]
 | |
|         extra_packages = list_entry["extra_packages"]
 | |
|         exclude_packages = list_entry["exclude_packages"]
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     main_package = if package_name.present?
 | |
|       Package.new(package_name)
 | |
|     else
 | |
|       begin
 | |
|         Package.new(formula.stable.url, is_url: true)
 | |
|       rescue ArgumentError
 | |
|         nil
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     if main_package.blank?
 | |
|       return if ignore_non_pypi_packages
 | |
| 
 | |
|       odie <<~EOS
 | |
|         Could not infer PyPI package name from URL:
 | |
|           #{Formatter.url(formula.stable.url)}
 | |
|       EOS
 | |
|     end
 | |
| 
 | |
|     unless main_package.valid_pypi_package?
 | |
|       return if ignore_non_pypi_packages
 | |
| 
 | |
|       odie "\"#{main_package}\" is not available on PyPI."
 | |
|     end
 | |
| 
 | |
|     main_package.version = version if version.present?
 | |
| 
 | |
|     extra_packages = (extra_packages || []).map { |p| Package.new p }
 | |
|     exclude_packages = (exclude_packages || []).map { |p| Package.new p }
 | |
|     exclude_packages += %W[#{main_package.name} argparse pip setuptools wheel wsgiref].map { |p| Package.new p }
 | |
|     # remove packages from the exclude list if we've explicitly requested them as an extra package
 | |
|     exclude_packages.delete_if { |package| extra_packages.include?(package) }
 | |
| 
 | |
|     input_packages = [main_package]
 | |
|     extra_packages.each do |extra_package|
 | |
|       if !extra_package.valid_pypi_package? && !ignore_non_pypi_packages
 | |
|         odie "\"#{extra_package}\" is not available on PyPI."
 | |
|       end
 | |
| 
 | |
|       input_packages.each do |existing_package|
 | |
|         if existing_package.same_package?(extra_package) && existing_package.version != extra_package.version
 | |
|           odie "Conflicting versions specified for the `#{extra_package.name}` package: "\
 | |
|                 "#{existing_package.version}, #{extra_package.version}"
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       input_packages << extra_package unless input_packages.include? extra_package
 | |
|     end
 | |
| 
 | |
|     formula.resources.each do |resource|
 | |
|       if !print_only && !resource.url.start_with?(PYTHONHOSTED_URL_PREFIX)
 | |
|         odie "\"#{formula.name}\" contains non-PyPI resources. Please update the resources manually."
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     @pipgrip_installed ||= Formula["pipgrip"].any_version_installed?
 | |
|     odie '"pipgrip" must be installed (`brew install pipgrip`)' unless @pipgrip_installed
 | |
| 
 | |
|     ohai "Retrieving PyPI dependencies for \"#{input_packages.join(" ")}\"..." if !print_only && !silent
 | |
|     command = [Formula["pipgrip"].bin/"pipgrip", "--json", "--no-cache-dir", *input_packages.map(&:to_s)]
 | |
|     pipgrip_output = Utils.popen_read(*command)
 | |
|     unless $CHILD_STATUS.success?
 | |
|       odie <<~EOS
 | |
|         Unable to determine dependencies for \"#{input_packages.join(" ")}\" because of a failure when running
 | |
|         `#{command.join(" ")}`.
 | |
|         Please update the resources for \"#{formula.name}\" manually.
 | |
|       EOS
 | |
|     end
 | |
| 
 | |
|     found_packages = JSON.parse(pipgrip_output).to_h.map do |new_name, new_version|
 | |
|       Package.new("#{new_name}==#{new_version}")
 | |
|     end
 | |
| 
 | |
|     new_resource_blocks = ""
 | |
|     found_packages.sort.each do |package|
 | |
|       if exclude_packages.include? package
 | |
|         ohai "Excluding \"#{package}\"" if !print_only && !silent
 | |
|         next
 | |
|       end
 | |
| 
 | |
|       ohai "Getting PyPI info for \"#{package}\"" if !print_only && !silent
 | |
|       name, url, checksum = package.pypi_info
 | |
|       # Fail if unable to find name, url or checksum for any resource
 | |
|       if name.blank?
 | |
|         odie "Unable to resolve some dependencies. Please update the resources for \"#{formula.name}\" manually."
 | |
|       elsif url.blank? || checksum.blank?
 | |
|         odie <<~EOS
 | |
|           Unable to find the URL and/or sha256 for the \"#{name}\" resource.
 | |
|           Please update the resources for \"#{formula.name}\" manually.
 | |
|         EOS
 | |
|       end
 | |
| 
 | |
|       # Append indented resource block
 | |
|       new_resource_blocks += <<-EOS
 | |
|   resource "#{name}" do
 | |
|     url "#{url}"
 | |
|     sha256 "#{checksum}"
 | |
|   end
 | |
| 
 | |
|       EOS
 | |
|     end
 | |
| 
 | |
|     if print_only
 | |
|       puts new_resource_blocks.chomp
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     # Check whether resources already exist (excluding virtualenv dependencies)
 | |
|     if formula.resources.all? { |resource| resource.name.start_with?("homebrew-") }
 | |
|       # Place resources above install method
 | |
|       inreplace_regex = /  def install/
 | |
|       new_resource_blocks += "  def install"
 | |
|     else
 | |
|       # Replace existing resource blocks with new resource blocks
 | |
|       inreplace_regex = /  (resource .* do\s+url .*\s+sha256 .*\s+ end\s*)+/
 | |
|       new_resource_blocks += "  "
 | |
|     end
 | |
| 
 | |
|     ohai "Updating resource blocks" unless silent
 | |
|     Utils::Inreplace.inreplace formula.path do |s|
 | |
|       if s.inreplace_string.scan(inreplace_regex).length > 1
 | |
|         odie "Unable to update resource blocks for \"#{formula.name}\" automatically. Please update them manually."
 | |
|       end
 | |
|       s.sub! inreplace_regex, new_resource_blocks
 | |
|     end
 | |
| 
 | |
|     true
 | |
|   end
 | |
| end
 | 
