Run Node.js in the browser. Filesystem, shell, npm packages, HTTP servers, no backend required.
npm install @scelar/nodepodNodepod uses a service worker to route preview iframes and virtual HTTP
servers. It has to be served from your own origin at /__sw__.js,
browsers won't register a SW from node_modules.
Pick the one-liner for your framework:
// vite.config.ts
import { defineConfig } from 'vite';
import nodepod from '@scelar/nodepod/vite';
export default defineConfig({
plugins: [nodepod()],
});// app/__sw__.js/route.ts
export { GET } from '@scelar/nodepod/next';If you already have a proxy.ts (Next 16+) or middleware.ts (Next <=15),
compose nodepodProxy / nodepodMiddleware alongside your own handler
instead. See docs/sw-setup.md.
import { serveSW } from '@scelar/nodepod/server';
app.get('/__sw__.js', () => serveSW());import { serveSWNode } from '@scelar/nodepod/server';
app.get('/__sw__.js', async (_req, res) => {
const { body, headers } = await serveSWNode();
for (const [k, v] of Object.entries(headers)) res.setHeader(k, v);
res.status(200).send(body);
});No server code? Copy the file once into your public directory:
cp node_modules/@scelar/nodepod/dist/__sw__.js public/__sw__.jsSee docs/sw-setup.md for the full story and how to customise the URL.
import { Nodepod } from '@scelar/nodepod';
const nodepod = await Nodepod.boot({
files: {
'/index.js': 'console.log("Hello from the browser!")',
},
});
const proc = await nodepod.spawn('node', ['index.js']);
proc.on('output', (text) => console.log(text));
await proc.completion;If the service worker isn't reachable at /__sw__.js, boot() throws a
NodepodSWSetupError with the one-liner for your framework. Pass
{ serviceWorker: false } to skip SW setup entirely (SSR, Node tests).
Plug in xterm.js for an interactive shell:
import { Terminal } from '@xterm/xterm';
import { FitAddon } from '@xterm/addon-fit';
const terminal = nodepod.createTerminal({ Terminal, FitAddon });
terminal.attach('#terminal-container');await nodepod.install(['express']);
const proc = await nodepod.spawn('node', ['server.js']);Works with Express, Hono, Vite, and anything that calls listen():
const nodepod = await Nodepod.boot({
files: {
'/server.js': `
const express = require('express');
const app = express();
app.get('/', (req, res) => res.json({ ok: true }));
app.listen(3000);
`,
},
});
await nodepod.install(['express']);
await nodepod.spawn('node', ['server.js']);
const response = await nodepod.request(3000, 'GET', '/');
console.log(response.body); // { ok: true }Save and restore the filesystem:
const snapshot = await nodepod.snapshot();
// ... later
await nodepod.restore(snapshot);| Option | Type | Description |
|---|---|---|
files |
Record<string, string | Uint8Array> |
Initial files |
workdir |
string |
Working directory (default "/") |
env |
Record<string, string> |
Environment variables |
swUrl |
string |
Service Worker URL for preview iframes |
watermark |
boolean |
Show nodepod badge in previews (default true) |
onServerReady |
(port, url) => void |
Called when a virtual server starts |
allowedFetchDomains |
string[] | null |
Extra CORS proxy domains. null = allow all |
| Method | Description |
|---|---|
spawn(cmd, args?, opts?) |
Run a command |
install(packages) |
Install npm packages |
createTerminal(opts) |
Create an xterm.js terminal |
fs.readFile(path, enc?) |
Read a file |
fs.writeFile(path, data) |
Write a file |
fs.readdir(path) |
List directory |
fs.stat(path) |
File stats |
fs.mkdir(path, opts?) |
Create directory |
fs.rm(path, opts?) |
Remove file/directory |
snapshot() |
Capture filesystem state |
restore(snapshot) |
Restore from snapshot |
request(port, method, path) |
Send request to virtual server |
port(num) |
Get preview URL for a port |
setPreviewScript(js) |
Inject JS into preview iframes |
clearPreviewScript() |
Remove injected script |
spawn() returns a NodepodProcess:
proc.on('output', (text) => { }); // stdout
proc.on('error', (text) => { }); // stderr
proc.on('exit', (code) => { }); // exit code
await proc.completion; // wait for exitFull: fs, path, events, stream, buffer, process, http, https, net, crypto, zlib, url, querystring, util, os, tty, child_process, assert, readline, module, timers, string_decoder, perf_hooks, constants, punycode
Stubs: dns, worker_threads, vm, v8, tls, dgram, cluster, http2, inspector, domain, diagnostics_channel, async_hooks
In development: Native WASI/WASM loading for napi-rs based packages (rolldown, lightningcss, etc.)
We built nodepod for Scelar, an AI app builder that takes you from idea to production in minutes. Scelar needed a way to run real Node.js code directly in the browser so users could build, preview, and interact with their apps instantly without waiting for remote servers to spin up. No containers, no cold starts, no infrastructure to manage.
We open-sourced it because we think running Node in the browser shouldn't be a proprietary black box. If you're building a web IDE, coding playground, AI dev tool, or anything that needs server-side JS in the browser, nodepod is for you.
git clone https://github.com/ScelarOrg/Nodepod.git
cd Nodepod
npm install
npm run build:publish # build library + types
npm test # run testsnpm version patch # or minor / major
npm publish # auto-builds before publishing
git push && git push --tagsContributions are really appreciated. This is a big project and it's hard to maintain and push updates on my own. If you want to help out, feel free to open a PR.
Note that nodepod is not my main focus, Scelar is. I work on nodepod when I have time for it, so responses to issues and PRs might take a bit.
Before opening a PR, make sure these pass:
npm run type-check # 0 TypeScript errors
npm run build:publish # builds cleanly
npm test # tests passCode style:
- Files are kebab-case (
shell-parser.ts,memory-volume.ts). Polyfills match their Node.js module name (fs.ts,crypto.ts). - Classes and types are PascalCase (
MemoryVolume,ShellResult). Functions and variables are camelCase. - Private properties and internal helpers use a leading underscore (
_registry,_ensureSlot()). - Use named exports. Default exports only for polyfills that need to match Node's
module.exportsshape.
Commit messages follow conventional commits:
feat: add readline support
fix: resolve path edge case in fs.watch
chore: bump dependencies
If you're writing a polyfill:
- Polyfill files live in
src/polyfills/and must be named after the Node.js module they replace. - EventEmitter methods must use
_reg()for lazy init, never accessthis._registrydirectly. - Polyfills registered in
CORE_MODULESmust not useasyncfunctions. - ESM-to-CJS replacement strings must include trailing semicolons.
MIT + Commons Clause. Use it in anything, just don't resell nodepod itself.