Skip to content

Commit 801246b

Browse files
jacobparisnatemoo-re
andauthoredDec 19, 2024··
feature: add abort controller (#216)
Co-authored-by: jacobparis <jacobparis@users.noreply.github.com> Co-authored-by: Nate Moore <nate@natemoo.re> Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com>
1 parent 4c98bd2 commit 801246b

File tree

3 files changed

+92
-7
lines changed

3 files changed

+92
-7
lines changed
 

‎.changeset/lucky-maps-beam.md

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
---
2+
"@clack/core": minor
3+
"@clack/prompts": minor
4+
---
5+
6+
Adds a new `signal` option to support programmatic prompt cancellation with an [abort controller](https://kettanaito.com/blog/dont-sleep-on-abort-controller).
7+
8+
One example use case is automatically cancelling a prompt after a timeout.
9+
10+
```ts
11+
const shouldContinue = await confirm({
12+
message: 'This message will self destruct in 5 seconds',
13+
signal: AbortSignal.timeout(5000),
14+
});
15+
```
16+
17+
Another use case is racing a long running task with a manual prompt.
18+
19+
```ts
20+
const abortController = new AbortController()
21+
22+
const projectType = await Promise.race([
23+
detectProjectType({
24+
signal: abortController.signal
25+
}),
26+
select({
27+
message: 'Pick a project type.',
28+
options: [
29+
{ value: 'ts', label: 'TypeScript' },
30+
{ value: 'js', label: 'JavaScript' },
31+
{ value: 'coffee', label: 'CoffeeScript', hint: 'oh no'},
32+
],
33+
signal: abortController.signal,
34+
})
35+
])
36+
37+
abortController.abort()
38+
```

‎packages/core/src/prompts/prompt.ts

+25-7
Original file line numberDiff line numberDiff line change
@@ -17,13 +17,15 @@ export interface PromptOptions<Self extends Prompt> {
1717
input?: Readable;
1818
output?: Writable;
1919
debug?: boolean;
20+
signal?: AbortSignal;
2021
}
2122

2223
export default class Prompt {
2324
protected input: Readable;
2425
protected output: Writable;
26+
private _abortSignal?: AbortSignal;
2527

26-
private rl!: ReadLine;
28+
private rl: ReadLine | undefined;
2729
private opts: Omit<PromptOptions<Prompt>, 'render' | 'input' | 'output'>;
2830
private _render: (context: Omit<Prompt, 'prompt'>) => string | undefined;
2931
private _track = false;
@@ -36,14 +38,15 @@ export default class Prompt {
3638
public value: any;
3739

3840
constructor(options: PromptOptions<Prompt>, trackValue = true) {
39-
const { input = stdin, output = stdout, render, ...opts } = options;
41+
const { input = stdin, output = stdout, render, signal, ...opts } = options;
4042

4143
this.opts = opts;
4244
this.onKeypress = this.onKeypress.bind(this);
4345
this.close = this.close.bind(this);
4446
this.render = this.render.bind(this);
4547
this._render = render.bind(this);
4648
this._track = trackValue;
49+
this._abortSignal = signal;
4750

4851
this.input = input;
4952
this.output = output;
@@ -111,11 +114,25 @@ export default class Prompt {
111114

112115
public prompt() {
113116
return new Promise<string | symbol>((resolve, reject) => {
117+
if (this._abortSignal) {
118+
if (this._abortSignal.aborted) {
119+
this.state = 'cancel';
120+
121+
this.close();
122+
return resolve(CANCEL_SYMBOL);
123+
}
124+
125+
this._abortSignal.addEventListener('abort', () => {
126+
this.state = 'cancel';
127+
this.close();
128+
}, { once: true });
129+
}
130+
114131
const sink = new WriteStream(0);
115132
sink._write = (chunk, encoding, done) => {
116133
if (this._track) {
117-
this.value = this.rl.line.replace(/\t/g, '');
118-
this._cursor = this.rl.cursor;
134+
this.value = this.rl?.line.replace(/\t/g, '');
135+
this._cursor = this.rl?.cursor ?? 0;
119136
this.emit('value', this.value);
120137
}
121138
done();
@@ -171,7 +188,7 @@ export default class Prompt {
171188
}
172189
if (char === '\t' && this.opts.placeholder) {
173190
if (!this.value) {
174-
this.rl.write(this.opts.placeholder);
191+
this.rl?.write(this.opts.placeholder);
175192
this.emit('value', this.opts.placeholder);
176193
}
177194
}
@@ -185,7 +202,7 @@ export default class Prompt {
185202
if (problem) {
186203
this.error = problem;
187204
this.state = 'error';
188-
this.rl.write(this.value);
205+
this.rl?.write(this.value);
189206
}
190207
}
191208
if (this.state !== 'error') {
@@ -210,7 +227,8 @@ export default class Prompt {
210227
this.input.removeListener('keypress', this.onKeypress);
211228
this.output.write('\n');
212229
setRawMode(this.input, false);
213-
this.rl.close();
230+
this.rl?.close();
231+
this.rl = undefined;
214232
this.emit(`${this.state}`, this.value);
215233
this.unsubscribe();
216234
}

‎packages/core/test/prompts/prompt.test.ts

+29
Original file line numberDiff line numberDiff line change
@@ -229,4 +229,33 @@ describe('Prompt', () => {
229229
expect(eventSpy).toBeCalledWith(key);
230230
}
231231
});
232+
233+
test('aborts on abort signal', () => {
234+
const abortController = new AbortController();
235+
236+
const instance = new Prompt({
237+
input,
238+
output,
239+
render: () => 'foo',
240+
signal: abortController.signal,
241+
});
242+
243+
instance.prompt();
244+
245+
expect(instance.state).to.equal('active');
246+
247+
abortController.abort();
248+
249+
expect(instance.state).to.equal('cancel');
250+
});
251+
252+
test('returns immediately if signal is already aborted', () => {
253+
const abortController = new AbortController();
254+
abortController.abort();
255+
256+
const instance = new Prompt({ input, output, render: () => 'foo', signal: abortController.signal });
257+
instance.prompt();
258+
259+
expect(instance.state).to.equal('cancel');
260+
});
232261
});

0 commit comments

Comments
 (0)
Please sign in to comment.