Compare commits

..

31 Commits

Author SHA1 Message Date
c731c4c698 fix: use git add -f to force-add gitignored artifact dirs (dist/, bin/) 2026-03-15 22:55:20 +01:00
407884e044 fix: patch Vec3::toOf() glm branch to return glm::vec3 not ofVec3f 2026-03-15 22:43:29 +01:00
0baf1715ed fix: define OF_VERSION_MAJOR/MINOR so Vec3/Vec2 compile the glm branch 2026-03-15 22:29:33 +01:00
564ede7bd9 fix: rename .build -> _build to avoid OF addon source grep exclusion 2026-03-15 22:15:42 +01:00
d1c00820d3 fix: explicitly link bundled OF static libs (tess2, kiss) via USER_LDFLAGS 2026-03-15 22:11:55 +01:00
8dfc7669a5 fix: add libblas-dev:arm64 and liblapack-dev:arm64 for linker 2026-03-15 22:09:13 +01:00
4aa9b5a540 fix: include addon libs/ dir directly for flat header layout (tinyxml.h) 2026-03-15 22:04:33 +01:00
1644159194 fix: include addon-bundled libs/*/include dirs (tinyxml in ofxXmlSettings) 2026-03-15 22:01:36 +01:00
76bf9feb81 fix: include bundled OF addon src dirs (ofxXmlSettings etc) from addons.make 2026-03-15 21:57:07 +01:00
64c1f26000 fix: add ofxPiMapper src subdirs to include path for FboSource.h 2026-03-15 21:52:28 +01:00
538043b7f2 build: explicitly inject OF include paths to fix cross-compile find() failures 2026-03-15 21:47:30 +01:00
ff3fa766ae build: pass PKG_CONFIG_LIBDIR to make for cross-compile pkg-config checks 2026-03-15 21:25:59 +01:00
93e7bd58b2 build: add librtaudio-dev:arm64 for RtAudio.h cross-compile 2026-03-15 21:19:34 +01:00
863b480a3f fix: add GST_VERSION variable for aarch64 cross-compilation 2026-03-15 21:11:45 +01:00
1423f8d390 fix: use pkg-config wrapper script instead of exporting PKG_CONFIG_PATH (avoids OF Makefile bug) 2026-03-15 21:07:07 +01:00
1dd92750cf fix: use --force-overwrite for arm64 dev packages to bypass shared man page conflicts 2026-03-15 20:39:53 +01:00
9c8a37f59b fix: avoid arm64/amd64 -dev header conflicts, generate missing .pc files for zmq/psl 2026-03-15 20:29:55 +01:00
a35863b3c5 fix: add arm64 multiarch dev packages and PKG_CONFIG_PATH for OF cross-compile 2026-03-15 20:17:55 +01:00
fa595d655e fix: cross-compile ofxPiMapper via PLATFORM_ARCH=aarch64 make override (no uname fake needed) 2026-03-15 20:15:51 +01:00
242dfb267a fix: move ofxPiMapper native build to setup.sh (OF Makefile doesn't support cross-compile) 2026-03-15 18:02:52 +01:00
7617034320 fix: remove --omit=optional from frontend npm install (rollup native deps) 2026-03-15 17:56:53 +01:00
98c3a12bbe fix: build.sh now accepts REPO_URL arg, clones repo, and auto-pushes artifacts 2026-03-15 17:46:30 +01:00
722908e235 feat: implement setup and build scripts for MPVJ deployment and cross-compilation 2026-03-15 17:39:24 +01:00
f7897dfa83 fix: add fallback for hostname setting in setup script for non-systemd environments 2026-03-15 17:34:38 +01:00
093319b903 fix: improve systemd service setup check for active status in setup script 2026-03-15 17:32:41 +01:00
a1039259a2 fix: enhance systemd service setup for containerized environments and add cross-compile support 2026-03-15 17:29:32 +01:00
ea11fc95f2 fix: enhance user detection for non-login environments in setup script 2026-03-15 17:16:39 +01:00
2a4a3a15ac fix: install whiptail if not present for interactive prompts in setup script 2026-03-15 17:13:05 +01:00
82813baa58 fix: implement path-agnostic source patching and refine global compatibility header logic 2026-03-15 15:56:17 +01:00
705b698b1d fix: enforce -O0 optimization and VRP disable to bypass compiler crashes on Pi 3B 2026-03-15 09:16:35 +01:00
e0e06978b2 fix: enforce clean ofxPiMapper addon state and implement high-powered global compatibility header to resolve syntax and namespace errors 2026-03-15 09:04:01 +01:00
2 changed files with 412 additions and 211 deletions

332
scripts/build.sh Normal file
View File

@ -0,0 +1,332 @@
#!/bin/bash
# MPVJ Build Script
# Runs on x86_64 (e.g., Docker) to cross-compile artifacts for aarch64 (RPi 3B).
# Outputs (committed back to the repo):
# frontend/dist/ — pre-built Vite bundle
# bin/ofxPiMapper — aarch64 binary
#
# Usage:
# bash build.sh <REPO_URL> # clones repo into ./mapper, then builds
# bash build.sh <REPO_URL> <DIR> # clones into DIR, then builds
set -e
REPO_URL="${1:-}"
REPO_DIR="${2:-$(pwd)/mapper}"
OF_VERSION="0.12.1"
OF_DIR="$REPO_DIR/_build/openFrameworks"
OF_FILE="of_v${OF_VERSION}_linuxaarch64_release.tar.gz"
OF_URL="https://github.com/openframeworks/openFrameworks/releases/download/${OF_VERSION}/${OF_FILE}"
BIN_OUT="$REPO_DIR/bin"
if [ -z "$REPO_URL" ]; then
echo "Usage: bash $0 <REPO_URL> [CLONE_DIR]"
exit 1
fi
# ── Clone repo if needed ─────────────────────────────────────────────────────
if [ ! -d "$REPO_DIR/frontend" ] || [ ! -d "$REPO_DIR/backend" ]; then
echo "Cloning $REPO_URL into $REPO_DIR ..."
git clone "$REPO_URL" "$REPO_DIR"
fi
echo "Building from: $REPO_DIR"
# ── 1. Install build dependencies ───────────────────────────────────────────
echo "Installing build dependencies..."
if command -v apt-get > /dev/null 2>&1; then
# Enable arm64 packages alongside x86_64 (Debian multiarch)
dpkg --add-architecture arm64
apt-get update -y
# Cross-compiler toolchain (host)
DEBIAN_FRONTEND=noninteractive apt-get install -y \
git curl wget unzip build-essential pkg-config \
gcc-aarch64-linux-gnu g++-aarch64-linux-gnu \
binutils-aarch64-linux-gnu
# aarch64 dev libraries (provides the .pc files pkg-config needs)
# Note: some arm64 packages share man pages with their amd64 counterparts —
# --force-overwrite is safe here as we only care about headers/.so/.pc files.
DEBIAN_FRONTEND=noninteractive apt-get install -y \
-o Dpkg::Options::="--force-overwrite" \
libmpg123-dev:arm64 libsndfile1-dev:arm64 libopenal-dev:arm64 libassimp-dev:arm64 \
librtaudio-dev:arm64 \
libglew-dev:arm64 libglfw3-dev:arm64 liburiparser-dev:arm64 \
libcurl4-openssl-dev:arm64 libpugixml-dev:arm64 libasound2-dev:arm64 \
libgstreamer1.0-dev:arm64 libgstreamer-plugins-base1.0-dev:arm64 \
libgtk-3-dev:arm64 libboost-filesystem-dev:arm64 \
libfontconfig1-dev:arm64 libfreetype-dev:arm64 libx11-dev:arm64 \
libxrandr-dev:arm64 libxinerama-dev:arm64 libxcursor-dev:arm64 libxi-dev:arm64 \
libpulse-dev:arm64 libudev-dev:arm64 libfreeimage-dev:arm64 \
freeglut3-dev:arm64 libxxf86vm-dev:arm64 \
libcairo2-dev:arm64 \
libblas-dev:arm64 liblapack-dev:arm64
# These conflict with amd64 -dev headers; install runtime-only for .so + .pc
DEBIAN_FRONTEND=noninteractive apt-get install -y \
-o Dpkg::Options::="--force-overwrite" \
libzmq5:arm64 libpsl5:arm64 || true
# Repair any partially-broken state from conflicts
apt-get -f install -y
# Generate minimal .pc files for runtime-only arm64 packages that lack one
if [ ! -f /usr/lib/aarch64-linux-gnu/pkgconfig/libzmq.pc ]; then
ZMQ_VER=$(dpkg -l libzmq5:arm64 2>/dev/null | awk '/^ii/{print $3}' | cut -d- -f1 || echo "4.3.5")
mkdir -p /usr/lib/aarch64-linux-gnu/pkgconfig
cat > /usr/lib/aarch64-linux-gnu/pkgconfig/libzmq.pc <<EOF
prefix=/usr
exec_prefix=\${prefix}
libdir=\${prefix}/lib/aarch64-linux-gnu
includedir=\${prefix}/include
Name: libzmq
Description: ZeroMQ messaging library
Version: $ZMQ_VER
Libs: -L\${libdir} -lzmq
Cflags: -I\${includedir}
EOF
fi
fi
# Install Node.js if missing
if ! command -v node > /dev/null 2>&1; then
echo "Installing Node.js 20..."
curl -fsSL https://deb.nodesource.com/setup_20.x | bash -
apt-get install -y nodejs
fi
# ── 2. Build frontend ────────────────────────────────────────────────────────
echo ""
echo "=== Building frontend ==="
cd "$REPO_DIR/frontend"
rm -rf node_modules package-lock.json
npm install --package-lock=false
NODE_OPTIONS="--max-old-space-size=512" npm run build
echo "Frontend built → $REPO_DIR/frontend/dist/"
# ── 3. Download OpenFrameworks aarch64 release ──────────────────────────────
echo ""
echo "=== Downloading OpenFrameworks ${OF_VERSION} (aarch64) ==="
mkdir -p "$REPO_DIR/_build"
if [ ! -d "$OF_DIR" ]; then
cd "$REPO_DIR/_build"
[ ! -f "$OF_FILE" ] && (wget "$OF_URL" || curl -L -O "$OF_URL")
tar -xzf "$OF_FILE"
mv "of_v${OF_VERSION}_linux"*"_release" openFrameworks
rm -f "$OF_FILE"
echo "OpenFrameworks extracted to $OF_DIR"
else
echo "OpenFrameworks already present, skipping download."
fi
# ── 4. Clone & patch ofxPiMapper ────────────────────────────────────────────
echo ""
echo "=== Cloning & patching ofxPiMapper ==="
ADDON_DIR="$OF_DIR/addons/ofxPiMapper"
rm -rf "$ADDON_DIR"
git clone --depth 1 https://github.com/kr15h/ofxPiMapper.git "$ADDON_DIR"
cd "$ADDON_DIR"
# 4a. OscControl.cpp highlight patch
OSC_FILE=$(find src -name "OscControl.cpp" | head -n 1)
if [ -n "$OSC_FILE" ]; then
sed -i '/if (m.getAddress() == "\/ofxPiMapper\/surface\/select"){/r /dev/stdin' "$OSC_FILE" <<'EOF_PATCH'
if (m.getAddress() == "/ofxPiMapper/surface/highlight"){
int surfaceIndex = m.getArgAsInt32(0);
mapper->getSurfaceManager()->selectSurface(surfaceIndex);
return;
}
EOF_PATCH
fi
# 4b. C++11 compat header
cat <<'EOF_COMPAT' > src/cpp11_compat.h
#ifndef OFXPIMAPPER_CPP11_COMPAT_H
#define OFXPIMAPPER_CPP11_COMPAT_H
#include <memory>
#include <utility>
#include <vector>
#include <string>
#include <algorithm>
#if __cplusplus < 201402L
namespace std {
template<typename T, typename... Args>
std::unique_ptr<T> make_unique(Args&&... args) {
return std::unique_ptr<T>(new T(std::forward<Args>(args)...));
}
}
#endif
using std::unique_ptr;
using std::make_unique;
using std::vector;
using std::string;
#endif
EOF_COMPAT
sed -i '1i #include "cpp11_compat.h"' src/ofxPiMapper.h
find src -type f \( -name "*.h" -o -name "*.cpp" \) \
-exec sed -i 's/\([^:a-zA-Z0-9]\)unique_ptr\b/\1std::unique_ptr/g' {} + \
-exec sed -i 's/\([^:a-zA-Z0-9]\)make_unique\b/\1std::make_unique/g' {} +
for f in $(grep -l "unique_ptr" src/*.h 2>/dev/null || true); do
grep -q "<memory>" "$f" || sed -i '1i #include <memory>' "$f"
done
# 4c. Fix Vec3.cpp: in the #else (glm) branch, toOf() incorrectly returns
# ofVec3f instead of glm::vec3 — a bug in ofxPiMapper upstream.
sed -i '/^#else/,/^#endif/{
s/return ofVec3f(x, y, z);/return glm::vec3(x, y, z);/
}' src/Types/Vec3.cpp
# ── 5. Cross-compile ofxPiMapper for aarch64 ────────────────────────────────
# OF's Makefile supports PLATFORM_ARCH and CROSS_COMPILING as make variables —
# no need to fake uname. Passing them directly bypasses the uname -m check.
echo ""
echo "=== Cross-compiling ofxPiMapper (aarch64) ==="
if ! command -v aarch64-linux-gnu-g++ > /dev/null 2>&1; then
echo "ERROR: aarch64-linux-gnu-g++ not found. Install gcc-aarch64-linux-gnu."
exit 1
fi
cd "$ADDON_DIR/example_basic"
GLOBAL_FLAGS="-Wno-error -Wno-reorder -Wno-sign-compare -Wno-delete-non-virtual-dtor -std=c++11 -O0 -fno-tree-vrp -DOF_VERSION_MAJOR=0 -DOF_VERSION_MINOR=12"
# The OF Makefile has a bug: it calls `export $(PKG_CONFIG_PATH)` which passes the
# VALUE as a variable name under /bin/sh, causing "bad variable name" errors.
# Workaround: don't set PKG_CONFIG_PATH in the environment at all.
# Instead, create a thin wrapper script and pass it via OF's PKG_CONFIG make variable.
PKGCFG_WRAPPER="/usr/local/bin/aarch64-pkg-config"
cat > "$PKGCFG_WRAPPER" << 'PKGCFG_EOF'
#!/bin/sh
exec env \
PKG_CONFIG_PATH=/usr/lib/aarch64-linux-gnu/pkgconfig:/usr/share/pkgconfig \
PKG_CONFIG_LIBDIR=/usr/lib/aarch64-linux-gnu/pkgconfig:/usr/share/pkgconfig \
PKG_CONFIG_SYSROOT_DIR="" \
pkg-config "$@"
PKGCFG_EOF
chmod +x "$PKGCFG_WRAPPER"
# Ensure PKG_CONFIG_PATH is not set in our environment (avoid the OF Makefile
# "export $(PKG_CONFIG_PATH)" bug which treats its value as a variable name).
unset PKG_CONFIG_PATH
unset PKG_CONFIG_LIBDIR
ARM64_PKG_CONFIG_PATH="/usr/lib/aarch64-linux-gnu/pkgconfig:/usr/share/pkgconfig"
# OF's config.shared.mk uses $(shell find ...) to discover OF source subdirectory
# include paths and thirdparty lib headers. This can return empty in Docker
# cross-compile environments, causing "ofConstants.h: No such file or directory".
# Work around it by computing the include paths here and injecting them via
# PLATFORM_CFLAGS (which appears in the project compile command regardless).
OF_EXTRA_INCLUDES="-I$OF_DIR/libs/openFrameworks"
# OF source subdirectories (app, utils, math, gl, graphics, events, etc.)
for d in "$OF_DIR/libs/openFrameworks"/*/; do
[ -d "$d" ] && OF_EXTRA_INCLUDES="$OF_EXTRA_INCLUDES -I${d%/}"
done
# Thirdparty header-only libs bundled with OF (glm, json, kiss, tess2, utf8, ...)
for lib in "$OF_DIR/libs"/*/; do
libname=$(basename "$lib")
[ "$libname" = "openFrameworks" ] && continue
[ "$libname" = "openFrameworksCompiled" ] && continue
incdir="${lib}include"
[ -d "$incdir" ] || continue
while IFS= read -r subdir; do
OF_EXTRA_INCLUDES="$OF_EXTRA_INCLUDES -I$subdir"
done < <(find "$incdir" -type d)
done
# ofxPiMapper addon subdirectories (Sources/, Surfaces/, Textures/, etc.)
while IFS= read -r subdir; do
OF_EXTRA_INCLUDES="$OF_EXTRA_INCLUDES -I$subdir"
done < <(find "$ADDON_DIR/src" -type d)
# Bundled OF addons that ofxPiMapper depends on (e.g. ofxXmlSettings, ofxOsc, ...)
# Read the list from addons.make if present, otherwise include all bundled addon src dirs.
ADDONS_MAKE="$ADDON_DIR/example_basic/addons.make"
if [ -f "$ADDONS_MAKE" ]; then
while IFS= read -r addon_name; do
addon_name="${addon_name%%#*}" # strip comments
addon_name="$(echo "$addon_name" | tr -d '[:space:]')"
[ -z "$addon_name" ] && continue
addon_src="$OF_DIR/addons/$addon_name/src"
[ -d "$addon_src" ] || continue
while IFS= read -r subdir; do
OF_EXTRA_INCLUDES="$OF_EXTRA_INCLUDES -I$subdir"
done < <(find "$addon_src" -type d)
# Also include any libs/ bundled inside the addon (e.g. tinyxml in ofxXmlSettings)
addon_libs="$OF_DIR/addons/$addon_name/libs"
if [ -d "$addon_libs" ]; then
# Some addons (ofxXmlSettings) keep headers directly in libs/ (flat layout)
OF_EXTRA_INCLUDES="$OF_EXTRA_INCLUDES -I$addon_libs"
for alib in "$addon_libs"/*/; do
[ -d "$alib" ] || continue
# libs/<name>/include/ layout (e.g. other addons)
alib_inc="${alib}include"
if [ -d "$alib_inc" ]; then
while IFS= read -r subdir; do
OF_EXTRA_INCLUDES="$OF_EXTRA_INCLUDES -I$subdir"
done < <(find "$alib_inc" -type d)
else
# headers directly in libs/<name>/ (no include/ subdir)
OF_EXTRA_INCLUDES="$OF_EXTRA_INCLUDES -I${alib%/}"
fi
done
fi
done < "$ADDONS_MAKE"
fi
# Collect bundled static libs from OF's libs/*/lib/linuxaarch64/*.a
# The Makefile's auto-discovery skips these in cross-compile mode, so
# we add them explicitly via USER_LDFLAGS.
BUNDLED_LDFLAGS=""
for lib in "$OF_DIR/libs"/*/; do
libname=$(basename "$lib")
[ "$libname" = "openFrameworks" ] && continue
[ "$libname" = "openFrameworksCompiled" ] && continue
libdir="${lib}lib/linuxaarch64"
[ -d "$libdir" ] || continue
for a in "$libdir"/*.a; do
[ -f "$a" ] && BUNDLED_LDFLAGS="$BUNDLED_LDFLAGS $a"
done
done
make clean 2>/dev/null || true
make -j1 \
PKG_CONFIG="$PKGCFG_WRAPPER" \
PKG_CONFIG_LIBDIR="$ARM64_PKG_CONFIG_PATH" \
GST_VERSION=1.0 \
PLATFORM_ARCH=aarch64 \
CROSS_COMPILING=1 \
CC=aarch64-linux-gnu-gcc \
CXX=aarch64-linux-gnu-g++ \
AR=aarch64-linux-gnu-ar \
LD=aarch64-linux-gnu-ld \
PLATFORM_CFLAGS="$GLOBAL_FLAGS $OF_EXTRA_INCLUDES -I$ADDON_DIR/src" \
USER_CFLAGS="$GLOBAL_FLAGS" \
USER_CPPFLAGS="-std=c++11" \
USER_LDFLAGS="$BUNDLED_LDFLAGS" \
OF_PROJECT_OPTIMIZATION_FLAGS="-O0"
# ── 6. Collect artifacts ─────────────────────────────────────────────────────
echo ""
echo "=== Collecting artifacts ==="
mkdir -p "$BIN_OUT"
cp bin/example_basic "$BIN_OUT/ofxPiMapper"
chmod +x "$BIN_OUT/ofxPiMapper"
# ── 7. Commit and push artifacts ─────────────────────────────────────────────
echo ""
echo "=== Pushing artifacts to remote ==="
cd "$REPO_DIR"
git config user.email "build-bot@mpvj" 2>/dev/null || true
git config user.name "MPVJ Build Bot" 2>/dev/null || true
git add -f frontend/dist/ bin/ofxPiMapper
git commit -m "ci: add pre-built artifacts (frontend/dist + bin/ofxPiMapper)" || echo "(nothing new to commit)"
git push
echo ""
echo "============================================"
echo " BUILD COMPLETE + PUSHED"
echo "============================================"
echo " Frontend: $REPO_DIR/frontend/dist/"
echo " ofxPiMapper: $BIN_OUT/ofxPiMapper (aarch64)"
echo ""
echo " Run setup.sh on the Pi to deploy."
echo "============================================"

View File

@ -1,146 +1,114 @@
#!/bin/bash
# MPVJ Headless Setup Script (Final Polish & Robustness)
# Usage: curl -sSL <URL>/setup.sh | sudo bash -s -- <REPO_URL>
# MPVJ Deployment Script
# Runs on the Raspberry Pi. Expects pre-built artifacts in the repo:
# frontend/dist/ — built by scripts/build.sh
# bin/ofxPiMapper — aarch64 binary built by scripts/build.sh
#
# Usage: sudo bash setup.sh <REPO_URL>
set -e
REPO_URL=$1
if [ -z "$REPO_URL" ]; then
echo "Usage: $0 <REPO_URL>"
exit 1
echo "Usage: sudo bash $0 <REPO_URL>"
exit 1
fi
# 0. BASH & ROOT CHECK
if [ -z "$BASH_VERSION" ]; then
echo "ERROR: This script must be run with BASH, not SH."
echo "Please use: sudo bash setup.sh $REPO_URL"
echo "ERROR: This script must be run with bash."
exit 1
fi
if [ "$(id -u)" -ne 0 ]; then
echo "ERROR: This script must be run as root (use sudo)."
echo "ERROR: Must be run as root (sudo)."
exit 1
fi
# Detect user
REAL_USER=${SUDO_USER:-$(logname)}
REAL_HOME=$(eval echo "~$REAL_USER")
# Detect real user
REAL_USER=${SUDO_USER:-$(logname 2>/dev/null || true)}
[ -z "$REAL_USER" ] && REAL_USER=root
REAL_HOME=$(eval echo "~$REAL_USER" 2>/dev/null || echo "/root")
echo "Target User: $REAL_USER ($REAL_HOME)"
# 0.5 SCRUB & SANITIZE PHASE
echo "Scrubbing system for a clean installation..."
# Ensure whiptail is available
if ! command -v whiptail > /dev/null 2>&1; then
apt-get update -y
DEBIAN_FRONTEND=noninteractive apt-get install -y whiptail
fi
# ── 0. SCRUB ─────────────────────────────────────────────────────────────────
echo "Stopping any running services..."
systemctl stop mpvj-backend.service 2>/dev/null || true
pkill -9 ofxPiMapper 2>/dev/null || true
pkill -9 node 2>/dev/null || true
# Remove old setup artifacts
swapoff /swapfile_mpvj 2>/dev/null || true
rm -f /swapfile_mpvj 2>/dev/null || true
# Clear any stale apt locks
rm -f /var/lib/dpkg/lock-frontend /var/lib/apt/lists/lock /var/cache/apt/archives/lock 2>/dev/null || true
# Ask to wipe previous data for a fresh start
if [ -d "$REAL_HOME/mpvj" ] || [ -d "$REAL_HOME/openFrameworks" ]; then
if whiptail --title "Cleanup" --yesno "Found existing MPVJ or OpenFrameworks data. Wipe them for a clean install?" 10 60 < /dev/tty; then
echo "Wiping old directories..."
rm -rf "$REAL_HOME/mpvj" "$REAL_HOME/openFrameworks"
if [ -d "$REAL_HOME/mpvj" ]; then
if whiptail --title "Cleanup" --yesno "Found existing MPVJ install. Wipe it for a clean install?" 10 60 < /dev/tty; then
rm -rf "$REAL_HOME/mpvj"
fi
fi
# 1. INTERROGATION PHASE
echo "Interrogating user for configuration..."
# 1.1 Hostname
HOSTNAME=$(whiptail --inputbox "Enter Hostname (e.g. mpvj)" 8 45 "mpvj" --title "Hostname Configuration" 3>&1 1>&2 2>&3 < /dev/tty) || exit 1
# ── 1. CONFIGURATION ─────────────────────────────────────────────────────────
HOSTNAME=$(whiptail --inputbox "Enter Hostname" 8 45 "mpvj" --title "Hostname" 3>&1 1>&2 2>&3 < /dev/tty) || exit 1
HOSTNAME=${HOSTNAME:-mpvj}
# 1.2 SSID
WIFI_SSID=$(whiptail --inputbox "Enter WiFi SSID (e.g. MPVJ-AP)" 8 45 "MPVJ-AP" --title "WiFi SSID Configuration" 3>&1 1>&2 2>&3 < /dev/tty) || exit 1
WIFI_SSID=$(whiptail --inputbox "Enter WiFi SSID" 8 45 "MPVJ-AP" --title "WiFi SSID" 3>&1 1>&2 2>&3 < /dev/tty) || exit 1
WIFI_SSID=${WIFI_SSID:-MPVJ-AP}
# 1.3 Country Code & Password
while true; do
WIFI_COUNTRY=$(whiptail --inputbox "Enter 2-letter Country Code (e.g. US, GB)" 8 45 "US" --title "WiFi Country Configuration" 3>&1 1>&2 2>&3 < /dev/tty) || exit 1
WIFI_COUNTRY=$(whiptail --inputbox "Enter 2-letter Country Code (e.g. US, GB)" 8 45 "US" --title "WiFi Country" 3>&1 1>&2 2>&3 < /dev/tty) || exit 1
WIFI_COUNTRY=$(echo "$WIFI_COUNTRY" | tr '[:lower:]' '[:upper:]')
if ! echo "$WIFI_COUNTRY" | grep -qE "^[A-Z]{2}$"; then
whiptail --msgbox "Error: Country Code must be 2 letters." 8 45 < /dev/tty
else
break
fi
echo "$WIFI_COUNTRY" | grep -qE "^[A-Z]{2}$" && break
whiptail --msgbox "Country Code must be exactly 2 letters." 8 45 < /dev/tty
done
while true; do
WIFI_PASS=$(whiptail --passwordbox "Enter WiFi Password (min 8 chars)" 8 45 --title "WiFi Password Configuration" 3>&1 1>&2 2>&3 < /dev/tty) || exit 1
if [ ${#WIFI_PASS} -lt 8 ]; then
whiptail --msgbox "Error: Minimum 8 characters required." 8 45 < /dev/tty
else
break
fi
WIFI_PASS=$(whiptail --passwordbox "Enter WiFi Password (min 8 chars)" 8 45 --title "WiFi Password" 3>&1 1>&2 2>&3 < /dev/tty) || exit 1
[ ${#WIFI_PASS} -ge 8 ] && break
whiptail --msgbox "Minimum 8 characters required." 8 45 < /dev/tty
done
# 2. SYSTEM STAGING PHASE
echo "System Staging Phase..."
# 2.1 Pre-Flight Disk Check
# ── 2. SYSTEM SETUP ──────────────────────────────────────────────────────────
FREE_SPACE_KB=$(df / --output=avail | tail -n1)
if [ "$FREE_SPACE_KB" -lt 4194304 ]; then
whiptail --msgbox "Error: 4GB free space required." 8 45 < /dev/tty
if [ "$FREE_SPACE_KB" -lt 2097152 ]; then
whiptail --msgbox "Error: At least 2GB free space required." 8 45 < /dev/tty
exit 1
fi
# 2.2 System Updates
if command -v apt-get > /dev/null; then
apt-get update -y && DEBIAN_FRONTEND=noninteractive apt-get upgrade -y
elif command -v pacman > /dev/null; then
pacman -Syu --noconfirm
echo "Installing runtime dependencies..."
apt-get update -y
DEBIAN_FRONTEND=noninteractive apt-get install -y \
git hostapd dnsmasq avahi-daemon curl ffmpeg \
libzmq3-dev libavahi-compat-libdnssd-dev
if ! command -v node > /dev/null 2>&1; then
curl -fsSL https://deb.nodesource.com/setup_20.x | bash -
apt-get install -y nodejs
fi
# 2.3 Install Dependencies
echo "Installing Core Dependencies..."
if command -v apt-get > /dev/null; then
DEBIAN_FRONTEND=noninteractive apt-get install -y git hostapd dnsmasq avahi-daemon curl ffmpeg build-essential libzmq3-dev libavahi-compat-libdnssd-dev wget
elif command -v pacman > /dev/null; then
pacman -S --noconfirm git hostapd dnsmasq avahi curl ffmpeg base-devel zeromq wget
fi
# Install Node.js if missing
if ! command -v node > /dev/null; then
curl -fsSL https://deb.nodesource.com/setup_20.x | bash - || true
command -v apt-get > /dev/null && apt-get install -y nodejs
fi
# 2.4 Hostname & WiFi Country
# Hostname
hostnamectl set-hostname "$HOSTNAME"
sed -i "s/127.0.1.1.*/127.0.1.1\t$HOSTNAME/g" /etc/hosts || true
# 2.5 OS-Agnostic Swap Increase (2GB)
echo "Ensuring 2GB Swap for build..."
if [ ! -f /swapfile_mpvj ]; then
fallocate -l 2G /swapfile_mpvj || dd if=/dev/zero of=/swapfile_mpvj bs=1M count=2048
chmod 600 /swapfile_mpvj
mkswap /swapfile_mpvj
swapon /swapfile_mpvj
fi
# 3. NETWORKING
echo "Configuring Networking..."
# hostapd
# ── 3. NETWORKING ────────────────────────────────────────────────────────────
echo "Configuring networking..."
cat <<EOF > /etc/hostapd/hostapd.conf
interface=wlan0
driver=nl80211
ssid=$WIFI_SSID
hw_mode=g
channel=7
country_code=$WIFI_COUNTRY
wpa=2
wpa_passphrase=$WIFI_PASS
wpa_key_mgmt=WPA-PSK
EOF
[ -f /etc/default/hostapd ] && sed -i 's/#DAEMON_CONF=""/DAEMON_CONF="\/etc\/hostapd\/hostapd.conf"/g' /etc/default/hostapd
[ -f /etc/default/hostapd ] && \
sed -i 's|#DAEMON_CONF=""|DAEMON_CONF="/etc/hostapd/hostapd.conf"|g' /etc/default/hostapd
# dnsmasq
[ -f /etc/dnsmasq.conf ] && mv /etc/dnsmasq.conf /etc/dnsmasq.conf.bak || true
cat <<EOF > /etc/dnsmasq.conf
interface=wlan0
@ -148,38 +116,39 @@ dhcp-range=192.168.4.2,192.168.4.20,255.255.255.0,24h
address=/$HOSTNAME.local/192.168.4.1
EOF
# dhcpcd / network manager
if [ -f /etc/dhcpcd.conf ]; then
if ! grep -q "interface wlan0" /etc/dhcpcd.conf; then
echo -e "\ninterface wlan0\n static ip_address=192.168.4.1/24\n nohook wpa_supplicant" >> /etc/dhcpcd.conf
fi
if [ -f /etc/dhcpcd.conf ] && ! grep -q "interface wlan0" /etc/dhcpcd.conf; then
printf '\ninterface wlan0\n static ip_address=192.168.4.1/24\n nohook wpa_supplicant\n' >> /etc/dhcpcd.conf
fi
# 4. DEPLOYMENT
echo "Deploying Application..."
# ── 4. DEPLOY APP ────────────────────────────────────────────────────────────
echo "Deploying application from $REPO_URL ..."
[ -d "$REAL_HOME/mpvj" ] && mv "$REAL_HOME/mpvj" "$REAL_HOME/mpvj.old.$(date +%s)"
sudo -u "$REAL_USER" git clone "$REPO_URL" "$REAL_HOME/mpvj"
# Verify pre-built artifacts exist
if [ ! -d "$REAL_HOME/mpvj/frontend/dist" ]; then
echo "ERROR: frontend/dist/ not found in repo."
echo "Run scripts/build.sh on your x86_64 machine first and commit the artifacts."
exit 1
fi
if [ ! -f "$REAL_HOME/mpvj/bin/ofxPiMapper" ]; then
echo "ERROR: bin/ofxPiMapper not found in repo."
echo "Run scripts/build.sh on your x86_64 machine first and commit the artifacts."
exit 1
fi
# Install npm deps for backend only (no build needed)
cd "$REAL_HOME/mpvj/backend"
sudo -u "$REAL_USER" npm install --omit=optional --package-lock=false
# Install ofxPiMapper binary
install -m 755 "$REAL_HOME/mpvj/bin/ofxPiMapper" /usr/local/bin/ofxPiMapper
# Media directory
mkdir -p "$REAL_HOME/media"
chown -R "$REAL_USER:$REAL_USER" "$REAL_HOME/mpvj" "$REAL_HOME/media"
# 4.2 Build
export NODE_OPTIONS="--max-old-space-size=512"
cd "$REAL_HOME/mpvj/backend"
echo "Installing backend dependencies..."
rm -rf node_modules package-lock.json
sudo -u "$REAL_USER" npm install --omit=optional --package-lock=false
cd "$REAL_HOME/mpvj/frontend"
if [ ! -d "dist" ]; then
echo "Installing frontend dependencies and building..."
rm -rf node_modules package-lock.json
sudo -u "$REAL_USER" npm install --omit=optional --package-lock=false
sudo -u "$REAL_USER" npm run build
else
echo "Pre-built frontend detected. Skipping frontend build."
fi
# 4.3 Setup Systemd
# ── 5. SYSTEMD SERVICE ───────────────────────────────────────────────────────
cat <<EOF > /etc/systemd/system/mpvj-backend.service
[Unit]
Description=MPVJ Backend
@ -196,7 +165,7 @@ EOF
systemctl daemon-reload
systemctl enable mpvj-backend.service
# 4.4 .env
# Write .env
cat <<EOF > "$REAL_HOME/mpvj/backend/.env"
PORT=80
MPVJ_HOSTNAME=$HOSTNAME
@ -204,106 +173,6 @@ MPVJ_SSID=$WIFI_SSID
MPVJ_MEDIA_DIR=$REAL_HOME/media
EOF
# 4.5 THE PRE-BUILT ENGINE SOLUTION
echo "Installing Pre-compiled OpenFrameworks (FAST)..."
cd "$REAL_HOME"
if [ ! -d "openFrameworks" ]; then
echo "Downloading OpenFrameworks v0.12.1..."
ARCH=$(uname -m)
OF_FILE="of_v0.12.1_linuxaarch64_release.tar.gz"
[ "$ARCH" != "aarch64" ] && [ "$ARCH" != "x86_64" ] && OF_FILE="of_v0.12.1_linuxarmv6l_release.tar.gz"
DOWNLOAD_URL="https://github.com/openframeworks/openFrameworks/releases/download/0.12.1/$OF_FILE"
if ! sudo -u "$REAL_USER" wget "$DOWNLOAD_URL"; then
sudo -u "$REAL_USER" curl -L -O "$DOWNLOAD_URL"
fi
sudo -u "$REAL_USER" tar -xzf "$OF_FILE"
sudo -u "$REAL_USER" mv of_v0.12.1_linux*_release openFrameworks
sudo -u "$REAL_USER" rm "$OF_FILE"
fi
# ALWAYS verify dependencies, even if OF is already present
echo "Ensuring modern dependencies for Debian Trixie..."
if command -v apt-get > /dev/null; then
DEBIAN_FRONTEND=noninteractive apt-get install -y \
libmpg123-dev libsndfile1-dev libopenal-dev libassimp-dev \
libglew-dev libglfw3-dev liburiparser-dev \
libcurl4-openssl-dev libpugixml-dev libasound2-dev \
libgstreamer1.0-dev libgstreamer-plugins-base1.0-dev \
gstreamer1.0-plugins-good gstreamer1.0-plugins-bad \
gstreamer1.0-plugins-ugly gstreamer1.0-libav \
libgtk-3-dev libboost-filesystem-dev \
libfontconfig1-dev libfreetype-dev libx11-dev \
libxrandr-dev libxinerama-dev libxcursor-dev libxi-dev \
libpulse-dev libudev-dev libfreeimage-dev librtaudio-dev \
freeglut3-dev libxxf86vm-dev
fi
echo "Downloading and building ofxPiMapper Addon..."
cd "$REAL_HOME/openFrameworks/addons"
# FORCE CLEAN CLONE to fix corrupted source files from previous failed runs
rm -rf ofxPiMapper
sudo -u "$REAL_USER" git clone --depth 1 https://github.com/kr15h/ofxPiMapper.git
# Apply patches to the addon before building
cd "$REAL_HOME/openFrameworks/addons/ofxPiMapper"
echo "Applying clean Global Compatibility Patches..."
# 1. Fix highlight patch
cat <<'EOF_PATCH' > /tmp/highlight.patch
if (m.getAddress() == "/ofxPiMapper/surface/highlight"){
int surfaceIndex = m.getArgAsInt32(0);
mapper->getSurfaceManager()->selectSurface(surfaceIndex);
return;
}
EOF_PATCH
sed -i '/if (m.getAddress() == "\/ofxPiMapper\/surface\/select"){/r /tmp/highlight.patch' src/Osc/OscControl.cpp
rm -f /tmp/highlight.patch
# 2. Fix unique_ptr and make_unique globally (Only if std:: is missing)
find src -type f \( -name "*.h" -o -name "*.cpp" \) -exec sed -i 's/\([^:a-zA-Z0-9]\)unique_ptr\b/\1std::unique_ptr/g' {} +
find src -type f \( -name "*.h" -o -name "*.cpp" \) -exec sed -i 's/\([^:a-zA-Z0-9]\)make_unique\b/\1std::make_unique/g' {} +
# 3. Create a single compatibility header for C++11
cat <<'EOF_COMPAT' > src/cpp11_compat.h
#ifndef OFXPIMAPPER_CPP11_COMPAT_H
#define OFXPIMAPPER_CPP11_COMPAT_H
#include <memory>
#include <utility>
#if __cplusplus < 201402L
namespace std {
template<typename T, typename... Args>
std::unique_ptr<T> make_unique(Args&&... args) {
return std::unique_ptr<T>(new T(std::forward<Args>(args)...));
}
}
#endif
#endif
EOF_COMPAT
# 4. Include the compat header in the main addon header
sed -i '1i #include "cpp11_compat.h"' src/ofxPiMapper.h
# 5. Add memory header to all headers where unique_ptr is used
for f in $(grep -l "unique_ptr" src/*.h 2>/dev/null); do
sed -i '1i #include <memory>' "$f"
done
# Build Example
cd "$REAL_HOME/openFrameworks/addons/ofxPiMapper/example_basic"
echo "Cleaning old objects..."
sudo -u "$REAL_USER" make clean
echo "Starting compilation (Forcing C++11 for Symbol Compatibility)..."
# Use -O2 and -fno-tree-vrp to prevent Internal Compiler Errors (ICE) on ARM
GLOBAL_FLAGS="-Wno-error -Wno-reorder -Wno-sign-compare -Wno-delete-non-virtual-dtor -std=c++11 -O2 -fno-tree-vrp"
sudo -u "$REAL_USER" make -j1 PLATFORM_CFLAGS="$GLOBAL_FLAGS" USER_CFLAGS="$GLOBAL_FLAGS" USER_CPPFLAGS="-std=c++11"
cp bin/example_basic /usr/local/bin/ofxPiMapper
chmod +x /usr/local/bin/ofxPiMapper
# 5. FINALIZATION
echo "Cleaning up..."
swapoff /swapfile_mpvj 2>/dev/null || true
rm /swapfile_mpvj 2>/dev/null || true
# ── 6. DONE ───────────────────────────────────────────────────────────────────
whiptail --title "Success" --msgbox "INSTALL COMPLETE!\n\nRebooting into AP Mode.\nSSID: $WIFI_SSID\nIP: 192.168.4.1" 12 45 < /dev/tty
reboot