wyrm-mcp 3.2.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/LICENSE +667 -0
- package/README.md +384 -0
- package/dist/analytics.d.ts +100 -0
- package/dist/analytics.d.ts.map +1 -0
- package/dist/analytics.js +368 -0
- package/dist/analytics.js.map +1 -0
- package/dist/auto-orchestrator.d.ts +118 -0
- package/dist/auto-orchestrator.d.ts.map +1 -0
- package/dist/auto-orchestrator.js +325 -0
- package/dist/auto-orchestrator.js.map +1 -0
- package/dist/autoconfig.d.ts +89 -0
- package/dist/autoconfig.d.ts.map +1 -0
- package/dist/autoconfig.js +576 -0
- package/dist/autoconfig.js.map +1 -0
- package/dist/cli.d.ts +148 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +281 -0
- package/dist/cli.js.map +1 -0
- package/dist/cloud-backup.d.ts +100 -0
- package/dist/cloud-backup.d.ts.map +1 -0
- package/dist/cloud-backup.js +545 -0
- package/dist/cloud-backup.js.map +1 -0
- package/dist/crypto.d.ts +72 -0
- package/dist/crypto.d.ts.map +1 -0
- package/dist/crypto.js +164 -0
- package/dist/crypto.js.map +1 -0
- package/dist/database.d.ts +218 -0
- package/dist/database.d.ts.map +1 -0
- package/dist/database.js +1058 -0
- package/dist/database.js.map +1 -0
- package/dist/http-auth.d.ts +68 -0
- package/dist/http-auth.d.ts.map +1 -0
- package/dist/http-auth.js +296 -0
- package/dist/http-auth.js.map +1 -0
- package/dist/http-fast.d.ts +13 -0
- package/dist/http-fast.d.ts.map +1 -0
- package/dist/http-fast.js +325 -0
- package/dist/http-fast.js.map +1 -0
- package/dist/http-server.d.ts +12 -0
- package/dist/http-server.d.ts.map +1 -0
- package/dist/http-server.js +383 -0
- package/dist/http-server.js.map +1 -0
- package/dist/index.d.ts +19 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +1695 -0
- package/dist/index.js.map +1 -0
- package/dist/license.d.ts +177 -0
- package/dist/license.d.ts.map +1 -0
- package/dist/license.js +405 -0
- package/dist/license.js.map +1 -0
- package/dist/logger.d.ts +76 -0
- package/dist/logger.d.ts.map +1 -0
- package/dist/logger.js +195 -0
- package/dist/logger.js.map +1 -0
- package/dist/performance.d.ts +114 -0
- package/dist/performance.d.ts.map +1 -0
- package/dist/performance.js +228 -0
- package/dist/performance.js.map +1 -0
- package/dist/resilience.d.ts +146 -0
- package/dist/resilience.d.ts.map +1 -0
- package/dist/resilience.js +563 -0
- package/dist/resilience.js.map +1 -0
- package/dist/security.d.ts +68 -0
- package/dist/security.d.ts.map +1 -0
- package/dist/security.js +215 -0
- package/dist/security.js.map +1 -0
- package/dist/setup.d.ts +21 -0
- package/dist/setup.d.ts.map +1 -0
- package/dist/setup.js +261 -0
- package/dist/setup.js.map +1 -0
- package/dist/summarizer.d.ts +30 -0
- package/dist/summarizer.d.ts.map +1 -0
- package/dist/summarizer.js +139 -0
- package/dist/summarizer.js.map +1 -0
- package/dist/sync.d.ts +39 -0
- package/dist/sync.d.ts.map +1 -0
- package/dist/sync.js +356 -0
- package/dist/sync.js.map +1 -0
- package/dist/types.d.ts +267 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +30 -0
- package/dist/types.js.map +1 -0
- package/dist/vectors.d.ts +103 -0
- package/dist/vectors.d.ts.map +1 -0
- package/dist/vectors.js +311 -0
- package/dist/vectors.js.map +1 -0
- package/package.json +73 -0
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Wyrm Summarizer - Compresses old sessions to save tokens
|
|
3
|
+
*
|
|
4
|
+
* Uses a simple extractive summarization:
|
|
5
|
+
* - Keeps key information (commits, files, major issues)
|
|
6
|
+
* - Drops verbose descriptions
|
|
7
|
+
* - Creates a condensed summary for archived sessions
|
|
8
|
+
*/
|
|
9
|
+
export function summarizeSession(session) {
|
|
10
|
+
const parts = [];
|
|
11
|
+
// Date
|
|
12
|
+
parts.push(`[${session.date}]`);
|
|
13
|
+
// Objectives - just first line
|
|
14
|
+
if (session.objectives) {
|
|
15
|
+
const firstLine = session.objectives.split('\n')[0].trim();
|
|
16
|
+
if (firstLine)
|
|
17
|
+
parts.push(`Goal: ${firstLine}`);
|
|
18
|
+
}
|
|
19
|
+
// Completed - extract bullet points
|
|
20
|
+
if (session.completed) {
|
|
21
|
+
const items = extractBulletPoints(session.completed);
|
|
22
|
+
if (items.length > 0) {
|
|
23
|
+
parts.push(`Done: ${items.slice(0, 3).join(', ')}`);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
// Issues - keep key problems
|
|
27
|
+
if (session.issues) {
|
|
28
|
+
const issues = extractKeyIssues(session.issues);
|
|
29
|
+
if (issues.length > 0) {
|
|
30
|
+
parts.push(`Fixed: ${issues.slice(0, 2).join('; ')}`);
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
// Commits - always keep
|
|
34
|
+
if (session.commits) {
|
|
35
|
+
const commits = session.commits.match(/[a-f0-9]{7,}/gi);
|
|
36
|
+
if (commits && commits.length > 0) {
|
|
37
|
+
parts.push(`Commits: ${commits.slice(0, 3).join(', ')}`);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
// Files - just count
|
|
41
|
+
if (session.files_changed) {
|
|
42
|
+
const files = session.files_changed.split('\n').filter(f => f.trim());
|
|
43
|
+
if (files.length > 0) {
|
|
44
|
+
parts.push(`Files: ${files.length} changed`);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
const summary = parts.join(' | ');
|
|
48
|
+
const tokenEstimate = Math.ceil(summary.length / 4);
|
|
49
|
+
return { summary, tokenEstimate };
|
|
50
|
+
}
|
|
51
|
+
export function summarizeMultipleSessions(sessions) {
|
|
52
|
+
const summaries = sessions.map(s => summarizeSession(s).summary);
|
|
53
|
+
const combined = summaries.join('\n');
|
|
54
|
+
return {
|
|
55
|
+
summary: combined,
|
|
56
|
+
tokenEstimate: Math.ceil(combined.length / 4)
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
function extractBulletPoints(text) {
|
|
60
|
+
const lines = text.split('\n');
|
|
61
|
+
const bullets = [];
|
|
62
|
+
for (const line of lines) {
|
|
63
|
+
const match = line.match(/^[\s]*[-*•\d.]+[\s]+(.+)/);
|
|
64
|
+
if (match) {
|
|
65
|
+
bullets.push(match[1].trim().slice(0, 50));
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
return bullets;
|
|
69
|
+
}
|
|
70
|
+
function extractKeyIssues(text) {
|
|
71
|
+
const issues = [];
|
|
72
|
+
// Look for "Problem:" or "Issue:" patterns
|
|
73
|
+
const problemMatch = text.match(/(?:problem|issue|bug|fix(?:ed)?)[:\s]+([^.\n]+)/gi);
|
|
74
|
+
if (problemMatch) {
|
|
75
|
+
for (const match of problemMatch.slice(0, 3)) {
|
|
76
|
+
const clean = match.replace(/^(?:problem|issue|bug|fix(?:ed)?)[:\s]+/i, '').trim();
|
|
77
|
+
if (clean.length > 5)
|
|
78
|
+
issues.push(clean.slice(0, 40));
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
return issues;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Estimate tokens in a string (rough approximation)
|
|
85
|
+
*/
|
|
86
|
+
export function estimateTokens(text) {
|
|
87
|
+
return Math.ceil(text.length / 4);
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Create a compressed context bundle under a token limit
|
|
91
|
+
*/
|
|
92
|
+
export function createContextBundle(project, currentContext, recentSessions, pendingQuests, maxTokens = 4000) {
|
|
93
|
+
const parts = [];
|
|
94
|
+
let currentTokens = 0;
|
|
95
|
+
// Project header (always include)
|
|
96
|
+
const header = `# ${project.name}\nStack: ${project.stack || 'Unknown'}`;
|
|
97
|
+
parts.push(header);
|
|
98
|
+
currentTokens += estimateTokens(header);
|
|
99
|
+
// Architecture/key info from context
|
|
100
|
+
const arch = currentContext['architecture'];
|
|
101
|
+
if (arch && currentTokens + estimateTokens(arch) < maxTokens * 0.4) {
|
|
102
|
+
parts.push(`\n## Architecture\n${arch}`);
|
|
103
|
+
currentTokens += estimateTokens(arch);
|
|
104
|
+
}
|
|
105
|
+
// Credentials (if any)
|
|
106
|
+
const creds = currentContext['credentials'];
|
|
107
|
+
if (creds && currentTokens + estimateTokens(creds) < maxTokens * 0.5) {
|
|
108
|
+
parts.push(`\n## Credentials\n${creds}`);
|
|
109
|
+
currentTokens += estimateTokens(creds);
|
|
110
|
+
}
|
|
111
|
+
// Recent sessions (summarized if needed)
|
|
112
|
+
if (recentSessions.length > 0) {
|
|
113
|
+
parts.push('\n## Recent Sessions');
|
|
114
|
+
for (const session of recentSessions) {
|
|
115
|
+
const sessionText = session.summary || summarizeSession(session).summary;
|
|
116
|
+
const tokens = estimateTokens(sessionText);
|
|
117
|
+
if (currentTokens + tokens < maxTokens * 0.8) {
|
|
118
|
+
parts.push(sessionText);
|
|
119
|
+
currentTokens += tokens;
|
|
120
|
+
}
|
|
121
|
+
else {
|
|
122
|
+
break;
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
// Pending quests
|
|
127
|
+
if (pendingQuests.length > 0 && currentTokens < maxTokens * 0.9) {
|
|
128
|
+
parts.push('\n## Pending Tasks');
|
|
129
|
+
for (const quest of pendingQuests.slice(0, 10)) {
|
|
130
|
+
const questText = `- [${quest.priority}] ${quest.title}`;
|
|
131
|
+
if (currentTokens + estimateTokens(questText) < maxTokens) {
|
|
132
|
+
parts.push(questText);
|
|
133
|
+
currentTokens += estimateTokens(questText);
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
return parts.join('\n');
|
|
138
|
+
}
|
|
139
|
+
//# sourceMappingURL=summarizer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"summarizer.js","sourceRoot":"","sources":["../src/summarizer.ts"],"names":[],"mappings":"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"}
|
package/dist/sync.d.ts
ADDED
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Wyrm File Sync - Watches and syncs .wyrm folder with database
|
|
3
|
+
*
|
|
4
|
+
* @copyright 2026 Ghost Protocol (Pvt) Ltd. All Rights Reserved.
|
|
5
|
+
* @license Proprietary - See LICENSE file for details.
|
|
6
|
+
*
|
|
7
|
+
* - Watches for changes to markdown files
|
|
8
|
+
* - Syncs changes to SQLite database
|
|
9
|
+
* - Exports database state back to markdown
|
|
10
|
+
*/
|
|
11
|
+
import { WyrmDB, Project } from './database.js';
|
|
12
|
+
export declare class WyrmSync {
|
|
13
|
+
private db;
|
|
14
|
+
private watchers;
|
|
15
|
+
constructor(db: WyrmDB);
|
|
16
|
+
/**
|
|
17
|
+
* Import markdown files from .wyrm folder into database
|
|
18
|
+
*/
|
|
19
|
+
importFromFolder(projectPath: string): Project;
|
|
20
|
+
/**
|
|
21
|
+
* Export database state to .wyrm folder
|
|
22
|
+
*/
|
|
23
|
+
exportToFolder(projectPath: string): void;
|
|
24
|
+
/**
|
|
25
|
+
* Watch .wyrm folder for changes
|
|
26
|
+
*/
|
|
27
|
+
watchFolder(projectPath: string): void;
|
|
28
|
+
stopWatching(projectPath: string): void;
|
|
29
|
+
private parseHoard;
|
|
30
|
+
private parseChronicles;
|
|
31
|
+
private parseSessionBody;
|
|
32
|
+
private parseQuests;
|
|
33
|
+
private generateHoard;
|
|
34
|
+
private generateChronicles;
|
|
35
|
+
private generateQuests;
|
|
36
|
+
private splitSections;
|
|
37
|
+
private extractSection;
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=sync.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync.d.ts","sourceRoot":"","sources":["../src/sync.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAIH,OAAO,EAAE,MAAM,EAAE,OAAO,EAAkB,MAAM,eAAe,CAAC;AA0ChE,qBAAa,QAAQ;IACnB,OAAO,CAAC,EAAE,CAAS;IACnB,OAAO,CAAC,QAAQ,CAAoD;gBAExD,EAAE,EAAE,MAAM;IAItB;;OAEG;IACH,gBAAgB,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO;IAyC9C;;OAEG;IACH,cAAc,CAAC,WAAW,EAAE,MAAM,GAAG,IAAI;IA2BzC;;OAEG;IACH,WAAW,CAAC,WAAW,EAAE,MAAM,GAAG,IAAI;IAmBtC,YAAY,CAAC,WAAW,EAAE,MAAM,GAAG,IAAI;IASvC,OAAO,CAAC,UAAU;IAUlB,OAAO,CAAC,eAAe;IAuBvB,OAAO,CAAC,gBAAgB;IAyBxB,OAAO,CAAC,WAAW;IA8BnB,OAAO,CAAC,aAAa;IAuBrB,OAAO,CAAC,kBAAkB;IAwC1B,OAAO,CAAC,cAAc;IAgEtB,OAAO,CAAC,aAAa;IAsBrB,OAAO,CAAC,cAAc;CAUvB"}
|
package/dist/sync.js
ADDED
|
@@ -0,0 +1,356 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Wyrm File Sync - Watches and syncs .wyrm folder with database
|
|
3
|
+
*
|
|
4
|
+
* @copyright 2026 Ghost Protocol (Pvt) Ltd. All Rights Reserved.
|
|
5
|
+
* @license Proprietary - See LICENSE file for details.
|
|
6
|
+
*
|
|
7
|
+
* - Watches for changes to markdown files
|
|
8
|
+
* - Syncs changes to SQLite database
|
|
9
|
+
* - Exports database state back to markdown
|
|
10
|
+
*/
|
|
11
|
+
import { watch, existsSync, readFileSync, writeFileSync, mkdirSync, statSync } from 'fs';
|
|
12
|
+
import { join, basename, resolve, normalize, relative, sep } from 'path';
|
|
13
|
+
// ==================== PATH SECURITY ====================
|
|
14
|
+
/**
|
|
15
|
+
* SECURITY: Validate path is within the base directory
|
|
16
|
+
*/
|
|
17
|
+
function validatePath(basePath, targetPath) {
|
|
18
|
+
const normalizedBase = normalize(resolve(basePath));
|
|
19
|
+
const normalizedTarget = normalize(resolve(basePath, targetPath));
|
|
20
|
+
// Check if target is within base
|
|
21
|
+
const rel = relative(normalizedBase, normalizedTarget);
|
|
22
|
+
if (rel.startsWith('..') || rel.startsWith(sep)) {
|
|
23
|
+
throw new Error('SECURITY: Path traversal detected');
|
|
24
|
+
}
|
|
25
|
+
if (!normalizedTarget.startsWith(normalizedBase)) {
|
|
26
|
+
throw new Error('SECURITY: Path traversal detected');
|
|
27
|
+
}
|
|
28
|
+
return normalizedTarget;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* SECURITY: Validate project path is a real directory
|
|
32
|
+
*/
|
|
33
|
+
function validateProjectPath(projectPath) {
|
|
34
|
+
const normalizedPath = normalize(resolve(projectPath));
|
|
35
|
+
if (!existsSync(normalizedPath)) {
|
|
36
|
+
throw new Error(`Project path does not exist: ${projectPath}`);
|
|
37
|
+
}
|
|
38
|
+
if (!statSync(normalizedPath).isDirectory()) {
|
|
39
|
+
throw new Error(`Project path is not a directory: ${projectPath}`);
|
|
40
|
+
}
|
|
41
|
+
return normalizedPath;
|
|
42
|
+
}
|
|
43
|
+
export class WyrmSync {
|
|
44
|
+
db;
|
|
45
|
+
watchers = new Map();
|
|
46
|
+
constructor(db) {
|
|
47
|
+
this.db = db;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Import markdown files from .wyrm folder into database
|
|
51
|
+
*/
|
|
52
|
+
importFromFolder(projectPath) {
|
|
53
|
+
// SECURITY: Validate project path
|
|
54
|
+
const validatedPath = validateProjectPath(projectPath);
|
|
55
|
+
const wyrmPath = validatePath(validatedPath, '.wyrm');
|
|
56
|
+
if (!existsSync(wyrmPath)) {
|
|
57
|
+
throw new Error(`No .wyrm folder found at ${validatedPath}`);
|
|
58
|
+
}
|
|
59
|
+
// Get or create project
|
|
60
|
+
const projectName = basename(validatedPath);
|
|
61
|
+
let project = this.db.getProject(validatedPath);
|
|
62
|
+
if (!project) {
|
|
63
|
+
project = this.db.registerProject(projectName, validatedPath);
|
|
64
|
+
}
|
|
65
|
+
// Import hoard.md - SECURITY: validatePath ensures no traversal
|
|
66
|
+
const hoardPath = validatePath(wyrmPath, 'hoard.md');
|
|
67
|
+
if (existsSync(hoardPath)) {
|
|
68
|
+
const content = readFileSync(hoardPath, 'utf-8');
|
|
69
|
+
this.parseHoard(project.id, content);
|
|
70
|
+
}
|
|
71
|
+
// Import chronicles.md
|
|
72
|
+
const chroniclesPath = validatePath(wyrmPath, 'chronicles.md');
|
|
73
|
+
if (existsSync(chroniclesPath)) {
|
|
74
|
+
const content = readFileSync(chroniclesPath, 'utf-8');
|
|
75
|
+
this.parseChronicles(project.id, content);
|
|
76
|
+
}
|
|
77
|
+
// Import quests.md
|
|
78
|
+
const questsPath = validatePath(wyrmPath, 'quests.md');
|
|
79
|
+
if (existsSync(questsPath)) {
|
|
80
|
+
const content = readFileSync(questsPath, 'utf-8');
|
|
81
|
+
this.parseQuests(project.id, content);
|
|
82
|
+
}
|
|
83
|
+
return project;
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Export database state to .wyrm folder
|
|
87
|
+
*/
|
|
88
|
+
exportToFolder(projectPath) {
|
|
89
|
+
// SECURITY: Validate project path
|
|
90
|
+
const validatedPath = validateProjectPath(projectPath);
|
|
91
|
+
const project = this.db.getProject(validatedPath);
|
|
92
|
+
if (!project) {
|
|
93
|
+
throw new Error(`Project not found: ${validatedPath}`);
|
|
94
|
+
}
|
|
95
|
+
const wyrmPath = validatePath(validatedPath, '.wyrm');
|
|
96
|
+
if (!existsSync(wyrmPath)) {
|
|
97
|
+
mkdirSync(wyrmPath, { recursive: true });
|
|
98
|
+
}
|
|
99
|
+
// Export hoard.md - SECURITY: validatePath ensures no traversal
|
|
100
|
+
const hoardContent = this.generateHoard(project);
|
|
101
|
+
writeFileSync(validatePath(wyrmPath, 'hoard.md'), hoardContent);
|
|
102
|
+
// Export chronicles.md
|
|
103
|
+
const chroniclesContent = this.generateChronicles(project.id);
|
|
104
|
+
writeFileSync(validatePath(wyrmPath, 'chronicles.md'), chroniclesContent);
|
|
105
|
+
// Export quests.md
|
|
106
|
+
const questsContent = this.generateQuests(project.id);
|
|
107
|
+
writeFileSync(validatePath(wyrmPath, 'quests.md'), questsContent);
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Watch .wyrm folder for changes
|
|
111
|
+
*/
|
|
112
|
+
watchFolder(projectPath) {
|
|
113
|
+
const wyrmPath = join(projectPath, '.wyrm');
|
|
114
|
+
if (!existsSync(wyrmPath))
|
|
115
|
+
return;
|
|
116
|
+
if (this.watchers.has(projectPath)) {
|
|
117
|
+
this.watchers.get(projectPath).close();
|
|
118
|
+
}
|
|
119
|
+
const watcher = watch(wyrmPath, { recursive: true }, (eventType, filename) => {
|
|
120
|
+
if (filename && filename.endsWith('.md')) {
|
|
121
|
+
console.log(`[Wyrm] File changed: ${filename}`);
|
|
122
|
+
this.importFromFolder(projectPath);
|
|
123
|
+
}
|
|
124
|
+
});
|
|
125
|
+
this.watchers.set(projectPath, watcher);
|
|
126
|
+
}
|
|
127
|
+
stopWatching(projectPath) {
|
|
128
|
+
const watcher = this.watchers.get(projectPath);
|
|
129
|
+
if (watcher) {
|
|
130
|
+
watcher.close();
|
|
131
|
+
this.watchers.delete(projectPath);
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
// Parse hoard.md
|
|
135
|
+
parseHoard(projectId, content) {
|
|
136
|
+
const sections = this.splitSections(content);
|
|
137
|
+
for (const [title, body] of Object.entries(sections)) {
|
|
138
|
+
const key = title.toLowerCase().replace(/[^a-z0-9]+/g, '_');
|
|
139
|
+
this.db.setContext(projectId, key, body);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
// Parse chronicles.md
|
|
143
|
+
parseChronicles(projectId, content) {
|
|
144
|
+
const sessionRegex = /## Session:\s*(\d{4}-\d{2}-\d{2})([\s\S]*?)(?=## Session:|$)/g;
|
|
145
|
+
let match;
|
|
146
|
+
while ((match = sessionRegex.exec(content)) !== null) {
|
|
147
|
+
const date = match[1];
|
|
148
|
+
const body = match[2];
|
|
149
|
+
// Check if session exists
|
|
150
|
+
const existing = this.db.getTodaySession(projectId);
|
|
151
|
+
if (existing && existing.date === date) {
|
|
152
|
+
// Update existing
|
|
153
|
+
this.db.updateSession(existing.id, this.parseSessionBody(body));
|
|
154
|
+
}
|
|
155
|
+
else {
|
|
156
|
+
// Create new
|
|
157
|
+
this.db.createSession(projectId, {
|
|
158
|
+
date,
|
|
159
|
+
...this.parseSessionBody(body)
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
parseSessionBody(body) {
|
|
165
|
+
const result = {};
|
|
166
|
+
const objectives = this.extractSection(body, 'Objectives', 'Mission');
|
|
167
|
+
if (objectives)
|
|
168
|
+
result.objectives = objectives;
|
|
169
|
+
const completed = this.extractSection(body, 'Completed', 'Done', 'Quests Completed');
|
|
170
|
+
if (completed)
|
|
171
|
+
result.completed = completed;
|
|
172
|
+
const issues = this.extractSection(body, 'Issues', 'Problems', 'Battles', 'Fixed');
|
|
173
|
+
if (issues)
|
|
174
|
+
result.issues = issues;
|
|
175
|
+
const commits = this.extractSection(body, 'Commits');
|
|
176
|
+
if (commits)
|
|
177
|
+
result.commits = commits;
|
|
178
|
+
const files = this.extractSection(body, 'Files', 'Changed');
|
|
179
|
+
if (files)
|
|
180
|
+
result.files_changed = files;
|
|
181
|
+
const notes = this.extractSection(body, 'Notes', 'Wisdom');
|
|
182
|
+
if (notes)
|
|
183
|
+
result.notes = notes;
|
|
184
|
+
return result;
|
|
185
|
+
}
|
|
186
|
+
// Parse quests.md
|
|
187
|
+
parseQuests(projectId, content) {
|
|
188
|
+
const lines = content.split('\n');
|
|
189
|
+
let currentPriority = 'medium';
|
|
190
|
+
for (const line of lines) {
|
|
191
|
+
// Check for priority headers
|
|
192
|
+
if (line.match(/critical|urgent/i))
|
|
193
|
+
currentPriority = 'critical';
|
|
194
|
+
else if (line.match(/high/i))
|
|
195
|
+
currentPriority = 'high';
|
|
196
|
+
else if (line.match(/medium/i))
|
|
197
|
+
currentPriority = 'medium';
|
|
198
|
+
else if (line.match(/low|future|backlog/i))
|
|
199
|
+
currentPriority = 'low';
|
|
200
|
+
// Check for task items
|
|
201
|
+
const taskMatch = line.match(/^[\s]*[-*]\s*\[([x\s])\]\s*\*?\*?(.+?)\*?\*?\s*$/i);
|
|
202
|
+
if (taskMatch) {
|
|
203
|
+
const isCompleted = taskMatch[1].toLowerCase() === 'x';
|
|
204
|
+
const title = taskMatch[2].trim();
|
|
205
|
+
// Add quest (avoid duplicates by title)
|
|
206
|
+
const existing = this.db.getPendingQuests(projectId).find(q => q.title === title);
|
|
207
|
+
if (!existing) {
|
|
208
|
+
const quest = this.db.addQuest(projectId, title, '', currentPriority);
|
|
209
|
+
if (isCompleted) {
|
|
210
|
+
this.db.updateQuest(quest.id, 'completed');
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
// Generate hoard.md from database
|
|
217
|
+
generateHoard(project) {
|
|
218
|
+
const context = this.db.getAllContext(project.id);
|
|
219
|
+
const lines = [
|
|
220
|
+
`# Wyrm Hoard // ${project.name}`,
|
|
221
|
+
'',
|
|
222
|
+
`> **Last Updated:** ${new Date().toISOString().split('T')[0]}`,
|
|
223
|
+
`> **Project:** ${project.name}`,
|
|
224
|
+
project.repo ? `> **Repo:** ${project.repo}` : '',
|
|
225
|
+
'',
|
|
226
|
+
'---',
|
|
227
|
+
''
|
|
228
|
+
];
|
|
229
|
+
// Add all context sections
|
|
230
|
+
for (const [key, value] of Object.entries(context)) {
|
|
231
|
+
const title = key.replace(/_/g, ' ').replace(/\b\w/g, c => c.toUpperCase());
|
|
232
|
+
lines.push(`## ${title}`, '', value, '');
|
|
233
|
+
}
|
|
234
|
+
return lines.filter(l => l !== undefined).join('\n');
|
|
235
|
+
}
|
|
236
|
+
// Generate chronicles.md from database
|
|
237
|
+
generateChronicles(projectId) {
|
|
238
|
+
const sessions = this.db.getRecentSessions(projectId, 20);
|
|
239
|
+
const lines = [
|
|
240
|
+
'# Wyrm Chronicles',
|
|
241
|
+
'',
|
|
242
|
+
'> Session history',
|
|
243
|
+
'',
|
|
244
|
+
'---',
|
|
245
|
+
''
|
|
246
|
+
];
|
|
247
|
+
for (const session of sessions) {
|
|
248
|
+
lines.push(`## Session: ${session.date}`, '');
|
|
249
|
+
if (session.objectives) {
|
|
250
|
+
lines.push('### Objectives', session.objectives, '');
|
|
251
|
+
}
|
|
252
|
+
if (session.completed) {
|
|
253
|
+
lines.push('### Completed', session.completed, '');
|
|
254
|
+
}
|
|
255
|
+
if (session.issues) {
|
|
256
|
+
lines.push('### Issues Solved', session.issues, '');
|
|
257
|
+
}
|
|
258
|
+
if (session.commits) {
|
|
259
|
+
lines.push('### Commits', session.commits, '');
|
|
260
|
+
}
|
|
261
|
+
if (session.files_changed) {
|
|
262
|
+
lines.push('### Files Changed', session.files_changed, '');
|
|
263
|
+
}
|
|
264
|
+
if (session.notes) {
|
|
265
|
+
lines.push('### Notes', session.notes, '');
|
|
266
|
+
}
|
|
267
|
+
lines.push('---', '');
|
|
268
|
+
}
|
|
269
|
+
return lines.join('\n');
|
|
270
|
+
}
|
|
271
|
+
// Generate quests.md from database
|
|
272
|
+
generateQuests(projectId) {
|
|
273
|
+
const pending = this.db.getPendingQuests(projectId);
|
|
274
|
+
const completed = this.db.getRecentlyCompleted(projectId, 10);
|
|
275
|
+
const lines = [
|
|
276
|
+
'# Wyrm Quests',
|
|
277
|
+
'',
|
|
278
|
+
'> Task queue',
|
|
279
|
+
'',
|
|
280
|
+
'---',
|
|
281
|
+
''
|
|
282
|
+
];
|
|
283
|
+
const byPriority = {
|
|
284
|
+
critical: pending.filter(q => q.priority === 'critical'),
|
|
285
|
+
high: pending.filter(q => q.priority === 'high'),
|
|
286
|
+
medium: pending.filter(q => q.priority === 'medium'),
|
|
287
|
+
low: pending.filter(q => q.priority === 'low')
|
|
288
|
+
};
|
|
289
|
+
if (byPriority.critical.length > 0) {
|
|
290
|
+
lines.push('## Critical', '');
|
|
291
|
+
for (const q of byPriority.critical) {
|
|
292
|
+
lines.push(`- [ ] **${q.title}**${q.description ? ` - ${q.description}` : ''}`);
|
|
293
|
+
}
|
|
294
|
+
lines.push('');
|
|
295
|
+
}
|
|
296
|
+
if (byPriority.high.length > 0) {
|
|
297
|
+
lines.push('## High Priority', '');
|
|
298
|
+
for (const q of byPriority.high) {
|
|
299
|
+
lines.push(`- [ ] **${q.title}**${q.description ? ` - ${q.description}` : ''}`);
|
|
300
|
+
}
|
|
301
|
+
lines.push('');
|
|
302
|
+
}
|
|
303
|
+
if (byPriority.medium.length > 0) {
|
|
304
|
+
lines.push('## Medium Priority', '');
|
|
305
|
+
for (const q of byPriority.medium) {
|
|
306
|
+
lines.push(`- [ ] ${q.title}${q.description ? ` - ${q.description}` : ''}`);
|
|
307
|
+
}
|
|
308
|
+
lines.push('');
|
|
309
|
+
}
|
|
310
|
+
if (byPriority.low.length > 0) {
|
|
311
|
+
lines.push('## Low Priority / Future', '');
|
|
312
|
+
for (const q of byPriority.low) {
|
|
313
|
+
lines.push(`- [ ] ${q.title}`);
|
|
314
|
+
}
|
|
315
|
+
lines.push('');
|
|
316
|
+
}
|
|
317
|
+
if (completed.length > 0) {
|
|
318
|
+
lines.push('## Completed', '');
|
|
319
|
+
for (const q of completed) {
|
|
320
|
+
lines.push(`- [x] ${q.title} (${q.completed_at?.split('T')[0]})`);
|
|
321
|
+
}
|
|
322
|
+
lines.push('');
|
|
323
|
+
}
|
|
324
|
+
return lines.join('\n');
|
|
325
|
+
}
|
|
326
|
+
// Helpers
|
|
327
|
+
splitSections(content) {
|
|
328
|
+
const sections = {};
|
|
329
|
+
const regex = /^##\s+(.+?)$/gm;
|
|
330
|
+
let lastTitle = '';
|
|
331
|
+
let lastIndex = 0;
|
|
332
|
+
let match;
|
|
333
|
+
while ((match = regex.exec(content)) !== null) {
|
|
334
|
+
if (lastTitle) {
|
|
335
|
+
sections[lastTitle] = content.slice(lastIndex, match.index).trim();
|
|
336
|
+
}
|
|
337
|
+
lastTitle = match[1];
|
|
338
|
+
lastIndex = match.index + match[0].length;
|
|
339
|
+
}
|
|
340
|
+
if (lastTitle) {
|
|
341
|
+
sections[lastTitle] = content.slice(lastIndex).trim();
|
|
342
|
+
}
|
|
343
|
+
return sections;
|
|
344
|
+
}
|
|
345
|
+
extractSection(body, ...titles) {
|
|
346
|
+
for (const title of titles) {
|
|
347
|
+
const regex = new RegExp(`###?\\s*${title}[^\\n]*\\n([\\s\\S]*?)(?=###|$)`, 'i');
|
|
348
|
+
const match = body.match(regex);
|
|
349
|
+
if (match) {
|
|
350
|
+
return match[1].trim();
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
return undefined;
|
|
354
|
+
}
|
|
355
|
+
}
|
|
356
|
+
//# sourceMappingURL=sync.js.map
|
package/dist/sync.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync.js","sourceRoot":"","sources":["../src/sync.ts"],"names":[],"mappings":"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"}
|