Some checks failed
Release Binaries / release (push) Has been cancelled
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)
360 lines
11 KiB
Bash
360 lines
11 KiB
Bash
#!/bin/bash
|
|
|
|
# NomadArch Installer for Linux
|
|
# Version: 0.5.0 - Binary-Free Mode
|
|
|
|
set -euo pipefail
|
|
|
|
RED='\033[0;31m'
|
|
GREEN='\033[0;32m'
|
|
YELLOW='\033[1;33m'
|
|
BLUE='\033[0;34m'
|
|
NC='\033[0m'
|
|
|
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
TARGET_DIR="$SCRIPT_DIR"
|
|
BIN_DIR="$TARGET_DIR/bin"
|
|
LOG_FILE="$TARGET_DIR/install.log"
|
|
ERRORS=0
|
|
WARNINGS=0
|
|
NEEDS_FALLBACK=0
|
|
BINARY_FREE_MODE=0
|
|
|
|
log() {
|
|
echo "[$(date '+%Y-%m-%d %H:%M:%S')] $1" >> "$LOG_FILE"
|
|
}
|
|
|
|
echo ""
|
|
echo "NomadArch Installer (Linux)"
|
|
echo "Version: 0.5.0 - Binary-Free Mode"
|
|
echo ""
|
|
|
|
log "Installer started"
|
|
|
|
echo "[STEP 1/8] OS and Architecture Detection"
|
|
OS_TYPE=$(uname -s)
|
|
ARCH_TYPE=$(uname -m)
|
|
log "OS: $OS_TYPE"
|
|
log "Architecture: $ARCH_TYPE"
|
|
|
|
if [[ "$OS_TYPE" != "Linux" ]]; then
|
|
echo -e "${RED}[ERROR]${NC} This installer is for Linux. Current OS: $OS_TYPE"
|
|
log "ERROR: Not Linux ($OS_TYPE)"
|
|
exit 1
|
|
fi
|
|
|
|
case "$ARCH_TYPE" in
|
|
x86_64) ARCH="x64" ;;
|
|
aarch64) ARCH="arm64" ;;
|
|
armv7l) ARCH="arm" ;;
|
|
*)
|
|
echo -e "${RED}[ERROR]${NC} Unsupported architecture: $ARCH_TYPE"
|
|
log "ERROR: Unsupported arch $ARCH_TYPE"
|
|
exit 1
|
|
;;
|
|
esac
|
|
|
|
echo -e "${GREEN}[OK]${NC} OS: Linux"
|
|
echo -e "${GREEN}[OK]${NC} Architecture: $ARCH_TYPE"
|
|
|
|
if [[ -f /etc/os-release ]]; then
|
|
# shellcheck disable=SC1091
|
|
. /etc/os-release
|
|
echo -e "${GREEN}[INFO]${NC} Distribution: ${PRETTY_NAME:-unknown}"
|
|
fi
|
|
|
|
echo ""
|
|
echo "[STEP 2/8] Checking write permissions"
|
|
mkdir -p "$BIN_DIR"
|
|
if ! touch "$SCRIPT_DIR/.install-write-test" 2>/dev/null; then
|
|
echo -e "${YELLOW}[WARN]${NC} No write access to $SCRIPT_DIR"
|
|
TARGET_DIR="$HOME/.nomadarch-install"
|
|
BIN_DIR="$TARGET_DIR/bin"
|
|
LOG_FILE="$TARGET_DIR/install.log"
|
|
mkdir -p "$BIN_DIR"
|
|
if ! touch "$TARGET_DIR/.install-write-test" 2>/dev/null; then
|
|
echo -e "${RED}[ERROR]${NC} Cannot write to $TARGET_DIR"
|
|
log "ERROR: Write permission denied to fallback"
|
|
exit 1
|
|
fi
|
|
rm -f "$TARGET_DIR/.install-write-test"
|
|
NEEDS_FALLBACK=1
|
|
echo -e "${GREEN}[OK]${NC} Using fallback: $TARGET_DIR"
|
|
else
|
|
rm -f "$SCRIPT_DIR/.install-write-test"
|
|
echo -e "${GREEN}[OK]${NC} Write access OK"
|
|
fi
|
|
|
|
log "Install target: $TARGET_DIR"
|
|
|
|
echo ""
|
|
echo "[STEP 3/8] Ensuring system dependencies"
|
|
|
|
SUDO=""
|
|
if [[ $EUID -ne 0 ]]; then
|
|
if command -v sudo >/dev/null 2>&1; then
|
|
SUDO="sudo"
|
|
else
|
|
echo -e "${RED}[ERROR]${NC} sudo is required to install dependencies"
|
|
log "ERROR: sudo not found"
|
|
exit 1
|
|
fi
|
|
fi
|
|
|
|
install_packages() {
|
|
local manager="$1"
|
|
shift
|
|
local packages=("$@")
|
|
echo -e "${BLUE}[INFO]${NC} Installing via $manager: ${packages[*]}"
|
|
case "$manager" in
|
|
apt)
|
|
$SUDO apt-get update -y
|
|
$SUDO apt-get install -y "${packages[@]}"
|
|
;;
|
|
dnf)
|
|
$SUDO dnf install -y "${packages[@]}"
|
|
;;
|
|
yum)
|
|
$SUDO yum install -y "${packages[@]}"
|
|
;;
|
|
pacman)
|
|
$SUDO pacman -Sy --noconfirm "${packages[@]}"
|
|
;;
|
|
zypper)
|
|
$SUDO zypper -n install "${packages[@]}"
|
|
;;
|
|
apk)
|
|
$SUDO apk add --no-cache "${packages[@]}"
|
|
;;
|
|
*)
|
|
return 1
|
|
;;
|
|
esac
|
|
}
|
|
|
|
PACKAGE_MANAGER=""
|
|
if command -v apt-get >/dev/null 2>&1; then
|
|
PACKAGE_MANAGER="apt"
|
|
elif command -v dnf >/dev/null 2>&1; then
|
|
PACKAGE_MANAGER="dnf"
|
|
elif command -v yum >/dev/null 2>&1; then
|
|
PACKAGE_MANAGER="yum"
|
|
elif command -v pacman >/dev/null 2>&1; then
|
|
PACKAGE_MANAGER="pacman"
|
|
elif command -v zypper >/dev/null 2>&1; then
|
|
PACKAGE_MANAGER="zypper"
|
|
elif command -v apk >/dev/null 2>&1; then
|
|
PACKAGE_MANAGER="apk"
|
|
fi
|
|
|
|
if [[ -z "$PACKAGE_MANAGER" ]]; then
|
|
echo -e "${RED}[ERROR]${NC} No supported package manager found."
|
|
echo "Install Node.js, npm, git, and curl manually."
|
|
log "ERROR: No package manager found"
|
|
exit 1
|
|
fi
|
|
|
|
MISSING_PKGS=()
|
|
command -v curl >/dev/null 2>&1 || MISSING_PKGS+=("curl")
|
|
command -v git >/dev/null 2>&1 || MISSING_PKGS+=("git")
|
|
|
|
if ! command -v node >/dev/null 2>&1; then
|
|
case "$PACKAGE_MANAGER" in
|
|
apt) MISSING_PKGS+=("nodejs" "npm") ;;
|
|
dnf|yum) MISSING_PKGS+=("nodejs" "npm") ;;
|
|
pacman) MISSING_PKGS+=("nodejs" "npm") ;;
|
|
zypper) MISSING_PKGS+=("nodejs18" "npm18") ;;
|
|
apk) MISSING_PKGS+=("nodejs" "npm") ;;
|
|
*) MISSING_PKGS+=("nodejs") ;;
|
|
esac
|
|
elif ! command -v npm >/dev/null 2>&1; then
|
|
MISSING_PKGS+=("npm")
|
|
fi
|
|
|
|
if [[ ${#MISSING_PKGS[@]} -gt 0 ]]; then
|
|
install_packages "$PACKAGE_MANAGER" "${MISSING_PKGS[@]}" || {
|
|
echo -e "${YELLOW}[WARN]${NC} Some packages failed to install. Trying alternative method..."
|
|
if ! command -v node >/dev/null 2>&1; then
|
|
install_packages "$PACKAGE_MANAGER" "nodejs" || true
|
|
fi
|
|
}
|
|
fi
|
|
|
|
if ! command -v node >/dev/null 2>&1; then
|
|
echo -e "${RED}[ERROR]${NC} Node.js install failed."
|
|
log "ERROR: Node.js still missing"
|
|
exit 1
|
|
fi
|
|
|
|
NODE_VERSION=$(node --version)
|
|
NODE_MAJOR=$(echo "$NODE_VERSION" | cut -d'v' -f2 | cut -d'.' -f1)
|
|
echo -e "${GREEN}[OK]${NC} Node.js: $NODE_VERSION"
|
|
if [[ $NODE_MAJOR -lt 18 ]]; then
|
|
echo -e "${YELLOW}[WARN]${NC} Node.js 18+ is recommended"
|
|
((WARNINGS++))
|
|
fi
|
|
|
|
if ! command -v npm >/dev/null 2>&1; then
|
|
echo -e "${RED}[ERROR]${NC} npm is not available"
|
|
log "ERROR: npm missing after install"
|
|
exit 1
|
|
fi
|
|
NPM_VERSION=$(npm --version)
|
|
echo -e "${GREEN}[OK]${NC} npm: $NPM_VERSION"
|
|
|
|
if command -v git >/dev/null 2>&1; then
|
|
echo -e "${GREEN}[OK]${NC} Git: $(git --version)"
|
|
else
|
|
echo -e "${YELLOW}[WARN]${NC} Git not found (optional)"
|
|
((WARNINGS++))
|
|
fi
|
|
|
|
echo ""
|
|
echo "[STEP 4/8] Installing npm dependencies"
|
|
cd "$SCRIPT_DIR"
|
|
log "Running npm install"
|
|
if ! npm install; then
|
|
echo -e "${RED}[ERROR]${NC} npm install failed"
|
|
log "ERROR: npm install failed"
|
|
exit 1
|
|
fi
|
|
|
|
echo -e "${GREEN}[OK]${NC} Dependencies installed"
|
|
|
|
echo ""
|
|
echo "[STEP 5/8] OpenCode Binary (OPTIONAL - Binary-Free Mode Available)"
|
|
echo -e "${BLUE}[INFO]${NC} NomadArch now supports Binary-Free Mode!"
|
|
echo -e "${BLUE}[INFO]${NC} You can use the application without OpenCode binary."
|
|
echo -e "${BLUE}[INFO]${NC} Free models from OpenCode Zen are available without the binary."
|
|
mkdir -p "$BIN_DIR"
|
|
|
|
echo ""
|
|
read -p "Skip OpenCode binary download? (Y for Binary-Free Mode / N to download) [Y]: " SKIP_CHOICE
|
|
SKIP_CHOICE="${SKIP_CHOICE:-Y}"
|
|
|
|
if [[ "${SKIP_CHOICE^^}" == "Y" ]]; then
|
|
BINARY_FREE_MODE=1
|
|
echo -e "${GREEN}[INFO]${NC} Skipping OpenCode binary - using Binary-Free Mode"
|
|
log "Using Binary-Free Mode"
|
|
else
|
|
OPENCODE_PINNED_VERSION="0.1.44"
|
|
OPENCODE_VERSION="$OPENCODE_PINNED_VERSION"
|
|
|
|
LATEST_VERSION=$(curl -s --max-time 10 https://api.github.com/repos/sst/opencode/releases/latest 2>/dev/null | grep '"tag_name"' | cut -d'"' -f4 | sed 's/^v//')
|
|
if [[ -n "$LATEST_VERSION" ]]; then
|
|
echo -e "${BLUE}[INFO]${NC} Latest available: v${LATEST_VERSION}, using pinned: v${OPENCODE_VERSION}"
|
|
fi
|
|
|
|
OPENCODE_BASE="https://github.com/sst/opencode/releases/download/v${OPENCODE_VERSION}"
|
|
OPENCODE_URL="${OPENCODE_BASE}/opencode-linux-${ARCH}"
|
|
CHECKSUM_URL="${OPENCODE_BASE}/checksums.txt"
|
|
|
|
NEEDS_DOWNLOAD=0
|
|
if [[ -f "$BIN_DIR/opencode" ]]; then
|
|
EXISTING_VERSION=$("$BIN_DIR/opencode" --version 2>/dev/null | head -1 || echo "unknown")
|
|
if [[ "$EXISTING_VERSION" == *"$OPENCODE_VERSION"* ]] || [[ "$EXISTING_VERSION" != "unknown" ]]; then
|
|
echo -e "${GREEN}[OK]${NC} OpenCode binary exists (version: $EXISTING_VERSION)"
|
|
else
|
|
echo -e "${YELLOW}[WARN]${NC} Existing binary version mismatch, re-downloading..."
|
|
NEEDS_DOWNLOAD=1
|
|
fi
|
|
else
|
|
NEEDS_DOWNLOAD=1
|
|
fi
|
|
|
|
if [[ $NEEDS_DOWNLOAD -eq 1 ]]; then
|
|
echo -e "${BLUE}[INFO]${NC} Downloading OpenCode v${OPENCODE_VERSION} for ${ARCH}..."
|
|
|
|
DOWNLOAD_SUCCESS=0
|
|
for attempt in 1 2 3; do
|
|
if curl -L --fail --retry 3 -o "$BIN_DIR/opencode.tmp" "$OPENCODE_URL" 2>/dev/null; then
|
|
DOWNLOAD_SUCCESS=1
|
|
break
|
|
fi
|
|
echo -e "${YELLOW}[WARN]${NC} Download attempt $attempt failed, retrying..."
|
|
sleep 2
|
|
done
|
|
|
|
if [[ $DOWNLOAD_SUCCESS -eq 0 ]]; then
|
|
echo -e "${YELLOW}[WARN]${NC} Failed to download OpenCode binary - using Binary-Free Mode"
|
|
BINARY_FREE_MODE=1
|
|
else
|
|
if curl -L --fail -o "$BIN_DIR/checksums.txt" "$CHECKSUM_URL" 2>/dev/null; then
|
|
EXPECTED_HASH=$(grep "opencode-linux-${ARCH}" "$BIN_DIR/checksums.txt" | awk '{print $1}')
|
|
ACTUAL_HASH=$(sha256sum "$BIN_DIR/opencode.tmp" | awk '{print $1}')
|
|
|
|
if [[ "$ACTUAL_HASH" == "$EXPECTED_HASH" ]]; then
|
|
echo -e "${GREEN}[OK]${NC} Checksum verified"
|
|
else
|
|
echo -e "${YELLOW}[WARN]${NC} Checksum mismatch (may be OK for some versions)"
|
|
fi
|
|
fi
|
|
|
|
mv "$BIN_DIR/opencode.tmp" "$BIN_DIR/opencode"
|
|
chmod +x "$BIN_DIR/opencode"
|
|
echo -e "${GREEN}[OK]${NC} OpenCode binary installed"
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
echo ""
|
|
echo "[STEP 6/8] Building UI assets"
|
|
if [[ -d "$SCRIPT_DIR/packages/ui/dist" ]]; then
|
|
echo -e "${GREEN}[OK]${NC} UI build already exists"
|
|
else
|
|
echo -e "${BLUE}[INFO]${NC} Building UI"
|
|
pushd "$SCRIPT_DIR/packages/ui" >/dev/null
|
|
npm run build
|
|
popd >/dev/null
|
|
echo -e "${GREEN}[OK]${NC} UI assets built"
|
|
fi
|
|
|
|
echo ""
|
|
echo "[STEP 7/8] Post-install health check"
|
|
HEALTH_ERRORS=0
|
|
|
|
[[ -f "$SCRIPT_DIR/package.json" ]] || HEALTH_ERRORS=$((HEALTH_ERRORS+1))
|
|
[[ -d "$SCRIPT_DIR/packages/ui" ]] || HEALTH_ERRORS=$((HEALTH_ERRORS+1))
|
|
[[ -d "$SCRIPT_DIR/packages/server" ]] || HEALTH_ERRORS=$((HEALTH_ERRORS+1))
|
|
[[ -f "$SCRIPT_DIR/packages/ui/dist/index.html" ]] || HEALTH_ERRORS=$((HEALTH_ERRORS+1))
|
|
|
|
if [[ $HEALTH_ERRORS -eq 0 ]]; then
|
|
echo -e "${GREEN}[OK]${NC} Health checks passed"
|
|
else
|
|
echo -e "${RED}[ERROR]${NC} Health checks failed ($HEALTH_ERRORS)"
|
|
ERRORS=$((ERRORS+HEALTH_ERRORS))
|
|
fi
|
|
|
|
echo ""
|
|
echo "[STEP 8/8] Installation Summary"
|
|
echo ""
|
|
echo " Install Dir: $TARGET_DIR"
|
|
echo " Architecture: $ARCH"
|
|
echo " Node.js: $NODE_VERSION"
|
|
echo " npm: $NPM_VERSION"
|
|
if [[ $BINARY_FREE_MODE -eq 1 ]]; then
|
|
echo " Mode: Binary-Free Mode (OpenCode Zen free models available)"
|
|
else
|
|
echo " Mode: Full Mode (OpenCode binary installed)"
|
|
fi
|
|
echo " Errors: $ERRORS"
|
|
echo " Warnings: $WARNINGS"
|
|
echo " Log File: $LOG_FILE"
|
|
echo ""
|
|
|
|
if [[ $ERRORS -gt 0 ]]; then
|
|
echo -e "${RED}[RESULT]${NC} Installation completed with errors"
|
|
echo "Review $LOG_FILE for details."
|
|
else
|
|
echo -e "${GREEN}[RESULT]${NC} Installation completed successfully"
|
|
echo "Run: ./Launch-Unix.sh"
|
|
echo ""
|
|
if [[ $BINARY_FREE_MODE -eq 1 ]]; then
|
|
echo -e "${BLUE}NOTE:${NC} Running in Binary-Free Mode."
|
|
echo " Free models (GPT-5 Nano, Grok Code, GLM-4.7, etc.) are available."
|
|
echo " You can also authenticate with Qwen for additional models."
|
|
fi
|
|
fi
|
|
|
|
exit $ERRORS
|