I made this dummy code to have a better understanding on how promises work, mimicking a more complex software that I have to "promisify". In the attached code, I wanted the events to fire and be logged in the following order:
But as you can see if you run it, the "after" string is printed between steps 2 and 3. Clearly, I must be doing something wrong in handling async logic. Thanks for your help!
const obj = {
"rows": [{
"type": "A",
"value": 0
}, {
"type": "B",
"value": 0
}, {
"type": "C",
"value": 0
}]
}
let promises = [];
function delay(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
const alter_value = async(row, to_add, time) => {
await delay(time);
row.value = row.value + to_add;
console.log("done with " + row.type);
return true;
}
const two = async() => {
obj.rows.forEach(async(row) => {
switch (row.type) {
case "A":
console.log("A detected");
promises.push(alter_value(row, 1, 1000))
promises.push(alter_value(row, 2, 1800))
break;
case "B":
console.log("B detected");
promises.push(alter_value(row, 5, 1400))
break;
case "C":
console.log("C detected");
promises.push(alter_value(row, 200, 2400))
break;
}
});
return promises;
}
const one = async() => {
console.log("before");
Promise.all(two()).then(console.log("after"));
}
one();
I see at least two issues with your code, explaining the result you're getting:
two
function should not be async
. async
functions return an implicit Promise
. Here you just want to return an array of Promise
s that you already construct yourself, so a normal function is what you need..then(console.log("after"))
will execute the console.log
right away: then()
expects a function to execute later, so you have to change it to .then(() => console.log("after"))
.This becomes:
const obj = {
"rows": [{
"type": "A",
"value": 0
}, {
"type": "B",
"value": 0
}, {
"type": "C",
"value": 0
}]
};
function delay(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
const alter_value = async (row, to_add, time) => {
await delay(time);
row.value = row.value + to_add;
console.log("done with " + row.type);
return true;
};
const two = () => {
const promises = [];
obj.rows.forEach(async(row) => {
switch (row.type) {
case "A":
console.log("A detected");
promises.push(alter_value(row, 1, 1000));
promises.push(alter_value(row, 2, 1800));
break;
case "B":
console.log("B detected");
promises.push(alter_value(row, 5, 1400));
break;
case "C":
console.log("C detected");
promises.push(alter_value(row, 200, 2400));
break;
}
});
return promises;
};
const one = async () => {
console.log('before');
Promise.all(two()).then(() => console.log('after'));
};
one();
Note that, as an alternative to .then()
, you can simply use await
on Promise.all
as well, to make your code more consistent:
await Promise.all(two());
console.log('after');