'Node.js readline inside of promises

I'm trying to use the node.js package readline to get user input on the command line, and I want to pipe the entered input through promises. However, the input never gets through the then chain. I think the problem could come from the fact that the promises are fulfilled in the callback method, but I don't know how to solve that problem.

An example of this problem looks like this:

import rlp = require('readline');

const rl = rlp.createInterface({
  input: process.stdin,
  output: process.stdout
});  

let prom = new Promise(resolve => {
  rl.question('Enter input: ', input => rl.close() && resolve(input));
});

prom
  .then(result => { console.log(result); return prom; })
  .then(result => { console.log(result); return prom; })
  .then(result => console.log(result));

If run in node.js, the question will appear once, after input has been entered the program just stops. I want it to wait until the first input has been entered, then it should print this input and ask for the next input.

Thanks in advance!



Solution 1:[1]

Once your promise is resolved, there's no use of waiting for that again. I also moved the rl.close() call to the end, as it's needed to be called only once.

const rlp = require('readline');

const rl = rlp.createInterface({
  input: process.stdin,
  output: process.stdout
});

function ask() {
  return new Promise(resolve => {
    rl.question('Enter input: ', input => resolve(input));
  });
}

ask()
  .then(result => { console.log(result); return ask(); })
  .then(result => { console.log(result); return ask(); })
  .then(result => { console.log(result); rl.close() });

Solution 2:[2]

Here's an answer from this question here for which I deserve no credit.


// Function
function Ask(query) {
  const readline = require("readline").createInterface({
    input: process.stdin,
    output: process.stdout
  })

  return  new Promise(resolve => readline.question(query, ans => {
  readline.close();
  resolve(ans);
}))
}

 
// example useage
async function main() {

 var name = await Ask("whats you name")
 console.log(`nice to meet you ${name}`)

 var age = await Ask("How old are you?")
 console.log(`Wow what a fantastic age, imagine just being ${age}`)
}

main()

Solution 3:[3]


const readline = require('readline');

const rl = readline.createInterface({ input: process.stdin, output: process.stdout });
const ask = (query) => new Promise((resolve) => rl.question(query, resolve));

ask('A: ').then(async (a) => {
  const b = await ask('B: ');
  const c = await ask('B: ');

  console.log(a, b, c);

  rl.close();
});

rl.on('close', () => process.exit(0));

Solution 4:[4]

Node.js 17 is here with new promise-based APIs for readline module:

import * as readline from 'node:readline/promises'
import { stdin as input, stdout as output } from 'process'

const rl = readline.createInterface({input, output})

const answer = await rl.question('What do you think of Node.js? ')

console.log(`Thank you for your valuable feedback: ${answer}`)

rl.close()

https://nodejs.org/api/readline.html#readline

Solution 5:[5]

node prompt.mjs

import { createInterface as createQuestionInterface } from 'readline';

const rl = createQuestionInterface({
  input: process.stdin,
  output: process.stdout
});

function questionLine(multiline, resolve, i, input, rl) {
  if (!multiline) {
    resolve(i);
  } else {
    if (input && !i) {
      resolve(input);
    } else {
      return input + i + "\r\n";
    }
  }
  return input;
}

function promptMultiLine(questionText) { // This is async by returning promise
  return prompt(questionText, true);
}

async function prompt(questionText, multiline = false) {
  return await (new Promise((resolve, reject) => {
    let input = '';
    rl.question(`${questionText}: `, (i) => {
      input = questionLine(multiline, resolve, i, input, rl);
    });
    rl.on('line', (i) => {
      input = questionLine(multiline, resolve, i, input, rl);
    });
  }));
}

async function run() {
  const question = prompt("please enter response [enter to complete]");
  console.log(question);
  const questionMultiLine = promptMultiLine("please enter response [enter text and enter twice]");
  console.log(questionMultiLine);
}

run();

Sources

This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.

Source: Stack Overflow

Solution Source
Solution 1 Jurosh
Solution 2 Max Carroll
Solution 3 Chris Wiles
Solution 4
Solution 5