7

I'm trying to write a simple function that converts node-style callback functions to promises, so I can use them with async/await.

Current code:

function toPromise(ctx, func, ...args) {
  let newPromise;

  args.push((err, res) => {
    newPromise = new Promise((resolve, reject)=> {
       if(err) reject(err);
       else{
        resolve(res) 
      };
     });
    });

   func.apply(ctx, args);

   return newPromise;
}

example usage:

const match = await toPromise(user, user.comparePassword, password);
//trying to avoid the following:
user.comparePassword(password, (err, res) => {
     ... });

This probably doesn't make any sense with some great libraries out there, but I'm just trying to code this as an exercise.

Problem is of course match evaluates to undefined, and apparently the promise gets resolved after the await syntax line.

Any idea how I can resolve this issue?

Seneca
  • 2,392
  • 2
  • 18
  • 33

2 Answers2

10

Since node v8.0.0 they added util.promisify.

const util = require('util');
const fs = require('fs');

const stat = util.promisify(fs.stat);
stat('.').then((stats) => {
  // Do something with `stats`
}).catch((error) => {
  // Handle the error.
});

Ref: https://nodejs.org/api/util.html#util_util_promisify_original

aegyed
  • 1,320
  • 3
  • 20
  • 39
4

Your problem is that you are constructing the newPromise inside of the asynchronous callback. So you've still got undefined when you return it. Instead, you will need to call the Promise constructor immediately, and only put the resolve/reject in the asynchronous callback:

function toPromise(ctx, func, ...args) {
    return new Promise((resolve, reject) => {
        args.push((err, res) => {
            if (err) reject(err);
            else resolve(res);
        });
        func.apply(ctx, args);
    });
}

See also How do I convert an existing callback API to promises?

Community
  • 1
  • 1
Bergi
  • 630,263
  • 148
  • 957
  • 1,375