Promises with NodeJS












0















I'd like to use promises with NodeJS but it seems not working as expected.



I'd like to see LOG 1, LOG 2 and LOG 3 for all files in this order. And LOG 4 is hidden...



But I get this :



LOG 1
LOG 3
LOG 1
LOG 3
LOG 1
LOG 3
SUCCESS LOG 2
SUCCESS LOG 2
SUCCESS LOG 2
SUCCESS LOG 2


My code :



var filePromise = _.map(files, function(file) {

var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
});

console.log('LOG 3');

});
}
});
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});









share|improve this question


















  • 1





    Your database query us async call. It will skip that block and begins next execution. You can use async/await or place your code inside of the then block of the query

    – Sagar
    Nov 22 '18 at 13:43











  • As @Sagar said, the execution does not pause at the db.query async call. Instead once the async action has been triggered, the main thread will jump to the next line and continue execution.

    – Luke Stoward
    Nov 22 '18 at 14:41
















0















I'd like to use promises with NodeJS but it seems not working as expected.



I'd like to see LOG 1, LOG 2 and LOG 3 for all files in this order. And LOG 4 is hidden...



But I get this :



LOG 1
LOG 3
LOG 1
LOG 3
LOG 1
LOG 3
SUCCESS LOG 2
SUCCESS LOG 2
SUCCESS LOG 2
SUCCESS LOG 2


My code :



var filePromise = _.map(files, function(file) {

var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
});

console.log('LOG 3');

});
}
});
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});









share|improve this question


















  • 1





    Your database query us async call. It will skip that block and begins next execution. You can use async/await or place your code inside of the then block of the query

    – Sagar
    Nov 22 '18 at 13:43











  • As @Sagar said, the execution does not pause at the db.query async call. Instead once the async action has been triggered, the main thread will jump to the next line and continue execution.

    – Luke Stoward
    Nov 22 '18 at 14:41














0












0








0








I'd like to use promises with NodeJS but it seems not working as expected.



I'd like to see LOG 1, LOG 2 and LOG 3 for all files in this order. And LOG 4 is hidden...



But I get this :



LOG 1
LOG 3
LOG 1
LOG 3
LOG 1
LOG 3
SUCCESS LOG 2
SUCCESS LOG 2
SUCCESS LOG 2
SUCCESS LOG 2


My code :



var filePromise = _.map(files, function(file) {

var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
});

console.log('LOG 3');

});
}
});
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});









share|improve this question














I'd like to use promises with NodeJS but it seems not working as expected.



I'd like to see LOG 1, LOG 2 and LOG 3 for all files in this order. And LOG 4 is hidden...



But I get this :



LOG 1
LOG 3
LOG 1
LOG 3
LOG 1
LOG 3
SUCCESS LOG 2
SUCCESS LOG 2
SUCCESS LOG 2
SUCCESS LOG 2


My code :



var filePromise = _.map(files, function(file) {

var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
});

console.log('LOG 3');

});
}
});
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});






node.js promise






share|improve this question













share|improve this question











share|improve this question




share|improve this question










asked Nov 22 '18 at 13:37









tonymx227tonymx227

1,676113161




1,676113161








  • 1





    Your database query us async call. It will skip that block and begins next execution. You can use async/await or place your code inside of the then block of the query

    – Sagar
    Nov 22 '18 at 13:43











  • As @Sagar said, the execution does not pause at the db.query async call. Instead once the async action has been triggered, the main thread will jump to the next line and continue execution.

    – Luke Stoward
    Nov 22 '18 at 14:41














  • 1





    Your database query us async call. It will skip that block and begins next execution. You can use async/await or place your code inside of the then block of the query

    – Sagar
    Nov 22 '18 at 13:43











  • As @Sagar said, the execution does not pause at the db.query async call. Instead once the async action has been triggered, the main thread will jump to the next line and continue execution.

    – Luke Stoward
    Nov 22 '18 at 14:41








1




1





Your database query us async call. It will skip that block and begins next execution. You can use async/await or place your code inside of the then block of the query

– Sagar
Nov 22 '18 at 13:43





Your database query us async call. It will skip that block and begins next execution. You can use async/await or place your code inside of the then block of the query

– Sagar
Nov 22 '18 at 13:43













As @Sagar said, the execution does not pause at the db.query async call. Instead once the async action has been triggered, the main thread will jump to the next line and continue execution.

– Luke Stoward
Nov 22 '18 at 14:41





