@evomap/evolver 1.88.3 → 1.89.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.
Files changed (94) hide show
  1. package/.cursor/BUGBOT.md +182 -0
  2. package/.env.example +68 -0
  3. package/.git-commit-guard-token +1 -0
  4. package/.github/CODEOWNERS +63 -0
  5. package/.github/ISSUE_TEMPLATE/good_first_issue.md +23 -0
  6. package/.github/pull_request_template.md +45 -0
  7. package/.github/workflows/test.yml +75 -0
  8. package/CHANGELOG.md +1123 -0
  9. package/README.md +86 -543
  10. package/README.public.md +594 -0
  11. package/SECURITY.md +108 -0
  12. package/assets/gep/events.jsonl +3 -0
  13. package/examples/atp-consumer-quickstart.md +100 -0
  14. package/examples/hello-world.md +38 -0
  15. package/index.js +30 -1
  16. package/package.json +5 -15
  17. package/proxy-package.json +39 -0
  18. package/public.manifest.json +141 -0
  19. package/src/evolve/guards.js +721 -1
  20. package/src/evolve/pipeline/collect.js +1283 -1
  21. package/src/evolve/pipeline/dispatch.js +421 -1
  22. package/src/evolve/pipeline/enrich.js +434 -1
  23. package/src/evolve/pipeline/hub.js +319 -1
  24. package/src/evolve/pipeline/select.js +274 -1
  25. package/src/evolve/pipeline/signals.js +206 -1
  26. package/src/evolve/utils.js +264 -1
  27. package/src/evolve.js +350 -1
  28. package/src/gep/a2aProtocol.js +4395 -1
  29. package/src/gep/assetCallLog.js +30 -2
  30. package/src/gep/autoDistillConv.js +205 -1
  31. package/src/gep/autoDistillLlm.js +315 -1
  32. package/src/gep/candidateEval.js +92 -1
  33. package/src/gep/candidates.js +198 -1
  34. package/src/gep/contentHash.js +30 -1
  35. package/src/gep/conversationSniffer.js +266 -1
  36. package/src/gep/crypto.js +89 -1
  37. package/src/gep/curriculum.js +163 -1
  38. package/src/gep/deviceId.js +218 -1
  39. package/src/gep/envFingerprint.js +118 -1
  40. package/src/gep/epigenetics.js +31 -1
  41. package/src/gep/execBridge.js +711 -1
  42. package/src/gep/explore.js +289 -1
  43. package/src/gep/hash.js +15 -1
  44. package/src/gep/hubFetch.js +359 -1
  45. package/src/gep/hubReview.js +207 -1
  46. package/src/gep/hubSearch.js +526 -1
  47. package/src/gep/hubVerify.js +306 -1
  48. package/src/gep/learningSignals.js +89 -1
  49. package/src/gep/memoryGraph.js +1374 -1
  50. package/src/gep/memoryGraphAdapter.js +203 -1
  51. package/src/gep/mutation.js +203 -1
  52. package/src/gep/narrativeMemory.js +108 -1
  53. package/src/gep/oauthLogin.js +143 -0
  54. package/src/gep/openPRRegistry.js +205 -1
  55. package/src/gep/personality.js +423 -1
  56. package/src/gep/policyCheck.js +599 -1
  57. package/src/gep/prompt.js +836 -1
  58. package/src/gep/recallInject.js +409 -1
  59. package/src/gep/recallVerifier.js +318 -1
  60. package/src/gep/reflection.js +177 -1
  61. package/src/gep/schemas/capsule.js +26 -0
  62. package/src/gep/selector.js +602 -1
  63. package/src/gep/skillDistiller.js +1294 -1
  64. package/src/gep/solidify.js +1699 -1
  65. package/src/gep/strategy.js +136 -1
  66. package/src/gep/tokenSavings.js +88 -0
  67. package/src/gep/workspaceKeychain.js +174 -1
  68. package/src/proxy/extensions/traceControl.js +99 -1
  69. package/src/proxy/index.js +36 -6
  70. package/src/proxy/inject.js +52 -1
  71. package/src/proxy/trace/extractor.js +534 -1
  72. package/src/proxy/trace/usage.js +105 -0
  73. package/CONTRIBUTING.md +0 -11
  74. package/assets/cover.png +0 -0
  75. package/scripts/a2a_export.js +0 -63
  76. package/scripts/a2a_ingest.js +0 -79
  77. package/scripts/a2a_promote.js +0 -118
  78. package/scripts/analyze_by_skill.js +0 -121
  79. package/scripts/build_binaries.js +0 -479
  80. package/scripts/check-changelog.js +0 -166
  81. package/scripts/extract_log.js +0 -85
  82. package/scripts/generate_history.js +0 -75
  83. package/scripts/gep_append_event.js +0 -96
  84. package/scripts/gep_personality_report.js +0 -234
  85. package/scripts/human_report.js +0 -147
  86. package/scripts/recall-verify-report.js +0 -234
  87. package/scripts/recover_loop.js +0 -61
  88. package/scripts/seed-merchants.js +0 -91
  89. package/scripts/suggest_version.js +0 -89
  90. package/scripts/validate-modules.js +0 -38
  91. package/scripts/validate-suite.js +0 -78
  92. package/skills/index.json +0 -14
  93. /package/assets/gep/{genes.seed.json → genes.json} +0 -0
  94. /package/{skills → bundled-skills}/_meta/SKILL.md +0 -0
