claude-live 2.0.5 → 2.0.7

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.
@@ -1,101 +1,101 @@
1
1
  {
2
- "description": "claude-live hooks — forwards Claude Code events to the visualizer",
2
+ "description": "claude-live hooks — auto-starts server on SessionStart, forwards all events to visualizer",
3
3
  "hooks": {
4
4
  "SessionStart": [{
5
5
  "hooks": [{
6
6
  "type": "command",
7
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
7
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" session-start",
8
8
  "async": true
9
9
  }]
10
10
  }],
11
11
  "InstructionsLoaded": [{
12
12
  "hooks": [{
13
13
  "type": "command",
14
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
14
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
15
15
  "async": true
16
16
  }]
17
17
  }],
18
18
  "WorktreeCreate": [{
19
19
  "hooks": [{
20
20
  "type": "command",
21
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
21
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
22
22
  "async": true
23
23
  }]
24
24
  }],
25
25
  "WorktreeRemove": [{
26
26
  "hooks": [{
27
27
  "type": "command",
28
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
28
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
29
29
  "async": true
30
30
  }]
31
31
  }],
32
32
  "PreToolUse": [{
33
33
  "hooks": [{
34
34
  "type": "command",
35
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
35
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
36
36
  "async": true
37
37
  }]
38
38
  }],
39
39
  "PostToolUse": [{
40
40
  "hooks": [{
41
41
  "type": "command",
42
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
42
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
43
43
  "async": true
44
44
  }]
45
45
  }],
46
46
  "Stop": [{
47
47
  "hooks": [{
48
48
  "type": "command",
49
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
49
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
50
50
  "async": true
51
51
  }]
52
52
  }],
53
53
  "Notification": [{
54
54
  "hooks": [{
55
55
  "type": "command",
56
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
56
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
57
57
  "async": true
58
58
  }]
59
59
  }],
60
60
  "PermissionRequest": [{
61
61
  "hooks": [{
62
62
  "type": "command",
63
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
63
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
64
64
  "async": true
65
65
  }]
66
66
  }],
67
67
  "SubagentStart": [{
68
68
  "hooks": [{
69
69
  "type": "command",
70
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
70
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
71
71
  "async": true
72
72
  }]
73
73
  }],
74
74
  "SubagentStop": [{
75
75
  "hooks": [{
76
76
  "type": "command",
77
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
77
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
78
78
  "async": true
79
79
  }]
80
80
  }],
81
81
  "SessionEnd": [{
82
82
  "hooks": [{
83
83
  "type": "command",
84
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
84
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
85
85
  "async": true
86
86
  }]
87
87
  }],
88
88
  "PostToolUseFailure": [{
89
89
  "hooks": [{
90
90
  "type": "command",
91
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
91
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
92
92
  "async": true
93
93
  }]
94
94
  }],
95
95
  "UserPromptSubmit": [{
96
96
  "hooks": [{
97
97
  "type": "command",
98
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
98
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
99
99
  "async": true
100
100
  }]
101
101
  }],
@@ -103,7 +103,7 @@
103
103
  "match": { "trigger": ["manual", "auto"] },
104
104
  "hooks": [{
105
105
  "type": "command",
106
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
106
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
107
107
  "async": true
108
108
  }]
109
109
  }],
@@ -111,49 +111,49 @@
111
111
  "match": { "trigger": ["manual", "auto"] },
112
112
  "hooks": [{
113
113
  "type": "command",
114
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
114
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
115
115
  "async": true
116
116
  }]
117
117
  }],
118
118
  "StopFailure": [{
119
119
  "hooks": [{
120
120
  "type": "command",
121
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
121
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
122
122
  "async": true
123
123
  }]
124
124
  }],
125
125
  "TeammateIdle": [{
126
126
  "hooks": [{
127
127
  "type": "command",
128
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
128
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
129
129
  "async": true
130
130
  }]
131
131
  }],
