opc-agent 1.2.1 → 1.3.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.
Files changed (152) hide show
  1. package/CONTRIBUTING.md +75 -75
  2. package/README.md +235 -358
  3. package/README.zh-CN.md +415 -415
  4. package/dist/channels/web.js +256 -256
  5. package/dist/core/knowledge.d.ts +5 -0
  6. package/dist/core/knowledge.js +39 -2
  7. package/dist/deploy/hermes.js +22 -22
  8. package/dist/deploy/openclaw.js +31 -31
  9. package/dist/index.d.ts +0 -4
  10. package/dist/index.js +1 -7
  11. package/dist/providers/index.d.ts +1 -1
  12. package/dist/providers/index.js +158 -14
  13. package/dist/schema/oad.d.ts +3 -3
  14. package/dist/templates/code-reviewer.js +5 -5
  15. package/dist/templates/customer-service.js +2 -2
  16. package/dist/templates/data-analyst.js +5 -5
  17. package/dist/templates/knowledge-base.js +2 -2
  18. package/dist/templates/sales-assistant.js +4 -4
  19. package/dist/templates/teacher.js +6 -6
  20. package/docs/.vitepress/config.ts +103 -103
  21. package/docs/api/cli.md +48 -48
  22. package/docs/api/oad-schema.md +64 -64
  23. package/docs/api/sdk.md +80 -80
  24. package/docs/guide/concepts.md +51 -51
  25. package/docs/guide/configuration.md +79 -79
  26. package/docs/guide/deployment.md +42 -42
  27. package/docs/guide/getting-started.md +44 -44
  28. package/docs/guide/templates.md +28 -28
  29. package/docs/guide/testing.md +84 -84
  30. package/docs/index.md +27 -27
  31. package/docs/zh/api/cli.md +54 -54
  32. package/docs/zh/api/oad-schema.md +87 -87
  33. package/docs/zh/api/sdk.md +102 -102
  34. package/docs/zh/guide/concepts.md +104 -104
  35. package/docs/zh/guide/configuration.md +135 -135
  36. package/docs/zh/guide/deployment.md +81 -81
  37. package/docs/zh/guide/getting-started.md +82 -82
  38. package/docs/zh/guide/templates.md +84 -84
  39. package/docs/zh/guide/testing.md +88 -88
  40. package/docs/zh/index.md +27 -27
  41. package/examples/customer-service-demo/README.md +90 -90
  42. package/examples/customer-service-demo/oad.yaml +107 -107
  43. package/package.json +50 -50
  44. package/src/analytics/index.ts +66 -66
  45. package/src/channels/discord.ts +192 -192
  46. package/src/channels/email.ts +177 -177
  47. package/src/channels/feishu.ts +236 -236
  48. package/src/channels/index.ts +15 -15
  49. package/src/channels/slack.ts +160 -160
  50. package/src/channels/telegram.ts +90 -90
  51. package/src/channels/voice.ts +106 -106
  52. package/src/channels/webhook.ts +199 -199
  53. package/src/channels/websocket.ts +87 -87
  54. package/src/channels/wechat.ts +149 -149
  55. package/src/cli.ts +119 -1
  56. package/src/core/a2a.ts +143 -143
  57. package/src/core/agent.ts +152 -152
  58. package/src/core/analytics-engine.ts +186 -186
  59. package/src/core/auth.ts +57 -57
  60. package/src/core/cache.ts +141 -141
  61. package/src/core/compose.ts +77 -77
  62. package/src/core/config.ts +14 -14
  63. package/src/core/errors.ts +148 -148
  64. package/src/core/hitl.ts +138 -138
  65. package/src/core/logger.ts +57 -57
  66. package/src/core/orchestrator.ts +215 -215
  67. package/src/core/performance.ts +187 -187
  68. package/src/core/rate-limiter.ts +128 -128
  69. package/src/core/room.ts +109 -109
  70. package/src/core/runtime.ts +152 -152
  71. package/src/core/sandbox.ts +101 -101
  72. package/src/core/security.ts +171 -171
  73. package/src/core/types.ts +68 -68
  74. package/src/core/versioning.ts +106 -106
  75. package/src/core/watch.ts +178 -178
  76. package/src/core/workflow.ts +235 -235
  77. package/src/deploy/hermes.ts +156 -156
  78. package/src/deploy/openclaw.ts +200 -200
  79. package/src/i18n/index.ts +216 -216
  80. package/src/index.ts +6 -2
  81. package/src/memory/deepbrain.ts +108 -108
  82. package/src/memory/index.ts +34 -34
  83. package/src/plugins/index.ts +208 -208
  84. package/src/schema/oad.ts +154 -155
  85. package/src/skills/base.ts +16 -16
  86. package/src/skills/document.ts +100 -100
  87. package/src/skills/http.ts +35 -35
  88. package/src/skills/index.ts +27 -27
  89. package/src/skills/scheduler.ts +80 -80
  90. package/src/skills/webhook-trigger.ts +59 -59
  91. package/src/templates/code-reviewer.ts +30 -34
  92. package/src/templates/customer-service.ts +76 -80
  93. package/src/templates/data-analyst.ts +66 -70
  94. package/src/templates/executive-assistant.ts +71 -71
  95. package/src/templates/financial-advisor.ts +60 -60
  96. package/src/templates/knowledge-base.ts +27 -31
  97. package/src/templates/legal-assistant.ts +71 -71
  98. package/src/templates/sales-assistant.ts +75 -79
  99. package/src/templates/teacher.ts +75 -79
  100. package/src/testing/index.ts +181 -181
  101. package/src/tools/calculator.ts +73 -73
  102. package/src/tools/datetime.ts +149 -149
  103. package/src/tools/json-transform.ts +187 -187
  104. package/src/tools/mcp.ts +76 -76
  105. package/src/tools/text-analysis.ts +116 -116
  106. package/src/traces/index.ts +132 -0
  107. package/templates/Dockerfile +15 -15
  108. package/templates/code-reviewer/README.md +27 -27
  109. package/templates/code-reviewer/oad.yaml +41 -41
  110. package/templates/customer-service/README.md +22 -22
  111. package/templates/customer-service/oad.yaml +36 -36
  112. package/templates/docker-compose.yml +21 -21
  113. package/templates/ecommerce-assistant/README.md +45 -45
  114. package/templates/ecommerce-assistant/oad.yaml +47 -47
  115. package/templates/knowledge-base/README.md +28 -28
  116. package/templates/knowledge-base/oad.yaml +38 -38
  117. package/templates/sales-assistant/README.md +26 -26
  118. package/templates/sales-assistant/oad.yaml +43 -43
  119. package/templates/tech-support/README.md +43 -43
  120. package/templates/tech-support/oad.yaml +45 -45
  121. package/tests/a2a.test.ts +66 -66
  122. package/tests/agent.test.ts +72 -72
  123. package/tests/analytics.test.ts +50 -50
  124. package/tests/channel.test.ts +39 -39
  125. package/tests/e2e.test.ts +134 -134
  126. package/tests/errors.test.ts +83 -83
  127. package/tests/hitl.test.ts +71 -71
  128. package/tests/i18n.test.ts +41 -41
  129. package/tests/mcp.test.ts +54 -54
  130. package/tests/oad.test.ts +68 -68
  131. package/tests/performance.test.ts +115 -115
  132. package/tests/plugin.test.ts +74 -74
  133. package/tests/room.test.ts +106 -106
  134. package/tests/runtime.test.ts +42 -42
  135. package/tests/sandbox.test.ts +46 -46
  136. package/tests/security.test.ts +60 -60
  137. package/tests/templates.test.ts +77 -77
  138. package/tests/v070.test.ts +76 -76
  139. package/tests/versioning.test.ts +75 -75
  140. package/tests/voice.test.ts +61 -61
  141. package/tests/webhook.test.ts +29 -29
  142. package/tests/workflow.test.ts +143 -143
  143. package/tsconfig.json +19 -19
  144. package/vitest.config.ts +9 -9
  145. package/dist/core/streaming.d.ts +0 -56
  146. package/dist/core/streaming.js +0 -160
  147. package/dist/tools/gateway.d.ts +0 -28
  148. package/dist/tools/gateway.js +0 -177
  149. package/src/dtv/data.ts +0 -29
  150. package/src/dtv/trust.ts +0 -43
  151. package/src/dtv/value.ts +0 -47
  152. package/src/marketplace/index.ts +0 -223
@@ -1,236 +1,236 @@
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
+
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,15 +1,15 @@
1
- import type { IChannel, Message } from '../core/types';
2
-
3
- export type { IChannel } from '../core/types';
4
-
5
- export abstract class BaseChannel implements IChannel {
6
- abstract type: string;
7
- protected handler: ((message: Message) => Promise<Message>) | null = null;
8
-
9
- onMessage(handler: (message: Message) => Promise<Message>): void {
10
- this.handler = handler;
11
- }
12
-
13
- abstract start(): Promise<void>;
14
- abstract stop(): Promise<void>;
15
- }
1
+ import type { IChannel, Message } from '../core/types';
2
+
3
+ export type { IChannel } from '../core/types';
4
+
5
+ export abstract class BaseChannel implements IChannel {
6
+ abstract type: string;
7
+ protected handler: ((message: Message) => Promise<Message>) | null = null;
8
+
9
+ onMessage(handler: (message: Message) => Promise<Message>): void {
10
+ this.handler = handler;
11
+ }
12
+
13
+ abstract start(): Promise<void>;
14
+ abstract stop(): Promise<void>;
15
+ }