@brainfish-ai/devdoc 0.1.31 → 0.1.33
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/ai-agents/.claude/skills/bootstrap-docs/SKILL.md +221 -64
- package/ai-agents/.claude/skills/check-docs/SKILL.md +21 -2
- package/ai-agents/.claude/skills/create-doc-page/SKILL.md +242 -25
- package/ai-agents/.claude/skills/docs-from-code/SKILL.md +9 -1
- package/ai-agents/.claude/skills/sync-docs/SKILL.md +36 -2
- package/ai-agents/.cursor/rules/devdoc-bootstrap.mdc +90 -36
- package/ai-agents/.cursor/rules/devdoc-sync.mdc +46 -11
- package/ai-agents/CLAUDE.md +9 -0
- package/dist/cli/commands/ai.d.ts +1 -0
- package/dist/cli/commands/ai.js +100 -35
- package/dist/cli/commands/domain.js +31 -10
- package/dist/cli/index.js +3 -2
- package/package.json +1 -1
- package/renderer/components/docs-viewer/sidebar/index.tsx +118 -87
|
@@ -1,13 +1,47 @@
|
|
|
1
1
|
---
|
|
2
2
|
name: sync-docs
|
|
3
|
-
description: Analyze existing documentation against codebase and identify/fix outdated content
|
|
3
|
+
description: Analyze existing documentation against codebase and identify/fix outdated content. Reads docType from docs.json config.
|
|
4
4
|
---
|
|
5
5
|
|
|
6
6
|
## Instructions
|
|
7
7
|
|
|
8
8
|
When syncing documentation with the codebase:
|
|
9
9
|
|
|
10
|
-
### Step
|
|
10
|
+
### Step 0: Locate Source Code
|
|
11
|
+
|
|
12
|
+
First, determine where the source code is relative to current directory:
|
|
13
|
+
- If `docs.json` exists here → you're in docs folder, source code is in `../`
|
|
14
|
+
- If `package.json` and `src/` exist here → you're at repo root
|
|
15
|
+
- Check both current directory AND parent directory for source files
|
|
16
|
+
|
|
17
|
+
### Step 1: Get Documentation Type from Config
|
|
18
|
+
|
|
19
|
+
**Read `docs.json` and check for `docType` field:**
|
|
20
|
+
|
|
21
|
+
```json
|
|
22
|
+
{
|
|
23
|
+
"docType": "api" // "internal" | "api" | "product"
|
|
24
|
+
}
|
|
25
|
+
```
|
|
26
|
+
|
|
27
|
+
**If `docType` is set:** Use that value automatically.
|
|
28
|
+
|
|
29
|
+
**If `docType` is NOT set, detect from structure:**
|
|
30
|
+
- Has `architecture/`, `development/` → "internal"
|
|
31
|
+
- Has `api-reference/`, `sdks/` → "api"
|
|
32
|
+
- Has `features/`, `tutorials/` → "product"
|
|
33
|
+
|
|
34
|
+
**If still unclear, ask:**
|
|
35
|
+
"What type of documentation is this?
|
|
36
|
+
1. internal - for your team
|
|
37
|
+
2. api - for developers using your product
|
|
38
|
+
3. product - for end users"
|
|
39
|
+
|
|
40
|
+
**After user answers, save to docs.json:**
|
|
41
|
+
Add `"docType": "{choice}"` to docs.json so it's remembered.
|
|
42
|
+
Tell user: "Saved docType to docs.json - you won't be asked again."
|
|
43
|
+
|
|
44
|
+
### Step 2: Build Documentation Inventory
|
|
11
45
|
|
|
12
46
|
Scan all MDX files and extract:
|
|
13
47
|
- Documented functions, classes, components
|
|
@@ -7,60 +7,114 @@ globs: ["**/README.md", "**/package.json", "**/src/**", "**/lib/**"]
|
|
|
7
7
|
|
|
8
8
|
When asked to bootstrap or generate initial documentation:
|
|
9
9
|
|
|
10
|
-
##
|
|
10
|
+
## Step 1: Check for docType Configuration
|
|
11
|
+
|
|
12
|
+
**Read `docs.json` and check for `docType` field:**
|
|
13
|
+
|
|
14
|
+
```json
|
|
15
|
+
{
|
|
16
|
+
"name": "Project Name",
|
|
17
|
+
"docType": "api" // "internal" | "api" | "product"
|
|
18
|
+
}
|
|
19
|
+
```
|
|
20
|
+
|
|
21
|
+
**If `docType` is set:** Use that value automatically. Tell user: "Using docType: {type} from docs.json"
|
|
22
|
+
|
|
23
|
+
**If `docType` is NOT set, ask:**
|
|
24
|
+
|
|
25
|
+
"What type of documentation are you creating?
|
|
26
|
+
1. **internal** - For your team: setup guides, architecture
|
|
27
|
+
2. **api** - For developers using your product: API reference, SDKs
|
|
28
|
+
3. **product** - For end users: feature guides, tutorials"
|
|
29
|
+
|
|
30
|
+
**After user answers, immediately save to docs.json:**
|
|
31
|
+
Add `"docType": "{choice}"` to docs.json so it's saved for future use.
|
|
32
|
+
Tell user: "Saved docType: {type} to docs.json - you won't be asked again."
|
|
33
|
+
|
|
34
|
+
## Step 2: Locate Source Code
|
|
35
|
+
|
|
36
|
+
- If `docs.json` exists → you're in docs folder, source is in `../`
|
|
37
|
+
- If `package.json` + `src/` exist → you're at repo root
|
|
38
|
+
- Otherwise, ask user for source code path
|
|
39
|
+
|
|
40
|
+
## Step 3: Analysis Checklist
|
|
11
41
|
|
|
12
42
|
1. Read README.md for project overview
|
|
13
|
-
2. Check package.json for
|
|
14
|
-
- Project name and description
|
|
15
|
-
- Dependencies (to understand tech stack)
|
|
16
|
-
- Scripts (for CLI documentation)
|
|
43
|
+
2. Check package.json for name, description, dependencies, scripts
|
|
17
44
|
3. Scan source directory structure
|
|
18
45
|
4. Identify API endpoints or exported functions
|
|
19
|
-
5. Look for
|
|
46
|
+
5. Look for OpenAPI/GraphQL specs
|
|
20
47
|
6. Check for example files or tests
|
|
21
48
|
|
|
22
|
-
##
|
|
49
|
+
## Step 4: Generate Based on docType
|
|
23
50
|
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
51
|
+
### docType: "internal"
|
|
52
|
+
```
|
|
53
|
+
docs/
|
|
54
|
+
├── docs.json (with "docType": "internal")
|
|
55
|
+
├── index.mdx # Project overview
|
|
56
|
+
├── getting-started/
|
|
57
|
+
│ ├── setup.mdx # Dev environment setup
|
|
58
|
+
│ ├── prerequisites.mdx
|
|
59
|
+
│ └── first-contribution.mdx
|
|
60
|
+
├── architecture/
|
|
61
|
+
│ ├── overview.mdx # System design
|
|
62
|
+
│ └── folder-structure.mdx
|
|
63
|
+
├── development/
|
|
64
|
+
│ ├── workflow.mdx # Git workflow, PR process
|
|
65
|
+
│ ├── testing.mdx
|
|
66
|
+
│ └── deployment.mdx
|
|
67
|
+
└── contributing.mdx
|
|
68
|
+
```
|
|
37
69
|
|
|
70
|
+
### docType: "api"
|
|
38
71
|
```
|
|
39
72
|
docs/
|
|
40
|
-
├── docs.json
|
|
41
|
-
├── index.mdx
|
|
42
|
-
├── quickstart.mdx
|
|
73
|
+
├── docs.json (with "docType": "api")
|
|
74
|
+
├── index.mdx # Product intro
|
|
75
|
+
├── quickstart.mdx # 5-minute getting started
|
|
76
|
+
├── authentication.mdx # API keys, OAuth
|
|
43
77
|
├── guides/
|
|
44
|
-
│ └── overview.mdx
|
|
45
|
-
├── api-reference/
|
|
78
|
+
│ └── overview.mdx
|
|
79
|
+
├── api-reference/
|
|
46
80
|
│ ├── introduction.mdx
|
|
81
|
+
│ ├── errors.mdx
|
|
47
82
|
│ └── openapi.json
|
|
48
|
-
|
|
83
|
+
├── sdks/ # If SDKs exist
|
|
84
|
+
│ └── ...
|
|
85
|
+
└── changelog.mdx
|
|
86
|
+
```
|
|
87
|
+
|
|
88
|
+
### docType: "product"
|
|
89
|
+
```
|
|
90
|
+
docs/
|
|
91
|
+
├── docs.json (with "docType": "product")
|
|
92
|
+
├── index.mdx # Product overview
|
|
93
|
+
├── getting-started/
|
|
94
|
+
│ ├── quickstart.mdx
|
|
95
|
+
│ └── key-concepts.mdx
|
|
96
|
+
├── features/
|
|
97
|
+
│ └── {feature}.mdx
|
|
98
|
+
├── tutorials/
|
|
99
|
+
│ └── {workflow}.mdx
|
|
100
|
+
├── troubleshooting/
|
|
101
|
+
│ ├── common-issues.mdx
|
|
102
|
+
│ └── faq.mdx
|
|
103
|
+
└── release-notes.mdx
|
|
49
104
|
```
|
|
50
105
|
|
|
51
|
-
## Content
|
|
106
|
+
## Content Guidelines by docType
|
|
52
107
|
|
|
53
|
-
|
|
|
54
|
-
|
|
55
|
-
|
|
|
56
|
-
|
|
|
57
|
-
|
|
|
58
|
-
| tests/ | Usage examples |
|
|
59
|
-
| .env.example | Configuration options |
|
|
108
|
+
| docType | Focus | Tone | Code |
|
|
109
|
+
|---------|-------|------|------|
|
|
110
|
+
| internal | Setup, architecture, processes | Technical | Heavy |
|
|
111
|
+
| api | Endpoints, auth, examples | Technical | Heavy |
|
|
112
|
+
| product | Features, workflows, FAQs | Friendly | Minimal |
|
|
60
113
|
|
|
61
114
|
## Quality Guidelines
|
|
62
115
|
|
|
63
116
|
- Extract real examples, don't fabricate
|
|
64
|
-
- Note areas needing human review
|
|
117
|
+
- Note areas needing human review with TODO
|
|
65
118
|
- Generate SEO-friendly descriptions
|
|
66
|
-
-
|
|
119
|
+
- Match tone to audience
|
|
120
|
+
- Always set `docType` in generated docs.json
|
|
@@ -7,7 +7,34 @@ globs: ["**/*.mdx", "**/docs.json"]
|
|
|
7
7
|
|
|
8
8
|
When asked to sync, update, or check documentation:
|
|
9
9
|
|
|
10
|
-
##
|
|
10
|
+
## Step 1: Locate Source Code
|
|
11
|
+
|
|
12
|
+
- If `docs.json` exists here → source code is in `../`
|
|
13
|
+
- If `package.json` + `src/` exist → you're at repo root
|
|
14
|
+
- Check both current directory AND parent for source files
|
|
15
|
+
|
|
16
|
+
## Step 2: Get Documentation Type from Config
|
|
17
|
+
|
|
18
|
+
**Read `docs.json` for `docType` field:**
|
|
19
|
+
|
|
20
|
+
```json
|
|
21
|
+
{ "docType": "api" } // "internal" | "api" | "product"
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
**If `docType` is set:** Use that value automatically.
|
|
25
|
+
|
|
26
|
+
**If `docType` is NOT set, detect from structure:**
|
|
27
|
+
- Has `architecture/`, `development/` → "internal"
|
|
28
|
+
- Has `api-reference/`, `sdks/` → "api"
|
|
29
|
+
- Has `features/`, `tutorials/` → "product"
|
|
30
|
+
|
|
31
|
+
**If still unclear, ask:** "What type of documentation is this?"
|
|
32
|
+
|
|
33
|
+
**After user answers, save to docs.json:**
|
|
34
|
+
Add `"docType": "{choice}"` to docs.json.
|
|
35
|
+
Tell user: "Saved docType to docs.json - you won't be asked again."
|
|
36
|
+
|
|
37
|
+
## Step 3: Analysis Process
|
|
11
38
|
|
|
12
39
|
1. Compare documented APIs against source code exports
|
|
13
40
|
2. Verify code examples compile/run correctly
|
|
@@ -15,6 +42,24 @@ When asked to sync, update, or check documentation:
|
|
|
15
42
|
4. Validate all internal links resolve
|
|
16
43
|
5. Identify new features needing documentation
|
|
17
44
|
|
|
45
|
+
## Type-Specific Checks
|
|
46
|
+
|
|
47
|
+
### For docType: "internal"
|
|
48
|
+
- Dev setup instructions still work
|
|
49
|
+
- Architecture diagrams match current code
|
|
50
|
+
- Internal tool versions are current
|
|
51
|
+
|
|
52
|
+
### For docType: "api"
|
|
53
|
+
- API endpoints match OpenAPI spec
|
|
54
|
+
- Auth examples are correct
|
|
55
|
+
- SDK versions are current
|
|
56
|
+
- Error codes are complete
|
|
57
|
+
|
|
58
|
+
### For docType: "product"
|
|
59
|
+
- Screenshots match current UI
|
|
60
|
+
- Feature descriptions are accurate
|
|
61
|
+
- Tutorials still work
|
|
62
|
+
|
|
18
63
|
## Common Outdated Patterns
|
|
19
64
|
|
|
20
65
|
- Function signatures that changed
|
|
@@ -32,16 +77,6 @@ When asked to sync, update, or check documentation:
|
|
|
32
77
|
- Create stubs for undocumented new features
|
|
33
78
|
- Mark deprecated features with Warning callout
|
|
34
79
|
|
|
35
|
-
## Detection Checklist
|
|
36
|
-
|
|
37
|
-
| Check | How to Verify |
|
|
38
|
-
|-------|---------------|
|
|
39
|
-
| Function exists | Search source for export |
|
|
40
|
-
| Signature matches | Compare params and return type |
|
|
41
|
-
| Version current | Check package.json |
|
|
42
|
-
| Route exists | Search API routes |
|
|
43
|
-
| Component props | Check TypeScript interface |
|
|
44
|
-
|
|
45
80
|
## Update Patterns
|
|
46
81
|
|
|
47
82
|
### Changed Signature
|
package/ai-agents/CLAUDE.md
CHANGED
|
@@ -2,6 +2,15 @@
|
|
|
2
2
|
|
|
3
3
|
This is a DevDoc documentation project using MDX (Markdown + React components).
|
|
4
4
|
|
|
5
|
+
## Source Code Location
|
|
6
|
+
|
|
7
|
+
If this docs folder is inside a larger repository, the source code is in the parent directory:
|
|
8
|
+
- Source code: `../src/` or `../lib/`
|
|
9
|
+
- Package config: `../package.json`
|
|
10
|
+
- README: `../README.md`
|
|
11
|
+
|
|
12
|
+
When generating documentation, always check the parent directory (`../`) for source code to document.
|
|
13
|
+
|
|
5
14
|
## Project Structure
|
|
6
15
|
|
|
7
16
|
```
|
package/dist/cli/commands/ai.js
CHANGED
|
@@ -125,27 +125,29 @@ function getTemplateDir() {
|
|
|
125
125
|
}
|
|
126
126
|
/**
|
|
127
127
|
* Copy CLAUDE.md to project root
|
|
128
|
+
* Returns 'created' | 'updated' | 'skipped'
|
|
128
129
|
*/
|
|
129
130
|
function copyClaudeMd(projectPath, templateDir) {
|
|
130
131
|
const sourcePath = path_1.default.join(templateDir, 'CLAUDE.md');
|
|
131
132
|
const destPath = path_1.default.join(projectPath, 'CLAUDE.md');
|
|
132
133
|
if (!fs_extra_1.default.existsSync(sourcePath)) {
|
|
133
|
-
|
|
134
|
-
return false;
|
|
134
|
+
return 'skipped';
|
|
135
135
|
}
|
|
136
|
+
const existed = fs_extra_1.default.existsSync(destPath);
|
|
136
137
|
fs_extra_1.default.copySync(sourcePath, destPath);
|
|
137
|
-
return
|
|
138
|
+
return existed ? 'updated' : 'created';
|
|
138
139
|
}
|
|
139
140
|
/**
|
|
140
141
|
* Copy Claude Code skills to .claude/skills/
|
|
142
|
+
* Returns array of { skill, status: 'created' | 'updated' }
|
|
141
143
|
*/
|
|
142
144
|
function copyClaudeSkills(projectPath, templateDir) {
|
|
143
|
-
const
|
|
145
|
+
const results = [];
|
|
144
146
|
const sourceSkillsDir = path_1.default.join(templateDir, '.claude', 'skills');
|
|
145
147
|
const destSkillsDir = path_1.default.join(projectPath, '.claude', 'skills');
|
|
146
148
|
if (!fs_extra_1.default.existsSync(sourceSkillsDir)) {
|
|
147
149
|
logger_1.logger.warn('Claude skills template directory not found');
|
|
148
|
-
return
|
|
150
|
+
return results;
|
|
149
151
|
}
|
|
150
152
|
// Ensure destination directory exists
|
|
151
153
|
fs_extra_1.default.ensureDirSync(destSkillsDir);
|
|
@@ -153,22 +155,24 @@ function copyClaudeSkills(projectPath, templateDir) {
|
|
|
153
155
|
const sourceDir = path_1.default.join(sourceSkillsDir, skill);
|
|
154
156
|
const destDir = path_1.default.join(destSkillsDir, skill);
|
|
155
157
|
if (fs_extra_1.default.existsSync(sourceDir)) {
|
|
158
|
+
const existed = fs_extra_1.default.existsSync(destDir);
|
|
156
159
|
fs_extra_1.default.copySync(sourceDir, destDir);
|
|
157
|
-
|
|
160
|
+
results.push({ skill, status: existed ? 'updated' : 'created' });
|
|
158
161
|
}
|
|
159
162
|
}
|
|
160
|
-
return
|
|
163
|
+
return results;
|
|
161
164
|
}
|
|
162
165
|
/**
|
|
163
166
|
* Copy Cursor rules to .cursor/rules/
|
|
167
|
+
* Returns array of { rule, status: 'created' | 'updated' }
|
|
164
168
|
*/
|
|
165
169
|
function copyCursorRules(projectPath, templateDir) {
|
|
166
|
-
const
|
|
170
|
+
const results = [];
|
|
167
171
|
const sourceRulesDir = path_1.default.join(templateDir, '.cursor', 'rules');
|
|
168
172
|
const destRulesDir = path_1.default.join(projectPath, '.cursor', 'rules');
|
|
169
173
|
if (!fs_extra_1.default.existsSync(sourceRulesDir)) {
|
|
170
174
|
logger_1.logger.warn('Cursor rules template directory not found');
|
|
171
|
-
return
|
|
175
|
+
return results;
|
|
172
176
|
}
|
|
173
177
|
// Ensure destination directory exists
|
|
174
178
|
fs_extra_1.default.ensureDirSync(destRulesDir);
|
|
@@ -176,17 +180,36 @@ function copyCursorRules(projectPath, templateDir) {
|
|
|
176
180
|
const sourcePath = path_1.default.join(sourceRulesDir, rule);
|
|
177
181
|
const destPath = path_1.default.join(destRulesDir, rule);
|
|
178
182
|
if (fs_extra_1.default.existsSync(sourcePath)) {
|
|
183
|
+
const existed = fs_extra_1.default.existsSync(destPath);
|
|
179
184
|
fs_extra_1.default.copySync(sourcePath, destPath);
|
|
180
|
-
|
|
185
|
+
results.push({ rule, status: existed ? 'updated' : 'created' });
|
|
181
186
|
}
|
|
182
187
|
}
|
|
183
|
-
return
|
|
188
|
+
return results;
|
|
189
|
+
}
|
|
190
|
+
/**
|
|
191
|
+
* Check if AI tools are already installed
|
|
192
|
+
*/
|
|
193
|
+
function checkExistingInstallation(projectPath) {
|
|
194
|
+
const claudeDir = path_1.default.join(projectPath, '.claude', 'skills');
|
|
195
|
+
const cursorDir = path_1.default.join(projectPath, '.cursor', 'rules');
|
|
196
|
+
return {
|
|
197
|
+
claude: fs_extra_1.default.existsSync(claudeDir) && fs_extra_1.default.readdirSync(claudeDir).length > 0,
|
|
198
|
+
cursor: fs_extra_1.default.existsSync(cursorDir) && fs_extra_1.default.readdirSync(cursorDir).some(f => f.startsWith('devdoc')),
|
|
199
|
+
};
|
|
184
200
|
}
|
|
185
201
|
async function ai(options) {
|
|
186
|
-
console.log();
|
|
187
|
-
logger_1.logger.info('DevDoc AI Agent Setup');
|
|
188
202
|
console.log();
|
|
189
203
|
const projectPath = process.cwd();
|
|
204
|
+
const existing = checkExistingInstallation(projectPath);
|
|
205
|
+
const isUpdate = options.update || existing.claude || existing.cursor;
|
|
206
|
+
if (isUpdate) {
|
|
207
|
+
logger_1.logger.info('DevDoc AI Agent Update');
|
|
208
|
+
}
|
|
209
|
+
else {
|
|
210
|
+
logger_1.logger.info('DevDoc AI Agent Setup');
|
|
211
|
+
}
|
|
212
|
+
console.log();
|
|
190
213
|
// Check if this is a DevDoc project
|
|
191
214
|
const docsJsonPath = path_1.default.join(projectPath, 'docs.json');
|
|
192
215
|
const hasDocsJson = fs_extra_1.default.existsSync(docsJsonPath);
|
|
@@ -206,44 +229,82 @@ async function ai(options) {
|
|
|
206
229
|
// Get tool selection if not provided
|
|
207
230
|
let tool = options.tool || 'both';
|
|
208
231
|
if (!options.tool) {
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
232
|
+
// If updating, default to updating what's already installed
|
|
233
|
+
if (isUpdate && existing.claude && !existing.cursor) {
|
|
234
|
+
tool = 'claude';
|
|
235
|
+
}
|
|
236
|
+
else if (isUpdate && existing.cursor && !existing.claude) {
|
|
237
|
+
tool = 'cursor';
|
|
238
|
+
}
|
|
239
|
+
else {
|
|
240
|
+
const toolChoices = [
|
|
241
|
+
{ value: 'both', label: 'Both - Claude Code and Cursor' },
|
|
242
|
+
{ value: 'claude', label: 'Claude Code - Skills and CLAUDE.md' },
|
|
243
|
+
{ value: 'cursor', label: 'Cursor - Rules (.cursor/rules/)' },
|
|
244
|
+
];
|
|
245
|
+
tool = await promptSelect('Which AI tool do you use?', toolChoices);
|
|
246
|
+
}
|
|
215
247
|
}
|
|
216
248
|
console.log();
|
|
217
|
-
|
|
249
|
+
if (isUpdate) {
|
|
250
|
+
logger_1.logger.info('Updating AI agent configuration...');
|
|
251
|
+
}
|
|
252
|
+
else {
|
|
253
|
+
logger_1.logger.info('Setting up AI agent configuration...');
|
|
254
|
+
}
|
|
218
255
|
console.log();
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
claudeSkills: [],
|
|
222
|
-
cursorRules: [],
|
|
223
|
-
};
|
|
256
|
+
let createdCount = 0;
|
|
257
|
+
let updatedCount = 0;
|
|
224
258
|
// Setup Claude Code
|
|
225
259
|
if (tool === 'claude' || tool === 'both') {
|
|
226
260
|
// Copy CLAUDE.md
|
|
227
|
-
|
|
228
|
-
if (
|
|
261
|
+
const claudeMdStatus = copyClaudeMd(projectPath, templateDir);
|
|
262
|
+
if (claudeMdStatus === 'created') {
|
|
229
263
|
logger_1.logger.success('Created CLAUDE.md');
|
|
264
|
+
createdCount++;
|
|
265
|
+
}
|
|
266
|
+
else if (claudeMdStatus === 'updated') {
|
|
267
|
+
logger_1.logger.success('Updated CLAUDE.md');
|
|
268
|
+
updatedCount++;
|
|
230
269
|
}
|
|
231
270
|
// Copy skills
|
|
232
|
-
|
|
233
|
-
for (const skill of
|
|
234
|
-
|
|
271
|
+
const skillResults = copyClaudeSkills(projectPath, templateDir);
|
|
272
|
+
for (const { skill, status } of skillResults) {
|
|
273
|
+
if (status === 'created') {
|
|
274
|
+
logger_1.logger.success(`Created .claude/skills/${skill}/SKILL.md`);
|
|
275
|
+
createdCount++;
|
|
276
|
+
}
|
|
277
|
+
else {
|
|
278
|
+
logger_1.logger.success(`Updated .claude/skills/${skill}/SKILL.md`);
|
|
279
|
+
updatedCount++;
|
|
280
|
+
}
|
|
235
281
|
}
|
|
236
282
|
}
|
|
237
283
|
// Setup Cursor
|
|
238
284
|
if (tool === 'cursor' || tool === 'both') {
|
|
239
|
-
|
|
240
|
-
for (const rule of
|
|
241
|
-
|
|
285
|
+
const ruleResults = copyCursorRules(projectPath, templateDir);
|
|
286
|
+
for (const { rule, status } of ruleResults) {
|
|
287
|
+
if (status === 'created') {
|
|
288
|
+
logger_1.logger.success(`Created .cursor/rules/${rule}`);
|
|
289
|
+
createdCount++;
|
|
290
|
+
}
|
|
291
|
+
else {
|
|
292
|
+
logger_1.logger.success(`Updated .cursor/rules/${rule}`);
|
|
293
|
+
updatedCount++;
|
|
294
|
+
}
|
|
242
295
|
}
|
|
243
296
|
}
|
|
244
297
|
// Summary
|
|
245
298
|
console.log();
|
|
246
|
-
|
|
299
|
+
if (updatedCount > 0 && createdCount > 0) {
|
|
300
|
+
logger_1.logger.success(`AI agent configuration complete! (${createdCount} created, ${updatedCount} updated)`);
|
|
301
|
+
}
|
|
302
|
+
else if (updatedCount > 0) {
|
|
303
|
+
logger_1.logger.success(`AI agent configuration updated! (${updatedCount} files)`);
|
|
304
|
+
}
|
|
305
|
+
else {
|
|
306
|
+
logger_1.logger.success('AI agent configuration complete!');
|
|
307
|
+
}
|
|
247
308
|
console.log();
|
|
248
309
|
if (tool === 'claude' || tool === 'both') {
|
|
249
310
|
console.log('Available Claude Code commands:');
|
|
@@ -276,5 +337,9 @@ async function ai(options) {
|
|
|
276
337
|
console.log(' Cursor: Ask "generate initial documentation from this repo"');
|
|
277
338
|
}
|
|
278
339
|
console.log();
|
|
340
|
+
if (isUpdate) {
|
|
341
|
+
console.log('Tip: Restart Claude Code or Cursor to load the updated skills/rules.');
|
|
342
|
+
console.log();
|
|
343
|
+
}
|
|
279
344
|
}
|
|
280
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ai.js","sourceRoot":"","sources":["../../../src/cli/commands/ai.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path';\nimport fs from 'fs-extra';\nimport { logger } from '../../utils/logger';\n\ntype AITool = 'claude' | 'cursor' | 'both';\n\ninterface AIOptions {\n  tool?: AITool;\n}\n\n// Simple prompt helper using readline\nasync function prompt(question: string, defaultValue?: string): Promise<string> {\n  const readline = await import('readline');\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  });\n\n  return new Promise((resolve) => {\n    const displayQuestion = defaultValue \n      ? `${question} (${defaultValue}): `\n      : `${question}: `;\n    \n    rl.question(displayQuestion, (answer) => {\n      rl.close();\n      resolve(answer.trim() || defaultValue || '');\n    });\n  });\n}\n\nasync function promptSelect(question: string, choices: { value: string; label: string }[]): Promise<string> {\n  console.log(`\\n${question}\\n`);\n  choices.forEach((choice, i) => {\n    console.log(`  ${i + 1}. ${choice.label}`);\n  });\n  console.log();\n  \n  const readline = await import('readline');\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  });\n\n  return new Promise((resolve) => {\n    rl.question('Enter number: ', (answer) => {\n      rl.close();\n      const index = parseInt(answer.trim(), 10) - 1;\n      if (index >= 0 && index < choices.length) {\n        resolve(choices[index].value);\n      } else {\n        // Default to first choice\n        resolve(choices[0].value);\n      }\n    });\n  });\n}\n\n// Claude Code skills to create\nconst CLAUDE_SKILLS = [\n  'bootstrap-docs',\n  'migrate-docs',\n  'import-api-spec',\n  'sync-docs',\n  'check-docs',\n  'create-doc-page',\n  'update-docs-json',\n  'generate-api-docs',\n  'docs-from-code',\n];\n\n// Cursor rules to create\nconst CURSOR_RULES = [\n  'devdoc.mdc',\n  'devdoc-bootstrap.mdc',\n  'devdoc-migrate.mdc',\n  'devdoc-sync.mdc',\n];\n\n/**\n * Get the AI agents template directory path (handles both development and installed package scenarios)\n */\nfunction getTemplateDir(): string {\n  // Try relative to compiled dist (packages/devdoc/dist/cli/commands -> packages/devdoc/ai-agents)\n  let templateDir = path.join(__dirname, '..', '..', '..', 'ai-agents');\n  \n  if (fs.existsSync(templateDir)) {\n    return templateDir;\n  }\n  \n  // Try relative to source (packages/devdoc/src/cli/commands -> packages/devdoc/ai-agents)\n  templateDir = path.join(__dirname, '..', '..', '..', 'ai-agents');\n  \n  if (fs.existsSync(templateDir)) {\n    return templateDir;\n  }\n  \n  // Try devdoc/templates at repo root (development - monorepo structure)\n  // __dirname is packages/devdoc/dist/cli/commands -> go up 5 levels to repo root\n  templateDir = path.join(__dirname, '..', '..', '..', '..', '..', 'devdoc', 'templates', 'starter');\n  \n  if (fs.existsSync(templateDir)) {\n    return templateDir;\n  }\n  \n  throw new Error('Could not find template directory');\n}\n\n/**\n * Copy CLAUDE.md to project root\n */\nfunction copyClaudeMd(projectPath: string, templateDir: string): boolean {\n  const sourcePath = path.join(templateDir, 'CLAUDE.md');\n  const destPath = path.join(projectPath, 'CLAUDE.md');\n  \n  if (!fs.existsSync(sourcePath)) {\n    logger.warn('CLAUDE.md template not found');\n    return false;\n  }\n  \n  fs.copySync(sourcePath, destPath);\n  return true;\n}\n\n/**\n * Copy Claude Code skills to .claude/skills/\n */\nfunction copyClaudeSkills(projectPath: string, templateDir: string): string[] {\n  const copied: string[] = [];\n  const sourceSkillsDir = path.join(templateDir, '.claude', 'skills');\n  const destSkillsDir = path.join(projectPath, '.claude', 'skills');\n  \n  if (!fs.existsSync(sourceSkillsDir)) {\n    logger.warn('Claude skills template directory not found');\n    return copied;\n  }\n  \n  // Ensure destination directory exists\n  fs.ensureDirSync(destSkillsDir);\n  \n  for (const skill of CLAUDE_SKILLS) {\n    const sourceDir = path.join(sourceSkillsDir, skill);\n    const destDir = path.join(destSkillsDir, skill);\n    \n    if (fs.existsSync(sourceDir)) {\n      fs.copySync(sourceDir, destDir);\n      copied.push(skill);\n    }\n  }\n  \n  return copied;\n}\n\n/**\n * Copy Cursor rules to .cursor/rules/\n */\nfunction copyCursorRules(projectPath: string, templateDir: string): string[] {\n  const copied: string[] = [];\n  const sourceRulesDir = path.join(templateDir, '.cursor', 'rules');\n  const destRulesDir = path.join(projectPath, '.cursor', 'rules');\n  \n  if (!fs.existsSync(sourceRulesDir)) {\n    logger.warn('Cursor rules template directory not found');\n    return copied;\n  }\n  \n  // Ensure destination directory exists\n  fs.ensureDirSync(destRulesDir);\n  \n  for (const rule of CURSOR_RULES) {\n    const sourcePath = path.join(sourceRulesDir, rule);\n    const destPath = path.join(destRulesDir, rule);\n    \n    if (fs.existsSync(sourcePath)) {\n      fs.copySync(sourcePath, destPath);\n      copied.push(rule);\n    }\n  }\n  \n  return copied;\n}\n\nexport async function ai(options: AIOptions): Promise<void> {\n  console.log();\n  logger.info('DevDoc AI Agent Setup');\n  console.log();\n\n  const projectPath = process.cwd();\n  \n  // Check if this is a DevDoc project\n  const docsJsonPath = path.join(projectPath, 'docs.json');\n  const hasDocsJson = fs.existsSync(docsJsonPath);\n  \n  if (!hasDocsJson) {\n    logger.warn('No docs.json found. This may not be a DevDoc project.');\n    console.log();\n  }\n\n  // Get template directory\n  let templateDir: string;\n  try {\n    templateDir = getTemplateDir();\n  } catch {\n    logger.error('Could not find AI agent templates. Please ensure DevDoc is installed correctly.');\n    process.exit(1);\n  }\n\n  // Get tool selection if not provided\n  let tool: AITool = options.tool || 'both';\n  \n  if (!options.tool) {\n    const toolChoices = [\n      { value: 'both', label: 'Both - Claude Code and Cursor' },\n      { value: 'claude', label: 'Claude Code - Skills and CLAUDE.md' },\n      { value: 'cursor', label: 'Cursor - Rules (.cursor/rules/)' },\n    ];\n\n    tool = await promptSelect('Which AI tool do you use?', toolChoices) as AITool;\n  }\n\n  console.log();\n  logger.info('Setting up AI agent configuration...');\n  console.log();\n\n  const results = {\n    claudeMd: false,\n    claudeSkills: [] as string[],\n    cursorRules: [] as string[],\n  };\n\n  // Setup Claude Code\n  if (tool === 'claude' || tool === 'both') {\n    // Copy CLAUDE.md\n    results.claudeMd = copyClaudeMd(projectPath, templateDir);\n    if (results.claudeMd) {\n      logger.success('Created CLAUDE.md');\n    }\n    \n    // Copy skills\n    results.claudeSkills = copyClaudeSkills(projectPath, templateDir);\n    for (const skill of results.claudeSkills) {\n      logger.success(`Created .claude/skills/${skill}/SKILL.md`);\n    }\n  }\n\n  // Setup Cursor\n  if (tool === 'cursor' || tool === 'both') {\n    results.cursorRules = copyCursorRules(projectPath, templateDir);\n    for (const rule of results.cursorRules) {\n      logger.success(`Created .cursor/rules/${rule}`);\n    }\n  }\n\n  // Summary\n  console.log();\n  logger.success('AI agent configuration complete!');\n  console.log();\n\n  if (tool === 'claude' || tool === 'both') {\n    console.log('Available Claude Code commands:');\n    console.log('  /bootstrap-docs    - Analyze repo and generate initial documentation');\n    console.log('  /migrate-docs      - Migrate from Mintlify, Docusaurus, GitBook, etc.');\n    console.log('  /import-api-spec   - Import OpenAPI, GraphQL, or AsyncAPI specs');\n    console.log('  /sync-docs         - Find and fix outdated documentation');\n    console.log('  /check-docs        - Quick health check without changes');\n    console.log('  /create-doc-page   - Create a new documentation page');\n    console.log('  /update-docs-json  - Add pages to navigation');\n    console.log('  /generate-api-docs - Generate API documentation');\n    console.log('  /docs-from-code    - Generate docs from specific code files');\n    console.log();\n  }\n\n  if (tool === 'cursor' || tool === 'both') {\n    console.log('Cursor rules configured for *.mdx files.');\n    console.log();\n    console.log('Suggested prompts in Agent mode:');\n    console.log('  \"Analyze this repo and generate initial documentation\"');\n    console.log('  \"Migrate my Mintlify docs to DevDoc format\"');\n    console.log('  \"Check my docs for outdated content\"');\n    console.log('  \"Create a new guide about authentication\"');\n    console.log();\n  }\n\n  console.log('Quick start:');\n  if (tool === 'claude' || tool === 'both') {\n    console.log('  Claude Code: /bootstrap-docs');\n  }\n  if (tool === 'cursor' || tool === 'both') {\n    console.log('  Cursor: Ask \"generate initial documentation from this repo\"');\n  }\n  console.log();\n}\n"]}
|
|
345
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ai.js","sourceRoot":"","sources":["../../../src/cli/commands/ai.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path';\nimport fs from 'fs-extra';\nimport { logger } from '../../utils/logger';\n\ntype AITool = 'claude' | 'cursor' | 'both';\n\ninterface AIOptions {\n  tool?: AITool;\n  update?: boolean;\n}\n\n// Simple prompt helper using readline\nasync function prompt(question: string, defaultValue?: string): Promise<string> {\n  const readline = await import('readline');\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  });\n\n  return new Promise((resolve) => {\n    const displayQuestion = defaultValue \n      ? `${question} (${defaultValue}): `\n      : `${question}: `;\n    \n    rl.question(displayQuestion, (answer) => {\n      rl.close();\n      resolve(answer.trim() || defaultValue || '');\n    });\n  });\n}\n\nasync function promptSelect(question: string, choices: { value: string; label: string }[]): Promise<string> {\n  console.log(`\\n${question}\\n`);\n  choices.forEach((choice, i) => {\n    console.log(`  ${i + 1}. ${choice.label}`);\n  });\n  console.log();\n  \n  const readline = await import('readline');\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  });\n\n  return new Promise((resolve) => {\n    rl.question('Enter number: ', (answer) => {\n      rl.close();\n      const index = parseInt(answer.trim(), 10) - 1;\n      if (index >= 0 && index < choices.length) {\n        resolve(choices[index].value);\n      } else {\n        // Default to first choice\n        resolve(choices[0].value);\n      }\n    });\n  });\n}\n\n// Claude Code skills to create\nconst CLAUDE_SKILLS = [\n  'bootstrap-docs',\n  'migrate-docs',\n  'import-api-spec',\n  'sync-docs',\n  'check-docs',\n  'create-doc-page',\n  'update-docs-json',\n  'generate-api-docs',\n  'docs-from-code',\n];\n\n// Cursor rules to create\nconst CURSOR_RULES = [\n  'devdoc.mdc',\n  'devdoc-bootstrap.mdc',\n  'devdoc-migrate.mdc',\n  'devdoc-sync.mdc',\n];\n\n/**\n * Get the AI agents template directory path (handles both development and installed package scenarios)\n */\nfunction getTemplateDir(): string {\n  // Try relative to compiled dist (packages/devdoc/dist/cli/commands -> packages/devdoc/ai-agents)\n  let templateDir = path.join(__dirname, '..', '..', '..', 'ai-agents');\n  \n  if (fs.existsSync(templateDir)) {\n    return templateDir;\n  }\n  \n  // Try relative to source (packages/devdoc/src/cli/commands -> packages/devdoc/ai-agents)\n  templateDir = path.join(__dirname, '..', '..', '..', 'ai-agents');\n  \n  if (fs.existsSync(templateDir)) {\n    return templateDir;\n  }\n  \n  // Try devdoc/templates at repo root (development - monorepo structure)\n  // __dirname is packages/devdoc/dist/cli/commands -> go up 5 levels to repo root\n  templateDir = path.join(__dirname, '..', '..', '..', '..', '..', 'devdoc', 'templates', 'starter');\n  \n  if (fs.existsSync(templateDir)) {\n    return templateDir;\n  }\n  \n  throw new Error('Could not find template directory');\n}\n\n/**\n * Copy CLAUDE.md to project root\n * Returns 'created' | 'updated' | 'skipped'\n */\nfunction copyClaudeMd(projectPath: string, templateDir: string): 'created' | 'updated' | 'skipped' {\n  const sourcePath = path.join(templateDir, 'CLAUDE.md');\n  const destPath = path.join(projectPath, 'CLAUDE.md');\n  \n  if (!fs.existsSync(sourcePath)) {\n    return 'skipped';\n  }\n  \n  const existed = fs.existsSync(destPath);\n  fs.copySync(sourcePath, destPath);\n  return existed ? 'updated' : 'created';\n}\n\n/**\n * Copy Claude Code skills to .claude/skills/\n * Returns array of { skill, status: 'created' | 'updated' }\n */\nfunction copyClaudeSkills(projectPath: string, templateDir: string): { skill: string; status: 'created' | 'updated' }[] {\n  const results: { skill: string; status: 'created' | 'updated' }[] = [];\n  const sourceSkillsDir = path.join(templateDir, '.claude', 'skills');\n  const destSkillsDir = path.join(projectPath, '.claude', 'skills');\n  \n  if (!fs.existsSync(sourceSkillsDir)) {\n    logger.warn('Claude skills template directory not found');\n    return results;\n  }\n  \n  // Ensure destination directory exists\n  fs.ensureDirSync(destSkillsDir);\n  \n  for (const skill of CLAUDE_SKILLS) {\n    const sourceDir = path.join(sourceSkillsDir, skill);\n    const destDir = path.join(destSkillsDir, skill);\n    \n    if (fs.existsSync(sourceDir)) {\n      const existed = fs.existsSync(destDir);\n      fs.copySync(sourceDir, destDir);\n      results.push({ skill, status: existed ? 'updated' : 'created' });\n    }\n  }\n  \n  return results;\n}\n\n/**\n * Copy Cursor rules to .cursor/rules/\n * Returns array of { rule, status: 'created' | 'updated' }\n */\nfunction copyCursorRules(projectPath: string, templateDir: string): { rule: string; status: 'created' | 'updated' }[] {\n  const results: { rule: string; status: 'created' | 'updated' }[] = [];\n  const sourceRulesDir = path.join(templateDir, '.cursor', 'rules');\n  const destRulesDir = path.join(projectPath, '.cursor', 'rules');\n  \n  if (!fs.existsSync(sourceRulesDir)) {\n    logger.warn('Cursor rules template directory not found');\n    return results;\n  }\n  \n  // Ensure destination directory exists\n  fs.ensureDirSync(destRulesDir);\n  \n  for (const rule of CURSOR_RULES) {\n    const sourcePath = path.join(sourceRulesDir, rule);\n    const destPath = path.join(destRulesDir, rule);\n    \n    if (fs.existsSync(sourcePath)) {\n      const existed = fs.existsSync(destPath);\n      fs.copySync(sourcePath, destPath);\n      results.push({ rule, status: existed ? 'updated' : 'created' });\n    }\n  }\n  \n  return results;\n}\n\n/**\n * Check if AI tools are already installed\n */\nfunction checkExistingInstallation(projectPath: string): { claude: boolean; cursor: boolean } {\n  const claudeDir = path.join(projectPath, '.claude', 'skills');\n  const cursorDir = path.join(projectPath, '.cursor', 'rules');\n  \n  return {\n    claude: fs.existsSync(claudeDir) && fs.readdirSync(claudeDir).length > 0,\n    cursor: fs.existsSync(cursorDir) && fs.readdirSync(cursorDir).some(f => f.startsWith('devdoc')),\n  };\n}\n\nexport async function ai(options: AIOptions): Promise<void> {\n  console.log();\n  \n  const projectPath = process.cwd();\n  const existing = checkExistingInstallation(projectPath);\n  const isUpdate = options.update || existing.claude || existing.cursor;\n  \n  if (isUpdate) {\n    logger.info('DevDoc AI Agent Update');\n  } else {\n    logger.info('DevDoc AI Agent Setup');\n  }\n  console.log();\n\n  // Check if this is a DevDoc project\n  const docsJsonPath = path.join(projectPath, 'docs.json');\n  const hasDocsJson = fs.existsSync(docsJsonPath);\n  \n  if (!hasDocsJson) {\n    logger.warn('No docs.json found. This may not be a DevDoc project.');\n    console.log();\n  }\n\n  // Get template directory\n  let templateDir: string;\n  try {\n    templateDir = getTemplateDir();\n  } catch {\n    logger.error('Could not find AI agent templates. Please ensure DevDoc is installed correctly.');\n    process.exit(1);\n  }\n\n  // Get tool selection if not provided\n  let tool: AITool = options.tool || 'both';\n  \n  if (!options.tool) {\n    // If updating, default to updating what's already installed\n    if (isUpdate && existing.claude && !existing.cursor) {\n      tool = 'claude';\n    } else if (isUpdate && existing.cursor && !existing.claude) {\n      tool = 'cursor';\n    } else {\n      const toolChoices = [\n        { value: 'both', label: 'Both - Claude Code and Cursor' },\n        { value: 'claude', label: 'Claude Code - Skills and CLAUDE.md' },\n        { value: 'cursor', label: 'Cursor - Rules (.cursor/rules/)' },\n      ];\n\n      tool = await promptSelect('Which AI tool do you use?', toolChoices) as AITool;\n    }\n  }\n\n  console.log();\n  if (isUpdate) {\n    logger.info('Updating AI agent configuration...');\n  } else {\n    logger.info('Setting up AI agent configuration...');\n  }\n  console.log();\n\n  let createdCount = 0;\n  let updatedCount = 0;\n\n  // Setup Claude Code\n  if (tool === 'claude' || tool === 'both') {\n    // Copy CLAUDE.md\n    const claudeMdStatus = copyClaudeMd(projectPath, templateDir);\n    if (claudeMdStatus === 'created') {\n      logger.success('Created CLAUDE.md');\n      createdCount++;\n    } else if (claudeMdStatus === 'updated') {\n      logger.success('Updated CLAUDE.md');\n      updatedCount++;\n    }\n    \n    // Copy skills\n    const skillResults = copyClaudeSkills(projectPath, templateDir);\n    for (const { skill, status } of skillResults) {\n      if (status === 'created') {\n        logger.success(`Created .claude/skills/${skill}/SKILL.md`);\n        createdCount++;\n      } else {\n        logger.success(`Updated .claude/skills/${skill}/SKILL.md`);\n        updatedCount++;\n      }\n    }\n  }\n\n  // Setup Cursor\n  if (tool === 'cursor' || tool === 'both') {\n    const ruleResults = copyCursorRules(projectPath, templateDir);\n    for (const { rule, status } of ruleResults) {\n      if (status === 'created') {\n        logger.success(`Created .cursor/rules/${rule}`);\n        createdCount++;\n      } else {\n        logger.success(`Updated .cursor/rules/${rule}`);\n        updatedCount++;\n      }\n    }\n  }\n\n  // Summary\n  console.log();\n  if (updatedCount > 0 && createdCount > 0) {\n    logger.success(`AI agent configuration complete! (${createdCount} created, ${updatedCount} updated)`);\n  } else if (updatedCount > 0) {\n    logger.success(`AI agent configuration updated! (${updatedCount} files)`);\n  } else {\n    logger.success('AI agent configuration complete!');\n  }\n  console.log();\n\n  if (tool === 'claude' || tool === 'both') {\n    console.log('Available Claude Code commands:');\n    console.log('  /bootstrap-docs    - Analyze repo and generate initial documentation');\n    console.log('  /migrate-docs      - Migrate from Mintlify, Docusaurus, GitBook, etc.');\n    console.log('  /import-api-spec   - Import OpenAPI, GraphQL, or AsyncAPI specs');\n    console.log('  /sync-docs         - Find and fix outdated documentation');\n    console.log('  /check-docs        - Quick health check without changes');\n    console.log('  /create-doc-page   - Create a new documentation page');\n    console.log('  /update-docs-json  - Add pages to navigation');\n    console.log('  /generate-api-docs - Generate API documentation');\n    console.log('  /docs-from-code    - Generate docs from specific code files');\n    console.log();\n  }\n\n  if (tool === 'cursor' || tool === 'both') {\n    console.log('Cursor rules configured for *.mdx files.');\n    console.log();\n    console.log('Suggested prompts in Agent mode:');\n    console.log('  \"Analyze this repo and generate initial documentation\"');\n    console.log('  \"Migrate my Mintlify docs to DevDoc format\"');\n    console.log('  \"Check my docs for outdated content\"');\n    console.log('  \"Create a new guide about authentication\"');\n    console.log();\n  }\n\n  console.log('Quick start:');\n  if (tool === 'claude' || tool === 'both') {\n    console.log('  Claude Code: /bootstrap-docs');\n  }\n  if (tool === 'cursor' || tool === 'both') {\n    console.log('  Cursor: Ask \"generate initial documentation from this repo\"');\n  }\n  console.log();\n  \n  if (isUpdate) {\n    console.log('Tip: Restart Claude Code or Cursor to load the updated skills/rules.');\n    console.log();\n  }\n}\n"]}
|