132
132
  "TaskCompleted": [{
133
133
  "hooks": [{
134
134
  "type": "command",
135
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
135
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
136
136
  "async": true
137
137
  }]
138
138
  }],
139
139
  "ConfigChange": [{
140
140
  "hooks": [{
141
141
  "type": "command",
142
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
142
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
143
143
  "async": true
144
144
  }]
145
145
  }],
146
146
  "Elicitation": [{
147
147
  "hooks": [{
148
148
  "type": "command",
149
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
149
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
150
150
  "async": true
151
151
  }]
152
152
  }],
153
153
  "ElicitationResult": [{
154
154
  "hooks": [{
155
155
  "type": "command",
156
- "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
156
+ "command": "\"${CLAUDE_PLUGIN_ROOT}/hooks/run-hook.cmd\" send-event",
157
157
  "async": true
158
158
  }]
159
159
  }]
package/README.md CHANGED
@@ -37,9 +37,16 @@ Run `npm root -g` to find your global node_modules path if the above doesn't wor
37
37
 
38
38
  ## Use
39
39
 
40
- ```bash
41
- claude-live # Start server (foreground)
42
- claude-live start # Start server (detached)
40
+ Once installed, use the `/claude-live:server` slash command in Claude Code:
41
+
42
+ ```
43
+ /claude-live:server # Check status, auto-start if needed
44
+ /claude-live:server stop # Stop the server
45
+ /claude-live:server restart # Restart the server
46
+ /claude-live:server logs # Show last 30 log lines
47
+ /claude-live:server config # Show current endpoint URL
48
+ /claude-live:server config http://192.168.1.50:43451 # Set remote endpoint
49
+ /claude-live:server config reset # Reset to localhost default
43
50
  ```
44
51
 
45
52
  Then open http://localhost:43451 in your browser.
@@ -57,13 +64,22 @@ Multiple Claude sessions show as separate star systems. Prompts fly inward, resp
57
64
 
58
65
  Under the hood: the plugin sends every event to a lightweight Node.js server (pure passthrough, no persistence). The server broadcasts events to a PixiJS frontend via Server-Sent Events (SSE).
59
66
 
60
- ## CLI
67
+ ## Server Endpoints
61
68
 
62
- ```bash
63
- claude-live start # Open visualization in browser
64
- claude-live stop # Stop the server
65
- claude-live status # Show server status
66
- ```
69
+ | Endpoint | Method | Description |
70
+ |---|---|---|
71
+ | `/hook` | POST | Receive hook events, broadcast to SSE clients |
72
+ | `/events` | GET | SSE stream for the frontend |
73
+ | `/health` | GET | Health check — returns `{"ok":true,"clients":<N>,"port":43451}` |
74
+
75
+ ## Troubleshooting
76
+
77
+ | Symptom | Cause | Fix |
78
+ |---|---|---|
79
+ | No activity in browser | Hooks not firing | Run `/reload-plugins` or check `settings.json` hooks |
80
+ | Server not reachable | Server not running | `/claude-live:server` auto-starts it |
81
+ | `clients: 0` in `/health` | Server up, no browser tab open | Open `http://localhost:43451` |
82
+ | Hook logs location | Debug delivery failures | `~/.config/claude-live/logs/YYYY-MM-DD.jsonl` |
67
83
 
68
84
  ## Development
69
85
 
@@ -20,23 +20,38 @@ User's argument: $ARGUMENTS
20
20
 
21
21
  ## Status output (default and after start)
22
22
 
23
- Run these and display in a compact summary:
23
+ Use `/health` endpoint not SSE stream (SSE causes false negatives with short timeouts):
24
24
 
25
25
  ```
26
- curl -sf http://localhost:43451/events -o /dev/null -m 1 2>&1 && echo "● running on port 43451 — http://localhost:43451" || echo "○ stopped"
26
+ curl -sf http://localhost:43451/health -m 2
27
+ ```
28
+
29
+ Returns `{"ok":true,"version":"X.Y.Z","clients":<N>,"port":43451}` when running.
30
+
31
+ Also show:
32
+ ```
27
33
  ps -eo pid,etime,cmd | grep "node.*server/index.js" | grep -v grep
28
34
  tail -5 /tmp/claude-live.log 2>/dev/null
29
35
  ```
