Class: Bundler::Retry
| Relationships & Source Files | |
| Inherits: | Object |
| Defined in: | lib/bundler/retry.rb |
Overview
General purpose class for retrying code that may fail
Class Attribute Summary
Class Method Summary
-
.attempts
Alias for .default_attempts.
- .default_attempts (also: .attempts)
- .default_retries
- .new(name, exceptions = nil, retries = self.class.default_retries, opts = {}) ⇒ Retry constructor
Instance Attribute Summary
- #current_run rw
- #name rw
- #total_runs rw
- #keep_trying? ⇒ Boolean readonly private
- #last_attempt? ⇒ Boolean readonly private
Instance Method Summary
- #attempt(&block) (also: #attempts)
-
#attempts(&block)
Alias for #attempt.
- #backoff_sleep private
- #fail_attempt(e) private
- #run(&block) private
- #sleep(duration) private
Constructor Details
.new(name, exceptions = nil, retries = self.class.default_retries, opts = {}) ⇒ Retry
# File 'lib/bundler/retry.rb', line 24
def initialize(name, exceptions = nil, retries = self.class.default_retries, opts = {}) @name = name @retries = retries @exceptions = Array(exceptions) || [] @total_runs = @retries + 1 # will run once, then upto attempts.times @base_delay = opts[:base_delay] || self.class.default_base_delay @max_delay = opts[:max_delay] || 60.0 @jitter = opts[:jitter] || 0.5 end
Class Attribute Details
.default_base_delay (rw)
[ GitHub ]# File 'lib/bundler/retry.rb', line 9
attr_accessor :default_base_delay
Class Method Details
.attempts
Alias for .default_attempts.
# File 'lib/bundler/retry.rb', line 14
alias_method :attempts, :default_attempts
.default_attempts Also known as: .attempts
[ GitHub ]# File 'lib/bundler/retry.rb', line 11
def default_attempts default_retries + 1 end
.default_retries
[ GitHub ]Instance Attribute Details
#current_run (rw)
[ GitHub ]# File 'lib/bundler/retry.rb', line 6
attr_accessor :name, :total_runs, :current_run
#keep_trying? ⇒ Boolean (readonly, private)
[ GitHub ]
# File 'lib/bundler/retry.rb', line 82
def return true if current_run.zero? return false if last_attempt? true if @failed end
#last_attempt? ⇒ Boolean (readonly, private)
[ GitHub ]
# File 'lib/bundler/retry.rb', line 88
def last_attempt? current_run >= total_runs end
#name (rw)
[ GitHub ]# File 'lib/bundler/retry.rb', line 6
attr_accessor :name, :total_runs, :current_run
#total_runs (rw)
[ GitHub ]# File 'lib/bundler/retry.rb', line 6
attr_accessor :name, :total_runs, :current_run
Instance Method Details
#attempt(&block) Also known as: #attempts
[ GitHub ]# File 'lib/bundler/retry.rb', line 34
def attempt(&block) @current_run = 0 @failed = false @error = nil run(&block) while @result end
#attempts(&block)
Alias for #attempt.
# File 'lib/bundler/retry.rb', line 41
alias_method :attempts, :attempt
#backoff_sleep (private)
[ GitHub ]# File 'lib/bundler/retry.rb', line 67
def backoff_sleep # Exponential backoff: delay = base_delay * 2^(attempt - 1) # Add jitter to prevent thundering herd: random value between 0 and jitter seconds delay = @base_delay * (2**(@current_run - 1)) delay = [@max_delay, delay].min jitter_amount = rand * @jitter total_delay = delay + jitter_amount Bundler.ui.debug "Sleeping for #{total_delay.round(2)} seconds before retry" sleep(total_delay) end
#fail_attempt(e) (private)
[ GitHub ]# File 'lib/bundler/retry.rb', line 53
def fail_attempt(e) @failed = true if last_attempt? || @exceptions.any? {|k| e.is_a?(k) } Bundler.ui.info "" unless Bundler.ui.debug? raise e end if name Bundler.ui.info "" unless Bundler.ui.debug? # Add new line in case dots preceded this Bundler.ui.warn "Retrying #{name} due to error (#{current_run.next}/#{total_runs}): #{e.class} #{e.}", true end backoff_sleep if @base_delay > 0 true end
#run(&block) (private)
[ GitHub ]# File 'lib/bundler/retry.rb', line 45
def run(&block) @failed = false @current_run += 1 @result = block.call rescue StandardError => e fail_attempt(e) end
#sleep(duration) (private)
[ GitHub ]# File 'lib/bundler/retry.rb', line 78
def sleep(duration) Kernel.sleep(duration) end