lightman-agent 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/agent.config.template.json +30 -0
- package/bin/cms-agent.js +233 -0
- package/nssm/nssm.exe +0 -0
- package/package.json +52 -0
- package/public/assets/index-CcBNCz6h.css +1 -0
- package/public/assets/index-H-8HDl46.js +1 -0
- package/public/index.html +19 -0
- package/scripts/guardian.ps1 +75 -0
- package/scripts/install-linux.sh +134 -0
- package/scripts/install-rpi.sh +117 -0
- package/scripts/install-windows.ps1 +529 -0
- package/scripts/launch-kiosk.vbs +101 -0
- package/scripts/lightman-agent.logrotate +12 -0
- package/scripts/lightman-agent.service +38 -0
- package/scripts/lightman-shell.bat +128 -0
- package/scripts/reinstall-windows.ps1 +26 -0
- package/scripts/restore-desktop.ps1 +32 -0
- package/scripts/setup.ps1 +116 -0
- package/scripts/setup.sh +115 -0
- package/scripts/uninstall-linux.sh +50 -0
- package/scripts/uninstall-windows.ps1 +54 -0
- package/src/commands/display.ts +177 -0
- package/src/commands/kiosk.ts +113 -0
- package/src/commands/maintenance.ts +106 -0
- package/src/commands/network.ts +129 -0
- package/src/commands/power.ts +163 -0
- package/src/commands/rpi.ts +45 -0
- package/src/commands/screenshot.ts +166 -0
- package/src/commands/serial.ts +17 -0
- package/src/commands/update.ts +124 -0
- package/src/index.ts +652 -0
- package/src/lib/config.ts +69 -0
- package/src/lib/identity.ts +40 -0
- package/src/lib/logger.ts +137 -0
- package/src/lib/platform.ts +10 -0
- package/src/lib/rpi.ts +180 -0
- package/src/lib/screens.ts +128 -0
- package/src/lib/types.ts +176 -0
- package/src/services/commands.ts +107 -0
- package/src/services/health.ts +161 -0
- package/src/services/kiosk.ts +395 -0
- package/src/services/localEvents.ts +60 -0
- package/src/services/logForwarder.ts +72 -0
- package/src/services/multiScreenKiosk.ts +324 -0
- package/src/services/oscBridge.ts +186 -0
- package/src/services/powerScheduler.ts +260 -0
- package/src/services/provisioning.ts +120 -0
- package/src/services/serialBridge.ts +230 -0
- package/src/services/serviceLauncher.ts +183 -0
- package/src/services/staticServer.ts +226 -0
- package/src/services/updater.ts +249 -0
- package/src/services/watchdog.ts +310 -0
- package/src/services/websocket.ts +152 -0
- package/tsconfig.json +28 -0
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
|
|
6
|
+
<meta name="mobile-web-app-capable" content="yes" />
|
|
7
|
+
<meta name="apple-mobile-web-app-capable" content="yes" />
|
|
8
|
+
<meta name="apple-mobile-web-app-status-bar-style" content="black" />
|
|
9
|
+
<title>LIGHTMAN Display</title>
|
|
10
|
+
<style>
|
|
11
|
+
body { margin: 0; padding: 0; background: #000; overflow: hidden; }
|
|
12
|
+
</style>
|
|
13
|
+
<script type="module" crossorigin src="/display/assets/index-H-8HDl46.js"></script>
|
|
14
|
+
<link rel="stylesheet" crossorigin href="/display/assets/index-CcBNCz6h.css">
|
|
15
|
+
</head>
|
|
16
|
+
<body>
|
|
17
|
+
<div id="root"></div>
|
|
18
|
+
</body>
|
|
19
|
+
</html>
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
# LIGHTMAN Guardian - Service Health Monitor
|
|
2
|
+
# Runs every 5 minutes via Task Scheduler.
|
|
3
|
+
# Restarts the NSSM service if it's down. Checks Chrome kiosk health.
|
|
4
|
+
|
|
5
|
+
$LogDir = "C:\ProgramData\Lightman\logs"
|
|
6
|
+
$LogFile = Join-Path $LogDir "guardian.log"
|
|
7
|
+
$ServiceName = "LightmanAgent"
|
|
8
|
+
$NssmExe = "C:\ProgramData\Lightman\nssm\nssm.exe"
|
|
9
|
+
|
|
10
|
+
function Write-GuardianLog($msg) {
|
|
11
|
+
$ts = Get-Date -Format "yyyy-MM-dd HH:mm:ss"
|
|
12
|
+
try {
|
|
13
|
+
if (-not (Test-Path $LogDir)) { New-Item -ItemType Directory -Force -Path $LogDir | Out-Null }
|
|
14
|
+
Add-Content -Path $LogFile -Value "[$ts] $msg" -ErrorAction SilentlyContinue
|
|
15
|
+
if ((Get-Item $LogFile -ErrorAction SilentlyContinue).Length -gt 1MB) {
|
|
16
|
+
$rotated = "$LogFile.old"
|
|
17
|
+
if (Test-Path $rotated) { Remove-Item $rotated -Force }
|
|
18
|
+
Rename-Item $LogFile $rotated -Force
|
|
19
|
+
}
|
|
20
|
+
} catch { }
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
try {
|
|
24
|
+
# 1. Check LIGHTMAN service
|
|
25
|
+
$svc = Get-Service -Name $ServiceName -ErrorAction SilentlyContinue
|
|
26
|
+
if (-not $svc) {
|
|
27
|
+
$svc = Get-Service -DisplayName "LIGHTMAN*" -ErrorAction SilentlyContinue | Select-Object -First 1
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
if (-not $svc) {
|
|
31
|
+
Write-GuardianLog "CRITICAL: Service not found!"
|
|
32
|
+
exit 1
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
if ($svc.Status -ne 'Running') {
|
|
36
|
+
Write-GuardianLog "WARNING: Service is $($svc.Status). Restarting..."
|
|
37
|
+
|
|
38
|
+
if ($svc.Status -eq 'Stopped') {
|
|
39
|
+
if (Test-Path $NssmExe) {
|
|
40
|
+
& $NssmExe start $ServiceName 2>$null
|
|
41
|
+
} else {
|
|
42
|
+
Start-Service -Name $svc.Name -ErrorAction SilentlyContinue
|
|
43
|
+
}
|
|
44
|
+
Start-Sleep -Seconds 5
|
|
45
|
+
$svc.Refresh()
|
|
46
|
+
Write-GuardianLog "After restart: $($svc.Status)"
|
|
47
|
+
}
|
|
48
|
+
elseif ($svc.Status -in @('StartPending', 'StopPending')) {
|
|
49
|
+
Write-GuardianLog "Service stuck in $($svc.Status). Force killing node.exe..."
|
|
50
|
+
Get-Process -Name "node" -ErrorAction SilentlyContinue | Stop-Process -Force -ErrorAction SilentlyContinue
|
|
51
|
+
Start-Sleep -Seconds 3
|
|
52
|
+
if (Test-Path $NssmExe) { & $NssmExe start $ServiceName 2>$null }
|
|
53
|
+
else { Start-Service -Name $svc.Name -ErrorAction SilentlyContinue }
|
|
54
|
+
Start-Sleep -Seconds 5
|
|
55
|
+
$svc.Refresh()
|
|
56
|
+
Write-GuardianLog "After force restart: $($svc.Status)"
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
# 2. Check Chrome kiosk
|
|
61
|
+
$chrome = Get-Process -Name "chrome" -ErrorAction SilentlyContinue
|
|
62
|
+
if (-not $chrome) {
|
|
63
|
+
$vbsPath = "C:\Program Files\Lightman\Agent\launch-kiosk.vbs"
|
|
64
|
+
if (Test-Path $vbsPath) {
|
|
65
|
+
Start-Sleep -Seconds 10
|
|
66
|
+
$chromeRecheck = Get-Process -Name "chrome" -ErrorAction SilentlyContinue
|
|
67
|
+
if (-not $chromeRecheck) {
|
|
68
|
+
Write-GuardianLog "Chrome not running. Launching via VBS..."
|
|
69
|
+
Start-Process "wscript.exe" -ArgumentList """$vbsPath""" -WindowStyle Hidden
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
} catch {
|
|
74
|
+
Write-GuardianLog "Guardian error: $_"
|
|
75
|
+
}
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# LIGHTMAN Agent — Linux Installer
|
|
3
|
+
# Run as root:
|
|
4
|
+
# sudo bash install-linux.sh --slug f-av01 --server http://192.168.1.100:3401
|
|
5
|
+
set -euo pipefail
|
|
6
|
+
|
|
7
|
+
INSTALL_DIR="/opt/lightman/agent"
|
|
8
|
+
LOG_DIR="/var/log/lightman"
|
|
9
|
+
SERVICE_NAME="lightman-agent"
|
|
10
|
+
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
11
|
+
AGENT_DIR="$(dirname "$SCRIPT_DIR")"
|
|
12
|
+
|
|
13
|
+
# Defaults
|
|
14
|
+
SLUG=""
|
|
15
|
+
SERVER=""
|
|
16
|
+
TIMEZONE="Asia/Kolkata"
|
|
17
|
+
|
|
18
|
+
# ── Parse arguments ──
|
|
19
|
+
while [[ $# -gt 0 ]]; do
|
|
20
|
+
case $1 in
|
|
21
|
+
--slug) SLUG="$2"; shift 2 ;;
|
|
22
|
+
--server) SERVER="$2"; shift 2 ;;
|
|
23
|
+
--timezone) TIMEZONE="$2"; shift 2 ;;
|
|
24
|
+
-h|--help)
|
|
25
|
+
echo "Usage: sudo bash install-linux.sh --slug f-av01 --server http://192.168.1.100:3401 [--timezone Asia/Kolkata]"
|
|
26
|
+
exit 0
|
|
27
|
+
;;
|
|
28
|
+
*) echo "Unknown option: $1"; exit 1 ;;
|
|
29
|
+
esac
|
|
30
|
+
done
|
|
31
|
+
|
|
32
|
+
# ── Validate required args ──
|
|
33
|
+
if [[ -z "$SLUG" ]]; then
|
|
34
|
+
echo "Error: --slug is required"
|
|
35
|
+
echo "Usage: sudo bash install-linux.sh --slug f-av01 --server http://192.168.1.100:3401"
|
|
36
|
+
exit 1
|
|
37
|
+
fi
|
|
38
|
+
|
|
39
|
+
if [[ -z "$SERVER" ]]; then
|
|
40
|
+
echo "Error: --server is required"
|
|
41
|
+
echo "Usage: sudo bash install-linux.sh --slug f-av01 --server http://192.168.1.100:3401"
|
|
42
|
+
exit 1
|
|
43
|
+
fi
|
|
44
|
+
|
|
45
|
+
# ── Pre-checks ──
|
|
46
|
+
if [[ $EUID -ne 0 ]]; then
|
|
47
|
+
echo "Error: This script must be run as root (use sudo)."
|
|
48
|
+
exit 1
|
|
49
|
+
fi
|
|
50
|
+
|
|
51
|
+
if ! command -v node &> /dev/null; then
|
|
52
|
+
echo "Error: Node.js is not installed. Install Node.js 20+ first."
|
|
53
|
+
exit 1
|
|
54
|
+
fi
|
|
55
|
+
|
|
56
|
+
NODE_VERSION=$(node -v | sed 's/v//' | cut -d. -f1)
|
|
57
|
+
if [[ "$NODE_VERSION" -lt 20 ]]; then
|
|
58
|
+
echo "Error: Node.js 20+ required, found v${NODE_VERSION}."
|
|
59
|
+
exit 1
|
|
60
|
+
fi
|
|
61
|
+
|
|
62
|
+
echo ""
|
|
63
|
+
echo "=== LIGHTMAN Agent — Linux Installer ==="
|
|
64
|
+
echo " Device slug : $SLUG"
|
|
65
|
+
echo " Server URL : $SERVER"
|
|
66
|
+
echo ""
|
|
67
|
+
|
|
68
|
+
# --- Create user/group ---
|
|
69
|
+
if ! id -u lightman &> /dev/null; then
|
|
70
|
+
echo "[1/7] Creating lightman user and group..."
|
|
71
|
+
groupadd --system lightman
|
|
72
|
+
useradd --system --gid lightman --home-dir /opt/lightman --shell /usr/sbin/nologin lightman
|
|
73
|
+
else
|
|
74
|
+
echo "[1/7] User 'lightman' already exists, skipping."
|
|
75
|
+
fi
|
|
76
|
+
|
|
77
|
+
# --- Create directories ---
|
|
78
|
+
echo "[2/7] Creating directories..."
|
|
79
|
+
mkdir -p "$INSTALL_DIR"
|
|
80
|
+
mkdir -p "$LOG_DIR"
|
|
81
|
+
|
|
82
|
+
# --- Copy agent files ---
|
|
83
|
+
echo "[3/7] Copying agent files to ${INSTALL_DIR}..."
|
|
84
|
+
cp -r "$AGENT_DIR/dist" "$INSTALL_DIR/"
|
|
85
|
+
cp "$AGENT_DIR/package.json" "$INSTALL_DIR/"
|
|
86
|
+
cp "$AGENT_DIR/package-lock.json" "$INSTALL_DIR/" 2>/dev/null || true
|
|
87
|
+
# Copy template so setup.sh can use it post-install
|
|
88
|
+
cp "$AGENT_DIR/agent.config.template.json" "$INSTALL_DIR/"
|
|
89
|
+
|
|
90
|
+
# Install production dependencies
|
|
91
|
+
echo "[4/7] Installing production dependencies..."
|
|
92
|
+
cd "$INSTALL_DIR"
|
|
93
|
+
npm ci --omit=dev --ignore-scripts 2>/dev/null || npm install --omit=dev --ignore-scripts
|
|
94
|
+
|
|
95
|
+
# --- Configure this device (generates agent.config.json, clears any stale identity) ---
|
|
96
|
+
echo "[5/7] Configuring device '$SLUG'..."
|
|
97
|
+
bash "$SCRIPT_DIR/setup.sh" \
|
|
98
|
+
--slug "$SLUG" \
|
|
99
|
+
--server "$SERVER" \
|
|
100
|
+
--timezone "$TIMEZONE" \
|
|
101
|
+
--dir "$INSTALL_DIR"
|
|
102
|
+
|
|
103
|
+
# --- Set permissions ---
|
|
104
|
+
echo "[6/7] Setting permissions..."
|
|
105
|
+
chown -R lightman:lightman "$INSTALL_DIR"
|
|
106
|
+
chown -R lightman:lightman "$LOG_DIR"
|
|
107
|
+
chmod 750 "$INSTALL_DIR"
|
|
108
|
+
chmod 750 "$LOG_DIR"
|
|
109
|
+
|
|
110
|
+
# --- Install systemd service ---
|
|
111
|
+
echo "[7/7] Installing systemd service..."
|
|
112
|
+
cp "$SCRIPT_DIR/lightman-agent.service" "/etc/systemd/system/${SERVICE_NAME}.service"
|
|
113
|
+
systemctl daemon-reload
|
|
114
|
+
systemctl enable "$SERVICE_NAME"
|
|
115
|
+
|
|
116
|
+
# --- Install logrotate config ---
|
|
117
|
+
cp "$SCRIPT_DIR/lightman-agent.logrotate" "/etc/logrotate.d/${SERVICE_NAME}" 2>/dev/null || true
|
|
118
|
+
|
|
119
|
+
echo ""
|
|
120
|
+
echo "=== Installation Complete ==="
|
|
121
|
+
echo ""
|
|
122
|
+
echo " Device slug : $SLUG"
|
|
123
|
+
echo " Server : $SERVER"
|
|
124
|
+
echo " Install dir : ${INSTALL_DIR}"
|
|
125
|
+
echo " Log dir : ${LOG_DIR}"
|
|
126
|
+
echo " Service : ${SERVICE_NAME}"
|
|
127
|
+
echo ""
|
|
128
|
+
echo " Start : sudo systemctl start ${SERVICE_NAME}"
|
|
129
|
+
echo " Status : sudo systemctl status ${SERVICE_NAME}"
|
|
130
|
+
echo " Logs : sudo journalctl -u ${SERVICE_NAME} -f"
|
|
131
|
+
echo ""
|
|
132
|
+
echo "The agent will now provision with the LIGHTMAN server."
|
|
133
|
+
echo "If pairing is needed, a 6-digit code will appear in the logs."
|
|
134
|
+
echo ""
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# LIGHTMAN Agent — Raspberry Pi Installer
|
|
3
|
+
# Extends the standard Linux install with RPi-specific optimizations.
|
|
4
|
+
# Run as root: sudo bash install-rpi.sh
|
|
5
|
+
set -euo pipefail
|
|
6
|
+
|
|
7
|
+
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
8
|
+
INSTALL_DIR="/opt/lightman/agent"
|
|
9
|
+
|
|
10
|
+
# --- Pre-checks ---
|
|
11
|
+
if [[ $EUID -ne 0 ]]; then
|
|
12
|
+
echo "Error: This script must be run as root (use sudo)."
|
|
13
|
+
exit 1
|
|
14
|
+
fi
|
|
15
|
+
|
|
16
|
+
if [[ ! -f /proc/device-tree/model ]]; then
|
|
17
|
+
echo "Error: This does not appear to be a Raspberry Pi."
|
|
18
|
+
exit 1
|
|
19
|
+
fi
|
|
20
|
+
|
|
21
|
+
MODEL=$(tr -d '\0' < /proc/device-tree/model)
|
|
22
|
+
echo "=== LIGHTMAN Agent — Raspberry Pi Installer ==="
|
|
23
|
+
echo " Detected: ${MODEL}"
|
|
24
|
+
echo ""
|
|
25
|
+
|
|
26
|
+
# --- Run standard Linux install first ---
|
|
27
|
+
echo "[RPi 1/6] Running standard Linux install..."
|
|
28
|
+
bash "$SCRIPT_DIR/install-linux.sh"
|
|
29
|
+
echo ""
|
|
30
|
+
|
|
31
|
+
# --- GPU memory split ---
|
|
32
|
+
echo "[RPi 2/6] Configuring GPU memory..."
|
|
33
|
+
CONFIG_FILE="/boot/config.txt"
|
|
34
|
+
if [[ -f /boot/firmware/config.txt ]]; then
|
|
35
|
+
CONFIG_FILE="/boot/firmware/config.txt"
|
|
36
|
+
fi
|
|
37
|
+
|
|
38
|
+
if ! grep -q "^gpu_mem=" "$CONFIG_FILE" 2>/dev/null; then
|
|
39
|
+
echo "gpu_mem=128" >> "$CONFIG_FILE"
|
|
40
|
+
echo " Set gpu_mem=128 in ${CONFIG_FILE}"
|
|
41
|
+
else
|
|
42
|
+
echo " gpu_mem already configured in ${CONFIG_FILE}"
|
|
43
|
+
fi
|
|
44
|
+
|
|
45
|
+
# --- Disable screensaver / screen blanking ---
|
|
46
|
+
echo "[RPi 3/6] Disabling screen blanking..."
|
|
47
|
+
# For console blanking
|
|
48
|
+
if ! grep -q "consoleblank=0" /boot/cmdline.txt 2>/dev/null; then
|
|
49
|
+
if [[ -f /boot/cmdline.txt ]]; then
|
|
50
|
+
sed -i 's/$/ consoleblank=0/' /boot/cmdline.txt
|
|
51
|
+
echo " Added consoleblank=0 to /boot/cmdline.txt"
|
|
52
|
+
fi
|
|
53
|
+
fi
|
|
54
|
+
|
|
55
|
+
# For X11 screensaver
|
|
56
|
+
LIGHTDM_CONF="/etc/lightdm/lightdm.conf"
|
|
57
|
+
if [[ -f "$LIGHTDM_CONF" ]]; then
|
|
58
|
+
if ! grep -q "xserver-command.*-s 0" "$LIGHTDM_CONF" 2>/dev/null; then
|
|
59
|
+
sed -i '/^\[Seat:\*\]/a xserver-command=X -s 0 -dpms' "$LIGHTDM_CONF" 2>/dev/null || true
|
|
60
|
+
echo " Disabled X11 screensaver via lightdm"
|
|
61
|
+
fi
|
|
62
|
+
fi
|
|
63
|
+
|
|
64
|
+
# --- Hardware watchdog ---
|
|
65
|
+
echo "[RPi 4/6] Configuring hardware watchdog..."
|
|
66
|
+
if ! grep -q "^dtparam=watchdog=on" "$CONFIG_FILE" 2>/dev/null; then
|
|
67
|
+
echo "dtparam=watchdog=on" >> "$CONFIG_FILE"
|
|
68
|
+
echo " Enabled BCM2835 watchdog in ${CONFIG_FILE}"
|
|
69
|
+
fi
|
|
70
|
+
|
|
71
|
+
# Load watchdog module
|
|
72
|
+
if ! lsmod | grep -q bcm2835_wdt 2>/dev/null; then
|
|
73
|
+
modprobe bcm2835_wdt 2>/dev/null || true
|
|
74
|
+
fi
|
|
75
|
+
|
|
76
|
+
if ! grep -q "bcm2835_wdt" /etc/modules 2>/dev/null; then
|
|
77
|
+
echo "bcm2835_wdt" >> /etc/modules
|
|
78
|
+
fi
|
|
79
|
+
|
|
80
|
+
# Set watchdog device permissions for lightman user
|
|
81
|
+
if [[ -e /dev/watchdog ]]; then
|
|
82
|
+
chown root:lightman /dev/watchdog
|
|
83
|
+
chmod 660 /dev/watchdog
|
|
84
|
+
echo " Set /dev/watchdog permissions for lightman group"
|
|
85
|
+
fi
|
|
86
|
+
|
|
87
|
+
# --- tmpfs for logs (reduce SD card wear) ---
|
|
88
|
+
echo "[RPi 5/6] Configuring tmpfs for logs..."
|
|
89
|
+
FSTAB_ENTRY="tmpfs /var/log/lightman tmpfs defaults,noatime,nosuid,nodev,noexec,mode=0750,size=50M,uid=lightman,gid=lightman 0 0"
|
|
90
|
+
if ! grep -q "/var/log/lightman" /etc/fstab 2>/dev/null; then
|
|
91
|
+
echo "$FSTAB_ENTRY" >> /etc/fstab
|
|
92
|
+
mount -a 2>/dev/null || true
|
|
93
|
+
echo " Added tmpfs mount for /var/log/lightman"
|
|
94
|
+
fi
|
|
95
|
+
|
|
96
|
+
# --- Sudoers for agent power commands ---
|
|
97
|
+
echo "[RPi 6/6] Configuring sudoers..."
|
|
98
|
+
SUDOERS_FILE="/etc/sudoers.d/lightman-agent"
|
|
99
|
+
cat > "$SUDOERS_FILE" << 'SUDOERS'
|
|
100
|
+
# LIGHTMAN Agent - allow power commands without password
|
|
101
|
+
lightman ALL=(ALL) NOPASSWD: /sbin/shutdown
|
|
102
|
+
lightman ALL=(ALL) NOPASSWD: /sbin/reboot
|
|
103
|
+
lightman ALL=(ALL) NOPASSWD: /usr/bin/vcgencmd
|
|
104
|
+
SUDOERS
|
|
105
|
+
chmod 440 "$SUDOERS_FILE"
|
|
106
|
+
echo " Installed sudoers for lightman agent"
|
|
107
|
+
|
|
108
|
+
echo ""
|
|
109
|
+
echo "=== Raspberry Pi Setup Complete ==="
|
|
110
|
+
echo ""
|
|
111
|
+
echo " IMPORTANT: Reboot required for:"
|
|
112
|
+
echo " - GPU memory allocation"
|
|
113
|
+
echo " - Hardware watchdog"
|
|
114
|
+
echo " - Screen blanking changes"
|
|
115
|
+
echo ""
|
|
116
|
+
echo " Reboot now: sudo reboot"
|
|
117
|
+
echo ""
|