runway-cli 1.2.0 → 1.3.1

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.
@@ -1,4 +1,37 @@
1
1
  "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
2
35
  var __importDefault = (this && this.__importDefault) || function (mod) {
3
36
  return (mod && mod.__esModule) ? mod : { "default": mod };
4
37
  };
@@ -124,6 +157,12 @@ async function deployCommand(options) {
124
157
  logger_1.logger.error('CLI not configured. Run "runway init" first.');
125
158
  return;
126
159
  }
160
+ // Check token validity before proceeding
161
+ if ((0, config_1.isTokenExpired)()) {
162
+ logger_1.logger.error('Your authentication token has expired.');
163
+ logger_1.logger.info('Run "runway login" to re-authenticate.');
164
+ return;
165
+ }
127
166
  const config = (0, config_1.getConfig)();
128
167
  logger_1.logger.dim(`Server: ${config.serverUrl}`);
129
168
  logger_1.logger.blank();
@@ -139,8 +178,13 @@ async function deployCommand(options) {
139
178
  logger_1.logger.error(error instanceof Error ? error.message : 'Unknown error');
140
179
  return;
141
180
  }
142
- // Determine project name
143
- let projectName = options.name || detectedProject.name;
181
+ // Determine project name — sanitize from package.json
182
+ const suggestedName = (detectedProject.name || path_1.default.basename(process.cwd()))
183
+ .replace(/^@[^/]+\//, '') // Remove npm scope (@scope/name → name)
184
+ .replace(/[^a-zA-Z0-9-_]/g, '-') // Replace invalid chars with hyphens
185
+ .replace(/^-+|-+$/g, '') // Trim leading/trailing hyphens
186
+ .toLowerCase();
187
+ let projectName = options.name || suggestedName;
144
188
  // Interactive mode if name not provided
145
189
  if (!options.name) {
146
190
  const answers = await inquirer_1.default.prompt([
@@ -249,6 +293,12 @@ async function deployCommand(options) {
249
293
  let buildOutputDir = detectedProject.buildOutputDir;
250
294
  if (buildMode === 'local') {
251
295
  logger_1.logger.step(1, 4, 'Building project...');
296
+ // Patch React Router basename before build (if applicable)
297
+ let didPatch = false;
298
+ if (projectType === 'react') {
299
+ const { ReactPatcher } = await Promise.resolve().then(() => __importStar(require('../services/reactPatcher')));
300
+ didPatch = await ReactPatcher.patch(process.cwd());
301
+ }
252
302
  const buildResult = await buildService_1.buildService.build({
253
303
  projectPath: process.cwd(),
254
304
  projectType,
@@ -256,6 +306,11 @@ async function deployCommand(options) {
256
306
  packageManager: detectedProject.packageManager,
257
307
  envFile: envFilePath,
258
308
  });
309
+ // Revert patch after build (keep user's source clean)
310
+ if (didPatch) {
311
+ const { ReactPatcher } = await Promise.resolve().then(() => __importStar(require('../services/reactPatcher')));
312
+ await ReactPatcher.revert(process.cwd());
313
+ }
259
314
  if (!buildResult.success) {
260
315
  logger_1.logger.error(`Build failed: ${buildResult.error}`);
261
316
  return;
@@ -382,6 +437,11 @@ async function deployCommand(options) {
382
437
  logger_1.logger.blank();
383
438
  logger_1.logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);
384
439
  }
440
+ // Show health warning if service didn't respond on its allocated port
441
+ if (finalStatus.healthWarning) {
442
+ logger_1.logger.blank();
443
+ logger_1.logger.warn(`⚠ Health check: ${finalStatus.healthWarning}`);
444
+ }
385
445
  }
386
446
  else {
387
447
  logger_1.logger.error(`Deployment failed: ${finalStatus.error || 'Unknown error'}`);
@@ -404,4 +464,4 @@ async function deployCommand(options) {
404
464
  }
405
465
  logger_1.logger.blank();
406
466
  }
407
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy.js","sourceRoot":"","sources":["../../src/commands/deploy.ts"],"names":[],"mappings":"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","sourcesContent":["import inquirer from 'inquirer';\nimport ora from 'ora';\nimport fs from 'fs';\nimport path from 'path';\nimport axios from 'axios';\nimport { ProjectType, BuildMode } from '../types';\nimport { detectProject } from '../services/projectDetector';\nimport { buildService } from '../services/buildService';\nimport { packageService } from '../services/packageService';\nimport { createUploadService } from '../services/uploadService';\nimport { isConfigured, getConfig } from '../utils/config';\nimport { logger } from '../utils/logger';\n\ninterface DeployOptions {\n  name?: string;\n  type?: ProjectType;\n  version?: string;\n  buildLocal?: boolean;\n  buildServer?: boolean;\n  envFile?: string;\n  skipEnvPrompt?: boolean;\n}\n\n/**\n * Parse a .env file into a Record\n */\nfunction parseEnvFile(filePath: string): Record<string, string> {\n  const content = fs.readFileSync(filePath, 'utf-8');\n  const vars: Record<string, string> = {};\n\n  for (const line of content.split('\\n')) {\n    const trimmed = line.trim();\n    // Skip empty lines and comments\n    if (!trimmed || trimmed.startsWith('#')) continue;\n\n    const eqIndex = trimmed.indexOf('=');\n    if (eqIndex > 0) {\n      const key = trimmed.slice(0, eqIndex).trim();\n      let value = trimmed.slice(eqIndex + 1).trim();\n      // Remove surrounding quotes if present\n      if ((value.startsWith('\"') && value.endsWith('\"')) ||\n          (value.startsWith(\"'\") && value.endsWith(\"'\"))) {\n        value = value.slice(1, -1);\n      }\n      vars[key] = value;\n    }\n  }\n\n  return vars;\n}\n\n/**\n * Prompt user for manual ENV variable entry\n */\nasync function promptManualEnvVars(): Promise<Record<string, string>> {\n  const vars: Record<string, string> = {};\n\n  logger.dim('Enter environment variables (empty name to finish):');\n\n  while (true) {\n    const { key } = await inquirer.prompt([\n      {\n        type: 'input',\n        name: 'key',\n        message: 'Variable name:',\n      },\n    ]);\n\n    if (!key || !key.trim()) {\n      break;\n    }\n\n    const { value } = await inquirer.prompt([\n      {\n        type: 'input',\n        name: 'value',\n        message: `Value for ${key}:`,\n      },\n    ]);\n\n    vars[key.toUpperCase().replace(/[^A-Z0-9_]/g, '')] = value;\n  }\n\n  return vars;\n}\n\n/**\n * Prompt user for environment variable options when no .env file exists\n */\nasync function promptEnvOptions(): Promise<string | undefined> {\n  const { envChoice } = await inquirer.prompt([\n    {\n      type: 'list',\n      name: 'envChoice',\n      message: 'How would you like to provide environment variables?',\n      choices: [\n        { name: 'Specify path to .env file', value: 'path' },\n        { name: 'Enter variables manually (creates .env)', value: 'manual' },\n        { name: 'Continue without environment variables', value: 'skip' },\n      ],\n    },\n  ]);\n\n  if (envChoice === 'path') {\n    const { envPath } = await inquirer.prompt([\n      {\n        type: 'input',\n        name: 'envPath',\n        message: 'Path to env file:',\n        validate: (input: string) => {\n          if (!input.trim()) return 'Please enter a path';\n          if (!fs.existsSync(input)) return 'File not found';\n          return true;\n        },\n      },\n    ]);\n    logger.success(`Using env file: ${envPath}`);\n    return envPath;\n  }\n\n  if (envChoice === 'manual') {\n    const vars = await promptManualEnvVars();\n    if (Object.keys(vars).length > 0) {\n      // Write to .env file in project\n      const envContent = Object.entries(vars)\n        .map(([k, v]) => `${k}=${v}`)\n        .join('\\n');\n      const envPath = path.join(process.cwd(), '.env');\n      fs.writeFileSync(envPath, envContent);\n      logger.success(`Created .env with ${Object.keys(vars).length} variables`);\n      return envPath;\n    }\n  }\n\n  return undefined;\n}\n\nexport async function deployCommand(options: DeployOptions): Promise<void> {\n  logger.header('Runway Deploy');\n\n  // Check configuration\n  if (!isConfigured()) {\n    logger.error('CLI not configured. Run \"runway init\" first.');\n    return;\n  }\n\n  const config = getConfig();\n  logger.dim(`Server: ${config.serverUrl}`);\n  logger.blank();\n\n  // Detect project\n  const spinner = ora('Detecting project...').start();\n  let detectedProject;\n\n  try {\n    detectedProject = await detectProject();\n    spinner.succeed(`Detected: ${detectedProject.type} project (${detectedProject.packageManager})`);\n  } catch (error) {\n    spinner.fail('Failed to detect project');\n    logger.error(error instanceof Error ? error.message : 'Unknown error');\n    return;\n  }\n\n  // Determine project name\n  let projectName = options.name || detectedProject.name;\n\n  // Interactive mode if name not provided\n  if (!options.name) {\n    const answers = await inquirer.prompt([\n      {\n        type: 'input',\n        name: 'name',\n        message: 'Project name:',\n        default: projectName,\n        validate: (input: string) => {\n          if (input.length < 2) return 'Name must be at least 2 characters';\n          if (!/^[a-zA-Z0-9-_]+$/.test(input)) return 'Name can only contain letters, numbers, hyphens, and underscores';\n          return true;\n        },\n      },\n    ]);\n    projectName = answers.name;\n  }\n\n  // Ask if environment variables are required\n  const { needsEnv } = await inquirer.prompt([\n    {\n      type: 'confirm',\n      name: 'needsEnv',\n      message: 'Does this project require environment variables?',\n      default: false,\n    },\n  ]);\n\n  let envFilePath: string | undefined;\n  let envVars: Record<string, string> = {};\n  let envInjected = false;\n\n  if (needsEnv) {\n    const defaultEnvPath = path.join(process.cwd(), '.env');\n    const hasEnvFile = fs.existsSync(defaultEnvPath);\n\n    if (hasEnvFile) {\n      // .env found - confirm with user\n      const { useExisting } = await inquirer.prompt([\n        {\n          type: 'confirm',\n          name: 'useExisting',\n          message: 'Found .env file in project. Use it?',\n          default: true,\n        },\n      ]);\n\n      if (useExisting) {\n        envFilePath = defaultEnvPath;\n        envVars = parseEnvFile(defaultEnvPath);\n        envInjected = Object.keys(envVars).length > 0;\n        logger.success(`Loaded ${Object.keys(envVars).length} variables from .env`);\n      } else {\n        // User declined existing .env - show other options\n        envFilePath = await promptEnvOptions();\n        if (envFilePath) {\n          envVars = parseEnvFile(envFilePath);\n          envInjected = Object.keys(envVars).length > 0;\n        }\n      }\n    } else {\n      // No .env found - show options\n      logger.warn('No .env file found in project directory.');\n      envFilePath = await promptEnvOptions();\n      if (envFilePath) {\n        envVars = parseEnvFile(envFilePath);\n        envInjected = Object.keys(envVars).length > 0;\n      }\n    }\n  }\n\n  // Determine project type\n  const projectType = options.type || detectedProject.type;\n\n  // Determine build mode\n  let buildMode: BuildMode;\n  if (options.buildServer) {\n    buildMode = 'server';\n  } else if (options.buildLocal) {\n    buildMode = 'local';\n  } else {\n    buildMode = config.defaultBuildMode || 'local';\n  }\n\n  logger.blank();\n  logger.info(`Project: ${projectName}`);\n  logger.info(`Type: ${projectType}`);\n  logger.info(`Build mode: ${buildMode}`);\n  if (options.version) {\n    logger.info(`Version: ${options.version}`);\n  }\n  logger.blank();\n\n  // Confirm deployment\n  const { confirm } = await inquirer.prompt([\n    {\n      type: 'confirm',\n      name: 'confirm',\n      message: 'Proceed with deployment?',\n      default: true,\n    },\n  ]);\n\n  if (!confirm) {\n    logger.warn('Deployment cancelled.');\n    return;\n  }\n\n  logger.blank();\n\n  // Step 1: Build (for local-build mode)\n  let buildOutputDir = detectedProject.buildOutputDir;\n\n  if (buildMode === 'local') {\n    logger.step(1, 4, 'Building project...');\n\n    const buildResult = await buildService.build({\n      projectPath: process.cwd(),\n      projectType,\n      projectName,\n      packageManager: detectedProject.packageManager,\n      envFile: envFilePath,\n    });\n\n    if (!buildResult.success) {\n      logger.error(`Build failed: ${buildResult.error}`);\n      return;\n    }\n\n    buildOutputDir = buildResult.outputDir;\n    logger.success(`Build completed in ${(buildResult.duration / 1000).toFixed(1)}s`);\n    logger.blank();\n  }\n\n  // Step 2: Package\n  const packageStep = buildMode === 'local' ? 2 : 1;\n  const totalSteps = buildMode === 'local' ? 4 : 3;\n\n  logger.step(packageStep, totalSteps, 'Creating deployment package...');\n\n  let packageResult;\n  try {\n    packageResult = await packageService.package({\n      projectPath: process.cwd(),\n      projectType,\n      buildOutputDir,\n      includeSource: buildMode === 'server',\n      envFile: projectType === 'node' ? envFilePath : undefined,\n    });\n  } catch (error) {\n    logger.error(`Packaging failed: ${error instanceof Error ? error.message : 'Unknown error'}`);\n    return;\n  }\n\n  logger.blank();\n\n  // Step 3: Analyze & Upload\n  logger.step(packageStep + 1, totalSteps, 'Analyzing and uploading to server...');\n\n  let uploadService;\n  try {\n    uploadService = createUploadService();\n  } catch (error) {\n    logger.error(error instanceof Error ? error.message : 'Unknown error');\n    packageService.cleanup(packageResult.zipPath);\n    return;\n  }\n\n  // Analyze package first (for server-build mode to get warnings)\n  let confirmServerBuild = false;\n  if (buildMode === 'server') {\n    const analyzeResult = await uploadService.analyzePackage(packageResult.zipPath, projectType);\n\n    if (analyzeResult.success && analyzeResult.analysis) {\n      const analysis = analyzeResult.analysis;\n\n      // Display warnings\n      if (analysis.warnings && analysis.warnings.length > 0) {\n        logger.blank();\n        logger.warn('Server Analysis:');\n        for (const warning of analysis.warnings) {\n          const prefix = warning.level === 'critical' ? '❌' : warning.level === 'warning' ? '⚠️' : 'ℹ️';\n          logger.dim(`  ${prefix} ${warning.message}`);\n        }\n        logger.blank();\n      }\n\n      // Handle confirmation for server-side build\n      if (analysis.requiresConfirmation) {\n        const { confirmBuild } = await inquirer.prompt([\n          {\n            type: 'confirm',\n            name: 'confirmBuild',\n            message: `${analysis.confirmationReason || 'Server-side build required'}. This may consume significant resources. Continue?`,\n            default: true,\n          },\n        ]);\n\n        if (!confirmBuild) {\n          logger.warn('Deployment cancelled by user.');\n          packageService.cleanup(packageResult.zipPath);\n          return;\n        }\n        confirmServerBuild = true;\n      }\n    }\n  }\n\n  const uploadResult = await uploadService.upload({\n    zipPath: packageResult.zipPath,\n    projectName,\n    projectType,\n    version: options.version,\n    buildMode,\n    confirmServerBuild,\n    // ENV mutability tracking\n    deploymentSource: 'cli',\n    envInjected,\n  });\n\n  // Cleanup zip file\n  packageService.cleanup(packageResult.zipPath);\n\n  if (!uploadResult.success) {\n    logger.error(`Upload failed: ${uploadResult.error}`);\n    return;\n  }\n\n  logger.success('Upload complete');\n  logger.blank();\n\n  // Step 4: Wait for deployment (if deployment ID available)\n  if (uploadResult.deploymentId) {\n    logger.step(packageStep + 2, totalSteps, 'Waiting for deployment to complete...');\n\n    try {\n      const finalStatus = await uploadService.pollDeploymentStatus(\n        uploadResult.deploymentId,\n        (status) => {\n          if (status.progress !== undefined) {\n            process.stdout.write(`\\r  Progress: ${status.progress}%`);\n          }\n        }\n      );\n\n      process.stdout.write('\\n');\n\n      if (finalStatus.status === 'success') {\n        logger.blank();\n        logger.success('Deployment successful!');\n\n        const safeName = projectName.toLowerCase().replace(/[^a-z0-9]/g, '-');\n\n        // Fetch domain config to show proper URL\n        try {\n          const domainResponse = await axios.get(`${config.serverUrl}/api/domain`, {\n            headers: config.token ? { Authorization: `Bearer ${config.token}` } : {},\n          });\n          const domainConfig = domainResponse.data;\n\n          if (domainConfig.domain?.active && domainConfig.securityMode === 'domain-https') {\n            logger.blank();\n            logger.info(`Your app is available at: https://${domainConfig.domain.domain}/app/${safeName}`);\n          } else {\n            logger.blank();\n            logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);\n          }\n        } catch {\n          // Fallback to server URL if domain fetch fails\n          logger.blank();\n          logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);\n        }\n      } else {\n        logger.error(`Deployment failed: ${finalStatus.error || 'Unknown error'}`);\n        if (finalStatus.logs) {\n          logger.blank();\n          logger.dim('Logs:');\n          console.log(finalStatus.logs);\n        }\n      }\n    } catch (error) {\n      logger.warn('Could not track deployment status');\n      logger.dim('Check the web UI for deployment status');\n    }\n  } else {\n    logger.blank();\n    logger.success('Upload successful!');\n    logger.dim('Deployment is being processed. Check the web UI for status.');\n  }\n\n  logger.blank();\n}\n"]}
467
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy.js","sourceRoot":"","sources":["../../src/commands/deploy.ts"],"names":[],"mappings":"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","sourcesContent":["import inquirer from 'inquirer';\nimport ora from 'ora';\nimport fs from 'fs';\nimport path from 'path';\nimport axios from 'axios';\nimport { ProjectType, BuildMode } from '../types';\nimport { detectProject } from '../services/projectDetector';\nimport { buildService } from '../services/buildService';\nimport { packageService } from '../services/packageService';\nimport { createUploadService } from '../services/uploadService';\nimport { isConfigured, getConfig, isTokenExpired } from '../utils/config';\nimport { logger } from '../utils/logger';\n\ninterface DeployOptions {\n  name?: string;\n  type?: ProjectType;\n  version?: string;\n  buildLocal?: boolean;\n  buildServer?: boolean;\n  envFile?: string;\n  skipEnvPrompt?: boolean;\n}\n\n/**\n * Parse a .env file into a Record\n */\nfunction parseEnvFile(filePath: string): Record<string, string> {\n  const content = fs.readFileSync(filePath, 'utf-8');\n  const vars: Record<string, string> = {};\n\n  for (const line of content.split('\\n')) {\n    const trimmed = line.trim();\n    // Skip empty lines and comments\n    if (!trimmed || trimmed.startsWith('#')) continue;\n\n    const eqIndex = trimmed.indexOf('=');\n    if (eqIndex > 0) {\n      const key = trimmed.slice(0, eqIndex).trim();\n      let value = trimmed.slice(eqIndex + 1).trim();\n      // Remove surrounding quotes if present\n      if ((value.startsWith('\"') && value.endsWith('\"')) ||\n          (value.startsWith(\"'\") && value.endsWith(\"'\"))) {\n        value = value.slice(1, -1);\n      }\n      vars[key] = value;\n    }\n  }\n\n  return vars;\n}\n\n/**\n * Prompt user for manual ENV variable entry\n */\nasync function promptManualEnvVars(): Promise<Record<string, string>> {\n  const vars: Record<string, string> = {};\n\n  logger.dim('Enter environment variables (empty name to finish):');\n\n  while (true) {\n    const { key } = await inquirer.prompt([\n      {\n        type: 'input',\n        name: 'key',\n        message: 'Variable name:',\n      },\n    ]);\n\n    if (!key || !key.trim()) {\n      break;\n    }\n\n    const { value } = await inquirer.prompt([\n      {\n        type: 'input',\n        name: 'value',\n        message: `Value for ${key}:`,\n      },\n    ]);\n\n    vars[key.toUpperCase().replace(/[^A-Z0-9_]/g, '')] = value;\n  }\n\n  return vars;\n}\n\n/**\n * Prompt user for environment variable options when no .env file exists\n */\nasync function promptEnvOptions(): Promise<string | undefined> {\n  const { envChoice } = await inquirer.prompt([\n    {\n      type: 'list',\n      name: 'envChoice',\n      message: 'How would you like to provide environment variables?',\n      choices: [\n        { name: 'Specify path to .env file', value: 'path' },\n        { name: 'Enter variables manually (creates .env)', value: 'manual' },\n        { name: 'Continue without environment variables', value: 'skip' },\n      ],\n    },\n  ]);\n\n  if (envChoice === 'path') {\n    const { envPath } = await inquirer.prompt([\n      {\n        type: 'input',\n        name: 'envPath',\n        message: 'Path to env file:',\n        validate: (input: string) => {\n          if (!input.trim()) return 'Please enter a path';\n          if (!fs.existsSync(input)) return 'File not found';\n          return true;\n        },\n      },\n    ]);\n    logger.success(`Using env file: ${envPath}`);\n    return envPath;\n  }\n\n  if (envChoice === 'manual') {\n    const vars = await promptManualEnvVars();\n    if (Object.keys(vars).length > 0) {\n      // Write to .env file in project\n      const envContent = Object.entries(vars)\n        .map(([k, v]) => `${k}=${v}`)\n        .join('\\n');\n      const envPath = path.join(process.cwd(), '.env');\n      fs.writeFileSync(envPath, envContent);\n      logger.success(`Created .env with ${Object.keys(vars).length} variables`);\n      return envPath;\n    }\n  }\n\n  return undefined;\n}\n\nexport async function deployCommand(options: DeployOptions): Promise<void> {\n  logger.header('Runway Deploy');\n\n  // Check configuration\n  if (!isConfigured()) {\n    logger.error('CLI not configured. Run \"runway init\" first.');\n    return;\n  }\n\n  // Check token validity before proceeding\n  if (isTokenExpired()) {\n    logger.error('Your authentication token has expired.');\n    logger.info('Run \"runway login\" to re-authenticate.');\n    return;\n  }\n\n  const config = getConfig();\n  logger.dim(`Server: ${config.serverUrl}`);\n  logger.blank();\n\n  // Detect project\n  const spinner = ora('Detecting project...').start();\n  let detectedProject;\n\n  try {\n    detectedProject = await detectProject();\n    spinner.succeed(`Detected: ${detectedProject.type} project (${detectedProject.packageManager})`);\n  } catch (error) {\n    spinner.fail('Failed to detect project');\n    logger.error(error instanceof Error ? error.message : 'Unknown error');\n    return;\n  }\n\n  // Determine project name — sanitize from package.json\n  const suggestedName = (detectedProject.name || path.basename(process.cwd()))\n    .replace(/^@[^/]+\\//, '')        // Remove npm scope (@scope/name → name)\n    .replace(/[^a-zA-Z0-9-_]/g, '-')  // Replace invalid chars with hyphens\n    .replace(/^-+|-+$/g, '')          // Trim leading/trailing hyphens\n    .toLowerCase();\n  let projectName = options.name || suggestedName;\n\n  // Interactive mode if name not provided\n  if (!options.name) {\n    const answers = await inquirer.prompt([\n      {\n        type: 'input',\n        name: 'name',\n        message: 'Project name:',\n        default: projectName,\n        validate: (input: string) => {\n          if (input.length < 2) return 'Name must be at least 2 characters';\n          if (!/^[a-zA-Z0-9-_]+$/.test(input)) return 'Name can only contain letters, numbers, hyphens, and underscores';\n          return true;\n        },\n      },\n    ]);\n    projectName = answers.name;\n  }\n\n  // Ask if environment variables are required\n  const { needsEnv } = await inquirer.prompt([\n    {\n      type: 'confirm',\n      name: 'needsEnv',\n      message: 'Does this project require environment variables?',\n      default: false,\n    },\n  ]);\n\n  let envFilePath: string | undefined;\n  let envVars: Record<string, string> = {};\n  let envInjected = false;\n\n  if (needsEnv) {\n    const defaultEnvPath = path.join(process.cwd(), '.env');\n    const hasEnvFile = fs.existsSync(defaultEnvPath);\n\n    if (hasEnvFile) {\n      // .env found - confirm with user\n      const { useExisting } = await inquirer.prompt([\n        {\n          type: 'confirm',\n          name: 'useExisting',\n          message: 'Found .env file in project. Use it?',\n          default: true,\n        },\n      ]);\n\n      if (useExisting) {\n        envFilePath = defaultEnvPath;\n        envVars = parseEnvFile(defaultEnvPath);\n        envInjected = Object.keys(envVars).length > 0;\n        logger.success(`Loaded ${Object.keys(envVars).length} variables from .env`);\n      } else {\n        // User declined existing .env - show other options\n        envFilePath = await promptEnvOptions();\n        if (envFilePath) {\n          envVars = parseEnvFile(envFilePath);\n          envInjected = Object.keys(envVars).length > 0;\n        }\n      }\n    } else {\n      // No .env found - show options\n      logger.warn('No .env file found in project directory.');\n      envFilePath = await promptEnvOptions();\n      if (envFilePath) {\n        envVars = parseEnvFile(envFilePath);\n        envInjected = Object.keys(envVars).length > 0;\n      }\n    }\n  }\n\n  // Determine project type\n  const projectType = options.type || detectedProject.type;\n\n  // Determine build mode\n  let buildMode: BuildMode;\n  if (options.buildServer) {\n    buildMode = 'server';\n  } else if (options.buildLocal) {\n    buildMode = 'local';\n  } else {\n    buildMode = config.defaultBuildMode || 'local';\n  }\n\n  logger.blank();\n  logger.info(`Project: ${projectName}`);\n  logger.info(`Type: ${projectType}`);\n  logger.info(`Build mode: ${buildMode}`);\n  if (options.version) {\n    logger.info(`Version: ${options.version}`);\n  }\n  logger.blank();\n\n  // Confirm deployment\n  const { confirm } = await inquirer.prompt([\n    {\n      type: 'confirm',\n      name: 'confirm',\n      message: 'Proceed with deployment?',\n      default: true,\n    },\n  ]);\n\n  if (!confirm) {\n    logger.warn('Deployment cancelled.');\n    return;\n  }\n\n  logger.blank();\n\n  // Step 1: Build (for local-build mode)\n  let buildOutputDir = detectedProject.buildOutputDir;\n\n  if (buildMode === 'local') {\n    logger.step(1, 4, 'Building project...');\n\n    // Patch React Router basename before build (if applicable)\n    let didPatch = false;\n    if (projectType === 'react') {\n      const { ReactPatcher } = await import('../services/reactPatcher');\n      didPatch = await ReactPatcher.patch(process.cwd());\n    }\n\n    const buildResult = await buildService.build({\n      projectPath: process.cwd(),\n      projectType,\n      projectName,\n      packageManager: detectedProject.packageManager,\n      envFile: envFilePath,\n    });\n\n    // Revert patch after build (keep user's source clean)\n    if (didPatch) {\n      const { ReactPatcher } = await import('../services/reactPatcher');\n      await ReactPatcher.revert(process.cwd());\n    }\n\n    if (!buildResult.success) {\n      logger.error(`Build failed: ${buildResult.error}`);\n      return;\n    }\n\n    buildOutputDir = buildResult.outputDir;\n    logger.success(`Build completed in ${(buildResult.duration / 1000).toFixed(1)}s`);\n    logger.blank();\n  }\n\n  // Step 2: Package\n  const packageStep = buildMode === 'local' ? 2 : 1;\n  const totalSteps = buildMode === 'local' ? 4 : 3;\n\n  logger.step(packageStep, totalSteps, 'Creating deployment package...');\n\n  let packageResult;\n  try {\n    packageResult = await packageService.package({\n      projectPath: process.cwd(),\n      projectType,\n      buildOutputDir,\n      includeSource: buildMode === 'server',\n      envFile: projectType === 'node' ? envFilePath : undefined,\n    });\n  } catch (error) {\n    logger.error(`Packaging failed: ${error instanceof Error ? error.message : 'Unknown error'}`);\n    return;\n  }\n\n  logger.blank();\n\n  // Step 3: Analyze & Upload\n  logger.step(packageStep + 1, totalSteps, 'Analyzing and uploading to server...');\n\n  let uploadService;\n  try {\n    uploadService = createUploadService();\n  } catch (error) {\n    logger.error(error instanceof Error ? error.message : 'Unknown error');\n    packageService.cleanup(packageResult.zipPath);\n    return;\n  }\n\n  // Analyze package first (for server-build mode to get warnings)\n  let confirmServerBuild = false;\n  if (buildMode === 'server') {\n    const analyzeResult = await uploadService.analyzePackage(packageResult.zipPath, projectType);\n\n    if (analyzeResult.success && analyzeResult.analysis) {\n      const analysis = analyzeResult.analysis;\n\n      // Display warnings\n      if (analysis.warnings && analysis.warnings.length > 0) {\n        logger.blank();\n        logger.warn('Server Analysis:');\n        for (const warning of analysis.warnings) {\n          const prefix = warning.level === 'critical' ? '❌' : warning.level === 'warning' ? '⚠️' : 'ℹ️';\n          logger.dim(`  ${prefix} ${warning.message}`);\n        }\n        logger.blank();\n      }\n\n      // Handle confirmation for server-side build\n      if (analysis.requiresConfirmation) {\n        const { confirmBuild } = await inquirer.prompt([\n          {\n            type: 'confirm',\n            name: 'confirmBuild',\n            message: `${analysis.confirmationReason || 'Server-side build required'}. This may consume significant resources. Continue?`,\n            default: true,\n          },\n        ]);\n\n        if (!confirmBuild) {\n          logger.warn('Deployment cancelled by user.');\n          packageService.cleanup(packageResult.zipPath);\n          return;\n        }\n        confirmServerBuild = true;\n      }\n    }\n  }\n\n  const uploadResult = await uploadService.upload({\n    zipPath: packageResult.zipPath,\n    projectName,\n    projectType,\n    version: options.version,\n    buildMode,\n    confirmServerBuild,\n    // ENV mutability tracking\n    deploymentSource: 'cli',\n    envInjected,\n  });\n\n  // Cleanup zip file\n  packageService.cleanup(packageResult.zipPath);\n\n  if (!uploadResult.success) {\n    logger.error(`Upload failed: ${uploadResult.error}`);\n    return;\n  }\n\n  logger.success('Upload complete');\n  logger.blank();\n\n  // Step 4: Wait for deployment (if deployment ID available)\n  if (uploadResult.deploymentId) {\n    logger.step(packageStep + 2, totalSteps, 'Waiting for deployment to complete...');\n\n    try {\n      const finalStatus = await uploadService.pollDeploymentStatus(\n        uploadResult.deploymentId,\n        (status) => {\n          if (status.progress !== undefined) {\n            process.stdout.write(`\\r  Progress: ${status.progress}%`);\n          }\n        }\n      );\n\n      process.stdout.write('\\n');\n\n      if (finalStatus.status === 'success') {\n        logger.blank();\n        logger.success('Deployment successful!');\n\n        const safeName = projectName.toLowerCase().replace(/[^a-z0-9]/g, '-');\n\n        // Fetch domain config to show proper URL\n        try {\n          const domainResponse = await axios.get(`${config.serverUrl}/api/domain`, {\n            headers: config.token ? { Authorization: `Bearer ${config.token}` } : {},\n          });\n          const domainConfig = domainResponse.data;\n\n          if (domainConfig.domain?.active && domainConfig.securityMode === 'domain-https') {\n            logger.blank();\n            logger.info(`Your app is available at: https://${domainConfig.domain.domain}/app/${safeName}`);\n          } else {\n            logger.blank();\n            logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);\n          }\n        } catch {\n          // Fallback to server URL if domain fetch fails\n          logger.blank();\n          logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);\n        }\n\n        // Show health warning if service didn't respond on its allocated port\n        if (finalStatus.healthWarning) {\n          logger.blank();\n          logger.warn(`⚠  Health check: ${finalStatus.healthWarning}`);\n        }\n      } else {\n        logger.error(`Deployment failed: ${finalStatus.error || 'Unknown error'}`);\n        if (finalStatus.logs) {\n          logger.blank();\n          logger.dim('Logs:');\n          console.log(finalStatus.logs);\n        }\n      }\n    } catch (error) {\n      logger.warn('Could not track deployment status');\n      logger.dim('Check the web UI for deployment status');\n    }\n  } else {\n    logger.blank();\n    logger.success('Upload successful!');\n    logger.dim('Deployment is being processed. Check the web UI for status.');\n  }\n\n  logger.blank();\n}\n"]}
@@ -0,0 +1 @@
1
+ export declare function loginCommand(): Promise<void>;
@@ -0,0 +1,73 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.loginCommand = loginCommand;
7
+ const inquirer_1 = __importDefault(require("inquirer"));
8
+ const config_1 = require("../utils/config");
9
+ const logger_1 = require("../utils/logger");
10
+ const authService_1 = require("../services/authService");
11
+ async function loginCommand() {
12
+ logger_1.logger.header('Runway Login');
13
+ // Check if init has been run (we need a server URL at minimum)
14
+ const config = (0, config_1.getConfig)();
15
+ if (!config.serverUrl) {
16
+ logger_1.logger.error('CLI not configured. Run "runway init" first to set up your server URL.');
17
+ return;
18
+ }
19
+ logger_1.logger.dim(`Server: ${config.serverUrl}`);
20
+ logger_1.logger.blank();
21
+ // Initialize auth service
22
+ const authService = new authService_1.AuthService(config.serverUrl);
23
+ // Check security mode
24
+ let securityInfo;
25
+ try {
26
+ securityInfo = await authService.getSecurityMode();
27
+ }
28
+ catch (error) {
29
+ logger_1.logger.error('Failed to connect to server. Make sure it is running.');
30
+ return;
31
+ }
32
+ if (securityInfo.securityMode === 'domain-https') {
33
+ logger_1.logger.success(`Server has HTTPS enabled: ${securityInfo.domain}`);
34
+ }
35
+ else {
36
+ logger_1.logger.warn('Server is running in HTTP mode.');
37
+ }
38
+ // Get credentials
39
+ logger_1.logger.blank();
40
+ const credentials = await inquirer_1.default.prompt([
41
+ {
42
+ type: 'input',
43
+ name: 'username',
44
+ message: 'Username:',
45
+ validate: (input) => input.length > 0 || 'Username is required',
46
+ },
47
+ {
48
+ type: 'password',
49
+ name: 'password',
50
+ message: 'Password:',
51
+ validate: (input) => input.length > 0 || 'Password is required',
52
+ },
53
+ ]);
54
+ // Authenticate
55
+ logger_1.logger.info('Authenticating...');
56
+ try {
57
+ const authResult = await authService.authenticate(credentials.username, credentials.password);
58
+ // Save new auth data (keep existing server URL)
59
+ (0, config_1.setAuthData)(authResult.token, authResult.expiresAt, authResult.securityMode);
60
+ logger_1.logger.blank();
61
+ logger_1.logger.success('Authentication successful! Token updated.');
62
+ // Show token expiry warning for HTTP mode
63
+ if (authResult.securityMode === 'ip-http') {
64
+ logger_1.logger.warn('Note: Token expires in 15 minutes due to HTTP mode.');
65
+ }
66
+ }
67
+ catch (error) {
68
+ logger_1.logger.blank();
69
+ logger_1.logger.dim('Check your credentials and try again.');
70
+ }
71
+ logger_1.logger.blank();
72
+ }
73
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9naW4uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29tbWFuZHMvbG9naW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFLQSxvQ0F3RUM7QUE3RUQsd0RBQWdDO0FBQ2hDLDRDQUF1RTtBQUN2RSw0Q0FBeUM7QUFDekMseURBQXNEO0FBRS9DLEtBQUssVUFBVSxZQUFZO0lBQ2hDLGVBQU0sQ0FBQyxNQUFNLENBQUMsY0FBYyxDQUFDLENBQUM7SUFFOUIsK0RBQStEO0lBQy9ELE1BQU0sTUFBTSxHQUFHLElBQUEsa0JBQVMsR0FBRSxDQUFDO0lBQzNCLElBQUksQ0FBQyxNQUFNLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDdEIsZUFBTSxDQUFDLEtBQUssQ0FBQyx3RUFBd0UsQ0FBQyxDQUFDO1FBQ3ZGLE9BQU87SUFDVCxDQUFDO0lBRUQsZUFBTSxDQUFDLEdBQUcsQ0FBQyxXQUFXLE1BQU0sQ0FBQyxTQUFTLEVBQUUsQ0FBQyxDQUFDO0lBQzFDLGVBQU0sQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUVmLDBCQUEwQjtJQUMxQixNQUFNLFdBQVcsR0FBRyxJQUFJLHlCQUFXLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBRXRELHNCQUFzQjtJQUN0QixJQUFJLFlBQVksQ0FBQztJQUNqQixJQUFJLENBQUM7UUFDSCxZQUFZLEdBQUcsTUFBTSxXQUFXLENBQUMsZUFBZSxFQUFFLENBQUM7SUFDckQsQ0FBQztJQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7UUFDZixlQUFNLENBQUMsS0FBSyxDQUFDLHVEQUF1RCxDQUFDLENBQUM7UUFDdEUsT0FBTztJQUNULENBQUM7SUFFRCxJQUFJLFlBQVksQ0FBQyxZQUFZLEtBQUssY0FBYyxFQUFFLENBQUM7UUFDakQsZUFBTSxDQUFDLE9BQU8sQ0FBQyw2QkFBNkIsWUFBWSxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUM7SUFDckUsQ0FBQztTQUFNLENBQUM7UUFDTixlQUFNLENBQUMsSUFBSSxDQUFDLGlDQUFpQyxDQUFDLENBQUM7SUFDakQsQ0FBQztJQUVELGtCQUFrQjtJQUNsQixlQUFNLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDZixNQUFNLFdBQVcsR0FBRyxNQUFNLGtCQUFRLENBQUMsTUFBTSxDQUFDO1FBQ3hDO1lBQ0UsSUFBSSxFQUFFLE9BQU87WUFDYixJQUFJLEVBQUUsVUFBVTtZQUNoQixPQUFPLEVBQUUsV0FBVztZQUNwQixRQUFRLEVBQUUsQ0FBQyxLQUFhLEVBQUUsRUFBRSxDQUFDLEtBQUssQ0FBQyxNQUFNLEdBQUcsQ0FBQyxJQUFJLHNCQUFzQjtTQUN4RTtRQUNEO1lBQ0UsSUFBSSxFQUFFLFVBQVU7WUFDaEIsSUFBSSxFQUFFLFVBQVU7WUFDaEIsT0FBTyxFQUFFLFdBQVc7WUFDcEIsUUFBUSxFQUFFLENBQUMsS0FBYSxFQUFFLEVBQUUsQ0FBQyxLQUFLLENBQUMsTUFBTSxHQUFHLENBQUMsSUFBSSxzQkFBc0I7U0FDeEU7S0FDRixDQUFDLENBQUM7SUFFSCxlQUFlO0lBQ2YsZUFBTSxDQUFDLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO0lBQ2pDLElBQUksQ0FBQztRQUNILE1BQU0sVUFBVSxHQUFHLE1BQU0sV0FBVyxDQUFDLFlBQVksQ0FDL0MsV0FBVyxDQUFDLFFBQVEsRUFDcEIsV0FBVyxDQUFDLFFBQVEsQ0FDckIsQ0FBQztRQUVGLGdEQUFnRDtRQUNoRCxJQUFBLG9CQUFXLEVBQUMsVUFBVSxDQUFDLEtBQUssRUFBRSxVQUFVLENBQUMsU0FBUyxFQUFFLFVBQVUsQ0FBQyxZQUFZLENBQUMsQ0FBQztRQUU3RSxlQUFNLENBQUMsS0FBSyxFQUFFLENBQUM7UUFDZixlQUFNLENBQUMsT0FBTyxDQUFDLDJDQUEyQyxDQUFDLENBQUM7UUFFNUQsMENBQTBDO1FBQzFDLElBQUksVUFBVSxDQUFDLFlBQVksS0FBSyxTQUFTLEVBQUUsQ0FBQztZQUMxQyxlQUFNLENBQUMsSUFBSSxDQUFDLHFEQUFxRCxDQUFDLENBQUM7UUFDckUsQ0FBQztJQUNILENBQUM7SUFBQyxPQUFPLEtBQUssRUFBRSxDQUFDO1FBQ2YsZUFBTSxDQUFDLEtBQUssRUFBRSxDQUFDO1FBQ2YsZUFBTSxDQUFDLEdBQUcsQ0FBQyx1Q0FBdUMsQ0FBQyxDQUFDO0lBQ3RELENBQUM7SUFFRCxlQUFNLENBQUMsS0FBSyxFQUFFLENBQUM7QUFDakIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBpbnF1aXJlciBmcm9tICdpbnF1aXJlcic7XG5pbXBvcnQgeyBnZXRDb25maWcsIHNldEF1dGhEYXRhLCBpc0NvbmZpZ3VyZWQgfSBmcm9tICcuLi91dGlscy9jb25maWcnO1xuaW1wb3J0IHsgbG9nZ2VyIH0gZnJvbSAnLi4vdXRpbHMvbG9nZ2VyJztcbmltcG9ydCB7IEF1dGhTZXJ2aWNlIH0gZnJvbSAnLi4vc2VydmljZXMvYXV0aFNlcnZpY2UnO1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gbG9naW5Db21tYW5kKCk6IFByb21pc2U8dm9pZD4ge1xuICBsb2dnZXIuaGVhZGVyKCdSdW53YXkgTG9naW4nKTtcblxuICAvLyBDaGVjayBpZiBpbml0IGhhcyBiZWVuIHJ1biAod2UgbmVlZCBhIHNlcnZlciBVUkwgYXQgbWluaW11bSlcbiAgY29uc3QgY29uZmlnID0gZ2V0Q29uZmlnKCk7XG4gIGlmICghY29uZmlnLnNlcnZlclVybCkge1xuICAgIGxvZ2dlci5lcnJvcignQ0xJIG5vdCBjb25maWd1cmVkLiBSdW4gXCJydW53YXkgaW5pdFwiIGZpcnN0IHRvIHNldCB1cCB5b3VyIHNlcnZlciBVUkwuJyk7XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgbG9nZ2VyLmRpbShgU2VydmVyOiAke2NvbmZpZy5zZXJ2ZXJVcmx9YCk7XG4gIGxvZ2dlci5ibGFuaygpO1xuXG4gIC8vIEluaXRpYWxpemUgYXV0aCBzZXJ2aWNlXG4gIGNvbnN0IGF1dGhTZXJ2aWNlID0gbmV3IEF1dGhTZXJ2aWNlKGNvbmZpZy5zZXJ2ZXJVcmwpO1xuXG4gIC8vIENoZWNrIHNlY3VyaXR5IG1vZGVcbiAgbGV0IHNlY3VyaXR5SW5mbztcbiAgdHJ5IHtcbiAgICBzZWN1cml0eUluZm8gPSBhd2FpdCBhdXRoU2VydmljZS5nZXRTZWN1cml0eU1vZGUoKTtcbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICBsb2dnZXIuZXJyb3IoJ0ZhaWxlZCB0byBjb25uZWN0IHRvIHNlcnZlci4gTWFrZSBzdXJlIGl0IGlzIHJ1bm5pbmcuJyk7XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgaWYgKHNlY3VyaXR5SW5mby5zZWN1cml0eU1vZGUgPT09ICdkb21haW4taHR0cHMnKSB7XG4gICAgbG9nZ2VyLnN1Y2Nlc3MoYFNlcnZlciBoYXMgSFRUUFMgZW5hYmxlZDogJHtzZWN1cml0eUluZm8uZG9tYWlufWApO1xuICB9IGVsc2Uge1xuICAgIGxvZ2dlci53YXJuKCdTZXJ2ZXIgaXMgcnVubmluZyBpbiBIVFRQIG1vZGUuJyk7XG4gIH1cblxuICAvLyBHZXQgY3JlZGVudGlhbHNcbiAgbG9nZ2VyLmJsYW5rKCk7XG4gIGNvbnN0IGNyZWRlbnRpYWxzID0gYXdhaXQgaW5xdWlyZXIucHJvbXB0KFtcbiAgICB7XG4gICAgICB0eXBlOiAnaW5wdXQnLFxuICAgICAgbmFtZTogJ3VzZXJuYW1lJyxcbiAgICAgIG1lc3NhZ2U6ICdVc2VybmFtZTonLFxuICAgICAgdmFsaWRhdGU6IChpbnB1dDogc3RyaW5nKSA9PiBpbnB1dC5sZW5ndGggPiAwIHx8ICdVc2VybmFtZSBpcyByZXF1aXJlZCcsXG4gICAgfSxcbiAgICB7XG4gICAgICB0eXBlOiAncGFzc3dvcmQnLFxuICAgICAgbmFtZTogJ3Bhc3N3b3JkJyxcbiAgICAgIG1lc3NhZ2U6ICdQYXNzd29yZDonLFxuICAgICAgdmFsaWRhdGU6IChpbnB1dDogc3RyaW5nKSA9PiBpbnB1dC5sZW5ndGggPiAwIHx8ICdQYXNzd29yZCBpcyByZXF1aXJlZCcsXG4gICAgfSxcbiAgXSk7XG5cbiAgLy8gQXV0aGVudGljYXRlXG4gIGxvZ2dlci5pbmZvKCdBdXRoZW50aWNhdGluZy4uLicpO1xuICB0cnkge1xuICAgIGNvbnN0IGF1dGhSZXN1bHQgPSBhd2FpdCBhdXRoU2VydmljZS5hdXRoZW50aWNhdGUoXG4gICAgICBjcmVkZW50aWFscy51c2VybmFtZSxcbiAgICAgIGNyZWRlbnRpYWxzLnBhc3N3b3JkXG4gICAgKTtcblxuICAgIC8vIFNhdmUgbmV3IGF1dGggZGF0YSAoa2VlcCBleGlzdGluZyBzZXJ2ZXIgVVJMKVxuICAgIHNldEF1dGhEYXRhKGF1dGhSZXN1bHQudG9rZW4sIGF1dGhSZXN1bHQuZXhwaXJlc0F0LCBhdXRoUmVzdWx0LnNlY3VyaXR5TW9kZSk7XG5cbiAgICBsb2dnZXIuYmxhbmsoKTtcbiAgICBsb2dnZXIuc3VjY2VzcygnQXV0aGVudGljYXRpb24gc3VjY2Vzc2Z1bCEgVG9rZW4gdXBkYXRlZC4nKTtcblxuICAgIC8vIFNob3cgdG9rZW4gZXhwaXJ5IHdhcm5pbmcgZm9yIEhUVFAgbW9kZVxuICAgIGlmIChhdXRoUmVzdWx0LnNlY3VyaXR5TW9kZSA9PT0gJ2lwLWh0dHAnKSB7XG4gICAgICBsb2dnZXIud2FybignTm90ZTogVG9rZW4gZXhwaXJlcyBpbiAxNSBtaW51dGVzIGR1ZSB0byBIVFRQIG1vZGUuJyk7XG4gICAgfVxuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGxvZ2dlci5ibGFuaygpO1xuICAgIGxvZ2dlci5kaW0oJ0NoZWNrIHlvdXIgY3JlZGVudGlhbHMgYW5kIHRyeSBhZ2Fpbi4nKTtcbiAgfVxuXG4gIGxvZ2dlci5ibGFuaygpO1xufVxuIl19
@@ -0,0 +1,6 @@
1
+ interface LogsOptions {
2
+ lines?: string;
3
+ type?: string;
4
+ }
5
+ export declare function logsCommand(project?: string, options?: LogsOptions): Promise<void>;
6
+ export {};
@@ -0,0 +1,112 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.logsCommand = logsCommand;
7
+ const inquirer_1 = __importDefault(require("inquirer"));
8
+ const axios_1 = __importDefault(require("axios"));
9
+ const chalk_1 = __importDefault(require("chalk"));
10
+ const config_1 = require("../utils/config");
11
+ const uploadService_1 = require("../services/uploadService");
12
+ const logger_1 = require("../utils/logger");
13
+ async function logsCommand(project, options = {}) {
14
+ logger_1.logger.header('Runway Logs');
15
+ // Check configuration
16
+ if (!(0, config_1.isConfigured)()) {
17
+ logger_1.logger.error('CLI not configured. Run "runway init" first.');
18
+ return;
19
+ }
20
+ // Check token validity
21
+ if ((0, config_1.isTokenExpired)()) {
22
+ logger_1.logger.error('Your authentication token has expired.');
23
+ logger_1.logger.info('Run "runway login" to re-authenticate.');
24
+ return;
25
+ }
26
+ const config = (0, config_1.getConfig)();
27
+ // If no project specified, list and let user choose
28
+ let projectId = project;
29
+ let projectName = project;
30
+ if (!projectId) {
31
+ let uploadService;
32
+ try {
33
+ uploadService = (0, uploadService_1.createUploadService)();
34
+ }
35
+ catch (error) {
36
+ logger_1.logger.error(error instanceof Error ? error.message : 'Unknown error');
37
+ return;
38
+ }
39
+ let projects;
40
+ try {
41
+ projects = await uploadService.listProjects();
42
+ }
43
+ catch (error) {
44
+ logger_1.logger.error('Failed to fetch projects');
45
+ return;
46
+ }
47
+ if (projects.length === 0) {
48
+ logger_1.logger.warn('No projects deployed.');
49
+ return;
50
+ }
51
+ // Filter to only PM2-managed projects (node, next)
52
+ const pm2Projects = projects.filter(p => p.type === 'node' || p.type === 'next');
53
+ if (pm2Projects.length === 0) {
54
+ logger_1.logger.warn('No Node.js or Next.js projects found. Logs are only available for PM2-managed projects.');
55
+ return;
56
+ }
57
+ const { selected } = await inquirer_1.default.prompt([
58
+ {
59
+ type: 'list',
60
+ name: 'selected',
61
+ message: 'Select project to view logs:',
62
+ choices: pm2Projects.map(p => ({
63
+ name: `${p.name} (${p.type}) — ${p.status}`,
64
+ value: p,
65
+ })),
66
+ },
67
+ ]);
68
+ projectId = selected.id;
69
+ projectName = selected.name;
70
+ }
71
+ // Fetch logs
72
+ const lines = parseInt(options.lines || '100');
73
+ const logType = options.type || 'all';
74
+ logger_1.logger.dim(`Fetching last ${lines} lines for ${projectName}...`);
75
+ logger_1.logger.blank();
76
+ try {
77
+ const response = await axios_1.default.get(`${config.serverUrl}/api/process/${projectId}/logs`, {
78
+ params: { lines, type: logType },
79
+ headers: config.token ? { Authorization: `Bearer ${config.token}` } : {},
80
+ timeout: 15000,
81
+ });
82
+ const data = response.data.data;
83
+ if (data.stdout) {
84
+ console.log(chalk_1.default.bold.green('━━━ stdout ━━━'));
85
+ console.log(data.stdout);
86
+ }
87
+ if (data.stderr) {
88
+ if (data.stdout)
89
+ console.log(); // spacer
90
+ console.log(chalk_1.default.bold.red('━━━ stderr ━━━'));
91
+ console.log(data.stderr);
92
+ }
93
+ if (!data.stdout && !data.stderr) {
94
+ logger_1.logger.dim('No logs found for this project.');
95
+ }
96
+ }
97
+ catch (error) {
98
+ if (axios_1.default.isAxiosError(error)) {
99
+ if (error.response?.status === 404) {
100
+ logger_1.logger.error('Project not found. Use "runway list" to see deployed projects.');
101
+ }
102
+ else {
103
+ logger_1.logger.error(`Failed to fetch logs: ${error.response?.data?.error || error.message}`);
104
+ }
105
+ }
106
+ else {
107
+ logger_1.logger.error(`Failed to fetch logs: ${error.message}`);
108
+ }
109
+ }
110
+ logger_1.logger.blank();
111
+ }
112
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logs.js","sourceRoot":"","sources":["../../src/commands/logs.ts"],"names":[],"mappings":"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","sourcesContent":["import inquirer from 'inquirer';\nimport axios from 'axios';\nimport chalk from 'chalk';\nimport { isConfigured, getConfig, isTokenExpired } from '../utils/config';\nimport { createUploadService } from '../services/uploadService';\nimport { logger } from '../utils/logger';\n\ninterface LogsOptions {\n  lines?: string;\n  type?: string;\n}\n\nexport async function logsCommand(project?: string, options: LogsOptions = {}): Promise<void> {\n  logger.header('Runway Logs');\n\n  // Check configuration\n  if (!isConfigured()) {\n    logger.error('CLI not configured. Run \"runway init\" first.');\n    return;\n  }\n\n  // Check token validity\n  if (isTokenExpired()) {\n    logger.error('Your authentication token has expired.');\n    logger.info('Run \"runway login\" to re-authenticate.');\n    return;\n  }\n\n  const config = getConfig();\n\n  // If no project specified, list and let user choose\n  let projectId = project;\n  let projectName = project;\n\n  if (!projectId) {\n    let uploadService;\n    try {\n      uploadService = createUploadService();\n    } catch (error) {\n      logger.error(error instanceof Error ? error.message : 'Unknown error');\n      return;\n    }\n\n    let projects;\n    try {\n      projects = await uploadService.listProjects();\n    } catch (error) {\n      logger.error('Failed to fetch projects');\n      return;\n    }\n\n    if (projects.length === 0) {\n      logger.warn('No projects deployed.');\n      return;\n    }\n\n    // Filter to only PM2-managed projects (node, next)\n    const pm2Projects = projects.filter(p => p.type === 'node' || p.type === 'next');\n\n    if (pm2Projects.length === 0) {\n      logger.warn('No Node.js or Next.js projects found. Logs are only available for PM2-managed projects.');\n      return;\n    }\n\n    const { selected } = await inquirer.prompt([\n      {\n        type: 'list',\n        name: 'selected',\n        message: 'Select project to view logs:',\n        choices: pm2Projects.map(p => ({\n          name: `${p.name} (${p.type}) — ${p.status}`,\n          value: p,\n        })),\n      },\n    ]);\n\n    projectId = selected.id;\n    projectName = selected.name;\n  }\n\n  // Fetch logs\n  const lines = parseInt(options.lines || '100');\n  const logType = options.type || 'all';\n\n  logger.dim(`Fetching last ${lines} lines for ${projectName}...`);\n  logger.blank();\n\n  try {\n    const response = await axios.get(\n      `${config.serverUrl}/api/process/${projectId}/logs`,\n      {\n        params: { lines, type: logType },\n        headers: config.token ? { Authorization: `Bearer ${config.token}` } : {},\n        timeout: 15000,\n      }\n    );\n\n    const data = response.data.data;\n\n    if (data.stdout) {\n      console.log(chalk.bold.green('━━━ stdout ━━━'));\n      console.log(data.stdout);\n    }\n\n    if (data.stderr) {\n      if (data.stdout) console.log(); // spacer\n      console.log(chalk.bold.red('━━━ stderr ━━━'));\n      console.log(data.stderr);\n    }\n\n    if (!data.stdout && !data.stderr) {\n      logger.dim('No logs found for this project.');\n    }\n  } catch (error: any) {\n    if (axios.isAxiosError(error)) {\n      if (error.response?.status === 404) {\n        logger.error('Project not found. Use \"runway list\" to see deployed projects.');\n      } else {\n        logger.error(`Failed to fetch logs: ${error.response?.data?.error || error.message}`);\n      }\n    } else {\n      logger.error(`Failed to fetch logs: ${error.message}`);\n    }\n  }\n\n  logger.blank();\n}\n"]}
@@ -0,0 +1 @@
1
+ export declare function updateCommand(): Promise<void>;
@@ -0,0 +1,244 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
37
+ };
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ exports.updateCommand = updateCommand;
40
+ const inquirer_1 = __importDefault(require("inquirer"));
41
+ const ora_1 = __importDefault(require("ora"));
42
+ const axios_1 = __importDefault(require("axios"));
43
+ const projectDetector_1 = require("../services/projectDetector");
44
+ const buildService_1 = require("../services/buildService");
45
+ const packageService_1 = require("../services/packageService");
46
+ const uploadService_1 = require("../services/uploadService");
47
+ const config_1 = require("../utils/config");
48
+ const logger_1 = require("../utils/logger");
49
+ async function updateCommand() {
50
+ logger_1.logger.header('Runway Update');
51
+ // Check configuration
52
+ if (!(0, config_1.isConfigured)()) {
53
+ logger_1.logger.error('CLI not configured. Run "runway init" first.');
54
+ return;
55
+ }
56
+ // Check token validity
57
+ if ((0, config_1.isTokenExpired)()) {
58
+ logger_1.logger.error('Your authentication token has expired.');
59
+ logger_1.logger.info('Run "runway login" to re-authenticate.');
60
+ return;
61
+ }
62
+ const config = (0, config_1.getConfig)();
63
+ logger_1.logger.dim(`Server: ${config.serverUrl}`);
64
+ logger_1.logger.blank();
65
+ // Fetch list of deployed projects
66
+ let uploadService;
67
+ try {
68
+ uploadService = (0, uploadService_1.createUploadService)();
69
+ }
70
+ catch (error) {
71
+ logger_1.logger.error(error instanceof Error ? error.message : 'Unknown error');
72
+ return;
73
+ }
74
+ const spinner = (0, ora_1.default)('Fetching deployed projects...').start();
75
+ let projects;
76
+ try {
77
+ projects = await uploadService.listProjects();
78
+ spinner.succeed(`Found ${projects.length} deployed projects`);
79
+ }
80
+ catch (error) {
81
+ spinner.fail('Failed to fetch projects');
82
+ logger_1.logger.error(error instanceof Error ? error.message : 'Unknown error');
83
+ return;
84
+ }
85
+ if (projects.length === 0) {
86
+ logger_1.logger.warn('No projects deployed. Use "runway deploy" to deploy a new project.');
87
+ return;
88
+ }
89
+ // Let user pick a project
90
+ const { selectedProject } = await inquirer_1.default.prompt([
91
+ {
92
+ type: 'list',
93
+ name: 'selectedProject',
94
+ message: 'Select project to update:',
95
+ choices: projects.map(p => ({
96
+ name: `${p.name} (${p.type}) — ${p.status}`,
97
+ value: p,
98
+ })),
99
+ },
100
+ ]);
101
+ const projectName = selectedProject.name;
102
+ const projectType = selectedProject.type;
103
+ logger_1.logger.blank();
104
+ logger_1.logger.info(`Updating: ${projectName} (${projectType})`);
105
+ logger_1.logger.blank();
106
+ // Detect current directory project
107
+ const detectedSpinner = (0, ora_1.default)('Detecting local project...').start();
108
+ let detectedProject;
109
+ try {
110
+ detectedProject = await (0, projectDetector_1.detectProject)();
111
+ detectedSpinner.succeed(`Detected: ${detectedProject.type} project (${detectedProject.packageManager})`);
112
+ }
113
+ catch (error) {
114
+ detectedSpinner.fail('Failed to detect project in current directory');
115
+ logger_1.logger.error(error instanceof Error ? error.message : 'Unknown error');
116
+ return;
117
+ }
118
+ // Confirm
119
+ const { confirm } = await inquirer_1.default.prompt([
120
+ {
121
+ type: 'confirm',
122
+ name: 'confirm',
123
+ message: `Build current directory and update "${projectName}"?`,
124
+ default: true,
125
+ },
126
+ ]);
127
+ if (!confirm) {
128
+ logger_1.logger.warn('Update cancelled.');
129
+ return;
130
+ }
131
+ logger_1.logger.blank();
132
+ // Step 1: Build locally
133
+ logger_1.logger.step(1, 3, 'Building project...');
134
+ // Patch React Router basename before build (if applicable)
135
+ let didPatch = false;
136
+ if (projectType === 'react') {
137
+ const { ReactPatcher } = await Promise.resolve().then(() => __importStar(require('../services/reactPatcher')));
138
+ didPatch = await ReactPatcher.patch(process.cwd());
139
+ }
140
+ const buildResult = await buildService_1.buildService.build({
141
+ projectPath: process.cwd(),
142
+ projectType,
143
+ projectName,
144
+ packageManager: detectedProject.packageManager,
145
+ });
146
+ // Revert patch after build (keep user's source clean)
147
+ if (didPatch) {
148
+ const { ReactPatcher } = await Promise.resolve().then(() => __importStar(require('../services/reactPatcher')));
149
+ await ReactPatcher.revert(process.cwd());
150
+ }
151
+ if (!buildResult.success) {
152
+ logger_1.logger.error(`Build failed: ${buildResult.error}`);
153
+ return;
154
+ }
155
+ logger_1.logger.success(`Build completed in ${(buildResult.duration / 1000).toFixed(1)}s`);
156
+ logger_1.logger.blank();
157
+ // Step 2: Package
158
+ logger_1.logger.step(2, 3, 'Creating deployment package...');
159
+ let packageResult;
160
+ try {
161
+ packageResult = await packageService_1.packageService.package({
162
+ projectPath: process.cwd(),
163
+ projectType,
164
+ buildOutputDir: buildResult.outputDir,
165
+ includeSource: false,
166
+ });
167
+ }
168
+ catch (error) {
169
+ logger_1.logger.error(`Packaging failed: ${error instanceof Error ? error.message : 'Unknown error'}`);
170
+ return;
171
+ }
172
+ logger_1.logger.blank();
173
+ // Step 3: Upload as update
174
+ logger_1.logger.step(3, 3, 'Uploading update to server...');
175
+ const uploadResult = await uploadService.upload({
176
+ zipPath: packageResult.zipPath,
177
+ projectName,
178
+ projectType,
179
+ buildMode: 'local',
180
+ deploymentSource: 'cli',
181
+ envInjected: false,
182
+ });
183
+ // Cleanup
184
+ packageService_1.packageService.cleanup(packageResult.zipPath);
185
+ if (!uploadResult.success) {
186
+ logger_1.logger.error(`Upload failed: ${uploadResult.error}`);
187
+ return;
188
+ }
189
+ logger_1.logger.success('Upload complete');
190
+ logger_1.logger.blank();
191
+ // Wait for deployment
192
+ if (uploadResult.deploymentId) {
193
+ try {
194
+ const finalStatus = await uploadService.pollDeploymentStatus(uploadResult.deploymentId, (status) => {
195
+ if (status.progress !== undefined) {
196
+ process.stdout.write(`\r Progress: ${status.progress}%`);
197
+ }
198
+ });
199
+ process.stdout.write('\n');
200
+ if (finalStatus.status === 'success') {
201
+ logger_1.logger.blank();
202
+ logger_1.logger.success(`✅ "${projectName}" updated successfully!`);
203
+ const safeName = projectName.toLowerCase().replace(/[^a-z0-9]/g, '-');
204
+ // Fetch domain config to show proper URL (same pattern as deploy.ts)
205
+ try {
206
+ const domainResponse = await axios_1.default.get(`${config.serverUrl}/api/domain`, {
207
+ headers: config.token ? { Authorization: `Bearer ${config.token}` } : {},
208
+ });
209
+ const domainConfig = domainResponse.data;
210
+ if (domainConfig.domain?.active && domainConfig.securityMode === 'domain-https') {
211
+ logger_1.logger.info(`Your app is available at: https://${domainConfig.domain.domain}/app/${safeName}`);
212
+ }
213
+ else {
214
+ logger_1.logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);
215
+ }
216
+ }
217
+ catch {
218
+ logger_1.logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);
219
+ }
220
+ // Show health warning if service didn't respond on its allocated port
221
+ if (finalStatus.healthWarning) {
222
+ logger_1.logger.blank();
223
+ logger_1.logger.warn(`⚠ Health check: ${finalStatus.healthWarning}`);
224
+ }
225
+ }
226
+ else {
227
+ logger_1.logger.error(`Update failed: ${finalStatus.error || 'Unknown error'}`);
228
+ if (finalStatus.logs) {
229
+ logger_1.logger.blank();
230
+ logger_1.logger.dim('Logs:');
231
+ console.log(finalStatus.logs);
232
+ }
233
+ }
234
+ }
235
+ catch {
236
+ logger_1.logger.warn('Could not track deployment status. Check the web UI.');
237
+ }
238
+ }
239
+ else {
240
+ logger_1.logger.success('Upload successful! Check web UI for deployment status.');
241
+ }
242
+ logger_1.logger.blank();
243
+ }
244
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"update.js","sourceRoot":"","sources":["../../src/commands/update.ts"],"names":[],"mappings":"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","sourcesContent":["import inquirer from 'inquirer';\nimport ora from 'ora';\nimport path from 'path';\nimport axios from 'axios';\nimport { ProjectType } from '../types';\nimport { detectProject } from '../services/projectDetector';\nimport { buildService } from '../services/buildService';\nimport { packageService } from '../services/packageService';\nimport { createUploadService } from '../services/uploadService';\nimport { isConfigured, getConfig, isTokenExpired } from '../utils/config';\nimport { logger } from '../utils/logger';\n\nexport async function updateCommand(): Promise<void> {\n  logger.header('Runway Update');\n\n  // Check configuration\n  if (!isConfigured()) {\n    logger.error('CLI not configured. Run \"runway init\" first.');\n    return;\n  }\n\n  // Check token validity\n  if (isTokenExpired()) {\n    logger.error('Your authentication token has expired.');\n    logger.info('Run \"runway login\" to re-authenticate.');\n    return;\n  }\n\n  const config = getConfig();\n  logger.dim(`Server: ${config.serverUrl}`);\n  logger.blank();\n\n  // Fetch list of deployed projects\n  let uploadService;\n  try {\n    uploadService = createUploadService();\n  } catch (error) {\n    logger.error(error instanceof Error ? error.message : 'Unknown error');\n    return;\n  }\n\n  const spinner = ora('Fetching deployed projects...').start();\n  let projects;\n  try {\n    projects = await uploadService.listProjects();\n    spinner.succeed(`Found ${projects.length} deployed projects`);\n  } catch (error) {\n    spinner.fail('Failed to fetch projects');\n    logger.error(error instanceof Error ? error.message : 'Unknown error');\n    return;\n  }\n\n  if (projects.length === 0) {\n    logger.warn('No projects deployed. Use \"runway deploy\" to deploy a new project.');\n    return;\n  }\n\n  // Let user pick a project\n  const { selectedProject } = await inquirer.prompt([\n    {\n      type: 'list',\n      name: 'selectedProject',\n      message: 'Select project to update:',\n      choices: projects.map(p => ({\n        name: `${p.name} (${p.type}) — ${p.status}`,\n        value: p,\n      })),\n    },\n  ]);\n\n  const projectName = selectedProject.name;\n  const projectType: ProjectType = selectedProject.type;\n\n  logger.blank();\n  logger.info(`Updating: ${projectName} (${projectType})`);\n  logger.blank();\n\n  // Detect current directory project\n  const detectedSpinner = ora('Detecting local project...').start();\n  let detectedProject;\n  try {\n    detectedProject = await detectProject();\n    detectedSpinner.succeed(`Detected: ${detectedProject.type} project (${detectedProject.packageManager})`);\n  } catch (error) {\n    detectedSpinner.fail('Failed to detect project in current directory');\n    logger.error(error instanceof Error ? error.message : 'Unknown error');\n    return;\n  }\n\n  // Confirm\n  const { confirm } = await inquirer.prompt([\n    {\n      type: 'confirm',\n      name: 'confirm',\n      message: `Build current directory and update \"${projectName}\"?`,\n      default: true,\n    },\n  ]);\n\n  if (!confirm) {\n    logger.warn('Update cancelled.');\n    return;\n  }\n\n  logger.blank();\n\n  // Step 1: Build locally\n  logger.step(1, 3, 'Building project...');\n\n  // Patch React Router basename before build (if applicable)\n  let didPatch = false;\n  if (projectType === 'react') {\n    const { ReactPatcher } = await import('../services/reactPatcher');\n    didPatch = await ReactPatcher.patch(process.cwd());\n  }\n\n  const buildResult = await buildService.build({\n    projectPath: process.cwd(),\n    projectType,\n    projectName,\n    packageManager: detectedProject.packageManager,\n  });\n\n  // Revert patch after build (keep user's source clean)\n  if (didPatch) {\n    const { ReactPatcher } = await import('../services/reactPatcher');\n    await ReactPatcher.revert(process.cwd());\n  }\n\n  if (!buildResult.success) {\n    logger.error(`Build failed: ${buildResult.error}`);\n    return;\n  }\n\n  logger.success(`Build completed in ${(buildResult.duration / 1000).toFixed(1)}s`);\n  logger.blank();\n\n  // Step 2: Package\n  logger.step(2, 3, 'Creating deployment package...');\n\n  let packageResult;\n  try {\n    packageResult = await packageService.package({\n      projectPath: process.cwd(),\n      projectType,\n      buildOutputDir: buildResult.outputDir,\n      includeSource: false,\n    });\n  } catch (error) {\n    logger.error(`Packaging failed: ${error instanceof Error ? error.message : 'Unknown error'}`);\n    return;\n  }\n\n  logger.blank();\n\n  // Step 3: Upload as update\n  logger.step(3, 3, 'Uploading update to server...');\n\n  const uploadResult = await uploadService.upload({\n    zipPath: packageResult.zipPath,\n    projectName,\n    projectType,\n    buildMode: 'local',\n    deploymentSource: 'cli',\n    envInjected: false,\n  });\n\n  // Cleanup\n  packageService.cleanup(packageResult.zipPath);\n\n  if (!uploadResult.success) {\n    logger.error(`Upload failed: ${uploadResult.error}`);\n    return;\n  }\n\n  logger.success('Upload complete');\n  logger.blank();\n\n  // Wait for deployment\n  if (uploadResult.deploymentId) {\n    try {\n      const finalStatus = await uploadService.pollDeploymentStatus(\n        uploadResult.deploymentId,\n        (status) => {\n          if (status.progress !== undefined) {\n            process.stdout.write(`\\r  Progress: ${status.progress}%`);\n          }\n        }\n      );\n\n      process.stdout.write('\\n');\n\n      if (finalStatus.status === 'success') {\n        logger.blank();\n        logger.success(`✅ \"${projectName}\" updated successfully!`);\n\n        const safeName = projectName.toLowerCase().replace(/[^a-z0-9]/g, '-');\n\n        // Fetch domain config to show proper URL (same pattern as deploy.ts)\n        try {\n          const domainResponse = await axios.get(`${config.serverUrl}/api/domain`, {\n            headers: config.token ? { Authorization: `Bearer ${config.token}` } : {},\n          });\n          const domainConfig = domainResponse.data;\n          if (domainConfig.domain?.active && domainConfig.securityMode === 'domain-https') {\n            logger.info(`Your app is available at: https://${domainConfig.domain.domain}/app/${safeName}`);\n          } else {\n            logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);\n          }\n        } catch {\n          logger.info(`Your app is available at: ${config.serverUrl}/app/${safeName}`);\n        }\n\n        // Show health warning if service didn't respond on its allocated port\n        if (finalStatus.healthWarning) {\n          logger.blank();\n          logger.warn(`⚠  Health check: ${finalStatus.healthWarning}`);\n        }\n      } else {\n        logger.error(`Update failed: ${finalStatus.error || 'Unknown error'}`);\n        if (finalStatus.logs) {\n          logger.blank();\n          logger.dim('Logs:');\n          console.log(finalStatus.logs);\n        }\n      }\n    } catch {\n      logger.warn('Could not track deployment status. Check the web UI.');\n    }\n  } else {\n    logger.success('Upload successful! Check web UI for deployment status.');\n  }\n\n  logger.blank();\n}\n"]}
package/dist/index.js CHANGED
@@ -9,6 +9,9 @@ const status_1 = require("./commands/status");
9
9
  const reset_1 = require("./commands/reset");
10
10
  const domain_1 = require("./commands/domain");
11
11
  const metrics_1 = require("./commands/metrics");
12
+ const login_1 = require("./commands/login");
13
+ const update_1 = require("./commands/update");
14
+ const logs_1 = require("./commands/logs");
12
15
  const program = new commander_1.Command();
13
16
  program
14
17
  .name('runway')
@@ -20,6 +23,11 @@ program
20
23
  .description('Configure the Runway CLI')
21
24
  .option('-s, --server <url>', 'Server URL')
22
25
  .action(init_1.initCommand);
26
+ // Login command (re-authenticate without full init)
27
+ program
28
+ .command('login')
29
+ .description('Re-authenticate with the Runway server')
30
+ .action(login_1.loginCommand);
23
31
  // Deploy command
24
32
  program
25
33
  .command('deploy')
@@ -31,6 +39,11 @@ program
31
39
  .option('--build-server', 'Upload source and build on server')
32
40
  .option('-e, --env-file <path>', 'Path to environment file')
33
41
  .action(deploy_1.deployCommand);
42
+ // Update command (update existing deployed project)
43
+ program
44
+ .command('update')
45
+ .description('Update an existing deployed project')
46
+ .action(update_1.updateCommand);
34
47
  // List command
35
48
  program
36
49
  .command('list')
@@ -42,6 +55,13 @@ program
42
55
  .command('status <project>')
43
56
  .description('Get status of a deployed project')
44
57
  .action(status_1.statusCommand);
58
+ // Logs command
59
+ program
60
+ .command('logs [project]')
61
+ .description('View logs for a deployed project')
62
+ .option('-n, --lines <count>', 'Number of lines to show', '100')
63
+ .option('--type <type>', 'Log type: out, error, or all', 'all')
64
+ .action(logs_1.logsCommand);
45
65
  // Reset command
46
66
  program
47
67
  .command('reset')
@@ -59,4 +79,4 @@ program
59
79
  .action(metrics_1.metricsCommand);
60
80
  // Parse command line arguments
61
81
  program.parse();
62
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBRUEseUNBQW9DO0FBQ3BDLDBDQUE4QztBQUM5Qyw4Q0FBa0Q7QUFDbEQsMENBQThDO0FBQzlDLDhDQUFrRDtBQUNsRCw0Q0FBZ0Q7QUFDaEQsOENBQWtEO0FBQ2xELGdEQUFvRDtBQUVwRCxNQUFNLE9BQU8sR0FBRyxJQUFJLG1CQUFPLEVBQUUsQ0FBQztBQUU5QixPQUFPO0tBQ0osSUFBSSxDQUFDLFFBQVEsQ0FBQztLQUNkLFdBQVcsQ0FBQywyQ0FBMkMsQ0FBQztLQUN4RCxPQUFPLENBQUMsT0FBTyxDQUFDLENBQUM7QUFFcEIsZUFBZTtBQUNmLE9BQU87S0FDSixPQUFPLENBQUMsTUFBTSxDQUFDO0tBQ2YsV0FBVyxDQUFDLDBCQUEwQixDQUFDO0tBQ3ZDLE1BQU0sQ0FBQyxvQkFBb0IsRUFBRSxZQUFZLENBQUM7S0FDMUMsTUFBTSxDQUFDLGtCQUFXLENBQUMsQ0FBQztBQUV2QixpQkFBaUI7QUFDakIsT0FBTztLQUNKLE9BQU8sQ0FBQyxRQUFRLENBQUM7S0FDakIsV0FBVyxDQUFDLDRCQUE0QixDQUFDO0tBQ3pDLE1BQU0sQ0FBQyxtQkFBbUIsRUFBRSxjQUFjLENBQUM7S0FDM0MsTUFBTSxDQUFDLG1CQUFtQixFQUFFLGtDQUFrQyxDQUFDO0tBQy9ELE1BQU0sQ0FBQyx5QkFBeUIsRUFBRSxnQkFBZ0IsQ0FBQztLQUNuRCxNQUFNLENBQUMsZUFBZSxFQUFFLDBDQUEwQyxDQUFDO0tBQ25FLE1BQU0sQ0FBQyxnQkFBZ0IsRUFBRSxtQ0FBbUMsQ0FBQztLQUM3RCxNQUFNLENBQUMsdUJBQXVCLEVBQUUsMEJBQTBCLENBQUM7S0FDM0QsTUFBTSxDQUFDLHNCQUFhLENBQUMsQ0FBQztBQUV6QixlQUFlO0FBQ2YsT0FBTztLQUNKLE9BQU8sQ0FBQyxNQUFNLENBQUM7S0FDZixLQUFLLENBQUMsSUFBSSxDQUFDO0tBQ1gsV0FBVyxDQUFDLHdCQUF3QixDQUFDO0tBQ3JDLE1BQU0sQ0FBQyxrQkFBVyxDQUFDLENBQUM7QUFFdkIsaUJBQWlCO0FBQ2pCLE9BQU87S0FDSixPQUFPLENBQUMsa0JBQWtCLENBQUM7S0FDM0IsV0FBVyxDQUFDLGtDQUFrQyxDQUFDO0tBQy9DLE1BQU0sQ0FBQyxzQkFBYSxDQUFDLENBQUM7QUFFekIsZ0JBQWdCO0FBQ2hCLE9BQU87S0FDSixPQUFPLENBQUMsT0FBTyxDQUFDO0tBQ2hCLFdBQVcsQ0FBQyx5Q0FBeUMsQ0FBQztLQUN0RCxNQUFNLENBQUMsb0JBQVksQ0FBQyxDQUFDO0FBRXhCLGlCQUFpQjtBQUNqQixPQUFPO0tBQ0osT0FBTyxDQUFDLFFBQVEsQ0FBQztLQUNqQixXQUFXLENBQUMsc0NBQXNDLENBQUM7S0FDbkQsTUFBTSxDQUFDLHNCQUFhLENBQUMsQ0FBQztBQUV6QixrQkFBa0I7QUFDbEIsT0FBTztLQUNKLE9BQU8sQ0FBQyxTQUFTLENBQUM7S0FDbEIsV0FBVyxDQUFDLCtCQUErQixDQUFDO0tBQzVDLE1BQU0sQ0FBQyx3QkFBYyxDQUFDLENBQUM7QUFFMUIsK0JBQStCO0FBQy9CLE9BQU8sQ0FBQyxLQUFLLEVBQUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIiMhL3Vzci9iaW4vZW52IG5vZGVcblxuaW1wb3J0IHsgQ29tbWFuZCB9IGZyb20gJ2NvbW1hbmRlcic7XG5pbXBvcnQgeyBpbml0Q29tbWFuZCB9IGZyb20gJy4vY29tbWFuZHMvaW5pdCc7XG5pbXBvcnQgeyBkZXBsb3lDb21tYW5kIH0gZnJvbSAnLi9jb21tYW5kcy9kZXBsb3knO1xuaW1wb3J0IHsgbGlzdENvbW1hbmQgfSBmcm9tICcuL2NvbW1hbmRzL2xpc3QnO1xuaW1wb3J0IHsgc3RhdHVzQ29tbWFuZCB9IGZyb20gJy4vY29tbWFuZHMvc3RhdHVzJztcbmltcG9ydCB7IHJlc2V0Q29tbWFuZCB9IGZyb20gJy4vY29tbWFuZHMvcmVzZXQnO1xuaW1wb3J0IHsgZG9tYWluQ29tbWFuZCB9IGZyb20gJy4vY29tbWFuZHMvZG9tYWluJztcbmltcG9ydCB7IG1ldHJpY3NDb21tYW5kIH0gZnJvbSAnLi9jb21tYW5kcy9tZXRyaWNzJztcblxuY29uc3QgcHJvZ3JhbSA9IG5ldyBDb21tYW5kKCk7XG5cbnByb2dyYW1cbiAgLm5hbWUoJ3J1bndheScpXG4gIC5kZXNjcmlwdGlvbignQ0xJIHRvb2wgZm9yIGRlcGxveWluZyBwcm9qZWN0cyB0byBSdW53YXknKVxuICAudmVyc2lvbignMC4wLjEnKTtcblxuLy8gSW5pdCBjb21tYW5kXG5wcm9ncmFtXG4gIC5jb21tYW5kKCdpbml0JylcbiAgLmRlc2NyaXB0aW9uKCdDb25maWd1cmUgdGhlIFJ1bndheSBDTEknKVxuICAub3B0aW9uKCctcywgLS1zZXJ2ZXIgPHVybD4nLCAnU2VydmVyIFVSTCcpXG4gIC5hY3Rpb24oaW5pdENvbW1hbmQpO1xuXG4vLyBEZXBsb3kgY29tbWFuZFxucHJvZ3JhbVxuICAuY29tbWFuZCgnZGVwbG95JylcbiAgLmRlc2NyaXB0aW9uKCdEZXBsb3kgdGhlIGN1cnJlbnQgcHJvamVjdCcpXG4gIC5vcHRpb24oJy1uLCAtLW5hbWUgPG5hbWU+JywgJ1Byb2plY3QgbmFtZScpXG4gIC5vcHRpb24oJy10LCAtLXR5cGUgPHR5cGU+JywgJ1Byb2plY3QgdHlwZSAocmVhY3QsIG5leHQsIG5vZGUpJylcbiAgLm9wdGlvbignLXYsIC0tdmVyc2lvbiA8dmVyc2lvbj4nLCAnVmVyc2lvbiBzdHJpbmcnKVxuICAub3B0aW9uKCctLWJ1aWxkLWxvY2FsJywgJ0J1aWxkIGxvY2FsbHkgYmVmb3JlIHVwbG9hZGluZyAoZGVmYXVsdCknKVxuICAub3B0aW9uKCctLWJ1aWxkLXNlcnZlcicsICdVcGxvYWQgc291cmNlIGFuZCBidWlsZCBvbiBzZXJ2ZXInKVxuICAub3B0aW9uKCctZSwgLS1lbnYtZmlsZSA8cGF0aD4nLCAnUGF0aCB0byBlbnZpcm9ubWVudCBmaWxlJylcbiAgLmFjdGlvbihkZXBsb3lDb21tYW5kKTtcblxuLy8gTGlzdCBjb21tYW5kXG5wcm9ncmFtXG4gIC5jb21tYW5kKCdsaXN0JylcbiAgLmFsaWFzKCdscycpXG4gIC5kZXNjcmlwdGlvbignTGlzdCBkZXBsb3llZCBwcm9qZWN0cycpXG4gIC5hY3Rpb24obGlzdENvbW1hbmQpO1xuXG4vLyBTdGF0dXMgY29tbWFuZFxucHJvZ3JhbVxuICAuY29tbWFuZCgnc3RhdHVzIDxwcm9qZWN0PicpXG4gIC5kZXNjcmlwdGlvbignR2V0IHN0YXR1cyBvZiBhIGRlcGxveWVkIHByb2plY3QnKVxuICAuYWN0aW9uKHN0YXR1c0NvbW1hbmQpO1xuXG4vLyBSZXNldCBjb21tYW5kXG5wcm9ncmFtXG4gIC5jb21tYW5kKCdyZXNldCcpXG4gIC5kZXNjcmlwdGlvbignUmVzZXQgQ0xJIGNvbmZpZ3VyYXRpb24gYW5kIHJlLXJ1biBpbml0JylcbiAgLmFjdGlvbihyZXNldENvbW1hbmQpO1xuXG4vLyBEb21haW4gY29tbWFuZFxucHJvZ3JhbVxuICAuY29tbWFuZCgnZG9tYWluJylcbiAgLmRlc2NyaXB0aW9uKCdDb25maWd1cmUgc2VydmVyIGRvbWFpbiAoc2FtZSBhcyBVSSknKVxuICAuYWN0aW9uKGRvbWFpbkNvbW1hbmQpO1xuXG4vLyBNZXRyaWNzIGNvbW1hbmRcbnByb2dyYW1cbiAgLmNvbW1hbmQoJ21ldHJpY3MnKVxuICAuZGVzY3JpcHRpb24oJ0Rpc3BsYXkgc2VydmVyIHN5c3RlbSBtZXRyaWNzJylcbiAgLmFjdGlvbihtZXRyaWNzQ29tbWFuZCk7XG5cbi8vIFBhcnNlIGNvbW1hbmQgbGluZSBhcmd1bWVudHNcbnByb2dyYW0ucGFyc2UoKTtcbiJdfQ==
82
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBRUEseUNBQW9DO0FBQ3BDLDBDQUE4QztBQUM5Qyw4Q0FBa0Q7QUFDbEQsMENBQThDO0FBQzlDLDhDQUFrRDtBQUNsRCw0Q0FBZ0Q7QUFDaEQsOENBQWtEO0FBQ2xELGdEQUFvRDtBQUNwRCw0Q0FBZ0Q7QUFDaEQsOENBQWtEO0FBQ2xELDBDQUE4QztBQUU5QyxNQUFNLE9BQU8sR0FBRyxJQUFJLG1CQUFPLEVBQUUsQ0FBQztBQUU5QixPQUFPO0tBQ0osSUFBSSxDQUFDLFFBQVEsQ0FBQztLQUNkLFdBQVcsQ0FBQywyQ0FBMkMsQ0FBQztLQUN4RCxPQUFPLENBQUMsT0FBTyxDQUFDLENBQUM7QUFFcEIsZUFBZTtBQUNmLE9BQU87S0FDSixPQUFPLENBQUMsTUFBTSxDQUFDO0tBQ2YsV0FBVyxDQUFDLDBCQUEwQixDQUFDO0tBQ3ZDLE1BQU0sQ0FBQyxvQkFBb0IsRUFBRSxZQUFZLENBQUM7S0FDMUMsTUFBTSxDQUFDLGtCQUFXLENBQUMsQ0FBQztBQUV2QixvREFBb0Q7QUFDcEQsT0FBTztLQUNKLE9BQU8sQ0FBQyxPQUFPLENBQUM7S0FDaEIsV0FBVyxDQUFDLHdDQUF3QyxDQUFDO0tBQ3JELE1BQU0sQ0FBQyxvQkFBWSxDQUFDLENBQUM7QUFFeEIsaUJBQWlCO0FBQ2pCLE9BQU87S0FDSixPQUFPLENBQUMsUUFBUSxDQUFDO0tBQ2pCLFdBQVcsQ0FBQyw0QkFBNEIsQ0FBQztLQUN6QyxNQUFNLENBQUMsbUJBQW1CLEVBQUUsY0FBYyxDQUFDO0tBQzNDLE1BQU0sQ0FBQyxtQkFBbUIsRUFBRSxrQ0FBa0MsQ0FBQztLQUMvRCxNQUFNLENBQUMseUJBQXlCLEVBQUUsZ0JBQWdCLENBQUM7S0FDbkQsTUFBTSxDQUFDLGVBQWUsRUFBRSwwQ0FBMEMsQ0FBQztLQUNuRSxNQUFNLENBQUMsZ0JBQWdCLEVBQUUsbUNBQW1DLENBQUM7S0FDN0QsTUFBTSxDQUFDLHVCQUF1QixFQUFFLDBCQUEwQixDQUFDO0tBQzNELE1BQU0sQ0FBQyxzQkFBYSxDQUFDLENBQUM7QUFFekIsb0RBQW9EO0FBQ3BELE9BQU87S0FDSixPQUFPLENBQUMsUUFBUSxDQUFDO0tBQ2pCLFdBQVcsQ0FBQyxxQ0FBcUMsQ0FBQztLQUNsRCxNQUFNLENBQUMsc0JBQWEsQ0FBQyxDQUFDO0FBRXpCLGVBQWU7QUFDZixPQUFPO0tBQ0osT0FBTyxDQUFDLE1BQU0sQ0FBQztLQUNmLEtBQUssQ0FBQyxJQUFJLENBQUM7S0FDWCxXQUFXLENBQUMsd0JBQXdCLENBQUM7S0FDckMsTUFBTSxDQUFDLGtCQUFXLENBQUMsQ0FBQztBQUV2QixpQkFBaUI7QUFDakIsT0FBTztLQUNKLE9BQU8sQ0FBQyxrQkFBa0IsQ0FBQztLQUMzQixXQUFXLENBQUMsa0NBQWtDLENBQUM7S0FDL0MsTUFBTSxDQUFDLHNCQUFhLENBQUMsQ0FBQztBQUV6QixlQUFlO0FBQ2YsT0FBTztLQUNKLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQztLQUN6QixXQUFXLENBQUMsa0NBQWtDLENBQUM7S0FDL0MsTUFBTSxDQUFDLHFCQUFxQixFQUFFLHlCQUF5QixFQUFFLEtBQUssQ0FBQztLQUMvRCxNQUFNLENBQUMsZUFBZSxFQUFFLDhCQUE4QixFQUFFLEtBQUssQ0FBQztLQUM5RCxNQUFNLENBQUMsa0JBQVcsQ0FBQyxDQUFDO0FBRXZCLGdCQUFnQjtBQUNoQixPQUFPO0tBQ0osT0FBTyxDQUFDLE9BQU8sQ0FBQztLQUNoQixXQUFXLENBQUMseUNBQXlDLENBQUM7S0FDdEQsTUFBTSxDQUFDLG9CQUFZLENBQUMsQ0FBQztBQUV4QixpQkFBaUI7QUFDakIsT0FBTztLQUNKLE9BQU8sQ0FBQyxRQUFRLENBQUM7S0FDakIsV0FBVyxDQUFDLHNDQUFzQyxDQUFDO0tBQ25ELE1BQU0sQ0FBQyxzQkFBYSxDQUFDLENBQUM7QUFFekIsa0JBQWtCO0FBQ2xCLE9BQU87S0FDSixPQUFPLENBQUMsU0FBUyxDQUFDO0tBQ2xCLFdBQVcsQ0FBQywrQkFBK0IsQ0FBQztLQUM1QyxNQUFNLENBQUMsd0JBQWMsQ0FBQyxDQUFDO0FBRTFCLCtCQUErQjtBQUMvQixPQUFPLENBQUMsS0FBSyxFQUFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXG5cbmltcG9ydCB7IENvbW1hbmQgfSBmcm9tICdjb21tYW5kZXInO1xuaW1wb3J0IHsgaW5pdENvbW1hbmQgfSBmcm9tICcuL2NvbW1hbmRzL2luaXQnO1xuaW1wb3J0IHsgZGVwbG95Q29tbWFuZCB9IGZyb20gJy4vY29tbWFuZHMvZGVwbG95JztcbmltcG9ydCB7IGxpc3RDb21tYW5kIH0gZnJvbSAnLi9jb21tYW5kcy9saXN0JztcbmltcG9ydCB7IHN0YXR1c0NvbW1hbmQgfSBmcm9tICcuL2NvbW1hbmRzL3N0YXR1cyc7XG5pbXBvcnQgeyByZXNldENvbW1hbmQgfSBmcm9tICcuL2NvbW1hbmRzL3Jlc2V0JztcbmltcG9ydCB7IGRvbWFpbkNvbW1hbmQgfSBmcm9tICcuL2NvbW1hbmRzL2RvbWFpbic7XG5pbXBvcnQgeyBtZXRyaWNzQ29tbWFuZCB9IGZyb20gJy4vY29tbWFuZHMvbWV0cmljcyc7XG5pbXBvcnQgeyBsb2dpbkNvbW1hbmQgfSBmcm9tICcuL2NvbW1hbmRzL2xvZ2luJztcbmltcG9ydCB7IHVwZGF0ZUNvbW1hbmQgfSBmcm9tICcuL2NvbW1hbmRzL3VwZGF0ZSc7XG5pbXBvcnQgeyBsb2dzQ29tbWFuZCB9IGZyb20gJy4vY29tbWFuZHMvbG9ncyc7XG5cbmNvbnN0IHByb2dyYW0gPSBuZXcgQ29tbWFuZCgpO1xuXG5wcm9ncmFtXG4gIC5uYW1lKCdydW53YXknKVxuICAuZGVzY3JpcHRpb24oJ0NMSSB0b29sIGZvciBkZXBsb3lpbmcgcHJvamVjdHMgdG8gUnVud2F5JylcbiAgLnZlcnNpb24oJzAuMC4xJyk7XG5cbi8vIEluaXQgY29tbWFuZFxucHJvZ3JhbVxuICAuY29tbWFuZCgnaW5pdCcpXG4gIC5kZXNjcmlwdGlvbignQ29uZmlndXJlIHRoZSBSdW53YXkgQ0xJJylcbiAgLm9wdGlvbignLXMsIC0tc2VydmVyIDx1cmw+JywgJ1NlcnZlciBVUkwnKVxuICAuYWN0aW9uKGluaXRDb21tYW5kKTtcblxuLy8gTG9naW4gY29tbWFuZCAocmUtYXV0aGVudGljYXRlIHdpdGhvdXQgZnVsbCBpbml0KVxucHJvZ3JhbVxuICAuY29tbWFuZCgnbG9naW4nKVxuICAuZGVzY3JpcHRpb24oJ1JlLWF1dGhlbnRpY2F0ZSB3aXRoIHRoZSBSdW53YXkgc2VydmVyJylcbiAgLmFjdGlvbihsb2dpbkNvbW1hbmQpO1xuXG4vLyBEZXBsb3kgY29tbWFuZFxucHJvZ3JhbVxuICAuY29tbWFuZCgnZGVwbG95JylcbiAgLmRlc2NyaXB0aW9uKCdEZXBsb3kgdGhlIGN1cnJlbnQgcHJvamVjdCcpXG4gIC5vcHRpb24oJy1uLCAtLW5hbWUgPG5hbWU+JywgJ1Byb2plY3QgbmFtZScpXG4gIC5vcHRpb24oJy10LCAtLXR5cGUgPHR5cGU+JywgJ1Byb2plY3QgdHlwZSAocmVhY3QsIG5leHQsIG5vZGUpJylcbiAgLm9wdGlvbignLXYsIC0tdmVyc2lvbiA8dmVyc2lvbj4nLCAnVmVyc2lvbiBzdHJpbmcnKVxuICAub3B0aW9uKCctLWJ1aWxkLWxvY2FsJywgJ0J1aWxkIGxvY2FsbHkgYmVmb3JlIHVwbG9hZGluZyAoZGVmYXVsdCknKVxuICAub3B0aW9uKCctLWJ1aWxkLXNlcnZlcicsICdVcGxvYWQgc291cmNlIGFuZCBidWlsZCBvbiBzZXJ2ZXInKVxuICAub3B0aW9uKCctZSwgLS1lbnYtZmlsZSA8cGF0aD4nLCAnUGF0aCB0byBlbnZpcm9ubWVudCBmaWxlJylcbiAgLmFjdGlvbihkZXBsb3lDb21tYW5kKTtcblxuLy8gVXBkYXRlIGNvbW1hbmQgKHVwZGF0ZSBleGlzdGluZyBkZXBsb3llZCBwcm9qZWN0KVxucHJvZ3JhbVxuICAuY29tbWFuZCgndXBkYXRlJylcbiAgLmRlc2NyaXB0aW9uKCdVcGRhdGUgYW4gZXhpc3RpbmcgZGVwbG95ZWQgcHJvamVjdCcpXG4gIC5hY3Rpb24odXBkYXRlQ29tbWFuZCk7XG5cbi8vIExpc3QgY29tbWFuZFxucHJvZ3JhbVxuICAuY29tbWFuZCgnbGlzdCcpXG4gIC5hbGlhcygnbHMnKVxuICAuZGVzY3JpcHRpb24oJ0xpc3QgZGVwbG95ZWQgcHJvamVjdHMnKVxuICAuYWN0aW9uKGxpc3RDb21tYW5kKTtcblxuLy8gU3RhdHVzIGNvbW1hbmRcbnByb2dyYW1cbiAgLmNvbW1hbmQoJ3N0YXR1cyA8cHJvamVjdD4nKVxuICAuZGVzY3JpcHRpb24oJ0dldCBzdGF0dXMgb2YgYSBkZXBsb3llZCBwcm9qZWN0JylcbiAgLmFjdGlvbihzdGF0dXNDb21tYW5kKTtcblxuLy8gTG9ncyBjb21tYW5kXG5wcm9ncmFtXG4gIC5jb21tYW5kKCdsb2dzIFtwcm9qZWN0XScpXG4gIC5kZXNjcmlwdGlvbignVmlldyBsb2dzIGZvciBhIGRlcGxveWVkIHByb2plY3QnKVxuICAub3B0aW9uKCctbiwgLS1saW5lcyA8Y291bnQ+JywgJ051bWJlciBvZiBsaW5lcyB0byBzaG93JywgJzEwMCcpXG4gIC5vcHRpb24oJy0tdHlwZSA8dHlwZT4nLCAnTG9nIHR5cGU6IG91dCwgZXJyb3IsIG9yIGFsbCcsICdhbGwnKVxuICAuYWN0aW9uKGxvZ3NDb21tYW5kKTtcblxuLy8gUmVzZXQgY29tbWFuZFxucHJvZ3JhbVxuICAuY29tbWFuZCgncmVzZXQnKVxuICAuZGVzY3JpcHRpb24oJ1Jlc2V0IENMSSBjb25maWd1cmF0aW9uIGFuZCByZS1ydW4gaW5pdCcpXG4gIC5hY3Rpb24ocmVzZXRDb21tYW5kKTtcblxuLy8gRG9tYWluIGNvbW1hbmRcbnByb2dyYW1cbiAgLmNvbW1hbmQoJ2RvbWFpbicpXG4gIC5kZXNjcmlwdGlvbignQ29uZmlndXJlIHNlcnZlciBkb21haW4gKHNhbWUgYXMgVUkpJylcbiAgLmFjdGlvbihkb21haW5Db21tYW5kKTtcblxuLy8gTWV0cmljcyBjb21tYW5kXG5wcm9ncmFtXG4gIC5jb21tYW5kKCdtZXRyaWNzJylcbiAgLmRlc2NyaXB0aW9uKCdEaXNwbGF5IHNlcnZlciBzeXN0ZW0gbWV0cmljcycpXG4gIC5hY3Rpb24obWV0cmljc0NvbW1hbmQpO1xuXG4vLyBQYXJzZSBjb21tYW5kIGxpbmUgYXJndW1lbnRzXG5wcm9ncmFtLnBhcnNlKCk7XG5cbiJdfQ==
@@ -15,6 +15,7 @@ export declare class PackageService {
15
15
  private addSourceFiles;
16
16
  private addBuildArtifacts;
17
17
  private addDirectory;
18
+ private addDirectoryFiltered;
18
19
  private addLockFile;
19
20
  cleanup(zipPath: string): void;
20
21
  }
@@ -75,8 +75,8 @@ class PackageService {
75
75
  logger_1.logger.dim(`Including build output from ${buildOutputDir}`);
76
76
  break;
77
77
  case 'next':
78
- // Include .next directory
79
- this.addDirectory(archive, path_1.default.join(projectPath, '.next'), '.next');
78
+ // Include .next directory but exclude cache (can be 200+ MB of build cache)
79
+ this.addDirectoryFiltered(archive, path_1.default.join(projectPath, '.next'), '.next', ['cache/**']);
80
80
  // Include public directory if exists
81
81
  const publicDir = path_1.default.join(projectPath, 'public');
82
82
  if (fs_1.default.existsSync(publicDir)) {
@@ -139,6 +139,15 @@ class PackageService {
139
139
  archive.directory(dirPath, destPath);
140
140
  }
141
141
  }
142
+ addDirectoryFiltered(archive, dirPath, destPath, ignore) {
143
+ if (fs_1.default.existsSync(dirPath)) {
144
+ archive.glob('**/*', {
145
+ cwd: dirPath,
146
+ ignore,
147
+ dot: true,
148
+ }, { prefix: destPath });
149
+ }
150
+ }
142
151
  addLockFile(archive, projectPath) {
143
152
  const lockFiles = ['package-lock.json', 'yarn.lock', 'pnpm-lock.yaml'];
144
153
  for (const lockFile of lockFiles) {
@@ -157,4 +166,4 @@ class PackageService {
157
166
  }
158
167
  exports.PackageService = PackageService;
159
168
  exports.packageService = new PackageService();
160
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"packageService.js","sourceRoot":"","sources":["../../src/services/packageService.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport path from 'path';\nimport archiver from 'archiver';\nimport { ProjectType } from '../types';\nimport { logger } from '../utils/logger';\n\nexport interface PackageOptions {\n  projectPath: string;\n  projectType: ProjectType;\n  buildOutputDir: string;\n  includeSource: boolean; // For server-build mode\n  envFile?: string; // Path to env file to include (for Node.js projects)\n}\n\nexport interface PackageResult {\n  zipPath: string;\n  size: number;\n}\n\nexport class PackageService {\n  async package(options: PackageOptions): Promise<PackageResult> {\n    const { projectPath, projectType, buildOutputDir, includeSource } = options;\n\n    const zipPath = path.join(projectPath, '.runway-deploy.zip');\n\n    // Remove existing zip if present\n    if (fs.existsSync(zipPath)) {\n      fs.unlinkSync(zipPath);\n    }\n\n    logger.info('Creating deployment package...');\n\n    const output = fs.createWriteStream(zipPath);\n    const archive = archiver('zip', {\n      zlib: { level: 9 }, // Maximum compression\n    });\n\n    return new Promise((resolve, reject) => {\n      output.on('close', () => {\n        const size = archive.pointer();\n        logger.success(`Package created: ${(size / 1024 / 1024).toFixed(2)} MB`);\n        resolve({ zipPath, size });\n      });\n\n      archive.on('error', (err) => {\n        reject(err);\n      });\n\n      archive.pipe(output);\n\n      if (includeSource) {\n        // Server-build mode: include source files\n        this.addSourceFiles(archive, projectPath);\n      } else {\n        // Local-build mode: include only build artifacts\n        this.addBuildArtifacts(archive, projectPath, projectType, buildOutputDir, options.envFile);\n      }\n\n      archive.finalize();\n    });\n  }\n\n  private addSourceFiles(archive: archiver.Archiver, projectPath: string): void {\n    // Include all files except node_modules, .git, and build outputs\n    const ignorePatterns = [\n      'node_modules/**',\n      '.git/**',\n      '.next/**',\n      'dist/**',\n      'build/**',\n      '.runway-deploy.zip',\n      '*.log',\n      '.env.local',\n      '.env.*.local',\n    ];\n\n    archive.glob('**/*', {\n      cwd: projectPath,\n      ignore: ignorePatterns,\n      dot: true, // Include dotfiles\n    });\n\n    logger.dim('Including source files for server-side build');\n  }\n\n  private addBuildArtifacts(\n    archive: archiver.Archiver,\n    projectPath: string,\n    projectType: ProjectType,\n    buildOutputDir: string,\n    envFile?: string\n  ): void {\n    // Include package.json if it exists (optional for static sites)\n    const packageJsonPath = path.join(projectPath, 'package.json');\n    if (fs.existsSync(packageJsonPath)) {\n      archive.file(packageJsonPath, { name: 'package.json' });\n    }\n\n    switch (projectType) {\n      case 'react':\n        // Include built static files\n        this.addDirectory(archive, buildOutputDir, 'dist');\n        logger.dim(`Including build output from ${buildOutputDir}`);\n        break;\n\n      case 'next':\n        // Include .next directory\n        this.addDirectory(archive, path.join(projectPath, '.next'), '.next');\n\n        // Include public directory if exists\n        const publicDir = path.join(projectPath, 'public');\n        if (fs.existsSync(publicDir)) {\n          this.addDirectory(archive, publicDir, 'public');\n        }\n\n        // Include next.config.js if exists\n        for (const configFile of ['next.config.js', 'next.config.mjs', 'next.config.ts']) {\n          const configPath = path.join(projectPath, configFile);\n          if (fs.existsSync(configPath)) {\n            archive.file(configPath, { name: configFile });\n          }\n        }\n\n        // Include package-lock.json or yarn.lock for production dependencies\n        this.addLockFile(archive, projectPath);\n\n        logger.dim('Including Next.js build artifacts');\n        break;\n\n      case 'node':\n        // For Node.js, include everything except node_modules and env files\n        const nodeIgnorePatterns = [\n          'node_modules/**',\n          '.git/**',\n          '.runway-deploy.zip',\n          '*.log',\n          '.env',      // Exclude default .env (we'll add specified one explicitly)\n          '.env.*',\n        ];\n\n        archive.glob('**/*', {\n          cwd: projectPath,\n          ignore: nodeIgnorePatterns,\n          dot: true,\n        });\n\n        // Include the specified env file as .env\n        if (envFile && fs.existsSync(envFile)) {\n          archive.file(envFile, { name: '.env' });\n          logger.dim('Including environment file');\n        }\n\n        logger.dim('Including Node.js project files');\n        break;\n\n      case 'static':\n        // For static sites, include HTML, CSS, JS, and common assets\n        const staticIgnorePatterns = [\n          'node_modules/**',\n          '.git/**',\n          '.runway-deploy.zip',\n          '*.log',\n          '.env',\n          '.env.*',\n        ];\n\n        archive.glob('**/*', {\n          cwd: projectPath,\n          ignore: staticIgnorePatterns,\n          dot: false, // Don't include dotfiles for static sites\n        });\n\n        logger.dim('Including static site files');\n        break;\n    }\n  }\n\n  private addDirectory(archive: archiver.Archiver, dirPath: string, destPath: string): void {\n    if (fs.existsSync(dirPath)) {\n      archive.directory(dirPath, destPath);\n    }\n  }\n\n  private addLockFile(archive: archiver.Archiver, projectPath: string): void {\n    const lockFiles = ['package-lock.json', 'yarn.lock', 'pnpm-lock.yaml'];\n\n    for (const lockFile of lockFiles) {\n      const lockPath = path.join(projectPath, lockFile);\n      if (fs.existsSync(lockPath)) {\n        archive.file(lockPath, { name: lockFile });\n        break;\n      }\n    }\n  }\n\n  cleanup(zipPath: string): void {\n    if (fs.existsSync(zipPath)) {\n      fs.unlinkSync(zipPath);\n    }\n  }\n}\n\nexport const packageService = new PackageService();\n"]}
169
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"packageService.js","sourceRoot":"","sources":["../../src/services/packageService.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport path from 'path';\nimport archiver from 'archiver';\nimport { ProjectType } from '../types';\nimport { logger } from '../utils/logger';\n\nexport interface PackageOptions {\n  projectPath: string;\n  projectType: ProjectType;\n  buildOutputDir: string;\n  includeSource: boolean; // For server-build mode\n  envFile?: string; // Path to env file to include (for Node.js projects)\n}\n\nexport interface PackageResult {\n  zipPath: string;\n  size: number;\n}\n\nexport class PackageService {\n  async package(options: PackageOptions): Promise<PackageResult> {\n    const { projectPath, projectType, buildOutputDir, includeSource } = options;\n\n    const zipPath = path.join(projectPath, '.runway-deploy.zip');\n\n    // Remove existing zip if present\n    if (fs.existsSync(zipPath)) {\n      fs.unlinkSync(zipPath);\n    }\n\n    logger.info('Creating deployment package...');\n\n    const output = fs.createWriteStream(zipPath);\n    const archive = archiver('zip', {\n      zlib: { level: 9 }, // Maximum compression\n    });\n\n    return new Promise((resolve, reject) => {\n      output.on('close', () => {\n        const size = archive.pointer();\n        logger.success(`Package created: ${(size / 1024 / 1024).toFixed(2)} MB`);\n        resolve({ zipPath, size });\n      });\n\n      archive.on('error', (err) => {\n        reject(err);\n      });\n\n      archive.pipe(output);\n\n      if (includeSource) {\n        // Server-build mode: include source files\n        this.addSourceFiles(archive, projectPath);\n      } else {\n        // Local-build mode: include only build artifacts\n        this.addBuildArtifacts(archive, projectPath, projectType, buildOutputDir, options.envFile);\n      }\n\n      archive.finalize();\n    });\n  }\n\n  private addSourceFiles(archive: archiver.Archiver, projectPath: string): void {\n    // Include all files except node_modules, .git, and build outputs\n    const ignorePatterns = [\n      'node_modules/**',\n      '.git/**',\n      '.next/**',\n      'dist/**',\n      'build/**',\n      '.runway-deploy.zip',\n      '*.log',\n      '.env.local',\n      '.env.*.local',\n    ];\n\n    archive.glob('**/*', {\n      cwd: projectPath,\n      ignore: ignorePatterns,\n      dot: true, // Include dotfiles\n    });\n\n    logger.dim('Including source files for server-side build');\n  }\n\n  private addBuildArtifacts(\n    archive: archiver.Archiver,\n    projectPath: string,\n    projectType: ProjectType,\n    buildOutputDir: string,\n    envFile?: string\n  ): void {\n    // Include package.json if it exists (optional for static sites)\n    const packageJsonPath = path.join(projectPath, 'package.json');\n    if (fs.existsSync(packageJsonPath)) {\n      archive.file(packageJsonPath, { name: 'package.json' });\n    }\n\n    switch (projectType) {\n      case 'react':\n        // Include built static files\n        this.addDirectory(archive, buildOutputDir, 'dist');\n        logger.dim(`Including build output from ${buildOutputDir}`);\n        break;\n\n      case 'next':\n        // Include .next directory but exclude cache (can be 200+ MB of build cache)\n        this.addDirectoryFiltered(\n          archive,\n          path.join(projectPath, '.next'),\n          '.next',\n          ['cache/**']\n        );\n\n        // Include public directory if exists\n        const publicDir = path.join(projectPath, 'public');\n        if (fs.existsSync(publicDir)) {\n          this.addDirectory(archive, publicDir, 'public');\n        }\n\n        // Include next.config.js if exists\n        for (const configFile of ['next.config.js', 'next.config.mjs', 'next.config.ts']) {\n          const configPath = path.join(projectPath, configFile);\n          if (fs.existsSync(configPath)) {\n            archive.file(configPath, { name: configFile });\n          }\n        }\n\n        // Include package-lock.json or yarn.lock for production dependencies\n        this.addLockFile(archive, projectPath);\n\n        logger.dim('Including Next.js build artifacts');\n        break;\n\n      case 'node':\n        // For Node.js, include everything except node_modules and env files\n        const nodeIgnorePatterns = [\n          'node_modules/**',\n          '.git/**',\n          '.runway-deploy.zip',\n          '*.log',\n          '.env',      // Exclude default .env (we'll add specified one explicitly)\n          '.env.*',\n        ];\n\n        archive.glob('**/*', {\n          cwd: projectPath,\n          ignore: nodeIgnorePatterns,\n          dot: true,\n        });\n\n        // Include the specified env file as .env\n        if (envFile && fs.existsSync(envFile)) {\n          archive.file(envFile, { name: '.env' });\n          logger.dim('Including environment file');\n        }\n\n        logger.dim('Including Node.js project files');\n        break;\n\n      case 'static':\n        // For static sites, include HTML, CSS, JS, and common assets\n        const staticIgnorePatterns = [\n          'node_modules/**',\n          '.git/**',\n          '.runway-deploy.zip',\n          '*.log',\n          '.env',\n          '.env.*',\n        ];\n\n        archive.glob('**/*', {\n          cwd: projectPath,\n          ignore: staticIgnorePatterns,\n          dot: false, // Don't include dotfiles for static sites\n        });\n\n        logger.dim('Including static site files');\n        break;\n    }\n  }\n\n  private addDirectory(archive: archiver.Archiver, dirPath: string, destPath: string): void {\n    if (fs.existsSync(dirPath)) {\n      archive.directory(dirPath, destPath);\n    }\n  }\n\n  private addDirectoryFiltered(\n    archive: archiver.Archiver,\n    dirPath: string,\n    destPath: string,\n    ignore: string[]\n  ): void {\n    if (fs.existsSync(dirPath)) {\n      archive.glob('**/*', {\n        cwd: dirPath,\n        ignore,\n        dot: true,\n      }, { prefix: destPath });\n    }\n  }\n\n  private addLockFile(archive: archiver.Archiver, projectPath: string): void {\n    const lockFiles = ['package-lock.json', 'yarn.lock', 'pnpm-lock.yaml'];\n\n    for (const lockFile of lockFiles) {\n      const lockPath = path.join(projectPath, lockFile);\n      if (fs.existsSync(lockPath)) {\n        archive.file(lockPath, { name: lockFile });\n        break;\n      }\n    }\n  }\n\n  cleanup(zipPath: string): void {\n    if (fs.existsSync(zipPath)) {\n      fs.unlinkSync(zipPath);\n    }\n  }\n}\n\nexport const packageService = new PackageService();\n"]}
@@ -0,0 +1,22 @@
1
+ /**
2
+ * React Router patcher for CLI builds.
3
+ * Injects `basename={import.meta.env.BASE_URL}` into <BrowserRouter>
4
+ * so client-side routing works under /app/<project-name>/ subpath.
5
+ *
6
+ * This is a port of the server's reactPatcher.ts adapted for CLI usage.
7
+ */
8
+ export declare class ReactPatcher {
9
+ /**
10
+ * Check if React Router patching is needed and apply it.
11
+ * Returns true if any patching was done.
12
+ */
13
+ static patch(projectPath: string): Promise<boolean>;
14
+ /**
15
+ * Revert the basename patch (restore original state after build).
16
+ */
17
+ static revert(projectPath: string): Promise<void>;
18
+ /**
19
+ * Recursively find the file containing <BrowserRouter> usage.
20
+ */
21
+ private static findRouterFile;
22
+ }
@@ -0,0 +1,111 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.ReactPatcher = void 0;
7
+ const fs_1 = __importDefault(require("fs"));
8
+ const path_1 = __importDefault(require("path"));
9
+ const logger_1 = require("../utils/logger");
10
+ /**
11
+ * React Router patcher for CLI builds.
12
+ * Injects `basename={import.meta.env.BASE_URL}` into <BrowserRouter>
13
+ * so client-side routing works under /app/<project-name>/ subpath.
14
+ *
15
+ * This is a port of the server's reactPatcher.ts adapted for CLI usage.
16
+ */
17
+ class ReactPatcher {
18
+ /**
19
+ * Check if React Router patching is needed and apply it.
20
+ * Returns true if any patching was done.
21
+ */
22
+ static async patch(projectPath) {
23
+ const pkgPath = path_1.default.join(projectPath, 'package.json');
24
+ if (!fs_1.default.existsSync(pkgPath))
25
+ return false;
26
+ const pkg = JSON.parse(fs_1.default.readFileSync(pkgPath, 'utf-8'));
27
+ const deps = { ...pkg.dependencies, ...pkg.devDependencies };
28
+ // Only patch if react-router-dom is a dependency
29
+ if (!deps['react-router-dom'])
30
+ return false;
31
+ logger_1.logger.info('React Router detected — checking for BrowserRouter...');
32
+ const srcDir = path_1.default.join(projectPath, 'src');
33
+ const searchDir = fs_1.default.existsSync(srcDir) ? srcDir : projectPath;
34
+ const routerFile = ReactPatcher.findRouterFile(searchDir);
35
+ if (!routerFile) {
36
+ logger_1.logger.warn('⚠️ React Router detected but <BrowserRouter> usage not found.');
37
+ logger_1.logger.warn(' Ensure you add `basename={import.meta.env.BASE_URL}` to your Router manually.');
38
+ return false;
39
+ }
40
+ let content = fs_1.default.readFileSync(routerFile, 'utf-8');
41
+ // Match <BrowserRouter> that does NOT already have basename
42
+ const tagRegex = /<BrowserRouter(?![^>]*basename=)(\s|>)/;
43
+ if (!tagRegex.test(content)) {
44
+ logger_1.logger.info('React Router already configured (basename prop found)');
45
+ return false;
46
+ }
47
+ // Inject basename
48
+ content = content.replace(/<BrowserRouter(\s|>)/, '<BrowserRouter basename={import.meta.env.BASE_URL}$1');
49
+ fs_1.default.writeFileSync(routerFile, content, 'utf-8');
50
+ logger_1.logger.success(`Patched React Router basename in ${path_1.default.basename(routerFile)}`);
51
+ return true;
52
+ }
53
+ /**
54
+ * Revert the basename patch (restore original state after build).
55
+ */
56
+ static async revert(projectPath) {
57
+ const pkgPath = path_1.default.join(projectPath, 'package.json');
58
+ if (!fs_1.default.existsSync(pkgPath))
59
+ return;
60
+ const pkg = JSON.parse(fs_1.default.readFileSync(pkgPath, 'utf-8'));
61
+ const deps = { ...pkg.dependencies, ...pkg.devDependencies };
62
+ if (!deps['react-router-dom'])
63
+ return;
64
+ const srcDir = path_1.default.join(projectPath, 'src');
65
+ const searchDir = fs_1.default.existsSync(srcDir) ? srcDir : projectPath;
66
+ const routerFile = ReactPatcher.findRouterFile(searchDir);
67
+ if (!routerFile)
68
+ return;
69
+ let content = fs_1.default.readFileSync(routerFile, 'utf-8');
70
+ // Remove the injected basename prop
71
+ const injectedPattern = /<BrowserRouter basename=\{import\.meta\.env\.BASE_URL\}(\s|>)/;
72
+ if (injectedPattern.test(content)) {
73
+ content = content.replace(injectedPattern, '<BrowserRouter$1');
74
+ fs_1.default.writeFileSync(routerFile, content, 'utf-8');
75
+ logger_1.logger.dim('Reverted React Router basename patch');
76
+ }
77
+ }
78
+ /**
79
+ * Recursively find the file containing <BrowserRouter> usage.
80
+ */
81
+ static findRouterFile(dir) {
82
+ try {
83
+ const entries = fs_1.default.readdirSync(dir);
84
+ for (const entry of entries) {
85
+ const fullPath = path_1.default.join(dir, entry);
86
+ const stat = fs_1.default.statSync(fullPath);
87
+ if (stat.isDirectory()) {
88
+ // Skip non-source directories
89
+ if (entry.startsWith('.') || entry === 'node_modules' || entry === 'dist' ||
90
+ entry === 'build' || entry === 'vendor' || entry === 'public')
91
+ continue;
92
+ const found = ReactPatcher.findRouterFile(fullPath);
93
+ if (found)
94
+ return found;
95
+ }
96
+ else if (/^(main|index|layout|home|App)\.(tsx|jsx|js)$/i.test(entry)) {
97
+ const content = fs_1.default.readFileSync(fullPath, 'utf-8');
98
+ if (content.includes('react-router-dom') && content.includes('<BrowserRouter')) {
99
+ return fullPath;
100
+ }
101
+ }
102
+ }
103
+ }
104
+ catch {
105
+ // Ignore read errors
106
+ }
107
+ return null;
108
+ }
109
+ }
110
+ exports.ReactPatcher = ReactPatcher;
111
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reactPatcher.js","sourceRoot":"","sources":["../../src/services/reactPatcher.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport path from 'path';\nimport { logger } from '../utils/logger';\n\n/**\n * React Router patcher for CLI builds.\n * Injects `basename={import.meta.env.BASE_URL}` into <BrowserRouter>\n * so client-side routing works under /app/<project-name>/ subpath.\n * \n * This is a port of the server's reactPatcher.ts adapted for CLI usage.\n */\nexport class ReactPatcher {\n  /**\n   * Check if React Router patching is needed and apply it.\n   * Returns true if any patching was done.\n   */\n  static async patch(projectPath: string): Promise<boolean> {\n    const pkgPath = path.join(projectPath, 'package.json');\n    if (!fs.existsSync(pkgPath)) return false;\n\n    const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'));\n    const deps = { ...pkg.dependencies, ...pkg.devDependencies };\n\n    // Only patch if react-router-dom is a dependency\n    if (!deps['react-router-dom']) return false;\n\n    logger.info('React Router detected — checking for BrowserRouter...');\n\n    const srcDir = path.join(projectPath, 'src');\n    const searchDir = fs.existsSync(srcDir) ? srcDir : projectPath;\n    const routerFile = ReactPatcher.findRouterFile(searchDir);\n\n    if (!routerFile) {\n      logger.warn('⚠️  React Router detected but <BrowserRouter> usage not found.');\n      logger.warn('   Ensure you add `basename={import.meta.env.BASE_URL}` to your Router manually.');\n      return false;\n    }\n\n    let content = fs.readFileSync(routerFile, 'utf-8');\n\n    // Match <BrowserRouter> that does NOT already have basename\n    const tagRegex = /<BrowserRouter(?![^>]*basename=)(\\s|>)/;\n\n    if (!tagRegex.test(content)) {\n      logger.info('React Router already configured (basename prop found)');\n      return false;\n    }\n\n    // Inject basename\n    content = content.replace(\n      /<BrowserRouter(\\s|>)/,\n      '<BrowserRouter basename={import.meta.env.BASE_URL}$1'\n    );\n    fs.writeFileSync(routerFile, content, 'utf-8');\n    logger.success(`Patched React Router basename in ${path.basename(routerFile)}`);\n    return true;\n  }\n\n  /**\n   * Revert the basename patch (restore original state after build).\n   */\n  static async revert(projectPath: string): Promise<void> {\n    const pkgPath = path.join(projectPath, 'package.json');\n    if (!fs.existsSync(pkgPath)) return;\n\n    const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'));\n    const deps = { ...pkg.dependencies, ...pkg.devDependencies };\n    if (!deps['react-router-dom']) return;\n\n    const srcDir = path.join(projectPath, 'src');\n    const searchDir = fs.existsSync(srcDir) ? srcDir : projectPath;\n    const routerFile = ReactPatcher.findRouterFile(searchDir);\n\n    if (!routerFile) return;\n\n    let content = fs.readFileSync(routerFile, 'utf-8');\n\n    // Remove the injected basename prop\n    const injectedPattern = /<BrowserRouter basename=\\{import\\.meta\\.env\\.BASE_URL\\}(\\s|>)/;\n    if (injectedPattern.test(content)) {\n      content = content.replace(injectedPattern, '<BrowserRouter$1');\n      fs.writeFileSync(routerFile, content, 'utf-8');\n      logger.dim('Reverted React Router basename patch');\n    }\n  }\n\n  /**\n   * Recursively find the file containing <BrowserRouter> usage.\n   */\n  private static findRouterFile(dir: string): string | null {\n    try {\n      const entries = fs.readdirSync(dir);\n\n      for (const entry of entries) {\n        const fullPath = path.join(dir, entry);\n        const stat = fs.statSync(fullPath);\n\n        if (stat.isDirectory()) {\n          // Skip non-source directories\n          if (entry.startsWith('.') || entry === 'node_modules' || entry === 'dist' ||\n              entry === 'build' || entry === 'vendor' || entry === 'public') continue;\n          const found = ReactPatcher.findRouterFile(fullPath);\n          if (found) return found;\n        } else if (/^(main|index|layout|home|App)\\.(tsx|jsx|js)$/i.test(entry)) {\n          const content = fs.readFileSync(fullPath, 'utf-8');\n          if (content.includes('react-router-dom') && content.includes('<BrowserRouter')) {\n            return fullPath;\n          }\n        }\n      }\n    } catch {\n      // Ignore read errors\n    }\n    return null;\n  }\n}\n"]}
@@ -20,6 +20,7 @@ export interface DeploymentStatus {
20
20
  progress?: number;
21
21
  logs?: string;
22
22
  error?: string;
23
+ healthWarning?: string;
23
24
  }
24
25
  export interface DeployWarning {
25
26
  level: 'info' | 'warning' | 'critical';
@@ -219,4 +219,4 @@ const createUploadService = () => {
219
219
  return new UploadService();
220
220
  };
221
221
  exports.createUploadService = createUploadService;
222
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"uploadService.js","sourceRoot":"","sources":["../../src/services/uploadService.ts"],"names":[],"mappings":"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","sourcesContent":["import axios, { AxiosProgressEvent } from 'axios';\nimport FormData from 'form-data';\nimport fs from 'fs';\nimport { ProjectType, BuildMode } from '../types';\nimport { getConfig } from '../utils/config';\nimport { logger } from '../utils/logger';\n\nexport interface UploadOptions {\n  zipPath: string;\n  projectName: string;\n  projectType: ProjectType;\n  version?: string;\n  buildMode: BuildMode;\n  confirmServerBuild?: boolean;\n  // ENV mutability tracking\n  deploymentSource?: 'ui' | 'cli';\n  envInjected?: boolean;\n}\n\nexport interface UploadResult {\n  success: boolean;\n  projectId?: string;\n  deploymentId?: string;\n  error?: string;\n}\n\nexport interface DeploymentStatus {\n  status: 'queued' | 'building' | 'deploying' | 'success' | 'failed';\n  progress?: number;\n  logs?: string;\n  error?: string;\n}\n\nexport interface DeployWarning {\n  level: 'info' | 'warning' | 'critical';\n  message: string;\n  code: string;\n}\n\n/**\n * Analysis result from backend\n * Backend trusts user-declared type - no auto-detection\n */\nexport interface DeployAnalysis {\n  // User's declared type (trusted, not validated)\n  declaredType: ProjectType;\n  // Package state\n  hasPackageJson: boolean;\n  hasBuildScript: boolean;\n  hasStartScript: boolean;\n  // Build state (generic detection)\n  hasBuildOutput: boolean;\n  buildOutputDir: string | null;\n  requiresBuild: boolean;\n  // Prebuilt detection (generic)\n  isPrebuiltProject: boolean;\n  // Static site detection (generic)\n  isStaticSite: boolean;\n  // Deployment strategy\n  strategy: 'static' | 'build-and-serve' | 'serve-prebuilt';\n  serveMethod: 'caddy-static' | 'pm2-proxy';\n  // Warnings\n  warnings: DeployWarning[];\n  requiresConfirmation: boolean;\n  confirmationReason?: string;\n}\n\nexport interface AnalyzeResult {\n  success: boolean;\n  analysis?: DeployAnalysis;\n  error?: string;\n}\n\nexport class UploadService {\n  private serverUrl: string;\n  private token: string;\n\n  constructor() {\n    const config = getConfig();\n    if (!config.serverUrl || !config.token) {\n      throw new Error('CLI not configured. Run \"runway init\" first.');\n    }\n    this.serverUrl = config.serverUrl;\n    this.token = config.token;\n  }\n\n  async upload(options: UploadOptions): Promise<UploadResult> {\n    const { zipPath, projectName, projectType, version, buildMode, confirmServerBuild, deploymentSource, envInjected } = options;\n\n    if (!fs.existsSync(zipPath)) {\n      return {\n        success: false,\n        error: `Zip file not found: ${zipPath}`,\n      };\n    }\n\n    const formData = new FormData();\n    formData.append('file', fs.createReadStream(zipPath));\n    formData.append('name', projectName);\n    formData.append('type', projectType);\n    formData.append('buildMode', buildMode);\n    if (version) {\n      formData.append('version', version);\n    }\n    if (confirmServerBuild) {\n      formData.append('confirmServerBuild', 'true');\n    }\n    // ENV mutability tracking\n    if (deploymentSource) {\n      formData.append('deploymentSource', deploymentSource);\n    }\n    if (envInjected !== undefined) {\n      formData.append('envInjected', envInjected ? 'true' : 'false');\n    }\n\n    // Choose endpoint based on build mode\n    const endpoint = buildMode === 'local'\n      ? '/api/project/deploy-prebuilt'\n      : '/api/project/deploy';\n\n    logger.info(`Uploading to ${this.serverUrl}${endpoint}...`);\n\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}${endpoint}`,\n        formData,\n        {\n          headers: {\n            ...formData.getHeaders(),\n            Authorization: `Bearer ${this.token}`,\n          },\n          maxBodyLength: Infinity,\n          maxContentLength: Infinity,\n          timeout: 300000, // 5 minutes\n          onUploadProgress: (progressEvent: AxiosProgressEvent) => {\n            if (progressEvent.total) {\n              const percent = Math.round((progressEvent.loaded * 100) / progressEvent.total);\n              process.stdout.write(`\\r  Uploading: ${percent}%`);\n            }\n          },\n        }\n      );\n\n      process.stdout.write('\\n');\n\n      if (response.data.success) {\n        return {\n          success: true,\n          projectId: response.data.data?.projectId,\n          deploymentId: response.data.data?.deploymentId,\n        };\n      } else {\n        return {\n          success: false,\n          error: response.data.error || 'Unknown error',\n        };\n      }\n    } catch (error) {\n      process.stdout.write('\\n');\n\n      if (axios.isAxiosError(error)) {\n        const message = error.response?.data?.error || error.message;\n        return {\n          success: false,\n          error: `Upload failed: ${message}`,\n        };\n      }\n\n      return {\n        success: false,\n        error: error instanceof Error ? error.message : 'Unknown error',\n      };\n    }\n  }\n\n  async getDeploymentStatus(deploymentId: string): Promise<DeploymentStatus> {\n    try {\n      const response = await axios.get(\n        `${this.serverUrl}/api/project/status/${deploymentId}`,\n        {\n          headers: {\n            Authorization: `Bearer ${this.token}`,\n          },\n        }\n      );\n\n      return response.data.data;\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  async pollDeploymentStatus(\n    deploymentId: string,\n    onUpdate: (status: DeploymentStatus) => void,\n    timeoutMs: number = 300000 // 5 minutes\n  ): Promise<DeploymentStatus> {\n    const startTime = Date.now();\n    const pollInterval = 2000; // 2 seconds\n\n    while (Date.now() - startTime < timeoutMs) {\n      const status = await this.getDeploymentStatus(deploymentId);\n      onUpdate(status);\n\n      if (status.status === 'success' || status.status === 'failed') {\n        return status;\n      }\n\n      await new Promise(resolve => setTimeout(resolve, pollInterval));\n    }\n\n    throw new Error('Deployment timed out');\n  }\n\n  async login(username: string, password: string): Promise<string> {\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}/api/auth/login`,\n        { username, password },\n        {\n          timeout: 10000,\n        }\n      );\n\n      if (response.data.success && response.data.data?.token) {\n        return response.data.data.token;\n      }\n\n      throw new Error(response.data.error || 'Login failed');\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  async listProjects(): Promise<Array<{ id: string; name: string; type: ProjectType; status: string }>> {\n    try {\n      const response = await axios.get(\n        `${this.serverUrl}/api/project`,\n        {\n          headers: {\n            Authorization: `Bearer ${this.token}`,\n          },\n        }\n      );\n\n      return response.data.data || [];\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Analyze a package before deployment to get server warnings and recommendations\n   */\n  /**\n   * Analyze a package before deployment\n   * @param zipPath - Path to the zip file\n   * @param declaredType - REQUIRED - User-selected project type (backend trusts this)\n   */\n  async analyzePackage(zipPath: string, declaredType: ProjectType): Promise<AnalyzeResult> {\n    if (!fs.existsSync(zipPath)) {\n      return {\n        success: false,\n        error: `Zip file not found: ${zipPath}`,\n      };\n    }\n\n    const formData = new FormData();\n    formData.append('file', fs.createReadStream(zipPath));\n    formData.append('type', declaredType);\n\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}/api/project/analyze`,\n        formData,\n        {\n          headers: {\n            ...formData.getHeaders(),\n            Authorization: `Bearer ${this.token}`,\n          },\n          maxBodyLength: Infinity,\n          maxContentLength: Infinity,\n          timeout: 60000, // 1 minute for analysis\n        }\n      );\n\n      if (response.data.success) {\n        return {\n          success: true,\n          analysis: response.data.data,\n        };\n      } else {\n        return {\n          success: false,\n          error: response.data.error || 'Analysis failed',\n        };\n      }\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        const message = error.response?.data?.error || error.message;\n        return {\n          success: false,\n          error: `Analysis failed: ${message}`,\n        };\n      }\n\n      return {\n        success: false,\n        error: error instanceof Error ? error.message : 'Unknown error',\n      };\n    }\n  }\n}\n\nexport const createUploadService = (): UploadService => {\n  return new UploadService();\n};\n"]}
222
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"uploadService.js","sourceRoot":"","sources":["../../src/services/uploadService.ts"],"names":[],"mappings":"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","sourcesContent":["import axios, { AxiosProgressEvent } from 'axios';\nimport FormData from 'form-data';\nimport fs from 'fs';\nimport { ProjectType, BuildMode } from '../types';\nimport { getConfig } from '../utils/config';\nimport { logger } from '../utils/logger';\n\nexport interface UploadOptions {\n  zipPath: string;\n  projectName: string;\n  projectType: ProjectType;\n  version?: string;\n  buildMode: BuildMode;\n  confirmServerBuild?: boolean;\n  // ENV mutability tracking\n  deploymentSource?: 'ui' | 'cli';\n  envInjected?: boolean;\n}\n\nexport interface UploadResult {\n  success: boolean;\n  projectId?: string;\n  deploymentId?: string;\n  error?: string;\n}\n\nexport interface DeploymentStatus {\n  status: 'queued' | 'building' | 'deploying' | 'success' | 'failed';\n  progress?: number;\n  logs?: string;\n  error?: string;\n  healthWarning?: string;\n}\n\nexport interface DeployWarning {\n  level: 'info' | 'warning' | 'critical';\n  message: string;\n  code: string;\n}\n\n/**\n * Analysis result from backend\n * Backend trusts user-declared type - no auto-detection\n */\nexport interface DeployAnalysis {\n  // User's declared type (trusted, not validated)\n  declaredType: ProjectType;\n  // Package state\n  hasPackageJson: boolean;\n  hasBuildScript: boolean;\n  hasStartScript: boolean;\n  // Build state (generic detection)\n  hasBuildOutput: boolean;\n  buildOutputDir: string | null;\n  requiresBuild: boolean;\n  // Prebuilt detection (generic)\n  isPrebuiltProject: boolean;\n  // Static site detection (generic)\n  isStaticSite: boolean;\n  // Deployment strategy\n  strategy: 'static' | 'build-and-serve' | 'serve-prebuilt';\n  serveMethod: 'caddy-static' | 'pm2-proxy';\n  // Warnings\n  warnings: DeployWarning[];\n  requiresConfirmation: boolean;\n  confirmationReason?: string;\n}\n\nexport interface AnalyzeResult {\n  success: boolean;\n  analysis?: DeployAnalysis;\n  error?: string;\n}\n\nexport class UploadService {\n  private serverUrl: string;\n  private token: string;\n\n  constructor() {\n    const config = getConfig();\n    if (!config.serverUrl || !config.token) {\n      throw new Error('CLI not configured. Run \"runway init\" first.');\n    }\n    this.serverUrl = config.serverUrl;\n    this.token = config.token;\n  }\n\n  async upload(options: UploadOptions): Promise<UploadResult> {\n    const { zipPath, projectName, projectType, version, buildMode, confirmServerBuild, deploymentSource, envInjected } = options;\n\n    if (!fs.existsSync(zipPath)) {\n      return {\n        success: false,\n        error: `Zip file not found: ${zipPath}`,\n      };\n    }\n\n    const formData = new FormData();\n    formData.append('file', fs.createReadStream(zipPath));\n    formData.append('name', projectName);\n    formData.append('type', projectType);\n    formData.append('buildMode', buildMode);\n    if (version) {\n      formData.append('version', version);\n    }\n    if (confirmServerBuild) {\n      formData.append('confirmServerBuild', 'true');\n    }\n    // ENV mutability tracking\n    if (deploymentSource) {\n      formData.append('deploymentSource', deploymentSource);\n    }\n    if (envInjected !== undefined) {\n      formData.append('envInjected', envInjected ? 'true' : 'false');\n    }\n\n    // Choose endpoint based on build mode\n    const endpoint = buildMode === 'local'\n      ? '/api/project/deploy-prebuilt'\n      : '/api/project/deploy';\n\n    logger.info(`Uploading to ${this.serverUrl}${endpoint}...`);\n\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}${endpoint}`,\n        formData,\n        {\n          headers: {\n            ...formData.getHeaders(),\n            Authorization: `Bearer ${this.token}`,\n          },\n          maxBodyLength: Infinity,\n          maxContentLength: Infinity,\n          timeout: 300000, // 5 minutes\n          onUploadProgress: (progressEvent: AxiosProgressEvent) => {\n            if (progressEvent.total) {\n              const percent = Math.round((progressEvent.loaded * 100) / progressEvent.total);\n              process.stdout.write(`\\r  Uploading: ${percent}%`);\n            }\n          },\n        }\n      );\n\n      process.stdout.write('\\n');\n\n      if (response.data.success) {\n        return {\n          success: true,\n          projectId: response.data.data?.projectId,\n          deploymentId: response.data.data?.deploymentId,\n        };\n      } else {\n        return {\n          success: false,\n          error: response.data.error || 'Unknown error',\n        };\n      }\n    } catch (error) {\n      process.stdout.write('\\n');\n\n      if (axios.isAxiosError(error)) {\n        const message = error.response?.data?.error || error.message;\n        return {\n          success: false,\n          error: `Upload failed: ${message}`,\n        };\n      }\n\n      return {\n        success: false,\n        error: error instanceof Error ? error.message : 'Unknown error',\n      };\n    }\n  }\n\n  async getDeploymentStatus(deploymentId: string): Promise<DeploymentStatus> {\n    try {\n      const response = await axios.get(\n        `${this.serverUrl}/api/project/status/${deploymentId}`,\n        {\n          headers: {\n            Authorization: `Bearer ${this.token}`,\n          },\n        }\n      );\n\n      return response.data.data;\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  async pollDeploymentStatus(\n    deploymentId: string,\n    onUpdate: (status: DeploymentStatus) => void,\n    timeoutMs: number = 300000 // 5 minutes\n  ): Promise<DeploymentStatus> {\n    const startTime = Date.now();\n    const pollInterval = 2000; // 2 seconds\n\n    while (Date.now() - startTime < timeoutMs) {\n      const status = await this.getDeploymentStatus(deploymentId);\n      onUpdate(status);\n\n      if (status.status === 'success' || status.status === 'failed') {\n        return status;\n      }\n\n      await new Promise(resolve => setTimeout(resolve, pollInterval));\n    }\n\n    throw new Error('Deployment timed out');\n  }\n\n  async login(username: string, password: string): Promise<string> {\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}/api/auth/login`,\n        { username, password },\n        {\n          timeout: 10000,\n        }\n      );\n\n      if (response.data.success && response.data.data?.token) {\n        return response.data.data.token;\n      }\n\n      throw new Error(response.data.error || 'Login failed');\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  async listProjects(): Promise<Array<{ id: string; name: string; type: ProjectType; status: string }>> {\n    try {\n      const response = await axios.get(\n        `${this.serverUrl}/api/project`,\n        {\n          headers: {\n            Authorization: `Bearer ${this.token}`,\n          },\n        }\n      );\n\n      return response.data.data || [];\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Analyze a package before deployment to get server warnings and recommendations\n   */\n  /**\n   * Analyze a package before deployment\n   * @param zipPath - Path to the zip file\n   * @param declaredType - REQUIRED - User-selected project type (backend trusts this)\n   */\n  async analyzePackage(zipPath: string, declaredType: ProjectType): Promise<AnalyzeResult> {\n    if (!fs.existsSync(zipPath)) {\n      return {\n        success: false,\n        error: `Zip file not found: ${zipPath}`,\n      };\n    }\n\n    const formData = new FormData();\n    formData.append('file', fs.createReadStream(zipPath));\n    formData.append('type', declaredType);\n\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}/api/project/analyze`,\n        formData,\n        {\n          headers: {\n            ...formData.getHeaders(),\n            Authorization: `Bearer ${this.token}`,\n          },\n          maxBodyLength: Infinity,\n          maxContentLength: Infinity,\n          timeout: 60000, // 1 minute for analysis\n        }\n      );\n\n      if (response.data.success) {\n        return {\n          success: true,\n          analysis: response.data.data,\n        };\n      } else {\n        return {\n          success: false,\n          error: response.data.error || 'Analysis failed',\n        };\n      }\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        const message = error.response?.data?.error || error.message;\n        return {\n          success: false,\n          error: `Analysis failed: ${message}`,\n        };\n      }\n\n      return {\n        success: false,\n        error: error instanceof Error ? error.message : 'Unknown error',\n      };\n    }\n  }\n}\n\nexport const createUploadService = (): UploadService => {\n  return new UploadService();\n};\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "runway-cli",
3
- "version": "1.2.0",
3
+ "version": "1.3.1",
4
4
  "description": "CLI tool for deploying projects to Runway",
5
5
  "main": "dist/index.js",
6
6
  "bin": {