@brainfish-ai/devdoc 0.1.23 → 0.1.24

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.
@@ -11,6 +11,38 @@ const net_1 = __importDefault(require("net"));
11
11
  const config_1 = require("../../config");
12
12
  const logger_1 = require("../../utils/logger");
13
13
  const paths_1 = require("../../utils/paths");
14
+ /**
15
+ * Open URL in the default browser (cross-platform)
16
+ */
17
+ async function openBrowser(url) {
18
+ const { platform } = process;
19
+ try {
20
+ let command;
21
+ let args;
22
+ switch (platform) {
23
+ case 'darwin': // macOS
24
+ command = 'open';
25
+ args = [url];
26
+ break;
27
+ case 'win32': // Windows
28
+ command = 'cmd';
29
+ args = ['/c', 'start', '', url];
30
+ break;
31
+ default: // Linux and others
32
+ command = 'xdg-open';
33
+ args = [url];
34
+ break;
35
+ }
36
+ const child = (0, child_process_1.spawn)(command, args, {
37
+ stdio: 'ignore',
38
+ detached: true,
39
+ });
40
+ child.unref();
41
+ }
42
+ catch {
43
+ // Silently fail if browser can't be opened
44
+ }
45
+ }
14
46
  /**
15
47
  * Check if a port is available
16
48
  */
@@ -39,6 +71,132 @@ async function findAvailablePort(startPort, host) {
39
71
  }
40
72
  throw new Error(`No available port found between ${startPort} and ${startPort + maxAttempts - 1}`);
41
73
  }