As @Sagar said, the execution does not pause at the db.query async call. Instead once the async action has been triggered, the main thread will jump to the next line and continue execution.

– Luke Stoward
Nov 22 '18 at 14:41












1 Answer
1






active

oldest

votes


















1














In order for filePromise to contain a collection of promises, you need to return a promise for each iteration of your map call. Only then will the call to Promise.all be evaluated correctly.



var filePromise = _.map(files, function(file) {
return new Promise(function(resolve, reject) {
var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
resolve();
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
reject();
});

console.log('LOG 3');

});
} else {
reject(new Error("failed condition"))
}
});
})
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});


This does not fix the ordering of your console logs. That is a separate issue. Your console logs are not actually out of order, they are logging as expected. However, this is due to the async call you are making to db.query which, when triggered does not block waiting for a response. Instead the next line is executed immediately after the function call returns.



Side note: Make sure to handle your err arguments passed to the inline callback functions of fs.lstat and fs.readFile. Failure to do so will result in the promise never completing in the event of an error.






share|improve this answer
























  • Thank you. :) It works !

    – tonymx227
    Nov 23 '18 at 12:47











  • @tonymx227 not sure I'd be swallowing the internal errors as displayed here (and in your original code), the errors should be propagated up the stack and handled elsewhere or at least be logged better inline.

    – James
    Nov 23 '18 at 15:46











  • @James I did state that in the "Side note" below the code.

    – Luke Stoward
    Nov 26 '18 at 16:04











  • @LukeStoward you did, but as a bare minimum to ensure your code works e.g. your code doesn't propagate the error from db.query so if that fails, the callstack is lost. Just emphasising the point to the OP that swallowing errors, in general, usually isn't a good idea.

    – James
    Nov 26 '18 at 16:17











  • @James Agree completely. I (perhaps wrongly) left the error handling logic out, since the example was already quite big. Hopefully, the OP read all of the solution and is handling errors.

    – Luke Stoward
    Nov 26 '18 at 16:50













Your Answer






StackExchange.ifUsing("editor", function () {
StackExchange.using("externalEditor", function () {
StackExchange.using("snippets", function () {
StackExchange.snippets.init();
});
});
}, "code-snippets");

StackExchange.ready(function() {
var channelOptions = {
tags: "".split(" "),
id: "1"
};
initTagRenderer("".split(" "), "".split(" "), channelOptions);

StackExchange.using("externalEditor", function() {
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled) {
StackExchange.using("snippets", function() {
createEditor();
});
}
else {
createEditor();
}
});

function createEditor() {
StackExchange.prepareEditor({
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: true,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: 10,
bindNavPrevention: true,
postfix: "",
imageUploader: {
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
},
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
});


}
});














draft saved

draft discarded


















StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53432227%2fpromises-with-nodejs%23new-answer', 'question_page');
}
);

Post as a guest















Required, but never shown

























1 Answer
1






active

oldest

votes








1 Answer
1






active

oldest

votes









active

oldest

votes






active

oldest

votes









1














In order for filePromise to contain a collection of promises, you need to return a promise for each iteration of your map call. Only then will the call to Promise.all be evaluated correctly.



var filePromise = _.map(files, function(file) {
return new Promise(function(resolve, reject) {
var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
resolve();
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
reject();
});

console.log('LOG 3');

});
} else {
reject(new Error("failed condition"))
}
});
})
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});


This does not fix the ordering of your console logs. That is a separate issue. Your console logs are not actually out of order, they are logging as expected. However, this is due to the async call you are making to db.query which, when triggered does not block waiting for a response. Instead the next line is executed immediately after the function call returns.



Side note: Make sure to handle your err arguments passed to the inline callback functions of fs.lstat and fs.readFile. Failure to do so will result in the promise never completing in the event of an error.






share|improve this answer
























  • Thank you. :) It works !

    – tonymx227
    Nov 23 '18 at 12:47











  • @tonymx227 not sure I'd be swallowing the internal errors as displayed here (and in your original code), the errors should be propagated up the stack and handled elsewhere or at least be logged better inline.

    – James
    Nov 23 '18 at 15:46











  • @James I did state that in the "Side note" below the code.

    – Luke Stoward
    Nov 26 '18 at 16:04











  • @LukeStoward you did, but as a bare minimum to ensure your code works e.g. your code doesn't propagate the error from db.query so if that fails, the callstack is lost. Just emphasising the point to the OP that swallowing errors, in general, usually isn't a good idea.

    – James
    Nov 26 '18 at 16:17











  • @James Agree completely. I (perhaps wrongly) left the error handling logic out, since the example was already quite big. Hopefully, the OP read all of the solution and is handling errors.

    – Luke Stoward
    Nov 26 '18 at 16:50


















