wize-dev-kit 0.3.0 → 0.3.1

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 (31) hide show
  1. package/AGENTS.md +21 -0
  2. package/ARCH.md +40 -4
  3. package/CHANGELOG.md +26 -0
  4. package/README.md +3 -1
  5. package/package.json +3 -2
  6. package/src/method-skills/1-analysis/wize-document-project/documentation-requirements.csv +12 -0
  7. package/src/method-skills/1-analysis/wize-document-project/templates/api-contracts-template.md +38 -0
  8. package/src/method-skills/1-analysis/wize-document-project/templates/architecture-template.md +54 -0
  9. package/src/method-skills/1-analysis/wize-document-project/templates/component-inventory-template.md +40 -0
  10. package/src/method-skills/1-analysis/wize-document-project/templates/contribution-guide-template.md +34 -0
  11. package/src/method-skills/1-analysis/wize-document-project/templates/data-models-template.md +36 -0
  12. package/src/method-skills/1-analysis/wize-document-project/templates/deep-dive-template.md +312 -0
  13. package/src/method-skills/1-analysis/wize-document-project/templates/deployment-guide-template.md +42 -0
  14. package/src/method-skills/1-analysis/wize-document-project/templates/development-guide-template.md +61 -0
  15. package/src/method-skills/1-analysis/wize-document-project/templates/index-template.md +185 -0
  16. package/src/method-skills/1-analysis/wize-document-project/templates/project-overview-template.md +110 -0
  17. package/src/method-skills/1-analysis/wize-document-project/templates/project-scan-report-schema.json +159 -0
  18. package/src/method-skills/1-analysis/wize-document-project/templates/source-tree-template.md +142 -0
  19. package/src/method-skills/1-analysis/wize-document-project/workflow.md +23 -0
  20. package/src/tea-skills/wize-tea-risk/workflow.md +11 -0
  21. package/tools/installer/commands/doctor.js +27 -1
  22. package/tools/installer/commands/document-project.js +93 -0
  23. package/tools/installer/document-project/batch-scanner.js +93 -0
  24. package/tools/installer/document-project/classify.js +170 -0
  25. package/tools/installer/document-project/modes/deep-dive.js +196 -0
  26. package/tools/installer/document-project/modes/full-rescan.js +15 -0
  27. package/tools/installer/document-project/modes/initial-scan.js +100 -0
  28. package/tools/installer/document-project/modes/quick.js +211 -0
  29. package/tools/installer/document-project/render-index.js +101 -0
  30. package/tools/installer/document-project/state.js +110 -0
  31. package/tools/installer/wize-cli.js +46 -30
