119 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			119 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # frozen_string_literal: true
 | |
| 
 | |
| require "extend/ENV"
 | |
| require "sandbox"
 | |
| require "timeout"
 | |
| require "cli/parser"
 | |
| 
 | |
| module Homebrew
 | |
|   module_function
 | |
| 
 | |
|   def test_args
 | |
|     Homebrew::CLI::Parser.new do
 | |
|       usage_banner <<~EOS
 | |
|         `test` [<options>] <formula>
 | |
| 
 | |
|         Run the test method provided by an installed formula.
 | |
|         There is no standard output or return code, but generally it should notify the
 | |
|         user if something is wrong with the installed formula.
 | |
| 
 | |
|         *Example:* `brew install jruby && brew test jruby`
 | |
|       EOS
 | |
|       switch "--devel",
 | |
|              description: "Test the development version of a formula."
 | |
|       switch "--HEAD",
 | |
|              description: "Test the head version of a formula."
 | |
|       switch "--keep-tmp",
 | |
|              description: "Retain the temporary files created for the test."
 | |
|       switch :verbose
 | |
|       switch :debug
 | |
|       conflicts "--devel", "--HEAD"
 | |
|       min_named :formula
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def test
 | |
|     test_args.parse
 | |
| 
 | |
|     require "formula_assertions"
 | |
|     require "formula_free_port"
 | |
| 
 | |
|     args.resolved_formulae.each do |f|
 | |
|       # Cannot test uninstalled formulae
 | |
|       unless f.latest_version_installed?
 | |
|         ofail "Testing requires the latest version of #{f.full_name}"
 | |
|         next
 | |
|       end
 | |
| 
 | |
|       # Cannot test formulae without a test method
 | |
|       unless f.test_defined?
 | |
|         ofail "#{f.full_name} defines no test"
 | |
|         next
 | |
|       end
 | |
| 
 | |
|       # Don't test unlinked formulae
 | |
|       if !args.force? && !f.keg_only? && !f.linked?
 | |
|         ofail "#{f.full_name} is not linked"
 | |
|         next
 | |
|       end
 | |
| 
 | |
|       # Don't test formulae missing test dependencies
 | |
|       missing_test_deps = f.recursive_dependencies do |_, dependency|
 | |
|         Dependency.prune if dependency.installed?
 | |
|         next if dependency.test?
 | |
| 
 | |
|         Dependency.prune if dependency.optional?
 | |
|         Dependency.prune if dependency.build?
 | |
|       end.map(&:to_s)
 | |
|       unless missing_test_deps.empty?
 | |
|         ofail "#{f.full_name} is missing test dependencies: #{missing_test_deps.join(" ")}"
 | |
|         next
 | |
|       end
 | |
| 
 | |
|       puts "Testing #{f.full_name}"
 | |
| 
 | |
|       env = ENV.to_hash
 | |
| 
 | |
|       begin
 | |
|         exec_args = %W[
 | |
|           #{RUBY_PATH}
 | |
|           -W0
 | |
|           -I #{$LOAD_PATH.join(File::PATH_SEPARATOR)}
 | |
|           --
 | |
|           #{HOMEBREW_LIBRARY_PATH}/test.rb
 | |
|           #{f.path}
 | |
|         ].concat(args.options_only)
 | |
| 
 | |
|         if f.head?
 | |
|           exec_args << "--HEAD"
 | |
|         elsif f.devel?
 | |
|           exec_args << "--devel"
 | |
|         end
 | |
| 
 | |
|         Utils.safe_fork do
 | |
|           if Sandbox.available?
 | |
|             sandbox = Sandbox.new
 | |
|             f.logs.mkpath
 | |
|             sandbox.record_log(f.logs/"test.sandbox.log")
 | |
|             sandbox.allow_write_temp_and_cache
 | |
|             sandbox.allow_write_log(f)
 | |
|             sandbox.allow_write_xcode
 | |
|             sandbox.allow_write_path(HOMEBREW_PREFIX/"var/cache")
 | |
|             sandbox.allow_write_path(HOMEBREW_PREFIX/"var/homebrew/locks")
 | |
|             sandbox.allow_write_path(HOMEBREW_PREFIX/"var/log")
 | |
|             sandbox.allow_write_path(HOMEBREW_PREFIX/"var/run")
 | |
|             sandbox.exec(*exec_args)
 | |
|           else
 | |
|             exec(*exec_args)
 | |
|           end
 | |
|         end
 | |
|       rescue Exception => e # rubocop:disable Lint/RescueException
 | |
|         ofail "#{f.full_name}: failed"
 | |
|         puts e, e.backtrace
 | |
|       ensure
 | |
|         ENV.replace(env)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 | 
