Bun

Catalogs

Catalogs in Bun provide a straightforward way to share common dependency versions across multiple packages in a monorepo. Rather than specifying the same versions repeatedly in each workspace package, you define them once in the root package.json and reference them consistently throughout your project.

Overview

Unlike traditional dependency management where each workspace package needs to independently specify versions, catalogs let you:

  1. Define version catalogs in the root package.json
  2. Reference these versions with a simple catalog: protocol
  3. Update all packages simultaneously by changing the version in just one place

This is especially useful in large monorepos where dozens of packages need to use the same version of key dependencies.

How to Use Catalogs

Directory Structure Example

Consider a monorepo with the following structure:

my-monorepo/
├── package.json
├── bun.lock
└── packages/
    ├── app/
    │   └── package.json
    ├── ui/
    │   └── package.json
    └── utils/
        └── package.json

1. Define Catalogs in Root package.json

In your root-level package.json, add a catalog or catalogs field within the workspaces object:

{
  "name": "my-monorepo",
  "workspaces": {
    "packages": ["packages/*"],
    "catalog": {
      "react": "^19.0.0",
      "react-dom": "^19.0.0"
    },
    "catalogs": {
      "testing": {
        "jest": "30.0.0",
        "testing-library": "14.0.0"
      }
    }
  }
}

2. Reference Catalog Versions in Workspace Packages

In your workspace packages, use the catalog: protocol to reference versions:

packages/app/package.json

{
  "name": "app",
  "dependencies": {
    "react": "catalog:",
    "react-dom": "catalog:",
    "jest": "catalog:testing"
  }
}

packages/ui/package.json

{
  "name": "ui",
  "dependencies": {
    "react": "catalog:",
    "react-dom": "catalog:"
  },
  "devDependencies": {
    "jest": "catalog:testing",
    "testing-library": "catalog:testing"
  }
}

3. Run Bun Install

Run bun install to install all dependencies according to the catalog versions.

Catalog vs Catalogs

Bun supports two ways to define catalogs:

  1. catalog (singular): A single default catalog for commonly used dependencies

    "catalog": {
      "react": "^19.0.0",
      "react-dom": "^19.0.0"
    }
    

    Reference with simply catalog::

    "dependencies": {
      "react": "catalog:"
    }
    
  2. catalogs (plural): Multiple named catalogs for grouping dependencies

    "catalogs": {
      "testing": {
        "jest": "30.0.0"
      },
      "ui": {
        "tailwind": "4.0.0"
      }
    }
    

    Reference with catalog:<name>:

    "dependencies": {
      "jest": "catalog:testing",
      "tailwind": "catalog:ui"
    }
    

Benefits of Using Catalogs

  • Consistency: Ensures all packages use the same version of critical dependencies
  • Maintenance: Update a dependency version in one place instead of across multiple package.json files
  • Clarity: Makes it obvious which dependencies are standardized across your monorepo
  • Simplicity: No need for complex version resolution strategies or external tools

Real-World Example

Here's a more comprehensive example for a React application:

Root package.json

{
  "name": "react-monorepo",
  "workspaces": {
    "packages": ["packages/*"],
    "catalog": {
      "react": "^19.0.0",
      "react-dom": "^19.0.0",
      "react-router-dom": "^6.15.0"
    },
    "catalogs": {
      "build": {
        "webpack": "5.88.2",
        "babel": "7.22.10"
      },
      "testing": {
        "jest": "29.6.2",
        "react-testing-library": "14.0.0"
      }
    }
  },
  "devDependencies": {
    "typescript": "5.1.6"
  }
}

packages/app/package.json

{
  "name": "app",
  "dependencies": {
    "react": "catalog:",
    "react-dom": "catalog:",
    "react-router-dom": "catalog:",
    "@monorepo/ui": "workspace:*",
    "@monorepo/utils": "workspace:*"
  },
  "devDependencies": {
    "webpack": "catalog:build",
    "babel": "catalog:build",
    "jest": "catalog:testing",
    "react-testing-library": "catalog:testing"
  }
}

packages/ui/package.json

{
  "name": "@monorepo/ui",
  "dependencies": {
    "react": "catalog:",
    "react-dom": "catalog:"
  },
  "devDependencies": {
    "jest": "catalog:testing",
    "react-testing-library": "catalog:testing"
  }
}

packages/utils/package.json

{
  "name": "@monorepo/utils",
  "dependencies": {
    "react": "catalog:"
  },
  "devDependencies": {
    "jest": "catalog:testing"
  }
}

Updating Versions

To update versions across all packages, simply change the version in the root package.json:

"catalog": {
  "react": "^19.1.0",  // Updated from ^19.0.0
  "react-dom": "^19.1.0"  // Updated from ^19.0.0
}

Then run bun install to update all packages.

Lockfile Integration

Bun's lockfile tracks catalog versions, making it easy to ensure consistent installations across different environments. The lockfile includes:

  • The catalog definitions from your package.json
  • The resolution of each cataloged dependency
// bun.lock (excerpt)
{
  "lockfileVersion": 1,
  "workspaces": {
    "": {
      "name": "react-monorepo",
    },
    "packages/app": {
      "name": "app",
      "dependencies": {
        "react": "catalog:",
        "react-dom": "catalog:",
        ...
      },
    },
    ...
  },
  "catalog": {
    "react": "^19.0.0",
    "react-dom": "^19.0.0",
    ...
  },
  "catalogs": {
    "build": {
      "webpack": "5.88.2",
      ...
    },
    ...
  },
  "packages": {
    ...
  }
}

Limitations and Edge Cases

  • Catalog references must match a dependency defined in either catalog or one of the named catalogs
  • Empty strings and whitespace in catalog names are ignored (treated as default catalog)
  • Invalid dependency versions in catalogs will fail to resolve during bun install
  • Catalogs are only available within workspaces; they cannot be used outside the monorepo

Bun's catalog system provides a powerful yet simple way to maintain consistency across your monorepo without introducing additional complexity to your workflow.