You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

227 lines
8.8 KiB

4 years ago
  1. <!-- badges/ -->
  2. [![Build Status](https://secure.travis-ci.org/tim-kos/node-retry.png?branch=master)](http://travis-ci.org/tim-kos/node-retry "Check this project's build status on TravisCI")
  3. [![codecov](https://codecov.io/gh/tim-kos/node-retry/branch/master/graph/badge.svg)](https://codecov.io/gh/tim-kos/node-retry)
  4. <!-- /badges -->
  5. # retry
  6. Abstraction for exponential and custom retry strategies for failed operations.
  7. ## Installation
  8. npm install retry
  9. ## Current Status
  10. This module has been tested and is ready to be used.
  11. ## Tutorial
  12. The example below will retry a potentially failing `dns.resolve` operation
  13. `10` times using an exponential backoff strategy. With the default settings, this
  14. means the last attempt is made after `17 minutes and 3 seconds`.
  15. ``` javascript
  16. var dns = require('dns');
  17. var retry = require('retry');
  18. function faultTolerantResolve(address, cb) {
  19. var operation = retry.operation();
  20. operation.attempt(function(currentAttempt) {
  21. dns.resolve(address, function(err, addresses) {
  22. if (operation.retry(err)) {
  23. return;
  24. }
  25. cb(err ? operation.mainError() : null, addresses);
  26. });
  27. });
  28. }
  29. faultTolerantResolve('nodejs.org', function(err, addresses) {
  30. console.log(err, addresses);
  31. });
  32. ```
  33. Of course you can also configure the factors that go into the exponential
  34. backoff. See the API documentation below for all available settings.
  35. currentAttempt is an int representing the number of attempts so far.
  36. ``` javascript
  37. var operation = retry.operation({
  38. retries: 5,
  39. factor: 3,
  40. minTimeout: 1 * 1000,
  41. maxTimeout: 60 * 1000,
  42. randomize: true,
  43. });
  44. ```
  45. ## API
  46. ### retry.operation([options])
  47. Creates a new `RetryOperation` object. `options` is the same as `retry.timeouts()`'s `options`, with two additions:
  48. * `forever`: Whether to retry forever, defaults to `false`.
  49. * `unref`: Whether to [unref](https://nodejs.org/api/timers.html#timers_unref) the setTimeout's, defaults to `false`.
  50. * `maxRetryTime`: The maximum time (in milliseconds) that the retried operation is allowed to run. Default is `Infinity`.
  51. ### retry.timeouts([options])
  52. Returns an array of timeouts. All time `options` and return values are in
  53. milliseconds. If `options` is an array, a copy of that array is returned.
  54. `options` is a JS object that can contain any of the following keys:
  55. * `retries`: The maximum amount of times to retry the operation. Default is `10`. Seting this to `1` means `do it once, then retry it once`.
  56. * `factor`: The exponential factor to use. Default is `2`.
  57. * `minTimeout`: The number of milliseconds before starting the first retry. Default is `1000`.
  58. * `maxTimeout`: The maximum number of milliseconds between two retries. Default is `Infinity`.
  59. * `randomize`: Randomizes the timeouts by multiplying with a factor between `1` to `2`. Default is `false`.
  60. The formula used to calculate the individual timeouts is:
  61. ```
  62. Math.min(random * minTimeout * Math.pow(factor, attempt), maxTimeout)
  63. ```
  64. Have a look at [this article][article] for a better explanation of approach.
  65. If you want to tune your `factor` / `times` settings to attempt the last retry
  66. after a certain amount of time, you can use wolfram alpha. For example in order
  67. to tune for `10` attempts in `5 minutes`, you can use this equation:
  68. ![screenshot](https://github.com/tim-kos/node-retry/raw/master/equation.gif)
  69. Explaining the various values from left to right:
  70. * `k = 0 ... 9`: The `retries` value (10)
  71. * `1000`: The `minTimeout` value in ms (1000)
  72. * `x^k`: No need to change this, `x` will be your resulting factor
  73. * `5 * 60 * 1000`: The desired total amount of time for retrying in ms (5 minutes)
  74. To make this a little easier for you, use wolfram alpha to do the calculations:
  75. <http://www.wolframalpha.com/input/?i=Sum%5B1000*x^k%2C+{k%2C+0%2C+9}%5D+%3D+5+*+60+*+1000>
  76. [article]: http://dthain.blogspot.com/2009/02/exponential-backoff-in-distributed.html
  77. ### retry.createTimeout(attempt, opts)
  78. Returns a new `timeout` (integer in milliseconds) based on the given parameters.
  79. `attempt` is an integer representing for which retry the timeout should be calculated. If your retry operation was executed 4 times you had one attempt and 3 retries. If you then want to calculate a new timeout, you should set `attempt` to 4 (attempts are zero-indexed).
  80. `opts` can include `factor`, `minTimeout`, `randomize` (boolean) and `maxTimeout`. They are documented above.
  81. `retry.createTimeout()` is used internally by `retry.timeouts()` and is public for you to be able to create your own timeouts for reinserting an item, see [issue #13](https://github.com/tim-kos/node-retry/issues/13).
  82. ### retry.wrap(obj, [options], [methodNames])
  83. Wrap all functions of the `obj` with retry. Optionally you can pass operation options and
  84. an array of method names which need to be wrapped.
  85. ```
  86. retry.wrap(obj)
  87. retry.wrap(obj, ['method1', 'method2'])
  88. retry.wrap(obj, {retries: 3})
  89. retry.wrap(obj, {retries: 3}, ['method1', 'method2'])
  90. ```
  91. The `options` object can take any options that the usual call to `retry.operation` can take.
  92. ### new RetryOperation(timeouts, [options])
  93. Creates a new `RetryOperation` where `timeouts` is an array where each value is
  94. a timeout given in milliseconds.
  95. Available options:
  96. * `forever`: Whether to retry forever, defaults to `false`.
  97. * `unref`: Wether to [unref](https://nodejs.org/api/timers.html#timers_unref) the setTimeout's, defaults to `false`.
  98. If `forever` is true, the following changes happen:
  99. * `RetryOperation.errors()` will only output an array of one item: the last error.
  100. * `RetryOperation` will repeatedly use the `timeouts` array. Once all of its timeouts have been used up, it restarts with the first timeout, then uses the second and so on.
  101. #### retryOperation.errors()
  102. Returns an array of all errors that have been passed to `retryOperation.retry()` so far. The
  103. returning array has the errors ordered chronologically based on when they were passed to
  104. `retryOperation.retry()`, which means the first passed error is at index zero and the last is
  105. at the last index.
  106. #### retryOperation.mainError()
  107. A reference to the error object that occured most frequently. Errors are
  108. compared using the `error.message` property.
  109. If multiple error messages occured the same amount of time, the last error
  110. object with that message is returned.
  111. If no errors occured so far, the value is `null`.
  112. #### retryOperation.attempt(fn, timeoutOps)
  113. Defines the function `fn` that is to be retried and executes it for the first
  114. time right away. The `fn` function can receive an optional `currentAttempt` callback that represents the number of attempts to execute `fn` so far.
  115. Optionally defines `timeoutOps` which is an object having a property `timeout` in miliseconds and a property `cb` callback function.
  116. Whenever your retry operation takes longer than `timeout` to execute, the timeout callback function `cb` is called.
  117. #### retryOperation.try(fn)
  118. This is an alias for `retryOperation.attempt(fn)`. This is deprecated. Please use `retryOperation.attempt(fn)` instead.
  119. #### retryOperation.start(fn)
  120. This is an alias for `retryOperation.attempt(fn)`. This is deprecated. Please use `retryOperation.attempt(fn)` instead.
  121. #### retryOperation.retry(error)
  122. Returns `false` when no `error` value is given, or the maximum amount of retries
  123. has been reached.
  124. Otherwise it returns `true`, and retries the operation after the timeout for
  125. the current attempt number.
  126. #### retryOperation.stop()
  127. Allows you to stop the operation being retried. Useful for aborting the operation on a fatal error etc.
  128. #### retryOperation.reset()
  129. Resets the internal state of the operation object, so that you can call `attempt()` again as if this was a new operation object.
  130. #### retryOperation.attempts()
  131. Returns an int representing the number of attempts it took to call `fn` before it was successful.
  132. ## License
  133. retry is licensed under the MIT license.
  134. # Changelog
  135. 0.10.0 Adding `stop` functionality, thanks to @maxnachlinger.
  136. 0.9.0 Adding `unref` functionality, thanks to @satazor.
  137. 0.8.0 Implementing retry.wrap.
  138. 0.7.0 Some bug fixes and made retry.createTimeout() public. Fixed issues [#10](https://github.com/tim-kos/node-retry/issues/10), [#12](https://github.com/tim-kos/node-retry/issues/12), and [#13](https://github.com/tim-kos/node-retry/issues/13).
  139. 0.6.0 Introduced optional timeOps parameter for the attempt() function which is an object having a property timeout in milliseconds and a property cb callback function. Whenever your retry operation takes longer than timeout to execute, the timeout callback function cb is called.
  140. 0.5.0 Some minor refactoring.
  141. 0.4.0 Changed retryOperation.try() to retryOperation.attempt(). Deprecated the aliases start() and try() for it.
  142. 0.3.0 Added retryOperation.start() which is an alias for retryOperation.try().
  143. 0.2.0 Added attempts() function and parameter to retryOperation.try() representing the number of attempts it took to call fn().