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.
Files changed (54) hide show
  1. package/agent.config.template.json +30 -0
  2. package/bin/cms-agent.js +233 -0
  3. package/nssm/nssm.exe +0 -0
  4. package/package.json +52 -0
  5. package/public/assets/index-CcBNCz6h.css +1 -0
  6. package/public/assets/index-H-8HDl46.js +1 -0
  7. package/public/index.html +19 -0
  8. package/scripts/guardian.ps1 +75 -0
  9. package/scripts/install-linux.sh +134 -0
  10. package/scripts/install-rpi.sh +117 -0
  11. package/scripts/install-windows.ps1 +529 -0
  12. package/scripts/launch-kiosk.vbs +101 -0
  13. package/scripts/lightman-agent.logrotate +12 -0
  14. package/scripts/lightman-agent.service +38 -0
  15. package/scripts/lightman-shell.bat +128 -0
  16. package/scripts/reinstall-windows.ps1 +26 -0
  17. package/scripts/restore-desktop.ps1 +32 -0
  18. package/scripts/setup.ps1 +116 -0
  19. package/scripts/setup.sh +115 -0
  20. package/scripts/uninstall-linux.sh +50 -0
  21. package/scripts/uninstall-windows.ps1 +54 -0
  22. package/src/commands/display.ts +177 -0
  23. package/src/commands/kiosk.ts +113 -0
  24. package/src/commands/maintenance.ts +106 -0
  25. package/src/commands/network.ts +129 -0
  26. package/src/commands/power.ts +163 -0
  27. package/src/commands/rpi.ts +45 -0
  28. package/src/commands/screenshot.ts +166 -0
  29. package/src/commands/serial.ts +17 -0
  30. package/src/commands/update.ts +124 -0
  31. package/src/index.ts +652 -0
  32. package/src/lib/config.ts +69 -0
  33. package/src/lib/identity.ts +40 -0
  34. package/src/lib/logger.ts +137 -0
  35. package/src/lib/platform.ts +10 -0
  36. package/src/lib/rpi.ts +180 -0
  37. package/src/lib/screens.ts +128 -0
  38. package/src/lib/types.ts +176 -0
  39. package/src/services/commands.ts +107 -0
  40. package/src/services/health.ts +161 -0
  41. package/src/services/kiosk.ts +395 -0
  42. package/src/services/localEvents.ts +60 -0
  43. package/src/services/logForwarder.ts +72 -0
  44. package/src/services/multiScreenKiosk.ts +324 -0
  45. package/src/services/oscBridge.ts +186 -0
  46. package/src/services/powerScheduler.ts +260 -0
  47. package/src/services/provisioning.ts +120 -0
  48. package/src/services/serialBridge.ts +230 -0
  49. package/src/services/serviceLauncher.ts +183 -0
  50. package/src/services/staticServer.ts +226 -0
  51. package/src/services/updater.ts +249 -0
  52. package/src/services/watchdog.ts +310 -0
  53. package/src/services/websocket.ts +152 -0
  54. 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 ""