Publishing a TypeScript project
This article augments TypeScript's own Publishing guide with specifics for native node support.
Some important things to note:
-
Everything from Publishing a package applies here.
-
Fields like
main
operate on published content, so when TypeScript source-code is transpiled to JavaScript, JavaScript is the published content andmain
would point to a JavaScript file with a JavaScript file extension (exmain.ts
→"main": "main.js"
). -
Fields like
scripts.test
operate on source-code, so they would use the file extensions of the source code (ex"test": "node --test './src/**/*.test.ts'
).
-
-
Node runs TypeScript code via a process called "type stripping", wherein node (via Amaro) removes TypeScript-specific syntax, leaving behind vanilla JavaScript (which node already understands). This behaviour is enabled by default as of node version 23.6.0.
- Node does not strip types in
node_modules
because it can cause significant performance issues for the official TypeScript compiler (tsc
) and parts of VS Code, so the TypeScript maintainers would like to discourage people publishing raw TypeScript, at least for now.
- Node does not strip types in
-
Consuming TypeScript-specific features like
enum
in node still requires a flag (--experimental-transform-types
). There are often better alternatives for these anyway. -
Use dependabot to keep your dependencies current, including those in github actions. It's a very easy set-and-forget configuration.
-
.nvmrc
comes from NVM, a multi-version manager for node. It allows you to specify the version of node the project should generally use.
A repository would look something like:
example-ts-pkg/
├ .github/
├ workflows/
├ ci.yml
└ publish.yml
└ dependabot.yml
├ src/
├ foo.fixture.js
├ main.ts
├ main.test.ts
├ some-util.ts
└ some-util.test.ts
├ LICENSE
├ package.json
├ README.md
└ tsconfig.json
And its published package would look something like:
example-ts-pkg/
├ LICENSE
├ main.d.ts
├ main.d.ts.map
├ main.js
├ package.json
├ README.md
├ some-util.d.ts
├ some-util.d.ts.map
└ some-util.js
What to do with your types
Treat types like a test
The purpose of types is to warn an implementation will not work:
const foo = 'a';
const bar: number = 1 + foo;
// ^^^ Type 'string' is not assignable to type 'number'.
TypeScript has warned that the above code will not behave as intended, just like a unit test warns that code does not behave as intended. They are complementary and verify different things—you should have both.
Your editor (ex VS Code) likely has built-in support for TypeScript, displaying errors as you work. If not, and/or you missed those, CI will have your back.
The following GitHub Action sets up a CI task to automatically check (and require) types pass inspection for a PR into the main
branch.
name: Tests
on:
pull_request:
branches: ['main']
jobs:
check-types:
# Separate these from tests because
# they are platform and node-version independent
# and need be run only once.
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with:
node-version-file: '.nvmrc'
cache: 'npm'
- name: npm clean install
run: npm ci
# You may want to run a lint check here too
- run: node --run types:check
test:
runs-on: ubuntu-latest
strategy:
matrix:
node:
- version: 23.x
- version: 22.x
fail-fast: false # Prevent a failure in one version cancelling other runs
steps:
- uses: actions/checkout@v4
- name: Use node ${{ matrix.node.version }}
uses: actions/setup-node@v4
with:
node-version: ${{ matrix.node.version }}
cache: 'npm'
- name: npm clean install
run: npm ci
- run: node --run test
Generate type declarations
Type declarations (.d.ts
and friends) provide type information as a sidecar file, allowing the execution code to be vanilla JavaScript whilst still having types.
Since these are generated based on source code, they can be built as part of your publication process and do not need to be checked into your repository.
Take the following example, where the type declarations are generated just before publishing to the NPM registry.
name: Publish to NPM
on:
push:
tags:
- '**@*'
jobs:
build:
runs-on: ubuntu-latest
permissions:
contents: read
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with:
node-version-file: '.nvmrc'
registry-url: 'https://registry.npmjs.org'
- run: npm ci
# You can probably ignore the boilerplate config above
- name: Publish with provenance
env:
NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }}
run: npm publish --access public --provenance
npm publish
will automatically run prepack
beforehand. npm
will also run prepack
automatically before npm pack --dry-run
(so you can easily see what your published package will be without actually publishing it). Beware, node --run
does not do that. You can't use node --run
for this step, so that caveat does not apply here, but it can for other steps.
Breaking this down
Generating type declarations is deterministic: you'll get the same output from the same input, every time. So there is no need to commit these to git.
npm publish
grabs everything applicable and available at the moment the command is run; so generating type declarations immediately before means those are available and will get picked up.
By default, npm publish
grabs (almost) everything (see Files included in package). In order to keep your published package minimal (see the "Heaviest Objects in the Universe" meme about node_modules
), you want to exclude certain files (like tests and test fixtures) from from packaging. Add these to the opt-out list specified in .npmignore
; ensure the !*.d.ts
exception is listed, or the generated type declartions will not be published! Alternatively, you can use package.json "files" to create an opt-in list.