mcp-context-sync 1.0.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 (72) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +85 -0
  3. package/dist/cli.d.ts +12 -0
  4. package/dist/cli.js +255 -0
  5. package/dist/cli.js.map +1 -0
  6. package/dist/db/connection.d.ts +3 -0
  7. package/dist/db/connection.js +44 -0
  8. package/dist/db/connection.js.map +1 -0
  9. package/dist/db/queries.d.ts +23 -0
  10. package/dist/db/queries.js +121 -0
  11. package/dist/db/queries.js.map +1 -0
  12. package/dist/db/schema.d.ts +2 -0
  13. package/dist/db/schema.js +103 -0
  14. package/dist/db/schema.js.map +1 -0
  15. package/dist/index.d.ts +1 -0
  16. package/dist/index.js +56 -0
  17. package/dist/index.js.map +1 -0
  18. package/dist/lib/project-id.d.ts +13 -0
  19. package/dist/lib/project-id.js +27 -0
  20. package/dist/lib/project-id.js.map +1 -0
  21. package/dist/lib/snapshot-renderer.d.ts +9 -0
  22. package/dist/lib/snapshot-renderer.js +77 -0
  23. package/dist/lib/snapshot-renderer.js.map +1 -0
  24. package/dist/lib/types.d.ts +290 -0
  25. package/dist/lib/types.js +122 -0
  26. package/dist/lib/types.js.map +1 -0
  27. package/dist/lib/uuid.d.ts +1 -0
  28. package/dist/lib/uuid.js +5 -0
  29. package/dist/lib/uuid.js.map +1 -0
  30. package/dist/lib/validation.d.ts +18 -0
  31. package/dist/lib/validation.js +63 -0
  32. package/dist/lib/validation.js.map +1 -0
  33. package/dist/prompts/auto-resume.d.ts +3 -0
  34. package/dist/prompts/auto-resume.js +70 -0
  35. package/dist/prompts/auto-resume.js.map +1 -0
  36. package/dist/prompts/resume-work.d.ts +2 -0
  37. package/dist/prompts/resume-work.js +29 -0
  38. package/dist/prompts/resume-work.js.map +1 -0
  39. package/dist/prompts/sync-session.d.ts +2 -0
  40. package/dist/prompts/sync-session.js +33 -0
  41. package/dist/prompts/sync-session.js.map +1 -0
  42. package/dist/resources/project-decisions.d.ts +3 -0
  43. package/dist/resources/project-decisions.js +69 -0
  44. package/dist/resources/project-decisions.js.map +1 -0
  45. package/dist/resources/project-snapshot.d.ts +3 -0
  46. package/dist/resources/project-snapshot.js +49 -0
  47. package/dist/resources/project-snapshot.js.map +1 -0
  48. package/dist/resources/status.d.ts +3 -0
  49. package/dist/resources/status.js +25 -0
  50. package/dist/resources/status.js.map +1 -0
  51. package/dist/tools/amend-snapshot.d.ts +3 -0
  52. package/dist/tools/amend-snapshot.js +126 -0
  53. package/dist/tools/amend-snapshot.js.map +1 -0
  54. package/dist/tools/get-history.d.ts +3 -0
  55. package/dist/tools/get-history.js +58 -0
  56. package/dist/tools/get-history.js.map +1 -0
  57. package/dist/tools/list-projects.d.ts +3 -0
  58. package/dist/tools/list-projects.js +40 -0
  59. package/dist/tools/list-projects.js.map +1 -0
  60. package/dist/tools/log-decision.d.ts +3 -0
  61. package/dist/tools/log-decision.js +46 -0
  62. package/dist/tools/log-decision.js.map +1 -0
  63. package/dist/tools/resume.d.ts +3 -0
  64. package/dist/tools/resume.js +113 -0
  65. package/dist/tools/resume.js.map +1 -0
  66. package/dist/tools/search-snapshots.d.ts +3 -0
  67. package/dist/tools/search-snapshots.js +71 -0
  68. package/dist/tools/search-snapshots.js.map +1 -0
  69. package/dist/tools/sync.d.ts +3 -0
  70. package/dist/tools/sync.js +102 -0
  71. package/dist/tools/sync.js.map +1 -0
  72. package/package.json +48 -0
