Module Resolution
This section assumes some basic knowledge about modules. Please see the Modules documentation for more information.
Module resolution is the process the compiler uses to figure out what an import refers to. Consider an import statement like import { a } from "moduleA"
; in order to check any use of a
, the compiler needs to know exactly what it represents, and will need to check its definition moduleA
.
At this point, the compiler will ask “what’s the shape of moduleA
?” While this sounds straightforward, moduleA
could be defined in one of your own .ts
/.tsx
files, or in a .d.ts
that your code depends on.
First, the compiler will try to locate a file that represents the imported module. To do so the compiler follows one of two different strategies: Classic or Node. These strategies tell the compiler where to look for moduleA
.
If that didn’t work and if the module name is non-relative (and in the case of "moduleA"
, it is), then the compiler will attempt to locate an ambient module declaration. We’ll cover non-relative imports next.
Finally, if the compiler could not resolve the module, it will log an error. In this case, the error would be something like error TS2307: Cannot find module 'moduleA'.
Relative vs. Non-relative module imports
Module imports are resolved differently based on whether the module reference is relative or non-relative.
A relative import is one that starts with /
, ./
or ../
. Some examples include:
import Entry from "./components/Entry";
import { DefaultHeaders } from "../constants/http";
import "/mod";
Any other import is considered non-relative. Some examples include:
import * as $ from "jquery";
import { Component } from "@angular/core";
A relative import is resolved relative to the importing file and cannot resolve to an ambient module declaration. You should use relative imports for your own modules that are guaranteed to maintain their relative location at runtime.
A non-relative import can be resolved relative to baseUrl
, or through path mapping, which we’ll cover below. They can also resolve to ambient module declarations. Use non-relative paths when importing any of your external dependencies.
Module Resolution Strategies
There are two possible module resolution strategies: Node and Classic. You can use the moduleResolution
option to specify the module resolution strategy. If not specified, the default is Node for --module commonjs
, and Classic otherwise (including when module
is set to amd
, system
, umd
, es2015
, esnext
, etc.).
Note:
node
module resolution is the most-commonly used in the TypeScript community and is recommended for most projects. If you are having resolution problems withimport
s andexport
s in TypeScript, try settingmoduleResolution: "node"
to see if it fixes the issue.
Classic
This used to be TypeScript’s default resolution strategy. Nowadays, this strategy is mainly present for backward compatibility.
A relative import will be resolved relative to the importing file. So import { b } from "./moduleB"
in source file /root/src/folder/A.ts
would result in the following lookups:
/root/src/folder/moduleB.ts
/root/src/folder/moduleB.d.ts
For non-relative module imports, however, the compiler walks up the directory tree starting with the directory containing the importing file, trying to locate a matching definition file.
For example:
A non-relative import to moduleB
such as import { b } from "moduleB"
, in a source file /root/src/folder/A.ts
, would result in attempting the following locations for locating "moduleB"
:
/root/src/folder/moduleB.ts
/root/src/folder/moduleB.d.ts
/root/src/moduleB.ts
/root/src/moduleB.d.ts
/root/moduleB.ts
/root/moduleB.d.ts
/moduleB.ts
/moduleB.d.ts
Node
This resolution strategy attempts to mimic the Node.js module resolution mechanism at runtime. The full Node.js resolution algorithm is outlined in Node.js module documentation.
How Node.js resolves modules
To understand what steps the TS compiler will follow, it is important to shed some light on Node.js modules. Traditionally, imports in Node.js are performed by calling a function named require
. The behavior Node.js takes will differ depending on if require
is given a relative path or a non-relative path.
Relative paths are fairly straightforward. As an example, let’s consider a file located at /root/src/moduleA.js
, which contains the import var x = require("./moduleB");
Node.js resolves that import in the following order:
-
Ask the file named
/root/src/moduleB.js
, if it exists. -
Ask the folder
/root/src/moduleB
if it contains a file namedpackage.json
that specifies a"main"
module. In our example, if Node.js found the file/root/src/moduleB/package.json
containing{ "main": "lib/mainModule.js" }
, then Node.js will refer to/root/src/moduleB/lib/mainModule.js
. -
Ask the folder
/root/src/moduleB
if it contains a file namedindex.js
. That file is implicitly considered that folder’s “main” module.
You can read more about this in Node.js documentation on file modules and folder modules.
However, resolution for a non-relative module name is performed differently. Node will look for your modules in special folders named node_modules
. A node_modules
folder can be on the same level as the current file, or higher up in the directory chain. Node will walk up the directory chain, looking through each node_modules
until it finds the module you tried to load.
Following up our example above, consider if /root/src/moduleA.js
instead used a non-relative path and had the import var x = require("moduleB");
. Node would then try to resolve moduleB
to each of the locations until one worked.
/root/src/node_modules/moduleB.js
-
/root/src/node_modules/moduleB/package.json
(if it specifies a"main"
property) -
/root/src/node_modules/moduleB/index.js
/root/node_modules/moduleB.js
-
/root/node_modules/moduleB/package.json
(if it specifies a"main"
property) -
/root/node_modules/moduleB/index.js
/node_modules/moduleB.js
-
/node_modules/moduleB/package.json
(if it specifies a"main"
property) /node_modules/moduleB/index.js
Notice that Node.js jumped up a directory in steps (4) and (7).
You can read more about the process in Node.js documentation on loading modules from node_modules
.
How TypeScript resolves modules
TypeScript will mimic the Node.js run-time resolution strategy in order to locate definition files for modules at compile-time. To accomplish this, TypeScript overlays the TypeScript source file extensions (.ts
, .tsx
, and .d.ts
) over Node’s resolution logic. TypeScript will also use a field in package.json
named types
to mirror the purpose of "main"
- the compiler will use it to find the “main” definition file to consult.
For example, an import statement like import { b } from "./moduleB"
in /root/src/moduleA.ts
would result in attempting the following locations for locating "./moduleB"
:
/root/src/moduleB.ts
/root/src/moduleB.tsx
/root/src/moduleB.d.ts
-
/root/src/moduleB/package.json
(if it specifies atypes
property) /root/src/moduleB/index.ts
/root/src/moduleB/index.tsx
/root/src/moduleB/index.d.ts
Recall that Node.js looked for a file named moduleB.js
, then an applicable package.json
, and then for an index.js
.
Similarly, a non-relative import will follow the Node.js resolution logic, first looking up a file, then looking up an applicable folder. So import { b } from "moduleB"
in source file /root/src/moduleA.ts
would result in the following lookups:
/root/src/node_modules/moduleB.ts
/root/src/node_modules/moduleB.tsx
/root/src/node_modules/moduleB.d.ts
-
/root/src/node_modules/moduleB/package.json
(if it specifies atypes
property) /root/src/node_modules/@types/moduleB.d.ts
/root/src/node_modules/moduleB/index.ts
/root/src/node_modules/moduleB/index.tsx
-
/root/src/node_modules/moduleB/index.d.ts
/root/node_modules/moduleB.ts
/root/node_modules/moduleB.tsx
/root/node_modules/moduleB.d.ts
-
/root/node_modules/moduleB/package.json
(if it specifies atypes
property) /root/node_modules/@types/moduleB.d.ts
/root/node_modules/moduleB/index.ts
/root/node_modules/moduleB/index.tsx
-
/root/node_modules/moduleB/index.d.ts
/node_modules/moduleB.ts
/node_modules/moduleB.tsx
/node_modules/moduleB.d.ts
-
/node_modules/moduleB/package.json
(if it specifies atypes
property) /node_modules/@types/moduleB.d.ts
/node_modules/moduleB/index.ts
/node_modules/moduleB/index.tsx
/node_modules/moduleB/index.d.ts
Don’t be intimidated by the number of steps here - TypeScript is still only jumping up directories twice at steps (9) and (17). This is really no more complex than what Node.js itself is doing.
Additional module resolution flags
A project source layout sometimes does not match that of the output. Usually a set of build steps result in generating the final output. These include compiling .ts
files into .js
, and copying dependencies from different source locations to a single output location. The net result is that modules at runtime may have different names than the source files containing their definitions. Or module paths in the final output may not match their corresponding source file paths at compile time.
The TypeScript compiler has a set of additional flags to inform the compiler of transformations that are expected to happen to the sources to generate the final output.
It is important to note that the compiler will not perform any of these transformations; it just uses these pieces of information to guide the process of resolving a module import to its definition file.
Base URL
Using a baseUrl
is a common practice in applications using AMD module loaders where modules are “deployed” to a single folder at run-time. The sources of these modules can live in different directories, but a build script will put them all together.
Setting baseUrl
informs the compiler where to find modules. All module imports with non-relative names are assumed to be relative to the baseUrl
.
Value of baseUrl is determined as either:
- value of baseUrl command line argument (if given path is relative, it is computed based on current directory)
- value of baseUrl property in ‘tsconfig.json’ (if given path is relative, it is computed based on the location of ‘tsconfig.json’)
Note that relative module imports are not impacted by setting the baseUrl, as they are always resolved relative to their importing files.
You can find more documentation on baseUrl in RequireJS and SystemJS documentation.
Path mapping
Sometimes modules are not directly located under baseUrl. For instance, an import to a module "jquery"
would be translated at runtime to "node_modules/jquery/dist/jquery.slim.min.js"
. Loaders use a mapping configuration to map module names to files at run-time, see RequireJs documentation and SystemJS documentation.
The TypeScript compiler supports the declaration of such mappings using paths
property in tsconfig.json
files. Here is an example for how to specify the paths
property for jquery
.
{ "compilerOptions": { "baseUrl": ".", // This must be specified if "paths" is. "paths": { "jquery": ["node_modules/jquery/dist/jquery"] // This mapping is relative to "baseUrl" } } }
Please notice that paths
are resolved relative to baseUrl
. When setting baseUrl
to another value than "."
, i.e. the directory of tsconfig.json
, the mappings must be changed accordingly. Say, you set "baseUrl": "./src"
in the above example, then jquery should be mapped to "../node_modules/jquery/dist/jquery"
.
Using paths
also allows for more sophisticated mappings including multiple fall back locations. Consider a project configuration where only some modules are available in one location, and the rest are in another. A build step would put them all together in one place. The project layout may look like:
projectRoot ├── folder1 │ ├── file1.ts (imports 'folder1/file2' and 'folder2/file3') │ └── file2.ts ├── generated │ ├── folder1 │ └── folder2 │ └── file3.ts └── tsconfig.jsonYes
© 2012-2021 Microsoft
Licensed under the Apache License, Version 2.0.
https://www.typescriptlang.org/docs/handbook/module-resolution.html