rembrembdocs

Hot Module Replacement (HMR) allows you to update modules in a running application without needing a full page reload. This preserves the application state and improves the development experience.

Bun implements a client-side HMR API modeled after Vite’s import.meta.hot API. It can be checked for with if (import.meta.hot), tree-shaking it in production.

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

if (import.meta.hot) {
  // HMR APIs are available.
}

However, this check is often not needed as Bun will dead-code-eliminate calls to all of the HMR APIs in production builds.

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

// This entire function call will be removed in production!
import.meta.hot.dispose(() => {
  console.log("dispose");
});

API Methods

Method

Status

Notes

hot.accept()

Indicate that a hot update can be replaced gracefully.

hot.data

Persist data between module evaluations.

hot.dispose()

Add a callback function to run when a module is about to be replaced.

hot.invalidate()

hot.on()

Attach an event listener

hot.off()

Remove an event listener from on.

hot.send()

hot.prune()

🚧

NOTE: Callback is currently never called.

hot.decline()

No-op to match Vite’s import.meta.hot

The accept() method indicates that a module can be hot-replaced. When called without arguments, it indicates that this module can be replaced by re-evaluating the file. After a hot update, importers of this module will be automatically patched.

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

// index.ts
import { getCount } from "./foo.ts";

console.log("count is ", getCount());

import.meta.hot.accept();

export function getNegativeCount() {
  return -getCount();
}

This creates a hot-reloading boundary for all of the files that index.ts imports. That means whenever foo.ts or any of its dependencies are saved, the update will bubble up to index.ts will re-evaluate. Files that import index.ts will then be patched to import the new version of getNegativeCount(). If only index.ts is updated, only the one file will be re-evaluated, and the counter in foo.ts is reused. This may be used in combination with import.meta.hot.data to transfer state from the previous module to the new one.

With callback

When provided one callback, import.meta.hot.accept will function how it does in Vite. Instead of patching the importers of this module, it will call the callback with the new module.

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

export const count = 0;

import.meta.hot.accept(newModule => {
  if (newModule) {
    // newModule is undefined when SyntaxError happened
    console.log("updated: count is now ", newModule.count);
  }
});

Accepting other modules

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

import { count } from "./foo";

import.meta.hot.accept("./foo", () => {
  if (!newModule) return;

  console.log("updated: count is now ", count);
});

Indicates that a dependency’s module can be accepted. When the dependency is updated, the callback will be called with the new module.

With multiple dependencies

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

import.meta.hot.accept(["./foo", "./bar"], newModules => {
  // newModules is an array where each item corresponds to the updated module
  // or undefined if that module had a syntax error
});

Indicates that multiple dependencies’ modules can be accepted. This variant accepts an array of dependencies, where the callback will receive the updated modules, and undefined for any that had errors.

import.meta.hot.data maintains state between module instances during hot replacement, enabling data transfer from previous to new versions. When import.meta.hot.data is written into, Bun will also mark this module as capable of self-accepting (equivalent of calling import.meta.hot.accept()).

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.tsx

import { createRoot } from "react-dom/client";
import { App } from "./app";

const root = (import.meta.hot.data.root ??= createRoot(elem));
root.render(<App />); // re-use an existing root

In production, data is inlined to be {}, meaning it cannot be used as a state holder.

Attaches an on-dispose callback. This is called:

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

const sideEffect = setupSideEffect();

import.meta.hot.dispose(() => {
  sideEffect.cleanup();
});

Returning a promise will delay module replacement until the module is disposed. All dispose callbacks are called in parallel.

Attaches an on-prune callback. This is called when all imports to this module are removed, but the module was previously loaded. This can be used to clean up resources that were created when the module was loaded. Unlike import.meta.hot.dispose(), this pairs much better with accept and data to manage stateful resources. A full example managing a WebSocket:

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

import { something } from "./something";

// Initialize or re-use a WebSocket connection
export const ws = (import.meta.hot.data.ws ??= new WebSocket(location.origin));

// If the module's import is removed, clean up the WebSocket connection.
import.meta.hot.prune(() => {
  ws.close();
});

on() and off() are used to listen for events from the HMR runtime. Event names are prefixed with a prefix so that plugins do not conflict with each other.

https://mintcdn.com/bun-1dd33a4e/JUhaF6Mf68z_zHyy/icons/typescript.svg?fit=max&auto=format&n=JUhaF6Mf68z_zHyy&q=85&s=7ac549adaea8d5487d8fbd58cc3ea35bindex.ts

import.meta.hot.on("bun:beforeUpdate", () => {
  console.log("before a hot update");
});

When a file is replaced, all of its event listeners are automatically removed.

Built-in events

Event

Emitted when

bun:beforeUpdate

before a hot update is applied.

bun:afterUpdate

after a hot update is applied.

bun:beforeFullReload

before a full page reload happens.

bun:beforePrune

before prune callbacks are called.

bun:invalidate

when a module is invalidated with import.meta.hot.invalidate()

bun:error

when a build or runtime error occurs

bun:ws:disconnect

when the HMR WebSocket connection is lost. This can indicate the development server is offline.

bun:ws:connect

when the HMR WebSocket connects or re-connects.