TypeScript build process

TypeScript build process

Applications written in TypeScript must be compiled into JavaScript before you can run them in production.

Compiling TypeScript source files can be performed using many different build tools. However, with AdonisJS, we stick to the most straightforward approach and use the following time-tested tools.

All the below-mentioned tools come pre-installed as development dependencies with official starter kits.

  • TSC is the TypeScript's official compiler. We use TSC to perform type-checking and create the production build.

  • TS Node Maintained is a Just-in-Time compiler for TypeScript. It allows you to execute TypeScript files without compiling them to JavaScript and proves to be a great tool for development.

  • SWC is a TypeScript compiler written in Rust. We use it during development with TS Node to make the JIT process extremely fast.

ToolUsed forType checking
TSCCreating production buildYes
TS NodeDevelopmentNo
SWCDevelopmentNo

Executing TypeScript files without compilation

You may execute the TypeScript files without compiling them using the ts-node-maintained/register/esm hook. For example, you may start the HTTP server by running the following command.

node --import=ts-node-maintained/register/esm bin/server.js
  • --import: The import flag allows you to specify a module that exports customization hooks for module resolution and loading. For more information refer to the Node.js customization hooks documentation.

  • ts-node-maintained/register/esm: The path to the ts-node-maintained/register/esm script that registers lifecycle hooks to perform Just-in-Time compilation of TypeScript source to JavaScript.

  • bin/server.js: The path to the AdonisJS HTTP server entry point file. See also: A note on file extensions

You may repeat this process for other TypeScript files as well. For example:

Run tests
node --import=ts-node-maintained/register/esm bin/test.ts
Run ace commands
node --import=ts-node-maintained/register/esm bin/console.ts
Run some other TypeScript file
node --import=ts-node-maintained/register/esm path/to/file.ts

A note on file extensions

You might have noticed us using the .js file extension everywhere, even though the file on disk is saved with the .ts file extension.

This is because, with ES modules, TypeScript forces you to use the .js extension in imports and when running scripts. You can learn about the thesis behind this choice in TypeScript documentation.

If you are using TypeScript 5.7 or later, you can import TypeScript files using the .ts extension. This is made possible by the path rewriting for relative paths feature.

Since some runtimes allow you to run TypeScript code "in-place" and require the .ts extension, you may prefer to already use .ts extension for future compatibility.

Running the development server

Instead of running the bin/server.js file directly, we recommend using the serve command for the following reasons.

  • The command includes a file watcher and restarts the development server on file change.
  • The serve command detects the frontend assets bundler your app is using and starts its development server. For example, If you have a vite.config.js file in your project root, the serve command will start the vite dev server.
node ace serve --watch

You may pass arguments to the Vite dev server using the --assets-args command line flag.

node ace serve --watch --assets-args="--debug --base=/public"

You may use the --no-assets flag to disable the Vite dev server.

node ace serve --watch --no-assets

Passing options to the Node.js commandline

The serve command starts the development server (bin/server.ts file) as a child process. If you want to pass node arguments to the child process, you can define them before the command name.

node ace --no-warnings --inspect serve --watch

Creating production build

The production build of your AdonisJS application is created using the node ace build command. The build command performs the following operations to create a standalone JavaScript application inside the ./build directory.

  • Remove the existing ./build folder (if any).
  • Rewrite the ace.js file from scratch to remove the ts-node/esm loader.
  • Compile frontend assets using Vite (if configured).
  • Compile TypeScript source code to JavaScript using tsc.
  • Copy non-TypeScript files registered under the metaFiles array to the ./build folder.
  • Copy the package.json and package-lock.json/yarn.lock files to the ./build folder.

Any modifications to the ace.js file will be lost during the build process since the file is rewritten from scratch. If you want to have any additional code that runs before Ace starts, you should instead do it inside the bin/console.ts file.

And that is all!

node ace build

Once the build has been created, you can cd into the build folder, install production dependencies, and run your application.

cd build
# Install production dependencies
npm ci --omit=dev
# Run server
node bin/server.js

You may pass arguments to the Vite build command using the --assets-args command line flag.

node ace build --assets-args="--debug --base=/public"

You may use the --no-assets flag to avoid compiling the frontend assets.

node ace build --no-assets

What is a standalone build?

Standalone build refers to the JavaScript output of your application that you can run without the original TypeScript source.

Creating a standalone build helps reduce the size of the code you deploy on your production server, as you do not have to copy both the source files and the JavaScript output.

After creating the production build, you can copy the ./build to your production server, install dependencies, define environment variables, and run the application.