Skip to content
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

Enable consuming of ES modules in extensions #130367

Open
igorskyflyer opened this issue Aug 8, 2021 · 103 comments
Open

Enable consuming of ES modules in extensions #130367

igorskyflyer opened this issue Aug 8, 2021 · 103 comments
Assignees
Labels
extension-host Extension host issues extensions Issues concerning extensions feature-request Request for new features or functionality
Milestone

Comments

@igorskyflyer
Copy link

When developing extensions and using either JavaScript or TypeScript we are unable to consume ES modules, only somewhat legacy CommonJS modules, setting the type to module and rewriting the extension to use import instead of require breaks the extension, generating an exception that states that all modules should use import instead of require in internal VS Code JavaScript files, I conclude it's caused by the type: module that forces Node to treat all .js files as ES modules. Tried using TypeScript which transpiles its own syntax to CommonJS module - so that's a no, I have also tried using just .mjs extension, again the same issue.

What is the status of this issue and are there plans to enable using of ES modules in extension development? That (could) bring somewhat big performance gains when bundling extensions with, for example, esbuild because it would enable tree-shaking - dead code removal, thus loading only necessary code. But I think this is not an extension API only issue, right? This needs to be done for VS Code itself?

@vscodebot
Copy link

vscodebot bot commented Aug 8, 2021

(Experimental duplicate detection)
Thanks for submitting this issue. Please also check if it is already covered by an existing one, like:

@igorskyflyer
Copy link
Author

Closing after 15 days of no reply.

@alexdima
Copy link
Member

cc @jrieken

@alexdima alexdima added extension-host Extension host issues feature-request Request for new features or functionality web Issues related to running VSCode in the web and removed web Issues related to running VSCode in the web labels Aug 25, 2021
@Lemmingh
Copy link
Contributor

Lemmingh commented Sep 3, 2021

Sounds like a duplicate of #116056. Is it possible to reopen that issue?


The VS Code extension host currently only accepts CJS module, as shown in the (trimmed) error message below:

Activating extension failed

Error [ERR_REQUIRE_ESM]: Must use import to load ES Module
require() of ES modules is not supported.

Instead change the requiring code to use import()

at internal/modules/cjs/loader.js:823:14

at require (internal/modules/cjs/helpers.js:88:18)
at Function.t [as __$__nodeRequire] (c:\Program Files\Microsoft VS Code\resources\app\out\vs\loader.js:5:101)
at v._loadCommonJSModule
at v._doActivateExtension
at v._activateExtension (c:\Program Files\Microsoft VS Code\resources\app\out\vs\workbench\services\extensions\node\extensionHostProcess.js:99:11695)

However, I think ES module will be more convenient in future.

