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
-
#parse_global_options ⇒ Object
Temporary workaround for issues mentioned in github.com/fastlane/fastlane/pull/18760 Code taken from github.com/commander-rb/commander/blob/40d06bfbc54906d0de7c72ac73f4e9188c9ca294/lib/commander/runner.rb#L372-L385.
- #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
185 186 187 188 189 190 191 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 185 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
304 305 306 307 308 309 310 311 312 313 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 304 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
231 232 233 234 235 236 237 238 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 231 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
223 224 225 226 227 228 229 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 223 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
279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 279 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 cannot 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 |
#parse_global_options ⇒ Object
Temporary workaround for issues mentioned in github.com/fastlane/fastlane/pull/18760 Code taken from github.com/commander-rb/commander/blob/40d06bfbc54906d0de7c72ac73f4e9188c9ca294/lib/commander/runner.rb#L372-L385
Problem:
`optparse` is guessing that command option `-e` is referring to global option `--env` (because it starts with an e).
This is raising OptionParser::MissingArgument error because `--env` takes a string argument.
A command of `-e --verbose` works because `--verbose` is seen as the argument.
A command of `--verbose -e` doesn't work because no argument after `-e` so MissingArgument is raised again.
This broke somewhere between Ruby 2.5 and Ruby 2.6
Solution:
Proper solution is to set `parser.require_exact = true` but this only available on `optparse` version 0.1.1
which is not used by Commander.
`require_exact` will prevent OptionParser from assuming `-e` is the same as `--env STRING`
Even if it was on RubyGems, it would require Commander to allow this option to be set on OptionParser
This work around was made on 2021-08-13
When fixed:
This method implementation overrides one provided by Commander::Runner already. Just delete this method
so the existing one can be used
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 58 def parser = .inject(OptionParser.new) do |, option| .on(*option[:args], &global_option_proc(option[:switches], &option[:proc])) end # This is the actual solution but is only in version 0.1.1 of optparse and its not in Commander # This is the only change from Commanders implementation of parse_global_options parser.require_exact = true = @args.dup begin parser.parse!() rescue OptionParser::InvalidOption => e # Remove the offending args and retry. = .reject { |o| e.args.include?(o) } retry end end |
#reraise_formatted!(e, message) ⇒ Object
315 316 317 318 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 315 def reraise_formatted!(e, ) backtrace = FastlaneCore::Env.truthy?("FASTLANE_HIDE_BACKTRACE") ? [] : e.backtrace raise e, "[!] #{}".red, backtrace end |
#rescue_connection_failed_error(e) ⇒ Object
202 203 204 205 206 207 208 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 202 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
216 217 218 219 220 221 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 216 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
193 194 195 196 197 198 199 200 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 193 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
210 211 212 213 214 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 210 def rescue_unknown_error(e) action_completed(@program[:name], status: FastlaneCore::ActionCompletionStatus::FAILED, exception: e) handle_unknown_error!(e) end |
#run! ⇒ Object
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 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 77 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) # Trainer has been added to fastlane as of 2.201.0 # We need to make sure that the trainer fastlane plugin is no longer installed # to avoid any clashes if Gem::Specification.any? { |s| (s.name == 'fastlane-plugin-trainer') && Gem::Requirement.default =~ (s.version) } FastlaneCore::UI.user_error!("Migration Needed: As of 2.201.0, trainer is included in fastlane. Please remove the trainer plugin from your Gemfile or Pluginfile (or with 'gem uninstall fastlane-plugin-trainer') - More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/") end 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
320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 320 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
240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 |
# File 'fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb', line 240 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.") 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 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 |