@@ -0,0 +1,170 @@
1
+ // Project-type classifier for `wize-dev-kit document-project`.
2
+ //
3
+ // Detects the project type(s) of a repo from file patterns, then decides
4
+ // whether the repo is a monolith, multi-part, or monorepo.
5
+ // No source files are read — this is pattern-only detection.
6
+
7
+ 'use strict';
8
+
9
+ const fs = require('node:fs');
10
+ const path = require('node:path');
11
+
12
+ const MULTI_PART_FOLDERS = ['client', 'server', 'api', 'web', 'app', 'frontend', 'backend', 'mobile', 'desktop'];
13
+
14
+ function toBool(v) {
15
+ return String(v).toLowerCase() === 'true';
16
+ }
17
+
18
+ function splitPatterns(str) {
19
+ if (!str || str === 'N/A') return [];
20
+ return str.split(';').map(s => s.trim()).filter(Boolean);
21
+ }
22
+
23
+ function loadRequirements(csvPath) {
24
+ const content = fs.readFileSync(csvPath, 'utf-8');
25
+ const lines = content.split(/\r?\n/).filter(l => l.trim());
26
+ const header = lines[0].split(',').map(h => h.trim());
27
+ const rows = [];
28
+ for (let i = 1; i < lines.length; i++) {
29
+ const values = lines[i].split(',').map(v => v.trim());
30
+ const row = {};
31
+ for (let j = 0; j < header.length; j++) {
32
+ const key = header[j];
33
+ const raw = values[j];
34
+ if (key.startsWith('requires_')) {
35
+ row[key] = toBool(raw);
36
+ } else {
37
+ row[key] = raw || '';
38
+ }
39
+ }
40
+ rows.push(row);
41
+ }
42
+ return rows;
43
+ }
44
+
45
+ function globMatch(rootDir, pattern) {
46
+ // Very small pattern matcher: supports * wildcard and trailing / for dirs.
47
+ const isDirPattern = pattern.endsWith('/');
48
+ const base = isDirPattern ? pattern.slice(0, -1) : pattern;
49
+ const parts = base.split('/');
50
+ return walkMatch(rootDir, parts, 0, isDirPattern);
51
+ }
52
+
53
+ function walkMatch(dir, parts, depth, isDirPattern) {
54
+ if (depth >= parts.length) return true;
55
+ let entries;
56
+ try { entries = fs.readdirSync(dir, { withFileTypes: true }); } catch { return false; }
57
+ const part = parts[depth];
58
+ const isLast = depth === parts.length - 1;
59
+ const regex = globToRegex(part);
60
+ for (const e of entries) {
61
+ if (!regex.test(e.name)) continue;
62
+ if (isLast) {
63
+ if (isDirPattern && e.isDirectory()) return true;
64
+ if (!isDirPattern && e.isFile()) return true;
65
+ if (!isDirPattern && e.isDirectory()) {
66
+ // pattern like vite.config.* can match a directory named vite.config.ts? no.
67
+ // But pattern may be a prefix; treat directory match as false for files.
68
+ return false;
69
+ }
70
+ } else if (e.isDirectory()) {
71
+ if (walkMatch(path.join(dir, e.name), parts, depth + 1, isDirPattern)) return true;
72
+ }
73
+ }
74
+ return false;
75
+ }
76
+
77
+ function globToRegex(pattern) {
78
+ const escaped = pattern
79
+ .replace(/\./g, '\\.')
80
+ .replace(/\*/g, '.*')
81
+ .replace(/\?/g, '.');
82
+ return new RegExp(`^${escaped}$`);
83
+ }
84
+
85
+ function scorePart(rootDir, row, { prefer = [] } = {}) {
86
+ let score = 0;
87
+ const keyPatterns = splitPatterns(row.key_file_patterns);
88
+ const dirPatterns = splitPatterns(row.critical_directories);
89
+ for (const p of keyPatterns) {
90
+ if (globMatch(rootDir, p)) score += 2;
91
+ }
92
+ for (const p of dirPatterns) {
93
+ if (globMatch(rootDir, p)) score += 1;
94
+ }
95
+ if (prefer.includes(row.project_type_id)) score += 0.5;
96
+ return score;
97
+ }
98
+
99
+ function detectParts(rootDir) {
100
+ const parts = [];
101
+ const candidates = [];
102
+ let entries;
103
+ try { entries = fs.readdirSync(rootDir, { withFileTypes: true }); } catch { return parts; }
104
+
105
+ for (const e of entries) {
106
+ if (!e.isDirectory()) continue;
107
+ if (!MULTI_PART_FOLDERS.includes(e.name.toLowerCase())) continue;
108
+ candidates.push(e.name);
109
+ }
110
+
111
+ if (candidates.length < 2) return parts;
112
+
113
+ const rows = loadRequirements(requirementsPath());
114
+ for (const partId of candidates) {
115
+ const partRoot = path.join(rootDir, partId);
116
+ const prefer = partId.toLowerCase() === 'server' || partId.toLowerCase() === 'backend' ? ['backend'] : [];
117
+ const scored = rows.map(r => ({ ...r, score: scorePart(partRoot, r, { prefer }) }));
118
+ scored.sort((a, b) => b.score - a.score);
119
+ const best = scored[0];
120
+ if (best && best.score > 0) {
121
+ parts.push({ part_id: partId, project_type_id: best.project_type_id, root_path: partRoot });
122
+ }
123
+ }
124
+ return parts;
125
+ }
126
+
127
+ function isMultiPart(rootDir) {
128
+ return detectParts(rootDir).length >= 2;
129
+ }
130
+
131
+ function classifyProject(rootDir, options = {}) {
132
+ const csvPath = options.csvPath || requirementsPath();
133
+ const rows = loadRequirements(csvPath);
134
+ const parts = detectParts(rootDir);
135
+
136
+ if (parts.length >= 2) {
137
+ return { projectTypes: [...new Set(parts.map(p => p.project_type_id))], parts };
138
+ }
139
+
140
+ const scored = rows.map(r => ({ ...r, score: scorePart(rootDir, r, { prefer: options.prefer || [] }) }));
141
+ scored.sort((a, b) => b.score - a.score);
142
+ const best = scored[0];
143
+ const tied = scored.filter(s => s.score === best.score).map(s => s.project_type_id);
144
+
145
+ if (!best || best.score === 0) {
146
+ return { projectTypes: [], parts: [] };
147
+ }
148
+
149
+ // Cli + library often appear together; prefer both when tied at the top.
150
+ const projectTypes = tied.includes('cli') && tied.includes('library')
151
+ ? ['cli', 'library']
152
+ : [best.project_type_id];
153
+
154
+ return {
155
+ projectTypes,
156
+ parts: [{ part_id: 'root', project_type_id: best.project_type_id, root_path: rootDir }]
157
+ };
158
+ }
159
+
160
+ function requirementsPath() {
161
+ // When running from the installed package, the CSV is next to this file under
162
+ // node_modules/wize-dev-kit/... When running from the source repo, resolve from
163
+ // the repo root via the known relative path.
164
+ const installed = path.resolve(__dirname, '..', '..', '..', 'src', 'method-skills', '1-analysis', 'wize-document-project', 'documentation-requirements.csv');
165
+ if (fs.existsSync(installed)) return installed;
166
+ // Fallback for source repo layout.
167
+ return path.resolve(__dirname, '..', '..', '..', '..', 'src', 'method-skills', '1-analysis', 'wize-document-project', 'documentation-requirements.csv');
168
+ }
169
+
170
+ module.exports = { classifyProject, loadRequirements, isMultiPart, scorePart, requirementsPath, globMatch };
@@ -0,0 +1,196 @@
1
+ // Deep-dive mode for `wize-dev-kit document-project`.
2
+ //
3
+ // Exhaustive documentation of a specific folder/file/feature.
4
+
5
+ 'use strict';
6
+
7
+ const fs = require('node:fs');
8
+ const path = require('node:path');
9
+ const { batchScanner, scanFolder } = require('../batch-scanner.js');
10
+ const { updateState } = require('../state.js');
11
+ const { renderIndex } = require('../render-index.js');
12
+
13
+ function sanitize(name) {
14
+ return name.replace(/[^a-z0-9_-]/gi, '-').toLowerCase();
15
+ }
16
+
17
+ function walkAllFiles(rootDir, ignore = ['node_modules', '.git', 'dist', 'build', 'coverage']) {
18
+ const files = [];
19
+ const stack = [rootDir];
20
+ while (stack.length) {
21
+ const dir = stack.pop();
22
+ let entries;
23
+ try { entries = fs.readdirSync(dir, { withFileTypes: true }); } catch { continue; }
24
+ for (const e of entries) {
25
+ const full = path.join(dir, e.name);
26
+ if (e.isDirectory()) {
27
+ if (!ignore.includes(e.name)) stack.push(full);
28
+ continue;
29
+ }
30
+ if (e.isFile() && /\.(js|ts|jsx|tsx|mjs|cjs)$/.test(e.name)) files.push(full);
31
+ }
32
+ }
33
+ return files;
34
+ }
35
+
36
+ function matchesTarget(filePath, projectRoot, targetType, targetName) {
37
+ const rel = path.relative(projectRoot, filePath).toLowerCase();
38
+ const name = targetName.toLowerCase();
39
+ if (targetType === 'feature') {
40
+ return rel.includes(name);
41
+ }
42
+ if (targetType === 'api_group') {
43
+ const inApiFolder = /\/(routes|api|controllers|endpoints)\//.test(rel);
44
+ return inApiFolder && rel.includes(name);
45
+ }
46
+ if (targetType === 'component_group') {
47
+ const inComponentFolder = /\/(components|ui|widgets)\//.test(rel);
48
+ return inComponentFolder && rel.includes(name);
49
+ }
50
+ return false;
51
+ }
52
+
53
+ function resolveTarget(projectRoot, target) {
54
+ if (!target) return { targetType: null, targetName: null, targetPath: null, files: [] };
55
+
56
+ // Explicit typed target: "feature:auth", "api:users", "component:Button"
57
+ const typedMatch = target.match(/^([a-z_]+):(.+)$/i);
58
+ if (typedMatch) {
59
+ const targetType = typedMatch[1].toLowerCase();
60
+ const targetName = typedMatch[2];
61
+ if (['feature', 'api_group', 'component_group'].includes(targetType)) {
62
+ const all = walkAllFiles(projectRoot);
63
+ const files = all.filter(f => matchesTarget(f, projectRoot, targetType, targetName));
64
+ return { targetType, targetName, targetPath: target, files };
65
+ }
66
+ return { targetType, targetName, targetPath: target, files: [], error: `unknown target type: ${targetType}` };
67
+ }
68
+
69
+ // Path-based target (folder or file)
70
+ const targetPath = path.resolve(projectRoot, target);
71
+ if (fs.existsSync(targetPath)) {
72
+ const stats = fs.statSync(targetPath);
73
+ const targetType = stats.isFile() ? 'file' : 'folder';
74
+ const files = stats.isFile() ? [targetPath] : scanFolder(targetPath, { ignore: ['node_modules', '.git', 'dist', 'build', 'coverage'] }).files.map(f => f.path);
75
+ return { targetType, targetName: path.basename(targetPath), targetPath, files };
76
+ }
77
+
78
+ // Bare name treated as feature search
79
+ const all = walkAllFiles(projectRoot);
80
+ const files = all.filter(f => matchesTarget(f, projectRoot, 'feature', target));
81
+ return { targetType: 'feature', targetName: target, targetPath: target, files };
82
+ }
83
+
84
+ function listFilesInScope(rootDir, targetPath) {
85
+ const stats = fs.statSync(targetPath);
86
+ if (stats.isFile()) return [targetPath];
87
+ const scan = scanFolder(targetPath, { ignore: ['node_modules', '.git', 'dist', 'build', 'coverage'] });
88
+ return scan.files.map(f => f.path);
89
+ }
90
+
91
+ function readFileSummary(filePath) {
92
+ try {
93
+ const content = fs.readFileSync(filePath, 'utf-8');
94
+ const lines = content.split('\n');
95
+ const exports = [];
96
+ const todos = [];
97
+ for (let i = 0; i < lines.length; i++) {
98
+ const line = lines[i];
99
+ if (/\b(module\.exports|exports\.|export\s+(default\s+)?|function\s+|class\s+)/.test(line)) {
100
+ exports.push(line.trim().slice(0, 80));
101
+ }
102
+ if (/\b(TODO|FIXME|HACK|XXX)\b/i.test(line)) {
103
+ todos.push(`${i + 1}: ${line.trim().slice(0, 80)}`);
104
+ }
105
+ }
106
+ return {
107
+ path: filePath,
108
+ loc: lines.length,
109
+ exports: exports.slice(0, 10),
110
+ todos: todos.slice(0, 10)
111
+ };
112
+ } catch {
113
+ return { path: filePath, loc: 0, exports: [], todos: [] };
114
+ }
115
+ }
116
+
117
+ function runDeepDive(projectRoot, { target, log = console.log } = {}) {
118
+ if (!target) {
119
+ return { ok: false, error: 'missing --target for deep_dive', exitCode: 1 };
120
+ }
121
+
122
+ const resolved = resolveTarget(projectRoot, target);
123
+ if (resolved.error) {
124
+ return { ok: false, error: resolved.error, exitCode: 1 };
125
+ }
126
+ if (resolved.targetType === 'file' || resolved.targetType === 'folder') {
127
+ if (!fs.existsSync(resolved.targetPath)) {
128
+ return { ok: false, error: `target not found: ${target}`, exitCode: 1 };
129
+ }
130
+ }
131
+ if (resolved.files.length === 0) {
132
+ return { ok: false, error: `target not found or no files matched: ${target}`, exitCode: 1 };
133
+ }
134
+
135
+ const files = resolved.files;
136
+ const summaries = files.map(readFileSummary);
137
+ const targetName = resolved.targetName;
138
+ const outputName = `deep-dive-${sanitize(targetName)}.md`;
139
+ const outputPath = path.join(projectRoot, '.wize', 'knowledge', 'document-project', outputName);
140
+
141
+ const lines = [
142
+ '---',
143
+ 'status: baseline',
144
+ 'owner: Pepper Potts + Tony Stark',
145
+ `created: ${new Date().toISOString().slice(0, 10)}`,
146
+ `last_refreshed: ${new Date().toISOString().slice(0, 10)}`,
147
+ '---',
148
+ '',
149
+ `# Deep Dive — ${targetName}`,
150
+ '',
151
+ `**Type:** ${resolved.targetType}`,
152
+ `**Scope:** ${target}`,
153
+ `**Files analyzed:** ${files.length}`,
154
+ '',
155
+ '## File Inventory',
156
+ ''
157
+ ];
158
+
159
+ for (const s of summaries) {
160
+ lines.push(`### ${path.relative(projectRoot, s.path)}`);
161
+ lines.push(`- **LOC:** ${s.loc}`);
162
+ if (s.exports.length) {
163
+ lines.push('- **Exports/signatures:**');
164
+ for (const e of s.exports) lines.push(` - \`${e}\``);
165
+ }
166
+ if (s.todos.length) {
167
+ lines.push('- **TODOs/FIXMEs:**');
168
+ for (const t of s.todos) lines.push(` - ${t}`);
169
+ }
170
+ lines.push('');
171
+ }
172
+
173
+ lines.push('## Modification Guidance', '', '- Verify tests before changing exported signatures.', '');
174
+
175
+ fs.mkdirSync(path.dirname(outputPath), { recursive: true });
176
+ fs.writeFileSync(outputPath, lines.join('\n'), 'utf-8');
177
+
178
+ updateState(projectRoot, {
179
+ deep_dive_targets: [{
180
+ target_name: targetName,
181
+ target_path: resolved.targetPath,
182
+ files_analyzed: files.length,
183
+ output_file: outputName,
184
+ timestamp: new Date().toISOString()
185
+ }],
186
+ outputs_generated: [outputName]
187
+ });
188
+
189
+ // Update master index with a link to this deep-dive doc.
190
+ renderIndex(projectRoot, { deepDiveFiles: [outputName] });
191
+
192
+ log(`Deep-dive written to ${outputPath}`);
193
+ return { ok: true, mode: 'deep_dive', changed: true, written: [outputName] };
194
+ }
195
+
196
+ module.exports = { runDeepDive, listFilesInScope, readFileSummary, resolveTarget };
@@ -0,0 +1,15 @@
1
+ // Full rescan mode for `wize-dev-kit document-project`.
2
+ //
3
+ // Archives the previous state and re-runs an initial scan from scratch.
4
+
5
+ 'use strict';
6
+
7
+ const { archiveOldState } = require('../state.js');
8
+ const { runInitialScan } = require('./initial-scan.js');
9
+
10
+ function runFullRescan(projectRoot, options = {}) {
11
+ archiveOldState(projectRoot);
12
+ return runInitialScan(projectRoot, { ...options, mode: 'full_rescan' });
13
+ }
14
+
15
+ module.exports = { runFullRescan };
@@ -0,0 +1,100 @@
1
+ // Initial scan mode for `wize-dev-kit document-project`.
2
+ //
3
+ // Runs project-type classification, conditional scans, and generates index + docs.
4
+
5
+ 'use strict';
6
+
7
+ const fs = require('node:fs');
8
+ const path = require('node:path');
9
+ const { classifyProject } = require('../classify.js');
10
+ const { batchScanner } = require('../batch-scanner.js');
11
+ const { renderIndex } = require('../render-index.js');
12
+ const { initState, updateState } = require('../state.js');
13
+ const { runQuick } = require('./quick.js');
14
+
15
+ function now() {
16
+ return new Date().toISOString().slice(0, 10);
17
+ }
18
+
19
+ function readRequirements(row) {
20
+ return {
21
+ api: row.requires_api_scan,
22
+ data: row.requires_data_models,
23
+ state: row.requires_state_management,
24
+ ui: row.requires_ui_components,
25
+ deploy: row.requires_deployment_config,
26
+ hardware: row.requires_hardware_docs,
27
+ assets: row.requires_asset_inventory
28
+ };
29
+ }
30
+
31
+ function existingDocs(root) {
32
+ const candidates = [
33
+ 'README.md',
34
+ 'CONTRIBUTING.md',
35
+ 'ARCHITECTURE.md',
36
+ 'DEPLOYMENT.md',
37
+ 'API.md'
38
+ ];
39
+ return candidates.filter(f => fs.existsSync(path.join(root, f))).map(f => path.join(root, f));
40
+ }
41
+
42
+ function runInitialScan(projectRoot, { scanLevel = 'quick', log = console.log, csvPath } = {}) {
43
+ const classification = classifyProject(projectRoot, csvPath ? { csvPath } : {});
44
+ const state = initState(projectRoot, 'initial_scan', scanLevel);
45
+
46
+ updateState(projectRoot, {
47
+ completed_steps: [{ step: 'classify', status: 'completed', timestamp: new Date().toISOString(), summary: `Classified as ${classification.projectTypes.join(', ')}` }],
48
+ current_step: 'scan',
49
+ findings: {
50
+ project_classification: {
51
+ repository_type: classification.parts.length >= 2 ? 'multi-part' : 'monolith',
52
+ parts_count: classification.parts.length,
53
+ primary_language: 'unknown',
54
+ architecture_type: classification.projectTypes[0] || 'unknown'
55
+ }
56
+ }
57
+ });
58
+
59
+ let generated = [];
60
+ if (scanLevel === 'deep' || scanLevel === 'exhaustive') {
61
+ const batches = batchScanner(projectRoot, { ignore: ['.wize', '.git', 'node_modules'] });
62
+ updateState(projectRoot, {
63
+ completed_steps: [{ step: 'scan', status: 'completed', timestamp: new Date().toISOString(), summary: `${batches.length} batches scanned` }],
64
+ current_step: 'generate',
65
+ findings: {
66
+ batches_completed: batches.map(b => ({ path: b.folder, files_scanned: b.fileCount, summary: `${b.totalLoc} LOC` }))
67
+ }
68
+ });
69
+ }
70
+
71
+ // Always write baseline + index.
72
+ const quick = runQuick(projectRoot, { log });
73
+ generated.push(...quick.written.map(f => path.join(projectRoot, '.wize', 'knowledge', 'document-project', f)));
74
+
75
+ const indexResult = renderIndex(projectRoot, {
76
+ projectTypes: classification.projectTypes,
77
+ parts: classification.parts,
78
+ generated,
79
+ existing: existingDocs(projectRoot)
80
+ });
81
+ generated.push(path.join(projectRoot, '.wize', 'knowledge', 'document-project', 'index.md'));
82
+
83
+ updateState(projectRoot, {
84
+ completed_steps: [{ step: 'generate', status: 'completed', timestamp: new Date().toISOString(), summary: `Generated ${generated.length} files` }],
85
+ current_step: 'completed',
86
+ outputs_generated: generated.map(g => path.basename(g)),
87
+ resume_instructions: 'Scan complete'
88
+ });
89
+
90
+ return {
91
+ changed: true,
92
+ mode: 'initial_scan',
93
+ scanLevel,
94
+ projectTypes: classification.projectTypes,
95
+ parts: classification.parts,
96
+ generated: generated.map(g => path.basename(g))
97
+ };
98
+ }
99
+
100
+ module.exports = { runInitialScan };
@@ -0,0 +1,211 @@
1
+ // Quick baseline mode for `wize-dev-kit document-project`.
2
+ //
3
+ // Produces the 6 lightweight baseline files. Does not read source files.
4
+
5
+ 'use strict';
6
+
7
+ const fs = require('node:fs');
8
+ const path = require('node:path');
9
+
10
+ const BASELINE_FILES = {
11
+ 'overview.md': overviewMarkdown,
12
+ 'architecture-snapshot.md': architectureMarkdown,
13
+ 'conventions.md': conventionsMarkdown,
14
+ 'dependencies.md': dependenciesMarkdown,
15
+ 'risk-spots.md': riskMarkdown,
16
+ 'open-questions.md': questionsMarkdown
17
+ };
18
+
19
+ function now() {
20
+ return new Date().toISOString().slice(0, 10);
21
+ }
22
+
23
+ function readPackage(root) {
24
+ try {
25
+ return JSON.parse(fs.readFileSync(path.join(root, 'package.json'), 'utf-8'));
26
+ } catch {
27
+ return {};
28
+ }
29
+ }
30
+
31
+ function countLines(root) {
32
+ let total = 0;
33
+ const dirs = ['src', 'tools', 'adapters', 'schemas', 'test'];
34
+ for (const dir of dirs) {
35
+ const full = path.join(root, dir);
36
+ if (!fs.existsSync(full)) continue;
37
+ total += walkLines(full);
38
+ }
39
+ return total;
40
+ }
41
+
42
+ function walkLines(dir) {
43
+ let total = 0;
44
+ const entries = fs.readdirSync(dir, { withFileTypes: true });
45
+ for (const e of entries) {
46
+ const full = path.join(dir, e.name);
47
+ if (e.isDirectory()) total += walkLines(full);
48
+ else if (e.isFile() && /\.(js|md|yaml|json)$/.test(e.name)) {
49
+ const content = fs.readFileSync(full, 'utf-8');
50
+ total += content.split('\n').length;
51
+ }
52
+ }
53
+ return total;
54
+ }
55
+
56
+ function recentCommitCount(root) {
57
+ // Avoid shelling out; return 0 if not a git repo or git unavailable.
58
+ // Caller can overlay a real count from CLI.
59
+ return 0;
60
+ }
61
+
62
+ function overviewMarkdown(root, pkg) {
63
+ return `---
64
+ status: baseline
65
+ owner: Pepper Potts + Peggy Carter
66
+ created: ${now()}
67
+ last_refreshed: ${now()}
68
+ ---
69
+
70
+ # Overview
71
+
72
+ **Project:** ${pkg.name || path.basename(root)}
73
+ **What it is:** ${pkg.description || 'Project documented by Wize Dev Kit.'}
74
+ **Current version:** ${pkg.version || 'unknown'}
75
+
76
+ ## Size
77
+
78
+ - Source lines of code: ~${countLines(root)} LOC.
79
+ - Runtime dependencies: ${Object.keys(pkg.dependencies || {}).length}.
80
+ - Dev dependencies: ${Object.keys(pkg.devDependencies || {}).length}.
81
+
82
+ ## What it ships
83
+
84
+ - See architecture-snapshot.md for components and entry points.
85
+ - See conventions.md for coding patterns.
86
+ - See dependencies.md for runtime and dev dependency roles.
87
+ `;
88
+ }
89
+
90
+ function architectureMarkdown() {
91
+ return `---
92
+ status: baseline
93
+ owner: Pepper Potts + Tony Stark
94
+ created: ${now()}
95
+ last_refreshed: ${now()}
96
+ ---
97
+
98
+ # Architecture Snapshot
99
+
100
+ ## Entry points
101
+
102
+ - Documented after a deeper scan (initial_scan mode).
103
+
104
+ ## Components
105
+
106
+ - Run \`wize-dev-kit document-project initial_scan deep\` for component-level detail.
107
+
108
+ ## Integration surface
109
+
110
+ - See dependencies.md for external integrations.
111
+ `;
112
+ }
113
+
114
+ function conventionsMarkdown() {
115
+ return `---
116
+ status: baseline
117
+ owner: Peggy Carter
118
+ created: ${now()}
119
+ last_refreshed: ${now()}
120
+ sampled: "pending deeper scan"
121
+ ---
122
+
123
+ # Conventions (observed, not prescribed)
124
+
125
+ ## Quick notes
126
+
127
+ - Run \`wize-dev-kit document-project initial_scan deep\` to sample files and fill this doc.
128
+ `;
129
+ }
130
+
131
+ function dependenciesMarkdown(root, pkg) {
132
+ const runtime = Object.entries(pkg.dependencies || {})
133
+ .map(([name, version]) => `| ${name} | ${version} | | |`).join('\n');
134
+ const dev = Object.entries(pkg.devDependencies || {})
135
+ .map(([name, version]) => `| ${name} | ${version} | | |`).join('\n');
136
+
137
+ return `---
138
+ status: baseline
139
+ owner: Pepper Potts
140
+ created: ${now()}
141
+ last_refreshed: ${now()}
142
+ ---
143
+
144
+ # Dependencies
145
+
146
+ ## Runtime
147
+
148
+ | Name | Version | Role in this repo | Load-bearing? |
149
+ |---|---|---|---|
150
+ ${runtime || '| — | — | — | — |'}
151
+
152
+ ## Dev / bundled
153
+
154
+ | Name | Version | Role in this repo | Load-bearing? |
155
+ |---|---|---|---|
156
+ ${dev || '| — | — | — | — |'}
157
+ `;
158
+ }
159
+
160
+ function riskMarkdown() {
161
+ return `---
162
+ status: baseline
163
+ owner: Pepper Potts + Tony Stark
164
+ created: ${now()}
165
+ last_refreshed: ${now()}
166
+ ---
167
+
168
+ # Risk Spots
169
+
170
+ | Area | Symptom | Likely cause | Confidence |
171
+ |---|---|---|---|
172
+ | | | | |
173
+
174
+ Run \`wize-dev-kit document-project initial_scan deep\` to populate risk spots.
175
+ `;
176
+ }
177
+
178
+ function questionsMarkdown() {
179
+ return `---
180
+ status: baseline
181
+ owner: Pepper Potts + Peggy Carter
182
+ created: ${now()}
183
+ last_refreshed: ${now()}
184
+ ---
185
+
186
+ # Open Questions
187
+
188
+ | Question | Why it matters | Owner to ask |
189
+ |---|---|---|
190
+ | | | |
191
+ `;
192
+ }
193
+
194
+ function runQuick(projectRoot, opts = {}) {
195
+ const pkg = readPackage(projectRoot);
196
+ const dir = path.join(projectRoot, '.wize', 'knowledge', 'document-project');
197
+ fs.mkdirSync(dir, { recursive: true });
198
+
199
+ const written = [];
200
+ for (const [name, factory] of Object.entries(BASELINE_FILES)) {
201
+ const file = path.join(dir, name);
202
+ const content = factory(projectRoot, pkg);
203
+ fs.writeFileSync(file, content, 'utf-8');
204
+ written.push(name);
205
+ }
206
+
207
+ if (opts.log) opts.log(`Quick baseline written to ${dir}`);
208
+ return { changed: true, mode: 'quick', written };
209
+ }
210
+
211
+ module.exports = { runQuick, BASELINE_FILES, countLines };