Class: Commander::Runner
- Inherits:
-
Object
- Object
- Commander::Runner
- Defined in:
- fastlane_core/lib/fastlane_core/ui/fastlane_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_fastlane_error(e) ⇒ Object
- #rescue_file_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 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 35 def collector @collector end |
Instance Method Details
#action_completed(action_name, status: nil, exception: nil) ⇒ Object
137 138 139 140 141 142 143 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 137 def action_completed(action_name, status: nil, exception: nil) # https://github.com/fastlane/fastlane/issues/11913 # if exception.nil? || exception.fastlane_should_report_metrics? # action_completion_context = FastlaneCore::ActionCompletionContext.context_for_action_name(action_name, args: ARGV, status: status) # FastlaneCore.session.action_completed(completion_context: action_completion_context) # end end |
#display_user_error!(e, message) ⇒ Object
260 261 262 263 264 265 266 267 268 269 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 260 def display_user_error!(e, ) if FastlaneCore::Globals.verbose? # with stack trace reraise_formatted!(e, ) else # without stack trace action_completed(@program[:name], status: FastlaneCore::ActionCompletionStatus::USER_ERROR, exception: e) abort("\n[!] #{}".red) end end |
#handle_ssl_error!(e) ⇒ Object
183 184 185 186 187 188 189 190 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 183 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/fastlane/fastlane/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
175 176 177 178 179 180 181 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 175 def handle_tls_error!(e) # Apple has upgraded its App Store Connect servers to require TLS 1.2, but # system Ruby 2.0 does not support it. 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
235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 235 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, fastlane_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
271 272 273 274 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 271 def reraise_formatted!(e, ) backtrace = FastlaneCore::Env.truthy?("FASTLANE_HIDE_BACKTRACE") ? [] : e.backtrace raise e, "[!] #{}".red, backtrace end |
#rescue_connection_failed_error(e) ⇒ Object
154 155 156 157 158 159 160 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 154 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_fastlane_error(e) ⇒ Object
168 169 170 171 172 173 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 168 def rescue_fastlane_error(e) action_completed(@program[:name], status: FastlaneCore::ActionCompletionStatus::USER_ERROR, exception: e) show_github_issues(e.) if e.show_github_issues display_user_error!(e, e.) end |
#rescue_file_error(e) ⇒ Object
145 146 147 148 149 150 151 152 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 145 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("") FastlaneCore::UI.important("Error accessing file, this might be due to fastlane's directory handling") FastlaneCore::UI.important("Check out https://docs.fastlane.tools/advanced/#directory-behavior for more details") puts("") raise e end |
#rescue_unknown_error(e) ⇒ Object
162 163 164 165 166 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 162 def rescue_unknown_error(e) action_completed(@program[:name], status: FastlaneCore::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 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_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) xcode_outdated = false begin unless FastlaneCore::Helper.xcode_at_least?(Fastlane::MINIMUM_XCODE_RELEASE) xcode_outdated = true end rescue # We don't care about exceptions here # We'll land here if the user doesn't have Xcode at all for example # which is fine for someone who uses fastlane just for Android project # What we *do* care about is when someone links an old version of Xcode end begin if xcode_outdated # We have to raise that error within this `begin` block to show a nice user error without a stack trace FastlaneCore::UI.user_error!("fastlane requires a minimum version of Xcode #{Fastlane::MINIMUM_XCODE_RELEASE}, please upgrade and make sure to use `sudo xcode-select -s /Applications/Xcode.app`") end is_swift = FastlaneCore::FastlaneFolder.swift? fastlane_client_language = is_swift ? :swift : :ruby action_launch_context = FastlaneCore::ActionLaunchContext.context_for_action_name(@program[:name], fastlane_client_language: fastlane_client_language, args: ARGV) FastlaneCore.session.action_launched(launch_context: action_launch_context) return_value = run_active_command action_completed(@program[:name], status: FastlaneCore::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 FastlaneCore::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 FastlaneCore::Globals.verbose? raise e else action_completed(@program[:name], status: FastlaneCore::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 FastlaneCore::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 `fastlane 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 FastlaneCore::Interface::FastlaneCommonException => e # these are exceptions that we don't count as crashes display_user_error!(e, e.to_s) rescue FastlaneCore::Interface::FastlaneError => e # user_error! rescue_fastlane_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 FastlaneCore::Interface::FastlaneCrash, but can be anything else rescue_unknown_error(e) ensure FastlaneCore.session.finalize_session end end |
#show_github_issues(message_or_error) ⇒ Object
276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 276 def show_github_issues() return if FastlaneCore::Env.truthy?("FASTLANE_HIDE_GITHUB_ISSUES") return if FastlaneCore::Helper.test? require 'gh_inspector' require 'fastlane_core/ui/github_issue_inspector_reporter' inspector = GhInspector::Inspector.new("fastlane", "fastlane", verbose: FastlaneCore::Globals.verbose?) delegate = Fastlane::InspectorReporter.new if .kind_of?(String) inspector.search_query(, delegate) else inspector.search_exception(, delegate) end rescue => ex FastlaneCore::UI.error("Error finding relevant GitHub issues: #{ex}") if FastlaneCore::Globals.verbose? end |
#suggest_ruby_reinstall(e) ⇒ Object
192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 192 def suggest_ruby_reinstall(e) ui = FastlaneCore::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("Apple has recently changed their servers to require TLS 1.2, 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 fastlane 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 fastlane using:") ui.error(" - `brew cask install fastlane`") ui.error(" - Use One-Click-Installer:") ui.error(" - download fastlane at https://download.fastlane.tools") ui.error(" - extract the archive and double click the `install`") ui.error("-----------------------------------------------------------") ui.error("for more details on ways to install fastlane please refer the documentation:") ui.error("-----------------------------------------------------------") ui.error(" 🚀 https://docs.fastlane.tools 🚀 ") 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 |