Skip to content

Commit 98a9abb

Browse files
committed
Add retry & echo
1 parent 3e4df75 commit 98a9abb

File tree

5 files changed

+108
-1
lines changed

5 files changed

+108
-1
lines changed

README.md

+27
Original file line numberDiff line numberDiff line change
@@ -338,6 +338,33 @@ files (when using `zx` executable).
338338
let {version} = require('./package.json')
339339
```
340340
341+
### Experimental
342+
343+
The zx also provides a few experimental functions. Please leave a feedback about
344+
those features in [the discussion](https://github.com/google/zx/discussions/299).
345+
346+
#### `retry()`
347+
348+
Retries a command a few times. Will return after the first
349+
successful attempt, or will throw after specifies attempts count.
350+
351+
```js
352+
import {retry} from 'zx/experimental'
353+
354+
let {stdout} = await retry(5)`curl localhost`
355+
```
356+
357+
#### ``echo`...` ``
358+
359+
A `console.log()` alternative which can take [ProcessOutput](#processoutput).
360+
361+
```js
362+
import {echo} from 'zx/experimental'
363+
364+
let branch = await $`git branch --show-current`
365+
echo`Current branch is ${branch}.`
366+
```
367+
341368
### FAQ
342369
343370
#### Passing env variables

experimental.d.ts

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
// Copyright 2021 Google LLC
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
import {ProcessOutput} from './index'
16+
17+
interface Echo {
18+
(pieces: TemplateStringsArray, ...args: any[]): Promise<void>
19+
}
20+
21+
interface Retry {
22+
(pieces: TemplateStringsArray, ...args: any[]): Promise<ProcessOutput>
23+
}
24+
25+
export const echo: Echo
26+
export const retry: (count: number) => Retry

experimental.mjs

+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
// Copyright 2021 Google LLC
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
import {ProcessOutput} from './index.mjs'
16+
17+
// Retries a command a few times. Will return after the first
18+
// successful attempt, or will throw after specifies attempts count.
19+
export const retry = (count = 5) => async (cmd, ...args) => {
20+
while (count --> 0) try {
21+
return await $(cmd, ...args)
22+
} catch (p) {
23+
if (count === 0) throw p
24+
}
25+
}
26+
27+
// A console.log() alternative which can take ProcessOutput.
28+
export const echo = (pieces, ...args) => {
29+
if (!Array.isArray(pieces) || pieces.length - 1 !== args.length) {
30+
throw new Error('The echo is a template string. Use as echo`...`.')
31+
}
32+
let msg = pieces[0], i = 0
33+
while (i < args.length) {
34+
msg += stringify(args[i]) + pieces[++i]
35+
}
36+
console.log(msg)
37+
}
38+
39+
function stringify(arg) {
40+
if (arg instanceof ProcessOutput) {
41+
return arg.toString().replace(/\n$/, '')
42+
}
43+
return `${arg}`
44+
}

package.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,8 @@
55
"main": "./index.mjs",
66
"exports": {
77
".": "./index.mjs",
8-
"./globals": "./globals.mjs"
8+
"./globals": "./globals.mjs",
9+
"./experimental": "./experimental.mjs"
910
},
1011
"types": "index.d.ts",
1112
"bin": {

test.mjs

+9
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
// limitations under the License.
1414

1515
import {strict as assert} from 'assert'
16+
import {retry} from './experimental.mjs'
1617

1718
{ // Only stdout is used during command substitution
1819
let hello = await $`echo Error >&2; echo Hello`
@@ -218,6 +219,14 @@ import {strict as assert} from 'assert'
218219
console.log(chalk.greenBright('YAML works'))
219220
}
220221

222+
{ // Retry works.
223+
try {
224+
await retry(5)`exit 123`
225+
} catch (p) {
226+
assert.equal(p.exitCode, 123)
227+
}
228+
}
229+
221230
{ // require() is working in ESM
222231
const {name, version} = require('./package.json')
223232
assert(typeof name === 'string')

0 commit comments

Comments
 (0)