| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119 |
- #!/usr/bin/env bash
- #
- # Assemble the npm thin-installer packages from built bundles (esbuild pattern).
- #
- # Produces, under release/npm/:
- # codegraph-<target>/ 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-<target>
- *.zip) base="${fname%.zip}" ;;
- esac
- target="${base#codegraph-}" # <target>, 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"
|