HMR is enabled by default when using Bun’s full-stack development server.
import.meta.hot
API Reference
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.
For this to work, Bun forces these APIs to be called without indirection. That means the following do not work:
index.ts
The HMR API is still a work in progress. Some features are missing. HMR can be disabled in
Bun.serve
by setting the development option to { hmr: false }
.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 |
import.meta.hot.accept()
Theaccept()
method indicates that a module can be hot-replaced. When called without arguments, it indicates that this module can be replaced simply by re-evaluating the file. After a hot update, importers of this module will be automatically patched.
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.
When no modules call
import.meta.hot.accept()
(and there isn’t React Fast Refresh or a plugin
calling it for you), the page will reload when the file updates, and a console warning shows which
files were invalidated. This warning is safe to ignore if it makes more sense to rely on full page
reloads.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.
Prefer using
import.meta.hot.accept()
without an argument as it usually makes your code easier
to understand.Accepting other modules
With multiple dependencies
undefined
for any that had errors.
import.meta.hot.data
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()
).
data
is inlined to be {}
, meaning it cannot be used as a state holder.
The above pattern is recommended for stateful modules because Bun knows it can minify
{}.prop ??= value
into value
in production.import.meta.hot.dispose()
Attaches an on-dispose callback. This is called:- Just before the module is replaced with another copy (before the next is loaded)
- After the module is detached (removing all imports to this module, see
import.meta.hot.prune()
)
This callback is not called on route navigation or when the browser tab closes.
import.meta.hot.prune()
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. Unlikeimport.meta.hot.dispose()
, this pairs much better with accept
and data
to manage stateful resources. A full example managing a WebSocket:
If
dispose
was used instead, the WebSocket would close and re-open on every hot update. Both
versions of the code will prevent page reloads when imported files are updated.import.meta.hot.on() and off()
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.
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. |
For compatibility with Vite, the above events are also available via
vite:*
prefix instead of
bun:*
.