Blog
Juri Strumpflohner
July 26, 2023

Evergreen Tooling — More than Just CodeMods

Evergreen Tooling — More than Just CodeMods

As developers we always want to use the latest shiny tools. There’s a new bundler? Let’s try! A new code editor, I’m in! For your side-project: for sure! At work: nah, not really. Keeping your tooling up to date with the rather fast moving JS ecosystem can be a major challenge. Nx provides a mechanism that can help mitigate that, by providing a command to upgrade your tooling automatically:

npx nx migrate latest

Prefer a video? I’ve got you covered!

TL;DR

You can run the following command to automatically upgrade your Nx workspace to the latest version:

npx nx migrate latest

The Balancing Act: Updating Tooling vs Shipping Features

If you’re anything like me, you’ve probably found that discussions about updating tooling tend to fall to the bottom of the priority list when talking to your product owner. It’s understandable — their primary goal is to ship features. However, sticking with outdated tooling can impact our ability to deliver these features swiftly (not to speak about potential security concerns due to outdated libraries).

Don’t get me wrong, I’m not suggesting that we should always be on the bleeding edge of technological innovation — especially in an enterprise environment.

Jason Lengstorf has some opinions there as well: “The Hidden Danger of Switching Tech Stacks in 2023?).

It’s wise to let security patches land and initial bugs get fixed before jumping on the upgrade bandwagon. But here’s the catch — don’t wait too long. The longer you delay upgrading, the more challenging and time-consuming it becomes. And the more effort it requires, the harder it is to sell the idea to your product owner.

The Key: Making Updates Easy(ier)!

Updating tooling is never easy, but the Nx team aims at making it “easier” at least. We try to embrace the concept of “evergreen tooling”, a strategy that’s been around since Google decided to automatically update Chrome for all users. The Angular team adopted this approach for their Angular CLI, and Nx has followed suit. But what exactly is it, and how does it work?

What if I told you Nx users have been automatically updating their React applications from Webpack 4 to Webpack 5!

The “why” is pretty straightforward. From the perspective of an open-source project, you want users to adopt the latest version as quickly as possible. This minimizes the maintenance work involved in supporting older versions, which can be a real headache. Looking at how Nx manages it, it seems to be successful in this regard (Source):

The distribution of Nx installs by version demonstrates the effectiveness of this approach. For instance, v16.5, which accounts for 19.7% of all versions, has already been adopted by many users, despite its recent release. The latest major accounts for 34.7% already and 41.4% are on the previous v15, a large majority of which is on the latest 15.9 minor. Hence, v16 & v15 make up 3/4 of all Nx installs.

How? Database Migration Scripts for Code?

If you know what “database migration scripts” are, then yes, it’s the same concept but applied at the code level. A series of small functions invoked to bring your workspace from version X to version Y (usually the latest). That includes:

  • update nx itself
  • update all Nx plugins and the technology they are responsible for (for example: @nx/react will upgrade React as well, @nx/webpack is upgrading Webpack)
  • automatically adjust relevant config files and source code (e.g., adjusting imports, functions etc..)

Everything that is required to get you to the latest version and still have a running code, even if there have been breaking changes.

How?! Because the Nx team (and plugin authors) do the work for you! Nx has a built-in mechanism where you can define so-called “migrations” for each Nx package. Here’s an excerpt of the @nx/webpack's migration file.

1{ 2 “generators”: { 3 "add-babel-inputs": { 4 “cli”: “nx”, 5 “version”: “15.0.0-beta.0”, 6 “description”: “Adds babel.config.json to the hash of all tasks”, 7 "factory": "./src/migrations/update-15-0-0/add-babel-inputs" 8 }, 9 "remove-es2015-polyfills-option": { 10 “cli”: “nx”, 11 “version”: “15.4.5-beta.0”, 12 “description”: “Removes es2015Polyfills option since legacy browsers are no longer supported.”, 13 "factory": "./src/migrations/update-15-4-5/remove-es2015-polyfills-option" 14 }, 15 "webpack-config-setup": { 16 “cli”: “nx”, 17 “version”: “15.6.3-beta.0”, 18 “description”: “Creates or updates webpack.config.js file with the new options for webpack.”, 19 "factory": "./src/migrations/update-15-6-3/webpack-config-setup" 20 }, 21 "add-babelUpwardRootMode-flag": { 22 “cli”: “nx”, 23 “version”: “15.7.2-beta.0”, 24 “description”: “Add the babelUpwardRootMode option to the build executor options.”, 25 "factory": "./src/migrations/update-15-7-2/add-babelUpwardRootMode-flag" 26 }, 27 "update-16-0-0-add-nx-packages": { 28 “cli”: “nx”, 29 “version”: “16.0.0-beta.1”, 30 "description": "Replace @nrwl/webpack with @nx/webpack", 31 "implementation": "./src/migrations/update-16-0-0-add-nx-packages/update-16-0-0-add-nx-packages" 32 } 33 }, 34 ... 35} 36

