@brainfish-ai/devdoc 0.1.50 → 0.1.51

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.
@@ -131,7 +131,7 @@ async function domainAdd(customDomain, options) {
131
131
  }
132
132
  console.log(' Next, add these DNS records to your domain provider:');
133
133
  console.log('');
134
- // Handle new Vercel verification format (array of records)
134
+ // Display verification records from Vercel
135
135
  if (result.verification && Array.isArray(result.verification)) {
136
136
  result.verification.forEach((record, index) => {
137
137
  console.log(` ${index + 1}. ${record.type} Record:`);
@@ -140,21 +140,6 @@ async function domainAdd(customDomain, options) {
140
140
  console.log('');
141
141
  });
142
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
- }
158
143
  logger_1.logger.info('After adding DNS records, run:');
159
144
  console.log(' devdoc domain verify');
160
145
  console.log('');
@@ -229,7 +214,6 @@ async function domainStatus(options) {
229
214
  console.log(` ${result.projectUrl}`);
230
215
  }
231
216
  else if (result.status === 'pending') {
232
- // Handle new Vercel verification format
233
217
  if (result.verification && result.verification.length > 0) {
234
218
  console.log(' Add these DNS records to your domain:');
235
219
  console.log('');
@@ -241,20 +225,6 @@ async function domainStatus(options) {
241
225
  });
242
226
  logger_1.logger.info('After adding DNS records, run: devdoc domain verify');
243
227
  }
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
- }
258
228
  }