74
+ /**
75
+ * Create a progress bar string
76
+ */
77
+ function createProgressBar(progress, width = 30) {
78
+ const filled = Math.round(width * progress);
79
+ const empty = width - filled;
80
+ const bar = '█'.repeat(filled) + '░'.repeat(empty);
81
+ const percentage = Math.round(progress * 100);
82
+ return `[${bar}] ${percentage}%`;
83
+ }
84
+ /**
85
+ * Install dependencies with progress display
86
+ */
87
+ async function installWithProgress(cwd) {
88
+ return new Promise((resolve, reject) => {
89
+ const installChild = (0, child_process_1.spawn)('npm', ['install', '--progress'], {
90
+ cwd,
91
+ stdio: ['pipe', 'pipe', 'pipe'],
92
+ shell: true,
93
+ });
94
+ let packagesAdded = 0;
95
+ let totalPackages = 0;
96
+ let currentStep = '';
97
+ let lastLine = '';
98
+ const updateProgress = (line) => {
99
+ // Parse npm output to track progress
100
+ // Match "added X packages" pattern
101
+ const addedMatch = line.match(/added (\d+) packages?/i);
102
+ if (addedMatch) {
103
+ packagesAdded = parseInt(addedMatch[1], 10);
104
+ }
105
+ // Match package count from "reify:package-name: timing" or similar
106
+ const reifyMatch = line.match(/reify:([^:]+)/);
107
+ if (reifyMatch) {
108
+ currentStep = `Installing ${reifyMatch[1].trim()}`;
109
+ }
110
+ // Match "timing reifyNode:node_modules/X Completed in Xms"
111
+ const completedMatch = line.match(/Completed in \d+/);
112
+ if (completedMatch && totalPackages > 0) {
113
+ packagesAdded++;
114
+ }
115
+ // Match initial package count from lockfile or package.json analysis
116
+ const totalMatch = line.match(/(\d+) packages? (?:are|to be|will be)/i);
117
+ if (totalMatch) {
118
+ totalPackages = parseInt(totalMatch[1], 10);
119
+ }
120
+ // Match "idealTree" phase
121
+ if (line.includes('idealTree')) {
122
+ currentStep = 'Resolving dependencies...';
123
+ }
124
+ // Match "reify" phase start
125
+ if (line.includes('reify:') && !currentStep.startsWith('Installing')) {
126
+ currentStep = 'Installing packages...';
127
+ }
128
+ // Match "audit" phase
129
+ if (line.includes('audit')) {
130
+ currentStep = 'Running security audit...';
131
+ }
132
+ // Only update display if we have meaningful info
133
+ if (currentStep || packagesAdded > 0) {
134
+ // Clear line and show progress
135
+ process.stdout.clearLine(0);
136
+ process.stdout.cursorTo(0);
137
+ if (totalPackages > 0 && packagesAdded > 0) {
138
+ const progress = Math.min(packagesAdded / totalPackages, 1);
139
+ process.stdout.write(` ${createProgressBar(progress)} ${packagesAdded}/${totalPackages} packages`);
140
+ }
141
+ else if (currentStep) {
142
+ process.stdout.write(` ${logger_1.logger.cyan('○')} ${currentStep}`);
143
+ }
144
+ }
145
+ };
146
+ // Process stdout
147
+ installChild.stdout?.on('data', (data) => {
148
+ const lines = data.toString().split('\n');
149
+ for (const line of lines) {
150
+ if (line.trim()) {
151
+ lastLine = line;
152
+ updateProgress(line);
153
+ }
154
+ }
155
+ });
156
+ // Process stderr (npm often outputs progress to stderr)
157
+ installChild.stderr?.on('data', (data) => {
158
+ const lines = data.toString().split('\n');
159
+ for (const line of lines) {
160
+ if (line.trim()) {
161
+ // Skip WARN messages but process progress info
162
+ if (!line.includes('WARN') && !line.includes('npm warn')) {
163
+ lastLine = line;
164
+ updateProgress(line);
165
+ }
166
+ }
167
+ }
168
+ });
169
+ installChild.on('exit', (code) => {
170
+ // Clear the progress line
171
+ process.stdout.clearLine(0);
172
+ process.stdout.cursorTo(0);
173
+ if (code === 0) {
174
+ // Show final summary
175
+ if (packagesAdded > 0) {
176
+ console.log(` ${logger_1.logger.green('✓')} Installed ${packagesAdded} packages`);
177
+ }
178
+ else {
179
+ console.log(` ${logger_1.logger.green('✓')} Installation complete`);
180
+ }
181
+ resolve();
182
+ }
183
+ else {
184
+ console.log(` ${logger_1.logger.red('✗')} Installation failed`);
185
+ if (lastLine) {
186
+ console.log(` ${lastLine}`);
187
+ }
188
+ reject(new Error(`npm install failed with code ${code}`));
189
+ }
190
+ });
191
+ installChild.on('error', (error) => {
192
+ process.stdout.clearLine(0);
193
+ process.stdout.cursorTo(0);
194
+ reject(error);
195
+ });
196
+ // Show initial message
197
+ process.stdout.write(` ${logger_1.logger.cyan('○')} Resolving dependencies...`);
198
+ });
199
+ }
42
200
  async function dev(options) {
43
201
  const projectRoot = process.cwd();
44
202
  logger_1.logger.info('Starting DevDoc development server...');
@@ -61,7 +219,8 @@ async function dev(options) {
61
219
  logger_1.logger.success(`Loaded configuration: ${config.name || 'Untitled'}`);
62
220
  }
63
221
  catch (error) {
64
- logger_1.logger.error(`Failed to load docs.json: ${error.message}`);
222
+ const message = error instanceof Error ? error.message : String(error);
223
+ logger_1.logger.error(`Failed to load docs.json: ${message}`);
65
224
  process.exit(1);
66
225
  }
67
226
  // Get the renderer directory (bundled with this package)
@@ -75,20 +234,9 @@ async function dev(options) {
75
234
  const nodeModulesPath = path_1.default.join(rendererDir, 'node_modules');
76
235
  if (!fs_extra_1.default.existsSync(nodeModulesPath)) {
77
236
  logger_1.logger.info('Installing renderer dependencies (first run)...');
78
- const installChild = (0, child_process_1.spawn)('npm', ['install'], {
79
- cwd: rendererDir,
80
- stdio: 'inherit',
81
- shell: true,
82
- });
83
- await new Promise((resolve, reject) => {
84
- installChild.on('exit', (code) => {
85
- if (code === 0)
86
- resolve();
87
- else
88
- reject(new Error(`npm install failed with code ${code}`));
89
- });
90
- installChild.on('error', reject);
91
- });
237
+ console.log('');
238
+ await installWithProgress(rendererDir);
239
+ console.log('');
92
240
  logger_1.logger.success('Dependencies installed');
93
241
  }
94
242
  // Find available port
@@ -101,7 +249,8 @@ async function dev(options) {
101
249
  }
102
250
  }
103
251
  catch (error) {
104
- logger_1.logger.error(error.message);
252
+ const message = error instanceof Error ? error.message : String(error);
253
+ logger_1.logger.error(message);
105
254
  process.exit(1);
106
255
  }
107
256
  // Set environment variables - use STARTER_PATH with absolute path
@@ -111,8 +260,9 @@ async function dev(options) {
111
260
  PORT: String(actualPort),
112
261
  HOSTNAME: options.host,
113
262
  };
263
+ const serverUrl = `http://${options.host}:${actualPort}`;
114
264
  logger_1.logger.info(`Content directory: ${projectRoot}`);
115
- logger_1.logger.info(`Starting server at http://${options.host}:${actualPort}`);
265
+ logger_1.logger.info(`Starting server at ${serverUrl}`);
116
266
  logger_1.logger.info('Press Ctrl+C to stop\n');
117
267
  // Start Next.js dev server with Turbopack for better path alias support
118
268
  const child = (0, child_process_1.spawn)('npx', ['next', 'dev', '--turbo', '-p', String(actualPort), '-H', options.host], {
@@ -121,6 +271,13 @@ async function dev(options) {
121
271
  stdio: 'inherit',
122
272
  shell: true,
123
273
  });
274
+ // Open browser after a short delay to let the server start
275
+ if (options.open) {
276
+ setTimeout(async () => {
277
+ logger_1.logger.info(`Opening ${serverUrl} in your browser...`);
278
+ await openBrowser(serverUrl);
279
+ }, 3000); // Wait 3 seconds for server to be ready
280
+ }
124
281
  child.on('error', (error) => {
125
282
  logger_1.logger.error(`Failed to start server: ${error.message}`);
126
283
  process.exit(1);
@@ -136,4 +293,4 @@ async function dev(options) {
136
293
  child.kill('SIGTERM');
137
294
  });
138
295
  }
139
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dev.js","sourceRoot":"","sources":["../../../src/cli/commands/dev.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn } from 'child_process';\nimport path from 'path';\nimport fs from 'fs-extra';\nimport net from 'net';\nimport { loadConfig, validateConfig } from '../../config';\nimport { logger } from '../../utils/logger';\nimport { getRendererDir } from '../../utils/paths';\n\ninterface DevOptions {\n  port: string;\n  host: string;\n}\n\n/**\n * Check if a port is available\n */\nasync function isPortAvailable(port: number, host: string): Promise<boolean> {\n  return new Promise((resolve) => {\n    const server = net.createServer();\n    server.once('error', () => resolve(false));\n    server.once('listening', () => {\n      server.close();\n      resolve(true);\n    });\n    server.listen(port, host);\n  });\n}\n\n/**\n * Find an available port starting from the given port\n */\nasync function findAvailablePort(startPort: number, host: string): Promise<number> {\n  let port = startPort;\n  const maxAttempts = 10;\n  \n  for (let i = 0; i < maxAttempts; i++) {\n    if (await isPortAvailable(port, host)) {\n      return port;\n    }\n    port++;\n  }\n  \n  throw new Error(`No available port found between ${startPort} and ${startPort + maxAttempts - 1}`);\n}\n\nexport async function dev(options: DevOptions): Promise<void> {\n  const projectRoot = process.cwd();\n  \n  logger.info('Starting DevDoc development server...');\n  \n  // Check for docs.json\n  const configPath = path.join(projectRoot, 'docs.json');\n  if (!fs.existsSync(configPath)) {\n    logger.error('docs.json not found in current directory');\n    logger.info('Run \"devdoc init\" to create a new documentation project');\n    process.exit(1);\n  }\n  \n  // Load and validate config\n  try {\n    const config = await loadConfig(projectRoot);\n    const validation = validateConfig(config);\n    \n    if (!validation.valid) {\n      logger.error('Invalid docs.json configuration:');\n      validation.errors.forEach(err => logger.error(`  - ${err}`));\n      process.exit(1);\n    }\n    \n    logger.success(`Loaded configuration: ${config.name || 'Untitled'}`);\n  } catch (error: any) {\n    logger.error(`Failed to load docs.json: ${error.message}`);\n    process.exit(1);\n  }\n  \n  // Get the renderer directory (bundled with this package)\n  const rendererDir = getRendererDir();\n  \n  if (!rendererDir || !fs.existsSync(rendererDir)) {\n    logger.error('Renderer not found. Package may be corrupted.');\n    logger.debug(`Looked for renderer at: ${rendererDir}`);\n    process.exit(1);\n  }\n  \n  // Check if renderer has node_modules\n  const nodeModulesPath = path.join(rendererDir, 'node_modules');\n  if (!fs.existsSync(nodeModulesPath)) {\n    logger.info('Installing renderer dependencies (first run)...');\n    const installChild = spawn('npm', ['install'], {\n      cwd: rendererDir,\n      stdio: 'inherit',\n      shell: true,\n    });\n    \n    await new Promise<void>((resolve, reject) => {\n      installChild.on('exit', (code) => {\n        if (code === 0) resolve();\n        else reject(new Error(`npm install failed with code ${code}`));\n      });\n      installChild.on('error', reject);\n    });\n    logger.success('Dependencies installed');\n  }\n  \n  // Find available port\n  const requestedPort = parseInt(options.port, 10);\n  let actualPort: number;\n  \n  try {\n    actualPort = await findAvailablePort(requestedPort, options.host);\n    if (actualPort !== requestedPort) {\n      logger.warn(`Port ${requestedPort} is in use, using ${actualPort} instead`);\n    }\n  } catch (error: any) {\n    logger.error(error.message);\n    process.exit(1);\n  }\n  \n  // Set environment variables - use STARTER_PATH with absolute path\n  const env = {\n    ...process.env,\n    STARTER_PATH: projectRoot, // Absolute path to user's project\n    PORT: String(actualPort),\n    HOSTNAME: options.host,\n  };\n  \n  logger.info(`Content directory: ${projectRoot}`);\n  logger.info(`Starting server at http://${options.host}:${actualPort}`);\n  logger.info('Press Ctrl+C to stop\\n');\n  \n  // Start Next.js dev server with Turbopack for better path alias support\n  const child = spawn(\n    'npx',\n    ['next', 'dev', '--turbo', '-p', String(actualPort), '-H', options.host],\n    {\n      cwd: rendererDir,\n      env,\n      stdio: 'inherit',\n      shell: true,\n    }\n  );\n  \n  child.on('error', (error) => {\n    logger.error(`Failed to start server: ${error.message}`);\n    process.exit(1);\n  });\n  \n  child.on('exit', (code) => {\n    process.exit(code || 0);\n  });\n  \n  // Handle termination\n  process.on('SIGINT', () => {\n    child.kill('SIGINT');\n  });\n  \n  process.on('SIGTERM', () => {\n    child.kill('SIGTERM');\n  });\n}\n"]}
296
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dev.js","sourceRoot":"","sources":["../../../src/cli/commands/dev.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn } from 'child_process';\nimport path from 'path';\nimport fs from 'fs-extra';\nimport net from 'net';\nimport { loadConfig, validateConfig } from '../../config';\nimport { logger } from '../../utils/logger';\nimport { getRendererDir } from '../../utils/paths';\n\ninterface DevOptions {\n  port: string;\n  host: string;\n  open: boolean;\n}\n\n/**\n * Open URL in the default browser (cross-platform)\n */\nasync function openBrowser(url: string): Promise<void> {\n  const { platform } = process;\n  \n  try {\n    let command: string;\n    let args: string[];\n    \n    switch (platform) {\n      case 'darwin': // macOS\n        command = 'open';\n        args = [url];\n        break;\n      case 'win32': // Windows\n        command = 'cmd';\n        args = ['/c', 'start', '', url];\n        break;\n      default: // Linux and others\n        command = 'xdg-open';\n        args = [url];\n        break;\n    }\n    \n    const child = spawn(command, args, {\n      stdio: 'ignore',\n      detached: true,\n    });\n    \n    child.unref();\n  } catch {\n    // Silently fail if browser can't be opened\n  }\n}\n\n/**\n * Check if a port is available\n */\nasync function isPortAvailable(port: number, host: string): Promise<boolean> {\n  return new Promise((resolve) => {\n    const server = net.createServer();\n    server.once('error', () => resolve(false));\n    server.once('listening', () => {\n      server.close();\n      resolve(true);\n    });\n    server.listen(port, host);\n  });\n}\n\n/**\n * Find an available port starting from the given port\n */\nasync function findAvailablePort(startPort: number, host: string): Promise<number> {\n  let port = startPort;\n  const maxAttempts = 10;\n  \n  for (let i = 0; i < maxAttempts; i++) {\n    if (await isPortAvailable(port, host)) {\n      return port;\n    }\n    port++;\n  }\n  \n  throw new Error(`No available port found between ${startPort} and ${startPort + maxAttempts - 1}`);\n}\n\n/**\n * Create a progress bar string\n */\nfunction createProgressBar(progress: number, width: number = 30): string {\n  const filled = Math.round(width * progress);\n  const empty = width - filled;\n  const bar = '█'.repeat(filled) + '░'.repeat(empty);\n  const percentage = Math.round(progress * 100);\n  return `[${bar}] ${percentage}%`;\n}\n\n/**\n * Install dependencies with progress display\n */\nasync function installWithProgress(cwd: string): Promise<void> {\n  return new Promise((resolve, reject) => {\n    const installChild = spawn('npm', ['install', '--progress'], {\n      cwd,\n      stdio: ['pipe', 'pipe', 'pipe'],\n      shell: true,\n    });\n    \n    let packagesAdded = 0;\n    let totalPackages = 0;\n    let currentStep = '';\n    let lastLine = '';\n    \n    const updateProgress = (line: string) => {\n      // Parse npm output to track progress\n      \n      // Match \"added X packages\" pattern\n      const addedMatch = line.match(/added (\\d+) packages?/i);\n      if (addedMatch) {\n        packagesAdded = parseInt(addedMatch[1], 10);\n      }\n      \n      // Match package count from \"reify:package-name: timing\" or similar\n      const reifyMatch = line.match(/reify:([^:]+)/);\n      if (reifyMatch) {\n        currentStep = `Installing ${reifyMatch[1].trim()}`;\n      }\n      \n      // Match \"timing reifyNode:node_modules/X Completed in Xms\"\n      const completedMatch = line.match(/Completed in \\d+/);\n      if (completedMatch && totalPackages > 0) {\n        packagesAdded++;\n      }\n      \n      // Match initial package count from lockfile or package.json analysis\n      const totalMatch = line.match(/(\\d+) packages? (?:are|to be|will be)/i);\n      if (totalMatch) {\n        totalPackages = parseInt(totalMatch[1], 10);\n      }\n      \n      // Match \"idealTree\" phase\n      if (line.includes('idealTree')) {\n        currentStep = 'Resolving dependencies...';\n      }\n      \n      // Match \"reify\" phase start\n      if (line.includes('reify:') && !currentStep.startsWith('Installing')) {\n        currentStep = 'Installing packages...';\n      }\n      \n      // Match \"audit\" phase\n      if (line.includes('audit')) {\n        currentStep = 'Running security audit...';\n      }\n      \n      // Only update display if we have meaningful info\n      if (currentStep || packagesAdded > 0) {\n        // Clear line and show progress\n        process.stdout.clearLine(0);\n        process.stdout.cursorTo(0);\n        \n        if (totalPackages > 0 && packagesAdded > 0) {\n          const progress = Math.min(packagesAdded / totalPackages, 1);\n          process.stdout.write(`  ${createProgressBar(progress)} ${packagesAdded}/${totalPackages} packages`);\n        } else if (currentStep) {\n          process.stdout.write(`  ${logger.cyan('○')} ${currentStep}`);\n        }\n      }\n    };\n    \n    // Process stdout\n    installChild.stdout?.on('data', (data: Buffer) => {\n      const lines = data.toString().split('\\n');\n      for (const line of lines) {\n        if (line.trim()) {\n          lastLine = line;\n          updateProgress(line);\n        }\n      }\n    });\n    \n    // Process stderr (npm often outputs progress to stderr)\n    installChild.stderr?.on('data', (data: Buffer) => {\n      const lines = data.toString().split('\\n');\n      for (const line of lines) {\n        if (line.trim()) {\n          // Skip WARN messages but process progress info\n          if (!line.includes('WARN') && !line.includes('npm warn')) {\n            lastLine = line;\n            updateProgress(line);\n          }\n        }\n      }\n    });\n    \n    installChild.on('exit', (code) => {\n      // Clear the progress line\n      process.stdout.clearLine(0);\n      process.stdout.cursorTo(0);\n      \n      if (code === 0) {\n        // Show final summary\n        if (packagesAdded > 0) {\n          console.log(`  ${logger.green('✓')} Installed ${packagesAdded} packages`);\n        } else {\n          console.log(`  ${logger.green('✓')} Installation complete`);\n        }\n        resolve();\n      } else {\n        console.log(`  ${logger.red('✗')} Installation failed`);\n        if (lastLine) {\n          console.log(`  ${lastLine}`);\n        }\n        reject(new Error(`npm install failed with code ${code}`));\n      }\n    });\n    \n    installChild.on('error', (error) => {\n      process.stdout.clearLine(0);\n      process.stdout.cursorTo(0);\n      reject(error);\n    });\n    \n    // Show initial message\n    process.stdout.write(`  ${logger.cyan('○')} Resolving dependencies...`);\n  });\n}\n\nexport async function dev(options: DevOptions): Promise<void> {\n  const projectRoot = process.cwd();\n  \n  logger.info('Starting DevDoc development server...');\n  \n  // Check for docs.json\n  const configPath = path.join(projectRoot, 'docs.json');\n  if (!fs.existsSync(configPath)) {\n    logger.error('docs.json not found in current directory');\n    logger.info('Run \"devdoc init\" to create a new documentation project');\n    process.exit(1);\n  }\n  \n  // Load and validate config\n  try {\n    const config = await loadConfig(projectRoot);\n    const validation = validateConfig(config);\n    \n    if (!validation.valid) {\n      logger.error('Invalid docs.json configuration:');\n      validation.errors.forEach(err => logger.error(`  - ${err}`));\n      process.exit(1);\n    }\n    \n    logger.success(`Loaded configuration: ${config.name || 'Untitled'}`);\n  } catch (error: unknown) {\n    const message = error instanceof Error ? error.message : String(error);\n    logger.error(`Failed to load docs.json: ${message}`);\n    process.exit(1);\n  }\n  \n  // Get the renderer directory (bundled with this package)\n  const rendererDir = getRendererDir();\n  \n  if (!rendererDir || !fs.existsSync(rendererDir)) {\n    logger.error('Renderer not found. Package may be corrupted.');\n    logger.debug(`Looked for renderer at: ${rendererDir}`);\n    process.exit(1);\n  }\n  \n  // Check if renderer has node_modules\n  const nodeModulesPath = path.join(rendererDir, 'node_modules');\n  if (!fs.existsSync(nodeModulesPath)) {\n    logger.info('Installing renderer dependencies (first run)...');\n    console.log('');\n    \n    await installWithProgress(rendererDir);\n    \n    console.log('');\n    logger.success('Dependencies installed');\n  }\n  \n  // Find available port\n  const requestedPort = parseInt(options.port, 10);\n  let actualPort: number;\n  \n  try {\n    actualPort = await findAvailablePort(requestedPort, options.host);\n    if (actualPort !== requestedPort) {\n      logger.warn(`Port ${requestedPort} is in use, using ${actualPort} instead`);\n    }\n  } catch (error: unknown) {\n    const message = error instanceof Error ? error.message : String(error);\n    logger.error(message);\n    process.exit(1);\n  }\n  \n  // Set environment variables - use STARTER_PATH with absolute path\n  const env = {\n    ...process.env,\n    STARTER_PATH: projectRoot, // Absolute path to user's project\n    PORT: String(actualPort),\n    HOSTNAME: options.host,\n  };\n  \n  const serverUrl = `http://${options.host}:${actualPort}`;\n  \n  logger.info(`Content directory: ${projectRoot}`);\n  logger.info(`Starting server at ${serverUrl}`);\n  logger.info('Press Ctrl+C to stop\\n');\n  \n  // Start Next.js dev server with Turbopack for better path alias support\n  const child = spawn(\n    'npx',\n    ['next', 'dev', '--turbo', '-p', String(actualPort), '-H', options.host],\n    {\n      cwd: rendererDir,\n      env,\n      stdio: 'inherit',\n      shell: true,\n    }\n  );\n  \n  // Open browser after a short delay to let the server start\n  if (options.open) {\n    setTimeout(async () => {\n      logger.info(`Opening ${serverUrl} in your browser...`);\n      await openBrowser(serverUrl);\n    }, 3000); // Wait 3 seconds for server to be ready\n  }\n  \n  child.on('error', (error) => {\n    logger.error(`Failed to start server: ${error.message}`);\n    process.exit(1);\n  });\n  \n  child.on('exit', (code) => {\n    process.exit(code || 0);\n  });\n  \n  // Handle termination\n  process.on('SIGINT', () => {\n    child.kill('SIGINT');\n  });\n  \n  process.on('SIGTERM', () => {\n    child.kill('SIGTERM');\n  });\n}\n"]}
@@ -5,7 +5,8 @@ interface InitOptions {
5
5
  url?: string;
6
6
  }
7
7
  /**
8
- * Initialize a DevDoc project with .devdoc.json and register with Brainfish
8
+ * Initialize a DevDoc project with .devdoc.json
9
+ * Note: Subdomain is only reserved when documentation is actually deployed (devdoc deploy)
9
10
  */
10
11
  export declare function init(options: InitOptions): Promise<void>;
11
12
  export {};
@@ -93,13 +93,14 @@ async function checkSubdomainAvailability(subdomain, apiUrl) {
93
93
  const result = await response.json();
94
94
  return result;
95
95
  }
96
- catch (error) {
96
+ catch {
97
97
  // If API is unavailable, allow proceeding (will fail at registration if invalid)
98
98
  return { available: true };
99
99
  }
100
100
  }
101
101
  /**
102
- * Initialize a DevDoc project with .devdoc.json and register with Brainfish
102
+ * Initialize a DevDoc project with .devdoc.json
103
+ * Note: Subdomain is only reserved when documentation is actually deployed (devdoc deploy)
103
104
  */
104
105
  async function init(options) {
105
106
  const projectRoot = process.cwd();
@@ -112,12 +113,12 @@ async function init(options) {
112
113
  logger_1.logger.info('Make sure you are in a DevDoc documentation project directory');
113
114
  process.exit(1);
114
115
  }
115
- // Check if .devdoc.json already exists with API key
116
+ // Check if .devdoc.json already exists with API key (already deployed)
116
117
  const devdocConfigPath = path_1.default.join(projectRoot, '.devdoc.json');
117
118
  if (fs_extra_1.default.existsSync(devdocConfigPath) && !options.force) {
118
119
  const existingConfig = fs_extra_1.default.readJsonSync(devdocConfigPath);
119
120
  if (existingConfig.apiKey) {
120
- logger_1.logger.warn('.devdoc.json already exists with API key');
121
+ logger_1.logger.warn('.devdoc.json already exists with API key (project is deployed)');
121
122
  logger_1.logger.info('Use --force to overwrite and create a new project');
122
123
  process.exit(1);
123
124
  }
@@ -148,6 +149,7 @@ async function init(options) {
148
149
  process.exit(1);
149
150
  }
150
151
  // Check subdomain availability via API (server validates blacklist)
152
+ // Note: This only checks availability, it does NOT reserve the subdomain
151
153
  logger_1.logger.info(`Checking if ${subdomain}.devdoc.sh is available...`);
152
154
  const availability = await checkSubdomainAvailability(subdomain, apiUrl);
153
155
  if (!availability.available) {
@@ -159,65 +161,25 @@ async function init(options) {
159
161
  }
160
162
  logger_1.logger.success(`✓ ${subdomain}.devdoc.sh is available!`);
161
163
  console.log('');
162
- // Register project with Brainfish API
163
- logger_1.logger.info('Registering project with Brainfish...');
164
- try {
165
- const response = await fetch(`${apiUrl}/api/projects/register`, {
166
- method: 'POST',
167
- headers: {
168
- 'Content-Type': 'application/json',
169
- },
170
- body: JSON.stringify({
171
- name: projectName,
172
- slug,
173
- subdomain,
174
- }),
175
- });
176
- if (!response.ok) {
177
- const errorData = await response.json().catch(() => ({ error: 'Unknown error' }));
178
- const errorMessage = errorData.error || `HTTP ${response.status}`;
179
- const details = errorData.details ? `\n Details: ${errorData.details}` : '';
180
- throw new Error(`${errorMessage}${details}`);
181
- }
182
- const result = await response.json();
183
- // Create .devdoc.json with API key
184
- const devdocConfig = {
185
- projectId: result.projectId,
186
- name: projectName,
187
- slug: result.slug,
188
- subdomain: result.subdomain,
189
- apiKey: result.apiKey,
190
- createdAt: new Date().toISOString(),
191
- };
192
- fs_extra_1.default.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });
193
- logger_1.logger.success('✓ Project registered successfully');
194
- console.log('');
195
- console.log(' Project ID:', result.projectId);
196
- console.log(' URL:', `https://${result.subdomain}.devdoc.sh`);
197
- console.log(' API Key:', result.apiKey);
198
- console.log('');
199
- logger_1.logger.warn('⚠️ API key saved to .devdoc.json - add to .gitignore!');
200
- console.log('');
201
- logger_1.logger.info('Run "devdoc deploy" to deploy your documentation');
202
- }
203
- catch (error) {
204
- const message = error instanceof Error ? error.message : String(error);
205
- logger_1.logger.error(`Failed to register project: ${message}`);
206
- // Create local .devdoc.json without API key (offline mode)
207
- logger_1.logger.warn('Creating local config without API key...');
208
- const projectId = `${slug}-${generateId()}`;
209
- const devdocConfig = {
210
- projectId,
211
- name: projectName,
212
- slug,
213
- subdomain,
214
- createdAt: new Date().toISOString(),
215
- };
216
- fs_extra_1.default.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });
217
- logger_1.logger.success('✓ Created .devdoc.json (offline mode)');
218
- console.log('');
219
- logger_1.logger.info('Run "devdoc init" again when online to register and get API key');
220
- }
164
+ // Create .devdoc.json with subdomain (but NO API key - not registered yet)
165
+ // The subdomain will only be reserved when the user actually deploys
166
+ const projectId = `${slug}-${generateId()}`;
167
+ const devdocConfig = {
168
+ projectId,
169
+ name: projectName,
170
+ slug,
171
+ subdomain,
172
+ createdAt: new Date().toISOString(),
173
+ // Note: No apiKey - subdomain is not reserved until deploy
174
+ };
175
+ fs_extra_1.default.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 });
176
+ logger_1.logger.success('✓ Project initialized');
177
+ console.log('');
178
+ console.log(' Subdomain:', `${subdomain}.devdoc.sh`);
179
+ console.log(' Status:', 'Not yet deployed');
180
+ console.log('');
181
+ logger_1.logger.info('Note: The subdomain is not reserved until you deploy.');
182
+ logger_1.logger.info('Run "devdoc deploy" to deploy and claim your subdomain.');
221
183
  }
