opc-agent 2.0.0 → 2.0.1

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.
Files changed (156) hide show
  1. package/dist/channels/email.d.ts +32 -26
  2. package/dist/channels/email.js +239 -62
  3. package/dist/channels/feishu.d.ts +21 -6
  4. package/dist/channels/feishu.js +225 -126
  5. package/dist/channels/websocket.d.ts +46 -3
  6. package/dist/channels/websocket.js +306 -37
  7. package/dist/channels/wechat.d.ts +33 -13
  8. package/dist/channels/wechat.js +229 -42
  9. package/dist/cli.js +712 -11
  10. package/dist/core/a2a.d.ts +17 -0
  11. package/dist/core/a2a.js +43 -1
  12. package/dist/core/agent.d.ts +16 -0
  13. package/dist/core/agent.js +108 -0
  14. package/dist/core/runtime.d.ts +6 -0
  15. package/dist/core/runtime.js +161 -2
  16. package/dist/core/sandbox.d.ts +26 -0
  17. package/dist/core/sandbox.js +117 -0
  18. package/dist/core/workflow-graph.d.ts +93 -0
  19. package/dist/core/workflow-graph.js +247 -0
  20. package/dist/doctor.d.ts +15 -0
  21. package/dist/doctor.js +183 -0
  22. package/dist/eval/index.d.ts +65 -0
  23. package/dist/eval/index.js +191 -0
  24. package/dist/index.d.ts +30 -6
  25. package/dist/index.js +60 -4
  26. package/dist/plugins/content-filter.d.ts +7 -0
  27. package/dist/plugins/content-filter.js +25 -0
  28. package/dist/plugins/index.d.ts +42 -0
  29. package/dist/plugins/index.js +108 -2
  30. package/dist/plugins/logger.d.ts +6 -0
  31. package/dist/plugins/logger.js +20 -0
  32. package/dist/plugins/rate-limiter.d.ts +7 -0
  33. package/dist/plugins/rate-limiter.js +35 -0
  34. package/dist/protocols/a2a/client.d.ts +25 -0
  35. package/dist/protocols/a2a/client.js +115 -0
  36. package/dist/protocols/a2a/index.d.ts +6 -0
  37. package/dist/protocols/a2a/index.js +12 -0
  38. package/dist/protocols/a2a/server.d.ts +41 -0
  39. package/dist/protocols/a2a/server.js +295 -0
  40. package/dist/protocols/a2a/types.d.ts +91 -0
  41. package/dist/protocols/a2a/types.js +15 -0
  42. package/dist/protocols/a2a/utils.d.ts +6 -0
  43. package/dist/protocols/a2a/utils.js +47 -0
  44. package/dist/protocols/agui/client.d.ts +10 -0
  45. package/dist/protocols/agui/client.js +75 -0
  46. package/dist/protocols/agui/index.d.ts +4 -0
  47. package/dist/protocols/agui/index.js +25 -0
  48. package/dist/protocols/agui/server.d.ts +37 -0
  49. package/dist/protocols/agui/server.js +191 -0
  50. package/dist/protocols/agui/types.d.ts +107 -0
  51. package/dist/protocols/agui/types.js +17 -0
  52. package/dist/protocols/index.d.ts +2 -0
  53. package/dist/protocols/index.js +19 -0
  54. package/dist/protocols/mcp/agent-tools.d.ts +11 -0
  55. package/dist/protocols/mcp/agent-tools.js +129 -0
  56. package/dist/protocols/mcp/index.d.ts +5 -0
  57. package/dist/protocols/mcp/index.js +11 -0
  58. package/dist/protocols/mcp/server.d.ts +31 -0
  59. package/dist/protocols/mcp/server.js +248 -0
  60. package/dist/protocols/mcp/types.d.ts +92 -0
  61. package/dist/protocols/mcp/types.js +17 -0
  62. package/dist/publish/index.d.ts +45 -0
  63. package/dist/publish/index.js +350 -0
  64. package/dist/schema/oad.d.ts +682 -65
  65. package/dist/schema/oad.js +36 -3
  66. package/dist/security/approval.d.ts +36 -0
  67. package/dist/security/approval.js +113 -0
  68. package/dist/security/index.d.ts +4 -0
  69. package/dist/security/index.js +8 -0
  70. package/dist/security/keys.d.ts +16 -0
  71. package/dist/security/keys.js +117 -0
  72. package/dist/studio/server.d.ts +63 -0
  73. package/dist/studio/server.js +625 -0
  74. package/dist/studio-ui/index.html +662 -0
  75. package/dist/telemetry/index.d.ts +93 -0
  76. package/dist/telemetry/index.js +285 -0
  77. package/package.json +5 -3
  78. package/scripts/install.ps1 +31 -0
  79. package/scripts/install.sh +40 -0
  80. package/src/channels/email.ts +351 -177
  81. package/src/channels/feishu.ts +349 -236
  82. package/src/channels/websocket.ts +399 -87
  83. package/src/channels/wechat.ts +329 -149
  84. package/src/cli.ts +783 -12
  85. package/src/core/a2a.ts +60 -0
  86. package/src/core/agent.ts +125 -0
  87. package/src/core/runtime.ts +127 -0
  88. package/src/core/sandbox.ts +143 -0
  89. package/src/core/workflow-graph.ts +365 -0
  90. package/src/doctor.ts +156 -0
  91. package/src/eval/index.ts +211 -0
  92. package/src/eval/suites/basic.json +16 -0
  93. package/src/eval/suites/memory.json +12 -0
  94. package/src/eval/suites/safety.json +14 -0
  95. package/src/index.ts +54 -6
  96. package/src/plugins/content-filter.ts +23 -0
  97. package/src/plugins/index.ts +133 -2
  98. package/src/plugins/logger.ts +18 -0
  99. package/src/plugins/rate-limiter.ts +38 -0
  100. package/src/protocols/a2a/client.ts +132 -0
  101. package/src/protocols/a2a/index.ts +8 -0
  102. package/src/protocols/a2a/server.ts +333 -0
  103. package/src/protocols/a2a/types.ts +88 -0
  104. package/src/protocols/a2a/utils.ts +50 -0
  105. package/src/protocols/agui/client.ts +83 -0
  106. package/src/protocols/agui/index.ts +4 -0
  107. package/src/protocols/agui/server.ts +218 -0
  108. package/src/protocols/agui/types.ts +153 -0
  109. package/src/protocols/index.ts +2 -0
  110. package/src/protocols/mcp/agent-tools.ts +134 -0
  111. package/src/protocols/mcp/index.ts +8 -0
  112. package/src/protocols/mcp/server.ts +262 -0
  113. package/src/protocols/mcp/types.ts +69 -0
  114. package/src/publish/index.ts +376 -0
  115. package/src/schema/oad.ts +39 -2
  116. package/src/security/approval.ts +131 -0
  117. package/src/security/index.ts +3 -0
  118. package/src/security/keys.ts +87 -0
  119. package/src/studio/server.ts +629 -0
  120. package/src/studio-ui/index.html +662 -0
  121. package/src/telemetry/index.ts +324 -0
  122. package/src/types/agent-workstation.d.ts +2 -0
  123. package/tests/a2a-protocol.test.ts +285 -0
  124. package/tests/agui-protocol.test.ts +246 -0
  125. package/tests/channels/discord.test.ts +79 -0
  126. package/tests/channels/email.test.ts +148 -0
  127. package/tests/channels/feishu.test.ts +123 -0
  128. package/tests/channels/telegram.test.ts +129 -0
  129. package/tests/channels/websocket.test.ts +53 -0
  130. package/tests/channels/wechat.test.ts +170 -0
  131. package/tests/chat-cli.test.ts +160 -0
  132. package/tests/daemon.test.ts +135 -0
  133. package/tests/deepbrain-wire.test.ts +234 -0
  134. package/tests/doctor.test.ts +38 -0
  135. package/tests/eval.test.ts +173 -0
  136. package/tests/init-role.test.ts +124 -0
  137. package/tests/mcp-client.test.ts +92 -0
  138. package/tests/mcp-server.test.ts +178 -0
  139. package/tests/plugin-a2a-enhanced.test.ts +230 -0
  140. package/tests/publish.test.ts +231 -0
  141. package/tests/scheduler.test.ts +200 -0
  142. package/tests/security-enhanced.test.ts +233 -0
  143. package/tests/skill-learner.test.ts +161 -0
  144. package/tests/studio.test.ts +229 -0
  145. package/tests/subagent.test.ts +63 -0
  146. package/tests/telemetry.test.ts +186 -0
  147. package/tests/tools/builtin-extended.test.ts +138 -0
  148. package/tests/workflow-graph.test.ts +279 -0
  149. package/tutorial/customer-service-agent/README.md +612 -0
  150. package/tutorial/customer-service-agent/SOUL.md +26 -0
  151. package/tutorial/customer-service-agent/agent.yaml +63 -0
  152. package/tutorial/customer-service-agent/package.json +19 -0
  153. package/tutorial/customer-service-agent/src/index.ts +69 -0
  154. package/tutorial/customer-service-agent/src/skills/faq.ts +27 -0
  155. package/tutorial/customer-service-agent/src/skills/ticket.ts +22 -0
  156. package/tutorial/customer-service-agent/tsconfig.json +14 -0