259
229
  else if (result.message) {
260
230
  console.log(` ${result.message}`);
@@ -277,11 +247,7 @@ async function domainStatus(options) {
277
247
  function getStatusDisplay(status) {
278
248
  switch (status) {
279
249
  case 'pending':
280
- return logger_1.logger.yellow('⏳ Pending DNS configuration');
281
- case 'dns_verified':
282
- return logger_1.logger.cyan('✓ DNS verified, SSL provisioning...');
283
- case 'ssl_provisioning':
284
- return logger_1.logger.cyan('🔒 SSL certificate provisioning...');
250
+ return logger_1.logger.yellow('⏳ Pending DNS verification');
285
251
  case 'active':
286
252
  return logger_1.logger.green('✓ Active');
287
253
  case 'error':
@@ -336,82 +302,34 @@ async function domainVerify(options) {
336
302
  console.log('');
337
303
  console.log(` Domain: ${result.domain}`);
338
304
  console.log('');
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
- }
370
- }
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
305
+ if (result.verified) {
306
+ console.log(` Status: ${logger_1.logger.green('✓ Verified')}`);
387
307
  console.log('');
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
- }
308
+ logger_1.logger.success('Domain verified!');
399
309
  console.log('');
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}`);
310
+ console.log(` ${result.message}`);
311
+ console.log('');
312
+ console.log(' Your docs will be available at:');
313
+ console.log(` ${logger_1.logger.cyan(`https://${result.domain}`)}`);
314
+ }
315
+ else {
316
+ console.log(` Status: ${logger_1.logger.yellow('⏳ Pending verification')}`);
317
+ console.log('');
318
+ // Show what DNS records are needed
319
+ if (result.verification && result.verification.length > 0) {
320
+ console.log(' Required DNS records:');
411
321
  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.');
322
+ result.verification.forEach((record, index) => {
323
+ console.log(` ${index + 1}. ${record.type} Record:`);
324
+ console.log(` Name: ${record.name}`);
325
+ console.log(` Value: ${record.value}`);
326
+ console.log('');
327
+ });
414
328
  }
329
+ console.log(` ${result.message}`);
330
+ console.log('');
331
+ logger_1.logger.info('DNS changes can take up to 48 hours to propagate.');
332
+ logger_1.logger.info('Run "devdoc domain verify" again later.');
415
333
  }
416
334
  console.log('');
417
335
  }
@@ -479,4 +397,4 @@ async function domainRemove(customDomain, options) {
479
397
  process.exit(1);
480
398
  }
481
399
  }
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"]}
400
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"domain.js","sourceRoot":"","sources":["../../../src/cli/commands/domain.ts"],"names":[],"mappings":";;;;;AAyIA,8BA4FC;AASD,oCA6FC;AAyBD,oCAsFC;AASD,oCA2DC;AA9fD,gDAAuB;AACvB,wDAAyB;AACzB,+CAA2C;AAC3C,+CAAiD;AAmEjD;;GAEG;AACH,SAAS,gBAAgB,CAAC,WAAmB;IAC3C,MAAM,UAAU,GAAG,cAAI,CAAC,IAAI,CAAC,WAAW,EAAE,cAAc,CAAC,CAAA;IACzD,IAAI,CAAC,kBAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;QAC/B,OAAO,IAAI,CAAA;IACb,CAAC;IACD,IAAI,CAAC;QACH,OAAO,kBAAE,CAAC,YAAY,CAAC,UAAU,CAAiB,CAAA;IACpD,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAA;IACb,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,SAAS,CAAC,OAAsB,EAAE,MAA2B;IACpE,OAAO,OAAO,CAAC,MAAM,IAAI,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,MAAM,EAAE,MAAM,IAAI,IAAI,CAAA;AAC/E,CAAC;AAED;;GAEG;AACH,SAAS,mBAAmB,CAAC,MAAc;IACzC,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,oBAAoB,EAAE,CAAA;IACtD,CAAC;IAED,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtB,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,sCAAsC,EAAE,CAAA;IACxE,CAAC;IAED,4BAA4B;IAC5B,MAAM,WAAW,GAAG,wDAAwD,CAAA;IAC5E,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;QAC9B,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,kDAAkD,EAAE,CAAA;IACpF,CAAC;IAED,6BAA6B;IAC7B,MAAM,QAAQ,GAAG,CAAC,WAAW,EAAE,WAAW,EAAE,YAAY,CAAC,CAAA;IACzD,IAAI,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,MAAM,KAAK,CAAC,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC;QACjE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,qCAAqC,EAAE,CAAA;IACvE,CAAC;IAED,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,CAAA;AACxB,CAAC;AAED;;GAEG;AACH,SAAS,eAAe,CAAC,MAAc;IACrC,IAAI,UAAU,GAAG,MAAM,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAA;IAC5C,UAAU,GAAG,UAAU,CAAC,OAAO,CAAC,cAAc,EAAE,EAAE,CAAC,CAAA;IACnD,UAAU,GAAG,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAA;IACrC,UAAU,GAAG,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAA;IACrC,OAAO,UAAU,CAAA;AACnB,CAAC;AAED,gFAAgF;AAChF,qBAAqB;AACrB,gFAAgF;AAEhF;;GAEG;AACI,KAAK,UAAU,SAAS,CAAC,YAAoB,EAAE,OAAsB;IAC1E,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAA;IACjC,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,IAAI,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,2BAAe,CAAA;IAE3E,eAAM,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAA;IAExC,cAAc;IACd,MAAM,MAAM,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAA;IAC5C,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,eAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAA;QACxC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,eAAM,CAAC,IAAI,CAAC,kEAAkE,CAAC,CAAA;QAC/E,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,wEAAwE;IACxE,MAAM,MAAM,GAAG,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC,CAAA;IACzC,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,eAAM,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAA;QACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,OAAO,CAAC,GAAG,CAAC,8CAA8C,CAAC,CAAA;QAC3D,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,OAAO,CAAC,GAAG,CAAC,8BAA8B,CAAC,CAAA;QAC3C,OAAO,CAAC,GAAG,CAAC,kDAAkD,CAAC,CAAA;QAC/D,OAAO,CAAC,GAAG,CAAC,uDAAuD,GAAG,YAAY,CAAC,CAAA;QACnF,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,gCAAgC;IAChC,MAAM,MAAM,GAAG,eAAe,CAAC,YAAY,CAAC,CAAA;IAC5C,MAAM,UAAU,GAAG,mBAAmB,CAAC,MAAM,CAAC,CAAA;IAC9C,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;QACtB,eAAM,CAAC,KAAK,CAAC,UAAU,CAAC,KAAM,CAAC,CAAA;QAC/B,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,eAAM,CAAC,IAAI,CAAC,WAAW,MAAM,EAAE,CAAC,CAAA;IAChC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAEf,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,MAAM,kBAAkB,EAAE;YACxD,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;gBAClC,eAAe,EAAE,UAAU,MAAM,EAAE;aACpC;YACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,YAAY,EAAE,MAAM,EAAE,CAAC;SAC/C,CAAC,CAAA;QAEF,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAuB,CAAA;QAEzD,IAAI,CAAC,QAAQ,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpC,eAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,sBAAsB,CAAC,CAAA;YACpD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACjB,CAAC;QAED,eAAM,CAAC,OAAO,CAAC,YAAY,MAAM,SAAS,CAAC,CAAA;QAC3C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QAEf,4FAA4F;QAC5F,IAAI,MAAM,CAAC,QAAQ,EAAE,CAAC;YACpB,OAAO,CAAC,GAAG,CAAC,IAAI,GAAG,eAAM,CAAC,KAAK,CAAC,wCAAwC,CAAC,CAAC,CAAA;YAC1E,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,OAAO,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAA;YAChD,OAAO,CAAC,GAAG,CAAC,KAAK,eAAM,CAAC,IAAI,CAAC,WAAW,MAAM,EAAE,CAAC,EAAE,CAAC,CAAA;YACpD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,OAAM;QACR,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,wDAAwD,CAAC,CAAA;QACrE,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QAEf,2CAA2C;QAC3C,IAAI,MAAM,CAAC,YAAY,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,YAAY,CAAC,EAAE,CAAC;YAC9D,MAAM,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;gBAC5C,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,GAAG,CAAC,KAAK,MAAM,CAAC,IAAI,UAAU,CAAC,CAAA;gBACrD,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,IAAI,EAAE,CAAC,CAAA;gBACzC,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,KAAK,EAAE,CAAC,CAAA;gBAC1C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACjB,CAAC,CAAC,CAAA;QACJ,CAAC;QAED,eAAM,CAAC,IAAI,CAAC,gCAAgC,CAAC,CAAA;QAC7C,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAA;QACrC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAEjB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;QACtE,eAAM,CAAC,KAAK,CAAC,yBAAyB,OAAO,EAAE,CAAC,CAAA;QAChD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;AACH,CAAC;AAED,gFAAgF;AAChF,wBAAwB;AACxB,gFAAgF;AAEhF;;GAEG;AACI,KAAK,UAAU,YAAY,CAAC,OAAsB;IACvD,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAA;IACjC,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,IAAI,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,2BAAe,CAAA;IAE3E,cAAc;IACd,MAAM,MAAM,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAA;IAC5C,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,eAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAA;QACxC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,eAAM,CAAC,IAAI,CAAC,kEAAkE,CAAC,CAAA;QAC/E,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,cAAc;IACd,MAAM,MAAM,GAAG,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC,CAAA;IACzC,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,eAAM,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAA;QACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,eAAM,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAA;QAC7D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,MAAM,qBAAqB,EAAE;YAC3D,MAAM,EAAE,KAAK;YACb,OAAO,EAAE;gBACP,eAAe,EAAE,UAAU,MAAM,EAAE;aACpC;SACF,CAAC,CAAA;QAEF,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,IAAI,EAA0B,CAAA;QAE5D,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,eAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,6BAA6B,CAAC,CAAA;YAC3D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACjB,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAA;QACrC,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAA;QAErC,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC;YAC5B,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,OAAO,CAAC,GAAG,CAAC,gCAAgC,CAAC,CAAA;YAC7C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,OAAO,CAAC,GAAG,CAAC,+BAA+B,CAAC,CAAA;YAC5C,OAAO,CAAC,GAAG,CAAC,KAAK,eAAM,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC,CAAA;YAClD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,eAAM,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAA;YAC3C,OAAO,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAA;YACtD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,OAAM;QACR,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,OAAO,CAAC,GAAG,CAAC,cAAc,MAAM,CAAC,MAAM,EAAE,CAAC,CAAA;QAC1C,OAAO,CAAC,GAAG,CAAC,cAAc,gBAAgB,CAAC,MAAM,CAAC,MAAM,IAAI,SAAS,CAAC,EAAE,CAAC,CAAA;QACzE,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QAEf,IAAI,MAAM,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;YAC/B,OAAO,CAAC,GAAG,CAAC,+BAA+B,CAAC,CAAA;YAC5C,OAAO,CAAC,GAAG,CAAC,KAAK,eAAM,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,IAAI,WAAW,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,CAAC,CAAA;YAC/E,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAA;YACpC,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,UAAU,EAAE,CAAC,CAAA;QACvC,CAAC;aAAM,IAAI,MAAM,CAAC,MAAM,KAAK,SAAS,EAAE,CAAC;YACvC,IAAI,MAAM,CAAC,YAAY,IAAI,MAAM,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC1D,OAAO,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAA;gBACtD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;gBACf,MAAM,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;oBAC5C,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,GAAG,CAAC,KAAK,MAAM,CAAC,IAAI,UAAU,CAAC,CAAA;oBACrD,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,IAAI,EAAE,CAAC,CAAA;oBACzC,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,KAAK,EAAE,CAAC,CAAA;oBAC1C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;gBACjB,CAAC,CAAC,CAAA;gBACF,eAAM,CAAC,IAAI,CAAC,qDAAqD,CAAC,CAAA;YACpE,CAAC;QACH,CAAC;aAAM,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;YAC1B,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,OAAO,EAAE,CAAC,CAAA;QACpC,CAAC;QAED,IAAI,MAAM,CAAC,QAAQ,EAAE,CAAC;YACpB,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,eAAM,CAAC,IAAI,CAAC,SAAS,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAA;QACzC,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAEjB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;QACtE,eAAM,CAAC,KAAK,CAAC,gCAAgC,OAAO,EAAE,CAAC,CAAA;QACvD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,gBAAgB,CAAC,MAAc;IACtC,QAAQ,MAAM,EAAE,CAAC;QACf,KAAK,SAAS;YACZ,OAAO,eAAM,CAAC,MAAM,CAAC,4BAA4B,CAAC,CAAA;QACpD,KAAK,QAAQ;YACX,OAAO,eAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAA;QACjC,KAAK,OAAO;YACV,OAAO,eAAM,CAAC,GAAG,CAAC,SAAS,CAAC,CAAA;QAC9B;YACE,OAAO,MAAM,CAAA;IACjB,CAAC;AACH,CAAC;AAED,gFAAgF;AAChF,wBAAwB;AACxB,gFAAgF;AAEhF;;GAEG;AACI,KAAK,UAAU,YAAY,CAAC,OAAsB;IACvD,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAA;IACjC,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,IAAI,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,2BAAe,CAAA;IAE3E,eAAM,CAAC,IAAI,CAAC,kCAAkC,CAAC,CAAA;IAE/C,cAAc;IACd,MAAM,MAAM,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAA;IAC5C,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,eAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAA;QACxC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,eAAM,CAAC,IAAI,CAAC,kEAAkE,CAAC,CAAA;QAC/E,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,cAAc;IACd,MAAM,MAAM,GAAG,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC,CAAA;IACzC,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,eAAM,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAA;QACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,eAAM,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAA;QAC7D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,MAAM,qBAAqB,EAAE;YAC3D,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;gBAClC,eAAe,EAAE,UAAU,MAAM,EAAE;aACpC;YACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC;SACzB,CAAC,CAAA;QAEF,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,IAAI,EAA0B,CAAA;QAE5D,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,eAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,yBAAyB,CAAC,CAAA;YACvD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACjB,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAA;QACzC,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAA;QACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,OAAO,CAAC,GAAG,CAAC,aAAa,MAAM,CAAC,MAAM,EAAE,CAAC,CAAA;QACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QAEf,IAAI,MAAM,CAAC,QAAQ,EAAE,CAAC;YACpB,OAAO,CAAC,GAAG,CAAC,aAAa,eAAM,CAAC,KAAK,CAAC,YAAY,CAAC,EAAE,CAAC,CAAA;YACtD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,eAAM,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAA;YAClC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,OAAO,EAAE,CAAC,CAAA;YAClC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,OAAO,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAA;YAChD,OAAO,CAAC,GAAG,CAAC,KAAK,eAAM,CAAC,IAAI,CAAC,WAAW,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,CAAC,CAAA;QAC7D,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,aAAa,eAAM,CAAC,MAAM,CAAC,wBAAwB,CAAC,EAAE,CAAC,CAAA;YACnE,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YAEf,mCAAmC;YACnC,IAAI,MAAM,CAAC,YAAY,IAAI,MAAM,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC1D,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAA;gBACtC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;gBACf,MAAM,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;oBAC5C,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,GAAG,CAAC,KAAK,MAAM,CAAC,IAAI,UAAU,CAAC,CAAA;oBACrD,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,IAAI,EAAE,CAAC,CAAA;oBACzC,OAAO,CAAC,GAAG,CAAC,eAAe,MAAM,CAAC,KAAK,EAAE,CAAC,CAAA;oBAC1C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;gBACjB,CAAC,CAAC,CAAA;YACJ,CAAC;YAED,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,OAAO,EAAE,CAAC,CAAA;YAClC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;YACf,eAAM,CAAC,IAAI,CAAC,mDAAmD,CAAC,CAAA;YAChE,eAAM,CAAC,IAAI,CAAC,yCAAyC,CAAC,CAAA;QACxD,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAEjB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;QACtE,eAAM,CAAC,KAAK,CAAC,4BAA4B,OAAO,EAAE,CAAC,CAAA;QACnD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;AACH,CAAC;AAED,gFAAgF;AAChF,wBAAwB;AACxB,gFAAgF;AAEhF;;GAEG;AACI,KAAK,UAAU,YAAY,CAAC,YAAgC,EAAE,OAAsB;IACzF,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAA;IACjC,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,IAAI,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,2BAAe,CAAA;IAE3E,eAAM,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAA;IAE1C,cAAc;IACd,MAAM,MAAM,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAA;IAC5C,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,eAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAA;QACxC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,eAAM,CAAC,IAAI,CAAC,kEAAkE,CAAC,CAAA;QAC/E,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,cAAc;IACd,MAAM,MAAM,GAAG,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC,CAAA;IACzC,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,eAAM,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAA;QACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,eAAM,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAA;QAC7D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;IAED,IAAI,CAAC;QACH,MAAM,IAAI,GAA2B,EAAE,CAAA;QACvC,IAAI,YAAY,EAAE,CAAC;YACjB,IAAI,CAAC,YAAY,GAAG,eAAe,CAAC,YAAY,CAAC,CAAA;QACnD,CAAC;QAED,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,MAAM,qBAAqB,EAAE;YAC3D,MAAM,EAAE,QAAQ;YAChB,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;gBAClC,eAAe,EAAE,UAAU,MAAM,EAAE;aACpC;YACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;SAC3B,CAAC,CAAA;QAEF,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,IAAI,EAA0B,CAAA;QAE5D,IAAI,CAAC,QAAQ,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpC,eAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,yBAAyB,CAAC,CAAA;YACvD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACjB,CAAC;QAED,eAAM,CAAC,OAAO,CAAC,YAAY,MAAM,CAAC,MAAM,UAAU,CAAC,CAAA;QACnD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,OAAO,CAAC,GAAG,CAAC,yDAAyD,CAAC,CAAA;QACtE,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;QACf,eAAM,CAAC,IAAI,CAAC,+CAA+C,CAAC,CAAA;QAC5D,OAAO,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAA;QACtD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAEjB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;QACtE,eAAM,CAAC,KAAK,CAAC,4BAA4B,OAAO,EAAE,CAAC,CAAA;QACnD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;AACH,CAAC","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  verification?: VerificationRecord[]\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  verification?: VerificationRecord[]\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  verification?: VerificationRecord[]\n  instructions?: string[]\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    // Display verification records from Vercel\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    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      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    } 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 verification')\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    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    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.50",
3
+ "version": "0.1.51",
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",
@@ -1,17 +1,17 @@
1
1
  import { NextResponse } from 'next/server';
2
2
  import { validateApiKey, addCustomDomain, isCustomDomainRegistered, updateCustomDomainStatus } from '@/lib/storage/blob';
3
3
  import { isValidDomain, normalizeDomain } from '@/lib/docs/config';
4
- import { addDomainToProject, isVercelIntegrationEnabled, formatVerificationInstructions, getDomainType } from '@/lib/vercel/domains';
4
+ import { addDomainToProject, isVercelIntegrationEnabled, formatVerificationInstructions } from '@/lib/vercel/domains';
5
5
  /**
6
6
  * POST /api/domains/add
7
7
  *
8
- * Add a custom domain to a project.
8
+ * Add a custom domain to a project via Vercel Domains API.
9
9
  * Each project can have ONE custom domain (free).
10
10
  *
11
- * When Vercel integration is enabled (VERCEL_API_TOKEN + VERCEL_PROJECT_ID):
12
- * - Domain is added to Vercel project
13
- * - Real DNS verification instructions from Vercel are returned
14
- * - SSL is provisioned automatically by Vercel after verification
11
+ * Requires environment variables:
12
+ * - VERCEL_API_TOKEN (or VERCEL_TOKEN)
13
+ * - VERCEL_PROJECT_ID
14
+ * - VERCEL_TEAM_ID (optional)
15
15
  *
16
16
  * Headers:
17
17
  * Authorization: Bearer <api_key>
@@ -25,7 +25,7 @@ import { addDomainToProject, isVercelIntegrationEnabled, formatVerificationInstr
25
25
  * domain: "docs.example.com",
26
26
  * status: "pending",
27
27
  * verification: [
28
- * { type: "TXT", domain: "_vercel.docs.example.com", value: "vc-domain-verify=..." }
28
+ * { type: "TXT", name: "_vercel.example.com", value: "vc-domain-verify=..." }
29
29
  * ]
30
30
  * }
31
31
  */ export async function POST(request) {
@@ -77,115 +77,79 @@ import { addDomainToProject, isVercelIntegrationEnabled, formatVerificationInstr
77
77
  status: 409
78
78
  });
79
79
  }
80
- // Check if Vercel integration is enabled
81
- if (isVercelIntegrationEnabled()) {
82
- // Add domain to Vercel project
83
- const vercelResult = await addDomainToProject(customDomain);
84
- if (!vercelResult.success) {
85
- return NextResponse.json({
86
- error: vercelResult.error,
87
- vercelError: vercelResult.vercelError
88
- }, {
89
- status: 400
90
- });
91
- }
92
- const vercelDomain = vercelResult.domain;
93
- // Add to our internal registry with Vercel data
94
- const result = await addCustomDomain(projectSlug, customDomain);
95
- if (!result.success) {
96
- // Rollback: Try to remove from Vercel if we couldn't add to registry
97
- console.error('[Domains API] Failed to add to registry, domain added to Vercel:', customDomain);
98
- return NextResponse.json({
99
- error: result.error
100
- }, {
101
- status: 400
102
- });
103
- }
104
- // Update with Vercel domain ID
105
- await updateCustomDomainStatus(customDomain, vercelDomain.verified ? 'active' : 'pending', {
106
- vercelDomainId: vercelDomain.projectId
80
+ // Require Vercel integration
81
+ if (!isVercelIntegrationEnabled()) {
82
+ return NextResponse.json({
83
+ error: 'Vercel integration not configured. Set VERCEL_API_TOKEN and VERCEL_PROJECT_ID environment variables.'
84
+ }, {
85
+ status: 503
107
86
  });
108
- // If Vercel already verified the domain (e.g., previously configured DNS)
109
- if (vercelDomain.verified) {
110
- return NextResponse.json({
111
- success: true,
112
- domain: customDomain,
113
- projectSlug,
114
- status: 'active',
115
- verified: true,
116
- message: 'Domain is already verified and active!'
117
- });
118
- }
119
- // Format Vercel's verification instructions
120
- const verification = vercelDomain.verification || [];
121
- const { records, instructions } = formatVerificationInstructions(verification);
122
- // Add CNAME instruction for subdomains (Vercel may not always include it)
123
- const domainType = getDomainType(customDomain);
124
- const parts = customDomain.split('.');
125
- const subdomain = parts.length > 2 ? parts[0] : '@';
126
- const cnameInstruction = domainType === 'subdomain' ? `\nAlso add a CNAME record:\n Name: ${subdomain}\n Value: cname.vercel-dns.com` : `\nFor apex domains, add an A record:\n Name: @\n Value: 76.76.21.21`;
87
+ }
88
+ // Add domain to Vercel project
89
+ const vercelResult = await addDomainToProject(customDomain);
90
+ if (!vercelResult.success) {
127
91
  return NextResponse.json({
128
- success: true,
129
- domain: customDomain,
130
- projectSlug,
131
- status: 'pending',
132
- verified: false,
133
- verification: records,
134
- instructions: [
135
- ...instructions,
136
- cnameInstruction
137
- ],
138
- vercelVerification: verification
92
+ error: vercelResult.error,
93
+ vercelError: vercelResult.vercelError
94
+ }, {
95
+ status: 400
139
96
  });
140
- } else {
141
- // Fallback: No Vercel integration, use legacy behavior
142
- console.warn('[Domains API] Vercel integration not configured, using legacy mode');
143
- const result = await addCustomDomain(projectSlug, customDomain);
144
- if (!result.success) {
145
- return NextResponse.json({
146
- error: result.error
147
- }, {
148
- status: 400
149
- });
150
- }
151
- // Legacy hardcoded DNS instructions
152
- const parts = customDomain.split('.');
153
- const subdomain = parts.length > 2 ? parts[0] : '@';
97
+ }
98
+ const vercelDomain = vercelResult.domain;
99
+ // Add to our internal registry with Vercel data
100
+ const result = await addCustomDomain(projectSlug, customDomain);
101
+ if (!result.success) {
102
+ // Rollback: Try to remove from Vercel if we couldn't add to registry
103
+ console.error('[Domains API] Failed to add to registry, domain added to Vercel:', customDomain);
104
+ return NextResponse.json({
105
+ error: result.error
106
+ }, {
107
+ status: 400
108
+ });
109
+ }
110
+ // Update with Vercel domain ID
111
+ await updateCustomDomainStatus(customDomain, vercelDomain.verified ? 'active' : 'pending', {
112
+ vercelDomainId: vercelDomain.projectId
113
+ });
114
+ // If Vercel already verified the domain (e.g., previously configured DNS)
115
+ if (vercelDomain.verified) {
154
116
  return NextResponse.json({
155
117
  success: true,
156
118
  domain: customDomain,
157
119
  projectSlug,
158
- status: result.entry.status,
159
- verified: false,
160
- verification: [
161
- {
162
- type: 'CNAME',
163
- name: subdomain,
164
- value: 'cname.vercel-dns.com'
165
- },
166
- {
167
- type: 'TXT',
168
- name: `_devdoc-verify.${customDomain}`,
169
- value: result.entry.verificationToken || ''
170
- }
171
- ],
172
- instructions: [
173
- 'Add the following DNS records to your domain:',
174
- '',
175
- '1. CNAME Record:',
176
- ` Name: ${subdomain}`,
177
- ' Value: cname.vercel-dns.com',
178
- '',
179
- '2. TXT Record (for verification):',
180
- ` Name: _devdoc-verify.${customDomain}`,
181
- ` Value: ${result.entry.verificationToken || ''}`,
182
- '',
183
- 'After adding DNS records, run "devdoc domain verify" to verify.',
184
- '',
185
- 'Note: Vercel integration not configured. Set VERCEL_API_TOKEN and VERCEL_PROJECT_ID for automatic SSL.'
186
- ]
120
+ status: 'active',
121
+ verified: true,
122
+ message: 'Domain is already verified and active!'
187
123
  });
188
124
  }
125
+ // Format Vercel's verification instructions
126
+ const verification = vercelDomain.verification || [];
127
+ const { records } = formatVerificationInstructions(verification);
128
+ // Add CNAME/A record for routing (Vercel only returns TXT for verification)
129
+ const parts = customDomain.split('.');
130
+ const isApexDomain = parts.length <= 2;
131
+ const subdomain = parts.length > 2 ? parts[0] : '@';
132
+ // Add routing record first, then verification records
133
+ const allRecords = [
134
+ isApexDomain ? {
135
+ type: 'A',
136
+ name: '@',
137
+ value: '76.76.21.21'
138
+ } : {
139
+ type: 'CNAME',
140
+ name: subdomain,
141
+ value: 'cname.vercel-dns.com'
142
+ },
143
+ ...records
144
+ ];
145
+ return NextResponse.json({
146
+ success: true,
147
+ domain: customDomain,
148
+ projectSlug,
149
+ status: 'pending',
150
+ verified: false,
151
+ verification: allRecords
152
+ });
189
153
  } catch (error) {
190
154
  console.error('[Domains API] Error adding domain:', error);
191
155
  const message = error instanceof Error ? error.message : String(error);
@@ -5,11 +5,13 @@ import { removeDomain as vercelRemoveDomain, isVercelIntegrationEnabled } from '
5
5
  /**
6
6
  * DELETE /api/domains/remove
7
7
  *
8
- * Remove a custom domain from a project.
8
+ * Remove a custom domain from a project via Vercel API.
9
+ * Removes from both Vercel and internal registry.
9
10
  *
10
- * When Vercel integration is enabled:
11
- * - Removes domain from Vercel project
12
- * - Removes from internal registry
11
+ * Requires environment variables:
12
+ * - VERCEL_API_TOKEN (or VERCEL_TOKEN)
13
+ * - VERCEL_PROJECT_ID
14
+ * - VERCEL_TEAM_ID (optional)
13
15
  *
14
16
  * Headers:
15
17
  * Authorization: Bearer <api_key>
@@ -57,15 +59,21 @@ import { removeDomain as vercelRemoveDomain, isVercelIntegrationEnabled } from '
57
59
  }
58
60
  customDomain = projectDomain.customDomain;
59
61
  }
60
- // Remove from Vercel if integration is enabled
61
- if (isVercelIntegrationEnabled()) {
62
- const vercelResult = await vercelRemoveDomain(customDomain);
63
- if (!vercelResult.success) {
64
- // Log warning but continue - domain might not exist in Vercel
65
- console.warn('[Domains API] Failed to remove from Vercel:', vercelResult.error);
66
- } else {
67
- console.log('[Domains API] Domain removed from Vercel:', customDomain);
68
- }
62
+ // Require Vercel integration
63
+ if (!isVercelIntegrationEnabled()) {
64
+ return NextResponse.json({
65
+ error: 'Vercel integration not configured. Set VERCEL_API_TOKEN and VERCEL_PROJECT_ID environment variables.'
66
+ }, {
67
+ status: 503
68
+ });
69
+ }
70
+ // Remove from Vercel
71
+ const vercelResult = await vercelRemoveDomain(customDomain);
72
+ if (!vercelResult.success) {
73
+ // Log warning but continue - domain might not exist in Vercel
74
+ console.warn('[Domains API] Failed to remove from Vercel:', vercelResult.error);
75
+ } else {
76
+ console.log('[Domains API] Domain removed from Vercel:', customDomain);
69
77
  }
70
78
  // Remove from internal registry
71
79
  const result = await removeCustomDomain(customDomain, projectSlug);
@@ -1,11 +1,17 @@
1
1
  import { NextResponse } from 'next/server';
2
2
  import { validateApiKey, getProjectCustomDomain, getCustomDomainEntry } from '@/lib/storage/blob';
3
- import { normalizeDomain, getDnsInstructions } from '@/lib/docs/config';
3
+ import { normalizeDomain } from '@/lib/docs/config';
4
4
  import { getProjectUrl } from '@/lib/multi-tenant/context';
5
+ import { getDomainConfig, isVercelIntegrationEnabled, formatVerificationInstructions } from '@/lib/vercel/domains';
5
6
  /**
6
7
  * GET /api/domains/status
7
8
  *
8
- * Get the status of a custom domain.
9
+ * Get the status of a custom domain via Vercel API.
10
+ *
11
+ * Requires environment variables:
12
+ * - VERCEL_API_TOKEN (or VERCEL_TOKEN)
13
+ * - VERCEL_PROJECT_ID
14
+ * - VERCEL_TEAM_ID (optional)
9
15
  *
10
16
  * Headers:
11
17
  * Authorization: Bearer <api_key>
@@ -94,26 +100,18 @@ import { getProjectUrl } from '@/lib/multi-tenant/context';
94
100
  switch(domainEntry.status){
95
101
  case 'pending':
96
102
  {
97
- const dnsInstructions = getDnsInstructions(domainEntry.customDomain);
98
- dnsInstructions.txt.value = domainEntry.verificationToken || '';
99
103
  response.message = 'Waiting for DNS configuration';
100
- response.dnsRecords = {
101
- cname: dnsInstructions.cname,
102
- txt: dnsInstructions.txt
103
- };
104
- response.nextStep = 'Add the DNS records above, then run "devdoc domain verify"';
104
+ response.nextStep = 'Add the DNS records below, then run "devdoc domain verify"';
105
+ // Get verification instructions from Vercel if available
106
+ if (isVercelIntegrationEnabled()) {
107
+ const configResult = await getDomainConfig(domainEntry.customDomain);
108
+ if (configResult.success && configResult.domain?.verification) {
109
+ const { records } = formatVerificationInstructions(configResult.domain.verification);
110
+ response.verification = records;
111
+ }
112
+ }
105
113
  break;
106
114
  }
107
- case 'dns_verified':
108
- response.message = 'DNS verified, SSL certificate provisioning in progress';
109
- response.verifiedAt = domainEntry.verifiedAt;
110
- response.nextStep = 'SSL certificate should be ready within 1-24 hours';
111
- break;
112
- case 'ssl_provisioning':
113
- response.message = 'SSL certificate being provisioned';
114
- response.verifiedAt = domainEntry.verifiedAt;
115
- response.nextStep = 'Almost ready! Check back in a few minutes';
116
- break;
117
115
  case 'active':
118
116
  response.message = 'Domain is active and working';
119
117
  response.customUrl = `https://${domainEntry.customDomain}`;
@@ -2,21 +2,16 @@ import { NextResponse } from 'next/server';
2
2
  import { validateApiKey, getProjectCustomDomain, getCustomDomainEntry, updateCustomDomainStatus } from '@/lib/storage/blob';
3
3
  import { normalizeDomain } from '@/lib/docs/config';
4
4
  import { verifyDomain as vercelVerifyDomain, getDomainConfig, isVercelIntegrationEnabled, formatVerificationInstructions } from '@/lib/vercel/domains';
5
- import dns from 'dns';
6
- import { promisify } from 'util';
7
- const resolveCname = promisify(dns.resolveCname);
8
- const resolveTxt = promisify(dns.resolveTxt);
9
5
  /**
10
6
  * POST /api/domains/verify
11
7
  *
12
- * Verify DNS configuration for a custom domain.
8
+ * Verify DNS configuration for a custom domain via Vercel API.
9
+ * Vercel handles DNS verification and automatic SSL provisioning.
13
10
  *
14
- * When Vercel integration is enabled:
15
- * - Calls Vercel's verify endpoint directly
16
- * - Vercel handles DNS verification and SSL provisioning
17
- *
18
- * Legacy mode (no Vercel integration):
19
- * - Checks CNAME and TXT records manually
11
+ * Requires environment variables:
12
+ * - VERCEL_API_TOKEN (or VERCEL_TOKEN)
13
+ * - VERCEL_PROJECT_ID
14
+ * - VERCEL_TEAM_ID (optional)
20
15
  *
21
16
  * Headers:
22
17
  * Authorization: Bearer <api_key>
@@ -84,142 +79,71 @@ const resolveTxt = promisify(dns.resolveTxt);
84
79
  status: 403
85
80
  });
86
81
  }
87
- // Use Vercel integration if available
88
- if (isVercelIntegrationEnabled()) {
89
- // Call Vercel's verify endpoint
90
- const verifyResult = await vercelVerifyDomain(customDomain);
91
- if (!verifyResult.success) {
92
- // Get current domain config to show what's needed
93
- const configResult = await getDomainConfig(customDomain);
94
- const verification = configResult.domain?.verification || [];
95
- let instructions = [];
96
- if (verification.length > 0) {
97
- const formatted = formatVerificationInstructions(verification);
98
- instructions = formatted.instructions;
99
- }
100
- return NextResponse.json({
101
- success: false,
102
- domain: customDomain,
103
- projectSlug,
104
- status: domainEntry.status,
105
- verified: false,
106
- message: verifyResult.error || 'DNS verification failed. Please check your DNS records.',
107
- verification: verification.map((v)=>({
108
- type: v.type,
109
- name: v.domain,
110
- value: v.value
111
- })),
112
- instructions
113
- });
114
- }
115
- // Domain verified successfully
116
- if (verifyResult.verified) {
117
- // Update our registry to mark as active
118
- await updateCustomDomainStatus(customDomain, 'active');
119
- return NextResponse.json({
120
- success: true,
121
- domain: customDomain,
122
- projectSlug,
123
- status: 'active',
124
- verified: true,
125
- message: 'Domain verified! SSL certificate will be provisioned automatically by Vercel.'
126
- });
127
- } else {
128
- // Vercel didn't verify yet, return what's needed
129
- const verification = verifyResult.domain?.verification || [];
130
- const { instructions } = formatVerificationInstructions(verification);
131
- return NextResponse.json({
132
- success: false,
133
- domain: customDomain,
134
- projectSlug,
135
- status: 'pending',
136
- verified: false,
137
- message: 'DNS records not yet propagated. This can take up to 48 hours.',
138
- verification: verification.map((v)=>({
139
- type: v.type,
140
- name: v.domain,
141
- value: v.value
142
- })),
143
- instructions
144
- });
145
- }
146
- }
147
- // Legacy verification (no Vercel integration)
148
- console.warn('[Domains API] Vercel integration not configured, using legacy DNS verification');
149
- // Check DNS records manually
150
- const checks = {
151
- cname: {
152
- found: false,
153
- value: null,
154
- expected: 'cname.vercel-dns.com'
155
- },
156
- txt: {
157
- found: false,
158
- verified: false,
159
- expected: domainEntry.verificationToken
160
- }
161
- };
162
- // Check CNAME record
163
- try {
164
- const cnameRecords = await resolveCname(customDomain);
165
- if (cnameRecords && cnameRecords.length > 0) {
166
- checks.cname.found = true;
167
- checks.cname.value = cnameRecords[0];
168
- }
169
- } catch {
170
- // CNAME not found or DNS error
82
+ // Require Vercel integration
83
+ if (!isVercelIntegrationEnabled()) {
84
+ return NextResponse.json({
85
+ error: 'Vercel integration not configured. Set VERCEL_API_TOKEN and VERCEL_PROJECT_ID environment variables.'
86
+ }, {
87
+ status: 503
88
+ });
171
89
  }
172
- // Check TXT record for verification
173
- const txtRecordName = `_devdoc-verify.${customDomain}`;
174
- try {
175
- const txtRecords = await resolveTxt(txtRecordName);
176
- if (txtRecords && txtRecords.length > 0) {
177
- checks.txt.found = true;
178
- // TXT records can be arrays, flatten and check
179
- const allTxtValues = txtRecords.flat();
180
- checks.txt.verified = allTxtValues.some((val)=>val === domainEntry.verificationToken);
90
+ // Call Vercel's verify endpoint
91
+ const verifyResult = await vercelVerifyDomain(customDomain);
92
+ if (!verifyResult.success) {
93
+ // Get current domain config to show what's needed
94
+ const configResult = await getDomainConfig(customDomain);
95
+ const verification = configResult.domain?.verification || [];
96
+ let instructions = [];
97
+ if (verification.length > 0) {
98
+ const formatted = formatVerificationInstructions(verification);
99
+ instructions = formatted.instructions;
181
100
  }
182
- } catch {
183
- // TXT not found or DNS error
101
+ return NextResponse.json({
102
+ success: false,
103
+ domain: customDomain,
104
+ projectSlug,
105
+ status: domainEntry.status,
106
+ verified: false,
107
+ message: verifyResult.error || 'DNS verification failed. Please check your DNS records.',
108
+ verification: verification.map((v)=>({
109
+ type: v.type,
110
+ name: v.domain,
111
+ value: v.value
112
+ })),
113
+ instructions
114
+ });
184
115
  }
185
- // Determine overall status
186
- const cnameValid = checks.cname.found && checks.cname.value?.toLowerCase().includes('vercel');
187
- const txtValid = checks.txt.found && checks.txt.verified;
188
- let newStatus = domainEntry.status;
189
- let message = '';
190
- if (cnameValid && txtValid) {
191
- // DNS is verified
192
- newStatus = 'active';
193
- message = 'DNS verified! Domain is now active.';
116
+ // Domain verified successfully
117
+ if (verifyResult.verified) {
118
+ // Update our registry to mark as active
194
119
  await updateCustomDomainStatus(customDomain, 'active');
195
- } else if (cnameValid && !txtValid) {
196
- message = 'CNAME record found, but TXT verification record is missing or incorrect.';
197
- } else if (!cnameValid && txtValid) {
198
- message = 'TXT verification found, but CNAME record is missing or incorrect.';
120
+ return NextResponse.json({
121
+ success: true,
122
+ domain: customDomain,
123
+ projectSlug,
124
+ status: 'active',
125
+ verified: true,
126
+ message: 'Domain verified! SSL certificate will be provisioned automatically by Vercel.'
127
+ });
199
128
  } else {
200
- message = 'DNS records not found. Please add the required CNAME and TXT records.';
129
+ // Vercel didn't verify yet, return what's needed
130
+ const verification = verifyResult.domain?.verification || [];
131
+ const { instructions } = formatVerificationInstructions(verification);
132
+ return NextResponse.json({
133
+ success: false,
134
+ domain: customDomain,
135
+ projectSlug,
136
+ status: 'pending',
137
+ verified: false,
138
+ message: 'DNS records not yet propagated. This can take up to 48 hours.',
139
+ verification: verification.map((v)=>({
140
+ type: v.type,
141
+ name: v.domain,
142
+ value: v.value
143
+ })),
144
+ instructions
145
+ });
201
146
  }
202
- return NextResponse.json({
203
- success: cnameValid && txtValid,
204
- domain: customDomain,
205
- projectSlug,
206
- status: newStatus,
207
- verified: cnameValid && txtValid,
208
- message,
209
- checks: {
210
- cname: {
211
- found: checks.cname.found,
212
- value: checks.cname.value,
213
- valid: cnameValid,
214
- expected: checks.cname.expected
215
- },
216
- txt: {
217
- found: checks.txt.found,
218
- verified: checks.txt.verified,
219
- expected: checks.txt.expected
220
- }
221
- }
222
- });
223
147
  } catch (error) {
224
148
  console.error('[Domains API] Error verifying domain:', error);
225
149
  const message = error instanceof Error ? error.message : String(error);
@@ -143,41 +143,3 @@ export const domainConfigSchema = z.object({
143
143
  normalized = normalized.split(':')[0];
144
144
  return normalized;
145
145
  }
146
- /**
147
- * Get DNS instructions for a custom domain
148
- *
149
- * NOTE: When Vercel integration is enabled (VERCEL_API_TOKEN + VERCEL_PROJECT_ID),
150
- * the actual DNS instructions come from Vercel's API response.
151
- * This function is used as a fallback for legacy/local development mode.
152
- *
153
- * For Vercel-hosted projects:
154
- * - Subdomains: CNAME to cname.vercel-dns.com
155
- * - Apex domains: A record to 76.76.21.21
156
- *
157
- * @param customDomain - The custom domain (e.g., "docs.example.com")
158
- * @returns DNS instructions for CNAME and TXT records
159
- */ export function getDnsInstructions(customDomain) {
160
- const parts = customDomain.split('.');
161
- const subdomain = parts.length > 2 ? parts[0] : '@';
162
- const isApexDomain = parts.length <= 2;
163
- return {
164
- cname: {
165
- name: subdomain === '@' ? customDomain : subdomain,
166
- // Use Vercel's actual DNS target
167
- value: isApexDomain ? '76.76.21.21' : 'cname.vercel-dns.com'
168
- },
169
- txt: {
170
- name: `_devdoc-verify.${customDomain}`,
171
- value: ''
172
- }
173
- };
174
- }
175
- /**
176
- * Get human-readable DNS type based on domain
177
- *
178
- * @param customDomain - The custom domain
179
- * @returns 'A' for apex domains, 'CNAME' for subdomains
180
- */ export function getDnsRecordType(customDomain) {
181
- const parts = customDomain.split('.');
182
- return parts.length <= 2 ? 'A' : 'CNAME';
183
- }
@@ -1,6 +1,6 @@
1
1
  /**
2
2
  * Configuration Module Exports
3
3
  */ export { docsConfigSchema, parseDocsConfig, safeParseDocsConfig, getDefaultDocsConfig } from './schema';
4
- export { domainConfigSchema, parseDomainConfig, safeParseDomainConfig, isValidDomain, normalizeDomain, getDnsInstructions, getDnsRecordType } from './domain-schema';
4
+ export { domainConfigSchema, parseDomainConfig, safeParseDomainConfig, isValidDomain, normalizeDomain } from './domain-schema';
5
5
  export { loadDocsConfig, safeLoadDocsConfig, clearConfigCache, hasDocsConfig, getContentDir, resolvePagePath, loadPageContent, listMdxFiles } from './loader';
6
6
  export { isDevMode, isProductionMode, shouldShowItem } from './environment';