Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/michaelbull/kotlin-retry
A multiplatform higher-order function for retrying operations that may fail.
https://github.com/michaelbull/kotlin-retry
backoff functional-programming high-order-function io jitter kotlin kotlin-multiplatform retry
Last synced: 7 days ago
JSON representation
A multiplatform higher-order function for retrying operations that may fail.
- Host: GitHub
- URL: https://github.com/michaelbull/kotlin-retry
- Owner: michaelbull
- License: isc
- Created: 2019-08-24T18:56:32.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2024-04-20T16:18:24.000Z (9 months ago)
- Last Synced: 2024-12-29T01:20:58.383Z (14 days ago)
- Topics: backoff, functional-programming, high-order-function, io, jitter, kotlin, kotlin-multiplatform, retry
- Language: Kotlin
- Homepage:
- Size: 260 KB
- Stars: 346
- Watchers: 4
- Forks: 9
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-ccamel - michaelbull/kotlin-retry - A multiplatform higher-order function for retrying operations that may fail. (Kotlin)
README
# kotlin-retry
[![Maven Central](https://img.shields.io/maven-central/v/com.michael-bull.kotlin-retry/kotlin-retry.svg)](https://search.maven.org/search?q=g:com.michael-bull.kotlin-retry)
[![CI](https://github.com/michaelbull/kotlin-retry/actions/workflows/ci.yaml/badge.svg)](https://github.com/michaelbull/kotlin-retry/actions/workflows/ci.yaml)
[![License](https://img.shields.io/github/license/michaelbull/kotlin-retry.svg)](https://github.com/michaelbull/kotlin-retry/blob/master/LICENSE)![badge][badge-android]
![badge][badge-jvm]
![badge][badge-js]
![badge][badge-nodejs]
![badge][badge-linux]
![badge][badge-windows]
![badge][badge-wasm]
![badge][badge-ios]
![badge][badge-mac]
![badge][badge-tvos]
![badge][badge-watchos]
![badge][badge-js-ir]
![badge][badge-android-native]
![badge][badge-apple-silicon]A multiplatform higher-order function for retrying operations that may fail.
```kotlin
val everySecondTenTimes = constantDelay(delayMillis = 1000L) + stopAtAttempts(10)retry(everySecondTenTimes) {
/* your code */
}
```## Installation
```groovy
repositories {
mavenCentral()
}dependencies {
implementation("com.michael-bull.kotlin-retry:kotlin-retry:2.0.1")
}
```## Introduction
IO operations often experience temporary failures that warrant re-execution,
e.g. a database transaction that may fail due to a deadlock.[1][innodb-deadlocks][2][postgres-deadlocks]> _“even if your application logic is correct, you must still handle the case
> where a transaction must be retried”_
>
> — _[Deadlocks in InnoDB][innodb-deadlocks]_The [`retry`][retry] function simplifies this process by wrapping the
application logic and applying a specified [`RetryPolicy`][retry-policy].In the example below, either of the calls to `customers.nameFromId` may fail,
abandoning the remaining logic within the `printExchangeBetween` function. As
such, we may want to retry this operation until 5 invocations in total have been
executed:```kotlin
import com.github.michaelbull.retry.policy.stopAtAttempts
import com.github.michaelbull.retry.retry
import kotlinx.coroutines.runBlockingsuspend fun printExchangeBetween(a: Long, b: Long) {
val customer1 = customers.nameFromId(a)
val customer2 = customers.nameFromId(b)
println("$customer1 exchanged with $customer2")
}val fiveTimes = stopAtAttempts(5)
fun main() = runBlocking {
retry(fiveTimes) {
printExchangeBetween(1L, 2L)
}
}
```We can also provide a [`RetryPolicy`][retry-policy] that only retries failures
of a specific type. The example below will retry the operation only if the
reason for failure was a `SQLDataException`, pausing for 20 milliseconds before
retrying and stopping after 5 total invocations.```kotlin
import com.github.michaelbull.retry.ContinueRetrying
import com.github.michaelbull.retry.StopRetrying
import com.github.michaelbull.retry.policy.RetryPolicy
import com.github.michaelbull.retry.policy.constantDelay
import com.github.michaelbull.retry.policy.continueIf
import com.github.michaelbull.retry.policy.plus
import com.github.michaelbull.retry.policy.stopAtAttempts
import com.github.michaelbull.retry.retry
import kotlinx.coroutines.runBlocking
import java.sql.SQLDataExceptionsuspend fun printExchangeBetween(a: Long, b: Long) {
val customer1 = customers.nameFromId(a)
val customer2 = customers.nameFromId(b)
println("$customer1 exchanged with $customer2")
}val continueOnTimeout = continueIf { (failure) ->
failure is SQLDataException
}val timeoutsEverySecondFiveTimes = continueOnTimeout + constantDelay(1000) + stopAtAttempts(5)
fun main() = runBlocking {
retry(timeoutsEverySecondFiveTimes) {
printExchangeBetween(1L, 2L)
}
}
```## Integration with [kotlin-result][kotlin-result]
If the code you wish to try returns a `Result` instead of throwing an
`Exception`, add the following dependency for access to a Result-based `retry`
function that shares the same policy code:```groovy
repositories {
mavenCentral()
}dependencies {
implementation("com.michael-bull.kotlin-retry:kotlin-retry-result:2.0.1")
}
```Usage:
```kotlin
import com.github.michaelbull.result.Result
import com.github.michaelbull.retry.policy.constantDelay
import com.github.michaelbull.retry.result.retryfun somethingThatCanFail(): Result = TODO()
val everyTwoSeconds = constantDelay(2000)
fun main() = runBlocking {
val result: Result = retry(everyTwoSeconds) {
somethingThatCanFail()
}
}
```## Backoff
The examples above retry executions immediately after they fail, however we may
wish to spread out retries with an ever-increasing delay. This is known as a
"backoff" and comes in many forms. This library includes all the forms of
backoff strategy detailed the article by Marc Brooker on the AWS Architecture
Blog entitled ["Exponential Backoff And Jitter"][aws-backoff].#### Binary Exponential
> _“exponential backoff means that clients multiply their backoff by a constant
> after each attempt, up to some maximum value”_
>
> ```
> sleep = min(cap, base * 2 ** attempt)
> ``````kotlin
retry(binaryExponentialBackoff(min = 10L, max = 5000L)) {
/* code */
}
```#### Full Jitter
> _“trying to improve the performance of a system by adding randomness ... we
> want to spread out the spikes to an approximately constant rate”_
>
> ```
> sleep = random_between(0, min(cap, base * 2 ** attempt))
> ``````kotlin
retry(fullJitterBackoff(min = 10L, max = 5000L)) {
/* code */
}
```#### Equal Jitter
> _“Equal Jitter, where we always keep some of the backoff and jitter by a
> smaller amount”_
>
> ```
> temp = min(cap, base * 2 ** attempt)
> sleep = temp / 2 + random_between(0, temp / 2)
> ``````kotlin
retry(equalJitterBackoff(min = 10L, max = 5000L)) {
/* code */
}
```#### Decorrelated Jitter
> _“Decorrelated Jitter, which is similar to “Full Jitter”, but we also
> increase the maximum jitter based on the last random value”_
>
> ```
> sleep = min(cap, random_between(base, sleep * 3))
> ``````kotlin
retry(decorrelatedJitterBackoff(min = 10L, max = 5000L)) {
/* code */
}
```## Inspiration
- [Control.Retry](http://hackage.haskell.org/package/retry-0.8.0.1/docs/Control-Retry.html)
- [tokio_retry](https://docs.rs/tokio-retry/0.2.0/tokio_retry/)## Contributing
Bug reports and pull requests are welcome on [GitHub][github].
## License
This project is available under the terms of the ISC license. See the
[`LICENSE`](LICENSE) file for the copyright information and licensing terms.[github]: https://github.com/michaelbull/kotlin-retry
[retry]: https://github.com/michaelbull/kotlin-retry/blob/master/kotlin-retry/src/commonMain/kotlin/com/github/michaelbull/retry/Retry.kt
[innodb-deadlocks]: https://dev.mysql.com/doc/refman/8.0/en/innodb-deadlocks.html
[postgres-deadlocks]: https://www.postgresql.org/docs/current/explicit-locking.html#LOCKING-DEADLOCKS
[retry-policy]: https://github.com/michaelbull/kotlin-retry/blob/master/kotlin-retry/src/commonMain/kotlin/com/github/michaelbull/retry/policy/RetryPolicy.kt
[aws-backoff]: https://aws.amazon.com/blogs/architecture/exponential-backoff-and-jitter/
[haskell-retry]: http://hackage.haskell.org/package/retry-0.8.0.1/docs/Control-Retry.html
[kotlin-result]: https://github.com/michaelbull/kotlin-result[badge-android]: http://img.shields.io/badge/-android-6EDB8D.svg?style=flat
[badge-android-native]: http://img.shields.io/badge/support-[AndroidNative]-6EDB8D.svg?style=flat
[badge-jvm]: http://img.shields.io/badge/-jvm-DB413D.svg?style=flat
[badge-js]: http://img.shields.io/badge/-js-F8DB5D.svg?style=flat
[badge-js-ir]: https://img.shields.io/badge/support-[IR]-AAC4E0.svg?style=flat
[badge-nodejs]: https://img.shields.io/badge/-nodejs-68a063.svg?style=flat
[badge-linux]: http://img.shields.io/badge/-linux-2D3F6C.svg?style=flat
[badge-windows]: http://img.shields.io/badge/-windows-4D76CD.svg?style=flat
[badge-wasm]: https://img.shields.io/badge/-wasm-624FE8.svg?style=flat
[badge-apple-silicon]: http://img.shields.io/badge/support-[AppleSilicon]-43BBFF.svg?style=flat
[badge-ios]: http://img.shields.io/badge/-ios-CDCDCD.svg?style=flat
[badge-mac]: http://img.shields.io/badge/-macos-111111.svg?style=flat
[badge-watchos]: http://img.shields.io/badge/-watchos-C0C0C0.svg?style=flat
[badge-tvos]: http://img.shields.io/badge/-tvos-808080.svg?style=flat