@brainfish-ai/devdoc 0.1.26 → 0.1.27
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/dist/cli/commands/create.d.ts +3 -0
- package/dist/cli/commands/create.js +162 -16
- package/package.json +1 -1
- package/renderer/app/api/docs/route.ts +6 -2
- package/renderer/app/icon.svg +4 -0
- package/renderer/components/docs-viewer/content/doc-page.tsx +17 -2
- package/renderer/components/docs-viewer/content/not-found-page.tsx +330 -0
- package/renderer/components/docs-viewer/index.tsx +15 -2
- package/renderer/app/favicon.ico +0 -0
|
@@ -2,14 +2,17 @@ declare const TEMPLATES: {
|
|
|
2
2
|
readonly basic: {
|
|
3
3
|
readonly name: "Basic";
|
|
4
4
|
readonly description: "Simple documentation site with guides and pages";
|
|
5
|
+
readonly color: "#10b981";
|
|
5
6
|
};
|
|
6
7
|
readonly openapi: {
|
|
7
8
|
readonly name: "OpenAPI";
|
|
8
9
|
readonly description: "Documentation with REST API reference (OpenAPI/Swagger)";
|
|
10
|
+
readonly color: "#10b981";
|
|
9
11
|
};
|
|
10
12
|
readonly graphql: {
|
|
11
13
|
readonly name: "GraphQL";
|
|
12
14
|
readonly description: "Documentation with GraphQL API playground";
|
|
15
|
+
readonly color: "#e535ab";
|
|
13
16
|
};
|
|
14
17
|
};
|
|
15
18
|
type TemplateType = keyof typeof TEMPLATES;
|
|
@@ -43,21 +43,24 @@ const fs_extra_1 = __importDefault(require("fs-extra"));
|
|
|
43
43
|
const giget_1 = require("giget");
|
|
44
44
|
const logger_1 = require("../../utils/logger");
|
|
45
45
|
const constants_1 = require("../../constants");
|
|
46
|
-
// Remote template repository
|
|
47
|
-
const TEMPLATE_REPO = 'github:brainfish-ai/devdoc/templates';
|
|
48
|
-
// Available
|
|
46
|
+
// Remote template repository - single starter template
|
|
47
|
+
const TEMPLATE_REPO = 'github:brainfish-ai/devdoc/templates/starter';
|
|
48
|
+
// Available template types (all use the same starter template, just different configs)
|
|
49
49
|
const TEMPLATES = {
|
|
50
50
|
basic: {
|
|
51
51
|
name: 'Basic',
|
|
52
52
|
description: 'Simple documentation site with guides and pages',
|
|
53
|
+
color: '#10b981',
|
|
53
54
|
},
|
|
54
55
|
openapi: {
|
|
55
56
|
name: 'OpenAPI',
|
|
56
57
|
description: 'Documentation with REST API reference (OpenAPI/Swagger)',
|
|
58
|
+
color: '#10b981',
|
|
57
59
|
},
|
|
58
60
|
graphql: {
|
|
59
61
|
name: 'GraphQL',
|
|
60
62
|
description: 'Documentation with GraphQL API playground',
|
|
63
|
+
color: '#e535ab',
|
|
61
64
|
},
|
|
62
65
|
};
|
|
63
66
|
// Simple prompt helper using readline
|
|
@@ -188,6 +191,139 @@ function getPackageManager() {
|
|
|
188
191
|
}
|
|
189
192
|
return 'npm';
|
|
190
193
|
}
|
|
194
|
+
/**
|
|
195
|
+
* Generate docs.json configuration based on template type
|
|
196
|
+
*/
|
|
197
|
+
function generateDocsConfig(projectName, templateType) {
|
|
198
|
+
const formattedName = formatProjectName(projectName);
|
|
199
|
+
const template = TEMPLATES[templateType];
|
|
200
|
+
// Base configuration (shared across all templates)
|
|
201
|
+
const baseConfig = {
|
|
202
|
+
$schema: 'https://devdoc.sh/docs.json',
|
|
203
|
+
name: formattedName,
|
|
204
|
+
favicon: '/favicon.svg',
|
|
205
|
+
logo: {
|
|
206
|
+
light: '/logo.svg',
|
|
207
|
+
dark: '/logo.svg',
|
|
208
|
+
},
|
|
209
|
+
colors: {
|
|
210
|
+
primary: template.color,
|
|
211
|
+
},
|
|
212
|
+
navigation: {
|
|
213
|
+
tabs: [],
|
|
214
|
+
global: {
|
|
215
|
+
anchors: [
|
|
216
|
+
{
|
|
217
|
+
anchor: 'GitHub',
|
|
218
|
+
href: 'https://github.com/your-org/your-repo',
|
|
219
|
+
icon: 'github-logo',
|
|
220
|
+
},
|
|
221
|
+
],
|
|
222
|
+
},
|
|
223
|
+
},
|
|
224
|
+
};
|
|
225
|
+
// Documentation tab (always present)
|
|
226
|
+
const docsTab = {
|
|
227
|
+
tab: 'Documentation',
|
|
228
|
+
type: 'docs',
|
|
229
|
+
groups: [
|
|
230
|
+
{
|
|
231
|
+
group: 'Getting Started',
|
|
232
|
+
icon: 'rocket-launch',
|
|
233
|
+
pages: ['index', 'quickstart'],
|
|
234
|
+
},
|
|
235
|
+
{
|
|
236
|
+
group: 'Guides',
|
|
237
|
+
icon: 'book-open',
|
|
238
|
+
pages: ['guides/overview', 'guides/configuration'],
|
|
239
|
+
},
|
|
240
|
+
],
|
|
241
|
+
};
|
|
242
|
+
baseConfig.navigation.tabs.push(docsTab);
|
|
243
|
+
// Add API reference tab based on template type
|
|
244
|
+
if (templateType === 'openapi') {
|
|
245
|
+
baseConfig.navigation.tabs.push({
|
|
246
|
+
tab: 'API Reference',
|
|
247
|
+
type: 'openapi',
|
|
248
|
+
path: '/api-reference',
|
|
249
|
+
versions: [
|
|
250
|
+
{ version: 'v1', spec: 'api-reference/openapi.json', default: true },
|
|
251
|
+
],
|
|
252
|
+
groups: [
|
|
253
|
+
{
|
|
254
|
+
group: 'Overview',
|
|
255
|
+
icon: 'book-open',
|
|
256
|
+
pages: [
|
|
257
|
+
'api-reference/introduction',
|
|
258
|
+
'api-reference/authentication',
|
|
259
|
+
'api-reference/errors',
|
|
260
|
+
],
|
|
261
|
+
},
|
|
262
|
+
],
|
|
263
|
+
});
|
|
264
|
+
// Add API playground config
|
|
265
|
+
baseConfig.api = {
|
|
266
|
+
baseUrl: 'https://api.example.com',
|
|
267
|
+
playground: {
|
|
268
|
+
mode: 'show',
|
|
269
|
+
},
|
|
270
|
+
};
|
|
271
|
+
}
|
|
272
|
+
else if (templateType === 'graphql') {
|
|
273
|
+
baseConfig.navigation.tabs.push({
|
|
274
|
+
tab: 'GraphQL API',
|
|
275
|
+
type: 'graphql',
|
|
276
|
+
path: '/graphql-api',
|
|
277
|
+
schema: 'api-reference/schema.graphql',
|
|
278
|
+
endpoint: 'https://api.example.com/graphql',
|
|
279
|
+
groups: [
|
|
280
|
+
{
|
|
281
|
+
group: 'Overview',
|
|
282
|
+
icon: 'book-open',
|
|
283
|
+
pages: [
|
|
284
|
+
'api-reference/introduction',
|
|
285
|
+
'api-reference/authentication',
|
|
286
|
+
],
|
|
287
|
+
},
|
|
288
|
+
],
|
|
289
|
+
});
|
|
290
|
+
}
|
|
291
|
+
return baseConfig;
|
|
292
|
+
}
|
|
293
|
+
/**
|
|
294
|
+
* Clean up unused API files based on template type
|
|
295
|
+
*/
|
|
296
|
+
function cleanupUnusedFiles(resolvedPath, templateType) {
|
|
297
|
+
const apiRefPath = path_1.default.join(resolvedPath, 'api-reference');
|
|
298
|
+
if (templateType === 'basic') {
|
|
299
|
+
// Remove entire api-reference folder for basic template
|
|
300
|
+
if (fs_extra_1.default.existsSync(apiRefPath)) {
|
|
301
|
+
fs_extra_1.default.removeSync(apiRefPath);
|
|
302
|
+
logger_1.logger.debug('Removed api-reference folder (not needed for basic template)');
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
else if (templateType === 'openapi') {
|
|
306
|
+
// Remove GraphQL schema for OpenAPI template
|
|
307
|
+
const graphqlSchema = path_1.default.join(apiRefPath, 'schema.graphql');
|
|
308
|
+
if (fs_extra_1.default.existsSync(graphqlSchema)) {
|
|
309
|
+
fs_extra_1.default.removeSync(graphqlSchema);
|
|
310
|
+
logger_1.logger.debug('Removed schema.graphql (not needed for OpenAPI template)');
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
else if (templateType === 'graphql') {
|
|
314
|
+
// Remove OpenAPI spec and errors.mdx for GraphQL template
|
|
315
|
+
const openapiSpec = path_1.default.join(apiRefPath, 'openapi.json');
|
|
316
|
+
const errorsFile = path_1.default.join(apiRefPath, 'errors.mdx');
|
|
317
|
+
if (fs_extra_1.default.existsSync(openapiSpec)) {
|
|
318
|
+
fs_extra_1.default.removeSync(openapiSpec);
|
|
319
|
+
logger_1.logger.debug('Removed openapi.json (not needed for GraphQL template)');
|
|
320
|
+
}
|
|
321
|
+
if (fs_extra_1.default.existsSync(errorsFile)) {
|
|
322
|
+
fs_extra_1.default.removeSync(errorsFile);
|
|
323
|
+
logger_1.logger.debug('Removed errors.mdx (not needed for GraphQL template)');
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
}
|
|
191
327
|
async function create(projectDirectory, options) {
|
|
192
328
|
console.log();
|
|
193
329
|
logger_1.logger.info('🐟 Create DevDoc Doc');
|
|
@@ -280,26 +416,25 @@ async function create(projectDirectory, options) {
|
|
|
280
416
|
console.log();
|
|
281
417
|
// Create project directory
|
|
282
418
|
fs_extra_1.default.ensureDirSync(resolvedPath);
|
|
283
|
-
// Download template from remote repository
|
|
419
|
+
// Download template from remote repository (always use starter template)
|
|
284
420
|
logger_1.logger.info('Downloading template...');
|
|
285
421
|
try {
|
|
286
422
|
// Try to download from remote repository first
|
|
287
|
-
|
|
288
|
-
await (0, giget_1.downloadTemplate)(templateSource, {
|
|
423
|
+
await (0, giget_1.downloadTemplate)(TEMPLATE_REPO, {
|
|
289
424
|
dir: resolvedPath,
|
|
290
425
|
force: true,
|
|
291
426
|
});
|
|
292
|
-
logger_1.logger.success(
|
|
427
|
+
logger_1.logger.success('Downloaded starter template');
|
|
293
428
|
}
|
|
294
429
|
catch (downloadError) {
|
|
295
430
|
// Fall back to local templates (for development or offline usage)
|
|
296
431
|
logger_1.logger.debug(`Remote download failed: ${downloadError}`);
|
|
297
432
|
logger_1.logger.info('Falling back to local template...');
|
|
298
|
-
const templateDir = path_1.default.join(__dirname, '..', '..', '..', 'templates',
|
|
433
|
+
const templateDir = path_1.default.join(__dirname, '..', '..', '..', 'templates', 'starter');
|
|
299
434
|
if (!fs_extra_1.default.existsSync(templateDir)) {
|
|
300
|
-
logger_1.logger.error(
|
|
435
|
+
logger_1.logger.error('Template "starter" not found');
|
|
301
436
|
logger_1.logger.debug(`Looked for template at: ${templateDir}`);
|
|
302
|
-
logger_1.logger.debug('Remote source: ' +
|
|
437
|
+
logger_1.logger.debug('Remote source: ' + TEMPLATE_REPO);
|
|
303
438
|
process.exit(1);
|
|
304
439
|
}
|
|
305
440
|
fs_extra_1.default.copySync(templateDir, resolvedPath, { overwrite: true });
|
|
@@ -312,13 +447,24 @@ async function create(projectDirectory, options) {
|
|
|
312
447
|
pkg.name = projectName;
|
|
313
448
|
fs_extra_1.default.writeJsonSync(pkgPath, pkg, { spaces: 2 });
|
|
314
449
|
}
|
|
315
|
-
//
|
|
450
|
+
// Generate docs.json based on template type
|
|
451
|
+
const docsConfig = generateDocsConfig(projectName, template);
|
|
316
452
|
const docsPath = path_1.default.join(resolvedPath, 'docs.json');
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
453
|
+
fs_extra_1.default.writeJsonSync(docsPath, docsConfig, { spaces: 2 });
|
|
454
|
+
logger_1.logger.success(`Generated docs.json for ${selectedTemplate.name} template`);
|
|
455
|
+
// Update theme.json with template color
|
|
456
|
+
const themePath = path_1.default.join(resolvedPath, 'theme.json');
|
|
457
|
+
if (fs_extra_1.default.existsSync(themePath)) {
|
|
458
|
+
const theme = fs_extra_1.default.readJsonSync(themePath);
|
|
459
|
+
theme.colors = {
|
|
460
|
+
primary: selectedTemplate.color,
|
|
461
|
+
primaryLight: selectedTemplate.color,
|
|
462
|
+
primaryDark: selectedTemplate.color,
|
|
463
|
+
};
|
|
464
|
+
fs_extra_1.default.writeJsonSync(themePath, theme, { spaces: 2 });
|
|
321
465
|
}
|
|
466
|
+
// Clean up unused files based on template type
|
|
467
|
+
cleanupUnusedFiles(resolvedPath, template);
|
|
322
468
|
// Create .devdoc.json with subdomain (not registered until deploy)
|
|
323
469
|
// The subdomain will only be reserved when the user actually deploys
|
|
324
470
|
const devdocConfigPath = path_1.default.join(resolvedPath, '.devdoc.json');
|
|
@@ -391,4 +537,4 @@ async function create(projectDirectory, options) {
|
|
|
391
537
|
console.log('Happy documenting! 📚');
|
|
392
538
|
console.log();
|
|
393
539
|
}
|
|
394
|
-
//# 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 { downloadTemplate } from 'giget';\nimport { logger } from '../../utils/logger';\nimport { DEFAULT_API_URL } from '../../constants';\n\n// Remote template repository\nconst TEMPLATE_REPO = 'github:brainfish-ai/devdoc/templates';\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 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  // Loop until we find an available subdomain\n  let subdomainAvailable = false;\n  while (!subdomainAvailable) {\n    logger.info(`Checking if ${subdomain}.devdoc.sh is available...`);\n    const availability = await checkSubdomainAvailability(subdomain, apiUrl);\n    \n    if (!availability.available) {\n      console.log();\n      logger.warn(availability.error || `Subdomain \"${subdomain}\" is not available`);\n      \n      const suggestion = availability.suggestion || `${subdomain}-docs`;\n      console.log();\n      logger.info(`Suggestion: ${suggestion}.devdoc.sh`);\n      console.log();\n      \n      // Prompt for new subdomain\n      subdomain = await prompt('Enter a different subdomain', suggestion);\n      subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '');\n      \n      // Validate format\n      const formatCheck = isValidSubdomainFormat(subdomain);\n      if (!formatCheck.valid) {\n        logger.error(formatCheck.error!);\n        continue;\n      }\n      \n      console.log();\n    } else {\n      subdomainAvailable = true;\n      logger.success(`✓ ${subdomain}.devdoc.sh is available!`);\n    }\n  }\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  // Download template from remote repository\n  logger.info('Downloading template...');\n  \n  try {\n    // Try to download from remote repository first\n    const templateSource = `${TEMPLATE_REPO}/${template}`;\n    await downloadTemplate(templateSource, {\n      dir: resolvedPath,\n      force: true,\n    });\n    logger.success(`Downloaded ${template} template`);\n  } catch (downloadError) {\n    // Fall back to local templates (for development or offline usage)\n    logger.debug(`Remote download failed: ${downloadError}`);\n    logger.info('Falling back to local template...');\n    \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      logger.debug('Remote source: ' + `${TEMPLATE_REPO}/${template}`);\n      process.exit(1);\n    }\n\n    fs.copySync(templateDir, resolvedPath, { overwrite: true });\n    logger.success('Copied local template');\n  }\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  // Create .devdoc.json with subdomain (not registered until deploy)\n  // The subdomain will only be reserved when the user actually deploys\n  const devdocConfigPath = path.join(resolvedPath, '.devdoc.json');\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    // Note: No apiKey - subdomain is not reserved until deploy\n  };\n  fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });\n  logger.success('Created .devdoc.json');\n  console.log();\n  console.log('  Subdomain:', `${subdomain}.devdoc.sh`);\n  console.log('  Status:', 'Not yet deployed (subdomain will be reserved on first deploy)');\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  logger.info('Note: The subdomain will be reserved when you run \"devdoc deploy\"');\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 and claims your subdomain.');\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"]}
|
|
540
|
+
//# 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 { downloadTemplate } from 'giget';\nimport { logger } from '../../utils/logger';\nimport { DEFAULT_API_URL } from '../../constants';\n\n// Remote template repository - single starter template\nconst TEMPLATE_REPO = 'github:brainfish-ai/devdoc/templates/starter';\n\n// Available template types (all use the same starter template, just different configs)\nconst TEMPLATES = {\n  basic: {\n    name: 'Basic',\n    description: 'Simple documentation site with guides and pages',\n    color: '#10b981',\n  },\n  openapi: {\n    name: 'OpenAPI',\n    description: 'Documentation with REST API reference (OpenAPI/Swagger)',\n    color: '#10b981',\n  },\n  graphql: {\n    name: 'GraphQL',\n    description: 'Documentation with GraphQL API playground',\n    color: '#e535ab',\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 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\n/**\n * Generate docs.json configuration based on template type\n */\nfunction generateDocsConfig(projectName: string, templateType: TemplateType): object {\n  const formattedName = formatProjectName(projectName);\n  const template = TEMPLATES[templateType];\n  \n  // Base configuration (shared across all templates)\n  const baseConfig = {\n    $schema: 'https://devdoc.sh/docs.json',\n    name: formattedName,\n    favicon: '/favicon.svg',\n    logo: {\n      light: '/logo.svg',\n      dark: '/logo.svg',\n    },\n    colors: {\n      primary: template.color,\n    },\n    navigation: {\n      tabs: [] as object[],\n      global: {\n        anchors: [\n          {\n            anchor: 'GitHub',\n            href: 'https://github.com/your-org/your-repo',\n            icon: 'github-logo',\n          },\n        ],\n      },\n    },\n  };\n\n  // Documentation tab (always present)\n  const docsTab = {\n    tab: 'Documentation',\n    type: 'docs',\n    groups: [\n      {\n        group: 'Getting Started',\n        icon: 'rocket-launch',\n        pages: ['index', 'quickstart'],\n      },\n      {\n        group: 'Guides',\n        icon: 'book-open',\n        pages: ['guides/overview', 'guides/configuration'],\n      },\n    ],\n  };\n\n  baseConfig.navigation.tabs.push(docsTab);\n\n  // Add API reference tab based on template type\n  if (templateType === 'openapi') {\n    baseConfig.navigation.tabs.push({\n      tab: 'API Reference',\n      type: 'openapi',\n      path: '/api-reference',\n      versions: [\n        { version: 'v1', spec: 'api-reference/openapi.json', default: true },\n      ],\n      groups: [\n        {\n          group: 'Overview',\n          icon: 'book-open',\n          pages: [\n            'api-reference/introduction',\n            'api-reference/authentication',\n            'api-reference/errors',\n          ],\n        },\n      ],\n    });\n\n    // Add API playground config\n    (baseConfig as any).api = {\n      baseUrl: 'https://api.example.com',\n      playground: {\n        mode: 'show',\n      },\n    };\n  } else if (templateType === 'graphql') {\n    baseConfig.navigation.tabs.push({\n      tab: 'GraphQL API',\n      type: 'graphql',\n      path: '/graphql-api',\n      schema: 'api-reference/schema.graphql',\n      endpoint: 'https://api.example.com/graphql',\n      groups: [\n        {\n          group: 'Overview',\n          icon: 'book-open',\n          pages: [\n            'api-reference/introduction',\n            'api-reference/authentication',\n          ],\n        },\n      ],\n    });\n  }\n\n  return baseConfig;\n}\n\n/**\n * Clean up unused API files based on template type\n */\nfunction cleanupUnusedFiles(resolvedPath: string, templateType: TemplateType): void {\n  const apiRefPath = path.join(resolvedPath, 'api-reference');\n  \n  if (templateType === 'basic') {\n    // Remove entire api-reference folder for basic template\n    if (fs.existsSync(apiRefPath)) {\n      fs.removeSync(apiRefPath);\n      logger.debug('Removed api-reference folder (not needed for basic template)');\n    }\n  } else if (templateType === 'openapi') {\n    // Remove GraphQL schema for OpenAPI template\n    const graphqlSchema = path.join(apiRefPath, 'schema.graphql');\n    if (fs.existsSync(graphqlSchema)) {\n      fs.removeSync(graphqlSchema);\n      logger.debug('Removed schema.graphql (not needed for OpenAPI template)');\n    }\n  } else if (templateType === 'graphql') {\n    // Remove OpenAPI spec and errors.mdx for GraphQL template\n    const openapiSpec = path.join(apiRefPath, 'openapi.json');\n    const errorsFile = path.join(apiRefPath, 'errors.mdx');\n    if (fs.existsSync(openapiSpec)) {\n      fs.removeSync(openapiSpec);\n      logger.debug('Removed openapi.json (not needed for GraphQL template)');\n    }\n    if (fs.existsSync(errorsFile)) {\n      fs.removeSync(errorsFile);\n      logger.debug('Removed errors.mdx (not needed for GraphQL template)');\n    }\n  }\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  // Loop until we find an available subdomain\n  let subdomainAvailable = false;\n  while (!subdomainAvailable) {\n    logger.info(`Checking if ${subdomain}.devdoc.sh is available...`);\n    const availability = await checkSubdomainAvailability(subdomain, apiUrl);\n    \n    if (!availability.available) {\n      console.log();\n      logger.warn(availability.error || `Subdomain \"${subdomain}\" is not available`);\n      \n      const suggestion = availability.suggestion || `${subdomain}-docs`;\n      console.log();\n      logger.info(`Suggestion: ${suggestion}.devdoc.sh`);\n      console.log();\n      \n      // Prompt for new subdomain\n      subdomain = await prompt('Enter a different subdomain', suggestion);\n      subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '');\n      \n      // Validate format\n      const formatCheck = isValidSubdomainFormat(subdomain);\n      if (!formatCheck.valid) {\n        logger.error(formatCheck.error!);\n        continue;\n      }\n      \n      console.log();\n    } else {\n      subdomainAvailable = true;\n      logger.success(`✓ ${subdomain}.devdoc.sh is available!`);\n    }\n  }\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  // Download template from remote repository (always use starter template)\n  logger.info('Downloading template...');\n  \n  try {\n    // Try to download from remote repository first\n    await downloadTemplate(TEMPLATE_REPO, {\n      dir: resolvedPath,\n      force: true,\n    });\n    logger.success('Downloaded starter template');\n  } catch (downloadError) {\n    // Fall back to local templates (for development or offline usage)\n    logger.debug(`Remote download failed: ${downloadError}`);\n    logger.info('Falling back to local template...');\n    \n    const templateDir = path.join(__dirname, '..', '..', '..', 'templates', 'starter');\n    \n    if (!fs.existsSync(templateDir)) {\n      logger.error('Template \"starter\" not found');\n      logger.debug(`Looked for template at: ${templateDir}`);\n      logger.debug('Remote source: ' + TEMPLATE_REPO);\n      process.exit(1);\n    }\n\n    fs.copySync(templateDir, resolvedPath, { overwrite: true });\n    logger.success('Copied local template');\n  }\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  // Generate docs.json based on template type\n  const docsConfig = generateDocsConfig(projectName, template);\n  const docsPath = path.join(resolvedPath, 'docs.json');\n  fs.writeJsonSync(docsPath, docsConfig, { spaces: 2 });\n  logger.success(`Generated docs.json for ${selectedTemplate.name} template`);\n\n  // Update theme.json with template color\n  const themePath = path.join(resolvedPath, 'theme.json');\n  if (fs.existsSync(themePath)) {\n    const theme = fs.readJsonSync(themePath);\n    theme.colors = {\n      primary: selectedTemplate.color,\n      primaryLight: selectedTemplate.color,\n      primaryDark: selectedTemplate.color,\n    };\n    fs.writeJsonSync(themePath, theme, { spaces: 2 });\n  }\n\n  // Clean up unused files based on template type\n  cleanupUnusedFiles(resolvedPath, template);\n\n  // Create .devdoc.json with subdomain (not registered until deploy)\n  // The subdomain will only be reserved when the user actually deploys\n  const devdocConfigPath = path.join(resolvedPath, '.devdoc.json');\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    // Note: No apiKey - subdomain is not reserved until deploy\n  };\n  fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });\n  logger.success('Created .devdoc.json');\n  console.log();\n  console.log('  Subdomain:', `${subdomain}.devdoc.sh`);\n  console.log('  Status:', 'Not yet deployed (subdomain will be reserved on first deploy)');\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  logger.info('Note: The subdomain will be reserved when you run \"devdoc deploy\"');\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 and claims your subdomain.');\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"]}
|
package/package.json
CHANGED
|
@@ -29,6 +29,7 @@ const prettyCodeOptions = {
|
|
|
29
29
|
export async function GET(request: NextRequest) {
|
|
30
30
|
const searchParams = request.nextUrl.searchParams
|
|
31
31
|
const slug = searchParams.get('slug')
|
|
32
|
+
const is404Request = searchParams.get('is404') === 'true'
|
|
32
33
|
|
|
33
34
|
if (!slug) {
|
|
34
35
|
return NextResponse.json(
|
|
@@ -42,7 +43,7 @@ export async function GET(request: NextRequest) {
|
|
|
42
43
|
|
|
43
44
|
// If multi-tenant, fetch from Blob Storage
|
|
44
45
|
if (projectSlug && !projectSlug.startsWith('custom:')) {
|
|
45
|
-
return handleMultiTenantDocs(projectSlug, slug)
|
|
46
|
+
return handleMultiTenantDocs(projectSlug, slug, is404Request)
|
|
46
47
|
}
|
|
47
48
|
|
|
48
49
|
try {
|
|
@@ -55,6 +56,8 @@ export async function GET(request: NextRequest) {
|
|
|
55
56
|
}
|
|
56
57
|
|
|
57
58
|
if (!existsSync(fullPath)) {
|
|
59
|
+
// For 404 page requests, return 404 status (not a server error)
|
|
60
|
+
// The client will handle showing the default 404 if custom doesn't exist
|
|
58
61
|
return NextResponse.json(
|
|
59
62
|
{ error: 'Page not found' },
|
|
60
63
|
{ status: 404 }
|
|
@@ -122,7 +125,7 @@ export async function GET(request: NextRequest) {
|
|
|
122
125
|
/**
|
|
123
126
|
* Handle multi-tenant docs request - fetch from Blob Storage
|
|
124
127
|
*/
|
|
125
|
-
async function handleMultiTenantDocs(projectSlug: string, slug: string): Promise<Response> {
|
|
128
|
+
async function handleMultiTenantDocs(projectSlug: string, slug: string, is404Request: boolean = false): Promise<Response> {
|
|
126
129
|
try {
|
|
127
130
|
// Try .mdx then .md
|
|
128
131
|
let fileContent = await getProjectFile(projectSlug, `${slug}.mdx`)
|
|
@@ -131,6 +134,7 @@ async function handleMultiTenantDocs(projectSlug: string, slug: string): Promise
|
|
|
131
134
|
}
|
|
132
135
|
|
|
133
136
|
if (!fileContent) {
|
|
137
|
+
// For 404 page requests, return 404 status
|
|
134
138
|
return NextResponse.json(
|
|
135
139
|
{ error: 'Page not found' },
|
|
136
140
|
{ status: 404 }
|
|
@@ -5,6 +5,7 @@ import { Spinner } from '@phosphor-icons/react'
|
|
|
5
5
|
import { MDXRemote, MDXRemoteSerializeResult } from 'next-mdx-remote'
|
|
6
6
|
import { useDocsNavigation } from '@/lib/docs-navigation-context'
|
|
7
7
|
import { useCodeCopy } from '@/hooks/use-code-copy'
|
|
8
|
+
import { NotFoundPage } from './not-found-page'
|
|
8
9
|
|
|
9
10
|
// Custom Link component for MDX - uses docs navigation context
|
|
10
11
|
function MdxLink({ href, children, ...props }: React.AnchorHTMLAttributes<HTMLAnchorElement>) {
|
|
@@ -225,6 +226,7 @@ const mdxComponents = {
|
|
|
225
226
|
|
|
226
227
|
interface DocPageProps {
|
|
227
228
|
slug: string
|
|
229
|
+
onSearch?: () => void
|
|
228
230
|
}
|
|
229
231
|
|
|
230
232
|
interface DocPageData {
|
|
@@ -242,10 +244,11 @@ interface DocPageData {
|
|
|
242
244
|
mdxSource: MDXRemoteSerializeResult
|
|
243
245
|
}
|
|
244
246
|
|
|
245
|
-
export function DocPage({ slug }: DocPageProps) {
|
|
247
|
+
export function DocPage({ slug, onSearch }: DocPageProps) {
|
|
246
248
|
const [pageData, setPageData] = useState<DocPageData | null>(null)
|
|
247
249
|
const [loading, setLoading] = useState(true)
|
|
248
250
|
const [error, setError] = useState<string | null>(null)
|
|
251
|
+
const [isNotFound, setIsNotFound] = useState(false)
|
|
249
252
|
const contentRef = useRef<HTMLDivElement>(null)
|
|
250
253
|
|
|
251
254
|
// Add copy buttons to code blocks after content loads
|
|
@@ -256,9 +259,15 @@ export function DocPage({ slug }: DocPageProps) {
|
|
|
256
259
|
try {
|
|
257
260
|
setLoading(true)
|
|
258
261
|
setError(null)
|
|
262
|
+
setIsNotFound(false)
|
|
259
263
|
|
|
260
264
|
const response = await fetch(`/api/docs?slug=${encodeURIComponent(slug)}`)
|
|
261
265
|
|
|
266
|
+
if (response.status === 404) {
|
|
267
|
+
setIsNotFound(true)
|
|
268
|
+
return
|
|
269
|
+
}
|
|
270
|
+
|
|
262
271
|
if (!response.ok) {
|
|
263
272
|
throw new Error('Failed to load page')
|
|
264
273
|
}
|
|
@@ -286,11 +295,17 @@ export function DocPage({ slug }: DocPageProps) {
|
|
|
286
295
|
)
|
|
287
296
|
}
|
|
288
297
|
|
|
298
|
+
// Show 404 page for not found errors
|
|
299
|
+
if (isNotFound) {
|
|
300
|
+
return <NotFoundPage slug={slug} onSearch={onSearch} />
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
// Show generic error for other errors
|
|
289
304
|
if (error || !pageData) {
|
|
290
305
|
return (
|
|
291
306
|
<div className="docs-page docs-page-error w-full min-h-[200px]">
|
|
292
307
|
<div className="docs-error text-center py-12">
|
|
293
|
-
<p className="text-destructive">{error || '
|
|
308
|
+
<p className="text-destructive">{error || 'Failed to load page'}</p>
|
|
294
309
|
</div>
|
|
295
310
|
</div>
|
|
296
311
|
)
|
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { useEffect, useState, useCallback, useRef } from 'react'
|
|
4
|
+
import { FileX, House, ArrowLeft, MagnifyingGlass, Spinner } from '@phosphor-icons/react'
|
|
5
|
+
import { Button } from '@/components/ui/button'
|
|
6
|
+
import { useDocsNavigation } from '@/lib/docs-navigation-context'
|
|
7
|
+
import { MDXRemote, MDXRemoteSerializeResult } from 'next-mdx-remote'
|
|
8
|
+
import { useCodeCopy } from '@/hooks/use-code-copy'
|
|
9
|
+
|
|
10
|
+
// Import MDX components for custom 404 pages
|
|
11
|
+
import {
|
|
12
|
+
Note,
|
|
13
|
+
Warning,
|
|
14
|
+
Info,
|
|
15
|
+
Tip,
|
|
16
|
+
Check,
|
|
17
|
+
Error as ErrorCallout,
|
|
18
|
+
Callout,
|
|
19
|
+
Card,
|
|
20
|
+
CardGroup,
|
|
21
|
+
Accordion,
|
|
22
|
+
AccordionGroup,
|
|
23
|
+
Steps,
|
|
24
|
+
Step,
|
|
25
|
+
Tabs,
|
|
26
|
+
Tab,
|
|
27
|
+
CodeGroup,
|
|
28
|
+
Frame,
|
|
29
|
+
Columns,
|
|
30
|
+
Snippet,
|
|
31
|
+
Latex,
|
|
32
|
+
ParamField,
|
|
33
|
+
ResponseField,
|
|
34
|
+
Expandable,
|
|
35
|
+
Iframe,
|
|
36
|
+
Video,
|
|
37
|
+
Loom,
|
|
38
|
+
Image,
|
|
39
|
+
Screenshot,
|
|
40
|
+
Logo,
|
|
41
|
+
Icon,
|
|
42
|
+
Highlight,
|
|
43
|
+
Marker,
|
|
44
|
+
Underline,
|
|
45
|
+
Badge,
|
|
46
|
+
Mermaid,
|
|
47
|
+
PDF,
|
|
48
|
+
Audio,
|
|
49
|
+
Download,
|
|
50
|
+
Hero,
|
|
51
|
+
Pre,
|
|
52
|
+
Tagline,
|
|
53
|
+
Headline,
|
|
54
|
+
Description,
|
|
55
|
+
CommandBox,
|
|
56
|
+
Section,
|
|
57
|
+
Center,
|
|
58
|
+
FeatureGrid,
|
|
59
|
+
FeatureItem,
|
|
60
|
+
ButtonLink,
|
|
61
|
+
Spacer,
|
|
62
|
+
Divider,
|
|
63
|
+
} from '../../docs/mdx/index'
|
|
64
|
+
|
|
65
|
+
// MDX components mapping for custom 404 pages
|
|
66
|
+
const mdxComponents = {
|
|
67
|
+
Note,
|
|
68
|
+
Warning,
|
|
69
|
+
Info,
|
|
70
|
+
Tip,
|
|
71
|
+
Check,
|
|
72
|
+
Error: ErrorCallout,
|
|
73
|
+
Callout,
|
|
74
|
+
Card,
|
|
75
|
+
CardGroup,
|
|
76
|
+
Accordion,
|
|
77
|
+
AccordionGroup,
|
|
78
|
+
Steps,
|
|
79
|
+
Step,
|
|
80
|
+
Tabs,
|
|
81
|
+
Tab,
|
|
82
|
+
CodeGroup,
|
|
83
|
+
Frame,
|
|
84
|
+
Columns,
|
|
85
|
+
Snippet,
|
|
86
|
+
Latex,
|
|
87
|
+
ParamField,
|
|
88
|
+
ResponseField,
|
|
89
|
+
Expandable,
|
|
90
|
+
Iframe,
|
|
91
|
+
Video,
|
|
92
|
+
Loom,
|
|
93
|
+
Image,
|
|
94
|
+
Screenshot,
|
|
95
|
+
Logo,
|
|
96
|
+
Icon,
|
|
97
|
+
Highlight,
|
|
98
|
+
Marker,
|
|
99
|
+
Underline,
|
|
100
|
+
Badge,
|
|
101
|
+
Mermaid,
|
|
102
|
+
PDF,
|
|
103
|
+
Audio,
|
|
104
|
+
Download,
|
|
105
|
+
Hero,
|
|
106
|
+
Pre,
|
|
107
|
+
Tagline,
|
|
108
|
+
Headline,
|
|
109
|
+
Description,
|
|
110
|
+
CommandBox,
|
|
111
|
+
Section,
|
|
112
|
+
Center,
|
|
113
|
+
FeatureGrid,
|
|
114
|
+
FeatureItem,
|
|
115
|
+
ButtonLink,
|
|
116
|
+
Spacer,
|
|
117
|
+
Divider,
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
interface NotFoundPageProps {
|
|
121
|
+
slug?: string
|
|
122
|
+
onGoBack?: () => void
|
|
123
|
+
onSearch?: () => void
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
interface Custom404Data {
|
|
127
|
+
frontmatter: {
|
|
128
|
+
title?: string
|
|
129
|
+
description?: string
|
|
130
|
+
mode?: 'default' | 'wide' | 'custom'
|
|
131
|
+
hideHeader?: boolean
|
|
132
|
+
background?: string
|
|
133
|
+
}
|
|
134
|
+
mdxSource: MDXRemoteSerializeResult
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
export function NotFoundPage({ slug, onGoBack, onSearch }: NotFoundPageProps) {
|
|
138
|
+
const docsNav = useDocsNavigation()
|
|
139
|
+
const [customPage, setCustomPage] = useState<Custom404Data | null>(null)
|
|
140
|
+
const [loading, setLoading] = useState(true)
|
|
141
|
+
const contentRef = useRef<HTMLDivElement>(null)
|
|
142
|
+
|
|
143
|
+
// Add copy buttons to code blocks after content loads
|
|
144
|
+
useCodeCopy(contentRef, [customPage])
|
|
145
|
+
|
|
146
|
+
// Try to load custom 404.mdx page
|
|
147
|
+
useEffect(() => {
|
|
148
|
+
async function loadCustom404() {
|
|
149
|
+
try {
|
|
150
|
+
const response = await fetch('/api/docs?slug=404&is404=true')
|
|
151
|
+
if (response.ok) {
|
|
152
|
+
const data = await response.json()
|
|
153
|
+
setCustomPage(data)
|
|
154
|
+
}
|
|
155
|
+
} catch {
|
|
156
|
+
// No custom 404 page, use default
|
|
157
|
+
} finally {
|
|
158
|
+
setLoading(false)
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
loadCustom404()
|
|
162
|
+
}, [])
|
|
163
|
+
|
|
164
|
+
const handleGoHome = useCallback(() => {
|
|
165
|
+
// Navigate to the first page of the current tab or just switch to first tab
|
|
166
|
+
if (docsNav?.switchToTab) {
|
|
167
|
+
window.location.hash = ''
|
|
168
|
+
window.location.reload()
|
|
169
|
+
} else {
|
|
170
|
+
window.location.hash = ''
|
|
171
|
+
}
|
|
172
|
+
}, [docsNav])
|
|
173
|
+
|
|
174
|
+
const handleGoBack = useCallback(() => {
|
|
175
|
+
if (onGoBack) {
|
|
176
|
+
onGoBack()
|
|
177
|
+
} else {
|
|
178
|
+
window.history.back()
|
|
179
|
+
}
|
|
180
|
+
}, [onGoBack])
|
|
181
|
+
|
|
182
|
+
// Show loading state briefly while checking for custom 404
|
|
183
|
+
if (loading) {
|
|
184
|
+
return (
|
|
185
|
+
<div className="docs-page docs-page-not-found w-full min-h-[200px]">
|
|
186
|
+
<div className="flex items-center justify-center py-12">
|
|
187
|
+
<Spinner className="h-6 w-6 animate-spin text-muted-foreground" />
|
|
188
|
+
</div>
|
|
189
|
+
</div>
|
|
190
|
+
)
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
// Render custom 404 page if available
|
|
194
|
+
if (customPage) {
|
|
195
|
+
const { mode = 'default', hideHeader, background } = customPage.frontmatter
|
|
196
|
+
const isCustomMode = mode === 'custom'
|
|
197
|
+
const showHeader = !hideHeader && !isCustomMode
|
|
198
|
+
|
|
199
|
+
// Custom mode: Full-width layout
|
|
200
|
+
if (isCustomMode) {
|
|
201
|
+
return (
|
|
202
|
+
<div
|
|
203
|
+
ref={contentRef}
|
|
204
|
+
className="docs-page docs-page-not-found docs-content w-full min-h-full"
|
|
205
|
+
style={{ background: background || 'var(--background)' }}
|
|
206
|
+
>
|
|
207
|
+
<div className="docs-custom-content [&>*]:w-full">
|
|
208
|
+
<MDXRemote {...customPage.mdxSource} components={mdxComponents} />
|
|
209
|
+
</div>
|
|
210
|
+
</div>
|
|
211
|
+
)
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
// Default mode with optional header
|
|
215
|
+
const containerClass = mode === 'wide' ? 'max-w-6xl' : 'max-w-4xl'
|
|
216
|
+
|
|
217
|
+
return (
|
|
218
|
+
<div ref={contentRef} className={`docs-page docs-page-not-found docs-content ${containerClass} mx-auto px-4 py-6 sm:px-8 sm:py-8`}>
|
|
219
|
+
{showHeader && customPage.frontmatter.title && (
|
|
220
|
+
<div className="docs-page-header mb-6">
|
|
221
|
+
<h1 className="docs-content-title text-2xl sm:text-3xl font-bold mb-2 text-foreground">
|
|
222
|
+
{customPage.frontmatter.title}
|
|
223
|
+
</h1>
|
|
224
|
+
{customPage.frontmatter.description && (
|
|
225
|
+
<p className="docs-content-description text-lg text-muted-foreground">
|
|
226
|
+
{customPage.frontmatter.description}
|
|
227
|
+
</p>
|
|
228
|
+
)}
|
|
229
|
+
</div>
|
|
230
|
+
)}
|
|
231
|
+
<div className="docs-prose prose prose-sm max-w-none prose-headings:text-foreground prose-p:text-muted-foreground prose-strong:text-foreground">
|
|
232
|
+
<MDXRemote {...customPage.mdxSource} components={mdxComponents} />
|
|
233
|
+
</div>
|
|
234
|
+
|
|
235
|
+
{/* Always show navigation actions for custom 404 */}
|
|
236
|
+
<div className="mt-8 pt-6 border-t border-border">
|
|
237
|
+
<div className="flex flex-col sm:flex-row items-center justify-center gap-3">
|
|
238
|
+
<Button variant="outline" size="sm" onClick={handleGoBack} className="gap-2 min-w-[120px]">
|
|
239
|
+
<ArrowLeft className="w-4 h-4" />
|
|
240
|
+
Go back
|
|
241
|
+
</Button>
|
|
242
|
+
<Button variant="default" size="sm" onClick={handleGoHome} className="gap-2 min-w-[120px]">
|
|
243
|
+
<House className="w-4 h-4" />
|
|
244
|
+
Go home
|
|
245
|
+
</Button>
|
|
246
|
+
{onSearch && (
|
|
247
|
+
<Button variant="outline" size="sm" onClick={onSearch} className="gap-2 min-w-[120px]">
|
|
248
|
+
<MagnifyingGlass className="w-4 h-4" />
|
|
249
|
+
Search
|
|
250
|
+
</Button>
|
|
251
|
+
)}
|
|
252
|
+
</div>
|
|
253
|
+
</div>
|
|
254
|
+
</div>
|
|
255
|
+
)
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
// Default 404 page
|
|
259
|
+
return (
|
|
260
|
+
<div className="docs-page docs-page-not-found w-full min-h-[calc(100vh-200px)] flex items-center justify-center">
|
|
261
|
+
<div className="text-center px-4 py-12 max-w-md mx-auto">
|
|
262
|
+
{/* Icon */}
|
|
263
|
+
<div className="mb-6">
|
|
264
|
+
<div className="inline-flex items-center justify-center w-20 h-20 rounded-full bg-muted/50 text-muted-foreground">
|
|
265
|
+
<FileX className="w-10 h-10" weight="light" />
|
|
266
|
+
</div>
|
|
267
|
+
</div>
|
|
268
|
+
|
|
269
|
+
{/* Title */}
|
|
270
|
+
<h1 className="text-2xl font-semibold text-foreground mb-2">
|
|
271
|
+
Page not found
|
|
272
|
+
</h1>
|
|
273
|
+
|
|
274
|
+
{/* Description */}
|
|
275
|
+
<p className="text-muted-foreground mb-2">
|
|
276
|
+
The page you're looking for doesn't exist or has been moved.
|
|
277
|
+
</p>
|
|
278
|
+
|
|
279
|
+
{/* Show the slug that wasn't found */}
|
|
280
|
+
{slug && (
|
|
281
|
+
<p className="text-sm text-muted-foreground/70 mb-6 font-mono bg-muted/30 px-3 py-1.5 rounded-md inline-block">
|
|
282
|
+
/{slug}
|
|
283
|
+
</p>
|
|
284
|
+
)}
|
|
285
|
+
|
|
286
|
+
{!slug && <div className="mb-6" />}
|
|
287
|
+
|
|
288
|
+
{/* Actions */}
|
|
289
|
+
<div className="flex flex-col sm:flex-row items-center justify-center gap-3">
|
|
290
|
+
<Button
|
|
291
|
+
variant="outline"
|
|
292
|
+
size="sm"
|
|
293
|
+
onClick={handleGoBack}
|
|
294
|
+
className="gap-2 min-w-[120px]"
|
|
295
|
+
>
|
|
296
|
+
<ArrowLeft className="w-4 h-4" />
|
|
297
|
+
Go back
|
|
298
|
+
</Button>
|
|
299
|
+
|
|
300
|
+
<Button
|
|
301
|
+
variant="default"
|
|
302
|
+
size="sm"
|
|
303
|
+
onClick={handleGoHome}
|
|
304
|
+
className="gap-2 min-w-[120px]"
|
|
305
|
+
>
|
|
306
|
+
<House className="w-4 h-4" />
|
|
307
|
+
Go home
|
|
308
|
+
</Button>
|
|
309
|
+
|
|
310
|
+
{onSearch && (
|
|
311
|
+
<Button
|
|
312
|
+
variant="outline"
|
|
313
|
+
size="sm"
|
|
314
|
+
onClick={onSearch}
|
|
315
|
+
className="gap-2 min-w-[120px]"
|
|
316
|
+
>
|
|
317
|
+
<MagnifyingGlass className="w-4 h-4" />
|
|
318
|
+
Search
|
|
319
|
+
</Button>
|
|
320
|
+
)}
|
|
321
|
+
</div>
|
|
322
|
+
|
|
323
|
+
{/* Help text */}
|
|
324
|
+
<p className="text-xs text-muted-foreground/60 mt-8">
|
|
325
|
+
If you believe this is an error, please check the URL or contact the documentation maintainer.
|
|
326
|
+
</p>
|
|
327
|
+
</div>
|
|
328
|
+
</div>
|
|
329
|
+
)
|
|
330
|
+
}
|
|
@@ -4,6 +4,7 @@ import { useEffect, useState, useCallback, useRef } from 'react'
|
|
|
4
4
|
import { Spinner } from '@phosphor-icons/react'
|
|
5
5
|
import type { BrainfishCollection, BrainfishRESTRequest, BrainfishDocGroup } from '@/lib/api-docs/types'
|
|
6
6
|
import { DocsSidebar } from './sidebar'
|
|
7
|
+
import { NotFoundPage } from './content/not-found-page'
|
|
7
8
|
import { ApiPlayground } from './playground'
|
|
8
9
|
import type { DebugContext } from './playground/response-viewer'
|
|
9
10
|
import { RightSidebar } from './sidebar/right-sidebar'
|
|
@@ -391,6 +392,7 @@ function DocsContent() {
|
|
|
391
392
|
const [isVersionLoading, setIsVersionLoading] = useState(false) // For version switch only
|
|
392
393
|
const [error, setError] = useState<string | null>(null)
|
|
393
394
|
const [showAuthModal, setShowAuthModal] = useState(false)
|
|
395
|
+
const [notFoundSlug, setNotFoundSlug] = useState<string | null>(null) // Track 404 for invalid URLs
|
|
394
396
|
|
|
395
397
|
// Prefill context for agent
|
|
396
398
|
const { setPrefill } = usePlaygroundPrefill()
|
|
@@ -481,13 +483,16 @@ function DocsContent() {
|
|
|
481
483
|
setSelectedRequest(request)
|
|
482
484
|
setSelectedDocSection(null)
|
|
483
485
|
setSelectedDocPage(null)
|
|
486
|
+
setNotFoundSlug(null)
|
|
484
487
|
} else {
|
|
485
|
-
// Endpoint not found -
|
|
488
|
+
// Endpoint not found - show 404 page
|
|
486
489
|
setSelectedRequest(null)
|
|
487
490
|
setSelectedDocSection(null)
|
|
488
491
|
setSelectedDocPage(null)
|
|
492
|
+
setNotFoundSlug(`endpoint/${actualId}`)
|
|
489
493
|
}
|
|
490
494
|
} else if (actualType === 'page' && actualId) {
|
|
495
|
+
setNotFoundSlug(null) // Clear not found state - DocPage handles its own 404
|
|
491
496
|
setSelectedDocPage(actualId)
|
|
492
497
|
setSelectedRequest(null)
|
|
493
498
|
setSelectedDocSection(null)
|
|
@@ -515,6 +520,7 @@ function DocsContent() {
|
|
|
515
520
|
setSelectedRequest(request)
|
|
516
521
|
setSelectedDocSection(null)
|
|
517
522
|
setSelectedDocPage(null)
|
|
523
|
+
setNotFoundSlug(null) // Clear 404 state when selecting an endpoint
|
|
518
524
|
updateUrlHash(`endpoint/${request.id}`)
|
|
519
525
|
// Reset tab navigation so the new endpoint can determine its default tab
|
|
520
526
|
resetNavigation()
|
|
@@ -527,6 +533,7 @@ function DocsContent() {
|
|
|
527
533
|
setSelectedDocSection(isIntro ? null : headingId)
|
|
528
534
|
setSelectedRequest(null)
|
|
529
535
|
setSelectedDocPage(null)
|
|
536
|
+
setNotFoundSlug(null) // Clear 404 state
|
|
530
537
|
|
|
531
538
|
updateUrlHash(isIntro ? '' : `doc/${headingId}`)
|
|
532
539
|
|
|
@@ -599,6 +606,7 @@ function DocsContent() {
|
|
|
599
606
|
setSelectedDocPage(pageSlug)
|
|
600
607
|
setSelectedRequest(null)
|
|
601
608
|
setSelectedDocSection(null)
|
|
609
|
+
setNotFoundSlug(null) // Clear 404 state
|
|
602
610
|
updateUrlHash(`page/${pageSlug}`, targetTab)
|
|
603
611
|
switchToDocs()
|
|
604
612
|
|
|
@@ -1162,6 +1170,7 @@ function DocsContent() {
|
|
|
1162
1170
|
selectedApiVersion={selectedApiVersion}
|
|
1163
1171
|
handleApiVersionChange={handleApiVersionChange}
|
|
1164
1172
|
isVersionLoading={isVersionLoading}
|
|
1173
|
+
notFoundSlug={notFoundSlug}
|
|
1165
1174
|
/>
|
|
1166
1175
|
</NavigationProvider>
|
|
1167
1176
|
)
|
|
@@ -1193,6 +1202,7 @@ interface DocsWithModeProps {
|
|
|
1193
1202
|
selectedApiVersion: string | null
|
|
1194
1203
|
handleApiVersionChange: (version: string) => void
|
|
1195
1204
|
isVersionLoading: boolean
|
|
1205
|
+
notFoundSlug: string | null
|
|
1196
1206
|
}
|
|
1197
1207
|
|
|
1198
1208
|
// Mode toggle tabs - switches between Docs, API Client, and Notes
|
|
@@ -1268,6 +1278,7 @@ function DocsWithMode({
|
|
|
1268
1278
|
selectedApiVersion,
|
|
1269
1279
|
handleApiVersionChange,
|
|
1270
1280
|
isVersionLoading,
|
|
1281
|
+
notFoundSlug,
|
|
1271
1282
|
}: DocsWithModeProps) {
|
|
1272
1283
|
const { mode, switchToDocs } = useModeContext()
|
|
1273
1284
|
const { setTheme } = useTheme()
|
|
@@ -1562,7 +1573,9 @@ function DocsWithMode({
|
|
|
1562
1573
|
) : selectedRequest ? (
|
|
1563
1574
|
<RequestDetails request={selectedRequest} />
|
|
1564
1575
|
) : selectedDocPage ? (
|
|
1565
|
-
<DocPage slug={selectedDocPage} />
|
|
1576
|
+
<DocPage slug={selectedDocPage} onSearch={openSearch} />
|
|
1577
|
+
) : notFoundSlug ? (
|
|
1578
|
+
<NotFoundPage slug={notFoundSlug} onSearch={openSearch} />
|
|
1566
1579
|
) : (
|
|
1567
1580
|
<div className="flex-1 flex items-center justify-center bg-background">
|
|
1568
1581
|
<div className="text-center text-muted-foreground">
|
package/renderer/app/favicon.ico
DELETED
|
Binary file
|