claude-live 2.0.3 → 2.0.5
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/.claude-plugin/hooks/hooks.json +22 -22
- package/.claude-plugin/skills/diagnostics.md +47 -0
- package/README.md +3 -3
- package/bin/hook.js +68 -14
- package/package.json +1 -1
|
@@ -4,98 +4,98 @@
|
|
|
4
4
|
"SessionStart": [{
|
|
5
5
|
"hooks": [{
|
|
6
6
|
"type": "command",
|
|
7
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
7
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
8
8
|
"async": true
|
|
9
9
|
}]
|
|
10
10
|
}],
|
|
11
11
|
"InstructionsLoaded": [{
|
|
12
12
|
"hooks": [{
|
|
13
13
|
"type": "command",
|
|
14
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
14
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
15
15
|
"async": true
|
|
16
16
|
}]
|
|
17
17
|
}],
|
|
18
18
|
"WorktreeCreate": [{
|
|
19
19
|
"hooks": [{
|
|
20
20
|
"type": "command",
|
|
21
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
21
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
22
22
|
"async": true
|
|
23
23
|
}]
|
|
24
24
|
}],
|
|
25
25
|
"WorktreeRemove": [{
|
|
26
26
|
"hooks": [{
|
|
27
27
|
"type": "command",
|
|
28
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
28
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
29
29
|
"async": true
|
|
30
30
|
}]
|
|
31
31
|
}],
|
|
32
32
|
"PreToolUse": [{
|
|
33
33
|
"hooks": [{
|
|
34
34
|
"type": "command",
|
|
35
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
35
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
36
36
|
"async": true
|
|
37
37
|
}]
|
|
38
38
|
}],
|
|
39
39
|
"PostToolUse": [{
|
|
40
40
|
"hooks": [{
|
|
41
41
|
"type": "command",
|
|
42
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
42
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
43
43
|
"async": true
|
|
44
44
|
}]
|
|
45
45
|
}],
|
|
46
46
|
"Stop": [{
|
|
47
47
|
"hooks": [{
|
|
48
48
|
"type": "command",
|
|
49
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
49
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
50
50
|
"async": true
|
|
51
51
|
}]
|
|
52
52
|
}],
|
|
53
53
|
"Notification": [{
|
|
54
54
|
"hooks": [{
|
|
55
55
|
"type": "command",
|
|
56
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
56
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
57
57
|
"async": true
|
|
58
58
|
}]
|
|
59
59
|
}],
|
|
60
60
|
"PermissionRequest": [{
|
|
61
61
|
"hooks": [{
|
|
62
62
|
"type": "command",
|
|
63
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
63
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
64
64
|
"async": true
|
|
65
65
|
}]
|
|
66
66
|
}],
|
|
67
67
|
"SubagentStart": [{
|
|
68
68
|
"hooks": [{
|
|
69
69
|
"type": "command",
|
|
70
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
70
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
71
71
|
"async": true
|
|
72
72
|
}]
|
|
73
73
|
}],
|
|
74
74
|
"SubagentStop": [{
|
|
75
75
|
"hooks": [{
|
|
76
76
|
"type": "command",
|
|
77
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
77
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
78
78
|
"async": true
|
|
79
79
|
}]
|
|
80
80
|
}],
|
|
81
81
|
"SessionEnd": [{
|
|
82
82
|
"hooks": [{
|
|
83
83
|
"type": "command",
|
|
84
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
84
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
85
85
|
"async": true
|
|
86
86
|
}]
|
|
87
87
|
}],
|
|
88
88
|
"PostToolUseFailure": [{
|
|
89
89
|
"hooks": [{
|
|
90
90
|
"type": "command",
|
|
91
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
91
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
92
92
|
"async": true
|
|
93
93
|
}]
|
|
94
94
|
}],
|
|
95
95
|
"UserPromptSubmit": [{
|
|
96
96
|
"hooks": [{
|
|
97
97
|
"type": "command",
|
|
98
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
98
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
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}/bin/hook.js",
|
|
106
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
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}/bin/hook.js",
|
|
114
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
115
115
|
"async": true
|
|
116
116
|
}]
|
|
117
117
|
}],
|
|
118
118
|
"StopFailure": [{
|
|
119
119
|
"hooks": [{
|
|
120
120
|
"type": "command",
|
|
121
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
121
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
122
122
|
"async": true
|
|
123
123
|
}]
|
|
124
124
|
}],
|
|
125
125
|
"TeammateIdle": [{
|
|
126
126
|
"hooks": [{
|
|
127
127
|
"type": "command",
|
|
128
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
128
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
129
129
|
"async": true
|
|
130
130
|
}]
|
|
131
131
|
}],
|
|
132
132
|
"TaskCompleted": [{
|
|
133
133
|
"hooks": [{
|
|
134
134
|
"type": "command",
|
|
135
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
135
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
136
136
|
"async": true
|
|
137
137
|
}]
|
|
138
138
|
}],
|
|
139
139
|
"ConfigChange": [{
|
|
140
140
|
"hooks": [{
|
|
141
141
|
"type": "command",
|
|
142
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
142
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
143
143
|
"async": true
|
|
144
144
|
}]
|
|
145
145
|
}],
|
|
146
146
|
"Elicitation": [{
|
|
147
147
|
"hooks": [{
|
|
148
148
|
"type": "command",
|
|
149
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
149
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
150
150
|
"async": true
|
|
151
151
|
}]
|
|
152
152
|
}],
|
|
153
153
|
"ElicitationResult": [{
|
|
154
154
|
"hooks": [{
|
|
155
155
|
"type": "command",
|
|
156
|
-
"command": "node ${CLAUDE_PLUGIN_ROOT}/bin/hook.js",
|
|
156
|
+
"command": "node ${CLAUDE_PLUGIN_ROOT:-$HOME/.claude/plugins/marketplaces/claude-live}/bin/hook.js",
|
|
157
157
|
"async": true
|
|
158
158
|
}]
|
|
159
159
|
}]
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: diagnostics
|
|
3
|
+
description: Show claude-live hook health — recent log entries, success/failure rates per target, and last errors
|
|
4
|
+
trigger: explicit
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
Show a diagnostics report for claude-live hooks.
|
|
8
|
+
|
|
9
|
+
## Log location
|
|
10
|
+
|
|
11
|
+
Logs are written to `~/.config/claude-live/logs/YYYY-MM-DD.jsonl` — one JSON line per hook fire.
|
|
12
|
+
|
|
13
|
+
## Steps
|
|
14
|
+
|
|
15
|
+
1. Read today's log file: `~/.config/claude-live/logs/<today>.jsonl`
|
|
16
|
+
- If it doesn't exist, also try yesterday's
|
|
17
|
+
- If no logs exist at all, report: "No hook logs found — hooks may not be firing"
|
|
18
|
+
|
|
19
|
+
2. Parse each line as JSON with fields: `ts`, `session_id`, `hook_event_name`, `tool_name`, `targets[]`
|
|
20
|
+
- Each target entry has: `url`, `ok` (bool), `ms`, optionally `error`
|
|
21
|
+
|
|
22
|
+
3. Report:
|
|
23
|
+
|
|
24
|
+
```
|
|
25
|
+
claude-live diagnostics
|
|
26
|
+
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
|
|
27
|
+
Log: ~/.config/claude-live/logs/<date>.jsonl
|
|
28
|
+
Entries: <N> hooks fired
|
|
29
|
+
|
|
30
|
+
Targets:
|
|
31
|
+
http://localhost:43451 ✓ 142/150 (94%) avg 3ms
|
|
32
|
+
http://192.168.1.50:43451 ✗ 0/150 (0%) — connection_error
|
|
33
|
+
|
|
34
|
+
Last 5 entries:
|
|
35
|
+
14:23:01 PreToolUse Bash ✓ 2ms
|
|
36
|
+
14:23:00 PreToolUse Read ✓ 3ms
|
|
37
|
+
14:22:58 PostToolUse Bash ✓ 2ms
|
|
38
|
+
...
|
|
39
|
+
|
|
40
|
+
Recent errors (if any):
|
|
41
|
+
14:20:11 PostToolUse Bash http://192.168.1.50:43451 connection_error
|
|
42
|
+
```
|
|
43
|
+
|
|
44
|
+
4. If all targets are failing: suggest running `/claude-live start` or checking `/claude-live status`
|
|
45
|
+
5. If no log file: suggest the user check that the plugin is enabled with `/reload-plugins`
|
|
46
|
+
|
|
47
|
+
Use the Read tool to read the log file and Bash for `date` if needed. Keep output compact.
|
package/README.md
CHANGED
|
@@ -27,13 +27,13 @@ Then add hooks to your `~/.claude/settings.json`:
|
|
|
27
27
|
```json
|
|
28
28
|
{
|
|
29
29
|
"hooks": {
|
|
30
|
-
"PreToolUse": [{ "hooks": [{ "type": "command", "command": "node /
|
|
31
|
-
"PostToolUse": [{ "hooks": [{ "type": "command", "command": "node /
|
|
30
|
+
"PreToolUse": [{ "hooks": [{ "type": "command", "command": "node $(npm root -g)/claude-live/bin/hook.js", "async": true }] }],
|
|
31
|
+
"PostToolUse": [{ "hooks": [{ "type": "command", "command": "node $(npm root -g)/claude-live/bin/hook.js", "async": true }] }]
|
|
32
32
|
}
|
|
33
33
|
}
|
|
34
34
|
```
|
|
35
35
|
|
|
36
|
-
|
|
36
|
+
Run `npm root -g` to find your global node_modules path if the above doesn't work on your shell.
|
|
37
37
|
|
|
38
38
|
## Use
|
|
39
39
|
|
package/bin/hook.js
CHANGED
|
@@ -1,24 +1,78 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
// Reads a JSON event from stdin and POSTs it to the claude-live server.
|
|
3
|
-
// Fails silently — must never block Claude Code.
|
|
4
2
|
import { request } from 'http'
|
|
3
|
+
import { request as httpsRequest } from 'https'
|
|
4
|
+
import { readFileSync, mkdirSync, appendFileSync } from 'fs'
|
|
5
|
+
import { homedir } from 'os'
|
|
6
|
+
import { join } from 'path'
|
|
7
|
+
|
|
8
|
+
const CONFIG_PATH = join(homedir(), '.config', 'claude-live', 'config.json')
|
|
9
|
+
const LOG_DIR = join(homedir(), '.config', 'claude-live', 'logs')
|
|
10
|
+
|
|
11
|
+
function getTargets() {
|
|
12
|
+
try {
|
|
13
|
+
const cfg = JSON.parse(readFileSync(CONFIG_PATH, 'utf8'))
|
|
14
|
+
if (Array.isArray(cfg.targets) && cfg.targets.length > 0) return cfg.targets
|
|
15
|
+
if (cfg.url) return [cfg.url]
|
|
16
|
+
} catch {}
|
|
17
|
+
return ['http://localhost:43451']
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
function writeLog(entry) {
|
|
21
|
+
try {
|
|
22
|
+
mkdirSync(LOG_DIR, { recursive: true })
|
|
23
|
+
const date = new Date().toISOString().slice(0, 10)
|
|
24
|
+
appendFileSync(join(LOG_DIR, `${date}.jsonl`), JSON.stringify(entry) + '\n')
|
|
25
|
+
} catch {}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
function postToTarget(url, body) {
|
|
29
|
+
return new Promise(resolve => {
|
|
30
|
+
const start = Date.now()
|
|
31
|
+
try {
|
|
32
|
+
const parsed = new URL(url)
|
|
33
|
+
const isHttps = parsed.protocol === 'https:'
|
|
34
|
+
const req = (isHttps ? httpsRequest : request)({
|
|
35
|
+
hostname: parsed.hostname,
|
|
36
|
+
port: parsed.port || (isHttps ? 443 : 80),
|
|
37
|
+
path: '/hook',
|
|
38
|
+
method: 'POST',
|
|
39
|
+
headers: { 'Content-Type': 'application/json', 'Content-Length': body.length },
|
|
40
|
+
timeout: 2000,
|
|
41
|
+
}, res => {
|
|
42
|
+
res.resume()
|
|
43
|
+
resolve({ url, ok: res.statusCode === 200, ms: Date.now() - start })
|
|
44
|
+
})
|
|
45
|
+
req.on('error', () => resolve({ url, ok: false, ms: Date.now() - start, error: 'connection_error' }))
|
|
46
|
+
req.on('timeout', () => { req.destroy(); resolve({ url, ok: false, ms: Date.now() - start, error: 'timeout' }) })
|
|
47
|
+
req.end(body)
|
|
48
|
+
} catch (e) {
|
|
49
|
+
resolve({ url, ok: false, ms: Date.now() - start, error: e.message })
|
|
50
|
+
}
|
|
51
|
+
})
|
|
52
|
+
}
|
|
5
53
|
|
|
6
54
|
let data = ''
|
|
7
55
|
process.stdin.setEncoding('utf8')
|
|
8
56
|
process.stdin.on('data', c => data += c)
|
|
9
|
-
process.stdin.on('end', () => {
|
|
57
|
+
process.stdin.on('end', async () => {
|
|
10
58
|
if (!data.trim()) process.exit(0)
|
|
59
|
+
|
|
11
60
|
const body = Buffer.from(data)
|
|
12
|
-
const
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
61
|
+
const targets = getTargets()
|
|
62
|
+
|
|
63
|
+
let event = {}
|
|
64
|
+
try { event = JSON.parse(data) } catch {}
|
|
65
|
+
|
|
66
|
+
const results = await Promise.all(targets.map(url => postToTarget(url, body)))
|
|
67
|
+
|
|
68
|
+
writeLog({
|
|
69
|
+
ts: new Date().toISOString(),
|
|
70
|
+
session_id: event.session_id ?? null,
|
|
71
|
+
hook_event_name: event.hook_event_name ?? null,
|
|
72
|
+
tool_name: event.tool_name ?? null,
|
|
73
|
+
targets: results,
|
|
19
74
|
})
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
req.end(body)
|
|
75
|
+
|
|
76
|
+
process.exit(0)
|
|
23
77
|
})
|
|
24
|
-
process.stdin.on('error', () =>
|
|
78
|
+
process.stdin.on('error', () => process.exit(0))
|