30
36
 
31
- Show: running/stopped, port (always 43451), PID and uptime if running, last 5 log lines.
37
+ Show: running/stopped, port (always 43451), PID and uptime if running, client count, last 5 log lines.
38
+
39
+ ## Version check
40
+
41
+ Compare the running server version (from `/health` response `version` field) against the installed plugin version (from `${CLAUDE_PLUGIN_ROOT}/package.json`). If they differ, warn the user:
42
+
43
+ ```
44
+ ⚠ Version mismatch: server running v2.0.6 but plugin is v2.0.7
45
+ Run `/claude-live:server restart` to pick up the new version.
46
+ ```
32
47
 
33
48
  ## Config Management
34
49
 
35
- The `/claude-live config` subcommands manage where hooks send events.
50
+ The `/claude-live:server config` subcommands manage where hooks send events.
36
51
 
37
52
  ### Display current URL
38
53
  ```
39
- /claude-live config
54
+ /claude-live:server config
40
55
  ```
41
56
  Output example: `● Configured: http://192.168.1.50:43451 (from global config)`
42
57
 
@@ -48,7 +63,7 @@ Possible sources shown:
48
63
 
49
64
  ### Set global URL
50
65
  ```
51
- /claude-live config http://192.168.1.50:43451
66
+ /claude-live:server config http://192.168.1.50:43451
52
67
  ```
53
68
  Output: `✓ URL set to http://192.168.1.50:43451`
54
69
 
@@ -70,7 +85,7 @@ Invalid URLs:
70
85
 
71
86
  ### Reset to default
72
87
  ```
73
- /claude-live config reset
88
+ /claude-live:server config reset
74
89
  ```
75
90
  Output: `✓ Reset to default (localhost:43451)`
76
91
 
