memgrid 0.5.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 (96) hide show
  1. package/CHANGELOG.md +80 -0
  2. package/LICENSE +21 -0
  3. package/README.md +251 -0
  4. package/dist/index.d.ts +2 -0
  5. package/dist/index.d.ts.map +1 -0
  6. package/dist/index.js +2 -0
  7. package/dist/index.js.map +1 -0
  8. package/dist/learn/index.d.ts +44 -0
  9. package/dist/learn/index.d.ts.map +1 -0
  10. package/dist/learn/index.js +234 -0
  11. package/dist/learn/index.js.map +1 -0
  12. package/dist/memgrid.d.ts +50 -0
  13. package/dist/memgrid.d.ts.map +1 -0
  14. package/dist/memgrid.js +175 -0
  15. package/dist/memgrid.js.map +1 -0
  16. package/dist/retrieve/index.d.ts +27 -0
  17. package/dist/retrieve/index.d.ts.map +1 -0
  18. package/dist/retrieve/index.js +209 -0
  19. package/dist/retrieve/index.js.map +1 -0
  20. package/dist/retrieve/semantic.d.ts +67 -0
  21. package/dist/retrieve/semantic.d.ts.map +1 -0
  22. package/dist/retrieve/semantic.js +240 -0
  23. package/dist/retrieve/semantic.js.map +1 -0
  24. package/dist/scanner/composite.d.ts +27 -0
  25. package/dist/scanner/composite.d.ts.map +1 -0
  26. package/dist/scanner/composite.js +58 -0
  27. package/dist/scanner/composite.js.map +1 -0
  28. package/dist/scanner/config.d.ts +15 -0
  29. package/dist/scanner/config.d.ts.map +1 -0
  30. package/dist/scanner/config.js +167 -0
  31. package/dist/scanner/config.js.map +1 -0
  32. package/dist/scanner/golang.d.ts +19 -0
  33. package/dist/scanner/golang.d.ts.map +1 -0
  34. package/dist/scanner/golang.js +190 -0
  35. package/dist/scanner/golang.js.map +1 -0
  36. package/dist/scanner/index.d.ts +11 -0
  37. package/dist/scanner/index.d.ts.map +1 -0
  38. package/dist/scanner/index.js +10 -0
  39. package/dist/scanner/index.js.map +1 -0
  40. package/dist/scanner/javascript.d.ts +20 -0
  41. package/dist/scanner/javascript.d.ts.map +1 -0
  42. package/dist/scanner/javascript.js +167 -0
  43. package/dist/scanner/javascript.js.map +1 -0
  44. package/dist/scanner/markdown.d.ts +17 -0
  45. package/dist/scanner/markdown.d.ts.map +1 -0
  46. package/dist/scanner/markdown.js +106 -0
  47. package/dist/scanner/markdown.js.map +1 -0
  48. package/dist/scanner/python.d.ts +19 -0
  49. package/dist/scanner/python.d.ts.map +1 -0
  50. package/dist/scanner/python.js +177 -0
  51. package/dist/scanner/python.js.map +1 -0
  52. package/dist/scanner/rules.d.ts +15 -0
  53. package/dist/scanner/rules.d.ts.map +1 -0
  54. package/dist/scanner/rules.js +86 -0
  55. package/dist/scanner/rules.js.map +1 -0
  56. package/dist/scanner/rust.d.ts +18 -0
  57. package/dist/scanner/rust.d.ts.map +1 -0
  58. package/dist/scanner/rust.js +178 -0
  59. package/dist/scanner/rust.js.map +1 -0
  60. package/dist/scanner/scanner.d.ts +33 -0
  61. package/dist/scanner/scanner.d.ts.map +1 -0
  62. package/dist/scanner/scanner.js +2 -0
  63. package/dist/scanner/scanner.js.map +1 -0
  64. package/dist/scanner/typescript.d.ts +28 -0
  65. package/dist/scanner/typescript.d.ts.map +1 -0
  66. package/dist/scanner/typescript.js +522 -0
  67. package/dist/scanner/typescript.js.map +1 -0
  68. package/dist/serve/cli.d.ts +3 -0
  69. package/dist/serve/cli.d.ts.map +1 -0
  70. package/dist/serve/cli.js +145 -0
  71. package/dist/serve/cli.js.map +1 -0
  72. package/dist/serve/mcp-server.d.ts +11 -0
  73. package/dist/serve/mcp-server.d.ts.map +1 -0
  74. package/dist/serve/mcp-server.js +276 -0
  75. package/dist/serve/mcp-server.js.map +1 -0
  76. package/dist/shared/constants.d.ts +2 -0
  77. package/dist/shared/constants.d.ts.map +1 -0
  78. package/dist/shared/constants.js +2 -0
  79. package/dist/shared/constants.js.map +1 -0
  80. package/dist/shared/types.d.ts +99 -0
  81. package/dist/shared/types.d.ts.map +1 -0
  82. package/dist/shared/types.js +3 -0
  83. package/dist/shared/types.js.map +1 -0
  84. package/dist/store/file-store.d.ts +62 -0
  85. package/dist/store/file-store.d.ts.map +1 -0
  86. package/dist/store/file-store.js +241 -0
  87. package/dist/store/file-store.js.map +1 -0
  88. package/dist/store/index.d.ts +3 -0
  89. package/dist/store/index.d.ts.map +1 -0
  90. package/dist/store/index.js +2 -0
  91. package/dist/store/index.js.map +1 -0
  92. package/dist/sync/index.d.ts +56 -0
  93. package/dist/sync/index.d.ts.map +1 -0
  94. package/dist/sync/index.js +580 -0
  95. package/dist/sync/index.js.map +1 -0
  96. package/package.json +51 -0
