| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  | # typed: strict | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  | require "abstract_command" | 
					
						
							| 
									
										
										
										
											2019-03-09 13:00:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-27 12:07:30 +00:00
										 |  |  | module Homebrew | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |   module DevCmd | 
					
						
							|  |  |  |     class Prof < AbstractCommand | 
					
						
							| 
									
										
										
										
											2024-03-05 10:40:02 -08:00
										 |  |  |       cmd_args do | 
					
						
							|  |  |  |         description <<~EOS | 
					
						
							|  |  |  |           Run Homebrew with a Ruby profiler. For example, `brew prof readall`. | 
					
						
							|  |  |  |         EOS | 
					
						
							|  |  |  |         switch "--stackprof", | 
					
						
							|  |  |  |                description: "Use `stackprof` instead of `ruby-prof` (the default)." | 
					
						
							| 
									
										
										
										
											2024-05-17 23:18:00 +09:00
										 |  |  |         switch "--vernier", | 
					
						
							|  |  |  |                description: "Use `vernier` instead of `ruby-prof` (the default)." | 
					
						
							| 
									
										
										
										
											2018-02-27 12:07:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-05 10:40:02 -08:00
										 |  |  |         named_args :command, min: 1
 | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2021-01-10 14:26:40 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |       sig { override.void } | 
					
						
							|  |  |  |       def run | 
					
						
							|  |  |  |         Homebrew.install_bundler_gems!(groups: ["prof"], setup_path: false) | 
					
						
							| 
									
										
										
										
											2018-10-08 22:48:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |         brew_rb = (HOMEBREW_LIBRARY_PATH/"brew.rb").resolved_path | 
					
						
							|  |  |  |         FileUtils.mkdir_p "prof" | 
					
						
							|  |  |  |         cmd = args.named.first | 
					
						
							| 
									
										
										
										
											2019-03-27 11:49:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |         case Commands.path(cmd)&.extname | 
					
						
							|  |  |  |         when ".rb" | 
					
						
							|  |  |  |           # expected file extension so we do nothing | 
					
						
							|  |  |  |         when ".sh" | 
					
						
							|  |  |  |           raise UsageError, <<~EOS | 
					
						
							|  |  |  |             `#{cmd}` is a Bash command! | 
					
						
							|  |  |  |             Try `hyperfine` for benchmarking instead. | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           raise UsageError, "`#{cmd}` is an unknown command!" | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2023-04-10 13:11:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |         Homebrew.setup_gem_environment! | 
					
						
							| 
									
										
										
										
											2023-02-15 00:50:21 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-15 12:51:03 -07:00
										 |  |  |         if args.stackprof? | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |           with_env HOMEBREW_STACKPROF: "1" do | 
					
						
							|  |  |  |             system(*HOMEBREW_RUBY_EXEC_ARGS, brew_rb, *args.named) | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |           output_filename = "prof/d3-flamegraph.html" | 
					
						
							|  |  |  |           safe_system "stackprof --d3-flamegraph prof/stackprof.dump > #{output_filename}" | 
					
						
							| 
									
										
										
										
											2024-05-17 23:18:00 +09:00
										 |  |  |           exec_browser output_filename | 
					
						
							|  |  |  |         elsif args.vernier? | 
					
						
							|  |  |  |           output_filename = "prof/vernier.json" | 
					
						
							|  |  |  |           Process::UID.change_privilege(Process.euid) if Process.euid != Process.uid | 
					
						
							|  |  |  |           safe_system "vernier", "run", "--output=#{output_filename}", "--allocation_sample_rate=500", "--", | 
					
						
							|  |  |  |                       RUBY_PATH, brew_rb, *args.named | 
					
						
							|  |  |  |           ohai "Profiling complete!" | 
					
						
							|  |  |  |           puts "Upload the results from #{output_filename} to:" | 
					
						
							|  |  |  |           puts "  #{Formatter.url("https://vernier.prof")}" | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |         else | 
					
						
							|  |  |  |           output_filename = "prof/call_stack.html" | 
					
						
							|  |  |  |           safe_system "ruby-prof", "--printer=call_stack", "--file=#{output_filename}", brew_rb, "--", *args.named | 
					
						
							| 
									
										
										
										
											2024-05-17 23:18:00 +09:00
										 |  |  |           exec_browser output_filename | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |         end | 
					
						
							|  |  |  |       rescue OptionParser::InvalidOption => e | 
					
						
							|  |  |  |         ofail e | 
					
						
							| 
									
										
										
										
											2023-08-21 14:22:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-04 10:29:47 -08:00
										 |  |  |         # The invalid option could have been meant for the subcommand. | 
					
						
							|  |  |  |         # Suggest `brew prof list -r` -> `brew prof -- list -r` | 
					
						
							|  |  |  |         args = ARGV - ["--"] | 
					
						
							|  |  |  |         puts "Try `brew prof -- #{args.join(" ")}` instead." | 
					
						
							| 
									
										
										
										
											2020-08-20 13:01:58 +01:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-02-27 12:07:30 +00:00
										 |  |  |   end | 
					
						
							|  |  |  | end |