@huo15/dingtalk-connector-pro 1.0.4 → 1.0.7

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 (142) hide show
  1. package/README.en.md +106 -384
  2. package/README.md +14 -18
  3. package/dist/index.js +17 -0
  4. package/dist/openclaw.plugin.json +498 -0
  5. package/dist/package.json +91 -0
  6. package/dist/src/channel.js +415 -0
  7. package/dist/src/config/accounts.js +182 -0
  8. package/dist/src/config/schema.js +135 -0
  9. package/dist/src/core/connection.js +561 -0
  10. package/dist/src/core/message-handler.js +1422 -0
  11. package/dist/src/core/provider.js +59 -0
  12. package/dist/src/core/state.js +49 -0
  13. package/dist/src/directory.js +53 -0
  14. package/dist/src/docs.js +209 -0
  15. package/dist/src/gateway-methods.js +360 -0
  16. package/dist/src/onboarding.js +337 -0
  17. package/dist/src/policy.js +15 -0
  18. package/dist/src/probe.js +144 -0
  19. package/dist/src/reply-dispatcher.js +435 -0
  20. package/dist/src/runtime.js +26 -0
  21. package/dist/src/sdk/helpers.js +237 -0
  22. package/dist/src/sdk/types.js +13 -0
  23. package/dist/src/secret-input.js +13 -0
  24. package/dist/src/services/media/audio.js +40 -0
  25. package/dist/src/services/media/chunk-upload.js +211 -0
  26. package/dist/src/services/media/common.js +120 -0
  27. package/dist/src/services/media/file.js +54 -0
  28. package/dist/src/services/media/image.js +59 -0
  29. package/dist/src/services/media/index.js +9 -0
  30. package/dist/src/services/media/video.js +133 -0
  31. package/dist/src/services/media.js +889 -0
  32. package/dist/src/services/messaging/card.js +234 -0
  33. package/dist/src/services/messaging/index.js +8 -0
  34. package/dist/src/services/messaging/send.js +85 -0
  35. package/dist/src/services/messaging.js +680 -0
  36. package/dist/src/targets.js +38 -0
  37. package/dist/src/types/index.js +1 -0
  38. package/dist/src/utils/agent.js +55 -0
  39. package/dist/src/utils/async.js +40 -0
  40. package/dist/src/utils/constants.js +24 -0
  41. package/dist/src/utils/http-client.js +33 -0
  42. package/dist/src/utils/index.js +7 -0
  43. package/dist/src/utils/logger.js +76 -0
  44. package/dist/src/utils/session.js +95 -0
  45. package/dist/src/utils/token.js +71 -0
  46. package/dist/src/utils/utils-legacy.js +393 -0
  47. package/index.ts +3 -3
  48. package/openclaw.plugin.json +1 -1
  49. package/package.json +16 -5
  50. package/src/channel.js +415 -0
  51. package/src/channel.ts +12 -12
  52. package/src/config/accounts.js +182 -0
  53. package/src/config/accounts.ts +2 -2
  54. package/src/config/schema.js +135 -0
  55. package/src/config/schema.ts +2 -2
  56. package/src/core/connection.js +561 -0
  57. package/src/core/connection.ts +2 -2
  58. package/src/core/message-handler.js +1422 -0
  59. package/src/core/message-handler.ts +12 -12
  60. package/src/core/provider.js +59 -0
  61. package/src/core/provider.ts +4 -4
  62. package/src/core/state.js +49 -0
  63. package/src/directory.js +53 -0
  64. package/src/directory.ts +2 -2
  65. package/src/docs.js +209 -0
  66. package/src/docs.ts +3 -3
  67. package/src/gateway-methods.js +360 -0
  68. package/src/gateway-methods.ts +5 -5
  69. package/src/onboarding.js +337 -0
  70. package/src/onboarding.ts +4 -4
  71. package/src/policy.js +15 -0
  72. package/src/policy.ts +1 -1
  73. package/src/probe.js +144 -0
  74. package/src/probe.ts +2 -2
  75. package/src/reply-dispatcher.js +435 -0
  76. package/src/reply-dispatcher.ts +9 -9
  77. package/src/runtime.js +26 -0
  78. package/src/sdk/helpers.js +237 -0
  79. package/src/sdk/helpers.ts +1 -1
  80. package/src/sdk/types.js +13 -0
  81. package/src/secret-input.js +13 -0
  82. package/src/secret-input.ts +1 -1
  83. package/src/services/media/audio.js +40 -0
  84. package/src/services/media/audio.ts +2 -2
  85. package/src/services/media/chunk-upload.js +211 -0
  86. package/src/services/media/chunk-upload.ts +2 -2
  87. package/src/services/media/common.js +120 -0
  88. package/src/services/media/common.ts +3 -3
  89. package/src/services/media/file.js +54 -0
  90. package/src/services/media/file.ts +2 -2
  91. package/src/services/media/image.js +59 -0
  92. package/src/services/media/image.ts +2 -2
  93. package/src/services/media/index.js +9 -0
  94. package/src/services/media/index.ts +6 -6
  95. package/src/services/media/video.js +133 -0
  96. package/src/services/media/video.ts +2 -2
  97. package/src/services/media.js +889 -0
  98. package/src/services/media.ts +12 -12
  99. package/src/services/messaging/card.js +234 -0
  100. package/src/services/messaging/card.ts +3 -3
  101. package/src/services/messaging/index.js +8 -0
  102. package/src/services/messaging/index.ts +3 -3
  103. package/src/services/messaging/send.js +85 -0
  104. package/src/services/messaging/send.ts +3 -3
  105. package/src/services/messaging.js +680 -0
  106. package/src/services/messaging.ts +8 -8
  107. package/src/targets.js +38 -0
  108. package/src/targets.ts +1 -1
  109. package/src/types/index.js +1 -0
  110. package/src/types/index.ts +1 -1
  111. package/src/utils/agent.js +55 -0
  112. package/src/utils/async.js +40 -0
  113. package/src/utils/constants.js +24 -0
  114. package/src/utils/http-client.js +33 -0
  115. package/src/utils/http-client.ts +1 -1
  116. package/src/utils/index.js +7 -0
  117. package/src/utils/index.ts +4 -4
  118. package/src/utils/logger.js +76 -0
  119. package/src/utils/session.js +95 -0
  120. package/src/utils/session.ts +1 -1
  121. package/src/utils/token.js +71 -0
  122. package/src/utils/token.ts +2 -2
  123. package/src/utils/utils-legacy.js +393 -0
  124. package/src/utils/utils-legacy.ts +8 -8
  125. package/CHANGELOG.md +0 -485
  126. package/SKILL.md +0 -40
  127. package/_meta.json +0 -4
  128. package/docs/AGENT_ROUTING.md +0 -335
  129. package/docs/DEAP_AGENT_GUIDE.en.md +0 -115
  130. package/docs/DEAP_AGENT_GUIDE.md +0 -115
  131. package/docs/images/dingtalk.svg +0 -1
  132. package/docs/images/image-1.png +0 -0
  133. package/docs/images/image-2.png +0 -0
  134. package/docs/images/image-3.png +0 -0
  135. package/docs/images/image-4.png +0 -0
  136. package/docs/images/image-5.png +0 -0
  137. package/docs/images/image-6.png +0 -0
  138. package/docs/images/image-7.png +0 -0
  139. package/install-beta.sh +0 -438
  140. package/install-npm.sh +0 -167
  141. package/src/hooks/init.ts +0 -16
  142. package/tsconfig.json +0 -20
