From e3536097012b57f8d645947d2bdb581711b74b9e Mon Sep 17 00:00:00 2001 From: Joshua Liebowitz Date: Fri, 20 Apr 2018 08:33:34 -0700 Subject: [PATCH 1/3] Handle connection error when github can't be found Prevent scheduler from getting into bad state when exception happens --- app/shared/github_handler.rb | 10 +++++++++- app/workers/worker_base.rb | 32 ++++++++++++++++++++++++-------- 2 files changed, 33 insertions(+), 9 deletions(-) diff --git a/app/shared/github_handler.rb b/app/shared/github_handler.rb index e8bc4a1a..8f319994 100644 --- a/app/shared/github_handler.rb +++ b/app/shared/github_handler.rb @@ -1,4 +1,5 @@ require "octokit" +require "faraday" require_relative "logging_module" # TODO: eventually we'll want to consider handling all these: @@ -66,7 +67,14 @@ def github_action(client, &block) sleep(sleep_length) retry end - + raise ex + rescue Octokit::TooManyRequests => ex + logger.error(ex) + raise ex + rescue Octokit::Unauthorized => ex # Maybe the token does not give access to rate limits. + logger.error(ex) + rescue Faraday::ConnectionFailed => ex + logger.error("Some GitHub API seems to be down right now, got a connection failure") raise ex end end diff --git a/app/workers/worker_base.rb b/app/workers/worker_base.rb index 78c75f25..9c88bf52 100644 --- a/app/workers/worker_base.rb +++ b/app/workers/worker_base.rb @@ -17,23 +17,39 @@ def thread_id end def initialize + # TODO: do we need a thread here to do the work or does `scheduler.schedule` handle that? @thread = Thread.new do begin # We have the `work` inside a `begin rescue` # so that if something fails, the thread still is alive scheduler.schedule do - work - # If we're running in debug mode, don't run these things continuously - if ENV["FASTLANE_CI_THREAD_DEBUG_MODE"] - logger.debug("Stopping worker after this work unit") - die! + begin + # This can cause an exception, in production mode, we don't re-raise the exception + # in development mode, we re-raise so we can catch it and understand how to handle it + work + # If we're running in debug mode, don't run these things continuously + if ENV["FASTLANE_CI_THREAD_DEBUG_MODE"] + logger.debug("Stopping worker after this work unit") + die! + end + rescue StandardError => ex + logger.error("[#{self.class} Exception], work unit caused exception: #{ex}: ") + logger.error(ex.backtrace.join("\n")) + if Thread.abort_on_exception == true + logger.error("[#{self.class}] Thread.abort_on_exception is `true`, killing task re-raising exception") + die! + raise ex + end end end rescue StandardError => ex - logger.error("[#{self.class} Exception]: #{ex}: ") + logger.error("Worker scheduler had a problem") logger.error(ex.backtrace.join("\n")) - logger.error("[#{self.class}] Killing thread #{thread_id} due to exception\n") - die! + if Thread.abort_on_exception == true + logger.error("[#{self.class}] Thread.abort_on_exception is `true`, killing task and re-raising exception") + die! + raise ex + end end end end From 2e2b56cddf87c5b9a4cb50fc2644956c0274b99b Mon Sep 17 00:00:00 2001 From: Joshua Liebowitz Date: Fri, 20 Apr 2018 09:30:31 -0700 Subject: [PATCH 2/3] update exception handling for github actions --- app/shared/github_handler.rb | 29 +++++++++-------------------- 1 file changed, 9 insertions(+), 20 deletions(-) diff --git a/app/shared/github_handler.rb b/app/shared/github_handler.rb index 8f319994..5aae28fb 100644 --- a/app/shared/github_handler.rb +++ b/app/shared/github_handler.rb @@ -41,25 +41,19 @@ def self.included(klass) end def github_action(client, &block) - # `retry` retains the variables through iterations so we assign to 0 the first time. - retry_count ||= 0 if client.kind_of?(Octokit::Client) - begin - if client.rate_limit!.remaining.zero? - sleep_time = client.rate_limit!.resets_in - logger.debug("Rate Limit exceeded, sleeping for #{sleep_time} seconds") - sleep(sleep_time) - end - rescue Octokit::TooManyRequests => ex - logger.error(ex) - raise ex - rescue Octokit::Unauthorized => ex # Maybe the token does not give access to rate limits. - logger.error(ex) + if client.rate_limit!.remaining.zero? + sleep_time = client.rate_limit!.resets_in + logger.debug("Rate Limit exceeded, sleeping for #{sleep_time} seconds") + sleep(sleep_time) end end + + # `retry` retains the variables through iterations so we assign to 0 the first time. + retry_count ||= 0 begin return block.call(client) - rescue Octokit::ServerError => ex + rescue Octokit::ServerError, Octokit::TooManyRequests, Faraday::ConnectionFailed => ex if (retry_count += 1) < 5 # exponential backoff sleep_length = 2**retry_count @@ -68,14 +62,9 @@ def github_action(client, &block) retry end raise ex - rescue Octokit::TooManyRequests => ex - logger.error(ex) - raise ex rescue Octokit::Unauthorized => ex # Maybe the token does not give access to rate limits. + logger.error("Your GitHub Personal Auth Token is unauthorized to perform the github action") logger.error(ex) - rescue Faraday::ConnectionFailed => ex - logger.error("Some GitHub API seems to be down right now, got a connection failure") - raise ex end end end From 92754a35b40734d5bcd18e5ac9693e3131a2dfa3 Mon Sep 17 00:00:00 2001 From: Joshua Liebowitz Date: Fri, 20 Apr 2018 09:44:31 -0700 Subject: [PATCH 3/3] add some extra exception handling --- app/shared/github_handler.rb | 40 +++++++++++++++++++++++++++++++----- 1 file changed, 35 insertions(+), 5 deletions(-) diff --git a/app/shared/github_handler.rb b/app/shared/github_handler.rb index 5aae28fb..11dc23d6 100644 --- a/app/shared/github_handler.rb +++ b/app/shared/github_handler.rb @@ -42,14 +42,37 @@ def self.included(klass) def github_action(client, &block) if client.kind_of?(Octokit::Client) - if client.rate_limit!.remaining.zero? - sleep_time = client.rate_limit!.resets_in - logger.debug("Rate Limit exceeded, sleeping for #{sleep_time} seconds") - sleep(sleep_time) + # `rate_limit_retry_count` retains the variables through iterations so we assign to 0 the first time. + rate_limit_retry_count ||= 0 + begin + if client.rate_limit!.remaining.zero? + rate_limit_reset_time_length = client.rate_limit!.resets_in + logger.debug("Rate Limit exceeded, sleeping for #{rate_limit_reset_time_length} seconds") + sleep(rate_limit_reset_time_length) + end + rescue Octokit::Unauthorized => ex # Maybe the token does not give access to rate limits. + logger.error("Your GitHub Personal Auth Token is not unauthorized to check the rate_limit") + logger.error(ex) + # We want to die now, since this is a server config issue + # Ultimately, this shouldn't kill the server, but rather, send a notification + # TODO: accomplish the above ^ + raise ex + rescue Octokit::ServerError, Octokit::TooManyRequests, Faraday::ConnectionFailed => ex + if (rate_limit_retry_count += 1) < 5 + rate_limit_sleep_length = 2**rate_limit_retry_count + logger.debug("Unable to get rate limit, sleeping for #{rate_limit_sleep_length} seconds and retrying") + logger.debug(ex) + sleep(rate_limit_sleep_length) + retry + end + logger.debug("Unable to get rate limit after retrying multiple time, failing") + # Ultimately, this shouldn't kill the server, but rather, send a notification + # TODO: accomplish the above ^ + raise ex end end - # `retry` retains the variables through iterations so we assign to 0 the first time. + # `retry_count` retains the variables through iterations so we assign to 0 the first time. retry_count ||= 0 begin return block.call(client) @@ -58,13 +81,20 @@ def github_action(client, &block) # exponential backoff sleep_length = 2**retry_count logger.debug("A GitHub action failed, sleeping for #{sleep_length} seconds and retrying") + logger.debug(ex) sleep(sleep_length) retry end + logger.debug("Unable to perform GitHub action after retrying multiple time, failing") + # Ultimately, this shouldn't kill the server, but rather, send a notification + # TODO: accomplish the above ^ raise ex rescue Octokit::Unauthorized => ex # Maybe the token does not give access to rate limits. logger.error("Your GitHub Personal Auth Token is unauthorized to perform the github action") logger.error(ex) + # Ultimately, this shouldn't kill the server, but rather, send a notification + # TODO: accomplish the above ^ + raise ex end end end