@@ -0,0 +1,105 @@
1
+ 'use strict';
2
+
3
+ // Per-run token-usage rollup over the proxy trace log.
4
+ //
5
+ // The local proxy (src/proxy) meters real Anthropic input/output tokens for
6
+ // every Hand /v1/messages call into proxy-traces.jsonl. This reads that log
7
+ // back -- decrypting encrypted rows with the local EvoMap node secret -- and
8
+ // sums the real tokens spent within a time window, giving solidify the
9
+ // MEASURED cost of a derive loop.
10
+ //
11
+ // Best-effort by design: returns measured:false (and never throws) when the
12
+ // proxy was inactive, the node secret is missing, no rows fall in the window,
13
+ // or the in-window rows carried no usage (e.g. streamed-but-unobserved calls).
14
+ // Callers fall back to a grounded estimate in that case.
15
+
16
+ const fs = require('fs');
17
+ const {
18
+ resolveTraceFile,
19
+ resolveEvomapNodeSecret,
20
+ decryptTraceEnvelope,
21
+ } = require('./extractor');
22
+
23
+ const EMPTY = Object.freeze({
24
+ input_tokens: 0,
25
+ output_tokens: 0,
26
+ total_tokens: 0,
27
+ calls: 0,
28
+ measured: false,
29
+ });
30
+
31
+ function _rowTimestampMs(row) {
32
+ const iso = row && (row.timestamp || row.createdAtIso);
33
+ if (iso) {
34
+ const ms = Date.parse(iso);
35
+ if (Number.isFinite(ms)) return ms;
36
+ }
37
+ // createdAt is unix seconds in the Prism trace shape.
38
+ if (row && Number.isFinite(Number(row.createdAt))) return Number(row.createdAt) * 1000;
39
+ return null;
40
+ }
41
+
42
+ /**
43
+ * Sum the real token usage the proxy recorded within a run's time window.
44
+ *
45
+ * @param {object} opts
46
+ * @param {string} opts.sinceIso - REQUIRED lower bound (e.g. last_run.created_at).
47
+ * Without a window we cannot attribute traces to this run, so we report
48
+ * unmeasured rather than summing unrelated calls.
49
+ * @param {string} [opts.untilIso] - upper bound; defaults to now.
50
+ * @returns {{input_tokens:number,output_tokens:number,total_tokens:number,calls:number,measured:boolean}}
51
+ */
52
+ function sumRunUsage(opts = {}) {
53
+ const sinceMs = opts && opts.sinceIso != null ? Date.parse(opts.sinceIso) : NaN;
54
+ if (!Number.isFinite(sinceMs)) return { ...EMPTY };
55
+ const untilMs = opts && opts.untilIso != null && Number.isFinite(Date.parse(opts.untilIso))
56
+ ? Date.parse(opts.untilIso)
57
+ : Date.now();
58
+
59
+ let raw;
60
+ try {
61
+ const file = resolveTraceFile();
62
+ if (!fs.existsSync(file)) return { ...EMPTY };
63
+ raw = fs.readFileSync(file, 'utf8');
64
+ } catch (_) {
65
+ return { ...EMPTY };
66
+ }
67
+
68
+ let secret = null;
69
+ try { secret = resolveEvomapNodeSecret(); } catch (_) { secret = null; }
70
+
71
+ let input = 0;
72
+ let output = 0;
73
+ let calls = 0;
74
+ for (const line of raw.split('\n')) {
75
+ const s = line.trim();
76
+ if (!s) continue;
77
+ let row;
78
+ try { row = JSON.parse(s); } catch (_) { continue; }
79
+ if (row && row.encrypted) {
80
+ if (!secret) continue; // cannot decrypt -> treat as unobserved
81
+ try { row = decryptTraceEnvelope(row, secret); } catch (_) { continue; }
82
+ }
83
+ if (!row || typeof row !== 'object') continue;
84
+ const ms = _rowTimestampMs(row);
85
+ if (ms == null || ms < sinceMs || ms > untilMs) continue;
86
+ const i = Number(row.input_tokens);
87
+ const o = Number(row.output_tokens);
88
+ const hasI = Number.isFinite(i) && i > 0;
89
+ const hasO = Number.isFinite(o) && o > 0;
90
+ if (hasI) input += i;
91
+ if (hasO) output += o;
92
+ if (hasI || hasO) calls += 1;
93
+ }
94
+
95
+ if (calls === 0) return { ...EMPTY };
96
+ return {
97
+ input_tokens: input,
98
+ output_tokens: output,
99
+ total_tokens: input + output,
100
+ calls,
101
+ measured: true,
102
+ };
103
+ }
104
+
105
+ module.exports = { sumRunUsage };
package/CONTRIBUTING.md DELETED
@@ -1,11 +0,0 @@
1
- ## Contributing
2
-
3
- Thank you for contributing. Please follow these rules:
4
-
5
- - Do not use emoji (except the DNA emoji in documentation if needed).
6
- - Keep changes small and reviewable.
7
- - Update related documentation when you change behavior.
8
- - Run `node index.js` for a quick sanity check.
9
-
10
- Submit PRs with clear intent and scope.
11
-
package/assets/cover.png DELETED
Binary file
@@ -1,63 +0,0 @@
1
- const { loadGenes, loadCapsules, readAllEvents } = require('../src/gep/assetStore');
2
- const { exportEligibleCapsules, exportEligibleGenes, isAllowedA2AAsset } = require('../src/gep/a2a');
3
- const { buildPublish, buildHello, getTransport } = require('../src/gep/a2aProtocol');
4
- const { computeAssetId, SCHEMA_VERSION } = require('../src/gep/contentHash');
5
-
6
- function main() {
7
- var args = process.argv.slice(2);
8
- var asJson = args.includes('--json');
9
- var asProtocol = args.includes('--protocol');
10
- var withHello = args.includes('--hello');
11
- var persist = args.includes('--persist');
12
- var includeEvents = args.includes('--include-events');
13
-
14
- var capsules = loadCapsules();
15
- var genes = loadGenes();
16
- var events = readAllEvents();
17
-
18
- // Build eligible list: Capsules (filtered) + Genes (filtered) + Events (opt-in)
19
- var eligibleCapsules = exportEligibleCapsules({ capsules: capsules, events: events });
20
- var eligibleGenes = exportEligibleGenes({ genes: genes });
21
- var eligible = eligibleCapsules.concat(eligibleGenes);
22
-
23
- if (includeEvents) {
24
- var eligibleEvents = (Array.isArray(events) ? events : []).filter(function (e) {
25
- return isAllowedA2AAsset(e) && e.type === 'EvolutionEvent';
26
- });
27
- for (var ei = 0; ei < eligibleEvents.length; ei++) {
28
- var ev = eligibleEvents[ei];
29
- if (!ev.schema_version) ev.schema_version = SCHEMA_VERSION;
30
- if (!ev.asset_id) { try { ev.asset_id = computeAssetId(ev); } catch (e) {} }
31
- }
32
- eligible = eligible.concat(eligibleEvents);
33
- }
34
-
35
- if (withHello || asProtocol) {
36
- var hello = buildHello({ geneCount: genes.length, capsuleCount: capsules.length });
37
- process.stdout.write(JSON.stringify(hello) + '\n');
38
- if (persist) { try { getTransport().send(hello); } catch (e) {} }
39
- }
40
-
41
- if (asProtocol) {
42
- for (var i = 0; i < eligible.length; i++) {
43
- var msg = buildPublish({ asset: eligible[i] });
44
- process.stdout.write(JSON.stringify(msg) + '\n');
45
- if (persist) { try { getTransport().send(msg); } catch (e) {} }
46
- }
47
- return;
48
- }
49
-
50
- if (asJson) {
51
- process.stdout.write(JSON.stringify(eligible, null, 2) + '\n');
52
- return;
53
- }
54
-
55
- for (var j = 0; j < eligible.length; j++) {
56
- process.stdout.write(JSON.stringify(eligible[j]) + '\n');
57
- }
58
- }
59
-
60
- try { main(); } catch (e) {
61
- process.stderr.write((e && e.message ? e.message : String(e)) + '\n');
62
- process.exit(1);
63
- }
@@ -1,79 +0,0 @@
1
- var fs = require('fs');
2
- var assetStore = require('../src/gep/assetStore');
3
- var a2a = require('../src/gep/a2a');
4
- var memGraph = require('../src/gep/memoryGraphAdapter');
5
- var contentHash = require('../src/gep/contentHash');
6
- var a2aProto = require('../src/gep/a2aProtocol');
7
-
8
- function readStdin() {
9
- try { return fs.readFileSync(0, 'utf8'); } catch (e) { return ''; }
10
- }
11
-
12
- function parseSignalsFromEnv() {
13
- var raw = process.env.A2A_SIGNALS || '';
14
- if (!raw) return [];
15
- try {
16
- var maybe = JSON.parse(raw);
17
- if (Array.isArray(maybe)) return maybe.map(String).filter(Boolean);
18
- } catch (e) {}
19
- return String(raw).split(',').map(function (s) { return s.trim(); }).filter(Boolean);
20
- }
21
-
22
- function main() {
23
- var args = process.argv.slice(2);
24
- var inputPath = '';
25
- for (var i = 0; i < args.length; i++) {
26
- if (args[i] && !args[i].startsWith('--')) { inputPath = args[i]; break; }
27
- }
28
- var source = process.env.A2A_SOURCE || 'external';
29
- var factor = Number.isFinite(Number(process.env.A2A_EXTERNAL_CONFIDENCE_FACTOR))
30
- ? Number(process.env.A2A_EXTERNAL_CONFIDENCE_FACTOR) : 0.6;
31
-
32
- var text = inputPath ? a2a.readTextIfExists(inputPath) : readStdin();
33
- var parsed = a2a.parseA2AInput(text);
34
- var signals = parseSignalsFromEnv();
35
-
36
- var accepted = 0;
37
- var rejected = 0;
38
- var emitDecisions = process.env.A2A_EMIT_DECISIONS === 'true';
39
-
40
- for (var j = 0; j < parsed.length; j++) {
41
- var obj = parsed[j];
42
- if (!a2a.isAllowedA2AAsset(obj)) continue;
43
-
44
- if (obj.asset_id && typeof obj.asset_id === 'string') {
45
- if (!contentHash.verifyAssetId(obj)) {
46
- rejected += 1;
47
- if (emitDecisions) {
48
- try {
49
- var dm = a2aProto.buildDecision({ assetId: obj.asset_id, localId: obj.id, decision: 'reject', reason: 'asset_id integrity check failed' });
50
- a2aProto.getTransport().send(dm);
51
- } catch (e) {}
52
- }
53
- continue;
54
- }
55
- }
56
-
57
- var staged = a2a.lowerConfidence(obj, { source: source, factor: factor });
58
- if (!staged) continue;
59
-
60
- assetStore.appendExternalCandidateJsonl(staged);
61
- try { memGraph.recordExternalCandidate({ asset: staged, source: source, signals: signals }); } catch (e) {}
62
-
63
- if (emitDecisions) {
64
- try {
65
- var dm2 = a2aProto.buildDecision({ assetId: staged.asset_id, localId: staged.id, decision: 'quarantine', reason: 'staged as external candidate' });
66
- a2aProto.getTransport().send(dm2);
67
- } catch (e) {}
68
- }
69
-
70
- accepted += 1;
71
- }
72
-
73
- process.stdout.write('accepted=' + accepted + ' rejected=' + rejected + '\n');
74
- }
75
-
76
- try { main(); } catch (e) {
77
- process.stderr.write((e && e.message ? e.message : String(e)) + '\n');
78
- process.exit(1);
79
- }
@@ -1,118 +0,0 @@
1
- var assetStore = require('../src/gep/assetStore');
2
- var solidifyMod = require('../src/gep/solidify');
3
- var contentHash = require('../src/gep/contentHash');
4
- var a2aProto = require('../src/gep/a2aProtocol');
5
-
6
- function parseArgs(argv) {
7
- var out = { flags: new Set(), kv: new Map(), positionals: [] };
8
- for (var i = 0; i < argv.length; i++) {
9
- var a = argv[i];
10
- if (!a) continue;
11
- if (a.startsWith('--')) {
12
- var eq = a.indexOf('=');
13
- if (eq > -1) { out.kv.set(a.slice(2, eq), a.slice(eq + 1)); }
14
- else {
15
- var key = a.slice(2);
16
- var next = argv[i + 1];
17
- if (next && !String(next).startsWith('--')) { out.kv.set(key, next); i++; }
18
- else { out.flags.add(key); }
19
- }
20
- } else { out.positionals.push(a); }
21
- }
22
- return out;
23
- }
24
-
25
- function main() {
26
- var args = parseArgs(process.argv.slice(2));
27
- var id = String(args.kv.get('id') || '').trim();
28
- var typeRaw = String(args.kv.get('type') || '').trim().toLowerCase();
29
- var validated = args.flags.has('validated') || String(args.kv.get('validated') || '') === 'true';
30
- var limit = Number.isFinite(Number(args.kv.get('limit'))) ? Number(args.kv.get('limit')) : 500;
31
-
32
- if (!id || !typeRaw) throw new Error('Usage: node scripts/a2a_promote.js --type capsule|gene|event --id <id> --validated');
33
- if (!validated) throw new Error('Refusing to promote without --validated (local verification must be done first).');
34
-
35
- var type = typeRaw === 'capsule' ? 'Capsule' : typeRaw === 'gene' ? 'Gene' : typeRaw === 'event' ? 'EvolutionEvent' : '';
36
- if (!type) throw new Error('Invalid --type. Use capsule, gene, or event.');
37
-
38
- var external = assetStore.readRecentExternalCandidates(limit);
39
- var candidate = null;
40
- for (var i = 0; i < external.length; i++) {
41
- if (external[i] && external[i].type === type && String(external[i].id) === id) { candidate = external[i]; break; }
42
- }
43
- if (!candidate) throw new Error('Candidate not found in external zone: type=' + type + ' id=' + id);
44
-
45
- if (type === 'Gene') {
46
- var validation = Array.isArray(candidate.validation) ? candidate.validation : [];
47
- for (var j = 0; j < validation.length; j++) {
48
- var c = String(validation[j] || '').trim();
49
- if (!c) continue;
50
- if (!solidifyMod.isValidationCommandAllowed(c)) {
51
- throw new Error('Refusing to promote Gene ' + id + ': validation command rejected by safety check: "' + c + '". Only node/npm/npx commands without shell operators are allowed.');
52
- }
53
- }
54
- }
55
-
56
- var promoted = JSON.parse(JSON.stringify(candidate));
57
- if (!promoted.a2a || typeof promoted.a2a !== 'object') promoted.a2a = {};
58
- promoted.a2a.status = 'promoted';
59
- promoted.a2a.promoted_at = new Date().toISOString();
60
- if (!promoted.schema_version) promoted.schema_version = contentHash.SCHEMA_VERSION;
61
- promoted.asset_id = contentHash.computeAssetId(promoted);
62
-
63
- var emitDecisions = process.env.A2A_EMIT_DECISIONS === 'true';
64
-
65
- if (type === 'EvolutionEvent') {
66
- assetStore.appendEventJsonl(promoted);
67
- if (emitDecisions) {
68
- try {
69
- var dmEv = a2aProto.buildDecision({ assetId: promoted.asset_id, localId: id, decision: 'accept', reason: 'event promoted for provenance tracking' });
70
- a2aProto.getTransport().send(dmEv);
71
- } catch (e) {}
72
- }
73
- process.stdout.write('promoted_event=' + id + '\n');
74
- return;
75
- }
76
-
77
- if (type === 'Capsule') {
78
- assetStore.appendCapsule(promoted);
79
- if (emitDecisions) {
80
- try {
81
- var dm = a2aProto.buildDecision({ assetId: promoted.asset_id, localId: id, decision: 'accept', reason: 'capsule promoted after validation' });
82
- a2aProto.getTransport().send(dm);
83
- } catch (e) {}
84
- }
85
- process.stdout.write('promoted_capsule=' + id + '\n');
86
- return;
87
- }
88
-
89
- var localGenes = assetStore.loadGenes();
90
- var exists = false;
91
- for (var k = 0; k < localGenes.length; k++) {
92
- if (localGenes[k] && localGenes[k].type === 'Gene' && String(localGenes[k].id) === id) { exists = true; break; }
93
- }
94
- if (exists) {
95
- if (emitDecisions) {
96
- try {
97
- var dm2 = a2aProto.buildDecision({ assetId: promoted.asset_id, localId: id, decision: 'reject', reason: 'local gene with same ID already exists' });
98
- a2aProto.getTransport().send(dm2);
99
- } catch (e) {}
100
- }
101
- process.stdout.write('conflict_keep_local_gene=' + id + '\n');
102
- return;
103
- }
104
-
105
- assetStore.upsertGene(promoted);
106
- if (emitDecisions) {
107
- try {
108
- var dm3 = a2aProto.buildDecision({ assetId: promoted.asset_id, localId: id, decision: 'accept', reason: 'gene promoted after safety audit' });
109
- a2aProto.getTransport().send(dm3);
110
- } catch (e) {}
111
- }
112
- process.stdout.write('promoted_gene=' + id + '\n');
113
- }
114
-
115
- try { main(); } catch (e) {
116
- process.stderr.write((e && e.message ? e.message : String(e)) + '\n');
117
- process.exit(1);
118
- }
@@ -1,121 +0,0 @@
1
- const fs = require('fs');
2
- const path = require('path');
3
-
4
- const REPO_ROOT = path.resolve(__dirname, '..');
5
- const LOG_FILE = path.join(REPO_ROOT, 'evolution_history_full.md');
6
- const OUT_FILE = path.join(REPO_ROOT, 'evolution_detailed_report.md');
7
-
8
- function analyzeEvolution() {
9
- if (!fs.existsSync(LOG_FILE)) {
10
- console.error("Source file missing.");
11
- return;
12
- }
13
-
14
- const content = fs.readFileSync(LOG_FILE, 'utf8');
15
- // Split by divider
16
- const entries = content.split('---').map(e => e.trim()).filter(e => e.length > 0);
17
-
18
- const skillUpdates = {}; // Map<SkillName, Array<Changes>>
19
- const generalUpdates = []; // Array<Changes>
20
-
21
- // Regex to detect skills/paths
22
- // e.g. `skills/feishu-card/send.js` or **Target**: `skills/git-sync`
23
- const skillRegex = /skills\/([a-zA-Z0-9\-_]+)/;
24
- const actionRegex = /Action:\s*([\s\S]*?)(?=\n\n|\n[A-Z]|$)/i; // Capture Action text
25
- const statusRegex = /Status:\s*\[?([A-Z\s_]+)\]?/i;
26
-
27
- entries.forEach(entry => {
28
- // Extract basic info
29
- const statusMatch = entry.match(statusRegex);
30
- const status = statusMatch ? statusMatch[1].trim().toUpperCase() : 'UNKNOWN';
31
-
32
- // Skip routine checks if we want a *detailed evolution* report (focus on changes)
33
- // But user asked for "what happened", so routine scans might be boring unless they found something.
34
- // Let's filter out "STABILITY" or "RUNNING" unless there is a clear "Mutated" or "Fixed" keyword.
35
- const isInteresting =
36
- entry.includes('Fixed') ||
37
- entry.includes('Hardened') ||
38
- entry.includes('Optimized') ||
39
- entry.includes('Patched') ||
40
- entry.includes('Created') ||
41
- entry.includes('Added') ||
42
- status === 'SUCCESS' ||
43
- status === 'COMPLETED';
44
-
45
- if (!isInteresting) return;
46
-
47
- // Find associated skill
48
- const skillMatch = entry.match(skillRegex);
49
- let skillName = 'General / System';
50
- if (skillMatch) {
51
- skillName = skillMatch[1];
52
- } else {
53
- // Try heuristics
54
- if (entry.toLowerCase().includes('feishu card')) skillName = 'feishu-card';
55
- else if (entry.toLowerCase().includes('git sync')) skillName = 'git-sync';
56
- else if (entry.toLowerCase().includes('logger')) skillName = 'interaction-logger';
57
- else if (entry.toLowerCase().includes('evolve')) skillName = 'capability-evolver';
58
- }
59
-
60
- // Extract description
61
- let description = "";
62
- const actionMatch = entry.match(actionRegex);
63
- if (actionMatch) {
64
- description = actionMatch[1].trim();
65
- } else {
66
- // Fallback: take lines that look like bullet points or text after header
67
- const lines = entry.split('\n');
68
- description = lines.filter(l => l.match(/^[•\-\*]|\w/)).slice(1).join('\n').trim();
69
- }
70
-
71
- // Clean up description (remove duplicate "Action:" prefix if captured)
72
- description = description.replace(/^Action:\s*/i, '');
73
-
74
- if (!skillUpdates[skillName]) skillUpdates[skillName] = [];
75
-
76
- // Dedup descriptions slightly (simple check)
77
- const isDuplicate = skillUpdates[skillName].some(u => u.desc.includes(description.substring(0, 20)));
78
- if (!isDuplicate) {
79
- // Extract Date if possible
80
- const dateMatch = entry.match(/\((\d{4}\/\d{1,2}\/\d{1,2}.*?)\)/);
81
- const date = dateMatch ? dateMatch[1] : 'Unknown';
82
-
83
- skillUpdates[skillName].push({
84
- date,
85
- status,
86
- desc: description
87
- });
88
- }
89
- });
90
-
91
- // Generate Markdown
92
- let md = "# Detailed Evolution Report (By Skill)\n\n> Comprehensive breakdown of system changes.\n\n";
93
-
94
- // Sort skills alphabetically
95
- const sortedSkills = Object.keys(skillUpdates).sort();
96
-
97
- sortedSkills.forEach(skill => {
98
- md += `## ${skill}\n`;
99
- const updates = skillUpdates[skill];
100
-
101
- updates.forEach(u => {
102
- // Icon based on content
103
- let icon = '*';
104
- const lowerDesc = u.desc.toLowerCase();
105
- if (lowerDesc.includes('optimiz')) icon = '[optimize]';
106
- if (lowerDesc.includes('secur') || lowerDesc.includes('harden') || lowerDesc.includes('permission')) icon = '[security]';
107
- if (lowerDesc.includes('fix') || lowerDesc.includes('patch')) icon = '[repair]';
108
- if (lowerDesc.includes('creat') || lowerDesc.includes('add')) icon = '[add]';
109
-
110
- md += `### ${icon} ${u.date}\n`;
111
- md += `${u.desc}\n\n`;
112
- });
113
- md += `---\n`;
114
- });
115
-
116
- fs.writeFileSync(OUT_FILE, md);
117
- console.log(`Generated report for ${sortedSkills.length} skills.`);
118
- }
119
-
120
- analyzeEvolution();
121
-