222
184
  /**
223
185
  * Generate a URL-friendly slug from a name
@@ -235,4 +197,4 @@ function generateSlug(name) {
235
197
  function generateId() {
236
198
  return Math.random().toString(36).substring(2, 8);
237
199
  }
238
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"init.js","sourceRoot":"","sources":["../../../src/cli/commands/init.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path'\nimport fs from 'fs-extra'\nimport { loadConfig } from '../../config'\nimport { logger } from '../../utils/logger'\nimport { DEFAULT_API_URL } from '../../constants'\n\ninterface InitOptions {\n  slug?: string\n  subdomain?: string\n  force?: boolean\n  url?: string\n}\n\ninterface DevDocConfig {\n  projectId: string\n  name: string\n  slug: string\n  subdomain: string\n  apiKey?: string\n  createdAt: string\n}\n\ninterface RegisterResponse {\n  success: boolean\n  projectId: string\n  slug: string\n  subdomain: string\n  apiKey: string\n  error?: string\n}\n\ninterface CheckSubdomainResponse {\n  available: boolean\n  error?: string\n  suggestion?: string\n}\n\n// Simple prompt helper using readline\nasync function prompt(question: string, defaultValue?: string): Promise<string> {\n  const readline = await import('readline')\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  })\n\n  return new Promise((resolve) => {\n    const defaultHint = defaultValue ? ` (${defaultValue})` : ''\n    rl.question(`${question}${defaultHint}: `, (answer) => {\n      rl.close()\n      resolve(answer.trim() || defaultValue || '')\n    })\n  })\n}\n\n/**\n * Basic subdomain format validation (server does full validation)\n */\nfunction isValidSubdomainFormat(subdomain: string): { valid: boolean; error?: string } {\n  if (!subdomain) {\n    return { valid: false, error: 'Subdomain is required' }\n  }\n  \n  if (subdomain.length < 3) {\n    return { valid: false, error: 'Subdomain must be at least 3 characters' }\n  }\n  \n  if (subdomain.length > 63) {\n    return { valid: false, error: 'Subdomain must be 63 characters or less' }\n  }\n  \n  if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain must start and end with alphanumeric, can contain hyphens' }\n  }\n  \n  if (/--/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain cannot contain consecutive hyphens' }\n  }\n  \n  return { valid: true }\n}\n\n/**\n * Check subdomain availability via API (also validates against server blacklist)\n */\nasync function checkSubdomainAvailability(\n  subdomain: string,\n  apiUrl: string\n): Promise<CheckSubdomainResponse> {\n  try {\n    const response = await fetch(`${apiUrl}/api/subdomains/check`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify({ subdomain }),\n    })\n    \n    const result = await response.json() as CheckSubdomainResponse\n    return result\n  } catch (error) {\n    // If API is unavailable, allow proceeding (will fail at registration if invalid)\n    return { available: true }\n  }\n}\n\n/**\n * Initialize a DevDoc project with .devdoc.json and register with Brainfish\n */\nexport async function init(options: InitOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Initializing DevDoc project...\\n')\n  \n  // Check for docs.json\n  const configPath = path.join(projectRoot, 'docs.json')\n  if (!fs.existsSync(configPath)) {\n    logger.error('docs.json not found in current directory')\n    logger.info('Make sure you are in a DevDoc documentation project directory')\n    process.exit(1)\n  }\n  \n  // Check if .devdoc.json already exists with API key\n  const devdocConfigPath = path.join(projectRoot, '.devdoc.json')\n  if (fs.existsSync(devdocConfigPath) && !options.force) {\n    const existingConfig = fs.readJsonSync(devdocConfigPath) as DevDocConfig\n    if (existingConfig.apiKey) {\n      logger.warn('.devdoc.json already exists with API key')\n      logger.info('Use --force to overwrite and create a new project')\n      process.exit(1)\n    }\n  }\n  \n  // Load docs.json to get project name\n  let projectName = 'My Documentation'\n  try {\n    const config = await loadConfig(projectRoot)\n    projectName = config.name || projectName\n  } catch {\n    // Use default name\n  }\n  \n  // Generate slug from project name or use provided slug\n  const slug = options.slug || generateSlug(projectName)\n  \n  // Get subdomain - prompt if not provided\n  let subdomain = options.subdomain\n  \n  if (!subdomain) {\n    const suggestedSubdomain = slug\n    console.log('')\n    subdomain = await prompt(`Enter subdomain for ${suggestedSubdomain}.devdoc.sh`, suggestedSubdomain)\n    subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '')\n  }\n  \n  // Validate subdomain format locally\n  const formatCheck = isValidSubdomainFormat(subdomain)\n  if (!formatCheck.valid) {\n    logger.error(formatCheck.error!)\n    process.exit(1)\n  }\n  \n  // Check subdomain availability via API (server validates blacklist)\n  logger.info(`Checking if ${subdomain}.devdoc.sh is available...`)\n  const availability = await checkSubdomainAvailability(subdomain, apiUrl)\n  \n  if (!availability.available) {\n    logger.error(availability.error || `Subdomain \"${subdomain}\" is not available`)\n    if (availability.suggestion) {\n      logger.info(`Suggestion: Try \"${availability.suggestion}\"`)\n    }\n    process.exit(1)\n  }\n  \n  logger.success(`✓ ${subdomain}.devdoc.sh is available!`)\n  console.log('')\n  \n  // Register project with Brainfish API\n  logger.info('Registering project with Brainfish...')\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/projects/register`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify({\n        name: projectName,\n        slug,\n        subdomain,\n      }),\n    })\n    \n    if (!response.ok) {\n      const errorData = await response.json().catch(() => ({ error: 'Unknown error' })) as { error?: string; details?: string }\n      const errorMessage = errorData.error || `HTTP ${response.status}`\n      const details = errorData.details ? `\\n   Details: ${errorData.details}` : ''\n      throw new Error(`${errorMessage}${details}`)\n    }\n    \n    const result = await response.json() as RegisterResponse\n    \n    // Create .devdoc.json with API key\n    const devdocConfig: DevDocConfig = {\n      projectId: result.projectId,\n      name: projectName,\n      slug: result.slug,\n      subdomain: result.subdomain,\n      apiKey: result.apiKey,\n      createdAt: new Date().toISOString(),\n    }\n    \n    fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 })\n    \n    logger.success('✓ Project registered successfully')\n    console.log('')\n    console.log('  Project ID:', result.projectId)\n    console.log('  URL:', `https://${result.subdomain}.devdoc.sh`)\n    console.log('  API Key:', result.apiKey)\n    console.log('')\n    logger.warn('⚠️  API key saved to .devdoc.json - add to .gitignore!')\n    console.log('')\n    logger.info('Run \"devdoc deploy\" to deploy your documentation')\n    \n  } catch (error: unknown) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to register project: ${message}`)\n    \n    // Create local .devdoc.json without API key (offline mode)\n    logger.warn('Creating local config without API key...')\n    const projectId = `${slug}-${generateId()}`\n    const devdocConfig: DevDocConfig = {\n      projectId,\n      name: projectName,\n      slug,\n      subdomain,\n      createdAt: new Date().toISOString(),\n    }\n    \n    fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 })\n    \n    logger.success('✓ Created .devdoc.json (offline mode)')\n    console.log('')\n    logger.info('Run \"devdoc init\" again when online to register and get API key')\n  }\n}\n\n/**\n * Generate a URL-friendly slug from a name\n */\nfunction generateSlug(name: string): string {\n  return name\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, '-')\n    .replace(/^-|-$/g, '')\n    .substring(0, 50)\n}\n\n/**\n * Generate a short random ID\n */\nfunction generateId(): string {\n  return Math.random().toString(36).substring(2, 8)\n}\n"]}
200
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"init.js","sourceRoot":"","sources":["../../../src/cli/commands/init.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path'\nimport fs from 'fs-extra'\nimport { loadConfig } from '../../config'\nimport { logger } from '../../utils/logger'\nimport { DEFAULT_API_URL } from '../../constants'\n\ninterface InitOptions {\n  slug?: string\n  subdomain?: string\n  force?: boolean\n  url?: string\n}\n\ninterface DevDocConfig {\n  projectId: string\n  name: string\n  slug: string\n  subdomain: string\n  apiKey?: string\n  createdAt: string\n}\n\ninterface CheckSubdomainResponse {\n  available: boolean\n  error?: string\n  suggestion?: string\n}\n\n// Simple prompt helper using readline\nasync function prompt(question: string, defaultValue?: string): Promise<string> {\n  const readline = await import('readline')\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  })\n\n  return new Promise((resolve) => {\n    const defaultHint = defaultValue ? ` (${defaultValue})` : ''\n    rl.question(`${question}${defaultHint}: `, (answer) => {\n      rl.close()\n      resolve(answer.trim() || defaultValue || '')\n    })\n  })\n}\n\n/**\n * Basic subdomain format validation (server does full validation)\n */\nfunction isValidSubdomainFormat(subdomain: string): { valid: boolean; error?: string } {\n  if (!subdomain) {\n    return { valid: false, error: 'Subdomain is required' }\n  }\n  \n  if (subdomain.length < 3) {\n    return { valid: false, error: 'Subdomain must be at least 3 characters' }\n  }\n  \n  if (subdomain.length > 63) {\n    return { valid: false, error: 'Subdomain must be 63 characters or less' }\n  }\n  \n  if (!/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain must start and end with alphanumeric, can contain hyphens' }\n  }\n  \n  if (/--/.test(subdomain)) {\n    return { valid: false, error: 'Subdomain cannot contain consecutive hyphens' }\n  }\n  \n  return { valid: true }\n}\n\n/**\n * Check subdomain availability via API (also validates against server blacklist)\n */\nasync function checkSubdomainAvailability(\n  subdomain: string,\n  apiUrl: string\n): Promise<CheckSubdomainResponse> {\n  try {\n    const response = await fetch(`${apiUrl}/api/subdomains/check`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify({ subdomain }),\n    })\n    \n    const result = await response.json() as CheckSubdomainResponse\n    return result\n  } catch {\n    // If API is unavailable, allow proceeding (will fail at registration if invalid)\n    return { available: true }\n  }\n}\n\n/**\n * Initialize a DevDoc project with .devdoc.json\n * Note: Subdomain is only reserved when documentation is actually deployed (devdoc deploy)\n */\nexport async function init(options: InitOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  logger.info('Initializing DevDoc project...\\n')\n  \n  // Check for docs.json\n  const configPath = path.join(projectRoot, 'docs.json')\n  if (!fs.existsSync(configPath)) {\n    logger.error('docs.json not found in current directory')\n    logger.info('Make sure you are in a DevDoc documentation project directory')\n    process.exit(1)\n  }\n  \n  // Check if .devdoc.json already exists with API key (already deployed)\n  const devdocConfigPath = path.join(projectRoot, '.devdoc.json')\n  if (fs.existsSync(devdocConfigPath) && !options.force) {\n    const existingConfig = fs.readJsonSync(devdocConfigPath) as DevDocConfig\n    if (existingConfig.apiKey) {\n      logger.warn('.devdoc.json already exists with API key (project is deployed)')\n      logger.info('Use --force to overwrite and create a new project')\n      process.exit(1)\n    }\n  }\n  \n  // Load docs.json to get project name\n  let projectName = 'My Documentation'\n  try {\n    const config = await loadConfig(projectRoot)\n    projectName = config.name || projectName\n  } catch {\n    // Use default name\n  }\n  \n  // Generate slug from project name or use provided slug\n  const slug = options.slug || generateSlug(projectName)\n  \n  // Get subdomain - prompt if not provided\n  let subdomain = options.subdomain\n  \n  if (!subdomain) {\n    const suggestedSubdomain = slug\n    console.log('')\n    subdomain = await prompt(`Enter subdomain for ${suggestedSubdomain}.devdoc.sh`, suggestedSubdomain)\n    subdomain = subdomain.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/^-|-$/g, '')\n  }\n  \n  // Validate subdomain format locally\n  const formatCheck = isValidSubdomainFormat(subdomain)\n  if (!formatCheck.valid) {\n    logger.error(formatCheck.error!)\n    process.exit(1)\n  }\n  \n  // Check subdomain availability via API (server validates blacklist)\n  // Note: This only checks availability, it does NOT reserve the subdomain\n  logger.info(`Checking if ${subdomain}.devdoc.sh is available...`)\n  const availability = await checkSubdomainAvailability(subdomain, apiUrl)\n  \n  if (!availability.available) {\n    logger.error(availability.error || `Subdomain \"${subdomain}\" is not available`)\n    if (availability.suggestion) {\n      logger.info(`Suggestion: Try \"${availability.suggestion}\"`)\n    }\n    process.exit(1)\n  }\n  \n  logger.success(`✓ ${subdomain}.devdoc.sh is available!`)\n  console.log('')\n  \n  // Create .devdoc.json with subdomain (but NO API key - not registered yet)\n  // The subdomain will only be reserved when the user actually deploys\n  const projectId = `${slug}-${generateId()}`\n  const devdocConfig: DevDocConfig = {\n    projectId,\n    name: projectName,\n    slug,\n    subdomain,\n    createdAt: new Date().toISOString(),\n    // Note: No apiKey - subdomain is not reserved until deploy\n  }\n  \n  fs.writeJsonSync(devdocConfigPath, devdocConfig, { spaces: 2 })\n  \n  logger.success('✓ Project initialized')\n  console.log('')\n  console.log('  Subdomain:', `${subdomain}.devdoc.sh`)\n  console.log('  Status:', 'Not yet deployed')\n  console.log('')\n  logger.info('Note: The subdomain is not reserved until you deploy.')\n  logger.info('Run \"devdoc deploy\" to deploy and claim your subdomain.')\n}\n\n/**\n * Generate a URL-friendly slug from a name\n */\nfunction generateSlug(name: string): string {\n  return name\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, '-')\n    .replace(/^-|-$/g, '')\n    .substring(0, 50)\n}\n\n/**\n * Generate a short random ID\n */\nfunction generateId(): string {\n  return Math.random().toString(36).substring(2, 8)\n}\n"]}
package/dist/cli/index.js CHANGED
@@ -39,6 +39,8 @@ program
39
39
  .description('Start development server with hot reload')
