wangchuan 3.0.0 → 3.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/README.md +146 -174
- package/README.zh-CN.md +254 -0
- package/dist/bin/wangchuan.d.ts +1 -1
- package/dist/bin/wangchuan.js +2 -2
- package/dist/src/agents/codex.d.ts +9 -0
- package/dist/src/agents/codex.d.ts.map +1 -0
- package/dist/src/agents/codex.js +20 -0
- package/dist/src/agents/codex.js.map +1 -0
- package/dist/src/agents/index.d.ts.map +1 -1
- package/dist/src/agents/index.js +2 -0
- package/dist/src/agents/index.js.map +1 -1
- package/dist/src/commands/completions.js +2 -2
- package/dist/src/commands/completions.js.map +1 -1
- package/dist/src/commands/list.d.ts.map +1 -1
- package/dist/src/commands/list.js +1 -0
- package/dist/src/commands/list.js.map +1 -1
- package/dist/src/commands/template.js +1 -1
- package/dist/src/commands/template.js.map +1 -1
- package/dist/src/core/config.d.ts +9 -9
- package/dist/src/core/config.d.ts.map +1 -1
- package/dist/src/core/config.js +11 -11
- package/dist/src/core/config.js.map +1 -1
- package/dist/src/core/crypto.d.ts +10 -10
- package/dist/src/core/crypto.d.ts.map +1 -1
- package/dist/src/core/crypto.js +19 -19
- package/dist/src/core/crypto.js.map +1 -1
- package/dist/src/core/git.d.ts +2 -2
- package/dist/src/core/git.js +9 -9
- package/dist/src/core/git.js.map +1 -1
- package/dist/src/core/json-field.d.ts +5 -5
- package/dist/src/core/json-field.d.ts.map +1 -1
- package/dist/src/core/json-field.js +5 -5
- package/dist/src/core/json-field.js.map +1 -1
- package/dist/src/core/sync.d.ts +9 -9
- package/dist/src/core/sync.js +47 -47
- package/dist/src/core/sync.js.map +1 -1
- package/dist/src/i18n.js +2 -2
- package/dist/src/i18n.js.map +1 -1
- package/dist/src/types.d.ts +24 -23
- package/dist/src/types.d.ts.map +1 -1
- package/dist/src/types.js +3 -3
- package/dist/src/types.js.map +1 -1
- package/dist/src/utils/linediff.d.ts +6 -5
- package/dist/src/utils/linediff.d.ts.map +1 -1
- package/dist/src/utils/linediff.js +10 -9
- package/dist/src/utils/linediff.js.map +1 -1
- package/dist/test/crypto.test.d.ts +1 -1
- package/dist/test/crypto.test.js +16 -16
- package/dist/test/crypto.test.js.map +1 -1
- package/dist/test/json-field.test.d.ts +1 -1
- package/dist/test/json-field.test.js +13 -13
- package/dist/test/json-field.test.js.map +1 -1
- package/dist/test/sync.test.d.ts +8 -8
- package/dist/test/sync.test.js +122 -113
- package/dist/test/sync.test.js.map +1 -1
- package/package.json +1 -1
package/dist/test/crypto.test.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* crypto.test.ts — AES-256-GCM
|
|
2
|
+
* crypto.test.ts — AES-256-GCM encryption module unit tests
|
|
3
3
|
*/
|
|
4
4
|
import { describe, it, before, after } from 'node:test';
|
|
5
5
|
import assert from 'node:assert/strict';
|
|
@@ -13,13 +13,13 @@ after(() => {
|
|
|
13
13
|
fs.rmSync(TMP, { recursive: true, force: true });
|
|
14
14
|
});
|
|
15
15
|
describe('cryptoEngine.generateKey', () => {
|
|
16
|
-
it('
|
|
16
|
+
it('generates a 64-char hex key file', () => {
|
|
17
17
|
cryptoEngine.generateKey(KEY);
|
|
18
18
|
const hex = fs.readFileSync(KEY, 'utf-8').trim();
|
|
19
19
|
assert.equal(hex.length, 64);
|
|
20
20
|
assert.match(hex, /^[0-9a-f]+$/);
|
|
21
21
|
});
|
|
22
|
-
it('
|
|
22
|
+
it('file permissions are 0o600', () => {
|
|
23
23
|
if (process.platform !== 'win32') {
|
|
24
24
|
const stat = fs.statSync(KEY);
|
|
25
25
|
assert.equal((stat.mode & 0o777).toString(8), '600');
|
|
@@ -27,8 +27,8 @@ describe('cryptoEngine.generateKey', () => {
|
|
|
27
27
|
});
|
|
28
28
|
});
|
|
29
29
|
describe('cryptoEngine.hasKey', () => {
|
|
30
|
-
it('
|
|
31
|
-
it('
|
|
30
|
+
it('returns true when key exists', () => assert.equal(cryptoEngine.hasKey(KEY), true));
|
|
31
|
+
it('returns false when key missing', () => assert.equal(cryptoEngine.hasKey(`${KEY}.nope`), false));
|
|
32
32
|
});
|
|
33
33
|
describe('cryptoEngine.encryptString / decryptString', () => {
|
|
34
34
|
const cases = [
|
|
@@ -39,23 +39,23 @@ describe('cryptoEngine.encryptString / decryptString', () => {
|
|
|
39
39
|
'',
|
|
40
40
|
];
|
|
41
41
|
for (const plain of cases) {
|
|
42
|
-
it(
|
|
42
|
+
it(`round-trip: "${plain.slice(0, 40)}"`, () => {
|
|
43
43
|
const enc = cryptoEngine.encryptString(plain, KEY);
|
|
44
44
|
assert.ok(typeof enc === 'string' && enc.length > 0);
|
|
45
45
|
const dec = cryptoEngine.decryptString(enc, KEY);
|
|
46
46
|
assert.equal(dec, plain);
|
|
47
47
|
});
|
|
48
48
|
}
|
|
49
|
-
it('
|
|
49
|
+
it('same plaintext encrypted twice yields different ciphertext (random IV)', () => {
|
|
50
50
|
const a = cryptoEngine.encryptString('hello', KEY);
|
|
51
51
|
const b = cryptoEngine.encryptString('hello', KEY);
|
|
52
52
|
assert.notEqual(a, b);
|
|
53
53
|
});
|
|
54
|
-
it('
|
|
54
|
+
it('tampered ciphertext throws error (GCM AuthTag verification)', () => {
|
|
55
55
|
const enc = cryptoEngine.encryptString('sensitive', KEY);
|
|
56
56
|
const buf = Buffer.from(enc, 'base64');
|
|
57
57
|
const last = buf[buf.length - 1];
|
|
58
|
-
assert.ok(last !== undefined, '
|
|
58
|
+
assert.ok(last !== undefined, 'ciphertext should not be empty');
|
|
59
59
|
buf[buf.length - 1] = last ^ 0xff;
|
|
60
60
|
const tampered = buf.toString('base64');
|
|
61
61
|
assert.throws(() => cryptoEngine.decryptString(tampered, KEY));
|
|
@@ -69,25 +69,25 @@ describe('cryptoEngine.encryptFile / decryptFile', () => {
|
|
|
69
69
|
before(() => {
|
|
70
70
|
fs.writeFileSync(PLAIN_FILE, CONTENT, 'utf-8');
|
|
71
71
|
});
|
|
72
|
-
it('
|
|
72
|
+
it('encrypted file does not contain plaintext', () => {
|
|
73
73
|
cryptoEngine.encryptFile(PLAIN_FILE, ENC_FILE, KEY);
|
|
74
74
|
const raw = fs.readFileSync(ENC_FILE, 'utf-8');
|
|
75
75
|
assert.ok(!raw.includes('敏感配置'));
|
|
76
76
|
assert.ok(!raw.includes('sk-xxxxx'));
|
|
77
77
|
});
|
|
78
|
-
it('
|
|
78
|
+
it('decrypted content matches original', () => {
|
|
79
79
|
cryptoEngine.decryptFile(ENC_FILE, DEC_FILE, KEY);
|
|
80
80
|
assert.equal(fs.readFileSync(DEC_FILE, 'utf-8'), CONTENT);
|
|
81
81
|
});
|
|
82
82
|
});
|
|
83
|
-
describe('cryptoEngine
|
|
84
|
-
it('
|
|
85
|
-
assert.throws(() => cryptoEngine.encryptString('x', '/tmp/nonexistent.key'),
|
|
83
|
+
describe('cryptoEngine error paths', () => {
|
|
84
|
+
it('throws friendly error when key file missing', () => {
|
|
85
|
+
assert.throws(() => cryptoEngine.encryptString('x', '/tmp/nonexistent.key'), /Key file not found/);
|
|
86
86
|
});
|
|
87
|
-
it('
|
|
87
|
+
it('throws error on invalid key format', () => {
|
|
88
88
|
const badKey = path.join(TMP, 'bad.key');
|
|
89
89
|
fs.writeFileSync(badKey, 'tooshort', 'utf-8');
|
|
90
|
-
assert.throws(() => cryptoEngine.encryptString('x', badKey),
|
|
90
|
+
assert.throws(() => cryptoEngine.encryptString('x', badKey), /Invalid key file format/);
|
|
91
91
|
});
|
|
92
92
|
});
|
|
93
93
|
//# sourceMappingURL=crypto.test.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"crypto.test.js","sourceRoot":"","sources":["../../test/crypto.test.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,WAAW,CAAC;AACxD,OAAO,MAAM,MAAM,oBAAoB,CAAC;AACxC,OAAO,EAAE,MAAU,IAAI,CAAC;AACxB,OAAO,EAAE,MAAU,IAAI,CAAC;AACxB,OAAO,IAAI,MAAQ,MAAM,CAAC;AAC1B,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AAErD,MAAM,GAAG,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,UAAU,CAAC,CAAC,CAAC;AAC/D,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC;AAEzC,KAAK,CAAC,GAAG,EAAE;IACT,EAAE,CAAC,MAAM,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;AACnD,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,0BAA0B,EAAE,GAAG,EAAE;IACxC,EAAE,CAAC,
|
|
1
|
+
{"version":3,"file":"crypto.test.js","sourceRoot":"","sources":["../../test/crypto.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* json-field.test.ts — JSON
|
|
2
|
+
* json-field.test.ts — JSON field extraction/merge unit tests
|
|
3
3
|
*/
|
|
4
4
|
import { describe, it } from 'node:test';
|
|
5
5
|
import assert from 'node:assert/strict';
|
|
6
6
|
import { jsonField } from '../src/core/json-field.js';
|
|
7
7
|
describe('jsonField.extractFields', () => {
|
|
8
|
-
it('
|
|
8
|
+
it('extracts specified top-level fields', () => {
|
|
9
9
|
const obj = {
|
|
10
10
|
mcpServers: { playwright: { type: 'stdio' } },
|
|
11
11
|
tipsHistory: { 'new-user': 3 },
|
|
@@ -14,52 +14,52 @@ describe('jsonField.extractFields', () => {
|
|
|
14
14
|
const result = jsonField.extractFields(obj, ['mcpServers']);
|
|
15
15
|
assert.deepStrictEqual(result, { mcpServers: { playwright: { type: 'stdio' } } });
|
|
16
16
|
});
|
|
17
|
-
it('
|
|
17
|
+
it('extracts multiple fields', () => {
|
|
18
18
|
const obj = { security: { auth: 'gongfeng' }, model: { name: 'opus' }, ide: { seen: true } };
|
|
19
19
|
const result = jsonField.extractFields(obj, ['security', 'model']);
|
|
20
20
|
assert.deepStrictEqual(result, { security: { auth: 'gongfeng' }, model: { name: 'opus' } });
|
|
21
21
|
});
|
|
22
|
-
it('
|
|
22
|
+
it('ignores non-existent fields', () => {
|
|
23
23
|
const obj = { a: 1 };
|
|
24
24
|
const result = jsonField.extractFields(obj, ['a', 'nonExistent']);
|
|
25
25
|
assert.deepStrictEqual(result, { a: 1 });
|
|
26
26
|
});
|
|
27
|
-
it('
|
|
27
|
+
it('returns empty object for empty field list', () => {
|
|
28
28
|
const result = jsonField.extractFields({ a: 1, b: 2 }, []);
|
|
29
29
|
assert.deepStrictEqual(result, {});
|
|
30
30
|
});
|
|
31
31
|
});
|
|
32
32
|
describe('jsonField.mergeFields', () => {
|
|
33
|
-
it('
|
|
33
|
+
it('merges partial into target, preserving other fields', () => {
|
|
34
34
|
const target = { a: 1, b: 2, c: 3 };
|
|
35
35
|
const partial = { b: 99 };
|
|
36
36
|
const result = jsonField.mergeFields(target, partial);
|
|
37
37
|
assert.deepStrictEqual(result, { a: 1, b: 99, c: 3 });
|
|
38
38
|
});
|
|
39
|
-
it('partial
|
|
39
|
+
it('adds new fields from partial to target', () => {
|
|
40
40
|
const target = { a: 1 };
|
|
41
41
|
const partial = { b: 2 };
|
|
42
42
|
const result = jsonField.mergeFields(target, partial);
|
|
43
43
|
assert.deepStrictEqual(result, { a: 1, b: 2 });
|
|
44
44
|
});
|
|
45
|
-
it('target
|
|
45
|
+
it('returns partial when target is empty', () => {
|
|
46
46
|
const result = jsonField.mergeFields({}, { mcpServers: { gongfeng: {} } });
|
|
47
47
|
assert.deepStrictEqual(result, { mcpServers: { gongfeng: {} } });
|
|
48
48
|
});
|
|
49
|
-
it('extract
|
|
49
|
+
it('extract then merge-back preserves other fields', () => {
|
|
50
50
|
const original = {
|
|
51
51
|
mcpServers: { old: 'config' },
|
|
52
52
|
tipsHistory: { tip1: 5 },
|
|
53
53
|
numStartups: 10,
|
|
54
54
|
projects: { '/tmp': { cost: 100 } },
|
|
55
55
|
};
|
|
56
|
-
//
|
|
56
|
+
// Simulate push: extract mcpServers
|
|
57
57
|
const extracted = jsonField.extractFields(original, ['mcpServers']);
|
|
58
|
-
//
|
|
58
|
+
// Simulate remote modification
|
|
59
59
|
extracted.mcpServers = { new: 'config' };
|
|
60
|
-
//
|
|
60
|
+
// Simulate pull: merge back into original
|
|
61
61
|
const merged = jsonField.mergeFields(original, extracted);
|
|
62
|
-
// mcpServers
|
|
62
|
+
// mcpServers updated, other fields preserved
|
|
63
63
|
assert.deepStrictEqual(merged, {
|
|
64
64
|
mcpServers: { new: 'config' },
|
|
65
65
|
tipsHistory: { tip1: 5 },
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"json-field.test.js","sourceRoot":"","sources":["../../test/json-field.test.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,WAAW,CAAC;AACzC,OAAO,MAAM,MAAM,oBAAoB,CAAC;AACxC,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AAEtD,QAAQ,CAAC,yBAAyB,EAAE,GAAG,EAAE;IACvC,EAAE,CAAC,
|
|
1
|
+
{"version":3,"file":"json-field.test.js","sourceRoot":"","sources":["../../test/json-field.test.ts"],"names":[],"mappings":"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"}
|
package/dist/test/sync.test.d.ts
CHANGED
|
@@ -1,13 +1,13 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* sync.test.ts —
|
|
2
|
+
* sync.test.ts — Sync engine unit tests
|
|
3
3
|
*
|
|
4
|
-
*
|
|
5
|
-
* - buildFileEntries
|
|
6
|
-
* - distributeShared
|
|
7
|
-
* - stageToRepo + restoreFromRepo
|
|
8
|
-
* - jsonFields
|
|
9
|
-
* -
|
|
10
|
-
* -
|
|
4
|
+
* Coverage:
|
|
5
|
+
* - buildFileEntries entry generation (agents + shared + jsonFields + dedup)
|
|
6
|
+
* - distributeShared (skills distribution, MCP distribution)
|
|
7
|
+
* - stageToRepo + restoreFromRepo round-trip consistency
|
|
8
|
+
* - jsonFields field-level extraction/merge without data loss
|
|
9
|
+
* - New skill/MCP cross-agent sharing
|
|
10
|
+
* - Behavior after skill deletion (add-only model)
|
|
11
11
|
*/
|
|
12
12
|
export {};
|
|
13
13
|
//# sourceMappingURL=sync.test.d.ts.map
|