Customizing Inputs and Named Inputs
The inputs
property of a task allows you to define under what conditions the cache for that task should be invalidated and the task should be run again. Tasks can have inputs
defined for them globally in the nx.json
file or on a per-project basis in the project configuration. Those inputs
can be fileset
s, runtime
inputs or env
variables.
If you find yourself reusing the same inputs
definitions, you can instead create a namedInput
in the project configuration or nx.json
and use that namedInput
in your inputs
array.
The inputs
and namedInputs
are parsed with the following rules:
{projectRoot}
and{workspaceRoot}
are replaced with the appropriate path- A
^
character at the beginning of the string means this entry applies to the project dependencies of the project, not the project itself. - Everything else is processed with the minimatch library
Knowing the syntax doesn't always explain how you would use the feature, so here are two scenarios explaining how to customize inputs
to match your particular set up.
Defaults
If you don't specify any inputs
, Nx uses as inputs every file in the task's project and every file in that project's dependencies. It's the same as writing this:
1{
2 "namedInputs": {
3 "default": ["{projectRoot}/**/*"]
4 },
5 "targetDefaults": {
6 "build": {
7 "inputs": ["default", "^default"]
8 }
9 }
10}
11
This default behavior works and will always give you correct output, but it might re-run a task in some cases where the cache could have been used instead. Modifying inputs
and namedInputs
is all about getting the most you can out of the caching mechanism.
Scenario 1: React App
Let's say we have a repo with two projects. There's a React app (react-app
) that depends on a React UI library (ui-library
).
Global Settings
In the root nx.json
file, we can set the following namedInputs
:
1{
2 "namedInputs": {
3 "sharedGlobals": ["{workspaceRoot}/babel.config.json"],
4 "default": ["{projectRoot}/**/*", "sharedGlobals"],
5 "production": [
6 "default",
7 "!{projectRoot}/**/*.spec.ts",
8 "!{projectRoot}/**/*.md",
9 "!{projectRoot}/tsconfig.spec.json",
10 "!{projectRoot}/jest.config.ts",
11 "!{projectRoot}/.eslintrc.json"
12 ]
13 }
14}
15
This configuration defines three namedInputs
.
sharedGlobals
contains root level files that should invalidate the cache for all projects. Note thattsconfig.base.json
and thepackage.json
lock files are handled behind the scenes by Nx so that when you modify the specific properties that affect your project, the cache is invalidated.default
contains every file in the project and everything defined insharedGlobals
production
starts with every file indefault
, but removes.spec.ts
files, markdown files and some project-specific config files.
These names are not hard-coded. You can add your own namedInputs
or rename these as you see fit.
With these namedInputs
in place, we can set default inputs
for project targets in nx.json
:
1{
2 "namedInputs": {
3 "sharedGlobals": [
4 /* ... */
5 ],
6 "default": [
7 /* ... */
8 ],
9 "production": [
10 /* ... */
11 ]
12 },
13 "targetDefaults": {
14 "build": {
15 "inputs": ["production", "^production"],
16 "dependsOn": ["^build"]
17 },
18 "test": {
19 "inputs": ["default", "^production", "{workspaceRoot}/jest.preset.js"]
20 }
21 }
22}
23
In this configuration, build
needs to be re-run if the project's production
files are changed or the dependencies' production
files are changed. The "dependsOn": ["^build"]
setting ensures that the output files of the dependencies are correct, but it does not tell Nx to re-run this project's build target when those outputs change. That's why "^production"
is needed in the inputs
array.
Note that we are using ^production
instead of ^default
because changing test files of a dependency should not invalidate the build cache for this project.
The test
target cache is invalidated if test or production code is changed in this project (default
) or if production code is changed in a dependency ^production
. The tests also need to be run again if the root level jest.preset.js
file is modified.
Project Settings
In the ui-library
library, we have some *.spec.tsx
files that we don't want to count as production files. We can modify the project configuration to account for this:
1{
2 "namedInputs": {
3 "production": [
4 // copied this definition from nx.json
5 "default",
6 "!{projectRoot}/**/*.spec.ts",
7 "!{projectRoot}/**/*.spec.tsx", // added this line
8 "!{projectRoot}/**/*.md",
9 "!{projectRoot}/tsconfig.spec.json",
10 "!{projectRoot}/jest.config.ts",
11 "!{projectRoot}/.eslintrc.json"
12 ]
13 }
14}
15
Note that I need to copy the entire definition of production
from nx.json
. This entry overwrites the existing production
entry defined in nx.json
, but default
and sharedGlobals
are inherited. Now when I modify a spec.tsx
file in ui-library
, the build for ui-library
is unaffected and the build for react-app
is unaffected.
You could also handle this scenario by modifying the production
definition in the nx.json
so that it applies to all projects in the repo. In fact, the defaults created by Nx do exactly that: "!{projectRoot}/**/?(*.)+(spec|test).[jt]s?(x)?(.snap)"
. This is a complex glob expression though and is not particularly helpful for explaining the concepts of inputs
and namedInputs
.
The build
and test
inputs
do not need to be modified at the project level, since the defaults defined at the root are still correct.
Scenario 2: Generated Documentation Site
Let's say you have a repo with two projects, a docs-site
app that contains all the code for rendering a documentation site and a content
library that contains markdown files that are used as the content of the site.
The content
library has a node script .ts
file that is used to check through the markdown files for internal broken links. The test
target looks like this:
1{
2 "targets": {
3 "test": {
4 "inputs": ["default", "^production"],
5 "executor": "nx:run-commands",
6 "command": "ts-node ./check-for-broken-links.ts"
7 }
8 }
9}
10
Note that we are overwriting the default inputs
for test
because changes to the root jest.preset.js
file shouldn't affect this project's tests.
This library's production
definition should look like this:
1{
2 "namedInputs": {
3 "production": ["default", "!{projectRoot}/**/*.ts"]
4 }
5}
6
This ensures that the .md
files are included in the production
file set, but the check-for-broken-links.ts
script is not. Now changes to a markdown file in content
will force a re-build of docs-site
, even though changes to markdown files in docs-site
will not force a re-build.
Scenario 3: Deploying applications and publishing libraries
We can use nx:run-commands
and a custom deploy
target for deploying our applications. Normally, we want to deploy only the applications that have been affected by recent changes. The hash for checking whether a certain project was affected consists of several parts:
- The full command we are running (e.g.
nx run my-app:deploy
) - The hash of the project's files and files of the dependencies
- The hash of the target executor's dependencies Since
nx:run-commands
is a special executor we can't automatically say what packages this executor depends on, so any change in the installed packages will cause a cache miss. This is of course not ideal.
We can therefore specify what external packages our executor should depend on (in this case none, we only depend on fly
version)
1{
2 "targets": {
3 "deploy": {
4 "inputs": [
5 "production",
6 { "externalDependencies": [] }, // this explicitly tells hasher to ignore all external packages for executor
7 { "runtime": "fly version" }
8 ],
9 "dependsOn": ["build"],
10 "executor": "nx:run-commands",
11 "cwd": "dist/{projectRoot}",
12 "command": "fly deploy"
13 }
14 }
15}
16
Similarly, we can specify dependencies for publishing libraries:
1{
2 "targets": {
3 "publish": {
4 "inputs": [
5 "production",
6 { "externalDependencies": ["lerna"] } // we explicitly say that our run-commands depends on lerna only
7 ],
8 "dependsOn": ["build"],
9 "executor": "nx:run-commands",
10 "cwd": "dist/{projectRoot}",
11 "command": "lerna publish"
12 }
13 }
14}
15