Besides, TypeScript can transpile dynamic import() in CJS module in a surprising way (microsoft/TypeScript#43329), while it's never a problem in ES module where import() is emitted as is. VS Code extension authors can benefit from building extension as ES module.

@andyleejordan
Copy link
Member

For what it's worth, I ran into this trying to update node-fetch from 2.x to 3.x, since the newer version is ESM only. Looks to be the direction things are going, so extension developers are only more likely in the future to run into issues due to lack of ESM support.

@andyleejordan
Copy link
Member

FYI @TylerLeonhardt one of the Code issues I'd love to see fixed 😃

LinqLover added a commit to LinqLover/downstream-repository-mining that referenced this issue Oct 2, 2021
@Lemmingh
Copy link
Contributor

TypeScript 4.5 will perhaps have a new module option called node12, which preserves import() in CJS module. Then, although your entry point still have to be a CJS module now, you can load ES modules internally in an asynchronous manner.

See

@andyleejordan
Copy link
Member

Excuse me, @TylerLeonhardt, do you know why my comment above was marked as spam? I have to assume that was a mistake.

@alexdima
Copy link
Member

alexdima commented Nov 1, 2021

@andschwa I'm sorry, it might have been me that marked the comment as spam, but I don't remember doing it. I personally tend to hide comments that do not bring any value to the underlying discussion or distract from it.

@igorskyflyer
Copy link
Author

Latest vscode I use esm work fine 🎉🎉🎉

Which repositories?
This target is cjs yxw007/vscode-translate-next@c61d60d/esbuild.js#L32-L33 (blame)

Agreed, very important to clarify using ESM modules and compiling to CJS is not the same as using them "natively"

@yxw007,
this was always possible... Currently, only VS Code itself runs with ESM, extensions still don't allow it.

@yxw007
Copy link

yxw007 commented Oct 9, 2024

@igorskyflyer @CoffeeChaton @JustinGrote
I just changed the packaging results to esm for testing, and found that it cannot be used normally, but it is still possible to use esm in the code.

@JustinGrote
Copy link
Contributor

@igorskyflyer @CoffeeChaton @JustinGrote
I just changed the packaging results to esm for testing, and found that it cannot be used normally, but it is still possible to use esm in the code.

You need to be very careful what you are saying. You are not "using esm", you are transpiling to commonjs, it's a big difference.

@NullVoxPopuli
Copy link

So can we use untranspiled ESM?
like, if I author extension in JS (in ESM), I should be able to ship it without compiling?

@JustinGrote
Copy link
Contributor

@NullVoxPopuli not yet, while vscode itself is using ESM now, the extension engine is still bootstrapped via AMD, so extensions still have a commonjs entrypoint and thus need to continue to be commonjs when compiled.

@felipecrs
Copy link
Contributor

felipecrs commented Oct 31, 2024

Maybe someone from vscode can clarify what is the game plan to get ESM support for extensions?

@JustinGrote
Copy link
Contributor

JustinGrote commented Oct 31, 2024

I am not a vscode team member, but it has to be fixed in the vscode-loader repository is my current understanding, here is the upstream issue for that: microsoft/vscode-loader#56

@ilg-ul
Copy link

ilg-ul commented Oct 31, 2024

the upstream issue for that: microsoft/vscode-loader#56

That issue is dated February 29, 2024 and it doesn't seem very active. :-(

Is this the only remaining issue to be fixed?

ArquintL added a commit to viperproject/vs-verification-toolbox that referenced this issue Nov 7, 2024
@axefrog
Copy link

axefrog commented Nov 20, 2024

@alexdima @jrieken Hey any chance you could weigh in on this? I've been Googling for about half an hour and it's quite hard to find any kind of definitive advice on what those of us implementing a VSCode extensions should be doing given how much of the world has moved on to ESM modules by default.

@JustinGrote
Copy link
Contributor

@axefrog #130367 (comment)
Just use esbuild or bundler of your choice to compile to commonJS, you can use pretty much all the ESM libraries out there for now, it's just not optimal till this gets resolved.

@NullVoxPopuli
Copy link

Compilation is easy.
I want to debug source without transpilation and sourcemaps

@JustinGrote
Copy link
Contributor

Compilation is easy. I want to debug source without transpilation and sourcemaps

VSCode makes sourcemaps really easy to work with so that you pretty much can't tell the difference, you can also use tools like esbuild/register to work with the ts files "directly" (they get transpiled on the fly in node)

@axefrog
Copy link

axefrog commented Nov 20, 2024

@JustinGrote Thankyou! Works beautifully. For anyone else arriving here, I'm using Bun.build for this. It is shockingly fast.

CLI:

bun build --entrypoints ./src/extension.ts --outdir ./out --target node --format cjs --sourcemap=linked --external vscode --watch

Documentation: https://bun.sh/docs/bundler

@bmingles
Copy link

bmingles commented Nov 20, 2024

There are now instructions in vscode docs for configuring esbuild:
https://code.visualstudio.com/api/working-with-extensions/bundling-extension#using-esbuild

This can also be auto configured as one of the options when scaffolding a new extension using yeoman

@tats-u
Copy link
Contributor

tats-u commented Dec 10, 2024

require(esm), one of the candidates for the escape hatch, is unflagged (still marked as experimental) in the latest Node 22 (22.12).
Waiting for being merged into Electron and VS Code.

@JustinGrote
Copy link
Contributor

There are now instructions in vscode docs for configuring esbuild: code.visualstudio.com/api/working-with-extensions/bundling-extension#using-esbuild

This can also be auto configured as one of the options when scaffolding a new extension using yeoman

Just again want to make a note, because this is a thread about esm, these instructions are for using esbuild to transpile ES modules to commonjs, they do not provide native esm support.

@felipecrs
Copy link
Contributor

felipecrs commented Jan 2, 2025

For me, transpiling to commonjs with esbuild did not help. I'm still not able to consume ESM-only dependencies like newest version of execa.

I was able to write ESM-like code at least, but that isn't very useful.

@igorskyflyer
Copy link
Author

For me, transpiling to commonjs with esbuild did not help. I'm still not able to consume ESM-only dependencies like newest version of execa.

I was able to write ESM-like code at least, but that isn't very useful.

@felipecrs, do you have a working repository for it so I can do some tests tomorrow? 😄

@felipecrs
Copy link
Contributor

@igorskyflyer, yeah: https://github.com/vscode-shellcheck/vscode-shellcheck. My last attempt: vscode-shellcheck/vscode-shellcheck#1555.

@igorskyflyer
Copy link
Author

@igorskyflyer, yeah: https://github.com/vscode-shellcheck/vscode-shellcheck. My last attempt: vscode-shellcheck/vscode-shellcheck#1555.

Thanks, I'll check it out tomorrow. 🙂

@igorskyflyer
Copy link
Author

@igorskyflyer, yeah: vscode-shellcheck/vscode-shellcheck. My last attempt: vscode-shellcheck/vscode-shellcheck#1555.

@felipecrs, sorry for the delay, I was busy with personal stuff. Anyway, after upgrading execa and minimatch I got a few errors in the tool-check.ts and the linter.ts files. After fixing the given import errors, I got it to compile to a proper CommonJS (and a JS map file) but I cannot seem to test it, I get the error:

Error: there is no registered task type 'npm'. Did you miss installing an extension that provides a corresponding task provider?
Activating task providers npm
Error: Invalid problemMatcher reference: $esbuild-watch

Even though I have npm installed and the extension working and running. Did you get to that point or you weren't even able to compile it?

@yne
Copy link

yne commented Jan 5, 2025

Just to clarify, it has been possible to import .mjs from a .js file, here is a minimal example:

lib.mjs

export default 42

main.js (.mjs files are refused even if they have no import/export statements)

exports.activate = async function () {
    console.log((await import("./lib.mjs")).default)
}

package.json

{
	"name": "basic",
	"version": "0.1.0",
	"main": "./main.js",
	"engines": { "vscode": "^1.96.0" },
	"activationEvents": [ "onStartupFinished" ]
}

Open the .js file, run > start debugging > Vscode Extension, and you shall see a 42 in your host vscode's debug console

However, As @JustinGrote said, the goal is to bootstram .mjs extension. and the upstream

So I tried on 1.96 the following changes to the package.json, but with no success (and no surprises):

+	"type": "module",
+	"module": "./basic.mjs",
-	"main": "./main.js",

=> this does not even try to run basic.mjs

+	"type": "module",
 	"main": "./main.mjs",

=> this fail with require() of ES Module basic.mjs not supported

Hope this clarify the current state of the affair, because a lot of comment seems to be bundling/transpiling issues.

@felipecrs
Copy link
Contributor

@igorskyflyer let's continue that conversation in vscode-shellcheck/vscode-shellcheck#1555 (comment) to avoid spamming other one's inboxes.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
extension-host Extension host issues extensions Issues concerning extensions feature-request Request for new features or functionality
Projects
None yet
Development

Successfully merging a pull request may close this issue.