npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

dtsroll

v1.8.0

Published

Bundle dts files

Readme

Are you publishing a TypeScript project where consumers encounter type-checking errors like:

Cannot find module 'some-package' or its corresponding type declarations.

When you compile with tsc, the emitted declaration files (.d.ts files) preserve imports exactly as written. So if your published types import from a devDependency or a private package (e.g. an internal monorepo package), those imports cannot be resolved by the consumer.

// dist/index.d.ts (generated by tsc)
import type { SomeType } from 'my-private-dependency' // ❌ consumers can't resolve this

export declare function myUtility(): SomeType

If you can't move the dependency to dependencies, or you just want its types pulled directly into your published declarations, dtsroll is for you.

What is dtsroll?

dtsroll is a TypeScript declaration (.d.ts) file bundler. It's zero-config and reads your package.json to determine how your types should be bundled.

What dtsroll does

dtsroll runs after your build when .d.ts files have been emitted, and works in-place to bundle them to their entry points.

Since packages declared in devDependencies are not installed for the consumer, dtsroll assumes any imports referencing them should be bundled, as they would otherwise be unresolvable.

The result is a single, clean .d.ts output that works for consumers without extra installs.

// dist/index.d.ts (after dtsroll)
type SomeType = {
    value: string
}

export declare function myUtility(): SomeType

Features

  • Zero config — Automatically finds entry points from package.json.
  • Fixes missing-type errors — Inlines types from devDependencies so consumers don't need them installed.
  • Tree-shaken output — Unused types are removed to keep files small.
  • In-place — Designed to run directly on your dist folder after compilation.

Install

npm install --save-dev dtsroll

Quick start

1. Configure package.json

Point your types or exports to the final .d.ts file you want to publish.

{
	"name": "my-package",
	"exports": {
		"types": "./dist/index.d.ts", // dtsroll targets this
		"default": "./dist/index.js",
	},
	"scripts": {
		"build": "tsc && dtsroll",
	},
}

2. Build

npm run build

That's it.

[!WARNING] dtsroll modifies files in-place—bundled source files are removed and entry files are overwritten with bundled output. Use --dry-run first to see what it would change:

dtsroll --dry-run

Behavior

Automatic configuration

By default, dtsroll reads your package.json to determine which imports should be bundled and which should remain external. The recommended setup is to run dtsroll without any configuration and let it infer the correct behavior based on your dependency declarations.

| Dependency type | Action | Reason | | ---------------------- | ----------- | ----------------------------- | | devDependencies | Bundle | Consumers don't install these | | dependencies | Externalize | Consumers already have them | | optionalDependencies | Externalize | Consumers already have them | | peerDependencies | Externalize | Provided by the consumer |

If you have a @types/* package in devDependencies but the corresponding runtime package in dependencies, dtsroll will recommend moving the types package to dependencies, as this can otherwise result in missing types for consumers.

Manual configuration

If your project doesn't have a package.json file, you can still manually specify the input files (which entry files to collapse the imports into), and which packages to externalize.

Subpath imports

[!WARNING] Currently, dtsroll mistakenly resolves and bundles subpath imports. Subpath imports are intended to be dynamic aliases controlled by the consumer. In a future breaking release, dtsroll will likely externalize them to preserve this behavior.

Usage

dtsroll can be used in several ways.

CLI usage

dtsroll [flags] [...entry .d.ts files]

If no entry files are provided, dtsroll reads package.json to determine them.

Flags

| Flag | Alias | Description | | -------------- | ----- | ------------------------------------------------- | | --dry-run | -d | Show what would be bundled without writing files | | --sourcemap | -s | Generate source maps (.d.ts.map files) | | --conditions | -C | Resolution conditions for subpath exports (e.g. production) | | --external | -e | (Only when no package.json) Packages to externalize |

Why use --sourcemap?

Without source maps, "Go to Definition" in VS Code lands you in bundled .d.ts files—often a flattened wall of generated types that's hard to navigate.

With --sourcemap, dtsroll generates .d.ts.map files that map positions in the bundled output back to your original source files. This lets VS Code jump directly to the actual TypeScript implementation instead of the generated declarations.

This is especially useful for:

  • Monorepos — Navigate seamlessly across packages to real source
  • Library authors — Give consumers a better DX when exploring your types
  • Anyone debugging types — Understand types at their origin, not the emitted output

[!NOTE] For source navigation to work, the original .ts source files must be available (either shipped with your package or present locally). If they're not, VS Code falls back to the .d.ts file.

Vite plugin

If you use unplugin-dts, dtsroll will automatically bundle the emitted types immediately after generation:

import { defineConfig } from 'vitest/config'
import dts from 'unplugin-dts/vite'
import { dtsroll } from 'dtsroll/vite'

export default defineConfig({
    plugins: [
        dts(),
        dtsroll()
    ]
})

Node API

import { dtsroll } from 'dtsroll'

await dtsroll({
    cwd: process.cwd(),
    dryRun: false,
    sourcemap: true // generates .d.ts.map files
})

Related

  • pkgroll — Zero-config JS + DTS bundler