| 
									
										
										
										
											2023-02-27 11:43:16 -08:00
										 |  |  | # typed: true | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-26 14:50:34 +00:00
										 |  |  | # Contains shorthand Homebrew utility methods like `ohai`, `opoo`, `odisabled`. | 
					
						
							| 
									
										
										
										
											2023-02-28 12:50:05 +00:00
										 |  |  | # TODO: move these out of `Kernel`. | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  | module Kernel | 
					
						
							|  |  |  |   def require?(path) | 
					
						
							|  |  |  |     return false if path.nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     require path | 
					
						
							|  |  |  |     true | 
					
						
							|  |  |  |   rescue LoadError => e | 
					
						
							|  |  |  |     # we should raise on syntax errors but not if the file doesn't exist. | 
					
						
							|  |  |  |     raise unless e.message.include?(path) | 
					
						
							| 
									
										
										
										
											2017-10-29 14:44:43 +00:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2017-06-01 17:45:07 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def ohai_title(title) | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |     verbose = if respond_to?(:verbose?) | 
					
						
							| 
									
										
										
										
											2023-02-27 11:43:16 -08:00
										 |  |  |       T.unsafe(self).verbose? | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2020-08-02 14:32:31 +02:00
										 |  |  |       Context.current.verbose? | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-02 09:22:15 -04:00
										 |  |  |     title = Tty.truncate(title.to_s) if $stdout.tty? && !verbose | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     Formatter.headline(title, color: :blue) | 
					
						
							| 
									
										
										
										
											2017-10-29 14:44:43 +00:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2016-08-01 04:07:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def ohai(title, *sput) | 
					
						
							|  |  |  |     puts ohai_title(title) | 
					
						
							|  |  |  |     puts sput | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |   def odebug(title, *sput, always_display: false) | 
					
						
							| 
									
										
										
										
											2020-08-02 14:32:31 +02:00
										 |  |  |     debug = if respond_to?(:debug) | 
					
						
							| 
									
										
										
										
											2023-02-27 11:43:16 -08:00
										 |  |  |       T.unsafe(self).debug? | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2020-08-02 14:32:31 +02:00
										 |  |  |       Context.current.debug? | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 13:49:05 -08:00
										 |  |  |     return if !debug && !always_display | 
					
						
							| 
									
										
										
										
											2010-01-13 09:00:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-10 01:54:54 +02:00
										 |  |  |     $stderr.puts Formatter.headline(title, color: :magenta) | 
					
						
							|  |  |  |     $stderr.puts sput unless sput.empty? | 
					
						
							| 
									
										
										
										
											2009-11-09 17:44:29 +00:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2009-08-11 12:20:55 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 23:44:45 -07:00
										 |  |  |   def oh1_title(title, truncate: :auto) | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |     verbose = if respond_to?(:verbose?) | 
					
						
							| 
									
										
										
										
											2023-02-27 11:43:16 -08:00
										 |  |  |       T.unsafe(self).verbose? | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2020-08-02 14:32:31 +02:00
										 |  |  |       Context.current.verbose? | 
					
						
							| 
									
										
										
										
											2020-08-02 03:22:22 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-02 09:22:15 -04:00
										 |  |  |     title = Tty.truncate(title.to_s) if $stdout.tty? && !verbose && truncate == :auto | 
					
						
							| 
									
										
										
										
											2022-08-12 23:44:45 -07:00
										 |  |  |     Formatter.headline(title, color: :green) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def oh1(title, truncate: :auto) | 
					
						
							|  |  |  |     puts oh1_title(title, truncate: truncate) | 
					
						
							| 
									
										
										
										
											2014-03-16 11:52:11 -07:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2012-03-03 15:50:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-05 17:17:03 -05:00
										 |  |  |   # Print a message prefixed with "Warning" (do this rarely). | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def opoo(message) | 
					
						
							| 
									
										
										
										
											2020-09-17 04:18:13 +05:30
										 |  |  |     Tty.with($stderr) do |stderr| | 
					
						
							|  |  |  |       stderr.puts Formatter.warning(message, label: "Warning") | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-05 17:17:03 -05:00
										 |  |  |   # Print a message prefixed with "Error". | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def onoe(message) | 
					
						
							| 
									
										
										
										
											2020-09-17 04:18:13 +05:30
										 |  |  |     Tty.with($stderr) do |stderr| | 
					
						
							|  |  |  |       stderr.puts Formatter.error(message, label: "Error") | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def ofail(error) | 
					
						
							|  |  |  |     onoe error | 
					
						
							|  |  |  |     Homebrew.failed = true | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-13 18:31:26 -07:00
										 |  |  |   sig { params(error: T.any(String, Exception)).returns(T.noreturn) } | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def odie(error) | 
					
						
							|  |  |  |     onoe error | 
					
						
							|  |  |  |     exit 1
 | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-18 16:41:39 +01:00
										 |  |  |   def odeprecated(method, replacement = nil, | 
					
						
							|  |  |  |                   disable:                false, | 
					
						
							|  |  |  |                   disable_on:             nil, | 
					
						
							|  |  |  |                   disable_for_developers: true, | 
					
						
							|  |  |  |                   caller:                 send(:caller)) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     replacement_message = if replacement | 
					
						
							|  |  |  |       "Use #{replacement} instead." | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       "There is no replacement." | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unless disable_on.nil? | 
					
						
							|  |  |  |       if disable_on > Time.now | 
					
						
							|  |  |  |         will_be_disabled_message = " and will be disabled on #{disable_on.strftime("%Y-%m-%d")}" | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         disable = true | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     verb = if disable | 
					
						
							|  |  |  |       "disabled" | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       "deprecated#{will_be_disabled_message}" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Try to show the most relevant location in message, i.e. (if applicable): | 
					
						
							|  |  |  |     # - Location in a formula. | 
					
						
							|  |  |  |     # - Location of caller of deprecated method (if all else fails). | 
					
						
							|  |  |  |     backtrace = caller | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Don't throw deprecations at all for cached, .brew or .metadata files. | 
					
						
							|  |  |  |     return if backtrace.any? do |line| | 
					
						
							| 
									
										
										
										
											2021-03-24 09:04:49 +01:00
										 |  |  |       next true if line.include?(HOMEBREW_CACHE.to_s) | 
					
						
							| 
									
										
										
										
											2020-12-15 12:04:28 +00:00
										 |  |  |       next true if line.include?("/.brew/") | 
					
						
							|  |  |  |       next true if line.include?("/.metadata/") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       next false unless line.match?(HOMEBREW_TAP_PATH_REGEX) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       path = Pathname(line.split(":", 2).first) | 
					
						
							|  |  |  |       next false unless path.file? | 
					
						
							|  |  |  |       next false unless path.readable? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       formula_contents = path.read | 
					
						
							|  |  |  |       formula_contents.include?(" deprecate! ") || formula_contents.include?(" disable! ") | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 18:15:48 +01:00
										 |  |  |     tap_message = T.let(nil, T.nilable(String)) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     backtrace.each do |line| | 
					
						
							| 
									
										
										
										
											2021-02-12 18:33:37 +05:30
										 |  |  |       next unless (match = line.match(HOMEBREW_TAP_PATH_REGEX)) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |       tap = Tap.fetch(match[:user], match[:repo]) | 
					
						
							| 
									
										
										
										
											2023-02-10 23:15:40 -05:00
										 |  |  |       tap_message = +"\nPlease report this issue to the #{tap} tap (not Homebrew/brew or Homebrew/homebrew-core)" | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |       tap_message += ", or even better, submit a PR to fix it" if replacement | 
					
						
							| 
									
										
										
										
											2020-06-02 09:49:23 +01:00
										 |  |  |       tap_message << ":\n  #{line.sub(/^(.*:\d+):.*$/, '\1')}\n\n" | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |       break | 
					
						
							| 
									
										
										
										
											2016-01-02 23:08:51 +08:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     message = +"Calling #{method} is #{verb}! #{replacement_message}" | 
					
						
							|  |  |  |     message << tap_message if tap_message | 
					
						
							|  |  |  |     message.freeze | 
					
						
							| 
									
										
										
										
											2017-06-07 16:07:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-18 16:41:39 +01:00
										 |  |  |     disable = true if disable_for_developers && Homebrew::EnvConfig.developer? | 
					
						
							|  |  |  |     if disable || Homebrew.raise_deprecation_exceptions? | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |       exception = MethodDeprecatedError.new(message) | 
					
						
							|  |  |  |       exception.set_backtrace(backtrace) | 
					
						
							|  |  |  |       raise exception | 
					
						
							|  |  |  |     elsif !Homebrew.auditing? | 
					
						
							|  |  |  |       opoo message | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2017-04-24 08:49:11 +01:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2015-01-05 15:23:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-28 17:48:31 +01:00
										 |  |  |   def odisabled(method, replacement = nil, **options) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     options = { disable: true, caller: caller }.merge(options) | 
					
						
							| 
									
										
										
										
											2023-07-06 16:47:09 +01:00
										 |  |  |     # This odeprecated should stick around indefinitely. | 
					
						
							| 
									
										
										
										
											2023-10-28 17:48:31 +01:00
										 |  |  |     odeprecated(method, replacement, **options) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2015-01-05 15:23:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |   def pretty_installed(formula) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     if !$stdout.tty? | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       formula.to_s | 
					
						
							| 
									
										
										
										
											2020-04-05 15:44:50 +01:00
										 |  |  |     elsif Homebrew::EnvConfig.no_emoji? | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       Formatter.success("#{Tty.bold}#{formula} (installed)#{Tty.reset}") | 
					
						
							| 
									
										
										
										
											2020-04-05 15:44:50 +01:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       "#{Tty.bold}#{formula} #{Formatter.success("✔")}#{Tty.reset}" | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2011-12-16 14:27:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |   def pretty_outdated(formula) | 
					
						
							| 
									
										
										
										
											2022-06-03 19:16:15 +01:00
										 |  |  |     if !$stdout.tty? | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       formula.to_s | 
					
						
							| 
									
										
										
										
											2022-06-03 19:16:15 +01:00
										 |  |  |     elsif Homebrew::EnvConfig.no_emoji? | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       Formatter.error("#{Tty.bold}#{formula} (outdated)#{Tty.reset}") | 
					
						
							| 
									
										
										
										
											2022-06-03 19:16:15 +01:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       "#{Tty.bold}#{formula} #{Formatter.warning("⚠")}#{Tty.reset}" | 
					
						
							| 
									
										
										
										
											2022-06-03 19:16:15 +01:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |   def pretty_uninstalled(formula) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     if !$stdout.tty? | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       formula.to_s | 
					
						
							| 
									
										
										
										
											2020-04-05 15:44:50 +01:00
										 |  |  |     elsif Homebrew::EnvConfig.no_emoji? | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       Formatter.error("#{Tty.bold}#{formula} (uninstalled)#{Tty.reset}") | 
					
						
							| 
									
										
										
										
											2020-04-05 15:44:50 +01:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |       "#{Tty.bold}#{formula} #{Formatter.error("✘")}#{Tty.reset}" | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2012-12-27 23:34:29 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 23:55:16 +00:00
										 |  |  |   def pretty_duration(seconds) | 
					
						
							|  |  |  |     seconds = seconds.to_i | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     res = +"" | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-07 23:55:16 +00:00
										 |  |  |     if seconds > 59
 | 
					
						
							|  |  |  |       minutes = seconds / 60
 | 
					
						
							|  |  |  |       seconds %= 60
 | 
					
						
							| 
									
										
										
										
											2023-03-20 07:23:17 -04:00
										 |  |  |       res = +Utils.pluralize("minute", minutes, include_count: true) | 
					
						
							| 
									
										
										
										
											2023-03-07 23:55:16 +00:00
										 |  |  |       return res.freeze if seconds.zero? | 
					
						
							| 
									
										
										
										
											2019-02-10 23:30:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |       res << " " | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2009-09-08 15:31:28 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-20 07:23:17 -04:00
										 |  |  |     res << Utils.pluralize("second", seconds, include_count: true) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     res.freeze | 
					
						
							| 
									
										
										
										
											2010-09-12 22:55:52 +02:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2010-01-29 10:15:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |   def interactive_shell(formula = nil) | 
					
						
							|  |  |  |     unless formula.nil? | 
					
						
							|  |  |  |       ENV["HOMEBREW_DEBUG_PREFIX"] = formula.prefix | 
					
						
							|  |  |  |       ENV["HOMEBREW_DEBUG_INSTALL"] = formula.full_name | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-30 05:07:42 +01:00
										 |  |  |     if Utils::Shell.preferred == :zsh && (home = Dir.home).start_with?(HOMEBREW_TEMP.resolved_path.to_s) | 
					
						
							| 
									
										
										
										
											2020-11-23 18:15:48 +01:00
										 |  |  |       FileUtils.mkdir_p home | 
					
						
							|  |  |  |       FileUtils.touch "#{home}/.zshrc" | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-28 15:02:06 +00:00
										 |  |  |     Process.wait fork { exec Utils::Shell.preferred_path(default: "/bin/bash") } | 
					
						
							| 
									
										
										
										
											2009-10-15 12:36:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     return if $CHILD_STATUS.success? | 
					
						
							|  |  |  |     raise "Aborted due to non-zero exit status (#{$CHILD_STATUS.exitstatus})" if $CHILD_STATUS.exited? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     raise $CHILD_STATUS.inspect | 
					
						
							| 
									
										
										
										
											2012-09-11 20:59:59 -04:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2009-11-09 18:24:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 17:12:32 +01:00
										 |  |  |   def with_homebrew_path(&block) | 
					
						
							| 
									
										
										
										
											2022-06-15 05:40:43 +01:00
										 |  |  |     with_env(PATH: PATH.new(ORIGINAL_PATHS), &block) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2016-11-11 20:08:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 17:12:32 +01:00
										 |  |  |   def with_custom_locale(locale, &block) | 
					
						
							|  |  |  |     with_env(LC_ALL: locale, &block) | 
					
						
							| 
									
										
										
										
											2009-11-09 18:24:36 +00:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2010-04-07 21:01:12 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-05 17:17:03 -05:00
										 |  |  |   # Kernel.system but with exceptions. | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def safe_system(cmd, *args, **options) | 
					
						
							|  |  |  |     return if Homebrew.system(cmd, *args, **options) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     raise ErrorDuringExecution.new([cmd, *args], status: $CHILD_STATUS) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-05 17:17:03 -05:00
										 |  |  |   # Prints no output. | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def quiet_system(cmd, *args) | 
					
						
							|  |  |  |     Homebrew._system(cmd, *args) do | 
					
						
							|  |  |  |       # Redirect output streams to `/dev/null` instead of closing as some programs | 
					
						
							|  |  |  |       # will fail to execute if they can't write to an open stream. | 
					
						
							|  |  |  |       $stdout.reopen("/dev/null") | 
					
						
							|  |  |  |       $stderr.reopen("/dev/null") | 
					
						
							| 
									
										
										
										
											2013-09-15 20:11:17 -07:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2013-09-15 20:11:17 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-15 05:40:43 +01:00
										 |  |  |   def which(cmd, path = ENV.fetch("PATH")) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     PATH.new(path).each do |p| | 
					
						
							|  |  |  |       begin | 
					
						
							|  |  |  |         pcmd = File.expand_path(cmd, p) | 
					
						
							|  |  |  |       rescue ArgumentError | 
					
						
							|  |  |  |         # File.expand_path will raise an ArgumentError if the path is malformed. | 
					
						
							|  |  |  |         # See https://github.com/Homebrew/legacy-homebrew/issues/32789 | 
					
						
							|  |  |  |         next | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       return Pathname.new(pcmd) if File.file?(pcmd) && File.executable?(pcmd) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     nil | 
					
						
							| 
									
										
										
										
											2015-11-17 16:12:31 +05:30
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-15 05:40:43 +01:00
										 |  |  |   def which_all(cmd, path = ENV.fetch("PATH")) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     PATH.new(path).map do |p| | 
					
						
							|  |  |  |       begin | 
					
						
							|  |  |  |         pcmd = File.expand_path(cmd, p) | 
					
						
							|  |  |  |       rescue ArgumentError | 
					
						
							|  |  |  |         # File.expand_path will raise an ArgumentError if the path is malformed. | 
					
						
							|  |  |  |         # See https://github.com/Homebrew/legacy-homebrew/issues/32789 | 
					
						
							|  |  |  |         next | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       Pathname.new(pcmd) if File.file?(pcmd) && File.executable?(pcmd) | 
					
						
							|  |  |  |     end.compact.uniq | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2016-07-05 08:45:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-11 00:16:11 -08:00
										 |  |  |   def which_editor(silent: false) | 
					
						
							| 
									
										
										
										
											2020-04-05 15:44:50 +01:00
										 |  |  |     editor = Homebrew::EnvConfig.editor | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     return editor if editor | 
					
						
							| 
									
										
										
										
											2016-08-24 11:06:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 08:53:32 +00:00
										 |  |  |     # Find VS Code, Sublime Text, Textmate, BBEdit, or vim | 
					
						
							|  |  |  |     editor = %w[code subl mate bbedit vim].find do |candidate| | 
					
						
							| 
									
										
										
										
											2022-06-15 05:40:43 +01:00
										 |  |  |       candidate if which(candidate, ORIGINAL_PATHS) | 
					
						
							| 
									
										
										
										
											2017-07-13 17:14:21 -07:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     editor ||= "vim" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-11 00:16:11 -08:00
										 |  |  |     unless silent | 
					
						
							|  |  |  |       opoo <<~EOS | 
					
						
							|  |  |  |         Using #{editor} because no editor was set in the environment. | 
					
						
							| 
									
										
										
										
											2023-09-08 14:46:15 -04:00
										 |  |  |         This may change in the future, so we recommend setting EDITOR | 
					
						
							| 
									
										
										
										
											2023-02-11 00:16:11 -08:00
										 |  |  |         or HOMEBREW_EDITOR to your preferred text editor. | 
					
						
							|  |  |  |       EOS | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     editor | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def exec_editor(*args) | 
					
						
							|  |  |  |     puts "Editing #{args.join "\n"}" | 
					
						
							|  |  |  |     with_homebrew_path { safe_system(*which_editor.shellsplit, *args) } | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def exec_browser(*args) | 
					
						
							| 
									
										
										
										
											2020-04-05 15:44:50 +01:00
										 |  |  |     browser = Homebrew::EnvConfig.browser | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     browser ||= OS::PATH_OPEN if defined?(OS::PATH_OPEN) | 
					
						
							|  |  |  |     return unless browser | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 15:44:50 +01:00
										 |  |  |     ENV["DISPLAY"] = Homebrew::EnvConfig.display | 
					
						
							| 
									
										
										
										
											2017-07-13 17:14:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-30 04:37:09 +01:00
										 |  |  |     with_env(DBUS_SESSION_BUS_ADDRESS: ENV.fetch("HOMEBREW_DBUS_SESSION_BUS_ADDRESS", nil)) do | 
					
						
							| 
									
										
										
										
											2022-02-17 11:55:04 +08:00
										 |  |  |       safe_system(browser, *args) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-17 15:45:50 +01:00
										 |  |  |   def ignore_interrupts(_opt = nil) | 
					
						
							|  |  |  |     # rubocop:disable Style/GlobalVars | 
					
						
							|  |  |  |     $ignore_interrupts_nesting_level = 0 unless defined?($ignore_interrupts_nesting_level) | 
					
						
							|  |  |  |     $ignore_interrupts_nesting_level += 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     $ignore_interrupts_interrupted = false unless defined?($ignore_interrupts_interrupted) | 
					
						
							|  |  |  |     old_sigint_handler = trap(:INT) do | 
					
						
							|  |  |  |       $ignore_interrupts_interrupted = true | 
					
						
							|  |  |  |       $stderr.print "\n" | 
					
						
							|  |  |  |       $stderr.puts "One sec, cleaning up..." | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2020-12-17 15:45:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     begin | 
					
						
							|  |  |  |       yield | 
					
						
							|  |  |  |     ensure | 
					
						
							|  |  |  |       trap(:INT, old_sigint_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       $ignore_interrupts_nesting_level -= 1
 | 
					
						
							|  |  |  |       if $ignore_interrupts_nesting_level == 0 && $ignore_interrupts_interrupted | 
					
						
							|  |  |  |         $ignore_interrupts_interrupted = false | 
					
						
							|  |  |  |         raise Interrupt | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     # rubocop:enable Style/GlobalVars | 
					
						
							| 
									
										
										
										
											2017-07-13 17:14:21 -07:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2017-07-14 17:03:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-23 00:24:15 +08:00
										 |  |  |   def redirect_stdout(file) | 
					
						
							|  |  |  |     out = $stdout.dup | 
					
						
							|  |  |  |     $stdout.reopen(file) | 
					
						
							|  |  |  |     yield | 
					
						
							|  |  |  |   ensure | 
					
						
							|  |  |  |     $stdout.reopen(out) | 
					
						
							|  |  |  |     out.close | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-23 22:48:39 +08:00
										 |  |  |   # Ensure the given formula is installed | 
					
						
							|  |  |  |   # This is useful for installing a utility formula (e.g. `shellcheck` for `brew style`) | 
					
						
							| 
									
										
										
										
											2021-11-24 00:46:02 +08:00
										 |  |  |   def ensure_formula_installed!(formula_or_name, reason: "", latest: false, | 
					
						
							| 
									
										
										
										
											2021-11-23 23:09:42 +08:00
										 |  |  |                                 output_to_stderr: true, quiet: false) | 
					
						
							| 
									
										
										
										
											2021-11-23 22:48:39 +08:00
										 |  |  |     if output_to_stderr || quiet | 
					
						
							|  |  |  |       file = if quiet | 
					
						
							|  |  |  |         File::NULL | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         $stderr | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       # Call this method itself with redirected stdout | 
					
						
							|  |  |  |       redirect_stdout(file) do | 
					
						
							| 
									
										
										
										
											2021-11-23 23:25:32 +08:00
										 |  |  |         return ensure_formula_installed!(formula_or_name, latest: latest, | 
					
						
							|  |  |  |                                          reason: reason, output_to_stderr: false) | 
					
						
							| 
									
										
										
										
											2021-11-23 22:48:39 +08:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     require "formula" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     formula = if formula_or_name.is_a?(Formula) | 
					
						
							|  |  |  |       formula_or_name | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       Formula[formula_or_name] | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-23 23:59:09 +08:00
										 |  |  |     reason = " for #{reason}" if reason.present? | 
					
						
							| 
									
										
										
										
											2021-11-23 22:48:39 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     unless formula.any_version_installed? | 
					
						
							|  |  |  |       ohai "Installing `#{formula.name}`#{reason}..." | 
					
						
							|  |  |  |       safe_system HOMEBREW_BREW_FILE, "install", "--formula", formula.full_name | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if latest && !formula.latest_version_installed? | 
					
						
							|  |  |  |       ohai "Upgrading `#{formula.name}`#{reason}..." | 
					
						
							|  |  |  |       safe_system HOMEBREW_BREW_FILE, "upgrade", "--formula", formula.full_name | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     formula | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-23 23:29:05 +08:00
										 |  |  |   # Ensure the given executable is exist otherwise install the brewed version | 
					
						
							|  |  |  |   def ensure_executable!(name, formula_name = nil, reason: "") | 
					
						
							|  |  |  |     formula_name ||= name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     executable = [ | 
					
						
							|  |  |  |       which(name), | 
					
						
							| 
									
										
										
										
											2022-06-15 05:40:43 +01:00
										 |  |  |       which(name, ORIGINAL_PATHS), | 
					
						
							| 
									
										
										
										
											2021-11-23 23:29:05 +08:00
										 |  |  |       HOMEBREW_PREFIX/"bin/#{name}", | 
					
						
							|  |  |  |     ].compact.first | 
					
						
							|  |  |  |     return executable if executable.exist? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ensure_formula_installed!(formula_name, reason: reason).opt_bin/name | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def paths | 
					
						
							| 
									
										
										
										
											2022-06-15 05:40:43 +01:00
										 |  |  |     @paths ||= ORIGINAL_PATHS.uniq.map(&:to_s) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2019-06-28 14:50:38 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def disk_usage_readable(size_in_bytes) | 
					
						
							|  |  |  |     if size_in_bytes >= 1_073_741_824
 | 
					
						
							|  |  |  |       size = size_in_bytes.to_f / 1_073_741_824
 | 
					
						
							|  |  |  |       unit = "GB" | 
					
						
							|  |  |  |     elsif size_in_bytes >= 1_048_576
 | 
					
						
							|  |  |  |       size = size_in_bytes.to_f / 1_048_576
 | 
					
						
							|  |  |  |       unit = "MB" | 
					
						
							|  |  |  |     elsif size_in_bytes >= 1_024
 | 
					
						
							|  |  |  |       size = size_in_bytes.to_f / 1_024
 | 
					
						
							|  |  |  |       unit = "KB" | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       size = size_in_bytes | 
					
						
							|  |  |  |       unit = "B" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # avoid trailing zero after decimal point | 
					
						
							|  |  |  |     if ((size * 10).to_i % 10).zero? | 
					
						
							|  |  |  |       "#{size.to_i}#{unit}" | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2019-10-03 08:50:45 +02:00
										 |  |  |       "#{format("%<size>.1f", size: size)}#{unit}" | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def number_readable(number) | 
					
						
							|  |  |  |     numstr = number.to_i.to_s | 
					
						
							|  |  |  |     (numstr.size - 3).step(1, -3) { |i| numstr.insert(i, ",") } | 
					
						
							|  |  |  |     numstr | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Truncates a text string to fit within a byte size constraint, | 
					
						
							|  |  |  |   # preserving character encoding validity. The returned string will | 
					
						
							|  |  |  |   # be not much longer than the specified max_bytes, though the exact | 
					
						
							|  |  |  |   # shortfall or overrun may vary. | 
					
						
							| 
									
										
										
										
											2023-03-07 23:55:16 +00:00
										 |  |  |   def truncate_text_to_approximate_size(str, max_bytes, options = {}) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     front_weight = options.fetch(:front_weight, 0.5) | 
					
						
							|  |  |  |     raise "opts[:front_weight] must be between 0.0 and 1.0" if front_weight < 0.0 || front_weight > 1.0
 | 
					
						
							| 
									
										
										
										
											2023-03-07 23:55:16 +00:00
										 |  |  |     return str if str.bytesize <= max_bytes | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     glue = "\n[...snip...]\n" | 
					
						
							|  |  |  |     max_bytes_in = [max_bytes - glue.bytesize, 1].max | 
					
						
							| 
									
										
										
										
											2023-03-07 23:55:16 +00:00
										 |  |  |     bytes = str.dup.force_encoding("BINARY") | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     glue_bytes = glue.encode("BINARY") | 
					
						
							|  |  |  |     n_front_bytes = (max_bytes_in * front_weight).floor | 
					
						
							|  |  |  |     n_back_bytes = max_bytes_in - n_front_bytes | 
					
						
							|  |  |  |     if n_front_bytes.zero? | 
					
						
							|  |  |  |       front = bytes[1..0] | 
					
						
							| 
									
										
										
										
											2020-05-21 10:15:34 +01:00
										 |  |  |       back = bytes[-max_bytes_in..] | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     elsif n_back_bytes.zero? | 
					
						
							|  |  |  |       front = bytes[0..(max_bytes_in - 1)] | 
					
						
							|  |  |  |       back = bytes[1..0] | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       front = bytes[0..(n_front_bytes - 1)] | 
					
						
							| 
									
										
										
										
											2020-05-21 10:15:34 +01:00
										 |  |  |       back = bytes[-n_back_bytes..] | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |     end | 
					
						
							|  |  |  |     out = front + glue_bytes + back | 
					
						
							|  |  |  |     out.force_encoding("UTF-8") | 
					
						
							|  |  |  |     out.encode!("UTF-16", invalid: :replace) | 
					
						
							|  |  |  |     out.encode!("UTF-8") | 
					
						
							|  |  |  |     out | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Calls the given block with the passed environment variables | 
					
						
							|  |  |  |   # added to ENV, then restores ENV afterwards. | 
					
						
							|  |  |  |   # <pre>with_env(PATH: "/bin") do | 
					
						
							|  |  |  |   #   system "echo $PATH" | 
					
						
							|  |  |  |   # end</pre> | 
					
						
							|  |  |  |   # | 
					
						
							| 
									
										
										
										
											2020-11-05 15:19:56 -05:00
										 |  |  |   # @note This method is *not* thread-safe - other threads | 
					
						
							|  |  |  |   #   which happen to be scheduled during the block will also | 
					
						
							|  |  |  |   #   see these environment variables. | 
					
						
							| 
									
										
										
										
											2020-11-26 16:11:33 -08:00
										 |  |  |   # @api public | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |   def with_env(hash) | 
					
						
							|  |  |  |     old_values = {} | 
					
						
							|  |  |  |     begin | 
					
						
							|  |  |  |       hash.each do |key, value| | 
					
						
							|  |  |  |         key = key.to_s | 
					
						
							|  |  |  |         old_values[key] = ENV.delete(key) | 
					
						
							|  |  |  |         ENV[key] = value | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       yield if block_given? | 
					
						
							|  |  |  |     ensure | 
					
						
							|  |  |  |       ENV.update(old_values) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def tap_and_name_comparison | 
					
						
							|  |  |  |     proc do |a, b| | 
					
						
							| 
									
										
										
										
											2020-12-01 17:04:59 +00:00
										 |  |  |       if a.include?("/") && b.exclude?("/") | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |         1
 | 
					
						
							| 
									
										
										
										
											2020-12-01 17:04:59 +00:00
										 |  |  |       elsif a.exclude?("/") && b.include?("/") | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |         -1
 | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         a <=> b | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def redact_secrets(input, secrets) | 
					
						
							|  |  |  |     secrets.compact | 
					
						
							|  |  |  |            .reduce(input) { |str, secret| str.gsub secret, "******" } | 
					
						
							|  |  |  |            .freeze | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2019-06-28 14:50:38 +08:00
										 |  |  | end |