Features: - Binary-Free Mode: No OpenCode binary required - NomadArch Native mode with free Zen models - Native session management - Provider routing (Zen, Qwen, Z.AI) - Fixed MCP connection with explicit connectAll() - Updated installers and launchers for all platforms - UI binary selector with Native option Free Models Available: - GPT-5 Nano (400K context) - Grok Code Fast 1 (256K context) - GLM-4.7 (205K context) - Doubao Seed Code (256K context) - Big Pickle (200K context)
11 KiB
FINAL EXECUTION PLAN - 8 Fixes with Proof Deliverables
Fix Summary
| Fix | Files | Deliverables |
|---|---|---|
| C1 | Install-Windows.bat, Install-Mac.sh, Install-Linux.sh, Launch-Windows.bat, Launch-Dev-Windows.bat, Launch-Unix.sh | 9 path diffs + dir packages\ui\dist verification |
| C2 | packages/ui/vite.config.ts, Launch-Dev-Windows.bat, Launch-Dev-Unix.sh (NEW) | vite.config.ts diff + 2 launcher diffs + Vite log showing port |
| C3 | Launch-Windows.bat, Launch-Dev-Windows.bat, Launch-Unix.sh | 3 CLI_PORT env var diffs + server log showing port |
| C4 | Install-Windows.bat, Install-Mac.sh, Install-Linux.sh | 3 download/checksum diffs + log verification |
| C5 | Install-Windows.bat | Certutil parsing diff + hash output |
| C6 | Install-Windows.bat, Install-Mac.sh, Install-Linux.sh | 3 TARGET_DIR/BIN_DIR diffs + fallback test output |
| C7 | Install-Windows.bat, Install-Mac.sh, Install-Linux.sh | 3 health check path diffs + health check output |
| C8 | Launch-Dev-Windows.bat | 1 path diff + grep verification |
C1: UI Build Path Correction
Files: Install-Windows.bat (lines 194, 245), Install-Mac.sh (204, 256), Install-Linux.sh (220, 272), Launch-Windows.bat (185), Launch-Dev-Windows.bat (144), Launch-Unix.sh (178)
Diff:
# All Windows scripts - replace:
packages\ui\src\renderer\dist
# With:
packages\ui\dist
# All Unix scripts - replace:
packages/ui/src/renderer/dist
# With:
packages/ui/dist
Verification: dir packages\ui\dist + dir packages\ui\dist\index.html
C2: Vite Dev Server Port Wiring
File 1: packages/ui/vite.config.ts (line 23)
- server: {
- port: 3000,
- },
+ server: {
+ port: Number(process.env.VITE_PORT ?? 3000),
+ },
File 2: Launch-Dev-Windows.bat (after port detection)
- start "NomadArch UI" cmd /k "cd /d \"%~dp0packages\ui\" && set VITE_PORT=!UI_PORT! && npm run dev"
+ start "NomadArch UI" cmd /k "cd /d \"%~dp0packages\ui\" && set VITE_PORT=!UI_PORT! && npm run dev -- --port !UI_PORT!"
File 3: Launch-Dev-Unix.sh (NEW FILE)
#!/bin/bash
set -e
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
cd "$SCRIPT_DIR"
# Port detection
DEFAULT_SERVER_PORT=3001
DEFAULT_UI_PORT=5173
SERVER_PORT=$DEFAULT_SERVER_PORT
UI_PORT=$DEFAULT_UI_PORT
echo "[INFO] Detecting available ports..."
# Server port (3001-3050)
for port in {3001..3050}; do
if ! lsof -i :$port -sTCP:LISTEN -t > /dev/null 2>&1; then
SERVER_PORT=$port
break
fi
done
# UI port (5173-5200)
for port in {5173..5200}; do
if ! lsof -i :$port -sTCP:LISTEN -t > /dev/null 2>&1; then
UI_PORT=$port
break
fi
done
echo "[INFO] Using server port: $SERVER_PORT"
echo "[INFO] Using UI port: $UI_PORT"
# Start server with CLI_PORT
echo "[INFO] Starting Backend Server..."
cd packages/server
export CLI_PORT=$SERVER_PORT
npm run dev &
SERVER_PID=$!
sleep 3
# Start UI with VITE_PORT + --port flag
echo "[INFO] Starting Frontend UI..."
cd "$SCRIPT_DIR/packages/ui"
export VITE_PORT=$UI_PORT
npm run dev -- --port $UI_PORT &
UI_PID=$!
sleep 3
# Start Electron
echo "[INFO] Starting Electron..."
cd "$SCRIPT_DIR/packages/electron-app"
npm run dev
# Cleanup on exit
trap "kill $SERVER_PID $UI_PID 2>/dev/null; exit" INT TERM
Verification: Vite log output showing Local: http://localhost:<detected_port>
C3: Server Port Environment Variable
Launch-Windows.bat (before npm run dev:electron):
echo [INFO] Starting NomadArch...
set SERVER_URL=http://localhost:!SERVER_PORT!
echo [INFO] Server will run on http://localhost:!SERVER_PORT!
+
+ set CLI_PORT=!SERVER_PORT!
call npm run dev:electron
Launch-Dev-Windows.bat (server start command):
echo [INFO] Starting Backend Server...
- start "NomadArch Server" cmd /k "cd /d \"%~dp0packages\server\" && npm run dev"
+ start "NomadArch Server" cmd /k "cd /d \"%~dp0packages\server\" && set CLI_PORT=!SERVER_PORT! && npm run dev"
Launch-Unix.sh (before npm run dev:electron):
echo -e "${GREEN}[INFO]${NC} Starting NomadArch..."
SERVER_URL="http://localhost:$SERVER_PORT"
echo -e "${GREEN}[INFO]${NC} Server will run on http://localhost:$SERVER_PORT"
export CLI_PORT=$SERVER_PORT
npm run dev:electron
Verification: Server log showing CodeNomad Server is ready at http://127.0.0.1:<detected_port>
C4: OpenCode Download with Dynamic Version + Checksum
Install-Windows.bat (lines 165-195):
set TARGET_DIR=%SCRIPT_DIR%
set BIN_DIR=%TARGET_DIR%\bin
if not exist "%BIN_DIR%" mkdir "%BIN_DIR%"
:: Resolve latest version from GitHub API
echo [INFO] Resolving latest OpenCode version...
for /f "delims=" %%v in ('curl -s https://api.github.com/repos/sst/opencode/releases/latest ^| findstr "\"tag_name\""') do (
set OPENCODE_VERSION=%%v
set OPENCODE_VERSION=!OPENCODE_VERSION:~18,-2!
)
set OPENCODE_BASE=https://github.com/sst/opencode/releases/download/v%OPENCODE_VERSION%
set OPENCODE_URL=%OPENCODE_BASE%/opencode-windows-%ARCH%.exe
set CHECKSUM_URL=%OPENCODE_BASE%/checksums.txt
if exist "%BIN_DIR%\opencode.exe" (
echo [OK] OpenCode binary already exists
) else (
echo [INFO] Downloading OpenCode v%OPENCODE_VERSION%...
echo Downloading from: %OPENCODE_URL%
:: Download binary to BIN_DIR
curl -L -o "%BIN_DIR%\opencode.exe.tmp" "%OPENCODE_URL%"
if %ERRORLEVEL% neq 0 (
echo [ERROR] Download failed!
set /a ERRORS+=1
goto :skip_opencode
)
:: Download checksums
curl -L -o "%BIN_DIR%\checksums.txt" "%CHECKSUM_URL%"
:: Extract expected checksum
set EXPECTED_HASH=
for /f "tokens=1,2" %%h in ('type "%BIN_DIR%\checksums.txt" ^| findstr /i "opencode-windows-%ARCH%"') do (
set EXPECTED_HASH=%%h
)
:: Calculate actual hash (line 2 from certutil)
set ACTUAL_HASH=
for /f "skip=1 tokens=*" %%h in ('certutil -hashfile "%BIN_DIR%\opencode.exe.tmp" SHA256 ^| findstr /v "CertUtil" ^| findstr /v "hash of"') do (
set ACTUAL_HASH=%%h
goto :hash_found
)
:hash_found
:: Verify and output hashes
echo Expected hash: !EXPECTED_HASH!
echo Actual hash: !ACTUAL_HASH!
if "!ACTUAL_HASH!"=="!EXPECTED_HASH!" (
move /Y "%BIN_DIR%\opencode.exe.tmp" "%BIN_DIR%\opencode.exe"
echo [OK] OpenCode downloaded and verified
echo [%date% %time%] OpenCode v%OPENCODE_VERSION% downloaded, checksum verified >> "%TARGET_DIR%\install.log"
) else (
echo [ERROR] Checksum mismatch!
del "%BIN_DIR%\opencode.exe.tmp"
set /a ERRORS+=1
)
)
:skip_opencode
Install-Mac.sh / Install-Linux.sh: Similar pattern with opencode-darwin-${ARCH} and opencode-linux-${ARCH}, using TARGET_DIR/bin
Verification: Log shows OpenCode v<x.y.z> downloaded, checksum verified + ls TARGET_DIR/bin/opencode exists
C5: Windows Checksum Parsing
Included in C4 above. Key change:
:: Parse certutil output - hash is on line 2
for /f "skip=1 tokens=*" %%h in ('certutil -hashfile "%BIN_DIR%\opencode.exe.tmp" SHA256 ^| findstr /v "CertUtil" ^| findstr /v "hash of"') do (
set ACTUAL_HASH=%%h
goto :hash_found
)
Verification: Output shows matching hashes:
Expected hash: abc123def456...
Actual hash: abc123def456...
C6: Permission Fallback with TARGET_DIR/BIN_DIR
Install-Windows.bat (lines 125-160):
set TARGET_DIR=%SCRIPT_DIR%
set BIN_DIR=%TARGET_DIR%\bin
set NEEDS_FALLBACK=0
echo [STEP 2/10] Checking Write Permissions...
echo.
echo. > "%SCRIPT_DIR%\test-write.tmp" 2>nul
if %ERRORLEVEL% neq 0 (
echo [WARN] Cannot write to current directory: %SCRIPT_DIR%
echo [INFO] Setting fallback for install outputs...
set TARGET_DIR=%USERPROFILE%\NomadArch-Install
set BIN_DIR=%TARGET_DIR%\bin
if not exist "%TARGET_DIR%" mkdir "%TARGET_DIR%"
if not exist "%BIN_DIR%" mkdir "%BIN_DIR%"
echo. > "%TARGET_DIR%\test-write.tmp" 2>nul
if %ERRORLEVEL% neq 0 (
echo [ERROR] Cannot write to fallback directory either!
set /a ERRORS+=1
goto :final_check
)
echo [OK] Using fallback for outputs: %TARGET_DIR%
echo [%date% %time%] Using fallback: %TARGET_DIR% >> "%TARGET_DIR%\install.log"
set NEEDS_FALLBACK=1
del "%TARGET_DIR%\test-write.tmp"
) else (
if not exist "%BIN_DIR%" mkdir "%BIN_DIR%"
del "%SCRIPT_DIR%\test-write.tmp"
echo [OK] Write permissions verified
)
:: All log writes use TARGET_DIR
set LOG_FILE=%TARGET_DIR%\install.log
Install-Mac.sh / Install-Linux.sh: Similar pattern with TARGET_DIR=$HOME/.nomadarch-install, BIN_DIR=$TARGET_DIR/bin
Verification: Run from read-only directory, output shows Using fallback for outputs: C:\Users\xxx\NomadArch-Install
C7: Health Check Path Corrections
Install-Windows.bat (health check section):
:: UI health check
- if exist "%SCRIPT_DIR%\packages\ui\src\renderer\dist" (
+ if exist "%SCRIPT_DIR%\packages\ui\dist\index.html" (
echo [OK] UI build directory exists
) else (
- echo [ERROR] UI build directory not found
+ echo [ERROR] UI build directory not found at packages\ui\dist
set /a HEALTH_ERRORS+=1
)
:: Electron health check
- if exist "%SCRIPT_DIR%\packages\electron-app\dist\main.js" (
+ if exist "%SCRIPT_DIR%\packages\electron-app\dist\main\main.js" (
echo [OK] Electron main.js exists
) else (
echo [WARN] Electron build not found (will build on launch)
)
Install-Mac.sh / Install-Linux.sh: Same logic with shell syntax
Verification: Health check output:
[OK] UI build directory exists
[OK] Electron main.js exists
C8: Launch-Dev-Windows Electron Path Fix
Launch-Dev-Windows.bat line 162:
- if not exist "electron-app\dist\main.js" (
+ if not exist "packages\electron-app\dist\main\main.js" (
Verification: grep -n "electron-app" Launch-Dev-Windows.bat shows no electron-app\ references remaining
Execution Order
- C6 (TARGET_DIR/BIN_DIR) - Foundation for C4
- C7 (Health checks) - Independent path fixes
- C1 (UI paths) - Quick path replacements
- C8 (Launch-Dev-Windows) - Quick path fix
- C2 (Vite port) - Includes new file creation
- C3 (Server port) - Quick env var changes
- C4 (OpenCode download) - Depends on C6, includes C5
- Run build for C1/C7 verification
Verification Commands to Run
| Fix | Command | Expected Output |
|---|---|---|
| C1 | dir packages\ui\dist |
Shows index.html, assets/ |
| C2 | Run Launch-Dev, check Vite log | Local: http://localhost:3001 |
| C3 | Run launcher, check server log | CodeNomad Server is ready at http://127.0.0.1:3001 |
| C4 | Run install, grep log | OpenCode v<x.y.z> downloaded, checksum verified |
| C5 | Run install, check log | Hashes match in output |
| C6 | Run from read-only dir | Using fallback: C:\Users\xxx\NomadArch-Install |
| C7 | Run install, check output | UI build directory exists + Electron main.js exists |
| C8 | grep -n "electron-app" Launch-Dev-Windows.bat |
Only packages\electron-app or commented lines |
Files Modified/Created
| File | Action |
|---|---|
| Install-Windows.bat | Edit (C1, C4, C5, C6, C7) |
| Install-Mac.sh | Edit (C1, C4, C6, C7) |
| Install-Linux.sh | Edit (C1, C4, C6, C7) |
| Launch-Windows.bat | Edit (C1, C3) |
| Launch-Dev-Windows.bat | Edit (C1, C2, C3, C8) |
| Launch-Unix.sh | Edit (C1, C3) |
| Launch-Dev-Unix.sh | CREATE (C2) |
| packages/ui/vite.config.ts | Edit (C2) |