 c5dbd3ca24
			
		
	
	
		c5dbd3ca24
		
			
		
	
	
	
	
		
			
			This improves the load time of most brew commands. For an example of
one of the simplest commands this speeds up:
Without Bootsnap:
```
$ hyperfine 'git checkout master; brew help' 'git checkout optimise_requires; brew help'
Benchmark 1: git checkout master; brew help
  Time (mean ± σ):     525.0 ms ±  35.8 ms    [User: 229.9 ms, System: 113.1 ms]
  Range (min … max):   465.3 ms … 576.6 ms    10 runs
Benchmark 2: git checkout optimise_requires; brew help
  Time (mean ± σ):     383.3 ms ±  25.1 ms    [User: 133.0 ms, System: 72.1 ms]
  Range (min … max):   353.0 ms … 443.6 ms    10 runs
Summary
  git checkout optimise_requires; brew help ran
    1.37 ± 0.13 times faster than git checkout master; brew help
```
With Bootsnap:
```
$ hyperfine 'git checkout master; brew help' 'git checkout optimise_requires; brew help'
Benchmark 1: git checkout master; brew help
  Time (mean ± σ):     386.0 ms ±  30.9 ms    [User: 130.2 ms, System: 93.8 ms]
  Range (min … max):   359.5 ms … 469.3 ms    10 runs
Benchmark 2: git checkout optimise_requires; brew help
  Time (mean ± σ):     330.2 ms ±  32.4 ms    [User: 93.4 ms, System: 73.0 ms]
  Range (min … max):   302.9 ms … 413.9 ms    10 runs
Summary
  git checkout optimise_requires; brew help ran
    1.17 ± 0.15 times faster than git checkout master; brew help
```
		
	
			
		
			
				
	
	
		
			167 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			167 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: true
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "context"
 | |
| 
 | |
| module Homebrew
 | |
|   extend Context
 | |
| 
 | |
|   def self._system(cmd, *args, **options)
 | |
|     pid = fork do
 | |
|       yield if block_given?
 | |
|       args.map!(&:to_s)
 | |
|       begin
 | |
|         exec(cmd, *args, **options)
 | |
|       rescue
 | |
|         nil
 | |
|       end
 | |
|       exit! 1 # never gets here unless exec failed
 | |
|     end
 | |
|     Process.wait(T.must(pid))
 | |
|     $CHILD_STATUS.success?
 | |
|   end
 | |
| 
 | |
|   def self.system(cmd, *args, **options)
 | |
|     if verbose?
 | |
|       out = (options[:out] == :err) ? $stderr : $stdout
 | |
|       out.puts "#{cmd} #{args * " "}".gsub(RUBY_PATH, "ruby")
 | |
|                                      .gsub($LOAD_PATH.join(File::PATH_SEPARATOR).to_s, "$LOAD_PATH")
 | |
|     end
 | |
|     _system(cmd, *args, **options)
 | |
|   end
 | |
| 
 | |
|   # rubocop:disable Style/GlobalVars
 | |
|   sig { params(the_module: Module, pattern: Regexp).void }
 | |
|   def self.inject_dump_stats!(the_module, pattern)
 | |
|     @injected_dump_stat_modules ||= {}
 | |
|     @injected_dump_stat_modules[the_module] ||= []
 | |
|     injected_methods = @injected_dump_stat_modules[the_module]
 | |
|     the_module.module_eval do
 | |
|       instance_methods.grep(pattern).each do |name|
 | |
|         next if injected_methods.include? name
 | |
| 
 | |
|         method = instance_method(name)
 | |
|         define_method(name) do |*args, &block|
 | |
|           require "time"
 | |
| 
 | |
|           time = Time.now
 | |
| 
 | |
|           begin
 | |
|             method.bind_call(self, *args, &block)
 | |
|           ensure
 | |
|             $times[name] ||= 0
 | |
|             $times[name] += Time.now - time
 | |
|           end
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     return unless $times.nil?
 | |
| 
 | |
|     $times = {}
 | |
|     at_exit do
 | |
|       col_width = [$times.keys.map(&:size).max.to_i + 2, 15].max
 | |
|       $times.sort_by { |_k, v| v }.each do |method, time|
 | |
|         puts format("%<method>-#{col_width}s %<time>0.4f sec", method: "#{method}:", time:)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
|   # rubocop:enable Style/GlobalVars
 | |
| end
 | |
| 
 | |
| module Utils
 | |
|   # Removes the rightmost segment from the constant expression in the string.
 | |
|   #
 | |
|   #   deconstantize('Net::HTTP')   # => "Net"
 | |
|   #   deconstantize('::Net::HTTP') # => "::Net"
 | |
|   #   deconstantize('String')      # => ""
 | |
|   #   deconstantize('::String')    # => ""
 | |
|   #   deconstantize('')            # => ""
 | |
