@brainfish-ai/devdoc 0.1.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +33 -0
- package/README.md +415 -0
- package/bin/devdoc.js +13 -0
- package/dist/cli/commands/build.d.ts +5 -0
- package/dist/cli/commands/build.js +87 -0
- package/dist/cli/commands/check.d.ts +1 -0
- package/dist/cli/commands/check.js +143 -0
- package/dist/cli/commands/create.d.ts +24 -0
- package/dist/cli/commands/create.js +387 -0
- package/dist/cli/commands/deploy.d.ts +9 -0
- package/dist/cli/commands/deploy.js +433 -0
- package/dist/cli/commands/dev.d.ts +6 -0
- package/dist/cli/commands/dev.js +139 -0
- package/dist/cli/commands/init.d.ts +11 -0
- package/dist/cli/commands/init.js +238 -0
- package/dist/cli/commands/keys.d.ts +12 -0
- package/dist/cli/commands/keys.js +165 -0
- package/dist/cli/commands/start.d.ts +5 -0
- package/dist/cli/commands/start.js +56 -0
- package/dist/cli/commands/upload.d.ts +13 -0
- package/dist/cli/commands/upload.js +238 -0
- package/dist/cli/commands/whoami.d.ts +8 -0
- package/dist/cli/commands/whoami.js +91 -0
- package/dist/cli/index.d.ts +1 -0
- package/dist/cli/index.js +106 -0
- package/dist/config/index.d.ts +80 -0
- package/dist/config/index.js +133 -0
- package/dist/constants.d.ts +9 -0
- package/dist/constants.js +13 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.js +12 -0
- package/dist/utils/logger.d.ts +16 -0
- package/dist/utils/logger.js +61 -0
- package/dist/utils/paths.d.ts +16 -0
- package/dist/utils/paths.js +50 -0
- package/package.json +51 -0
- package/renderer/app/api/assets/[...path]/route.ts +123 -0
- package/renderer/app/api/assets/route.ts +124 -0
- package/renderer/app/api/assets/upload/route.ts +177 -0
- package/renderer/app/api/auth-schemes/route.ts +77 -0
- package/renderer/app/api/chat/route.ts +858 -0
- package/renderer/app/api/codegen/route.ts +72 -0
- package/renderer/app/api/collections/route.ts +1016 -0
- package/renderer/app/api/debug/route.ts +53 -0
- package/renderer/app/api/deploy/route.ts +234 -0
- package/renderer/app/api/device/route.ts +42 -0
- package/renderer/app/api/docs/route.ts +187 -0
- package/renderer/app/api/keys/regenerate/route.ts +80 -0
- package/renderer/app/api/openapi-spec/route.ts +151 -0
- package/renderer/app/api/projects/[slug]/route.ts +153 -0
- package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
- package/renderer/app/api/projects/register/route.ts +152 -0
- package/renderer/app/api/proxy/route.ts +149 -0
- package/renderer/app/api/proxy-stream/route.ts +168 -0
- package/renderer/app/api/redirects/route.ts +47 -0
- package/renderer/app/api/schema/route.ts +65 -0
- package/renderer/app/api/subdomains/check/route.ts +172 -0
- package/renderer/app/api/suggestions/route.ts +144 -0
- package/renderer/app/favicon.ico +0 -0
- package/renderer/app/globals.css +1103 -0
- package/renderer/app/layout.tsx +47 -0
- package/renderer/app/llms-full.txt/route.ts +346 -0
- package/renderer/app/llms.txt/route.ts +279 -0
- package/renderer/app/page.tsx +14 -0
- package/renderer/app/robots.txt/route.ts +84 -0
- package/renderer/app/sitemap.xml/route.ts +199 -0
- package/renderer/components/docs/index.ts +12 -0
- package/renderer/components/docs/mdx/accordion.tsx +169 -0
- package/renderer/components/docs/mdx/badge.tsx +132 -0
- package/renderer/components/docs/mdx/callouts.tsx +154 -0
- package/renderer/components/docs/mdx/cards.tsx +213 -0
- package/renderer/components/docs/mdx/changelog.tsx +120 -0
- package/renderer/components/docs/mdx/code-block.tsx +186 -0
- package/renderer/components/docs/mdx/code-group.tsx +421 -0
- package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
- package/renderer/components/docs/mdx/frame.tsx +112 -0
- package/renderer/components/docs/mdx/highlight.tsx +151 -0
- package/renderer/components/docs/mdx/iframe.tsx +134 -0
- package/renderer/components/docs/mdx/image.tsx +235 -0
- package/renderer/components/docs/mdx/index.ts +204 -0
- package/renderer/components/docs/mdx/mermaid.tsx +240 -0
- package/renderer/components/docs/mdx/param-field.tsx +200 -0
- package/renderer/components/docs/mdx/steps.tsx +113 -0
- package/renderer/components/docs/mdx/tabs.tsx +86 -0
- package/renderer/components/docs/mdx-renderer.tsx +100 -0
- package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
- package/renderer/components/docs/navigation/index.ts +8 -0
- package/renderer/components/docs/navigation/page-nav.tsx +64 -0
- package/renderer/components/docs/navigation/sidebar.tsx +515 -0
- package/renderer/components/docs/navigation/toc.tsx +113 -0
- package/renderer/components/docs/notice.tsx +105 -0
- package/renderer/components/docs-header.tsx +274 -0
- package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
- package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
- package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
- package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
- package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
- package/renderer/components/docs-viewer/agent/index.tsx +6 -0
- package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
- package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
- package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
- package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
- package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
- package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
- package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
- package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
- package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
- package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
- package/renderer/components/docs-viewer/content/index.tsx +29 -0
- package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
- package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
- package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
- package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
- package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
- package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
- package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
- package/renderer/components/docs-viewer/index.tsx +1466 -0
- package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
- package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
- package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
- package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
- package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
- package/renderer/components/docs-viewer/playground/index.tsx +671 -0
- package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
- package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
- package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
- package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
- package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
- package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
- package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
- package/renderer/components/docs-viewer/search/index.ts +2 -0
- package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
- package/renderer/components/docs-viewer/search/use-search.ts +117 -0
- package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
- package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
- package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
- package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
- package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
- package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
- package/renderer/components/theme-provider.tsx +11 -0
- package/renderer/components/theme-toggle.tsx +76 -0
- package/renderer/components/ui/badge.tsx +46 -0
- package/renderer/components/ui/button.tsx +59 -0
- package/renderer/components/ui/dialog.tsx +118 -0
- package/renderer/components/ui/dropdown-menu.tsx +257 -0
- package/renderer/components/ui/input.tsx +21 -0
- package/renderer/components/ui/label.tsx +24 -0
- package/renderer/components/ui/navigation-menu.tsx +168 -0
- package/renderer/components/ui/select.tsx +190 -0
- package/renderer/components/ui/spinner.tsx +114 -0
- package/renderer/components/ui/tabs.tsx +66 -0
- package/renderer/components/ui/tooltip.tsx +61 -0
- package/renderer/hooks/use-code-copy.ts +88 -0
- package/renderer/hooks/use-openapi-title.ts +44 -0
- package/renderer/lib/api-docs/agent/index.ts +6 -0
- package/renderer/lib/api-docs/agent/indexer.ts +323 -0
- package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
- package/renderer/lib/api-docs/agent/types.ts +116 -0
- package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
- package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
- package/renderer/lib/api-docs/auth/crypto.ts +89 -0
- package/renderer/lib/api-docs/auth/index.ts +4 -0
- package/renderer/lib/api-docs/code-editor/db.ts +164 -0
- package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
- package/renderer/lib/api-docs/code-editor/index.ts +6 -0
- package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
- package/renderer/lib/api-docs/code-editor/types.ts +105 -0
- package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
- package/renderer/lib/api-docs/codegen/har.ts +251 -0
- package/renderer/lib/api-docs/codegen/index.ts +159 -0
- package/renderer/lib/api-docs/factories.ts +151 -0
- package/renderer/lib/api-docs/index.ts +17 -0
- package/renderer/lib/api-docs/mobile-context.tsx +112 -0
- package/renderer/lib/api-docs/navigation-context.tsx +88 -0
- package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
- package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
- package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
- package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
- package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
- package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
- package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
- package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
- package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
- package/renderer/lib/api-docs/playground/context.tsx +107 -0
- package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
- package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
- package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
- package/renderer/lib/api-docs/playground/types.ts +35 -0
- package/renderer/lib/api-docs/types.ts +269 -0
- package/renderer/lib/api-docs/utils.ts +311 -0
- package/renderer/lib/cache.ts +193 -0
- package/renderer/lib/docs/config/index.ts +29 -0
- package/renderer/lib/docs/config/loader.ts +142 -0
- package/renderer/lib/docs/config/schema.ts +298 -0
- package/renderer/lib/docs/index.ts +12 -0
- package/renderer/lib/docs/mdx/compiler.ts +176 -0
- package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
- package/renderer/lib/docs/mdx/index.ts +26 -0
- package/renderer/lib/docs/navigation/generator.ts +348 -0
- package/renderer/lib/docs/navigation/index.ts +12 -0
- package/renderer/lib/docs/navigation/types.ts +123 -0
- package/renderer/lib/docs-navigation-context.tsx +80 -0
- package/renderer/lib/multi-tenant/context.ts +105 -0
- package/renderer/lib/storage/blob.ts +845 -0
- package/renderer/lib/utils.ts +6 -0
- package/renderer/next.config.ts +76 -0
- package/renderer/package.json +66 -0
- package/renderer/postcss.config.mjs +5 -0
- package/renderer/public/assets/images/screenshot.png +0 -0
- package/renderer/public/assets/logo/dark.svg +9 -0
- package/renderer/public/assets/logo/light.svg +9 -0
- package/renderer/public/assets/logo.svg +9 -0
- package/renderer/public/file.svg +1 -0
- package/renderer/public/globe.svg +1 -0
- package/renderer/public/icon.png +0 -0
- package/renderer/public/logo.svg +9 -0
- package/renderer/public/window.svg +1 -0
- package/renderer/tsconfig.json +28 -0
- package/templates/basic/README.md +139 -0
- package/templates/basic/assets/favicon.svg +4 -0
- package/templates/basic/assets/logo.svg +9 -0
- package/templates/basic/docs.json +47 -0
- package/templates/basic/guides/configuration.mdx +149 -0
- package/templates/basic/guides/overview.mdx +96 -0
- package/templates/basic/index.mdx +39 -0
- package/templates/basic/package.json +14 -0
- package/templates/basic/quickstart.mdx +92 -0
- package/templates/basic/vercel.json +6 -0
- package/templates/graphql/README.md +139 -0
- package/templates/graphql/api-reference/schema.graphql +305 -0
- package/templates/graphql/assets/favicon.svg +4 -0
- package/templates/graphql/assets/logo.svg +9 -0
- package/templates/graphql/docs.json +54 -0
- package/templates/graphql/guides/configuration.mdx +149 -0
- package/templates/graphql/guides/overview.mdx +96 -0
- package/templates/graphql/index.mdx +39 -0
- package/templates/graphql/package.json +14 -0
- package/templates/graphql/quickstart.mdx +92 -0
- package/templates/graphql/vercel.json +6 -0
- package/templates/openapi/README.md +139 -0
- package/templates/openapi/api-reference/openapi.json +419 -0
- package/templates/openapi/assets/favicon.svg +4 -0
- package/templates/openapi/assets/logo.svg +9 -0
- package/templates/openapi/docs.json +61 -0
- package/templates/openapi/guides/configuration.mdx +149 -0
- package/templates/openapi/guides/overview.mdx +96 -0
- package/templates/openapi/index.mdx +39 -0
- package/templates/openapi/package.json +14 -0
- package/templates/openapi/quickstart.mdx +92 -0
- package/templates/openapi/vercel.json +6 -0
|
@@ -0,0 +1,143 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.check = check;
|
|
7
|
+
const path_1 = __importDefault(require("path"));
|
|
8
|
+
const fs_extra_1 = __importDefault(require("fs-extra"));
|
|
9
|
+
const config_1 = require("../../config");
|
|
10
|
+
const logger_1 = require("../../utils/logger");
|
|
11
|
+
async function check() {
|
|
12
|
+
const projectRoot = process.cwd();
|
|
13
|
+
let hasErrors = false;
|
|
14
|
+
logger_1.logger.info('Checking documentation project...\n');
|
|
15
|
+
// Check for docs.json
|
|
16
|
+
const configPath = path_1.default.join(projectRoot, 'docs.json');
|
|
17
|
+
if (!fs_extra_1.default.existsSync(configPath)) {
|
|
18
|
+
logger_1.logger.error('✗ docs.json not found');
|
|
19
|
+
process.exit(1);
|
|
20
|
+
}
|
|
21
|
+
logger_1.logger.success('✓ docs.json found');
|
|
22
|
+
// Load and validate config
|
|
23
|
+
let config;
|
|
24
|
+
try {
|
|
25
|
+
config = await (0, config_1.loadConfig)(projectRoot);
|
|
26
|
+
logger_1.logger.success('✓ docs.json is valid JSON');
|
|
27
|
+
}
|
|
28
|
+
catch (error) {
|
|
29
|
+
logger_1.logger.error(`✗ Invalid JSON in docs.json: ${error.message}`);
|
|
30
|
+
process.exit(1);
|
|
31
|
+
}
|
|
32
|
+
// Validate config schema
|
|
33
|
+
const validation = (0, config_1.validateConfig)(config);
|
|
34
|
+
if (validation.valid) {
|
|
35
|
+
logger_1.logger.success('✓ docs.json schema is valid');
|
|
36
|
+
}
|
|
37
|
+
else {
|
|
38
|
+
logger_1.logger.error('✗ docs.json schema validation failed:');
|
|
39
|
+
validation.errors.forEach(err => {
|
|
40
|
+
logger_1.logger.error(` - ${err}`);
|
|
41
|
+
});
|
|
42
|
+
hasErrors = true;
|
|
43
|
+
}
|
|
44
|
+
// Check navigation pages exist
|
|
45
|
+
if (config.navigation) {
|
|
46
|
+
logger_1.logger.info('\nChecking navigation pages...');
|
|
47
|
+
// Handle both old array format and new tabs format
|
|
48
|
+
const groups = [];
|
|
49
|
+
if (Array.isArray(config.navigation)) {
|
|
50
|
+
// Old format: navigation is an array of groups
|
|
51
|
+
groups.push(...config.navigation);
|
|
52
|
+
}
|
|
53
|
+
else if (config.navigation.tabs) {
|
|
54
|
+
// New format: navigation.tabs contains tabs with groups
|
|
55
|
+
for (const tab of config.navigation.tabs) {
|
|
56
|
+
if (tab.groups) {
|
|
57
|
+
groups.push(...tab.groups);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
for (const group of groups) {
|
|
62
|
+
if (group.pages) {
|
|
63
|
+
checkPages(projectRoot, group.pages, (found, pagePath) => {
|
|
64
|
+
if (found) {
|
|
65
|
+
logger_1.logger.success(` ✓ ${pagePath}`);
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
logger_1.logger.error(` ✗ ${pagePath} - file not found`);
|
|
69
|
+
hasErrors = true;
|
|
70
|
+
}
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
// Check for MDX files
|
|
76
|
+
logger_1.logger.info('\nScanning for MDX files...');
|
|
77
|
+
const mdxFiles = findMdxFiles(projectRoot);
|
|
78
|
+
logger_1.logger.info(` Found ${mdxFiles.length} MDX file(s)`);
|
|
79
|
+
// Check for public assets
|
|
80
|
+
const publicDir = path_1.default.join(projectRoot, 'public');
|
|
81
|
+
if (fs_extra_1.default.existsSync(publicDir)) {
|
|
82
|
+
const publicFiles = await fs_extra_1.default.readdir(publicDir);
|
|
83
|
+
logger_1.logger.success(`✓ public/ directory found (${publicFiles.length} files)`);
|
|
84
|
+
}
|
|
85
|
+
else {
|
|
86
|
+
logger_1.logger.warn('⚠ public/ directory not found (optional)');
|
|
87
|
+
}
|
|
88
|
+
// Summary
|
|
89
|
+
console.log('');
|
|
90
|
+
if (hasErrors) {
|
|
91
|
+
logger_1.logger.error('Check completed with errors');
|
|
92
|
+
process.exit(1);
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
logger_1.logger.success('All checks passed!');
|
|
96
|
+
process.exit(0);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Recursively check pages in navigation (handles nested groups)
|
|
101
|
+
*/
|
|
102
|
+
function checkPages(projectRoot, pages, callback) {
|
|
103
|
+
for (const page of pages) {
|
|
104
|
+
if (typeof page === 'string') {
|
|
105
|
+
// Simple page path
|
|
106
|
+
const mdxPath = path_1.default.join(projectRoot, `${page}.mdx`);
|
|
107
|
+
const mdPath = path_1.default.join(projectRoot, `${page}.md`);
|
|
108
|
+
callback(fs_extra_1.default.existsSync(mdxPath) || fs_extra_1.default.existsSync(mdPath), page);
|
|
109
|
+
}
|
|
110
|
+
else if (page.group && page.pages) {
|
|
111
|
+
// Nested group
|
|
112
|
+
checkPages(projectRoot, page.pages, callback);
|
|
113
|
+
}
|
|
114
|
+
else if (page.page) {
|
|
115
|
+
// Page object with explicit page property
|
|
116
|
+
const mdxPath = path_1.default.join(projectRoot, `${page.page}.mdx`);
|
|
117
|
+
const mdPath = path_1.default.join(projectRoot, `${page.page}.md`);
|
|
118
|
+
callback(fs_extra_1.default.existsSync(mdxPath) || fs_extra_1.default.existsSync(mdPath), page.page);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Recursively find all MDX files in a directory
|
|
124
|
+
*/
|
|
125
|
+
function findMdxFiles(dir, files = []) {
|
|
126
|
+
const items = fs_extra_1.default.readdirSync(dir);
|
|
127
|
+
for (const item of items) {
|
|
128
|
+
// Skip common directories
|
|
129
|
+
if (['node_modules', 'dist', '.next', '.git'].includes(item)) {
|
|
130
|
+
continue;
|
|
131
|
+
}
|
|
132
|
+
const fullPath = path_1.default.join(dir, item);
|
|
133
|
+
const stat = fs_extra_1.default.statSync(fullPath);
|
|
134
|
+
if (stat.isDirectory()) {
|
|
135
|
+
findMdxFiles(fullPath, files);
|
|
136
|
+
}
|
|
137
|
+
else if (item.endsWith('.mdx') || item.endsWith('.md')) {
|
|
138
|
+
files.push(fullPath);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
return files;
|
|
142
|
+
}
|
|
143
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"check.js","sourceRoot":"","sources":["../../../src/cli/commands/check.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path';\nimport fs from 'fs-extra';\nimport { loadConfig, validateConfig } from '../../config';\nimport { logger } from '../../utils/logger';\n\nexport async function check(): Promise<void> {\n  const projectRoot = process.cwd();\n  let hasErrors = false;\n  \n  logger.info('Checking documentation project...\\n');\n  \n  // Check for docs.json\n  const configPath = path.join(projectRoot, 'docs.json');\n  if (!fs.existsSync(configPath)) {\n    logger.error('✗ docs.json not found');\n    process.exit(1);\n  }\n  logger.success('✓ docs.json found');\n  \n  // Load and validate config\n  let config: any;\n  try {\n    config = await loadConfig(projectRoot);\n    logger.success('✓ docs.json is valid JSON');\n  } catch (error: any) {\n    logger.error(`✗ Invalid JSON in docs.json: ${error.message}`);\n    process.exit(1);\n  }\n  \n  // Validate config schema\n  const validation = validateConfig(config);\n  if (validation.valid) {\n    logger.success('✓ docs.json schema is valid');\n  } else {\n    logger.error('✗ docs.json schema validation failed:');\n    validation.errors.forEach(err => {\n      logger.error(`  - ${err}`);\n    });\n    hasErrors = true;\n  }\n  \n  // Check navigation pages exist\n  if (config.navigation) {\n    logger.info('\\nChecking navigation pages...');\n    \n    // Handle both old array format and new tabs format\n    const groups: any[] = [];\n    \n    if (Array.isArray(config.navigation)) {\n      // Old format: navigation is an array of groups\n      groups.push(...config.navigation);\n    } else if (config.navigation.tabs) {\n      // New format: navigation.tabs contains tabs with groups\n      for (const tab of config.navigation.tabs) {\n        if (tab.groups) {\n          groups.push(...tab.groups);\n        }\n      }\n    }\n    \n    for (const group of groups) {\n      if (group.pages) {\n        checkPages(projectRoot, group.pages, (found, pagePath) => {\n          if (found) {\n            logger.success(`  ✓ ${pagePath}`);\n          } else {\n            logger.error(`  ✗ ${pagePath} - file not found`);\n            hasErrors = true;\n          }\n        });\n      }\n    }\n  }\n  \n  // Check for MDX files\n  logger.info('\\nScanning for MDX files...');\n  const mdxFiles = findMdxFiles(projectRoot);\n  logger.info(`  Found ${mdxFiles.length} MDX file(s)`);\n  \n  // Check for public assets\n  const publicDir = path.join(projectRoot, 'public');\n  if (fs.existsSync(publicDir)) {\n    const publicFiles = await fs.readdir(publicDir);\n    logger.success(`✓ public/ directory found (${publicFiles.length} files)`);\n  } else {\n    logger.warn('⚠ public/ directory not found (optional)');\n  }\n  \n  // Summary\n  console.log('');\n  if (hasErrors) {\n    logger.error('Check completed with errors');\n    process.exit(1);\n  } else {\n    logger.success('All checks passed!');\n    process.exit(0);\n  }\n}\n\n/**\n * Recursively check pages in navigation (handles nested groups)\n */\nfunction checkPages(\n  projectRoot: string, \n  pages: any[], \n  callback: (found: boolean, pagePath: string) => void\n): void {\n  for (const page of pages) {\n    if (typeof page === 'string') {\n      // Simple page path\n      const mdxPath = path.join(projectRoot, `${page}.mdx`);\n      const mdPath = path.join(projectRoot, `${page}.md`);\n      callback(fs.existsSync(mdxPath) || fs.existsSync(mdPath), page);\n    } else if (page.group && page.pages) {\n      // Nested group\n      checkPages(projectRoot, page.pages, callback);\n    } else if (page.page) {\n      // Page object with explicit page property\n      const mdxPath = path.join(projectRoot, `${page.page}.mdx`);\n      const mdPath = path.join(projectRoot, `${page.page}.md`);\n      callback(fs.existsSync(mdxPath) || fs.existsSync(mdPath), page.page);\n    }\n  }\n}\n\n/**\n * Recursively find all MDX files in a directory\n */\nfunction findMdxFiles(dir: string, files: string[] = []): string[] {\n  const items = fs.readdirSync(dir);\n  \n  for (const item of items) {\n    // Skip common directories\n    if (['node_modules', 'dist', '.next', '.git'].includes(item)) {\n      continue;\n    }\n    \n    const fullPath = path.join(dir, item);\n    const stat = fs.statSync(fullPath);\n    \n    if (stat.isDirectory()) {\n      findMdxFiles(fullPath, files);\n    } else if (item.endsWith('.mdx') || item.endsWith('.md')) {\n      files.push(fullPath);\n    }\n  }\n  \n  return files;\n}\n"]}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
declare const TEMPLATES: {
|
|
2
|
+
readonly basic: {
|
|
3
|
+
readonly name: "Basic";
|
|
4
|
+
readonly description: "Simple documentation site with guides and pages";
|
|
5
|
+
};
|
|
6
|
+
readonly openapi: {
|
|
7
|
+
readonly name: "OpenAPI";
|
|
8
|
+
readonly description: "Documentation with REST API reference (OpenAPI/Swagger)";
|
|
9
|
+
};
|
|
10
|
+
readonly graphql: {
|
|
11
|
+
readonly name: "GraphQL";
|
|
12
|
+
readonly description: "Documentation with GraphQL API playground";
|
|
13
|
+
};
|
|
14
|
+
};
|
|
15
|
+
type TemplateType = keyof typeof TEMPLATES;
|
|
16
|
+
interface CreateOptions {
|
|
17
|
+
template?: TemplateType;
|
|
18
|
+
git?: boolean;
|
|
19
|
+
install?: boolean;
|
|
20
|
+
url?: string;
|
|
21
|
+
subdomain?: string;
|
|
22
|
+
}
|
|
23
|
+
export declare function create(projectDirectory: string | undefined, options: CreateOptions): Promise<void>;
|
|
24
|
+
export {};
|
|
@@ -0,0 +1,387 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.create = create;
|
|
40
|
+
const child_process_1 = require("child_process");
|
|
41
|
+
const path_1 = __importDefault(require("path"));
|
|
42
|
+
const fs_extra_1 = __importDefault(require("fs-extra"));
|
|
43
|
+
const logger_1 = require("../../utils/logger");
|
|
44
|
+
const constants_1 = require("../../constants");
|
|
45
|
+
// Available templates
|
|
46
|
+
const TEMPLATES = {
|
|
47
|
+
basic: {
|
|
48
|
+
name: 'Basic',
|
|
49
|
+
description: 'Simple documentation site with guides and pages',
|
|
50
|
+
},
|
|
51
|
+
openapi: {
|
|
52
|
+
name: 'OpenAPI',
|
|
53
|
+
description: 'Documentation with REST API reference (OpenAPI/Swagger)',
|
|
54
|
+
},
|
|
55
|
+
graphql: {
|
|
56
|
+
name: 'GraphQL',
|
|
57
|
+
description: 'Documentation with GraphQL API playground',
|
|
58
|
+
},
|
|
59
|
+
};
|
|
60
|
+
// Simple prompt helper using readline
|
|
61
|
+
async function prompt(question, defaultValue) {
|
|
62
|
+
const readline = await Promise.resolve().then(() => __importStar(require('readline')));
|
|
63
|
+
const rl = readline.createInterface({
|
|
64
|
+
input: process.stdin,
|
|
65
|
+
output: process.stdout,
|
|
66
|
+
});
|
|
67
|
+
return new Promise((resolve) => {
|
|
68
|
+
const displayQuestion = defaultValue
|
|
69
|
+
? `${question} (${defaultValue}): `
|
|
70
|
+
: `${question}: `;
|
|
71
|
+
rl.question(displayQuestion, (answer) => {
|
|
72
|
+
rl.close();
|
|
73
|
+
resolve(answer.trim() || defaultValue || '');
|
|
74
|
+
});
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
async function promptSelect(question, choices) {
|
|
78
|
+
console.log(`\n${question}\n`);
|
|
79
|
+
choices.forEach((choice, i) => {
|
|
80
|
+
console.log(` ${i + 1}. ${choice.label}`);
|
|
81
|
+
});
|
|
82
|
+
console.log();
|
|
83
|
+
const readline = await Promise.resolve().then(() => __importStar(require('readline')));
|
|
84
|
+
const rl = readline.createInterface({
|
|
85
|
+
input: process.stdin,
|
|
86
|
+
output: process.stdout,
|
|
87
|
+
});
|
|
88
|
+
return new Promise((resolve) => {
|
|
89
|
+
rl.question('Enter number: ', (answer) => {
|
|
90
|
+
rl.close();
|
|
91
|
+
const index = parseInt(answer.trim(), 10) - 1;
|
|
92
|
+
if (index >= 0 && index < choices.length) {
|
|
93
|
+
resolve(choices[index].value);
|
|
94
|
+
}
|
|
95
|
+
else {
|
|
96
|
+
// Default to first choice
|
|
97
|
+
resolve(choices[0].value);
|
|
98
|
+
}
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
async function promptConfirm(question, defaultValue = false) {
|
|
103
|
+
const readline = await Promise.resolve().then(() => __importStar(require('readline')));
|
|
104
|
+
const rl = readline.createInterface({
|
|
105
|
+
input: process.stdin,
|
|
106
|
+
output: process.stdout,
|
|
107
|
+
});
|
|
108
|
+
return new Promise((resolve) => {
|
|
109
|
+
const hint = defaultValue ? '[Y/n]' : '[y/N]';
|
|
110
|
+
rl.question(`${question} ${hint}: `, (answer) => {
|
|
111
|
+
rl.close();
|
|
112
|
+
const normalized = answer.trim().toLowerCase();
|
|
113
|
+
if (normalized === '') {
|
|
114
|
+
resolve(defaultValue);
|
|
115
|
+
}
|
|
116
|
+
else {
|
|
117
|
+
resolve(normalized === 'y' || normalized === 'yes');
|
|
118
|
+
}
|
|
119
|
+
});
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
function validateProjectName(name) {
|
|
123
|
+
if (!name) {
|
|
124
|
+
return { valid: false, error: 'Project name is required' };
|
|
125
|
+
}
|
|
126
|
+
// Check for valid npm package name
|
|
127
|
+
if (!/^[a-z0-9][a-z0-9-._]*$/i.test(name)) {
|
|
128
|
+
return { valid: false, error: 'Invalid project name. Use lowercase letters, numbers, and dashes.' };
|
|
129
|
+
}
|
|
130
|
+
return { valid: true };
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Basic subdomain format validation (server does full validation including blacklist)
|
|
134
|
+
*/
|
|
135
|
+
function isValidSubdomainFormat(subdomain) {
|
|
136
|
+
if (!subdomain) {
|
|
137
|
+
return { valid: false, error: 'Subdomain is required' };
|
|
138
|
+
}
|
|
139
|
+
if (subdomain.length < 3) {
|
|
140
|
+
return { valid: false, error: 'Subdomain must be at least 3 characters' };
|
|
141
|
+
}
|
|
142
|
+
if (subdomain.length > 63) {
|
|
143
|
+
return { valid: false, error: 'Subdomain must be 63 characters or less' };
|
|
144
|
+
}
|
|
145
|
+
if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(subdomain)) {
|
|
146
|
+
return { valid: false, error: 'Subdomain must start and end with alphanumeric, can contain hyphens' };
|
|
147
|
+
}
|
|
148
|
+
if (/--/.test(subdomain)) {
|
|
149
|
+
return { valid: false, error: 'Subdomain cannot contain consecutive hyphens' };
|
|
150
|
+
}
|
|
151
|
+
return { valid: true };
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Check subdomain availability via API (also validates against server blacklist)
|
|
155
|
+
*/
|
|
156
|
+
async function checkSubdomainAvailability(subdomain, apiUrl) {
|
|
157
|
+
try {
|
|
158
|
+
const response = await fetch(`${apiUrl}/api/subdomains/check`, {
|
|
159
|
+
method: 'POST',
|
|
160
|
+
headers: {
|
|
161
|
+
'Content-Type': 'application/json',
|
|
162
|
+
},
|
|
163
|
+
body: JSON.stringify({ subdomain }),
|
|
164
|
+
});
|
|
165
|
+
const result = await response.json();
|
|
166
|
+
return result;
|
|
167
|
+
}
|
|
168
|
+
catch {
|
|
169
|
+
// If API is unavailable, allow proceeding (will fail at registration if invalid)
|
|
170
|
+
return { available: true };
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
function formatProjectName(name) {
|
|
174
|
+
return name
|
|
175
|
+
.replace(/-/g, ' ')
|
|
176
|
+
.replace(/\b\w/g, (char) => char.toUpperCase());
|
|
177
|
+
}
|
|
178
|
+
function getPackageManager() {
|
|
179
|
+
const userAgent = process.env.npm_config_user_agent || '';
|
|
180
|
+
if (userAgent.startsWith('yarn')) {
|
|
181
|
+
return 'yarn';
|
|
182
|
+
}
|
|
183
|
+
if (userAgent.startsWith('pnpm')) {
|
|
184
|
+
return 'pnpm';
|
|
185
|
+
}
|
|
186
|
+
return 'npm';
|
|
187
|
+
}
|
|
188
|
+
async function create(projectDirectory, options) {
|
|
189
|
+
console.log();
|
|
190
|
+
logger_1.logger.info('🐟 Create DevDoc Doc');
|
|
191
|
+
console.log();
|
|
192
|
+
const apiUrl = options.url || process.env.DEVDOC_API_URL || constants_1.DEFAULT_API_URL;
|
|
193
|
+
// Get project name if not provided
|
|
194
|
+
let projectPath = projectDirectory;
|
|
195
|
+
if (!projectPath) {
|
|
196
|
+
projectPath = await prompt('What is your project named?', 'my-docs');
|
|
197
|
+
if (!projectPath) {
|
|
198
|
+
logger_1.logger.error('Project name is required');
|
|
199
|
+
process.exit(1);
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
// Validate project name
|
|
203
|
+
const validation = validateProjectName(projectPath);
|
|
204
|
+
if (!validation.valid) {
|
|
205
|
+
logger_1.logger.error(validation.error || 'Invalid project name');
|
|
206
|
+
process.exit(1);
|
|
207
|
+
}
|
|
208
|
+
// Get template selection if not provided
|
|
209
|
+
let template = options.template || 'basic';
|
|
210
|
+
if (!options.template) {
|
|
211
|
+
const templateChoices = Object.entries(TEMPLATES).map(([key, value]) => ({
|
|
212
|
+
value: key,
|
|
213
|
+
label: `${value.name} - ${value.description}`,
|
|
214
|
+
}));
|
|
215
|
+
template = await promptSelect('Which template would you like to use?', templateChoices);
|
|
216
|
+
}
|
|
217
|
+
// Resolve full path
|
|
218
|
+
const resolvedPath = path_1.default.resolve(projectPath);
|
|
219
|
+
const projectName = path_1.default.basename(resolvedPath);
|
|
220
|
+
const slug = projectName.toLowerCase().replace(/[^a-z0-9]+/g, '-').replace(/^-|-$/g, '');
|
|
221
|
+
// Get subdomain - prompt if not provided
|
|
222
|
+
let subdomain = options.subdomain;
|
|
223
|
+
if (!subdomain) {
|
|
224
|
+
console.log();
|
|
225
|
+
subdomain = await prompt(`Enter subdomain for your docs (${slug}.devdoc.sh)`, slug);
|
|
226
|
+
subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '');
|
|
227
|
+
}
|
|
228
|
+
// Validate subdomain format locally
|
|
229
|
+
const formatCheck = isValidSubdomainFormat(subdomain);
|
|
230
|
+
if (!formatCheck.valid) {
|
|
231
|
+
logger_1.logger.error(formatCheck.error);
|
|
232
|
+
process.exit(1);
|
|
233
|
+
}
|
|
234
|
+
// Check subdomain availability via API (server validates blacklist)
|
|
235
|
+
logger_1.logger.info(`Checking if ${subdomain}.devdoc.sh is available...`);
|
|
236
|
+
const availability = await checkSubdomainAvailability(subdomain, apiUrl);
|
|
237
|
+
if (!availability.available) {
|
|
238
|
+
logger_1.logger.error(availability.error || `Subdomain "${subdomain}" is not available`);
|
|
239
|
+
if (availability.suggestion) {
|
|
240
|
+
logger_1.logger.info(`Suggestion: Try "${availability.suggestion}"`);
|
|
241
|
+
}
|
|
242
|
+
process.exit(1);
|
|
243
|
+
}
|
|
244
|
+
logger_1.logger.success(`✓ ${subdomain}.devdoc.sh is available!`);
|
|
245
|
+
// Check if directory exists
|
|
246
|
+
if (fs_extra_1.default.existsSync(resolvedPath)) {
|
|
247
|
+
const files = fs_extra_1.default.readdirSync(resolvedPath);
|
|
248
|
+
if (files.length > 0) {
|
|
249
|
+
const overwrite = await promptConfirm(`Directory ${projectName} is not empty. Continue anyway?`, false);
|
|
250
|
+
if (!overwrite) {
|
|
251
|
+
logger_1.logger.error('Operation cancelled');
|
|
252
|
+
process.exit(1);
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
const selectedTemplate = TEMPLATES[template];
|
|
257
|
+
console.log();
|
|
258
|
+
logger_1.logger.info(`Creating a new ${selectedTemplate.name} DevDoc documentation site in ${resolvedPath}`);
|
|
259
|
+
console.log();
|
|
260
|
+
// Create project directory
|
|
261
|
+
fs_extra_1.default.ensureDirSync(resolvedPath);
|
|
262
|
+
// Copy template
|
|
263
|
+
const templateDir = path_1.default.join(__dirname, '..', '..', '..', 'templates', template);
|
|
264
|
+
if (!fs_extra_1.default.existsSync(templateDir)) {
|
|
265
|
+
logger_1.logger.error(`Template "${template}" not found`);
|
|
266
|
+
logger_1.logger.debug(`Looked for template at: ${templateDir}`);
|
|
267
|
+
process.exit(1);
|
|
268
|
+
}
|
|
269
|
+
logger_1.logger.info('Copying template files...');
|
|
270
|
+
fs_extra_1.default.copySync(templateDir, resolvedPath, { overwrite: true });
|
|
271
|
+
// Update package.json with project name
|
|
272
|
+
const pkgPath = path_1.default.join(resolvedPath, 'package.json');
|
|
273
|
+
if (fs_extra_1.default.existsSync(pkgPath)) {
|
|
274
|
+
const pkg = fs_extra_1.default.readJsonSync(pkgPath);
|
|
275
|
+
pkg.name = projectName;
|
|
276
|
+
fs_extra_1.default.writeJsonSync(pkgPath, pkg, { spaces: 2 });
|
|
277
|
+
}
|
|
278
|
+
// Update docs.json with project name
|
|
279
|
+
const docsPath = path_1.default.join(resolvedPath, 'docs.json');
|
|
280
|
+
if (fs_extra_1.default.existsSync(docsPath)) {
|
|
281
|
+
const docs = fs_extra_1.default.readJsonSync(docsPath);
|
|
282
|
+
docs.name = formatProjectName(projectName);
|
|
283
|
+
fs_extra_1.default.writeJsonSync(docsPath, docs, { spaces: 2 });
|
|
284
|
+
}
|
|
285
|
+
// Register project with Brainfish API and create .devdoc.json
|
|
286
|
+
const devdocConfigPath = path_1.default.join(resolvedPath, '.devdoc.json');
|
|
287
|
+
logger_1.logger.info('Registering project with Brainfish...');
|
|
288
|
+
try {
|
|
289
|
+
const response = await fetch(`${apiUrl}/api/projects/register`, {
|
|
290
|
+
method: 'POST',
|
|
291
|
+
headers: {
|
|
292
|
+
'Content-Type': 'application/json',
|
|
293
|
+
},
|
|
294
|
+
body: JSON.stringify({
|
|
295
|
+
name: formatProjectName(projectName),
|
|
296
|
+
slug,
|
|
297
|
+
subdomain,
|
|
298
|
+
}),
|
|
299
|
+
});
|
|
300
|
+
if (!response.ok) {
|
|
301
|
+
const errorData = await response.json().catch(() => ({ error: 'Unknown error' }));
|
|
302
|
+
throw new Error(errorData.error || `HTTP ${response.status}`);
|
|
303
|
+
}
|
|
304
|
+
const result = await response.json();
|
|
305
|
+
const devdocConfig = {
|
|
306
|
+
projectId: result.projectId,
|
|
307
|
+
name: formatProjectName(projectName),
|
|
308
|
+
slug: result.slug,
|
|
309
|
+
subdomain: result.subdomain,
|
|
310
|
+
apiKey: result.apiKey,
|
|
311
|
+
createdAt: new Date().toISOString(),
|
|
312
|
+
};
|
|
313
|
+
fs_extra_1.default.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });
|
|
314
|
+
logger_1.logger.success('Project registered - API key saved to .devdoc.json');
|
|
315
|
+
console.log();
|
|
316
|
+
console.log(' URL:', `https://${result.subdomain}.devdoc.sh`);
|
|
317
|
+
console.log(' API Key:', result.apiKey);
|
|
318
|
+
}
|
|
319
|
+
catch (error) {
|
|
320
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
321
|
+
logger_1.logger.warn(`Could not register project: ${message}`);
|
|
322
|
+
logger_1.logger.info('Creating local config - run "devdoc init" when online to register');
|
|
323
|
+
const devdocConfig = {
|
|
324
|
+
projectId: `${slug}-${Math.random().toString(36).substring(2, 8)}`,
|
|
325
|
+
name: formatProjectName(projectName),
|
|
326
|
+
slug: slug,
|
|
327
|
+
subdomain: subdomain,
|
|
328
|
+
createdAt: new Date().toISOString(),
|
|
329
|
+
};
|
|
330
|
+
fs_extra_1.default.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });
|
|
331
|
+
logger_1.logger.success('Created .devdoc.json (offline mode)');
|
|
332
|
+
}
|
|
333
|
+
// Initialize git
|
|
334
|
+
if (options.git !== false) {
|
|
335
|
+
logger_1.logger.info('Initializing git repository...');
|
|
336
|
+
try {
|
|
337
|
+
(0, child_process_1.execSync)('git init', { cwd: resolvedPath, stdio: 'ignore' });
|
|
338
|
+
(0, child_process_1.execSync)('git add -A', { cwd: resolvedPath, stdio: 'ignore' });
|
|
339
|
+
(0, child_process_1.execSync)('git commit -m "Initial commit from devdoc create"', {
|
|
340
|
+
cwd: resolvedPath,
|
|
341
|
+
stdio: 'ignore',
|
|
342
|
+
});
|
|
343
|
+
logger_1.logger.success('Git repository initialized');
|
|
344
|
+
}
|
|
345
|
+
catch {
|
|
346
|
+
logger_1.logger.warn('Could not initialize git repository');
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
// Install dependencies
|
|
350
|
+
if (options.install !== false) {
|
|
351
|
+
logger_1.logger.info('Installing dependencies...');
|
|
352
|
+
try {
|
|
353
|
+
const packageManager = getPackageManager();
|
|
354
|
+
(0, child_process_1.execSync)(`${packageManager} install`, {
|
|
355
|
+
cwd: resolvedPath,
|
|
356
|
+
stdio: 'inherit',
|
|
357
|
+
});
|
|
358
|
+
logger_1.logger.success('Dependencies installed');
|
|
359
|
+
}
|
|
360
|
+
catch {
|
|
361
|
+
logger_1.logger.warn('Could not install dependencies');
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
// Success message
|
|
365
|
+
console.log();
|
|
366
|
+
logger_1.logger.success(`Created ${projectName} at ${resolvedPath}`);
|
|
367
|
+
console.log();
|
|
368
|
+
console.log('Your docs will be available at:');
|
|
369
|
+
console.log(` https://${subdomain}.devdoc.sh`);
|
|
370
|
+
console.log();
|
|
371
|
+
console.log('Inside that directory, you can run several commands:');
|
|
372
|
+
console.log();
|
|
373
|
+
console.log(' npm run dev');
|
|
374
|
+
console.log(' Starts the development server.');
|
|
375
|
+
console.log();
|
|
376
|
+
console.log(' devdoc deploy');
|
|
377
|
+
console.log(' Deploys your documentation to Brainfish.');
|
|
378
|
+
console.log();
|
|
379
|
+
console.log('We suggest that you begin by typing:');
|
|
380
|
+
console.log();
|
|
381
|
+
console.log(` cd ${projectName}`);
|
|
382
|
+
console.log(' npm run dev');
|
|
383
|
+
console.log();
|
|
384
|
+
console.log('Happy documenting! 📚');
|
|
385
|
+
console.log();
|
|
386
|
+
}
|
|
387
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create.js","sourceRoot":"","sources":["../../../src/cli/commands/create.ts"],"names":[],"mappings":"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","sourcesContent":["import { execSync } from 'child_process';\nimport path from 'path';\nimport fs from 'fs-extra';\nimport { logger } from '../../utils/logger';\nimport { DEFAULT_API_URL } from '../../constants';\n\n// Available templates\nconst TEMPLATES = {\n  basic: {\n    name: 'Basic',\n    description: 'Simple documentation site with guides and pages',\n  },\n  openapi: {\n    name: 'OpenAPI',\n    description: 'Documentation with REST API reference (OpenAPI/Swagger)',\n  },\n  graphql: {\n    name: 'GraphQL',\n    description: 'Documentation with GraphQL API playground',\n  },\n} as const;\n\ntype TemplateType = keyof typeof TEMPLATES;\n\ninterface CreateOptions {\n  template?: TemplateType;\n  git?: boolean;\n  install?: boolean;\n  url?: string;\n  subdomain?: string;\n}\n\ninterface RegisterResponse {\n  projectId: string;\n  slug: string;\n  subdomain: string;\n  apiKey: string;\n}\n\ninterface CheckSubdomainResponse {\n  available: boolean;\n  error?: string;\n  suggestion?: string;\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\nasync function promptConfirm(question: string, defaultValue = false): Promise<boolean> {\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 hint = defaultValue ? '[Y/n]' : '[y/N]';\n    rl.question(`${question} ${hint}: `, (answer) => {\n      rl.close();\n      const normalized = answer.trim().toLowerCase();\n      if (normalized === '') {\n        resolve(defaultValue);\n      } else {\n        resolve(normalized === 'y' || normalized === 'yes');\n      }\n    });\n  });\n}\n\nfunction validateProjectName(name: string): { valid: boolean; error?: string } {\n  if (!name) {\n    return { valid: false, error: 'Project name is required' };\n  }\n  \n  // Check for valid npm package name\n  if (!/^[a-z0-9][a-z0-9-._]*$/i.test(name)) {\n    return { valid: false, error: 'Invalid project name. Use lowercase letters, numbers, and dashes.' };\n  }\n  \n  return { valid: true };\n}\n\n/**\n * Basic subdomain format validation (server does full validation including blacklist)\n */\nfunction isValidSubdomainFormat(subdomain: string): { valid: boolean; error?: string } {\n  if (!subdomain) {\n    return { valid: false, error: 'Subdomain is required' };\n  }\n  \n  if (subdomain.length < 3) {\n    return { valid: false, error: 'Subdomain must be at least 3 characters' };\n  }\n  \n  if (subdomain.length > 63) {\n    return { valid: false, error: 'Subdomain must be 63 characters or less' };\n  }\n  \n  if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain must start and end with alphanumeric, can contain hyphens' };\n  }\n  \n  if (/--/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain cannot contain consecutive hyphens' };\n  }\n  \n  return { valid: true };\n}\n\n/**\n * Check subdomain availability via API (also validates against server blacklist)\n */\nasync function checkSubdomainAvailability(\n  subdomain: string,\n  apiUrl: string\n): Promise<CheckSubdomainResponse> {\n  try {\n    const response = await fetch(`${apiUrl}/api/subdomains/check`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify({ subdomain }),\n    });\n    \n    const result = await response.json() as CheckSubdomainResponse;\n    return result;\n  } catch {\n    // If API is unavailable, allow proceeding (will fail at registration if invalid)\n    return { available: true };\n  }\n}\n\nfunction formatProjectName(name: string): string {\n  return name\n    .replace(/-/g, ' ')\n    .replace(/\\b\\w/g, (char) => char.toUpperCase());\n}\n\nfunction getPackageManager(): string {\n  const userAgent = process.env.npm_config_user_agent || '';\n  \n  if (userAgent.startsWith('yarn')) {\n    return 'yarn';\n  }\n  \n  if (userAgent.startsWith('pnpm')) {\n    return 'pnpm';\n  }\n  \n  return 'npm';\n}\n\nexport async function create(projectDirectory: string | undefined, options: CreateOptions): Promise<void> {\n  console.log();\n  logger.info('🐟 Create DevDoc Doc');\n  console.log();\n\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL;\n\n  // Get project name if not provided\n  let projectPath = projectDirectory;\n\n  if (!projectPath) {\n    projectPath = await prompt('What is your project named?', 'my-docs');\n    \n    if (!projectPath) {\n      logger.error('Project name is required');\n      process.exit(1);\n    }\n  }\n\n  // Validate project name\n  const validation = validateProjectName(projectPath);\n  if (!validation.valid) {\n    logger.error(validation.error || 'Invalid project name');\n    process.exit(1);\n  }\n\n  // Get template selection if not provided\n  let template: TemplateType = options.template || 'basic';\n  \n  if (!options.template) {\n    const templateChoices = Object.entries(TEMPLATES).map(([key, value]) => ({\n      value: key,\n      label: `${value.name} - ${value.description}`,\n    }));\n\n    template = await promptSelect('Which template would you like to use?', templateChoices) as TemplateType;\n  }\n\n  // Resolve full path\n  const resolvedPath = path.resolve(projectPath);\n  const projectName = path.basename(resolvedPath);\n  const slug = projectName.toLowerCase().replace(/[^a-z0-9]+/g, '-').replace(/^-|-$/g, '');\n\n  // Get subdomain - prompt if not provided\n  let subdomain = options.subdomain;\n  \n  if (!subdomain) {\n    console.log();\n    subdomain = await prompt(`Enter subdomain for your docs (${slug}.devdoc.sh)`, slug);\n    subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '');\n  }\n\n  // Validate subdomain format locally\n  const formatCheck = isValidSubdomainFormat(subdomain);\n  if (!formatCheck.valid) {\n    logger.error(formatCheck.error!);\n    process.exit(1);\n  }\n\n  // Check subdomain availability via API (server validates blacklist)\n  logger.info(`Checking if ${subdomain}.devdoc.sh is available...`);\n  const availability = await checkSubdomainAvailability(subdomain, apiUrl);\n  \n  if (!availability.available) {\n    logger.error(availability.error || `Subdomain \"${subdomain}\" is not available`);\n    if (availability.suggestion) {\n      logger.info(`Suggestion: Try \"${availability.suggestion}\"`);\n    }\n    process.exit(1);\n  }\n  \n  logger.success(`✓ ${subdomain}.devdoc.sh is available!`);\n\n  // Check if directory exists\n  if (fs.existsSync(resolvedPath)) {\n    const files = fs.readdirSync(resolvedPath);\n    if (files.length > 0) {\n      const overwrite = await promptConfirm(`Directory ${projectName} is not empty. Continue anyway?`, false);\n      \n      if (!overwrite) {\n        logger.error('Operation cancelled');\n        process.exit(1);\n      }\n    }\n  }\n\n  const selectedTemplate = TEMPLATES[template];\n  console.log();\n  logger.info(`Creating a new ${selectedTemplate.name} DevDoc documentation site in ${resolvedPath}`);\n  console.log();\n\n  // Create project directory\n  fs.ensureDirSync(resolvedPath);\n\n  // Copy template\n  const templateDir = path.join(__dirname, '..', '..', '..', 'templates', template);\n  \n  if (!fs.existsSync(templateDir)) {\n    logger.error(`Template \"${template}\" not found`);\n    logger.debug(`Looked for template at: ${templateDir}`);\n    process.exit(1);\n  }\n\n  logger.info('Copying template files...');\n  fs.copySync(templateDir, resolvedPath, { overwrite: true });\n\n  // Update package.json with project name\n  const pkgPath = path.join(resolvedPath, 'package.json');\n  if (fs.existsSync(pkgPath)) {\n    const pkg = fs.readJsonSync(pkgPath);\n    pkg.name = projectName;\n    fs.writeJsonSync(pkgPath, pkg, { spaces: 2 });\n  }\n\n  // Update docs.json with project name\n  const docsPath = path.join(resolvedPath, 'docs.json');\n  if (fs.existsSync(docsPath)) {\n    const docs = fs.readJsonSync(docsPath);\n    docs.name = formatProjectName(projectName);\n    fs.writeJsonSync(docsPath, docs, { spaces: 2 });\n  }\n\n  // Register project with Brainfish API and create .devdoc.json\n  const devdocConfigPath = path.join(resolvedPath, '.devdoc.json');\n  \n  logger.info('Registering project with Brainfish...');\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/projects/register`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify({\n        name: formatProjectName(projectName),\n        slug,\n        subdomain,\n      }),\n    });\n    \n    if (!response.ok) {\n      const errorData = await response.json().catch(() => ({ error: 'Unknown error' })) as { error?: string };\n      throw new Error(errorData.error || `HTTP ${response.status}`);\n    }\n    \n    const result = await response.json() as RegisterResponse;\n    \n    const devdocConfig = {\n      projectId: result.projectId,\n      name: formatProjectName(projectName),\n      slug: result.slug,\n      subdomain: result.subdomain,\n      apiKey: result.apiKey,\n      createdAt: new Date().toISOString(),\n    };\n    fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });\n    logger.success('Project registered - API key saved to .devdoc.json');\n    console.log();\n    console.log('  URL:', `https://${result.subdomain}.devdoc.sh`);\n    console.log('  API Key:', result.apiKey);\n    \n  } catch (error: unknown) {\n    const message = error instanceof Error ? error.message : String(error);\n    logger.warn(`Could not register project: ${message}`);\n    logger.info('Creating local config - run \"devdoc init\" when online to register');\n    \n    const devdocConfig = {\n      projectId: `${slug}-${Math.random().toString(36).substring(2, 8)}`,\n      name: formatProjectName(projectName),\n      slug: slug,\n      subdomain: subdomain,\n      createdAt: new Date().toISOString(),\n    };\n    fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });\n    logger.success('Created .devdoc.json (offline mode)');\n  }\n\n  // Initialize git\n  if (options.git !== false) {\n    logger.info('Initializing git repository...');\n    try {\n      execSync('git init', { cwd: resolvedPath, stdio: 'ignore' });\n      execSync('git add -A', { cwd: resolvedPath, stdio: 'ignore' });\n      execSync('git commit -m \"Initial commit from devdoc create\"', {\n        cwd: resolvedPath,\n        stdio: 'ignore',\n      });\n      logger.success('Git repository initialized');\n    } catch {\n      logger.warn('Could not initialize git repository');\n    }\n  }\n\n  // Install dependencies\n  if (options.install !== false) {\n    logger.info('Installing dependencies...');\n    try {\n      const packageManager = getPackageManager();\n      execSync(`${packageManager} install`, {\n        cwd: resolvedPath,\n        stdio: 'inherit',\n      });\n      logger.success('Dependencies installed');\n    } catch {\n      logger.warn('Could not install dependencies');\n    }\n  }\n\n  // Success message\n  console.log();\n  logger.success(`Created ${projectName} at ${resolvedPath}`);\n  console.log();\n  console.log('Your docs will be available at:');\n  console.log(`  https://${subdomain}.devdoc.sh`);\n  console.log();\n  console.log('Inside that directory, you can run several commands:');\n  console.log();\n  console.log('  npm run dev');\n  console.log('    Starts the development server.');\n  console.log();\n  console.log('  devdoc deploy');\n  console.log('    Deploys your documentation to Brainfish.');\n  console.log();\n  console.log('We suggest that you begin by typing:');\n  console.log();\n  console.log(`  cd ${projectName}`);\n  console.log('  npm run dev');\n  console.log();\n  console.log('Happy documenting! 📚');\n  console.log();\n}\n"]}
|