TypeScript Monorepos with Yarn

In a past article in this monorepo series, we’ve discussed setting up CI/CD for JavaScript packages using Yarn Workspaces. This time, we will figure out the same for TypeScript. We’ll learn how to build and test TypeScript projects at scale with Yarn and Semaphore.

At the end of the tutorial, we’re going to have a continuous integration pipeline that builds only the code that changes.

Uniting Yarn and TypeScript

TypeScript extends JavaScript by adding everything it was missing: types, stricter checks, and a deeper IDE integration. TypeScript code is easier to read and debug, helping us write more robust code.

Compared to JavaScript, however, TypeScript saddles us with one more layer of complexity: code must be compiled first before it can be executed or used as a dependency. For instance, say we have two packages, “child” and “parent”. The child is easy to compile since it has no other dependencies:

$ npm install -g typescript
$ cd child
$ tsc

Yet, when we try to do the same with the parent that depends on it, we get an error since the local dependency is not found.

$ cd parent
$ tsc

src/index.ts:1:20 - error TS2307: Cannot find module 'child' or its corresponding type declarations.

1 import { moduleName } from 'child';

Found 1 error.

Without specialized tooling, we have to build and link packages by hand while preserving the correct build order. Yarn Workspaces already solves problems like these in JavaScript. Fortunately, with a bit of tweaking, we can extend it to TypeScript.

Setting up Workspaces in Yarn

Fork and clone the following GitHub repository, which has a couple of packages to experiment with.

GitHub logo TomFern / semaphore-demo-monorepo-typescript

Yarn Workspaces + TypeScript monorepo

Monorepo TypeScript Demo

Build Status

A hello world type monorepo demo for TypeScript and Yarn Workspaces.

Before Yarn Workspaces

Withouth workspaces, you have to build and link each project separately. For instance:

$ npm install -g typescript
$ cd shared
$ tsc
Enter fullscreen mode Exit fullscreen mode

This builds the shared package. But when we try to do the same with sayhi, we get an error since the local dependency is not found:

$ cd ..
$ cd sayhi
$ tsc

src/sayhi.ts:1:20 - error TS2307: Cannot find module 'shared' or its corresponding type declarations.

1 import { hi } from 'shared';
                     ~~~~~~~~
Found 1 error.
Enter fullscreen mode Exit fullscreen mode

Yarn workspaces help us link projects while keeping each in its own separate folder.

Configure Yarn Workspaces and TypeScript

To configure workspaces, first install the latest Yarn version:

$ yarn set version berry
Enter fullscreen mode Exit fullscreen mode

This creates .yarn and .yarnrc.yml

Initialize workspaces, this creates the packages folder…

We’re going to build a TypeScript monorepo made of two small packages:

  • shared: contains a few utility functions.
  • sayhi: the main package provides a “hello, world” program.

Let’s get going. To configure workspaces, switch to the latest Yarn version:

$ yarn set version berry

Yarn installs on .yarn/releases and can be safely checked in the repo.

Then, initialize workspaces. This creates the packages folder, a .gitignore, and the package.json and yarn.lock.

$ yarn init -w

You can add root-level dependencies to build all projects at once with:

$ yarn add -D typescript

Optionally, you may want to install the TypeScript plugin, which handles types for you. The foreach plugin is also convenient for running commands in many packages at the same time.

Next, move the code into packages.

$ git mv sayhi shared packages/

To confirm that workspaces have been detected, run:

$ yarn workspaces list --json

{"location":".","name":"semaphore-demo-monorepo-typescript"}
{"location":"packages/sayhi","name":"sayhi"}
{"location":"packages/shared","name":"shared"}

If this were a JavaScript monorepo, we would be finished. The following section introduces TypeScript builds into the mix.

TypeScript Workspaces

Our demo packages already come with a working tsconfig.json, albeit a straightforward one. Yet, we haven’t done anything to link them up — thus far, they have been completely isolated and don’t reference each other.

We can link TypeScript packages using project references. This feature, which was introduced on TypeScript 3.0, allows us to break an application into small pieces and build them piecemeal.

First, we need a root-level tsconfig.json with the following contents:

{
  "exclude": [
    "packages/**/tests/**",
    "packages/**/dist/**"
  ],
  "references": [
    {
      "path": "./packages/shared"
    },
    {
      "path": "./packages/sayhi"
    }
  ]
}

