Skip to content

[@rollup/plugin-typescript] { compilerOptions: { module: "nodeNext" } } overwritten when coming from "extends" in tsconfig.json #1583

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

Closed
jessevanassen opened this issue Sep 14, 2023 · 5 comments

Comments

@jessevanassen
Copy link

jessevanassen commented Sep 14, 2023

Expected Behavior

I have two typescript config files: tsconfig.base.json and tsconfig.json.
In the tsconfig.base.json I have { compilerOptions: { module: "nodeNext" } }.
tsconfig.json extends tsconfig.base.json.

When I run the typescript config, I expect that it takes the module: "nodeNext" into account that I configured in the tsconfig.base.ts.

Actual Behavior

The plugin overwrites { compilerOptions: { module: "nodeNext" } } with { compilerOptions: { module: "esNext" } }. This produces the following error with TypeScript 5.2 and up (see also microsoft/TypeScript#54567):

(!) Plugin typescript: @rollup/plugin-typescript TS5110: Option 'module' must be set to 'NodeNext' when option 'moduleResolution' is set to 'NodeNext'.

Additional Information

Cause of the bug

https://github.com/rollup/plugins/blob/typescript-v11.1.3/packages/typescript/src/options/tsconfig.ts#L155-L196

export const DEFAULT_COMPILER_OPTIONS: PartialCompilerOptions = {
  module: 'esnext',
  skipLibCheck: true
};

// ...

ts.parseJsonConfigFileContent(
  {
    ...tsConfigFile,
    compilerOptions: {
      ...DEFAULT_COMPILER_OPTIONS,
      ...tsConfigFile.compilerOptions
    }
  },
  // ...
);

This block gets the JSON parsed tsConfigFile that's used to process the project (in the reproduction, thats ./tsconfig.json), adds the default options to the compilerOptions, and passes it on to TypeScript to completely parse the file, fill in defaults, etc.

However, if the tsconfig.json contains an extends which defines the module, the tsconfig.json ends up like this:

{
  "compilerOptions": {
    module: 'esnext',
    skipLibCheck: true,
    composite: true,
  },
  extends: "./tsconfig.base.json",
}

These compilerOptions take precedence over the one in the tsconfig.base.json, effectively overriding the project's module setting.

Workaround

Move / copy { compilerOptions: { module: "nodeNext" } } to the tsconfig.json file.
This way, ...tsConfigFile.compilerOptions overwrites the property from ...DEFAULT_COMPILER_OPTIONS. However, it does lead to repetition that the extends should solve.

@jessevanassen jessevanassen changed the title [typescript] { compilerOptions: { module: "nodeNext" } } overwritten when using "extends" in tsconfig.json [@rollup/plugin-typescript] { compilerOptions: { module: "nodeNext" } } overwritten when using "extends" in tsconfig.json Sep 14, 2023
@jessevanassen jessevanassen changed the title [@rollup/plugin-typescript] { compilerOptions: { module: "nodeNext" } } overwritten when using "extends" in tsconfig.json [@rollup/plugin-typescript] { compilerOptions: { module: "nodeNext" } } overwritten when coming from "extends" in tsconfig.json Sep 14, 2023
@romainmnr
Copy link

I'm facing the same issue...

Move / copy { compilerOptions: { module: "nodeNext" } } to the tsconfig.json file.

This workaround works, but we loose all the purpose of extending a config...

@renke0
Copy link

renke0 commented Jan 29, 2024

Explicitly adding it to the plugin declaration also works. Not ideal, but IMO it's better than having to change tsconfig.json.

plugins: [typescript({ module: 'nodeNext' })]

@sam-mfb
Copy link

sam-mfb commented Jul 29, 2024

you can also dynamically add it to to the plugin declaration using something like this in a rollup.config.mjs file (or .cjs file with some modifications):

import {
  parseJsonConfigFileContent,
  sys
} from "typescript"
import { readFileSync } from "fs"

const getTsConfig = configFilePath => {
  const configFile = readFileSync(configFilePath, "utf8")
  const configObject = JSON.parse(configFile)
  const configParseResult = parseJsonConfigFileContent(
      configObject,
      sys,
      process.cwd()   )
  return configParseResult
}
const configObj = getTsConfig("./tsconfig.rollup.json")

// other stuff

plugins: [typescript({
  module: configObj.options.module === 199 ? "nodeNext" : undefined //199 is what the typescript enum value ModuleKind.NodeNext resolves to
})]

Presumably some version of the getTsConfig function above could be used to actually fix the readTsConfigFile that is the source of the problem. the basic issue is that it uses ts.readConfigFile() rather than ts.parseJsonConfigFileContent() to parse the tsconfig.json file and so it doesn't handle walking through the extends files. i apologize i don't have time to figure that out and do a PR now.

@jelovac
Copy link

jelovac commented Nov 7, 2024

Run into this issue as well.

The affected code is located here:

I believe that the whole tsconfig.ts module is due for a refactor as its current behavior is not correct and misleading.

The following discussion might help with resolution: microsoft/TypeScript#44573 . There I found mention of https://github.com/dominikg/tsconfck package which can help with parsing of tsconfig files.

jelovac added a commit to jelovac/rollup-plugins that referenced this issue Nov 8, 2024
This refactor updates readTsConfigFile to correctly handle nested "extends" properties by using TypeScript’s parsing mechanism instead of basic JSON reading. The new behavior returns a unified JSON object that consolidates all properties from extended tsconfig files.

This change addresses issue rollup#1583, ensuring complete config resolution across extended tsconfigs while retaining existing behavior with minimal changes and no breaking modifications.
jelovac added a commit to jelovac/rollup-plugins that referenced this issue Nov 8, 2024
…tion" from extended config

This test asserts that the fix in rollup#1812, which addresses issue rollup#1583, fully inherits both "module" and "moduleResolution" options from the extended tsconfig. Without this fix, the TypeScript compiler would throw an error as described in rollup#1583.
jelovac added a commit to jelovac/rollup-plugins that referenced this issue Nov 11, 2024
@stale stale bot added the x⁷ ⋅ stale label Apr 26, 2025
Copy link

stale bot commented Apr 27, 2025

Hey folks. This issue hasn't received any traction for 60 days, so we're going to close this for housekeeping. If this is still an ongoing issue, please do consider contributing a Pull Request to resolve it. Further discussion is always welcome even with the issue closed. If anything actionable is posted in the comments, we'll consider reopening it.

@stale stale bot closed this as completed Apr 27, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

6 participants
@jessevanassen @jelovac @romainmnr @renke0 @sam-mfb and others