|   #
 | |
|   # See also #demodulize.
 | |
|   # @see https://github.com/rails/rails/blob/b0dd7c7/activesupport/lib/active_support/inflector/methods.rb#L247-L258
 | |
|   #   `ActiveSupport::Inflector.deconstantize`
 | |
|   sig { params(path: String).returns(String) }
 | |
|   def self.deconstantize(path)
 | |
|     T.must(path[0, path.rindex("::") || 0]) # implementation based on the one in facets' Module#spacename
 | |
|   end
 | |
| 
 | |
|   # Removes the module part from the expression in the string.
 | |
|   #
 | |
|   #   demodulize('ActiveSupport::Inflector::Inflections') # => "Inflections"
 | |
|   #   demodulize('Inflections')                           # => "Inflections"
 | |
|   #   demodulize('::Inflections')                         # => "Inflections"
 | |
|   #   demodulize('')                                      # => ""
 | |
|   #
 | |
|   # See also #deconstantize.
 | |
|   # @see https://github.com/rails/rails/blob/b0dd7c7/activesupport/lib/active_support/inflector/methods.rb#L230-L245
 | |
|   #   `ActiveSupport::Inflector.demodulize`
 | |
|   sig { params(path: String).returns(String) }
 | |
|   def self.demodulize(path)
 | |
|     if (i = path.rindex("::"))
 | |
|       T.must(path[(i + 2)..])
 | |
|     else
 | |
|       path
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # A lightweight alternative to `ActiveSupport::Inflector.pluralize`:
 | |
|   # Combines `stem` with the `singular` or `plural` suffix based on `count`.
 | |
|   # Adds a prefix of the count value if `include_count` is set to true.
 | |
|   sig {
 | |
|     params(stem: String, count: Integer, plural: String, singular: String, include_count: T::Boolean).returns(String)
 | |
|   }
 | |
|   def self.pluralize(stem, count, plural: "s", singular: "", include_count: false)
 | |
|     prefix = include_count ? "#{count} " : ""
 | |
|     suffix = (count == 1) ? singular : plural
 | |
|     "#{prefix}#{stem}#{suffix}"
 | |
|   end
 | |
| 
 | |
|   sig { params(author: String).returns({ email: String, name: String }) }
 | |
|   def self.parse_author!(author)
 | |
|     match_data = /^(?<name>[^<]+?)[ \t]*<(?<email>[^>]+?)>$/.match(author)
 | |
|     if match_data
 | |
|       name = match_data[:name]
 | |
|       email = match_data[:email]
 | |
|     end
 | |
|     raise UsageError, "Unable to parse name and email." if name.blank? && email.blank?
 | |
| 
 | |
|     { name: T.must(name), email: T.must(email) }
 | |
|   end
 | |
| 
 | |
|   # Makes an underscored, lowercase form from the expression in the string.
 | |
|   #
 | |
|   # Changes '::' to '/' to convert namespaces to paths.
 | |
|   #
 | |
|   #   underscore('ActiveModel')         # => "active_model"
 | |
|   #   underscore('ActiveModel::Errors') # => "active_model/errors"
 | |
|   #
 | |
|   # @see https://github.com/rails/rails/blob/v6.1.7.2/activesupport/lib/active_support/inflector/methods.rb#L81-L100
 | |
|   #   `ActiveSupport::Inflector.underscore`
 | |
|   sig { params(camel_cased_word: T.any(String, Symbol)).returns(String) }
 | |
|   def self.underscore(camel_cased_word)
 | |
|     return camel_cased_word.to_s unless /[A-Z-]|::/.match?(camel_cased_word)
 | |
| 
 | |
|     word = camel_cased_word.to_s.gsub("::", "/")
 | |
|     word.gsub!(/([A-Z])(?=[A-Z][a-z])|([a-z\d])(?=[A-Z])/) do
 | |
|       T.must(::Regexp.last_match(1) || ::Regexp.last_match(2)) << "_"
 | |
|     end
 | |
|     word.tr!("-", "_")
 | |
|     word.downcase!
 | |
|     word
 | |
|   end
 | |
| 
 | |
|   SAFE_FILENAME_REGEX = /[[:cntrl:]#{Regexp.escape("#{File::SEPARATOR}#{File::ALT_SEPARATOR}")}]/o
 | |
|   private_constant :SAFE_FILENAME_REGEX
 | |
| 
 | |
|   sig { params(basename: String).returns(T::Boolean) }
 | |
|   def self.safe_filename?(basename)
 | |
|     !SAFE_FILENAME_REGEX.match?(basename)
 | |
|   end
 | |
| 
 | |
|   sig { params(basename: String).returns(String) }
 | |
|   def self.safe_filename(basename)
 | |
|     basename.gsub(SAFE_FILENAME_REGEX, "")
 | |
|   end
 | |
| end
 |