| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  | # typed: strict | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-20 19:20:19 +01:00
										 |  |  |  | require "utils/output" | 
					
						
							| 
									
										
										
										
											2024-07-14 08:49:39 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  | module Kernel | 
					
						
							| 
									
										
										
										
											2025-06-09 19:06:16 +01:00
										 |  |  |  |   sig { params(env: T.nilable(String)).returns(T::Boolean) } | 
					
						
							|  |  |  |  |   def superenv?(env) | 
					
						
							|  |  |  |  |     return false if env == "std" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     !Superenv.bin.nil? | 
					
						
							|  |  |  |  |   end | 
					
						
							|  |  |  |  |   private :superenv? | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |   sig { params(formula: T.nilable(Formula)).void } | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |  |   def interactive_shell(formula = nil) | 
					
						
							|  |  |  |  |     unless formula.nil? | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |       ENV["HOMEBREW_DEBUG_PREFIX"] = formula.prefix.to_s | 
					
						
							| 
									
										
										
										
											2023-03-10 23:46:07 +00:00
										 |  |  |  |       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
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { type_parameters(:U).params(block: T.proc.returns(T.type_parameter(:U))).returns(T.type_parameter(:U)) } | 
					
						
							| 
									
										
										
										
											2020-08-19 17:12:32 +01:00
										 |  |  |  |   def with_homebrew_path(&block) | 
					
						
							| 
									
										
										
										
											2025-09-07 11:23:48 -07:00
										 |  |  |  |     with_env(PATH: PATH.new(ORIGINAL_PATHS).to_s, &block) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   end | 
					
						
							| 
									
										
										
										
											2016-11-11 20:08:26 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { | 
					
						
							|  |  |  |  |     type_parameters(:U) | 
					
						
							|  |  |  |  |       .params(locale: String, block: T.proc.returns(T.type_parameter(:U))) | 
					
						
							|  |  |  |  |       .returns(T.type_parameter(:U)) | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { | 
					
						
							|  |  |  |  |     params( | 
					
						
							|  |  |  |  |       cmd:     T.any(NilClass, Pathname, String, [String, String], T::Hash[String, T.nilable(String)]), | 
					
						
							|  |  |  |  |       argv0:   T.any(NilClass, Pathname, String, [String, String]), | 
					
						
							|  |  |  |  |       args:    T.any(NilClass, Pathname, String), | 
					
						
							|  |  |  |  |       options: T.untyped, | 
					
						
							|  |  |  |  |     ).void | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   def safe_system(cmd, argv0 = nil, *args, **options) | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |     # TODO: migrate to utils.rb Homebrew.safe_system | 
					
						
							| 
									
										
										
										
											2024-07-26 19:48:51 +01:00
										 |  |  |  |     require "utils" | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |     return if Homebrew.system(cmd, argv0, *args, **options) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |     raise ErrorDuringExecution.new([cmd, argv0, *args], status: $CHILD_STATUS) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   end | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-13 14:30:06 -07:00
										 |  |  |  |   # Run a system command without any output. | 
					
						
							| 
									
										
										
										
											2024-04-23 19:10:33 +02:00
										 |  |  |  |   # | 
					
						
							|  |  |  |  |   # @api internal | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { | 
					
						
							|  |  |  |  |     params( | 
					
						
							|  |  |  |  |       cmd:   T.any(NilClass, Pathname, String, [String, String], T::Hash[String, T.nilable(String)]), | 
					
						
							|  |  |  |  |       argv0: T.any(NilClass, String, [String, String]), | 
					
						
							|  |  |  |  |       args:  T.any(Pathname, String), | 
					
						
							|  |  |  |  |     ).returns(T::Boolean) | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   def quiet_system(cmd, argv0 = nil, *args) | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |     # TODO: migrate to utils.rb Homebrew.quiet_system | 
					
						
							| 
									
										
										
										
											2024-07-26 19:48:51 +01:00
										 |  |  |  |     require "utils" | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |     Homebrew._system(cmd, argv0, *args) do | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |       # 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. | 
					
						
							| 
									
										
										
										
											2024-11-26 11:50:45 -08:00
										 |  |  |  |       $stdout.reopen(File::NULL) | 
					
						
							|  |  |  |  |       $stderr.reopen(File::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
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-22 21:05:48 +02:00
										 |  |  |  |   # Find a command. | 
					
						
							|  |  |  |  |   # | 
					
						
							|  |  |  |  |   # @api public | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { params(cmd: String, path: PATH::Elements).returns(T.nilable(Pathname)) } | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { params(silent: T::Boolean).returns(String) } | 
					
						
							| 
									
										
										
										
											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
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-19 11:42:37 +00:00
										 |  |  |  |     # Find VS Code variants, Sublime Text, Textmate, BBEdit, or vim | 
					
						
							|  |  |  |  |     editor = %w[code codium cursor code-insiders 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 | 
					
						
							| 
									
										
										
										
											2025-08-20 19:20:19 +01:00
										 |  |  |  |       Utils::Output.opoo <<~EOS | 
					
						
							| 
									
										
										
										
											2023-02-11 00:16:11 -08:00
										 |  |  |  |         Using #{editor} because no editor was set in the environment. | 
					
						
							| 
									
										
										
										
											2025-08-02 23:27:59 -04:00
										 |  |  |  |         This may change in the future, so we recommend setting `$EDITOR` | 
					
						
							|  |  |  |  |         or `$HOMEBREW_EDITOR` to your preferred text editor. | 
					
						
							| 
									
										
										
										
											2023-02-11 00:16:11 -08:00
										 |  |  |  |       EOS | 
					
						
							|  |  |  |  |     end | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     editor | 
					
						
							|  |  |  |  |   end | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-03 16:35:53 +01:00
										 |  |  |  |   sig { params(filenames: T.any(String, Pathname)).void } | 
					
						
							|  |  |  |  |   def exec_editor(*filenames) | 
					
						
							|  |  |  |  |     puts "Editing #{filenames.join "\n"}" | 
					
						
							|  |  |  |  |     with_homebrew_path { safe_system(*which_editor.shellsplit, *filenames) } | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   end | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |   sig { params(args: T.any(String, Pathname)).void } | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   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 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |   IGNORE_INTERRUPTS_MUTEX = T.let(Thread::Mutex.new.freeze, Thread::Mutex) | 
					
						
							| 
									
										
										
										
											2020-12-17 15:45:50 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { type_parameters(:U).params(_block: T.proc.returns(T.type_parameter(:U))).returns(T.type_parameter(:U)) } | 
					
						
							|  |  |  |  |   def ignore_interrupts(&_block) | 
					
						
							| 
									
										
										
										
											2024-07-14 11:41:38 -04:00
										 |  |  |  |     IGNORE_INTERRUPTS_MUTEX.synchronize do | 
					
						
							| 
									
										
										
										
											2024-07-14 14:32:17 -04:00
										 |  |  |  |       interrupted = T.let(false, T::Boolean) | 
					
						
							| 
									
										
										
										
											2024-07-14 11:41:38 -04:00
										 |  |  |  |       old_sigint_handler = trap(:INT) do | 
					
						
							| 
									
										
										
										
											2024-07-14 13:36:43 -04:00
										 |  |  |  |         interrupted = true | 
					
						
							| 
									
										
										
										
											2024-07-14 13:18:33 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-14 15:08:37 -04:00
										 |  |  |  |         $stderr.print "\n" | 
					
						
							|  |  |  |  |         $stderr.puts "One sec, cleaning up..." | 
					
						
							| 
									
										
										
										
											2024-07-14 11:41:38 -04:00
										 |  |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-12-17 15:45:50 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-14 11:41:38 -04:00
										 |  |  |  |       begin | 
					
						
							|  |  |  |  |         yield | 
					
						
							|  |  |  |  |       ensure | 
					
						
							|  |  |  |  |         trap(:INT, old_sigint_handler) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-14 13:36:43 -04:00
										 |  |  |  |         raise Interrupt if interrupted | 
					
						
							| 
									
										
										
										
											2020-12-17 15:45:50 +01:00
										 |  |  |  |       end | 
					
						
							|  |  |  |  |     end | 
					
						
							| 
									
										
										
										
											2017-07-13 17:14:21 -07:00
										 |  |  |  |   end | 
					
						
							| 
									
										
										
										
											2017-07-14 17:03:33 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { | 
					
						
							|  |  |  |  |     type_parameters(:U) | 
					
						
							|  |  |  |  |       .params(file: T.any(IO, Pathname, String), _block: T.proc.returns(T.type_parameter(:U))) | 
					
						
							|  |  |  |  |       .returns(T.type_parameter(:U)) | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   def redirect_stdout(file, &_block) | 
					
						
							| 
									
										
										
										
											2021-11-23 00:24:15 +08:00
										 |  |  |  |     out = $stdout.dup | 
					
						
							|  |  |  |  |     $stdout.reopen(file) | 
					
						
							|  |  |  |  |     yield | 
					
						
							|  |  |  |  |   ensure | 
					
						
							|  |  |  |  |     $stdout.reopen(out) | 
					
						
							|  |  |  |  |     out.close | 
					
						
							|  |  |  |  |   end | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-23 23:29:05 +08:00
										 |  |  |  |   # Ensure the given executable is exist otherwise install the brewed version | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |   sig { params(name: String, formula_name: T.nilable(String), reason: String, latest: T::Boolean).returns(T.nilable(Pathname)) } | 
					
						
							| 
									
										
										
										
											2024-07-29 12:59:29 -04:00
										 |  |  |  |   def ensure_executable!(name, formula_name = nil, reason: "", latest: false) | 
					
						
							| 
									
										
										
										
											2021-11-23 23:29:05 +08:00
										 |  |  |  |     formula_name ||= name | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     executable = [ | 
					
						
							|  |  |  |  |       which(name), | 
					
						
							| 
									
										
										
										
											2022-06-15 05:40:43 +01:00
										 |  |  |  |       which(name, ORIGINAL_PATHS), | 
					
						
							| 
									
										
										
										
											2024-04-18 11:11:26 -07:00
										 |  |  |  |       # We prefer the opt_bin path to a formula's executable over the prefix | 
					
						
							|  |  |  |  |       # path where available, since the former is stable during upgrades. | 
					
						
							|  |  |  |  |       HOMEBREW_PREFIX/"opt/#{formula_name}/bin/#{name}", | 
					
						
							| 
									
										
										
										
											2021-11-23 23:29:05 +08:00
										 |  |  |  |       HOMEBREW_PREFIX/"bin/#{name}", | 
					
						
							|  |  |  |  |     ].compact.first | 
					
						
							|  |  |  |  |     return executable if executable.exist? | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-02 02:34:56 +08:00
										 |  |  |  |     require "formula" | 
					
						
							| 
									
										
										
										
											2025-08-02 03:37:31 +08:00
										 |  |  |  |     Formula[formula_name].ensure_installed!(reason:, latest:).opt_bin/name | 
					
						
							| 
									
										
										
										
											2021-11-23 23:29:05 +08:00
										 |  |  |  |   end | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |   sig { params(size_in_bytes: T.any(Integer, Float)).returns(String) } | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   def disk_usage_readable(size_in_bytes) | 
					
						
							| 
									
										
										
										
											2025-03-04 20:01:07 -05:00
										 |  |  |  |     if size_in_bytes.abs >= 1_073_741_824
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |       size = size_in_bytes.to_f / 1_073_741_824
 | 
					
						
							|  |  |  |  |       unit = "GB" | 
					
						
							| 
									
										
										
										
											2025-03-04 20:01:07 -05:00
										 |  |  |  |     elsif size_in_bytes.abs >= 1_048_576
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |       size = size_in_bytes.to_f / 1_048_576
 | 
					
						
							|  |  |  |  |       unit = "MB" | 
					
						
							| 
									
										
										
										
											2025-03-04 20:01:07 -05:00
										 |  |  |  |     elsif size_in_bytes.abs >= 1_024
 | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |       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 | 
					
						
							| 
									
										
										
										
											2024-03-07 16:20:20 +00:00
										 |  |  |  |       "#{format("%<size>.1f", size:)}#{unit}" | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |     end | 
					
						
							|  |  |  |  |   end | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { params(number: Integer).returns(String) } | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   def number_readable(number) | 
					
						
							|  |  |  |  |     numstr = number.to_i.to_s | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |     (numstr.size - 3).step(1, -3) { |i| numstr.insert(i.to_i, ",") } | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |     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. | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |   sig { params(str: String, max_bytes: Integer, options: T::Hash[Symbol, T.untyped]).returns(String) } | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |     out = T.must(front) + glue_bytes + T.must(back) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |     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 | 
					
						
							| 
									
										
										
										
											2024-04-26 20:55:51 +02:00
										 |  |  |  |   # added to `ENV`, then restores `ENV` afterwards. | 
					
						
							|  |  |  |  |   # | 
					
						
							|  |  |  |  |   # NOTE: This method is **not** thread-safe – other threads | 
					
						
							|  |  |  |  |   #       which happen to be scheduled during the block will also | 
					
						
							|  |  |  |  |   #       see these environment variables. | 
					
						
							|  |  |  |  |   # | 
					
						
							|  |  |  |  |   # ### Example | 
					
						
							|  |  |  |  |   # | 
					
						
							|  |  |  |  |   # ```ruby | 
					
						
							|  |  |  |  |   # with_env(PATH: "/bin") do | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   #   system "echo $PATH" | 
					
						
							| 
									
										
										
										
											2024-04-26 20:55:51 +02:00
										 |  |  |  |   # end | 
					
						
							|  |  |  |  |   # ``` | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   # | 
					
						
							| 
									
										
										
										
											2020-11-26 16:11:33 -08:00
										 |  |  |  |   # @api public | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |   sig { | 
					
						
							|  |  |  |  |     type_parameters(:U) | 
					
						
							| 
									
										
										
										
											2025-09-07 12:33:49 -07:00
										 |  |  |  |       .params(hash: T::Hash[Object, | 
					
						
							|  |  |  |  |                             T.any(NilClass, PATH, Pathname, String)], _block: T.proc.returns(T.type_parameter(:U))) | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |       .returns(T.type_parameter(:U)) | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |   def with_env(hash, &_block) | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |     old_values = {} | 
					
						
							|  |  |  |  |     begin | 
					
						
							|  |  |  |  |       hash.each do |key, value| | 
					
						
							|  |  |  |  |         key = key.to_s | 
					
						
							|  |  |  |  |         old_values[key] = ENV.delete(key) | 
					
						
							| 
									
										
										
										
											2025-09-07 12:33:49 -07:00
										 |  |  |  |         ENV[key] = value&.to_s | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |       end | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-23 18:44:36 -07:00
										 |  |  |  |       yield | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |     ensure | 
					
						
							|  |  |  |  |       ENV.update(old_values) | 
					
						
							|  |  |  |  |     end | 
					
						
							|  |  |  |  |   end | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |   sig { returns(T.proc.params(a: String, b: String).returns(Integer)) } | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   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 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-31 17:31:27 +01:00
										 |  |  |  |   sig { params(input: String, secrets: T::Array[String]).returns(String) } | 
					
						
							| 
									
										
										
										
											2019-08-15 09:12:21 +02:00
										 |  |  |  |   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 |