1














In order for filePromise to contain a collection of promises, you need to return a promise for each iteration of your map call. Only then will the call to Promise.all be evaluated correctly.



var filePromise = _.map(files, function(file) {
return new Promise(function(resolve, reject) {
var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
resolve();
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
reject();
});

console.log('LOG 3');

});
} else {
reject(new Error("failed condition"))
}
});
})
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});


This does not fix the ordering of your console logs. That is a separate issue. Your console logs are not actually out of order, they are logging as expected. However, this is due to the async call you are making to db.query which, when triggered does not block waiting for a response. Instead the next line is executed immediately after the function call returns.



Side note: Make sure to handle your err arguments passed to the inline callback functions of fs.lstat and fs.readFile. Failure to do so will result in the promise never completing in the event of an error.






share|improve this answer
























  • Thank you. :) It works !

    – tonymx227
    Nov 23 '18 at 12:47











  • @tonymx227 not sure I'd be swallowing the internal errors as displayed here (and in your original code), the errors should be propagated up the stack and handled elsewhere or at least be logged better inline.

    – James
    Nov 23 '18 at 15:46











  • @James I did state that in the "Side note" below the code.

    – Luke Stoward
    Nov 26 '18 at 16:04











  • @LukeStoward you did, but as a bare minimum to ensure your code works e.g. your code doesn't propagate the error from db.query so if that fails, the callstack is lost. Just emphasising the point to the OP that swallowing errors, in general, usually isn't a good idea.

    – James
    Nov 26 '18 at 16:17











  • @James Agree completely. I (perhaps wrongly) left the error handling logic out, since the example was already quite big. Hopefully, the OP read all of the solution and is handling errors.

    – Luke Stoward
    Nov 26 '18 at 16:50
















1












1








1







In order for filePromise to contain a collection of promises, you need to return a promise for each iteration of your map call. Only then will the call to Promise.all be evaluated correctly.



var filePromise = _.map(files, function(file) {
return new Promise(function(resolve, reject) {
var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
resolve();
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
reject();
});

console.log('LOG 3');

});
} else {
reject(new Error("failed condition"))
}
});
})
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});


This does not fix the ordering of your console logs. That is a separate issue. Your console logs are not actually out of order, they are logging as expected. However, this is due to the async call you are making to db.query which, when triggered does not block waiting for a response. Instead the next line is executed immediately after the function call returns.



Side note: Make sure to handle your err arguments passed to the inline callback functions of fs.lstat and fs.readFile. Failure to do so will result in the promise never completing in the event of an error.






share|improve this answer













In order for filePromise to contain a collection of promises, you need to return a promise for each iteration of your map call. Only then will the call to Promise.all be evaluated correctly.



var filePromise = _.map(files, function(file) {
return new Promise(function(resolve, reject) {
var filePath = './sql/' + file;

fs.lstat(filePath, function(err, stats) {

if (stats.isFile() && file !== '.gitignore' && file !== 'index.js') {

fs.readFile(filePath, 'utf-8', function(err, data) {

console.log('LOG 1');

db.query(data).then(function() {
console.log(colors.green('SUCCESS LOG 2'));
resolve();
}).catch(function() {
console.log(colors.red('ERROR LOG 2'));
reject();
});

console.log('LOG 3');

});
} else {
reject(new Error("failed condition"))
}
});
})
});

Promise.all(filePromise).then(function() {
console.log(colors.green('LOG 4'));
});


This does not fix the ordering of your console logs. That is a separate issue. Your console logs are not actually out of order, they are logging as expected. However, this is due to the async call you are making to db.query which, when triggered does not block waiting for a response. Instead the next line is executed immediately after the function call returns.



Side note: Make sure to handle your err arguments passed to the inline callback functions of fs.lstat and fs.readFile. Failure to do so will result in the promise never completing in the event of an error.







share|improve this answer












share|improve this answer



share|improve this answer










answered Nov 22 '18 at 14:51









Luke StowardLuke Stoward

636917