@@ -0,0 +1,177 @@
1
+ import * as path from 'path';
2
+ import * as fs from 'fs';
3
+ /**
4
+ * Python project scanner.
5
+ * Extracts functions, classes, methods, decorators via regex-based parsing.
6
+ */
7
+ export class PythonScanner {
8
+ name = 'python';
9
+ projectRoot;
10
+ constructor(_store, projectRoot) {
11
+ this.projectRoot = projectRoot;
12
+ }
13
+ detect(projectRoot) {
14
+ return (fs.existsSync(path.join(projectRoot, 'pyproject.toml')) ||
15
+ fs.existsSync(path.join(projectRoot, 'setup.py')) ||
16
+ fs.existsSync(path.join(projectRoot, 'requirements.txt')) ||
17
+ fs.existsSync(path.join(projectRoot, 'Pipfile')) ||
18
+ this.findSourceDirs(projectRoot).length > 0);
19
+ }
20
+ async scan(_options) {
21
+ const units = [];
22
+ const srcDirs = this.findSourceDirs(this.projectRoot);
23
+ for (const dir of srcDirs) {
24
+ await this.scanDir(dir, '', units);
25
+ }
26
+ // Also scan root-level .py files (excluding setup.py, __init__.py scaffold)
27
+ const rootDirs = ['src', 'lib', 'tests', 'app'];
28
+ for (const d of rootDirs) {
29
+ const abs = path.join(this.projectRoot, d);
30
+ if (fs.existsSync(abs) && !srcDirs.includes(abs)) {
31
+ await this.scanDir(abs, '', units);
32
+ }
33
+ }
34
+ return units;
35
+ }
36
+ async scanDir(dir, relPrefix, units) {
37
+ if (!fs.existsSync(dir))
38
+ return;
39
+ for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
40
+ const abs = path.join(dir, entry.name);
41
+ const rel = relPrefix ? path.join(relPrefix, entry.name) : entry.name;
42
+ if (entry.isDirectory()) {
43
+ if (entry.name.startsWith('.') || entry.name === '__pycache__' || entry.name === 'node_modules')
44
+ continue;
45
+ if (entry.name === 'tests' || entry.name === 'test')
46
+ continue;
47
+ await this.scanDir(abs, rel, units);
48
+ }
49
+ else if (entry.name.endsWith('.py') && !entry.name.startsWith('test_') && !entry.name.endsWith('_test.py')) {
50
+ try {
51
+ const code = fs.readFileSync(abs, 'utf-8');
52
+ this.parsePythonFile(code, rel, units);
53
+ }
54
+ catch {
55
+ // skip unreadable files
56
+ }
57
+ }
58
+ }
59
+ }
60
+ parsePythonFile(code, file, units) {
61
+ const lines = code.split('\n');
62
+ // Class pattern: class ClassName(...):
63
+ const classPattern = /^class\s+(\w+)\s*(?:\(([^)]*)\))?\s*:/;
64
+ // Function/method pattern: def func_name(params):
65
+ const funcPattern = /^\s{0,8}def\s+(\w+)\s*\(([^)]*)\)\s*(?:->\s*(\S+))?\s*:/;
66
+ let currentClass = '';
67
+ let decorators = [];
68
+ for (let i = 0; i < lines.length; i++) {
69
+ const line = lines[i];
70
+ const stripped = line.trim();
71
+ // Track decorators
72
+ if (stripped.startsWith('@')) {
73
+ decorators.push(stripped);
74
+ continue;
75
+ }
76
+ // Class
77
+ const classMatch = stripped.match(classPattern);
78
+ if (classMatch) {
79
+ currentClass = classMatch[1];
80
+ const doc = this.extractDoc(lines, i + 1);
81
+ units.push({
82
+ id: `method_py_${this.sanitizeId(currentClass)}`,
83
+ type: 'method',
84
+ summary: `${currentClass}${doc ? ` — ${doc}` : ''}`,
85
+ source: { file, lines: `${i + 1}` },
86
+ signatures: [currentClass],
87
+ content: {
88
+ description: `Python class ${currentClass}${decorators.length ? ` [${decorators.join(', ')}]` : ''}`,
89
+ },
90
+ associations: [],
91
+ meta: {
92
+ created: new Date().toISOString(),
93
+ updated: new Date().toISOString(),
94
+ confidence: 0.7,
95
+ usage_count: 0,
96
+ status: 'active',
97
+ },
98
+ });
99
+ decorators = [];
100
+ continue;
101
+ }
102
+ // Function/method
103
+ const funcMatch = stripped.match(funcPattern);
104
+ if (funcMatch) {
105
+ const funcName = funcMatch[1];
106
+ if (funcName.startsWith('_') && funcName !== '__init__') {
107
+ decorators = [];
108
+ continue;
109
+ } // skip private
110
+ const params = funcMatch[2];
111
+ const returnType = funcMatch[3] || '';
112
+ const doc = this.extractDoc(lines, i + 1);
113
+ const fullName = currentClass ? `${currentClass}.${funcName}` : funcName;
114
+ const unit = {
115
+ id: `method_py_${this.sanitizeId(fullName)}`,
116
+ type: 'method',
117
+ summary: doc ? `${fullName}() — ${doc}` : `${fullName}()`,
118
+ source: { file, lines: `${i + 1}` },
119
+ signatures: [fullName, funcName],
120
+ content: {
121
+ description: doc || `${fullName}()`,
122
+ inputs: params || 'none',
123
+ outputs: returnType || 'unknown',
124
+ },
125
+ associations: [],
126
+ meta: {
127
+ created: new Date().toISOString(),
128
+ updated: new Date().toISOString(),
129
+ confidence: currentClass ? 0.75 : 0.7,
130
+ usage_count: 0,
131
+ status: 'active',
132
+ },
133
+ };
134
+ // Add decorator info
135
+ if (decorators.length > 0) {
136
+ unit.content.style_notes = `Decorators: ${decorators.join(', ')}`;
137
+ }
138
+ units.push(unit);
139
+ decorators = [];
140
+ }
141
+ }
142
+ }
143
+ extractDoc(lines, startLine) {
144
+ // Look for """ or ''' docstring on next non-empty line
145
+ for (let i = startLine; i < Math.min(startLine + 3, lines.length); i++) {
146
+ const line = lines[i].trim();
147
+ const match = line.match(/^["']{3}([^"']*)["']{3}$/);
148
+ if (match)
149
+ return match[1].trim();
150
+ if (line.startsWith('"""') || line.startsWith("'''")) {
151
+ const inner = line.replace(/^["']{3}|["']{3}$/g, '').trim();
152
+ return inner || '...';
153
+ }
154
+ }
155
+ return '';
156
+ }
157
+ findSourceDirs(projectRoot) {
158
+ const dirs = [];
159
+ for (const name of ['src', 'app', 'lib', 'modules']) {
160
+ const abs = path.join(projectRoot, name);
161
+ if (fs.existsSync(abs) && fs.statSync(abs).isDirectory()) {
162
+ dirs.push(abs);
163
+ }
164
+ }
165
+ return dirs;
166
+ }
167
+ sanitizeId(text) {
168
+ return text
169
+ .replace(/\./g, '_')
170
+ .replace(/__/g, '_dunder_')
171
+ .replace(/[^a-zA-Z0-9_\-]/g, '_')
172
+ .replace(/_+/g, '_')
173
+ .toLowerCase()
174
+ .replace(/^_|_$/g, '');
175
+ }
176
+ }
177
+ //# sourceMappingURL=python.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"python.js","sourceRoot":"","sources":["../../src/scanner/python.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ import type { MemoryUnit, ScanOptions } from '../shared/types.js';
2
+ import type { Scanner } from './scanner.js';
3
+ /**
4
+ * Shared scanner for .claude/rules/*.md files.
5
+ * Extracted from TypeScriptScanner — applies to any language project.
6
+ */
7
+ export declare class RulesScanner implements Scanner {
8
+ readonly name = "rules";
9
+ private projectRoot;
10
+ constructor(projectRoot: string);
11
+ detect(projectRoot: string): boolean;
12
+ scan(_options: ScanOptions): Promise<MemoryUnit[]>;
13
+ scanFiles(_files: string[], _options: ScanOptions): Promise<MemoryUnit[]>;
14
+ }
15
+ //# sourceMappingURL=rules.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rules.d.ts","sourceRoot":"","sources":["../../src/scanner/rules.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AAClE,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAE5C;;;GAGG;AACH,qBAAa,YAAa,YAAW,OAAO;IAC1C,QAAQ,CAAC,IAAI,WAAW;IACxB,OAAO,CAAC,WAAW,CAAS;gBAEhB,WAAW,EAAE,MAAM;IAI/B,MAAM,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO;IAK9B,IAAI,CAAC,QAAQ,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC;IAqElD,SAAS,CAAC,MAAM,EAAE,MAAM,EAAE,EAAE,QAAQ,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC;CAIhF"}
@@ -0,0 +1,86 @@
1
+ import * as path from 'path';
2
+ import * as fs from 'fs';
3
+ /**
4
+ * Shared scanner for .claude/rules/*.md files.
5
+ * Extracted from TypeScriptScanner — applies to any language project.
6
+ */
7
+ export class RulesScanner {
8
+ name = 'rules';
9
+ projectRoot;
10
+ constructor(projectRoot) {
11
+ this.projectRoot = projectRoot;
12
+ }
13
+ detect(projectRoot) {
14
+ const dir = path.join(projectRoot, '.claude', 'rules');
15
+ return fs.existsSync(dir) && fs.readdirSync(dir).some((f) => f.endsWith('.md'));
16
+ }
17
+ async scan(_options) {
18
+ const units = [];
19
+ const rulesDir = path.join(this.projectRoot, '.claude', 'rules');
20
+ if (!fs.existsSync(rulesDir))
21
+ return units;
22
+ for (const file of fs.readdirSync(rulesDir)) {
23
+ if (!file.endsWith('.md'))
24
+ continue;
25
+ const filePath = path.join(rulesDir, file);
26
+ const content = fs.readFileSync(filePath, 'utf-8');
27
+ const relativePath = path.relative(this.projectRoot, filePath);
28
+ // Extract sections from markdown
29
+ const sections = content.split(/^## /m).filter(Boolean);
30
+ for (const section of sections) {
31
+ const title = section.split('\n')[0].trim();
32
+ const body = section.split('\n').slice(1).join('\n').trim();
33
+ if (!title || body.length < 50)
34
+ continue;
35
+ const safeTitle = title.replace(/[^a-zA-Z0-9_\-]/g, '_').replace(/_+/g, '_').slice(0, 50).toLowerCase();
36
+ if (!safeTitle || safeTitle === '_')
37
+ continue;
38
+ const safeFile = file.replace('.md', '').replace(/[^a-zA-Z0-9_\-]/g, '_').replace(/_+/g, '_').slice(0, 30).toLowerCase();
39
+ units.push({
40
+ id: `rule_${safeFile}_${safeTitle}`,
41
+ type: 'pattern',
42
+ summary: `${file.replace('.md', '')}: ${title}`,
43
+ source: { file: relativePath },
44
+ signatures: [title, file.replace('.md', '').replace(/-/g, ' ')],
45
+ content: { description: body.slice(0, 500) },
46
+ associations: [],
47
+ meta: {
48
+ created: new Date().toISOString(),
49
+ updated: new Date().toISOString(),
50
+ confidence: 0.9,
51
+ usage_count: 0,
52
+ status: 'active',
53
+ },
54
+ });
55
+ }
56
+ // Also create a rule_trigger unit
57
+ const safeFile = file.replace('.md', '').replace(/[^a-zA-Z0-9_\-]/g, '_').replace(/_+/g, '_').slice(0, 30).toLowerCase();
58
+ units.push({
59
+ id: `trigger_rule_${safeFile}`,
60
+ type: 'rule_trigger',
61
+ summary: `When working on ${file.replace('.md', '').replace(/-/g, ' ')} → load ${file}`,
62
+ source: { file: relativePath },
63
+ signatures: [file.replace('.md', '').replace(/-/g, ' ')],
64
+ content: {
65
+ description: `Load ${relativePath} when working on relevant code`,
66
+ trigger: `Working on ${file.replace('.md', '').replace(/-/g, ' ')} related code`,
67
+ action: `Load ${relativePath}`,
68
+ },
69
+ associations: [],
70
+ meta: {
71
+ created: new Date().toISOString(),
72
+ updated: new Date().toISOString(),
73
+ confidence: 0.8,
74
+ usage_count: 0,
75
+ status: 'active',
76
+ },
77
+ });
78
+ }
79
+ return units;
80
+ }
81
+ async scanFiles(_files, _options) {
82
+ // Rules are small — just re-scan all
83
+ return this.scan(_options);
84
+ }
85
+ }
86
+ //# sourceMappingURL=rules.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rules.js","sourceRoot":"","sources":["../../src/scanner/rules.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ import type { MemoryUnit, ScanOptions } from '../shared/types.js';
2
+ import type { Scanner } from './scanner.js';
3
+ /**
4
+ * Rust project scanner.
5
+ * Regex-based: extracts fn, struct, impl, trait, enum declarations.
6
+ */
7
+ export declare class RustScanner implements Scanner {
8
+ readonly name = "rust";
9
+ private projectRoot;
10
+ constructor(_store: any, projectRoot: string);
11
+ detect(projectRoot: string): boolean;
12
+ scan(_options: ScanOptions): Promise<MemoryUnit[]>;
13
+ private scanDir;
14
+ private parseRustFile;
15
+ private findSourceDirs;
16
+ private sanitizeId;
17
+ }
18
+ //# sourceMappingURL=rust.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rust.d.ts","sourceRoot":"","sources":["../../src/scanner/rust.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AAClE,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAE5C;;;GAGG;AACH,qBAAa,WAAY,YAAW,OAAO;IACzC,QAAQ,CAAC,IAAI,UAAU;IACvB,OAAO,CAAC,WAAW,CAAS;gBAEhB,MAAM,EAAE,GAAG,EAAE,WAAW,EAAE,MAAM;IAI5C,MAAM,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO;IAI9B,IAAI,CAAC,QAAQ,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC;YAW1C,OAAO;IAmBrB,OAAO,CAAC,aAAa;IA2HrB,OAAO,CAAC,cAAc;IAWtB,OAAO,CAAC,UAAU;CAUnB"}
@@ -0,0 +1,178 @@
1
+ import * as path from 'path';
2
+ import * as fs from 'fs';
3
+ /**
4
+ * Rust project scanner.
5
+ * Regex-based: extracts fn, struct, impl, trait, enum declarations.
6
+ */
7
+ export class RustScanner {
8
+ name = 'rust';
9
+ projectRoot;
10
+ constructor(_store, projectRoot) {
11
+ this.projectRoot = projectRoot;
12
+ }
13
+ detect(projectRoot) {
14
+ return fs.existsSync(path.join(projectRoot, 'Cargo.toml'));
15
+ }
16
+ async scan(_options) {
17
+ const units = [];
18
+ const srcDirs = this.findSourceDirs(this.projectRoot);
19
+ for (const dir of srcDirs) {
20
+ await this.scanDir(dir, '', units);
21
+ }
22
+ return units;
23
+ }
24
+ async scanDir(dir, relPrefix, units) {
25
+ if (!fs.existsSync(dir))
26
+ return;
27
+ for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
28
+ const abs = path.join(dir, entry.name);
29
+ const rel = relPrefix ? path.join(relPrefix, entry.name) : entry.name;
30
+ if (entry.isDirectory()) {
31
+ if (entry.name.startsWith('.') || entry.name === 'target' || entry.name === 'node_modules')
32
+ continue;
33
+ await this.scanDir(abs, rel, units);
34
+ }
35
+ else if (entry.name.endsWith('.rs')) {
36
+ try {
37
+ const code = fs.readFileSync(abs, 'utf-8');
38
+ this.parseRustFile(code, rel, units);
39
+ }
40
+ catch { /* skip */ }
41
+ }
42
+ }
43
+ }
44
+ parseRustFile(code, file, units) {
45
+ const lines = code.split('\n');
46
+ let currentImpl = '';
47
+ for (let i = 0; i < lines.length; i++) {
48
+ const line = lines[i].trim();
49
+ // fn name(params) -> ReturnType {
50
+ let match = line.match(/^(?:pub(?:\s*\(\s*crate\s*\))?\s+)?(?:async\s+)?fn\s+(\w+)\s*(?:<[^>]*>)?\s*\(([^)]*)\)\s*(?:->\s*([^{]+?))?\s*\{/);
51
+ if (match) {
52
+ const fnName = match[1];
53
+ if (fnName === 'main' || fnName.startsWith('test_'))
54
+ continue;
55
+ const params = match[2];
56
+ const retType = match[3]?.trim() || '';
57
+ const fullName = currentImpl ? `${currentImpl}::${fnName}` : fnName;
58
+ units.push({
59
+ id: `method_rs_${this.sanitizeId(fullName)}`,
60
+ type: 'method',
61
+ summary: retType ? `${fullName}() → ${retType}` : `${fullName}()`,
62
+ source: { file, lines: `${i + 1}` },
63
+ signatures: [fullName, fnName],
64
+ content: {
65
+ description: `Rust function ${fullName}()`,
66
+ inputs: params || 'none',
67
+ outputs: retType || '()',
68
+ },
69
+ associations: [],
70
+ meta: {
71
+ created: new Date().toISOString(),
72
+ updated: new Date().toISOString(),
73
+ confidence: 0.7,
74
+ usage_count: 0,
75
+ status: 'active',
76
+ },
77
+ });
78
+ continue;
79
+ }
80
+ // pub struct Name {
81
+ match = line.match(/^(?:pub\s+)?struct\s+(\w+)\s*(?:<[^>]*>)?\s*\{/);
82
+ if (match) {
83
+ const structName = match[1];
84
+ units.push({
85
+ id: `method_rs_struct_${this.sanitizeId(structName)}`,
86
+ type: 'method',
87
+ summary: `${structName} (struct)`,
88
+ source: { file, lines: `${i + 1}` },
89
+ signatures: [structName],
90
+ content: { description: `Rust struct ${structName}` },
91
+ associations: [],
92
+ meta: {
93
+ created: new Date().toISOString(),
94
+ updated: new Date().toISOString(),
95
+ confidence: 0.65,
96
+ usage_count: 0,
97
+ status: 'active',
98
+ },
99
+ });
100
+ continue;
101
+ }
102
+ // pub enum Name {
103
+ match = line.match(/^(?:pub\s+)?enum\s+(\w+)\s*\{/);
104
+ if (match) {
105
+ units.push({
106
+ id: `method_rs_enum_${this.sanitizeId(match[1])}`,
107
+ type: 'pattern',
108
+ summary: `${match[1]} (enum)`,
109
+ source: { file, lines: `${i + 1}` },
110
+ signatures: [match[1]],
111
+ content: { description: `Rust enum ${match[1]}` },
112
+ associations: [],
113
+ meta: {
114
+ created: new Date().toISOString(),
115
+ updated: new Date().toISOString(),
116
+ confidence: 0.65,
117
+ usage_count: 0,
118
+ status: 'active',
119
+ },
120
+ });
121
+ continue;
122
+ }
123
+ // pub trait Name {
124
+ match = line.match(/^(?:pub\s+)?trait\s+(\w+)\s*\{/);
125
+ if (match) {
126
+ units.push({
127
+ id: `method_rs_trait_${this.sanitizeId(match[1])}`,
128
+ type: 'pattern',
129
+ summary: `${match[1]} (trait)`,
130
+ source: { file, lines: `${i + 1}` },
131
+ signatures: [match[1]],
132
+ content: { description: `Rust trait ${match[1]}` },
133
+ associations: [],
134
+ meta: {
135
+ created: new Date().toISOString(),
136
+ updated: new Date().toISOString(),
137
+ confidence: 0.7,
138
+ usage_count: 0,
139
+ status: 'active',
140
+ },
141
+ });
142
+ continue;
143
+ }
144
+ // impl Type { or impl Trait for Type {
145
+ match = line.match(/^impl(?:\s+(\w+)\s+for)?\s+(\w+)\s*\{/);
146
+ if (match) {
147
+ currentImpl = match[2];
148
+ continue;
149
+ }
150
+ // Close impl block
151
+ if (line === '}' && currentImpl) {
152
+ currentImpl = '';
153
+ continue;
154
+ }
155
+ }
156
+ }
157
+ findSourceDirs(projectRoot) {
158
+ const dirs = [];
159
+ for (const name of ['src', 'crates']) {
160
+ const abs = path.join(projectRoot, name);
161
+ if (fs.existsSync(abs) && fs.statSync(abs).isDirectory()) {
162
+ dirs.push(abs);
163
+ }
164
+ }
165
+ return dirs;
166
+ }
167
+ sanitizeId(text) {
168
+ return text
169
+ .replace(/::/g, '_')
170
+ .replace(/\./g, '_')
171
+ .replace(/<[^>]*>/g, '')
172
+ .replace(/[^a-zA-Z0-9_\-]/g, '_')
173
+ .replace(/_+/g, '_')
174
+ .toLowerCase()
175
+ .replace(/^_|_$/g, '');
176
+ }
177
+ }
178
+ //# sourceMappingURL=rust.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rust.js","sourceRoot":"","sources":["../../src/scanner/rust.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,33 @@
1
+ import type { MemoryUnit, ScanOptions } from '../shared/types.js';
2
+ /**
3
+ * Scanner interface — scan a project directory and produce memory units.
4
+ *
5
+ * MemGrid is language-agnostic. Each language/framework gets its own
6
+ * Scanner implementation. Scanners are composed via CompositeScanner.
7
+ *
8
+ * Implementations:
9
+ * - TypeScriptScanner — ts-morph AST extraction
10
+ * - (future) PythonScanner, GoScanner, MarkdownScanner, ComponentScanner...
11
+ */
12
+ export interface Scanner {
13
+ /** Human-readable name for CLI output */
14
+ readonly name: string;
15
+ /**
16
+ * Scan the configured project and return memory units.
17
+ * Called once during init, or per-changed-file during sync.
18
+ */
19
+ scan(options: ScanOptions): Promise<MemoryUnit[]>;
20
+ /**
21
+ * Scan only specific files (used by incremental sync).
22
+ * Default implementation: returns empty array (scanner doesn't support partial).
23
+ * Override for scanners that can parse individual files.
24
+ */
25
+ scanFiles?(files: string[], options: ScanOptions): Promise<MemoryUnit[]>;
26
+ /**
27
+ * Detect whether this scanner applies to the given project.
28
+ * e.g. TypeScriptScanner checks for tsconfig.json or .ts files.
29
+ * Used by auto-detection in CLI.
30
+ */
31
+ detect(projectRoot: string): boolean;
32
+ }
33
+ //# sourceMappingURL=scanner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scanner.d.ts","sourceRoot":"","sources":["../../src/scanner/scanner.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AAElE;;;;;;;;;GASG;AACH,MAAM,WAAW,OAAO;IACtB,yCAAyC;IACzC,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IAEtB;;;OAGG;IACH,IAAI,CAAC,OAAO,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC,CAAC;IAElD;;;;OAIG;IACH,SAAS,CAAC,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE,OAAO,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC,CAAC;IAEzE;;;;OAIG;IACH,MAAM,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC;CACtC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=scanner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scanner.js","sourceRoot":"","sources":["../../src/scanner/scanner.ts"],"names":[],"mappings":""}
@@ -0,0 +1,28 @@
1
+ import type { MemoryUnit, ScanOptions } from '../shared/types.js';
2
+ import type { FileStore } from '../store/file-store.js';
3
+ import type { Scanner } from './scanner.js';
4
+ export declare class TypeScriptScanner implements Scanner {
5
+ readonly name = "typescript";
6
+ private store;
7
+ private projectRoot;
8
+ constructor(store: FileStore, projectRoot: string);
9
+ detect(projectRoot: string): boolean;
10
+ scan(options: ScanOptions): Promise<MemoryUnit[]>;
11
+ private scanTypeScript;
12
+ private scanRules;
13
+ private scanExamples;
14
+ private scanConfig;
15
+ private sanitizeId;
16
+ private buildAssociations;
17
+ private updateGrid;
18
+ private extractJsDoc;
19
+ private extractDependencies;
20
+ /**
21
+ * Extract function/method names called in a code snippet.
22
+ * Matches patterns like: foo(), await foo(), this.foo(), obj.foo()
23
+ */
24
+ private extractCalledNames;
25
+ private isCommonKeyword;
26
+ private findSourceDirs;
27
+ }
28
+ //# sourceMappingURL=typescript.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"typescript.d.ts","sourceRoot":"","sources":["../../src/scanner/typescript.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,UAAU,EAA2B,WAAW,EAAE,MAAM,oBAAoB,CAAC;AAC3F,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,wBAAwB,CAAC;AACxD,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAE5C,qBAAa,iBAAkB,YAAW,OAAO;IAC/C,QAAQ,CAAC,IAAI,gBAAgB;IAC7B,OAAO,CAAC,KAAK,CAAY;IACzB,OAAO,CAAC,WAAW,CAAS;gBAEhB,KAAK,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM;IAKjD,MAAM,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO;IAO9B,IAAI,CAAC,OAAO,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC;YAkCzC,cAAc;YA8Hd,SAAS;YA2ET,YAAY;YA8CZ,UAAU;IAgExB,OAAO,CAAC,UAAU;IAIlB,OAAO,CAAC,iBAAiB;IAkGzB,OAAO,CAAC,UAAU;IA4BlB,OAAO,CAAC,YAAY;IASpB,OAAO,CAAC,mBAAmB;IAc3B;;;OAGG;IACH,OAAO,CAAC,kBAAkB;IAoC1B,OAAO,CAAC,eAAe;IAYvB,OAAO,CAAC,cAAc;CAkBvB"}