botmux 1.0.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/.env.example +24 -0
- package/LICENSE +21 -0
- package/README.en.md +267 -0
- package/README.md +267 -0
- package/dist/adapters/backend/pty-backend.d.ts +13 -0
- package/dist/adapters/backend/pty-backend.d.ts.map +1 -0
- package/dist/adapters/backend/pty-backend.js +39 -0
- package/dist/adapters/backend/pty-backend.js.map +1 -0
- package/dist/adapters/backend/tmux-backend.d.ts +20 -0
- package/dist/adapters/backend/tmux-backend.d.ts.map +1 -0
- package/dist/adapters/backend/tmux-backend.js +30 -0
- package/dist/adapters/backend/tmux-backend.js.map +1 -0
- package/dist/adapters/backend/types.d.ts +19 -0
- package/dist/adapters/backend/types.d.ts.map +1 -0
- package/dist/adapters/backend/types.js +2 -0
- package/dist/adapters/backend/types.js.map +1 -0
- package/dist/adapters/cli/aiden.d.ts +4 -0
- package/dist/adapters/cli/aiden.d.ts.map +1 -0
- package/dist/adapters/cli/aiden.js +63 -0
- package/dist/adapters/cli/aiden.js.map +1 -0
- package/dist/adapters/cli/claude-code.d.ts +4 -0
- package/dist/adapters/cli/claude-code.d.ts.map +1 -0
- package/dist/adapters/cli/claude-code.js +59 -0
- package/dist/adapters/cli/claude-code.js.map +1 -0
- package/dist/adapters/cli/coco.d.ts +4 -0
- package/dist/adapters/cli/coco.d.ts.map +1 -0
- package/dist/adapters/cli/coco.js +46 -0
- package/dist/adapters/cli/coco.js.map +1 -0
- package/dist/adapters/cli/codex.d.ts +4 -0
- package/dist/adapters/cli/codex.d.ts.map +1 -0
- package/dist/adapters/cli/codex.js +47 -0
- package/dist/adapters/cli/codex.js.map +1 -0
- package/dist/adapters/cli/registry.d.ts +13 -0
- package/dist/adapters/cli/registry.d.ts.map +1 -0
- package/dist/adapters/cli/registry.js +42 -0
- package/dist/adapters/cli/registry.js.map +1 -0
- package/dist/adapters/cli/types.d.ts +39 -0
- package/dist/adapters/cli/types.d.ts.map +1 -0
- package/dist/adapters/cli/types.js +2 -0
- package/dist/adapters/cli/types.js.map +1 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +245 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +24 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +39 -0
- package/dist/config.js.map +1 -0
- package/dist/core/command-handler.d.ts +11 -0
- package/dist/core/command-handler.d.ts.map +1 -0
- package/dist/core/command-handler.js +322 -0
- package/dist/core/command-handler.js.map +1 -0
- package/dist/core/cost-calculator.d.ts +12 -0
- package/dist/core/cost-calculator.d.ts.map +1 -0
- package/dist/core/cost-calculator.js +61 -0
- package/dist/core/cost-calculator.js.map +1 -0
- package/dist/core/scheduler.d.ts +38 -0
- package/dist/core/scheduler.d.ts.map +1 -0
- package/dist/core/scheduler.js +221 -0
- package/dist/core/scheduler.js.map +1 -0
- package/dist/core/session-manager.d.ts +13 -0
- package/dist/core/session-manager.d.ts.map +1 -0
- package/dist/core/session-manager.js +125 -0
- package/dist/core/session-manager.js.map +1 -0
- package/dist/core/types.d.ts +29 -0
- package/dist/core/types.d.ts.map +1 -0
- package/dist/core/types.js +2 -0
- package/dist/core/types.js.map +1 -0
- package/dist/core/worker-pool.d.ts +26 -0
- package/dist/core/worker-pool.d.ts.map +1 -0
- package/dist/core/worker-pool.js +261 -0
- package/dist/core/worker-pool.js.map +1 -0
- package/dist/daemon.d.ts +3 -0
- package/dist/daemon.d.ts.map +1 -0
- package/dist/daemon.js +344 -0
- package/dist/daemon.js.map +1 -0
- package/dist/im/lark/card-builder.d.ts +18 -0
- package/dist/im/lark/card-builder.d.ts.map +1 -0
- package/dist/im/lark/card-builder.js +194 -0
- package/dist/im/lark/card-builder.js.map +1 -0
- package/dist/im/lark/card-handler.d.ts +9 -0
- package/dist/im/lark/card-handler.d.ts.map +1 -0
- package/dist/im/lark/card-handler.js +131 -0
- package/dist/im/lark/card-handler.js.map +1 -0
- package/dist/im/lark/client.d.ts +23 -0
- package/dist/im/lark/client.d.ts.map +1 -0
- package/dist/im/lark/client.js +259 -0
- package/dist/im/lark/client.js.map +1 -0
- package/dist/im/lark/event-dispatcher.d.ts +34 -0
- package/dist/im/lark/event-dispatcher.d.ts.map +1 -0
- package/dist/im/lark/event-dispatcher.js +195 -0
- package/dist/im/lark/event-dispatcher.js.map +1 -0
- package/dist/im/lark/message-parser.d.ts +45 -0
- package/dist/im/lark/message-parser.d.ts.map +1 -0
- package/dist/im/lark/message-parser.js +132 -0
- package/dist/im/lark/message-parser.js.map +1 -0
- package/dist/im/types.d.ts +78 -0
- package/dist/im/types.d.ts.map +1 -0
- package/dist/im/types.js +2 -0
- package/dist/im/types.js.map +1 -0
- package/dist/index-daemon.d.ts +3 -0
- package/dist/index-daemon.d.ts.map +1 -0
- package/dist/index-daemon.js +21 -0
- package/dist/index-daemon.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +16 -0
- package/dist/index.js.map +1 -0
- package/dist/scheduler.d.ts +38 -0
- package/dist/scheduler.d.ts.map +1 -0
- package/dist/scheduler.js +221 -0
- package/dist/scheduler.js.map +1 -0
- package/dist/server.d.ts +3 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +23 -0
- package/dist/server.js.map +1 -0
- package/dist/services/lark-client.d.ts +22 -0
- package/dist/services/lark-client.d.ts.map +1 -0
- package/dist/services/lark-client.js +238 -0
- package/dist/services/lark-client.js.map +1 -0
- package/dist/services/lark-ws.d.ts +5 -0
- package/dist/services/lark-ws.d.ts.map +1 -0
- package/dist/services/lark-ws.js +79 -0
- package/dist/services/lark-ws.js.map +1 -0
- package/dist/services/message-queue.d.ts +9 -0
- package/dist/services/message-queue.d.ts.map +1 -0
- package/dist/services/message-queue.js +77 -0
- package/dist/services/message-queue.js.map +1 -0
- package/dist/services/message-router.d.ts +8 -0
- package/dist/services/message-router.d.ts.map +1 -0
- package/dist/services/message-router.js +73 -0
- package/dist/services/message-router.js.map +1 -0
- package/dist/services/project-scanner.d.ts +12 -0
- package/dist/services/project-scanner.d.ts.map +1 -0
- package/dist/services/project-scanner.js +109 -0
- package/dist/services/project-scanner.js.map +1 -0
- package/dist/services/schedule-store.d.ts +14 -0
- package/dist/services/schedule-store.d.ts.map +1 -0
- package/dist/services/schedule-store.js +89 -0
- package/dist/services/schedule-store.js.map +1 -0
- package/dist/services/session-store.d.ts +8 -0
- package/dist/services/session-store.d.ts.map +1 -0
- package/dist/services/session-store.js +93 -0
- package/dist/services/session-store.js.map +1 -0
- package/dist/tools/close-session.d.ts +22 -0
- package/dist/tools/close-session.d.ts.map +1 -0
- package/dist/tools/close-session.js +38 -0
- package/dist/tools/close-session.js.map +1 -0
- package/dist/tools/create-session.d.ts +28 -0
- package/dist/tools/create-session.d.ts.map +1 -0
- package/dist/tools/create-session.js +40 -0
- package/dist/tools/create-session.js.map +1 -0
- package/dist/tools/get-thread-messages.d.ts +26 -0
- package/dist/tools/get-thread-messages.d.ts.map +1 -0
- package/dist/tools/get-thread-messages.js +33 -0
- package/dist/tools/get-thread-messages.js.map +1 -0
- package/dist/tools/index.d.ts +9 -0
- package/dist/tools/index.d.ts.map +1 -0
- package/dist/tools/index.js +10 -0
- package/dist/tools/index.js.map +1 -0
- package/dist/tools/react-to-message.d.ts +41 -0
- package/dist/tools/react-to-message.d.ts.map +1 -0
- package/dist/tools/react-to-message.js +30 -0
- package/dist/tools/react-to-message.js.map +1 -0
- package/dist/tools/send-to-thread.d.ts +24 -0
- package/dist/tools/send-to-thread.d.ts.map +1 -0
- package/dist/tools/send-to-thread.js +83 -0
- package/dist/tools/send-to-thread.js.map +1 -0
- package/dist/tools/wait-for-reply.d.ts +17 -0
- package/dist/tools/wait-for-reply.d.ts.map +1 -0
- package/dist/tools/wait-for-reply.js +65 -0
- package/dist/tools/wait-for-reply.js.map +1 -0
- package/dist/types.d.ts +85 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +6 -0
- package/dist/types.js.map +1 -0
- package/dist/utils/card-builder.d.ts +16 -0
- package/dist/utils/card-builder.d.ts.map +1 -0
- package/dist/utils/card-builder.js +183 -0
- package/dist/utils/card-builder.js.map +1 -0
- package/dist/utils/idle-detector.d.ts +21 -0
- package/dist/utils/idle-detector.d.ts.map +1 -0
- package/dist/utils/idle-detector.js +95 -0
- package/dist/utils/idle-detector.js.map +1 -0
- package/dist/utils/logger.d.ts +7 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +22 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/message-parser.d.ts +45 -0
- package/dist/utils/message-parser.d.ts.map +1 -0
- package/dist/utils/message-parser.js +132 -0
- package/dist/utils/message-parser.js.map +1 -0
- package/dist/utils/terminal-renderer.d.ts +39 -0
- package/dist/utils/terminal-renderer.d.ts.map +1 -0
- package/dist/utils/terminal-renderer.js +186 -0
- package/dist/utils/terminal-renderer.js.map +1 -0
- package/dist/worker.d.ts +3 -0
- package/dist/worker.d.ts.map +1 -0
- package/dist/worker.js +411 -0
- package/dist/worker.js.map +1 -0
- package/ecosystem.config.cjs +15 -0
- package/package.json +60 -0
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
import { logger } from '../../utils/logger.js';
|
|
2
|
+
export function extractResources(msgType, rawContent) {
|
|
3
|
+
try {
|
|
4
|
+
const parsed = JSON.parse(rawContent);
|
|
5
|
+
if (msgType === 'image') {
|
|
6
|
+
const imageKey = parsed.image_key;
|
|
7
|
+
if (imageKey) {
|
|
8
|
+
return [{ type: 'image', key: imageKey, name: `${imageKey}.jpg` }];
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
if (msgType === 'file') {
|
|
12
|
+
const fileKey = parsed.file_key;
|
|
13
|
+
if (fileKey) {
|
|
14
|
+
return [{ type: 'file', key: fileKey, name: parsed.file_name ?? fileKey }];
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
if (msgType === 'post') {
|
|
18
|
+
const resources = [];
|
|
19
|
+
const { content: contentBlocks } = resolvePostBody(parsed);
|
|
20
|
+
for (const block of contentBlocks) {
|
|
21
|
+
const nodes = Array.isArray(block) ? block : [block];
|
|
22
|
+
for (const node of nodes) {
|
|
23
|
+
if (node.tag === 'img' && node.image_key) {
|
|
24
|
+
resources.push({ type: 'image', key: node.image_key, name: `${node.image_key}.jpg` });
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
return resources;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
catch {
|
|
32
|
+
// ignore parse errors
|
|
33
|
+
}
|
|
34
|
+
return [];
|
|
35
|
+
}
|
|
36
|
+
export function parseEventMessage(data) {
|
|
37
|
+
const { sender, message } = data;
|
|
38
|
+
// Debug: log raw message for non-text types
|
|
39
|
+
if (message.message_type !== 'text') {
|
|
40
|
+
logger.info(`[parser] type=${message.message_type} content=${message.content} keys=${Object.keys(message).join(',')}`);
|
|
41
|
+
}
|
|
42
|
+
const resources = extractResources(message.message_type, message.content);
|
|
43
|
+
const parsed = {
|
|
44
|
+
messageId: message.message_id,
|
|
45
|
+
rootId: message.root_id ?? '',
|
|
46
|
+
senderId: sender.sender_id?.open_id ?? '',
|
|
47
|
+
senderType: sender.sender_type,
|
|
48
|
+
msgType: message.message_type,
|
|
49
|
+
content: extractTextContent(message.message_type, message.content, message.mentions),
|
|
50
|
+
createTime: message.create_time,
|
|
51
|
+
};
|
|
52
|
+
return { parsed, resources };
|
|
53
|
+
}
|
|
54
|
+
export function parseApiMessage(msg) {
|
|
55
|
+
return {
|
|
56
|
+
messageId: msg.message_id ?? '',
|
|
57
|
+
rootId: msg.root_id ?? msg.thread_id ?? '',
|
|
58
|
+
senderId: msg.sender?.id ?? '',
|
|
59
|
+
senderType: msg.sender?.sender_type ?? 'unknown',
|
|
60
|
+
msgType: msg.msg_type ?? 'text',
|
|
61
|
+
content: extractTextContent(msg.msg_type ?? 'text', msg.body?.content ?? ''),
|
|
62
|
+
createTime: msg.create_time ?? '',
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
/** Resolve post body from either wrapped {"zh_cn":{title,content}} or unwrapped {title,content} format */
|
|
66
|
+
function resolvePostBody(parsed) {
|
|
67
|
+
// Unwrapped: has content array directly
|
|
68
|
+
if (Array.isArray(parsed.content)) {
|
|
69
|
+
return { title: parsed.title ?? '', content: parsed.content };
|
|
70
|
+
}
|
|
71
|
+
// Wrapped in language key: {"zh_cn": {title, content}}
|
|
72
|
+
for (const key of Object.keys(parsed)) {
|
|
73
|
+
const val = parsed[key];
|
|
74
|
+
if (val && typeof val === 'object' && Array.isArray(val.content)) {
|
|
75
|
+
return { title: val.title ?? '', content: val.content };
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
return { title: '', content: [] };
|
|
79
|
+
}
|
|
80
|
+
function resolveMentions(text, mentions) {
|
|
81
|
+
if (!mentions || mentions.length === 0) {
|
|
82
|
+
// No mention info available — strip placeholders
|
|
83
|
+
return text.replace(/@_user_\d+/g, '').replace(/\s{2,}/g, ' ').trim();
|
|
84
|
+
}
|
|
85
|
+
let result = text;
|
|
86
|
+
for (const m of mentions) {
|
|
87
|
+
result = result.replace(m.key, `@${m.name}`);
|
|
88
|
+
}
|
|
89
|
+
return result.trim();
|
|
90
|
+
}
|
|
91
|
+
function extractTextContent(msgType, rawContent, mentions) {
|
|
92
|
+
try {
|
|
93
|
+
if (msgType === 'text') {
|
|
94
|
+
const parsed = JSON.parse(rawContent);
|
|
95
|
+
return resolveMentions(parsed.text ?? rawContent, mentions);
|
|
96
|
+
}
|
|
97
|
+
if (msgType === 'post') {
|
|
98
|
+
const parsed = JSON.parse(rawContent);
|
|
99
|
+
const { title, content } = resolvePostBody(parsed);
|
|
100
|
+
const body = content
|
|
101
|
+
.flat()
|
|
102
|
+
.filter((node) => node.tag === 'text' || node.tag === 'a' || node.tag === 'at')
|
|
103
|
+
.map((node) => {
|
|
104
|
+
if (node.tag === 'at')
|
|
105
|
+
return `@${node.user_name ?? 'unknown'}`;
|
|
106
|
+
return node.text ?? node.href ?? '';
|
|
107
|
+
})
|
|
108
|
+
.join('');
|
|
109
|
+
return title ? `${title}\n${body}` : body;
|
|
110
|
+
}
|
|
111
|
+
if (msgType === 'image') {
|
|
112
|
+
return '[图片]';
|
|
113
|
+
}
|
|
114
|
+
if (msgType === 'file') {
|
|
115
|
+
try {
|
|
116
|
+
const p = JSON.parse(rawContent);
|
|
117
|
+
return `[文件: ${p.file_name ?? 'unknown'}]`;
|
|
118
|
+
}
|
|
119
|
+
catch {
|
|
120
|
+
return '[文件]';
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
if (msgType === 'interactive') {
|
|
124
|
+
return '[interactive card]';
|
|
125
|
+
}
|
|
126
|
+
return rawContent;
|
|
127
|
+
}
|
|
128
|
+
catch {
|
|
129
|
+
return rawContent;
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
//# sourceMappingURL=message-parser.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"message-parser.js","sourceRoot":"","sources":["../../../src/im/lark/message-parser.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
export interface ImMessage {
|
|
2
|
+
id: string;
|
|
3
|
+
threadId: string;
|
|
4
|
+
senderId: string;
|
|
5
|
+
senderType: 'user' | 'bot';
|
|
6
|
+
content: string;
|
|
7
|
+
msgType: string;
|
|
8
|
+
attachments?: ImAttachment[];
|
|
9
|
+
createTime: string;
|
|
10
|
+
}
|
|
11
|
+
export interface ImAttachment {
|
|
12
|
+
type: 'image' | 'file';
|
|
13
|
+
path: string;
|
|
14
|
+
name: string;
|
|
15
|
+
}
|
|
16
|
+
export interface ImUser {
|
|
17
|
+
id: string;
|
|
18
|
+
identifier: string;
|
|
19
|
+
}
|
|
20
|
+
export interface ImCard {
|
|
21
|
+
payload: unknown;
|
|
22
|
+
}
|
|
23
|
+
export interface ImCardAction {
|
|
24
|
+
actionType: string;
|
|
25
|
+
threadId: string;
|
|
26
|
+
operatorId?: string;
|
|
27
|
+
value?: Record<string, unknown>;
|
|
28
|
+
}
|
|
29
|
+
export interface ImEventHandler {
|
|
30
|
+
onNewTopic(msg: ImMessage, chatId: string, chatType: 'group' | 'p2p'): Promise<void>;
|
|
31
|
+
onThreadReply(msg: ImMessage, threadId: string): Promise<void>;
|
|
32
|
+
onCardAction(action: ImCardAction): Promise<void>;
|
|
33
|
+
}
|
|
34
|
+
export interface ImCardBuilder {
|
|
35
|
+
buildSessionCard(opts: {
|
|
36
|
+
sessionId: string;
|
|
37
|
+
rootMessageId: string;
|
|
38
|
+
terminalUrl: string;
|
|
39
|
+
title: string;
|
|
40
|
+
}): ImCard;
|
|
41
|
+
buildStreamingCard(opts: {
|
|
42
|
+
sessionId: string;
|
|
43
|
+
rootMessageId: string;
|
|
44
|
+
terminalUrl: string;
|
|
45
|
+
title: string;
|
|
46
|
+
content: string;
|
|
47
|
+
status: 'starting' | 'working' | 'idle';
|
|
48
|
+
}): ImCard;
|
|
49
|
+
buildRepoSelectCard(opts: {
|
|
50
|
+
projects: Array<{
|
|
51
|
+
name: string;
|
|
52
|
+
path: string;
|
|
53
|
+
description: string;
|
|
54
|
+
}>;
|
|
55
|
+
currentCwd: string;
|
|
56
|
+
rootMessageId: string;
|
|
57
|
+
}): ImCard;
|
|
58
|
+
}
|
|
59
|
+
export interface ImAdapter {
|
|
60
|
+
start(handler: ImEventHandler): Promise<void>;
|
|
61
|
+
stop(): Promise<void>;
|
|
62
|
+
cards: ImCardBuilder;
|
|
63
|
+
sendMessage(threadId: string, content: string, format: 'text' | 'rich'): Promise<string>;
|
|
64
|
+
replyMessage(messageId: string, content: string, format: 'text' | 'rich'): Promise<string>;
|
|
65
|
+
updateMessage(messageId: string, content: string): Promise<void>;
|
|
66
|
+
sendCard(threadId: string, card: ImCard): Promise<string>;
|
|
67
|
+
updateCard(messageId: string, card: ImCard): Promise<void>;
|
|
68
|
+
resolveUsers(identifiers: string[]): Promise<ImUser[]>;
|
|
69
|
+
sendDirectMessage(userId: string, content: string): Promise<void>;
|
|
70
|
+
downloadAttachment(messageId: string, resourceKey: string): Promise<string>;
|
|
71
|
+
getThreadMessages(threadId: string, limit: number): Promise<ImMessage[]>;
|
|
72
|
+
/** Add a reaction emoji to a message. Returns the reaction ID. */
|
|
73
|
+
addReaction(messageId: string, emojiType: string): Promise<string>;
|
|
74
|
+
/** Remove a reaction by its reaction ID (not emoji type). */
|
|
75
|
+
removeReaction(messageId: string, reactionId: string): Promise<void>;
|
|
76
|
+
getBotUserId(): string | undefined;
|
|
77
|
+
}
|
|
78
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/im/types.ts"],"names":[],"mappings":"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"}
|
package/dist/im/types.js
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/im/types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index-daemon.d.ts","sourceRoot":"","sources":["../src/index-daemon.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { config as dotenvConfig } from 'dotenv';
|
|
3
|
+
import { join } from 'node:path';
|
|
4
|
+
import { homedir } from 'node:os';
|
|
5
|
+
import { existsSync } from 'node:fs';
|
|
6
|
+
// Load .env BEFORE any other module reads process.env
|
|
7
|
+
// Try ~/.botmux/.env first (npm global install), then CWD/.env (dev)
|
|
8
|
+
const globalEnv = join(homedir(), '.botmux', '.env');
|
|
9
|
+
dotenvConfig({ path: existsSync(globalEnv) ? globalEnv : '.env' });
|
|
10
|
+
async function main() {
|
|
11
|
+
// Dynamic import so config.ts reads env vars AFTER dotenv has loaded them
|
|
12
|
+
const { startDaemon } = await import('./daemon.js');
|
|
13
|
+
const { logger } = await import('./utils/logger.js');
|
|
14
|
+
logger.info('Starting botmux daemon...');
|
|
15
|
+
await startDaemon();
|
|
16
|
+
}
|
|
17
|
+
main().catch((err) => {
|
|
18
|
+
console.error(`Fatal error: ${err}`);
|
|
19
|
+
process.exit(1);
|
|
20
|
+
});
|
|
21
|
+
//# sourceMappingURL=index-daemon.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index-daemon.js","sourceRoot":"","sources":["../src/index-daemon.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,MAAM,IAAI,YAAY,EAAE,MAAM,QAAQ,CAAC;AAChD,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AAErC,sDAAsD;AACtD,qEAAqE;AACrE,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,MAAM,CAAC,CAAC;AACrD,YAAY,CAAC,EAAE,IAAI,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC;AAEnE,KAAK,UAAU,IAAI;IACjB,0EAA0E;IAC1E,MAAM,EAAE,WAAW,EAAE,GAAG,MAAM,MAAM,CAAC,aAAa,CAAC,CAAC;IACpD,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,mBAAmB,CAAC,CAAC;IACrD,MAAM,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;IACzC,MAAM,WAAW,EAAE,CAAC;AACtB,CAAC;AAED,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;IACnB,OAAO,CAAC,KAAK,CAAC,gBAAgB,GAAG,EAAE,CAAC,CAAC;IACrC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA,OAAO,eAAe,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import 'dotenv/config';
|
|
3
|
+
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
|
|
4
|
+
import { createServer } from './server.js';
|
|
5
|
+
import { logger } from './utils/logger.js';
|
|
6
|
+
async function main() {
|
|
7
|
+
const server = createServer();
|
|
8
|
+
const transport = new StdioServerTransport();
|
|
9
|
+
await server.connect(transport);
|
|
10
|
+
logger.info('botmux MCP server running (stdio)');
|
|
11
|
+
}
|
|
12
|
+
main().catch((err) => {
|
|
13
|
+
logger.error(`Fatal error: ${err}`);
|
|
14
|
+
process.exit(1);
|
|
15
|
+
});
|
|
16
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA,OAAO,eAAe,CAAC;AACvB,OAAO,EAAE,oBAAoB,EAAE,MAAM,2CAA2C,CAAC;AACjF,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAC3C,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAE3C,KAAK,UAAU,IAAI;IACjB,MAAM,MAAM,GAAG,YAAY,EAAE,CAAC;IAC9B,MAAM,SAAS,GAAG,IAAI,oBAAoB,EAAE,CAAC;IAC7C,MAAM,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;IAChC,MAAM,CAAC,IAAI,CAAC,mCAAmC,CAAC,CAAC;AACnD,CAAC;AAED,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;IACnB,MAAM,CAAC,KAAK,CAAC,gBAAgB,GAAG,EAAE,CAAC,CAAC;IACpC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import type { ScheduledTask } from './types.js';
|
|
2
|
+
export declare function setExecuteCallback(cb: (task: ScheduledTask) => Promise<void>): void;
|
|
3
|
+
interface ParseResult {
|
|
4
|
+
cron: string;
|
|
5
|
+
type: ScheduledTask['type'];
|
|
6
|
+
prompt: string;
|
|
7
|
+
name: string;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* Parse natural language schedule expression into cron + prompt.
|
|
11
|
+
*
|
|
12
|
+
* Examples:
|
|
13
|
+
* "每日17:50给我帮我看看AI新闻" → { cron: "50 17 * * *", prompt: "帮我看看AI新闻" }
|
|
14
|
+
* "每天9:00 检查服务状态" → { cron: "0 9 * * *", prompt: "检查服务状态" }
|
|
15
|
+
* "每周一10:30 生成周报" → { cron: "30 10 * * 1", prompt: "生成周报" }
|
|
16
|
+
* "工作日每天9:00 检查邮件" → { cron: "0 9 * * 1-5", prompt: "检查邮件" }
|
|
17
|
+
* "每小时 检查服务" → { cron: "0 * * * *", prompt: "检查服务" }
|
|
18
|
+
* "每30分钟 ping" → { cron: "*\/30 * * * *", prompt: "ping" }
|
|
19
|
+
* "每月1号9:00 生成月报" → { cron: "0 9 1 * *", prompt: "生成月报" }
|
|
20
|
+
*/
|
|
21
|
+
export declare function parseNaturalSchedule(input: string): ParseResult | null;
|
|
22
|
+
export declare function startScheduler(): void;
|
|
23
|
+
export declare function stopScheduler(): void;
|
|
24
|
+
export declare function addTask(params: {
|
|
25
|
+
name: string;
|
|
26
|
+
type: ScheduledTask['type'];
|
|
27
|
+
schedule: string;
|
|
28
|
+
prompt: string;
|
|
29
|
+
workingDir: string;
|
|
30
|
+
chatId: string;
|
|
31
|
+
}): ScheduledTask;
|
|
32
|
+
export declare function removeTask(id: string): boolean;
|
|
33
|
+
export declare function enableTask(id: string): boolean;
|
|
34
|
+
export declare function disableTask(id: string): boolean;
|
|
35
|
+
export declare function runTaskNow(id: string): boolean;
|
|
36
|
+
export declare function getNextRun(id: string): Date | null;
|
|
37
|
+
export {};
|
|
38
|
+
//# sourceMappingURL=scheduler.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"scheduler.d.ts","sourceRoot":"","sources":["../src/scheduler.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,YAAY,CAAC;AAShD,wBAAgB,kBAAkB,CAAC,EAAE,EAAE,CAAC,IAAI,EAAE,aAAa,KAAK,OAAO,CAAC,IAAI,CAAC,GAAG,IAAI,CAEnF;AASD,UAAU,WAAW;IACnB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,aAAa,CAAC,MAAM,CAAC,CAAC;IAC5B,MAAM,EAAE,MAAM,CAAC;IACf,IAAI,EAAE,MAAM,CAAC;CACd;AAED;;;;;;;;;;;GAWG;AACH,wBAAgB,oBAAoB,CAAC,KAAK,EAAE,MAAM,GAAG,WAAW,GAAG,IAAI,CA4GtE;AA0CD,wBAAgB,cAAc,IAAI,IAAI,CAQrC;AAED,wBAAgB,aAAa,IAAI,IAAI,CAKpC;AAED,wBAAgB,OAAO,CAAC,MAAM,EAAE;IAC9B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,aAAa,CAAC,MAAM,CAAC,CAAC;IAC5B,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,EAAE,MAAM,CAAC;IACnB,MAAM,EAAE,MAAM,CAAC;CAChB,GAAG,aAAa,CAIhB;AAED,wBAAgB,UAAU,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAG9C;AAED,wBAAgB,UAAU,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAO9C;AAED,wBAAgB,WAAW,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAM/C;AAED,wBAAgB,UAAU,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAa9C;AAED,wBAAgB,UAAU,CAAC,EAAE,EAAE,MAAM,GAAG,IAAI,GAAG,IAAI,CAGlD"}
|
|
@@ -0,0 +1,221 @@
|
|
|
1
|
+
import { Cron } from 'croner';
|
|
2
|
+
import * as scheduleStore from './services/schedule-store.js';
|
|
3
|
+
import { logger } from './utils/logger.js';
|
|
4
|
+
// ─── Active cron instances ──────────────────────────────────────────────────
|
|
5
|
+
const cronJobs = new Map();
|
|
6
|
+
// Callback set by daemon to execute a scheduled task
|
|
7
|
+
let executeCallback = null;
|
|
8
|
+
export function setExecuteCallback(cb) {
|
|
9
|
+
executeCallback = cb;
|
|
10
|
+
}
|
|
11
|
+
// ─── Natural language → cron parser ─────────────────────────────────────────
|
|
12
|
+
const WEEKDAY_MAP = {
|
|
13
|
+
'一': 1, '二': 2, '三': 3, '四': 4, '五': 5, '六': 6, '日': 0, '天': 0,
|
|
14
|
+
'周一': 1, '周二': 2, '周三': 3, '周四': 4, '周五': 5, '周六': 6, '周日': 0,
|
|
15
|
+
};
|
|
16
|
+
/**
|
|
17
|
+
* Parse natural language schedule expression into cron + prompt.
|
|
18
|
+
*
|
|
19
|
+
* Examples:
|
|
20
|
+
* "每日17:50给我帮我看看AI新闻" → { cron: "50 17 * * *", prompt: "帮我看看AI新闻" }
|
|
21
|
+
* "每天9:00 检查服务状态" → { cron: "0 9 * * *", prompt: "检查服务状态" }
|
|
22
|
+
* "每周一10:30 生成周报" → { cron: "30 10 * * 1", prompt: "生成周报" }
|
|
23
|
+
* "工作日每天9:00 检查邮件" → { cron: "0 9 * * 1-5", prompt: "检查邮件" }
|
|
24
|
+
* "每小时 检查服务" → { cron: "0 * * * *", prompt: "检查服务" }
|
|
25
|
+
* "每30分钟 ping" → { cron: "*\/30 * * * *", prompt: "ping" }
|
|
26
|
+
* "每月1号9:00 生成月报" → { cron: "0 9 1 * *", prompt: "生成月报" }
|
|
27
|
+
*/
|
|
28
|
+
export function parseNaturalSchedule(input) {
|
|
29
|
+
let rest = input.trim();
|
|
30
|
+
let cron = '';
|
|
31
|
+
let type = 'cron';
|
|
32
|
+
// Helper: parse time like "17:50", "17:50", "9点", "9点30", "17点30分"
|
|
33
|
+
function parseTime(s) {
|
|
34
|
+
// HH:MM or HH:MM
|
|
35
|
+
let tm = s.match(/^(\d{1,2})[:::](\d{2})\s*(.*)/s);
|
|
36
|
+
if (tm)
|
|
37
|
+
return { hour: parseInt(tm[1]), minute: parseInt(tm[2]), rest: tm[3] };
|
|
38
|
+
// X点Y分 or X点Y
|
|
39
|
+
tm = s.match(/^(\d{1,2})点(\d{1,2})分?\s*(.*)/s);
|
|
40
|
+
if (tm)
|
|
41
|
+
return { hour: parseInt(tm[1]), minute: parseInt(tm[2]), rest: tm[3] };
|
|
42
|
+
// X点 (whole hour)
|
|
43
|
+
tm = s.match(/^(\d{1,2})点\s*(.*)/s);
|
|
44
|
+
if (tm)
|
|
45
|
+
return { hour: parseInt(tm[1]), minute: 0, rest: tm[2] };
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
48
|
+
// Pattern: 工作日/每个工作日 HH:MM
|
|
49
|
+
let m = rest.match(/^(?:每个?工作日|工作日每[天日])\s*(.*)/);
|
|
50
|
+
if (m) {
|
|
51
|
+
const t = parseTime(m[1]);
|
|
52
|
+
if (t) {
|
|
53
|
+
cron = `${t.minute} ${t.hour} * * 1-5`;
|
|
54
|
+
rest = t.rest;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
// Pattern: 每日/每天 HH:MM
|
|
58
|
+
if (!cron) {
|
|
59
|
+
m = rest.match(/^每[天日]\s*(.*)/);
|
|
60
|
+
if (m) {
|
|
61
|
+
const t = parseTime(m[1]);
|
|
62
|
+
if (t) {
|
|
63
|
+
cron = `${t.minute} ${t.hour} * * *`;
|
|
64
|
+
rest = t.rest;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
// Pattern: 每周X HH:MM
|
|
69
|
+
if (!cron) {
|
|
70
|
+
m = rest.match(/^每周([一二三四五六日天])\s*(.*)/);
|
|
71
|
+
if (m) {
|
|
72
|
+
const day = WEEKDAY_MAP[m[1]] ?? 1;
|
|
73
|
+
const t = parseTime(m[2]);
|
|
74
|
+
if (t) {
|
|
75
|
+
cron = `${t.minute} ${t.hour} * * ${day}`;
|
|
76
|
+
rest = t.rest;
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
// Pattern: 每月X号 HH:MM
|
|
81
|
+
if (!cron) {
|
|
82
|
+
m = rest.match(/^每月(\d{1,2})[号日]\s*(.*)/);
|
|
83
|
+
if (m) {
|
|
84
|
+
const dayOfMonth = parseInt(m[1]);
|
|
85
|
+
const t = parseTime(m[2]);
|
|
86
|
+
if (t) {
|
|
87
|
+
cron = `${t.minute} ${t.hour} ${dayOfMonth} * *`;
|
|
88
|
+
rest = t.rest;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
// Pattern: 每N小时
|
|
93
|
+
if (!cron) {
|
|
94
|
+
m = rest.match(/^每(\d+)小时\s*(.*)/);
|
|
95
|
+
if (m) {
|
|
96
|
+
const hours = parseInt(m[1]);
|
|
97
|
+
cron = hours === 1 ? '0 * * * *' : `0 */${hours} * * *`;
|
|
98
|
+
rest = m[2];
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
// Pattern: 每小时
|
|
102
|
+
if (!cron) {
|
|
103
|
+
m = rest.match(/^每小时\s*(.*)/);
|
|
104
|
+
if (m) {
|
|
105
|
+
cron = '0 * * * *';
|
|
106
|
+
rest = m[1];
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
// Pattern: 每N分钟
|
|
110
|
+
if (!cron) {
|
|
111
|
+
m = rest.match(/^每(\d+)分钟\s*(.*)/);
|
|
112
|
+
if (m) {
|
|
113
|
+
cron = `*/${parseInt(m[1])} * * * *`;
|
|
114
|
+
rest = m[2];
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
if (!cron)
|
|
118
|
+
return null;
|
|
119
|
+
// Clean prompt: remove leading connectors like "给我" "帮我" etc.
|
|
120
|
+
let prompt = rest.replace(/^[给帮]我\s*/, '').trim();
|
|
121
|
+
// Remove surrounding quotes
|
|
122
|
+
prompt = prompt.replace(/^["'"「](.+?)["'"」]$/, '$1').trim();
|
|
123
|
+
if (!prompt)
|
|
124
|
+
return null;
|
|
125
|
+
// Auto-generate name from prompt (first 20 chars)
|
|
126
|
+
const name = prompt.length > 20 ? prompt.substring(0, 20) + '...' : prompt;
|
|
127
|
+
return { cron, type, prompt, name };
|
|
128
|
+
}
|
|
129
|
+
// ─── Cron management ────────────────────────────────────────────────────────
|
|
130
|
+
function scheduleCronJob(task) {
|
|
131
|
+
// Stop existing job if any
|
|
132
|
+
stopCronJob(task.id);
|
|
133
|
+
if (!task.enabled)
|
|
134
|
+
return;
|
|
135
|
+
try {
|
|
136
|
+
const job = new Cron(task.schedule, { timezone: 'Asia/Shanghai' }, async () => {
|
|
137
|
+
logger.info(`[scheduler] Task "${task.name}" (${task.id}) triggered`);
|
|
138
|
+
scheduleStore.updateTask(task.id, { lastRunAt: new Date().toISOString() });
|
|
139
|
+
if (executeCallback) {
|
|
140
|
+
try {
|
|
141
|
+
await executeCallback(task);
|
|
142
|
+
}
|
|
143
|
+
catch (err) {
|
|
144
|
+
logger.error(`[scheduler] Failed to execute task "${task.name}": ${err.message}`);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
});
|
|
148
|
+
cronJobs.set(task.id, job);
|
|
149
|
+
const next = job.nextRun();
|
|
150
|
+
logger.info(`[scheduler] Scheduled "${task.name}" (${task.id}): ${task.schedule}, next run: ${next?.toLocaleString('zh-CN', { timeZone: 'Asia/Shanghai' }) ?? 'N/A'}`);
|
|
151
|
+
}
|
|
152
|
+
catch (err) {
|
|
153
|
+
logger.error(`[scheduler] Invalid cron expression for task "${task.name}": ${task.schedule} — ${err.message}`);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
function stopCronJob(taskId) {
|
|
157
|
+
const job = cronJobs.get(taskId);
|
|
158
|
+
if (job) {
|
|
159
|
+
job.stop();
|
|
160
|
+
cronJobs.delete(taskId);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
// ─── Public API ─────────────────────────────────────────────────────────────
|
|
164
|
+
export function startScheduler() {
|
|
165
|
+
const tasks = scheduleStore.listTasks();
|
|
166
|
+
const enabled = tasks.filter(t => t.enabled);
|
|
167
|
+
logger.info(`[scheduler] Starting with ${enabled.length}/${tasks.length} enabled tasks`);
|
|
168
|
+
for (const task of enabled) {
|
|
169
|
+
scheduleCronJob(task);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
export function stopScheduler() {
|
|
173
|
+
for (const [id] of cronJobs) {
|
|
174
|
+
stopCronJob(id);
|
|
175
|
+
}
|
|
176
|
+
logger.info('[scheduler] All cron jobs stopped');
|
|
177
|
+
}
|
|
178
|
+
export function addTask(params) {
|
|
179
|
+
const task = scheduleStore.createTask(params);
|
|
180
|
+
scheduleCronJob(task);
|
|
181
|
+
return task;
|
|
182
|
+
}
|
|
183
|
+
export function removeTask(id) {
|
|
184
|
+
stopCronJob(id);
|
|
185
|
+
return scheduleStore.removeTask(id);
|
|
186
|
+
}
|
|
187
|
+
export function enableTask(id) {
|
|
188
|
+
const task = scheduleStore.getTask(id);
|
|
189
|
+
if (!task)
|
|
190
|
+
return false;
|
|
191
|
+
scheduleStore.updateTask(id, { enabled: true });
|
|
192
|
+
task.enabled = true;
|
|
193
|
+
scheduleCronJob(task);
|
|
194
|
+
return true;
|
|
195
|
+
}
|
|
196
|
+
export function disableTask(id) {
|
|
197
|
+
const task = scheduleStore.getTask(id);
|
|
198
|
+
if (!task)
|
|
199
|
+
return false;
|
|
200
|
+
scheduleStore.updateTask(id, { enabled: false });
|
|
201
|
+
stopCronJob(id);
|
|
202
|
+
return true;
|
|
203
|
+
}
|
|
204
|
+
export function runTaskNow(id) {
|
|
205
|
+
const task = scheduleStore.getTask(id);
|
|
206
|
+
if (!task)
|
|
207
|
+
return false;
|
|
208
|
+
logger.info(`[scheduler] Manual run of task "${task.name}" (${task.id})`);
|
|
209
|
+
scheduleStore.updateTask(id, { lastRunAt: new Date().toISOString() });
|
|
210
|
+
if (executeCallback) {
|
|
211
|
+
executeCallback(task).catch(err => {
|
|
212
|
+
logger.error(`[scheduler] Manual run failed for "${task.name}": ${err.message}`);
|
|
213
|
+
});
|
|
214
|
+
}
|
|
215
|
+
return true;
|
|
216
|
+
}
|
|
217
|
+
export function getNextRun(id) {
|
|
218
|
+
const job = cronJobs.get(id);
|
|
219
|
+
return job?.nextRun() ?? null;
|
|
220
|
+
}
|
|
221
|
+
//# sourceMappingURL=scheduler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"scheduler.js","sourceRoot":"","sources":["../src/scheduler.ts"],"names":[],"mappings":"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"}
|
package/dist/server.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAKpE,wBAAgB,YAAY,IAAI,SAAS,CAoBxC"}
|
package/dist/server.js
ADDED
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
2
|
+
import { validateConfig } from './config.js';
|
|
3
|
+
import { tools } from './tools/index.js';
|
|
4
|
+
import { logger } from './utils/logger.js';
|
|
5
|
+
export function createServer() {
|
|
6
|
+
validateConfig();
|
|
7
|
+
const server = new McpServer({
|
|
8
|
+
name: 'botmux',
|
|
9
|
+
version: '1.0.0',
|
|
10
|
+
});
|
|
11
|
+
// Register all tools
|
|
12
|
+
for (const [name, tool] of Object.entries(tools)) {
|
|
13
|
+
server.tool(name, tool.description, tool.schema.shape, async (args) => {
|
|
14
|
+
logger.info(`Tool called: ${name}`, args);
|
|
15
|
+
const result = await tool.execute(args);
|
|
16
|
+
return {
|
|
17
|
+
content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],
|
|
18
|
+
};
|
|
19
|
+
});
|
|
20
|
+
}
|
|
21
|
+
return server;
|
|
22
|
+
}
|
|
23
|
+
//# sourceMappingURL=server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AACpE,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,KAAK,EAAE,MAAM,kBAAkB,CAAC;AACzC,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAE3C,MAAM,UAAU,YAAY;IAC1B,cAAc,EAAE,CAAC;IAEjB,MAAM,MAAM,GAAG,IAAI,SAAS,CAAC;QAC3B,IAAI,EAAE,QAAQ;QACd,OAAO,EAAE,OAAO;KACjB,CAAC,CAAC;IAEH,qBAAqB;IACrB,KAAK,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QACjD,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,KAAK,EAAE,IAAS,EAAE,EAAE;YACzE,MAAM,CAAC,IAAI,CAAC,gBAAgB,IAAI,EAAE,EAAE,IAAI,CAAC,CAAC;YAC1C,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YACxC,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;aAC5E,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import * as Lark from '@larksuiteoapi/node-sdk';
|
|
2
|
+
export declare function getLarkClient(): Lark.Client;
|
|
3
|
+
export declare function sendMessage(chatId: string, content: string, msgType?: string): Promise<string>;
|
|
4
|
+
export declare function replyMessage(messageId: string, content: string, msgType?: string, replyInThread?: boolean): Promise<string>;
|
|
5
|
+
export declare function addReaction(messageId: string, emojiType: string): Promise<string>;
|
|
6
|
+
export declare function removeReaction(messageId: string, reactionId: string): Promise<void>;
|
|
7
|
+
export declare function sendUserMessage(openId: string, content: string, msgType?: string): Promise<string>;
|
|
8
|
+
export declare function getChatInfo(chatId: string): Promise<{
|
|
9
|
+
userCount: number;
|
|
10
|
+
}>;
|
|
11
|
+
export declare function updateMessage(messageId: string, cardJson: string): Promise<void>;
|
|
12
|
+
export declare function getMessageDetail(messageId: string): Promise<any>;
|
|
13
|
+
export declare function downloadMessageResource(messageId: string, fileKey: string, type: 'image' | 'file', savePath: string): Promise<void>;
|
|
14
|
+
/**
|
|
15
|
+
* Resolve email prefixes (e.g. "shenhan.sh") to Lark open_ids via batch user lookup.
|
|
16
|
+
* Accepts mixed input: items starting with "ou_" are kept as-is; everything else
|
|
17
|
+
* is treated as an email prefix and looked up as `${prefix}@bytedance.com`.
|
|
18
|
+
* Returns an array of open_ids (unresolvable entries are dropped with a warning).
|
|
19
|
+
*/
|
|
20
|
+
export declare function resolveAllowedUsers(raw: string[]): Promise<string[]>;
|
|
21
|
+
export declare function listThreadMessages(chatId: string, rootMessageId: string, pageSize?: number): Promise<any[]>;
|
|
22
|
+
//# sourceMappingURL=lark-client.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"lark-client.d.ts","sourceRoot":"","sources":["../../src/services/lark-client.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,IAAI,MAAM,yBAAyB,CAAC;AAQhD,wBAAgB,aAAa,IAAI,IAAI,CAAC,MAAM,CAQ3C;AAED,wBAAsB,WAAW,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,OAAO,GAAE,MAAe,GAAG,OAAO,CAAC,MAAM,CAAC,CAqB5G;AAED,wBAAsB,YAAY,CAAC,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,OAAO,GAAE,MAAe,EAAE,aAAa,GAAE,OAAe,GAAG,OAAO,CAAC,MAAM,CAAC,CAqBhJ;AAED,wBAAsB,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAYvF;AAED,wBAAsB,cAAc,CAAC,SAAS,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CASzF;AAED,wBAAsB,eAAe,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,OAAO,GAAE,MAAe,GAAG,OAAO,CAAC,MAAM,CAAC,CAqBhH;AAED,wBAAsB,WAAW,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC;IAAE,SAAS,EAAE,MAAM,CAAA;CAAE,CAAC,CAUhF;AAED,wBAAsB,aAAa,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAStF;AAED,wBAAsB,gBAAgB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,CAStE;AAED,wBAAsB,uBAAuB,CAAC,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO,GAAG,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CA2BzI;AAED;;;;;GAKG;AACH,wBAAsB,mBAAmB,CAAC,GAAG,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,CAoC1E;AAED,wBAAsB,kBAAkB,CAAC,MAAM,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,EAAE,QAAQ,GAAE,MAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC,CAuCrH"}
|