-
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
Migration from process.binding #22064
Comments
i have reservations about exposing anything on process.binding('util') publicly... stack decoration shouldn't have a node-specific solution and promise and proxy details can't be equally represented by the engines that might want to run node. taking a step back, i think we should look at how the things these apis enabled would be implemented if these apis had never existed, instead of straight-up replacements. |
Cool. As long as there's some way, it would be nice. (#21958 is a step to providing a way)
Right, Chakra had to do extra work to support those.
👏 That's a great exercise! |
@jdalton have you seen https://nodejs.org/api/inspector.html#inspector_inspector_console to print messages into the inspector console? |
Yes. I believe you're saying I can create my own form of |
It would be great if you could confirm that, so we can rule out one item from that list. |
Confirmed. Looks like |
I am not sure that I am not missing some context, but the vital part of console.log behavior at least on inspector side - it is providing stack trace that points to the place where console.log was called. I am worried that with plain JS function we will get JS stack frame that contains location inside this function. |
Would you be up for providing a small repro case so I can test it out and see what's what? |
node/lib/internal/bootstrap/node.js Lines 441 to 444 in ce98e2e
right here the two functions are wrapped to be called by a native function (consoleCall is c++) Lines 157 to 181 in ce98e2e
|
@jdalton you can try replacing the c++ consoleCall with this if you wanna play around with it. // because this is written in js it will be added as an additional frame
// which doesn't show up in the c++ version
function consoleCall(inspector_method, node_method, config_object, ...args) {
const in_call_key = 'in_call';
if (!(in_call_key in config_object)) {
config_object[in_call_key] = true;
inspector_method(...args);
delete config_object[in_call_key];
}
node_method(...args);
} |
Thanks! I've got something on my end as well. I'm just looking for a test case to run it against is all. |
Test script: 'use strict';
const { Console } = require('console');
const originalConsole = require('inspector').console;
const config = {};
// Pretty straightforward port of InspectorConsoleCall() to JavaScript.
function consoleCall(inspectorMethod, nodeMethod, config, ...args) {
// Assume inspector is always on.
if (!('in_call' in config)) {
config.in_call = true;
inspectorMethod.apply(this, args);
}
delete config.in_call;
nodeMethod.apply(this, args);
}
const jsWrappedConsole = new Console(process.stdout);
jsWrappedConsole.error = consoleCall.bind(jsWrappedConsole, originalConsole.error, jsWrappedConsole.error, config);
function usesCPPWrapped() {
console.error('bleh');
}
function usesJSWrapped() {
jsWrappedConsole.error('bleh');
}
usesCPPWrapped();
usesJSWrapped(); Note how Hope this helps. |
Ah yep okay, since |
@jdalton you can do if my changes to error stack decoration ever land there will be no need for i would recommend creating os.getenv and os.setenv as a replacement for safeGetenv. i would strongly disagree with exposing proxy inspection as an api and i will to the best of my ability block exposing promise inspection |
Thanks @TimothyGu ! @jdalton, you can check the same thing inside ndb, just run it and evaluate console.trace in console, top frame should point to anonymous script, not to some line inside internal scripts. BTW, regardless inspector binding that I use inside ndb. In ideal world I would like to extend |
Let's hold off on elevating tone and focused on something more productive like following up on your previous comments...
Thank you for the feedback! |
@jdalton sorry i didn't mean to elevate, i was just trying to "check off" the items or so to speak |
@TimothyGu Maybe I'm not understading the use case for #22064 (comment). Is that a common use case for that API? |
I don't know why someone would need to do the same thing we are doing for I'm just demonstrating how the function must be written in C++, and could not be written in JavaScript with the same effect, which was @jdalton's question. |
again i think what we need to do with this is just ask "what would people be doing with the console if these apis never existed" |
@jdalton ... be careful relying on |
Tracking issue at #22160. |
This threat seems to have no mention of I am facing an issue where a website responds with an invalid header. https://gist.github.com/gajus/45e37f8e46e400f377baa8a8dd39c92a In earlier Node.js versions, it was possible to workaround this by overriding So aside from using Attempt to |
I have worked around this specific issue by using /**
* ___utmv* headers when server by Incapsula CDN may include invalid chracters
* that are causing 'Parse Error: Invalid header value char'.
* We work around the Parse Error by using `--http-parser=legacy`.
* However, we also strip the offending headers to avoid passing them
* down if the request is being proxied.
* @see https://gist.github.com/gajus/d02aa2c6c705fb8c3256a0108e4e03ff
*/
const sanitizeHeaders = (headers) => {
const append = {};
if (headers['set-cookie']) {
append['set-cookie'] = [];
for (const value of headers['set-cookie']) {
if (!value.startsWith('___utmv')) {
append['set-cookie'].push(value);
}
}
}
return {
...headers,
...append,
};
};
const main = async () => {
const got = require('got');
const response = await got('https://entradas.cinesa.es/compra/?performanceCode=185560&s=192');
console.log(sanitizeHeaders(response.headers));
};
main(); However, this means that I would not be able to workaround the issue if it was not possible to override |
Is this going to get fixed? I'd really prefer not to litter my codebase(s) with a fix for this. |
cc @nodejs/http ^ |
The legacy http parser has been removed from master as of ac59dc4 (#29589) and the above workaround will not work from Node.js 13+ forward. As @bnoordhuis points out in the thread in #21509, the current behavior is intentional as to avoid a very real security vulnerability, and it is working as intended. |
Is #3591 related to this at all? It still has not been resolved. |
To be honest, I don't really know @stevenvachon, it's been a while since I've looked that issue over. |
What is the security vulnerability? He doesn't state in that comment the implications. What is the workaround for when a remote service returns invalid response, such as in a scenario I describe in #22064 (comment) ? |
Strict handling of invalid characters is necessary to prevent a variety of request smuggling and response splitting style attacks. The correct workaround is to try to get the broken remote service fixed. Not sure what else we can do now that the legacy parser has been removed. |
was the question here about loosening how llhttp works or allowing the parser to be overridden? the latter seems like a reasonable request to me. |
That is an insane position to take. I run a large data aggregation network. There are literally hundreds of websites misbehaving even within our relatively restricted domain operation. This change literally makes Node.js not usable for us. |
@gajus Although I see the relationship, I feel like this is turning into a separate discussion from what the issue is about, and I’d suggest opening a new issue, ideally with a description of what exact “features” you need from the newer parser that the old one had. |
This has been discussed in the past (repeatedly and at length) and the answer has always been 'no' because it turns too many implementation details into frozen-forever public API. Custom JS parsers exist (link). You could even WASM-ify http-parser if you want bug-for-bug compatibility. I'm its maintainer and I'd be open to that. |
@devsnek overriding |
The HTTPParser override does not work in v12 and above. I have raised a new issue #30573. |
For all internal uses, we have migrated away from using |
With #22004 landed to doc deprecate
process.binding
I was encouraged to open an issue related to my use of it. Now is a great time to starting examining what fromprocess.binding
can be exposed in a user-friendly way.In my own experience I've attempted to use (if available and there were no user-facing options) these bindings. My usage fits into these categories
The API run down as follows:
process.binding("config")
experimentalREPLAwait (command line checks)experimentalWorker (command line checks)exposeInternals (command line checks)preserveSymlinks (command line checks)preserveSymlinksMain (command line checks)process.binding("inspector")
open (only existence checks, much like Node does)process.binding("util")
For chrome inspector wiring there has been some work to expose things like
originalConsole
(#21659), but more in this area is needed.For command-line checks the answer may be to use
process.execArgv
, but a moreconfig
like object form would be handy.For custom error stack decoration there is early feelers in #21958.
Custom inspection helpers like
getProxyDetails
/getPromiseDetails
andsafeGetenv
have nothing simmering at the moment.Updated:
I
crossed throughAPI that have user-facing options.Update:
Replaced
process.binding("inspector").open
inferences withprocess.config.variables.v8_enable_inspector
checks.The text was updated successfully, but these errors were encountered: