 a1f112f3fe
			
		
	
	
		a1f112f3fe
		
			
		
	
	
	
	
		
			
			This reduces the surface area of our `Kernel` monkeypatch and removes the need to `include Kernel` in a bunch of modules. While we're here, also move `Kernel#require?` to `Homebrew` and fully scope the calls to it.
		
			
				
	
	
		
			157 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			157 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: strict
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "development_tools"
 | |
| require "messages"
 | |
| require "utils/output"
 | |
| 
 | |
| # Needed to handle circular require dependency.
 | |
| # rubocop:disable Lint/EmptyClass
 | |
| class FormulaInstaller; end
 | |
| # rubocop:enable Lint/EmptyClass
 | |
| 
 | |
| module Homebrew
 | |
|   module Reinstall
 | |
|     extend Utils::Output::Mixin
 | |
| 
 | |
|     class InstallationContext < T::Struct
 | |
|       const :formula_installer, ::FormulaInstaller
 | |
|       const :keg, T.nilable(Keg)
 | |
|       const :formula, Formula
 | |
|       const :options, Options
 | |
|     end
 | |
| 
 | |
|     class << self
 | |
|       sig {
 | |
|         params(
 | |
|           formula: Formula, flags: T::Array[String], force_bottle: T::Boolean,
 | |
|           build_from_source_formulae: T::Array[String], interactive: T::Boolean, keep_tmp: T::Boolean,
 | |
|           debug_symbols: T::Boolean, force: T::Boolean, debug: T::Boolean, quiet: T::Boolean,
 | |
|           verbose: T::Boolean, git: T::Boolean
 | |
|         ).returns(InstallationContext)
 | |
|       }
 | |
|       def build_install_context(
 | |
|         formula,
 | |
|         flags:,
 | |
|         force_bottle: false,
 | |
|         build_from_source_formulae: [],
 | |
|         interactive: false,
 | |
|         keep_tmp: false,
 | |
|         debug_symbols: false,
 | |
|         force: false,
 | |
|         debug: false,
 | |
|         quiet: false,
 | |
|         verbose: false,
 | |
|         git: false
 | |
|       )
 | |
|         if formula.opt_prefix.directory?
 | |
|           keg = Keg.new(formula.opt_prefix.resolved_path)
 | |
|           tab = keg.tab
 | |
|           link_keg = keg.linked?
 | |
|           installed_as_dependency = tab.installed_as_dependency == true
 | |
|           installed_on_request = tab.installed_on_request == true
 | |
|           build_bottle = tab.built_bottle?
 | |
|           backup keg
 | |
|         else
 | |
|           link_keg = nil
 | |
|           installed_as_dependency = false
 | |
|           installed_on_request = true
 | |
|           build_bottle = false
 | |
|         end
 | |
| 
 | |
|         build_options = BuildOptions.new(Options.create(flags), formula.options)
 | |
|         options = build_options.used_options
 | |
|         options |= formula.build.used_options
 | |
|         options &= formula.options
 | |
| 
 | |
|         formula_installer = FormulaInstaller.new(
 | |
|           formula,
 | |
|           **{
 | |
|             options:,
 | |
|             link_keg:,
 | |
|             installed_as_dependency:,
 | |
|             installed_on_request:,
 | |
|             build_bottle:,
 | |
|             force_bottle:,
 | |
|             build_from_source_formulae:,
 | |
|             git:,
 | |
|             interactive:,
 | |
|             keep_tmp:,
 | |
|             debug_symbols:,
 | |
|             force:,
 | |
|             debug:,
 | |
|             quiet:,
 | |
|             verbose:,
 | |
|           }.compact,
 | |
|         )
 | |
|         InstallationContext.new(formula_installer:, keg:, formula:, options:)
 | |
|       end
 | |
| 
 | |
|       sig { params(install_context: InstallationContext).void }
 | |
|       def reinstall_formula(install_context)
 | |
|         formula_installer = install_context.formula_installer
 | |
|         keg = install_context.keg
 | |
|         formula = install_context.formula
 | |
|         options = install_context.options
 | |
|         link_keg = keg&.linked?
 | |
|         verbose = formula_installer.verbose?
 | |
| 
 | |
|         oh1 "Reinstalling #{Formatter.identifier(formula.full_name)} #{options.to_a.join " "}"
 | |
| 
 | |
|         formula_installer.install
 | |
|         formula_installer.finish
 | |
|       rescue FormulaInstallationAlreadyAttemptedError
 | |
|         nil
 | |
|         # Any other exceptions we want to restore the previous keg and report the error.
 | |
|       rescue Exception # rubocop:disable Lint/RescueException
 | |
|         ignore_interrupts { restore_backup(keg, link_keg, verbose:) if keg }
 | |
|         raise
 | |
|       else
 | |
|         if keg
 | |
|           backup_keg = backup_path(keg)
 | |
|           begin
 | |
|             FileUtils.rm_r(backup_keg) if backup_keg.exist?
 | |
|           rescue Errno::EACCES, Errno::ENOTEMPTY
 | |
|             odie <<~EOS
 | |
|               Could not remove #{backup_keg.parent.basename} backup keg! Do so manually:
 | |
|                 sudo rm -rf #{backup_keg}
 | |
|             EOS
 | |
|           end
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       private
 | |
| 
 | |
|       sig { params(keg: Keg).void }
 | |
|       def backup(keg)
 | |
|         keg.unlink
 | |
|         begin
 | |
|           keg.rename backup_path(keg)
 | |
|         rescue Errno::EACCES, Errno::ENOTEMPTY
 | |
|           odie <<~EOS
 | |
|             Could not rename #{keg.name} keg! Check/fix its permissions:
 | |
|               sudo chown -R #{ENV.fetch("USER", "$(whoami)")} #{keg}
 | |
|           EOS
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       sig { params(keg: Keg, keg_was_linked: T::Boolean, verbose: T::Boolean).void }
 | |
|       def restore_backup(keg, keg_was_linked, verbose:)
 | |
|         path = backup_path(keg)
 | |
| 
 | |
|         return unless path.directory?
 | |
| 
 | |
|         FileUtils.rm_r(Pathname.new(keg)) if keg.exist?
 | |
| 
 | |
|         path.rename keg.to_s
 | |
|         keg.link(verbose:) if keg_was_linked
 | |
|       end
 | |
| 
 | |
|       sig { params(keg: Keg).returns(Pathname) }
 | |
|       def backup_path(keg)
 | |
|         Pathname.new "#{keg}.reinstall"
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 |