I have been reading nodejs lately, trying to understand how it handles multiple concurrent requests, I know nodejs is a single threaded event loop based architecture, at a given point of time only one statement is gonna execute i.e, on main thread and blocking code/IO calls are handled by the worker threads (default is 4).
Now my question is what happens when a web server built using nodejs receives multiple requests, I know, there are a lots of Stack overflow thread that has similar questions, but didn't find a concrete answer to this.
So I am putting an example here, let's say we have following code inside a route like /index.
app.use('/index', function(req, res, next) { console.log("hello index routes was invoked"); readImage("path", function(err, content) { status = "Success"; if(err) { console.log("err :", err); status = "Error" } else { console.log("Image read"); } return res.send({ status: status }); }); var a = 4, b = 5; console.log("sum =", a + b); });
Let's assume that readImage() takes around 1 min to read that Image. If two request T1, and T2 came concurently, How nodejs is gonna process these request ?
Does it going to take first request T1, process it while queueing the request T2 (please correct me if my understanding is wrong here), if any async/blocking stuff is encountered like readImage, it then sends that to worker thread (some point later when async stuff is done it notifies the main thread and main thread starts executing the callback), move ahead by executing the next line of code. When it is done with T1 then picks T2 request ? Is it correct? or it can process T2 code in between (meaning while readImage is called, it can start processing T2)?
I would really appreciate if anyone can help me finding an answer to this question
6 Answers
Answers 1
Your confusion might be coming from not focusing on the event loop enough. clearly you have an idea of how this works, but maybe not the full picture.
Part 1, Event Loop Basics
When you call the use
method, what happens behind the scenes is another thread is created to listen for connections.
However, when a request comes in, because we're in a different thread than the V8 engine (and cannot directly invoke the route function), a serialized call to the function is appended onto the shared event loop, for it to be called later. (event loop is a poor name in this context, as it operates more like a queue or stack)
at the end of the js file, V8 will check if there are any running theads or messages in the event loop. If there are none, it will exit 0 (this is why server code keeps the process running). So the first Timing nuance to understand is that no request will be processed until the synchronous end of the js file is reached.
If the event loop was appended to while the process was starting up, each function call on the event loop will be handled one by one, in its entirety, synchronously.
For simplicity, let me break down your example into something more expressive.
function callback() { setTimeout(function inner() { console.log('hello inner!'); }, 0); // † console.log('hello callback!'); } setTimeout(callback, 0); setTimeout(callback, 0);
† setTimeout
with a time of 0, is a quick and easy way to put something on the event loop without any timer complications, since no matter what, it has always been at least 0ms.
In this example, the output will always be:
hello callback! hello callback! hello inner! hello inner!
Both serialized calls to callback
are appended to the event loop before either of them is called, guaranteed. This happens because nothing can be invoked from the event loop until after the full synchronous execution of the file.
It can be helpful to think of the execution of your file, as the first thing on the event loop. Because each invocation from the event loop can only happen in series, it becomes a logical consequence, that no other event loop invocation can occur during its execution; Only when its finished, can another event loop function be invoked.
Part 2, The inner Callback
The same logic applies to the inner callback as well, and can be used to explain why the program will never output:
hello callback! hello inner! hello callback! hello inner!
Like you might expect.
By the end of the execution of the file, 2 serialized function calls will be on the event loop, both for callback
. As the Event loop is a FIFO (first in, first out), the setTimeout
that came first, will be be invoked first.
The first thing callback
does is perform another setTimeout
. As before, this will append a serialized call, this time to the inner
function, to the event loop. setTimeout
immediately returns, and execution will move on to the first console.log
.
At this time, the event loop looks like this:
1 [callback] (executing) 2 [callback] (next in line) 3 [inner] (just added by callback)
The return of callback
is the signal for the event loop to remove that invocation from itself. This leaves 2 things on the event loop now: 1 more call to callback
, and 1 call to inner
.
callback
is the next function in line, so it will be invoked next. The process repeats itself. A call to inner
is appended to the event loop. A console.log
prints Hello Callback!
and we finish by removing this invocation of callback
from the event loop.
This leaves the event loop with 2 more functions:
1 [inner] (next in line) 2 [inner] (added by most recent callback)
Neither of these functions mess with the event loop any farther, they execute one after the other; The second one waiting for the first one's return. Then when the second one returns, The event loop is left empty. This combined wit the fact that there are no other threads currently running, triggers the end of the process. exit 0.
Part 3, Relating to the Original Example
The first thing that happens in your example, is that a thread is created, within the process, that will create a server bound to a particular port. Note, this is happening in precompiled C++, not javascript, and is not a separate process, its a thread within the same process. see: C++ Thread Tutorial
So now, whenever a request comes in, the execution of your original code wont be disturbed. Instead, incoming connection requests will be opened, held onto, and appended to the event loop.
The use
function, is the gateway into catching the events for incoming requests. Its an abstraction layer, but for the sake of simplicity, Its helpful to think of the use
function like you would a setTimeout
. Except, instead of waiting a set amount of time, it appends the callback to the event loop upon incoming http requests.
So, lets assume that there are two requests coming in to the server: T1 and T2. In your question you say they come in concurrently, since this is technically impossible, I'm going to assume they are one after the other, with a negligible time in between them.
Whichever request comes in first, will be handled first by the secondary thread from earlier. once that connection has been opened, its appended to the event loop, and we move on to the next request, and repeat.
At any point after the first request is added to the event loop, V8 can begin execution of the use
callback.
a quick aside about readImage
Since its unclear whether readImage
is from a particular library,, something you wrote or otherwise, its impossible to tell exactly what it will do in this case. There are only 2 possibilities though, so here they are:
// in this example definition of readImage, its entirely // synchronous, never using an alternate thread or the // event loop function readImage (path, callback) { let image = fs.readFileSync(path); callback(null, image); // a definition like this will force the callback to // fully return before readImage returns. This means // means readImage will block any subsequent calls. } // in this alternate example definition its entirely // asynchronous, and take advantage of fs' async // callback. function readImage (path, callback) { fs.readFile(path, (err, data) => { callback(err, data); }); // a definition like this will force the readImage // to immediately return, and allow exectution // to continue. }
For the purposes of explanation, I'll be operating under the assumption that readImage will immediately return, as proper asynchronous functions should.
Once the use
callback execution is started, the following will happen:
- The first console log will print.
- readImage will kick off a worker thread and immediately return.
- The second console log will print.
During all of this, its important to note, these operations are happening synchronously; No other event loop invocation can start until these are finished. readImage may be asynchronous, but calling it is not, the callback and usage of a worker thread is what makes it asynchronous.
After this use
callback returns, the next request has probably already finished parsing and was added to the event loop, while V8 was busy doing our console logs and readImage call.
So the next use
callback is invoked, and repeats the same process: log, kick off a readImage thread, log again, return.
After this point, the read Images (depending on how long they take) have probably already retrieved what they needed and appended their callback to the event loop. So they will get executed next, in order of whichever one retrieved its data first. remember, These operations were happening in separate threads, so the happened not only parallel to the main javascript thread, but also parallel to each other, so here, it doesnt matter which one got called first, it matters which one finished first, and got dibs on the event loop.
Whichever readImage completed first will be the first one to execute. so, assuming no errors, we'll print out to the console, then write to the response for the corresponding request, held in lexical scope.
When that send returns, the next readImage callback will begin execution: console log, and writing to the response.
at this point, both readImage threads have died, and the event loop is empty, but the thread that holds the server port binding is keeping the process alive, waiting for something else to add to the event loop, and the cycle to continue.
I hope this helps you understand the mechanics behind the asynchronous nature of the example you provided
Answers 2
For each incoming request, node will handle it one by one. That means there must be order, just like the queue, first in first serve. When node starts processing request, all synchronous code will execute, and asynchronous will pass to work thread, so node can start to process the next request. When the asynchrous part is done, it will go back to main thread and keep going.
So when your synchronous code takes too long, you block the main thread, node won't be able to handle other request, it's easy to test.
app.use('/index', function(req, res, next) { // synchronous part console.log("hello index routes was invoked"); var sum = 0; // useless heavy task to keep running and block the main thread for (var i = 0; i < 100000000000000000; i++) { sum += i; } // asynchronous part, pass to work thread readImage("path", function(err, content) { // when work thread finishes, add this to the end of the event loop and wait to be processed by main thread status = "Success"; if(err) { console.log("err :", err); status = "Error" } else { console.log("Image read"); } return res.send({ status: status }); }); // continue synchronous part at the same time. var a = 4, b = 5; console.log("sum =", a + b); });
Node won't start processing the next request until finish all synchronous part. So people said don't block the main thread.
Answers 3
You can simply create child process by shifting readImage() function in a different file using fork().
The parent file, parent.js
:
const { fork } = require('child_process'); const forked = fork('child.js'); forked.on('message', (msg) => { console.log('Message from child', msg); }); forked.send({ hello: 'world' });
The child file, child.js
:
process.on('message', (msg) => { console.log('Message from parent:', msg); }); let counter = 0; setInterval(() => { process.send({ counter: counter++ }); }, 1000);
Above article might be useful to you .
In the parent file above, we fork child.js
(which will execute the file with the node command) and then we listen for the message
event. The message
event will be emitted whenever the child uses process.send
, which we’re doing every second.
To pass down messages from the parent to the child, we can execute the send
function on the forked object itself, and then, in the child script, we can listen to the message
event on the global process
object.
When executing the parent.js
file above, it’ll first send down the { hello: 'world' }
object to be printed by the forked child process and then the forked child process will send an incremented counter value every second to be printed by the parent process.
Answers 4
There are a number of articles that explain this such as this one
The long and the short of it is that nodejs
is not really a single threaded application, its an illusion. The diagram at the top of the above link explains it reasonably well, however as a summary
- NodeJS event-loop runs in a single thread
- When it gets a request, it hands that request off to a new thread
So, in your code, your running application will have a PID of 1 for example. When you get request T1 it creates PID 2 that processes that request (taking 1 minute). While thats running you get request T2 which spawns PID 3 also taking 1 minute. Both PID 2 and 3 will end after their task is completed, however PID 1 will continue listening and handing off events as and when they come in.
In summary, NodeJS
being 'single threaded' is true, however its just an event-loop listener. When events are heard (requests), it passes them off to a pool of threads that execute asynchronously, meaning its not blocking other requests.
Answers 5
The V8 JS interpeter (ie: Node) is basically single threaded. But, the processes it kicks off can be async, example: 'fs.readFile'.
As the express server runs, it will open new processes as it needs to complete the requests. So the 'readImage' function will be kicked off (usually asynchronously) meaning that they will return in any order. However the server will manage which response goes to which request automatically.
So you will NOT have to manage which readImage
response goes to which request.
So basically, T1 and T2, will not return concurrently, this is virtually impossible. They are both heavily reliant on the Filesystem to complete the 'read' and they may finish in ANY ORDER (this cannot be predicted). Note that processes are handled by the OS layer and are by nature multithreaded (in a modern computer).
If you are looking for a queue system, it should not be too hard to implement/ensure that images are read/returned in the exact order that they are requested.
Answers 6
Since there's not really more to add to the previous answer from Marcus - here's a graphic that explains the single threaded event-loop mechanism:
0 comments:
Post a Comment