673 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			673 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: false
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "keg_relocate"
 | |
| require "language/python"
 | |
| require "lock_file"
 | |
| require "ostruct"
 | |
| require "extend/cachable"
 | |
| 
 | |
| # Installation prefix of a formula.
 | |
| #
 | |
| # @api private
 | |
| class Keg
 | |
|   extend T::Sig
 | |
| 
 | |
|   extend Cachable
 | |
| 
 | |
|   # Error for when a keg is already linked.
 | |
|   class AlreadyLinkedError < RuntimeError
 | |
|     def initialize(keg)
 | |
|       super <<~EOS
 | |
|         Cannot link #{keg.name}
 | |
|         Another version is already linked: #{keg.linked_keg_record.resolved_path}
 | |
|       EOS
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # Error for when a keg cannot be linked.
 | |
|   class LinkError < RuntimeError
 | |
|     attr_reader :keg, :src, :dst
 | |
| 
 | |
|     def initialize(keg, src, dst, cause)
 | |
|       @src = src
 | |
|       @dst = dst
 | |
|       @keg = keg
 | |
|       @cause = cause
 | |
|       super(cause.message)
 | |
|       set_backtrace(cause.backtrace)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # Error for when a file already exists or belongs to another keg.
 | |
|   class ConflictError < LinkError
 | |
|     extend T::Sig
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def suggestion
 | |
|       conflict = Keg.for(dst)
 | |
|     rescue NotAKegError, Errno::ENOENT
 | |
|       "already exists. You may want to remove it:\n  rm '#{dst}'\n"
 | |
|     else
 | |
|       <<~EOS
 | |
|         is a symlink belonging to #{conflict.name}. You can unlink it:
 | |
|           brew unlink #{conflict.name}
 | |
|       EOS
 | |
|     end
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def to_s
 | |
|       s = []
 | |
|       s << "Could not symlink #{src}"
 | |
|       s << "Target #{dst}" << suggestion
 | |
|       s << <<~EOS
 | |
|         To force the link and overwrite all conflicting files:
 | |
|           brew link --overwrite #{keg.name}
 | |
| 
 | |
|         To list all files that would be deleted:
 | |
|           brew link --overwrite --dry-run #{keg.name}
 | |
|       EOS
 | |
|       s.join("\n")
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # Error for when a directory is not writable.
 | |
|   class DirectoryNotWritableError < LinkError
 | |
|     extend T::Sig
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def to_s
 | |
|       <<~EOS
 | |
|         Could not symlink #{src}
 | |
|         #{dst.dirname} is not writable.
 | |
|       EOS
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   # Locale-specific directories have the form `language[_territory][.codeset][@modifier]`
 | |
|   LOCALEDIR_RX = %r{(locale|man)/([a-z]{2}|C|POSIX)(_[A-Z]{2})?(\.[a-zA-Z\-0-9]+(@.+)?)?}.freeze
 | |
|   INFOFILE_RX = %r{info/([^.].*?\.info|dir)$}.freeze
 | |
|   KEG_LINK_DIRECTORIES = %w[
 | |
|     bin etc include lib sbin share var
 | |
|   ].freeze
 | |
|   MUST_EXIST_SUBDIRECTORIES = (
 | |
|     KEG_LINK_DIRECTORIES - %w[var] + %w[
 | |
|       opt
 | |
|       var/homebrew/linked
 | |
|     ]
 | |
|   ).map { |dir| HOMEBREW_PREFIX/dir }.sort.uniq.freeze
 | |
| 
 | |
|   # Keep relatively in sync with
 | |
|   # {https://github.com/Homebrew/install/blob/HEAD/install.sh}
 | |
|   MUST_EXIST_DIRECTORIES = (MUST_EXIST_SUBDIRECTORIES + [
 | |
|     HOMEBREW_CELLAR,
 | |
|   ].sort.uniq).freeze
 | |
