lightman-agent 1.0.11 → 1.0.14

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/bin/cms-agent.js CHANGED
@@ -8,7 +8,7 @@ import { spawnSync } from 'child_process';
8
8
  import { createInterface } from 'readline/promises';
9
9
  import { stdin as input, stdout as output, cwd, platform, exit } from 'process';
10
10
 
11
- const DEFAULT_SERVER = 'http://192.168.1.100:3401';
11
+ const DEFAULT_SERVER = 'http://192.168.10.100:3401';
12
12
  const INSTALL_CONFIG_PATH = 'C:\\Program Files\\Lightman\\Agent\\agent.config.json';
13
13
 
14
14
  function printUsage() {
@@ -22,7 +22,7 @@ Commands:
22
22
 
23
23
  Options:
24
24
  --slug <value> Device slug (example: C-AV01)
25
- --server <url> Server URL or IP (example: 192.168.1.100 or http://192.168.1.100:3401)
25
+ --server <url> Server URL or IP (example: 192.168.10.100 or http://192.168.10.100:3401)
26
26
  --timezone <tz> Timezone override (default: Asia/Kolkata)
27
27
  --pair-timeout <s> Wait time for pairing in seconds (default: 900, 0 = no timeout)
28
28
  --no-restart Skip reboot after successful install/update
@@ -92,7 +92,7 @@ function normalizeServer(server) {
92
92
  try {
93
93
  parsed = new URL(candidate);
94
94
  } catch {
95
- throw new Error('Invalid server IP/URL. Use an IP like 192.168.1.100 or a URL like http://192.168.1.100:3401.');
95
+ throw new Error('Invalid server IP/URL. Use an IP like 192.168.10.100 or a URL like http://192.168.10.100:3401.');
96
96
  }
97
97
 
98
98
  if (!parsed.port) {
@@ -162,7 +162,7 @@ async function promptServer(defaultServer) {
162
162
  while (true) {
163
163
  const prompt = defaultServer
164
164
  ? `Enter server IP or URL [${defaultServer}]: `
165
- : 'Enter server IP or URL (example 192.168.1.100 or http://192.168.1.100:3401): ';
165
+ : 'Enter server IP or URL (example 192.168.10.100 or http://192.168.10.100:3401): ';
166
166
  const answer = (await rl.question(prompt)).trim();
167
167
  const server = answer || defaultServer || '';
168
168
 
package/package.json CHANGED
@@ -1,53 +1,53 @@
1
- {
2
- "name": "lightman-agent",
3
- "version": "1.0.11",
4
- "description": "LIGHTMAN Agent - System-level daemon for museum display machines",
5
- "private": false,
6
- "type": "module",
7
- "bin": {
8
- "cms-agent": "bin/cms-agent.js",
9
- "lightman-agent": "bin/cms-agent.js"
10
- },
11
- "files": [
12
- "bin/",
13
- "nssm/",
14
- "public/",
15
- "scripts/",
16
- "src/index.ts",
17
- "src/commands/",
18
- "src/lib/",
19
- "src/services/",
20
- "agent.config.json",
21
- "agent.config.template.json",
22
- "package-lock.json",
23
- "tsconfig.json"
24
- ],
25
- "publishConfig": {
26
- "access": "public"
27
- },
28
- "scripts": {
29
- "dev": "tsx watch src/index.ts",
30
- "cms-agent": "node ./bin/cms-agent.js",
31
- "sync-display": "node ./scripts/sync-display.mjs",
32
- "build": "tsc",
33
- "start": "node dist/index.js",
34
- "typecheck": "tsc --noEmit",
35
- "test": "vitest run",
36
- "test:watch": "vitest"
37
- },
38
- "dependencies": {
39
- "systeminformation": "^5.23.0",
40
- "ws": "^8.18.0",
41
- "zod": "^3.24.1"
42
- },
43
- "devDependencies": {
44
- "@types/node": "^22.10.0",
45
- "@types/ws": "^8.5.13",
46
- "tsx": "^4.19.2",
47
- "typescript": "^5.7.2",
48
- "vitest": "^3.1.1"
49
- },
50
- "engines": {
51
- "node": ">=20.0.0"
52
- }
53
- }
1
+ {
2
+ "name": "lightman-agent",
3
+ "version": "1.0.14",
4
+ "description": "LIGHTMAN Agent - System-level daemon for museum display machines",
5
+ "private": false,
6
+ "type": "module",
7
+ "bin": {
8
+ "cms-agent": "bin/cms-agent.js",
9
+ "lightman-agent": "bin/cms-agent.js"
10
+ },
11
+ "files": [
12
+ "bin/",
13
+ "nssm/",
14
+ "public/",
15
+ "scripts/",
16
+ "src/index.ts",
17
+ "src/commands/",
18
+ "src/lib/",
19
+ "src/services/",
20
+ "agent.config.json",
21
+ "agent.config.template.json",
22
+ "package-lock.json",
23
+ "tsconfig.json"
24
+ ],
25
+ "publishConfig": {
26
+ "access": "public"
27
+ },
28
+ "scripts": {
29
+ "dev": "tsx watch src/index.ts",
30
+ "cms-agent": "node ./bin/cms-agent.js",
31
+ "sync-display": "node ./scripts/sync-display.mjs",
32
+ "build": "tsc",
33
+ "start": "node dist/index.js",
34
+ "typecheck": "tsc --noEmit",
35
+ "test": "vitest run",
36
+ "test:watch": "vitest"
37
+ },
38
+ "dependencies": {
39
+ "systeminformation": "^5.23.0",
40
+ "ws": "^8.18.0",
41
+ "zod": "^3.24.1"
42
+ },
43
+ "devDependencies": {
44
+ "@types/node": "^22.10.0",
45
+ "@types/ws": "^8.5.13",
46
+ "tsx": "^4.19.2",
47
+ "typescript": "^5.7.2",
48
+ "vitest": "^3.1.1"
49
+ },
50
+ "engines": {
51
+ "node": ">=20.0.0"
52
+ }
53
+ }
@@ -1,94 +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
- $ConfigPath = "C:\Program Files\Lightman\Agent\agent.config.json"
10
-
11
- function Write-GuardianLog($msg) {
12
- $ts = Get-Date -Format "yyyy-MM-dd HH:mm:ss"
13
- try {
14
- if (-not (Test-Path $LogDir)) { New-Item -ItemType Directory -Force -Path $LogDir | Out-Null }
15
- Add-Content -Path $LogFile -Value "[$ts] $msg" -ErrorAction SilentlyContinue
16
- if ((Get-Item $LogFile -ErrorAction SilentlyContinue).Length -gt 1MB) {
17
- $rotated = "$LogFile.old"
18
- if (Test-Path $rotated) { Remove-Item $rotated -Force }
19
- Rename-Item $LogFile $rotated -Force
20
- }
21
- } catch { }
22
- }
23
-
24
- function Get-BrowserProcessName {
25
- try {
26
- if (Test-Path $ConfigPath) {
27
- $config = Get-Content $ConfigPath -Raw | ConvertFrom-Json
28
- $browserPath = $config.kiosk.browserPath
29
- if ($browserPath) {
30
- $browserLeaf = Split-Path $browserPath -Leaf
31
- if ($browserLeaf) {
32
- return [System.IO.Path]::GetFileNameWithoutExtension($browserLeaf)
33
- }
34
- }
35
- }
36
- } catch { }
37
-
38
- return "chrome"
39
- }
40
-
41
- try {
42
- # 1. Check LIGHTMAN service
43
- $svc = Get-Service -Name $ServiceName -ErrorAction SilentlyContinue
44
- if (-not $svc) {
45
- $svc = Get-Service -DisplayName "LIGHTMAN*" -ErrorAction SilentlyContinue | Select-Object -First 1
46
- }
47
-
48
- if (-not $svc) {
49
- Write-GuardianLog "CRITICAL: Service not found!"
50
- exit 1
51
- }
52
-
53
- if ($svc.Status -ne 'Running') {
54
- Write-GuardianLog "WARNING: Service is $($svc.Status). Restarting..."
55
-
56
- if ($svc.Status -eq 'Stopped') {
57
- if (Test-Path $NssmExe) {
58
- & $NssmExe start $ServiceName 2>$null
59
- } else {
60
- Start-Service -Name $svc.Name -ErrorAction SilentlyContinue
61
- }
62
- Start-Sleep -Seconds 5
63
- $svc.Refresh()
64
- Write-GuardianLog "After restart: $($svc.Status)"
65
- }
66
- elseif ($svc.Status -in @('StartPending', 'StopPending')) {
67
- Write-GuardianLog "Service stuck in $($svc.Status). Force killing node.exe..."
68
- Get-Process -Name "node" -ErrorAction SilentlyContinue | Stop-Process -Force -ErrorAction SilentlyContinue
69
- Start-Sleep -Seconds 3
70
- if (Test-Path $NssmExe) { & $NssmExe start $ServiceName 2>$null }
71
- else { Start-Service -Name $svc.Name -ErrorAction SilentlyContinue }
72
- Start-Sleep -Seconds 5
73
- $svc.Refresh()
74
- Write-GuardianLog "After force restart: $($svc.Status)"
75
- }
76
- }
77
-
78
- # 2. Check kiosk browser
79
- $browserProcess = Get-BrowserProcessName
80
- $browser = Get-Process -Name $browserProcess -ErrorAction SilentlyContinue
81
- if (-not $browser) {
82
- $vbsPath = "C:\Program Files\Lightman\Agent\launch-kiosk.vbs"
83
- if (Test-Path $vbsPath) {
84
- Start-Sleep -Seconds 10
85
- $browserRecheck = Get-Process -Name $browserProcess -ErrorAction SilentlyContinue
86
- if (-not $browserRecheck) {
87
- Write-GuardianLog "Browser '$browserProcess' not running. Launching via VBS..."
88
- Start-Process "wscript.exe" -ArgumentList """$vbsPath""" -WindowStyle Hidden
89
- }
90
- }
91
- }
92
- } catch {
93
- Write-GuardianLog "Guardian error: $_"
94
- }
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
+ }
@@ -1,134 +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 ""
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 ""