#!/usr/bin/env bash # Build a Furtka live ISO. # # From the repo root or from iso/ on any host with Docker: # ./iso/build.sh # # The build runs inside a privileged `archlinux:latest` container because # mkarchiso needs root + loop mounts + an Arch package manager, which # Ubuntu doesn't provide natively. Output ISO goes to iso/out/. set -euo pipefail SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" REPO_ROOT="$(cd "$SCRIPT_DIR/.." && pwd)" OUT_DIR="$SCRIPT_DIR/out" if [[ "${FURTKA_ISO_INNER:-0}" != "1" ]]; then mkdir -p "$OUT_DIR" echo "==> Launching build container" exec docker run --rm --privileged \ -v "$REPO_ROOT:/work" \ -w /work \ -e FURTKA_ISO_INNER=1 \ archlinux:latest \ bash /work/iso/build.sh fi # ---- inside the container from here on ---- echo "==> Syncing pacman, installing archiso" pacman -Syu --noconfirm --needed archiso PROFILE_SRC="/usr/share/archiso/configs/releng" PROFILE_WORK="/tmp/furtka-profile" BUILD_WORK="/tmp/furtka-build" OUT_IN_CONTAINER="/work/iso/out" rm -rf "$PROFILE_WORK" "$BUILD_WORK" cp -a "$PROFILE_SRC" "$PROFILE_WORK" echo "==> Overlaying Furtka customizations" cat "$SCRIPT_DIR/overlay/packages.extra" >> "$PROFILE_WORK/packages.x86_64" cat "$SCRIPT_DIR/overlay/profiledef.sh" >> "$PROFILE_WORK/profiledef.sh" cp -a "$SCRIPT_DIR/overlay/airootfs/." "$PROFILE_WORK/airootfs/" echo "==> Rebranding boot menu (GRUB + syslinux + systemd-boot)" # releng ships menu entries labelled "Arch Linux install medium" across three # bootloader configs (BIOS syslinux, GRUB, systemd-boot for UEFI). Rewrite to # our brand. Done with sed (not a static overlay) so upstream archiso file # moves don't silently leave stale Arch labels behind. # # Also rebrands the syslinux menu header ("MENU TITLE Arch Linux") and the # per-entry HELP text shown at the bottom of the BIOS screen. GRUB/efiboot # don't ship equivalent long descriptions, so menu-entry rename is enough there. find "$PROFILE_WORK/grub" "$PROFILE_WORK/syslinux" "$PROFILE_WORK/efiboot" \ -type f \( -name "*.cfg" -o -name "*.conf" \) -print0 \ | xargs -0 sed -i \ -e 's/Arch Linux install medium/Furtka Live Installer/g' \ -e 's/Arch Linux live medium/Furtka Live Installer/g' \ -e 's/install Arch Linux or perform system maintenance/install Furtka or perform system maintenance/g' \ -e 's/^MENU TITLE Arch Linux$/MENU TITLE Furtka/' # Mark the default entry as (Recommended) so first-time users know which to # pick. Targets the main entry only — speech/accessibility variants stay # unlabeled to avoid suggesting they're the normal choice. sed -i 's/^title Furtka Live Installer (%ARCH%, UEFI)$/title (Recommended) Furtka Live Installer (%ARCH%, UEFI)/' \ "$PROFILE_WORK/efiboot/loader/entries/01-archiso-linux.conf" sed -i 's/^MENU LABEL Furtka Live Installer (%ARCH%, BIOS)$/MENU LABEL (Recommended) Furtka Live Installer (%ARCH%, BIOS)/' \ "$PROFILE_WORK/syslinux/archiso_sys-linux.cfg" sed -i "/--id 'archlinux'/s/menuentry \"Furtka Live Installer/menuentry \"(Recommended) Furtka Live Installer/" \ "$PROFILE_WORK/grub/grub.cfg" "$PROFILE_WORK/grub/loopback.cfg" mkdir -p "$PROFILE_WORK/airootfs/opt/furtka" cp -a "$REPO_ROOT/webinstaller/." "$PROFILE_WORK/airootfs/opt/furtka/" # Ship the post-install asset tree (HTML, CSS, systemd units, scripts, …) # next to webinstaller/app.py so _resolve_assets_dir() finds it at runtime. cp -a "$REPO_ROOT/assets" "$PROFILE_WORK/airootfs/opt/furtka/assets" rm -rf "$PROFILE_WORK/airootfs/opt/furtka/__pycache__" # Pack the resource manager (furtka/ Python package + bundled apps/) as a # tarball that webinstaller hands to archinstall via custom_commands. Lives at # a fixed path in the live ISO; the installed system reads it back, untars # into /opt/furtka/versions//, and gets a working `furtka` CLI + the # fileshare app. Same tarball shape as Phase-2 self-update releases, so an # ISO-installed box and an updated box converge on the same layout. echo "==> Bundling resource manager payload" PAYLOAD_STAGE="$(mktemp -d)" cp -a "$REPO_ROOT/furtka" "$PAYLOAD_STAGE/" cp -a "$REPO_ROOT/apps" "$PAYLOAD_STAGE/" # assets/ ships at the tarball root (not inside the Python package) because # Caddy, systemd, and the updater all expect it at /opt/furtka/current/assets/. cp -a "$REPO_ROOT/assets" "$PAYLOAD_STAGE/" find "$PAYLOAD_STAGE" -type d -name __pycache__ -exec rm -rf {} + # VERSION at tarball root: the installer reads it to choose the versions// # directory name and /opt/furtka/current/VERSION reports it at runtime. grep -E '^version = ' "$REPO_ROOT/pyproject.toml" | head -1 \ | sed 's/.*= "\(.*\)"/\1/' > "$PAYLOAD_STAGE/VERSION" tar -czf "$PROFILE_WORK/airootfs/opt/furtka-resource-manager.tar.gz" \ -C "$PAYLOAD_STAGE" . rm -rf "$PAYLOAD_STAGE" mkdir -p "$PROFILE_WORK/airootfs/etc/systemd/system/avahi-daemon.service.d" ln -sf /usr/lib/systemd/system/avahi-daemon.service \ "$PROFILE_WORK/airootfs/etc/systemd/system/multi-user.target.wants/avahi-daemon.service" echo "==> Building ISO (mkarchiso)" mkdir -p "$OUT_IN_CONTAINER" mkarchiso -v -w "$BUILD_WORK" -o "$OUT_IN_CONTAINER" "$PROFILE_WORK" echo echo "==> Done. ISO(s) in $OUT_IN_CONTAINER (on host: iso/out/):" ls -lh "$OUT_IN_CONTAINER"