@@ -0,0 +1,360 @@
1
+ /**
2
+ * Gateway Methods 注册
3
+ *
4
+ * 提供钉钉插件的 RPC 接口,允许外部系统、AI Agent 和其他插件调用钉钉功能
5
+ */
6
+ import { resolveDingtalkAccount } from "./config/accounts.js";
7
+ import { DingtalkDocsClient } from "./docs.js";
8
+ import { sendProactive } from "./services/messaging.js";
9
+ import { getUnionId } from "./utils/utils-legacy.js";
10
+ /**
11
+ * 注册所有 Gateway Methods
12
+ */
13
+ export function registerGatewayMethods(api) {
14
+ const log = api.logger;
15
+ // ============ 消息发送类 ============
16
+ /**
17
+ * 主动发送单聊消息
18
+ *
19
+ * @example
20
+ * ```typescript
21
+ * await gateway.call('dingtalk-connector.sendToUser', {
22
+ * userId: 'user123',
23
+ * content: '任务已完成!',
24
+ * useAICard: true
25
+ * });
26
+ * ```
27
+ */
28
+ api.registerGatewayMethod('dingtalk-connector.sendToUser', async ({ context, params, respond }) => {
29
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
30
+ const cfg = loadConfig();
31
+ try {
32
+ const { userId, userIds, content, msgType, title, useAICard, fallbackToNormal, accountId } = params || {};
33
+ const account = resolveDingtalkAccount({ cfg, accountId });
34
+ if (!account.config?.clientId) {
35
+ return respond(false, { error: 'DingTalk not configured' });
36
+ }
37
+ const targetUserIds = userIds || (userId ? [userId] : []);
38
+ if (targetUserIds.length === 0) {
39
+ return respond(false, { error: 'userId or userIds is required' });
40
+ }
41
+ if (!content) {
42
+ return respond(false, { error: 'content is required' });
43
+ }
44
+ // 构建目标
45
+ const target = targetUserIds.length === 1
46
+ ? { userId: targetUserIds[0] }
47
+ : { userIds: targetUserIds };
48
+ const result = await sendProactive(account.config, target, content, {
49
+ msgType,
50
+ title,
51
+ log,
52
+ useAICard: useAICard !== false,
53
+ fallbackToNormal: fallbackToNormal !== false,
54
+ });
55
+ respond(result.ok, result);
56
+ }
57
+ catch (err) {
58
+ log?.error?.(`[Gateway][sendToUser] 错误: ${err.message}`);
59
+ respond(false, { error: err.message });
60
+ }
61
+ });
62
+ /**
63
+ * 主动发送群聊消息
64
+ *
65
+ * @example
66
+ * ```typescript
67
+ * await gateway.call('dingtalk-connector.sendToGroup', {
68
+ * openConversationId: 'cid123',
69
+ * content: '构建失败,请检查日志',
70
+ * useAICard: true
71
+ * });
72
+ * ```
73
+ */
74
+ api.registerGatewayMethod('dingtalk-connector.sendToGroup', async ({ context, params, respond }) => {
75
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
76
+ const cfg = loadConfig();
77
+ try {
78
+ const { openConversationId, content, msgType, title, useAICard, fallbackToNormal, accountId } = params || {};
79
+ const account = resolveDingtalkAccount({ cfg, accountId });
80
+ if (!account.config?.clientId) {
81
+ return respond(false, { error: 'DingTalk not configured' });
82
+ }
83
+ if (!openConversationId) {
84
+ return respond(false, { error: 'openConversationId is required' });
85
+ }
86
+ if (!content) {
87
+ return respond(false, { error: 'content is required' });
88
+ }
89
+ const result = await sendProactive(account.config, { openConversationId }, content, {
90
+ msgType,
91
+ title,
92
+ log,
93
+ useAICard: useAICard !== false,
94
+ fallbackToNormal: fallbackToNormal !== false,
95
+ });
96
+ respond(result.ok, result);
97
+ }
98
+ catch (err) {
99
+ log?.error?.(`[Gateway][sendToGroup] 错误: ${err.message}`);
100
+ console.error(err);
101
+ respond(false, { error: err.message });
102
+ }
103
+ });
104
+ api.registerGatewayMethod('dingtalk-connector.send', async ({ context, params, respond }) => {
105
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
106
+ const cfg = loadConfig();
107
+ try {
108
+ const { target, content, message, msgType, title, useAICard, fallbackToNormal, accountId } = params || {};
109
+ const actualContent = content || message;
110
+ const account = resolveDingtalkAccount({ cfg, accountId });
111
+ log?.info?.(`[Gateway][send] 收到请求: target=${target}, contentLen=${actualContent?.length}`);
112
+ if (!account.config?.clientId) {
113
+ return respond(false, { error: 'DingTalk not configured' });
114
+ }
115
+ if (!target) {
116
+ return respond(false, { error: 'target is required (format: user:<userId> or group:<openConversationId>)' });
117
+ }
118
+ if (!actualContent) {
119
+ return respond(false, { error: 'content is required' });
120
+ }
121
+ const targetStr = String(target);
122
+ let sendTarget;
123
+ if (targetStr.startsWith('user:')) {
124
+ sendTarget = { userId: targetStr.slice(5) };
125
+ }
126
+ else if (targetStr.startsWith('group:')) {
127
+ sendTarget = { openConversationId: targetStr.slice(6) };
128
+ }
129
+ else {
130
+ // 默认当作 userId
131
+ sendTarget = { userId: targetStr };
132
+ }
133
+ const result = await sendProactive(account.config, sendTarget, actualContent, {
134
+ msgType,
135
+ title,
136
+ log,
137
+ useAICard: useAICard !== false,
138
+ fallbackToNormal: fallbackToNormal !== false,
139
+ });
140
+ respond(result.ok, result);
141
+ }
142
+ catch (err) {
143
+ log?.error?.(`[Gateway][send] 错误: ${err.message}`);
144
+ respond(false, { error: err.message });
145
+ }
146
+ });
147
+ // ============ 文档操作类 ============
148
+ api.registerGatewayMethod('dingtalk-connector.docs.read', async ({ context, params, respond }) => {
149
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
150
+ const cfg = loadConfig();
151
+ try {
152
+ const { docId, operatorId: rawOperatorId, accountId } = params || {};
153
+ const account = resolveDingtalkAccount({ cfg, accountId });
154
+ if (!account.config?.clientId) {
155
+ return respond(false, { error: 'DingTalk not configured' });
156
+ }
157
+ if (!docId) {
158
+ return respond(false, { error: 'docId is required' });
159
+ }
160
+ if (!rawOperatorId) {
161
+ return respond(false, { error: 'operatorId (unionId or staffId) is required' });
162
+ }
163
+ // 如果 operatorId 不像 unionId,尝试转换
164
+ let operatorId = rawOperatorId;
165
+ if (!rawOperatorId.includes('$')) {
166
+ const resolved = await getUnionId(rawOperatorId, account.config, log);
167
+ if (resolved)
168
+ operatorId = resolved;
169
+ }
170
+ const client = new DingtalkDocsClient(account.config, log);
171
+ const content = await client.readDoc(docId, operatorId);
172
+ if (content !== null) {
173
+ respond(true, { content });
174
+ }
175
+ else {
176
+ respond(false, { error: 'Failed to read document node' });
177
+ }
178
+ }
179
+ catch (err) {
180
+ log?.error?.(`[Gateway][docs.read] 错误: ${err.message}`);
181
+ respond(false, { error: err.message });
182
+ }
183
+ });
184
+ /**
185
+ * 创建钉钉文档
186
+ *
187
+ * @example
188
+ * ```typescript
189
+ * const result = await gateway.call('dingtalk-connector.docs.create', {
190
+ * spaceId: 'workspace123',
191
+ * title: '会议纪要',
192
+ * content: '今天讨论了...'
193
+ * });
194
+ * console.log('文档ID:', result.docId);
195
+ * ```
196
+ */
197
+ api.registerGatewayMethod('dingtalk-connector.docs.create', async ({ context, params, respond }) => {
198
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
199
+ const cfg = loadConfig();
200
+ try {
201
+ const { spaceId, title, content, accountId } = params || {};
202
+ const account = resolveDingtalkAccount({ cfg, accountId });
203
+ if (!account.config?.clientId) {
204
+ return respond(false, { error: 'DingTalk not configured' });
205
+ }
206
+ if (!spaceId || !title) {
207
+ return respond(false, { error: 'spaceId and title are required' });
208
+ }
209
+ const client = new DingtalkDocsClient(account.config, log);
210
+ const doc = await client.createDoc(spaceId, title, content);
211
+ if (doc) {
212
+ respond(true, doc);
213
+ }
214
+ else {
215
+ respond(false, { error: 'Failed to create document' });
216
+ }
217
+ }
218
+ catch (err) {
219
+ log?.error?.(`[Gateway][docs.create] 错误: ${err.message}`);
220
+ respond(false, { error: err.message });
221
+ }
222
+ });
223
+ /**
224
+ * 向钉钉文档追加内容
225
+ *
226
+ * @example
227
+ * ```typescript
228
+ * await gateway.call('dingtalk-connector.docs.append', {
229
+ * docId: 'doc123',
230
+ * content: '补充内容...'
231
+ * });
232
+ * ```
233
+ */
234
+ api.registerGatewayMethod('dingtalk-connector.docs.append', async ({ context, params, respond }) => {
235
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
236
+ const cfg = loadConfig();
237
+ try {
238
+ const { docId, content, accountId } = params || {};
239
+ const account = resolveDingtalkAccount({ cfg, accountId });
240
+ if (!account.config?.clientId) {
241
+ return respond(false, { error: 'DingTalk not configured' });
242
+ }
243
+ if (!docId || !content) {
244
+ return respond(false, { error: 'docId and content are required' });
245
+ }
246
+ const client = new DingtalkDocsClient(account.config, log);
247
+ const ok = await client.appendToDoc(docId, content);
248
+ respond(ok, ok ? { success: true } : { error: 'Failed to append to document' });
249
+ }
250
+ catch (err) {
251
+ log?.error?.(`[Gateway][docs.append] 错误: ${err.message}`);
252
+ respond(false, { error: err.message });
253
+ }
254
+ });
255
+ /**
256
+ * 搜索钉钉文档
257
+ *
258
+ * @example
259
+ * ```typescript
260
+ * const result = await gateway.call('dingtalk-connector.docs.search', {
261
+ * keyword: '项目规范',
262
+ * spaceId: 'workspace123' // 可选
263
+ * });
264
+ * console.log('找到文档:', result.docs);
265
+ * ```
266
+ */
267
+ api.registerGatewayMethod('dingtalk-connector.docs.search', async ({ context, params, respond }) => {
268
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
269
+ const cfg = loadConfig();
270
+ try {
271
+ const { keyword, spaceId, accountId } = params || {};
272
+ const account = resolveDingtalkAccount({ cfg, accountId });
273
+ if (!account.config?.clientId) {
274
+ return respond(false, { error: 'DingTalk not configured' });
275
+ }
276
+ if (!keyword) {
277
+ return respond(false, { error: 'keyword is required' });
278
+ }
279
+ const client = new DingtalkDocsClient(account.config, log);
280
+ const docs = await client.searchDocs(keyword, spaceId);
281
+ respond(true, { docs });
282
+ }
283
+ catch (err) {
284
+ log?.error?.(`[Gateway][docs.search] 错误: ${err.message}`);
285
+ respond(false, { error: err.message });
286
+ }
287
+ });
288
+ /**
289
+ * 列出空间下的文档
290
+ *
291
+ * @example
292
+ * ```typescript
293
+ * const result = await gateway.call('dingtalk-connector.docs.list', {
294
+ * spaceId: 'workspace123',
295
+ * parentId: 'folder456' // 可选,不传则列出根目录
296
+ * });
297
+ * console.log('文档列表:', result.docs);
298
+ * ```
299
+ */
300
+ api.registerGatewayMethod('dingtalk-connector.docs.list', async ({ context, params, respond }) => {
301
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
302
+ const cfg = loadConfig();
303
+ try {
304
+ const { spaceId, parentId, accountId } = params || {};
305
+ const account = resolveDingtalkAccount({ cfg, accountId });
306
+ if (!account.config?.clientId) {
307
+ return respond(false, { error: 'DingTalk not configured' });
308
+ }
309
+ if (!spaceId) {
310
+ return respond(false, { error: 'spaceId is required' });
311
+ }
312
+ const client = new DingtalkDocsClient(account.config, log);
313
+ const docs = await client.listDocs(spaceId, parentId);
314
+ respond(true, { docs });
315
+ }
316
+ catch (err) {
317
+ log?.error?.(`[Gateway][docs.list] 错误: ${err.message}`);
318
+ respond(false, { error: err.message });
319
+ }
320
+ });
321
+ // ============ 状态检查类 ============
322
+ api.registerGatewayMethod('dingtalk-connector.status', async ({ context, params, respond }) => {
323
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
324
+ const cfg = loadConfig();
325
+ try {
326
+ const accountId = params?.accountId;
327
+ const account = resolveDingtalkAccount({ cfg, accountId });
328
+ const hasClientId = !!account.config?.clientId;
329
+ const hasClientSecret = !!account.config?.clientSecret;
330
+ respond(true, {
331
+ configured: hasClientId && hasClientSecret,
332
+ enabled: account.enabled,
333
+ accountId: account.accountId,
334
+ clientId: hasClientId ? String(account.config.clientId).substring(0, 8) + '...' : undefined,
335
+ });
336
+ }
337
+ catch (err) {
338
+ log?.error?.(`[Gateway][status] 错误: ${err.message}`);
339
+ respond(false, { error: err.message });
340
+ }
341
+ });
342
+ api.registerGatewayMethod('dingtalk-connector.probe', async ({ context, respond }) => {
343
+ const { loadConfig } = await import('openclaw/plugin-sdk/config-runtime');
344
+ const cfg = loadConfig();
345
+ try {
346
+ const account = resolveDingtalkAccount({ cfg });
347
+ if (!account.config?.clientId || !account.config?.clientSecret) {
348
+ return respond(false, { error: 'Not configured' });
349
+ }
350
+ // 尝试获取 access token 来验证连接
351
+ const { getAccessToken } = await import('./utils/utils-legacy.js');
352
+ await getAccessToken(account.config);
353
+ respond(true, { ok: true, details: { clientId: account.config.clientId } });
354
+ }
355
+ catch (err) {
356
+ log?.error?.(`[Gateway][probe] 错误: ${err.message}`);
357
+ respond(false, { ok: false, error: err.message });
358
+ }
359
+ });
360
+ }
@@ -5,10 +5,10 @@
5
5
  */
6
6
 
7
7
  import type { OpenClawPluginApi } from "openclaw/plugin-sdk";
8
- import { resolveDingtalkAccount } from "./config/accounts.ts";
9
- import { DingtalkDocsClient } from "./docs.ts";
10
- import { sendProactive } from "./services/messaging.ts";
11
- import { getUnionId } from "./utils/utils-legacy.ts";
8
+ import { resolveDingtalkAccount } from "./config/accounts.js";
9
+ import { DingtalkDocsClient } from "./docs.js";
10
+ import { sendProactive } from "./services/messaging.js";
11
+ import { getUnionId } from "./utils/utils-legacy.js";
12
12
 
13
13
  /**
14
14
  * 注册所有 Gateway Methods
@@ -391,7 +391,7 @@ export function registerGatewayMethods(api: OpenClawPluginApi) {
391
391
  }
392
392
 
393
393
  // 尝试获取 access token 来验证连接
394
- const { getAccessToken } = await import('./utils/utils-legacy.ts');
394
+ const { getAccessToken } = await import('./utils/utils-legacy.js');
395
395
  await getAccessToken(account.config);
396
396
 
397
397
  respond(true, { ok: true, details: { clientId: account.config.clientId } });