#!/usr/bin/env bash # # Assemble the npm thin-installer packages from built bundles (esbuild pattern). # # Produces, under release/npm/: # codegraph-/ one per built bundle — the vendored Node + app, tagged # with os/cpu so npm installs only the matching one. # main/ the @colbymchenry/codegraph shim package: a tiny bin # that execs the matching platform bundle, with every # platform package in optionalDependencies. # # The release pipeline then `npm publish`es each dir. This does NOT touch the # repo's package.json — the dev/from-source path keeps working; the *published* # main package's shape is generated here. # # Prereq: run build-bundle.sh for each target first (release/codegraph-*.tar.gz). # Usage: scripts/pack-npm.sh [version] (default: version from package.json) set -euo pipefail ROOT="$(cd "$(dirname "$0")/.." && pwd)" VERSION="${1:-$(node -p "require('$ROOT/package.json').version")}" SCOPE="@colbymchenry" REL="$ROOT/release" NPM="$REL/npm" rm -rf "$NPM" mkdir -p "$NPM/main" shopt -s nullglob archives=("$REL"/codegraph-*.tar.gz "$REL"/codegraph-*.zip) [ ${#archives[@]} -gt 0 ] || { echo "[pack-npm] no bundles in $REL — run build-bundle.sh first" >&2; exit 1; } targets=() for archive in "${archives[@]}"; do fname="$(basename "$archive")" case "$fname" in *.tar.gz) base="${fname%.tar.gz}" ;; # codegraph- *.zip) base="${fname%.zip}" ;; esac target="${base#codegraph-}" # , e.g. darwin-arm64 / win32-x64 os="${target%-*}" # darwin | linux | win32 arch="${target##*-}" # arm64 | x64 pkgdir="$NPM/$base" mkdir -p "$pkgdir" case "$fname" in *.zip) tmpx="$(mktemp -d)" unzip -q "$archive" -d "$tmpx" mv "$tmpx/codegraph-${target}"/* "$pkgdir"/ rm -rf "$tmpx" nodefile="node.exe" ;; *) tar -xzf "$archive" -C "$pkgdir" --strip-components=1 nodefile="node" ;; esac VERSION="$VERSION" SCOPE="$SCOPE" TARGET="$target" OSV="$os" ARCHV="$arch" NODEFILE="$nodefile" \ node -e ' const fs=require("fs"); fs.writeFileSync(process.argv[1], JSON.stringify({ name: `${process.env.SCOPE}/codegraph-${process.env.TARGET}`, version: process.env.VERSION, description: `CodeGraph self-contained bundle for ${process.env.TARGET}`, os: [process.env.OSV], cpu: [process.env.ARCHV], files: [process.env.NODEFILE, "lib", "bin"], license: "MIT" }, null, 2) + "\n"); ' "$pkgdir/package.json" targets+=("$target") echo "[pack-npm] ${SCOPE}/codegraph-${target}@${VERSION}" done # Main shim package. # npm-shim.js CLI/MCP launcher (execs the bundled Node) — the `bin`. # npm-sdk.js programmatic/embedded entry (#354): re-exports the installed # platform bundle's compiled library — the `main`. # dist/ the .d.ts tree only (types). The runtime .js stays in the # per-platform bundle so its deps aren't duplicated here. cp "$ROOT/scripts/npm-shim.js" "$NPM/main/npm-shim.js" cp "$ROOT/scripts/npm-sdk.js" "$NPM/main/npm-sdk.js" [ -f "$ROOT/README.md" ] && cp "$ROOT/README.md" "$NPM/main/README.md" # Ship the type declarations so `types`/`exports.types` resolve. Built from this # same release, so they can't skew from the runtime npm-sdk.js re-exports. [ -f "$ROOT/dist/index.d.ts" ] || ( echo "[pack-npm] building dist for .d.ts" >&2 && cd "$ROOT" && npm run build >/dev/null ) ROOT="$ROOT" DEST="$NPM/main" node -e ' const fs=require("fs"), path=require("path"); const src=path.join(process.env.ROOT,"dist"), dest=path.join(process.env.DEST,"dist"); fs.cpSync(src, dest, { recursive:true, filter(s){ try { return fs.statSync(s).isDirectory() || s.endsWith(".d.ts"); } catch (e) { return false; } }}); ' VERSION="$VERSION" SCOPE="$SCOPE" TARGETS="${targets[*]}" \ node -e ' const fs=require("fs"); const opt={}; for (const t of process.env.TARGETS.split(/\s+/).filter(Boolean)) opt[`${process.env.SCOPE}/codegraph-${t}`]=process.env.VERSION; fs.writeFileSync(process.argv[1], JSON.stringify({ name: `${process.env.SCOPE}/codegraph`, version: process.env.VERSION, description: "Local-first code intelligence for AI agents (MCP). Self-contained — bundles its own runtime.", bin: { codegraph: "npm-shim.js" }, main: "npm-sdk.js", types: "dist/index.d.ts", exports: { ".": { types: "./dist/index.d.ts", default: "./npm-sdk.js" }, "./package.json": "./package.json" }, optionalDependencies: opt, files: ["npm-shim.js","npm-sdk.js","dist","README.md"], license: "MIT" }, null, 2) + "\n"); ' "$NPM/main/package.json" echo "[pack-npm] ${SCOPE}/codegraph@${VERSION} (${#targets[@]} platform packages in optionalDependencies)" echo "[pack-npm] output: $NPM"