636917













  • Thank you. :) It works !

    – tonymx227
    Nov 23 '18 at 12:47











  • @tonymx227 not sure I'd be swallowing the internal errors as displayed here (and in your original code), the errors should be propagated up the stack and handled elsewhere or at least be logged better inline.

    – James
    Nov 23 '18 at 15:46











  • @James I did state that in the "Side note" below the code.

    – Luke Stoward
    Nov 26 '18 at 16:04











  • @LukeStoward you did, but as a bare minimum to ensure your code works e.g. your code doesn't propagate the error from db.query so if that fails, the callstack is lost. Just emphasising the point to the OP that swallowing errors, in general, usually isn't a good idea.

    – James
    Nov 26 '18 at 16:17











  • @James Agree completely. I (perhaps wrongly) left the error handling logic out, since the example was already quite big. Hopefully, the OP read all of the solution and is handling errors.

    – Luke Stoward
    Nov 26 '18 at 16:50





















  • Thank you. :) It works !

    – tonymx227
    Nov 23 '18 at 12:47











  • @tonymx227 not sure I'd be swallowing the internal errors as displayed here (and in your original code), the errors should be propagated up the stack and handled elsewhere or at least be logged better inline.

    – James
    Nov 23 '18 at 15:46











  • @James I did state that in the "Side note" below the code.

    – Luke Stoward
    Nov 26 '18 at 16:04











  • @LukeStoward you did, but as a bare minimum to ensure your code works e.g. your code doesn't propagate the error from db.query so if that fails, the callstack is lost. Just emphasising the point to the OP that swallowing errors, in general, usually isn't a good idea.

    – James
    Nov 26 '18 at 16:17











  • @James Agree completely. I (perhaps wrongly) left the error handling logic out, since the example was already quite big. Hopefully, the OP read all of the solution and is handling errors.

    – Luke Stoward
    Nov 26 '18 at 16:50



















Thank you. :) It works !

– tonymx227
Nov 23 '18 at 12:47





Thank you. :) It works !

– tonymx227
Nov 23 '18 at 12:47













@tonymx227 not sure I'd be swallowing the internal errors as displayed here (and in your original code), the errors should be propagated up the stack and handled elsewhere or at least be logged better inline.

– James
Nov 23 '18 at 15:46





@tonymx227 not sure I'd be swallowing the internal errors as displayed here (and in your original code), the errors should be propagated up the stack and handled elsewhere or at least be logged better inline.

– James
Nov 23 '18 at 15:46













@James I did state that in the "Side note" below the code.

– Luke Stoward
Nov 26 '18 at 16:04





@James I did state that in the "Side note" below the code.

– Luke Stoward
Nov 26 '18 at 16:04













@LukeStoward you did, but as a bare minimum to ensure your code works e.g. your code doesn't propagate the error from db.query so if that fails, the callstack is lost. Just emphasising the point to the OP that swallowing errors, in general, usually isn't a good idea.

– James
Nov 26 '18 at 16:17





@LukeStoward you did, but as a bare minimum to ensure your code works e.g. your code doesn't propagate the error from db.query so if that fails, the callstack is lost. Just emphasising the point to the OP that swallowing errors, in general, usually isn't a good idea.

– James
Nov 26 '18 at 16:17













@James Agree completely. I (perhaps wrongly) left the error handling logic out, since the example was already quite big. Hopefully, the OP read all of the solution and is handling errors.

– Luke Stoward
Nov 26 '18 at 16:50







@James Agree completely. I (perhaps wrongly) left the error handling logic out, since the example was already quite big. Hopefully, the OP read all of the solution and is handling errors.

– Luke Stoward
Nov 26 '18 at 16:50






















draft saved

draft discarded




















































Thanks for contributing an answer to Stack Overflow!


  • Please be sure to answer the question. Provide details and share your research!

But avoid



  • Asking for help, clarification, or responding to other answers.

  • Making statements based on opinion; back them up with references or personal experience.


To learn more, see our tips on writing great answers.




draft saved


draft discarded














StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53432227%2fpromises-with-nodejs%23new-answer', 'question_page');
}
);

Post as a guest















Required, but never shown





















































Required, but never shown














Required, but never shown












Required, but never shown







Required, but never shown

































Required, but never shown














Required, but never shown












Required, but never shown







Required, but never shown







Popular posts from this blog

MongoDB - Not Authorized To Execute Command

in spring boot 2.1 many test slices are not allowed anymore due to multiple @BootstrapWith

How to fix TextFormField cause rebuild widget in Flutter