@@ -0,0 +1,161 @@
1
+ {
2
+ "description": "claude-live hooks — forwards Claude Code events to the visualizer",
3
+ "hooks": {
4
+ "SessionStart": [{
5
+ "hooks": [{
6
+ "type": "command",
7
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
8
+ "async": true
9
+ }]
10
+ }],
11
+ "InstructionsLoaded": [{
12
+ "hooks": [{
13
+ "type": "command",
14
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
15
+ "async": true
16
+ }]
17
+ }],
18
+ "WorktreeCreate": [{
19
+ "hooks": [{
20
+ "type": "command",
21
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
22
+ "async": true
23
+ }]
24
+ }],
25
+ "WorktreeRemove": [{
26
+ "hooks": [{
27
+ "type": "command",
28
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
29
+ "async": true
30
+ }]
31
+ }],
32
+ "PreToolUse": [{
33
+ "hooks": [{
34
+ "type": "command",
35
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
36
+ "async": true
37
+ }]
38
+ }],
39
+ "PostToolUse": [{
40
+ "hooks": [{
41
+ "type": "command",
42
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
43
+ "async": true
44
+ }]
45
+ }],
46
+ "Stop": [{
47
+ "hooks": [{
48
+ "type": "command",
49
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
50
+ "async": true
51
+ }]
52
+ }],
53
+ "Notification": [{
54
+ "hooks": [{
55
+ "type": "command",
56
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
57
+ "async": true
58
+ }]
59
+ }],
60
+ "PermissionRequest": [{
61
+ "hooks": [{
62
+ "type": "command",
63
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
64
+ "async": true
65
+ }]
66
+ }],
67
+ "SubagentStart": [{
68
+ "hooks": [{
69
+ "type": "command",
70
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
71
+ "async": true
72
+ }]
73
+ }],
74
+ "SubagentStop": [{
75
+ "hooks": [{
76
+ "type": "command",
77
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
78
+ "async": true
79
+ }]
80
+ }],
81
+ "SessionEnd": [{
82
+ "hooks": [{
83
+ "type": "command",
84
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
85
+ "async": true
86
+ }]
87
+ }],
88
+ "PostToolUseFailure": [{
89
+ "hooks": [{
90
+ "type": "command",
91
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
92
+ "async": true
93
+ }]
94
+ }],
95
+ "UserPromptSubmit": [{
96
+ "hooks": [{
97
+ "type": "command",
98
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
99
+ "async": true
100
+ }]
101
+ }],
102
+ "PreCompact": [{
103
+ "match": { "trigger": ["manual", "auto"] },
104
+ "hooks": [{
105
+ "type": "command",
106
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
107
+ "async": true
108
+ }]
109
+ }],
110
+ "PostCompact": [{
111
+ "match": { "trigger": ["manual", "auto"] },
112
+ "hooks": [{
113
+ "type": "command",
114
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
115
+ "async": true
116
+ }]
117
+ }],
118
+ "StopFailure": [{
119
+ "hooks": [{
120
+ "type": "command",
121
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
122
+ "async": true
123
+ }]
124
+ }],
125
+ "TeammateIdle": [{
126
+ "hooks": [{
127
+ "type": "command",
128
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
129
+ "async": true
130
+ }]
131
+ }],
132
+ "TaskCompleted": [{
133
+ "hooks": [{
134
+ "type": "command",
135
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
136
+ "async": true
137
+ }]
138
+ }],
139
+ "ConfigChange": [{
140
+ "hooks": [{
141
+ "type": "command",
142
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
143
+ "async": true
144
+ }]
145
+ }],
146
+ "Elicitation": [{
147
+ "hooks": [{
148
+ "type": "command",
149
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
150
+ "async": true
151
+ }]
152
+ }],
153
+ "ElicitationResult": [{
154
+ "hooks": [{
155
+ "type": "command",
156
+ "command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
157
+ "async": true
158
+ }]
159
+ }]
160
+ }
161
+ }
@@ -0,0 +1,38 @@
1
+ : << 'CMDBLOCK'
2
+ @echo off
3
+ REM Cross-platform polyglot wrapper for hook scripts.
4
+ REM On Windows: cmd.exe runs the batch portion.
5
+ REM On Unix: the shell interprets this as a script (: is a no-op).
6
+
7
+ if "%~1"=="" (
8
+ echo run-hook.cmd: missing script name >&2
9
+ exit /b 1
10
+ )
11
+
12
+ set "HOOK_DIR=%~dp0"
13
+
14
+ if exist "C:\Program Files\Git\bin\bash.exe" (
15
+ "C:\Program Files\Git\bin\bash.exe" "%HOOK_DIR%%~1" %2 %3 %4 %5 %6 %7 %8 %9
16
+ exit /b %ERRORLEVEL%
17
+ )
18
+ if exist "C:\Program Files (x86)\Git\bin\bash.exe" (
19
+ "C:\Program Files (x86)\Git\bin\bash.exe" "%HOOK_DIR%%~1" %2 %3 %4 %5 %6 %7 %8 %9
20
+ exit /b %ERRORLEVEL%
21
+ )
22
+ where bash >nul 2>nul
23
+ if %ERRORLEVEL% equ 0 (
24
+ bash "%HOOK_DIR%%~1" %2 %3 %4 %5 %6 %7 %8 %9
25
+ exit /b %ERRORLEVEL%
26
+ )
27
+ exit /b 0
28
+ CMDBLOCK
29
+
30
+ # Unix: run the named script directly
31
+ SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
32
+ if [ -z "${1:-}" ]; then
33
+ echo "run-hook.cmd: missing script name" >&2
34
+ exit 1
35
+ fi
36
+ SCRIPT_NAME="$1"
37
+ shift
38
+ exec bash "${SCRIPT_DIR}/${SCRIPT_NAME}" "$@"
@@ -0,0 +1,5 @@
1
+ #!/usr/bin/env bash
2
+ # Forwards stdin event JSON to bin/hook.js. Fails silently.
3
+ SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
4
+ PLUGIN_ROOT="$(cd "${SCRIPT_DIR}/.." && pwd)"
5
+ exec node "${PLUGIN_ROOT}/bin/hook.js"
@@ -0,0 +1,40 @@
1
+ #!/usr/bin/env bash
2
+ # SessionStart hook: auto-starts claude-live server if not running, then forwards event.
3
+ set -euo pipefail
4
+
5
+ SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
6
+ PLUGIN_ROOT="$(cd "${SCRIPT_DIR}/.." && pwd)"
7
+ SERVER="${PLUGIN_ROOT}/server/index.js"
8
+ LOG="/tmp/claude-live.log"
9
+ PORT="${CLAUDE_LIVE_PORT:-43451}"
10
+
11
+ # Read stdin so it's available to forward
12
+ input="$(cat)"
13
+
14
+ # Check if server is already running.
15
+ # /events is an SSE stream that never closes, so curl exits with 28 (timeout) on success.
16
+ # We treat exit 0 (got data + closed) and 28 (timeout = streaming) as "server up".
17
+ server_up() {
18
+ local code
19
+ curl -s "http://127.0.0.1:${PORT}/events" -o /dev/null -m 1 2>/dev/null
20
+ code=$?
21
+ [ "$code" -eq 0 ] || [ "$code" -eq 28 ]
22
+ }
23
+
24
+ if ! server_up; then
25
+ # Start server detached
26
+ nohup node "${SERVER}" >"${LOG}" 2>&1 &
27
+ disown
28
+ # Wait up to 3s for it to come up
29
+ for i in 1 2 3; do
30
+ sleep 1
31
+ server_up && break
32
+ done
33
+ fi
34
+
35
+ # Forward event to server (if it came up)
36
+ if [ -n "$input" ]; then
37
+ printf '%s' "$input" | node "${PLUGIN_ROOT}/bin/hook.js" || true
38
+ fi
39
+
40
+ exit 0
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "claude-live",
3
- "version": "2.0.5",
3
+ "version": "2.0.7",
4
4
  "description": "Realtime Claude Code activity visualizer",