@@ -0,0 +1,126 @@
1
+ import { AmendSnapshotInputSchema } from '../lib/types.js';
2
+ import { projectIdFromPath, displayNameFromPath, } from '../lib/project-id.js';
3
+ import { getLatestSnapshot, updateSnapshot, insertDecision } from '../db/queries.js';
4
+ import { generateId } from '../lib/uuid.js';
5
+ export function registerAmendSnapshotTool(server, db) {
6
+ server.registerTool('amend-snapshot', {
7
+ title: 'Amend Last Snapshot',
8
+ description: 'Update the most recent snapshot for a project. Useful for adding forgotten tasks, ' +
9
+ 'correcting decisions, or updating next steps before another agent resumes.',
10
+ inputSchema: AmendSnapshotInputSchema,
11
+ annotations: {
12
+ readOnlyHint: false,
13
+ destructiveHint: false,
14
+ idempotentHint: false,
15
+ openWorldHint: false,
16
+ },
17
+ }, async (input) => {
18
+ const projectId = projectIdFromPath(input.projectDir);
19
+ const displayName = displayNameFromPath(input.projectDir);
20
+ const snapshot = getLatestSnapshot(db, projectId);
21
+ if (!snapshot) {
22
+ return {
23
+ content: [
24
+ {
25
+ type: 'text',
26
+ text: `No snapshot found for project "${displayName}" to amend.`,
27
+ },
28
+ ],
29
+ };
30
+ }
31
+ const now = new Date().toISOString();
32
+ const changes = [];
33
+ // Tasks completed
34
+ if (input.addTasksCompleted?.length) {
35
+ const current = JSON.parse(snapshot.tasks_completed);
36
+ current.push(...input.addTasksCompleted);
37
+ updateSnapshot(db, snapshot.id, {
38
+ tasks_completed: JSON.stringify(current),
39
+ });
40
+ changes.push(`Added ${input.addTasksCompleted.length} completed tasks`);
41
+ }
42
+ // Tasks remaining
43
+ let remaining = JSON.parse(snapshot.tasks_remaining);
44
+ if (input.removeTasksRemaining?.length) {
45
+ const toRemove = new Set(input.removeTasksRemaining);
46
+ remaining = remaining.filter((t) => !toRemove.has(t));
47
+ changes.push(`Removed ${input.removeTasksRemaining.length} remaining tasks`);
48
+ }
49
+ if (input.addTasksRemaining?.length) {
50
+ remaining.push(...input.addTasksRemaining);
51
+ changes.push(`Added ${input.addTasksRemaining.length} remaining tasks`);
52
+ }
53
+ if (input.removeTasksRemaining?.length || input.addTasksRemaining?.length) {
54
+ updateSnapshot(db, snapshot.id, {
55
+ tasks_remaining: JSON.stringify(remaining),
56
+ });
57
+ }
58
+ // Decisions
59
+ if (input.addDecisions?.length) {
60
+ const current = JSON.parse(snapshot.decisions);
61
+ current.push(...input.addDecisions);
62
+ updateSnapshot(db, snapshot.id, {
63
+ decisions: JSON.stringify(current),
64
+ });
65
+ for (const d of input.addDecisions) {
66
+ insertDecision(db, {
67
+ id: generateId(),
68
+ project_id: projectId,
69
+ snapshot_id: snapshot.id,
70
+ agent: input.agent,
71
+ decision: d.decision,
72
+ reasoning: d.reasoning,
73
+ alternatives: JSON.stringify(d.alternatives ?? []),
74
+ category: null,
75
+ related_files: '[]',
76
+ created_at: now,
77
+ });
78
+ }
79
+ changes.push(`Added ${input.addDecisions.length} decisions`);
80
+ }
81
+ // Next steps
82
+ if (input.updateNextSteps) {
83
+ updateSnapshot(db, snapshot.id, {
84
+ next_steps: input.updateNextSteps,
85
+ });
86
+ changes.push('Updated next steps');
87
+ }
88
+ // Blockers
89
+ let blockers = JSON.parse(snapshot.blockers);
90
+ if (input.removeBlockers?.length) {
91
+ const toRemove = new Set(input.removeBlockers);
92
+ blockers = blockers.filter((b) => !toRemove.has(b));
93
+ changes.push(`Removed ${input.removeBlockers.length} blockers`);
94
+ }
95
+ if (input.addBlockers?.length) {
96
+ blockers.push(...input.addBlockers);
97
+ changes.push(`Added ${input.addBlockers.length} blockers`);
98
+ }
99
+ if (input.removeBlockers?.length || input.addBlockers?.length) {
100
+ updateSnapshot(db, snapshot.id, {
101
+ blockers: JSON.stringify(blockers),
102
+ });
103
+ }
104
+ // Tags
105
+ if (input.addTags?.length) {
106
+ const current = JSON.parse(snapshot.tags);
107
+ const unique = [...new Set([...current, ...input.addTags])];
108
+ updateSnapshot(db, snapshot.id, { tags: JSON.stringify(unique) });
109
+ changes.push(`Added ${input.addTags.length} tags`);
110
+ }
111
+ // Mark as amended
112
+ updateSnapshot(db, snapshot.id, {
113
+ amended_at: now,
114
+ amended_by: input.agent,
115
+ });
116
+ return {
117
+ content: [
118
+ {
119
+ type: 'text',
120
+ text: `Snapshot #${snapshot.sequence_number} amended for "${displayName}":\n\n${changes.map((c) => `- ${c}`).join('\n')}`,
121
+ },
122
+ ],
123
+ };
124
+ });
125
+ }
126
+ //# sourceMappingURL=amend-snapshot.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"amend-snapshot.js","sourceRoot":"","sources":["../../src/tools/amend-snapshot.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import type Database from 'better-sqlite3';
3
+ export declare function registerGetHistoryTool(server: McpServer, db: Database.Database): void;
@@ -0,0 +1,58 @@
1
+ import { GetHistoryInputSchema } from '../lib/types.js';
2
+ import { projectIdFromPath, displayNameFromPath, } from '../lib/project-id.js';
3
+ import { getProject, getSnapshotsByProject, getHandoffsByProject, } from '../db/queries.js';
4
+ export function registerGetHistoryTool(server, db) {
5
+ server.registerTool('get-history', {
6
+ title: 'Get Project Handoff History',
7
+ description: 'View the timeline of agent handoffs, snapshots, and decisions for a project.',
8
+ inputSchema: GetHistoryInputSchema,
9
+ annotations: {
10
+ readOnlyHint: true,
11
+ destructiveHint: false,
12
+ idempotentHint: true,
13
+ openWorldHint: false,
14
+ },
15
+ }, async (input) => {
16
+ const projectId = projectIdFromPath(input.projectDir);
17
+ const displayName = displayNameFromPath(input.projectDir);
18
+ const project = getProject(db, projectId);
19
+ if (!project) {
20
+ return {
21
+ content: [
22
+ {
23
+ type: 'text',
24
+ text: `No history found for project "${displayName}".`,
25
+ },
26
+ ],
27
+ };
28
+ }
29
+ let snapshots = getSnapshotsByProject(db, projectId, input.limit);
30
+ if (input.agentFilter !== 'all') {
31
+ snapshots = snapshots.filter((s) => s.agent === input.agentFilter);
32
+ }
33
+ const handoffs = getHandoffsByProject(db, projectId, 50);
34
+ const lines = [];
35
+ lines.push(`# Project Timeline: ${displayName}`);
36
+ lines.push(`Total: ${project.snapshot_count} snapshots, ${project.handoff_count} handoffs`);
37
+ lines.push('');
38
+ for (const s of snapshots.reverse()) {
39
+ lines.push(`## Snapshot #${s.sequence_number} -- ${s.agent} -- ${s.created_at}`);
40
+ lines.push(s.summary);
41
+ if (input.includeDecisions) {
42
+ const decs = JSON.parse(s.decisions);
43
+ for (const d of decs) {
44
+ lines.push(` Decision: ${d.decision}`);
45
+ }
46
+ }
47
+ const ho = handoffs.find((h) => h.snapshot_id === s.id);
48
+ if (ho) {
49
+ lines.push(` Handoff: ${ho.from_agent} -> ${ho.to_agent} (${ho.to_timestamp})`);
50
+ }
51
+ lines.push('');
52
+ }
53
+ return {
54
+ content: [{ type: 'text', text: lines.join('\n') }],
55
+ };
56
+ });
57
+ }
58
+ //# sourceMappingURL=get-history.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"get-history.js","sourceRoot":"","sources":["../../src/tools/get-history.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import type Database from 'better-sqlite3';
3
+ export declare function registerListProjectsTool(server: McpServer, db: Database.Database): void;
@@ -0,0 +1,40 @@
1
+ import { ListProjectsInputSchema } from '../lib/types.js';
2
+ import { listProjects } from '../db/queries.js';
3
+ export function registerListProjectsTool(server, db) {
4
+ server.registerTool('list-projects', {
5
+ title: 'List Synced Projects',
6
+ description: 'List all projects that have context-sync snapshots, with summary stats.',
7
+ inputSchema: ListProjectsInputSchema,
8
+ annotations: {
9
+ readOnlyHint: true,
10
+ destructiveHint: false,
11
+ idempotentHint: true,
12
+ openWorldHint: false,
13
+ },
14
+ }, async () => {
15
+ const projects = listProjects(db);
16
+ if (projects.length === 0) {
17
+ return {
18
+ content: [
19
+ {
20
+ type: 'text',
21
+ text: 'No projects have been synced yet. Use the "sync" tool to create your first snapshot.',
22
+ },
23
+ ],
24
+ };
25
+ }
26
+ const lines = ['# Synced Projects', ''];
27
+ for (const p of projects) {
28
+ lines.push(`## ${p.display_name}`);
29
+ lines.push(`- **Path:** ${p.normalized_path}`);
30
+ lines.push(`- **Snapshots:** ${p.snapshot_count}`);
31
+ lines.push(`- **Handoffs:** ${p.handoff_count}`);
32
+ lines.push(`- **Last synced:** ${p.last_synced_at ?? 'never'}`);
33
+ lines.push('');
34
+ }
35
+ return {
36
+ content: [{ type: 'text', text: lines.join('\n') }],
37
+ };
38
+ });
39
+ }
40
+ //# sourceMappingURL=list-projects.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"list-projects.js","sourceRoot":"","sources":["../../src/tools/list-projects.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,uBAAuB,EAAE,MAAM,iBAAiB,CAAC;AAC1D,OAAO,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAEhD,MAAM,UAAU,wBAAwB,CACtC,MAAiB,EACjB,EAAqB;IAErB,MAAM,CAAC,YAAY,CACjB,eAAe,EACf;QACE,KAAK,EAAE,sBAAsB;QAC7B,WAAW,EACT,yEAAyE;QAC3E,WAAW,EAAE,uBAAuB;QACpC,WAAW,EAAE;YACX,YAAY,EAAE,IAAI;YAClB,eAAe,EAAE,KAAK;YACtB,cAAc,EAAE,IAAI;YACpB,aAAa,EAAE,KAAK;SACrB;KACF,EACD,KAAK,IAAI,EAAE;QACT,MAAM,QAAQ,GAAG,YAAY,CAAC,EAAE,CAAC,CAAC;QAElC,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC1B,OAAO;gBACL,OAAO,EAAE;oBACP;wBACE,IAAI,EAAE,MAAe;wBACrB,IAAI,EAAE,sFAAsF;qBAC7F;iBACF;aACF,CAAC;QACJ,CAAC;QAED,MAAM,KAAK,GAAa,CAAC,mBAAmB,EAAE,EAAE,CAAC,CAAC;QAClD,KAAK,MAAM,CAAC,IAAI,QAAQ,EAAE,CAAC;YACzB,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,YAAY,EAAE,CAAC,CAAC;YACnC,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,eAAe,EAAE,CAAC,CAAC;YAC/C,KAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,cAAc,EAAE,CAAC,CAAC;YACnD,KAAK,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,aAAa,EAAE,CAAC,CAAC;YACjD,KAAK,CAAC,IAAI,CACR,sBAAsB,CAAC,CAAC,cAAc,IAAI,OAAO,EAAE,CACpD,CAAC;YACF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,OAAO;YACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;SAC7D,CAAC;IACJ,CAAC,CACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import type Database from 'better-sqlite3';
3
+ export declare function registerLogDecisionTool(server: McpServer, db: Database.Database): void;
@@ -0,0 +1,46 @@
1
+ import { LogDecisionInputSchema } from '../lib/types.js';
2
+ import { projectIdFromPath, normalizePath, displayNameFromPath, } from '../lib/project-id.js';
3
+ import { generateId } from '../lib/uuid.js';
4
+ import { upsertProject, insertDecision } from '../db/queries.js';
5
+ export function registerLogDecisionTool(server, db) {
6
+ server.registerTool('log-decision', {
7
+ title: 'Log Decision',
8
+ description: 'Record an architectural or implementation decision for the project. ' +
9
+ 'Useful during a session when a significant decision is made that future agents should know about.',
10
+ inputSchema: LogDecisionInputSchema,
11
+ annotations: {
12
+ readOnlyHint: false,
13
+ destructiveHint: false,
14
+ idempotentHint: false,
15
+ openWorldHint: false,
16
+ },
17
+ }, async (input) => {
18
+ const projectId = projectIdFromPath(input.projectDir);
19
+ const normalized = normalizePath(input.projectDir);
20
+ const displayName = displayNameFromPath(input.projectDir);
21
+ const now = new Date().toISOString();
22
+ const decisionId = generateId();
23
+ upsertProject(db, projectId, normalized, displayName);
24
+ insertDecision(db, {
25
+ id: decisionId,
26
+ project_id: projectId,
27
+ snapshot_id: null,
28
+ agent: input.agent,
29
+ decision: input.decision,
30
+ reasoning: input.reasoning,
31
+ alternatives: JSON.stringify(input.alternatives ?? []),
32
+ category: input.category ?? null,
33
+ related_files: JSON.stringify(input.relatedFiles ?? []),
34
+ created_at: now,
35
+ });
36
+ return {
37
+ content: [
38
+ {
39
+ type: 'text',
40
+ text: `Decision logged for "${displayName}":\n\n**${input.decision}**\n${input.reasoning}\n\nID: ${decisionId}`,
41
+ },
42
+ ],
43
+ };
44
+ });
45
+ }
46
+ //# sourceMappingURL=log-decision.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"log-decision.js","sourceRoot":"","sources":["../../src/tools/log-decision.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,sBAAsB,EAAE,MAAM,iBAAiB,CAAC;AACzD,OAAO,EACL,iBAAiB,EACjB,aAAa,EACb,mBAAmB,GACpB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EAAE,aAAa,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAEjE,MAAM,UAAU,uBAAuB,CACrC,MAAiB,EACjB,EAAqB;IAErB,MAAM,CAAC,YAAY,CACjB,cAAc,EACd;QACE,KAAK,EAAE,cAAc;QACrB,WAAW,EACT,sEAAsE;YACtE,mGAAmG;QACrG,WAAW,EAAE,sBAAsB;QACnC,WAAW,EAAE;YACX,YAAY,EAAE,KAAK;YACnB,eAAe,EAAE,KAAK;YACtB,cAAc,EAAE,KAAK;YACrB,aAAa,EAAE,KAAK;SACrB;KACF,EACD,KAAK,EAAE,KAAK,EAAE,EAAE;QACd,MAAM,SAAS,GAAG,iBAAiB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACtD,MAAM,UAAU,GAAG,aAAa,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACnD,MAAM,WAAW,GAAG,mBAAmB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAC1D,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QACrC,MAAM,UAAU,GAAG,UAAU,EAAE,CAAC;QAEhC,aAAa,CAAC,EAAE,EAAE,SAAS,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC;QAEtD,cAAc,CAAC,EAAE,EAAE;YACjB,EAAE,EAAE,UAAU;YACd,UAAU,EAAE,SAAS;YACrB,WAAW,EAAE,IAAI;YACjB,KAAK,EAAE,KAAK,CAAC,KAAK;YAClB,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,SAAS,EAAE,KAAK,CAAC,SAAS;YAC1B,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,YAAY,IAAI,EAAE,CAAC;YACtD,QAAQ,EAAE,KAAK,CAAC,QAAQ,IAAI,IAAI;YAChC,aAAa,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,YAAY,IAAI,EAAE,CAAC;YACvD,UAAU,EAAE,GAAG;SAChB,CAAC,CAAC;QAEH,OAAO;YACL,OAAO,EAAE;gBACP;oBACE,IAAI,EAAE,MAAe;oBACrB,IAAI,EAAE,wBAAwB,WAAW,WAAW,KAAK,CAAC,QAAQ,OAAO,KAAK,CAAC,SAAS,WAAW,UAAU,EAAE;iBAChH;aACF;SACF,CAAC;IACJ,CAAC,CACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import type Database from 'better-sqlite3';
3
+ export declare function registerResumeTool(server: McpServer, db: Database.Database): void;
@@ -0,0 +1,113 @@
1
+ import { ResumeInputSchema } from '../lib/types.js';
2
+ import { projectIdFromPath, displayNameFromPath, normalizePath, } from '../lib/project-id.js';
3
+ import { generateId } from '../lib/uuid.js';
4
+ import { renderSnapshotMarkdown } from '../lib/snapshot-renderer.js';
5
+ import { getProject, getLatestSnapshot, getSnapshotById, insertHandoff, updateProjectSyncStats, getProjectAgents, listProjects, } from '../db/queries.js';
6
+ export function registerResumeTool(server, db) {
7
+ server.registerTool('resume', {
8
+ title: 'Resume from Context Snapshot',
9
+ description: 'Load the latest context snapshot for a project to continue where another agent left off. ' +
10
+ 'Returns the full snapshot with summary, pending tasks, decisions, and next steps.',
11
+ inputSchema: ResumeInputSchema,
12
+ annotations: {
13
+ readOnlyHint: false, // writes a handoff record
14
+ destructiveHint: false,
15
+ idempotentHint: false,
16
+ openWorldHint: false,
17
+ },
18
+ }, async (input) => {
19
+ const projectId = projectIdFromPath(input.projectDir);
20
+ const displayName = displayNameFromPath(input.projectDir);
21
+ const project = getProject(db, projectId);
22
+ if (!project) {
23
+ return { content: [{ type: 'text', text: buildNotFoundMessage(db, input.projectDir, displayName, projectId) }] };
24
+ }
25
+ const snapshot = input.snapshotId
26
+ ? getSnapshotById(db, input.snapshotId)
27
+ : getLatestSnapshot(db, projectId);
28
+ if (!snapshot) {
29
+ return {
30
+ content: [{
31
+ type: 'text',
32
+ text: `Project "${displayName}" exists but has no snapshots${input.snapshotId ? ` with ID "${input.snapshotId}"` : ''}.`,
33
+ }],
34
+ };
35
+ }
36
+ // Register the handoff in a transaction
37
+ const now = new Date().toISOString();
38
+ const agents = db.transaction(() => {
39
+ insertHandoff(db, {
40
+ id: generateId(),
41
+ project_id: projectId,
42
+ snapshot_id: snapshot.id,
43
+ from_agent: snapshot.agent,
44
+ to_agent: input.agent,
45
+ from_timestamp: snapshot.created_at,
46
+ to_timestamp: now,
47
+ created_at: now,
48
+ });
49
+ updateProjectSyncStats(db, projectId);
50
+ return getProjectAgents(db, projectId);
51
+ })();
52
+ const markdown = renderSnapshotMarkdown(snapshot, displayName, {
53
+ totalSnapshots: project.snapshot_count,
54
+ totalHandoffs: project.handoff_count + 1,
55
+ agents,
56
+ });
57
+ return { content: [{ type: 'text', text: markdown }] };
58
+ });
59
+ }
60
+ // ---------------------------------------------------------------------------
61
+ // Error message builder
62
+ // ---------------------------------------------------------------------------
63
+ function buildNotFoundMessage(db, inputPath, displayName, projectId) {
64
+ const normalized = normalizePath(inputPath);
65
+ const allProjects = listProjects(db);
66
+ const lines = [
67
+ `No context-sync data found for project "${displayName}"`,
68
+ '',
69
+ '**Searched for:**',
70
+ `- Path: \`${normalized}\``,
71
+ `- ID: \`${projectId}\``,
72
+ '',
73
+ ];
74
+ if (allProjects.length === 0) {
75
+ lines.push('No projects have been synced yet. Use the `sync` tool first.');
76
+ return lines.join('\n');
77
+ }
78
+ lines.push(`**Available projects (${allProjects.length}):**`);
79
+ // Rank projects by path similarity (shared prefix segments)
80
+ const ranked = allProjects
81
+ .map((p) => ({ ...p, score: pathSimilarity(normalized, p.normalized_path) }))
82
+ .sort((a, b) => b.score - a.score);
83
+ for (const p of ranked) {
84
+ const marker = p === ranked[0] ? '> ' : ' ';
85
+ lines.push(`${marker}\`${p.normalized_path}\` (${p.snapshot_count} snapshots)`);
86
+ }
87
+ // Suggest the best match if reasonably close
88
+ const best = ranked[0];
89
+ if (best.score > 0.5) {
90
+ lines.push('', `**Suggestion:** Did you mean \`${best.normalized_path}\`?`);
91
+ }
92
+ lines.push('', '**Troubleshooting:**', '- Use the absolute path to the **project root**, not a subdirectory', '- Path normalization is automatic (lowercase, forward slashes)', '- If none match, create a new snapshot with `sync`');
93
+ return lines.join('\n');
94
+ }
95
+ // ---------------------------------------------------------------------------
96
+ // Path similarity: ratio of shared prefix segments to total segments
97
+ // ---------------------------------------------------------------------------
98
+ function pathSimilarity(a, b) {
99
+ const segsA = a.split('/').filter(Boolean);
100
+ const segsB = b.split('/').filter(Boolean);
101
+ const maxLen = Math.max(segsA.length, segsB.length);
102
+ if (maxLen === 0)
103
+ return 0;
104
+ let shared = 0;
105
+ for (let i = 0; i < Math.min(segsA.length, segsB.length); i++) {
106
+ if (segsA[i] === segsB[i])
107
+ shared++;
108
+ else
109
+ break;
110
+ }
111
+ return shared / maxLen;
112
+ }
113
+ //# sourceMappingURL=resume.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resume.js","sourceRoot":"","sources":["../../src/tools/resume.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AACpD,OAAO,EACL,iBAAiB,EACjB,mBAAmB,EACnB,aAAa,GACd,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EAAE,sBAAsB,EAAE,MAAM,6BAA6B,CAAC;AACrE,OAAO,EACL,UAAU,EACV,iBAAiB,EACjB,eAAe,EACf,aAAa,EACb,sBAAsB,EACtB,gBAAgB,EAChB,YAAY,GACb,MAAM,kBAAkB,CAAC;AAE1B,MAAM,UAAU,kBAAkB,CAAC,MAAiB,EAAE,EAAqB;IACzE,MAAM,CAAC,YAAY,CACjB,QAAQ,EACR;QACE,KAAK,EAAE,8BAA8B;QACrC,WAAW,EACT,2FAA2F;YAC3F,mFAAmF;QACrF,WAAW,EAAE,iBAAiB;QAC9B,WAAW,EAAE;YACX,YAAY,EAAE,KAAK,EAAE,0BAA0B;YAC/C,eAAe,EAAE,KAAK;YACtB,cAAc,EAAE,KAAK;YACrB,aAAa,EAAE,KAAK;SACrB;KACF,EACD,KAAK,EAAE,KAAK,EAAE,EAAE;QACd,MAAM,SAAS,GAAG,iBAAiB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACtD,MAAM,WAAW,GAAG,mBAAmB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAE1D,MAAM,OAAO,GAAG,UAAU,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;QAC1C,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,oBAAoB,CAAC,EAAE,EAAE,KAAK,CAAC,UAAU,EAAE,WAAW,EAAE,SAAS,CAAC,EAAE,CAAC,EAAE,CAAC;QAC5H,CAAC;QAED,MAAM,QAAQ,GAAG,KAAK,CAAC,UAAU;YAC/B,CAAC,CAAC,eAAe,CAAC,EAAE,EAAE,KAAK,CAAC,UAAU,CAAC;YACvC,CAAC,CAAC,iBAAiB,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;QAErC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,OAAO;gBACL,OAAO,EAAE,CAAC;wBACR,IAAI,EAAE,MAAe;wBACrB,IAAI,EAAE,YAAY,WAAW,gCAAgC,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,aAAa,KAAK,CAAC,UAAU,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG;qBACzH,CAAC;aACH,CAAC;QACJ,CAAC;QAED,wCAAwC;QACxC,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QACrC,MAAM,MAAM,GAAG,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE;YACjC,aAAa,CAAC,EAAE,EAAE;gBAChB,EAAE,EAAE,UAAU,EAAE;gBAChB,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,QAAQ,CAAC,EAAE;gBACxB,UAAU,EAAE,QAAQ,CAAC,KAAK;gBAC1B,QAAQ,EAAE,KAAK,CAAC,KAAK;gBACrB,cAAc,EAAE,QAAQ,CAAC,UAAU;gBACnC,YAAY,EAAE,GAAG;gBACjB,UAAU,EAAE,GAAG;aAChB,CAAC,CAAC;YACH,sBAAsB,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;YACtC,OAAO,gBAAgB,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;QACzC,CAAC,CAAC,EAAE,CAAC;QAEL,MAAM,QAAQ,GAAG,sBAAsB,CAAC,QAAQ,EAAE,WAAW,EAAE;YAC7D,cAAc,EAAE,OAAO,CAAC,cAAc;YACtC,aAAa,EAAE,OAAO,CAAC,aAAa,GAAG,CAAC;YACxC,MAAM;SACP,CAAC,CAAC;QAEH,OAAO,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAC;IAClE,CAAC,CACF,CAAC;AACJ,CAAC;AAED,8EAA8E;AAC9E,wBAAwB;AACxB,8EAA8E;AAE9E,SAAS,oBAAoB,CAC3B,EAAqB,EACrB,SAAiB,EACjB,WAAmB,EACnB,SAAiB;IAEjB,MAAM,UAAU,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC;IAC5C,MAAM,WAAW,GAAG,YAAY,CAAC,EAAE,CAAC,CAAC;IAErC,MAAM,KAAK,GAAa;QACtB,2CAA2C,WAAW,GAAG;QACzD,EAAE;QACF,mBAAmB;QACnB,aAAa,UAAU,IAAI;QAC3B,WAAW,SAAS,IAAI;QACxB,EAAE;KACH,CAAC;IAEF,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC7B,KAAK,CAAC,IAAI,CAAC,8DAA8D,CAAC,CAAC;QAC3E,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,yBAAyB,WAAW,CAAC,MAAM,MAAM,CAAC,CAAC;IAE9D,4DAA4D;IAC5D,MAAM,MAAM,GAAG,WAAW;SACvB,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,KAAK,EAAE,cAAc,CAAC,UAAU,EAAE,CAAC,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC;SAC5E,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC;IAErC,KAAK,MAAM,CAAC,IAAI,MAAM,EAAE,CAAC;QACvB,MAAM,MAAM,GAAG,CAAC,KAAK,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;QAC7C,KAAK,CAAC,IAAI,CAAC,GAAG,MAAM,KAAK,CAAC,CAAC,eAAe,QAAQ,CAAC,CAAC,cAAc,aAAa,CAAC,CAAC;IACnF,CAAC;IAED,6CAA6C;IAC7C,MAAM,IAAI,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IACvB,IAAI,IAAI,CAAC,KAAK,GAAG,GAAG,EAAE,CAAC;QACrB,KAAK,CAAC,IAAI,CAAC,EAAE,EAAE,kCAAkC,IAAI,CAAC,eAAe,KAAK,CAAC,CAAC;IAC9E,CAAC;IAED,KAAK,CAAC,IAAI,CACR,EAAE,EACF,sBAAsB,EACtB,qEAAqE,EACrE,gEAAgE,EAChE,oDAAoD,CACrD,CAAC;IAEF,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AAED,8EAA8E;AAC9E,qEAAqE;AACrE,8EAA8E;AAE9E,SAAS,cAAc,CAAC,CAAS,EAAE,CAAS;IAC1C,MAAM,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IAC3C,MAAM,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IAC3C,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC;IACpD,IAAI,MAAM,KAAK,CAAC;QAAE,OAAO,CAAC,CAAC;IAE3B,IAAI,MAAM,GAAG,CAAC,CAAC;IACf,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,EAAE,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QAC9D,IAAI,KAAK,CAAC,CAAC,CAAC,KAAK,KAAK,CAAC,CAAC,CAAC;YAAE,MAAM,EAAE,CAAC;;YAC/B,MAAM;IACb,CAAC;IACD,OAAO,MAAM,GAAG,MAAM,CAAC;AACzB,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import type Database from 'better-sqlite3';
3
+ export declare function registerSearchSnapshotsTool(server: McpServer, db: Database.Database): void;
@@ -0,0 +1,71 @@
1
+ import { SearchSnapshotsInputSchema } from '../lib/types.js';
2
+ import { projectIdFromPath } from '../lib/project-id.js';
3
+ import { searchSnapshots, getSnapshotsByProject } from '../db/queries.js';
4
+ export function registerSearchSnapshotsTool(server, db) {
5
+ server.registerTool('search-snapshots', {
6
+ title: 'Search Snapshots',
7
+ description: 'Search across context snapshots by keyword, tag, date range, or agent.',
8
+ inputSchema: SearchSnapshotsInputSchema,
9
+ annotations: {
10
+ readOnlyHint: true,
11
+ destructiveHint: false,
12
+ idempotentHint: true,
13
+ openWorldHint: false,
14
+ },
15
+ }, async (input) => {
16
+ const projectId = input.projectDir
17
+ ? projectIdFromPath(input.projectDir)
18
+ : undefined;
19
+ let results;
20
+ if (input.query) {
21
+ results = searchSnapshots(db, input.query, projectId, input.agent, input.since, input.until, input.limit);
22
+ }
23
+ else if (projectId) {
24
+ // No query text, just filter by project
25
+ results = getSnapshotsByProject(db, projectId, input.limit);
26
+ if (input.agent) {
27
+ results = results.filter((s) => s.agent === input.agent);
28
+ }
29
+ if (input.since) {
30
+ results = results.filter((s) => s.created_at >= input.since);
31
+ }
32
+ if (input.until) {
33
+ results = results.filter((s) => s.created_at <= input.until);
34
+ }
35
+ }
36
+ else {
37
+ return {
38
+ content: [
39
+ {
40
+ type: 'text',
41
+ text: 'Please provide a "query" for full-text search or a "projectDir" to list snapshots.',
42
+ },
43
+ ],
44
+ };
45
+ }
46
+ if (!results || results.length === 0) {
47
+ return {
48
+ content: [
49
+ {
50
+ type: 'text',
51
+ text: `No snapshots found matching your search criteria.`,
52
+ },
53
+ ],
54
+ };
55
+ }
56
+ const lines = [`# Search Results (${results.length} found)`, ''];
57
+ for (const s of results) {
58
+ lines.push(`### Snapshot #${s.sequence_number} -- ${s.agent} -- ${s.created_at}`);
59
+ lines.push(s.summary);
60
+ const tags = JSON.parse(s.tags);
61
+ if (tags.length > 0) {
62
+ lines.push(`Tags: ${tags.map((t) => `\`${t}\``).join(', ')}`);
63
+ }
64
+ lines.push('');
65
+ }
66
+ return {
67
+ content: [{ type: 'text', text: lines.join('\n') }],
68
+ };
69
+ });
70
+ }
71
+ //# sourceMappingURL=search-snapshots.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search-snapshots.js","sourceRoot":"","sources":["../../src/tools/search-snapshots.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import type Database from 'better-sqlite3';
3
+ export declare function registerSyncTool(server: McpServer, db: Database.Database): void;
@@ -0,0 +1,102 @@
1
+ import { SyncInputSchema } from '../lib/types.js';
2
+ import { projectIdFromPath, normalizePath, displayNameFromPath, } from '../lib/project-id.js';
3
+ import { generateId } from '../lib/uuid.js';
4
+ import { upsertProject, getNextSequenceNumber, insertSnapshot, insertDecision, updateProjectSyncStats, } from '../db/queries.js';
5
+ import { validateProjectDir, suggestProjectRoot, formatWarnings, } from '../lib/validation.js';
6
+ export function registerSyncTool(server, db) {
7
+ server.registerTool('sync', {
8
+ title: 'Sync Context Snapshot',
9
+ description: 'Save a structured snapshot of current work state for cross-agent handoff. ' +
10
+ 'Call this when finishing a work session so another agent can resume.',
11
+ inputSchema: SyncInputSchema,
12
+ annotations: {
13
+ readOnlyHint: false,
14
+ destructiveHint: false,
15
+ idempotentHint: false,
16
+ openWorldHint: false,
17
+ },
18
+ }, async (input) => {
19
+ const projectId = projectIdFromPath(input.projectDir);
20
+ const normalized = normalizePath(input.projectDir);
21
+ const displayName = displayNameFromPath(input.projectDir);
22
+ const now = new Date().toISOString();
23
+ const validation = validateProjectDir(input.projectDir);
24
+ const suggestedRoot = suggestProjectRoot(input.projectDir);
25
+ const txn = db.transaction(() => {
26
+ upsertProject(db, projectId, normalized, displayName);
27
+ const seqNum = getNextSequenceNumber(db, projectId);
28
+ const snapshotId = generateId();
29
+ // Serialize all arrays once
30
+ const serialized = {
31
+ decisions: JSON.stringify(input.decisions ?? []),
32
+ files: JSON.stringify(input.filesModified ?? []),
33
+ blockers: JSON.stringify(input.blockers ?? []),
34
+ tags: JSON.stringify(input.tags ?? []),
35
+ tasksCompleted: JSON.stringify(input.tasksCompleted),
36
+ tasksRemaining: JSON.stringify(input.tasksRemaining),
37
+ };
38
+ insertSnapshot(db, {
39
+ id: snapshotId,
40
+ project_id: projectId,
41
+ sequence_number: seqNum,
42
+ agent: input.agent,
43
+ summary: input.summary,
44
+ tasks_completed: serialized.tasksCompleted,
45
+ tasks_remaining: serialized.tasksRemaining,
46
+ decisions: serialized.decisions,
47
+ files_modified: serialized.files,
48
+ blockers: serialized.blockers,
49
+ next_steps: input.nextSteps,
50
+ tags: serialized.tags,
51
+ created_at: now,
52
+ });
53
+ // Also insert decisions as first-class entities
54
+ if (input.decisions) {
55
+ for (const d of input.decisions) {
56
+ insertDecision(db, {
57
+ id: generateId(),
58
+ project_id: projectId,
59
+ snapshot_id: snapshotId,
60
+ agent: input.agent,
61
+ decision: d.decision,
62
+ reasoning: d.reasoning,
63
+ alternatives: JSON.stringify(d.alternatives ?? []),
64
+ category: null,
65
+ related_files: '[]',
66
+ created_at: now,
67
+ });
68
+ }
69
+ }
70
+ updateProjectSyncStats(db, projectId);
71
+ return { snapshotId, seqNum };
72
+ });
73
+ const { snapshotId, seqNum } = txn();
74
+ // Build response
75
+ const lines = [
76
+ `Snapshot #${seqNum} saved for project "${displayName}".`,
77
+ '',
78
+ `- ID: \`${snapshotId}\``,
79
+ `- Agent: ${input.agent}`,
80
+ `- Time: ${now}`,
81
+ `- Path: \`${normalized}\``,
82
+ ];
83
+ // Append path warnings if any
84
+ if (suggestedRoot) {
85
+ lines.push('', `**Warning:** Path looks like a subdirectory. Consider using the project root:`, `\`${suggestedRoot}\``, 'Subdirectory paths cause mismatches when other agents resume.');
86
+ }
87
+ const warnings = formatWarnings(validation);
88
+ if (warnings)
89
+ lines.push('', warnings);
90
+ lines.push('', `Another agent can call \`resume\` with \`projectDir: "${normalized}"\` to continue.`);
91
+ const message = lines.join('\n');
92
+ return {
93
+ content: [
94
+ {
95
+ type: 'text',
96
+ text: message,
97
+ },
98
+ ],
99
+ };
100
+ });
101
+ }
102
+ //# sourceMappingURL=sync.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sync.js","sourceRoot":"","sources":["../../src/tools/sync.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAClD,OAAO,EACL,iBAAiB,EACjB,aAAa,EACb,mBAAmB,GACpB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EACL,aAAa,EACb,qBAAqB,EACrB,cAAc,EACd,cAAc,EACd,sBAAsB,GACvB,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EACL,kBAAkB,EAClB,kBAAkB,EAClB,cAAc,GACf,MAAM,sBAAsB,CAAC;AAE9B,MAAM,UAAU,gBAAgB,CAAC,MAAiB,EAAE,EAAqB;IACvE,MAAM,CAAC,YAAY,CACjB,MAAM,EACN;QACE,KAAK,EAAE,uBAAuB;QAC9B,WAAW,EACT,4EAA4E;YAC5E,sEAAsE;QACxE,WAAW,EAAE,eAAe;QAC5B,WAAW,EAAE;YACX,YAAY,EAAE,KAAK;YACnB,eAAe,EAAE,KAAK;YACtB,cAAc,EAAE,KAAK;YACrB,aAAa,EAAE,KAAK;SACrB;KACF,EACD,KAAK,EAAE,KAAK,EAAE,EAAE;QACd,MAAM,SAAS,GAAG,iBAAiB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACtD,MAAM,UAAU,GAAG,aAAa,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACnD,MAAM,WAAW,GAAG,mBAAmB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAC1D,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAErC,MAAM,UAAU,GAAG,kBAAkB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACxD,MAAM,aAAa,GAAG,kBAAkB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAE3D,MAAM,GAAG,GAAG,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE;YAC9B,aAAa,CAAC,EAAE,EAAE,SAAS,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC;YAEtD,MAAM,MAAM,GAAG,qBAAqB,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;YACpD,MAAM,UAAU,GAAG,UAAU,EAAE,CAAC;YAEhC,4BAA4B;YAC5B,MAAM,UAAU,GAAG;gBACjB,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,SAAS,IAAI,EAAE,CAAC;gBAChD,KAAK,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,aAAa,IAAI,EAAE,CAAC;gBAChD,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,QAAQ,IAAI,EAAE,CAAC;gBAC9C,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,IAAI,EAAE,CAAC;gBACtC,cAAc,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,cAAc,CAAC;gBACpD,cAAc,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,cAAc,CAAC;aACrD,CAAC;YAEF,cAAc,CAAC,EAAE,EAAE;gBACjB,EAAE,EAAE,UAAU;gBACd,UAAU,EAAE,SAAS;gBACrB,eAAe,EAAE,MAAM;gBACvB,KAAK,EAAE,KAAK,CAAC,KAAK;gBAClB,OAAO,EAAE,KAAK,CAAC,OAAO;gBACtB,eAAe,EAAE,UAAU,CAAC,cAAc;gBAC1C,eAAe,EAAE,UAAU,CAAC,cAAc;gBAC1C,SAAS,EAAE,UAAU,CAAC,SAAS;gBAC/B,cAAc,EAAE,UAAU,CAAC,KAAK;gBAChC,QAAQ,EAAE,UAAU,CAAC,QAAQ;gBAC7B,UAAU,EAAE,KAAK,CAAC,SAAS;gBAC3B,IAAI,EAAE,UAAU,CAAC,IAAI;gBACrB,UAAU,EAAE,GAAG;aAChB,CAAC,CAAC;YAEH,gDAAgD;YAChD,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;gBACpB,KAAK,MAAM,CAAC,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;oBAChC,cAAc,CAAC,EAAE,EAAE;wBACjB,EAAE,EAAE,UAAU,EAAE;wBAChB,UAAU,EAAE,SAAS;wBACrB,WAAW,EAAE,UAAU;wBACvB,KAAK,EAAE,KAAK,CAAC,KAAK;wBAClB,QAAQ,EAAE,CAAC,CAAC,QAAQ;wBACpB,SAAS,EAAE,CAAC,CAAC,SAAS;wBACtB,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,YAAY,IAAI,EAAE,CAAC;wBAClD,QAAQ,EAAE,IAAI;wBACd,aAAa,EAAE,IAAI;wBACnB,UAAU,EAAE,GAAG;qBAChB,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;YAED,sBAAsB,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;YAEtC,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,CAAC;QAChC,CAAC,CAAC,CAAC;QAEH,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,GAAG,EAAE,CAAC;QAErC,iBAAiB;QACjB,MAAM,KAAK,GAAa;YACtB,aAAa,MAAM,uBAAuB,WAAW,IAAI;YACzD,EAAE;YACF,WAAW,UAAU,IAAI;YACzB,YAAY,KAAK,CAAC,KAAK,EAAE;YACzB,WAAW,GAAG,EAAE;YAChB,aAAa,UAAU,IAAI;SAC5B,CAAC;QAEF,8BAA8B;QAC9B,IAAI,aAAa,EAAE,CAAC;YAClB,KAAK,CAAC,IAAI,CACR,EAAE,EACF,+EAA+E,EAC/E,KAAK,aAAa,IAAI,EACtB,+DAA+D,CAChE,CAAC;QACJ,CAAC;QACD,MAAM,QAAQ,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QAC5C,IAAI,QAAQ;YAAE,KAAK,CAAC,IAAI,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC;QAEvC,KAAK,CAAC,IAAI,CACR,EAAE,EACF,yDAAyD,UAAU,kBAAkB,CACtF,CAAC;QAEF,MAAM,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAEjC,OAAO;YACL,OAAO,EAAE;gBACP;oBACE,IAAI,EAAE,MAAe;oBACrB,IAAI,EAAE,OAAO;iBACd;aACF;SACF,CAAC;IACJ,CAAC,CACF,CAAC;AACJ,CAAC"}