Skip to content

Typescript project with gts and esbuild

Abstract

When starting a new typescript project you need to configure lots of things

  • npm project settings
  • typescript transpiler settings
  • folder structure
  • style checker and formatter
  • bundler

To minize the effort gts is a nice tool to help you setting up the project. Our of deployment/devliver purposes we also want to bundle the code to create nice and small artifacts. For those purposes we use esbuild. The whole example can be found here

Prerequisites

For starting this tutorial you need to have nodejs and npm (also npx) installed on your machine. Some basic knowledge about nodejs, npm and npx is helpful.

Create the project

Please keep in mind that some of the outputs can vary on your system related to the used versions.

So lets start some actions. First we want to create the npm project. Therefore please cd into a nice directory of your choice.

Now lets create the project directory and the npm typescript project with the help of gts.

Remark We add the -y flag to accept the default settings. If you want to adjust those, just do not add the flag.

mkdir ts_gts_esbuild
cd ts_gts_esbuild
npx gts init -y

Analyse

Now lets see what we got.

$ tree -a -L 1
.
├── .eslintignore
├── .eslintrc.json
├── node_modules
├── package.json
├── package-lock.json
├── .prettierrc.js
├── src
└── tsconfig.json

2 directories, 6 files

Because we do not want to add build artifacts and dependent libs to our git database we add them to .gitignore

Remark The build/ folder is used per default for the typescript build artefacts.

echo node_modules/ >> .gitignore
echo build/ >> .gitignore

We can see that we received some default settings

  • npm project settings
  • typescript transpiler settings
  • folder structure
  • style checker and formatter

NPM project settings

$ cat package.json
{
  "name": "",
  "version": "0.0.0",
  "description": "",
  "main": "build/src/index.js",
  "types": "build/src/index.d.ts",
  "files": [
    "build/src"
  ],
  "license": "Apache-2.0",
  "keywords": [],
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1",
    "lint": "gts lint",
    "clean": "gts clean",
    "compile": "tsc",
    "fix": "gts fix",
    "prepare": "npm run compile",
    "pretest": "npm run compile",
    "posttest": "npm run lint"
  },
  "devDependencies": {
    "gts": "^3.1.0",
    "typescript": "^4.0.3",
    "@types/node": "^14.11.2"
  }
}

As you can see some devDependencies have been installed. Also we got some preparations for delivering a npm package. Furthermore we got some npm scripts.

Typescript transpiler settings

npm run compile

This will compile the typescript sources defined in tsconfig.json file and put the built aretefacts into the outDir folder defined in tsconfig.json.

npm run clean

This will cleanup the typescript build artefacts (delete the outDir directory defined in tsconfig.json).

Style checker and formatter

npm run lint

This will run the style checker eslint.

npm run fix

This will fix some of the found lint issues.

bundler

Because we also want to bundle our project we first install esbuild as dev dependency

npm i -D esbuild

Lets test it.

npx esbuild src/index.ts

The output is the bundled file.

You can pass lots of options to the cli interface. However, using the command-line interface can become unwieldy if you need to pass many options to esbuild. For more sophisticated uses you will likely want to write a build script in JavaScript using esbuild's JavaScript API.

We will do that now. Lets create a small nodejs binary.

mkdir bin
touch bin/esbuild.mjs

now please copy the following content to the file

#!/usr/bin/env node

import esbuild from 'esbuild';

const build = esbuild.build({
  entryPoints: ["src/index.ts"],
  outdir: "dist",
  bundle: true,
  loader: {".ts": "ts"}
});

await build;

as you can see we will have the outdir dist set. We also need to add this to our .gitignore file.

echo dist/ >> .gitignore

Now lets test it

node bin/esbuild.mjs

As you can see the bundled package is built.

Now we can add a new npm script. Please add a new script within the script part in package.json. Because in addition we want to create the types we also add npx tsc --emitDeclarationOnly --outDir dist/types

...
"scripts": {
  ...
  "posttest": "npm run lint",
  "bundle": "npx tsc --emitDeclarationOnly --outDir dist/types && node bin/esbuild.mjs"
},
...

[option] Package

Now we have the option to create a nice small npm package. Therefore we need to add a .npmignore file and add some content.

touch .npmignore
echo node_modules/ >> .npmignore
echo build/ >> .npmignore
echo bin/ >> .npmignore
echo src/ >> .npmignore

Also we need to adjust some paths within our package.json

"name": "ts-gts-esbuild",
"version": "0.1.0",
"description": "",
"main": "dist/index.js",
"types": "dist/src/index.d.ts",
...

We can test it now

npm package

This will create a tar archive in your package folder. You can check it.

Cons

There are lots of pros related to this project setup. But of course there are also some cons. One of them is that gts is out of the box not properly working with npm workspaces yet gts issue. But you just need to do the gts init within the repos and adjust some path parameters in some files.

ko-fi