@rokrokss/claude-slack-channel 0.1.0
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.md +27 -0
- package/README.md +270 -0
- package/bun.lock +266 -0
- package/lib/audit.ts +24 -0
- package/lib/event.ts +30 -0
- package/lib/formatting.ts +73 -0
- package/lib/gate.ts +47 -0
- package/lib/index.ts +7 -0
- package/lib/permalink.ts +8 -0
- package/lib/resilience.ts +45 -0
- package/lib/security.ts +9 -0
- package/package.json +27 -0
- package/server.test.ts +722 -0
- package/server.ts +412 -0
- package/tools.ts +171 -0
- package/tsconfig.json +17 -0
package/CLAUDE.md
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
# Claude Slack Channel 개발 가이드
|
|
2
|
+
|
|
3
|
+
## 프로젝트 구조
|
|
4
|
+
|
|
5
|
+
- `server.ts` — MCP channel 서버 (stateful: Slack 클라이언트, 이벤트 핸들링)
|
|
6
|
+
- `tools.ts` — MCP 도구 등록 (reply, react, delete_bot_message, fetch_dm_thread)
|
|
7
|
+
- `lib/` — 순수 함수 모듈
|
|
8
|
+
- `gate.ts` — 접근 제어 (Access, GateOptions, gate)
|
|
9
|
+
- `security.ts` — 아웃바운드 게이트 (assertOutboundAllowed)
|
|
10
|
+
- `formatting.ts` — Slack mrkdwn 변환, 메시지 텍스트 추출
|
|
11
|
+
- `audit.ts` — 감사 로그 (AuditEntry, auditLog)
|
|
12
|
+
- `event.ts` — DM 판별, 스레드 해석, stale/empty 필터링
|
|
13
|
+
- `resilience.ts` — 이벤트 중복 제거 (EventDeduplicator)
|
|
14
|
+
- `permalink.ts` — Slack 퍼마링크 빌더
|
|
15
|
+
- `index.ts` — barrel re-export
|
|
16
|
+
- `server.test.ts` — lib/ 테스트 (bun:test)
|
|
17
|
+
|
|
18
|
+
## 개발
|
|
19
|
+
|
|
20
|
+
```bash
|
|
21
|
+
bun test # 테스트 실행
|
|
22
|
+
bun run typecheck # 타입 체크
|
|
23
|
+
```
|
|
24
|
+
|
|
25
|
+
## 참고
|
|
26
|
+
|
|
27
|
+
- Channels Reference: https://code.claude.com/docs/en/channels-reference
|
package/README.md
ADDED
|
@@ -0,0 +1,270 @@
|
|
|
1
|
+
# Claude Slack Channel
|
|
2
|
+
|
|
3
|
+
Claude Code 세션과 Slack을 양방향으로 연결합니다. Slack DM이나 채널 멘션으로 Claude Code와 대화할 수 있습니다.
|
|
4
|
+
|
|
5
|
+
## 아키텍처
|
|
6
|
+
|
|
7
|
+
```
|
|
8
|
+
┌─────────────────────────────────────────────────────────────────┐
|
|
9
|
+
│ Claude Code (Host) │
|
|
10
|
+
│ │
|
|
11
|
+
│ Claude LLM ◄──── <channel> notification ◄──── stdout │
|
|
12
|
+
│ │ ▲ │
|
|
13
|
+
│ │ tool call (reply/react/...) │ │
|
|
14
|
+
│ ▼ │ │
|
|
15
|
+
│ stdin ────► server.ts (MCP Server) ────► mcp.notification() │
|
|
16
|
+
│ │ ▲ │
|
|
17
|
+
│ │ │ │
|
|
18
|
+
│ ▼ │ │
|
|
19
|
+
│ Slack Web API Socket Mode (WebSocket) │
|
|
20
|
+
│ │ │ │
|
|
21
|
+
└────────────────────┼───────────┼────────────────────────────────┘
|
|
22
|
+
│ │
|
|
23
|
+
▼ │
|
|
24
|
+
┌──────────────────┴──────┐
|
|
25
|
+
│ Slack Platform │
|
|
26
|
+
│ (채널, DM, 스레드) │
|
|
27
|
+
└─────────────────────────┘
|
|
28
|
+
```
|
|
29
|
+
|
|
30
|
+
- **전송 프로토콜**: MCP (Model Context Protocol) over stdio
|
|
31
|
+
- **인바운드**: Slack → WebSocket(Socket Mode) → `server.ts` → `mcp.notification()` → stdout → Claude
|
|
32
|
+
- **아웃바운드**: Claude → tool call → stdin → `server.ts` → Slack Web API → Slack
|
|
33
|
+
- `console.error`만 디버그 로그에 사용 (stdout은 MCP 전용)
|
|
34
|
+
|
|
35
|
+
## 시작 흐름
|
|
36
|
+
|
|
37
|
+
```
|
|
38
|
+
main()
|
|
39
|
+
│
|
|
40
|
+
├── startSocketMode()
|
|
41
|
+
│ ├── web.auth.test() → botUserId 확인
|
|
42
|
+
│ │ └── 실패 ──► process.exit(1)
|
|
43
|
+
│ └── socket.start() → Slack 이벤트 수신 시작
|
|
44
|
+
│
|
|
45
|
+
└── mcp.connect(transport) → MCP stdio 연결
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
Socket Mode는 항상 연결됩니다. Claude Code가 `--dangerously-load-development-channels` 플래그 없이 실행되면 channel notification을 무시하므로, 서버 측에서 별도 게이팅 없이도 안전합니다.
|
|
49
|
+
|
|
50
|
+
## 인바운드 메시지 파이프라인
|
|
51
|
+
|
|
52
|
+
Slack에서 메시지가 들어오면 6단계 파이프라인을 거칩니다. 각 단계에서 조건을 만족하지 않으면 메시지를 drop합니다.
|
|
53
|
+
|
|
54
|
+
```
|
|
55
|
+
Slack Event (message / app_mention)
|
|
56
|
+
│
|
|
57
|
+
▼
|
|
58
|
+
┌─────────────┐
|
|
59
|
+
│ 1. Dedup │──── 이미 처리한 이벤트? ──── drop
|
|
60
|
+
└──────┬──────┘
|
|
61
|
+
│ 새 이벤트
|
|
62
|
+
▼
|
|
63
|
+
┌─────────────┐
|
|
64
|
+
│ 2. Stale │──── 10분 이상 된 이벤트? ──── drop
|
|
65
|
+
└──────┬──────┘
|
|
66
|
+
│ 최근 이벤트
|
|
67
|
+
▼
|
|
68
|
+
┌─────────────┐
|
|
69
|
+
│ 3. Empty │──── 텍스트/파일/블록 없음? ──── drop
|
|
70
|
+
└──────┬──────┘
|
|
71
|
+
│ 내용 있음
|
|
72
|
+
▼
|
|
73
|
+
┌─────────────┐
|
|
74
|
+
│ 4. Gate │──── 봇 자신? subtype 불허? ──── drop
|
|
75
|
+
│ (접근제어) │──── allowlist에 없음? ──────── drop
|
|
76
|
+
└──────┬──────┘
|
|
77
|
+
│ 허용된 사용자
|
|
78
|
+
▼
|
|
79
|
+
┌─────────────┐
|
|
80
|
+
│ 5. Deliver │──── permalink 생성
|
|
81
|
+
│ │──── ack reaction 추가
|
|
82
|
+
│ │──── mcp.notification() → Claude
|
|
83
|
+
└─────────────┘
|
|
84
|
+
```
|
|
85
|
+
|
|
86
|
+
## 아웃바운드 도구
|
|
87
|
+
|
|
88
|
+
Claude가 사용할 수 있는 MCP 도구:
|
|
89
|
+
|
|
90
|
+
```
|
|
91
|
+
Claude (tool call)
|
|
92
|
+
│
|
|
93
|
+
├── reply ──────────── 메시지 전송
|
|
94
|
+
│ chat.postMessage → mrkdwn 포맷 변환
|
|
95
|
+
│ ack reaction 자동 제거
|
|
96
|
+
│
|
|
97
|
+
├── react ─────────── 이모지 리액션 추가
|
|
98
|
+
│ reactions.add
|
|
99
|
+
│
|
|
100
|
+
├── delete_bot_message ── 봇 메시지 삭제
|
|
101
|
+
│ chat.delete (자기 메시지만)
|
|
102
|
+
│
|
|
103
|
+
└── fetch_dm_thread ── DM 스레드 읽기
|
|
104
|
+
conversations.replies
|
|
105
|
+
(is_dm=true일 때만 사용)
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
모든 아웃바운드 호출은 **audit log**에 기록되며, **outbound gate**를 통과해야 합니다 (인바운드를 받은 채널에만 응답 가능).
|
|
109
|
+
|
|
110
|
+
## 보안 레이어
|
|
111
|
+
|
|
112
|
+
```
|
|
113
|
+
┌─────────────────────────────────────────────────────┐
|
|
114
|
+
│ Inbound Security │
|
|
115
|
+
│ │
|
|
116
|
+
│ Allowlist Gate ── SLACK_ALLOW_FROM에 등록된 │
|
|
117
|
+
│ 사용자만 Claude에 메시지 전달 │
|
|
118
|
+
│ │
|
|
119
|
+
│ Bot Owner ────── SLACK_BOT_OWNER는 항상 허용 │
|
|
120
|
+
│ │
|
|
121
|
+
│ Self-loop ────── 봇 자신의 메시지 자동 drop │
|
|
122
|
+
│ │
|
|
123
|
+
│ Dedup ────────── 중복 이벤트 필터링 (TTL 10분) │
|
|
124
|
+
│ │
|
|
125
|
+
├─────────────────────────────────────────────────────┤
|
|
126
|
+
│ Outbound Security │
|
|
127
|
+
│ │
|
|
128
|
+
│ Outbound Gate ── 인바운드 수신 이력이 있는 │
|
|
129
|
+
│ 채널에만 응답 허용 │
|
|
130
|
+
│ │
|
|
131
|
+
│ Prompt Hardening ── 시스템 프롬프트에서 │
|
|
132
|
+
│ 설정 변경 요청 거부 │
|
|
133
|
+
└─────────────────────────────────────────────────────┘
|
|
134
|
+
```
|
|
135
|
+
|
|
136
|
+
## 프로젝트 구조
|
|
137
|
+
|
|
138
|
+
```
|
|
139
|
+
server.ts MCP 서버, Slack 클라이언트, 이벤트 핸들링
|
|
140
|
+
tools.ts MCP 도구 등록 (reply, react, delete_bot_message, fetch_dm_thread)
|
|
141
|
+
lib/
|
|
142
|
+
gate.ts 접근 제어 (Access, GateOptions, gate, clientSupportsChannels)
|
|
143
|
+
security.ts 아웃바운드 게이트 (assertOutboundAllowed)
|
|
144
|
+
formatting.ts Slack mrkdwn 변환, 메시지 텍스트 추출
|
|
145
|
+
audit.ts 감사 로그 (AuditEntry, auditLog)
|
|
146
|
+
event.ts DM 판별, 스레드 해석, stale/empty 필터링
|
|
147
|
+
resilience.ts 이벤트 중복 제거 (EventDeduplicator)
|
|
148
|
+
permalink.ts Slack 퍼마링크 빌더
|
|
149
|
+
index.ts barrel re-export
|
|
150
|
+
server.test.ts lib/ 테스트 (bun:test, 85개)
|
|
151
|
+
```
|
|
152
|
+
|
|
153
|
+
## 데이터 흐름 상세
|
|
154
|
+
|
|
155
|
+
### Permission Request 흐름
|
|
156
|
+
|
|
157
|
+
도구 실행에 권한 확인이 필요할 때 Slack으로 알림을 보냅니다:
|
|
158
|
+
|
|
159
|
+
```
|
|
160
|
+
Claude Code ──► permission_request notification
|
|
161
|
+
│
|
|
162
|
+
▼
|
|
163
|
+
server.ts 수신
|
|
164
|
+
│
|
|
165
|
+
▼
|
|
166
|
+
마지막 inbound 채널/스레드로
|
|
167
|
+
Slack 알림 전송 (경고색 #f0ad4e)
|
|
168
|
+
│
|
|
169
|
+
▼
|
|
170
|
+
사용자가 Slack에서 확인
|
|
171
|
+
(승인/거부는 터미널에서)
|
|
172
|
+
```
|
|
173
|
+
|
|
174
|
+
### Ack Reaction 흐름
|
|
175
|
+
|
|
176
|
+
메시지 수신 확인과 답장 완료를 이모지로 표시합니다:
|
|
177
|
+
|
|
178
|
+
```
|
|
179
|
+
메시지 수신 ──► ack reaction 추가 (예: 👀)
|
|
180
|
+
│
|
|
181
|
+
▼
|
|
182
|
+
Claude가 처리 중...
|
|
183
|
+
│
|
|
184
|
+
▼
|
|
185
|
+
reply 도구 호출 시
|
|
186
|
+
ack reaction 자동 제거
|
|
187
|
+
```
|
|
188
|
+
|
|
189
|
+
## 기존 Slack App에 추가 설정
|
|
190
|
+
|
|
191
|
+
이미 Slack App이 있다면 (예: slack MCP 서버용), 같은 앱에 아래 설정을 추가합니다.
|
|
192
|
+
|
|
193
|
+
### 1. Socket Mode 활성화
|
|
194
|
+
|
|
195
|
+
1. [api.slack.com/apps](https://api.slack.com/apps) → 앱 선택
|
|
196
|
+
2. **Settings → Socket Mode** → Enable
|
|
197
|
+
3. App-Level Token 생성 → `connections:write` 스코프 → `xapp-...` 토큰 복사
|
|
198
|
+
|
|
199
|
+
### 2. Event Subscriptions 활성화
|
|
200
|
+
|
|
201
|
+
**Event Subscriptions** → Enable → Bot Events 구독:
|
|
202
|
+
- `message.im` — DM
|
|
203
|
+
- `message.channels` — 공개 채널
|
|
204
|
+
- `message.groups` — 비공개 채널
|
|
205
|
+
- `app_mention` — @ 멘션
|
|
206
|
+
|
|
207
|
+
### 3. Bot Token Scopes 추가
|
|
208
|
+
|
|
209
|
+
**OAuth & Permissions** → Bot Token Scopes에 추가:
|
|
210
|
+
- `chat:write` — 메시지 전송
|
|
211
|
+
- `channels:history` — 공개 채널 읽기
|
|
212
|
+
- `groups:history` — 비공개 채널 읽기
|
|
213
|
+
- `im:history` — DM 읽기
|
|
214
|
+
- `reactions:write` — 리액션 추가/제거
|
|
215
|
+
- `users:read` — 사용자 이름 확인
|
|
216
|
+
|
|
217
|
+
### 4. 워크스페이스에 재설치
|
|
218
|
+
|
|
219
|
+
**OAuth & Permissions** → **Reinstall to Workspace** → Bot Token (`xoxb-...`) 복사
|
|
220
|
+
|
|
221
|
+
## 설치 및 설정
|
|
222
|
+
|
|
223
|
+
프로젝트 `.mcp.json` 또는 `~/.claude.json`에 추가합니다:
|
|
224
|
+
|
|
225
|
+
```json
|
|
226
|
+
{
|
|
227
|
+
"mcpServers": {
|
|
228
|
+
"slack-channel": {
|
|
229
|
+
"command": "bunx",
|
|
230
|
+
"args": ["@rokrokss/claude-slack-channel"],
|
|
231
|
+
"env": {
|
|
232
|
+
"SLACK_BOT_TOKEN": "xoxb-...",
|
|
233
|
+
"SLACK_APP_TOKEN": "xapp-...",
|
|
234
|
+
"SLACK_ALLOW_FROM": "U123,U456",
|
|
235
|
+
"SLACK_WORKSPACE": "your-workspace",
|
|
236
|
+
"SLACK_ACK_REACTION": "eyes"
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
```
|
|
242
|
+
|
|
243
|
+
| 환경변수 | 필수 | 설명 |
|
|
244
|
+
|---|---|---|
|
|
245
|
+
| `SLACK_BOT_TOKEN` | O | Bot User OAuth Token (`xoxb-...`) |
|
|
246
|
+
| `SLACK_APP_TOKEN` | O | App-Level Token (`xapp-...`) |
|
|
247
|
+
| `SLACK_ALLOW_FROM` | O | 허용할 Slack 사용자 ID (쉼표 구분) |
|
|
248
|
+
| `SLACK_WORKSPACE` | O | Slack 워크스페이스 서브도메인 (예: `msuniverse`). permalink 생성에 사용 |
|
|
249
|
+
| `SLACK_BOT_OWNER` | X | 봇 소유자 사용자 ID. allowlist 없이도 항상 허용 |
|
|
250
|
+
| `SLACK_ACK_REACTION` | X | 수신 확인 이모지 (예: `eyes`). 답장 후 자동 제거 |
|
|
251
|
+
| `SLACK_DEFAULT_COLOR` | X | 메시지 사이드바 색상 hex (기본: `#e5da9a`) |
|
|
252
|
+
|
|
253
|
+
허용된 사용자는 DM이든 채널이든 어디서든 봇과 대화할 수 있습니다.
|
|
254
|
+
|
|
255
|
+
## 실행
|
|
256
|
+
|
|
257
|
+
```bash
|
|
258
|
+
claude --dangerously-load-development-channels server:slack-channel
|
|
259
|
+
```
|
|
260
|
+
|
|
261
|
+
## 개발
|
|
262
|
+
|
|
263
|
+
```bash
|
|
264
|
+
bun test # 테스트 실행 (85개)
|
|
265
|
+
bun run typecheck # 타입 체크
|
|
266
|
+
```
|
|
267
|
+
|
|
268
|
+
## 라이선스
|
|
269
|
+
|
|
270
|
+
MIT
|
package/bun.lock
ADDED
|
@@ -0,0 +1,266 @@
|
|
|
1
|
+
{
|
|
2
|
+
"lockfileVersion": 1,
|
|
3
|
+
"configVersion": 1,
|
|
4
|
+
"workspaces": {
|
|
5
|
+
"": {
|
|
6
|
+
"name": "@rokrokss/claude-slack-channel",
|
|
7
|
+
"dependencies": {
|
|
8
|
+
"@modelcontextprotocol/sdk": "^1.0.0",
|
|
9
|
+
"@slack/socket-mode": "^2.0.0",
|
|
10
|
+
"@slack/web-api": "^7.0.0",
|
|
11
|
+
"zod": "^4.3.6",
|
|
12
|
+
},
|
|
13
|
+
"devDependencies": {
|
|
14
|
+
"@types/bun": "^1.0.0",
|
|
15
|
+
"typescript": "^5.4.0",
|
|
16
|
+
},
|
|
17
|
+
},
|
|
18
|
+
},
|
|
19
|
+
"packages": {
|
|
20
|
+
"@hono/node-server": ["@hono/node-server@1.19.11", "", { "peerDependencies": { "hono": "^4" } }, "sha512-dr8/3zEaB+p0D2n/IUrlPF1HZm586qgJNXK1a9fhg/PzdtkK7Ksd5l312tJX2yBuALqDYBlG20QEbayqPyxn+g=="],
|
|
21
|
+
|
|
22
|
+
"@modelcontextprotocol/sdk": ["@modelcontextprotocol/sdk@1.28.0", "", { "dependencies": { "@hono/node-server": "^1.19.9", "ajv": "^8.17.1", "ajv-formats": "^3.0.1", "content-type": "^1.0.5", "cors": "^2.8.5", "cross-spawn": "^7.0.5", "eventsource": "^3.0.2", "eventsource-parser": "^3.0.0", "express": "^5.2.1", "express-rate-limit": "^8.2.1", "hono": "^4.11.4", "jose": "^6.1.3", "json-schema-typed": "^8.0.2", "pkce-challenge": "^5.0.0", "raw-body": "^3.0.0", "zod": "^3.25 || ^4.0", "zod-to-json-schema": "^3.25.1" }, "peerDependencies": { "@cfworker/json-schema": "^4.1.1" }, "optionalPeers": ["@cfworker/json-schema"] }, "sha512-gmloF+i+flI8ouQK7MWW4mOwuMh4RePBuPFAEPC6+pdqyWOUMDOixb6qZ69owLJpz6XmyllCouc4t8YWO+E2Nw=="],
|
|
23
|
+
|
|
24
|
+
"@slack/logger": ["@slack/logger@4.0.1", "", { "dependencies": { "@types/node": ">=18" } }, "sha512-6cmdPrV/RYfd2U0mDGiMK8S7OJqpCTm7enMLRR3edccsPX8j7zXTLnaEF4fhxxJJTAIOil6+qZrnUPTuaLvwrQ=="],
|
|
25
|
+
|
|
26
|
+
"@slack/socket-mode": ["@slack/socket-mode@2.0.6", "", { "dependencies": { "@slack/logger": "^4.0.1", "@slack/web-api": "^7.15.0", "@types/node": ">=18", "@types/ws": "^8", "eventemitter3": "^5", "ws": "^8" } }, "sha512-Aj5RO3MoYVJ+b2tUjHUXuA3tiIaCUMOf1Ss5tPiz29XYVUi6qNac2A8ulcU1pUPERpXVHTmT1XW6HzQIO74daQ=="],
|
|
27
|
+
|
|
28
|
+
"@slack/types": ["@slack/types@2.20.1", "", {}, "sha512-eWX2mdt1ktpn8+40iiMc404uGrih+2fxiky3zBcPjtXKj6HLRdYlmhrPkJi7JTJm8dpXR6BWVWEDBXtaWMKD6A=="],
|
|
29
|
+
|
|
30
|
+
"@slack/web-api": ["@slack/web-api@7.15.0", "", { "dependencies": { "@slack/logger": "^4.0.1", "@slack/types": "^2.20.1", "@types/node": ">=18", "@types/retry": "0.12.0", "axios": "^1.13.5", "eventemitter3": "^5.0.1", "form-data": "^4.0.4", "is-electron": "2.2.2", "is-stream": "^2", "p-queue": "^6", "p-retry": "^4", "retry": "^0.13.1" } }, "sha512-va7zYIt3QHG1x9M/jqXXRPFMoOVlVSSRHC5YH+DzKYsrz5xUKOA3lR4THsu/Zxha9N1jOndbKFKLtr0WOPW1Vw=="],
|
|
31
|
+
|
|
32
|
+
"@types/bun": ["@types/bun@1.3.11", "", { "dependencies": { "bun-types": "1.3.11" } }, "sha512-5vPne5QvtpjGpsGYXiFyycfpDF2ECyPcTSsFBMa0fraoxiQyMJ3SmuQIGhzPg2WJuWxVBoxWJ2kClYTcw/4fAg=="],
|
|
33
|
+
|
|
34
|
+
"@types/node": ["@types/node@25.5.0", "", { "dependencies": { "undici-types": "~7.18.0" } }, "sha512-jp2P3tQMSxWugkCUKLRPVUpGaL5MVFwF8RDuSRztfwgN1wmqJeMSbKlnEtQqU8UrhTmzEmZdu2I6v2dpp7XIxw=="],
|
|
35
|
+
|
|
36
|
+
"@types/retry": ["@types/retry@0.12.0", "", {}, "sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA=="],
|
|
37
|
+
|
|
38
|
+
"@types/ws": ["@types/ws@8.18.1", "", { "dependencies": { "@types/node": "*" } }, "sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg=="],
|
|
39
|
+
|
|
40
|
+
"accepts": ["accepts@2.0.0", "", { "dependencies": { "mime-types": "^3.0.0", "negotiator": "^1.0.0" } }, "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng=="],
|
|
41
|
+
|
|
42
|
+
"ajv": ["ajv@8.18.0", "", { "dependencies": { "fast-deep-equal": "^3.1.3", "fast-uri": "^3.0.1", "json-schema-traverse": "^1.0.0", "require-from-string": "^2.0.2" } }, "sha512-PlXPeEWMXMZ7sPYOHqmDyCJzcfNrUr3fGNKtezX14ykXOEIvyK81d+qydx89KY5O71FKMPaQ2vBfBFI5NHR63A=="],
|
|
43
|
+
|
|
44
|
+
"ajv-formats": ["ajv-formats@3.0.1", "", { "dependencies": { "ajv": "^8.0.0" } }, "sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ=="],
|
|
45
|
+
|
|
46
|
+
"asynckit": ["asynckit@0.4.0", "", {}, "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q=="],
|
|
47
|
+
|
|
48
|
+
"axios": ["axios@1.13.6", "", { "dependencies": { "follow-redirects": "^1.15.11", "form-data": "^4.0.5", "proxy-from-env": "^1.1.0" } }, "sha512-ChTCHMouEe2kn713WHbQGcuYrr6fXTBiu460OTwWrWob16g1bXn4vtz07Ope7ewMozJAnEquLk5lWQWtBig9DQ=="],
|
|
49
|
+
|
|
50
|
+
"body-parser": ["body-parser@2.2.2", "", { "dependencies": { "bytes": "^3.1.2", "content-type": "^1.0.5", "debug": "^4.4.3", "http-errors": "^2.0.0", "iconv-lite": "^0.7.0", "on-finished": "^2.4.1", "qs": "^6.14.1", "raw-body": "^3.0.1", "type-is": "^2.0.1" } }, "sha512-oP5VkATKlNwcgvxi0vM0p/D3n2C3EReYVX+DNYs5TjZFn/oQt2j+4sVJtSMr18pdRr8wjTcBl6LoV+FUwzPmNA=="],
|
|
51
|
+
|
|
52
|
+
"bun-types": ["bun-types@1.3.11", "", { "dependencies": { "@types/node": "*" } }, "sha512-1KGPpoxQWl9f6wcZh57LvrPIInQMn2TQ7jsgxqpRzg+l0QPOFvJVH7HmvHo/AiPgwXy+/Thf6Ov3EdVn1vOabg=="],
|
|
53
|
+
|
|
54
|
+
"bytes": ["bytes@3.1.2", "", {}, "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg=="],
|
|
55
|
+
|
|
56
|
+
"call-bind-apply-helpers": ["call-bind-apply-helpers@1.0.2", "", { "dependencies": { "es-errors": "^1.3.0", "function-bind": "^1.1.2" } }, "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ=="],
|
|
57
|
+
|
|
58
|
+
"call-bound": ["call-bound@1.0.4", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "get-intrinsic": "^1.3.0" } }, "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg=="],
|
|
59
|
+
|
|
60
|
+
"combined-stream": ["combined-stream@1.0.8", "", { "dependencies": { "delayed-stream": "~1.0.0" } }, "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg=="],
|
|
61
|
+
|
|
62
|
+
"content-disposition": ["content-disposition@1.0.1", "", {}, "sha512-oIXISMynqSqm241k6kcQ5UwttDILMK4BiurCfGEREw6+X9jkkpEe5T9FZaApyLGGOnFuyMWZpdolTXMtvEJ08Q=="],
|
|
63
|
+
|
|
64
|
+
"content-type": ["content-type@1.0.5", "", {}, "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA=="],
|
|
65
|
+
|
|
66
|
+
"cookie": ["cookie@0.7.2", "", {}, "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w=="],
|
|
67
|
+
|
|
68
|
+
"cookie-signature": ["cookie-signature@1.2.2", "", {}, "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg=="],
|
|
69
|
+
|
|
70
|
+
"cors": ["cors@2.8.6", "", { "dependencies": { "object-assign": "^4", "vary": "^1" } }, "sha512-tJtZBBHA6vjIAaF6EnIaq6laBBP9aq/Y3ouVJjEfoHbRBcHBAHYcMh/w8LDrk2PvIMMq8gmopa5D4V8RmbrxGw=="],
|
|
71
|
+
|
|
72
|
+
"cross-spawn": ["cross-spawn@7.0.6", "", { "dependencies": { "path-key": "^3.1.0", "shebang-command": "^2.0.0", "which": "^2.0.1" } }, "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA=="],
|
|
73
|
+
|
|
74
|
+
"debug": ["debug@4.4.3", "", { "dependencies": { "ms": "^2.1.3" } }, "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA=="],
|
|
75
|
+
|
|
76
|
+
"delayed-stream": ["delayed-stream@1.0.0", "", {}, "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ=="],
|
|
77
|
+
|
|
78
|
+
"depd": ["depd@2.0.0", "", {}, "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw=="],
|
|
79
|
+
|
|
80
|
+
"dunder-proto": ["dunder-proto@1.0.1", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.1", "es-errors": "^1.3.0", "gopd": "^1.2.0" } }, "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A=="],
|
|
81
|
+
|
|
82
|
+
"ee-first": ["ee-first@1.1.1", "", {}, "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="],
|
|
83
|
+
|
|
84
|
+
"encodeurl": ["encodeurl@2.0.0", "", {}, "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg=="],
|
|
85
|
+
|
|
86
|
+
"es-define-property": ["es-define-property@1.0.1", "", {}, "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g=="],
|
|
87
|
+
|
|
88
|
+
"es-errors": ["es-errors@1.3.0", "", {}, "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw=="],
|
|
89
|
+
|
|
90
|
+
"es-object-atoms": ["es-object-atoms@1.1.1", "", { "dependencies": { "es-errors": "^1.3.0" } }, "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA=="],
|
|
91
|
+
|
|
92
|
+
"es-set-tostringtag": ["es-set-tostringtag@2.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "get-intrinsic": "^1.2.6", "has-tostringtag": "^1.0.2", "hasown": "^2.0.2" } }, "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA=="],
|
|
93
|
+
|
|
94
|
+
"escape-html": ["escape-html@1.0.3", "", {}, "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow=="],
|
|
95
|
+
|
|
96
|
+
"etag": ["etag@1.8.1", "", {}, "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg=="],
|
|
97
|
+
|
|
98
|
+
"eventemitter3": ["eventemitter3@5.0.4", "", {}, "sha512-mlsTRyGaPBjPedk6Bvw+aqbsXDtoAyAzm5MO7JgU+yVRyMQ5O8bD4Kcci7BS85f93veegeCPkL8R4GLClnjLFw=="],
|
|
99
|
+
|
|
100
|
+
"eventsource": ["eventsource@3.0.7", "", { "dependencies": { "eventsource-parser": "^3.0.1" } }, "sha512-CRT1WTyuQoD771GW56XEZFQ/ZoSfWid1alKGDYMmkt2yl8UXrVR4pspqWNEcqKvVIzg6PAltWjxcSSPrboA4iA=="],
|
|
101
|
+
|
|
102
|
+
"eventsource-parser": ["eventsource-parser@3.0.6", "", {}, "sha512-Vo1ab+QXPzZ4tCa8SwIHJFaSzy4R6SHf7BY79rFBDf0idraZWAkYrDjDj8uWaSm3S2TK+hJ7/t1CEmZ7jXw+pg=="],
|
|
103
|
+
|
|
104
|
+
"express": ["express@5.2.1", "", { "dependencies": { "accepts": "^2.0.0", "body-parser": "^2.2.1", "content-disposition": "^1.0.0", "content-type": "^1.0.5", "cookie": "^0.7.1", "cookie-signature": "^1.2.1", "debug": "^4.4.0", "depd": "^2.0.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "finalhandler": "^2.1.0", "fresh": "^2.0.0", "http-errors": "^2.0.0", "merge-descriptors": "^2.0.0", "mime-types": "^3.0.0", "on-finished": "^2.4.1", "once": "^1.4.0", "parseurl": "^1.3.3", "proxy-addr": "^2.0.7", "qs": "^6.14.0", "range-parser": "^1.2.1", "router": "^2.2.0", "send": "^1.1.0", "serve-static": "^2.2.0", "statuses": "^2.0.1", "type-is": "^2.0.1", "vary": "^1.1.2" } }, "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw=="],
|
|
105
|
+
|
|
106
|
+
"express-rate-limit": ["express-rate-limit@8.3.1", "", { "dependencies": { "ip-address": "10.1.0" }, "peerDependencies": { "express": ">= 4.11" } }, "sha512-D1dKN+cmyPWuvB+G2SREQDzPY1agpBIcTa9sJxOPMCNeH3gwzhqJRDWCXW3gg0y//+LQ/8j52JbMROWyrKdMdw=="],
|
|
107
|
+
|
|
108
|
+
"fast-deep-equal": ["fast-deep-equal@3.1.3", "", {}, "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q=="],
|
|
109
|
+
|
|
110
|
+
"fast-uri": ["fast-uri@3.1.0", "", {}, "sha512-iPeeDKJSWf4IEOasVVrknXpaBV0IApz/gp7S2bb7Z4Lljbl2MGJRqInZiUrQwV16cpzw/D3S5j5Julj/gT52AA=="],
|
|
111
|
+
|
|
112
|
+
"finalhandler": ["finalhandler@2.1.1", "", { "dependencies": { "debug": "^4.4.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "on-finished": "^2.4.1", "parseurl": "^1.3.3", "statuses": "^2.0.1" } }, "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA=="],
|
|
113
|
+
|
|
114
|
+
"follow-redirects": ["follow-redirects@1.15.11", "", {}, "sha512-deG2P0JfjrTxl50XGCDyfI97ZGVCxIpfKYmfyrQ54n5FO/0gfIES8C/Psl6kWVDolizcaaxZJnTS0QSMxvnsBQ=="],
|
|
115
|
+
|
|
116
|
+
"form-data": ["form-data@4.0.5", "", { "dependencies": { "asynckit": "^0.4.0", "combined-stream": "^1.0.8", "es-set-tostringtag": "^2.1.0", "hasown": "^2.0.2", "mime-types": "^2.1.12" } }, "sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w=="],
|
|
117
|
+
|
|
118
|
+
"forwarded": ["forwarded@0.2.0", "", {}, "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow=="],
|
|
119
|
+
|
|
120
|
+
"fresh": ["fresh@2.0.0", "", {}, "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A=="],
|
|
121
|
+
|
|
122
|
+
"function-bind": ["function-bind@1.1.2", "", {}, "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA=="],
|
|
123
|
+
|
|
124
|
+
"get-intrinsic": ["get-intrinsic@1.3.0", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "es-define-property": "^1.0.1", "es-errors": "^1.3.0", "es-object-atoms": "^1.1.1", "function-bind": "^1.1.2", "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-symbols": "^1.1.0", "hasown": "^2.0.2", "math-intrinsics": "^1.1.0" } }, "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ=="],
|
|
125
|
+
|
|
126
|
+
"get-proto": ["get-proto@1.0.1", "", { "dependencies": { "dunder-proto": "^1.0.1", "es-object-atoms": "^1.0.0" } }, "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g=="],
|
|
127
|
+
|
|
128
|
+
"gopd": ["gopd@1.2.0", "", {}, "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg=="],
|
|
129
|
+
|
|
130
|
+
"has-symbols": ["has-symbols@1.1.0", "", {}, "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ=="],
|
|
131
|
+
|
|
132
|
+
"has-tostringtag": ["has-tostringtag@1.0.2", "", { "dependencies": { "has-symbols": "^1.0.3" } }, "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw=="],
|
|
133
|
+
|
|
134
|
+
"hasown": ["hasown@2.0.2", "", { "dependencies": { "function-bind": "^1.1.2" } }, "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ=="],
|
|
135
|
+
|
|
136
|
+
"hono": ["hono@4.12.9", "", {}, "sha512-wy3T8Zm2bsEvxKZM5w21VdHDDcwVS1yUFFY6i8UobSsKfFceT7TOwhbhfKsDyx7tYQlmRM5FLpIuYvNFyjctiA=="],
|
|
137
|
+
|
|
138
|
+
"http-errors": ["http-errors@2.0.1", "", { "dependencies": { "depd": "~2.0.0", "inherits": "~2.0.4", "setprototypeof": "~1.2.0", "statuses": "~2.0.2", "toidentifier": "~1.0.1" } }, "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ=="],
|
|
139
|
+
|
|
140
|
+
"iconv-lite": ["iconv-lite@0.7.2", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" } }, "sha512-im9DjEDQ55s9fL4EYzOAv0yMqmMBSZp6G0VvFyTMPKWxiSBHUj9NW/qqLmXUwXrrM7AvqSlTCfvqRb0cM8yYqw=="],
|
|
141
|
+
|
|
142
|
+
"inherits": ["inherits@2.0.4", "", {}, "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="],
|
|
143
|
+
|
|
144
|
+
"ip-address": ["ip-address@10.1.0", "", {}, "sha512-XXADHxXmvT9+CRxhXg56LJovE+bmWnEWB78LB83VZTprKTmaC5QfruXocxzTZ2Kl0DNwKuBdlIhjL8LeY8Sf8Q=="],
|
|
145
|
+
|
|
146
|
+
"ipaddr.js": ["ipaddr.js@1.9.1", "", {}, "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="],
|
|
147
|
+
|
|
148
|
+
"is-electron": ["is-electron@2.2.2", "", {}, "sha512-FO/Rhvz5tuw4MCWkpMzHFKWD2LsfHzIb7i6MdPYZ/KW7AlxawyLkqdy+jPZP1WubqEADE3O4FUENlJHDfQASRg=="],
|
|
149
|
+
|
|
150
|
+
"is-promise": ["is-promise@4.0.0", "", {}, "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ=="],
|
|
151
|
+
|
|
152
|
+
"is-stream": ["is-stream@2.0.1", "", {}, "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg=="],
|
|
153
|
+
|
|
154
|
+
"isexe": ["isexe@2.0.0", "", {}, "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw=="],
|
|
155
|
+
|
|
156
|
+
"jose": ["jose@6.2.2", "", {}, "sha512-d7kPDd34KO/YnzaDOlikGpOurfF0ByC2sEV4cANCtdqLlTfBlw2p14O/5d/zv40gJPbIQxfES3nSx1/oYNyuZQ=="],
|
|
157
|
+
|
|
158
|
+
"json-schema-traverse": ["json-schema-traverse@1.0.0", "", {}, "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug=="],
|
|
159
|
+
|
|
160
|
+
"json-schema-typed": ["json-schema-typed@8.0.2", "", {}, "sha512-fQhoXdcvc3V28x7C7BMs4P5+kNlgUURe2jmUT1T//oBRMDrqy1QPelJimwZGo7Hg9VPV3EQV5Bnq4hbFy2vetA=="],
|
|
161
|
+
|
|
162
|
+
"math-intrinsics": ["math-intrinsics@1.1.0", "", {}, "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g=="],
|
|
163
|
+
|
|
164
|
+
"media-typer": ["media-typer@1.1.0", "", {}, "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw=="],
|
|
165
|
+
|
|
166
|
+
"merge-descriptors": ["merge-descriptors@2.0.0", "", {}, "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g=="],
|
|
167
|
+
|
|
168
|
+
"mime-db": ["mime-db@1.54.0", "", {}, "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ=="],
|
|
169
|
+
|
|
170
|
+
"mime-types": ["mime-types@3.0.2", "", { "dependencies": { "mime-db": "^1.54.0" } }, "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A=="],
|
|
171
|
+
|
|
172
|
+
"ms": ["ms@2.1.3", "", {}, "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="],
|
|
173
|
+
|
|
174
|
+
"negotiator": ["negotiator@1.0.0", "", {}, "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg=="],
|
|
175
|
+
|
|
176
|
+
"object-assign": ["object-assign@4.1.1", "", {}, "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg=="],
|
|
177
|
+
|
|
178
|
+
"object-inspect": ["object-inspect@1.13.4", "", {}, "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew=="],
|
|
179
|
+
|
|
180
|
+
"on-finished": ["on-finished@2.4.1", "", { "dependencies": { "ee-first": "1.1.1" } }, "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg=="],
|
|
181
|
+
|
|
182
|
+
"once": ["once@1.4.0", "", { "dependencies": { "wrappy": "1" } }, "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w=="],
|
|
183
|
+
|
|
184
|
+
"p-finally": ["p-finally@1.0.0", "", {}, "sha512-LICb2p9CB7FS+0eR1oqWnHhp0FljGLZCWBE9aix0Uye9W8LTQPwMTYVGWQWIw9RdQiDg4+epXQODwIYJtSJaow=="],
|
|
185
|
+
|
|
186
|
+
"p-queue": ["p-queue@6.6.2", "", { "dependencies": { "eventemitter3": "^4.0.4", "p-timeout": "^3.2.0" } }, "sha512-RwFpb72c/BhQLEXIZ5K2e+AhgNVmIejGlTgiB9MzZ0e93GRvqZ7uSi0dvRF7/XIXDeNkra2fNHBxTyPDGySpjQ=="],
|
|
187
|
+
|
|
188
|
+
"p-retry": ["p-retry@4.6.2", "", { "dependencies": { "@types/retry": "0.12.0", "retry": "^0.13.1" } }, "sha512-312Id396EbJdvRONlngUx0NydfrIQ5lsYu0znKVUzVvArzEIt08V1qhtyESbGVd1FGX7UKtiFp5uwKZdM8wIuQ=="],
|
|
189
|
+
|
|
190
|
+
"p-timeout": ["p-timeout@3.2.0", "", { "dependencies": { "p-finally": "^1.0.0" } }, "sha512-rhIwUycgwwKcP9yTOOFK/AKsAopjjCakVqLHePO3CC6Mir1Z99xT+R63jZxAT5lFZLa2inS5h+ZS2GvR99/FBg=="],
|
|
191
|
+
|
|
192
|
+
"parseurl": ["parseurl@1.3.3", "", {}, "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="],
|
|
193
|
+
|
|
194
|
+
"path-key": ["path-key@3.1.1", "", {}, "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q=="],
|
|
195
|
+
|
|
196
|
+
"path-to-regexp": ["path-to-regexp@8.4.0", "", {}, "sha512-PuseHIvAnz3bjrM2rGJtSgo1zjgxapTLZ7x2pjhzWwlp4SJQgK3f3iZIQwkpEnBaKz6seKBADpM4B4ySkuYypg=="],
|
|
197
|
+
|
|
198
|
+
"pkce-challenge": ["pkce-challenge@5.0.1", "", {}, "sha512-wQ0b/W4Fr01qtpHlqSqspcj3EhBvimsdh0KlHhH8HRZnMsEa0ea2fTULOXOS9ccQr3om+GcGRk4e+isrZWV8qQ=="],
|
|
199
|
+
|
|
200
|
+
"proxy-addr": ["proxy-addr@2.0.7", "", { "dependencies": { "forwarded": "0.2.0", "ipaddr.js": "1.9.1" } }, "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg=="],
|
|
201
|
+
|
|
202
|
+
"proxy-from-env": ["proxy-from-env@1.1.0", "", {}, "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="],
|
|
203
|
+
|
|
204
|
+
"qs": ["qs@6.15.0", "", { "dependencies": { "side-channel": "^1.1.0" } }, "sha512-mAZTtNCeetKMH+pSjrb76NAM8V9a05I9aBZOHztWy/UqcJdQYNsf59vrRKWnojAT9Y+GbIvoTBC++CPHqpDBhQ=="],
|
|
205
|
+
|
|
206
|
+
"range-parser": ["range-parser@1.2.1", "", {}, "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="],
|
|
207
|
+
|
|
208
|
+
"raw-body": ["raw-body@3.0.2", "", { "dependencies": { "bytes": "~3.1.2", "http-errors": "~2.0.1", "iconv-lite": "~0.7.0", "unpipe": "~1.0.0" } }, "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA=="],
|
|
209
|
+
|
|
210
|
+
"require-from-string": ["require-from-string@2.0.2", "", {}, "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw=="],
|
|
211
|
+
|
|
212
|
+
"retry": ["retry@0.13.1", "", {}, "sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg=="],
|
|
213
|
+
|
|
214
|
+
"router": ["router@2.2.0", "", { "dependencies": { "debug": "^4.4.0", "depd": "^2.0.0", "is-promise": "^4.0.0", "parseurl": "^1.3.3", "path-to-regexp": "^8.0.0" } }, "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ=="],
|
|
215
|
+
|
|
216
|
+
"safer-buffer": ["safer-buffer@2.1.2", "", {}, "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="],
|
|
217
|
+
|
|
218
|
+
"send": ["send@1.2.1", "", { "dependencies": { "debug": "^4.4.3", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "fresh": "^2.0.0", "http-errors": "^2.0.1", "mime-types": "^3.0.2", "ms": "^2.1.3", "on-finished": "^2.4.1", "range-parser": "^1.2.1", "statuses": "^2.0.2" } }, "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ=="],
|
|
219
|
+
|
|
220
|
+
"serve-static": ["serve-static@2.2.1", "", { "dependencies": { "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "parseurl": "^1.3.3", "send": "^1.2.0" } }, "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw=="],
|
|
221
|
+
|
|
222
|
+
"setprototypeof": ["setprototypeof@1.2.0", "", {}, "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="],
|
|
223
|
+
|
|
224
|
+
"shebang-command": ["shebang-command@2.0.0", "", { "dependencies": { "shebang-regex": "^3.0.0" } }, "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA=="],
|
|
225
|
+
|
|
226
|
+
"shebang-regex": ["shebang-regex@3.0.0", "", {}, "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A=="],
|
|
227
|
+
|
|
228
|
+
"side-channel": ["side-channel@1.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "object-inspect": "^1.13.3", "side-channel-list": "^1.0.0", "side-channel-map": "^1.0.1", "side-channel-weakmap": "^1.0.2" } }, "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw=="],
|
|
229
|
+
|
|
230
|
+
"side-channel-list": ["side-channel-list@1.0.0", "", { "dependencies": { "es-errors": "^1.3.0", "object-inspect": "^1.13.3" } }, "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA=="],
|
|
231
|
+
|
|
232
|
+
"side-channel-map": ["side-channel-map@1.0.1", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3" } }, "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA=="],
|
|
233
|
+
|
|
234
|
+
"side-channel-weakmap": ["side-channel-weakmap@1.0.2", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3", "side-channel-map": "^1.0.1" } }, "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A=="],
|
|
235
|
+
|
|
236
|
+
"statuses": ["statuses@2.0.2", "", {}, "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw=="],
|
|
237
|
+
|
|
238
|
+
"toidentifier": ["toidentifier@1.0.1", "", {}, "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA=="],
|
|
239
|
+
|
|
240
|
+
"type-is": ["type-is@2.0.1", "", { "dependencies": { "content-type": "^1.0.5", "media-typer": "^1.1.0", "mime-types": "^3.0.0" } }, "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw=="],
|
|
241
|
+
|
|
242
|
+
"typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="],
|
|
243
|
+
|
|
244
|
+
"undici-types": ["undici-types@7.18.2", "", {}, "sha512-AsuCzffGHJybSaRrmr5eHr81mwJU3kjw6M+uprWvCXiNeN9SOGwQ3Jn8jb8m3Z6izVgknn1R0FTCEAP2QrLY/w=="],
|
|
245
|
+
|
|
246
|
+
"unpipe": ["unpipe@1.0.0", "", {}, "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ=="],
|
|
247
|
+
|
|
248
|
+
"vary": ["vary@1.1.2", "", {}, "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg=="],
|
|
249
|
+
|
|
250
|
+
"which": ["which@2.0.2", "", { "dependencies": { "isexe": "^2.0.0" }, "bin": { "node-which": "./bin/node-which" } }, "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA=="],
|
|
251
|
+
|
|
252
|
+
"wrappy": ["wrappy@1.0.2", "", {}, "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ=="],
|
|
253
|
+
|
|
254
|
+
"ws": ["ws@8.20.0", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": ">=5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-sAt8BhgNbzCtgGbt2OxmpuryO63ZoDk/sqaB/znQm94T4fCEsy/yV+7CdC1kJhOU9lboAEU7R3kquuycDoibVA=="],
|
|
255
|
+
|
|
256
|
+
"zod": ["zod@4.3.6", "", {}, "sha512-rftlrkhHZOcjDwkGlnUtZZkvaPHCsDATp4pGpuOOMDaTdDDXF91wuVDJoWoPsKX/3YPQ5fHuF3STjcYyKr+Qhg=="],
|
|
257
|
+
|
|
258
|
+
"zod-to-json-schema": ["zod-to-json-schema@3.25.1", "", { "peerDependencies": { "zod": "^3.25 || ^4" } }, "sha512-pM/SU9d3YAggzi6MtR4h7ruuQlqKtad8e9S0fmxcMi+ueAK5Korys/aWcV9LIIHTVbj01NdzxcnXSN+O74ZIVA=="],
|
|
259
|
+
|
|
260
|
+
"form-data/mime-types": ["mime-types@2.1.35", "", { "dependencies": { "mime-db": "1.52.0" } }, "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw=="],
|
|
261
|
+
|
|
262
|
+
"p-queue/eventemitter3": ["eventemitter3@4.0.7", "", {}, "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw=="],
|
|
263
|
+
|
|
264
|
+
"form-data/mime-types/mime-db": ["mime-db@1.52.0", "", {}, "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg=="],
|
|
265
|
+
}
|
|
266
|
+
}
|
package/lib/audit.ts
ADDED
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { join } from 'path'
|
|
2
|
+
import { appendFileSync, mkdirSync } from 'fs'
|
|
3
|
+
|
|
4
|
+
export interface AuditEntry {
|
|
5
|
+
ts: string
|
|
6
|
+
direction: 'inbound' | 'outbound'
|
|
7
|
+
userId?: string
|
|
8
|
+
chatId: string
|
|
9
|
+
action: string
|
|
10
|
+
threadTs?: string
|
|
11
|
+
text?: string
|
|
12
|
+
replyTo?: string
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function formatAuditLine(entry: AuditEntry): string {
|
|
16
|
+
return JSON.stringify(entry) + '\n'
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function auditLog(stateDir: string, entry: AuditEntry): void {
|
|
20
|
+
const dir = join(stateDir, 'audit')
|
|
21
|
+
mkdirSync(dir, { recursive: true })
|
|
22
|
+
const date = new Date().toISOString().slice(0, 10)
|
|
23
|
+
appendFileSync(join(dir, `${date}.jsonl`), formatAuditLine(entry))
|
|
24
|
+
}
|
package/lib/event.ts
ADDED
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
export function isDm(event: Record<string, unknown>): boolean {
|
|
2
|
+
return event['channel_type'] === 'im'
|
|
3
|
+
}
|
|
4
|
+
|
|
5
|
+
export function resolveThreadTs(event: Record<string, unknown>): string {
|
|
6
|
+
return (event['thread_ts'] as string) || (event['ts'] as string) || ''
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export function parseSlackTimestamp(ts: string): Date | null {
|
|
10
|
+
if (!/^\d+\.\d+$/.test(ts)) return null
|
|
11
|
+
const sec = parseFloat(ts)
|
|
12
|
+
return new Date(sec * 1000)
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/** Default: 10 minutes in milliseconds */
|
|
16
|
+
export const DEFAULT_STALE_THRESHOLD_MS = 10 * 60 * 1000
|
|
17
|
+
|
|
18
|
+
export function isStaleEvent(eventTs: string, maxAgeMs: number = DEFAULT_STALE_THRESHOLD_MS): boolean {
|
|
19
|
+
const date = parseSlackTimestamp(eventTs)
|
|
20
|
+
if (!date) return false
|
|
21
|
+
return Date.now() - date.getTime() > maxAgeMs
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export function isEmptyMessage(event: Record<string, unknown>): boolean {
|
|
25
|
+
if (event['files'] && (event['files'] as unknown[]).length > 0) return false
|
|
26
|
+
if (event['blocks'] && (event['blocks'] as unknown[]).length > 0) return false
|
|
27
|
+
if (event['attachments'] && (event['attachments'] as unknown[]).length > 0) return false
|
|
28
|
+
const text = (event['text'] as string) || ''
|
|
29
|
+
return text.trim().length === 0
|
|
30
|
+
}
|