40
40
  .option('-p, --port <port>', 'Port to run the server on', '3333')
41
41
  .option('-H, --host <host>', 'Host to bind the server to', 'localhost')
42
+ .option('-o, --open', 'Open browser automatically', true)
43
+ .option('--no-open', 'Do not open browser automatically')
42
44
  .action(dev_1.dev);
43
45
  program
44
46
  .command('build')
@@ -103,4 +105,4 @@ program
103
105
  .option('-k, --api-key <key>', 'API key for authentication')
104
106
  .action(upload_1.upload);
105
107
  program.parse(process.argv);
106
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command } from 'commander';\nimport { create } from './commands/create';\nimport { init } from './commands/init';\nimport { dev } from './commands/dev';\nimport { build } from './commands/build';\nimport { start } from './commands/start';\nimport { check } from './commands/check';\nimport { deploy } from './commands/deploy';\nimport { listKeys, regenerateKey } from './commands/keys';\nimport { whoami } from './commands/whoami';\nimport { upload, listAssets } from './commands/upload';\n\nconst packageJson = require('../../package.json');\n\nconst program = new Command();\n\nprogram\n  .name('devdoc')\n  .description('Documentation framework for developers')\n  .version(packageJson.version);\n\nprogram\n  .command('create [project-directory]')\n  .description('Create a new DevDoc documentation site')\n  .option('-t, --template <type>', 'Template to use (basic, openapi, graphql)')\n  .option('-s, --subdomain <subdomain>', 'Subdomain for your docs (e.g., my-docs.devdoc.sh)')\n  .option('--no-git', 'Skip git initialization')\n  .option('--no-install', 'Skip installing dependencies')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .action(create);\n\nprogram\n  .command('init')\n  .description('Initialize/register project with Brainfish')\n  .option('-s, --slug <slug>', 'Project slug')\n  .option('--subdomain <subdomain>', 'Subdomain for your docs (e.g., my-docs.devdoc.sh)')\n  .option('-f, --force', 'Overwrite existing .devdoc.json')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .action(init);\n\nprogram\n  .command('dev')\n  .description('Start development server with hot reload')\n  .option('-p, --port <port>', 'Port to run the server on', '3333')\n  .option('-H, --host <host>', 'Host to bind the server to', 'localhost')\n  .action(dev);\n\nprogram\n  .command('build')\n  .description('Build documentation for production')\n  .option('-o, --output <dir>', 'Output directory', 'dist')\n  .action(build);\n\nprogram\n  .command('start')\n  .description('Start production server')\n  .option('-p, --port <port>', 'Port to run the server on', '3000')\n  .action(start);\n\nprogram\n  .command('check')\n  .description('Validate docs.json and MDX files')\n  .action(check);\n\nprogram\n  .command('deploy')\n  .description('Deploy documentation to DevDoc platform')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(deploy);\n\n// Keys management\nconst keysCmd = program\n  .command('keys')\n  .description('Manage API keys for your project');\n\nkeysCmd\n  .command('list')\n  .description('Show current API key info')\n  .option('-u, --url <url>', 'API URL')\n  .action(listKeys);\n\nkeysCmd\n  .command('regenerate')\n  .description('Generate a new API key (invalidates the old one)')\n  .option('-u, --url <url>', 'API URL')\n  .action(regenerateKey);\n\nprogram\n  .command('whoami')\n  .description('Show current project information')\n  .option('-u, --url <url>', 'API URL')\n  .action(whoami);\n\n// Assets management\nconst assetsCmd = program\n  .command('assets')\n  .description('Manage project assets (images, files)');\n\nassetsCmd\n  .command('upload <files...>')\n  .description('Upload assets to storage (max 25MB per file)')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(upload);\n\nassetsCmd\n  .command('list')\n  .description('List uploaded assets')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(listAssets);\n\n// Shortcut for upload\nprogram\n  .command('upload <files...>')\n  .description('Upload assets to storage (shortcut for \"devdoc assets upload\")')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(upload);\n\nprogram.parse(process.argv);\n"]}
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command } from 'commander';\nimport { create } from './commands/create';\nimport { init } from './commands/init';\nimport { dev } from './commands/dev';\nimport { build } from './commands/build';\nimport { start } from './commands/start';\nimport { check } from './commands/check';\nimport { deploy } from './commands/deploy';\nimport { listKeys, regenerateKey } from './commands/keys';\nimport { whoami } from './commands/whoami';\nimport { upload, listAssets } from './commands/upload';\n\nconst packageJson = require('../../package.json');\n\nconst program = new Command();\n\nprogram\n  .name('devdoc')\n  .description('Documentation framework for developers')\n  .version(packageJson.version);\n\nprogram\n  .command('create [project-directory]')\n  .description('Create a new DevDoc documentation site')\n  .option('-t, --template <type>', 'Template to use (basic, openapi, graphql)')\n  .option('-s, --subdomain <subdomain>', 'Subdomain for your docs (e.g., my-docs.devdoc.sh)')\n  .option('--no-git', 'Skip git initialization')\n  .option('--no-install', 'Skip installing dependencies')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .action(create);\n\nprogram\n  .command('init')\n  .description('Initialize/register project with Brainfish')\n  .option('-s, --slug <slug>', 'Project slug')\n  .option('--subdomain <subdomain>', 'Subdomain for your docs (e.g., my-docs.devdoc.sh)')\n  .option('-f, --force', 'Overwrite existing .devdoc.json')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .action(init);\n\nprogram\n  .command('dev')\n  .description('Start development server with hot reload')\n  .option('-p, --port <port>', 'Port to run the server on', '3333')\n  .option('-H, --host <host>', 'Host to bind the server to', 'localhost')\n  .option('-o, --open', 'Open browser automatically', true)\n  .option('--no-open', 'Do not open browser automatically')\n  .action(dev);\n\nprogram\n  .command('build')\n  .description('Build documentation for production')\n  .option('-o, --output <dir>', 'Output directory', 'dist')\n  .action(build);\n\nprogram\n  .command('start')\n  .description('Start production server')\n  .option('-p, --port <port>', 'Port to run the server on', '3000')\n  .action(start);\n\nprogram\n  .command('check')\n  .description('Validate docs.json and MDX files')\n  .action(check);\n\nprogram\n  .command('deploy')\n  .description('Deploy documentation to DevDoc platform')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(deploy);\n\n// Keys management\nconst keysCmd = program\n  .command('keys')\n  .description('Manage API keys for your project');\n\nkeysCmd\n  .command('list')\n  .description('Show current API key info')\n  .option('-u, --url <url>', 'API URL')\n  .action(listKeys);\n\nkeysCmd\n  .command('regenerate')\n  .description('Generate a new API key (invalidates the old one)')\n  .option('-u, --url <url>', 'API URL')\n  .action(regenerateKey);\n\nprogram\n  .command('whoami')\n  .description('Show current project information')\n  .option('-u, --url <url>', 'API URL')\n  .action(whoami);\n\n// Assets management\nconst assetsCmd = program\n  .command('assets')\n  .description('Manage project assets (images, files)');\n\nassetsCmd\n  .command('upload <files...>')\n  .description('Upload assets to storage (max 25MB per file)')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(upload);\n\nassetsCmd\n  .command('list')\n  .description('List uploaded assets')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(listAssets);\n\n// Shortcut for upload\nprogram\n  .command('upload <files...>')\n  .description('Upload assets to storage (shortcut for \"devdoc assets upload\")')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(upload);\n\nprogram.parse(process.argv);\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@brainfish-ai/devdoc",
3
- "version": "0.1.23",
3
+ "version": "0.1.24",
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",