Typescript: Resolve tsconfig.json `extends` path using node_modules resolution logic

Created on 30 Sep 2017  Β·  38Comments  Β·  Source: microsoft/TypeScript

I am trying to manage my tsconfigs by keeping them in a repo/subrepo, and was hoping this would work:

{
  "extends": "my-config-repo/tsconfig.standard",
   "compilerOptions": { }
 }

But I get

tsconfig.json(2,14): error TS18001: A path in an 'extends' option must be relative or rooted, but 'my-config-repo/tsconfig.standard' is not.

Is there some reason why the path given to extends, if neither relative nor absolute, could not be searched for using the node_modules lookup rules?

(I would also like multiple extends, but I suppose there was some reason for not doing that, and that would be another ticket anyway.)

In Discussion Suggestion

Most helpful comment

Another vote for this. I just want to keep my tsconfig in one place in a lerna repo. Rather surprised to see there's no support already for this.

In my case I have multiple packages and those packages can either be built standalone or with lerna. If they are built as standalone then they will have node_modules in the package. If they are built using lerna the node_modules get hoisted into the parent directory. Therefore I cannot use the following solution

{
  "extends": "./node_modules/@foo/bar/tsconfig.json"
}

because it may also be here

{
  "extends": "../../node_modules/@foo/bar/tsconfig.json"
}

The ideal solution appears to be

{
  "extends": "@foo/bar/tsconfig.json"
}

Using node module resolution would fix this issue.

All 38 comments

Seems like a good idea. If this feature is added, I think it would be important that it also work with "paths".

Maybe this would be a small
step toward Project References :wink:

We excluded it during the initial implementation due to complexity, but left an error for it in so we could go back and add it later if we needed to. An issue here is that the module resolution strategy we use is determined by your compiler options... and an extends option can be specifying just what that module resolution scheme is, which is kind of a circular dependency - you must know your module resolution scheme to resolve an extends option, but your extends option can lead to where you specify some module resolution scheme!

@weswigham yeah that is paradoxical :fearful:. We find the source of truth only to learn from it that we were never meant to discover it...

How about escaping the paradoxical loop by just letting Node.js resolve this as it normally would?

BTW, for reference and inspiration, all of these support resolving this to a package export:

How good of an idea would it be to use "extends": "./node_modules/typescript-config-foo/index.json" until this is possibly implemented?

@mightyjam that's a relative path, so it's fine.

How about escaping the paradoxical loop by just letting Node.js resolve this as it normally would?

