wx-bot-cli 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,207 @@
1
+ import fs from 'node:fs';
2
+ import { apiPost, LONG_POLL_TIMEOUT_MS } from './api.js';
3
+ import { sendTextMessage } from './api.js';
4
+ import { loadSession } from './auth.js';
5
+ import { openDb, insertMessage, countMessages } from './db.js';
6
+ import { createIpcServer } from './ipc.js';
7
+ import { DATA_DIR, SESSION_PATH, DB_PATH, SOCKET_PATH, PID_PATH, LOG_PATH } from './paths.js';
8
+ export function createUserSessionState() {
9
+ return {
10
+ activeUser: null,
11
+ sessions: new Map(),
12
+ lastPollAt: new Date().toISOString(),
13
+ sessionExpired: false,
14
+ startedAt: Date.now(),
15
+ };
16
+ }
17
+ export function processInboundMessage(state, msg) {
18
+ state.activeUser = msg.fromUserId;
19
+ state.sessions.set(msg.fromUserId, {
20
+ contextToken: msg.contextToken,
21
+ sentCount: 0,
22
+ exhausted: false,
23
+ });
24
+ }
25
+ export function recordOutboundSent(state, userId) {
26
+ const us = state.sessions.get(userId);
27
+ if (!us)
28
+ return 0;
29
+ us.sentCount++;
30
+ return 10 - us.sentCount;
31
+ }
32
+ export function shouldAutoNotify(state, userId) {
33
+ return (state.sessions.get(userId)?.sentCount ?? 0) === 9;
34
+ }
35
+ export function getEffectiveRemaining(state, userId) {
36
+ const us = state.sessions.get(userId);
37
+ if (!us)
38
+ return 0;
39
+ if (us.exhausted)
40
+ return 0;
41
+ return 10 - us.sentCount;
42
+ }
43
+ // ---- Logger -----------------------------------------------------------------
44
+ function logLine(level, msg, extra) {
45
+ const line = JSON.stringify({ ts: new Date().toISOString(), level, msg, ...extra });
46
+ fs.appendFileSync(LOG_PATH, line + '\n');
47
+ }
48
+ // ---- Main daemon ------------------------------------------------------------
49
+ export async function runService() {
50
+ fs.mkdirSync(DATA_DIR, { recursive: true });
51
+ logLine('INFO', 'service starting', { pid: process.pid });
52
+ const session = loadSession(SESSION_PATH);
53
+ if (!session) {
54
+ logLine('ERROR', 'no session found, exiting');
55
+ process.stderr.write('wxbot: no session found. Run wxbot login first.\n');
56
+ process.exit(1);
57
+ }
58
+ // Write PID file
59
+ fs.writeFileSync(PID_PATH, String(process.pid), 'utf-8');
60
+ const db = openDb(DB_PATH);
61
+ const state = createUserSessionState();
62
+ // ---- IPC server -----------------------------------------------------------
63
+ const server = createIpcServer(SOCKET_PATH, async (req) => {
64
+ if (req.type === 'status') {
65
+ const activeUs = state.activeUser ? state.sessions.get(state.activeUser) : undefined;
66
+ const currentSentCount = activeUs?.sentCount ?? 0;
67
+ const exhausted = activeUs?.exhausted ?? false;
68
+ return {
69
+ running: true,
70
+ pid: process.pid,
71
+ accountId: session.accountId,
72
+ lastPollAt: state.lastPollAt,
73
+ activeUser: state.activeUser,
74
+ totalMessages: countMessages(db),
75
+ uptime: Math.floor((Date.now() - state.startedAt) / 1000),
76
+ sessionExpired: state.sessionExpired,
77
+ currentSentCount,
78
+ exhausted,
79
+ remaining: exhausted ? 0 : 10 - currentSentCount,
80
+ };
81
+ }
82
+ if (req.type === 'send') {
83
+ if (!state.activeUser) {
84
+ return { ok: false, reason: 'no_active_user', message: '还没有收到任何消息,无法确定发送对象' };
85
+ }
86
+ const us = state.sessions.get(state.activeUser);
87
+ if (!us || us.exhausted) {
88
+ return { ok: false, reason: 'session_exhausted', message: '当前会话已满,等待用户回复以开启新会话' };
89
+ }
90
+ try {
91
+ await sendTextMessage({
92
+ baseUrl: session.baseUrl,
93
+ token: session.token,
94
+ toUserId: state.activeUser,
95
+ text: req.text,
96
+ contextToken: us.contextToken,
97
+ });
98
+ }
99
+ catch (err) {
100
+ logLine('ERROR', 'send failed', { err: String(err) });
101
+ return { ok: false, reason: 'api_error', message: String(err) };
102
+ }
103
+ const remaining = recordOutboundSent(state, state.activeUser);
104
+ insertMessage(db, {
105
+ ts: new Date().toISOString(),
106
+ direction: 'out',
107
+ user_id: state.activeUser,
108
+ text: req.text,
109
+ context_token: us.contextToken,
110
+ });
111
+ // Auto-notification at 9th message
112
+ if (shouldAutoNotify(state, state.activeUser)) {
113
+ const noticeText = '您好,当前会话已达到 10 条消息上限,请回复我一条消息以开启新会话。';
114
+ try {
115
+ await sendTextMessage({
116
+ baseUrl: session.baseUrl,
117
+ token: session.token,
118
+ toUserId: state.activeUser,
119
+ text: noticeText,
120
+ contextToken: us.contextToken,
121
+ });
122
+ insertMessage(db, {
123
+ ts: new Date().toISOString(),
124
+ direction: 'out',
125
+ user_id: state.activeUser,
126
+ text: noticeText,
127
+ context_token: us.contextToken,
128
+ });
129
+ }
130
+ catch (err) {
131
+ logLine('ERROR', 'auto-notification send failed', { err: String(err) });
132
+ }
133
+ us.exhausted = true;
134
+ }
135
+ return { ok: true, remaining };
136
+ }
137
+ return { ok: false, reason: 'api_error', message: 'unknown request type' };
138
+ });
139
+ await new Promise((resolve, reject) => {
140
+ server.on('error', reject);
141
+ server.listen(SOCKET_PATH, resolve);
142
+ });
143
+ logLine('INFO', 'IPC server listening', { socket: SOCKET_PATH });
144
+ // ---- Long-poll loop -------------------------------------------------------
145
+ let getUpdatesBuf = '';
146
+ let consecutiveFailures = 0;
147
+ const MAX_FAILURES = 3;
148
+ const BACKOFF_MS = 30_000;
149
+ while (!state.sessionExpired) {
150
+ try {
151
+ const resp = await apiPost({
152
+ baseUrl: session.baseUrl,
153
+ endpoint: 'ilink/bot/getupdates',
154
+ body: {
155
+ get_updates_buf: getUpdatesBuf,
156
+ base_info: { channel_version: 'standalone' },
157
+ },
158
+ token: session.token,
159
+ timeoutMs: LONG_POLL_TIMEOUT_MS + 5_000,
160
+ });
161
+ if (resp.errcode === -14 || resp.ret === -14) {
162
+ logLine('ERROR', 'session expired (-14)');
163
+ state.sessionExpired = true;
164
+ break;
165
+ }
166
+ if (resp.get_updates_buf)
167
+ getUpdatesBuf = resp.get_updates_buf;
168
+ state.lastPollAt = new Date().toISOString();
169
+ consecutiveFailures = 0;
170
+ for (const msg of resp.msgs ?? []) {
171
+ if (msg.message_type === 2)
172
+ continue; // skip bot's own
173
+ const from = msg.from_user_id ?? '';
174
+ const contextToken = msg.context_token ?? '';
175
+ if (from && contextToken) {
176
+ processInboundMessage(state, { fromUserId: from, contextToken });
177
+ }
178
+ const texts = (msg.item_list ?? [])
179
+ .filter((i) => i.type === 1)
180
+ .map((i) => i.text_item?.text ?? '')
181
+ .filter(Boolean);
182
+ if (texts.length > 0 && from) {
183
+ insertMessage(db, {
184
+ ts: new Date().toISOString(),
185
+ direction: 'in',
186
+ user_id: from,
187
+ text: texts.join(' '),
188
+ context_token: contextToken || null,
189
+ });
190
+ logLine('INFO', 'inbound message stored', { from, textLen: texts.join(' ').length });
191
+ }
192
+ }
193
+ }
194
+ catch (err) {
195
+ consecutiveFailures++;
196
+ logLine('WARN', 'getUpdates error', { err: String(err), consecutiveFailures });
197
+ if (consecutiveFailures >= MAX_FAILURES) {
198
+ logLine('WARN', `pausing ${BACKOFF_MS}ms after ${MAX_FAILURES} failures`);
199
+ await new Promise((r) => setTimeout(r, BACKOFF_MS));
200
+ consecutiveFailures = 0;
201
+ }
202
+ }
203
+ }
204
+ logLine('INFO', 'poll loop stopped');
205
+ // Keep server alive so status IPC still works
206
+ }
207
+ //# sourceMappingURL=service.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"service.js","sourceRoot":"","sources":["../src/service.ts"],"names":[],"mappings":"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"}
package/dist/tui.d.ts ADDED
@@ -0,0 +1,6 @@
1
+ type Props = {
2
+ onLoginRequested?: () => void;
3
+ };
4
+ export declare function App({ onLoginRequested }: Props): import("react/jsx-runtime").JSX.Element;
5
+ export {};
6
+ //# sourceMappingURL=tui.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tui.d.ts","sourceRoot":"","sources":["../src/tui.tsx"],"names":[],"mappings":"AAgBA,KAAK,KAAK,GAAG;IACX,gBAAgB,CAAC,EAAE,MAAM,IAAI,CAAC;CAC/B,CAAC;AAEF,wBAAgB,GAAG,CAAC,EAAE,gBAAgB,EAAE,EAAE,KAAK,2CAqI9C"}
package/dist/tui.js ADDED
@@ -0,0 +1,87 @@
1
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
2
+ import { useState, useEffect } from 'react';
3
+ import { Box, Text, useApp, useInput } from 'ink';
4
+ import { openDb, getRecentMessages, countMessages } from './db.js';
5
+ import { sendIpcRequest } from './ipc.js';
6
+ import { loadSession } from './auth.js';
7
+ import { DB_PATH, SOCKET_PATH, SESSION_PATH } from './paths.js';
8
+ const POLL_INTERVAL = 2000;
9
+ export function App({ onLoginRequested }) {
10
+ const { exit } = useApp();
11
+ const [messages, setMessages] = useState([]);
12
+ const [status, setStatus] = useState({ state: 'stopped' });
13
+ const [totalMessages, setTotalMessages] = useState(0);
14
+ const session = loadSession(SESSION_PATH);
15
+ useInput((input) => {
16
+ if (input === 'q' || input === 'Q')
17
+ exit();
18
+ if (input === 'r' || input === 'R')
19
+ pollAll();
20
+ if ((input === 'l' || input === 'L') && !session) {
21
+ onLoginRequested?.();
22
+ exit();
23
+ }
24
+ });
25
+ async function pollAll() {
26
+ // Poll SQLite for messages
27
+ try {
28
+ const db = openDb(DB_PATH, true);
29
+ const rows = getRecentMessages(db, 50).reverse();
30
+ const total = countMessages(db);
31
+ db.close();
32
+ setMessages(rows);
33
+ setTotalMessages(total);
34
+ }
35
+ catch { /* DB not ready yet */ }
36
+ // Poll IPC for service status
37
+ try {
38
+ const resp = await sendIpcRequest(SOCKET_PATH, { type: 'status' }, 2000);
39
+ if (resp.sessionExpired) {
40
+ setStatus({ state: 'expired' });
41
+ }
42
+ else {
43
+ setStatus({ state: 'running', data: resp });
44
+ }
45
+ }
46
+ catch {
47
+ setStatus({ state: 'stopped' });
48
+ }
49
+ }
50
+ useEffect(() => {
51
+ pollAll();
52
+ const timer = setInterval(pollAll, POLL_INTERVAL);
53
+ return () => clearInterval(timer);
54
+ }, []);
55
+ const statusDot = () => {
56
+ switch (status.state) {
57
+ case 'running': return _jsx(Text, { color: "green", children: "\u25CF \u670D\u52A1\u8FD0\u884C\u4E2D" });
58
+ case 'expired': return _jsx(Text, { color: "yellow", children: "\u26A0 \u4F1A\u8BDD\u8FC7\u671F" });
59
+ case 'error': return _jsx(Text, { color: "yellow", children: "\u26A0 \u9519\u8BEF" });
60
+ default: return _jsx(Text, { color: "red", children: "\u25CB \u5DF2\u505C\u6B62" });
61
+ }
62
+ };
63
+ const relativeTime = (isoStr) => {
64
+ const diff = Math.floor((Date.now() - new Date(isoStr).getTime()) / 1000);
65
+ if (diff < 60)
66
+ return `${diff}s 前`;
67
+ return `${Math.floor(diff / 60)}m 前`;
68
+ };
69
+ const runningData = status.state === 'running' ? status.data : null;
70
+ const remaining = runningData && !runningData.exhausted ? runningData.remaining : null;
71
+ if (!session) {
72
+ return (_jsxs(Box, { flexDirection: "column", padding: 2, children: [_jsx(Box, { borderStyle: "single", paddingX: 1, marginBottom: 1, children: _jsx(Text, { bold: true, children: "wx bot cli" }) }), _jsxs(Box, { flexDirection: "column", paddingX: 1, gap: 1, children: [_jsx(Text, { color: "red", children: "\u25CB \u672A\u767B\u5F55" }), _jsx(Text, { children: "\u5C1A\u672A\u767B\u5F55\u5FAE\u4FE1\u8D26\u53F7\u3002" }), _jsx(Text, { dimColor: true, children: "\u6309 [L] \u626B\u7801\u767B\u5F55 / \u6309 [Q] \u9000\u51FA" })] })] }));
73
+ }
74
+ return (_jsxs(Box, { flexDirection: "column", height: "100%", children: [_jsxs(Box, { borderStyle: "single", paddingX: 1, flexDirection: "column", children: [_jsxs(Box, { gap: 2, children: [_jsx(Text, { bold: true, children: "wx bot cli" }), _jsx(Text, { children: " \u2022 " }), statusDot()] }), _jsxs(Box, { gap: 2, children: [_jsxs(Text, { dimColor: true, children: ["\u8D26\u53F7: ", session?.accountId ?? '(未登录)'] }), runningData && (_jsxs(_Fragment, { children: [_jsxs(Text, { dimColor: true, children: [" \u4E0A\u6B21\u8F6E\u8BE2: ", relativeTime(runningData.lastPollAt)] }), _jsxs(Text, { dimColor: true, children: [" \u5171 ", totalMessages, " \u6761\u6D88\u606F"] })] }))] })] }), _jsx(Box, { flexGrow: 1, flexDirection: "column", paddingX: 1, overflowY: "hidden", children: messages.length === 0 ? (_jsx(Text, { dimColor: true, children: "\uFF08\u6682\u65E0\u6D88\u606F\uFF09" })) : (messages.map((row, i) => {
75
+ const time = new Date(row.created_at).toLocaleTimeString('zh-CN', { hour: '2-digit', minute: '2-digit' });
76
+ const dir = row.direction === 'in'
77
+ ? _jsxs(Text, { color: "cyan", children: ["\u2192 [", row.user_id, "]"] })
78
+ : _jsx(Text, { color: "green", children: "\u2190 [Bot] " });
79
+ // Show remaining annotation on most recent outbound message when remaining <= 3
80
+ const isLastOutbound = row.direction === 'out' && i === messages.length - 1;
81
+ const showWarning = isLastOutbound && remaining !== null && remaining <= 3;
82
+ return (_jsxs(Box, { gap: 1, children: [_jsx(Text, { dimColor: true, children: time }), dir, _jsx(Text, { children: row.text }), showWarning && _jsxs(Text, { color: "yellow", children: [" \u26A0\uFE0F \u5269 ", remaining, " \u6761"] })] }, row.id));
83
+ })) }), _jsxs(Box, { borderStyle: "single", paddingX: 1, children: [_jsx(Text, { dimColor: true, children: "[q] \u9000\u51FA [r] \u5237\u65B0 " }), runningData?.activeUser
84
+ ? _jsxs(Text, { dimColor: true, children: ["\u6D3B\u8DC3\u7528\u6237: ", runningData.activeUser] })
85
+ : _jsx(Text, { dimColor: true, children: "\u6D3B\u8DC3\u7528\u6237: (\u65E0)" })] })] }));
86
+ }
87
+ //# sourceMappingURL=tui.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tui.js","sourceRoot":"","sources":["../src/tui.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,54 @@
1
+ export type Session = {
2
+ token: string;
3
+ baseUrl: string;
4
+ accountId: string;
5
+ userId?: string;
6
+ savedAt: string;
7
+ };
8
+ export type MessageRow = {
9
+ id: number;
10
+ ts: string;
11
+ direction: 'in' | 'out';
12
+ user_id: string;
13
+ text: string;
14
+ context_token: string | null;
15
+ created_at: number;
16
+ };
17
+ export type IpcSendRequest = {
18
+ type: 'send';
19
+ text: string;
20
+ };
21
+ export type IpcStatusRequest = {
22
+ type: 'status';
23
+ };
24
+ export type IpcRequest = IpcSendRequest | IpcStatusRequest;
25
+ export type IpcSendOk = {
26
+ ok: true;
27
+ remaining: number;
28
+ };
29
+ export type IpcSendFail = {
30
+ ok: false;
31
+ reason: 'no_active_user' | 'session_exhausted' | 'api_error';
32
+ message: string;
33
+ };
34
+ export type IpcSendResponse = IpcSendOk | IpcSendFail;
35
+ export type IpcStatusResponse = {
36
+ running: true;
37
+ pid: number;
38
+ accountId: string;
39
+ lastPollAt: string;
40
+ activeUser: string | null;
41
+ totalMessages: number;
42
+ uptime: number;
43
+ sessionExpired: boolean;
44
+ currentSentCount: number;
45
+ exhausted: boolean;
46
+ remaining: number;
47
+ };
48
+ export type IpcResponse = IpcSendResponse | IpcStatusResponse;
49
+ export type UserSession = {
50
+ contextToken: string;
51
+ sentCount: number;
52
+ exhausted: boolean;
53
+ };
54
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAEA,MAAM,MAAM,OAAO,GAAG;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAIF,MAAM,MAAM,UAAU,GAAG;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,EAAE,EAAE,MAAM,CAAC;IACX,SAAS,EAAE,IAAI,GAAG,KAAK,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,CAAC;IACb,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,UAAU,EAAE,MAAM,CAAC;CACpB,CAAC;AAIF,MAAM,MAAM,cAAc,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,CAAC;AAC5D,MAAM,MAAM,gBAAgB,GAAG;IAAE,IAAI,EAAE,QAAQ,CAAA;CAAE,CAAC;AAClD,MAAM,MAAM,UAAU,GAAG,cAAc,GAAG,gBAAgB,CAAC;AAE3D,MAAM,MAAM,SAAS,GAAG;IAAE,EAAE,EAAE,IAAI,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,CAAC;AACxD,MAAM,MAAM,WAAW,GAAG;IACxB,EAAE,EAAE,KAAK,CAAC;IACV,MAAM,EAAE,gBAAgB,GAAG,mBAAmB,GAAG,WAAW,CAAC;IAC7D,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AACF,MAAM,MAAM,eAAe,GAAG,SAAS,GAAG,WAAW,CAAC;AAEtD,MAAM,MAAM,iBAAiB,GAAG;IAC9B,OAAO,EAAE,IAAI,CAAC;IACd,GAAG,EAAE,MAAM,CAAC;IACZ,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,aAAa,EAAE,MAAM,CAAC;IACtB,MAAM,EAAE,MAAM,CAAC;IACf,cAAc,EAAE,OAAO,CAAC;IACxB,gBAAgB,EAAE,MAAM,CAAC;IACzB,SAAS,EAAE,OAAO,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;CACnB,CAAC;AAEF,MAAM,MAAM,WAAW,GAAG,eAAe,GAAG,iBAAiB,CAAC;AAI9D,MAAM,MAAM,WAAW,GAAG;IACxB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,OAAO,CAAC;CACpB,CAAC"}
package/dist/types.js ADDED
@@ -0,0 +1,3 @@
1
+ // ---- Session ----------------------------------------------------------------
2
+ export {};
3
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,gFAAgF"}
package/package.json ADDED
@@ -0,0 +1,33 @@
1
+ {
2
+ "name": "wx-bot-cli",
3
+ "version": "0.1.0",
4
+ "description": "WeChat AI Bot CLI and TUI dashboard",
5
+ "license": "MIT",
6
+ "type": "module",
7
+ "bin": {
8
+ "wxbot": "./dist/bin.js"
9
+ },
10
+ "files": ["dist/"],
11
+ "engines": { "node": ">=22" },
12
+ "scripts": {
13
+ "build": "tsc",
14
+ "typecheck": "tsc --noEmit",
15
+ "test": "vitest run --coverage",
16
+ "prepublishOnly": "npm run typecheck && npm run build"
17
+ },
18
+ "dependencies": {
19
+ "better-sqlite3": "^9.6.0",
20
+ "commander": "^12.0.0",
21
+ "ink": "^5.0.0",
22
+ "qrcode-terminal": "^0.12.0",
23
+ "react": "^18.3.0"
24
+ },
25
+ "devDependencies": {
26
+ "@types/better-sqlite3": "^7.6.0",
27
+ "@types/node": "^22.0.0",
28
+ "@types/react": "^18.3.0",
29
+ "@vitest/coverage-v8": "^3.0.0",
30
+ "typescript": "^5.8.0",
31
+ "vitest": "^3.0.0"
32
+ }
33
+ }