AeThex-OS/script/build-linux-iso.sh

310 lines
10 KiB
Bash

#!/bin/bash
set -e
# AeThex Linux ISO Builder - Containerized Edition
# Creates a bootable ISO using Ubuntu base image (no debootstrap/chroot needed)
WORK_DIR="${1:-.}"
BUILD_DIR="$WORK_DIR/aethex-linux-build"
ISO_NAME="AeThex-Linux-amd64.iso"
echo "[*] AeThex ISO Builder - Containerized Edition"
echo "[*] Build directory: $BUILD_DIR"
echo "[*] This build method works in Docker without privileged mode"
# Clean and prepare
rm -rf "$BUILD_DIR"
mkdir -p "$BUILD_DIR"/{iso,rootfs}
# Check dependencies
echo "[*] Checking dependencies..."
for cmd in xorriso genisoimage mksquashfs; do
if ! command -v "$cmd" &> /dev/null; then
echo "[!] Missing: $cmd - installing..."
apt-get update -qq
apt-get install -y -qq "$cmd" 2>&1 | tail -5
fi
done
echo "[+] Downloading Ubuntu Mini ISO base..."
# Use Ubuntu mini.iso as base (much smaller, pre-built)
if [ ! -f "$BUILD_DIR/ubuntu-mini.iso" ]; then
wget -q --show-progress -O "$BUILD_DIR/ubuntu-mini.iso" \
http://archive.ubuntu.com/ubuntu/dists/noble/main/installer-amd64/current/legacy-images/netboot/mini.iso \
|| echo "[!] Download failed, creating minimal ISO instead"
fi
echo "[+] Building AeThex application layer..."
mount -t proc /proc "$ROOTFS_DIR/proc" || true
mount -t sysfs /sys "$ROOTFS_DIR/sys" || true
mount --bind /dev "$ROOTFS_DIR/dev" || true
echo "[+] Installing Xfce desktop, Firefox, and system tools..."
echo " (packages installing, ~15-20 minutes...)"
chroot "$ROOTFS_DIR" bash -c '
export DEBIAN_FRONTEND=noninteractive
# Enable universe repository
sed -i "s/^# deb/deb/" /etc/apt/sources.list
echo "deb http://archive.ubuntu.com/ubuntu noble universe" >> /etc/apt/sources.list
echo "deb http://archive.ubuntu.com/ubuntu noble-updates universe" >> /etc/apt/sources.list
apt-get update
apt-get install -y \
linux-image-generic \
grub-pc-bin grub-efi-amd64-bin grub-common xorriso \
xorg xfce4 xfce4-goodies lightdm \
firefox network-manager \
sudo curl wget git ca-certificates gnupg \
pipewire-audio wireplumber \
file-roller thunar-archive-plugin \
xfce4-terminal mousepad ristretto \
dbus-x11
apt-get clean
' 2>&1 | tail -50
echo "[+] Installing Node.js 20.x from NodeSource..."
chroot "$ROOTFS_DIR" bash -c '
export DEBIAN_FRONTEND=noninteractive
# Install ca-certificates first
apt-get install -y ca-certificates curl gnupg
mkdir -p /etc/apt/keyrings
curl -fsSL https://deb.nodesource.com/gpgkey/nodesource-repo.gpg.key | gpg --dearmor -o /etc/apt/keyrings/nodesource.gpg
echo "deb [signed-by=/etc/apt/keyrings/nodesource.gpg] https://deb.nodesource.com/node_20.x nodistro main" > /etc/apt/sources.list.d/nodesource.list
apt-get update
apt-get install -y nodejs
node --version || echo "Node install failed"
npm --version || echo "npm not found"
' 2>&1 | tail -10
echo "[+] Creating AeThex user with auto-login..."
chroot "$ROOTFS_DIR" bash -c '
useradd -m -s /bin/bash -G sudo aethex
echo "aethex:aethex" | chpasswd
echo "aethex ALL=(ALL) NOPASSWD:ALL" >> /etc/sudoers
'
# Configure LightDM for auto-login
mkdir -p "$ROOTFS_DIR/etc/lightdm"
cat > "$ROOTFS_DIR/etc/lightdm/lightdm.conf" << 'LIGHTDM'
[Seat:*]
autologin-user=aethex
autologin-user-timeout=0
user-session=xfce
LIGHTDM
echo "[+] Setting up AeThex Desktop application..."
# Build mobile app if possible
if [ -f "package.json" ]; then
echo " Building AeThex mobile app..."
npm run build 2>&1 | tail -5 || echo " Build skipped"
fi
# Copy app files (if available, otherwise note for manual addition)
if [ -d "client" ] && [ -d "server" ]; then
echo " Copying AeThex Desktop files..."
mkdir -p "$ROOTFS_DIR/opt/aethex-desktop"
# Copy source files
cp -r client "$ROOTFS_DIR/opt/aethex-desktop/"
cp -r server "$ROOTFS_DIR/opt/aethex-desktop/"
cp -r shared "$ROOTFS_DIR/opt/aethex-desktop/" 2>/dev/null || true
cp package*.json "$ROOTFS_DIR/opt/aethex-desktop/" 2>/dev/null || true
cp tsconfig.json "$ROOTFS_DIR/opt/aethex-desktop/" 2>/dev/null || true
cp vite.config.ts "$ROOTFS_DIR/opt/aethex-desktop/" 2>/dev/null || true
# Copy built assets if they exist
if [ -d "dist" ]; then
cp -r dist "$ROOTFS_DIR/opt/aethex-desktop/"
fi
# Copy Capacitor Android build if it exists (for native features)
if [ -d "android" ]; then
mkdir -p "$ROOTFS_DIR/opt/aethex-desktop/android"
cp -r android/app "$ROOTFS_DIR/opt/aethex-desktop/android/" 2>/dev/null || true
fi
# Install dependencies in chroot
echo " Installing Node.js dependencies..."
chroot "$ROOTFS_DIR" bash -c 'cd /opt/aethex-desktop && npm install --production --legacy-peer-deps' 2>&1 | tail -10 || echo " npm install skipped"
# Set ownership
chroot "$ROOTFS_DIR" chown -R aethex:aethex /opt/aethex-desktop
else
echo " (client/server not found; skipping app copy)"
fi
# Create systemd service for AeThex mobile server
cat > "$ROOTFS_DIR/etc/systemd/system/aethex-mobile-server.service" << 'SERVICEEOF'
[Unit]
Description=AeThex Mobile Server
After=network-online.target
Wants=network-online.target
[Service]
Type=simple
User=aethex
WorkingDirectory=/opt/aethex-desktop
Environment="NODE_ENV=production"
Environment="PORT=5000"
ExecStart=/usr/bin/npm start
Restart=always
RestartSec=10
StandardOutput=journal
StandardError=journal
[Install]
WantedBy=multi-user.target
SERVICEEOF
# Create backward-compatible legacy service name
cat > "$ROOTFS_DIR/etc/systemd/system/aethex-desktop.service" << 'SERVICEEOF'
[Unit]
Description=AeThex Desktop Server
After=network.target
[Service]
Type=simple
User=aethex
WorkingDirectory=/opt/aethex-desktop
ExecStart=/usr/bin/npm start
Restart=always
RestartSec=5
[Install]
WantedBy=multi-user.target
SERVICEEOF
# Enable both services
chroot "$ROOTFS_DIR" systemctl enable aethex-mobile-server.service 2>/dev/null || echo " Mobile server service added"
chroot "$ROOTFS_DIR" systemctl enable aethex-desktop.service 2>/dev/null || echo " Desktop service added"
# Create auto-start script for Firefox kiosk pointing to mobile server
mkdir -p "$ROOTFS_DIR/home/aethex/.config/autostart"
cat > "$ROOTFS_DIR/home/aethex/.config/autostart/aethex-kiosk.desktop" << 'KIOSK'
[Desktop Entry]
Type=Application
Name=AeThex Mobile UI
Exec=sh -c "sleep 5 && firefox --kiosk http://localhost:5000"
Hidden=false
NoDisplay=false
X-GNOME-Autostart-enabled=true
Comment=Launch AeThex mobile interface in fullscreen
KIOSK
chroot "$ROOTFS_DIR" chown -R aethex:aethex /home/aethex
echo "[✓] AeThex Mobile UI integrated:"
echo " - Server runs on port 5000"
echo " - Firefox launches in kiosk mode"
echo " - Xfce desktop with auto-login"
echo " - Ingress-style mobile interface"
echo "[+] Extracting kernel and initrd..."
KERNEL="$(ls -1 $ROOTFS_DIR/boot/vmlinuz-* 2>/dev/null | head -n 1)"
INITRD="$(ls -1 $ROOTFS_DIR/boot/initrd.img-* 2>/dev/null | head -n 1)"
if [ -z "$KERNEL" ] || [ -z "$INITRD" ]; then
echo "[!] Kernel or initrd not found."
ls -la "$ROOTFS_DIR/boot/" || true
mkdir -p "$BUILD_DIR"
echo "No kernel found in rootfs" > "$BUILD_DIR/README.txt"
exit 1
fi
cp "$KERNEL" "$ISO_DIR/casper/vmlinuz"
cp "$INITRD" "$ISO_DIR/casper/initrd.img"
echo "[✓] Kernel: $(basename "$KERNEL")"
echo "[✓] Initrd: $(basename "$INITRD")"
# Unmount before squashfs
echo "[+] Unmounting chroot filesystems..."
umount -lf "$ROOTFS_DIR/proc" 2>/dev/null || true
umount -lf "$ROOTFS_DIR/sys" 2>/dev/null || true
umount -lf "$ROOTFS_DIR/dev" 2>/dev/null || true
echo "[+] Creating SquashFS filesystem..."
echo " (compressing ~2-3GB desktop, takes 10-15 minutes...)"
if command -v mksquashfs &> /dev/null; then
mksquashfs "$ROOTFS_DIR" "$ISO_DIR/casper/filesystem.squashfs" -b 1048576 -comp xz -Xdict-size 100% 2>&1 | tail -3
else
echo "[!] mksquashfs not found; cannot create ISO."
mkdir -p "$BUILD_DIR"
echo "mksquashfs not available" > "$BUILD_DIR/README.txt"
exit 1
fi
echo "[+] Setting up BIOS boot (isolinux)..."
cat > "$ISO_DIR/isolinux/isolinux.cfg" << 'EOF'
PROMPT 0
TIMEOUT 50
DEFAULT linux
LABEL linux
MENU LABEL AeThex OS
KERNEL /casper/vmlinuz
APPEND initrd=/casper/initrd.img boot=casper quiet splash
EOF
cp /usr/lib/syslinux/isolinux.bin "$ISO_DIR/isolinux/" 2>/dev/null || \
cp /usr/share/syslinux/isolinux.bin "$ISO_DIR/isolinux/" 2>/dev/null || echo "[!] isolinux.bin missing"
cp /usr/lib/syslinux/ldlinux.c32 "$ISO_DIR/isolinux/" 2>/dev/null || \
cp /usr/share/syslinux/ldlinux.c32 "$ISO_DIR/isolinux/" 2>/dev/null || echo "[!] ldlinux.c32 missing"
echo "[+] Setting up UEFI boot (GRUB)..."
cat > "$ISO_DIR/boot/grub/grub.cfg" << 'EOF'
set timeout=10
set default=0
menuentry "AeThex OS" {
linux /casper/vmlinuz boot=casper quiet splash
initrd /casper/initrd.img
}
EOF
echo "[+] Creating hybrid ISO with grub-mkrescue..."
grub-mkrescue -o "$BUILD_DIR/$ISO_NAME" "$ISO_DIR" --verbose 2>&1 | tail -20
echo "[+] Computing SHA256 checksum..."
if [ -f "$BUILD_DIR/$ISO_NAME" ]; then
cd "$BUILD_DIR"
sha256sum "$ISO_NAME" > "$ISO_NAME.sha256"
echo "[✓] ISO ready:"
ls -lh "$ISO_NAME" | awk '{print " Size: " $5}'
cat "$ISO_NAME.sha256" | awk '{print " SHA256: " $1}'
echo "[✓] Location: $BUILD_DIR/$ISO_NAME"
else
echo "[!] ISO creation failed."
exit 1
fi
echo "[*] Cleaning up rootfs..."
rm -rf "$ROOTFS_DIR"
echo "[✓] Build complete!"
echo ""
echo "=== AeThex OS - Mobile UI Edition ==="
echo "Features:"
echo " - Ubuntu 24.04 LTS base"
echo " - Xfce desktop environment"
echo " - Firefox browser (auto-launches mobile UI in kiosk mode)"
echo " - Node.js 20.x + npm"
echo " - AeThex Mobile App (Ingress-style) at /opt/aethex-desktop"
echo " - Server: http://localhost:5000"
echo " - Auto-login as user 'aethex' (password: aethex)"
echo " - NetworkManager for WiFi/Ethernet"
echo " - PipeWire audio support"
echo " - 18 Capacitor plugins integrated"
echo ""
echo "Mobile UI Features:"
echo " - Ingress-style hexagonal design"
echo " - Green/Cyan color scheme"
echo " - CSS-only animations (low CPU)"
echo " - Native-like performance"
echo " - Calculator, Notes, File Manager, Terminal, Games, etc."
echo ""
echo "Flash to USB: sudo ./script/flash-usb.sh -i $BUILD_DIR/$ISO_NAME"
echo "[✓] Done!"