|   MUST_BE_WRITABLE_DIRECTORIES = (
 | |
|     %w[
 | |
|       etc/bash_completion.d lib/pkgconfig
 | |
|       share/aclocal share/doc share/info share/locale share/man
 | |
|       share/man/man1 share/man/man2 share/man/man3 share/man/man4
 | |
|       share/man/man5 share/man/man6 share/man/man7 share/man/man8
 | |
|       share/zsh share/zsh/site-functions
 | |
|       var/log
 | |
|     ].map { |dir| HOMEBREW_PREFIX/dir } + MUST_EXIST_SUBDIRECTORIES + [
 | |
|       HOMEBREW_CACHE,
 | |
|       HOMEBREW_CELLAR,
 | |
|       HOMEBREW_LOCKS,
 | |
|       HOMEBREW_LOGS,
 | |
|       HOMEBREW_REPOSITORY,
 | |
|       Language::Python.homebrew_site_packages,
 | |
|     ]
 | |
|   ).sort.uniq.freeze
 | |
| 
 | |
|   # These paths relative to the keg's share directory should always be real
 | |
|   # directories in the prefix, never symlinks.
 | |
|   SHARE_PATHS = %w[
 | |
|     aclocal doc info java locale man
 | |
|     man/man1 man/man2 man/man3 man/man4
 | |
|     man/man5 man/man6 man/man7 man/man8
 | |
|     man/cat1 man/cat2 man/cat3 man/cat4
 | |
|     man/cat5 man/cat6 man/cat7 man/cat8
 | |
|     applications gnome gnome/help icons
 | |
|     mime-info pixmaps sounds postgresql
 | |
|   ].freeze
 | |
| 
 | |
|   ELISP_EXTENSIONS = %w[.el .elc].freeze
 | |
|   PYC_EXTENSIONS = %w[.pyc .pyo].freeze
 | |
|   LIBTOOL_EXTENSIONS = %w[.la .lai].freeze
 | |
| 
 | |
|   # @param path if this is a file in a keg, returns the containing {Keg} object.
 | |
|   def self.for(path)
 | |
|     original_path = path
 | |
|     raise Errno::ENOENT, original_path.to_s unless original_path.exist?
 | |
| 
 | |
|     if (path = original_path.realpath)
 | |
|       until path.root?
 | |
|         return Keg.new(path) if path.parent.parent == HOMEBREW_CELLAR.realpath
 | |
| 
 | |
|         path = path.parent.realpath # realpath() prevents root? failing
 | |
|       end
 | |
|     end
 | |
|     raise NotAKegError, "#{original_path} is not inside a keg"
 | |
|   end
 | |
| 
 | |
|   def self.all
 | |
|     Formula.racks.flat_map(&:subdirs).map { |d| new(d) }
 | |
|   end
 | |
| 
 | |
|   attr_reader :path, :name, :linked_keg_record, :opt_record
 | |
| 
 | |
|   protected :path
 | |
| 
 | |
|   extend Forwardable
 | |
| 
 | |
|   def_delegators :path,
 | |
|                  :to_s, :hash, :abv, :disk_usage, :file_count, :directory?, :exist?, :/,
 | |
|                  :join, :rename, :find
 | |
| 
 | |
|   def initialize(path)
 | |
|     path = path.resolved_path if path.to_s.start_with?("#{HOMEBREW_PREFIX}/opt/")
 | |
|     raise "#{path} is not a valid keg" if path.parent.parent.realpath != HOMEBREW_CELLAR.realpath
 | |
|     raise "#{path} is not a directory" unless path.directory?
 | |
| 
 | |
|     @path = path
 | |
|     @name = path.parent.basename.to_s
 | |
|     @linked_keg_record = HOMEBREW_LINKED_KEGS/name
 | |
|     @opt_record = HOMEBREW_PREFIX/"opt/#{name}"
 | |
|     @require_relocation = false
 | |
|   end
 | |
| 
 | |
|   def rack
 | |
|     path.parent
 | |
|   end
 | |
| 
 | |
|   alias to_path to_s
 | |
| 
 | |
|   sig { returns(String) }
 | |
|   def inspect
 | |
|     "#<#{self.class.name}:#{path}>"
 | |
|   end
 | |
| 
 | |
|   def ==(other)
 | |
