-
Notifications
You must be signed in to change notification settings - Fork 30.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
http.Parser would be great to have. #21202
Comments
No, I think we’re good here.
Probably, yes.
If HTTP code would have to use an additional layer of streams (in particular of streams in the sense of the It might be easier to do the reverse thing and provide an API (maybe as an npm module) on top of our current code? I think the |
This is what I'm doing for requests from the client, but it doesn't cover reponses from the onward server (unless I've missed a trick). Until recently I didn't need access to individual responses and could pipe the onward server socket directly to the client socket. New requirements have meant I need access to individual responses now. I do have a solution which appears to work, but it's using private stuff and is pretty gnarly. EDIT: I need to read more carefully! This looks promising:
I'll definitely check that out. |
So, the issue is that http.request assumes a 1:1 relationship between request and responses, but in your model requests and responses are not necessarily correlated that way? Am I understanding that correctly? That doesn’t seem trivial to cover, yes… |
Ah, so my use case is rather strange. Amongst other things, the proxy I'm building caches certain responses from the onward server (each keyed on the request from a client which led to it). The need to build a key means parsing the request, and the need to cache the response means knowing where its boundaries are within the response stream (preferably more, but this would suffice). My crude first pass at this has at it's heart this code like: // where:
// - chunk is a chunk from the response stream
// - message is an array
// - parser is an HTTPParser instance
function write(chunk) {
for (const byte of chunk) {
message.push(byte);
parser.execute(Buffer.from([byte]));
}
}
This is going to perform badly since it's working byte-by-byte, but I have the benefit of knowing that |
I'd say Why don't you use https://www.npmjs.com/package/http-parser-js? It follows the http-parser API closely and is not much slower. (There's also https://github.com/bnoordhuis/node-http-parser but I haven't touched that in years and probably needs some work by now.) |
That'd be a drop-in replacement, so definitely worth keeping in mind should the binding change and break my proxy. This issue asked a few questions about the hypothetical Given this outcome, should the linked comment in the code be updated or removed? |
That comment looks outdated to me. Do you want to file a pull request removing it? |
No problem. Would you like me to remove the |
Yep, that'd be good. |
Thanks for the discussion folks! This was really interesting. |
Fixes: #21202 PR-URL: #21214 Reviewed-By: Ben Noordhuis <[email protected]> Reviewed-By: Michaël Zasso <[email protected]> Reviewed-By: Gireesh Punathil <[email protected]> Reviewed-By: Anna Henningsen <[email protected]> Reviewed-By: James M Snell <[email protected]> Reviewed-By: Weijia Wang <[email protected]> Reviewed-By: Luigi Pinca <[email protected]> Reviewed-By: Trivikram Kamat <[email protected]> Reviewed-By: Colin Ihrig <[email protected]>
Fixes: #21202 PR-URL: #21214 Reviewed-By: Ben Noordhuis <[email protected]> Reviewed-By: Michaël Zasso <[email protected]> Reviewed-By: Gireesh Punathil <[email protected]> Reviewed-By: Anna Henningsen <[email protected]> Reviewed-By: James M Snell <[email protected]> Reviewed-By: Weijia Wang <[email protected]> Reviewed-By: Luigi Pinca <[email protected]> Reviewed-By: Trivikram Kamat <[email protected]> Reviewed-By: Colin Ihrig <[email protected]>
I've recently been doing some work in which I needed access to the individual requests and responses from a pair of streams (it's a proxy of sorts and I'm only working with HTTP/1.1). For requests, recent changes to
http.Server
have meant I can fool a server instance into parsing request objects from a stream. Responses have been more challenging...As it stands I've had to make use of the
http_parser
binding andIncomingMessage.prototype._addHeaderLine
, both of which are internals. In my quest to understand how to use thehttp_parser
binding, I came across this comment:// TODO: http.Parser should be a Writable emits request/response events.
This
Parser
would eliminate all the hacks and use of private API in my code!Alas, the comment is rather old. It got me thinking about how it might be implemented though. My understanding is that
Parser
would have to be instantiated either for requests or responses since it would wrap an underlyingHTTPParser
instance. On that assumption, my questions are:Parser
in principle)?Parser
lead to a nasty performance hit which my limited understanding has missed?The text was updated successfully, but these errors were encountered: