## Summary - revert the Bun standalone desktop packaging path and restore the server's original `dist/bin.js` bootstrap flow - add a managed Node runtime for Electron and Tauri that downloads only the current platform/arch artifact into `~/.config/codenomad` - update desktop startup and packaging scripts so packaged apps use the managed runtime consistently, and clean up Electron's expected navigation-abort log noise ## Testing - npm run typecheck --workspace @neuralnomads/codenomad-electron-app - cargo check - npm run build --workspace @neuralnomads/codenomad - npm run build:mac --workspace @neuralnomads/codenomad-electron-app - launch `packages/electron-app/release/mac-arm64/CodeNomad.app/Contents/MacOS/CodeNomad` and verify the packaged server reaches ready with the managed Node runtime
148 lines
4.2 KiB
JavaScript
148 lines
4.2 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
import { spawn } from "child_process"
|
|
import { existsSync } from "fs"
|
|
import path, { join } from "path"
|
|
import { fileURLToPath } from "url"
|
|
|
|
const __dirname = fileURLToPath(new URL(".", import.meta.url))
|
|
const appDir = join(__dirname, "..")
|
|
const workspaceRoot = join(appDir, "..", "..")
|
|
|
|
const npmCmd = process.platform === "win32" ? "npm.cmd" : "npm"
|
|
const npxCmd = process.platform === "win32" ? "npx.cmd" : "npx"
|
|
const nodeModulesPath = join(appDir, "node_modules")
|
|
const workspaceNodeModulesPath = join(workspaceRoot, "node_modules")
|
|
|
|
const platforms = {
|
|
mac: {
|
|
args: ["--mac", "--x64", "--arm64"],
|
|
description: "macOS (Intel & Apple Silicon)",
|
|
},
|
|
"mac-x64": {
|
|
args: ["--mac", "--x64"],
|
|
description: "macOS (Intel only)",
|
|
},
|
|
"mac-arm64": {
|
|
args: ["--mac", "--arm64"],
|
|
description: "macOS (Apple Silicon only)",
|
|
},
|
|
win: {
|
|
args: ["--win", "--x64"],
|
|
description: "Windows (x64)",
|
|
},
|
|
"win-arm64": {
|
|
args: ["--win", "--arm64"],
|
|
description: "Windows (ARM64)",
|
|
},
|
|
linux: {
|
|
args: ["--linux", "--x64"],
|
|
description: "Linux (x64)",
|
|
},
|
|
"linux-arm64": {
|
|
args: ["--linux", "--arm64"],
|
|
description: "Linux (ARM64)",
|
|
},
|
|
"linux-rpm": {
|
|
args: ["--linux", "rpm", "--x64", "--arm64"],
|
|
description: "Linux RPM packages (x64 & ARM64)",
|
|
},
|
|
all: {
|
|
args: ["--mac", "--win", "--linux", "--x64", "--arm64"],
|
|
description: "All platforms (macOS, Windows, Linux)",
|
|
},
|
|
}
|
|
|
|
function run(command, args, options = {}) {
|
|
return new Promise((resolve, reject) => {
|
|
const env = { ...process.env, NODE_PATH: nodeModulesPath, ...(options.env || {}) }
|
|
const pathKey = Object.keys(env).find((key) => key.toLowerCase() === "path") ?? "PATH"
|
|
|
|
const binPaths = [
|
|
join(nodeModulesPath, ".bin"),
|
|
join(workspaceNodeModulesPath, ".bin"),
|
|
]
|
|
|
|
env[pathKey] = `${binPaths.join(path.delimiter)}${path.delimiter}${env[pathKey] ?? ""}`
|
|
|
|
const spawnOptions = {
|
|
cwd: appDir,
|
|
stdio: "inherit",
|
|
shell: process.platform === "win32",
|
|
...options,
|
|
env,
|
|
}
|
|
|
|
const child = spawn(command, args, spawnOptions)
|
|
|
|
child.on("error", reject)
|
|
child.on("exit", (code) => {
|
|
if (code === 0) {
|
|
resolve(undefined)
|
|
} else {
|
|
reject(new Error(`${command} ${args.join(" ")} exited with code ${code}`))
|
|
}
|
|
})
|
|
})
|
|
}
|
|
|
|
function printAvailablePlatforms() {
|
|
console.error(`\nAvailable platforms:`)
|
|
for (const [name, cfg] of Object.entries(platforms)) {
|
|
console.error(` - ${name.padEnd(12)} : ${cfg.description}`)
|
|
}
|
|
}
|
|
|
|
async function build(platform) {
|
|
const config = platforms[platform]
|
|
|
|
if (!config) {
|
|
console.error(`❌ Unknown platform: ${platform}`)
|
|
printAvailablePlatforms()
|
|
process.exit(1)
|
|
}
|
|
|
|
console.log(`\n🔨 Building for: ${config.description}\n`)
|
|
|
|
try {
|
|
console.log("📦 Step 1/3: Building CLI dependency...\n")
|
|
await run(npmCmd, ["run", "build", "--workspace", "@neuralnomads/codenomad"], {
|
|
cwd: workspaceRoot,
|
|
env: { NODE_PATH: workspaceNodeModulesPath },
|
|
})
|
|
|
|
console.log("\n📦 Step 1.5/3: Preparing packaged server resources...\n")
|
|
await run(process.execPath, [join(appDir, "scripts", "prepare-resources.js")], {
|
|
cwd: workspaceRoot,
|
|
env: { NODE_PATH: workspaceNodeModulesPath },
|
|
})
|
|
|
|
console.log("\n📦 Step 2/3: Building Electron app...\n")
|
|
await run(npmCmd, ["run", "build"])
|
|
|
|
console.log("\n📦 Step 3/3: Packaging binaries...\n")
|
|
const distPath = join(appDir, "dist")
|
|
if (!existsSync(distPath)) {
|
|
throw new Error("dist/ directory not found. Build failed.")
|
|
}
|
|
|
|
await run(npxCmd, ["electron-builder", "--publish=never", ...config.args])
|
|
|
|
console.log("\n✅ Build complete!")
|
|
console.log(`📁 Binaries available in: ${join(appDir, "release")}\n`)
|
|
} catch (error) {
|
|
console.error("\n❌ Build failed:", error)
|
|
process.exit(1)
|
|
}
|
|
}
|
|
|
|
const platform = process.argv[2] || "mac"
|
|
|
|
console.log(`
|
|
╔════════════════════════════════════════╗
|
|
║ CodeNomad - Binary Builder ║
|
|
╚════════════════════════════════════════╝
|
|
`)
|
|
|
|
await build(platform)
|