@brainfish-ai/devdoc 0.1.48 → 0.1.50

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (46) hide show
  1. package/dist/cli/commands/deploy.js +83 -10
  2. package/dist/cli/commands/domain.js +134 -59
  3. package/package.json +1 -1
  4. package/renderer/app/[...slug]/client.js +17 -0
  5. package/renderer/app/[...slug]/page.js +125 -0
  6. package/renderer/app/api/assets/[...path]/route.js +23 -4
  7. package/renderer/app/api/chat/route.js +188 -25
  8. package/renderer/app/api/collections/route.js +105 -3
  9. package/renderer/app/api/deploy/route.js +7 -3
  10. package/renderer/app/api/domains/add/route.js +118 -40
  11. package/renderer/app/api/domains/remove/route.js +16 -3
  12. package/renderer/app/api/domains/verify/route.js +82 -17
  13. package/renderer/app/api/schema/route.js +12 -11
  14. package/renderer/app/api/suggestions/route.js +98 -10
  15. package/renderer/app/globals.css +33 -0
  16. package/renderer/app/layout.js +83 -8
  17. package/renderer/components/docs/mdx/cards.js +16 -45
  18. package/renderer/components/docs/mdx/file-tree.js +102 -0
  19. package/renderer/components/docs/mdx/index.js +7 -0
  20. package/renderer/components/docs-header.js +11 -11
  21. package/renderer/components/docs-viewer/agent/agent-chat.js +75 -11
  22. package/renderer/components/docs-viewer/agent/messages/assistant-message.js +67 -3
  23. package/renderer/components/docs-viewer/agent/messages/tool-call-display.js +49 -4
  24. package/renderer/components/docs-viewer/content/content-router.js +1 -1
  25. package/renderer/components/docs-viewer/content/doc-page.js +36 -28
  26. package/renderer/components/docs-viewer/index.js +223 -58
  27. package/renderer/components/docs-viewer/playground/graphql-playground.js +131 -33
  28. package/renderer/components/docs-viewer/shared/method-badge.js +11 -2
  29. package/renderer/components/docs-viewer/sidebar/collection-tree.js +44 -6
  30. package/renderer/components/docs-viewer/sidebar/index.js +2 -1
  31. package/renderer/components/docs-viewer/sidebar/right-sidebar.js +3 -1
  32. package/renderer/components/docs-viewer/sidebar/sidebar-item.js +5 -7
  33. package/renderer/hooks/use-route-state.js +44 -56
  34. package/renderer/lib/api-docs/agent/indexer.js +73 -12
  35. package/renderer/lib/api-docs/agent/use-suggestions.js +26 -16
  36. package/renderer/lib/api-docs/code-editor/mode-context.js +16 -18
  37. package/renderer/lib/api-docs/parsers/openapi/transformer.js +8 -1
  38. package/renderer/lib/cache/purge.js +98 -0
  39. package/renderer/lib/docs/config/domain-schema.js +23 -1
  40. package/renderer/lib/docs/config/index.js +1 -1
  41. package/renderer/lib/docs-link-utils.js +146 -0
  42. package/renderer/lib/docs-navigation-context.js +3 -2
  43. package/renderer/lib/docs-navigation.js +50 -41
  44. package/renderer/lib/rate-limit.js +203 -0
  45. package/renderer/lib/storage/blob.js +4 -2
  46. package/renderer/lib/vercel/domains.js +275 -0
@@ -120,16 +120,41 @@ async function domainAdd(customDomain, options) {
120
120
  }
121
121
  logger_1.logger.success(`✓ Domain ${domain} added!`);
122
122
  console.log('');
