| 
									
										
										
										
											2016-01-04 16:22:26 -04:00
										 |  |  | require "extend/string" | 
					
						
							| 
									
										
										
										
											2016-03-07 18:04:25 +08:00
										 |  |  | require "tap_migrations" | 
					
						
							|  |  |  | require "formula_renames" | 
					
						
							| 
									
										
										
										
											2016-01-04 16:22:26 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  | # a {Tap} is used to extend the formulae provided by Homebrew core. | 
					
						
							|  |  |  | # Usually, it's synced with a remote git repository. And it's likely | 
					
						
							|  |  |  | # a Github repository with the name of `user/homebrew-repo`. In such | 
					
						
							|  |  |  | # case, `user/repo` will be used as the {#name} of this {Tap}, where | 
					
						
							|  |  |  | # {#user} represents Github username and {#repo} represents repository | 
					
						
							|  |  |  | # name without leading `homebrew-`. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  | class Tap | 
					
						
							|  |  |  |   TAP_DIRECTORY = HOMEBREW_LIBRARY/"Taps" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-27 16:52:14 +08:00
										 |  |  |   CACHE = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def self.clear_cache | 
					
						
							|  |  |  |     CACHE.clear | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:36:26 +08:00
										 |  |  |   def self.fetch(*args) | 
					
						
							|  |  |  |     case args.length | 
					
						
							|  |  |  |     when 1
 | 
					
						
							|  |  |  |       user, repo = args.first.split("/", 2) | 
					
						
							|  |  |  |     when 2
 | 
					
						
							|  |  |  |       user = args[0] | 
					
						
							|  |  |  |       repo = args[1] | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     raise "Invalid tap name" unless user && repo | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-02 14:35:42 +08:00
										 |  |  |     # we special case homebrew so users don't have to shift in a terminal | 
					
						
							|  |  |  |     user = "Homebrew" if user == "homebrew" | 
					
						
							|  |  |  |     repo = repo.strip_prefix "homebrew-" | 
					
						
							| 
									
										
										
										
											2015-12-06 21:36:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if user == "Homebrew" && repo == "homebrew" | 
					
						
							| 
									
										
										
										
											2016-03-07 18:04:25 +08:00
										 |  |  |       return CoreTap.instance | 
					
						
							| 
									
										
										
										
											2015-12-06 21:36:26 +08:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-27 16:52:14 +08:00
										 |  |  |     cache_key = "#{user}/#{repo}".downcase | 
					
						
							|  |  |  |     CACHE.fetch(cache_key) { |key| CACHE[key] = Tap.new(user, repo) } | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   extend Enumerable | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # The user name of this {Tap}. Usually, it's the Github username of | 
					
						
							|  |  |  |   # this #{Tap}'s remote repository. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   attr_reader :user | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   # The repository name of this {Tap} without leading `homebrew-`. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   attr_reader :repo | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   # The name of this {Tap}. It combines {#user} and {#repo} with a slash. | 
					
						
							|  |  |  |   # {#name} is always in lowercase. | 
					
						
							|  |  |  |   # e.g. `user/repo` | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   attr_reader :name | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   # The local path to this {Tap}. | 
					
						
							|  |  |  |   # e.g. `/usr/local/Library/Taps/user/homebrew-repo` | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   attr_reader :path | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-02 14:35:42 +08:00
										 |  |  |   # @private | 
					
						
							| 
									
										
										
										
											2015-06-13 01:57:00 +08:00
										 |  |  |   def initialize(user, repo) | 
					
						
							| 
									
										
										
										
											2015-12-02 14:35:42 +08:00
										 |  |  |     @user = user | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |     @repo = repo | 
					
						
							|  |  |  |     @name = "#{@user}/#{@repo}".downcase | 
					
						
							|  |  |  |     @path = TAP_DIRECTORY/"#{@user}/homebrew-#{@repo}".downcase | 
					
						
							| 
									
										
										
										
											2015-06-13 01:57:00 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-06 03:50:06 +08:00
										 |  |  |   # clear internal cache | 
					
						
							|  |  |  |   def clear_cache | 
					
						
							|  |  |  |     @remote = nil | 
					
						
							|  |  |  |     @formula_dir = nil | 
					
						
							|  |  |  |     @formula_files = nil | 
					
						
							|  |  |  |     @alias_files = nil | 
					
						
							|  |  |  |     @aliases = nil | 
					
						
							|  |  |  |     @alias_table = nil | 
					
						
							|  |  |  |     @alias_reverse_table = nil | 
					
						
							|  |  |  |     @command_files = nil | 
					
						
							|  |  |  |     @formula_renames = nil | 
					
						
							| 
									
										
										
										
											2016-02-25 14:14:33 +08:00
										 |  |  |     @tap_migrations = nil | 
					
						
							| 
									
										
										
										
											2016-02-06 03:50:06 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # The remote path to this {Tap}. | 
					
						
							|  |  |  |   # e.g. `https://github.com/user/homebrew-repo` | 
					
						
							| 
									
										
										
										
											2015-06-13 01:57:00 +08:00
										 |  |  |   def remote | 
					
						
							|  |  |  |     @remote ||= if installed? | 
					
						
							| 
									
										
										
										
											2016-03-05 20:03:43 +08:00
										 |  |  |       if git? && Utils.git_available? | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |         path.cd do | 
					
						
							| 
									
										
										
										
											2015-06-13 01:57:00 +08:00
										 |  |  |           Utils.popen_read("git", "config", "--get", "remote.origin.url").chomp | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |       end | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2015-06-13 01:57:00 +08:00
										 |  |  |       raise TapUnavailableError, name | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-30 16:51:00 +08:00
										 |  |  |   # True if this {Tap} is a git repository. | 
					
						
							|  |  |  |   def git? | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     (path/".git").exist? | 
					
						
							| 
									
										
										
										
											2015-07-30 16:51:00 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-05 20:03:43 +08:00
										 |  |  |   # git HEAD for this {Tap}. | 
					
						
							|  |  |  |   def git_head | 
					
						
							|  |  |  |     raise TapUnavailableError, name unless installed? | 
					
						
							|  |  |  |     return unless git? && Utils.git_available? | 
					
						
							|  |  |  |     path.cd { Utils.popen_read("git", "rev-parse", "--verify", "-q", "HEAD").chuzzle } | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # git HEAD in short format for this {Tap}. | 
					
						
							|  |  |  |   def git_short_head | 
					
						
							|  |  |  |     raise TapUnavailableError, name unless installed? | 
					
						
							|  |  |  |     return unless git? && Utils.git_available? | 
					
						
							|  |  |  |     path.cd { Utils.popen_read("git", "rev-parse", "--short=4", "--verify", "-q", "HEAD").chuzzle } | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # time since git last commit for this {Tap}. | 
					
						
							|  |  |  |   def git_last_commit | 
					
						
							|  |  |  |     raise TapUnavailableError, name unless installed? | 
					
						
							|  |  |  |     return unless git? && Utils.git_available? | 
					
						
							|  |  |  |     path.cd { Utils.popen_read("git", "show", "-s", "--format=%cr", "HEAD").chuzzle } | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # git last commit date for this {Tap}. | 
					
						
							|  |  |  |   def git_last_commit_date | 
					
						
							|  |  |  |     raise TapUnavailableError, name unless installed? | 
					
						
							|  |  |  |     return unless git? && Utils.git_available? | 
					
						
							|  |  |  |     path.cd { Utils.popen_read("git", "show", "-s", "--format=%cd", "--date=short", "HEAD").chuzzle } | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-28 13:23:22 +01:00
										 |  |  |   # The issues URL of this {Tap}. | 
					
						
							|  |  |  |   # e.g. `https://github.com/user/homebrew-repo/issues` | 
					
						
							|  |  |  |   def issues_url | 
					
						
							|  |  |  |     if official? || !custom_remote? | 
					
						
							|  |  |  |       "https://github.com/#{user}/homebrew-#{repo}/issues" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def to_s | 
					
						
							|  |  |  |     name | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # True if this {Tap} is an official Homebrew tap. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def official? | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     user == "Homebrew" | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # True if the remote of this {Tap} is a private repository. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def private? | 
					
						
							|  |  |  |     return true if custom_remote? | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     GitHub.private_repo?(user, "homebrew-#{repo}") | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   rescue GitHub::HTTPNotFoundError | 
					
						
							|  |  |  |     true | 
					
						
							|  |  |  |   rescue GitHub::Error | 
					
						
							|  |  |  |     false | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # True if this {Tap} has been installed. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def installed? | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     path.directory? | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-07 14:12:57 +08:00
										 |  |  |   # @private | 
					
						
							| 
									
										
										
										
											2016-03-07 18:04:25 +08:00
										 |  |  |   def core_tap? | 
					
						
							| 
									
										
										
										
											2015-12-07 14:12:57 +08:00
										 |  |  |     false | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |   # install this {Tap}. | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   # @param [Hash] options | 
					
						
							|  |  |  |   # @option options [String]  :clone_targe If passed, it will be used as the clone remote. | 
					
						
							|  |  |  |   # @option options [Boolean] :full_clone If set as true, full clone will be used. | 
					
						
							| 
									
										
										
										
											2016-02-25 21:09:50 +08:00
										 |  |  |   # @option options [Boolean] :quiet If set, suppress all output. | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |   def install(options = {}) | 
					
						
							| 
									
										
										
										
											2015-12-19 19:10:22 +08:00
										 |  |  |     require "descriptions" | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |     raise TapAlreadyTappedError, name if installed? | 
					
						
							| 
									
										
										
										
											2016-02-06 03:50:06 +08:00
										 |  |  |     clear_cache | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-25 21:09:50 +08:00
										 |  |  |     quiet = options.fetch(:quiet, false) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |     # ensure git is installed | 
					
						
							|  |  |  |     Utils.ensure_git_installed! | 
					
						
							| 
									
										
										
										
											2016-02-25 21:09:50 +08:00
										 |  |  |     ohai "Tapping #{name}" unless quiet | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     remote = options[:clone_target] || "https://github.com/#{user}/homebrew-#{repo}" | 
					
						
							|  |  |  |     args = %W[clone #{remote} #{path}] | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |     args << "--depth=1" unless options.fetch(:full_clone, false) | 
					
						
							| 
									
										
										
										
											2016-02-25 21:09:50 +08:00
										 |  |  |     args << "-q" if quiet | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     begin | 
					
						
							|  |  |  |       safe_system "git", *args | 
					
						
							|  |  |  |     rescue Interrupt, ErrorDuringExecution | 
					
						
							|  |  |  |       ignore_interrupts do | 
					
						
							|  |  |  |         sleep 0.1 # wait for git to cleanup the top directory when interrupt happens. | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |         path.parent.rmdir_if_possible | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |       end | 
					
						
							|  |  |  |       raise | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-08 17:26:53 +00:00
										 |  |  |     link_manpages | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |     formula_count = formula_files.size | 
					
						
							| 
									
										
										
										
											2016-02-25 21:09:50 +08:00
										 |  |  |     puts "Tapped #{formula_count} formula#{plural(formula_count, "e")} (#{path.abv})" unless quiet | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |     Descriptions.cache_formulae(formula_names) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-25 21:09:50 +08:00
										 |  |  |     if !options[:clone_target] && private? && !quiet | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |       puts <<-EOS.undent
 | 
					
						
							|  |  |  |         It looks like you tapped a private repository. To avoid entering your | 
					
						
							|  |  |  |         credentials each time you update, you can use git HTTP credential | 
					
						
							|  |  |  |         caching or issue the following command: | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |           cd #{path} | 
					
						
							|  |  |  |           git remote set-url origin git@github.com:#{user}/homebrew-#{repo}.git | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-08 17:26:53 +00:00
										 |  |  |   def link_manpages | 
					
						
							|  |  |  |     return unless (path/"man").exist? | 
					
						
							|  |  |  |     conflicts = [] | 
					
						
							|  |  |  |     (path/"man").find do |src| | 
					
						
							|  |  |  |       next if src.directory? | 
					
						
							|  |  |  |       dst = HOMEBREW_PREFIX/"share"/src.relative_path_from(path) | 
					
						
							|  |  |  |       next if dst.symlink? && src == dst.resolved_path | 
					
						
							|  |  |  |       if dst.exist? | 
					
						
							|  |  |  |         conflicts << dst | 
					
						
							|  |  |  |         next | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       dst.make_relative_symlink(src) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     unless conflicts.empty? | 
					
						
							|  |  |  |       onoe <<-EOS.undent
 | 
					
						
							|  |  |  |         Could not link #{name} manpages to: | 
					
						
							|  |  |  |           #{conflicts.join("\n")} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Please delete these files and run `brew tap --repair`. | 
					
						
							|  |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |   # uninstall this {Tap}. | 
					
						
							|  |  |  |   def uninstall | 
					
						
							| 
									
										
										
										
											2015-12-19 19:10:22 +08:00
										 |  |  |     require "descriptions" | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |     raise TapUnavailableError, name unless installed? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     puts "Untapping #{name}... (#{path.abv})" | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |     unpin if pinned? | 
					
						
							|  |  |  |     formula_count = formula_files.size | 
					
						
							|  |  |  |     Descriptions.uncache_formulae(formula_names) | 
					
						
							| 
									
										
										
										
											2015-12-08 17:26:53 +00:00
										 |  |  |     unlink_manpages | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     path.rmtree | 
					
						
							| 
									
										
										
										
											2015-12-08 17:26:53 +00:00
										 |  |  |     path.parent.rmdir_if_possible | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |     puts "Untapped #{formula_count} formula#{plural(formula_count, "e")}" | 
					
						
							| 
									
										
										
										
											2016-02-06 03:50:06 +08:00
										 |  |  |     clear_cache | 
					
						
							| 
									
										
										
										
											2015-11-07 16:25:34 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-08 17:26:53 +00:00
										 |  |  |   def unlink_manpages | 
					
						
							|  |  |  |     return unless (path/"man").exist? | 
					
						
							|  |  |  |     (path/"man").find do |src| | 
					
						
							|  |  |  |       next if src.directory? | 
					
						
							| 
									
										
										
										
											2015-12-19 20:51:00 +08:00
										 |  |  |       dst = HOMEBREW_PREFIX/"share"/src.relative_path_from(path) | 
					
						
							| 
									
										
										
										
											2015-12-08 17:26:53 +00:00
										 |  |  |       dst.delete if dst.symlink? && src == dst.resolved_path | 
					
						
							|  |  |  |       dst.parent.rmdir_if_possible | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # True if the {#remote} of {Tap} is customized. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def custom_remote? | 
					
						
							| 
									
										
										
										
											2015-06-13 01:57:00 +08:00
										 |  |  |     return true unless remote | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     remote.casecmp("https://github.com/#{user}/homebrew-#{repo}") != 0
 | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 20:59:31 +08:00
										 |  |  |   # path to the directory of all {Formula} files for this {Tap}. | 
					
						
							|  |  |  |   def formula_dir | 
					
						
							|  |  |  |     @formula_dir ||= [path/"Formula", path/"HomebrewFormula", path].detect(&:directory?) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # an array of all {Formula} files of this {Tap}. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def formula_files | 
					
						
							| 
									
										
										
										
											2015-12-06 20:59:31 +08:00
										 |  |  |     @formula_files ||= if formula_dir | 
					
						
							|  |  |  |       formula_dir.children.select { |p| p.extname == ".rb" } | 
					
						
							| 
									
										
										
										
											2015-09-27 16:52:14 +08:00
										 |  |  |     else | 
					
						
							|  |  |  |       [] | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-15 22:31:47 +08:00
										 |  |  |   # return true if given path would present a {Formula} file in this {Tap}. | 
					
						
							|  |  |  |   # accepts both absolute path and relative path (relative to this {Tap}'s path) | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def formula_file?(file) | 
					
						
							|  |  |  |     file = Pathname.new(file) unless file.is_a? Pathname | 
					
						
							|  |  |  |     file = file.expand_path(path) | 
					
						
							|  |  |  |     file.extname == ".rb" && file.parent == formula_dir | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # an array of all {Formula} names of this {Tap}. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def formula_names | 
					
						
							| 
									
										
										
										
											2015-12-06 21:30:23 +08:00
										 |  |  |     @formula_names ||= formula_files.map { |f| formula_file_to_name(f) } | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:15:43 +08:00
										 |  |  |   # path to the directory of all alias files for this {Tap}. | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def alias_dir | 
					
						
							|  |  |  |     path/"Aliases" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-12 18:21:22 +08:00
										 |  |  |   # an array of all alias files of this {Tap}. | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def alias_files | 
					
						
							| 
									
										
										
										
											2015-12-06 21:15:43 +08:00
										 |  |  |     @alias_files ||= Pathname.glob("#{alias_dir}/*").select(&:file?) | 
					
						
							| 
									
										
										
										
											2015-09-12 18:21:22 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # an array of all aliases of this {Tap}. | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def aliases | 
					
						
							| 
									
										
										
										
											2015-12-06 21:30:23 +08:00
										 |  |  |     @aliases ||= alias_files.map { |f| alias_file_to_name(f) } | 
					
						
							| 
									
										
										
										
											2015-09-12 18:21:22 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-07 17:54:20 +08:00
										 |  |  |   # a table mapping alias to formula name | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def alias_table | 
					
						
							|  |  |  |     return @alias_table if @alias_table | 
					
						
							|  |  |  |     @alias_table = Hash.new | 
					
						
							|  |  |  |     alias_files.each do |alias_file| | 
					
						
							| 
									
										
										
										
											2015-12-06 21:30:23 +08:00
										 |  |  |       @alias_table[alias_file_to_name(alias_file)] = formula_file_to_name(alias_file.resolved_path) | 
					
						
							| 
									
										
										
										
											2015-10-07 17:54:20 +08:00
										 |  |  |     end | 
					
						
							|  |  |  |     @alias_table | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # a table mapping formula name to aliases | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def alias_reverse_table | 
					
						
							|  |  |  |     return @alias_reverse_table if @alias_reverse_table | 
					
						
							|  |  |  |     @alias_reverse_table = Hash.new | 
					
						
							|  |  |  |     alias_table.each do |alias_name, formula_name| | 
					
						
							|  |  |  |       @alias_reverse_table[formula_name] ||= [] | 
					
						
							|  |  |  |       @alias_reverse_table[formula_name] << alias_name | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     @alias_reverse_table | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # an array of all commands files of this {Tap}. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def command_files | 
					
						
							| 
									
										
										
										
											2015-09-27 16:52:14 +08:00
										 |  |  |     @command_files ||= Pathname.glob("#{path}/cmd/brew-*").select(&:executable?) | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 16:00:45 +08:00
										 |  |  |   # path to the pin record for this {Tap}. | 
					
						
							|  |  |  |   # @private | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |   def pinned_symlink_path | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     HOMEBREW_LIBRARY/"PinnedTaps/#{name}" | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 16:00:45 +08:00
										 |  |  |   # True if this {Tap} has been pinned. | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |   def pinned? | 
					
						
							| 
									
										
										
										
											2015-09-27 16:52:14 +08:00
										 |  |  |     return @pinned if instance_variable_defined?(:@pinned) | 
					
						
							|  |  |  |     @pinned = pinned_symlink_path.directory? | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 16:00:45 +08:00
										 |  |  |   # pin this {Tap}. | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |   def pin | 
					
						
							|  |  |  |     raise TapUnavailableError, name unless installed? | 
					
						
							|  |  |  |     raise TapPinStatusError.new(name, true) if pinned? | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |     pinned_symlink_path.make_relative_symlink(path) | 
					
						
							| 
									
										
										
										
											2015-09-27 16:52:14 +08:00
										 |  |  |     @pinned = true | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 16:00:45 +08:00
										 |  |  |   # unpin this {Tap}. | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |   def unpin | 
					
						
							|  |  |  |     raise TapUnavailableError, name unless installed? | 
					
						
							|  |  |  |     raise TapPinStatusError.new(name, false) unless pinned? | 
					
						
							|  |  |  |     pinned_symlink_path.delete | 
					
						
							| 
									
										
										
										
											2015-12-08 17:26:53 +00:00
										 |  |  |     pinned_symlink_path.parent.rmdir_if_possible | 
					
						
							| 
									
										
										
										
											2015-12-19 20:32:16 +08:00
										 |  |  |     pinned_symlink_path.parent.parent.rmdir_if_possible | 
					
						
							| 
									
										
										
										
											2015-09-27 16:52:14 +08:00
										 |  |  |     @pinned = false | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def to_hash | 
					
						
							| 
									
										
										
										
											2015-08-13 14:56:14 +08:00
										 |  |  |     hash = { | 
					
						
							| 
									
										
										
										
											2015-12-06 20:57:28 +08:00
										 |  |  |       "name" => name, | 
					
						
							|  |  |  |       "user" => user, | 
					
						
							|  |  |  |       "repo" => repo, | 
					
						
							|  |  |  |       "path" => path.to_s, | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |       "installed" => installed?, | 
					
						
							|  |  |  |       "official" => official?, | 
					
						
							|  |  |  |       "formula_names" => formula_names, | 
					
						
							|  |  |  |       "formula_files" => formula_files.map(&:to_s), | 
					
						
							| 
									
										
										
										
											2015-08-09 22:42:46 +08:00
										 |  |  |       "command_files" => command_files.map(&:to_s), | 
					
						
							|  |  |  |       "pinned" => pinned? | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-08-13 14:56:14 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if installed? | 
					
						
							|  |  |  |       hash["remote"] = remote | 
					
						
							|  |  |  |       hash["custom_remote"] = custom_remote? | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hash | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-09 14:39:46 +03:00
										 |  |  |   # Hash with tap formula renames | 
					
						
							|  |  |  |   def formula_renames | 
					
						
							| 
									
										
										
										
											2015-12-19 19:10:22 +08:00
										 |  |  |     require "utils/json" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-09 14:39:46 +03:00
										 |  |  |     @formula_renames ||= if (rename_file = path/"formula_renames.json").file? | 
					
						
							|  |  |  |       Utils::JSON.load(rename_file.read) | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       {} | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-25 14:14:33 +08:00
										 |  |  |   # Hash with tap migrations | 
					
						
							|  |  |  |   def tap_migrations | 
					
						
							|  |  |  |     require "utils/json" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @tap_migrations ||= if (migration_file = path/"tap_migrations.json").file? | 
					
						
							|  |  |  |       Utils::JSON.load(migration_file.read) | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       {} | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 22:21:27 +08:00
										 |  |  |   def ==(other) | 
					
						
							|  |  |  |     other = Tap.fetch(other) if other.is_a?(String) | 
					
						
							|  |  |  |     self.class == other.class && self.name == other.name | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def self.each | 
					
						
							|  |  |  |     return unless TAP_DIRECTORY.directory? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     TAP_DIRECTORY.subdirs.each do |user| | 
					
						
							|  |  |  |       user.subdirs.each do |repo| | 
					
						
							| 
									
										
										
										
											2015-12-02 14:35:42 +08:00
										 |  |  |         yield fetch(user.basename.to_s, repo.basename.to_s) | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 01:53:55 +08:00
										 |  |  |   # an array of all installed {Tap} names. | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  |   def self.names | 
					
						
							|  |  |  |     map(&:name) | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2015-12-06 21:30:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-15 20:06:15 +08:00
										 |  |  |   # @private | 
					
						
							| 
									
										
										
										
											2015-12-06 21:30:23 +08:00
										 |  |  |   def formula_file_to_name(file) | 
					
						
							|  |  |  |     "#{name}/#{file.basename(".rb")}" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-15 20:06:15 +08:00
										 |  |  |   # @private | 
					
						
							| 
									
										
										
										
											2015-12-06 21:30:23 +08:00
										 |  |  |   def alias_file_to_name(file) | 
					
						
							|  |  |  |     "#{name}/#{file.basename}" | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2015-06-10 15:39:47 +08:00
										 |  |  | end | 
					
						
							| 
									
										
										
										
											2016-03-07 18:04:25 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | # A specialized {Tap} class to mimic the core formula file system, which shares many | 
					
						
							|  |  |  | # similarities with normal {Tap}. | 
					
						
							|  |  |  | # TODO Separate core formulae with core codes. See discussion below for future plan: | 
					
						
							|  |  |  | #      https://github.com/Homebrew/homebrew/pull/46735#discussion_r46820565 | 
					
						
							|  |  |  | class CoreTap < Tap | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def initialize | 
					
						
							|  |  |  |     @user = "Homebrew" | 
					
						
							|  |  |  |     @repo = "homebrew" | 
					
						
							|  |  |  |     @name = "Homebrew/homebrew" | 
					
						
							|  |  |  |     @path = HOMEBREW_REPOSITORY | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def self.instance | 
					
						
							|  |  |  |     @instance ||= CoreTap.new | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def uninstall | 
					
						
							|  |  |  |     raise "Tap#uninstall is not available for CoreTap" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def pin | 
					
						
							|  |  |  |     raise "Tap#pin is not available for CoreTap" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def unpin | 
					
						
							|  |  |  |     raise "Tap#unpin is not available for CoreTap" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def pinned? | 
					
						
							|  |  |  |     false | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def command_files | 
					
						
							|  |  |  |     [] | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def custom_remote? | 
					
						
							|  |  |  |     remote != "https://github.com/#{user}/#{repo}.git" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def core_tap? | 
					
						
							|  |  |  |     true | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def formula_dir | 
					
						
							|  |  |  |     HOMEBREW_LIBRARY/"Formula" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def alias_dir | 
					
						
							|  |  |  |     HOMEBREW_LIBRARY/"Aliases" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def formula_renames | 
					
						
							|  |  |  |     FORMULA_RENAMES | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def tap_migrations | 
					
						
							|  |  |  |     TAP_MIGRATIONS | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def formula_file_to_name(file) | 
					
						
							|  |  |  |     file.basename(".rb").to_s | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # @private | 
					
						
							|  |  |  |   def alias_file_to_name(file) | 
					
						
							|  |  |  |     file.basename.to_s | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end |