- 
                Notifications
    You must be signed in to change notification settings 
- Fork 13.1k
Description
TL;DR: A new compiler option mainFields for selecting multiple fields in package.json instead of just package.json#main.
There are lots of related issues to this one (which I link to below), but I want to focus on just this specific proposal.
Packages often look like this:
{
  "name": "my-package",
  "main": "dist/cjs/index.js",
  "module": "dist/esm/index.js"
  "source": "src/index.ts"
}Notice how we have multiple fields which specify multiple entry points. These entry points all refer to the same code, just in different compile states and configurations.
Many tools use these fields in order to find the entry point that they care about. For example, tools like Webpack and Rollup will use package.json#module in order to find ES modules. Other tools will use fields like package.json#source (or src) for local package development.
While these fields aren't part of the official Node module resolution algorithm. They are a community convention which has proven to be useful in lots of scenarios.
For TypeScript, one such scenario that this would be useful for is with multi-package repos or "monorepos". These are repositories where the code for multiple npm packages exist and are symlinked together locally.
/project/
  package.json
  /packages/
    /package-one/
      package.json
      /node_modules/
        /package-two/ -> ../../package-two (symlink)
    /package-two/
      package.json
Inside each package, you'll generally have a src/ directory that gets compiled to dist/
/package-two/
  package.json
  /src/
    index.ts
  /dist/
    index.js
    index.d.ts
Right now it is really painful to use TypeScript with one of these repos. This is because TypeScript will use the package.json#main to resolve to the packages dist folders. The problem with this is that the dist folders might not exist and if they do exist they might not be compiled from the most recent version of src.
To work around this today you can add a index.ts file in the root of each of your packages to point to the right location and make sure that the root index.ts file does not get shipped to npm.
/package-two/
  index.ts
  /src/index.ts
// package-two/index.ts
export * from './src/index'It sucks that you need this file, and if you ever forget to create it in a new package, you'll revert back to really crap behavior.
If, instead of all that, TypeScript supported a new compiler option mainFields which looked like:
{
  "compilerOptions": {
    "mainFields": ["source", "main"]
  }
}You could add package.json#source (in addition to package.json#main) and resolve it to the right location locally.
The algorithm would look like this:
For each mainField:
- Check if the package.jsonhas a field with that name
- If the package.json does not have the field, continue to next mainField
- If it field exists, check for a file at that location.
- If no file at that location exists, continue to the next mainField
- If the file exists, use that file as the resolved module and stop looking
I think this is the relevant code:
Related Issues:
- Path mapping in yarn workspaces #21137 "Path mapping in yarn workspaces"
- Module resolution for sub-packages picks d.ts file when .ts file is available #20248 "Module resolution for sub-packages picks d.ts file when .ts file is available"
- Support .mjsoutput #18442 "Support.mjsoutput"