wangchuan 1.0.2 → 2.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.
- package/.wangchuan/config.example.json +44 -19
- package/README.md +74 -31
- package/dist/bin/wangchuan.js +9 -0
- package/dist/bin/wangchuan.js.map +1 -1
- package/dist/src/commands/diff.d.ts.map +1 -1
- package/dist/src/commands/diff.js +3 -1
- package/dist/src/commands/diff.js.map +1 -1
- package/dist/src/commands/dump.d.ts +9 -0
- package/dist/src/commands/dump.d.ts.map +1 -0
- package/dist/src/commands/dump.js +68 -0
- package/dist/src/commands/dump.js.map +1 -0
- package/dist/src/commands/list.d.ts +1 -1
- package/dist/src/commands/list.d.ts.map +1 -1
- package/dist/src/commands/list.js +26 -12
- package/dist/src/commands/list.js.map +1 -1
- package/dist/src/commands/pull.d.ts.map +1 -1
- package/dist/src/commands/pull.js +11 -1
- package/dist/src/commands/pull.js.map +1 -1
- package/dist/src/commands/push.d.ts.map +1 -1
- package/dist/src/commands/push.js +11 -3
- package/dist/src/commands/push.js.map +1 -1
- package/dist/src/commands/status.d.ts.map +1 -1
- package/dist/src/commands/status.js +5 -2
- package/dist/src/commands/status.js.map +1 -1
- package/dist/src/core/config.d.ts +8 -1
- package/dist/src/core/config.d.ts.map +1 -1
- package/dist/src/core/config.js +54 -15
- package/dist/src/core/config.js.map +1 -1
- package/dist/src/core/json-field.d.ts +17 -0
- package/dist/src/core/json-field.d.ts.map +1 -0
- package/dist/src/core/json-field.js +21 -0
- package/dist/src/core/json-field.js.map +1 -0
- package/dist/src/core/migrate.d.ts +22 -0
- package/dist/src/core/migrate.d.ts.map +1 -0
- package/dist/src/core/migrate.js +183 -0
- package/dist/src/core/migrate.js.map +1 -0
- package/dist/src/core/sync.d.ts +7 -0
- package/dist/src/core/sync.d.ts.map +1 -1
- package/dist/src/core/sync.js +423 -68
- package/dist/src/core/sync.js.map +1 -1
- package/dist/src/types.d.ts +62 -17
- package/dist/src/types.d.ts.map +1 -1
- package/dist/test/json-field.test.d.ts +5 -0
- package/dist/test/json-field.test.d.ts.map +1 -0
- package/dist/test/json-field.test.js +71 -0
- package/dist/test/json-field.test.js.map +1 -0
- package/dist/test/sync.test.d.ts +13 -0
- package/dist/test/sync.test.d.ts.map +1 -0
- package/dist/test/sync.test.js +477 -0
- package/dist/test/sync.test.js.map +1 -0
- package/package.json +2 -2
- package/skill/SKILL.md +9 -3
|
@@ -0,0 +1,477 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* sync.test.ts — 同步引擎单元测试
|
|
3
|
+
*
|
|
4
|
+
* 覆盖:
|
|
5
|
+
* - buildFileEntries 条目生成(agents + shared + jsonFields + 去重)
|
|
6
|
+
* - distributeShared(skills 分发、MCP 分发)
|
|
7
|
+
* - stageToRepo + restoreFromRepo 往返一致性
|
|
8
|
+
* - jsonFields 字段级提取/合并不丢数据
|
|
9
|
+
* - 新增 skill/MCP 的跨 agent 共享
|
|
10
|
+
* - 删除 skill 后的行为(当前:只增不删)
|
|
11
|
+
*/
|
|
12
|
+
import { describe, it, before, after } from 'node:test';
|
|
13
|
+
import assert from 'node:assert/strict';
|
|
14
|
+
import fs from 'fs';
|
|
15
|
+
import os from 'os';
|
|
16
|
+
import path from 'path';
|
|
17
|
+
import { syncEngine, buildFileEntries } from '../src/core/sync.js';
|
|
18
|
+
import { cryptoEngine } from '../src/core/crypto.js';
|
|
19
|
+
// ── 测试工具 ────────────────────────────────────────────────────────
|
|
20
|
+
const TMP = fs.mkdtempSync(path.join(os.tmpdir(), 'wc-sync-'));
|
|
21
|
+
const KEY = path.join(TMP, 'master.key');
|
|
22
|
+
const REPO = path.join(TMP, 'repo');
|
|
23
|
+
const WS_OC = path.join(TMP, 'openclaw');
|
|
24
|
+
const WS_CL = path.join(TMP, 'claude');
|
|
25
|
+
const WS_GE = path.join(TMP, 'gemini');
|
|
26
|
+
function mkCfg(overrides) {
|
|
27
|
+
return {
|
|
28
|
+
repo: 'git@example.com:test.git',
|
|
29
|
+
branch: 'main',
|
|
30
|
+
localRepoPath: REPO,
|
|
31
|
+
keyPath: KEY,
|
|
32
|
+
hostname: 'test-host',
|
|
33
|
+
version: 2,
|
|
34
|
+
profiles: {
|
|
35
|
+
default: {
|
|
36
|
+
openclaw: {
|
|
37
|
+
enabled: true,
|
|
38
|
+
workspacePath: WS_OC,
|
|
39
|
+
syncFiles: [
|
|
40
|
+
{ src: 'MEMORY.md', encrypt: true },
|
|
41
|
+
{ src: 'SOUL.md', encrypt: false },
|
|
42
|
+
],
|
|
43
|
+
},
|
|
44
|
+
claude: {
|
|
45
|
+
enabled: true,
|
|
46
|
+
workspacePath: WS_CL,
|
|
47
|
+
syncFiles: [
|
|
48
|
+
{ src: 'CLAUDE.md', encrypt: false },
|
|
49
|
+
],
|
|
50
|
+
jsonFields: [
|
|
51
|
+
{ src: '.claude.json', fields: ['mcpServers'], repoName: 'mcpServers.json', encrypt: true },
|
|
52
|
+
],
|
|
53
|
+
},
|
|
54
|
+
gemini: {
|
|
55
|
+
enabled: true,
|
|
56
|
+
workspacePath: WS_GE,
|
|
57
|
+
syncFiles: [],
|
|
58
|
+
jsonFields: [
|
|
59
|
+
{ src: 'settings.json', fields: ['security', 'model'], repoName: 'settings-sync.json', encrypt: false },
|
|
60
|
+
],
|
|
61
|
+
},
|
|
62
|
+
},
|
|
63
|
+
},
|
|
64
|
+
shared: {
|
|
65
|
+
skills: {
|
|
66
|
+
sources: [
|
|
67
|
+
{ agent: 'claude', dir: 'skills/' },
|
|
68
|
+
{ agent: 'openclaw', dir: 'skills/' },
|
|
69
|
+
],
|
|
70
|
+
},
|
|
71
|
+
mcp: {
|
|
72
|
+
sources: [
|
|
73
|
+
{ agent: 'claude', src: '.claude.json', field: 'mcpServers' },
|
|
74
|
+
{ agent: 'openclaw', src: 'config/mcporter.json', field: 'mcpServers' },
|
|
75
|
+
],
|
|
76
|
+
},
|
|
77
|
+
syncFiles: [],
|
|
78
|
+
},
|
|
79
|
+
...overrides,
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
/** 写文件,自动建目录 */
|
|
83
|
+
function writeFile(filePath, content) {
|
|
84
|
+
fs.mkdirSync(path.dirname(filePath), { recursive: true });
|
|
85
|
+
fs.writeFileSync(filePath, content, 'utf-8');
|
|
86
|
+
}
|
|
87
|
+
// ── Setup / Teardown ────────────────────────────────────────────────
|
|
88
|
+
before(() => {
|
|
89
|
+
cryptoEngine.generateKey(KEY);
|
|
90
|
+
for (const d of [REPO, WS_OC, WS_CL, WS_GE]) {
|
|
91
|
+
fs.mkdirSync(d, { recursive: true });
|
|
92
|
+
}
|
|
93
|
+
});
|
|
94
|
+
after(() => {
|
|
95
|
+
fs.rmSync(TMP, { recursive: true, force: true });
|
|
96
|
+
});
|
|
97
|
+
// ── buildFileEntries 测试 ───────────────────────────────────────────
|
|
98
|
+
describe('buildFileEntries', () => {
|
|
99
|
+
it('为每个 agent 生成 agents/<name>/ 前缀的条目', () => {
|
|
100
|
+
const cfg = mkCfg();
|
|
101
|
+
const entries = buildFileEntries(cfg);
|
|
102
|
+
const ocEntries = entries.filter(e => e.agentName === 'openclaw');
|
|
103
|
+
assert.ok(ocEntries.length >= 2);
|
|
104
|
+
assert.ok(ocEntries.every(e => e.repoRel.startsWith('agents/openclaw/')));
|
|
105
|
+
const clEntries = entries.filter(e => e.agentName === 'claude');
|
|
106
|
+
assert.ok(clEntries.length >= 1);
|
|
107
|
+
assert.ok(clEntries.every(e => e.repoRel.startsWith('agents/claude/') || e.repoRel.startsWith('shared/')));
|
|
108
|
+
});
|
|
109
|
+
it('jsonFields 条目带 jsonExtract 元数据', () => {
|
|
110
|
+
const cfg = mkCfg();
|
|
111
|
+
const entries = buildFileEntries(cfg);
|
|
112
|
+
const jfEntries = entries.filter(e => e.jsonExtract);
|
|
113
|
+
assert.ok(jfEntries.length >= 2); // claude mcpServers + gemini settings
|
|
114
|
+
const claudeJf = jfEntries.find(e => e.repoRel.includes('mcpServers'));
|
|
115
|
+
assert.ok(claudeJf);
|
|
116
|
+
assert.deepStrictEqual(claudeJf.jsonExtract.fields, ['mcpServers']);
|
|
117
|
+
assert.ok(claudeJf.encrypt);
|
|
118
|
+
});
|
|
119
|
+
it('--agent 过滤只返回指定 agent 条目', () => {
|
|
120
|
+
const cfg = mkCfg();
|
|
121
|
+
const entries = buildFileEntries(cfg, undefined, 'gemini');
|
|
122
|
+
assert.ok(entries.length > 0);
|
|
123
|
+
assert.ok(entries.every(e => e.agentName === 'gemini'));
|
|
124
|
+
});
|
|
125
|
+
it('shared skills 条目带 shared 标识', () => {
|
|
126
|
+
// 先创建 skill 文件让 walkDir 能扫到
|
|
127
|
+
writeFile(path.join(WS_CL, 'skills', 'test.md'), '# test skill');
|
|
128
|
+
const cfg = mkCfg();
|
|
129
|
+
const entries = buildFileEntries(cfg);
|
|
130
|
+
const sharedSkills = entries.filter(e => e.agentName === 'shared' && e.repoRel.startsWith('shared/skills/'));
|
|
131
|
+
assert.ok(sharedSkills.length >= 1);
|
|
132
|
+
fs.rmSync(path.join(WS_CL, 'skills'), { recursive: true, force: true });
|
|
133
|
+
});
|
|
134
|
+
it('repoRel 去重:多 agent 同名 skill 只保留先出现者', () => {
|
|
135
|
+
writeFile(path.join(WS_CL, 'skills', 'dup.md'), 'claude version');
|
|
136
|
+
writeFile(path.join(WS_OC, 'skills', 'dup.md'), 'openclaw version');
|
|
137
|
+
const cfg = mkCfg();
|
|
138
|
+
const entries = buildFileEntries(cfg);
|
|
139
|
+
const dupEntries = entries.filter(e => e.repoRel === path.join('shared', 'skills', 'dup.md'));
|
|
140
|
+
assert.equal(dupEntries.length, 1, '同名 skill 应去重为 1 条');
|
|
141
|
+
// 清理
|
|
142
|
+
fs.rmSync(path.join(WS_CL, 'skills'), { recursive: true, force: true });
|
|
143
|
+
fs.rmSync(path.join(WS_OC, 'skills'), { recursive: true, force: true });
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
// ── stageToRepo + restoreFromRepo 往返测试 ──────────────────────────
|
|
147
|
+
describe('stageToRepo → restoreFromRepo 往返', () => {
|
|
148
|
+
it('整文件同步:push 后 pull 内容一致', async () => {
|
|
149
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# 记忆内容 v1');
|
|
150
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# 灵魂');
|
|
151
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# Claude 指令');
|
|
152
|
+
const cfg = mkCfg({ shared: { skills: { sources: [] }, mcp: { sources: [] }, syncFiles: [] } });
|
|
153
|
+
const pushResult = await syncEngine.stageToRepo(cfg);
|
|
154
|
+
assert.ok(pushResult.synced.length >= 3);
|
|
155
|
+
// repo 中文件应存在
|
|
156
|
+
assert.ok(fs.existsSync(path.join(REPO, 'agents/openclaw/MEMORY.md.enc')));
|
|
157
|
+
assert.ok(fs.existsSync(path.join(REPO, 'agents/openclaw/SOUL.md')));
|
|
158
|
+
assert.ok(fs.existsSync(path.join(REPO, 'agents/claude/CLAUDE.md')));
|
|
159
|
+
// 模拟新环境:删除本地文件
|
|
160
|
+
fs.unlinkSync(path.join(WS_OC, 'MEMORY.md'));
|
|
161
|
+
fs.unlinkSync(path.join(WS_CL, 'CLAUDE.md'));
|
|
162
|
+
const pullResult = await syncEngine.restoreFromRepo(cfg);
|
|
163
|
+
assert.ok(pullResult.synced.length >= 3);
|
|
164
|
+
// 还原后内容一致
|
|
165
|
+
assert.equal(fs.readFileSync(path.join(WS_OC, 'MEMORY.md'), 'utf-8'), '# 记忆内容 v1');
|
|
166
|
+
assert.equal(fs.readFileSync(path.join(WS_CL, 'CLAUDE.md'), 'utf-8'), '# Claude 指令');
|
|
167
|
+
});
|
|
168
|
+
it('jsonFields:push 只提取指定字段,pull merge 回不丢其他字段', async () => {
|
|
169
|
+
const claudeJson = {
|
|
170
|
+
mcpServers: { playwright: { type: 'stdio', cmd: 'npx' } },
|
|
171
|
+
tipsHistory: { tip1: 5 },
|
|
172
|
+
numStartups: 42,
|
|
173
|
+
projects: { '/tmp/proj': { cost: 100 } },
|
|
174
|
+
};
|
|
175
|
+
writeFile(path.join(WS_CL, '.claude.json'), JSON.stringify(claudeJson, null, 2));
|
|
176
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# Claude');
|
|
177
|
+
const geminiJson = {
|
|
178
|
+
security: { auth: 'gongfeng' },
|
|
179
|
+
model: { name: 'opus' },
|
|
180
|
+
ide: { seen: true },
|
|
181
|
+
cache: { size: 999 },
|
|
182
|
+
};
|
|
183
|
+
writeFile(path.join(WS_GE, 'settings.json'), JSON.stringify(geminiJson, null, 2));
|
|
184
|
+
const cfg = mkCfg({ shared: { skills: { sources: [] }, mcp: { sources: [] }, syncFiles: [] } });
|
|
185
|
+
await syncEngine.stageToRepo(cfg);
|
|
186
|
+
// repo 中 claude 的 mcpServers.json.enc 应只含 mcpServers
|
|
187
|
+
const claudeRepoFile = path.join(REPO, 'agents/claude/mcpServers.json.enc');
|
|
188
|
+
assert.ok(fs.existsSync(claudeRepoFile));
|
|
189
|
+
const decrypted = JSON.parse(cryptoEngine.decryptString(fs.readFileSync(claudeRepoFile, 'utf-8').trim(), KEY));
|
|
190
|
+
assert.deepStrictEqual(Object.keys(decrypted), ['mcpServers']);
|
|
191
|
+
assert.ok(!('tipsHistory' in decrypted), 'tipsHistory 不应被提取');
|
|
192
|
+
// repo 中 gemini 的 settings-sync.json 应只含 security + model
|
|
193
|
+
const geminiRepoFile = path.join(REPO, 'agents/gemini/settings-sync.json');
|
|
194
|
+
assert.ok(fs.existsSync(geminiRepoFile));
|
|
195
|
+
const geminiPartial = JSON.parse(fs.readFileSync(geminiRepoFile, 'utf-8'));
|
|
196
|
+
assert.deepStrictEqual(Object.keys(geminiPartial).sort(), ['model', 'security']);
|
|
197
|
+
// 模拟远端修改 mcpServers(添加新 server)
|
|
198
|
+
const modified = { mcpServers: { playwright: { type: 'stdio', cmd: 'npx' }, gongfeng: { type: 'sse' } } };
|
|
199
|
+
const enc = cryptoEngine.encryptString(JSON.stringify(modified, null, 2), KEY);
|
|
200
|
+
fs.writeFileSync(claudeRepoFile, enc, 'utf-8');
|
|
201
|
+
// pull 回来
|
|
202
|
+
await syncEngine.restoreFromRepo(cfg);
|
|
203
|
+
// .claude.json 应保留 tipsHistory/numStartups/projects,mcpServers 被更新
|
|
204
|
+
const restored = JSON.parse(fs.readFileSync(path.join(WS_CL, '.claude.json'), 'utf-8'));
|
|
205
|
+
assert.deepStrictEqual(restored.mcpServers, modified.mcpServers);
|
|
206
|
+
assert.equal(restored.tipsHistory.tip1, 5, 'tipsHistory 不应被破坏');
|
|
207
|
+
assert.equal(restored.numStartups, 42, 'numStartups 不应被破坏');
|
|
208
|
+
assert.deepStrictEqual(restored.projects, { '/tmp/proj': { cost: 100 } });
|
|
209
|
+
});
|
|
210
|
+
});
|
|
211
|
+
// ── distributeShared 测试(通过 stageToRepo 间接验证) ───────────────
|
|
212
|
+
describe('skill 跨 agent 共享', () => {
|
|
213
|
+
it('Claude 新增 skill → push 后 OpenClaw 也获得该 skill', async () => {
|
|
214
|
+
writeFile(path.join(WS_CL, 'skills', 'new-skill.md'), '# New Skill');
|
|
215
|
+
// OpenClaw 没有这个 skill
|
|
216
|
+
const ocSkillPath = path.join(WS_OC, 'skills', 'new-skill.md');
|
|
217
|
+
if (fs.existsSync(ocSkillPath))
|
|
218
|
+
fs.unlinkSync(ocSkillPath);
|
|
219
|
+
const cfg = mkCfg();
|
|
220
|
+
// 准备必需的工作区文件
|
|
221
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
222
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
223
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
224
|
+
writeFile(path.join(WS_CL, '.claude.json'), '{"mcpServers":{}}');
|
|
225
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
226
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), '{"mcpServers":{}}');
|
|
227
|
+
await syncEngine.stageToRepo(cfg);
|
|
228
|
+
// distributeShared 应将 skill 复制到 OpenClaw
|
|
229
|
+
assert.ok(fs.existsSync(ocSkillPath), 'OpenClaw 应获得 Claude 的 new-skill.md');
|
|
230
|
+
assert.equal(fs.readFileSync(ocSkillPath, 'utf-8'), '# New Skill');
|
|
231
|
+
// repo 中 shared/skills/ 也应有
|
|
232
|
+
assert.ok(fs.existsSync(path.join(REPO, 'shared', 'skills', 'new-skill.md')));
|
|
233
|
+
});
|
|
234
|
+
it('删除 skill 后 push — 另一个 agent 有则复制回来(只增不删)', async () => {
|
|
235
|
+
// 两个 agent 都有 old-skill
|
|
236
|
+
writeFile(path.join(WS_CL, 'skills', 'old-skill.md'), '# Old');
|
|
237
|
+
writeFile(path.join(WS_OC, 'skills', 'old-skill.md'), '# Old');
|
|
238
|
+
// 从 OpenClaw 删除
|
|
239
|
+
fs.unlinkSync(path.join(WS_OC, 'skills', 'old-skill.md'));
|
|
240
|
+
const cfg = mkCfg();
|
|
241
|
+
await syncEngine.stageToRepo(cfg);
|
|
242
|
+
// distributeShared 会从 Claude 复制回 OpenClaw
|
|
243
|
+
assert.ok(fs.existsSync(path.join(WS_OC, 'skills', 'old-skill.md')), '删除的 skill 被从另一个 agent 复制回来');
|
|
244
|
+
});
|
|
245
|
+
});
|
|
246
|
+
describe('MCP 跨 agent 共享', () => {
|
|
247
|
+
it('Claude 新增 MCP server → push 后 OpenClaw 也获得', async () => {
|
|
248
|
+
writeFile(path.join(WS_CL, '.claude.json'), JSON.stringify({
|
|
249
|
+
mcpServers: { playwright: { type: 'stdio' }, newServer: { type: 'sse' } },
|
|
250
|
+
tipsHistory: {},
|
|
251
|
+
}, null, 2));
|
|
252
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), JSON.stringify({
|
|
253
|
+
mcpServers: { playwright: { type: 'stdio' } },
|
|
254
|
+
}, null, 2));
|
|
255
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
256
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
257
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
258
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
259
|
+
const cfg = mkCfg();
|
|
260
|
+
await syncEngine.stageToRepo(cfg);
|
|
261
|
+
// OpenClaw 的 mcporter.json 应包含 newServer
|
|
262
|
+
const ocMcp = JSON.parse(fs.readFileSync(path.join(WS_OC, 'config', 'mcporter.json'), 'utf-8'));
|
|
263
|
+
assert.ok('newServer' in ocMcp.mcpServers, 'OpenClaw 应获得 Claude 的 newServer');
|
|
264
|
+
assert.deepStrictEqual(ocMcp.mcpServers.newServer, { type: 'sse' });
|
|
265
|
+
});
|
|
266
|
+
it('分发 MCP 不覆盖已有配置', async () => {
|
|
267
|
+
writeFile(path.join(WS_CL, '.claude.json'), JSON.stringify({
|
|
268
|
+
mcpServers: { playwright: { type: 'stdio', version: 'claude' } },
|
|
269
|
+
}, null, 2));
|
|
270
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), JSON.stringify({
|
|
271
|
+
mcpServers: { playwright: { type: 'stdio', version: 'openclaw' } },
|
|
272
|
+
}, null, 2));
|
|
273
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
274
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
275
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
276
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
277
|
+
const cfg = mkCfg();
|
|
278
|
+
await syncEngine.stageToRepo(cfg);
|
|
279
|
+
// 两个 agent 的 playwright 配置应保持各自的 version,不被覆盖
|
|
280
|
+
const clMcp = JSON.parse(fs.readFileSync(path.join(WS_CL, '.claude.json'), 'utf-8'));
|
|
281
|
+
const ocMcp = JSON.parse(fs.readFileSync(path.join(WS_OC, 'config', 'mcporter.json'), 'utf-8'));
|
|
282
|
+
assert.equal(clMcp.mcpServers.playwright.version, 'claude');
|
|
283
|
+
assert.equal(ocMcp.mcpServers.playwright.version, 'openclaw');
|
|
284
|
+
});
|
|
285
|
+
});
|
|
286
|
+
// ── 一键还原(新服务器场景) ────────────────────────────────────────
|
|
287
|
+
describe('新环境一键还原', () => {
|
|
288
|
+
it('工作区为空时 pull 能完整还原所有 agent 配置', async () => {
|
|
289
|
+
// 先 push 一份完整数据到 repo
|
|
290
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# 永久记忆');
|
|
291
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# 灵魂身份');
|
|
292
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# Claude 全局指令');
|
|
293
|
+
writeFile(path.join(WS_CL, '.claude.json'), JSON.stringify({
|
|
294
|
+
mcpServers: { server1: { cmd: 'test' } },
|
|
295
|
+
tipsHistory: { x: 1 },
|
|
296
|
+
}, null, 2));
|
|
297
|
+
writeFile(path.join(WS_GE, 'settings.json'), JSON.stringify({
|
|
298
|
+
security: { auth: 'test' },
|
|
299
|
+
model: { name: 'gemini-2' },
|
|
300
|
+
cache: { big: true },
|
|
301
|
+
}, null, 2));
|
|
302
|
+
writeFile(path.join(WS_CL, 'skills', 'review.md'), '# code review');
|
|
303
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), '{"mcpServers":{}}');
|
|
304
|
+
const cfg = mkCfg();
|
|
305
|
+
await syncEngine.stageToRepo(cfg);
|
|
306
|
+
// 模拟新服务器:清空所有工作区
|
|
307
|
+
for (const d of [WS_OC, WS_CL, WS_GE]) {
|
|
308
|
+
fs.rmSync(d, { recursive: true, force: true });
|
|
309
|
+
fs.mkdirSync(d, { recursive: true });
|
|
310
|
+
}
|
|
311
|
+
// pull 还原
|
|
312
|
+
const pullResult = await syncEngine.restoreFromRepo(cfg);
|
|
313
|
+
assert.ok(pullResult.synced.length >= 4);
|
|
314
|
+
// OpenClaw 还原
|
|
315
|
+
assert.equal(fs.readFileSync(path.join(WS_OC, 'MEMORY.md'), 'utf-8'), '# 永久记忆');
|
|
316
|
+
assert.equal(fs.readFileSync(path.join(WS_OC, 'SOUL.md'), 'utf-8'), '# 灵魂身份');
|
|
317
|
+
// Claude 还原
|
|
318
|
+
assert.equal(fs.readFileSync(path.join(WS_CL, 'CLAUDE.md'), 'utf-8'), '# Claude 全局指令');
|
|
319
|
+
// Claude jsonFields merge-back(新环境本地无 .claude.json → 从空 {} merge)
|
|
320
|
+
const clJson = JSON.parse(fs.readFileSync(path.join(WS_CL, '.claude.json'), 'utf-8'));
|
|
321
|
+
assert.deepStrictEqual(clJson.mcpServers, { server1: { cmd: 'test' } });
|
|
322
|
+
assert.ok(!('tipsHistory' in clJson), '新环境不应有 tipsHistory');
|
|
323
|
+
// Gemini jsonFields merge-back
|
|
324
|
+
const geJson = JSON.parse(fs.readFileSync(path.join(WS_GE, 'settings.json'), 'utf-8'));
|
|
325
|
+
assert.deepStrictEqual(geJson.security, { auth: 'test' });
|
|
326
|
+
assert.deepStrictEqual(geJson.model, { name: 'gemini-2' });
|
|
327
|
+
assert.ok(!('cache' in geJson), '新环境不应有 cache');
|
|
328
|
+
// shared skills 分发到所有 agent
|
|
329
|
+
assert.ok(fs.existsSync(path.join(WS_CL, 'skills', 'review.md')), 'Claude 应还原 skill');
|
|
330
|
+
assert.ok(fs.existsSync(path.join(WS_OC, 'skills', 'review.md')), 'OpenClaw 也应获得共享 skill');
|
|
331
|
+
});
|
|
332
|
+
});
|
|
333
|
+
// ── 删除传播测试 ────────────────────────────────────────────────────
|
|
334
|
+
describe('删除传播', () => {
|
|
335
|
+
it('所有 agent 都删除的 skill → push 后从 repo 清理', async () => {
|
|
336
|
+
// 先 push 一个 skill 到 repo
|
|
337
|
+
writeFile(path.join(WS_CL, 'skills', 'obsolete.md'), '# 过时的 skill');
|
|
338
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
339
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
340
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
341
|
+
writeFile(path.join(WS_CL, '.claude.json'), '{"mcpServers":{}}');
|
|
342
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
343
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), '{"mcpServers":{}}');
|
|
344
|
+
const cfg = mkCfg();
|
|
345
|
+
await syncEngine.stageToRepo(cfg);
|
|
346
|
+
assert.ok(fs.existsSync(path.join(REPO, 'shared', 'skills', 'obsolete.md')));
|
|
347
|
+
// 从所有 agent 删除(Claude 有,OpenClaw 也被分发了)
|
|
348
|
+
const clSkill = path.join(WS_CL, 'skills', 'obsolete.md');
|
|
349
|
+
const ocSkill = path.join(WS_OC, 'skills', 'obsolete.md');
|
|
350
|
+
if (fs.existsSync(clSkill))
|
|
351
|
+
fs.unlinkSync(clSkill);
|
|
352
|
+
if (fs.existsSync(ocSkill))
|
|
353
|
+
fs.unlinkSync(ocSkill);
|
|
354
|
+
// 再次 push
|
|
355
|
+
const result = await syncEngine.stageToRepo(cfg);
|
|
356
|
+
// repo 中应被清理
|
|
357
|
+
assert.ok(!fs.existsSync(path.join(REPO, 'shared', 'skills', 'obsolete.md')), '所有 agent 都删除后,repo 中也应被清理');
|
|
358
|
+
assert.ok(result.deleted.includes(path.join('shared', 'skills', 'obsolete.md')));
|
|
359
|
+
});
|
|
360
|
+
it('repo 有但本地所有 agent 都无的整文件 → push 后从 repo 清理', async () => {
|
|
361
|
+
// 手动在 repo 中创建一个「幽灵文件」(模拟旧版残留)
|
|
362
|
+
writeFile(path.join(REPO, 'agents', 'openclaw', 'GHOST.md'), '幽灵');
|
|
363
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
364
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
365
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
366
|
+
writeFile(path.join(WS_CL, '.claude.json'), '{"mcpServers":{}}');
|
|
367
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
368
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), '{"mcpServers":{}}');
|
|
369
|
+
const cfg = mkCfg();
|
|
370
|
+
const result = await syncEngine.stageToRepo(cfg);
|
|
371
|
+
assert.ok(!fs.existsSync(path.join(REPO, 'agents', 'openclaw', 'GHOST.md')), 'repo 中的幽灵文件应被清理');
|
|
372
|
+
assert.ok(result.deleted.includes(path.join('agents', 'openclaw', 'GHOST.md')));
|
|
373
|
+
});
|
|
374
|
+
it('pull 后 repo 中删除的 skill 不再分发到任何 agent', async () => {
|
|
375
|
+
// 先创建 skill 并 push
|
|
376
|
+
writeFile(path.join(WS_CL, 'skills', 'temp.md'), '# temp');
|
|
377
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
378
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
379
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
380
|
+
writeFile(path.join(WS_CL, '.claude.json'), '{"mcpServers":{}}');
|
|
381
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
382
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), '{"mcpServers":{}}');
|
|
383
|
+
const cfg = mkCfg();
|
|
384
|
+
await syncEngine.stageToRepo(cfg);
|
|
385
|
+
// 从所有 agent 删除并 push(触发 repo 清理)
|
|
386
|
+
for (const d of [WS_CL, WS_OC]) {
|
|
387
|
+
const f = path.join(d, 'skills', 'temp.md');
|
|
388
|
+
if (fs.existsSync(f))
|
|
389
|
+
fs.unlinkSync(f);
|
|
390
|
+
}
|
|
391
|
+
await syncEngine.stageToRepo(cfg);
|
|
392
|
+
assert.ok(!fs.existsSync(path.join(REPO, 'shared', 'skills', 'temp.md')));
|
|
393
|
+
// pull — temp.md 不应出现在任何 agent
|
|
394
|
+
await syncEngine.restoreFromRepo(cfg);
|
|
395
|
+
assert.ok(!fs.existsSync(path.join(WS_CL, 'skills', 'temp.md')), 'Claude 不应恢复已删除的 skill');
|
|
396
|
+
assert.ok(!fs.existsSync(path.join(WS_OC, 'skills', 'temp.md')), 'OpenClaw 不应恢复已删除的 skill');
|
|
397
|
+
});
|
|
398
|
+
});
|
|
399
|
+
// ── localOnly 检测测试 ──────────────────────────────────────────────
|
|
400
|
+
describe('pull 检测本地独有文件', () => {
|
|
401
|
+
it('本地有但 repo 无的文件标记为 localOnly', async () => {
|
|
402
|
+
// 清空 repo
|
|
403
|
+
fs.rmSync(REPO, { recursive: true, force: true });
|
|
404
|
+
fs.mkdirSync(REPO, { recursive: true });
|
|
405
|
+
// 本地有文件
|
|
406
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# 本地记忆');
|
|
407
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# 灵魂');
|
|
408
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# 指令');
|
|
409
|
+
writeFile(path.join(WS_CL, '.claude.json'), '{"mcpServers":{}}');
|
|
410
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
411
|
+
const cfg = mkCfg({ shared: { skills: { sources: [] }, mcp: { sources: [] }, syncFiles: [] } });
|
|
412
|
+
const result = await syncEngine.restoreFromRepo(cfg);
|
|
413
|
+
// 应检测到本地独有文件
|
|
414
|
+
assert.ok(result.localOnly.length >= 2, `应有 localOnly 文件,实际: ${result.localOnly.length}`);
|
|
415
|
+
assert.ok(result.localOnly.some(f => f.includes('MEMORY.md')), 'MEMORY.md 应标记为 localOnly');
|
|
416
|
+
});
|
|
417
|
+
it('jsonFields 的 localOnly 不误报 — 提取字段为空则不标记', async () => {
|
|
418
|
+
fs.rmSync(REPO, { recursive: true, force: true });
|
|
419
|
+
fs.mkdirSync(REPO, { recursive: true });
|
|
420
|
+
// .claude.json 存在但 mcpServers 字段为空
|
|
421
|
+
writeFile(path.join(WS_CL, '.claude.json'), JSON.stringify({
|
|
422
|
+
tipsHistory: { x: 1 },
|
|
423
|
+
numStartups: 5,
|
|
424
|
+
}, null, 2));
|
|
425
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
426
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{}');
|
|
427
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
428
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
429
|
+
const cfg = mkCfg({ shared: { skills: { sources: [] }, mcp: { sources: [] }, syncFiles: [] } });
|
|
430
|
+
const result = await syncEngine.restoreFromRepo(cfg);
|
|
431
|
+
// .claude.json 没有 mcpServers 字段,不应标记为 localOnly
|
|
432
|
+
assert.ok(!result.localOnly.some(f => f.includes('mcpServers')), 'mcpServers 字段为空时不应标记为 localOnly');
|
|
433
|
+
});
|
|
434
|
+
});
|
|
435
|
+
// ── shared MCP 跨 agent 分发(pull 时) ─────────────────────────────
|
|
436
|
+
describe('pull 时 shared MCP 跨 agent 分发', () => {
|
|
437
|
+
it('从 repo pull 的 MCP 配置应分发到所有 agent', async () => {
|
|
438
|
+
// 清理 repo
|
|
439
|
+
fs.rmSync(REPO, { recursive: true, force: true });
|
|
440
|
+
fs.mkdirSync(REPO, { recursive: true });
|
|
441
|
+
// 准备 Claude 有 MCP config,OpenClaw 没有
|
|
442
|
+
writeFile(path.join(WS_CL, '.claude.json'), JSON.stringify({
|
|
443
|
+
mcpServers: { playwright: { type: 'stdio' } },
|
|
444
|
+
}, null, 2));
|
|
445
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
446
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
447
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
448
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), '{"mcpServers":{}}');
|
|
449
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
450
|
+
const cfg = mkCfg();
|
|
451
|
+
// push 先(创建 repo 内容)
|
|
452
|
+
await syncEngine.stageToRepo(cfg);
|
|
453
|
+
// 模拟新环境:清空 OpenClaw 的 mcporter.json
|
|
454
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), '{"mcpServers":{}}');
|
|
455
|
+
// pull
|
|
456
|
+
await syncEngine.restoreFromRepo(cfg);
|
|
457
|
+
// OpenClaw 应从 shared MCP 获得 playwright
|
|
458
|
+
const ocMcp = JSON.parse(fs.readFileSync(path.join(WS_OC, 'config', 'mcporter.json'), 'utf-8'));
|
|
459
|
+
assert.ok('playwright' in (ocMcp.mcpServers ?? {}), 'OpenClaw 应通过 pull 获得 Claude 的 MCP 配置');
|
|
460
|
+
});
|
|
461
|
+
});
|
|
462
|
+
// ── JSON 解析失败容错 ───────────────────────────────────────────────
|
|
463
|
+
describe('JSON 解析失败容错', () => {
|
|
464
|
+
it('损坏的 JSON 源文件不导致 push 崩溃', async () => {
|
|
465
|
+
writeFile(path.join(WS_CL, '.claude.json'), '{ invalid json !!!');
|
|
466
|
+
writeFile(path.join(WS_CL, 'CLAUDE.md'), '# C');
|
|
467
|
+
writeFile(path.join(WS_OC, 'MEMORY.md'), '# M');
|
|
468
|
+
writeFile(path.join(WS_OC, 'SOUL.md'), '# S');
|
|
469
|
+
writeFile(path.join(WS_GE, 'settings.json'), '{"security":{},"model":{}}');
|
|
470
|
+
writeFile(path.join(WS_OC, 'config', 'mcporter.json'), '{"mcpServers":{}}');
|
|
471
|
+
const cfg = mkCfg({ shared: { skills: { sources: [] }, mcp: { sources: [] }, syncFiles: [] } });
|
|
472
|
+
// 不应抛异常
|
|
473
|
+
const result = await syncEngine.stageToRepo(cfg);
|
|
474
|
+
assert.ok(result.skipped.some(f => f.includes('mcpServers')), '损坏 JSON 对应的条目应被 skip');
|
|
475
|
+
});
|
|
476
|
+
});
|
|
477
|
+
//# sourceMappingURL=sync.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync.test.js","sourceRoot":"","sources":["../../test/sync.test.ts"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "wangchuan",
|
|
3
|
-
"version": "1.0
|
|
3
|
+
"version": "2.1.0",
|
|
4
4
|
"description": "忘川 · AI 记忆同步系统 — 智能体记忆永不遗失",
|
|
5
5
|
"bin": {
|
|
6
6
|
"wangchuan": "./dist/bin/wangchuan.js"
|
|
@@ -17,7 +17,7 @@
|
|
|
17
17
|
"postbuild": "chmod +x dist/bin/wangchuan.js",
|
|
18
18
|
"prepublishOnly": "npm run build",
|
|
19
19
|
"dev": "tsx bin/wangchuan.ts",
|
|
20
|
-
"test": "node --import tsx/esm --test test/crypto.test.ts",
|
|
20
|
+
"test": "node --import tsx/esm --test test/crypto.test.ts test/json-field.test.ts test/sync.test.ts",
|
|
21
21
|
"typecheck": "tsc --noEmit"
|
|
22
22
|
},
|
|
23
23
|
"keywords": [
|
package/skill/SKILL.md
CHANGED
|
@@ -13,6 +13,7 @@ wangchuan status [--agent openclaw|claude|gemini] 查看同步状态和差
|
|
|
13
13
|
wangchuan diff [--agent openclaw|claude|gemini] 显示行级文件差异
|
|
14
14
|
wangchuan pull [--agent openclaw|claude|gemini] 拉取远端配置,还原到本地
|
|
15
15
|
wangchuan push [--agent <name>] [-m "<描述>"] 加密推送本地配置到远端
|
|
16
|
+
wangchuan dump [--agent openclaw|claude|gemini] 生成明文快照到临时目录
|
|
16
17
|
wangchuan init --repo <git地址> 首次初始化
|
|
17
18
|
```
|
|
18
19
|
|
|
@@ -39,6 +40,7 @@ wangchuan init --repo <git地址> 首次初始化
|
|
|
39
40
|
- `✔ 本地 · 仓库` — 本地有但未推送
|
|
40
41
|
- `✖ 本地 ✔ 仓库` — 仓库有但本地缺失,执行 pull 还原
|
|
41
42
|
- `[enc]` — 该文件加密存储(AES-256-GCM)
|
|
43
|
+
- `[字段]` — JSON 字段级提取(只同步指定字段,不影响其他内容)
|
|
42
44
|
|
|
43
45
|
### diff
|
|
44
46
|
- `+` 绿色行 — 本地新增内容
|
|
@@ -48,6 +50,8 @@ wangchuan init --repo <git地址> 首次初始化
|
|
|
48
50
|
|
|
49
51
|
### push / pull
|
|
50
52
|
- `[已加密]` / `[已解密]` — 经过 AES-256-GCM 处理
|
|
53
|
+
- `[已清理]` — 所有 agent 都已删除的文件从 repo 移除(删除传播)
|
|
54
|
+
- `⚠ 本地独有` — pull 时检测到本地有但 repo 无的文件,建议 push 同步
|
|
51
55
|
|
|
52
56
|
## --agent 说明
|
|
53
57
|
|
|
@@ -55,9 +59,11 @@ wangchuan init --repo <git地址> 首次初始化
|
|
|
55
59
|
|
|
56
60
|
| 值 | 说明 |
|
|
57
61
|
|----|------|
|
|
58
|
-
| `openclaw` | ~/.openclaw/workspace/
|
|
59
|
-
| `claude` |
|
|
60
|
-
| `gemini` | ~/.gemini/
|
|
62
|
+
| `openclaw` | MEMORY.md(加密)、AGENTS.md、SOUL.md — 默认 ~/.openclaw/workspace/ |
|
|
63
|
+
| `claude` | CLAUDE.md、settings.json(加密)、.claude.json→mcpServers 字段级提取(加密) — 默认 ~/.claude-internal/ |
|
|
64
|
+
| `gemini` | settings.internal.json→security+model 字段级提取(加密) — 默认 ~/.gemini/ |
|
|
65
|
+
|
|
66
|
+
不指定 `--agent` 时,同时操作所有已启用的 agent 及 shared 共享层(skills/MCP/共享记忆)。
|
|
61
67
|
|
|
62
68
|
## 前置条件
|
|
63
69
|
|