chore: Debian 12 → 13 (Trixie) migration, service hardening

- Update all references from Debian 12 (Bookworm) to Debian 13 (Trixie)
- Enable SystemCallArchitectures, RestrictAddressFamilies, RestrictRealtime
  in archipelago.service (safe on systemd 256+ which respects NoNewPrivileges=no)
- Update GLIBC compatibility checks from 2.36 to 2.40
- ISO filename, build container, and docs updated throughout

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Dorian 2026-04-09 21:32:08 +02:00
parent b8a09b448b
commit 2517379ac3
19 changed files with 41 additions and 43 deletions

View File

@ -50,8 +50,8 @@ jobs:
# Build in persistent repo dir to reuse target/ cache # Build in persistent repo dir to reuse target/ cache
cd "$HOME/archy" cd "$HOME/archy"
export GIT_HASH=$(git rev-parse --short HEAD) export GIT_HASH=$(git rev-parse --short HEAD)
# Static musl build — runner is Debian 13 (glibc 2.41) but ISO rootfs # Static musl build for portability — ensures binary runs regardless
# is Debian 12/bookworm (glibc 2.36). Dynamic binary won't run. # of glibc version differences between build host and ISO rootfs.
cargo build --release --target x86_64-unknown-linux-musl --manifest-path core/Cargo.toml cargo build --release --target x86_64-unknown-linux-musl --manifest-path core/Cargo.toml
# Copy binary to workspace for downstream steps # Copy binary to workspace for downstream steps
mkdir -p "$GITHUB_WORKSPACE/core/target/release" mkdir -p "$GITHUB_WORKSPACE/core/target/release"

View File

@ -43,7 +43,7 @@ DEV_SERVER=archipelago@192.168.1.228 ./build-auto-installer-iso.sh
## What the ISO Includes ## What the ISO Includes
✅ Complete Debian 12 root filesystem ✅ Complete Debian 13 root filesystem
✅ Pre-built Archipelago backend ✅ Pre-built Archipelago backend
✅ Pre-built frontend (web UI) ✅ Pre-built frontend (web UI)
**Prepackaged container images** (Bitcoin Knots, LND, UIs, and other bundled apps), loaded on first boot **Prepackaged container images** (Bitcoin Knots, LND, UIs, and other bundled apps), loaded on first boot

View File

@ -2,7 +2,7 @@
Archipelago is a **Bitcoin Node OS** — bootable, self-sovereign personal server. Flash to USB, install on hardware, manage via web UI. Archipelago is a **Bitcoin Node OS** — bootable, self-sovereign personal server. Flash to USB, install on hardware, manage via web UI.
**Stack**: Rust backend + Vue 3 + TypeScript (strict) + Vite 7 + Tailwind + Pinia + Podman on Debian 12 **Stack**: Rust backend + Vue 3 + TypeScript (strict) + Vite 7 + Tailwind + Pinia + Podman on Debian 13
**Version**: 1.3.0 | **Target**: x86_64 and ARM64 **Version**: 1.3.0 | **Target**: x86_64 and ARM64
## Beta Freeze (2026-03-18) ## Beta Freeze (2026-03-18)
@ -22,7 +22,7 @@ cd neode-ui && npm run build # Build (outputs to web/dist/neode-ui/)
## Architecture ## Architecture
``` ```
Debian 12 Debian 13
├── Podman (rootless, user archipelago) ├── Podman (rootless, user archipelago)
├── Nginx (80/443 → backend, app proxies) ├── Nginx (80/443 → backend, app proxies)
├── Rust Backend (core/) on 127.0.0.1:5678 ├── Rust Backend (core/) on 127.0.0.1:5678

View File

@ -28,7 +28,7 @@ npm test # Run tests
### Backend (Rust) ### Backend (Rust)
Build on a Linux server (Debian 12), **not** macOS: Build on a Linux server (Debian 13), **not** macOS:
```bash ```bash
cargo clippy --all-targets --all-features cargo clippy --all-targets --all-features

View File

