Skip to content

Commit cb3020d

Browse files
rexagodnodejs-github-bot
authored andcommitted
lib: add error handling for input stream
This adds support for error handling in readline.createInterface() for cases where the input object is not supplied, the input stream is invalid, or the underlying buffer emits an error. Now, the 'error' emissions by the readline module are thrown but in order to log those in the specific case of await for loops, we still need to fix silent rejections (TODO added there) inside async iterators for the thenables to work. Fixes: #30831 PR-URL: #31603 Reviewed-By: Robert Nagy <ronagy@icloud.com> Reviewed-By: Ruben Bridgewater <ruben@bridgewater.de> Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com> Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com>
1 parent e40ed28 commit cb3020d

File tree

2 files changed

+53
-0
lines changed

2 files changed

+53
-0
lines changed

lib/readline.js

+13
Original file line numberDiff line numberDiff line change
@@ -190,6 +190,10 @@ function Interface(input, output, completer, terminal) {
190190
this._ttyWrite = _ttyWriteDumb.bind(this);
191191
}
192192

193+
function onerror(err) {
194+
self.emit('error', err);
195+
}
196+
193197
function ondata(data) {
194198
self._normalWrite(data);
195199
}
@@ -227,9 +231,12 @@ function Interface(input, output, completer, terminal) {
227231

228232
this[kLineObjectStream] = undefined;
229233

234+
input.on('error', onerror);
235+
230236
if (!this.terminal) {
231237
function onSelfCloseWithoutTerminal() {
232238
input.removeListener('data', ondata);
239+
input.removeListener('error', onerror);
233240
input.removeListener('end', onend);
234241
}
235242

@@ -240,6 +247,7 @@ function Interface(input, output, completer, terminal) {
240247
} else {
241248
function onSelfCloseWithTerminal() {
242249
input.removeListener('keypress', onkeypress);
250+
input.removeListener('error', onerror);
243251
input.removeListener('end', ontermend);
244252
if (output !== null && output !== undefined) {
245253
output.removeListener('resize', onresize);
@@ -1098,12 +1106,17 @@ Interface.prototype[SymbolAsyncIterator] = function() {
10981106
});
10991107
const lineListener = (input) => {
11001108
if (!readable.push(input)) {
1109+
// TODO(rexagod): drain to resume flow
11011110
this.pause();
11021111
}
11031112
};
11041113
const closeListener = () => {
11051114
readable.push(null);
11061115
};
1116+
const errorListener = (err) => {
1117+
readable.destroy(err);
1118+
};
1119+
this.on('error', errorListener);
11071120
this.on('line', lineListener);
11081121
this.on('close', closeListener);
11091122
this[kLineObjectStream] = readable;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
'use strict';
2+
const common = require('../common');
3+
const fs = require('fs');
4+
const readline = require('readline');
5+
const path = require('path');
6+
7+
async function processLineByLine_SymbolAsyncError(filename) {
8+
const fileStream = fs.createReadStream(filename);
9+
const rl = readline.createInterface({
10+
input: fileStream,
11+
crlfDelay: Infinity
12+
});
13+
// eslint-disable-next-line no-unused-vars
14+
for await (const line of rl) {
15+
/* check SymbolAsyncIterator `errorListener` */
16+
}
17+
}
18+
19+
const f = path.join(__dirname, 'file.txt');
20+
21+
// catch-able SymbolAsyncIterator `errorListener` error
22+
processLineByLine_SymbolAsyncError(f).catch(common.expectsError({
23+
code: 'ENOENT',
24+
message: `ENOENT: no such file or directory, open '${f}'`
25+
}));
26+
27+
async function processLineByLine_InterfaceErrorEvent(filename) {
28+
const fileStream = fs.createReadStream(filename);
29+
const rl = readline.createInterface({
30+
input: fileStream,
31+
crlfDelay: Infinity
32+
});
33+
rl.on('error', common.expectsError({
34+
code: 'ENOENT',
35+
message: `ENOENT: no such file or directory, open '${f}'`
36+
}));
37+
}
38+
39+
// check Interface 'error' event
40+
processLineByLine_InterfaceErrorEvent(f);

0 commit comments

Comments
 (0)