- 
                Notifications
    You must be signed in to change notification settings 
- Fork 2.1k
Use no-op for production instanceOf #4426
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
Conversation
| Tagging people from #4386 @JoviDeCroock @benjie @glasser | 
Re-implements graphql#4386 with a configurable function variable initialized to a no-op function in production builds. This is apparently optimizable by v8, as it has no performance hit when setEnv('development') is not called. This should also allow for tree-shaking in production builds, as the development instanceOf function should not be included within the production bundle if setEnv is not called, although this has not been confirmed. To avoid a performance hit, setEnv('development') calls setInstanceOfCheckForEnv('development') which sets the function variable to developmentInstanceOfCheck. Setting a property on globalThis was tested, but reading the property led to a small performance hit. Co-Authored-By: Yaacov Rydzinski <[email protected]>
| Just to note — this is still the same semantics as #4386 where the default is always production? Honestly I don't really understand the point of doing it this way. It's a really elaborate set of gymnastics to go through just to provide a better error if you've misconfigured things than things silently failing due to instanceof checks being spuriously false ... but you still have to know to turn it on? By what mechanism do we expect people to choose to turn this on? Is there going to be some sort of tell-tale error message or behavior that, when they Google it, will tell them "oh this might be a multiple-installation issue, run  But in that case, couldn't the instructions just say "run  If this was on by default and disabled by  | 
| In previous discussions, we basically landed on the fact that our defaults should make sure that devs not informed about our  More context: So that combination means that the error message will be less discoverable, usually reached by getting a particularly puzzling error message and getting eventually to some issues like: But actually a more cryptic error message after this change! 
 As seen in the above issue, it's sometimes difficult to see that there are really two realms or two different versions of graphql because of bundling, and having the additional access point may indeed be somewhat helpful. Basically, I think we acknowledge the issue of discoverability and reduced utility you raise, but we are forced there based on other constraints. At least, that's where the consensus seems to be. This seems like something we could discuss together at a graphql-js-wg meeting (which have been light on agenda recently). | 
| In Grafast we default to production BUT we output a message if you didn't specifically enable it - i.e. we tell the user that they should choose development or production mode, but we default to production because it's safer from a denial of service point of view (i.e. limits error count, less internal detail in errors, etc etc). Developers hate these kinds of messages, so it's a strong motivation for them to explicitly choose a suitable environment :) | 
| @benjie that's a helpful workflow! I added a commit to this PR that allows checking the value of  Integrations can then decide to appropriately emit a warning (or error message?) via the appropriate channel if the env was not set, as they might wish to provide this better workflow? In theory, we could internally within graphql-js check this value and print a one-off error message let's say on  | 
| To make sure tree shaking works , I think we would need separate environment sets for production and development. And of course we need to test that tree shaking actually still works. | 
| Sadly, if we make our no-op function log a message on first use if the environment is not set, it no longer, the performance hit reappears, presumably because it is no longer optimized out. We could: 
 | 
| On further reflection, the promise of dynamically setting the environment does not work well with this "no-op" trick. Once we switch once to development and use our function, the no-op will be deoptimized and not re-optimized if we switch back to production. I've added a commit that prevents the mode from being reset. Still doing option 2 from list above, defaulting to production. | 
| My instinct continues to be that this is a lot of complexity for a feature that's supposed to protect against accidental user error but is off by default. But I respect that I'm in the minority here. Just chiming in again because I had been explicitly tagged earlier — don't consider me a blocker on this! | 
| An ENVVAR is definitely the more standard way to do it. Inventing our own thing is likely to bring more problems over time. | 
| Closing in favor of #4437 | 


Re-implements #4386 with a configurable function variable initialized to a no-op function in production builds. This is apparently optimizable by v8, as it has no performance hit when setEnv('development') is not called.
This should also allow for tree-shaking in production builds, as the development instanceOf function should not be included within the production bundle if setEnv is not called, although this has not been confirmed.
To avoid a performance hit, setEnv('development') calls setInstanceOfCheckForEnv('development') which sets the function variable to developmentInstanceOfCheck. Setting a property on globalThis was tested, but reading the property led to a small performance hit.
UPDATE: this version only allows
setEnv(...)to be called once, as the no-op trick will not work if we initially set to development and then reset to production, we will have been deoptimized and will not be reoptimized.