|     instance_of?(other.class) && path == other.path
 | |
|   end
 | |
|   alias eql? ==
 | |
| 
 | |
|   sig { returns(T::Boolean) }
 | |
|   def empty_installation?
 | |
|     Pathname.glob("#{path}/*") do |file|
 | |
|       return false if file.directory? && !file.children.reject(&:ds_store?).empty?
 | |
| 
 | |
|       basename = file.basename.to_s
 | |
|       next if Metafiles.copy?(basename)
 | |
|       next if %w[.DS_Store INSTALL_RECEIPT.json].include?(basename)
 | |
| 
 | |
|       return false
 | |
|     end
 | |
| 
 | |
|     true
 | |
|   end
 | |
| 
 | |
|   def require_relocation?
 | |
|     @require_relocation
 | |
|   end
 | |
| 
 | |
|   def linked?
 | |
|     linked_keg_record.symlink? &&
 | |
|       linked_keg_record.directory? &&
 | |
|       path == linked_keg_record.resolved_path
 | |
|   end
 | |
| 
 | |
|   def remove_linked_keg_record
 | |
|     linked_keg_record.unlink
 | |
|     linked_keg_record.parent.rmdir_if_possible
 | |
|   end
 | |
| 
 | |
|   def optlinked?
 | |
|     opt_record.symlink? && path == opt_record.resolved_path
 | |
|   end
 | |
| 
 | |
|   def remove_old_aliases
 | |
|     opt = opt_record.parent
 | |
|     linkedkegs = linked_keg_record.parent
 | |
| 
 | |
|     tap = begin
 | |
|       to_formula.tap
 | |
|     rescue
 | |
|       # If the formula can't be found, just ignore aliases for now.
 | |
|       nil
 | |
|     end
 | |
| 
 | |
|     if tap
 | |
|       bad_tap_opt = opt/tap.user
 | |
|       FileUtils.rm_rf bad_tap_opt if !bad_tap_opt.symlink? && bad_tap_opt.directory?
 | |
|     end
 | |
| 
 | |
|     aliases.each do |a|
 | |
|       # versioned aliases are handled below
 | |
|       next if a.match?(/.+@./)
 | |
| 
 | |
|       remove_alias_symlink(opt/a, opt_record)
 | |
|       remove_alias_symlink(linkedkegs/a, linked_keg_record)
 | |
|     end
 | |
| 
 | |
|     Pathname.glob("#{opt_record}@*").each do |a|
 | |
|       a = a.basename.to_s
 | |
|       next if aliases.include?(a)
 | |
| 
 | |
|       remove_alias_symlink(opt/a, rack)
 | |
|       remove_alias_symlink(linkedkegs/a, rack)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def remove_opt_record
 | |
|     opt_record.unlink
 | |
|     opt_record.parent.rmdir_if_possible
 | |
|   end
 | |
| 
 | |
|   def uninstall(raise_failures: false)
 | |
|     CacheStoreDatabase.use(:linkage) do |db|
 | |
|       break unless db.created?
 | |
| 
 | |
|       LinkageCacheStore.new(path, db).delete!
 | |
|     end
 | |
| 
 | |
|     path.rmtree
 | |
|     path.parent.rmdir_if_possible
 | |
|     remove_opt_record if optlinked?
 | |
|     remove_linked_keg_record if linked?
 | |
|     remove_old_aliases
 | |
|     remove_oldname_opt_record
 | |
|   rescue Errno::EACCES, Errno::ENOTEMPTY
 | |
|     raise if raise_failures
 | |
| 
 | |
|     odie <<~EOS
 | |
|       Could not remove #{name} keg! Do so manually:
 | |
|         sudo rm -rf #{path}
 | |
|     EOS
 | |
|   end
 | |
| 
 | |
|   def unlink(verbose: false, dry_run: false)
 | |
|     ObserverPathnameExtension.reset_counts!
 | |
| 
 | |
|     dirs = []
 | |
| 
 | |
|     keg_directories = KEG_LINK_DIRECTORIES.map { |d| path/d }
 | |
|                                           .select(&:exist?)
 | |
|     keg_directories.each do |dir|
 | |
|       dir.find do |src|
 | |
|         dst = HOMEBREW_PREFIX + src.relative_path_from(path)
 | |
|         dst.extend(ObserverPathnameExtension)
 | |
| 
 | |
|         dirs << dst if dst.directory? && !dst.symlink?
 | |
| 
 | |
|         # check whether the file to be unlinked is from the current keg first
 | |
|         next unless dst.symlink?
 | |
|         next if src != dst.resolved_path
 | |
| 
 | |
|         if dry_run
 | |
|           puts dst
 | |
|           Find.prune if src.directory?
 | |
|           next
 | |
|         end
 | |
| 
 | |
|         dst.uninstall_info if dst.to_s.match?(INFOFILE_RX)
 | |
|         dst.unlink
 | |
|         Find.prune if src.directory?
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     unless dry_run
 | |
|       remove_old_aliases
 | |
|       remove_linked_keg_record if linked?
 | |
|       dirs.reverse_each(&:rmdir_if_possible)
 | |
|     end
 | |
| 
 | |
|     ObserverPathnameExtension.n
 | |
|   end
 | |
| 
 | |
|   def lock(&block)
 | |
|     FormulaLock.new(name).with_lock do
 | |
|       if oldname_opt_record
 | |
|         FormulaLock.new(oldname_opt_record.basename.to_s).with_lock(&block)
 | |
|       else
 | |
|         yield
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def completion_installed?(shell)
 | |
|     dir = case shell
 | |
|     when :bash then path/"etc/bash_completion.d"
 | |
|     when :zsh
 | |
|       dir = path/"share/zsh/site-functions"
 | |
|       dir if dir.directory? && dir.children.any? { |f| f.basename.to_s.start_with?("_") }
 | |
|     when :fish then path/"share/fish/vendor_completions.d"
 | |
|     end
 | |
|     dir&.directory? && !dir.children.empty?
 | |
|   end
 | |
| 
 | |
|   def functions_installed?(shell)
 | |
|     case shell
 | |
|     when :fish
 | |
|       dir = path/"share/fish/vendor_functions.d"
 | |
|       dir.directory? && !dir.children.empty?
 | |
|     when :zsh
 | |
|       # Check for non completion functions (i.e. files not started with an underscore),
 | |
|       # since those can be checked separately
 | |
|       dir = path/"share/zsh/site-functions"
 | |
|       dir.directory? && dir.children.any? { |f| !f.basename.to_s.start_with?("_") }
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   sig { returns(T::Boolean) }
 | |
|   def plist_installed?
 | |
|     !Dir["#{path}/*.plist"].empty?
 | |
|   end
 | |
| 
 | |
|   def python_site_packages_installed?
 | |
|     (path/"lib/python2.7/site-packages").directory?
 | |
|   end
 | |
| 
 | |
|   sig { returns(T::Boolean) }
 | |
|   def python_pth_files_installed?
 | |
|     !Dir["#{path}/lib/python2.7/site-packages/*.pth"].empty?
 | |
|   end
 | |
| 
 | |
|   sig { returns(T::Array[Pathname]) }
 | |
|   def apps
 | |
|     app_prefix = optlinked? ? opt_record : path
 | |
|     Pathname.glob("#{app_prefix}/{,libexec/}*.app")
 | |
|   end
 | |
| 
 | |
|   def elisp_installed?
 | |
|     return false unless (path/"share/emacs/site-lisp"/name).exist?
 | |
| 
 | |
|     (path/"share/emacs/site-lisp"/name).children.any? { |f| ELISP_EXTENSIONS.include? f.extname }
 | |
|   end
 | |
| 
 | |
|   def version
 | |
|     require "pkg_version"
 | |
|     PkgVersion.parse(path.basename.to_s)
 | |
|   end
 | |
| 
 | |
|   def to_formula
 | |
|     Formulary.from_keg(self)
 | |
|   end
 | |
| 
 | |
|   def oldname_opt_record
 | |
|     @oldname_opt_record ||= if (opt_dir = HOMEBREW_PREFIX/"opt").directory?
 | |
|       opt_dir.subdirs.find do |dir|
 | |
|         dir.symlink? && dir != opt_record && path.parent == dir.resolved_path.parent
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def link(verbose: false, dry_run: false, overwrite: false)
 | |
|     raise AlreadyLinkedError, self if linked_keg_record.directory?
 | |
| 
 | |
|     ObserverPathnameExtension.reset_counts!
 | |
| 
 | |
|     optlink(verbose: verbose, dry_run: dry_run, overwrite: overwrite) unless dry_run
 | |
| 
 | |
|     # yeah indeed, you have to force anything you need in the main tree into
 | |
|     # these dirs REMEMBER that *NOT* everything needs to be in the main tree
 | |
|     link_dir("etc", verbose: verbose, dry_run: dry_run, overwrite: overwrite) { :mkpath }
 | |
|     link_dir("bin", verbose: verbose, dry_run: dry_run, overwrite: overwrite) { :skip_dir }
 | |
|     link_dir("sbin", verbose: verbose, dry_run: dry_run, overwrite: overwrite) { :skip_dir }
 | |
|     link_dir("include", verbose: verbose, dry_run: dry_run, overwrite: overwrite) { :link }
 | |
| 
 | |
|     link_dir("share", verbose: verbose, dry_run: dry_run, overwrite: overwrite) do |relative_path|
 | |
|       case relative_path.to_s
 | |
|       when INFOFILE_RX then :info
 | |
|       when "locale/locale.alias",
 | |
|            %r{^icons/.*/icon-theme\.cache$}
 | |
|         :skip_file
 | |
|       when LOCALEDIR_RX,
 | |
|            %r{^icons/}, # all icons subfolders should also mkpath
 | |
|            /^zsh/,
 | |
|            /^fish/,
 | |
|            %r{^lua/}, #  Lua, Lua51, Lua53 all need the same handling.
 | |
|            %r{^guile/},
 | |
|            *SHARE_PATHS
 | |
|         :mkpath
 | |
|       else
 | |
|         :link
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     link_dir("lib", verbose: verbose, dry_run: dry_run, overwrite: overwrite) do |relative_path|
 | |
|       case relative_path.to_s
 | |
|       when "charset.alias"
 | |
|         :skip_file
 | |
|       when "pkgconfig", # pkg-config database gets explicitly created
 | |
|            "cmake",     # cmake database gets explicitly created
 | |
|            "dtrace",    # lib/language folders also get explicitly created
 | |
|            /^gdk-pixbuf/,
 | |
|            "ghc",
 | |
|            /^gio/,
 | |
|            /^lua/,
 | |
|            /^mecab/,
 | |
|            /^node/,
 | |
|            /^ocaml/,
 | |
|            /^perl5/,
 | |
|            "php",
 | |
|            /^python[23]\.\d+/,
 | |
|            /^R/,
 | |
|            /^ruby/
 | |
|         :mkpath
 | |
|       else
 | |
|         # Everything else is symlinked to the cellar
 | |
|         :link
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     link_dir("Frameworks", verbose: verbose, dry_run: dry_run, overwrite: overwrite) do |relative_path|
 | |
|       # Frameworks contain symlinks pointing into a subdir, so we have to use
 | |
|       # the :link strategy. However, for Foo.framework and
 | |
|       # Foo.framework/Versions we have to use :mkpath so that multiple formulae
 | |
|       # can link their versions into it and `brew [un]link` works.
 | |
|       if relative_path.to_s.match?(%r{[^/]*\.framework(/Versions)?$})
 | |
|         :mkpath
 | |
|       else
 | |
|         :link
 | |
|       end
 | |
|     end
 | |
|     unless dry_run
 | |
|       make_relative_symlink(linked_keg_record, path, verbose: verbose, dry_run: dry_run, overwrite: overwrite)
 | |
|     end
 | |
|   rescue LinkError
 | |
|     unlink(verbose: verbose)
 | |
|     raise
 | |
|   else
 | |
|     ObserverPathnameExtension.n
 | |