@ -4,7 +4,7 @@
**Archipelago** is a bootable personal server OS. Flash it to a USB drive, install on any x86_64 or ARM64 machine, and manage Bitcoin infrastructure, self-hosted apps, and decentralized identity through a glassmorphism web UI. **Archipelago** is a bootable personal server OS. Flash it to a USB drive, install on any x86_64 or ARM64 machine, and manage Bitcoin infrastructure, self-hosted apps, and decentralized identity through a glassmorphism web UI.
[![Debian 12](https://img.shields.io/badge/Debian-12%20Bookworm-a80030)](https://www.debian.org/) [![Debian 13](https://img.shields.io/badge/Debian-13%20Trixie-a80030)](https://www.debian.org/)
[![License](https://img.shields.io/badge/license-MIT-green)](LICENSE) [![License](https://img.shields.io/badge/license-MIT-green)](LICENSE)
[![Rust](https://img.shields.io/badge/rust-stable-orange)](https://www.rust-lang.org/) [![Rust](https://img.shields.io/badge/rust-stable-orange)](https://www.rust-lang.org/)
[![Vue.js](https://img.shields.io/badge/vue.js-3.5-brightgreen)](https://vuejs.org/) [![Vue.js](https://img.shields.io/badge/vue.js-3.5-brightgreen)](https://vuejs.org/)
@ -81,7 +81,7 @@ Bitcoin (ThunderHub), Storage (FileBrowser, Immich, Nextcloud), Productivity (Pe
### Prerequisites ### Prerequisites
- macOS or Linux for frontend development - macOS or Linux for frontend development
- Linux dev server (Debian 12) for backend builds — **never build Rust on macOS for Linux** - Linux dev server (Debian 13) for backend builds — **never build Rust on macOS for Linux**
- Node.js 20+, Rust stable toolchain - Node.js 20+, Rust stable toolchain
### Frontend Development ### Frontend Development
@ -112,7 +112,7 @@ sudo ./build-auto-installer-iso.sh
## Architecture ## Architecture
``` ```
Debian 12 (Bookworm) Debian 13 (Trixie)
├── Rootless Podman (30 containers, archy-net DNS) ├── Rootless Podman (30 containers, archy-net DNS)
├── Nginx (reverse proxy, security headers, rate limiting) ├── Nginx (reverse proxy, security headers, rate limiting)
├── Rust Backend (JSON-RPC API on 127.0.0.1:5678) ├── Rust Backend (JSON-RPC API on 127.0.0.1:5678)

View File

@ -1,7 +1,7 @@
# Archipelago v1.0.0 Release Notes # Archipelago v1.0.0 Release Notes
**Release Date**: March 2026 **Release Date**: March 2026
**Target Platform**: Debian 12 (Bookworm) — x86_64 and ARM64 **Target Platform**: Debian 13 (Trixie) — x86_64 and ARM64
## What is Archipelago? ## What is Archipelago?

View File

@ -1058,8 +1058,8 @@ autopilot.active=false\n",
} }
/// Resolve the host gateway IP for --add-host flag. /// Resolve the host gateway IP for --add-host flag.
/// Podman 4.3.x (Debian 12) doesn't support "host-gateway" in rootless mode, /// Resolve the default gateway IP from the routing table for --add-host flag.
/// so we resolve the default gateway IP from the routing table. /// Explicit IP avoids issues with "host-gateway" in rootless Podman.
async fn resolve_host_gateway() -> String { async fn resolve_host_gateway() -> String {
// Try `ip route` to get the default gateway // Try `ip route` to get the default gateway
if let Ok(output) = tokio::process::Command::new("ip") if let Ok(output) = tokio::process::Command::new("ip")

View File

@ -156,7 +156,7 @@ These must be tested in order on a fresh install:
- [ ] Boot from USB on x86_64 hardware - [ ] Boot from USB on x86_64 hardware
- [ ] Auto-installer partitions disk correctly - [ ] Auto-installer partitions disk correctly
- [ ] Debian 12 installs without errors - [ ] Debian 13 installs without errors
- [ ] Archipelago services start on first boot - [ ] Archipelago services start on first boot
- [ ] Web UI accessible at server IP within 3 minutes of first boot - [ ] Web UI accessible at server IP within 3 minutes of first boot

View File

@ -3,7 +3,7 @@
> **Bitcoin Node OS** — Flash to USB, install on hardware, manage via web UI. > **Bitcoin Node OS** — Flash to USB, install on hardware, manage via web UI.
**Stack**: Rust backend + Vue 3 + TypeScript (strict) + Vite 7 + Tailwind CSS + Pinia + rootless Podman **Stack**: Rust backend + Vue 3 + TypeScript (strict) + Vite 7 + Tailwind CSS + Pinia + rootless Podman
**Target OS**: Debian 12 (Bookworm) — x86_64 and ARM64 **Target OS**: Debian 13 (Trixie) — x86_64 and ARM64
**Status**: Beta freeze (Phase 1: Feature Testing) **Status**: Beta freeze (Phase 1: Feature Testing)
For the full interactive architecture review with diagrams and learning guide, see [`architecture-review.html`](architecture-review.html). For the full interactive architecture review with diagrams and learning guide, see [`architecture-review.html`](architecture-review.html).

View File

@ -75,7 +75,7 @@ archy/
### Prerequisites ### Prerequisites
- **macOS** (development machine): Node.js 20+, npm - **macOS** (development machine): Node.js 20+, npm
- **Linux server** (`192.168.1.228`): Rust toolchain, Podman, Nginx, Debian 12 - **Linux server** (`192.168.1.228`): Rust toolchain, Podman, Nginx, Debian 13
- SSH key: `~/.ssh/archipelago-deploy` - SSH key: `~/.ssh/archipelago-deploy`
### Local Frontend Development ### Local Frontend Development

View File

@ -74,7 +74,7 @@ From your dev server (192.168.1.228):
## Current Status ## Current Status
**Latest Working ISO**: `archipelago-debian-12-x86_64.iso` (469M, built 18:28) **Latest Working ISO**: `archipelago-debian-13-x86_64.iso` (469M, built 18:28)
- This ISO was built earlier today - This ISO was built earlier today
- Contains the auto-installer - Contains the auto-installer
- **Should be tested** - might already have your live server state - **Should be tested** - might already have your live server state

View File

@ -35,7 +35,7 @@ See the Architecture documentation for detailed system information.
## What's Included ## What's Included
- **Debian Linux Base**: Stable Debian 12 (Bookworm) distribution - **Debian Linux Base**: Stable Debian 13 (Trixie) distribution
- **Podman**: Container runtime for apps (rootless by default) - **Podman**: Container runtime for apps (rootless by default)
- **Archipelago Backend**: Rust-based API server - **Archipelago Backend**: Rust-based API server
- **Archipelago Frontend**: Vue.js web interface - **Archipelago Frontend**: Vue.js web interface
@ -44,7 +44,7 @@ See the Architecture documentation for detailed system information.
## Build Output ## Build Output
- `results/archipelago-debian-12-x86_64.iso` - Bootable hybrid ISO image - `results/archipelago-debian-13-x86_64.iso` - Bootable hybrid ISO image
## Supported Platforms ## Supported Platforms

View File

@ -905,10 +905,9 @@ mkdir -p "$ARCH_DIR"
mkdir -p "$ARCH_DIR/bin" mkdir -p "$ARCH_DIR/bin"
mkdir -p "$ARCH_DIR/scripts" mkdir -p "$ARCH_DIR/scripts"
# netavark + aardvark-dns are installed in the rootfs via Dockerfile.rootfs (Debian 12 packages). # netavark + aardvark-dns are installed in the rootfs via Dockerfile.rootfs (Debian 13 packages).
# Do NOT copy from the build host — the host may run a newer glibc (e.g. Debian 13) # Do NOT copy from the build host — the host may run a different glibc version.
# and the resulting binary will fail on the Debian 12 target with GLIBC_2.39 not found. echo " netavark + aardvark-dns: included in rootfs (Debian 13 packages)"
echo " netavark + aardvark-dns: included in rootfs (Debian 12 packages)"
# Copy the pre-built rootfs # Copy the pre-built rootfs
echo " Including root filesystem..." echo " Including root filesystem..."
@ -967,9 +966,9 @@ BACKENDFILE
fi fi
# Extract NostrVPN binary from container image (native system service, not a container app) # Extract NostrVPN binary from container image (native system service, not a container app)
# NOTE: The container image must be built against Debian 12's GLIBC (2.36). # NOTE: The container image must be built against Debian 13's GLIBC (2.40).
# If built against a newer GLIBC (e.g. 2.39 from Ubuntu 24.10), the binary will fail # If built against a newer GLIBC, the binary will fail at runtime.
# at runtime with "GLIBC_2.39 not found". Rebuild with: FROM debian:12 AS builder # Rebuild with: FROM debian:13 AS builder
echo " Extracting NostrVPN binary..." echo " Extracting NostrVPN binary..."
_NVPN_IMG="${NOSTR_VPN_IMAGE:-80.71.235.15:3000/archipelago/nostr-vpn:v0.3.7}" _NVPN_IMG="${NOSTR_VPN_IMAGE:-80.71.235.15:3000/archipelago/nostr-vpn:v0.3.7}"
NVPN_IMAGE_ID="$($CONTAINER_CMD images -q "$_NVPN_IMG" 2>/dev/null)" NVPN_IMAGE_ID="$($CONTAINER_CMD images -q "$_NVPN_IMG" 2>/dev/null)"
@ -982,17 +981,17 @@ if [ -n "$NVPN_CONTAINER" ]; then
chmod +x "$ARCH_DIR/bin/nvpn" && \ chmod +x "$ARCH_DIR/bin/nvpn" && \
echo " ✅ NostrVPN binary extracted ($(du -h "$ARCH_DIR/bin/nvpn" | cut -f1))" echo " ✅ NostrVPN binary extracted ($(du -h "$ARCH_DIR/bin/nvpn" | cut -f1))"
$CONTAINER_CMD rm "$NVPN_CONTAINER" 2>/dev/null || true $CONTAINER_CMD rm "$NVPN_CONTAINER" 2>/dev/null || true
# Check GLIBC compatibility — Debian 12 has GLIBC 2.36 # Check GLIBC compatibility — Debian 13 (Trixie) has GLIBC 2.40
if [ -f "$ARCH_DIR/bin/nvpn" ]; then if [ -f "$ARCH_DIR/bin/nvpn" ]; then
NVPN_GLIBC=$(objdump -T "$ARCH_DIR/bin/nvpn" 2>/dev/null | grep -oP 'GLIBC_\K[0-9.]+' | sort -V | tail -1) NVPN_GLIBC=$(objdump -T "$ARCH_DIR/bin/nvpn" 2>/dev/null | grep -oP 'GLIBC_\K[0-9.]+' | sort -V | tail -1)
if [ -n "$NVPN_GLIBC" ]; then if [ -n "$NVPN_GLIBC" ]; then
# Compare: if required GLIBC > 2.36, warn # Compare: if required GLIBC > 2.40, warn
if printf '%s\n' "2.36" "$NVPN_GLIBC" | sort -V | tail -1 | grep -qv "^2\.36$"; then if printf '%s\n' "2.40" "$NVPN_GLIBC" | sort -V | tail -1 | grep -qv "^2\.40$"; then
echo " ⚠ WARNING: nvpn binary requires GLIBC $NVPN_GLIBC but Debian 12 has 2.36" echo " ⚠ WARNING: nvpn binary requires GLIBC $NVPN_GLIBC but Debian 13 has 2.40"
echo " ⚠ The nvpn daemon will fail at runtime. Rebuild the container against Debian 12." echo " ⚠ The nvpn daemon will fail at runtime. Rebuild the container against Debian 13."
echo " ⚠ VPN invite/status will still work via Rust backend config.toml fallback." echo " ⚠ VPN invite/status will still work via Rust backend config.toml fallback."
else else
echo " ✅ nvpn GLIBC compatibility OK (requires $NVPN_GLIBC, target has 2.36)" echo " ✅ nvpn GLIBC compatibility OK (requires $NVPN_GLIBC, target has 2.40)"
fi fi
fi fi
fi fi
@ -2048,7 +2047,7 @@ REGCONF
chown -R 1000:1000 /mnt/target/home/archipelago/.config chown -R 1000:1000 /mnt/target/home/archipelago/.config
# Configure podman to use netavark backend (enables container DNS on archy-net). # Configure podman to use netavark backend (enables container DNS on archy-net).
# netavark + aardvark-dns binaries come from the rootfs (Debian 12 apt packages). # netavark + aardvark-dns binaries come from the rootfs (Debian 13 apt packages).
if [ -f /mnt/target/usr/lib/podman/netavark ]; then if [ -f /mnt/target/usr/lib/podman/netavark ]; then
mkdir -p /mnt/target/home/archipelago/.config/containers mkdir -p /mnt/target/home/archipelago/.config/containers
cat > /mnt/target/home/archipelago/.config/containers/containers.conf <<'CONTAINERSCONF' cat > /mnt/target/home/archipelago/.config/containers/containers.conf <<'CONTAINERSCONF'

View File

@ -34,12 +34,11 @@ NoNewPrivileges=no
PrivateDevices=no PrivateDevices=no
SupplementaryGroups=dialout debian-tor SupplementaryGroups=dialout debian-tor
# Network, syscall, and realtime restrictions DISABLED on Debian 12: # Syscall and network restrictions — safe on Debian 13 (systemd 256+)
# RestrictAddressFamilies, SystemCallArchitectures, and RestrictRealtime all use # which respects NoNewPrivileges=no as an explicit override for seccomp filters
# seccomp filters that force no_new_privs=1 in the kernel (systemd 252). SystemCallArchitectures=native
# This blocks sudo, which is required for archipelago-wg (WireGuard peer management). RestrictAddressFamilies=AF_UNIX AF_INET AF_INET6 AF_NETLINK
# Debian 13+ (systemd 256) respects NoNewPrivileges=no as an override, but Debian 12 does not. RestrictRealtime=yes
# Re-enable these when dropping Debian 12 support.
# MemoryDenyWriteExecute removed: ring (rustls) and secp256k1 (bitcoin/nostr) # MemoryDenyWriteExecute removed: ring (rustls) and secp256k1 (bitcoin/nostr)
# use assembly code that requires executable memory mappings on some platforms # use assembly code that requires executable memory mappings on some platforms

View File

@ -20,7 +20,7 @@ if [ -z "$1" ]; then
fi fi
USB_DISK="$1" USB_DISK="$1"
ISO_FILE="$SCRIPT_DIR/results/archipelago-debian-12-x86_64.iso" ISO_FILE="$SCRIPT_DIR/results/archipelago-debian-13-x86_64.iso"
WORK_DIR="$SCRIPT_DIR/build/usb-extract" WORK_DIR="$SCRIPT_DIR/build/usb-extract"
if [ ! -f "$ISO_FILE" ]; then if [ ! -f "$ISO_FILE" ]; then

View File

@ -32,7 +32,7 @@ if [ "$USE_DOCKER" = true ]; then
-v "$PROJECT_ROOT:/workspace" \ -v "$PROJECT_ROOT:/workspace" \
-v "$OUTPUT_DIR:/output" \ -v "$OUTPUT_DIR:/output" \
-w /workspace/core/archipelago \ -w /workspace/core/archipelago \
rust:bookworm \ rust:trixie \
sh -c ' sh -c '
echo "📦 Installing build dependencies..." echo "📦 Installing build dependencies..."
apt-get update && apt-get install -y pkg-config libssl-dev apt-get update && apt-get install -y pkg-config libssl-dev

View File

@ -17,7 +17,7 @@ fi
USB_DISK="$1" USB_DISK="$1"
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)" SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
ISO_FILE="$SCRIPT_DIR/results/archipelago-debian-12-x86_64.iso" ISO_FILE="$SCRIPT_DIR/results/archipelago-debian-13-x86_64.iso"
if [ ! -f "$ISO_FILE" ]; then if [ ! -f "$ISO_FILE" ]; then
echo "❌ ISO not found: $ISO_FILE" echo "❌ ISO not found: $ISO_FILE"

View File

@ -126,7 +126,7 @@ else
fi fi
# ── NostrVPN: configure native system service with node identity ────── # ── NostrVPN: configure native system service with node identity ──────
# The nvpn binary may have GLIBC mismatch (built for newer glibc than Debian 12). # The nvpn binary may have GLIBC mismatch (built for newer glibc than target OS).
# Write config.toml directly as fallback — the Rust backend reads it for vpn.invite/status. # Write config.toml directly as fallback — the Rust backend reads it for vpn.invite/status.
NOSTR_SECRET=$(cat /var/lib/archipelago/identity/nostr_secret 2>/dev/null) NOSTR_SECRET=$(cat /var/lib/archipelago/identity/nostr_secret 2>/dev/null)
NOSTR_PUBKEY=$(cat /var/lib/archipelago/identity/nostr_pubkey 2>/dev/null) NOSTR_PUBKEY=$(cat /var/lib/archipelago/identity/nostr_pubkey 2>/dev/null)

View File

@ -273,7 +273,7 @@ PHASE_DETAILS=(
"/dev/sda (465.8G) — TOSHIBA MQ01ACF0" "/dev/sda (465.8G) — TOSHIBA MQ01ACF0"
"BIOS boot + EFI + root + data" "BIOS boot + EFI + root + data"
"FAT32, ext4, LUKS2" "FAT32, ext4, LUKS2"
"debootstrap → Debian 12 minimal" "debootstrap → Debian 13 minimal"
"AES-256-XTS (AES-NI detected)" "AES-256-XTS (AES-NI detected)"
"GRUB: BIOS + UEFI hybrid" "GRUB: BIOS + UEFI hybrid"
) )