Project Configuration

Projects can be configured in package.json (if you use npm scripts and not Nx executors) and project.json (if you use task executors). Both package.json and project.json files are located in each project's folder. Nx merges the two files to get each project's configuration. This reference

The following configuration creates build and test targets for Nx.

1{
2  "name": "mylib",
3  "scripts": {
4    "test": "jest",
5    "build": "tsc -p tsconfig.lib.json" // the actual command here is arbitrary
6  }
7}
8

You can invoke nx build mylib or nx test mylib without any extra configuration.

You can add Nx-specific configuration as follows:

1{
2  "name": "mylib",
3  "scripts": {
4    "test: "jest",
5    "build": "tsc -p tsconfig.lib.json" // the actual command here is arbitrary
6  },
7  "nx": {
8    "namedInputs": {
9      "default": [
10        "{projectRoot}/**/*"
11      ],
12      "prod": [
13        "!{projectRoot}/**/*.spec.tsx"
14      ]
15    },
16    "targets": {
17      "build": {
18        "inputs": ["prod", "^prod"],
19        "outputs": ["dist/libs/mylib"],
20        "dependsOn": ["^build"]
21      },
22      "test": {
23        "inputs": ["default", "^prod"],
24        "outputs": [],
25        "dependsOn": ["build"]
26      }
27    }
28  }
29}
30

inputs & namedInputs

The inputs array tells Nx what to consider to determine whether a particular invocation of a script should be a cache hit or not. There are three types of inputs:

Filesets

Examples:

  • {projectRoot}/**.*.ts
  • same as {fileset: "{projectRoot}/**/*.ts"}
  • {workspaceRoot}/jest.config.ts
  • same as {fileset: "{workspaceRoot}/jest.config.ts}

Runtime Inputs

Examples:

  • {runtime: "node -v"}

Node the result value is hashed, so it is never displayed.

Env Variables

Examples:

  • {env: "MY_ENV_VAR"}

Node the result value is hashed, so it is never displayed.

Named Inputs

Examples:

  • inputs: ["prod"]
  • same as inputs: [{input: "prod", projects: "self"}]

Often the same glob will appear in many places (e.g., prod fileset will exclude spec files for all projects). Because keeping them in sync is error-prone, we recommend defining named inputs, which you can then reference in all of those places.

Using ^

Examples:

  • inputs: ["^prod"]
  • same as inputs: [{input: "prod", projects: "dependencies"}]

Similar to dependsOn, the "^" symbols means "dependencies". This is a very important idea, so let's illustrate it with an example.

1"test": {
2  "inputs": [ "default", "^prod" ]
3}
4

The configuration above means that the test target depends on all source files of a given project and only prod sources (non-test sources) of its dependencies. In other words, it treats test sources as private.

outputs

Targets may define outputs to tell Nx where the target is going to create file artifacts that Nx should cache. "outputs": ["dist/libs/mylib"] tells Nx where the build target is going to create file artifacts.

This configuration is usually not needed. Nx comes with reasonable defaults (imported in nx.json) which implement the configuration above.

Basic Example

Usually, a target writes to a specific directory or a file. The following instructs Nx to cache dist/libs/mylib and build/libs/mylib/main.js:

1{
2  "build": {
3    "outputs": ["dist/libs/mylib", "build/libs/mylib/main.js"]
4  }
5}
6

Specifying Globs

Sometimes, multiple targets might write to the same directory. When possible it is recommended to direct these targets into separate directories.

1{
2  "build-js": {
3    "outputs": ["dist/libs/mylib/js"]
4  },
5  "build-css": {
6    "outputs": ["dist/libs/mylib/css"]
7  }
8}
9

But if the above is not possible, globs can be specified as outputs to only cache a set of files rather than the whole directory.

1{
2  "build-js": {
3    "outputs": ["dist/libs/mylib/**/*.js"]
4  },
5  "build-css": {
6    "outputs": ["dist/libs/mylib/**/*.css"]
7  }
8}
9

dependsOn

Targets can depend on other targets. This is the relevant portion of the configuration file:

1"build": {
2  "dependsOn": ["^build"]
3},
4"test": {
5  "dependsOn": ["build"]
6}
7

A common scenario is having to build dependencies of a project first before building the project. This is what the "dependsOn": ["^build"] property of the build target configures. It tells Nx that before it can build mylib it needs to make sure that mylib's dependencies are built as well. This doesn't mean Nx is going to rerun those builds. If the right artifacts are already in the right place, Nx will do nothing. If they aren't in the right place, but they are available in the cache, Nx will retrieve them from the cache.

Another common scenario is for a target to depend on another target of the same project. For instance, "dependsOn": ["build"] of the test target tells Nx that before it can test mylib it needs to make sure that mylib is built, which will result in mylib's dependencies being built as well.

You can also express the same configuration using:

1"build": {
2  "dependsOn": [{ "projects": "dependencies", "target": "build" }]
3},
4"test": {
5  "dependsOn": [{ "projects": "self", "target": "build" }]
6}
7

With the expanded syntax, you also have a third option available to configure how to handle the params passed to the target. You can either forward them or you can ignore them (default).

1"build": {
2   // forward params passed to this target to the dependency targets
3  "dependsOn": [{ "projects": "dependencies", "target": "build", "params": "forward" }]
4},
5"test": {
6  // ignore params passed to this target, won't be forwarded to the dependency targets
7  "dependsOn": [{ "projects": "dependencies", "target": "build", "params": "ignore" }]
8}
9"lint": {
10  // ignore params passed to this target, won't be forwarded to the dependency targets
11  "dependsOn": [{ "projects": "dependencies", "target": "build" }]
12}
13

Obviously this also works when defining a relation for the target of the project itself using "projects": "self":

1"build": {
2   // forward params passed to this target to the project target
3  "dependsOn": [{ "projects": "self", "target": "pre-build", "params": "forward" }]
4}
5

This configuration is usually not needed. Nx comes with reasonable defaults (imported in nx.json) which implement the configuration above.

tags

You can annotate your projects with tags as follows:

1{
2  "name": "mylib",
3  "nx": {
4    "tags": ["scope:myteam"]
5  }
6}
7

You can configure lint rules using these tags to, for instance, ensure that libraries belonging to myteam are not depended on by libraries belong to theirteam.

implicitDependencies

Nx uses powerful source-code analysis to figure out your workspace's project graph. Some dependencies cannot be deduced statically, so you can set them manually like this:

1{
2  "name": "mylib",
3  "nx": {
4    "implicitDependencies": ["anotherlib"]
5  }
6}
7

You can also remove a dependency as follows:

1{
2  "name": "mylib",
3  "nx": {
4    "implicitDependencies": ["!anotherlib"] # regardless of what Nx thinks, "mylib" doesn't depend on "anotherlib"
5  }
6}
7

Ignoring a project

Nx will add every project with a package.json file in it to its project graph. If you want to ignore a particular project, add the following to its package.json:

1{
2  "name": "mylib",
3  "nx": {
4    "ignore": true
5  }
6}
7