|   end
 | |
| 
 | |
|   def prepare_debug_symbols; end
 | |
| 
 | |
|   def consistent_reproducible_symlink_permissions!; end
 | |
| 
 | |
|   def remove_oldname_opt_record
 | |
|     return unless oldname_opt_record
 | |
|     return if oldname_opt_record.resolved_path != path
 | |
| 
 | |
|     @oldname_opt_record.unlink
 | |
|     @oldname_opt_record.parent.rmdir_if_possible
 | |
|     @oldname_opt_record = nil
 | |
|   end
 | |
| 
 | |
|   def tab
 | |
|     Tab.for_keg(self)
 | |
|   end
 | |
| 
 | |
|   def runtime_dependencies
 | |
|     Keg.cache[:runtime_dependencies] ||= {}
 | |
|     Keg.cache[:runtime_dependencies][path] ||= tab.runtime_dependencies
 | |
|   end
 | |
| 
 | |
|   def aliases
 | |
|     tab.aliases || []
 | |
|   end
 | |
| 
 | |
|   def optlink(verbose: false, dry_run: false, overwrite: false)
 | |
|     opt_record.delete if opt_record.symlink? || opt_record.exist?
 | |
|     make_relative_symlink(opt_record, path, verbose: verbose, dry_run: dry_run, overwrite: overwrite)
 | |
|     aliases.each do |a|
 | |
|       alias_opt_record = opt_record.parent/a
 | |
|       alias_opt_record.delete if alias_opt_record.symlink? || alias_opt_record.exist?
 | |
|       make_relative_symlink(alias_opt_record, path, verbose: verbose, dry_run: dry_run, overwrite: overwrite)
 | |
|     end
 | |
| 
 | |
|     return unless oldname_opt_record
 | |
| 
 | |
|     oldname_opt_record.delete
 | |
|     make_relative_symlink(oldname_opt_record, path, verbose: verbose, dry_run: dry_run, overwrite: overwrite)
 | |
|   end
 | |
| 
 | |
|   def delete_pyc_files!
 | |
|     find { |pn| pn.delete if PYC_EXTENSIONS.include?(pn.extname) }
 | |
|     find { |pn| FileUtils.rm_rf pn if pn.basename.to_s == "__pycache__" }
 | |
|   end
 | |
| 
 | |
|   def binary_executable_or_library_files
 | |
|     []
 | |
|   end
 | |
| 
 | |
|   def codesign_patched_binary(file); end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def resolve_any_conflicts(dst, dry_run: false, verbose: false, overwrite: false)
 | |
|     return unless dst.symlink?
 | |
| 
 | |
|     src = dst.resolved_path
 | |
| 
 | |
|     # src itself may be a symlink, so check lstat to ensure we are dealing with
 | |
|     # a directory, and not a symlink pointing at a directory (which needs to be
 | |
|     # treated as a file). In other words, we only want to resolve one symlink.
 | |
| 
 | |
|     begin
 | |
|       stat = src.lstat
 | |
|     rescue Errno::ENOENT
 | |
|       # dst is a broken symlink, so remove it.
 | |
|       dst.unlink unless dry_run
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     return unless stat.directory?
 | |
| 
 | |
|     begin
 | |
|       keg = Keg.for(src)
 | |
|     rescue NotAKegError
 | |
|       puts "Won't resolve conflicts for symlink #{dst} as it doesn't resolve into the Cellar." if verbose
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     dst.unlink unless dry_run
 | |
|     keg.link_dir(src, dry_run: false, verbose: false, overwrite: false) { :mkpath }
 | |
|     true
 | |
|   end
 | |
| 
 | |
|   def make_relative_symlink(dst, src, verbose: false, dry_run: false, overwrite: false)
 | |
|     if dst.symlink? && src == dst.resolved_path
 | |
|       puts "Skipping; link already exists: #{dst}" if verbose
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     # cf. git-clean -n: list files to delete, don't really link or delete
 | |
|     if dry_run && overwrite
 | |
|       if dst.symlink?
 | |
|         puts "#{dst} -> #{dst.resolved_path}"
 | |
