TypeScript Monorepos with Yarn - Semaphore (2023)

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:

$ npminstall-gtypescript$ cdchild$ 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.

$ cdparent$ tsc​src/index.ts:1:20-error TS2307: Cannotfindmodule'child'or its corresponding type declarations.​1import { moduleName } from'child';​Found1error.

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.

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:

$ yarnsetversion 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.

$ yarninit-w

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

$ yarnadd-Dtypescript

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.

$ gitmvsayhi shared packages/

To confirm that workspaces have been detected, run:

$ yarnworkspaces 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.

$ yarntsc--build--force

You also can build each project separately with:

$ yarnworkspace shared build$ yarnworkspace sayhi build

And you can try running the main program.

$ yarnworkspace sayhinodedist/src/sayhi.jsHi, 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 TypeScript monorepo, so we’re ready to begin the next phase: automating testing with CI/CD.

$ gitadd-A$ gitcommit-m"Set up TS and Yarn"$ gitpush 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.

TypeScript Monorepos with Yarn - Semaphore (1)

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.

TypeScript Monorepos with Yarn - Semaphore (2)

The Workflow Builder opens to let you configure the pipeline.

TypeScript Monorepos with Yarn - Semaphore (3)

Building the TypeScript monorepo

We’ll set up a TypeScript monorepo 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-versionnode14.17.3checkoutyarn workspace shared build
TypeScript Monorepos with Yarn - Semaphore (4)

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.

TypeScript Monorepos with Yarn - Semaphore (5)

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

In the prologue, type:

sem-version node 14.17.3checkout

Create two jobs in the block:

  • Lint with the command: yarn workspace shared lint
  • Unit testing: yarn workspace shared test
TypeScript Monorepos with Yarn - Semaphore (6)

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

TypeScript Monorepos with Yarn - Semaphore (7)

Managing TypeScript monorepo 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.

TypeScript Monorepos with Yarn - Semaphore (8)

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.

TypeScript Monorepos with Yarn - Semaphore (9)

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 your TypeScript monorepo.

TypeScript Monorepos with Yarn - Semaphore (10)

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

TypeScript Monorepos with Yarn - Semaphore (11)

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.

TypeScript Monorepos with Yarn - Semaphore (12)

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:

  • Continuous Integration for Monorepos
  • Monorepo and Micro-Frontends with Jonathan Creamer
  • Monorepo and Building at Scale with Benjy Weinberger
  • JavaScript Monorepos with Lerna
Top Articles
Latest Posts
Article information

Author: Maia Crooks Jr

Last Updated: 12/20/2022

Views: 5591

Rating: 4.2 / 5 (63 voted)

Reviews: 86% of readers found this page helpful

Author information

Name: Maia Crooks Jr

Birthday: 1997-09-21

Address: 93119 Joseph Street, Peggyfurt, NC 11582

Phone: +2983088926881

Job: Principal Design Liaison

Hobby: Web surfing, Skiing, role-playing games, Sketching, Polo, Sewing, Genealogy

Introduction: My name is Maia Crooks Jr, I am a homely, joyous, shiny, successful, hilarious, thoughtful, joyous person who loves writing and wants to share my knowledge and understanding with you.