@brainfish-ai/devdoc 0.1.49 → 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.
- package/dist/cli/commands/deploy.js +72 -4
- package/dist/cli/commands/domain.js +51 -58
- package/package.json +1 -1
- package/renderer/app/api/collections/route.js +16 -7
- package/renderer/app/api/deploy/route.js +3 -3
- package/renderer/app/api/domains/add/route.js +73 -31
- package/renderer/app/api/domains/remove/route.js +25 -4
- package/renderer/app/api/domains/status/route.js +17 -19
- package/renderer/app/api/domains/verify/route.js +72 -83
- package/renderer/app/api/schema/route.js +12 -11
- package/renderer/components/docs-header.js +11 -11
- package/renderer/lib/docs/config/domain-schema.js +0 -16
- package/renderer/lib/docs/config/index.js +1 -1
- package/renderer/lib/storage/blob.js +4 -2
- package/renderer/lib/vercel/domains.js +275 -0
|
@@ -120,16 +120,26 @@ 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
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
134
|
+
// Display verification records from Vercel
|
|
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
|
+
}
|
|
133
143
|
logger_1.logger.info('After adding DNS records, run:');
|
|
134
144
|
console.log(' devdoc domain verify');
|
|
135
145
|
console.log('');
|
|
@@ -203,18 +213,18 @@ async function domainStatus(options) {
|
|
|
203
213
|
console.log(' Also accessible at:');
|
|
204
214
|
console.log(` ${result.projectUrl}`);
|
|
205
215
|
}
|
|
206
|
-
else if (result.status === 'pending'
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
216
|
+
else if (result.status === 'pending') {
|
|
217
|
+
if (result.verification && result.verification.length > 0) {
|
|
218
|
+
console.log(' Add these DNS records to your domain:');
|
|
219
|
+
console.log('');
|
|
220
|
+
result.verification.forEach((record, index) => {
|
|
221
|
+
console.log(` ${index + 1}. ${record.type} Record:`);
|
|
222
|
+
console.log(` Name: ${record.name}`);
|
|
223
|
+
console.log(` Value: ${record.value}`);
|
|
224
|
+
console.log('');
|
|
225
|
+
});
|
|
226
|
+
logger_1.logger.info('After adding DNS records, run: devdoc domain verify');
|
|
227
|
+
}
|
|
218
228
|
}
|
|
219
229
|
else if (result.message) {
|
|
220
230
|
console.log(` ${result.message}`);
|
|
@@ -237,11 +247,7 @@ async function domainStatus(options) {
|
|
|
237
247
|
function getStatusDisplay(status) {
|
|
238
248
|
switch (status) {
|
|
239
249
|
case 'pending':
|
|
240
|
-
return logger_1.logger.yellow('⏳ Pending DNS
|
|
241
|
-
case 'dns_verified':
|
|
242
|
-
return logger_1.logger.cyan('✓ DNS verified, SSL provisioning...');
|
|
243
|
-
case 'ssl_provisioning':
|
|
244
|
-
return logger_1.logger.cyan('🔒 SSL certificate provisioning...');
|
|
250
|
+
return logger_1.logger.yellow('⏳ Pending DNS verification');
|
|
245
251
|
case 'active':
|
|
246
252
|
return logger_1.logger.green('✓ Active');
|
|
247
253
|
case 'error':
|
|
@@ -296,46 +302,33 @@ async function domainVerify(options) {
|
|
|
296
302
|
console.log('');
|
|
297
303
|
console.log(` Domain: ${result.domain}`);
|
|
298
304
|
console.log('');
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
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}`);
|
|
307
|
-
}
|
|
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!');
|
|
305
|
+
if (result.verified) {
|
|
306
|
+
console.log(` Status: ${logger_1.logger.green('✓ Verified')}`);
|
|
307
|
+
console.log('');
|
|
308
|
+
logger_1.logger.success('Domain verified!');
|
|
328
309
|
console.log('');
|
|
329
310
|
console.log(` ${result.message}`);
|
|
330
311
|
console.log('');
|
|
331
|
-
|
|
312
|
+
console.log(' Your docs will be available at:');
|
|
313
|
+
console.log(` ${logger_1.logger.cyan(`https://${result.domain}`)}`);
|
|
332
314
|
}
|
|
333
315
|
else {
|
|
334
|
-
logger_1.logger.
|
|
316
|
+
console.log(` Status: ${logger_1.logger.yellow('⏳ Pending verification')}`);
|
|
335
317
|
console.log('');
|
|
318
|
+
// Show what DNS records are needed
|
|
319
|
+
if (result.verification && result.verification.length > 0) {
|
|
320
|
+
console.log(' Required DNS records:');
|
|
321
|
+
console.log('');
|
|
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
|
+
});
|
|
328
|
+
}
|
|
336
329
|
console.log(` ${result.message}`);
|
|
337
330
|
console.log('');
|
|
338
|
-
logger_1.logger.info('DNS changes can take
|
|
331
|
+
logger_1.logger.info('DNS changes can take up to 48 hours to propagate.');
|
|
339
332
|
logger_1.logger.info('Run "devdoc domain verify" again later.');
|
|
340
333
|
}
|
|
341
334
|
console.log('');
|
|
@@ -404,4 +397,4 @@ async function domainRemove(customDomain, options) {
|
|
|
404
397
|
process.exit(1);
|
|
405
398
|
}
|
|
406
399
|
}
|
|
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"]}
|
|
400
|
+
//# 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  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
|
@@ -123,14 +123,15 @@ async function buildCombinedEndpointIndexFromBlob(collection, navigationTabs, pr
|
|
|
123
123
|
if (tab.type === 'graphql' && tab.graphqlSchemas) {
|
|
124
124
|
for (const schemaConfig of tab.graphqlSchemas){
|
|
125
125
|
try {
|
|
126
|
-
// Load GraphQL schema from
|
|
127
|
-
const
|
|
128
|
-
)
|
|
129
|
-
|
|
130
|
-
|
|
126
|
+
// Load GraphQL schema from dedicated graphqlSchemas map (like OpenAPI specs)
|
|
127
|
+
const schemaPath = schemaConfig.schema;
|
|
128
|
+
const schemaContent = projectContent.graphqlSchemas?.[schemaPath] || projectContent.graphqlSchemas?.[schemaPath.replace(/^\//, '')] // Try without leading slash
|
|
129
|
+
;
|
|
130
|
+
if (!schemaContent) {
|
|
131
|
+
console.log('[Collections] GraphQL schema not found in blob:', schemaPath, 'Available:', Object.keys(projectContent.graphqlSchemas || {}));
|
|
131
132
|
continue;
|
|
132
133
|
}
|
|
133
|
-
const graphqlCollection = await importGraphQLSchema(
|
|
134
|
+
const graphqlCollection = await importGraphQLSchema(schemaContent, {
|
|
134
135
|
name: schemaConfig.name || 'GraphQL API',
|
|
135
136
|
endpoint: schemaConfig.endpoint || '/graphql'
|
|
136
137
|
});
|
|
@@ -475,6 +476,14 @@ function getOpenApiSpec(specPath) {
|
|
|
475
476
|
console.error('[Collections] Failed to parse theme.json:', e);
|
|
476
477
|
}
|
|
477
478
|
}
|
|
479
|
+
// Load custom CSS from project files if specified in theme.json
|
|
480
|
+
let customCssContent = null;
|
|
481
|
+
if (themeConfig?.customCss) {
|
|
482
|
+
const cssFile = projectContent.files.find((f)=>f.path === themeConfig.customCss);
|
|
483
|
+
if (cssFile) {
|
|
484
|
+
customCssContent = cssFile.content;
|
|
485
|
+
}
|
|
486
|
+
}
|
|
478
487
|
// Build navigation tabs and doc groups from config
|
|
479
488
|
// Multi-tenant projects are always in production mode (deployed), so devMode=false
|
|
480
489
|
const navigationTabs = buildNavigationTabs(docsConfig, false);
|
|
@@ -540,7 +549,7 @@ function getOpenApiSpec(specPath) {
|
|
|
540
549
|
docsNavbar: themeConfig?.navbar || null,
|
|
541
550
|
docsColors: themeConfig?.colors || null,
|
|
542
551
|
defaultTheme: themeConfig?.defaultTheme || null,
|
|
543
|
-
customCss:
|
|
552
|
+
customCss: customCssContent,
|
|
544
553
|
apiVersions,
|
|
545
554
|
selectedApiVersion: selectedVersion,
|
|
546
555
|
notice: docsConfig.notice || null,
|
|
@@ -21,7 +21,7 @@ import { purgeProjectCache } from '@/lib/cache/purge';
|
|
|
21
21
|
try {
|
|
22
22
|
const body = await request.json();
|
|
23
23
|
// Validate request body
|
|
24
|
-
const { name, slug: existingSlug, docsJson, themeJson, openApiSpecs, files } = body;
|
|
24
|
+
const { name, slug: existingSlug, docsJson, themeJson, openApiSpecs, graphqlSchemas, files } = body;
|
|
25
25
|
if (!name || typeof name !== 'string') {
|
|
26
26
|
return NextResponse.json({
|
|
27
27
|
error: 'Missing or invalid project name'
|
|
@@ -127,9 +127,9 @@ import { purgeProjectCache } from '@/lib/cache/purge';
|
|
|
127
127
|
// Store or update content
|
|
128
128
|
let result;
|
|
129
129
|
if (isUpdate) {
|
|
130
|
-
result = await updateProjectContent(slug, docsJson, validFiles, themeJson, openApiSpecs);
|
|
130
|
+
result = await updateProjectContent(slug, docsJson, validFiles, themeJson, openApiSpecs, graphqlSchemas);
|
|
131
131
|
} else {
|
|
132
|
-
result = await storeProjectContent(slug, name, docsJson, validFiles, themeJson, openApiSpecs);
|
|
132
|
+
result = await storeProjectContent(slug, name, docsJson, validFiles, themeJson, openApiSpecs, graphqlSchemas);
|
|
133
133
|
// Store the API key for new projects
|
|
134
134
|
if (apiKey) {
|
|
135
135
|
await storeProjectApiKey(slug, apiKey);
|
|
@@ -1,12 +1,18 @@
|
|
|
1
1
|
import { NextResponse } from 'next/server';
|
|
2
|
-
import { validateApiKey, addCustomDomain, isCustomDomainRegistered } from '@/lib/storage/blob';
|
|
3
|
-
import { isValidDomain, normalizeDomain
|
|
2
|
+
import { validateApiKey, addCustomDomain, isCustomDomainRegistered, updateCustomDomainStatus } from '@/lib/storage/blob';
|
|
3
|
+
import { isValidDomain, normalizeDomain } from '@/lib/docs/config';
|
|
4
|
+
import { addDomainToProject, isVercelIntegrationEnabled, formatVerificationInstructions } from '@/lib/vercel/domains';
|
|
4
5
|
/**
|
|
5
6
|
* POST /api/domains/add
|
|
6
7
|
*
|
|
7
|
-
* Add a custom domain to a project.
|
|
8
|
+
* Add a custom domain to a project via Vercel Domains API.
|
|
8
9
|
* Each project can have ONE custom domain (free).
|
|
9
10
|
*
|
|
11
|
+
* Requires environment variables:
|
|
12
|
+
* - VERCEL_API_TOKEN (or VERCEL_TOKEN)
|
|
13
|
+
* - VERCEL_PROJECT_ID
|
|
14
|
+
* - VERCEL_TEAM_ID (optional)
|
|
15
|
+
*
|
|
10
16
|
* Headers:
|
|
11
17
|
* Authorization: Bearer <api_key>
|
|
12
18
|
*
|
|
@@ -18,10 +24,9 @@ import { isValidDomain, normalizeDomain, getDnsInstructions } from '@/lib/docs/c
|
|
|
18
24
|
* success: true,
|
|
19
25
|
* domain: "docs.example.com",
|
|
20
26
|
* status: "pending",
|
|
21
|
-
* verification:
|
|
22
|
-
*
|
|
23
|
-
*
|
|
24
|
-
* }
|
|
27
|
+
* verification: [
|
|
28
|
+
* { type: "TXT", name: "_vercel.example.com", value: "vc-domain-verify=..." }
|
|
29
|
+
* ]
|
|
25
30
|
* }
|
|
26
31
|
*/ export async function POST(request) {
|
|
27
32
|
try {
|
|
@@ -63,7 +68,7 @@ import { isValidDomain, normalizeDomain, getDnsInstructions } from '@/lib/docs/c
|
|
|
63
68
|
status: 400
|
|
64
69
|
});
|
|
65
70
|
}
|
|
66
|
-
// Check if domain is already registered
|
|
71
|
+
// Check if domain is already registered in our registry
|
|
67
72
|
const isRegistered = await isCustomDomainRegistered(customDomain);
|
|
68
73
|
if (isRegistered) {
|
|
69
74
|
return NextResponse.json({
|
|
@@ -72,41 +77,78 @@ import { isValidDomain, normalizeDomain, getDnsInstructions } from '@/lib/docs/c
|
|
|
72
77
|
status: 409
|
|
73
78
|
});
|
|
74
79
|
}
|
|
75
|
-
//
|
|
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
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
// Add domain to Vercel project
|
|
89
|
+
const vercelResult = await addDomainToProject(customDomain);
|
|
90
|
+
if (!vercelResult.success) {
|
|
91
|
+
return NextResponse.json({
|
|
92
|
+
error: vercelResult.error,
|
|
93
|
+
vercelError: vercelResult.vercelError
|
|
94
|
+
}, {
|
|
95
|
+
status: 400
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
const vercelDomain = vercelResult.domain;
|
|
99
|
+
// Add to our internal registry with Vercel data
|
|
76
100
|
const result = await addCustomDomain(projectSlug, customDomain);
|
|
77
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);
|
|
78
104
|
return NextResponse.json({
|
|
79
105
|
error: result.error
|
|
80
106
|
}, {
|
|
81
107
|
status: 400
|
|
82
108
|
});
|
|
83
109
|
}
|
|
84
|
-
//
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
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) {
|
|
116
|
+
return NextResponse.json({
|
|
117
|
+
success: true,
|
|
118
|
+
domain: customDomain,
|
|
119
|
+
projectSlug,
|
|
120
|
+
status: 'active',
|
|
121
|
+
verified: true,
|
|
122
|
+
message: 'Domain is already verified and active!'
|
|
123
|
+
});
|
|
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
|
+
];
|
|
88
145
|
return NextResponse.json({
|
|
89
146
|
success: true,
|
|
90
147
|
domain: customDomain,
|
|
91
148
|
projectSlug,
|
|
92
|
-
status:
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
txt: dnsInstructions.txt
|
|
96
|
-
},
|
|
97
|
-
instructions: [
|
|
98
|
-
'Add the following DNS records to your domain:',
|
|
99
|
-
'',
|
|
100
|
-
`1. CNAME Record:`,
|
|
101
|
-
` Name: ${dnsInstructions.cname.name}`,
|
|
102
|
-
` Value: ${dnsInstructions.cname.value}`,
|
|
103
|
-
'',
|
|
104
|
-
`2. TXT Record (for verification):`,
|
|
105
|
-
` Name: ${dnsInstructions.txt.name}`,
|
|
106
|
-
` Value: ${dnsInstructions.txt.value}`,
|
|
107
|
-
'',
|
|
108
|
-
'After adding DNS records, run "devdoc domain verify" to verify.'
|
|
109
|
-
]
|
|
149
|
+
status: 'pending',
|
|
150
|
+
verified: false,
|
|
151
|
+
verification: allRecords
|
|
110
152
|
});
|
|
111
153
|
} catch (error) {
|
|
112
154
|
console.error('[Domains API] Error adding domain:', error);
|
|
@@ -1,10 +1,17 @@
|
|
|
1
1
|
import { NextResponse } from 'next/server';
|
|
2
2
|
import { validateApiKey, getProjectCustomDomain, removeCustomDomain } from '@/lib/storage/blob';
|
|
3
3
|
import { normalizeDomain } from '@/lib/docs/config';
|
|
4
|
+
import { removeDomain as vercelRemoveDomain, isVercelIntegrationEnabled } from '@/lib/vercel/domains';
|
|
4
5
|
/**
|
|
5
6
|
* DELETE /api/domains/remove
|
|
6
7
|
*
|
|
7
|
-
* 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.
|
|
10
|
+
*
|
|
11
|
+
* Requires environment variables:
|
|
12
|
+
* - VERCEL_API_TOKEN (or VERCEL_TOKEN)
|
|
13
|
+
* - VERCEL_PROJECT_ID
|
|
14
|
+
* - VERCEL_TEAM_ID (optional)
|
|
8
15
|
*
|
|
9
16
|
* Headers:
|
|
10
17
|
* Authorization: Bearer <api_key>
|
|
@@ -52,7 +59,23 @@ import { normalizeDomain } from '@/lib/docs/config';
|
|
|
52
59
|
}
|
|
53
60
|
customDomain = projectDomain.customDomain;
|
|
54
61
|
}
|
|
55
|
-
//
|
|
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);
|
|
77
|
+
}
|
|
78
|
+
// Remove from internal registry
|
|
56
79
|
const result = await removeCustomDomain(customDomain, projectSlug);
|
|
57
80
|
if (!result.success) {
|
|
58
81
|
return NextResponse.json({
|
|
@@ -61,8 +84,6 @@ import { normalizeDomain } from '@/lib/docs/config';
|
|
|
61
84
|
status: 400
|
|
62
85
|
});
|
|
63
86
|
}
|
|
64
|
-
// TODO: In production, also remove from Vercel via API
|
|
65
|
-
// await vercelApi.removeDomain(customDomain)
|
|
66
87
|
return NextResponse.json({
|
|
67
88
|
success: true,
|
|
68
89
|
message: `Domain ${customDomain} removed successfully`,
|