As you can see, we have one path item per package in the repo. The paths must point to folders containing package-specific tsconfig.json.

The referenced packages also need to have the composite option enabled. Add this line into packages/shared/tsconfig.json and packages/sayhi/tsconfig.json.

{
  "compilerOptions": {
     "composite": true

     . . .

  }
}

Packages that depend on other ones within the monorepo will need an extra reference. Add a references instruction in packages/sayhi/tsconfig.json (the parent package). The lines go at the top level of the file, outside compilerOptions.

{
  "references": [
    {
      "path": "../shared"
    }
  ]

  . . .

}

Install and build the combined dependencies with yarn install. Since we’re using the latest release of Yarn, it will generate a zero install file that can be checked into the repository.

Now that the configuration is ready, we need to run tsc to build everything for the first time.

$ yarn tsc --build --force

You also can build each project separately with:

$ yarn workspace shared build
$ yarn workspace sayhi build

And you can try running the main program.

$ yarn workspace sayhi node dist/src/sayhi.js
Hi, World

At the end of this section, the monorepo structure should look like this:

├── package.json
├── packages
│   ├── sayhi
│   │   ├── dist/
│   │   ├── src/
│   │   ├── package.json
│   │   └── tsconfig.json
│   └── shared
│       ├── dist/
│       ├── src/
│       ├── package.json
│       └── tsconfig.json
├── tsconfig.json
└── yarn.lock

That’s it, Yarn and TypeScript work together. Commit everything into the repository, so we’re ready to begin the next phase: automating testing with CI/CD.

$ git add -A
$ git commit -m "Set up TS and Yarn"
$ git push origin master

Building and testing with Semaphore

The demo includes a ready-to-work, change-based pipeline in the final branch. But we’ll learn faster by creating it from zero.

If you’ve never used Semaphore before, check out the getting started guide. Once you have added the forked demo repository into Semaphore, come back, and we’ll finish the setup.

We’ll start from scratch and use the starter single job template. Select “Single Job” and click on Customize.

The Workflow Builder opens to let you configure the pipeline.

Build Stage

We’ll set up a TypeScript build stage. The build stage compiles the code into JavaScript and runs tests such as linting and unit testing.

The first block will build the shared package. Add the following commands to the job.

sem-version node 14.17.3
checkout
yarn workspace shared build

The details are covered in-depth in the starter guide. But in a few words, sem-version switches the active version of Node (so we have version consistency), while checkout clones the repository into the CI machine.

Scroll down the right pane until you find Skip/Run conditions. Select “Run this block when conditions are met”. In the When? field type:

change_in('/packages/shared/')

The change_in function is an integral part of monorepo workflows. It scans the Git history to find which files have recently changed. In this case, we’re essentially asking Semaphore to skip the block if no files in the /packages/shared folders have changed.

Create a new block for testing. We’ll use it to run ESLint and unit tests with Jest.

In the prologue, type:

sem-version node 14.17.3
checkout

Create two jobs in the block:

  • Lint with the command: yarn workspace shared lint
  • Unit testing: yarn workspace shared test

Again, set the Skip/Run conditions and put the same condition as before.

Managing dependencies

We’ll repeat the steps for the sayhi package. Here, we only need to replace any instance of yarn workspace shared <command> with yarn workspace sayhi <command>.

Now, create a building block and uncheck the Dependencies section. Removing block dependencies in the pipeline makes blocks run in parallel.

Next, set the Skip/Run Condition on the new block to: change_in('/packages/sayhi/').

To finish, add a test block with a lint job and a unit test job. Since this package depends on shared, we can add a block-level dependency at this point. When done, you should have a total of four blocks.

The Skip/Run Condition, in this case, is different because the test block should run if either sayhi or shared change. Thus, we must supply an array instead of a single path in order to let change_in handle all cases correctly:

change_in(['/packages/sayhi', '/packages/shared'])

Running the Workflow

Click on Run the Workflow and then Start.

The first time the pipeline runs, all blocks will be executed.

On successive runs, only relevant blocks will start; the rest will be skipped, speeding up the pipeline considerably, especially if we’re dealing with tens or hundreds of packages in the repo.

Read Next

Adding TypeScript into the mix doesn’t complicate things too much. It’s a small effort that returns gains manifold with higher code readability and fewer errors.

Want to keep learning about monorepos? Check these excellent posts and tutorials:

30