Thanks to visit codestin.com
Credit goes to github.com

Skip to content

Gitoffthelawn/recur

 
 

Repository files navigation

recur

recur is a command-line tool that runs a single command repeatedly until it succeeds or no more attempts are left. It implements optional exponential backoff with configurable jitter. It lets you write the success condition in Starlark.

Installation

Prebuilt binaries

Prebuilt binaries for FreeBSD (amd64), Linux (aarch64, riscv64, x86_64), macOS (arm64, x86_64), NetBSD (amd64), OpenBSD (amd64), and Windows (amd64, arm64, x86) are attached to releases.

Homebrew

You can install recur from Homebrew on macOS and Linux:

brew install recur

Go

Install Go, then run the following command:

go install dbohdan.com/recur/v2@latest

Build requirements

  • Go 1.19
  • Task (go-task) 3.28

Usage

Command-line interface

Usage: recur [-h] [-V] [-a <attempts>] [-b <backoff>] [-c <condition>] [-d
<delay>] [-F] [-f] [-I] [-j <jitter>] [-m <max-delay>] [-O] [-r <reset-time>]
[-t <timeout>] [-v] [--] <command> [<arg> ...]

Retry a command with exponential backoff and jitter.

Arguments:
  <command>
          Command to run

  [<arg> ...]
          Arguments to the command

Options:
  -h, --help
          Print this help message and exit

  -V, --version
          Print version number and exit

  -a, --attempts 10
          Maximum number of attempts (negative for infinite)

  -b, --backoff 0
          Base for exponential backoff (duration)

  -c, --condition 'code == 0'
          Success condition (Starlark expression)

  -d, --delay 0
          Constant delay (duration)

  -F, --fib
          Add Fibonacci backoff

  -f, --forever
          Infinite attempts

  -I, --replay-stdin
          Read standard input until EOF at the start and replay it on each
attempt

  -j, --jitter '0,0'
          Additional random delay (maximum duration or 'min,max' duration)

  -m, --max-delay 1h
          Maximum allowed sum of constant delay, exponential backoff, and
Fibonacci backoff (duration)

  -O, --hold-stdout
          Buffer standard output for each attempt and only print it on success

  -r, --reset -1s
          Minimum attempt time that resets exponential and Fibonacci backoff
(duration; negative for no reset)

  -t, --timeout -1s
          Timeout for each attempt (duration; negative for no timeout)

  -v, --verbose
          Increase verbosity (up to 3 times)

The "duration" arguments take Go duration strings; for example, 0, 100ms, 2.5s, 0.5m, or 1h. The value of -j/--jitter must be either one duration string or two joined with a comma, like 1s,2s.

Setting the delay (-d/--delay) increases the maximum delay (-m/--max-delay) to that value when the maximum delay is shorter. Use -m/--max-delay after -d/--delay if you want a shorter maximum delay.

The following recur options run the command foo --config bar.cfg indefinitely. Every time foo exits, there is a delay that grows exponentially from two seconds to a minute. The delay resets back to two seconds if the command runs for at least five minutes.

recur --backoff 2s --condition False --forever --max-delay 1m --reset 5m foo --config bar.cfg

recur exits with the last command's exit code unless the user overrides this in the condition. When the command is not found during the last attempt, recur exits with the code 255.

Standard input

By default, the command run by recur inherits recur's standard input. This means that if standard input is a terminal, every attempt can read interactively from the terminal. If standard input is a pipe or a redirected file, the data are consumed on the first attempt; later attempts will see an immediate EOF.

To feed the command the same data every time, use the option -I/--replay-stdin. With this option, recur slurps its entire stdin into memory and replays the buffer on each attempt.

$ echo hi | recur -a 3 -c False cat
hi
recur [00:00:00.0]: maximum 3 attempts reached

$ echo hi | recur -a 3 -c False -I cat
hi
hi
hi
recur [00:00:00.0]: maximum 3 attempts reached

Because the data are buffered in memory, --replay-stdin is not recommended for very large inputs.

Standard output

The command's standard output is passed through to recur's standard output by default. To prevent this and only print the output of the final, successful attempt if there is a successful attempt, use the option -O/--hold-stdout. With this option, recur buffers the command's stdout in memory and only prints it if the success condition is met or the condition expression calls flush_stdout.