They are defined in a migrations.json config file within the NPM package. Each entry defines a version for which the entry should be run, a description (just for humans to read) and a factory property which points to a TypeScript file.

Example: if you’re on Nx 15.5 and you run nx migrate latest it would run the corresponding “factory functions” for:

  • webpack-config-setup
  • add-babelUpwardRootMode-flag
  • update-16-0-0-add-nx-packages

Depending on the nature of the update, these functions can be as simple as performing text replacements to more complex AST parsing and TypeScript source file manipulations. Let’s have a look at the add-babelUpwardRootMode-flag migration:

1import { 2 formatFiles, 3 readProjectConfiguration, 4 Tree, 5 updateProjectConfiguration, 6} from '@nx/devkit'; 7import { forEachExecutorOptions } from '@nx/devkit/src/generators/executor-options-utils'; 8import { WebpackExecutorOptions } from '../../executors/webpack/schema'; 9 10export default async function (tree: Tree) { 11 forEachExecutorOptions<WebpackExecutorOptions>( 12 tree, 13 ‘@nrwl/webpack:webpack’, 14 ( 15 options: WebpackExecutorOptions, 16 projectName, 17 targetName, 18 _configurationName 19 ) => { 20 if (options.babelUpwardRootMode !== undefined) { 21 return; 22 } 23 24 typconst projectConfiguration = readProjectConfiguration(tree, projectName); 25 projectConfiguration.targets[targetName].options.babelUpwardRootMode = 26 true; 27 updateProjectConfiguration(tree, projectName, projectConfiguration); 28 } 29 ); 30 31 await formatFiles(tree); 32} 33

It leverages the utility functions provided by the @nx/devkit package to read the various projects.json files to adjust the babelupwardRootMode property.

Nx’s modular design helps as each plugin is responsible for a particular area and can thus contribute according migration scripts. To give you some context. There is the nx package at the core — which you can use nicely in combination with a PNPM workspaces repo to speed things up — and then there are plugins built on top.

(Source: /getting-started/why-nx)

These plugins are usually technology-specific, like a plugin to help you manage React, Next, Remix, or Angular projects and tooling like ESLint, Cypress, Playwright, Vite, Jest, and so on. There are no limits as you can create your own. They are optional, in that you can use Nx and React and set everything up on your own. But it might be worth relying on them for some better DX and automation, such as the update mechanism we’re currently looking at.

Plugins are helpful here, because each plugin has a clearly defined responsibility. Like the @nx/webpack we looked at earlier, handles everything related to Webpack. So it’ll be responsible for updating the webpack NPM package and adjusting config Webpack-related files.

Performing the Update

Alright, we’ve learned how these updates work behind the scenes. Let’s look at what the experience looks like as a developer performing the update on your codebase.

Note, it is highly recommended to start with a clean Git workspace s.t. you can quickly revert the update.

To run the update, use the following command:

npx nx migrate latest

Note latest stands for the target version. You can also provide a specific Nx version if you cannot update to the latest one for some reason.

At this point, Nx

  • analyzes your workspace and finds all the plugins you’re using
  • downloads the version of the plugins specified in the migrate command above
  • collects all the migration.json files from these plugins
  • picks out the relevant ones based on your current workspace version
  • creates a migrations.json at the root of your workspace
  • updates the package.json to point to the matching NPM package versions (without performing an install just yet)

You can now inspect the migration.json and the package.json before you run the following command to run the migrations on your codebase.

npx nx migrate —-run-migrations

After that, your codebase should have been updated. Run your (ideally automated) sanity checks and fix the remaining issues that couldn’t be adjusted automatically.

Wrapping Up

That’s it! If you want to dive deeper, here are some potentially helpful links:

Also, if you haven’t already, give us a ⭐️ on Github: https://github.com/nrwl/nx. We’d appreciate it 😃.

Learn more