mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-03 09:17:09 +08:00
Replace the tsc + babel pipeline with a single esbuild invocation that produces a self-contained dist/entry.js. The nix TUI derivation no longer copies node_modules — only dist/ + package.json ship, shrinking the output from hundreds of MB to ~2.9 MB. - ui-tui/scripts/build.mjs: new esbuild bundler. Aliases @hermes/ink to source (esbuild's __esm helper doesn't await nested async init, which breaks lazy-assigned exports like 'render' when re-exporting through a prebuilt submodule). Stubs react-devtools-core (dev-only). Injects a createRequire shim for transitive CJS deps. Strips the shebang from src/entry.tsx because Nix patchShebangs mangles '/usr/bin/env -S node --max-old-space-size=8192 --expose-gc' — it drops the 'node' token. The Python launcher always invokes node explicitly, so the shebang is redundant. - nix/tui.nix: installPhase no longer copies node_modules or the @hermes/ink packages dir. - nix/checks.nix: drop the 'node_modules present' assertion. - hermes_cli/main.py: _tui_need_npm_install short-circuits when dist/entry.js exists and no package-lock.json is present. That is the prebuilt-bundle layout (nix / packaged release) and there is nothing to install. Without this, the launcher tried to npm install in a non-existent site-packages/ui-tui path.
62 lines
2.2 KiB
JavaScript
62 lines
2.2 KiB
JavaScript
#!/usr/bin/env node
|
|
// Bundles src/entry.tsx into a single self-contained dist/entry.js.
|
|
// No runtime node_modules needed.
|
|
import { build } from 'esbuild'
|
|
import { readFileSync, writeFileSync } from 'node:fs'
|
|
import { fileURLToPath } from 'node:url'
|
|
import { dirname, resolve } from 'node:path'
|
|
|
|
const here = dirname(fileURLToPath(import.meta.url))
|
|
const root = resolve(here, '..')
|
|
const out = resolve(root, 'dist/entry.js')
|
|
|
|
// `react-devtools-core` is only imported when DEV=true at runtime (Ink dev
|
|
// mode). Stub it out so the bundle doesn't carry the dep.
|
|
const stubDevtools = {
|
|
name: 'stub-react-devtools-core',
|
|
setup(b) {
|
|
b.onResolve({ filter: /^react-devtools-core$/ }, args => ({
|
|
path: args.path,
|
|
namespace: 'stub-devtools'
|
|
}))
|
|
b.onLoad({ filter: /.*/, namespace: 'stub-devtools' }, () => ({
|
|
contents: 'export default { initialize() {}, connectToDevTools() {} }',
|
|
loader: 'js'
|
|
}))
|
|
}
|
|
}
|
|
|
|
await build({
|
|
entryPoints: [resolve(root, 'src/entry.tsx')],
|
|
bundle: true,
|
|
platform: 'node',
|
|
format: 'esm',
|
|
target: 'node20',
|
|
outfile: out,
|
|
jsx: 'automatic',
|
|
jsxImportSource: 'react',
|
|
// Skip the prebuilt @hermes/ink bundle — esbuild's __esm helper doesn't
|
|
// await nested async init, which breaks lazy-initialized exports like
|
|
// `render`. Bundling from source sidesteps that.
|
|
alias: { '@hermes/ink': resolve(root, 'packages/hermes-ink/src/entry-exports.ts') },
|
|
plugins: [stubDevtools],
|
|
// Some transitive deps use CommonJS `require(...)` at runtime. ESM bundles
|
|
// don't get a `require` binding automatically, so we inject one.
|
|
banner: {
|
|
js: "import { createRequire as __cr } from 'node:module'; const require = __cr(import.meta.url);"
|
|
},
|
|
logLevel: 'info'
|
|
})
|
|
|
|
// esbuild preserves the shebang from src/entry.tsx into the bundle, but Nix's
|
|
// patchShebangs phase mangles `/usr/bin/env -S node --foo --bar` (it strips
|
|
// the `node` token, leaving a broken interpreter). The hermes_cli launcher
|
|
// always invokes this file as `node dist/entry.js` anyway, so the shebang is
|
|
// redundant — strip it.
|
|
const body = readFileSync(out, 'utf8')
|
|
if (body.startsWith('#!')) {
|
|
writeFileSync(out, body.slice(body.indexOf('\n') + 1))
|
|
}
|
|
|
|
console.log(`built ${out}`)
|