$ recur -c 'attempt == 3' sh -c 'echo "$RECUR_ATTEMPT"'
1
2
3

$ recur -c 'attempt == 3' -O sh -c 'echo "$RECUR_ATTEMPT"'
3

$ recur -c 'flush_stdout() or attempt == 3' -O sh -c 'echo "$RECUR_ATTEMPT"'
1
2
3

Because the data are buffered in memory, --hold-stdout is not recommended for commands that produce very large output.

Environment variables

recur sets the environment variable RECUR_ATTEMPT for the command it runs to the current attempt number. This way the command can access the attempt counter. recur also sets RECUR_MAX_ATTEMPTS to the value of -a/--attempts and RECUR_ATTEMPT_SINCE_RESET to the attempt number since exponential and Fibonacci backoff were reset.

The following command succeeds on the last attempt:

$ recur sh -c 'echo "Attempt $RECUR_ATTEMPT of $RECUR_MAX_ATTEMPTS"; exit $((RECUR_MAX_ATTEMPTS - RECUR_ATTEMPT))'
Attempt 1 of 10
Attempt 2 of 10
Attempt 3 of 10
Attempt 4 of 10
Attempt 5 of 10
Attempt 6 of 10
Attempt 7 of 10
Attempt 8 of 10
Attempt 9 of 10
Attempt 10 of 10

Conditions

recur supports a limited form of scripting. You can define the success condition using an expression in Starlark, a small scripting language derived from Python. The default condition is code == 0. It means recur will stop retrying when the exit code of the command is zero.

If you know Python, you can quickly start writing recur conditions in Starlark. The most significant differences between Starlark and Python for this purpose are:

  • Starlark has no is. You must write code == None, not code is None.
  • Starlark has no sets. Write code in (1, 2, 3) or code in [1, 2, 3] rather than code in {1, 2, 3}.

You can use the following variables in the condition expression:

  • attempt: int — the number of the current attempt, starting at one. Combine with --forever to use the condition instead of the built-in attempt counter.
  • attempt_since_reset: int — the attempt number since exponential and Fibonacci backoff were reset, starting at one.
  • code: int | None — the exit code of the last command. code is None when the command was not found.
  • command_found: bool — whether the last command was found.
  • max_attempts: int — the value of the option --attempts. --forever sets it to -1.
  • time: float — the time the most recent attempt took, in seconds.
  • total_time: float — the time between the start of the first attempt and the end of the most recent, again in seconds.

recur defines custom functions:

  • exit(code: int | None) -> None — exit with the exit code. If code is None, exit with the exit code for a missing command (255).
  • flush_stdout() -> None — if recur is running with the option -O/--hold-stdout, recur will output the command's buffered standard output when it has finished evaluating the condition. recur will print the standard output whether the condition is true or false and also if exit is called. The function does nothing without the option -O/--hold-stdout.
  • inspect(value: Any, *, prefix: str = "") -> Any — log value prefixed by prefix and return value. This is useful for debugging.

The exit function allows you to override the default behavior of returning the last command's exit code. For example, you can make recur exit with success when the command fails.

recur --condition 'code != 0 and exit(0)' sh -c 'exit 1'
# or
recur --condition 'False if code == 0 else exit(0)' sh -c 'exit 1'

In the following example we stop early and do not retry when the command's exit code indicates incorrect usage or a problem with the installation.

recur --condition 'code == 0 or (code in (1, 2, 3, 4) and exit(code))' curl "$url"

License

MIT.

Alternatives

recur was inspired by retry-cli. I wanted something like retry-cli but without the Node.js dependency.

There are other similar tools:

  • attempt. Written in Rust. cargo install attempt-cli.
  • eb. Written in Rust. cargo install eb.
  • retry (joshdk). Written in Go. go install github.com/joshdk/retry@master.
  • retry (kadwanev). Written in Bash.
  • retry (minfrin). Written in C. Packaged for Debian and Ubuntu. sudo apt install retry.
  • retry (timofurrer). Written in Rust. cargo install retry-cmd.
  • retry-cli. Written in JavaScript for Node.js. npx retry-cli.
  • SysBox includes the command splay. Written in Go. go install github.com/skx/sysbox@latest.

About

Retry a command with exponential backoff and jitter (+ Starlark expressions)

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • Go 93.3%
  • Shell 6.7%