We do not always use node module resolution for your package (you have to specify in your configuration if you want node module resolution or a custom scheme - that's effectively what "moduleResolution": "classic" is), and it stands to reason that we'd use the same module resolution for your configuration as for your other files. Which leads to the issue that it becomes possible to not know what module resolution scheme to use to find your configuration until we have found your configuration, which is an issue.

The solution can be something like setting the moduleResolution in the "children":

{
  "extends": "external-package/tsconfig",
  "compilerOptions": {
    "moduleResolution": "node"
  }
}

And, if the external-package/tsconfig.json file also have a moduleResolution, then you throw an error like: 'moduleResolution' can't be overwritten because 'external-package/tsconfig' is already setting a value. And, if not, then you resolve trough the node resolution.

This right now is a "blocker" for me, because I have a "core" module with the settings that will be shared among all my modules, and because the path is relative to node_modules, and npm has now a flat tree, this breaks.

For now I'm resolving it by adding setting the value to ../external-package/tsconfig instead of ./node_modules/external-package/tsconfig, but this solution is dirty because in development I need to have the core module one level above the module I'm working on in order to make it work.

TSLint already have this resolved, you can make something like the example above without any issues.

Yup. I'm the same use case as @lukeshiru.

I can't imagine why the node_modules being flat prevents you from resolving properly, @lukeshiru. I don't understand what you're doing there. Why do you resolve other than what my example shows?

tsconfig.json is not used after package is packaged, right? So my workaround should work.

How good of an idea would it be to use "extends": "./node_modules/typescript-config-foo/index.json" until this is possibly implemented?

Not a very good idea. In my mono-repo setup, the plan would be to have a config-like sub-repo, and it could end up getting put at a higher level, so you'd need to do ../node_modules/@myproject/config/tsconfig.json, essentially guessing where yarn would put things, which is exactly why I suggested some way to follow the built-in node module resolution algorithm.

@lukeshiru there is always a --moduleResolution, but it is often implied.
--module commonjs implies --moduleResolution node, otherwise the language defaults to --moduleResolution classic.

@mightyiam the problem is mentioned by @rtm. Let's say you have you children module using the parent's module config like this:

children-module/tsconfig

{
  "extends": "./node_modules/parent-module/tsconfig"
}

If you then, have that children module used somewhere else (let's calle it other-children-module), that reference becomes invalid, because instead of having this:

children-module/
└── node_modules/
    └── parend-module/

Now you have this:

other-children-module/
└── node_modules/
    β”œβ”€β”€ children-module/
    └── parent-module/

The flat dependency tree makes children and parent at the same level. So for that to work you need children-module to have the config changed to:

{
  "extends": "../parent-module/tsconfig"
}

The ideal scenario should be to solve it like TSLint does, by setting it like this:

{
  "extends": "parent-module/tsconfig"
}

And letting node resolve it.

@aluanhaddad that still doesn't solve the issue. See my comment above for clarification.

Although by no means the only use case, the requirement might be simplest to understand by looking at the case where I want to install some config globally:

npm install --global cool-tsconfigs

Then use it by saying

"extends": "cool-tsconfigs/tsconfig.test.json"

I don't agree with the use of global packages, I would be happy with tsconfig resolving extends like tslint does it, by using the npm module resolution (that's what I proposed first).

Right now my tslint files are beautiful:

{
  "extends": "@property/core-dev/tslint"
}

And my tsconfig are horrible :'( :

{
  // FIXME: Temporary fix until github.com/Microsoft/TypeScript/issues/18865 is fixed.
  // TODO: Change to @property/core/tsconfig when above is resolved.
  "extends": "../core/tsconfig",
  "compilerOptions": {
    "outDir": "./build"
  }
}

But the npm module resolution algorithm does resolve to global packages
anyway, you can't stop it.

On Thu, Oct 26, 2017 at 10:37 PM, Lucas Ciruzzi notifications@github.com
wrote:

I don't agree with the use of global packages, I would be happy with
tsconfig resolving extends like tslint does it, by using the npm module
resolution (that's what I proposed first
https://github.com/Microsoft/TypeScript/issues/18865#issuecomment-339489923
).

β€”
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
https://github.com/Microsoft/TypeScript/issues/18865#issuecomment-339733459,
or mute the thread
https://github.com/notifications/unsubscribe-auth/AABfR6e_cFNgz1N9b986dq_D6vNb-eoRks5swLw2gaJpZM4Ppcg1
.

I don't think is a good practice to require a global package from a module. Quoting npm itself:

In general, the rule is:

  1. If you’re installing something that you want to use in your program, using require('whatever'), then install it locally, at the root of your project.
  2. If you’re installing something that you want to use in your shell, on the command line or something, install it globally, so that its binaries end up in your PATH environment variable.

I think we're getting distracted. I was not proposing global installation
of repos with TS configs in them as a best practice. I was simply trying to
say that it to understand how this feature would work it might be useful to
realize that that would work too.

On Thu, Oct 26, 2017 at 10:54 PM, Lucas Ciruzzi notifications@github.com
wrote:

I don't think is a good practice to require a global package from a
module. Quoting npm itself:

In general, the rule of is:

  1. If you’re installing something that you want to use in your
    program, using require('whatever'), then install it locally, at the root of
    your project.
  2. If you’re installing something that you want to use in your shell,
    on the command line or something, install it globally, so that its binaries
    end up in your PATH environment variable.

β€”
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
https://github.com/Microsoft/TypeScript/issues/18865#issuecomment-339737362,
or mute the thread
https://github.com/notifications/unsubscribe-auth/AABfR9QCGz8YK6E27Gk8eDqC03lxcdTpks5swMA6gaJpZM4Ppcg1
.

We excluded it during the initial implementation due to complexity, but left an error for it in so we could go back and add it later if we needed to. An issue here is that the module resolution strategy we use is determined by your compiler options... and an extends option can be specifying just what that module resolution scheme is, which is kind of a circular dependency - you must know your module resolution scheme to resolve an extends option, but your extends option can lead to where you specify some module resolution scheme!

Only a problem if you don't specify the module resolution algorithm in that particular file. If you do, then extends should work and the final module resolution will not change since the final pkg config overrides it.

Of course that means you can't specify the module resolution in the base file. Personally, I think thats fine.

I understand this is a bit of a chicken-and-egg problem here but not being able to extend from another module without specifying a path seems really problematic, and makes the extension mechanism far less useful than it should be.

I can think of two trivial ways this could be done, the first resolving environment variables in the extends path, and the second allowing some kind of scheme in the extends path which indicates how to resolve the module. e.g.

{
  "extends": "$BASEDIR/tsconfig"
}

and

{
  "extends": "node:my-base-package"
}

currently I have a bunch of tsconfig.json.template files which are substituted via sed (sed 's@$BASEDIR@'"$LERNA_ROOT_PATH"'@' tsconfig.json.template > tsconfig.tmp.json) into tsconfig.tmp.json which has to be in the .gitignore.

I then build the project using tsc -p tsconfig.tmp.json.

This works for our lerna monorepo, but seems like a bunch of silly hoops to jump through.

@ravenscar I like the scheme idea!

@ravenscar Either of these looks good. However, I would point out that moduleResolution is a compilerOptions, which applies to the compilation process, and would not necessarily, and possibly should not, apply to the process by which TS reads in and processes its configurations. There is potentially no need to worry about any conflict between the compiler's module resolution setting and that applied by TS in resolving extends paths. I see no problem with the latter always using node resolution strategy. For instance, AFAIK tslint always uses node resolution semantics for its extends option.

If it is considered important to have both resolution strategies available for extends, then I would slightly prefer a new extendsResolution property at the top level of tsconfig.json instead of embedding this information in the extends path itself with the scheme-like node: notation. Or, allow extends: {path: "configrepo/tsconfig.json", resolution: "classic"}, with the default for extends.resolution hopefully being node.

IMO better to only support node than support nothing at all.

It's almost 100% certain people will be developing in a node environment. To me it seems like people here are confusing compiler settings with development settings.

NPM and yarn don't offer custom module resolution for a reason.

There is literally no use case for using anything other than node module resolution to resolve a base TSCONFIG

A use case for this; I have two libraries:

core-typescript: vends a baseline tsconfig.json for use in a bunch of projects (with things like lib, esModuleInterop, etc)

code-style: vends a style-oriented tsconfig.json that extends core-typescript's tsconfig (with things like strict, etc)

And I want application code to depend on code-style.

My current approach is for code-style to declare core-typescript as a _peer_ dependency, to enforce that it is installed at the top level of an application's node_modulesβ€”and _assumes_ that it will only ever be a top level dependency. Brittle :(

node_modules/
  core-typescript
    tsconfig.json
  code-style
    tsconfig.json // extends "../core-typescript/tsconfig.json"
tsconfig.json // extends "./node_modules/code-style/tsconfig.json"

Yup, @nevir's use case is exactly what I need.

Another vote for this. I just want to keep my tsconfig in one place in a lerna repo. Rather surprised to see there's no support already for this.

In my case I have multiple packages and those packages can either be built standalone or with lerna. If they are built as standalone then they will have node_modules in the package. If they are built using lerna the node_modules get hoisted into the parent directory. Therefore I cannot use the following solution

{
  "extends": "./node_modules/@foo/bar/tsconfig.json"
}

because it may also be here

{
  "extends": "../../node_modules/@foo/bar/tsconfig.json"
}

The ideal solution appears to be

{
  "extends": "@foo/bar/tsconfig.json"
}

Using node module resolution would fix this issue.

Just had a look in the source code for TypeScript and found the following in commandLineParser

If the path isn't a rooted or relative path, don't try to resolve it (we reserve the right to special case module-id like paths in the future

    function getExtendsConfigPath(
        extendedConfig: string,
        host: ParseConfigHost,
        basePath: string,
        errors: Push<Diagnostic>,
        createDiagnostic: (message: DiagnosticMessage, arg1?: string) => Diagnostic) {
        extendedConfig = normalizeSlashes(extendedConfig);
        // If the path isn't a rooted or relative path, don't try to resolve it (we reserve the right to special case module-id like paths in the future)
        if (!(isRootedDiskPath(extendedConfig) || startsWith(extendedConfig, "./") || startsWith(extendedConfig, "../"))) {
            errors.push(createDiagnostic(Diagnostics.A_path_in_an_extends_option_must_be_relative_or_rooted_but_0_is_not, extendedConfig));
            return undefined;
        }
        let extendedConfigPath = getNormalizedAbsolutePath(extendedConfig, basePath);
        /// ....

And the actual loader

    function getExtendedConfig(
        sourceFile: TsConfigSourceFile,
        extendedConfigPath: string,
        host: ParseConfigHost,
        basePath: string,
        resolutionStack: string[],
        errors: Push<Diagnostic>,
    ): ParsedTsconfig | undefined {
        const extendedResult = readJsonConfigFile(extendedConfigPath, path => host.readFile(path));
        // ...

Using the following read file util functions:

    /**
     * Read tsconfig.json file
     * @param fileName The path to the config file
     */
    export function readJsonConfigFile(fileName: string, readFile: (path: string) => string | undefined): TsConfigSourceFile {
        const textOrDiagnostic = tryReadFile(fileName, readFile);
        return isString(textOrDiagnostic) ? parseJsonText(fileName, textOrDiagnostic) : <TsConfigSourceFile>{ parseDiagnostics: [textOrDiagnostic] };
    }

    function tryReadFile(fileName: string, readFile: (path: string) => string | undefined): string | Diagnostic {
        let text: string | undefined;
        try {
            text = readFile(fileName);
        }
        catch (e) {
            return createCompilerDiagnostic(Diagnostics.Cannot_read_file_0_Colon_1, fileName, e.message);
        }
        return text === undefined ? createCompilerDiagnostic(Diagnostics.The_specified_path_does_not_exist_Colon_0, fileName) : text;
    }

A "hacky fix" could be to replace readFile with a version which first tries the readFile function passed in, then falls back to try node resolution using require

    function tryReadFile(fileName: string, readFile: (path: string) => string | undefined): string | Diagnostic {
        let text: string | undefined;
        try {
            text = readFile(fileName);
        }  catch (e) {
           try {
              // fallback to basic nodeJS resolution strategy using require
              text = require(fileName)
           } catch (e) {
             return createCompilerDiagnostic(Diagnostics.Cannot_read_file_0_Colon_1, fileName, e.message);
           }
        }
        return text === undefined ? createCompilerDiagnostic(Diagnostics.The_specified_path_does_not_exist_Colon_0, fileName) : text;
    }

To fix the first part, allowing a non relative extends path, requires a bit more thought.
Currently in getExtendsConfigPath, the path is normalized before being returned using getNormalizedAbsolutePath which doesn't make sense when specified as a node dependency.

    /**
     * Parse a path into an array containing a root component (at index 0) and zero or more path
     * components (at indices > 0). The result is normalized.
     * If the path is relative, the root component is `""`.
     * If the path is absolute, the root component includes the first path separator (`/`).
     */
    export function getNormalizedPathComponents(path: string, currentDirectory: string | undefined) {
        return reducePathComponents(getPathComponents(path, currentDirectory));
    }

    export function getNormalizedAbsolutePath(fileName: string, currentDirectory: string | undefined) {
        return getPathFromPathComponents(getNormalizedPathComponents(fileName, currentDirectory));
    }

    /**
     * Formats a parsed path consisting of a root component (at index 0) and zero or more path
     * segments (at indices > 0).
     */
    export function getPathFromPathComponents(pathComponents: ReadonlyArray<string>) {
        if (pathComponents.length === 0) return "";

        const root = pathComponents[0] && ensureTrailingDirectorySeparator(pathComponents[0]);
        return root + pathComponents.slice(1).join(directorySeparator);
    }

Instead we could have it try both strategies and only normalize etc. if node resolution fails

    function getExtendsConfigPath(
        extendedConfig: string,
        host: ParseConfigHost,
        basePath: string,
        errors: Push<Diagnostic>,
        createDiagnostic: (message: DiagnosticMessage, arg1?: string) => Diagnostic) {

        extendedConfig = normalizeSlashes(extendedConfig);

        try {
          // FIX: try basic node resolution first
          require(extendedConfig)
          // yup, worked! So return config path "as is"
          return extendedConfig
        } catch (e) {
          // node resolution failed, so instead try using normalized path strategy

          // If the path isn't a rooted or relative path, don't try to resolve it (we reserve the right to special case module-id like paths in the future)
          if (!(isRootedDiskPath(extendedConfig) || startsWith(extendedConfig, "./") || startsWith(extendedConfig, "../"))) {
            errors.push(createDiagnostic(Diagnostics.A_path_in_an_extends_option_must_be_relative_or_rooted_but_0_is_not, extendedConfig));
            return undefined;
          }
          let extendedConfigPath = getNormalizedAbsolutePath(extendedConfig, basePath);
          if (!host.fileExists(extendedConfigPath) && !endsWith(extendedConfigPath, Extension.Json)) {
            extendedConfigPath = `${extendedConfigPath}.json`;
            if (!host.fileExists(extendedConfigPath)) {
                errors.push(createDiagnostic(Diagnostics.File_0_does_not_exist, extendedConfig));
                return undefined;
            }
          }
          return extendedConfigPath;
       }
    }

Note: I haven't tried this strategy/solution myself but should provide a starting point for anyone trying to give it a go ;)

This just bit me because I had to extend a tsconfig.json in node_modules, that itself had to extend a tsconfig.json from a package in node_modules. But it can't, because ./node_modules/foo is now actually ../foo.

Hey @weswigham! I think that PR didn't fixed this issue. I have a repo with settings and I'm trying to extend them like this:

// child-module/tsconfig.json
{
  "extends": "@org-name/settings-repo/tsconfig.json"
}

And instead of looking for it in the node_modules directory, it looks for it in the cwd, which is not the expected behavior. If I use this instead:

// child-module/tsconfig.json
{
  "extends": "./node_modules/@org-name/settings-repo/tsconfig.json"
}

It will break because that only works in this scenario:

child-module/
└── node_modules/
    └── @org-name/settings-repo/

But then if I use children-module it breaks in this scenario:

other-module/
└── node_modules/
    β”œβ”€β”€ child-module/
    └── @org-name/settings-repo/

Because is not anymore in child-module/node_modules, instead it is in children-module/... You are forgetting about plain structures...

@weswigham Any chance this could be re-opened. As @lukeshiru points out this hasn't been properly implemented. The current implementation is not in line with the described Module resolution node in the typescript documentation.

@lukeshiru, @charrondev, please allow me to suggest that opening a new issue might help. You know how no-one likes reopening closed issues.

I am confused by this reported "bug" because I have been using this feature extensively since it was implemented and it works exactly as expected. Perhaps something else is wrong in your case. You may need to provide more information or even a sample repo.

This definitely work properly for me at this point on 3.4 release in most scenarios.

I'll try to put together my reproduction case soon though. It's something like the following:

- ~/workspace/oss-core/
- ~/workspace/oss-core/tsconfig.json (extends `@vanilla/tsconfig/core.json`)
- ~/workspace/oss-core/plugins/proprietary-plugin (symlink to other directory).
- ~/workspace/proprietary-plugin
- ~/workspace/proprietary-plugin/tsconfig.json (extends `@vanilla/tsconfig/core.json`)

In this file structure I would expect the tsconfig extension to be capable of being discovered from when looked at through the symlinked file.

Instead oss-core/tsconfig.json works, but proprietary-plugin/tsconfig.json does not.

I think that the real file path is being resolved before evaluating the config file. I believe this is similar to the issue @lukeshiru is describing, but slightly different.

In any case I can open a separate issue and try to make a sample repo for my issue.

I'm not sure how people are saying this works... My project structure is like so:

  /project
    /tsconfig.js
    /node_modules
      /@package
        /tsconfig.js

And contents of `parent/project/tsconfig.json is:

{
  "extends": "@package/tsconfig.json"
}

Yet it fails as it's trying to extend from parent/node_modules/package/tsconfig.json instead of from parent/project/node_modules/package/tsconfig.json. Why is it looking for node_modules in the parent of the root instead of root itself?

I also can confirm that this just doesn't work. Going through the code of that PR, I can only find out that the internal code is just over engineered and complicated for no apparent reason. Why is just doing require not an option here and just let node do his work?

I also want to note that it doesn't work by just doing:

"extends": "@company/config-typescript"

Why am I trying this? Because I want to have an .js file where I can dynamically build the config depending on the project extending it.

We don't support any kind of .js based configuration. At all. You should just have a tsconfig.json in the package root, which, ofc, can't do anything dynamic.

We don't support any kind of .js based configuration. At all. You should just have a tsconfig.json in the package root, which, ofc, can't do anything dynamic.

would such things be on the roadmap? As mono-repos are used more and more, it is useful to share config with the same includes following the globs that all projects inside the mono repo complies to and have dynamic things in it.

For example, we do this with our babel and nyc config, and that works like a charm.

Go open an issue - https://github.com/microsoft/TypeScript/issues/30400 kinda tracked it but was closed by the author.

Was this page helpful?
0 / 5 - 0 ratings

Related issues

remojansen picture remojansen  Β·  3Comments

siddjain picture siddjain  Β·  3Comments

wmaurer picture wmaurer  Β·  3Comments

jbondc picture jbondc  Β·  3Comments

DanielRosenwasser picture DanielRosenwasser  Β·  3Comments