Node.js Now Supports TypeScript By Default
TypeScript is coming to Node 23. Let's break down what that means.
In this guide, we'll go through every single step you need to take to publish a package to npm.
This is not a minimal guide. We'll be setting up a fully production-ready package from an empty directory. This will include:
If you want to see the finished product, check out this demo repo.
If you prefer video content, I've created a video walkthrough of this guide:
In this section, we'll create a new git repository, set up a .gitignore
, create an initial commit, create a new repository on GitHub, and push our code to GitHub.
Run the following command to initialize a new git repository:
git init
.gitignore
Create a .gitignore
file in the root of your project and add the following:
node_modules
Run the following command to create an initial commit:
git add .
git commit -m "Initial commit"
Using the GitHub CLI, run the following command to create a new repository. I've chosen the name tt-package-demo
for this example:
gh repo create tt-package-demo --source=. --public
Run the following command to push your code to GitHub:
git push --set-upstream origin main
package.json
In this section, we'll create a package.json
file, add a license
field, create a LICENSE
file, and add a README.md
file.
package.json
fileCreate a package.json
file with these values:
{
"name": "tt-package-demo",
"version": "1.0.0",
"description": "A demo package for Total TypeScript",
"keywords": ["demo", "typescript"],
"homepage": "https://github.com/mattpocock/tt-package-demo",
"bugs": {
"url": "https://github.com/mattpocock/tt-package-demo/issues"
},
"author": "Matt Pocock <team@totaltypescript.com> (https://totaltypescript.com)",
"repository": {
"type": "git",
"url": "git+https://github.com/mattpocock/tt-package-demo.git"
},
"files": ["dist"],
"type": "module"
}
name
is the name by which people will install your package. It must be unique on npm. You can create organization scopes (such as @total-typescript/demo
) for free, these can help make it unique.version
is the version of your package. It should follow semantic versioning: the 0.0.1
format. Each time you publish a new version, you should increment this number.description
and keywords
are short descriptions of your package. They're listed in searches in the npm registry.homepage
is the URL of your package's homepage. The GitHub repo is a good default, or a docs site if you have one.bugs
is the URL where people can report issues with your package.author
is you! You can add optionally add your email and website. If you have multiple contributors, you can specify them as an array of contributors
with the same formatting.repository
is the URL of your package's repository. This creates a link on the npm registry to your GitHub repo.files
is an array of files that should be included when people install your package. In this case, we're including the dist
folder. README.md
, package.json
and LICENSE
are included by default.type
is set to module
to indicate that your package uses ECMAScript modules, not CommonJS modules.license
fieldAdd a license
field to your package.json
. Choose a license here. I've chosen MIT.
{
"license": "MIT"
}
LICENSE
fileCreate a file called LICENSE
(no extension) containing the text of your license. For MIT, this is:
MIT License
Copyright (c) [year] [fullname]
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
Change the [year]
and [fullname]
placeholders to the current year and your name.
README.md
fileCreate a README.md
file with a description of your package. Here's an example:
**tt-package-demo**
A demo package for Total TypeScript.
This will be shown on the npm registry when people view your package.
In this section, we'll install TypeScript, set up a tsconfig.json
, create a source file, create an index file, set up a build
script, run our build, add dist
to .gitignore
, set up a ci
script, and configure our tsconfig.json
for the DOM.
Run the following command to install TypeScript:
npm install --save-dev typescript
We add --save-dev
to install TypeScript as a development dependency. This means it won't be included when people install your package.
tsconfig.json
Create a tsconfig.json
with the following values:
{
"compilerOptions": {
/* Base Options: */
"esModuleInterop": true,
"skipLibCheck": true,
"target": "es2022",
"allowJs": true,
"resolveJsonModule": true,
"moduleDetection": "force",
"isolatedModules": true,
"verbatimModuleSyntax": true,
/* Strictness */
"strict": true,
"noUncheckedIndexedAccess": true,
"noImplicitOverride": true,
/* If transpiling with TypeScript: */
"module": "NodeNext",
"outDir": "dist",
"rootDir": "src",
"sourceMap": true,
/* AND if you're building for a library: */
"declaration": true,
/* AND if you're building for a library in a monorepo: */
"declarationMap": true
}
}
These options are explained in detail in my TSConfig Cheat Sheet.
tsconfig.json
for the DOMIf your code runs in the DOM (i.e. requires access to document
, window
, or localStorage
etc), skip this step.
If your code doesn't require access to DOM API's, add the following to your tsconfig.json
:
{
"compilerOptions": {
// ...other options
"lib": ["es2022"]
}
}
This prevents the DOM typings from being available in your code.
If you're not sure, skip this step.
Create a src/utils.ts
file with the following content:
export const add = (a: number, b: number) => a + b;
Create a src/index.ts
file with the following content:
export { add } from "./utils.js";
The .js
extension will look odd. This article explains more.
build
scriptAdd a scripts
object to your package.json
with the following content:
{
"scripts": {
"build": "tsc"
}
}
This will compile your TypeScript code to JavaScript.
Run the following command to compile your TypeScript code:
npm run build
This will create a dist
folder with your compiled JavaScript code.
dist
to .gitignore
Add the dist
folder to your .gitignore
file:
dist
This will prevent your compiled code from being included in your git repository.
ci
scriptAdd a ci
script to your package.json
with the following content:
{
"scripts": {
"ci": "npm run build"
}
}
This gives us a quick shortcut for running all required operations on CI.
In this section, we'll install Prettier, set up a .prettierrc
, set up a format
script, run the format
script, set up a check-format
script, add the check-format
script to our CI
script, and run the CI
script.
Prettier is a code formatter that automatically formats your code to a consistent style. This makes your code easier to read and maintain.
Run the following command to install Prettier:
npm install --save-dev prettier
.prettierrc
Create a .prettierrc
file with the following content:
{
"semi": true,
"singleQuote": true,
"trailingComma": "all",
"printWidth": 80,
"tabWidth": 2
}
You can add more options to this file to customize Prettier's behavior. You can find a full list of options here.
format
scriptAdd a format
script to your package.json
with the following content:
{
"scripts": {
"format": "prettier --write ."
}
}
This will format all files in your project using Prettier.
format
scriptRun the following command to format all files in your project:
npm run format
You might notice some files change. Commit them with:
git add .
git commit -m "Format code with Prettier"
check-format
scriptAdd a check-format
script to your package.json
with the following content:
{
"scripts": {
"check-format": "prettier --check ."
}
}
This will check if all files in your project are formatted correctly.
CI
scriptAdd the check-format
script to your ci
script in your package.json
:
{
"scripts": {
"ci": "npm run build && npm run check-format"
}
}
This will run the check-format
script as part of your CI process.
exports
, main
and @arethetypeswrong/cli
In this section, we'll install @arethetypeswrong/cli
, set up a check-exports
script, run the check-exports
script, set up a main
field, run the check-exports
script again, set up a ci
script, and run the ci
script.
@arethetypeswrong/cli
is a tool that checks if your package exports are correct. This is important because these are easy to get wrong, and can cause issues for people using your package.
@arethetypeswrong/cli
Run the following command to install @arethetypeswrong/cli
:
npm install --save-dev @arethetypeswrong/cli
check-exports
scriptAdd a check-exports
script to your package.json
with the following content:
{
"scripts": {
"check-exports": "attw --pack ."
}
}
This will check if all exports from your package are correct.
check-exports
scriptRun the following command to check if all exports from your package are correct:
npm run check-exports
You should notice various errors:
┌───────────────────┬──────────────────────┐
│ │ "tt-package-demo" │
├───────────────────┼──────────────────────┤
│ node10 │ 💀 Resolution failed │
├───────────────────┼──────────────────────┤
│ node16 (from CJS) │ 💀 Resolution failed │
├───────────────────┼──────────────────────┤
│ node16 (from ESM) │ 💀 Resolution failed │
├───────────────────┼──────────────────────┤
│ bundler │ 💀 Resolution failed │
└───────────────────┴──────────────────────┘
This indicates that no version of Node, or any bundler, can use our package.
Let's fix this.
main
Add a main
field to your package.json
with the following content:
{
"main": "dist/index.js"
}
This tells Node where to find the entry point of your package.
check-exports
againRun the following command to check if all exports from your package are correct:
npm run check-exports
You should notice only one warning:
┌───────────────────┬──────────────────────────────┐
│ │ "tt-package-demo" │
├───────────────────┼──────────────────────────────┤
│ node10 │ 🟢 │
├───────────────────┼──────────────────────────────┤
│ node16 (from CJS) │ ⚠️ ESM (dynamic import only) │
├───────────────────┼──────────────────────────────┤
│ node16 (from ESM) │ 🟢 (ESM) │
├───────────────────┼──────────────────────────────┤
│ bundler │ 🟢 │
└───────────────────┴──────────────────────────────┘
This is telling us that our package is compatible with systems running ESM. People using CJS (often in legacy systems) will need to import it using a dynamic import.
If you don't want to support CJS (which I recommend), change the check-exports script to:
{
"scripts": {
"check-exports": "attw --pack . --ignore-rules=cjs-resolves-to-esm"
}
}
Now, running check-exports
will show everything as green:
┌───────────────────┬───────────────────┐
│ │ "tt-package-demo" │
├───────────────────┼───────────────────┤
│ node10 │ 🟢 │
├───────────────────┼───────────────────┤
│ node16 (from CJS) │ 🟢 (ESM) │
├───────────────────┼───────────────────┤
│ node16 (from ESM) │ 🟢 (ESM) │
├───────────────────┼───────────────────┤
│ bundler │ 🟢 │
└───────────────────┴───────────────────┘
If you prefer to dual publish CJS and ESM, skip this step.
CI
scriptAdd the check-exports
script to your ci
script in your package.json
:
{
"scripts": {
"ci": "npm run build && npm run check-format && npm run check-exports"
}
}
tsup
to Dual PublishIf you want to publish both CJS and ESM code, you can use tsup
. This is a tool built on top of esbuild
that compiles your TypeScript code into both formats.
My personal recommendation would be to skip this step, and only ship ES Modules. This makes your setup significantly simpler, and avoids many of the pitfalls of dual publishing, like Dual Package Hazard.
But if you want to, go ahead.
tsup
Run the following command to install tsup
:
npm install --save-dev tsup
tsup.config.ts
fileCreate a tsup.config.ts
file with the following content:
import { defineConfig } from "tsup";
export default defineConfig({
entryPoints: ["src/index.ts"],
format: ["cjs", "esm"],
dts: true,
outDir: "dist",
clean: true,
});
entryPoints
is an array of entry points for your package. In this case, we're using src/index.ts
.format
is an array of formats to output. We're using cjs
(CommonJS) and esm
(ECMAScript modules).dts
is a boolean that tells tsup
to generate declaration files.outDir
is the output directory for the compiled code.clean
tells tsup
to clean the output directory before building.build
scriptChange the build
script in your package.json
to the following:
{
"scripts": {
"build": "tsup"
}
}
We'll now be running tsup
to compile our code instead of tsc
.
exports
fieldAdd an exports
field to your package.json
with the following content:
{
"exports": {
"./package.json": "./package.json",
".": {
"import": "./dist/index.js",
"default": "./dist/index.cjs"
}
}
}
The exports
field tells programs consuming your package how to find the CJS and ESM versions of your package. In this case, we're pointing folks using import
to dist/index.js
and folks using require
to dist/index.cjs
.
It's also recommended to add ./package.json
to the exports
field. This is because certain tools need easy access to your package.json
file.
check-exports
againRun the following command to check if all exports from your package are correct:
npm run check-exports
Now, everything is green:
┌───────────────────┬───────────────────┐
│ │ "tt-package-demo" │
├───────────────────┼───────────────────┤
│ node10 │ 🟢 │
├───────────────────┼───────────────────┤
│ node16 (from CJS) │ 🟢 (CJS) │
├───────────────────┼───────────────────┤
│ node16 (from ESM) │ 🟢 (ESM) │
├───────────────────┼───────────────────┤
│ bundler │ 🟢 │
└───────────────────┴───────────────────┘
We're no longer running tsc
to compile our code. And tsup
doesn't actually check our code for errors - it just turns it into JavaScript.
This means that our ci
script won't error if we have TypeScript errors in our code. Eek.
Let's fix this.
noEmit
to tsconfig.json
Add a noEmit
field to your tsconfig.json
:
{
"compilerOptions": {
// ...other options
"noEmit": true
}
}
tsconfig.json
Remove the following fields from your tsconfig.json
:
outDir
rootDir
sourceMap
declaration
declarationMap
They are no longer needed in our new 'linting' setup.
module
to Preserve
Optionally, you can now change module
to Preserve
in your tsconfig.json
:
{
"compilerOptions": {
// ...other options
"module": "Preserve"
}
}
This means you'll no longer need to import your files with .js
extensions. This means that index.ts
can look like this instead:
export * from "./utils";
lint
scriptAdd a lint
script to your package.json
with the following content:
{
"scripts": {
"lint": "tsc"
}
}
This will run TypeScript as a linter.
lint
to your ci
scriptAdd the lint
script to your ci
script in your package.json
:
{
"scripts": {
"ci": "npm run build && npm run check-format && npm run check-exports && npm run lint"
}
}
Now, we'll get TypeScript errors as part of our CI process.
In this section, we'll install vitest
, create a test, set up a test
script, run the test
script, set up a dev
script, and add the test
script to our CI
script.
vitest
is a modern test runner for ESM and TypeScript. It's like Jest, but better.
vitest
Run the following command to install vitest
:
npm install --save-dev vitest
Create a src/utils.test.ts
file with the following content:
import { add } from "./utils.js";
import { test, expect } from "vitest";
test("add", () => {
expect(add(1, 2)).toBe(3);
});
This is a simple test that checks if the hello
function returns the correct value.
test
scriptAdd a test
script to your package.json
with the following content:
{
"scripts": {
"test": "vitest run"
}
}
vitest run
runs all tests in your project once, without watching.
test
scriptRun the following command to run your tests:
npm run test
You should see the following output:
✓ src/utils.test.ts (1)
✓ hello
Test Files 1 passed (1)
Tests 1 passed (1)
This indicates that your test passed successfully.
dev
scriptA common workflow is to run your tests in watch mode while developing. Add a dev
script to your package.json
with the following content:
{
"scripts": {
"dev": "vitest"
}
}
This will run your tests in watch mode.
CI
scriptAdd the test
script to your ci
script in your package.json
:
{
"scripts": {
"ci": "npm run build && npm run check-format && npm run check-exports && npm run lint && npm run test"
}
}
In this section, we'll create a GitHub Actions workflow that runs our CI process on every commit and pull request.
This is a crucial step in ensuring that our package is always in a working state.
Create a .github/workflows/ci.yml
file with the following content:
name: CI
on:
pull_request:
push:
branches:
- main
concurrency:
group: ${{ github.workflow }}-${{ github.ref }}
cancel-in-progress: true
jobs:
ci:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Use Node.js
uses: actions/setup-node@v4
with:
node-version: "20"
- name: Install dependencies
run: npm install
- name: Run CI
run: npm run ci
This file is what GitHub uses as its instructions for running your CI process.
name
is the name of the workflow.on
specifies when the workflow should run. In this case, it runs on pull requests and pushes to the main
branch.concurrency
prevents multiple instances of the workflow from running at the same time, using cancel-in-progress
to cancel any existing runs.jobs
is a set of jobs to run. In this case, we have one job called ci
.actions/checkout@v4
checks out the code from the repository.actions/setup-node@v4
sets up Node.js and npm.npm install
installs the project's dependencies.npm run ci
runs the project's CI script.If any part of our CI process fails, the workflow will fail and GitHub will let us know by showing a red cross next to our commit.
Push your changes to GitHub and check the Actions tab in your repository. You should see your workflow running.
This will give us a warning on every commit made, and every PR made to the repository.
In this section, we'll install @changesets/cli
, initialize Changesets, make changeset releases public, set commit
to true
, set up a local-release
script, add a changeset, commit your changes, run the local-release
script, and finally see your package on npm.
Changesets is a tool that helps you version and publish your package. It's an incredible tool that I recommend to anyone publishing packages to npm.
@changesets/cli
Run the following command to initialise Changesets:
npm install --save-dev @changesets/cli
Run the following command to initialize Changesets:
npx changeset init
This will create a .changeset
folder in your project, containing a config.json
file. This is also where your changesets will live.
In .changeset/config.json
, change the access
field to public
:
// .changeset/config.json
{
"access": "public"
}
Without changing this field, changesets
won't publish your package to npm.
commit
to true
:In .changeset/config.json
, change the commit
field to true
:
// .changeset/config.json
{
"commit": true
}
This will commit the changeset to your repository after versioning.
local-release
scriptAdd a local-release
script to your package.json
with the following content:
{
"scripts": {
"local-release": "changeset version && changeset publish"
}
}
This script will run your CI process and then publish your package to npm. This will be the command you run when you want to release a new version of your package from your local machine.
prepublishOnly
Add a prepublishOnly
script to your package.json
with the following content:
{
"scripts": {
"prepublishOnly": "npm run ci"
}
}
This will automatically run your CI process before publishing your package to npm.
This is useful to separate from the local-release
script in case a user accidentally runs npm publish
without running local-release
. Thanks to Jordan Harband for the suggestion!
Run the following command to add a changeset:
npx changeset
This will open an interactive prompt where you can add a changeset. Changesets are a way to group changes together and give them a version number.
Mark this release as a patch
release, and give it a description like "Initial release".
This will create a new file in the .changeset
folder with the changeset.
Commit your changes to your repository:
git add .
git commit -m "Prepare for initial release"
local-release
scriptRun the following command to release your package:
npm run local-release
This will run your CI process, version your package, and publish it to npm.
It will have created a CHANGELOG.md
file in your repository, detailing the changes in this release. This will be updated each time you release.
Go to:
http://npmjs.com/package/<your package name>
You should see your package there! You've done it! You've published to npm!
You now have a fully set up package. You've set up:
@arethetypeswrong/cli
, which checks that your package exports are correcttsup
, which compiles your TypeScript code to JavaScriptvitest
, which runs your testsFor further reading, I'd recommend setting up the Changesets GitHub action and PR bot to automatically recommend contributors add changesets to their PR's. They are both phenomenal.
And if you've got any more questions, let me know!
How To Create An NPM Package
TypeScript is coming to Node 23. Let's break down what that means.
Learn how to extract the type of an array element in TypeScript using the powerful Array[number]
trick.
Enums in TypeScript can be confusing, with differences between numeric and string enums causing unexpected behaviors.
Is TypeScript just a linter? No, but yes.
It's a massive ship day. We're launching a free TypeScript book, new course, giveaway, price cut, and sale.
Learn why the order you specify object properties in TypeScript matters and how it can affect type inference in your functions.