-
-
Notifications
You must be signed in to change notification settings - Fork 771
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
Spying on ESM default export fails/inexplicably blocked #2258
Comments
There's actually no mention of Could you post some straight vanilla javascript example where you are able to overwrite the (non-default) exports of an imported ES Module? You need two files This test suite documents Sinon's current behavior. |
FYI stub doesn't work for TypeScript@3.9.2+ here |
@fatso83 that old issue indeed does not (but I didn't say it did 😜). What I mean is, Sinon throws an error when detecting an ESM because later down the code object.__defineGetter__(property, methodDesc.value); Buuuut, when I tried modifying the Sinon code and running it, I got an undefined error on So I'm thinking perhaps a different way, such as re-creating the object (with new values) instead of overriding its properties. More complex, but then at least do-able. |
Same issue here last working version 3.8.3 |
@zorji and @aelbore : I have no idea why you are posting in this thread. If you are reporting a new issue, please create a separate issue and follow the normal steps for reporting (how to verify, actual code, versions, etc). We do not support Typescript and have never said so, although it should work, of course, but the details will depend on your transpilation step. |
Is it? An ES Module is a static feature that is bound at runtime, AFAIK, and you cannot modify it.
https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Modules Looking forward to seeing if you can get anywhere with this, but I would be surprised if you (or anyone!) did. |
This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions. |
Bad bot. Can this stupid thing be turned off? |
@OmgImAlexis do you think there's any reason to keep this issue open? |
Folks, please try to keep the issue on point and only discuss what's mentioned in the original post. It is challenging enough for the maintainers to spend our unpaid free time supporting free software, without also having to navigate orthogonal topics in the same issue. We are delighted that Sinon (a JavaScript tool) can also be useful in the TypeScript community. However, like @fatso83 said, we do not support Typescript. Any further off-topic comments about TypeScript in this issue will be deleted. |
@mroderick if you don't like the comments then ignore them. Actively removing comments which would help other users isn't helpful at all. Let's say you remove the comments then you'll just end up with similar users coming and commenting again and again which in turn would end up with the thread being locked. Edit: You can also use the super easy to find "unsubscribe" button. :) |
Ack sorry, I got very occupied at work. I'll try to take a look in a couple weeks. Several libraries accomplish this, so it must be possible 😁 |
I am a maintainer of this project, and try to read all the comments. I can't ignore comments or unsubscribe from an issue because people decide to not play by the rules set out, ignore the guidance offered and hijack the original issue for their own purposes. It is my responsibility to try to resolve issues in a polite, tactful and efficient manner. Hijacking issues makes it harder than it needs to be to support the community. Please help me help you and the rest of the community. |
This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions. |
Bad bot. |
@jshado1 do you think this issue should stay open? |
I see that @jshado1 says this used to work fine, but I don't think this has ever worked in vanilla un-transpiled javascript. Because it should not. It can work when transpiling modules using Babel/Webpack or similar, because you only emulate modules and not actually create a read only object in the runtime (which modules are). The throw clause was added by me, because it would otherwise throw a little bit later when you tried changing the unchangeable. Runtimes/module loaders such as If you think the behavior is wrong, I suggest just editing |
To quote @RyanCavanaugh (microsoft/TypeScript#38568 (comment))
Which is basically what I have been saying. So closing, as the results of this is entirely dependent on you using a transpilation step, how the transpiler does that step, etc. If it tries to create a result that is somewhat in accordance with the ESM spec, it should not work as the exports of the ES Modules are immutable (not the actual objects bound by those exports, of course). |
@mroderick @fatso83 sorry i never got that time off and have been slammed at work since. It's been several months, but if memory serves, I had a very rudimentary (and dirty AF) working example back then to confirm it did work (but indeed probably shouldn't have). BUT, I think that's not the way to go beccccauuse: I believe there is / will be an official way to do this via ems loaders: https://github.com/nodejs/node/blob/master/doc/api/esm.md#loaders These are currently unstable (explicitly stated to soon be changed), so probably hold off til they get more stable. But once they do, I think that's the way to go. |
Update: This is currently being discussed in nodejs/node#36396. I think my proposal in this comment would make this very simple. A user could manually supply a map, and the esm loader could use that to substitute: // const mocksMap = { 'serviceA.js': 'serviceA.mock.js' };
const mock = mocksMap[importPath]; // note: it's not called `importPath` in the loader hooks
if (mock) // … and/or, the esm loader could look for a mock match based on filename (quick n dirty example): const ext = path.ext(importPath);
const filename = path.basename(importPath, ext);
const mockFile = await import(`${filename}.mock${ext}`);
if (mockFile) // … |
@jshado1 Thanks for providing that link. The linked issue is essentially about trying to find a standardized way of handling the issue of substituing dependencies on the link level, which is similar to what tools such as rewire and proxyquire does (which we describe in this how-to on our home page). In any case, this is not something Sinon should or will solve, as it is out of scope and how to solve it will be entirely dependant on your environment. |
Describe the bug
Previously, a combination of
import * as foo from 'someModule'
+spy(foo, 'default')
would work just fine (and was even listed in this repo's issues as the solution); however, now this is explicitly blocked by a manual throw.Expected behavior
Sinon to spy on the default export.
Context (please complete the following information):
Additional context
I presume this is in order to avoid
TypeError: Cannot assign to read only property
However, this can be circumnavigated by using the old skool
__defineGetter__
The text was updated successfully, but these errors were encountered: