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 +4 -4
- package/package.json +53 -53
- package/scripts/guardian.ps1 +75 -94
- package/scripts/install-linux.sh +134 -134
- package/scripts/install-rpi.sh +117 -117
- package/scripts/install-windows.ps1 +529 -673
- package/scripts/launch-kiosk.vbs +101 -117
- package/scripts/lightman-shell.bat +128 -149
- package/scripts/reinstall-windows.ps1 +1 -1
- package/scripts/setup.ps1 +116 -132
- package/scripts/setup.sh +115 -115
- package/scripts/uninstall-linux.sh +50 -50
- package/src/index.ts +734 -652
- package/src/lib/config.ts +69 -90
- package/src/services/autoUpdater.ts +155 -0
- package/src/services/kiosk.ts +395 -414
- package/src/services/presenceSensor.ts +331 -0
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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
|
+
}
|
package/scripts/guardian.ps1
CHANGED
|
@@ -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
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
$rotated
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
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
|
+
}
|
package/scripts/install-linux.sh
CHANGED
|
@@ -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 ""
|