123
+ // Check if domain is already verified (Vercel may auto-verify if DNS is already configured)
124
+ if (result.verified) {
125
+ console.log(' ' + logger_1.logger.green('Domain is already verified and active!'));
126
+ console.log('');
127
+ console.log(' Your docs will be available at:');
128
+ console.log(` ${logger_1.logger.cyan(`https://${domain}`)}`);
129
+ console.log('');
130
+ return;
131
+ }
123
132
  console.log(' Next, add these DNS records to your domain provider:');
124
133
  console.log('');
125
- console.log(' 1. CNAME Record:');
126
- console.log(` Name: ${result.verification.cname.name}`);
127
- console.log(` Value: ${result.verification.cname.value}`);
128
- console.log('');
129
- console.log(' 2. TXT Record (for verification):');
130
- console.log(` Name: ${result.verification.txt.name}`);
131
- console.log(` Value: ${result.verification.txt.value}`);
132
- console.log('');
134
+ // Handle new Vercel verification format (array of records)
135
+ if (result.verification && Array.isArray(result.verification)) {
136
+ result.verification.forEach((record, index) => {
137
+ console.log(` ${index + 1}. ${record.type} Record:`);
138
+ console.log(` Name: ${record.name}`);
139
+ console.log(` Value: ${record.value}`);
140
+ console.log('');
141
+ });
142
+ }
143
+ // Show any additional instructions from the API
144
+ if (result.instructions && result.instructions.length > 0) {
145
+ // Filter out redundant lines that we've already shown
146
+ const additionalInstructions = result.instructions.filter(line => !line.includes('Add the following') &&
147
+ line.trim() !== '' &&
148
+ !line.match(/^\d+\.\s+(CNAME|TXT|A)\s+Record/));
149
+ if (additionalInstructions.length > 0) {
150
+ additionalInstructions.forEach(line => {
151
+ if (line.trim()) {
152
+ console.log(` ${line}`);
153
+ }
154
+ });
155
+ console.log('');
156
+ }
157
+ }
133
158
  logger_1.logger.info('After adding DNS records, run:');
134
159
  console.log(' devdoc domain verify');
135
160
  console.log('');
@@ -203,18 +228,33 @@ async function domainStatus(options) {
203
228
  console.log(' Also accessible at:');
204
229
  console.log(` ${result.projectUrl}`);
205
230
  }
206
- else if (result.status === 'pending' && result.dnsRecords) {
207
- console.log(' Add these DNS records to your domain:');
208
- console.log('');
209
- console.log(' CNAME Record:');
210
- console.log(` Name: ${result.dnsRecords.cname.name}`);
211
- console.log(` Value: ${result.dnsRecords.cname.value}`);
212
- console.log('');
213
- console.log(' TXT Record:');
214
- console.log(` Name: ${result.dnsRecords.txt.name}`);
215
- console.log(` Value: ${result.dnsRecords.txt.value}`);
216
- console.log('');
217
- logger_1.logger.info('After adding DNS records, run: devdoc domain verify');
231
+ else if (result.status === 'pending') {
232
+ // Handle new Vercel verification format
233
+ if (result.verification && result.verification.length > 0) {
234
+ console.log(' Add these DNS records to your domain:');
235
+ console.log('');
236
+ result.verification.forEach((record, index) => {
237
+ console.log(` ${index + 1}. ${record.type} Record:`);
238
+ console.log(` Name: ${record.name}`);
239
+ console.log(` Value: ${record.value}`);
240
+ console.log('');
241
+ });
242
+ logger_1.logger.info('After adding DNS records, run: devdoc domain verify');
243
+ }
244
+ // Handle legacy DNS records format
245
+ else if (result.dnsRecords) {
246
+ console.log(' Add these DNS records to your domain:');
247
+ console.log('');
248
+ console.log(' CNAME Record:');
249
+ console.log(` Name: ${result.dnsRecords.cname.name}`);
250
+ console.log(` Value: ${result.dnsRecords.cname.value}`);
251
+ console.log('');
252
+ console.log(' TXT Record:');
253
+ console.log(` Name: ${result.dnsRecords.txt.name}`);
254
+ console.log(` Value: ${result.dnsRecords.txt.value}`);
255
+ console.log('');
256
+ logger_1.logger.info('After adding DNS records, run: devdoc domain verify');
257
+ }
218
258
  }
219
259
  else if (result.message) {
220
260
  console.log(` ${result.message}`);
@@ -296,47 +336,82 @@ async function domainVerify(options) {
296
336
  console.log('');
297
337
  console.log(` Domain: ${result.domain}`);
298
338
  console.log('');
299
- // CNAME check
300
- console.log(' CNAME Record:');
301
- if (result.checks.cname.valid) {
302
- console.log(` ${logger_1.logger.green('✓')} Found: ${result.checks.cname.value}`);
303
- }
304
- else if (result.checks.cname.found) {
305
- console.log(` ${logger_1.logger.yellow('!')} Found but incorrect: ${result.checks.cname.value}`);
306
- console.log(` Expected: ${result.checks.cname.expected}`);
339
+ // Handle Vercel verification (new format)
340
+ if (result.verified !== undefined) {
341
+ if (result.verified) {
342
+ console.log(` Status: ${logger_1.logger.green('✓ Verified')}`);
343
+ console.log('');
344
+ logger_1.logger.success('Domain verified!');
345
+ console.log('');
346
+ console.log(` ${result.message}`);
347
+ console.log('');
348
+ console.log(' Your docs will be available at:');
349
+ console.log(` ${logger_1.logger.cyan(`https://${result.domain}`)}`);
350
+ }
351
+ else {
352
+ console.log(` Status: ${logger_1.logger.yellow('⏳ Pending verification')}`);
353
+ console.log('');
354
+ // Show what DNS records are needed
355
+ if (result.verification && result.verification.length > 0) {
356
+ console.log(' Required DNS records:');
357
+ console.log('');
358
+ result.verification.forEach((record, index) => {
359
+ console.log(` ${index + 1}. ${record.type} Record:`);
360
+ console.log(` Name: ${record.name}`);
361
+ console.log(` Value: ${record.value}`);
362
+ console.log('');
363
+ });
364
+ }
365
+ console.log(` ${result.message}`);
366
+ console.log('');
367
+ logger_1.logger.info('DNS changes can take up to 48 hours to propagate.');
368
+ logger_1.logger.info('Run "devdoc domain verify" again later.');
369
+ }
307
370
  }
308
- else {
309
- console.log(` ${logger_1.logger.red('✗')} Not found`);
310
- console.log(` Expected: ${result.checks.cname.expected}`);
311
- }
312
- // TXT check
313
- console.log('');
314
- console.log(' TXT Verification Record:');
315
- if (result.checks.txt.verified) {
316
- console.log(` ${logger_1.logger.green('')} Verified`);
317
- }
318
- else if (result.checks.txt.found) {
319
- console.log(` ${logger_1.logger.yellow('!')} Found but value doesn't match`);
320
- }
321
- else {
322
- console.log(` ${logger_1.logger.red('✗')} Not found`);
323
- console.log(` Expected: ${result.checks.txt.expected}`);
324
- }
325
- console.log('');
326
- if (result.success) {
327
- logger_1.logger.success('DNS verified!');
328
- console.log('');
329
- console.log(` ${result.message}`);
330
- console.log('');
331
- logger_1.logger.info('Run "devdoc domain status" to check SSL provisioning progress.');
332
- }
333
- else {
334
- logger_1.logger.warn('DNS verification incomplete');
371
+ // Handle legacy verification format (for backward compatibility)
372
+ else if (result.checks) {
373
+ // CNAME check
374
+ console.log(' CNAME Record:');
375
+ if (result.checks.cname.valid) {
376
+ console.log(` ${logger_1.logger.green('')} Found: ${result.checks.cname.value}`);
377
+ }
378
+ else if (result.checks.cname.found) {
379
+ console.log(` ${logger_1.logger.yellow('!')} Found but incorrect: ${result.checks.cname.value}`);
380
+ console.log(` Expected: ${result.checks.cname.expected}`);
381
+ }
382
+ else {
383
+ console.log(` ${logger_1.logger.red('✗')} Not found`);
384
+ console.log(` Expected: ${result.checks.cname.expected}`);
385
+ }
386
+ // TXT check
335
387
  console.log('');
336
- console.log(` ${result.message}`);
388
+ console.log(' TXT Verification Record:');
389
+ if (result.checks.txt.verified) {
390
+ console.log(` ${logger_1.logger.green('✓')} Verified`);
391
+ }
392
+ else if (result.checks.txt.found) {
393
+ console.log(` ${logger_1.logger.yellow('!')} Found but value doesn't match`);
394
+ }
395
+ else {
396
+ console.log(` ${logger_1.logger.red('✗')} Not found`);
397
+ console.log(` Expected: ${result.checks.txt.expected}`);
398
+ }
337
399
  console.log('');
338
- logger_1.logger.info('DNS changes can take 1-24 hours to propagate.');
339
- logger_1.logger.info('Run "devdoc domain verify" again later.');
400
+ if (result.success) {
401
+ logger_1.logger.success('DNS verified!');
402
+ console.log('');
403
+ console.log(` ${result.message}`);
404
+ console.log('');
405
+ logger_1.logger.info('Run "devdoc domain status" to check SSL provisioning progress.');
406
+ }
407
+ else {
408
+ logger_1.logger.warn('DNS verification incomplete');
409
+ console.log('');
410
+ console.log(` ${result.message}`);
411
+ console.log('');
412
+ logger_1.logger.info('DNS changes can take 1-24 hours to propagate.');
413
+ logger_1.logger.info('Run "devdoc domain verify" again later.');
414
+ }
340
415
  }
341
416
  console.log('');
342
417
  }
@@ -404,4 +479,4 @@ async function domainRemove(customDomain, options) {
404
479
  process.exit(1);
405
480
  }
406
481
  }
407
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"domain.js","sourceRoot":"","sources":["../../../src/cli/commands/domain.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path'\nimport fs from 'fs-extra'\nimport { logger } from '../../utils/logger'\nimport { DEFAULT_API_URL } from '../../constants'\n\ninterface DomainOptions {\n  url?: string\n  apiKey?: string\n}\n\ninterface DevDocConfig {\n  projectId?: string\n  name?: string\n  slug?: string\n  subdomain?: string\n  apiKey?: string\n  url?: string\n  lastDeployed?: string\n  createdAt?: string\n}\n\ninterface DomainAddResponse {\n  success: boolean\n  domain: string\n  projectSlug: string\n  status: string\n  verification: {\n    cname: { name: string; value: string }\n    txt: { name: string; value: string }\n  }\n  instructions: string[]\n  error?: string\n}\n\ninterface DomainStatusResponse {\n  hasCustomDomain: boolean\n  domain?: string\n  status?: string\n  projectSlug: string\n  projectUrl: string\n  customUrl?: string\n  message?: string\n  dnsRecords?: {\n    cname: { name: string; value: string }\n    txt: { name: string; value: string }\n  }\n  nextStep?: string\n  error?: string\n}\n\ninterface DomainVerifyResponse {\n  success: boolean\n  domain: string\n  status: string\n  message: string\n  checks: {\n    cname: { found: boolean; value: string | null; valid: boolean; expected: string }\n    txt: { found: boolean; verified: boolean; expected: string }\n  }\n  error?: string\n}\n\ninterface DomainRemoveResponse {\n  success: boolean\n  message: string\n  domain: string\n  error?: string\n}\n\n/**\n * Load .devdoc.json and get API key\n */\nfunction loadDevDocConfig(projectRoot: string): DevDocConfig | null {\n  const configPath = path.join(projectRoot, '.devdoc.json')\n  if (!fs.existsSync(configPath)) {\n    return null\n  }\n  try {\n    return fs.readJsonSync(configPath) as DevDocConfig\n  } catch {\n    return null\n  }\n}\n\n/**\n * Get API key from options, env, or config\n */\nfunction getApiKey(options: DomainOptions, config: DevDocConfig | null): string | null {\n  return options.apiKey || process.env.DEVDOC_API_KEY || config?.apiKey || null\n}\n\n/**\n * Validate domain format\n */\nfunction isValidDomainFormat(domain: string): { valid: boolean; error?: string } {\n  if (!domain) {\n    return { valid: false, error: 'Domain is required' }\n  }\n  \n  if (domain.length < 4) {\n    return { valid: false, error: 'Domain must be at least 4 characters' }\n  }\n  \n  // Basic domain format check\n  const domainRegex = /^(?!-)[a-zA-Z0-9-]+(?:\\.[a-zA-Z0-9-]+)*\\.[a-zA-Z]{2,}$/\n  if (!domainRegex.test(domain)) {\n    return { valid: false, error: 'Invalid domain format. Example: docs.example.com' }\n  }\n  \n  // Check for reserved domains\n  const reserved = ['devdoc.sh', 'devdoc.io', 'devdoc.com']\n  if (reserved.some(r => domain === r || domain.endsWith(`.${r}`))) {\n    return { valid: false, error: 'Cannot use a reserved DevDoc domain' }\n  }\n  \n  return { valid: true }\n}\n\n/**\n * Normalize domain (lowercase, remove protocol/path)\n */\nfunction normalizeDomain(domain: string): string {\n  let normalized = domain.toLowerCase().trim()\n  normalized = normalized.replace(/^https?:\\/\\//, '')\n  normalized = normalized.split('/')[0]\n  normalized = normalized.split(':')[0]\n  return normalized\n}\n\n// =============================================================================\n// Domain Add Command\n// =============================================================================\n\n/**\n * Add a custom domain to the project\n */\nexport async function domainAdd(customDomain: string, options: DomainOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Adding custom domain...\\n')\n  \n  // Load config\n  const config = loadDevDocConfig(projectRoot)\n  if (!config) {\n    logger.error('Project not initialized.')\n    console.log('')\n    logger.info('Run \"devdoc init\" to initialize, then \"devdoc deploy\" to deploy.')\n    process.exit(1)\n  }\n  \n  // Get API key - required because custom domains need a deployed project\n  const apiKey = getApiKey(options, config)\n  if (!apiKey) {\n    logger.error('Project not deployed yet.')\n    console.log('')\n    console.log('  Custom domains require a deployed project.')\n    console.log('')\n    console.log('  To set up a custom domain:')\n    console.log('    1. Deploy your project first:  devdoc deploy')\n    console.log('    2. Then add your domain:       devdoc domain add ' + customDomain)\n    console.log('')\n    process.exit(1)\n  }\n  \n  // Normalize and validate domain\n  const domain = normalizeDomain(customDomain)\n  const validation = isValidDomainFormat(domain)\n  if (!validation.valid) {\n    logger.error(validation.error!)\n    process.exit(1)\n  }\n  \n  logger.info(`Domain: ${domain}`)\n  console.log('')\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/domains/add`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${apiKey}`,\n      },\n      body: JSON.stringify({ customDomain: domain }),\n    })\n    \n    const result = await response.json() as DomainAddResponse\n    \n    if (!response.ok || !result.success) {\n      logger.error(result.error || 'Failed to add domain')\n      process.exit(1)\n    }\n    \n    logger.success(`✓ Domain ${domain} added!`)\n    console.log('')\n    console.log('  Next, add these DNS records to your domain provider:')\n    console.log('')\n    console.log('  1. CNAME Record:')\n    console.log(`     Name:  ${result.verification.cname.name}`)\n    console.log(`     Value: ${result.verification.cname.value}`)\n    console.log('')\n    console.log('  2. TXT Record (for verification):')\n    console.log(`     Name:  ${result.verification.txt.name}`)\n    console.log(`     Value: ${result.verification.txt.value}`)\n    console.log('')\n    logger.info('After adding DNS records, run:')\n    console.log('  devdoc domain verify')\n    console.log('')\n    \n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to add domain: ${message}`)\n    process.exit(1)\n  }\n}\n\n// =============================================================================\n// Domain Status Command\n// =============================================================================\n\n/**\n * Check status of custom domain\n */\nexport async function domainStatus(options: DomainOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  // Load config\n  const config = loadDevDocConfig(projectRoot)\n  if (!config) {\n    logger.error('Project not initialized.')\n    console.log('')\n    logger.info('Run \"devdoc init\" to initialize, then \"devdoc deploy\" to deploy.')\n    process.exit(1)\n  }\n  \n  // Get API key\n  const apiKey = getApiKey(options, config)\n  if (!apiKey) {\n    logger.error('Project not deployed yet.')\n    console.log('')\n    logger.info('Deploy your project first with \"devdoc deploy\"')\n    process.exit(1)\n  }\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/domains/status`, {\n      method: 'GET',\n      headers: {\n        'Authorization': `Bearer ${apiKey}`,\n      },\n    })\n    \n    const result = await response.json() as DomainStatusResponse\n    \n    if (!response.ok) {\n      logger.error(result.error || 'Failed to get domain status')\n      process.exit(1)\n    }\n    \n    console.log('')\n    console.log('  Custom Domain Status')\n    console.log('  --------------------')\n    \n    if (!result.hasCustomDomain) {\n      console.log('')\n      console.log('  No custom domain configured.')\n      console.log('')\n      console.log('  Your docs are available at:')\n      console.log(`  ${logger.cyan(result.projectUrl)}`)\n      console.log('')\n      logger.info('To add a custom domain, run:')\n      console.log('  devdoc domain add docs.yourdomain.com')\n      console.log('')\n      return\n    }\n    \n    console.log('')\n    console.log(`  Domain:  ${result.domain}`)\n    console.log(`  Status:  ${getStatusDisplay(result.status || 'unknown')}`)\n    console.log('')\n    \n    if (result.status === 'active') {\n      console.log('  Your docs are available at:')\n      console.log(`  ${logger.cyan(result.customUrl || `https://${result.domain}`)}`)\n      console.log('')\n      console.log('  Also accessible at:')\n      console.log(`  ${result.projectUrl}`)\n    } else if (result.status === 'pending' && result.dnsRecords) {\n      console.log('  Add these DNS records to your domain:')\n      console.log('')\n      console.log('  CNAME Record:')\n      console.log(`    Name:  ${result.dnsRecords.cname.name}`)\n      console.log(`    Value: ${result.dnsRecords.cname.value}`)\n      console.log('')\n      console.log('  TXT Record:')\n      console.log(`    Name:  ${result.dnsRecords.txt.name}`)\n      console.log(`    Value: ${result.dnsRecords.txt.value}`)\n      console.log('')\n      logger.info('After adding DNS records, run: devdoc domain verify')\n    } else if (result.message) {\n      console.log(`  ${result.message}`)\n    }\n    \n    if (result.nextStep) {\n      console.log('')\n      logger.info(`Next: ${result.nextStep}`)\n    }\n    \n    console.log('')\n    \n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to get domain status: ${message}`)\n    process.exit(1)\n  }\n}\n\n/**\n * Get display string for status\n */\nfunction getStatusDisplay(status: string): string {\n  switch (status) {\n    case 'pending':\n      return logger.yellow('⏳ Pending DNS configuration')\n    case 'dns_verified':\n      return logger.cyan('✓ DNS verified, SSL provisioning...')\n    case 'ssl_provisioning':\n      return logger.cyan('🔒 SSL certificate provisioning...')\n    case 'active':\n      return logger.green('✓ Active')\n    case 'error':\n      return logger.red('✗ Error')\n    default:\n      return status\n  }\n}\n\n// =============================================================================\n// Domain Verify Command\n// =============================================================================\n\n/**\n * Verify DNS configuration for custom domain\n */\nexport async function domainVerify(options: DomainOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Verifying DNS configuration...\\n')\n  \n  // Load config\n  const config = loadDevDocConfig(projectRoot)\n  if (!config) {\n    logger.error('Project not initialized.')\n    console.log('')\n    logger.info('Run \"devdoc init\" to initialize, then \"devdoc deploy\" to deploy.')\n    process.exit(1)\n  }\n  \n  // Get API key\n  const apiKey = getApiKey(options, config)\n  if (!apiKey) {\n    logger.error('Project not deployed yet.')\n    console.log('')\n    logger.info('Deploy your project first with \"devdoc deploy\"')\n    process.exit(1)\n  }\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/domains/verify`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${apiKey}`,\n      },\n      body: JSON.stringify({}),\n    })\n    \n    const result = await response.json() as DomainVerifyResponse\n    \n    if (!response.ok) {\n      logger.error(result.error || 'Failed to verify domain')\n      process.exit(1)\n    }\n    \n    console.log('')\n    console.log('  DNS Verification Results')\n    console.log('  ------------------------')\n    console.log('')\n    console.log(`  Domain: ${result.domain}`)\n    console.log('')\n    \n    // CNAME check\n    console.log('  CNAME Record:')\n    if (result.checks.cname.valid) {\n      console.log(`    ${logger.green('✓')} Found: ${result.checks.cname.value}`)\n    } else if (result.checks.cname.found) {\n      console.log(`    ${logger.yellow('!')} Found but incorrect: ${result.checks.cname.value}`)\n      console.log(`      Expected: ${result.checks.cname.expected}`)\n    } else {\n      console.log(`    ${logger.red('✗')} Not found`)\n      console.log(`      Expected: ${result.checks.cname.expected}`)\n    }\n    \n    // TXT check\n    console.log('')\n    console.log('  TXT Verification Record:')\n    if (result.checks.txt.verified) {\n      console.log(`    ${logger.green('✓')} Verified`)\n    } else if (result.checks.txt.found) {\n      console.log(`    ${logger.yellow('!')} Found but value doesn't match`)\n    } else {\n      console.log(`    ${logger.red('✗')} Not found`)\n      console.log(`      Expected: ${result.checks.txt.expected}`)\n    }\n    \n    console.log('')\n    \n    if (result.success) {\n      logger.success('DNS verified!')\n      console.log('')\n      console.log(`  ${result.message}`)\n      console.log('')\n      logger.info('Run \"devdoc domain status\" to check SSL provisioning progress.')\n    } else {\n      logger.warn('DNS verification incomplete')\n      console.log('')\n      console.log(`  ${result.message}`)\n      console.log('')\n      logger.info('DNS changes can take 1-24 hours to propagate.')\n      logger.info('Run \"devdoc domain verify\" again later.')\n    }\n    \n    console.log('')\n    \n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to verify domain: ${message}`)\n    process.exit(1)\n  }\n}\n\n// =============================================================================\n// Domain Remove Command\n// =============================================================================\n\n/**\n * Remove custom domain from project\n */\nexport async function domainRemove(customDomain: string | undefined, options: DomainOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Removing custom domain...\\n')\n  \n  // Load config\n  const config = loadDevDocConfig(projectRoot)\n  if (!config) {\n    logger.error('Project not initialized.')\n    console.log('')\n    logger.info('Run \"devdoc init\" to initialize, then \"devdoc deploy\" to deploy.')\n    process.exit(1)\n  }\n  \n  // Get API key\n  const apiKey = getApiKey(options, config)\n  if (!apiKey) {\n    logger.error('Project not deployed yet.')\n    console.log('')\n    logger.info('Deploy your project first with \"devdoc deploy\"')\n    process.exit(1)\n  }\n  \n  try {\n    const body: Record<string, string> = {}\n    if (customDomain) {\n      body.customDomain = normalizeDomain(customDomain)\n    }\n    \n    const response = await fetch(`${apiUrl}/api/domains/remove`, {\n      method: 'DELETE',\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${apiKey}`,\n      },\n      body: JSON.stringify(body),\n    })\n    \n    const result = await response.json() as DomainRemoveResponse\n    \n    if (!response.ok || !result.success) {\n      logger.error(result.error || 'Failed to remove domain')\n      process.exit(1)\n    }\n    \n    logger.success(`✓ Domain ${result.domain} removed`)\n    console.log('')\n    console.log('  Your docs remain accessible at the default subdomain.')\n    console.log('')\n    logger.info('You can add a new custom domain anytime with:')\n    console.log('  devdoc domain add docs.yourdomain.com')\n    console.log('')\n    \n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to remove domain: ${message}`)\n    process.exit(1)\n  }\n}\n"]}
482
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"domain.js","sourceRoot":"","sources":["../../../src/cli/commands/domain.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path'\nimport fs from 'fs-extra'\nimport { logger } from '../../utils/logger'\nimport { DEFAULT_API_URL } from '../../constants'\n\ninterface DomainOptions {\n  url?: string\n  apiKey?: string\n}\n\ninterface DevDocConfig {\n  projectId?: string\n  name?: string\n  slug?: string\n  subdomain?: string\n  apiKey?: string\n  url?: string\n  lastDeployed?: string\n  createdAt?: string\n}\n\ninterface VerificationRecord {\n  type: string\n  name: string\n  value: string\n}\n\ninterface DomainAddResponse {\n  success: boolean\n  domain: string\n  projectSlug: string\n  status: string\n  verified?: boolean\n  message?: string\n  // New Vercel format - array of verification records\n  verification?: VerificationRecord[]\n  instructions?: string[]\n  // Legacy format (for backward compatibility)\n  legacyVerification?: {\n    cname: { name: string; value: string }\n    txt: { name: string; value: string }\n  }\n  error?: string\n}\n\ninterface DomainStatusResponse {\n  hasCustomDomain: boolean\n  domain?: string\n  status?: string\n  projectSlug: string\n  projectUrl: string\n  customUrl?: string\n  message?: string\n  verification?: VerificationRecord[]\n  dnsRecords?: {\n    cname: { name: string; value: string }\n    txt: { name: string; value: string }\n  }\n  nextStep?: string\n  error?: string\n}\n\ninterface DomainVerifyResponse {\n  success: boolean\n  domain: string\n  status: string\n  message: string\n  verified?: boolean\n  // New Vercel format\n  verification?: VerificationRecord[]\n  instructions?: string[]\n  // Legacy format (for backward compatibility)\n  checks?: {\n    cname: { found: boolean; value: string | null; valid: boolean; expected: string }\n    txt: { found: boolean; verified: boolean; expected: string }\n  }\n  error?: string\n}\n\ninterface DomainRemoveResponse {\n  success: boolean\n  message: string\n  domain: string\n  error?: string\n}\n\n/**\n * Load .devdoc.json and get API key\n */\nfunction loadDevDocConfig(projectRoot: string): DevDocConfig | null {\n  const configPath = path.join(projectRoot, '.devdoc.json')\n  if (!fs.existsSync(configPath)) {\n    return null\n  }\n  try {\n    return fs.readJsonSync(configPath) as DevDocConfig\n  } catch {\n    return null\n  }\n}\n\n/**\n * Get API key from options, env, or config\n */\nfunction getApiKey(options: DomainOptions, config: DevDocConfig | null): string | null {\n  return options.apiKey || process.env.DEVDOC_API_KEY || config?.apiKey || null\n}\n\n/**\n * Validate domain format\n */\nfunction isValidDomainFormat(domain: string): { valid: boolean; error?: string } {\n  if (!domain) {\n    return { valid: false, error: 'Domain is required' }\n  }\n  \n  if (domain.length < 4) {\n    return { valid: false, error: 'Domain must be at least 4 characters' }\n  }\n  \n  // Basic domain format check\n  const domainRegex = /^(?!-)[a-zA-Z0-9-]+(?:\\.[a-zA-Z0-9-]+)*\\.[a-zA-Z]{2,}$/\n  if (!domainRegex.test(domain)) {\n    return { valid: false, error: 'Invalid domain format. Example: docs.example.com' }\n  }\n  \n  // Check for reserved domains\n  const reserved = ['devdoc.sh', 'devdoc.io', 'devdoc.com']\n  if (reserved.some(r => domain === r || domain.endsWith(`.${r}`))) {\n    return { valid: false, error: 'Cannot use a reserved DevDoc domain' }\n  }\n  \n  return { valid: true }\n}\n\n/**\n * Normalize domain (lowercase, remove protocol/path)\n */\nfunction normalizeDomain(domain: string): string {\n  let normalized = domain.toLowerCase().trim()\n  normalized = normalized.replace(/^https?:\\/\\//, '')\n  normalized = normalized.split('/')[0]\n  normalized = normalized.split(':')[0]\n  return normalized\n}\n\n// =============================================================================\n// Domain Add Command\n// =============================================================================\n\n/**\n * Add a custom domain to the project\n */\nexport async function domainAdd(customDomain: string, options: DomainOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Adding custom domain...\\n')\n  \n  // Load config\n  const config = loadDevDocConfig(projectRoot)\n  if (!config) {\n    logger.error('Project not initialized.')\n    console.log('')\n    logger.info('Run \"devdoc init\" to initialize, then \"devdoc deploy\" to deploy.')\n    process.exit(1)\n  }\n  \n  // Get API key - required because custom domains need a deployed project\n  const apiKey = getApiKey(options, config)\n  if (!apiKey) {\n    logger.error('Project not deployed yet.')\n    console.log('')\n    console.log('  Custom domains require a deployed project.')\n    console.log('')\n    console.log('  To set up a custom domain:')\n    console.log('    1. Deploy your project first:  devdoc deploy')\n    console.log('    2. Then add your domain:       devdoc domain add ' + customDomain)\n    console.log('')\n    process.exit(1)\n  }\n  \n  // Normalize and validate domain\n  const domain = normalizeDomain(customDomain)\n  const validation = isValidDomainFormat(domain)\n  if (!validation.valid) {\n    logger.error(validation.error!)\n    process.exit(1)\n  }\n  \n  logger.info(`Domain: ${domain}`)\n  console.log('')\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/domains/add`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${apiKey}`,\n      },\n      body: JSON.stringify({ customDomain: domain }),\n    })\n    \n    const result = await response.json() as DomainAddResponse\n    \n    if (!response.ok || !result.success) {\n      logger.error(result.error || 'Failed to add domain')\n      process.exit(1)\n    }\n    \n    logger.success(`✓ Domain ${domain} added!`)\n    console.log('')\n    \n    // Check if domain is already verified (Vercel may auto-verify if DNS is already configured)\n    if (result.verified) {\n      console.log('  ' + logger.green('Domain is already verified and active!'))\n      console.log('')\n      console.log('  Your docs will be available at:')\n      console.log(`  ${logger.cyan(`https://${domain}`)}`)\n      console.log('')\n      return\n    }\n    \n    console.log('  Next, add these DNS records to your domain provider:')\n    console.log('')\n    \n    // Handle new Vercel verification format (array of records)\n    if (result.verification && Array.isArray(result.verification)) {\n      result.verification.forEach((record, index) => {\n        console.log(`  ${index + 1}. ${record.type} Record:`)\n        console.log(`     Name:  ${record.name}`)\n        console.log(`     Value: ${record.value}`)\n        console.log('')\n      })\n    }\n    \n    // Show any additional instructions from the API\n    if (result.instructions && result.instructions.length > 0) {\n      // Filter out redundant lines that we've already shown\n      const additionalInstructions = result.instructions.filter(line => \n        !line.includes('Add the following') && \n        line.trim() !== '' &&\n        !line.match(/^\\d+\\.\\s+(CNAME|TXT|A)\\s+Record/)\n      )\n      if (additionalInstructions.length > 0) {\n        additionalInstructions.forEach(line => {\n          if (line.trim()) {\n            console.log(`  ${line}`)\n          }\n        })\n        console.log('')\n      }\n    }\n    \n    logger.info('After adding DNS records, run:')\n    console.log('  devdoc domain verify')\n    console.log('')\n    \n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to add domain: ${message}`)\n    process.exit(1)\n  }\n}\n\n// =============================================================================\n// Domain Status Command\n// =============================================================================\n\n/**\n * Check status of custom domain\n */\nexport async function domainStatus(options: DomainOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  // Load config\n  const config = loadDevDocConfig(projectRoot)\n  if (!config) {\n    logger.error('Project not initialized.')\n    console.log('')\n    logger.info('Run \"devdoc init\" to initialize, then \"devdoc deploy\" to deploy.')\n    process.exit(1)\n  }\n  \n  // Get API key\n  const apiKey = getApiKey(options, config)\n  if (!apiKey) {\n    logger.error('Project not deployed yet.')\n    console.log('')\n    logger.info('Deploy your project first with \"devdoc deploy\"')\n    process.exit(1)\n  }\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/domains/status`, {\n      method: 'GET',\n      headers: {\n        'Authorization': `Bearer ${apiKey}`,\n      },\n    })\n    \n    const result = await response.json() as DomainStatusResponse\n    \n    if (!response.ok) {\n      logger.error(result.error || 'Failed to get domain status')\n      process.exit(1)\n    }\n    \n    console.log('')\n    console.log('  Custom Domain Status')\n    console.log('  --------------------')\n    \n    if (!result.hasCustomDomain) {\n      console.log('')\n      console.log('  No custom domain configured.')\n      console.log('')\n      console.log('  Your docs are available at:')\n      console.log(`  ${logger.cyan(result.projectUrl)}`)\n      console.log('')\n      logger.info('To add a custom domain, run:')\n      console.log('  devdoc domain add docs.yourdomain.com')\n      console.log('')\n      return\n    }\n    \n    console.log('')\n    console.log(`  Domain:  ${result.domain}`)\n    console.log(`  Status:  ${getStatusDisplay(result.status || 'unknown')}`)\n    console.log('')\n    \n    if (result.status === 'active') {\n      console.log('  Your docs are available at:')\n      console.log(`  ${logger.cyan(result.customUrl || `https://${result.domain}`)}`)\n      console.log('')\n      console.log('  Also accessible at:')\n      console.log(`  ${result.projectUrl}`)\n    } else if (result.status === 'pending') {\n      // Handle new Vercel verification format\n      if (result.verification && result.verification.length > 0) {\n        console.log('  Add these DNS records to your domain:')\n        console.log('')\n        result.verification.forEach((record, index) => {\n          console.log(`  ${index + 1}. ${record.type} Record:`)\n          console.log(`     Name:  ${record.name}`)\n          console.log(`     Value: ${record.value}`)\n          console.log('')\n        })\n        logger.info('After adding DNS records, run: devdoc domain verify')\n      }\n      // Handle legacy DNS records format\n      else if (result.dnsRecords) {\n        console.log('  Add these DNS records to your domain:')\n        console.log('')\n        console.log('  CNAME Record:')\n        console.log(`    Name:  ${result.dnsRecords.cname.name}`)\n        console.log(`    Value: ${result.dnsRecords.cname.value}`)\n        console.log('')\n        console.log('  TXT Record:')\n        console.log(`    Name:  ${result.dnsRecords.txt.name}`)\n        console.log(`    Value: ${result.dnsRecords.txt.value}`)\n        console.log('')\n        logger.info('After adding DNS records, run: devdoc domain verify')\n      }\n    } else if (result.message) {\n      console.log(`  ${result.message}`)\n    }\n    \n    if (result.nextStep) {\n      console.log('')\n      logger.info(`Next: ${result.nextStep}`)\n    }\n    \n    console.log('')\n    \n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to get domain status: ${message}`)\n    process.exit(1)\n  }\n}\n\n/**\n * Get display string for status\n */\nfunction getStatusDisplay(status: string): string {\n  switch (status) {\n    case 'pending':\n      return logger.yellow('⏳ Pending DNS configuration')\n    case 'dns_verified':\n      return logger.cyan('✓ DNS verified, SSL provisioning...')\n    case 'ssl_provisioning':\n      return logger.cyan('🔒 SSL certificate provisioning...')\n    case 'active':\n      return logger.green('✓ Active')\n    case 'error':\n      return logger.red('✗ Error')\n    default:\n      return status\n  }\n}\n\n// =============================================================================\n// Domain Verify Command\n// =============================================================================\n\n/**\n * Verify DNS configuration for custom domain\n */\nexport async function domainVerify(options: DomainOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Verifying DNS configuration...\\n')\n  \n  // Load config\n  const config = loadDevDocConfig(projectRoot)\n  if (!config) {\n    logger.error('Project not initialized.')\n    console.log('')\n    logger.info('Run \"devdoc init\" to initialize, then \"devdoc deploy\" to deploy.')\n    process.exit(1)\n  }\n  \n  // Get API key\n  const apiKey = getApiKey(options, config)\n  if (!apiKey) {\n    logger.error('Project not deployed yet.')\n    console.log('')\n    logger.info('Deploy your project first with \"devdoc deploy\"')\n    process.exit(1)\n  }\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/domains/verify`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${apiKey}`,\n      },\n      body: JSON.stringify({}),\n    })\n    \n    const result = await response.json() as DomainVerifyResponse\n    \n    if (!response.ok) {\n      logger.error(result.error || 'Failed to verify domain')\n      process.exit(1)\n    }\n    \n    console.log('')\n    console.log('  DNS Verification Results')\n    console.log('  ------------------------')\n    console.log('')\n    console.log(`  Domain: ${result.domain}`)\n    console.log('')\n    \n    // Handle Vercel verification (new format)\n    if (result.verified !== undefined) {\n      if (result.verified) {\n        console.log(`  Status: ${logger.green('✓ Verified')}`)\n        console.log('')\n        logger.success('Domain verified!')\n        console.log('')\n        console.log(`  ${result.message}`)\n        console.log('')\n        console.log('  Your docs will be available at:')\n        console.log(`  ${logger.cyan(`https://${result.domain}`)}`)\n      } else {\n        console.log(`  Status: ${logger.yellow('⏳ Pending verification')}`)\n        console.log('')\n        \n        // Show what DNS records are needed\n        if (result.verification && result.verification.length > 0) {\n          console.log('  Required DNS records:')\n          console.log('')\n          result.verification.forEach((record, index) => {\n            console.log(`  ${index + 1}. ${record.type} Record:`)\n            console.log(`     Name:  ${record.name}`)\n            console.log(`     Value: ${record.value}`)\n            console.log('')\n          })\n        }\n        \n        console.log(`  ${result.message}`)\n        console.log('')\n        logger.info('DNS changes can take up to 48 hours to propagate.')\n        logger.info('Run \"devdoc domain verify\" again later.')\n      }\n    }\n    // Handle legacy verification format (for backward compatibility)\n    else if (result.checks) {\n      // CNAME check\n      console.log('  CNAME Record:')\n      if (result.checks.cname.valid) {\n        console.log(`    ${logger.green('✓')} Found: ${result.checks.cname.value}`)\n      } else if (result.checks.cname.found) {\n        console.log(`    ${logger.yellow('!')} Found but incorrect: ${result.checks.cname.value}`)\n        console.log(`      Expected: ${result.checks.cname.expected}`)\n      } else {\n        console.log(`    ${logger.red('✗')} Not found`)\n        console.log(`      Expected: ${result.checks.cname.expected}`)\n      }\n      \n      // TXT check\n      console.log('')\n      console.log('  TXT Verification Record:')\n      if (result.checks.txt.verified) {\n        console.log(`    ${logger.green('✓')} Verified`)\n      } else if (result.checks.txt.found) {\n        console.log(`    ${logger.yellow('!')} Found but value doesn't match`)\n      } else {\n        console.log(`    ${logger.red('✗')} Not found`)\n        console.log(`      Expected: ${result.checks.txt.expected}`)\n      }\n      \n      console.log('')\n      \n      if (result.success) {\n        logger.success('DNS verified!')\n        console.log('')\n        console.log(`  ${result.message}`)\n        console.log('')\n        logger.info('Run \"devdoc domain status\" to check SSL provisioning progress.')\n      } else {\n        logger.warn('DNS verification incomplete')\n        console.log('')\n        console.log(`  ${result.message}`)\n        console.log('')\n        logger.info('DNS changes can take 1-24 hours to propagate.')\n        logger.info('Run \"devdoc domain verify\" again later.')\n      }\n    }\n    \n    console.log('')\n    \n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to verify domain: ${message}`)\n    process.exit(1)\n  }\n}\n\n// =============================================================================\n// Domain Remove Command\n// =============================================================================\n\n/**\n * Remove custom domain from project\n */\nexport async function domainRemove(customDomain: string | undefined, options: DomainOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Removing custom domain...\\n')\n  \n  // Load config\n  const config = loadDevDocConfig(projectRoot)\n  if (!config) {\n    logger.error('Project not initialized.')\n    console.log('')\n    logger.info('Run \"devdoc init\" to initialize, then \"devdoc deploy\" to deploy.')\n    process.exit(1)\n  }\n  \n  // Get API key\n  const apiKey = getApiKey(options, config)\n  if (!apiKey) {\n    logger.error('Project not deployed yet.')\n    console.log('')\n    logger.info('Deploy your project first with \"devdoc deploy\"')\n    process.exit(1)\n  }\n  \n  try {\n    const body: Record<string, string> = {}\n    if (customDomain) {\n      body.customDomain = normalizeDomain(customDomain)\n    }\n    \n    const response = await fetch(`${apiUrl}/api/domains/remove`, {\n      method: 'DELETE',\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${apiKey}`,\n      },\n      body: JSON.stringify(body),\n    })\n    \n    const result = await response.json() as DomainRemoveResponse\n    \n    if (!response.ok || !result.success) {\n      logger.error(result.error || 'Failed to remove domain')\n      process.exit(1)\n    }\n    \n    logger.success(`✓ Domain ${result.domain} removed`)\n    console.log('')\n    console.log('  Your docs remain accessible at the default subdomain.')\n    console.log('')\n    logger.info('You can add a new custom domain anytime with:')\n    console.log('  devdoc domain add docs.yourdomain.com')\n    console.log('')\n    \n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to remove domain: ${message}`)\n    process.exit(1)\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@brainfish-ai/devdoc",
3
- "version": "0.1.48",
3
+ "version": "0.1.50",
4
4
  "description": "Documentation framework for developers. Write docs in MDX, preview locally, deploy to Brainfish.",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
@@ -0,0 +1,17 @@
1
+ 'use client';
2
+ import { jsx as _jsx } from "react/jsx-runtime";
3
+ import { Docs } from "@/components/docs-viewer";
4
+ import { MobileProvider } from "@/lib/api-docs/mobile-context";
5
+ /**
6
+ * Client component for rendering docs
7
+ * Receives the initial path from the server for hydration
8
+ */ export function DocsClientPage({ initialPath }) {
9
+ return /*#__PURE__*/ _jsx(MobileProvider, {
10
+ children: /*#__PURE__*/ _jsx("div", {
11
+ className: "min-h-screen bg-background",
12
+ children: /*#__PURE__*/ _jsx(Docs, {
13
+ initialPath: initialPath
14
+ })
15
+ })
16
+ });
17
+ }
@@ -0,0 +1,125 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { headers } from 'next/headers';
3
+ import { extractProjectContext } from '@/lib/multi-tenant/context';
4
+ import { getProjectContent } from '@/lib/storage/blob';
5
+ import { DocsClientPage } from './client';
6
+ /**
7
+ * Parse URL path into route components
8
+ */ function parseSlug(slug) {
9
+ if (!slug || slug.length === 0) {
10
+ return {
11
+ tab: null,
12
+ contentType: null,
13
+ contentId: null,
14
+ view: 'docs'
15
+ };
16
+ }
17
+ const [tab, type, ...rest] = slug;
18
+ // Check if last part is a view mode
19
+ const lastPart = rest[rest.length - 1];
20
+ const isViewSuffix = lastPart === 'playground' || lastPart === 'notes';
21
+ const view = isViewSuffix ? lastPart : 'docs';
22
+ // Get content ID (everything between type and view)
23
+ const idParts = isViewSuffix ? rest.slice(0, -1) : rest;
24
+ const contentId = idParts.join('/') || null;
25
+ // Determine content type
26
+ let contentType = null;
27
+ if (type === 'page') contentType = 'page';
28
+ else if (type === 'endpoint') contentType = 'endpoint';
29
+ return {
30
+ tab: tab || null,
31
+ contentType,
32
+ contentId,
33
+ view
34
+ };
35
+ }
36
+ /**
37
+ * Generate dynamic metadata for SEO and social sharing
38
+ */ export async function generateMetadata({ params }) {
39
+ const { slug } = await params;
40
+ const headersList = await headers();
41
+ const hostname = headersList.get('host') || 'localhost';
42
+ const context = extractProjectContext(hostname);
43
+ // Parse the route
44
+ const route = parseSlug(slug);
45
+ // Default metadata
46
+ const defaultMeta = {
47
+ title: 'Documentation',
48
+ description: 'API Documentation and Reference'
49
+ };
50
+ if (!context.isMultiTenant || !context.slug) {
51
+ return defaultMeta;
52
+ }
53
+ try {
54
+ const projectContent = await getProjectContent(context.slug);
55
+ if (!projectContent) {
56
+ return defaultMeta;
57
+ }
58
+ const docsConfig = JSON.parse(projectContent.docsJson);
59
+ const themeConfig = projectContent.themeJson ? JSON.parse(projectContent.themeJson) : null;
60
+ const projectName = docsConfig.name || 'Documentation';
61
+ const projectDescription = docsConfig.description || `${projectName} - API Documentation and Reference`;
62
+ // Build logo URL
63
+ const logoUrl = themeConfig?.logo?.url ? `https://${context.slug}.devdoc.sh/api/assets/${context.slug}${themeConfig.logo.url.replace('/assets', '')}` : null;
64
+ // Generate page-specific title and description
65
+ let pageTitle = projectName;
66
+ let pageDescription = projectDescription;
67
+ if (route.contentType === 'page' && route.contentId) {
68
+ // Find the page in docs config to get its title
69
+ const pageName = route.contentId.split('/').pop()?.replace(/-/g, ' ') || route.contentId;
70
+ pageTitle = `${pageName.charAt(0).toUpperCase() + pageName.slice(1)} | ${projectName}`;
71
+ pageDescription = `${pageName} - ${projectDescription}`;
72
+ } else if (route.contentType === 'endpoint' && route.contentId) {
73
+ // Format endpoint name
74
+ const endpointName = route.contentId.replace(/-/g, ' ');
75
+ pageTitle = `${endpointName.charAt(0).toUpperCase() + endpointName.slice(1)} | ${projectName}`;
76
+ pageDescription = `API endpoint: ${endpointName} - ${projectDescription}`;
77
+ } else if (route.tab) {
78
+ // Tab-level page
79
+ const tabName = route.tab.replace(/-/g, ' ');
80
+ pageTitle = `${tabName.charAt(0).toUpperCase() + tabName.slice(1)} | ${projectName}`;
81
+ }
82
+ return {
83
+ title: pageTitle,
84
+ description: pageDescription,
85
+ openGraph: {
86
+ title: pageTitle,
87
+ description: pageDescription,
88
+ siteName: projectName,
89
+ type: 'website',
90
+ url: `https://${context.slug}.devdoc.sh/${slug?.join('/') || ''}`,
91
+ ...logoUrl && {
92
+ images: [
93
+ {
94
+ url: logoUrl,
95
+ width: 1200,
96
+ height: 630,
97
+ alt: projectName
98
+ }
99
+ ]
100
+ }
101
+ },
102
+ twitter: {
103
+ card: logoUrl ? 'summary_large_image' : 'summary',
104
+ title: pageTitle,
105
+ description: pageDescription,
106
+ ...logoUrl && {
107
+ images: [
108
+ logoUrl
109
+ ]
110
+ }
111
+ }
112
+ };
113
+ } catch (error) {
114
+ console.error('[Metadata] Error:', error);
115
+ return defaultMeta;
116
+ }
117
+ }
118
+ /**
119
+ * Server component that renders the client docs page
120
+ */ export default async function DocsPage({ params }) {
121
+ const { slug } = await params;
122
+ return /*#__PURE__*/ _jsx(DocsClientPage, {
123
+ initialPath: `/${slug?.join('/') || ''}`
124
+ });
125
+ }
@@ -70,11 +70,30 @@ const LOCAL_STORAGE_DIR = path.join(process.cwd(), '.devdoc-storage');
70
70
  status: 404
71
71
  });
72
72
  }
73
- // Redirect to the blob URL (more efficient than proxying)
74
- return NextResponse.redirect(blobs[0].url, {
75
- status: 302,
73
+ // Fetch and proxy the content through Vercel's edge for CDN caching
74
+ // This reduces blob storage data transfer charges as edge caches the response
75
+ const blobResponse = await fetch(blobs[0].url);
76
+ if (!blobResponse.ok) {
77
+ return NextResponse.json({
78
+ error: 'Failed to fetch asset'
79
+ }, {
80
+ status: 502
81
+ });
82
+ }
83
+ const contentType = getContentType(fileName);
84
+ const arrayBuffer = await blobResponse.arrayBuffer();
85
+ return new NextResponse(arrayBuffer, {
86
+ status: 200,
76
87
  headers: {
77
- 'Cache-Control': 'public, max-age=31536000, immutable'
88
+ 'Content-Type': contentType,
89
+ // Aggressive CDN caching for static assets
90
+ // s-maxage: cache at edge for 1 year (assets are immutable by path)
91
+ // max-age: browser cache for 1 day (allows updates on hard refresh)
92
+ 'Cache-Control': 'public, max-age=86400, s-maxage=31536000, immutable',
93
+ // Vercel-specific CDN caching
94
+ 'CDN-Cache-Control': 'public, s-maxage=31536000, immutable',
95
+ // Cache tag for invalidation on redeploy
96
+ 'x-vercel-cache-tags': `assets-${slug}`
78
97
  }
79
98
  });
80
99
  } catch (error) {