|       elsif dst.exist?
 | |
|         puts dst
 | |
|       end
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     # list all link targets
 | |
|     if dry_run
 | |
|       puts dst
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     dst.delete if overwrite && (dst.exist? || dst.symlink?)
 | |
|     dst.make_relative_symlink(src)
 | |
|   rescue Errno::EEXIST => e
 | |
|     raise ConflictError.new(self, src.relative_path_from(path), dst, e) if dst.exist?
 | |
| 
 | |
|     if dst.symlink?
 | |
|       dst.unlink
 | |
|       retry
 | |
|     end
 | |
|   rescue Errno::EACCES => e
 | |
|     raise DirectoryNotWritableError.new(self, src.relative_path_from(path), dst, e)
 | |
|   rescue SystemCallError => e
 | |
|     raise LinkError.new(self, src.relative_path_from(path), dst, e)
 | |
|   end
 | |
| 
 | |
|   def remove_alias_symlink(alias_symlink, alias_match_path)
 | |
|     if alias_symlink.symlink? && alias_symlink.exist?
 | |
|       alias_symlink.delete if alias_match_path.exist? && alias_symlink.realpath == alias_match_path.realpath
 | |
|     elsif alias_symlink.symlink? || alias_symlink.exist?
 | |
|       alias_symlink.delete
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   protected
 | |
| 
 | |
|   # symlinks the contents of path+relative_dir recursively into #{HOMEBREW_PREFIX}/relative_dir
 | |
|   def link_dir(relative_dir, verbose: false, dry_run: false, overwrite: false)
 | |
|     root = path/relative_dir
 | |
|     return unless root.exist?
 | |
| 
 | |
|     root.find do |src|
 | |
|       next if src == root
 | |
| 
 | |
|       dst = HOMEBREW_PREFIX + src.relative_path_from(path)
 | |
|       dst.extend ObserverPathnameExtension
 | |
| 
 | |
|       if src.symlink? || src.file?
 | |
|         Find.prune if File.basename(src) == ".DS_Store"
 | |
|         Find.prune if src.resolved_path == dst
 | |
|         # Don't link pyc or pyo files because Python overwrites these
 | |
|         # cached object files and next time brew wants to link, the
 | |
|         # file is in the way.
 | |
|         Find.prune if PYC_EXTENSIONS.include?(src.extname) && src.to_s.include?("/site-packages/")
 | |
| 
 | |
|         case yield src.relative_path_from(root)
 | |
|         when :skip_file, nil
 | |
|           Find.prune
 | |
|         when :info
 | |
|           next if File.basename(src) == "dir" # skip historical local 'dir' files
 | |
| 
 | |
|           make_relative_symlink dst, src, verbose: verbose, dry_run: dry_run, overwrite: overwrite
 | |
|           dst.install_info
 | |
|         else
 | |
|           make_relative_symlink dst, src, verbose: verbose, dry_run: dry_run, overwrite: overwrite
 | |
|         end
 | |
|       elsif src.directory?
 | |
|         # if the dst dir already exists, then great! walk the rest of the tree tho
 | |
|         next if dst.directory? && !dst.symlink?
 | |
| 
 | |
|         # no need to put .app bundles in the path, the user can just use
 | |
|         # spotlight, or the open command and actual mac apps use an equivalent
 | |
|         Find.prune if src.extname == ".app"
 | |
| 
 | |
|         case yield src.relative_path_from(root)
 | |
|         when :skip_dir
 | |
|           Find.prune
 | |
|         when :mkpath
 | |
|           dst.mkpath unless resolve_any_conflicts(dst, verbose: verbose, dry_run: dry_run, overwrite: overwrite)
 | |
|         else
 | |
|           unless resolve_any_conflicts(dst, verbose: verbose, dry_run: dry_run, overwrite: overwrite)
 | |
|             make_relative_symlink dst, src, verbose: verbose, dry_run: dry_run, overwrite: overwrite
 | |
|             Find.prune
 | |
|           end
 | |
|         end
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 | |
| 
 | |
| require "extend/os/keg"
 | 
