Class: Commander::Runner
- Inherits:
-
Object
- Object
- Commander::Runner
- Defined in:
- pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb
Overview
This class override the run method with our custom stack trace handling In particular we want to distinguish between user_error! and crash! (one with, one without stack trace)
Instance Attribute Summary collapse
Instance Method Summary collapse
- #action_completed(action_name, status: nil, exception: nil) ⇒ Object
- #display_user_error!(e, message) ⇒ Object
- #handle_ssl_error!(e) ⇒ Object
- #handle_tls_error!(e) ⇒ Object
- #handle_unknown_error!(e) ⇒ Object
- #reraise_formatted!(e, message) ⇒ Object
- #rescue_connection_failed_error(e) ⇒ Object
- #rescue_file_error(e) ⇒ Object
- #rescue_pantograph_error(e) ⇒ Object
- #rescue_unknown_error(e) ⇒ Object
- #run! ⇒ Object
- #show_github_issues(message_or_error) ⇒ Object
- #suggest_ruby_reinstall(e) ⇒ Object
Instance Attribute Details
#collector ⇒ Object
35 36 37 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 35 def collector @collector end |
Instance Method Details
#action_completed(action_name, status: nil, exception: nil) ⇒ Object
119 120 121 122 123 124 125 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 119 def action_completed(action_name, status: nil, exception: nil) # https://github.com/urbanquakers/pantograph/issues/11913 # if exception.nil? || exception.pantograph_should_report_metrics? # action_completion_context = PantographCore::ActionCompletionContext.context_for_action_name(action_name, args: ARGV, status: status) # PantographCore.session.action_completed(completion_context: action_completion_context) # end end |
#display_user_error!(e, message) ⇒ Object
241 242 243 244 245 246 247 248 249 250 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 241 def display_user_error!(e, ) if PantographCore::Globals.verbose? # with stack trace reraise_formatted!(e, ) else # without stack trace action_completed(@program[:name], status: PantographCore::ActionCompletionStatus::USER_ERROR, exception: e) abort("\n[!] #{}".red) end end |
#handle_ssl_error!(e) ⇒ Object
164 165 166 167 168 169 170 171 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 164 def handle_ssl_error!(e) # SSL errors are very common when the Ruby or OpenSSL installation is somehow broken # We want to show a nice error message to the user here # We have over 20 GitHub issues just for this one error: # https://github.com/urbanquakers/pantograph/search?q=errno%3D0+state%3DSSLv3+read+server&type=Issues suggest_ruby_reinstall(e) display_user_error!(e, e.to_s) end |
#handle_tls_error!(e) ⇒ Object
157 158 159 160 161 162 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 157 def handle_tls_error!(e) # Apple system Ruby 2.0 does not support some things. # We want to suggest that users upgrade their Ruby version suggest_ruby_reinstall(e) display_user_error!(e, e.to_s) end |
#handle_unknown_error!(e) ⇒ Object
216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 216 def handle_unknown_error!(e) # Some spaceship exception classes implement #preferred_error_info in order to share error info # that we'd rather display instead of crashing with a stack trace. However, pantograph_core and # spaceship can not know about each other's classes! To make this information passing work, we # use a bit of Ruby duck-typing to check whether the unknown exception type implements the right # method. If so, we'll present any returned error info in the manner of a user_error! error_info = e.respond_to?(:preferred_error_info) ? e.preferred_error_info : nil should_show_github_issues = e.respond_to?(:show_github_issues) ? e.show_github_issues : true if error_info error_info = error_info.join("\n\t") if error_info.kind_of?(Array) show_github_issues(error_info) if should_show_github_issues display_user_error!(e, error_info) else # Pass the error instead of a message so that the inspector can do extra work to simplify the query show_github_issues(e) if should_show_github_issues # From https://stackoverflow.com/a/4789702/445598 # We do this to make the actual error message red and therefore more visible reraise_formatted!(e, e.) end end |
#reraise_formatted!(e, message) ⇒ Object
252 253 254 255 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 252 def reraise_formatted!(e, ) backtrace = PantographCore::Env.truthy?("PANTOGRAPH_HIDE_BACKTRACE") ? [] : e.backtrace raise e, "[!] #{}".red, backtrace end |
#rescue_connection_failed_error(e) ⇒ Object
136 137 138 139 140 141 142 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 136 def rescue_connection_failed_error(e) if e..include?('Connection reset by peer - SSL_connect') handle_tls_error!(e) else handle_unknown_error!(e) end end |
#rescue_file_error(e) ⇒ Object
127 128 129 130 131 132 133 134 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 127 def rescue_file_error(e) # We're also printing the new-lines, as otherwise the message is not very visible in-between the error and the stack trace puts("") PantographCore::UI.important("Error accessing file, this might be due to pantograph's directory handling") PantographCore::UI.important("Check out https://urbanquakers.github.io/pantograph/advanced/#directory-behavior for more details") puts("") raise e end |
#rescue_pantograph_error(e) ⇒ Object
150 151 152 153 154 155 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 150 def rescue_pantograph_error(e) action_completed(@program[:name], status: PantographCore::ActionCompletionStatus::USER_ERROR, exception: e) show_github_issues(e.) if e.show_github_issues display_user_error!(e, e.) end |
#rescue_unknown_error(e) ⇒ Object
144 145 146 147 148 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 144 def rescue_unknown_error(e) action_completed(@program[:name], status: PantographCore::ActionCompletionStatus::FAILED, exception: e) handle_unknown_error!(e) end |
#run! ⇒ Object
37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 37 def run! require_program(:version, :description) trap('INT') { abort(program(:int_message)) } if program(:int_message) trap('INT') { program(:int_block).call } if program(:int_block) global_option('-h', '--help', 'Display help documentation') do args = @args - %w(-h --help) command(:help).run(*args) return end global_option('-v', '--version', 'Display version information') do say(version) return end (, @args) begin pantograph_client_language = :ruby action_launch_context = PantographCore::ActionLaunchContext.context_for_action_name(@program[:name], pantograph_client_language: pantograph_client_language, args: ARGV) PantographCore.session.action_launched(launch_context: action_launch_context) return_value = run_active_command action_completed(@program[:name], status: PantographCore::ActionCompletionStatus::SUCCESS) return return_value rescue Commander::Runner::InvalidCommandError => e # calling `abort` makes it likely that tests stop without failing, so # we'll disable that during tests. if PantographCore::Helper.test? raise e else abort("#{e}. Use --help for more information") end rescue Interrupt => e # We catch it so that the stack trace is hidden by default when using ctrl + c if PantographCore::Globals.verbose? raise e else action_completed(@program[:name], status: PantographCore::ActionCompletionStatus::INTERRUPTED, exception: e) abort("\nCancelled... use --verbose to show the stack trace") end rescue \ OptionParser::InvalidOption, OptionParser::InvalidArgument, OptionParser::MissingArgument => e # calling `abort` makes it likely that tests stop without failing, so # we'll disable that during tests. if PantographCore::Helper.test? raise e else if self.active_command.name == "help" && @default_command == :help # need to access directly via @ # This is a special case, for example for pilot # when the user runs `pantograph pilot -u [email protected]` # This would be confusing, as the user probably wanted to use `pilot list` # or some other command. Because `-u` isn't available for the `pilot --help` # command it would show this very confusing error message otherwise abort("Please ensure to use one of the available commands (#{self.commands.keys.join(', ')})".red) else # This would print something like # # invalid option: -u # abort(e.to_s) end end rescue PantographCore::Interface::PantographCommonException => e # these are exceptions that we dont count as crashes display_user_error!(e, e.to_s) rescue PantographCore::Interface::PantographError => e # user_error! rescue_pantograph_error(e) rescue Errno::ENOENT => e rescue_file_error(e) rescue Faraday::SSLError, OpenSSL::SSL::SSLError => e # SSL issues are very common handle_ssl_error!(e) rescue Faraday::ConnectionFailed => e rescue_connection_failed_error(e) rescue => e # high chance this is actually PantographCore::Interface::PantographCrash, but can be anything else rescue_unknown_error(e) ensure PantographCore.session.finalize_session end end |
#show_github_issues(message_or_error) ⇒ Object
257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 257 def show_github_issues() return if PantographCore::Env.truthy?("PANTOGRAPH_HIDE_GITHUB_ISSUES") return if PantographCore::Helper.test? require 'gh_inspector' require 'pantograph_core/ui/github_issue_inspector_reporter' inspector = GhInspector::Inspector.new("pantograph", "pantograph", verbose: PantographCore::Globals.verbose?) delegate = Pantograph::InspectorReporter.new if .kind_of?(String) inspector.search_query(, delegate) else inspector.search_exception(, delegate) end rescue => ex PantographCore::UI.error("Error finding relevant GitHub issues: #{ex}") if PantographCore::Globals.verbose? end |
#suggest_ruby_reinstall(e) ⇒ Object
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 |
# File 'pantograph_core/lib/pantograph_core/ui/pantograph_runner.rb', line 173 def suggest_ruby_reinstall(e) ui = PantographCore::UI ui.error("-----------------------------------------------------------------------") ui.error(e.to_s) ui.error("") ui.error("SSL errors can be caused by various components on your local machine.") if Gem::Version.new(RUBY_VERSION) < Gem::Version.new('2.1') ui.error("AApple system Ruby 2.0 does not support some things, which may") ui.error("not be available to your system installed Ruby (#{RUBY_VERSION})") end ui.error("") ui.error("The best solution is to use the self-contained pantograph version.") ui.error("Which ships with a bundled OpenSSL,ruby and all gems - so you don't depend on system libraries") ui.error(" - Use Homebrew") ui.error(" - update brew with `brew update`") ui.error(" - install pantograph using:") ui.error(" - `brew cask install pantograph`") ui.error(" - Use One-Click-Installer:") ui.error(" - download pantograph at https://download.pantograph.tools") ui.error(" - extract the archive and double click the `install`") ui.error("-----------------------------------------------------------") ui.error("for more details on ways to install pantograph please refer the documentation:") ui.error("-----------------------------------------------------------") ui.error(" 🚀 https://urbanquakers.github.io/pantograph 🚀 ") ui.error("-----------------------------------------------------------") ui.error("") ui.error("You can also install a new version of Ruby") ui.error("") ui.error("- Make sure OpenSSL is installed with Homebrew: `brew update && brew upgrade openssl`") ui.error("- If you use system Ruby:") ui.error(" - Run `brew update && brew install ruby`") ui.error("- If you use rbenv with ruby-build:") ui.error(" - Run `brew update && brew upgrade ruby-build && rbenv install 2.3.1`") ui.error(" - Run `rbenv global 2.3.1` to make it the new global default Ruby version") ui.error("- If you use rvm:") ui.error(" - First run `rvm osx-ssl-certs update all`") ui.error(" - Then run `rvm reinstall ruby-2.3.1 --with-openssl-dir=/usr/local`") ui.error("") ui.error("If that doesn't fix your issue, please google for the following error message:") ui.error(" '#{e}'") ui.error("-----------------------------------------------------------------------") end |