@@ -1,236 +1,349 @@
1
- import { BaseChannel } from './index';
2
- import type { Message } from '../core/types';
3
-
4
- /**
5
- * Feishu / Lark Channel — v1.1.0
6
- *
7
- * Supports:
8
- * - Event Subscription (webhook) mode for receiving messages
9
- * - Bot send via Feishu Open API
10
- * - URL verification challenge
11
- * - Message card (interactive) responses
12
- * - Group chat & P2P messaging
13
- *
14
- * Env vars:
15
- * FEISHU_APP_ID, FEISHU_APP_SECRET app credentials
16
- * FEISHU_VERIFICATION_TOKEN event subscription verification
17
- * FEISHU_ENCRYPT_KEY — (optional) event encryption key
18
- */
19
-
20
- export interface FeishuChannelConfig {
21
- /** Feishu App ID */
22
- appId?: string;
23
- /** Feishu App Secret */
24
- appSecret?: string;
25
- /** Verification token for event subscription */
26
- verificationToken?: string;
27
- /** Encrypt key (optional, for encrypted events) */
28
- encryptKey?: string;
29
- /** Webhook server port (default: 3002) */
30
- port?: number;
31
- /** API base URL (use 'https://open.larksuite.com' for Lark international) */
32
- apiBase?: string;
33
- }
34
-
35
- interface FeishuTokenCache {
36
- token: string;
37
- expiresAt: number;
38
- }
39
-
40
- export class FeishuChannel extends BaseChannel {
41
- readonly type = 'feishu';
42
- private config: Required<Pick<FeishuChannelConfig, 'port' | 'apiBase'>> & FeishuChannelConfig;
43
- private server: import('http').Server | null = null;
44
- private tokenCache: FeishuTokenCache | null = null;
45
- private processedEvents = new Set<string>();
46
-
47
- constructor(config: FeishuChannelConfig = {}) {
48
- super();
49
- this.config = {
50
- appId: config.appId ?? process.env.FEISHU_APP_ID ?? '',
51
- appSecret: config.appSecret ?? process.env.FEISHU_APP_SECRET ?? '',
52
- verificationToken: config.verificationToken ?? process.env.FEISHU_VERIFICATION_TOKEN ?? '',
53
- encryptKey: config.encryptKey ?? process.env.FEISHU_ENCRYPT_KEY,
54
- port: config.port ?? 3002,
55
- apiBase: config.apiBase ?? 'https://open.feishu.cn',
56
- };
57
- }
58
-
59
- async start(): Promise<void> {
60
- if (!this.config.appId || !this.config.appSecret) {
61
- console.warn('[FeishuChannel] Missing appId/appSecret. Set FEISHU_APP_ID and FEISHU_APP_SECRET.');
62
- return;
63
- }
64
-
65
- const express = (await import('express')).default;
66
- const app = express();
67
- app.use(express.json());
68
-
69
- // Event subscription endpoint
70
- app.post('/feishu/event', async (req, res) => {
71
- try {
72
- const body = req.body;
73
-
74
- // URL verification challenge
75
- if (body.type === 'url_verification') {
76
- res.json({ challenge: body.challenge });
77
- return;
78
- }
79
-
80
- // Deduplicate events
81
- const eventId = body.header?.event_id;
82
- if (eventId && this.processedEvents.has(eventId)) {
83
- res.json({ ok: true });
84
- return;
85
- }
86
- if (eventId) {
87
- this.processedEvents.add(eventId);
88
- // Prune old events (keep last 1000)
89
- if (this.processedEvents.size > 1000) {
90
- const arr = [...this.processedEvents];
91
- this.processedEvents = new Set(arr.slice(-500));
92
- }
93
- }
94
-
95
- // Verify token
96
- if (this.config.verificationToken && body.header?.token !== this.config.verificationToken) {
97
- res.status(403).json({ error: 'Invalid verification token' });
98
- return;
99
- }
100
-
101
- // Handle im.message.receive_v1
102
- const event = body.event;
103
- if (body.header?.event_type === 'im.message.receive_v1' && this.handler) {
104
- const msgBody = event?.message;
105
- if (!msgBody) { res.json({ ok: true }); return; }
106
-
107
- // Only handle text messages for now
108
- const msgType = msgBody.message_type;
109
- let content = '';
110
- if (msgType === 'text') {
111
- try {
112
- const parsed = JSON.parse(msgBody.content);
113
- content = parsed.text ?? '';
114
- } catch {
115
- content = msgBody.content ?? '';
116
- }
117
- } else {
118
- // Acknowledge non-text silently
119
- res.json({ ok: true });
120
- return;
121
- }
122
-
123
- // Strip @bot mentions
124
- content = content.replace(/@_user_\d+/g, '').trim();
125
- if (!content) { res.json({ ok: true }); return; }
126
-
127
- const chatId = msgBody.chat_id;
128
- const senderId = event.sender?.sender_id?.open_id ?? 'unknown';
129
-
130
- const msg: Message = {
131
- id: `feishu_${msgBody.message_id}`,
132
- role: 'user',
133
- content,
134
- timestamp: parseInt(msgBody.create_time, 10) || Date.now(),
135
- metadata: {
136
- sessionId: `feishu_${chatId}`,
137
- chatId,
138
- userId: senderId,
139
- platform: 'feishu',
140
- messageId: msgBody.message_id,
141
- chatType: msgBody.chat_type, // 'p2p' or 'group'
142
- },
143
- };
144
-
145
- const response = await this.handler(msg);
146
- await this.sendTextMessage(chatId, response.content);
147
- }
148
-
149
- res.json({ ok: true });
150
- } catch (err) {
151
- console.error('[FeishuChannel] Error handling event:', err);
152
- res.status(500).json({ error: 'Internal error' });
153
- }
154
- });
155
-
156
- app.get('/health', (_req, res) => {
157
- res.json({ status: 'ok', channel: 'feishu' });
158
- });
159
-
160
- this.server = app.listen(this.config.port, () => {
161
- console.log(`[FeishuChannel] Listening on port ${this.config.port}`);
162
- });
163
- }
164
-
165
- async stop(): Promise<void> {
166
- if (this.server) {
167
- this.server.close();
168
- this.server = null;
169
- }
170
- }
171
-
172
- /** Get tenant access token (cached) */
173
- private async getAccessToken(): Promise<string> {
174
- if (this.tokenCache && Date.now() < this.tokenCache.expiresAt) {
175
- return this.tokenCache.token;
176
- }
177
-
178
- const resp = await fetch(`${this.config.apiBase}/open-apis/auth/v3/tenant_access_token/internal`, {
179
- method: 'POST',
180
- headers: { 'Content-Type': 'application/json' },
181
- body: JSON.stringify({
182
- app_id: this.config.appId,
183
- app_secret: this.config.appSecret,
184
- }),
185
- });
186
-
187
- const data = await resp.json() as { tenant_access_token: string; expire: number; code: number };
188
- if (data.code !== 0) {
189
- throw new Error(`[FeishuChannel] Failed to get access token: ${JSON.stringify(data)}`);
190
- }
191
-
192
- this.tokenCache = {
193
- token: data.tenant_access_token,
194
- expiresAt: Date.now() + (data.expire - 60) * 1000, // refresh 60s early
195
- };
196
- return this.tokenCache.token;
197
- }
198
-
199
- /** Send a text message to a chat */
200
- async sendTextMessage(chatId: string, text: string): Promise<void> {
201
- const token = await this.getAccessToken();
202
- const resp = await fetch(`${this.config.apiBase}/open-apis/im/v1/messages?receive_id_type=chat_id`, {
203
- method: 'POST',
204
- headers: {
205
- 'Content-Type': 'application/json',
206
- 'Authorization': `Bearer ${token}`,
207
- },
208
- body: JSON.stringify({
209
- receive_id: chatId,
210
- msg_type: 'text',
211
- content: JSON.stringify({ text }),
212
- }),
213
- });
214
-
215
- if (!resp.ok) {
216
- console.error('[FeishuChannel] Failed to send message:', await resp.text());
217
- }
218
- }
219
-
220
- /** Send an interactive card message */
221
- async sendCardMessage(chatId: string, card: Record<string, unknown>): Promise<void> {
222
- const token = await this.getAccessToken();
223
- await fetch(`${this.config.apiBase}/open-apis/im/v1/messages?receive_id_type=chat_id`, {
224
- method: 'POST',
225
- headers: {
226
- 'Content-Type': 'application/json',
227
- 'Authorization': `Bearer ${token}`,
228
- },
229
- body: JSON.stringify({
230
- receive_id: chatId,
231
- msg_type: 'interactive',
232
- content: JSON.stringify(card),
233
- }),
234
- });
235
- }
236
- }
1
+ import { BaseChannel } from './index';
2
+ import type { Message } from '../core/types';
3
+ import * as http from 'http';
4
+ import * as https from 'https';
5
+
6
+ /**
7
+ * Feishu / Lark Channel — v1.1.0
8
+ *
9
+ * Supports:
10
+ * - Event Subscription (webhook) mode for receiving messages
11
+ * - Bot send via Feishu Open API
12
+ * - URL verification challenge
13
+ * - Message card (interactive) responses
14
+ * - Group chat & P2P messaging
15
+ * - Event deduplication
16
+ * - No external dependencies (uses Node.js built-in http/https)
17
+ */
18
+
19
+ export interface FeishuChannelConfig {
20
+ /** Feishu App ID */
21
+ appId?: string;
22
+ /** Feishu App Secret */
23
+ appSecret?: string;
24
+ /** Verification token for event subscription */
25
+ verificationToken?: string;
26
+ /** Encrypt key (optional, for encrypted events) */
27
+ encryptKey?: string;
28
+ /** Webhook server port (default: 8081) */
29
+ port?: number;
30
+ /** API base URL (use 'https://open.larksuite.com' for Lark international) */
31
+ apiBase?: string;
32
+ }
33
+
34
+ interface FeishuTokenCache {
35
+ token: string;
36
+ expiresAt: number;
37
+ }
38
+
39
+ export class FeishuChannel extends BaseChannel {
40
+ readonly type = 'feishu';
41
+ private config: Required<Pick<FeishuChannelConfig, 'port' | 'apiBase'>> & FeishuChannelConfig;
42
+ private server: http.Server | null = null;
43
+ private tokenCache: FeishuTokenCache | null = null;
44
+ private processedEvents = new Set<string>();
45
+
46
+ constructor(config: FeishuChannelConfig = {}) {
47
+ super();
48
+ this.config = {
49
+ appId: config.appId ?? process.env.FEISHU_APP_ID ?? '',
50
+ appSecret: config.appSecret ?? process.env.FEISHU_APP_SECRET ?? '',
51
+ verificationToken: config.verificationToken ?? process.env.FEISHU_VERIFICATION_TOKEN ?? '',
52
+ encryptKey: config.encryptKey ?? process.env.FEISHU_ENCRYPT_KEY,
53
+ port: config.port ?? 8081,
54
+ apiBase: config.apiBase ?? 'https://open.feishu.cn',
55
+ };
56
+ }
57
+
58
+ async start(): Promise<void> {
59
+ if (!this.config.appId || !this.config.appSecret) {
60
+ console.warn('[FeishuChannel] Missing appId/appSecret. Set FEISHU_APP_ID and FEISHU_APP_SECRET.');
61
+ return;
62
+ }
63
+
64
+ this.server = http.createServer(async (req, res) => {
65
+ if (req.method === 'GET' && req.url === '/health') {
66
+ res.writeHead(200, { 'Content-Type': 'application/json' });
67
+ res.end(JSON.stringify({ status: 'ok', channel: 'feishu' }));
68
+ return;
69
+ }
70
+
71
+ if (req.method !== 'POST') {
72
+ res.writeHead(404);
73
+ res.end();
74
+ return;
75
+ }
76
+
77
+ try {
78
+ const body = await this.readBody(req);
79
+ const parsed = JSON.parse(body);
80
+ await this.handleEvent(parsed, res);
81
+ } catch (err) {
82
+ console.error('[FeishuChannel] Error:', err);
83
+ res.writeHead(500, { 'Content-Type': 'application/json' });
84
+ res.end(JSON.stringify({ error: 'Internal error' }));
85
+ }
86
+ });
87
+
88
+ return new Promise((resolve) => {
89
+ this.server!.listen(this.config.port, () => {
90
+ console.log(`[FeishuChannel] Listening on port ${this.config.port}`);
91
+ resolve();
92
+ });
93
+ });
94
+ }
95
+
96
+ async stop(): Promise<void> {
97
+ return new Promise((resolve, reject) => {
98
+ if (!this.server) return resolve();
99
+ this.server.close((err) => (err ? reject(err) : resolve()));
100
+ this.server = null;
101
+ });
102
+ }
103
+
104
+ /** Handle Feishu event */
105
+ private async handleEvent(body: any, res: http.ServerResponse): Promise<void> {
106
+ // URL verification challenge
107
+ if (body.type === 'url_verification') {
108
+ res.writeHead(200, { 'Content-Type': 'application/json' });
109
+ res.end(JSON.stringify({ challenge: body.challenge }));
110
+ return;
111
+ }
112
+
113
+ // Deduplicate events
114
+ const eventId = body.header?.event_id;
115
+ if (eventId && this.processedEvents.has(eventId)) {
116
+ res.writeHead(200, { 'Content-Type': 'application/json' });
117
+ res.end(JSON.stringify({ ok: true }));
118
+ return;
119
+ }
120
+ if (eventId) {
121
+ this.processedEvents.add(eventId);
122
+ if (this.processedEvents.size > 1000) {
123
+ const arr = [...this.processedEvents];
124
+ this.processedEvents = new Set(arr.slice(-500));
125
+ }
126
+ }
127
+
128
+ // Verify token
129
+ if (this.config.verificationToken && body.header?.token !== this.config.verificationToken) {
130
+ res.writeHead(403, { 'Content-Type': 'application/json' });
131
+ res.end(JSON.stringify({ error: 'Invalid verification token' }));
132
+ return;
133
+ }
134
+
135
+ // Handle im.message.receive_v1
136
+ const event = body.event;
137
+ if (body.header?.event_type === 'im.message.receive_v1' && this.handler) {
138
+ const msgBody = event?.message;
139
+ if (!msgBody) {
140
+ res.writeHead(200, { 'Content-Type': 'application/json' });
141
+ res.end(JSON.stringify({ ok: true }));
142
+ return;
143
+ }
144
+
145
+ const msgType = msgBody.message_type;
146
+ let content = '';
147
+ if (msgType === 'text') {
148
+ try {
149
+ const parsed = JSON.parse(msgBody.content);
150
+ content = parsed.text ?? '';
151
+ } catch {
152
+ content = msgBody.content ?? '';
153
+ }
154
+ } else {
155
+ res.writeHead(200, { 'Content-Type': 'application/json' });
156
+ res.end(JSON.stringify({ ok: true }));
157
+ return;
158
+ }
159
+
160
+ // Strip @bot mentions
161
+ content = content.replace(/@_user_\d+/g, '').trim();
162
+ if (!content) {
163
+ res.writeHead(200, { 'Content-Type': 'application/json' });
164
+ res.end(JSON.stringify({ ok: true }));
165
+ return;
166
+ }
167
+
168
+ const chatId = msgBody.chat_id;
169
+ const senderId = event.sender?.sender_id?.open_id ?? 'unknown';
170
+
171
+ const msg: Message = {
172
+ id: `feishu_${msgBody.message_id}`,
173
+ role: 'user',
174
+ content,
175
+ timestamp: parseInt(msgBody.create_time, 10) || Date.now(),
176
+ metadata: {
177
+ sessionId: `feishu_${chatId}`,
178
+ chatId,
179
+ userId: senderId,
180
+ platform: 'feishu',
181
+ messageId: msgBody.message_id,
182
+ chatType: msgBody.chat_type,
183
+ },
184
+ };
185
+
186
+ // Don't block the response
187
+ res.writeHead(200, { 'Content-Type': 'application/json' });
188
+ res.end(JSON.stringify({ ok: true }));
189
+
190
+ try {
191
+ const response = await this.handler(msg);
192
+ await this.sendTextMessage(chatId, response.content);
193
+ } catch (err) {
194
+ console.error('[FeishuChannel] Handler error:', err);
195
+ }
196
+ return;
197
+ }
198
+
199
+ res.writeHead(200, { 'Content-Type': 'application/json' });
200
+ res.end(JSON.stringify({ ok: true }));
201
+ }
202
+
203
+ /** Parse Feishu event body (exported for testing) */
204
+ static parseEventBody(body: any): { type: string; challenge?: string; eventType?: string; content?: string; chatId?: string; senderId?: string; messageId?: string } {
205
+ if (body.type === 'url_verification') {
206
+ return { type: 'url_verification', challenge: body.challenge };
207
+ }
208
+
209
+ const eventType = body.header?.event_type;
210
+ const event = body.event;
211
+
212
+ if (eventType === 'im.message.receive_v1' && event?.message) {
213
+ const msgBody = event.message;
214
+ let content = '';
215
+ if (msgBody.message_type === 'text') {
216
+ try {
217
+ const parsed = JSON.parse(msgBody.content);
218
+ content = parsed.text ?? '';
219
+ } catch {
220
+ content = msgBody.content ?? '';
221
+ }
222
+ }
223
+
224
+ return {
225
+ type: 'message',
226
+ eventType,
227
+ content: content.replace(/@_user_\d+/g, '').trim(),
228
+ chatId: msgBody.chat_id,
229
+ senderId: event.sender?.sender_id?.open_id,
230
+ messageId: msgBody.message_id,
231
+ };
232
+ }
233
+
234
+ return { type: 'unknown', eventType };
235
+ }
236
+
237
+ /** Get tenant access token (cached) */
238
+ private async getAccessToken(): Promise<string> {
239
+ if (this.tokenCache && Date.now() < this.tokenCache.expiresAt) {
240
+ return this.tokenCache.token;
241
+ }
242
+
243
+ const body = JSON.stringify({
244
+ app_id: this.config.appId,
245
+ app_secret: this.config.appSecret,
246
+ });
247
+
248
+ const result = await this.httpsPost(
249
+ `${this.config.apiBase}/open-apis/auth/v3/tenant_access_token/internal`,
250
+ body
251
+ );
252
+
253
+ const data = JSON.parse(result);
254
+ if (data.code !== 0) {
255
+ throw new Error(`[FeishuChannel] Failed to get access token: ${result}`);
256
+ }
257
+
258
+ this.tokenCache = {
259
+ token: data.tenant_access_token,
260
+ expiresAt: Date.now() + (data.expire - 60) * 1000,
261
+ };
262
+ return this.tokenCache.token;
263
+ }
264
+
265
+ /** Send a text message to a chat */
266
+ async sendTextMessage(chatId: string, text: string): Promise<void> {
267
+ const token = await this.getAccessToken();
268
+ const body = JSON.stringify({
269
+ receive_id: chatId,
270
+ msg_type: 'text',
271
+ content: JSON.stringify({ text }),
272
+ });
273
+
274
+ const url = `${this.config.apiBase}/open-apis/im/v1/messages?receive_id_type=chat_id`;
275
+ await this.httpsPostWithAuth(url, body, token);
276
+ }
277
+
278
+ /** Send an interactive card message */
279
+ async sendCardMessage(chatId: string, card: Record<string, unknown>): Promise<void> {
280
+ const token = await this.getAccessToken();
281
+ const body = JSON.stringify({
282
+ receive_id: chatId,
283
+ msg_type: 'interactive',
284
+ content: JSON.stringify(card),
285
+ });
286
+
287
+ const url = `${this.config.apiBase}/open-apis/im/v1/messages?receive_id_type=chat_id`;
288
+ await this.httpsPostWithAuth(url, body, token);
289
+ }
290
+
291
+ /** Read request body */
292
+ private readBody(req: http.IncomingMessage): Promise<string> {
293
+ return new Promise((resolve, reject) => {
294
+ const chunks: Buffer[] = [];
295
+ req.on('data', (chunk: Buffer) => chunks.push(chunk));
296
+ req.on('end', () => resolve(Buffer.concat(chunks).toString('utf-8')));
297
+ req.on('error', reject);
298
+ });
299
+ }
300
+
301
+ /** HTTPS POST */
302
+ private httpsPost(url: string, body: string): Promise<string> {
303
+ return new Promise((resolve, reject) => {
304
+ const parsed = new URL(url);
305
+ const req = https.request({
306
+ hostname: parsed.hostname,
307
+ path: parsed.pathname + parsed.search,
308
+ method: 'POST',
309
+ headers: {
310
+ 'Content-Type': 'application/json',
311
+ 'Content-Length': Buffer.byteLength(body),
312
+ },
313
+ }, (res) => {
314
+ const chunks: Buffer[] = [];
315
+ res.on('data', (chunk: Buffer) => chunks.push(chunk));
316
+ res.on('end', () => resolve(Buffer.concat(chunks).toString()));
317
+ res.on('error', reject);
318
+ });
319
+ req.on('error', reject);
320
+ req.write(body);
321
+ req.end();
322
+ });
323
+ }
324
+
325
+ /** HTTPS POST with Bearer auth */
326
+ private httpsPostWithAuth(url: string, body: string, token: string): Promise<string> {
327
+ return new Promise((resolve, reject) => {
328
+ const parsed = new URL(url);
329
+ const req = https.request({
330
+ hostname: parsed.hostname,
331
+ path: parsed.pathname + parsed.search,
332
+ method: 'POST',
333
+ headers: {
334
+ 'Content-Type': 'application/json',
335
+ 'Content-Length': Buffer.byteLength(body),
336
+ 'Authorization': `Bearer ${token}`,
337
+ },
338
+ }, (res) => {
339
+ const chunks: Buffer[] = [];
340
+ res.on('data', (chunk: Buffer) => chunks.push(chunk));
341
+ res.on('end', () => resolve(Buffer.concat(chunks).toString()));
342
+ res.on('error', reject);
343
+ });
344
+ req.on('error', reject);
345
+ req.write(body);
346
+ req.end();
347
+ });
348
+ }
349
+ }