{S}

Image for Turborepo for NextJs and Express with Typescript

Turborepo for NextJs and Express with Typescript

In this blog we will talk about how to setup a turborepo (monorepo) with nextjs as frontend and express js as the backend. This guide can we followed for any frameworks by just tweaking few steps.

  1. NextJS

In this blog we will talk about how to setup a turborepo (monorepo) with nextjs as frontend and express js as the backend. This guide can we followed for any frameworks by just tweaking few steps.

Why TurboRepo?

Turborepo makes managing and deploying monorepo a lot easier. It also have very good support for typescript and eslint as well. On the top of that you can use it with other languages as well. Read more about turborepo here


Initial Setup

We will use pnpm (as this is recommended by turborepo devs themselves) as the package manager for this project. You can install pnpm by running the following command.

## Method 1
npm i -g pnpm

## Method 2
corepack enable

# Verify that pnpm is installed
pnpm -v

Note: If you are on windows you might want to restart your terminal after installing pnpm


Create a new project

Create a new project using the following command

pnpm dlx create-turbo@latest
cd <project-name>

You will be asked few questions, answer them as per your project requirements. Turbo Setup

Make sure to select pnpm as your package manager, it will make your life a lot easier.

As turborepo is build by the same team who created nextjs, it defaults to nextjs as frontend and docs, but don’t worry you can just go ahead and delete the docs and web folder and use your own framework.

As I am going to use nextjs and express, I will not delete the web folder, but I will delete the docs folder.

Lets delete the docs folder in the apps directory

rm apps/docs -r

Lets create a App (api)

To create a new app you just do the following, you can follow same steps if you want to initiate vite or something else.

pnpm dlx turbo gen workspace

Make sure to install devDependencies such as @repo/eslint-config and @repo/typescript-config in the newly created app. This will help you to have a better development experience.

Create Turbo App

You will be asked few questions, answer them as per your project requirements.


Setting up API (Express)

pnpm --filter api add express
pnpm --filter api add @types/express typescript nodemon npm-run-all -D

Lets create a tsconfig.json file in the api folder

{
  "extends": "@repo/typescript-config/tsconfig.json",
  "compilerOptions": {
    "outDir": "dist",
    "rootDir": "src"
  },
}

Add two scripts in the package.json file in the api folder

...,
"scripts": {
  "dev": "nodemon",
  "build": "tsc -b",
  "start": "node dist/index.js"
}
...

Lets create a nodemon.json file in the api folder, although there are many tools but I still prefer nodemon.

{
  "watch": ["src"],
  "ext": "ts",
  "ignore": ["dist"],
  "exec": "npm-run-all build start"
}

Create a basic API server

Lets create a basic express server in the src/index.ts file

import express from 'express';

const app = express();
const PORT = process.env.PORT || 3000;

app.get('/', (req, res) => {
  res.send('Hello World');
});

app.listen(PORT, () => {
  console.log(`Server is running on http://localhost:${PORT}`);
});

Now make sure you are in the root directory of the project and run the following command

pnpm dev

Voila you have your express server running on http://localhost:5000

But wait, lets create a shared package as well. That’s the part where people get confused and couldn’t get it to work.


Creating a shared package

We will use same command to create a workspace, but this time we will select package as the workspace type. Also don’t forget to install @repo/eslint-config and @repo/typescript-config as devDependencies in the newly created package.

pnpm dlx turbo gen workspace

Create Turbo Package

Let’s create a shared utility that can be used in both frontend and backend.

pnpm --filter @repo/utils add typescript nodemon -D

# You can omit nodemon if you want and can use tsc -w instead, but for this example I will use nodemon

Lets create a tsconfig.json file in the utils folder

{
  "extends": "@repo/typescript-config/tsconfig.json",
  "compilerOptions": {
    "outDir": "dist",
    "rootDir": "src"
  },
}

Add two scripts in the package.json file in the utils folder

...,
"scripts": {
  "dev": "nodemon",
  "build": "tsc -b"
}
...

Create a nodemon.json file in the utils folder

{
  "watch": ["src"],
  "ext": "ts",
  "ignore": ["dist"],
  "exec": "npm run build"
}

Lets create a basic utility function in the src/index.ts file

export function sharedUtil(){
  console.log('This is a shared utility function');
}

Lets add this shared utility in the api and web folder as well. Edit package.json as follows

...,
"dependencies": {
  ...,
  "@repo/utils": "workspace:*"
}
...

Install the packages using pnpm

pnpm i

Lets try to import the shared utility in the api and web folder Edit api/src/index.ts and try to add this on top

import { sharedUtil } from '@repo/utils';

But we are getting a following import error. Import Error

This is because we also need to tell package.json that we should export specific files from the shared package. Edit the package.json file in the utils folder and add the following

...,
"exports": {
  "default": "./dist/index.js",
  "types": "./dist/index.d.ts",
},
...

But we are still getting the same error, this is because we need to build the shared package first. But this would become very hard to maintain in large scale projects. Right? Here comes the turborepo to the rescue.

You can specify on which packages/scripts your app depends, and turborepo will take care of building them in the right order.

Create a turbo.json in api directory and write the following

{
  "$schema": "https://turbo.build/schema.json",
  "extends": ["//"],
  "tasks": {
    "dev": {
      "dependsOn": ["@repo/utils#build"]
    }
  }
}

Now, lets import the shared utility in the api/src/index.ts file

import { sharedUtil } from '@repo/utils';
sharedUtil()
...

Also import it in the web/app/page.tsx file

import { sharedUtil } from '@repo/utils';
sharedUtil()
...

Now run the following command in the root directory of the project

pnpm dev

We should see the following output in the terminal Backend Output Frontend Output

Congratulations, you have successfully setup a turborepo with nextjs and express with typescript. You can now add more packages and apps as per your requirements.


Additional Steps

You can also setup a nextjs proxy to the express server, so that you can use the same domain for both frontend and backend. You can do this by adding the following code in the web/next.config.js file

/** @type {import('next').NextConfig} */
const nextConfig = {
  async rewrites(){
    return [
      {
        source: '/api/:path*',
        destination: 'http://localhost:3001/api/:path*',
      },
    ]
  }
};

export default nextConfig;

Thank you for reading, I hope this blog was helpful to you. If you have any questions or suggestions, feel free to comment below.

In the next blog we will talk about how to deploy this project using docker. Stay tuned.

Table of Contents
  1. Why TurboRepo?
    1. Initial Setup
      1. Create a new project
        1. Lets create a App (api)
          1. Setting up API (Express)
            1. Create a basic API server
            2. Creating a shared package
              1. Additional Steps