feique 1.3.3 → 1.4.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/README.en.md +3 -2
- package/README.md +3 -2
- package/dist/backend/claude.js +28 -54
- package/dist/backend/claude.js.map +1 -1
- package/dist/backend/factory.d.ts +28 -0
- package/dist/backend/factory.js +61 -0
- package/dist/backend/factory.js.map +1 -1
- package/dist/backend/probe.d.ts +29 -0
- package/dist/backend/probe.js +99 -0
- package/dist/backend/probe.js.map +1 -0
- package/dist/bridge/admin-config.d.ts +47 -0
- package/dist/bridge/admin-config.js +141 -0
- package/dist/bridge/admin-config.js.map +1 -0
- package/dist/bridge/collab-commands.d.ts +42 -0
- package/dist/bridge/collab-commands.js +254 -0
- package/dist/bridge/collab-commands.js.map +1 -0
- package/dist/bridge/commands.d.ts +1 -1
- package/dist/bridge/commands.js +3 -0
- package/dist/bridge/commands.js.map +1 -1
- package/dist/bridge/feishu-commands.d.ts +27 -0
- package/dist/bridge/feishu-commands.js +462 -0
- package/dist/bridge/feishu-commands.js.map +1 -0
- package/dist/bridge/lifecycle.d.ts +46 -0
- package/dist/bridge/lifecycle.js +228 -0
- package/dist/bridge/lifecycle.js.map +1 -0
- package/dist/bridge/memory-commands.d.ts +26 -0
- package/dist/bridge/memory-commands.js +330 -0
- package/dist/bridge/memory-commands.js.map +1 -0
- package/dist/bridge/reply-builders.d.ts +30 -0
- package/dist/bridge/reply-builders.js +72 -0
- package/dist/bridge/reply-builders.js.map +1 -0
- package/dist/bridge/run-pipeline.d.ts +86 -0
- package/dist/bridge/run-pipeline.js +442 -0
- package/dist/bridge/run-pipeline.js.map +1 -0
- package/dist/bridge/run-scheduler.d.ts +47 -0
- package/dist/bridge/run-scheduler.js +121 -0
- package/dist/bridge/run-scheduler.js.map +1 -0
- package/dist/bridge/service-utils.d.ts +47 -0
- package/dist/bridge/service-utils.js +309 -0
- package/dist/bridge/service-utils.js.map +1 -0
- package/dist/bridge/service.d.ts +114 -66
- package/dist/bridge/service.js +225 -2196
- package/dist/bridge/service.js.map +1 -1
- package/dist/config/load.js +1 -1
- package/dist/config/load.js.map +1 -1
- package/dist/config/paths.js +1 -20
- package/dist/config/paths.js.map +1 -1
- package/dist/config/schema.d.ts +3 -0
- package/dist/config/schema.js +3 -1
- package/dist/config/schema.js.map +1 -1
- package/dist/feishu/long-connection.js +1 -0
- package/dist/feishu/long-connection.js.map +1 -1
- package/dist/feishu/webhook.js +1 -0
- package/dist/feishu/webhook.js.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,228 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
import { AuditLog } from '../state/audit-log.js';
|
|
3
|
+
import { loadBridgeConfigFile } from '../config/load.js';
|
|
4
|
+
import { getProjectArchiveDir, getProjectAuditDir, getProjectAuditFile } from '../projects/paths.js';
|
|
5
|
+
import { buildTeamDigest, formatTeamDigest, createDigestPeriod } from '../collaboration/digest.js';
|
|
6
|
+
import { checkLongRunningAlerts, formatAlert } from '../collaboration/proactive-alerts.js';
|
|
7
|
+
import { diffConfigs } from './service-utils.js';
|
|
8
|
+
// ---------------------------------------------------------------------------
|
|
9
|
+
// Run state recovery (startup orphan handling)
|
|
10
|
+
// ---------------------------------------------------------------------------
|
|
11
|
+
export async function recoverRuntimeState(host) {
|
|
12
|
+
const recovered = await host.runStateStore.recoverOrphanedRuns();
|
|
13
|
+
for (const run of recovered) {
|
|
14
|
+
await host.auditLog.append({
|
|
15
|
+
type: 'codex.run.recovered',
|
|
16
|
+
run_id: run.run_id,
|
|
17
|
+
project_alias: run.project_alias,
|
|
18
|
+
conversation_key: run.conversation_key,
|
|
19
|
+
status: run.status,
|
|
20
|
+
pid: run.pid,
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
return recovered;
|
|
24
|
+
}
|
|
25
|
+
export async function reloadConfig(host, configPath) {
|
|
26
|
+
let newConfig;
|
|
27
|
+
try {
|
|
28
|
+
const { config } = await loadBridgeConfigFile(configPath);
|
|
29
|
+
newConfig = config;
|
|
30
|
+
}
|
|
31
|
+
catch (error) {
|
|
32
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
33
|
+
host.logger.error({ configPath, error: msg }, 'Config reload rejected — invalid config');
|
|
34
|
+
// Notify admin about the broken config
|
|
35
|
+
const alertText = `🔴 配置变更被拒绝\n\n文件: ${configPath}\n原因: ${msg}\n\n当前服务继续使用旧配置运行。请修正后重新保存。`;
|
|
36
|
+
for (const chatId of host.config.security.admin_chat_ids) {
|
|
37
|
+
try {
|
|
38
|
+
await host.feishuClient.sendText(chatId, alertText);
|
|
39
|
+
}
|
|
40
|
+
catch { /* best-effort */ }
|
|
41
|
+
}
|
|
42
|
+
await host.auditLog.append({
|
|
43
|
+
type: 'config.reload.rejected',
|
|
44
|
+
config_path: configPath,
|
|
45
|
+
error: msg,
|
|
46
|
+
});
|
|
47
|
+
return { ok: false, error: msg };
|
|
48
|
+
}
|
|
49
|
+
// Diff: what changed?
|
|
50
|
+
const changes = diffConfigs(host.config, newConfig);
|
|
51
|
+
if (changes.length === 0) {
|
|
52
|
+
host.logger.debug({ configPath }, 'Config file changed but no effective differences');
|
|
53
|
+
return { ok: true, changes: [] };
|
|
54
|
+
}
|
|
55
|
+
host.logger.info({ configPath, changeCount: changes.length }, 'Config reloaded');
|
|
56
|
+
// Notify admin — read admin_chat_ids from the OLD config in case the new
|
|
57
|
+
// config removed them, so we can still deliver the notification.
|
|
58
|
+
const changeList = changes.slice(0, 15).map((c) => ` • ${c}`).join('\n');
|
|
59
|
+
const truncated = changes.length > 15 ? `\n …及其他 ${changes.length - 15} 项变更` : '';
|
|
60
|
+
const notifyText = `✅ 配置已热加载\n\n${changes.length} 项变更:\n${changeList}${truncated}`;
|
|
61
|
+
const notifyChatIds = host.config.security.admin_chat_ids.length > 0
|
|
62
|
+
? host.config.security.admin_chat_ids
|
|
63
|
+
: newConfig.security.admin_chat_ids;
|
|
64
|
+
for (const chatId of notifyChatIds) {
|
|
65
|
+
try {
|
|
66
|
+
await host.feishuClient.sendText(chatId, notifyText);
|
|
67
|
+
}
|
|
68
|
+
catch { /* best-effort */ }
|
|
69
|
+
}
|
|
70
|
+
await host.auditLog.append({
|
|
71
|
+
type: 'config.reload.applied',
|
|
72
|
+
config_path: configPath,
|
|
73
|
+
change_count: changes.length,
|
|
74
|
+
changes: changes.slice(0, 20),
|
|
75
|
+
});
|
|
76
|
+
return { ok: true, changes, newConfig };
|
|
77
|
+
}
|
|
78
|
+
// ---------------------------------------------------------------------------
|
|
79
|
+
// Team digest cycle
|
|
80
|
+
// ---------------------------------------------------------------------------
|
|
81
|
+
export async function runDigestCycle(host) {
|
|
82
|
+
const chatIds = host.config.service.team_digest_chat_ids;
|
|
83
|
+
if (chatIds.length === 0)
|
|
84
|
+
return;
|
|
85
|
+
try {
|
|
86
|
+
const period = createDigestPeriod(host.config.service.team_digest_interval_hours);
|
|
87
|
+
const runs = await host.runStateStore.listRuns();
|
|
88
|
+
const memories = host.config.service.memory_enabled
|
|
89
|
+
? await host.memoryStore.listRecentMemories({ scope: 'project', project_alias: '' }, 100)
|
|
90
|
+
: [];
|
|
91
|
+
const auditEvents = await host.auditLog.tail(500);
|
|
92
|
+
const digest = buildTeamDigest(runs, memories, auditEvents, period);
|
|
93
|
+
if (digest.summary.total_runs === 0) {
|
|
94
|
+
return; // Nothing to report
|
|
95
|
+
}
|
|
96
|
+
const text = formatTeamDigest(digest);
|
|
97
|
+
for (const chatId of chatIds) {
|
|
98
|
+
try {
|
|
99
|
+
await host.feishuClient.sendText(chatId, text);
|
|
100
|
+
}
|
|
101
|
+
catch (error) {
|
|
102
|
+
host.logger.warn({ chatId, error }, 'Failed to send team digest');
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
await host.auditLog.append({
|
|
106
|
+
type: 'collaboration.digest.sent',
|
|
107
|
+
period_label: period.label,
|
|
108
|
+
total_runs: digest.summary.total_runs,
|
|
109
|
+
chat_ids: chatIds,
|
|
110
|
+
});
|
|
111
|
+
// Send per-project mini-digests to project notification chats
|
|
112
|
+
for (const projectDigest of digest.topProjects) {
|
|
113
|
+
const projectChatIds = host.config.projects[projectDigest.alias]?.notification_chat_ids ?? [];
|
|
114
|
+
if (projectChatIds.length === 0)
|
|
115
|
+
continue;
|
|
116
|
+
const successPct = Math.round(projectDigest.success_rate * 100);
|
|
117
|
+
const miniDigestText = [
|
|
118
|
+
`📊 项目摘要 [${projectDigest.alias}] — ${period.label}`,
|
|
119
|
+
`运行: ${projectDigest.runs} | 成功率: ${successPct}%`,
|
|
120
|
+
`参与者: ${projectDigest.actors.join(', ') || '无'}`,
|
|
121
|
+
].join('\n');
|
|
122
|
+
for (const chatId of projectChatIds) {
|
|
123
|
+
try {
|
|
124
|
+
await host.feishuClient.sendText(chatId, miniDigestText);
|
|
125
|
+
}
|
|
126
|
+
catch { /* best-effort */ }
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
catch (error) {
|
|
131
|
+
host.logger.error({ error }, 'Failed to generate team digest');
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
// ---------------------------------------------------------------------------
|
|
135
|
+
// Memory maintenance
|
|
136
|
+
// ---------------------------------------------------------------------------
|
|
137
|
+
export async function runMemoryMaintenance(host) {
|
|
138
|
+
if (!host.config.service.memory_enabled) {
|
|
139
|
+
return 0;
|
|
140
|
+
}
|
|
141
|
+
const cleaned = await host.memoryStore.cleanupExpiredMemories();
|
|
142
|
+
if (cleaned > 0) {
|
|
143
|
+
await host.auditLog.append({
|
|
144
|
+
type: 'memory.archive.expired.maintenance',
|
|
145
|
+
count: cleaned,
|
|
146
|
+
});
|
|
147
|
+
host.logger.info({ cleaned }, 'Expired memories cleaned by background maintenance');
|
|
148
|
+
}
|
|
149
|
+
return cleaned;
|
|
150
|
+
}
|
|
151
|
+
// ---------------------------------------------------------------------------
|
|
152
|
+
// Audit maintenance
|
|
153
|
+
// ---------------------------------------------------------------------------
|
|
154
|
+
function listManagedAuditTargets(config) {
|
|
155
|
+
const targets = [
|
|
156
|
+
{
|
|
157
|
+
stateDir: config.storage.dir,
|
|
158
|
+
fileName: 'audit.jsonl',
|
|
159
|
+
archiveDir: path.join(config.storage.dir, 'archive'),
|
|
160
|
+
},
|
|
161
|
+
{
|
|
162
|
+
stateDir: config.storage.dir,
|
|
163
|
+
fileName: 'admin-audit.jsonl',
|
|
164
|
+
archiveDir: path.join(config.storage.dir, 'archive'),
|
|
165
|
+
},
|
|
166
|
+
];
|
|
167
|
+
for (const [alias, project] of Object.entries(config.projects)) {
|
|
168
|
+
targets.push({
|
|
169
|
+
stateDir: getProjectAuditDir(config.storage.dir, alias, project),
|
|
170
|
+
fileName: path.basename(getProjectAuditFile(config.storage.dir, alias, project)),
|
|
171
|
+
archiveDir: getProjectArchiveDir(config.storage.dir, alias),
|
|
172
|
+
});
|
|
173
|
+
}
|
|
174
|
+
return targets;
|
|
175
|
+
}
|
|
176
|
+
export async function runAuditMaintenance(host) {
|
|
177
|
+
const auditTargets = listManagedAuditTargets(host.config);
|
|
178
|
+
let scanned = 0;
|
|
179
|
+
let archived = 0;
|
|
180
|
+
let removed = 0;
|
|
181
|
+
for (const target of auditTargets) {
|
|
182
|
+
const auditLog = new AuditLog(target.stateDir, target.fileName);
|
|
183
|
+
const result = await auditLog.cleanup({
|
|
184
|
+
retentionDays: host.config.service.audit_retention_days,
|
|
185
|
+
archiveAfterDays: host.config.service.audit_archive_after_days,
|
|
186
|
+
archiveDir: target.archiveDir,
|
|
187
|
+
});
|
|
188
|
+
scanned += 1;
|
|
189
|
+
archived += result.archived;
|
|
190
|
+
removed += result.removed;
|
|
191
|
+
}
|
|
192
|
+
if (archived > 0 || removed > 0) {
|
|
193
|
+
await host.auditLog.append({
|
|
194
|
+
type: 'audit.cleanup.completed',
|
|
195
|
+
scanned,
|
|
196
|
+
archived,
|
|
197
|
+
removed,
|
|
198
|
+
});
|
|
199
|
+
host.logger.info({ scanned, archived, removed }, 'Audit retention cleanup completed');
|
|
200
|
+
}
|
|
201
|
+
return { scanned, archived, removed };
|
|
202
|
+
}
|
|
203
|
+
// ---------------------------------------------------------------------------
|
|
204
|
+
// Combined maintenance cycle
|
|
205
|
+
// ---------------------------------------------------------------------------
|
|
206
|
+
export async function runMaintenanceCycle(host) {
|
|
207
|
+
if (host.config.service.memory_enabled) {
|
|
208
|
+
await runMemoryMaintenance(host);
|
|
209
|
+
}
|
|
210
|
+
await runAuditMaintenance(host);
|
|
211
|
+
// Proactive: check for long-running tasks
|
|
212
|
+
try {
|
|
213
|
+
const activeRuns = await host.runStateStore.listRuns();
|
|
214
|
+
const longAlerts = checkLongRunningAlerts(activeRuns);
|
|
215
|
+
for (const alert of longAlerts) {
|
|
216
|
+
const text = formatAlert(alert);
|
|
217
|
+
for (const chatId of host.config.security.admin_chat_ids) {
|
|
218
|
+
try {
|
|
219
|
+
await host.feishuClient.sendText(chatId, text);
|
|
220
|
+
}
|
|
221
|
+
catch { /* best-effort */ }
|
|
222
|
+
}
|
|
223
|
+
await host.notifyProjectChats(alert.project_alias, text);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
catch { /* best-effort */ }
|
|
227
|
+
}
|
|
228
|
+
//# sourceMappingURL=lifecycle.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"lifecycle.js","sourceRoot":"","sources":["../../src/bridge/lifecycle.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import type { BridgeConfig, ProjectConfig } from '../config/schema.js';
|
|
2
|
+
import type { IncomingMessageContext } from './types.js';
|
|
3
|
+
import type { FeishuClient } from '../feishu/client.js';
|
|
4
|
+
import type { AuditLog } from '../state/audit-log.js';
|
|
5
|
+
import type { SessionStore } from '../state/session-store.js';
|
|
6
|
+
import type { MemoryStore } from '../state/memory-store.js';
|
|
7
|
+
import type { MemoryCommandFilters, MemoryScopeTarget } from './commands.js';
|
|
8
|
+
/**
|
|
9
|
+
* Subset of FeiqueService that the /memory command handler needs. Same
|
|
10
|
+
* pattern as feishu-commands.ts: a structural interface keeps the new
|
|
11
|
+
* module honest about its dependencies and unit-testable in isolation.
|
|
12
|
+
*/
|
|
13
|
+
export interface MemoryCommandHost {
|
|
14
|
+
readonly config: BridgeConfig;
|
|
15
|
+
readonly feishuClient: FeishuClient;
|
|
16
|
+
readonly auditLog: AuditLog;
|
|
17
|
+
readonly sessionStore: SessionStore;
|
|
18
|
+
readonly memoryStore: MemoryStore;
|
|
19
|
+
sendTextReply(chatId: string, body: string, replyToMessageId?: string, originalText?: string): Promise<unknown>;
|
|
20
|
+
}
|
|
21
|
+
export interface MemoryProjectContext {
|
|
22
|
+
projectAlias: string;
|
|
23
|
+
project: ProjectConfig;
|
|
24
|
+
sessionKey: string;
|
|
25
|
+
}
|
|
26
|
+
export declare function handleMemoryCommand(host: MemoryCommandHost, context: IncomingMessageContext, projectContext: MemoryProjectContext, action: 'status' | 'stats' | 'search' | 'recent' | 'save' | 'pin' | 'unpin' | 'forget' | 'restore', scope: MemoryScopeTarget | undefined, value?: string, filters?: MemoryCommandFilters): Promise<void>;
|
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
import { truncateExcerpt } from './service-utils.js';
|
|
2
|
+
// ---------------------------------------------------------------------------
|
|
3
|
+
// Pure helpers (formerly private methods on FeiqueService)
|
|
4
|
+
// ---------------------------------------------------------------------------
|
|
5
|
+
function renderMemoryFilterLines(filters) {
|
|
6
|
+
return [
|
|
7
|
+
...(filters?.tag ? [`tag: ${filters.tag}`] : []),
|
|
8
|
+
...(filters?.source ? [`source: ${filters.source}`] : []),
|
|
9
|
+
...(filters?.created_by ? [`created_by: ${filters.created_by}`] : []),
|
|
10
|
+
];
|
|
11
|
+
}
|
|
12
|
+
function resolveMemoryTarget(config, context, requestedScope) {
|
|
13
|
+
if (requestedScope === 'group') {
|
|
14
|
+
if (!config.service.memory_group_enabled) {
|
|
15
|
+
throw new Error('群共享记忆未启用。请在配置中设置 `service.memory_group_enabled = true`。');
|
|
16
|
+
}
|
|
17
|
+
if (context.chat_type !== 'group') {
|
|
18
|
+
throw new Error('群共享记忆只能在群聊中使用。');
|
|
19
|
+
}
|
|
20
|
+
return {
|
|
21
|
+
scope: 'group',
|
|
22
|
+
chatId: context.chat_id,
|
|
23
|
+
label: '群共享记忆',
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
return {
|
|
27
|
+
scope: 'project',
|
|
28
|
+
label: '项目记忆',
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
function buildMemoryExpiresAt(config) {
|
|
32
|
+
const ttlDays = config.service.memory_default_ttl_days;
|
|
33
|
+
if (!ttlDays) {
|
|
34
|
+
return undefined;
|
|
35
|
+
}
|
|
36
|
+
return new Date(Date.now() + ttlDays * 24 * 60 * 60 * 1000).toISOString();
|
|
37
|
+
}
|
|
38
|
+
// ---------------------------------------------------------------------------
|
|
39
|
+
// /memory — full command body
|
|
40
|
+
// ---------------------------------------------------------------------------
|
|
41
|
+
export async function handleMemoryCommand(host, context, projectContext, action, scope, value, filters) {
|
|
42
|
+
if (!host.config.service.memory_enabled) {
|
|
43
|
+
await host.sendTextReply(context.chat_id, '当前未启用记忆功能。请在配置里设置 `service.memory_enabled = true`。', context.message_id, context.text);
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
try {
|
|
47
|
+
const explicitExpiredCleanup = action === 'forget' && value?.trim() === 'all-expired';
|
|
48
|
+
if (!explicitExpiredCleanup) {
|
|
49
|
+
await host.memoryStore.cleanupExpiredMemories();
|
|
50
|
+
}
|
|
51
|
+
const conversation = await host.sessionStore.getConversation(projectContext.sessionKey);
|
|
52
|
+
const activeThreadId = conversation?.projects[projectContext.projectAlias]?.thread_id;
|
|
53
|
+
const groupMemoryAvailable = host.config.service.memory_group_enabled && context.chat_type === 'group';
|
|
54
|
+
if (action === 'status') {
|
|
55
|
+
if (scope === 'group') {
|
|
56
|
+
const target = resolveMemoryTarget(host.config, context, 'group');
|
|
57
|
+
const [count, pinnedCount] = await Promise.all([
|
|
58
|
+
host.memoryStore.countGroupMemories(projectContext.projectAlias, target.chatId),
|
|
59
|
+
host.memoryStore.countPinnedGroupMemories(projectContext.projectAlias, target.chatId),
|
|
60
|
+
]);
|
|
61
|
+
await host.sendTextReply(context.chat_id, [
|
|
62
|
+
`项目: ${projectContext.projectAlias}`,
|
|
63
|
+
`群共享记忆数: ${count}`,
|
|
64
|
+
`Pinned 群共享记忆数: ${pinnedCount}`,
|
|
65
|
+
`群 chat_id: ${target.chatId}`,
|
|
66
|
+
].join('\n'), context.message_id, context.text);
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
const [count, pinnedCount, threadSummary, groupCount, groupPinnedCount] = await Promise.all([
|
|
70
|
+
host.memoryStore.countProjectMemories(projectContext.projectAlias),
|
|
71
|
+
host.memoryStore.countPinnedProjectMemories(projectContext.projectAlias),
|
|
72
|
+
activeThreadId ? host.memoryStore.getThreadSummary(projectContext.sessionKey, projectContext.projectAlias, activeThreadId) : Promise.resolve(null),
|
|
73
|
+
groupMemoryAvailable ? host.memoryStore.countGroupMemories(projectContext.projectAlias, context.chat_id) : Promise.resolve(0),
|
|
74
|
+
groupMemoryAvailable ? host.memoryStore.countPinnedGroupMemories(projectContext.projectAlias, context.chat_id) : Promise.resolve(0),
|
|
75
|
+
]);
|
|
76
|
+
await host.sendTextReply(context.chat_id, [
|
|
77
|
+
`项目: ${projectContext.projectAlias}`,
|
|
78
|
+
`项目记忆数: ${count}`,
|
|
79
|
+
`Pinned 项目记忆数: ${pinnedCount}`,
|
|
80
|
+
...(groupMemoryAvailable ? [`群共享记忆数: ${groupCount}`, `Pinned 群共享记忆数: ${groupPinnedCount}`] : []),
|
|
81
|
+
`当前会话: ${activeThreadId ?? '未开始'}`,
|
|
82
|
+
'',
|
|
83
|
+
threadSummary?.summary ?? '当前没有 thread summary。',
|
|
84
|
+
].join('\n'), context.message_id, context.text);
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
if (action === 'stats') {
|
|
88
|
+
const target = resolveMemoryTarget(host.config, context, scope);
|
|
89
|
+
const stats = await host.memoryStore.getMemoryStats({
|
|
90
|
+
scope: target.scope,
|
|
91
|
+
project_alias: projectContext.projectAlias,
|
|
92
|
+
chat_id: target.chatId,
|
|
93
|
+
});
|
|
94
|
+
await host.sendTextReply(context.chat_id, [
|
|
95
|
+
`项目: ${projectContext.projectAlias}`,
|
|
96
|
+
`${target.label}统计:`,
|
|
97
|
+
`active_count: ${stats.active_count}`,
|
|
98
|
+
`expired_count: ${stats.expired_count}`,
|
|
99
|
+
`pinned_count: ${stats.pinned_count}`,
|
|
100
|
+
`archived_count: ${stats.archived_count}`,
|
|
101
|
+
`latest_accessed_at: ${stats.latest_accessed_at ?? '-'}`,
|
|
102
|
+
`latest_updated_at: ${stats.latest_updated_at ?? '-'}`,
|
|
103
|
+
`latest_archived_at: ${stats.latest_archived_at ?? '-'}`,
|
|
104
|
+
].join('\n'), context.message_id, context.text);
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
if (action === 'recent') {
|
|
108
|
+
const target = resolveMemoryTarget(host.config, context, scope);
|
|
109
|
+
const recent = await host.memoryStore.listRecentMemories({ scope: target.scope, project_alias: projectContext.projectAlias, chat_id: target.chatId }, host.config.service.memory_recent_limit, filters);
|
|
110
|
+
if (recent.length === 0) {
|
|
111
|
+
await host.sendTextReply(context.chat_id, [
|
|
112
|
+
`项目: ${projectContext.projectAlias}`,
|
|
113
|
+
`当前没有可展示的${target.label}。`,
|
|
114
|
+
...renderMemoryFilterLines(filters),
|
|
115
|
+
].join('\n'), context.message_id, context.text);
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
await host.sendTextReply(context.chat_id, [
|
|
119
|
+
`项目: ${projectContext.projectAlias}`,
|
|
120
|
+
`最近${target.label}:`,
|
|
121
|
+
...renderMemoryFilterLines(filters),
|
|
122
|
+
'',
|
|
123
|
+
...recent.map((item, index) => [
|
|
124
|
+
`${index + 1}. ${item.title}${item.pinned ? ' [pinned]' : ''}`,
|
|
125
|
+
` id: ${item.id}`,
|
|
126
|
+
` source: ${item.source}`,
|
|
127
|
+
...(item.created_by ? [` created_by: ${item.created_by}`] : []),
|
|
128
|
+
...(item.tags.length > 0 ? [` tags: ${item.tags.join(', ')}`] : []),
|
|
129
|
+
` updated_at: ${item.updated_at}`,
|
|
130
|
+
...(item.last_accessed_at ? [` last_accessed_at: ${item.last_accessed_at}`] : []),
|
|
131
|
+
...(item.expires_at ? [` expires_at: ${item.expires_at}`] : []),
|
|
132
|
+
` ${truncateExcerpt(item.content, 180)}`,
|
|
133
|
+
].join('\n')),
|
|
134
|
+
].join('\n'), context.message_id, context.text);
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
if (action === 'search') {
|
|
138
|
+
if (!value?.trim()) {
|
|
139
|
+
await host.sendTextReply(context.chat_id, '用法: /memory search [--tag <tag>] [--source <source>] [--created-by <actor_id>] <query>', context.message_id, context.text);
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
const target = resolveMemoryTarget(host.config, context, scope);
|
|
143
|
+
const hits = await host.memoryStore.searchMemories({ scope: target.scope, project_alias: projectContext.projectAlias, chat_id: target.chatId }, value, host.config.service.memory_search_limit, filters);
|
|
144
|
+
await host.auditLog.append({
|
|
145
|
+
type: 'memory.search',
|
|
146
|
+
chat_id: context.chat_id,
|
|
147
|
+
actor_id: context.actor_id,
|
|
148
|
+
project_alias: projectContext.projectAlias,
|
|
149
|
+
scope: target.scope,
|
|
150
|
+
query: value,
|
|
151
|
+
result_count: hits.length,
|
|
152
|
+
});
|
|
153
|
+
if (hits.length === 0) {
|
|
154
|
+
await host.sendTextReply(context.chat_id, [`项目: ${projectContext.projectAlias}`, `${target.label}搜索: ${value}`, ...renderMemoryFilterLines(filters), '未找到匹配记忆。'].join('\n'), context.message_id, context.text);
|
|
155
|
+
return;
|
|
156
|
+
}
|
|
157
|
+
await host.sendTextReply(context.chat_id, [
|
|
158
|
+
`项目: ${projectContext.projectAlias}`,
|
|
159
|
+
`${target.label}搜索: ${value}`,
|
|
160
|
+
...renderMemoryFilterLines(filters),
|
|
161
|
+
'',
|
|
162
|
+
...hits.map((hit, index) => [
|
|
163
|
+
`${index + 1}. ${hit.title}${hit.pinned ? ' [pinned]' : ''}`,
|
|
164
|
+
` id: ${hit.id}`,
|
|
165
|
+
` source: ${hit.source}`,
|
|
166
|
+
...(hit.created_by ? [` created_by: ${hit.created_by}`] : []),
|
|
167
|
+
...(hit.tags.length > 0 ? [` tags: ${hit.tags.join(', ')}`] : []),
|
|
168
|
+
...(hit.last_accessed_at ? [` last_accessed_at: ${hit.last_accessed_at}`] : []),
|
|
169
|
+
` ${truncateExcerpt(hit.content, 180)}`,
|
|
170
|
+
].join('\n')),
|
|
171
|
+
].join('\n'), context.message_id, context.text);
|
|
172
|
+
return;
|
|
173
|
+
}
|
|
174
|
+
if (action === 'pin' || action === 'unpin' || action === 'forget' || action === 'restore') {
|
|
175
|
+
if (!value?.trim()) {
|
|
176
|
+
const usage = action === 'forget'
|
|
177
|
+
? '用法: /memory forget <id> 或 /memory forget group <id>'
|
|
178
|
+
: action === 'restore'
|
|
179
|
+
? '用法: /memory restore <id> 或 /memory restore group <id>'
|
|
180
|
+
: `用法: /memory ${action} <id> 或 /memory ${action} group <id>`;
|
|
181
|
+
await host.sendTextReply(context.chat_id, usage, context.message_id, context.text);
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
const target = resolveMemoryTarget(host.config, context, scope);
|
|
185
|
+
const selector = { scope: target.scope, project_alias: projectContext.projectAlias, chat_id: target.chatId };
|
|
186
|
+
if (action === 'forget' && value === 'all-expired') {
|
|
187
|
+
const cleaned = await host.memoryStore.cleanupExpiredMemories(selector);
|
|
188
|
+
await host.auditLog.append({
|
|
189
|
+
type: 'memory.archive.expired',
|
|
190
|
+
chat_id: context.chat_id,
|
|
191
|
+
actor_id: context.actor_id,
|
|
192
|
+
project_alias: projectContext.projectAlias,
|
|
193
|
+
scope: target.scope,
|
|
194
|
+
count: cleaned,
|
|
195
|
+
});
|
|
196
|
+
await host.sendTextReply(context.chat_id, `${target.label}已归档过期项: ${cleaned}`, context.message_id, context.text);
|
|
197
|
+
return;
|
|
198
|
+
}
|
|
199
|
+
const existing = await host.memoryStore.getMemoryById(selector, value, { includeArchived: action === 'restore', includeExpired: action === 'restore' });
|
|
200
|
+
if (!existing) {
|
|
201
|
+
await host.sendTextReply(context.chat_id, `未找到可更新的${target.label} ID: ${value}`, context.message_id, context.text);
|
|
202
|
+
return;
|
|
203
|
+
}
|
|
204
|
+
if (action === 'forget') {
|
|
205
|
+
const archived = await host.memoryStore.archiveMemory(selector, value, { archived_by: context.actor_id, reason: 'manual' });
|
|
206
|
+
if (archived) {
|
|
207
|
+
await host.auditLog.append({
|
|
208
|
+
type: 'memory.archive',
|
|
209
|
+
chat_id: context.chat_id,
|
|
210
|
+
actor_id: context.actor_id,
|
|
211
|
+
project_alias: projectContext.projectAlias,
|
|
212
|
+
scope: target.scope,
|
|
213
|
+
memory_id: value,
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
await host.sendTextReply(context.chat_id, archived
|
|
217
|
+
? [`${target.label}已归档: ${archived.title}`, `memory_id: ${archived.id}`, `可用 /memory restore${target.scope === 'group' ? ' group' : ''} ${archived.id} 恢复`].join('\n')
|
|
218
|
+
: `未找到可归档的${target.label} ID: ${value}`, context.message_id, context.text);
|
|
219
|
+
return;
|
|
220
|
+
}
|
|
221
|
+
if (action === 'restore') {
|
|
222
|
+
const restored = await host.memoryStore.restoreMemory(selector, value, context.actor_id);
|
|
223
|
+
if (restored) {
|
|
224
|
+
await host.auditLog.append({
|
|
225
|
+
type: 'memory.restore',
|
|
226
|
+
chat_id: context.chat_id,
|
|
227
|
+
actor_id: context.actor_id,
|
|
228
|
+
project_alias: projectContext.projectAlias,
|
|
229
|
+
scope: target.scope,
|
|
230
|
+
memory_id: value,
|
|
231
|
+
});
|
|
232
|
+
}
|
|
233
|
+
await host.sendTextReply(context.chat_id, restored ? `${target.label}已恢复: ${restored.title}\nmemory_id: ${restored.id}` : `未找到可恢复的${target.label} ID: ${value}`, context.message_id, context.text);
|
|
234
|
+
return;
|
|
235
|
+
}
|
|
236
|
+
const pinned = action === 'pin';
|
|
237
|
+
let agedOutMemoryTitle;
|
|
238
|
+
let agedOutMemoryId;
|
|
239
|
+
if (pinned && !existing.pinned) {
|
|
240
|
+
const pinnedCount = await host.memoryStore.countPinnedMemories(selector);
|
|
241
|
+
if (pinnedCount >= host.config.service.memory_max_pinned_per_scope) {
|
|
242
|
+
if (host.config.service.memory_pin_overflow_strategy === 'age-out') {
|
|
243
|
+
const oldest = await host.memoryStore.getOldestPinnedMemory(selector, host.config.service.memory_pin_age_basis);
|
|
244
|
+
if (oldest && oldest.id !== existing.id) {
|
|
245
|
+
await host.memoryStore.setMemoryPinned(selector, oldest.id, false);
|
|
246
|
+
agedOutMemoryTitle = oldest.title;
|
|
247
|
+
agedOutMemoryId = oldest.id;
|
|
248
|
+
await host.auditLog.append({
|
|
249
|
+
type: 'memory.pin.aged_out',
|
|
250
|
+
chat_id: context.chat_id,
|
|
251
|
+
actor_id: context.actor_id,
|
|
252
|
+
project_alias: projectContext.projectAlias,
|
|
253
|
+
scope: target.scope,
|
|
254
|
+
memory_id: oldest.id,
|
|
255
|
+
replaced_by: existing.id,
|
|
256
|
+
});
|
|
257
|
+
}
|
|
258
|
+
else {
|
|
259
|
+
await host.sendTextReply(context.chat_id, `${target.label}置顶数量已达上限 (${host.config.service.memory_max_pinned_per_scope})。请先取消置顶旧记录。`, context.message_id, context.text);
|
|
260
|
+
return;
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
else {
|
|
264
|
+
await host.sendTextReply(context.chat_id, `${target.label}置顶数量已达上限 (${host.config.service.memory_max_pinned_per_scope})。请先取消置顶旧记录。`, context.message_id, context.text);
|
|
265
|
+
return;
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
const updated = await host.memoryStore.setMemoryPinned(selector, value, pinned);
|
|
270
|
+
if (!updated) {
|
|
271
|
+
await host.sendTextReply(context.chat_id, `未找到可更新的${target.label} ID: ${value}`, context.message_id, context.text);
|
|
272
|
+
return;
|
|
273
|
+
}
|
|
274
|
+
await host.auditLog.append({
|
|
275
|
+
type: pinned ? 'memory.pin' : 'memory.unpin',
|
|
276
|
+
chat_id: context.chat_id,
|
|
277
|
+
actor_id: context.actor_id,
|
|
278
|
+
project_alias: projectContext.projectAlias,
|
|
279
|
+
scope: target.scope,
|
|
280
|
+
memory_id: value,
|
|
281
|
+
});
|
|
282
|
+
await host.sendTextReply(context.chat_id, [
|
|
283
|
+
`${target.label}${pinned ? '已置顶' : '已取消置顶'}: ${updated.title}`,
|
|
284
|
+
`memory_id: ${updated.id}`,
|
|
285
|
+
...(agedOutMemoryId ? [`已自动老化旧置顶: ${agedOutMemoryTitle} (${agedOutMemoryId})`] : []),
|
|
286
|
+
].join('\n'), context.message_id, context.text);
|
|
287
|
+
return;
|
|
288
|
+
}
|
|
289
|
+
// Default: action === 'save'
|
|
290
|
+
const content = value?.trim();
|
|
291
|
+
if (!content) {
|
|
292
|
+
await host.sendTextReply(context.chat_id, '用法: /memory save <text> 或 /memory save group <text>', context.message_id, context.text);
|
|
293
|
+
return;
|
|
294
|
+
}
|
|
295
|
+
const target = resolveMemoryTarget(host.config, context, scope);
|
|
296
|
+
const title = truncateExcerpt(content.replace(/\s+/g, ' ').trim(), 60);
|
|
297
|
+
const expiresAt = buildMemoryExpiresAt(host.config);
|
|
298
|
+
const saved = await host.memoryStore.saveMemory({
|
|
299
|
+
scope: target.scope,
|
|
300
|
+
project_alias: projectContext.projectAlias,
|
|
301
|
+
chat_id: target.chatId,
|
|
302
|
+
title,
|
|
303
|
+
content,
|
|
304
|
+
tags: filters?.tag ? [filters.tag] : undefined,
|
|
305
|
+
source: filters?.source ?? 'manual',
|
|
306
|
+
created_by: context.actor_id,
|
|
307
|
+
expires_at: expiresAt,
|
|
308
|
+
});
|
|
309
|
+
await host.auditLog.append({
|
|
310
|
+
type: 'memory.save',
|
|
311
|
+
chat_id: context.chat_id,
|
|
312
|
+
actor_id: context.actor_id,
|
|
313
|
+
project_alias: projectContext.projectAlias,
|
|
314
|
+
scope: target.scope,
|
|
315
|
+
memory_id: saved.id,
|
|
316
|
+
title: saved.title,
|
|
317
|
+
});
|
|
318
|
+
await host.sendTextReply(context.chat_id, [
|
|
319
|
+
`项目: ${projectContext.projectAlias}`,
|
|
320
|
+
`已保存${target.label}: ${saved.title}`,
|
|
321
|
+
`memory_id: ${saved.id}`,
|
|
322
|
+
...(saved.expires_at ? [`expires_at: ${saved.expires_at}`] : []),
|
|
323
|
+
].join('\n'), context.message_id, context.text);
|
|
324
|
+
}
|
|
325
|
+
catch (error) {
|
|
326
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
327
|
+
await host.sendTextReply(context.chat_id, message, context.message_id, context.text);
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
//# sourceMappingURL=memory-commands.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"memory-commands.js","sourceRoot":"","sources":["../../src/bridge/memory-commands.ts"],"names":[],"mappings":"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"}
|