airlock-bot 0.0.1 → 0.2.2
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/LICENSE +21 -0
- package/README.md +337 -0
- package/airlock.service +27 -0
- package/dist/allowlist/engine.d.ts +9 -0
- package/dist/allowlist/engine.d.ts.map +1 -0
- package/dist/allowlist/engine.js +24 -0
- package/dist/allowlist/engine.js.map +1 -0
- package/dist/allowlist/pattern.d.ts +13 -0
- package/dist/allowlist/pattern.d.ts.map +1 -0
- package/dist/allowlist/pattern.js +33 -0
- package/dist/allowlist/pattern.js.map +1 -0
- package/dist/audit/api.d.ts +7 -0
- package/dist/audit/api.d.ts.map +1 -0
- package/dist/audit/api.js +31 -0
- package/dist/audit/api.js.map +1 -0
- package/dist/audit/db.d.ts +44 -0
- package/dist/audit/db.d.ts.map +1 -0
- package/dist/audit/db.js +121 -0
- package/dist/audit/db.js.map +1 -0
- package/dist/audit/logger.d.ts +25 -0
- package/dist/audit/logger.d.ts.map +1 -0
- package/dist/audit/logger.js +58 -0
- package/dist/audit/logger.js.map +1 -0
- package/dist/audit/redactor.d.ts +5 -0
- package/dist/audit/redactor.d.ts.map +1 -0
- package/dist/audit/redactor.js +27 -0
- package/dist/audit/redactor.js.map +1 -0
- package/dist/backend/cli/adapter.d.ts +23 -0
- package/dist/backend/cli/adapter.d.ts.map +1 -0
- package/dist/backend/cli/adapter.js +176 -0
- package/dist/backend/cli/adapter.js.map +1 -0
- package/dist/backend/cli/builder.d.ts +3 -0
- package/dist/backend/cli/builder.d.ts.map +1 -0
- package/dist/backend/cli/builder.js +52 -0
- package/dist/backend/cli/builder.js.map +1 -0
- package/dist/backend/cli/escaper.d.ts +2 -0
- package/dist/backend/cli/escaper.d.ts.map +1 -0
- package/dist/backend/cli/escaper.js +8 -0
- package/dist/backend/cli/escaper.js.map +1 -0
- package/dist/backend/exec-adapter.d.ts +13 -0
- package/dist/backend/exec-adapter.d.ts.map +1 -0
- package/dist/backend/exec-adapter.js +39 -0
- package/dist/backend/exec-adapter.js.map +1 -0
- package/dist/backend/factory.d.ts +9 -0
- package/dist/backend/factory.d.ts.map +1 -0
- package/dist/backend/factory.js +35 -0
- package/dist/backend/factory.js.map +1 -0
- package/dist/backend/http-adapter.d.ts +15 -0
- package/dist/backend/http-adapter.d.ts.map +1 -0
- package/dist/backend/http-adapter.js +39 -0
- package/dist/backend/http-adapter.js.map +1 -0
- package/dist/backend/mcp-adapter.d.ts +14 -0
- package/dist/backend/mcp-adapter.d.ts.map +1 -0
- package/dist/backend/mcp-adapter.js +38 -0
- package/dist/backend/mcp-adapter.js.map +1 -0
- package/dist/backend/openapi/adapter.d.ts +17 -0
- package/dist/backend/openapi/adapter.d.ts.map +1 -0
- package/dist/backend/openapi/adapter.js +144 -0
- package/dist/backend/openapi/adapter.js.map +1 -0
- package/dist/backend/openapi/parser.d.ts +21 -0
- package/dist/backend/openapi/parser.d.ts.map +1 -0
- package/dist/backend/openapi/parser.js +145 -0
- package/dist/backend/openapi/parser.js.map +1 -0
- package/dist/backend/types.d.ts +9 -0
- package/dist/backend/types.d.ts.map +1 -0
- package/dist/backend/types.js +2 -0
- package/dist/backend/types.js.map +1 -0
- package/dist/config/loader.d.ts +12 -0
- package/dist/config/loader.d.ts.map +1 -0
- package/dist/config/loader.js +178 -0
- package/dist/config/loader.js.map +1 -0
- package/dist/config/profiles.d.ts +12 -0
- package/dist/config/profiles.d.ts.map +1 -0
- package/dist/config/profiles.js +34 -0
- package/dist/config/profiles.js.map +1 -0
- package/dist/config/schema.d.ts +2034 -0
- package/dist/config/schema.d.ts.map +1 -0
- package/dist/config/schema.js +257 -0
- package/dist/config/schema.js.map +1 -0
- package/dist/config/watcher.d.ts +11 -0
- package/dist/config/watcher.d.ts.map +1 -0
- package/dist/config/watcher.js +39 -0
- package/dist/config/watcher.js.map +1 -0
- package/dist/configure-agent/cli.d.ts +2 -0
- package/dist/configure-agent/cli.d.ts.map +1 -0
- package/dist/configure-agent/cli.js +390 -0
- package/dist/configure-agent/cli.js.map +1 -0
- package/dist/discover/cli.d.ts +2 -0
- package/dist/discover/cli.d.ts.map +1 -0
- package/dist/discover/cli.js +97 -0
- package/dist/discover/cli.js.map +1 -0
- package/dist/discover/index.d.ts +19 -0
- package/dist/discover/index.d.ts.map +1 -0
- package/dist/discover/index.js +70 -0
- package/dist/discover/index.js.map +1 -0
- package/dist/discover/openapi.d.ts +9 -0
- package/dist/discover/openapi.d.ts.map +1 -0
- package/dist/discover/openapi.js +47 -0
- package/dist/discover/openapi.js.map +1 -0
- package/dist/discover/strategies/fig.d.ts +29 -0
- package/dist/discover/strategies/fig.d.ts.map +1 -0
- package/dist/discover/strategies/fig.js +82 -0
- package/dist/discover/strategies/fig.js.map +1 -0
- package/dist/discover/strategies/help-parser.d.ts +21 -0
- package/dist/discover/strategies/help-parser.d.ts.map +1 -0
- package/dist/discover/strategies/help-parser.js +121 -0
- package/dist/discover/strategies/help-parser.js.map +1 -0
- package/dist/discover/writer.d.ts +5 -0
- package/dist/discover/writer.d.ts.map +1 -0
- package/dist/discover/writer.js +14 -0
- package/dist/discover/writer.js.map +1 -0
- package/dist/gateway.d.ts +20 -0
- package/dist/gateway.d.ts.map +1 -0
- package/dist/gateway.js +125 -0
- package/dist/gateway.js.map +1 -0
- package/dist/hitl/api.d.ts +7 -0
- package/dist/hitl/api.d.ts.map +1 -0
- package/dist/hitl/api.js +35 -0
- package/dist/hitl/api.js.map +1 -0
- package/dist/hitl/batcher.d.ts +11 -0
- package/dist/hitl/batcher.d.ts.map +1 -0
- package/dist/hitl/batcher.js +37 -0
- package/dist/hitl/batcher.js.map +1 -0
- package/dist/hitl/engine.d.ts +36 -0
- package/dist/hitl/engine.d.ts.map +1 -0
- package/dist/hitl/engine.js +150 -0
- package/dist/hitl/engine.js.map +1 -0
- package/dist/hitl/formatter.d.ts +4 -0
- package/dist/hitl/formatter.d.ts.map +1 -0
- package/dist/hitl/formatter.js +31 -0
- package/dist/hitl/formatter.js.map +1 -0
- package/dist/hitl/parser.d.ts +7 -0
- package/dist/hitl/parser.d.ts.map +1 -0
- package/dist/hitl/parser.js +17 -0
- package/dist/hitl/parser.js.map +1 -0
- package/dist/hitl/provider-factory.d.ts +4 -0
- package/dist/hitl/provider-factory.d.ts.map +1 -0
- package/dist/hitl/provider-factory.js +42 -0
- package/dist/hitl/provider-factory.js.map +1 -0
- package/dist/hitl/providers/composite.d.ts +9 -0
- package/dist/hitl/providers/composite.d.ts.map +1 -0
- package/dist/hitl/providers/composite.js +23 -0
- package/dist/hitl/providers/composite.js.map +1 -0
- package/dist/hitl/providers/dashboard.d.ts +17 -0
- package/dist/hitl/providers/dashboard.d.ts.map +1 -0
- package/dist/hitl/providers/dashboard.js +210 -0
- package/dist/hitl/providers/dashboard.js.map +1 -0
- package/dist/hitl/providers/macos.d.ts +10 -0
- package/dist/hitl/providers/macos.d.ts.map +1 -0
- package/dist/hitl/providers/macos.js +65 -0
- package/dist/hitl/providers/macos.js.map +1 -0
- package/dist/hitl/providers/openclaw.d.ts +21 -0
- package/dist/hitl/providers/openclaw.d.ts.map +1 -0
- package/dist/hitl/providers/openclaw.js +106 -0
- package/dist/hitl/providers/openclaw.js.map +1 -0
- package/dist/hitl/providers/slack.d.ts +12 -0
- package/dist/hitl/providers/slack.d.ts.map +1 -0
- package/dist/hitl/providers/slack.js +24 -0
- package/dist/hitl/providers/slack.js.map +1 -0
- package/dist/hitl/providers/stdio.d.ts +12 -0
- package/dist/hitl/providers/stdio.d.ts.map +1 -0
- package/dist/hitl/providers/stdio.js +41 -0
- package/dist/hitl/providers/stdio.js.map +1 -0
- package/dist/hitl/providers/telegram.d.ts +22 -0
- package/dist/hitl/providers/telegram.d.ts.map +1 -0
- package/dist/hitl/providers/telegram.js +87 -0
- package/dist/hitl/providers/telegram.js.map +1 -0
- package/dist/hitl/providers/tui.d.ts +16 -0
- package/dist/hitl/providers/tui.d.ts.map +1 -0
- package/dist/hitl/providers/tui.js +169 -0
- package/dist/hitl/providers/tui.js.map +1 -0
- package/dist/hitl/providers/types.d.ts +18 -0
- package/dist/hitl/providers/types.d.ts.map +1 -0
- package/dist/hitl/providers/types.js +2 -0
- package/dist/hitl/providers/types.js.map +1 -0
- package/dist/hitl/providers/webhook.d.ts +13 -0
- package/dist/hitl/providers/webhook.d.ts.map +1 -0
- package/dist/hitl/providers/webhook.js +27 -0
- package/dist/hitl/providers/webhook.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +115 -0
- package/dist/index.js.map +1 -0
- package/dist/middleware/chain-builder.d.ts +16 -0
- package/dist/middleware/chain-builder.d.ts.map +1 -0
- package/dist/middleware/chain-builder.js +139 -0
- package/dist/middleware/chain-builder.js.map +1 -0
- package/dist/middleware/compose.d.ts +3 -0
- package/dist/middleware/compose.d.ts.map +1 -0
- package/dist/middleware/compose.js +15 -0
- package/dist/middleware/compose.js.map +1 -0
- package/dist/middleware/core/allowlist.d.ts +3 -0
- package/dist/middleware/core/allowlist.d.ts.map +1 -0
- package/dist/middleware/core/allowlist.js +23 -0
- package/dist/middleware/core/allowlist.js.map +1 -0
- package/dist/middleware/core/exec-policy.d.ts +3 -0
- package/dist/middleware/core/exec-policy.d.ts.map +1 -0
- package/dist/middleware/core/exec-policy.js +30 -0
- package/dist/middleware/core/exec-policy.js.map +1 -0
- package/dist/middleware/core/execute.d.ts +3 -0
- package/dist/middleware/core/execute.d.ts.map +1 -0
- package/dist/middleware/core/execute.js +35 -0
- package/dist/middleware/core/execute.js.map +1 -0
- package/dist/middleware/core/hitl-gate.d.ts +3 -0
- package/dist/middleware/core/hitl-gate.d.ts.map +1 -0
- package/dist/middleware/core/hitl-gate.js +38 -0
- package/dist/middleware/core/hitl-gate.js.map +1 -0
- package/dist/middleware/core/rate-limiter.d.ts +10 -0
- package/dist/middleware/core/rate-limiter.d.ts.map +1 -0
- package/dist/middleware/core/rate-limiter.js +32 -0
- package/dist/middleware/core/rate-limiter.js.map +1 -0
- package/dist/middleware/core/schema-validator.d.ts +3 -0
- package/dist/middleware/core/schema-validator.d.ts.map +1 -0
- package/dist/middleware/core/schema-validator.js +31 -0
- package/dist/middleware/core/schema-validator.js.map +1 -0
- package/dist/middleware/detectors/injection-detector.d.ts +12 -0
- package/dist/middleware/detectors/injection-detector.d.ts.map +1 -0
- package/dist/middleware/detectors/injection-detector.js +129 -0
- package/dist/middleware/detectors/injection-detector.js.map +1 -0
- package/dist/middleware/detectors/sensitivity-classifier.d.ts +12 -0
- package/dist/middleware/detectors/sensitivity-classifier.d.ts.map +1 -0
- package/dist/middleware/detectors/sensitivity-classifier.js +125 -0
- package/dist/middleware/detectors/sensitivity-classifier.js.map +1 -0
- package/dist/middleware/post/canary-token-injector.d.ts +10 -0
- package/dist/middleware/post/canary-token-injector.d.ts.map +1 -0
- package/dist/middleware/post/canary-token-injector.js +53 -0
- package/dist/middleware/post/canary-token-injector.js.map +1 -0
- package/dist/middleware/post/output-injection-detector.d.ts +7 -0
- package/dist/middleware/post/output-injection-detector.d.ts.map +1 -0
- package/dist/middleware/post/output-injection-detector.js +46 -0
- package/dist/middleware/post/output-injection-detector.js.map +1 -0
- package/dist/middleware/post/output-size-limiter.d.ts +7 -0
- package/dist/middleware/post/output-size-limiter.d.ts.map +1 -0
- package/dist/middleware/post/output-size-limiter.js +47 -0
- package/dist/middleware/post/output-size-limiter.js.map +1 -0
- package/dist/middleware/post/output-summarizer.d.ts +15 -0
- package/dist/middleware/post/output-summarizer.d.ts.map +1 -0
- package/dist/middleware/post/output-summarizer.js +38 -0
- package/dist/middleware/post/output-summarizer.js.map +1 -0
- package/dist/middleware/post/strip-query-params.d.ts +3 -0
- package/dist/middleware/post/strip-query-params.d.ts.map +1 -0
- package/dist/middleware/post/strip-query-params.js +22 -0
- package/dist/middleware/post/strip-query-params.js.map +1 -0
- package/dist/middleware/post/untrusted-envelope.d.ts +3 -0
- package/dist/middleware/post/untrusted-envelope.d.ts.map +1 -0
- package/dist/middleware/post/untrusted-envelope.js +10 -0
- package/dist/middleware/post/untrusted-envelope.js.map +1 -0
- package/dist/middleware/types.d.ts +32 -0
- package/dist/middleware/types.d.ts.map +1 -0
- package/dist/middleware/types.js +2 -0
- package/dist/middleware/types.js.map +1 -0
- package/dist/pool/http-client.d.ts +26 -0
- package/dist/pool/http-client.d.ts.map +1 -0
- package/dist/pool/http-client.js +109 -0
- package/dist/pool/http-client.js.map +1 -0
- package/dist/pool/oauth-provider.d.ts +34 -0
- package/dist/pool/oauth-provider.d.ts.map +1 -0
- package/dist/pool/oauth-provider.js +135 -0
- package/dist/pool/oauth-provider.js.map +1 -0
- package/dist/pool/pool.d.ts +30 -0
- package/dist/pool/pool.d.ts.map +1 -0
- package/dist/pool/pool.js +119 -0
- package/dist/pool/pool.js.map +1 -0
- package/dist/pool/required-mcps.d.ts +7 -0
- package/dist/pool/required-mcps.d.ts.map +1 -0
- package/dist/pool/required-mcps.js +18 -0
- package/dist/pool/required-mcps.js.map +1 -0
- package/dist/pool/sse-client.d.ts +22 -0
- package/dist/pool/sse-client.d.ts.map +1 -0
- package/dist/pool/sse-client.js +70 -0
- package/dist/pool/sse-client.js.map +1 -0
- package/dist/pool/stdio-client.d.ts +24 -0
- package/dist/pool/stdio-client.d.ts.map +1 -0
- package/dist/pool/stdio-client.js +77 -0
- package/dist/pool/stdio-client.js.map +1 -0
- package/dist/registry/registry.d.ts +19 -0
- package/dist/registry/registry.d.ts.map +1 -0
- package/dist/registry/registry.js +85 -0
- package/dist/registry/registry.js.map +1 -0
- package/dist/registry/sanitizer.d.ts +2 -0
- package/dist/registry/sanitizer.d.ts.map +1 -0
- package/dist/registry/sanitizer.js +31 -0
- package/dist/registry/sanitizer.js.map +1 -0
- package/dist/security/blocked-hosts.d.ts +6 -0
- package/dist/security/blocked-hosts.d.ts.map +1 -0
- package/dist/security/blocked-hosts.js +26 -0
- package/dist/security/blocked-hosts.js.map +1 -0
- package/dist/security/domain-allowlist.d.ts +7 -0
- package/dist/security/domain-allowlist.d.ts.map +1 -0
- package/dist/security/domain-allowlist.js +19 -0
- package/dist/security/domain-allowlist.js.map +1 -0
- package/dist/stdio-mode.d.ts +3 -0
- package/dist/stdio-mode.d.ts.map +1 -0
- package/dist/stdio-mode.js +130 -0
- package/dist/stdio-mode.js.map +1 -0
- package/dist/tools/exec.d.ts +20 -0
- package/dist/tools/exec.d.ts.map +1 -0
- package/dist/tools/exec.js +105 -0
- package/dist/tools/exec.js.map +1 -0
- package/dist/tools/http.d.ts +13 -0
- package/dist/tools/http.d.ts.map +1 -0
- package/dist/tools/http.js +99 -0
- package/dist/tools/http.js.map +1 -0
- package/dist/transport/agent-server.d.ts +26 -0
- package/dist/transport/agent-server.d.ts.map +1 -0
- package/dist/transport/agent-server.js +55 -0
- package/dist/transport/agent-server.js.map +1 -0
- package/dist/transport/mcp-normalizer.d.ts +9 -0
- package/dist/transport/mcp-normalizer.d.ts.map +1 -0
- package/dist/transport/mcp-normalizer.js +12 -0
- package/dist/transport/mcp-normalizer.js.map +1 -0
- package/dist/transport/sse-server.d.ts +7 -0
- package/dist/transport/sse-server.d.ts.map +1 -0
- package/dist/transport/sse-server.js +94 -0
- package/dist/transport/sse-server.js.map +1 -0
- package/dist/transport/stdio-server.d.ts +3 -0
- package/dist/transport/stdio-server.d.ts.map +1 -0
- package/dist/transport/stdio-server.js +12 -0
- package/dist/transport/stdio-server.js.map +1 -0
- package/dist/types.d.ts +15 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/dist/util/id.d.ts +5 -0
- package/dist/util/id.d.ts.map +1 -0
- package/dist/util/id.js +16 -0
- package/dist/util/id.js.map +1 -0
- package/dist/util/logger.d.ts +4 -0
- package/dist/util/logger.d.ts.map +1 -0
- package/dist/util/logger.js +24 -0
- package/dist/util/logger.js.map +1 -0
- package/dist/version.d.ts +2 -0
- package/dist/version.d.ts.map +1 -0
- package/dist/version.js +4 -0
- package/dist/version.js.map +1 -0
- package/examples/claude-code-setup.md +77 -0
- package/examples/gateway.yaml +118 -0
- package/examples/local-dev.yaml +41 -0
- package/examples/openclaw-setup.md +52 -0
- package/examples/profiles.yaml +103 -0
- package/package.json +80 -3
- package/schema.json +943 -0
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { execFile } from 'child_process';
|
|
2
|
+
import { childLogger } from '../../util/logger.js';
|
|
3
|
+
const log = childLogger('hitl-macos');
|
|
4
|
+
export class MacosHitlProvider {
|
|
5
|
+
approvalApi;
|
|
6
|
+
constructor(approvalApi) {
|
|
7
|
+
this.approvalApi = approvalApi;
|
|
8
|
+
}
|
|
9
|
+
init() {
|
|
10
|
+
log.info('macOS dialog HITL provider ready');
|
|
11
|
+
return Promise.resolve();
|
|
12
|
+
}
|
|
13
|
+
stop() {
|
|
14
|
+
return Promise.resolve();
|
|
15
|
+
}
|
|
16
|
+
async notify(requests) {
|
|
17
|
+
// Show each request as a separate dialog (in parallel)
|
|
18
|
+
await Promise.all(requests.map((req) => this.showDialog(req)));
|
|
19
|
+
}
|
|
20
|
+
showDialog(req) {
|
|
21
|
+
const argSummary = Object.entries(req.args)
|
|
22
|
+
.map(([k, v]) => {
|
|
23
|
+
const val = typeof v === 'string'
|
|
24
|
+
? v.length > 100
|
|
25
|
+
? `${v.slice(0, 100)}...`
|
|
26
|
+
: v
|
|
27
|
+
: JSON.stringify(v);
|
|
28
|
+
return ` ${k}: ${val}`;
|
|
29
|
+
})
|
|
30
|
+
.join('\n');
|
|
31
|
+
const timeoutSec = Math.ceil(req.timeoutMs / 1000);
|
|
32
|
+
const message = `Agent: ${req.agentId}\\nTool: ${req.tool}\\n\\n${argSummary}\\n\\nCode: ${req.code}`;
|
|
33
|
+
return new Promise((resolve) => {
|
|
34
|
+
execFile('osascript', [
|
|
35
|
+
'-e',
|
|
36
|
+
`display dialog "${message}" with title "Airlock Approval" buttons {"Deny", "Approve"} default button "Approve" with icon caution giving up after ${timeoutSec}`,
|
|
37
|
+
], (err, stdout) => {
|
|
38
|
+
if (err) {
|
|
39
|
+
// User closed the dialog or hit cancel — treat as deny
|
|
40
|
+
log.info({ code: req.code }, 'macOS dialog dismissed, treating as deny');
|
|
41
|
+
this.approvalApi.deny(req.code, 'Dialog dismissed');
|
|
42
|
+
resolve();
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
const output = stdout.trim();
|
|
46
|
+
if (output.includes('gave up:true')) {
|
|
47
|
+
log.info({ code: req.code }, 'macOS dialog timed out');
|
|
48
|
+
// Let the engine's own timeout handle it
|
|
49
|
+
resolve();
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
if (output.includes('Approve')) {
|
|
53
|
+
log.info({ code: req.code }, 'Approved via macOS dialog');
|
|
54
|
+
this.approvalApi.approve(req.code);
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
log.info({ code: req.code }, 'Denied via macOS dialog');
|
|
58
|
+
this.approvalApi.deny(req.code, 'Denied via dialog');
|
|
59
|
+
}
|
|
60
|
+
resolve();
|
|
61
|
+
});
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
//# sourceMappingURL=macos.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"macos.js","sourceRoot":"","sources":["../../../src/hitl/providers/macos.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import type { HitlProvider, HitlNotification, ApprovalApi } from './types.js';
|
|
2
|
+
interface OpenClawConfig {
|
|
3
|
+
gateway_url: string;
|
|
4
|
+
token: string;
|
|
5
|
+
session_key: string;
|
|
6
|
+
}
|
|
7
|
+
export declare class OpenClawHitlProvider implements HitlProvider {
|
|
8
|
+
private config;
|
|
9
|
+
private approvalApi;
|
|
10
|
+
private ws?;
|
|
11
|
+
private reconnectTimer?;
|
|
12
|
+
private stopped;
|
|
13
|
+
constructor(config: OpenClawConfig, approvalApi: ApprovalApi);
|
|
14
|
+
init(): Promise<void>;
|
|
15
|
+
private connect;
|
|
16
|
+
private handleMessage;
|
|
17
|
+
notify(requests: HitlNotification[]): Promise<void>;
|
|
18
|
+
stop(): Promise<void>;
|
|
19
|
+
}
|
|
20
|
+
export {};
|
|
21
|
+
//# sourceMappingURL=openclaw.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"openclaw.d.ts","sourceRoot":"","sources":["../../../src/hitl/providers/openclaw.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,YAAY,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAK9E,UAAU,cAAc;IACtB,WAAW,EAAE,MAAM,CAAC;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,qBAAa,oBAAqB,YAAW,YAAY;IAMrD,OAAO,CAAC,MAAM;IACd,OAAO,CAAC,WAAW;IANrB,OAAO,CAAC,EAAE,CAAC,CAAiC;IAC5C,OAAO,CAAC,cAAc,CAAC,CAAiB;IACxC,OAAO,CAAC,OAAO,CAAS;gBAGd,MAAM,EAAE,cAAc,EACtB,WAAW,EAAE,WAAW;IAG5B,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAI3B,OAAO,CAAC,OAAO;IAsCf,OAAO,CAAC,aAAa;IAgBrB,MAAM,CAAC,QAAQ,EAAE,gBAAgB,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAsBnD,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;CAMtB"}
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import { WebSocket } from 'ws';
|
|
2
|
+
import { formatBatch } from '../formatter.js';
|
|
3
|
+
import { parseApprovalCommand } from '../parser.js';
|
|
4
|
+
import { generateId } from '../../util/id.js';
|
|
5
|
+
import { childLogger } from '../../util/logger.js';
|
|
6
|
+
const log = childLogger('hitl-openclaw');
|
|
7
|
+
export class OpenClawHitlProvider {
|
|
8
|
+
config;
|
|
9
|
+
approvalApi;
|
|
10
|
+
ws;
|
|
11
|
+
reconnectTimer;
|
|
12
|
+
stopped = false;
|
|
13
|
+
constructor(config, approvalApi) {
|
|
14
|
+
this.config = config;
|
|
15
|
+
this.approvalApi = approvalApi;
|
|
16
|
+
}
|
|
17
|
+
async init() {
|
|
18
|
+
await this.connect();
|
|
19
|
+
}
|
|
20
|
+
connect() {
|
|
21
|
+
return new Promise((resolve, reject) => {
|
|
22
|
+
const ws = new WebSocket(this.config.gateway_url, {
|
|
23
|
+
headers: { Authorization: `Bearer ${this.config.token}` },
|
|
24
|
+
});
|
|
25
|
+
ws.once('open', () => {
|
|
26
|
+
log.info({ url: this.config.gateway_url }, 'OpenClaw WS connected');
|
|
27
|
+
this.ws = ws;
|
|
28
|
+
resolve();
|
|
29
|
+
});
|
|
30
|
+
ws.once('error', (err) => {
|
|
31
|
+
log.error({ err }, 'OpenClaw WS connection error');
|
|
32
|
+
reject(err);
|
|
33
|
+
});
|
|
34
|
+
ws.on('message', (data) => {
|
|
35
|
+
try {
|
|
36
|
+
const msg = JSON.parse(data.toString());
|
|
37
|
+
this.handleMessage(msg);
|
|
38
|
+
}
|
|
39
|
+
catch {
|
|
40
|
+
// ignore non-JSON
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
ws.on('close', () => {
|
|
44
|
+
if (!this.stopped) {
|
|
45
|
+
log.warn('OpenClaw WS disconnected, reconnecting in 5s');
|
|
46
|
+
this.reconnectTimer = setTimeout(() => {
|
|
47
|
+
void this.connect().catch(() => { });
|
|
48
|
+
}, 5000);
|
|
49
|
+
this.reconnectTimer.unref();
|
|
50
|
+
}
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
handleMessage(msg) {
|
|
55
|
+
const text = extractText(msg);
|
|
56
|
+
if (!text)
|
|
57
|
+
return;
|
|
58
|
+
const parsed = parseApprovalCommand(text);
|
|
59
|
+
if (!parsed)
|
|
60
|
+
return;
|
|
61
|
+
if (parsed.type === 'approve') {
|
|
62
|
+
log.info({ code: parsed.code }, 'Approve via OpenClaw');
|
|
63
|
+
this.approvalApi.approve(parsed.code);
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
log.info({ code: parsed.code }, 'Deny via OpenClaw');
|
|
67
|
+
this.approvalApi.deny(parsed.code, parsed.reason);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
notify(requests) {
|
|
71
|
+
if (!this.ws || this.ws.readyState !== WebSocket.OPEN) {
|
|
72
|
+
log.error('OpenClaw WS not connected, cannot send HITL notification');
|
|
73
|
+
return Promise.resolve();
|
|
74
|
+
}
|
|
75
|
+
const message = formatBatch(requests);
|
|
76
|
+
const rpc = {
|
|
77
|
+
jsonrpc: '2.0',
|
|
78
|
+
id: generateId(),
|
|
79
|
+
method: 'chat.send',
|
|
80
|
+
params: {
|
|
81
|
+
sessionKey: this.config.session_key,
|
|
82
|
+
message,
|
|
83
|
+
idempotencyKey: generateId(),
|
|
84
|
+
},
|
|
85
|
+
};
|
|
86
|
+
this.ws.send(JSON.stringify(rpc));
|
|
87
|
+
return Promise.resolve();
|
|
88
|
+
}
|
|
89
|
+
stop() {
|
|
90
|
+
this.stopped = true;
|
|
91
|
+
clearTimeout(this.reconnectTimer);
|
|
92
|
+
this.ws?.close();
|
|
93
|
+
return Promise.resolve();
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
function extractText(msg) {
|
|
97
|
+
// Try common message shapes
|
|
98
|
+
const text = msg['text'] ?? msg['content'] ?? msg['message'] ?? msg['params'];
|
|
99
|
+
if (typeof text === 'string')
|
|
100
|
+
return text;
|
|
101
|
+
if (typeof text === 'object' && text !== null) {
|
|
102
|
+
return extractText(text);
|
|
103
|
+
}
|
|
104
|
+
return null;
|
|
105
|
+
}
|
|
106
|
+
//# sourceMappingURL=openclaw.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"openclaw.js","sourceRoot":"","sources":["../../../src/hitl/providers/openclaw.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import type { HitlProvider, HitlNotification } from './types.js';
|
|
2
|
+
export interface SlackHitlConfig {
|
|
3
|
+
webhook_url: string;
|
|
4
|
+
}
|
|
5
|
+
export declare class SlackHitlProvider implements HitlProvider {
|
|
6
|
+
private cfg;
|
|
7
|
+
constructor(cfg: SlackHitlConfig);
|
|
8
|
+
init(): Promise<void>;
|
|
9
|
+
stop(): Promise<void>;
|
|
10
|
+
notify(requests: HitlNotification[]): Promise<void>;
|
|
11
|
+
}
|
|
12
|
+
//# sourceMappingURL=slack.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"slack.d.ts","sourceRoot":"","sources":["../../../src/hitl/providers/slack.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAIjE,MAAM,WAAW,eAAe;IAC9B,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,qBAAa,iBAAkB,YAAW,YAAY;IACxC,OAAO,CAAC,GAAG;gBAAH,GAAG,EAAE,eAAe;IAElC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAErB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAErB,MAAM,CAAC,QAAQ,EAAE,gBAAgB,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;CAc1D"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { formatBatch } from '../formatter.js';
|
|
2
|
+
import { childLogger } from '../../util/logger.js';
|
|
3
|
+
const log = childLogger('hitl-slack');
|
|
4
|
+
export class SlackHitlProvider {
|
|
5
|
+
cfg;
|
|
6
|
+
constructor(cfg) {
|
|
7
|
+
this.cfg = cfg;
|
|
8
|
+
}
|
|
9
|
+
async init() { }
|
|
10
|
+
async stop() { }
|
|
11
|
+
async notify(requests) {
|
|
12
|
+
const text = formatBatch(requests);
|
|
13
|
+
const body = JSON.stringify({ text });
|
|
14
|
+
const res = await fetch(this.cfg.webhook_url, {
|
|
15
|
+
method: 'POST',
|
|
16
|
+
headers: { 'Content-Type': 'application/json' },
|
|
17
|
+
body,
|
|
18
|
+
});
|
|
19
|
+
if (!res.ok) {
|
|
20
|
+
log.warn({ status: res.status }, 'Slack webhook returned non-2xx');
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=slack.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"slack.js","sourceRoot":"","sources":["../../../src/hitl/providers/slack.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAGnD,MAAM,GAAG,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;AAMtC,MAAM,OAAO,iBAAiB;IACR;IAApB,YAAoB,GAAoB;QAApB,QAAG,GAAH,GAAG,CAAiB;IAAG,CAAC;IAE5C,KAAK,CAAC,IAAI,KAAmB,CAAC;IAE9B,KAAK,CAAC,IAAI,KAAmB,CAAC;IAE9B,KAAK,CAAC,MAAM,CAAC,QAA4B;QACvC,MAAM,IAAI,GAAG,WAAW,CAAC,QAAQ,CAAC,CAAC;QACnC,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QAEtC,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE;YAC5C,MAAM,EAAE,MAAM;YACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;YAC/C,IAAI;SACL,CAAC,CAAC;QAEH,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;YACZ,GAAG,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,MAAM,EAAE,EAAE,gCAAgC,CAAC,CAAC;QACrE,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import type { Readable } from 'stream';
|
|
2
|
+
import type { HitlProvider, HitlNotification, ApprovalApi } from './types.js';
|
|
3
|
+
export declare class StdioHitlProvider implements HitlProvider {
|
|
4
|
+
private approvalApi;
|
|
5
|
+
private inputStream;
|
|
6
|
+
private rl?;
|
|
7
|
+
constructor(approvalApi: ApprovalApi, inputStream?: Readable);
|
|
8
|
+
init(): Promise<void>;
|
|
9
|
+
notify(requests: HitlNotification[]): Promise<void>;
|
|
10
|
+
stop(): Promise<void>;
|
|
11
|
+
}
|
|
12
|
+
//# sourceMappingURL=stdio.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"stdio.d.ts","sourceRoot":"","sources":["../../../src/hitl/providers/stdio.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,QAAQ,CAAC;AAGvC,OAAO,KAAK,EAAE,YAAY,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAK9E,qBAAa,iBAAkB,YAAW,YAAY;IAIlD,OAAO,CAAC,WAAW;IACnB,OAAO,CAAC,WAAW;IAJrB,OAAO,CAAC,EAAE,CAAC,CAAqB;gBAGtB,WAAW,EAAE,WAAW,EACxB,WAAW,GAAE,QAA+C;IAGtE,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAkBrB,MAAM,CAAC,QAAQ,EAAE,gBAAgB,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAKnD,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;CAItB"}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import * as readline from 'readline';
|
|
2
|
+
import { formatBatch } from '../formatter.js';
|
|
3
|
+
import { parseApprovalCommand } from '../parser.js';
|
|
4
|
+
import { childLogger } from '../../util/logger.js';
|
|
5
|
+
const log = childLogger('hitl-stdio');
|
|
6
|
+
export class StdioHitlProvider {
|
|
7
|
+
approvalApi;
|
|
8
|
+
inputStream;
|
|
9
|
+
rl;
|
|
10
|
+
constructor(approvalApi, inputStream = process.stdin) {
|
|
11
|
+
this.approvalApi = approvalApi;
|
|
12
|
+
this.inputStream = inputStream;
|
|
13
|
+
}
|
|
14
|
+
init() {
|
|
15
|
+
this.rl = readline.createInterface({ input: this.inputStream, terminal: false });
|
|
16
|
+
this.rl.on('line', (line) => {
|
|
17
|
+
const parsed = parseApprovalCommand(line);
|
|
18
|
+
if (!parsed)
|
|
19
|
+
return;
|
|
20
|
+
if (parsed.type === 'approve') {
|
|
21
|
+
log.info({ code: parsed.code }, 'Approve received via stdio');
|
|
22
|
+
this.approvalApi.approve(parsed.code);
|
|
23
|
+
}
|
|
24
|
+
else {
|
|
25
|
+
log.info({ code: parsed.code }, 'Deny received via stdio');
|
|
26
|
+
this.approvalApi.deny(parsed.code, parsed.reason);
|
|
27
|
+
}
|
|
28
|
+
});
|
|
29
|
+
log.info('Stdio HITL provider ready. Type: approve <CODE> or deny <CODE>');
|
|
30
|
+
return Promise.resolve();
|
|
31
|
+
}
|
|
32
|
+
notify(requests) {
|
|
33
|
+
process.stderr.write('\n' + formatBatch(requests) + '\n\n');
|
|
34
|
+
return Promise.resolve();
|
|
35
|
+
}
|
|
36
|
+
stop() {
|
|
37
|
+
this.rl?.close();
|
|
38
|
+
return Promise.resolve();
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=stdio.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"stdio.js","sourceRoot":"","sources":["../../../src/hitl/providers/stdio.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,QAAQ,MAAM,UAAU,CAAC;AAErC,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAE,oBAAoB,EAAE,MAAM,cAAc,CAAC;AAEpD,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAEnD,MAAM,GAAG,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;AAEtC,MAAM,OAAO,iBAAiB;IAIlB;IACA;IAJF,EAAE,CAAsB;IAEhC,YACU,WAAwB,EACxB,cAAwB,OAAO,CAAC,KAA4B;QAD5D,gBAAW,GAAX,WAAW,CAAa;QACxB,gBAAW,GAAX,WAAW,CAAiD;IACnE,CAAC;IAEJ,IAAI;QACF,IAAI,CAAC,EAAE,GAAG,QAAQ,CAAC,eAAe,CAAC,EAAE,KAAK,EAAE,IAAI,CAAC,WAAW,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC,CAAC;QACjF,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;YAC1B,MAAM,MAAM,GAAG,oBAAoB,CAAC,IAAI,CAAC,CAAC;YAC1C,IAAI,CAAC,MAAM;gBAAE,OAAO;YAEpB,IAAI,MAAM,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;gBAC9B,GAAG,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,EAAE,4BAA4B,CAAC,CAAC;gBAC9D,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YACxC,CAAC;iBAAM,CAAC;gBACN,GAAG,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,EAAE,yBAAyB,CAAC,CAAC;gBAC3D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;YACpD,CAAC;QACH,CAAC,CAAC,CAAC;QACH,GAAG,CAAC,IAAI,CAAC,gEAAgE,CAAC,CAAC;QAC3E,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;IAC3B,CAAC;IAED,MAAM,CAAC,QAA4B;QACjC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,GAAG,WAAW,CAAC,QAAQ,CAAC,GAAG,MAAM,CAAC,CAAC;QAC5D,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;IAC3B,CAAC;IAED,IAAI;QACF,IAAI,CAAC,EAAE,EAAE,KAAK,EAAE,CAAC;QACjB,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;IAC3B,CAAC;CACF"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import type { HitlProvider, HitlNotification, ApprovalApi } from './types.js';
|
|
2
|
+
interface TelegramConfig {
|
|
3
|
+
bot_token: string;
|
|
4
|
+
chat_id: string;
|
|
5
|
+
}
|
|
6
|
+
export declare class TelegramHitlProvider implements HitlProvider {
|
|
7
|
+
private config;
|
|
8
|
+
private approvalApi;
|
|
9
|
+
private polling;
|
|
10
|
+
private lastUpdateId;
|
|
11
|
+
private pollTimer?;
|
|
12
|
+
constructor(config: TelegramConfig, approvalApi: ApprovalApi);
|
|
13
|
+
private get apiBase();
|
|
14
|
+
init(): Promise<void>;
|
|
15
|
+
notify(requests: HitlNotification[]): Promise<void>;
|
|
16
|
+
private schedulePoll;
|
|
17
|
+
private poll;
|
|
18
|
+
private handleUpdate;
|
|
19
|
+
stop(): Promise<void>;
|
|
20
|
+
}
|
|
21
|
+
export {};
|
|
22
|
+
//# sourceMappingURL=telegram.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"telegram.d.ts","sourceRoot":"","sources":["../../../src/hitl/providers/telegram.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,YAAY,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAK9E,UAAU,cAAc;IACtB,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,qBAAa,oBAAqB,YAAW,YAAY;IAMrD,OAAO,CAAC,MAAM;IACd,OAAO,CAAC,WAAW;IANrB,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,YAAY,CAAK;IACzB,OAAO,CAAC,SAAS,CAAC,CAAiB;gBAGzB,MAAM,EAAE,cAAc,EACtB,WAAW,EAAE,WAAW;IAGlC,OAAO,KAAK,OAAO,GAElB;IAED,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAOf,MAAM,CAAC,QAAQ,EAAE,gBAAgB,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAYzD,OAAO,CAAC,YAAY;YAQN,IAAI;IAoBlB,OAAO,CAAC,YAAY;IAyBpB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;CAKtB"}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import { formatBatch } from '../formatter.js';
|
|
2
|
+
import { parseApprovalCommand } from '../parser.js';
|
|
3
|
+
import { childLogger } from '../../util/logger.js';
|
|
4
|
+
const log = childLogger('hitl-telegram');
|
|
5
|
+
export class TelegramHitlProvider {
|
|
6
|
+
config;
|
|
7
|
+
approvalApi;
|
|
8
|
+
polling = false;
|
|
9
|
+
lastUpdateId = 0;
|
|
10
|
+
pollTimer;
|
|
11
|
+
constructor(config, approvalApi) {
|
|
12
|
+
this.config = config;
|
|
13
|
+
this.approvalApi = approvalApi;
|
|
14
|
+
}
|
|
15
|
+
get apiBase() {
|
|
16
|
+
return `https://api.telegram.org/bot${this.config.bot_token}`;
|
|
17
|
+
}
|
|
18
|
+
init() {
|
|
19
|
+
this.polling = true;
|
|
20
|
+
this.schedulePoll();
|
|
21
|
+
log.info({ chat_id: this.config.chat_id }, 'Telegram HITL provider started');
|
|
22
|
+
return Promise.resolve();
|
|
23
|
+
}
|
|
24
|
+
async notify(requests) {
|
|
25
|
+
const text = formatBatch(requests);
|
|
26
|
+
const res = await fetch(`${this.apiBase}/sendMessage`, {
|
|
27
|
+
method: 'POST',
|
|
28
|
+
headers: { 'Content-Type': 'application/json' },
|
|
29
|
+
body: JSON.stringify({ chat_id: this.config.chat_id, text, parse_mode: 'Markdown' }),
|
|
30
|
+
});
|
|
31
|
+
if (!res.ok) {
|
|
32
|
+
log.error({ status: res.status }, 'Failed to send Telegram notification');
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
schedulePoll() {
|
|
36
|
+
if (!this.polling)
|
|
37
|
+
return;
|
|
38
|
+
this.pollTimer = setTimeout(() => {
|
|
39
|
+
void this.poll();
|
|
40
|
+
}, 1000);
|
|
41
|
+
this.pollTimer.unref();
|
|
42
|
+
}
|
|
43
|
+
async poll() {
|
|
44
|
+
try {
|
|
45
|
+
const res = await fetch(`${this.apiBase}/getUpdates?offset=${this.lastUpdateId + 1}&timeout=25&allowed_updates=["message"]`);
|
|
46
|
+
if (!res.ok)
|
|
47
|
+
throw new Error(`getUpdates failed: ${res.status}`);
|
|
48
|
+
const data = (await res.json());
|
|
49
|
+
if (data.ok && data.result.length > 0) {
|
|
50
|
+
for (const update of data.result) {
|
|
51
|
+
this.lastUpdateId = Math.max(this.lastUpdateId, update.update_id);
|
|
52
|
+
this.handleUpdate(update);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
catch (err) {
|
|
57
|
+
log.warn({ err }, 'Telegram poll error');
|
|
58
|
+
}
|
|
59
|
+
this.schedulePoll();
|
|
60
|
+
}
|
|
61
|
+
handleUpdate(update) {
|
|
62
|
+
const chatId = update.message?.chat?.id;
|
|
63
|
+
const text = update.message?.text ?? '';
|
|
64
|
+
// Only process messages from the configured chat
|
|
65
|
+
if (!chatId || String(chatId) !== this.config.chat_id) {
|
|
66
|
+
log.warn({ chatId, expected: this.config.chat_id }, 'Ignoring message from unauthorized chat');
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
const parsed = parseApprovalCommand(text);
|
|
70
|
+
if (!parsed)
|
|
71
|
+
return;
|
|
72
|
+
if (parsed.type === 'approve') {
|
|
73
|
+
log.info({ code: parsed.code }, 'Approve via Telegram');
|
|
74
|
+
this.approvalApi.approve(parsed.code);
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
log.info({ code: parsed.code }, 'Deny via Telegram');
|
|
78
|
+
this.approvalApi.deny(parsed.code, parsed.reason);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
stop() {
|
|
82
|
+
this.polling = false;
|
|
83
|
+
clearTimeout(this.pollTimer);
|
|
84
|
+
return Promise.resolve();
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
//# sourceMappingURL=telegram.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"telegram.js","sourceRoot":"","sources":["../../../src/hitl/providers/telegram.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import type { HitlProvider, HitlNotification, ApprovalApi } from './types.js';
|
|
2
|
+
export declare class TuiHitlProvider implements HitlProvider {
|
|
3
|
+
private approvalApi;
|
|
4
|
+
private tty;
|
|
5
|
+
private items;
|
|
6
|
+
private selected;
|
|
7
|
+
private active;
|
|
8
|
+
constructor(approvalApi: ApprovalApi);
|
|
9
|
+
init(): Promise<void>;
|
|
10
|
+
stop(): Promise<void>;
|
|
11
|
+
notify(requests: HitlNotification[]): Promise<void>;
|
|
12
|
+
private handleKey;
|
|
13
|
+
private render;
|
|
14
|
+
private formatArgs;
|
|
15
|
+
}
|
|
16
|
+
//# sourceMappingURL=tui.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tui.d.ts","sourceRoot":"","sources":["../../../src/hitl/providers/tui.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,YAAY,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAmB9E,qBAAa,eAAgB,YAAW,YAAY;IAMtC,OAAO,CAAC,WAAW;IAL/B,OAAO,CAAC,GAAG,CAAsC;IACjD,OAAO,CAAC,KAAK,CAAqB;IAClC,OAAO,CAAC,QAAQ,CAAK;IACrB,OAAO,CAAC,MAAM,CAAS;gBAEH,WAAW,EAAE,WAAW;IAKtC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAgC3B,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAUrB,MAAM,CAAC,QAAQ,EAAE,gBAAgB,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAQnD,OAAO,CAAC,SAAS;IAoDjB,OAAO,CAAC,MAAM;IAgDd,OAAO,CAAC,UAAU;CAanB"}
|
|
@@ -0,0 +1,169 @@
|
|
|
1
|
+
import { createReadStream } from 'fs';
|
|
2
|
+
import { childLogger } from '../../util/logger.js';
|
|
3
|
+
const log = childLogger('hitl-tui');
|
|
4
|
+
// ANSI helpers
|
|
5
|
+
const ESC = '\x1b[';
|
|
6
|
+
const BOLD = `${ESC}1m`;
|
|
7
|
+
const DIM = `${ESC}2m`;
|
|
8
|
+
const RESET = `${ESC}0m`;
|
|
9
|
+
const GREEN = `${ESC}32m`;
|
|
10
|
+
const RED = `${ESC}31m`;
|
|
11
|
+
const YELLOW = `${ESC}33m`;
|
|
12
|
+
const CYAN = `${ESC}36m`;
|
|
13
|
+
export class TuiHitlProvider {
|
|
14
|
+
approvalApi;
|
|
15
|
+
tty;
|
|
16
|
+
items = [];
|
|
17
|
+
selected = 0;
|
|
18
|
+
active = false;
|
|
19
|
+
constructor(approvalApi) {
|
|
20
|
+
this.approvalApi = approvalApi;
|
|
21
|
+
// Read keystrokes from /dev/tty — doesn't touch stdin/stdout
|
|
22
|
+
this.tty = createReadStream('/dev/tty', { encoding: 'utf8' });
|
|
23
|
+
}
|
|
24
|
+
async init() {
|
|
25
|
+
this.active = true;
|
|
26
|
+
// Put tty in raw mode to get individual keypresses
|
|
27
|
+
const fd = this.tty.fd;
|
|
28
|
+
if (typeof fd === 'number') {
|
|
29
|
+
try {
|
|
30
|
+
const tty = await import('tty');
|
|
31
|
+
if (tty.isatty(fd)) {
|
|
32
|
+
// We need the actual tty fd for raw mode
|
|
33
|
+
const { openSync } = await import('fs');
|
|
34
|
+
const ttyFd = openSync('/dev/tty', 'r');
|
|
35
|
+
const ttyReadStream = new tty.ReadStream(ttyFd);
|
|
36
|
+
ttyReadStream.setRawMode(true);
|
|
37
|
+
ttyReadStream.setEncoding('utf8');
|
|
38
|
+
ttyReadStream.on('data', (key) => this.handleKey(key));
|
|
39
|
+
ttyReadStream.resume();
|
|
40
|
+
this.tty = ttyReadStream;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
catch {
|
|
44
|
+
// Fallback: line-based input
|
|
45
|
+
this.tty.setEncoding('utf8');
|
|
46
|
+
this.tty.on('data', (chunk) => {
|
|
47
|
+
for (const ch of String(chunk))
|
|
48
|
+
this.handleKey(ch);
|
|
49
|
+
});
|
|
50
|
+
this.tty.resume();
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
log.info('TUI HITL provider ready — approve/deny from your terminal');
|
|
54
|
+
}
|
|
55
|
+
stop() {
|
|
56
|
+
this.active = false;
|
|
57
|
+
try {
|
|
58
|
+
this.tty.destroy?.();
|
|
59
|
+
}
|
|
60
|
+
catch {
|
|
61
|
+
/* swallow */
|
|
62
|
+
}
|
|
63
|
+
return Promise.resolve();
|
|
64
|
+
}
|
|
65
|
+
notify(requests) {
|
|
66
|
+
for (const req of requests) {
|
|
67
|
+
this.items.push({ req, status: 'pending' });
|
|
68
|
+
}
|
|
69
|
+
this.render();
|
|
70
|
+
return Promise.resolve();
|
|
71
|
+
}
|
|
72
|
+
handleKey(key) {
|
|
73
|
+
if (!this.active)
|
|
74
|
+
return;
|
|
75
|
+
const pending = this.items.filter((i) => i.status === 'pending');
|
|
76
|
+
if (pending.length === 0)
|
|
77
|
+
return;
|
|
78
|
+
// Ctrl-C
|
|
79
|
+
if (key === '\x03') {
|
|
80
|
+
process.exit(0);
|
|
81
|
+
}
|
|
82
|
+
// j / down arrow
|
|
83
|
+
if (key === 'j' || key === `${ESC}B`) {
|
|
84
|
+
this.selected = Math.min(this.selected + 1, pending.length - 1);
|
|
85
|
+
this.render();
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
// k / up arrow
|
|
89
|
+
if (key === 'k' || key === `${ESC}A`) {
|
|
90
|
+
this.selected = Math.max(this.selected - 1, 0);
|
|
91
|
+
this.render();
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
// a = approve
|
|
95
|
+
if (key === 'a' || key === 'A') {
|
|
96
|
+
const item = pending[this.selected];
|
|
97
|
+
if (item) {
|
|
98
|
+
item.status = 'approved';
|
|
99
|
+
this.approvalApi.approve(item.req.code);
|
|
100
|
+
log.info({ code: item.req.code }, 'Approved via TUI');
|
|
101
|
+
this.selected = Math.min(this.selected, Math.max(pending.length - 2, 0));
|
|
102
|
+
this.render();
|
|
103
|
+
}
|
|
104
|
+
return;
|
|
105
|
+
}
|
|
106
|
+
// d = deny
|
|
107
|
+
if (key === 'd' || key === 'D') {
|
|
108
|
+
const item = pending[this.selected];
|
|
109
|
+
if (item) {
|
|
110
|
+
item.status = 'denied';
|
|
111
|
+
this.approvalApi.deny(item.req.code, 'Denied via TUI');
|
|
112
|
+
log.info({ code: item.req.code }, 'Denied via TUI');
|
|
113
|
+
this.selected = Math.min(this.selected, Math.max(pending.length - 2, 0));
|
|
114
|
+
this.render();
|
|
115
|
+
}
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
render() {
|
|
120
|
+
const out = process.stderr;
|
|
121
|
+
const pending = this.items.filter((i) => i.status === 'pending');
|
|
122
|
+
const resolved = this.items.filter((i) => i.status !== 'pending');
|
|
123
|
+
const lines = [];
|
|
124
|
+
lines.push('');
|
|
125
|
+
lines.push(`${BOLD}${CYAN} Airlock Approvals${RESET} ${DIM}[a]pprove [d]eny [j/k] navigate${RESET}`);
|
|
126
|
+
lines.push(`${DIM} ${'─'.repeat(60)}${RESET}`);
|
|
127
|
+
if (pending.length === 0) {
|
|
128
|
+
lines.push(`${DIM} No pending requests${RESET}`);
|
|
129
|
+
}
|
|
130
|
+
else {
|
|
131
|
+
for (let i = 0; i < pending.length; i++) {
|
|
132
|
+
const item = pending[i];
|
|
133
|
+
const selected = i === this.selected;
|
|
134
|
+
const prefix = selected ? `${BOLD}${YELLOW} ▸ ` : `${DIM} `;
|
|
135
|
+
const args = this.formatArgs(item.req.args);
|
|
136
|
+
lines.push(`${prefix}${selected ? BOLD : ''}${item.req.tool}${RESET}` +
|
|
137
|
+
` ${DIM}agent:${item.req.agentId}${RESET}` +
|
|
138
|
+
` ${DIM}[${item.req.code}]${RESET}`);
|
|
139
|
+
lines.push(`${selected ? ' ' : ' '}${DIM}${args}${RESET}`);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
// Show recently resolved
|
|
143
|
+
const recent = resolved.slice(-3);
|
|
144
|
+
if (recent.length > 0) {
|
|
145
|
+
lines.push(`${DIM} ${'─'.repeat(60)}${RESET}`);
|
|
146
|
+
for (const item of recent) {
|
|
147
|
+
const color = item.status === 'approved' ? GREEN : RED;
|
|
148
|
+
lines.push(`${DIM} ${color}${item.status}${RESET}${DIM} ${item.req.tool} [${item.req.code}]${RESET}`);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
lines.push('');
|
|
152
|
+
// Clear previous output and write new
|
|
153
|
+
out.write(`${ESC}2J${ESC}H`); // clear screen, cursor to top
|
|
154
|
+
out.write(lines.join('\n'));
|
|
155
|
+
}
|
|
156
|
+
formatArgs(args) {
|
|
157
|
+
const parts = Object.entries(args).map(([k, v]) => {
|
|
158
|
+
const val = typeof v === 'string'
|
|
159
|
+
? v.length > 40
|
|
160
|
+
? `"${v.slice(0, 40)}..."`
|
|
161
|
+
: `"${v}"`
|
|
162
|
+
: JSON.stringify(v);
|
|
163
|
+
return `${k}: ${val}`;
|
|
164
|
+
});
|
|
165
|
+
const str = parts.join(', ');
|
|
166
|
+
return str.length > 70 ? str.slice(0, 70) + '...' : str;
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
//# sourceMappingURL=tui.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tui.js","sourceRoot":"","sources":["../../../src/hitl/providers/tui.ts"],"names":[],"mappings":"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"}
|