5
5
  "license": "MIT",
6
6
  "repository": "https://github.com/marisancans/claude-live",
@@ -19,7 +19,8 @@
19
19
  "bin/",
20
20
  "client/dist/",
21
21
  ".claude-plugin/",
22
- "commands/"
22
+ "commands/",
23
+ "hooks/"
23
24
  ],
24
25
  "engines": {
25
26
  "node": ">=18"
package/server/index.js CHANGED
@@ -5,6 +5,7 @@ import { fileURLToPath } from 'url'
5
5
  import { dirname } from 'path'
6
6
 
7
7
  const __dirname = dirname(fileURLToPath(import.meta.url))
8
+ const VERSION = JSON.parse(readFileSync(join(__dirname, '..', 'package.json'), 'utf8')).version
8
9
  const PORT = parseInt(process.env.PORT || '43451', 10)
9
10
  const DIST = process.env.CLAUDE_LIVE_STATIC_DIR
10
11
  || join(__dirname, '..', 'client', 'dist')
@@ -52,6 +53,13 @@ const server = createServer((req, res) => {
52
53
  return
53
54
  }
54
55
 
56
+ // GET /health — health check
57
+ if (req.method === 'GET' && req.url === '/health') {
58
+ res.writeHead(200, { 'Content-Type': 'application/json' })
59
+ res.end(JSON.stringify({ ok: true, version: VERSION, clients: clients.size, port: PORT }))
60
+ return
61
+ }
62
+
55
63
  // GET /events — SSE stream
56
64
  if (req.method === 'GET' && req.url === '/events') {
57
65
  res.writeHead(200, {