prime-dev-cli 1.0.19 → 1.0.20

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (73) hide show
  1. package/dist/server/.next/BUILD_ID +1 -1
  2. package/dist/server/.next/app-build-manifest.json +1 -1
  3. package/dist/server/.next/build-manifest.json +2 -2
  4. package/dist/server/.next/cache/.rscinfo +1 -1
  5. package/dist/server/.next/cache/.tsbuildinfo +1 -1
  6. package/dist/server/.next/cache/eslint/.cache_1qa5vxt +1 -1
  7. package/dist/server/.next/fallback-build-manifest.json +2 -2
  8. package/dist/server/.next/next-minimal-server.js.nft.json +1 -1
  9. package/dist/server/.next/next-server.js.nft.json +1 -1
  10. package/dist/server/.next/prerender-manifest.json +9 -9
  11. package/dist/server/.next/server/app/_not-found.html +1 -1
  12. package/dist/server/.next/server/app/_not-found.rsc +1 -1
  13. package/dist/server/.next/server/app/api/uat-git-flow/route.js +2 -2
  14. package/dist/server/.next/server/app/api/uat-git-flow/route.js.nft.json +1 -1
  15. package/dist/server/.next/server/app/api/uat-git-flow/step/route.js +2 -2
  16. package/dist/server/.next/server/app/api/uat-git-flow/step/route.js.nft.json +1 -1
  17. package/dist/server/.next/server/app/configuration.html +1 -1
  18. package/dist/server/.next/server/app/configuration.rsc +1 -1
  19. package/dist/server/.next/server/app/index.html +1 -1
  20. package/dist/server/.next/server/app/index.rsc +1 -1
  21. package/dist/server/.next/server/app/initialization/page/app-build-manifest.json +1 -1
  22. package/dist/server/.next/server/app/initialization/page_client-reference-manifest.js +1 -1
  23. package/dist/server/.next/server/app/initialization.html +1 -1
  24. package/dist/server/.next/server/app/initialization.rsc +2 -2
  25. package/dist/server/.next/server/chunks/[root-of-the-server]__0879b2d6._.js.map +1 -1
  26. package/dist/server/.next/server/chunks/[root-of-the-server]__0d56f35b._.js.map +1 -1
  27. package/dist/server/.next/server/chunks/[root-of-the-server]__0e8a08fc._.js.map +1 -1
  28. package/dist/server/.next/server/chunks/[root-of-the-server]__1e792460._.js.map +1 -1
  29. package/dist/server/.next/server/chunks/[root-of-the-server]__2a29597f._.js.map +1 -1
  30. package/dist/server/.next/server/chunks/[root-of-the-server]__2a78acef._.js.map +1 -1
  31. package/dist/server/.next/server/chunks/[root-of-the-server]__2f8b52da._.js.map +1 -1
  32. package/dist/server/.next/server/chunks/[root-of-the-server]__2ffbf0d3._.js.map +1 -1
  33. package/dist/server/.next/server/chunks/[root-of-the-server]__452b5f9c._.js +13 -13
  34. package/dist/server/.next/server/chunks/[root-of-the-server]__452b5f9c._.js.map +1 -1
  35. package/dist/server/.next/server/chunks/{[root-of-the-server]__4ab3dd83._.js → [root-of-the-server]__4ad2e049._.js} +14 -14
  36. package/dist/server/.next/server/chunks/[root-of-the-server]__4ad2e049._.js.map +1 -0
  37. package/dist/server/.next/server/chunks/[root-of-the-server]__4ec3205b._.js.map +1 -1
  38. package/dist/server/.next/server/chunks/[root-of-the-server]__4fd52ac1._.js +13 -13
  39. package/dist/server/.next/server/chunks/[root-of-the-server]__4fd52ac1._.js.map +1 -1
  40. package/dist/server/.next/server/chunks/[root-of-the-server]__5c9f62f9._.js.map +1 -1
  41. package/dist/server/.next/server/chunks/[root-of-the-server]__8637b7f1._.js.map +1 -1
  42. package/dist/server/.next/server/chunks/[root-of-the-server]__a6fbefc6._.js.map +1 -1
  43. package/dist/server/.next/server/chunks/[root-of-the-server]__a74239fc._.js.map +1 -1
  44. package/dist/server/.next/server/chunks/{[root-of-the-server]__6cbd50e9._.js → [root-of-the-server]__a8ae0f8f._.js} +14 -14
  45. package/dist/server/.next/server/chunks/[root-of-the-server]__a8ae0f8f._.js.map +1 -0
  46. package/dist/server/.next/server/chunks/[root-of-the-server]__aaa3fe74._.js.map +1 -1
  47. package/dist/server/.next/server/chunks/[root-of-the-server]__b176e8df._.js +14 -14
  48. package/dist/server/.next/server/chunks/[root-of-the-server]__b176e8df._.js.map +1 -1
  49. package/dist/server/.next/server/chunks/[root-of-the-server]__c18498ef._.js.map +1 -1
  50. package/dist/server/.next/server/chunks/[root-of-the-server]__dda45cac._.js.map +1 -1
  51. package/dist/server/.next/server/chunks/[root-of-the-server]__de18efbd._.js +13 -13
  52. package/dist/server/.next/server/chunks/[root-of-the-server]__de18efbd._.js.map +1 -1
  53. package/dist/server/.next/server/chunks/[root-of-the-server]__e0b7050d._.js.map +1 -1
  54. package/dist/server/.next/server/chunks/[root-of-the-server]__ec681197._.js.map +1 -1
  55. package/dist/server/.next/server/chunks/_24087f66._.js +1 -1
  56. package/dist/server/.next/server/chunks/_24087f66._.js.map +1 -1
  57. package/dist/server/.next/server/chunks/_61c263b3._.js +1 -1
  58. package/dist/server/.next/server/chunks/_61c263b3._.js.map +1 -1
  59. package/dist/server/.next/server/chunks/ssr/packages_server_src_7cef6dbd._.js.map +1 -1
  60. package/dist/server/.next/server/pages/404.html +1 -1
  61. package/dist/server/.next/server/pages/500.html +1 -1
  62. package/dist/server/.next/server/server-reference-manifest.js +1 -1
  63. package/dist/server/.next/server/server-reference-manifest.json +1 -1
  64. package/dist/server/.next/static/chunks/{9bc1bbb3fe6e5eb3.js → 36cdd24a97bd8ed5.js} +1 -1
  65. package/dist/server/.next/static/chunks/df7954b7bd4ea3eb.js.map +1 -0
  66. package/dist/server/.next/trace +1 -1
  67. package/package.json +1 -1
  68. package/dist/server/.next/server/chunks/[root-of-the-server]__4ab3dd83._.js.map +0 -1
  69. package/dist/server/.next/server/chunks/[root-of-the-server]__6cbd50e9._.js.map +0 -1
  70. package/dist/server/.next/static/chunks/99154153690ad580.js.map +0 -1
  71. /package/dist/server/.next/static/{tz_bf3MbzRijrx8XdXPWA → y0HjvfId3sNkXoR9vLiLd}/_buildManifest.js +0 -0
  72. /package/dist/server/.next/static/{tz_bf3MbzRijrx8XdXPWA → y0HjvfId3sNkXoR9vLiLd}/_clientMiddlewareManifest.json +0 -0
  73. /package/dist/server/.next/static/{tz_bf3MbzRijrx8XdXPWA → y0HjvfId3sNkXoR9vLiLd}/_ssgManifest.js +0 -0
@@ -1 +1 @@
1
- {"version":3,"sources":["turbopack:///[project]/packages/server/src/lib/workspace-types.ts","turbopack:///[project]/packages/server/src/lib/workspace-config.ts","turbopack:///[project]/packages/server/src/lib/env-file-operations.ts","turbopack:///[project]/packages/server/src/lib/sse-connection-manager.ts","turbopack:///[project]/packages/server/src/app/api/microapp-proxy-config/route.ts","turbopack:///[project]/node_modules/.pnpm/next@15.3.4_react-dom@19.1.0_react@19.1.0__react@19.1.0/node_modules/next/dist/src/build/templates/app-route.ts"],"sourcesContent":["import { homedir, cpus } from 'os';\nimport { join } from 'path';\n\n/**\n * 工作空间配置常量\n */\nexport const WORKSPACE_CONFIG = {\n // 工作空间根目录路径\n WORKSPACE_ROOT: join(homedir(), 'Documents', 'prime-workspace'),\n\n // Git 操作超时时间 (毫秒)\n GIT_TIMEOUT: 300000, // 5分钟\n\n // 最大并发克隆数量(基于CPU核心数,最小2个,最大8个)\n MAX_CONCURRENT_CLONES: Math.max(2, Math.min(cpus().length, 8)),\n\n // 重试次数限制\n MAX_RETRY_ATTEMPTS: 3,\n} as const;\n\nexport const PROJECT_CONFIG_PATH = join(homedir(), '.prime-projects.json');\n\nexport interface ProjectConfig {\n repo: string;\n mockingIntercept?: boolean;\n mockOpen?: boolean;\n envs: {\n [envName: string]: {\n host: string;\n currentProxy: string;\n proxyEnv: {\n [key: string]: string;\n };\n envFileName: string;\n proxyKey: string;\n subApps?: string[];\n };\n };\n}\n\nexport interface ProjectsConfig {\n [projectName: string]: ProjectConfig;\n}\n\n/**\n * 项目状态枚举\n */\nexport enum ProjectStatus {\n PENDING = 'pending',\n EXISTING = 'existing',\n CLONING = 'cloning',\n SUCCESS = 'success',\n FAILED = 'failed',\n RETRYING = 'retrying',\n INCOMPLETE = 'incomplete', // 不完整的项目,需要清理\n}\n\n/**\n * 项目进度信息接口\n */\nexport interface ProjectProgress {\n projectName: string;\n status: ProjectStatus;\n progress: number; // 0-100\n message: string;\n error?: string;\n startTime?: number;\n endTime?: number;\n}\n\n/**\n * 初始化结果接口\n */\nexport interface InitializationResult {\n success: boolean;\n totalProjects: number;\n successCount: number;\n failedCount: number;\n failedProjects: string[];\n workspacePath: string;\n duration: number;\n}\n\n/**\n * Git 操作配置接口\n */\nexport interface GitOperationConfig {\n repoUrl: string;\n targetDir: string;\n timeout: number;\n onProgress?: (progress: number, message: string) => void;\n onError?: (error: string) => void;\n onLog?: (projectName: string, logType: 'stdout' | 'stderr' | 'progress', content: string) => void;\n} ","import { readFile, writeFile, access, constants, rename } from 'fs/promises';\nimport type { ProjectsConfig } from './workspace-types';\nimport { PROJECT_CONFIG_PATH } from './workspace-types';\n\n/**\n * 读取并解析项目配置文件\n * @returns {Promise<ProjectsConfig>} 项目配置对象\n */\nexport async function getProjectsConfig(): Promise<ProjectsConfig> {\n try {\n await access(PROJECT_CONFIG_PATH, constants.F_OK);\n } catch {\n // 文件不存在,创建并写入空对象\n await saveProjectsConfig({});\n return {};\n }\n\n try {\n const configContent = await readFile(PROJECT_CONFIG_PATH, 'utf-8');\n // 如果文件为空,返回空对象\n if (!configContent) {\n return {};\n }\n return JSON.parse(configContent);\n } catch (error) {\n console.error('读取或解析项目配置文件失败:', error);\n // 在出错时返回空对象或抛出错误,这里选择抛出以通知调用者\n throw new Error('无法加载项目配置');\n }\n}\n\n/**\n * 将项目配置写入文件(原子操作)\n * @param {ProjectsConfig} config - 要保存的配置对象\n */\nexport async function saveProjectsConfig(config: ProjectsConfig): Promise<void> {\n const tempConfigPath = PROJECT_CONFIG_PATH + '.tmp';\n try {\n await writeFile(tempConfigPath, JSON.stringify(config, null, 2), 'utf-8');\n await rename(tempConfigPath, PROJECT_CONFIG_PATH);\n } catch (error) {\n console.error('保存项目配置文件失败:', error);\n throw new Error('无法保存项目配置');\n }\n}\n\n/**\n * 项目状态枚举\n */\nexport enum ProjectStatus {\n PENDING = 'pending',\n EXISTING = 'existing',\n CLONING = 'cloning',\n SUCCESS = 'success',\n FAILED = 'failed',\n RETRYING = 'retrying',\n}\n\n/**\n * 项目进度信息接口\n */\nexport interface ProjectProgress {\n projectName: string;\n status: ProjectStatus;\n progress: number; // 0-100\n message: string;\n error?: string;\n startTime?: number;\n endTime?: number;\n}\n\n/**\n * 初始化结果接口\n */\nexport interface InitializationResult {\n success: boolean;\n totalProjects: number;\n successCount: number;\n failedCount: number;\n failedProjects: string[];\n workspacePath: string;\n duration: number;\n}\n\n/**\n * Git 操作配置接口\n */\nexport interface GitOperationConfig {\n repoUrl: string;\n targetDir: string;\n timeout: number;\n onProgress?: (progress: number, message: string) => void;\n onError?: (error: string) => void;\n}","import { promises as fs } from 'fs';\nimport { join } from 'path';\nimport type { ProjectConfig } from './workspace-types';\n\n/**\n * 为指定项目创建环境文件,如果文件已存在则跳过。\n * @param projectName - 项目名称\n * @param projectConfig - 项目配置\n * @param workspacePath - 工作空间根路径\n * @returns 一个包含已创建和已跳过文件列表的对象\n */\nexport async function createProjectEnvFiles(\n projectName: string,\n projectConfig: ProjectConfig,\n workspacePath: string,\n): Promise<{\n success: boolean;\n createdFiles: string[];\n skippedFiles: string[];\n error?: string;\n}> {\n const projectPath = join(workspacePath, projectName);\n const createdFiles: string[] = [];\n const skippedFiles: string[] = [];\n\n try {\n const envs = projectConfig.envs || {};\n for (const envName in envs) {\n const envConfig = envs[envName];\n if (envConfig.envFileName && envConfig.proxyKey) {\n const filePath = join(projectPath, envConfig.envFileName);\n \n try {\n await fs.access(filePath);\n // 文件已存在,跳过\n skippedFiles.push(envConfig.envFileName);\n } catch {\n // 文件不存在,创建它\n const currentProxyUrl = envConfig.proxyEnv[envConfig.currentProxy] || '';\n const content = `${envConfig.proxyKey}=${currentProxyUrl}`;\n await fs.writeFile(filePath, content, 'utf-8');\n createdFiles.push(envConfig.envFileName);\n }\n }\n }\n return { success: true, createdFiles, skippedFiles };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n console.error(`在为项目 ${projectName} 创建环境文件时出错:`, error);\n return { success: false, createdFiles, skippedFiles, error: errorMessage };\n }\n}\n\n/**\n * 为指定项目创建.mock文件夹\n * @param projectName - 项目名称\n * @param workspacePath - 工作空间根路径\n * @returns 创建结果\n */\nexport async function createProjectMockDirectories(\n projectName: string,\n workspacePath: string,\n): Promise<{\n success: boolean;\n createdDirectories: string[];\n skippedDirectories: string[];\n error?: string;\n}> {\n const projectPath = join(workspacePath, projectName);\n const mockPath = join(projectPath, '.mock');\n const createdDirectories: string[] = [];\n const skippedDirectories: string[] = [];\n\n try {\n // 创建 .mock 文件夹\n try {\n await fs.access(mockPath);\n skippedDirectories.push('.mock');\n } catch {\n await fs.mkdir(mockPath, { recursive: true });\n createdDirectories.push('.mock');\n }\n\n return { success: true, createdDirectories, skippedDirectories };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n console.error(`在为项目 ${projectName} 创建Mock目录时出错:`, error);\n return { success: false, createdDirectories, skippedDirectories, error: errorMessage };\n }\n}\n\n/**\n * 更新指定项目环境的环境文件内容\n * @param projectName - 项目名称\n * @param envName - 环境名称\n * @param projectConfig - 项目配置\n * @param workspacePath - 工作空间根路径\n * @returns 更新结果\n */\nexport async function updateProjectEnvFile(\n projectName: string,\n envName: string,\n projectConfig: ProjectConfig,\n workspacePath: string,\n): Promise<{\n success: boolean;\n updatedFile?: string;\n error?: string;\n}> {\n try {\n const envConfig = projectConfig.envs[envName];\n \n if (!envConfig) {\n return {\n success: false,\n error: `环境 ${envName} 不存在于项目 ${projectName} 配置中`,\n };\n }\n\n if (!envConfig.envFileName || !envConfig.proxyKey) {\n return {\n success: false,\n error: `环境 ${envName} 缺少 envFileName 或 proxyKey 配置`,\n };\n }\n\n const projectPath = join(workspacePath, projectName);\n const filePath = join(projectPath, envConfig.envFileName);\n \n // 获取当前代理URL\n const currentProxyUrl = envConfig.proxyEnv[envConfig.currentProxy] || envConfig.currentProxy;\n const content = `${envConfig.proxyKey}=${currentProxyUrl}`;\n \n // 写入文件内容\n await fs.writeFile(filePath, content, 'utf-8');\n \n console.log(`已更新环境文件: ${filePath}, 内容: ${content}`);\n \n return {\n success: true,\n updatedFile: envConfig.envFileName,\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n console.error(`更新项目 ${projectName} 环境 ${envName} 的环境文件时出错:`, error);\n return {\n success: false,\n error: errorMessage,\n };\n }\n} ","/**\n * SSE连接管理器\n * 用于管理MicroApp代理配置变化推送的SSE连接\n */\n\nexport interface ConnectionInfo {\n controller: ReadableStreamDefaultController<Uint8Array>;\n timestamp: number;\n clientId: string;\n envName: string;\n}\n\nexport interface ConfigChangeMessage {\n type: 'microapp-proxy-config-change';\n envName: string;\n timestamp: number;\n changes: {\n before: string[];\n after: string[];\n added: string[];\n removed: string[];\n };\n projectName: string;\n message: string;\n}\n\nexport interface HeartbeatMessage {\n type: 'heartbeat';\n timestamp: number;\n}\n\ntype SSEMessage = ConfigChangeMessage | HeartbeatMessage;\n\n/**\n * SSE连接管理器类\n */\nclass SSEConnectionManager {\n private connections = new Map<string, Set<ConnectionInfo>>();\n private encoder = new TextEncoder();\n private heartbeatInterval: NodeJS.Timeout | null = null;\n private readonly HEARTBEAT_INTERVAL = 30000; // 30秒心跳\n private readonly CONNECTION_TIMEOUT = 120000; // 2分钟连接超时\n private readonly MAX_CONNECTIONS_PER_ENV = 10; // 每个环境最多10个连接\n\n constructor() {\n this.startHeartbeat();\n }\n \n /**\n * 清理所有连接(用于内存清理)\n */\n clearAllConnections(): void {\n console.log('[SSE] 开始清理所有连接...');\n const totalConnections = Array.from(this.connections.values()).reduce((sum, set) => sum + set.size, 0);\n \n this.connections.forEach((connections) => {\n connections.forEach(conn => {\n try {\n conn.controller.close();\n } catch {\n // 忽略关闭错误\n }\n });\n });\n \n this.connections.clear();\n console.log(`[SSE] 已清理 ${totalConnections} 个连接`);\n }\n\n /**\n * 添加新的SSE连接\n */\n addConnection(envName: string, controller: ReadableStreamDefaultController<Uint8Array>, clientId: string): void {\n console.log(`[SSE] 添加连接: 环境=${envName}, 客户端=${clientId}`);\n \n if (!this.connections.has(envName)) {\n this.connections.set(envName, new Set());\n }\n \n const connections = this.connections.get(envName)!;\n \n // 检查连接数限制\n if (connections.size >= this.MAX_CONNECTIONS_PER_ENV) {\n console.warn(`[SSE] 环境 ${envName} 连接数已达上限,清理最旧的连接`);\n // 找到最旧的连接并关闭\n const oldestConn = Array.from(connections).sort((a, b) => a.timestamp - b.timestamp)[0];\n if (oldestConn) {\n try {\n oldestConn.controller.close();\n } catch {\n // 忽略关闭错误\n }\n connections.delete(oldestConn);\n }\n }\n \n const connectionInfo: ConnectionInfo = {\n controller,\n timestamp: Date.now(),\n clientId,\n envName,\n };\n \n connections.add(connectionInfo);\n \n // 发送连接确认消息\n this.sendToConnection(connectionInfo, {\n type: 'heartbeat',\n timestamp: Date.now(),\n });\n\n console.log(`[SSE] 当前连接统计: ${this.getConnectionStats()}`);\n }\n\n /**\n * 移除SSE连接\n */\n removeConnection(envName: string, clientId: string): void {\n console.log(`[SSE] 移除连接: 环境=${envName}, 客户端=${clientId}`);\n \n const connections = this.connections.get(envName);\n if (connections) {\n const connectionToRemove = Array.from(connections).find(conn => conn.clientId === clientId);\n if (connectionToRemove) {\n connections.delete(connectionToRemove);\n \n // 如果该环境没有连接了,删除整个entry\n if (connections.size === 0) {\n this.connections.delete(envName);\n }\n }\n }\n\n console.log(`[SSE] 当前连接统计: ${this.getConnectionStats()}`);\n }\n\n /**\n * 向指定环境广播消息\n */\n broadcastToEnv(envName: string, message: ConfigChangeMessage): void {\n console.log(`[SSE] 向环境 ${envName} 广播配置变化消息`);\n \n const connections = this.connections.get(envName);\n if (!connections || connections.size === 0) {\n console.log(`[SSE] 环境 ${envName} 没有活跃连接,跳过广播`);\n return;\n }\n\n const deadConnections: ConnectionInfo[] = [];\n \n connections.forEach(connectionInfo => {\n try {\n this.sendToConnection(connectionInfo, message);\n } catch (error) {\n console.error(`[SSE] 发送消息到客户端 ${connectionInfo.clientId} 失败:`, error);\n deadConnections.push(connectionInfo);\n }\n });\n\n // 清理无效连接\n deadConnections.forEach(conn => {\n connections.delete(conn);\n console.log(`[SSE] 清理无效连接: 客户端=${conn.clientId}`);\n });\n\n console.log(`[SSE] 成功发送给 ${connections.size} 个连接`);\n }\n\n /**\n * 向单个连接发送消息\n */\n private sendToConnection(connectionInfo: ConnectionInfo, message: SSEMessage): void {\n const data = `data: ${JSON.stringify(message)}\\n\\n`;\n const encodedData = this.encoder.encode(data);\n \n try {\n connectionInfo.controller.enqueue(encodedData);\n connectionInfo.timestamp = Date.now(); // 更新最后活跃时间\n } catch (error) {\n console.error(`[SSE] 发送消息失败:`, error);\n throw error;\n }\n }\n\n /**\n * 启动心跳机制\n */\n private startHeartbeat(): void {\n this.heartbeatInterval = setInterval(() => {\n this.sendHeartbeat();\n this.cleanupStaleConnections();\n }, this.HEARTBEAT_INTERVAL);\n \n console.log(`[SSE] 心跳机制已启动,间隔: ${this.HEARTBEAT_INTERVAL}ms`);\n }\n\n /**\n * 发送心跳消息\n */\n private sendHeartbeat(): void {\n const heartbeatMessage: HeartbeatMessage = {\n type: 'heartbeat',\n timestamp: Date.now(),\n };\n\n let totalConnections = 0;\n this.connections.forEach((connections) => {\n totalConnections += connections.size;\n const deadConnections: ConnectionInfo[] = [];\n \n connections.forEach(connectionInfo => {\n try {\n this.sendToConnection(connectionInfo, heartbeatMessage);\n } catch {\n deadConnections.push(connectionInfo);\n }\n });\n\n // 清理无效连接\n deadConnections.forEach(conn => connections.delete(conn));\n });\n\n if (totalConnections > 0) {\n console.log(`[SSE] 心跳发送完成,活跃连接数: ${totalConnections}`);\n }\n }\n\n /**\n * 清理过期连接\n */\n private cleanupStaleConnections(): void {\n const now = Date.now();\n let cleanedCount = 0;\n\n this.connections.forEach((connections, envName) => {\n const staleConnections: ConnectionInfo[] = [];\n \n connections.forEach(connectionInfo => {\n if (now - connectionInfo.timestamp > this.CONNECTION_TIMEOUT) {\n staleConnections.push(connectionInfo);\n }\n });\n\n staleConnections.forEach(conn => {\n connections.delete(conn);\n cleanedCount++;\n console.log(`[SSE] 清理过期连接: 环境=${envName}, 客户端=${conn.clientId}`);\n });\n\n // 如果该环境没有连接了,删除整个entry\n if (connections.size === 0) {\n this.connections.delete(envName);\n }\n });\n\n if (cleanedCount > 0) {\n console.log(`[SSE] 连接清理完成,清理数量: ${cleanedCount}`);\n }\n }\n\n /**\n * 获取连接统计信息\n */\n getConnectionStats(): string {\n const stats: string[] = [];\n let totalConnections = 0;\n \n this.connections.forEach((connections, envName) => {\n const count = connections.size;\n totalConnections += count;\n stats.push(`${envName}: ${count}`);\n });\n\n return `总计 ${totalConnections} 个连接 (${stats.join(', ')})`;\n }\n\n /**\n * 停止连接管理器\n */\n stop(): void {\n if (this.heartbeatInterval) {\n clearInterval(this.heartbeatInterval);\n this.heartbeatInterval = null;\n console.log(`[SSE] 心跳机制已停止`);\n }\n\n // 关闭所有连接\n this.connections.forEach((connections) => {\n connections.forEach(connectionInfo => {\n try {\n connectionInfo.controller.close();\n } catch (error) {\n console.error(`[SSE] 关闭连接失败:`, error);\n }\n });\n });\n\n this.connections.clear();\n console.log(`[SSE] 所有连接已关闭`);\n }\n}\n\n// 创建全局实例\nexport const sseConnectionManager = new SSEConnectionManager();\n\n// 在进程退出时清理资源\nprocess.on('SIGINT', () => {\n console.log('[SSE] 收到退出信号,正在清理连接...');\n sseConnectionManager.stop();\n process.exit(0);\n});\n\nprocess.on('SIGTERM', () => {\n console.log('[SSE] 收到终止信号,正在清理连接...');\n sseConnectionManager.stop();\n process.exit(0);\n}); ","import { NextRequest, NextResponse } from 'next/server';\nimport {\n getProjectsConfig,\n saveProjectsConfig,\n} from '@/lib/workspace-config';\nimport { updateProjectEnvFile } from '@/lib/env-file-operations';\nimport { WORKSPACE_CONFIG } from '@/lib/workspace-types';\nimport { sseConnectionManager } from '@/lib/sse-connection-manager';\nimport type { ConfigChangeMessage } from '@/lib/sse-connection-manager';\n\nexport async function PUT(request: NextRequest) {\n try {\n // 解析请求体\n const { envName, projectName, selectedItems } = await request.json();\n\n // 参数验证\n if (!envName || !projectName || !Array.isArray(selectedItems)) {\n return NextResponse.json(\n {\n success: false,\n error: '缺少必要参数:envName, projectName, selectedItems',\n },\n { status: 400 },\n );\n }\n\n // 读取当前配置文件\n const projectsConfig = await getProjectsConfig();\n\n // 检查项目是否存在\n if (!projectsConfig[projectName]) {\n return NextResponse.json(\n { success: false, error: `项目 ${projectName} 不存在` },\n { status: 404 },\n );\n }\n\n // 检查环境是否存在\n if (!projectsConfig[projectName].envs[envName]) {\n return NextResponse.json(\n {\n success: false,\n error: `项目 ${projectName} 中不存在环境 ${envName}`,\n },\n { status: 404 },\n );\n }\n\n // 获取当前环境的host\n const currentHost = projectsConfig[projectName].envs[envName].host;\n\n // 保存更新前的配置用于变化检测\n const beforeSubApps = projectsConfig[projectName].envs[envName].subApps || [];\n\n // 更新 subApps 配置\n projectsConfig[projectName].envs[envName].subApps = selectedItems;\n\n // 记录更新的项目和环境,用于后续更新环境文件\n const updatedProjects: Array<{projectName: string, envName: string}> = [];\n\n // 将选中的环境的代理改为当前环境的host\n selectedItems.forEach((selectedEnvName: string) => {\n // 遍历所有项目,找到包含该环境的项目并更新其代理配置\n Object.keys(projectsConfig).forEach(projName => {\n if (projectsConfig[projName].envs[selectedEnvName]) {\n // 找到匹配的环境,更新其代理配置\n const targetEnv = projectsConfig[projName].envs[selectedEnvName];\n \n // 将选中环境的代理设置为custom,并指向当前环境的host\n targetEnv.currentProxy = 'custom';\n targetEnv.proxyEnv.custom = currentHost;\n \n // 记录需要更新环境文件的项目和环境\n updatedProjects.push({\n projectName: projName,\n envName: selectedEnvName\n });\n }\n });\n });\n\n // 保存更新后的配置\n await saveProjectsConfig(projectsConfig);\n\n // 检测配置变化并推送通知\n const afterSubApps = selectedItems;\n const hasChanges = JSON.stringify(beforeSubApps.sort()) !== JSON.stringify(afterSubApps.sort());\n \n if (hasChanges) {\n console.log(`[配置变化] 环境 ${envName} 的 MicroApp 代理配置发生变化`);\n console.log(`[配置变化] 变化前: ${JSON.stringify(beforeSubApps)}`);\n console.log(`[配置变化] 变化后: ${JSON.stringify(afterSubApps)}`);\n\n // 计算具体变化\n const beforeSet = new Set(beforeSubApps);\n const afterSet = new Set(afterSubApps);\n const added = afterSubApps.filter(item => !beforeSet.has(item));\n const removed = beforeSubApps.filter(item => !afterSet.has(item));\n\n // 构建推送消息\n const changeMessage: ConfigChangeMessage = {\n type: 'microapp-proxy-config-change',\n envName,\n timestamp: Date.now(),\n changes: {\n before: beforeSubApps,\n after: afterSubApps,\n added,\n removed,\n },\n projectName,\n message: `环境 ${envName} 的 MicroApp 代理配置已更新`\n };\n\n // 推送给相关环境的连接\n try {\n sseConnectionManager.broadcastToEnv(envName, changeMessage);\n console.log(`[推送成功] 已向环境 ${envName} 推送配置变化通知`);\n } catch (pushError) {\n console.error(`[推送失败] 向环境 ${envName} 推送通知失败:`, pushError);\n // 推送失败不影响主要功能,只记录错误\n }\n } else {\n console.log(`[配置未变化] 环境 ${envName} 的 MicroApp 代理配置无变化,跳过推送`);\n }\n\n // 更新相关项目的环境文件\n const envFileUpdates = await Promise.allSettled(\n updatedProjects.map(async ({ projectName: projName, envName: envNameToUpdate }) => {\n return updateProjectEnvFile(\n projName,\n envNameToUpdate,\n projectsConfig[projName],\n WORKSPACE_CONFIG.WORKSPACE_ROOT,\n );\n })\n );\n\n // 检查环境文件更新结果\n const failedUpdates = envFileUpdates\n .map((result, index) => ({ result, project: updatedProjects[index] }))\n .filter(({ result }) => result.status === 'rejected' || (result.status === 'fulfilled' && !result.value.success));\n\n // 构建响应消息\n let message = 'MicroApp代理配置保存成功,已更新选中环境的代理地址';\n if (failedUpdates.length > 0) {\n message += `,但有 ${failedUpdates.length} 个环境文件更新失败`;\n }\n\n return NextResponse.json({\n success: true,\n message,\n data: {\n projectName,\n envName,\n subApps: selectedItems,\n currentHost,\n updatedProjects: updatedProjects.length,\n envFileUpdateFailures: failedUpdates.length,\n },\n });\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : '保存MicroApp代理配置失败';\n console.error('保存MicroApp代理配置失败:', error);\n return NextResponse.json(\n { success: false, error: errorMessage },\n { status: 500 },\n );\n }\n} ","import {\n AppRouteRouteModule,\n type AppRouteRouteModuleOptions,\n} from '../../server/route-modules/app-route/module.compiled'\nimport { RouteKind } from '../../server/route-kind'\nimport { patchFetch as _patchFetch } from '../../server/lib/patch-fetch'\n\nimport * as userland from 'VAR_USERLAND'\n\n// These are injected by the loader afterwards. This is injected as a variable\n// instead of a replacement because this could also be `undefined` instead of\n// an empty string.\ndeclare const nextConfigOutput: AppRouteRouteModuleOptions['nextConfigOutput']\n\n// We inject the nextConfigOutput here so that we can use them in the route\n// module.\n// INJECT:nextConfigOutput\n\nconst routeModule = new AppRouteRouteModule({\n definition: {\n kind: RouteKind.APP_ROUTE,\n page: 'VAR_DEFINITION_PAGE',\n pathname: 'VAR_DEFINITION_PATHNAME',\n filename: 'VAR_DEFINITION_FILENAME',\n bundlePath: 'VAR_DEFINITION_BUNDLE_PATH',\n },\n resolvedPagePath: 'VAR_RESOLVED_PAGE_PATH',\n nextConfigOutput,\n userland,\n})\n\n// Pull out the exports that we need to expose from the module. This should\n// be eliminated when we've moved the other routes to the new format. These\n// are used to hook into the route.\nconst { workAsyncStorage, workUnitAsyncStorage, serverHooks } = routeModule\n\nfunction patchFetch() {\n return _patchFetch({\n workAsyncStorage,\n workUnitAsyncStorage,\n })\n}\n\nexport {\n routeModule,\n workAsyncStorage,\n workUnitAsyncStorage,\n serverHooks,\n patchFetch,\n}\n"],"names":["AppRouteRouteModule","RouteKind","patchFetch","_patchFetch","userland","routeModule","definition","kind","APP_ROUTE","page","pathname","filename","bundlePath","resolvedPagePath","nextConfigOutput","workAsyncStorage","workUnitAsyncStorage","serverHooks"],"mappings":"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","ignoreList":[5]}
1
+ {"version":3,"sources":["turbopack:///[project]/packages/server/src/lib/workspace-types.ts","turbopack:///[project]/packages/server/src/lib/workspace-config.ts","turbopack:///[project]/packages/server/src/lib/env-file-operations.ts","turbopack:///[project]/packages/server/src/lib/sse-connection-manager.ts","turbopack:///[project]/packages/server/src/app/api/microapp-proxy-config/route.ts","turbopack:///[project]/node_modules/.pnpm/next@15.3.4_react-dom@19.1.0_react@19.1.0__react@19.1.0/node_modules/next/dist/src/build/templates/app-route.ts"],"sourcesContent":["import { homedir, cpus } from 'os';\nimport { join } from 'path';\n\n/**\n * 工作空间配置常量\n */\nexport const WORKSPACE_CONFIG = {\n // 工作空间根目录路径\n WORKSPACE_ROOT: join(homedir(), 'Documents', 'prime-workspace'),\n\n // Git 操作超时时间 (毫秒)\n GIT_TIMEOUT: 300000, // 5分钟\n\n // 最大并发克隆数量(基于CPU核心数,最小2个,最大8个)\n MAX_CONCURRENT_CLONES: Math.max(2, Math.min(cpus().length, 8)),\n\n // 重试次数限制\n MAX_RETRY_ATTEMPTS: 3,\n} as const;\n\nexport const PROJECT_CONFIG_PATH = join(homedir(), '.prime-projects.json');\n\nexport interface ProjectConfig {\n repo: string;\n mockingIntercept?: boolean;\n mockOpen?: boolean;\n envs: {\n [envName: string]: {\n host: string;\n currentProxy: string;\n proxyEnv: {\n [key: string]: string;\n };\n envFileName: string;\n proxyKey: string;\n subApps?: string[];\n };\n };\n}\n\nexport interface ProjectsConfig {\n [projectName: string]: ProjectConfig;\n}\n\n/**\n * 项目状态枚举\n */\nexport enum ProjectStatus {\n PENDING = 'pending',\n EXISTING = 'existing',\n CLONING = 'cloning',\n SUCCESS = 'success',\n FAILED = 'failed',\n RETRYING = 'retrying',\n INCOMPLETE = 'incomplete', // 不完整的项目,需要清理\n}\n\n/**\n * 项目进度信息接口\n */\nexport interface ProjectProgress {\n projectName: string;\n status: ProjectStatus;\n progress: number; // 0-100\n message: string;\n error?: string;\n startTime?: number;\n endTime?: number;\n}\n\n/**\n * 初始化结果接口\n */\nexport interface InitializationResult {\n success: boolean;\n totalProjects: number;\n successCount: number;\n failedCount: number;\n failedProjects: string[];\n workspacePath: string;\n duration: number;\n}\n\n/**\n * Git 操作配置接口\n */\nexport interface GitOperationConfig {\n repoUrl: string;\n projectName?: string;\n targetDir: string;\n timeout: number;\n onProgress?: (progress: number, message: string) => void;\n onError?: (error: string) => void;\n onLog?: (projectName: string, logType: 'stdout' | 'stderr' | 'progress', content: string) => void;\n}\n","import { readFile, writeFile, access, constants, rename } from 'fs/promises';\nimport type { ProjectsConfig } from './workspace-types';\nimport { PROJECT_CONFIG_PATH } from './workspace-types';\n\n/**\n * 读取并解析项目配置文件\n * @returns {Promise<ProjectsConfig>} 项目配置对象\n */\nexport async function getProjectsConfig(): Promise<ProjectsConfig> {\n try {\n await access(PROJECT_CONFIG_PATH, constants.F_OK);\n } catch {\n // 文件不存在,创建并写入空对象\n await saveProjectsConfig({});\n return {};\n }\n\n try {\n const configContent = await readFile(PROJECT_CONFIG_PATH, 'utf-8');\n // 如果文件为空,返回空对象\n if (!configContent) {\n return {};\n }\n return JSON.parse(configContent);\n } catch (error) {\n console.error('读取或解析项目配置文件失败:', error);\n // 在出错时返回空对象或抛出错误,这里选择抛出以通知调用者\n throw new Error('无法加载项目配置');\n }\n}\n\n/**\n * 将项目配置写入文件(原子操作)\n * @param {ProjectsConfig} config - 要保存的配置对象\n */\nexport async function saveProjectsConfig(config: ProjectsConfig): Promise<void> {\n const tempConfigPath = PROJECT_CONFIG_PATH + '.tmp';\n try {\n await writeFile(tempConfigPath, JSON.stringify(config, null, 2), 'utf-8');\n await rename(tempConfigPath, PROJECT_CONFIG_PATH);\n } catch (error) {\n console.error('保存项目配置文件失败:', error);\n throw new Error('无法保存项目配置');\n }\n}\n\n/**\n * 项目状态枚举\n */\nexport enum ProjectStatus {\n PENDING = 'pending',\n EXISTING = 'existing',\n CLONING = 'cloning',\n SUCCESS = 'success',\n FAILED = 'failed',\n RETRYING = 'retrying',\n}\n\n/**\n * 项目进度信息接口\n */\nexport interface ProjectProgress {\n projectName: string;\n status: ProjectStatus;\n progress: number; // 0-100\n message: string;\n error?: string;\n startTime?: number;\n endTime?: number;\n}\n\n/**\n * 初始化结果接口\n */\nexport interface InitializationResult {\n success: boolean;\n totalProjects: number;\n successCount: number;\n failedCount: number;\n failedProjects: string[];\n workspacePath: string;\n duration: number;\n}\n\n/**\n * Git 操作配置接口\n */\nexport interface GitOperationConfig {\n repoUrl: string;\n projectName?: string;\n targetDir: string;\n timeout: number;\n onProgress?: (progress: number, message: string) => void;\n onError?: (error: string) => void;\n}\n","import { promises as fs } from 'fs';\nimport { join } from 'path';\nimport type { ProjectConfig } from './workspace-types';\n\n/**\n * 为指定项目创建环境文件,如果文件已存在则跳过。\n * @param projectName - 项目名称\n * @param projectConfig - 项目配置\n * @param workspacePath - 工作空间根路径\n * @returns 一个包含已创建和已跳过文件列表的对象\n */\nexport async function createProjectEnvFiles(\n projectName: string,\n projectConfig: ProjectConfig,\n workspacePath: string,\n): Promise<{\n success: boolean;\n createdFiles: string[];\n skippedFiles: string[];\n error?: string;\n}> {\n const projectPath = join(workspacePath, projectName);\n const createdFiles: string[] = [];\n const skippedFiles: string[] = [];\n\n try {\n const envs = projectConfig.envs || {};\n for (const envName in envs) {\n const envConfig = envs[envName];\n if (envConfig.envFileName && envConfig.proxyKey) {\n const filePath = join(projectPath, envConfig.envFileName);\n \n try {\n await fs.access(filePath);\n // 文件已存在,跳过\n skippedFiles.push(envConfig.envFileName);\n } catch {\n // 文件不存在,创建它\n const currentProxyUrl = envConfig.proxyEnv[envConfig.currentProxy] || '';\n const content = `${envConfig.proxyKey}=${currentProxyUrl}`;\n await fs.writeFile(filePath, content, 'utf-8');\n createdFiles.push(envConfig.envFileName);\n }\n }\n }\n return { success: true, createdFiles, skippedFiles };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n console.error(`在为项目 ${projectName} 创建环境文件时出错:`, error);\n return { success: false, createdFiles, skippedFiles, error: errorMessage };\n }\n}\n\n/**\n * 为指定项目创建.mock文件夹\n * @param projectName - 项目名称\n * @param workspacePath - 工作空间根路径\n * @returns 创建结果\n */\nexport async function createProjectMockDirectories(\n projectName: string,\n workspacePath: string,\n): Promise<{\n success: boolean;\n createdDirectories: string[];\n skippedDirectories: string[];\n error?: string;\n}> {\n const projectPath = join(workspacePath, projectName);\n const mockPath = join(projectPath, '.mock');\n const createdDirectories: string[] = [];\n const skippedDirectories: string[] = [];\n\n try {\n // 创建 .mock 文件夹\n try {\n await fs.access(mockPath);\n skippedDirectories.push('.mock');\n } catch {\n await fs.mkdir(mockPath, { recursive: true });\n createdDirectories.push('.mock');\n }\n\n return { success: true, createdDirectories, skippedDirectories };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n console.error(`在为项目 ${projectName} 创建Mock目录时出错:`, error);\n return { success: false, createdDirectories, skippedDirectories, error: errorMessage };\n }\n}\n\n/**\n * 更新指定项目环境的环境文件内容\n * @param projectName - 项目名称\n * @param envName - 环境名称\n * @param projectConfig - 项目配置\n * @param workspacePath - 工作空间根路径\n * @returns 更新结果\n */\nexport async function updateProjectEnvFile(\n projectName: string,\n envName: string,\n projectConfig: ProjectConfig,\n workspacePath: string,\n): Promise<{\n success: boolean;\n updatedFile?: string;\n error?: string;\n}> {\n try {\n const envConfig = projectConfig.envs[envName];\n \n if (!envConfig) {\n return {\n success: false,\n error: `环境 ${envName} 不存在于项目 ${projectName} 配置中`,\n };\n }\n\n if (!envConfig.envFileName || !envConfig.proxyKey) {\n return {\n success: false,\n error: `环境 ${envName} 缺少 envFileName 或 proxyKey 配置`,\n };\n }\n\n const projectPath = join(workspacePath, projectName);\n const filePath = join(projectPath, envConfig.envFileName);\n \n // 获取当前代理URL\n const currentProxyUrl = envConfig.proxyEnv[envConfig.currentProxy] || envConfig.currentProxy;\n const content = `${envConfig.proxyKey}=${currentProxyUrl}`;\n \n // 写入文件内容\n await fs.writeFile(filePath, content, 'utf-8');\n \n console.log(`已更新环境文件: ${filePath}, 内容: ${content}`);\n \n return {\n success: true,\n updatedFile: envConfig.envFileName,\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n console.error(`更新项目 ${projectName} 环境 ${envName} 的环境文件时出错:`, error);\n return {\n success: false,\n error: errorMessage,\n };\n }\n} ","/**\n * SSE连接管理器\n * 用于管理MicroApp代理配置变化推送的SSE连接\n */\n\nexport interface ConnectionInfo {\n controller: ReadableStreamDefaultController<Uint8Array>;\n timestamp: number;\n clientId: string;\n envName: string;\n}\n\nexport interface ConfigChangeMessage {\n type: 'microapp-proxy-config-change';\n envName: string;\n timestamp: number;\n changes: {\n before: string[];\n after: string[];\n added: string[];\n removed: string[];\n };\n projectName: string;\n message: string;\n}\n\nexport interface HeartbeatMessage {\n type: 'heartbeat';\n timestamp: number;\n}\n\ntype SSEMessage = ConfigChangeMessage | HeartbeatMessage;\n\n/**\n * SSE连接管理器类\n */\nclass SSEConnectionManager {\n private connections = new Map<string, Set<ConnectionInfo>>();\n private encoder = new TextEncoder();\n private heartbeatInterval: NodeJS.Timeout | null = null;\n private readonly HEARTBEAT_INTERVAL = 30000; // 30秒心跳\n private readonly CONNECTION_TIMEOUT = 120000; // 2分钟连接超时\n private readonly MAX_CONNECTIONS_PER_ENV = 10; // 每个环境最多10个连接\n\n constructor() {\n this.startHeartbeat();\n }\n \n /**\n * 清理所有连接(用于内存清理)\n */\n clearAllConnections(): void {\n console.log('[SSE] 开始清理所有连接...');\n const totalConnections = Array.from(this.connections.values()).reduce((sum, set) => sum + set.size, 0);\n \n this.connections.forEach((connections) => {\n connections.forEach(conn => {\n try {\n conn.controller.close();\n } catch {\n // 忽略关闭错误\n }\n });\n });\n \n this.connections.clear();\n console.log(`[SSE] 已清理 ${totalConnections} 个连接`);\n }\n\n /**\n * 添加新的SSE连接\n */\n addConnection(envName: string, controller: ReadableStreamDefaultController<Uint8Array>, clientId: string): void {\n console.log(`[SSE] 添加连接: 环境=${envName}, 客户端=${clientId}`);\n \n if (!this.connections.has(envName)) {\n this.connections.set(envName, new Set());\n }\n \n const connections = this.connections.get(envName)!;\n \n // 检查连接数限制\n if (connections.size >= this.MAX_CONNECTIONS_PER_ENV) {\n console.warn(`[SSE] 环境 ${envName} 连接数已达上限,清理最旧的连接`);\n // 找到最旧的连接并关闭\n const oldestConn = Array.from(connections).sort((a, b) => a.timestamp - b.timestamp)[0];\n if (oldestConn) {\n try {\n oldestConn.controller.close();\n } catch {\n // 忽略关闭错误\n }\n connections.delete(oldestConn);\n }\n }\n \n const connectionInfo: ConnectionInfo = {\n controller,\n timestamp: Date.now(),\n clientId,\n envName,\n };\n \n connections.add(connectionInfo);\n \n // 发送连接确认消息\n this.sendToConnection(connectionInfo, {\n type: 'heartbeat',\n timestamp: Date.now(),\n });\n\n console.log(`[SSE] 当前连接统计: ${this.getConnectionStats()}`);\n }\n\n /**\n * 移除SSE连接\n */\n removeConnection(envName: string, clientId: string): void {\n console.log(`[SSE] 移除连接: 环境=${envName}, 客户端=${clientId}`);\n \n const connections = this.connections.get(envName);\n if (connections) {\n const connectionToRemove = Array.from(connections).find(conn => conn.clientId === clientId);\n if (connectionToRemove) {\n connections.delete(connectionToRemove);\n \n // 如果该环境没有连接了,删除整个entry\n if (connections.size === 0) {\n this.connections.delete(envName);\n }\n }\n }\n\n console.log(`[SSE] 当前连接统计: ${this.getConnectionStats()}`);\n }\n\n /**\n * 向指定环境广播消息\n */\n broadcastToEnv(envName: string, message: ConfigChangeMessage): void {\n console.log(`[SSE] 向环境 ${envName} 广播配置变化消息`);\n \n const connections = this.connections.get(envName);\n if (!connections || connections.size === 0) {\n console.log(`[SSE] 环境 ${envName} 没有活跃连接,跳过广播`);\n return;\n }\n\n const deadConnections: ConnectionInfo[] = [];\n \n connections.forEach(connectionInfo => {\n try {\n this.sendToConnection(connectionInfo, message);\n } catch (error) {\n console.error(`[SSE] 发送消息到客户端 ${connectionInfo.clientId} 失败:`, error);\n deadConnections.push(connectionInfo);\n }\n });\n\n // 清理无效连接\n deadConnections.forEach(conn => {\n connections.delete(conn);\n console.log(`[SSE] 清理无效连接: 客户端=${conn.clientId}`);\n });\n\n console.log(`[SSE] 成功发送给 ${connections.size} 个连接`);\n }\n\n /**\n * 向单个连接发送消息\n */\n private sendToConnection(connectionInfo: ConnectionInfo, message: SSEMessage): void {\n const data = `data: ${JSON.stringify(message)}\\n\\n`;\n const encodedData = this.encoder.encode(data);\n \n try {\n connectionInfo.controller.enqueue(encodedData);\n connectionInfo.timestamp = Date.now(); // 更新最后活跃时间\n } catch (error) {\n console.error(`[SSE] 发送消息失败:`, error);\n throw error;\n }\n }\n\n /**\n * 启动心跳机制\n */\n private startHeartbeat(): void {\n this.heartbeatInterval = setInterval(() => {\n this.sendHeartbeat();\n this.cleanupStaleConnections();\n }, this.HEARTBEAT_INTERVAL);\n \n console.log(`[SSE] 心跳机制已启动,间隔: ${this.HEARTBEAT_INTERVAL}ms`);\n }\n\n /**\n * 发送心跳消息\n */\n private sendHeartbeat(): void {\n const heartbeatMessage: HeartbeatMessage = {\n type: 'heartbeat',\n timestamp: Date.now(),\n };\n\n let totalConnections = 0;\n this.connections.forEach((connections) => {\n totalConnections += connections.size;\n const deadConnections: ConnectionInfo[] = [];\n \n connections.forEach(connectionInfo => {\n try {\n this.sendToConnection(connectionInfo, heartbeatMessage);\n } catch {\n deadConnections.push(connectionInfo);\n }\n });\n\n // 清理无效连接\n deadConnections.forEach(conn => connections.delete(conn));\n });\n\n if (totalConnections > 0) {\n console.log(`[SSE] 心跳发送完成,活跃连接数: ${totalConnections}`);\n }\n }\n\n /**\n * 清理过期连接\n */\n private cleanupStaleConnections(): void {\n const now = Date.now();\n let cleanedCount = 0;\n\n this.connections.forEach((connections, envName) => {\n const staleConnections: ConnectionInfo[] = [];\n \n connections.forEach(connectionInfo => {\n if (now - connectionInfo.timestamp > this.CONNECTION_TIMEOUT) {\n staleConnections.push(connectionInfo);\n }\n });\n\n staleConnections.forEach(conn => {\n connections.delete(conn);\n cleanedCount++;\n console.log(`[SSE] 清理过期连接: 环境=${envName}, 客户端=${conn.clientId}`);\n });\n\n // 如果该环境没有连接了,删除整个entry\n if (connections.size === 0) {\n this.connections.delete(envName);\n }\n });\n\n if (cleanedCount > 0) {\n console.log(`[SSE] 连接清理完成,清理数量: ${cleanedCount}`);\n }\n }\n\n /**\n * 获取连接统计信息\n */\n getConnectionStats(): string {\n const stats: string[] = [];\n let totalConnections = 0;\n \n this.connections.forEach((connections, envName) => {\n const count = connections.size;\n totalConnections += count;\n stats.push(`${envName}: ${count}`);\n });\n\n return `总计 ${totalConnections} 个连接 (${stats.join(', ')})`;\n }\n\n /**\n * 停止连接管理器\n */\n stop(): void {\n if (this.heartbeatInterval) {\n clearInterval(this.heartbeatInterval);\n this.heartbeatInterval = null;\n console.log(`[SSE] 心跳机制已停止`);\n }\n\n // 关闭所有连接\n this.connections.forEach((connections) => {\n connections.forEach(connectionInfo => {\n try {\n connectionInfo.controller.close();\n } catch (error) {\n console.error(`[SSE] 关闭连接失败:`, error);\n }\n });\n });\n\n this.connections.clear();\n console.log(`[SSE] 所有连接已关闭`);\n }\n}\n\n// 创建全局实例\nexport const sseConnectionManager = new SSEConnectionManager();\n\n// 在进程退出时清理资源\nprocess.on('SIGINT', () => {\n console.log('[SSE] 收到退出信号,正在清理连接...');\n sseConnectionManager.stop();\n process.exit(0);\n});\n\nprocess.on('SIGTERM', () => {\n console.log('[SSE] 收到终止信号,正在清理连接...');\n sseConnectionManager.stop();\n process.exit(0);\n}); ","import { NextRequest, NextResponse } from 'next/server';\nimport {\n getProjectsConfig,\n saveProjectsConfig,\n} from '@/lib/workspace-config';\nimport { updateProjectEnvFile } from '@/lib/env-file-operations';\nimport { WORKSPACE_CONFIG } from '@/lib/workspace-types';\nimport { sseConnectionManager } from '@/lib/sse-connection-manager';\nimport type { ConfigChangeMessage } from '@/lib/sse-connection-manager';\n\nexport async function PUT(request: NextRequest) {\n try {\n // 解析请求体\n const { envName, projectName, selectedItems } = await request.json();\n\n // 参数验证\n if (!envName || !projectName || !Array.isArray(selectedItems)) {\n return NextResponse.json(\n {\n success: false,\n error: '缺少必要参数:envName, projectName, selectedItems',\n },\n { status: 400 },\n );\n }\n\n // 读取当前配置文件\n const projectsConfig = await getProjectsConfig();\n\n // 检查项目是否存在\n if (!projectsConfig[projectName]) {\n return NextResponse.json(\n { success: false, error: `项目 ${projectName} 不存在` },\n { status: 404 },\n );\n }\n\n // 检查环境是否存在\n if (!projectsConfig[projectName].envs[envName]) {\n return NextResponse.json(\n {\n success: false,\n error: `项目 ${projectName} 中不存在环境 ${envName}`,\n },\n { status: 404 },\n );\n }\n\n // 获取当前环境的host\n const currentHost = projectsConfig[projectName].envs[envName].host;\n\n // 保存更新前的配置用于变化检测\n const beforeSubApps = projectsConfig[projectName].envs[envName].subApps || [];\n\n // 更新 subApps 配置\n projectsConfig[projectName].envs[envName].subApps = selectedItems;\n\n // 记录更新的项目和环境,用于后续更新环境文件\n const updatedProjects: Array<{projectName: string, envName: string}> = [];\n\n // 将选中的环境的代理改为当前环境的host\n selectedItems.forEach((selectedEnvName: string) => {\n // 遍历所有项目,找到包含该环境的项目并更新其代理配置\n Object.keys(projectsConfig).forEach(projName => {\n if (projectsConfig[projName].envs[selectedEnvName]) {\n // 找到匹配的环境,更新其代理配置\n const targetEnv = projectsConfig[projName].envs[selectedEnvName];\n \n // 将选中环境的代理设置为custom,并指向当前环境的host\n targetEnv.currentProxy = 'custom';\n targetEnv.proxyEnv.custom = currentHost;\n \n // 记录需要更新环境文件的项目和环境\n updatedProjects.push({\n projectName: projName,\n envName: selectedEnvName\n });\n }\n });\n });\n\n // 保存更新后的配置\n await saveProjectsConfig(projectsConfig);\n\n // 检测配置变化并推送通知\n const afterSubApps = selectedItems;\n const hasChanges = JSON.stringify(beforeSubApps.sort()) !== JSON.stringify(afterSubApps.sort());\n \n if (hasChanges) {\n console.log(`[配置变化] 环境 ${envName} 的 MicroApp 代理配置发生变化`);\n console.log(`[配置变化] 变化前: ${JSON.stringify(beforeSubApps)}`);\n console.log(`[配置变化] 变化后: ${JSON.stringify(afterSubApps)}`);\n\n // 计算具体变化\n const beforeSet = new Set(beforeSubApps);\n const afterSet = new Set(afterSubApps);\n const added = afterSubApps.filter(item => !beforeSet.has(item));\n const removed = beforeSubApps.filter(item => !afterSet.has(item));\n\n // 构建推送消息\n const changeMessage: ConfigChangeMessage = {\n type: 'microapp-proxy-config-change',\n envName,\n timestamp: Date.now(),\n changes: {\n before: beforeSubApps,\n after: afterSubApps,\n added,\n removed,\n },\n projectName,\n message: `环境 ${envName} 的 MicroApp 代理配置已更新`\n };\n\n // 推送给相关环境的连接\n try {\n sseConnectionManager.broadcastToEnv(envName, changeMessage);\n console.log(`[推送成功] 已向环境 ${envName} 推送配置变化通知`);\n } catch (pushError) {\n console.error(`[推送失败] 向环境 ${envName} 推送通知失败:`, pushError);\n // 推送失败不影响主要功能,只记录错误\n }\n } else {\n console.log(`[配置未变化] 环境 ${envName} 的 MicroApp 代理配置无变化,跳过推送`);\n }\n\n // 更新相关项目的环境文件\n const envFileUpdates = await Promise.allSettled(\n updatedProjects.map(async ({ projectName: projName, envName: envNameToUpdate }) => {\n return updateProjectEnvFile(\n projName,\n envNameToUpdate,\n projectsConfig[projName],\n WORKSPACE_CONFIG.WORKSPACE_ROOT,\n );\n })\n );\n\n // 检查环境文件更新结果\n const failedUpdates = envFileUpdates\n .map((result, index) => ({ result, project: updatedProjects[index] }))\n .filter(({ result }) => result.status === 'rejected' || (result.status === 'fulfilled' && !result.value.success));\n\n // 构建响应消息\n let message = 'MicroApp代理配置保存成功,已更新选中环境的代理地址';\n if (failedUpdates.length > 0) {\n message += `,但有 ${failedUpdates.length} 个环境文件更新失败`;\n }\n\n return NextResponse.json({\n success: true,\n message,\n data: {\n projectName,\n envName,\n subApps: selectedItems,\n currentHost,\n updatedProjects: updatedProjects.length,\n envFileUpdateFailures: failedUpdates.length,\n },\n });\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : '保存MicroApp代理配置失败';\n console.error('保存MicroApp代理配置失败:', error);\n return NextResponse.json(\n { success: false, error: errorMessage },\n { status: 500 },\n );\n }\n} ","import {\n AppRouteRouteModule,\n type AppRouteRouteModuleOptions,\n} from '../../server/route-modules/app-route/module.compiled'\nimport { RouteKind } from '../../server/route-kind'\nimport { patchFetch as _patchFetch } from '../../server/lib/patch-fetch'\n\nimport * as userland from 'VAR_USERLAND'\n\n// These are injected by the loader afterwards. This is injected as a variable\n// instead of a replacement because this could also be `undefined` instead of\n// an empty string.\ndeclare const nextConfigOutput: AppRouteRouteModuleOptions['nextConfigOutput']\n\n// We inject the nextConfigOutput here so that we can use them in the route\n// module.\n// INJECT:nextConfigOutput\n\nconst routeModule = new AppRouteRouteModule({\n definition: {\n kind: RouteKind.APP_ROUTE,\n page: 'VAR_DEFINITION_PAGE',\n pathname: 'VAR_DEFINITION_PATHNAME',\n filename: 'VAR_DEFINITION_FILENAME',\n bundlePath: 'VAR_DEFINITION_BUNDLE_PATH',\n },\n resolvedPagePath: 'VAR_RESOLVED_PAGE_PATH',\n nextConfigOutput,\n userland,\n})\n\n// Pull out the exports that we need to expose from the module. This should\n// be eliminated when we've moved the other routes to the new format. These\n// are used to hook into the route.\nconst { workAsyncStorage, workUnitAsyncStorage, serverHooks } = routeModule\n\nfunction patchFetch() {\n return _patchFetch({\n workAsyncStorage,\n workUnitAsyncStorage,\n })\n}\n\nexport {\n routeModule,\n workAsyncStorage,\n workUnitAsyncStorage,\n serverHooks,\n patchFetch,\n}\n"],"names":["AppRouteRouteModule","RouteKind","patchFetch","_patchFetch","userland","routeModule","definition","kind","APP_ROUTE","page","pathname","filename","bundlePath","resolvedPagePath","nextConfigOutput","workAsyncStorage","workUnitAsyncStorage","serverHooks"],"mappings":"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","ignoreList":[5]}
@@ -1,19 +1,19 @@
1
- module.exports={88941:e=>{"use strict";var{g:t,__dirname:r}=e;{e.s({PROJECT_CONFIG_PATH:()=>r,ProjectStatus:()=>o,WORKSPACE_CONFIG:()=>t});var s=e.i(13442),n=e.i(30331);let t={WORKSPACE_ROOT:(0,n.join)((0,s.homedir)(),"Documents","prime-workspace"),GIT_TIMEOUT:3e5,MAX_CONCURRENT_CLONES:Math.max(2,Math.min((0,s.cpus)().length,8)),MAX_RETRY_ATTEMPTS:3},r=(0,n.join)((0,s.homedir)(),".prime-projects.json");var o=function(e){return e.PENDING="pending",e.EXISTING="existing",e.CLONING="cloning",e.SUCCESS="success",e.FAILED="failed",e.RETRYING="retrying",e.INCOMPLETE="incomplete",e}({})}},30331:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("path",()=>require("path"))},29549:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/compiled/next-server/app-page-turbo.runtime.prod.js",()=>require("next/dist/compiled/next-server/app-page-turbo.runtime.prod.js"))},83943:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/server/app-render/work-unit-async-storage.external.js",()=>require("next/dist/server/app-render/work-unit-async-storage.external.js"))},86103:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/server/app-render/work-async-storage.external.js",()=>require("next/dist/server/app-render/work-async-storage.external.js"))},23430:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/compiled/@opentelemetry/api",()=>require("next/dist/compiled/@opentelemetry/api"))},52670:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/compiled/next-server/app-route-turbo.runtime.prod.js",()=>require("next/dist/compiled/next-server/app-route-turbo.runtime.prod.js"))},13442:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("os",()=>require("os"))},9892:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("fs/promises",()=>require("fs/promises"))},35692:e=>{"use strict";var{g:t,__dirname:r}=e;e.s({ProjectStatus:()=>c,getProjectsConfig:()=>o,saveProjectsConfig:()=>i});var s=e.i(9892),n=e.i(88941);async function o(){try{await (0,s.access)(n.PROJECT_CONFIG_PATH,s.constants.F_OK)}catch{return await i({}),{}}try{let e=await (0,s.readFile)(n.PROJECT_CONFIG_PATH,"utf-8");if(!e)return{};return JSON.parse(e)}catch(e){throw console.error("读取或解析项目配置文件失败:",e),Error("无法加载项目配置")}}async function i(e){let t=n.PROJECT_CONFIG_PATH+".tmp";try{await (0,s.writeFile)(t,JSON.stringify(e,null,2),"utf-8"),await (0,s.rename)(t,n.PROJECT_CONFIG_PATH)}catch(e){throw console.error("保存项目配置文件失败:",e),Error("无法保存项目配置")}}var c=function(e){return e.PENDING="pending",e.EXISTING="existing",e.CLONING="cloning",e.SUCCESS="success",e.FAILED="failed",e.RETRYING="retrying",e}({})},83886:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("fs",()=>require("fs"))},87485:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("child_process",()=>require("child_process"))},77159:e=>{"use strict";var{g:t,__dirname:r}=e;e.s({checkMultipleProjectsExist:()=>h,checkProjectExists:()=>g,checkProjectStatus:()=>p,cleanupIncompleteProject:()=>m,cloneRepository:()=>a,createWorkspaceDirectory:()=>c,isValidGitUrl:()=>u,sanitizeProjectName:()=>l});var s=e.i(87485),n=e.i(83886),o=e.i(30331),i=e.i(88941);async function c(e){try{return(0,n.existsSync)(e)||(0,n.mkdirSync)(e,{recursive:!0}),{success:!0}}catch(e){return{success:!1,error:`创建工作空间目录失败: ${e instanceof Error?e.message:String(e)}`}}}async function a(e){let{repoUrl:t,targetDir:r,timeout:n,onProgress:c,onError:a,onLog:u}=e,l=(0,o.basename)(t,".git");return new Promise(e=>{let p=Date.now(),g={projectName:l,status:i.ProjectStatus.CLONING,progress:0,message:"开始克隆项目...",startTime:p};c?.(0,"开始克隆项目...");let d=(0,o.join)(r,l),m=["clone","--progress","--verbose",t,d];u?.(l,"stdout",`[开始克隆] 执行命令: git ${m.join(" ")}
2
- `),u?.(l,"stdout",`[目标目录] ${d}
3
- `),u?.(l,"stdout",`[仓库地址] ${t}
4
- `);let h=(0,s.spawn)("git",m,{cwd:r,stdio:["pipe","pipe","pipe"]}),f="",y=0;u?.(l,"stdout",`[进程启动] Git 进程 PID: ${h.pid}
5
- `);let j=setTimeout(()=>{u?.(l,"stderr",`[超时警告] Git 进程运行超过 ${n/1e3}s,即将终止
6
- `),h.kill("SIGTERM");let t={...g,status:i.ProjectStatus.FAILED,progress:y,message:"克隆操作超时",error:`克隆操作超时 (${n/1e3}s)`,endTime:Date.now()};u?.(l,"stderr",`[进程终止] 克隆操作超时
7
- `),a?.(`${l}: 克隆操作超时`),e(t)},n);h.stdout?.on("data",e=>{let t=e.toString();u?.(l,"stdout",t),y=Math.min(y+10,80),c?.(y,"正在克隆代码...")}),h.stderr?.on("data",e=>{let t=e.toString();f+=t,t.includes("remote:")||t.includes("Receiving objects")||t.includes("Resolving deltas")||t.includes("Counting objects")||t.includes("Enumerating objects")||t.includes("Compressing objects")||t.includes("upload-pack")||t.includes("POST git-")||t.includes("正克隆到")||t.includes("Cloning into")?u?.(l,"progress",t):u?.(l,"stderr",t),t.includes("Receiving objects")||t.includes("Resolving deltas")?(y=Math.min(y+5,90),c?.(y,"正在接收对象...")):t.includes("Counting objects")?(y=Math.min(y+2,70),c?.(y,"正在计算对象...")):t.includes("Enumerating objects")&&(y=Math.min(y+2,50),c?.(y,"正在枚举对象..."))}),h.on("close",t=>{clearTimeout(j);let r=Date.now(),s=r-p;if(u?.(l,"stdout",`[进程结束] Git 进程退出,代码: ${t},耗时: ${Math.round(s/1e3)}s
8
- `),0===t){u?.(l,"stdout",`[克隆成功] 项目已成功克隆到: ${d}
9
- `);let t={...g,status:i.ProjectStatus.SUCCESS,progress:100,message:"克隆完成",endTime:r};c?.(100,"克隆完成"),e(t)}else{let s=function(e){let t=e.toLowerCase();return t.includes("authentication failed")||t.includes("access denied")?"身份验证失败,请检查仓库访问权限":t.includes("repository not found")||t.includes("not found")?"仓库不存在或无法访问":t.includes("network")||t.includes("connection")?"网络连接问题,请检查网络设置":t.includes("timeout")?"网络超时,请重试":t.includes("permission denied")?"权限被拒绝,请检查文件系统权限":t.includes("already exists")?"目标目录已存在":e.slice(0,200)+(e.length>200?"...":"")}(f);u?.(l,"stderr",`[克隆失败] 错误信息: ${s}
10
- `),u?.(l,"stderr",`[完整错误输出]
1
+ module.exports={88941:e=>{"use strict";var{g:t,__dirname:r}=e;{e.s({PROJECT_CONFIG_PATH:()=>r,ProjectStatus:()=>o,WORKSPACE_CONFIG:()=>t});var s=e.i(13442),n=e.i(30331);let t={WORKSPACE_ROOT:(0,n.join)((0,s.homedir)(),"Documents","prime-workspace"),GIT_TIMEOUT:3e5,MAX_CONCURRENT_CLONES:Math.max(2,Math.min((0,s.cpus)().length,8)),MAX_RETRY_ATTEMPTS:3},r=(0,n.join)((0,s.homedir)(),".prime-projects.json");var o=function(e){return e.PENDING="pending",e.EXISTING="existing",e.CLONING="cloning",e.SUCCESS="success",e.FAILED="failed",e.RETRYING="retrying",e.INCOMPLETE="incomplete",e}({})}},30331:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("path",()=>require("path"))},29549:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/compiled/next-server/app-page-turbo.runtime.prod.js",()=>require("next/dist/compiled/next-server/app-page-turbo.runtime.prod.js"))},83943:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/server/app-render/work-unit-async-storage.external.js",()=>require("next/dist/server/app-render/work-unit-async-storage.external.js"))},86103:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/server/app-render/work-async-storage.external.js",()=>require("next/dist/server/app-render/work-async-storage.external.js"))},23430:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/compiled/@opentelemetry/api",()=>require("next/dist/compiled/@opentelemetry/api"))},52670:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("next/dist/compiled/next-server/app-route-turbo.runtime.prod.js",()=>require("next/dist/compiled/next-server/app-route-turbo.runtime.prod.js"))},13442:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("os",()=>require("os"))},9892:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("fs/promises",()=>require("fs/promises"))},35692:e=>{"use strict";var{g:t,__dirname:r}=e;e.s({ProjectStatus:()=>c,getProjectsConfig:()=>o,saveProjectsConfig:()=>i});var s=e.i(9892),n=e.i(88941);async function o(){try{await (0,s.access)(n.PROJECT_CONFIG_PATH,s.constants.F_OK)}catch{return await i({}),{}}try{let e=await (0,s.readFile)(n.PROJECT_CONFIG_PATH,"utf-8");if(!e)return{};return JSON.parse(e)}catch(e){throw console.error("读取或解析项目配置文件失败:",e),Error("无法加载项目配置")}}async function i(e){let t=n.PROJECT_CONFIG_PATH+".tmp";try{await (0,s.writeFile)(t,JSON.stringify(e,null,2),"utf-8"),await (0,s.rename)(t,n.PROJECT_CONFIG_PATH)}catch(e){throw console.error("保存项目配置文件失败:",e),Error("无法保存项目配置")}}var c=function(e){return e.PENDING="pending",e.EXISTING="existing",e.CLONING="cloning",e.SUCCESS="success",e.FAILED="failed",e.RETRYING="retrying",e}({})},83886:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("fs",()=>require("fs"))},87485:function(e){var{g:t,__dirname:r,m:s,e:n}=e;s.exports=e.x("child_process",()=>require("child_process"))},77159:e=>{"use strict";var{g:t,__dirname:r}=e;e.s({checkMultipleProjectsExist:()=>h,checkProjectExists:()=>g,checkProjectStatus:()=>p,cleanupIncompleteProject:()=>m,cloneRepository:()=>a,createWorkspaceDirectory:()=>c,isValidGitUrl:()=>u,sanitizeProjectName:()=>l});var s=e.i(87485),n=e.i(83886),o=e.i(30331),i=e.i(88941);async function c(e){try{return(0,n.existsSync)(e)||(0,n.mkdirSync)(e,{recursive:!0}),{success:!0}}catch(e){return{success:!1,error:`创建工作空间目录失败: ${e instanceof Error?e.message:String(e)}`}}}async function a(e){let{repoUrl:t,projectName:r,targetDir:n,timeout:c,onProgress:a,onError:u,onLog:l}=e,p=r?.trim()||(0,o.basename)(t,".git");return new Promise(e=>{let r=Date.now(),g={projectName:p,status:i.ProjectStatus.CLONING,progress:0,message:"开始克隆项目...",startTime:r};a?.(0,"开始克隆项目...");let d=(0,o.join)(n,p),m=["clone","--progress","--verbose",t,d];l?.(p,"stdout",`[开始克隆] 执行命令: git ${m.join(" ")}
2
+ `),l?.(p,"stdout",`[目标目录] ${d}
3
+ `),l?.(p,"stdout",`[仓库地址] ${t}
4
+ `);let h=(0,s.spawn)("git",m,{cwd:n,stdio:["pipe","pipe","pipe"]}),f="",y=0;l?.(p,"stdout",`[进程启动] Git 进程 PID: ${h.pid}
5
+ `);let j=setTimeout(()=>{l?.(p,"stderr",`[超时警告] Git 进程运行超过 ${c/1e3}s,即将终止
6
+ `),h.kill("SIGTERM");let t={...g,status:i.ProjectStatus.FAILED,progress:y,message:"克隆操作超时",error:`克隆操作超时 (${c/1e3}s)`,endTime:Date.now()};l?.(p,"stderr",`[进程终止] 克隆操作超时
7
+ `),u?.(`${p}: 克隆操作超时`),e(t)},c);h.stdout?.on("data",e=>{let t=e.toString();l?.(p,"stdout",t),y=Math.min(y+10,80),a?.(y,"正在克隆代码...")}),h.stderr?.on("data",e=>{let t=e.toString();f+=t,t.includes("remote:")||t.includes("Receiving objects")||t.includes("Resolving deltas")||t.includes("Counting objects")||t.includes("Enumerating objects")||t.includes("Compressing objects")||t.includes("upload-pack")||t.includes("POST git-")||t.includes("正克隆到")||t.includes("Cloning into")?l?.(p,"progress",t):l?.(p,"stderr",t),t.includes("Receiving objects")||t.includes("Resolving deltas")?(y=Math.min(y+5,90),a?.(y,"正在接收对象...")):t.includes("Counting objects")?(y=Math.min(y+2,70),a?.(y,"正在计算对象...")):t.includes("Enumerating objects")&&(y=Math.min(y+2,50),a?.(y,"正在枚举对象..."))}),h.on("close",t=>{clearTimeout(j);let s=Date.now(),n=s-r;if(l?.(p,"stdout",`[进程结束] Git 进程退出,代码: ${t},耗时: ${Math.round(n/1e3)}s
8
+ `),0===t){l?.(p,"stdout",`[克隆成功] 项目已成功克隆到: ${d}
9
+ `);let t={...g,status:i.ProjectStatus.SUCCESS,progress:100,message:"克隆完成",endTime:s};a?.(100,"克隆完成"),e(t)}else{let r=function(e){let t=e.toLowerCase();return t.includes("authentication failed")||t.includes("access denied")?"身份验证失败,请检查仓库访问权限":t.includes("repository not found")||t.includes("not found")?"仓库不存在或无法访问":t.includes("network")||t.includes("connection")?"网络连接问题,请检查网络设置":t.includes("timeout")?"网络超时,请重试":t.includes("permission denied")?"权限被拒绝,请检查文件系统权限":t.includes("already exists")?"目标目录已存在":e.slice(0,200)+(e.length>200?"...":"")}(f);l?.(p,"stderr",`[克隆失败] 错误信息: ${r}
10
+ `),l?.(p,"stderr",`[完整错误输出]
11
11
  ${f}
12
- `);let n={...g,status:i.ProjectStatus.FAILED,progress:y,message:"克隆失败",error:s||`Git 进程退出,代码: ${t}`,endTime:r};a?.(`${l}: ${n.error}`),e(n)}}),h.on("error",t=>{clearTimeout(j),u?.(l,"stderr",`[进程错误] Git 进程启动失败: ${t.message}
13
- `),u?.(l,"stderr",`[错误详情] ${t.stack||"No stack trace"}
14
- `);let r={...g,status:i.ProjectStatus.FAILED,progress:y,message:"进程启动失败",error:`进程启动失败: ${t.message}`,endTime:Date.now()};a?.(`${l}: ${t.message}`),e(r)})})}function u(e){return/^(https?:\/\/|git@)[^\s]+\.git$/i.test(e)}function l(e){return e.replace(/[^a-zA-Z0-9\-_]/g,"")}async function p(e,t){try{let r=(0,o.join)(e,t);if(!(0,n.existsSync)(r))return{exists:!1,status:"not_exists",needsCleanup:!1};let s=(0,o.join)(r,".git");if(!(0,n.existsSync)(s))return{exists:!0,status:"incomplete",needsCleanup:!0};if(!await d(r))return console.log(`检测到不完整的Git仓库: ${r}`),{exists:!0,status:"incomplete",needsCleanup:!0};return{exists:!0,status:"complete",needsCleanup:!1}}catch(e){return console.error(`检查项目 ${t} 状态时出错:`,e),{exists:!1,status:"not_exists",needsCleanup:!1}}}async function g(e,t){return"complete"===(await p(e,t)).status}async function d(e){try{let t=(0,o.join)(e,".git");for(let e of["HEAD","config","refs"]){let r=(0,o.join)(t,e);if(!(0,n.existsSync)(r))return!1}let r=(await n.promises.readdir(e)).filter(e=>".git"!==e);if(0===r.length)return!1;let s=(0,o.join)(t,"HEAD"),i=await n.promises.readFile(s,"utf-8");if(!i.trim()||i.includes("unborn"))return!1;return!0}catch(e){return console.error("检查Git仓库完整性时出错:",e),!1}}async function m(e){try{console.log(`正在清理不完整的项目: ${e}`),await n.promises.rm(e,{recursive:!0,force:!0}),console.log(`清理完成: ${e}`)}catch(t){console.error(`清理项目失败 ${e}:`,t)}}async function h(e,t){let r={},s=t.map(async t=>{let s=await g(e,t);r[t]=s});return await Promise.all(s),r}},15079:e=>{"use strict";var{g:t,__dirname:r}=e;e.s({createProjectEnvFiles:()=>o,createProjectMockDirectories:()=>i,updateProjectEnvFile:()=>c});var s=e.i(83886),n=e.i(30331);async function o(e,t,r){let o=(0,n.join)(r,e),i=[],c=[];try{let e=t.envs||{};for(let t in e){let r=e[t];if(r.envFileName&&r.proxyKey){let e=(0,n.join)(o,r.envFileName);try{await s.promises.access(e),c.push(r.envFileName)}catch{let t=r.proxyEnv[r.currentProxy]||"",n=`${r.proxyKey}=${t}`;await s.promises.writeFile(e,n,"utf-8"),i.push(r.envFileName)}}}return{success:!0,createdFiles:i,skippedFiles:c}}catch(r){let t=r instanceof Error?r.message:String(r);return console.error(`在为项目 ${e} 创建环境文件时出错:`,r),{success:!1,createdFiles:i,skippedFiles:c,error:t}}}async function i(e,t){let r=(0,n.join)(t,e),o=(0,n.join)(r,".mock"),i=[],c=[];try{try{await s.promises.access(o),c.push(".mock")}catch{await s.promises.mkdir(o,{recursive:!0}),i.push(".mock")}return{success:!0,createdDirectories:i,skippedDirectories:c}}catch(r){let t=r instanceof Error?r.message:String(r);return console.error(`在为项目 ${e} 创建Mock目录时出错:`,r),{success:!1,createdDirectories:i,skippedDirectories:c,error:t}}}async function c(e,t,r,o){try{let i=r.envs[t];if(!i)return{success:!1,error:`环境 ${t} 不存在于项目 ${e} 配置中`};if(!i.envFileName||!i.proxyKey)return{success:!1,error:`环境 ${t} 缺少 envFileName 或 proxyKey 配置`};let c=(0,n.join)(o,e),a=(0,n.join)(c,i.envFileName),u=i.proxyEnv[i.currentProxy]||i.currentProxy,l=`${i.proxyKey}=${u}`;return await s.promises.writeFile(a,l,"utf-8"),console.log(`已更新环境文件: ${a}, 内容: ${l}`),{success:!0,updatedFile:i.envFileName}}catch(s){let r=s instanceof Error?s.message:String(s);return console.error(`更新项目 ${e} 环境 ${t} 的环境文件时出错:`,s),{success:!1,error:r}}}},6618:function(e){var{g:t,__dirname:r,m:s,e:n}=e},44055:e=>{"use strict";var{g:t,__dirname:r}=e;{e.s({ConcurrentController:()=>t});class t{running=new Set;queue=[];maxConcurrent;constructor(e){this.maxConcurrent=e}async add(e){return new Promise((t,r)=>{this.queue.push(async()=>{try{let r=await e();return t(r),r}catch(e){throw r(e),e}}),this.process()})}async process(){if(this.running.size>=this.maxConcurrent||0===this.queue.length)return;let e=this.queue.shift();if(!e)return;let t=e().finally(()=>{this.running.delete(t),this.process()});this.running.add(t)}async waitForAll(){for(;this.running.size>0||this.queue.length>0;)this.running.size>0&&await Promise.race(this.running),this.process()}getStatus(){return{running:this.running.size,queued:this.queue.length,maxConcurrent:this.maxConcurrent}}}}},77854:e=>{"use strict";var{g:t,__dirname:r}=e;e.s({GET:()=>a});var s=e.i(88941),n=e.i(35692),o=e.i(77159),i=e.i(15079),c=e.i(44055);async function a(){let e=new TextEncoder,t=new ReadableStream({async start(t){try{let r=r=>{"command_log"===r.type&&"string"==typeof r.content&&r.content.length>8192&&(r.content=r.content.slice(0,8192)+"\n[...日志内容已截断...]");let s=`data: ${JSON.stringify(r)}
12
+ `);let n={...g,status:i.ProjectStatus.FAILED,progress:y,message:"克隆失败",error:r||`Git 进程退出,代码: ${t}`,endTime:s};u?.(`${p}: ${n.error}`),e(n)}}),h.on("error",t=>{clearTimeout(j),l?.(p,"stderr",`[进程错误] Git 进程启动失败: ${t.message}
13
+ `),l?.(p,"stderr",`[错误详情] ${t.stack||"No stack trace"}
14
+ `);let r={...g,status:i.ProjectStatus.FAILED,progress:y,message:"进程启动失败",error:`进程启动失败: ${t.message}`,endTime:Date.now()};u?.(`${p}: ${t.message}`),e(r)})})}function u(e){return/^(https?:\/\/|git@)[^\s]+\.git$/i.test(e)}function l(e){return e.replace(/[^a-zA-Z0-9\-_]/g,"")}async function p(e,t){try{let r=(0,o.join)(e,t);if(!(0,n.existsSync)(r))return{exists:!1,status:"not_exists",needsCleanup:!1};let s=(0,o.join)(r,".git");if(!(0,n.existsSync)(s))return{exists:!0,status:"incomplete",needsCleanup:!0};if(!await d(r))return console.log(`检测到不完整的Git仓库: ${r}`),{exists:!0,status:"incomplete",needsCleanup:!0};return{exists:!0,status:"complete",needsCleanup:!1}}catch(e){return console.error(`检查项目 ${t} 状态时出错:`,e),{exists:!1,status:"not_exists",needsCleanup:!1}}}async function g(e,t){return"complete"===(await p(e,t)).status}async function d(e){try{let t=(0,o.join)(e,".git");for(let e of["HEAD","config","refs"]){let r=(0,o.join)(t,e);if(!(0,n.existsSync)(r))return!1}let r=(await n.promises.readdir(e)).filter(e=>".git"!==e);if(0===r.length)return!1;let s=(0,o.join)(t,"HEAD"),i=await n.promises.readFile(s,"utf-8");if(!i.trim()||i.includes("unborn"))return!1;return!0}catch(e){return console.error("检查Git仓库完整性时出错:",e),!1}}async function m(e){try{console.log(`正在清理不完整的项目: ${e}`),await n.promises.rm(e,{recursive:!0,force:!0}),console.log(`清理完成: ${e}`)}catch(t){console.error(`清理项目失败 ${e}:`,t)}}async function h(e,t){let r={},s=t.map(async t=>{let s=await g(e,t);r[t]=s});return await Promise.all(s),r}},15079:e=>{"use strict";var{g:t,__dirname:r}=e;e.s({createProjectEnvFiles:()=>o,createProjectMockDirectories:()=>i,updateProjectEnvFile:()=>c});var s=e.i(83886),n=e.i(30331);async function o(e,t,r){let o=(0,n.join)(r,e),i=[],c=[];try{let e=t.envs||{};for(let t in e){let r=e[t];if(r.envFileName&&r.proxyKey){let e=(0,n.join)(o,r.envFileName);try{await s.promises.access(e),c.push(r.envFileName)}catch{let t=r.proxyEnv[r.currentProxy]||"",n=`${r.proxyKey}=${t}`;await s.promises.writeFile(e,n,"utf-8"),i.push(r.envFileName)}}}return{success:!0,createdFiles:i,skippedFiles:c}}catch(r){let t=r instanceof Error?r.message:String(r);return console.error(`在为项目 ${e} 创建环境文件时出错:`,r),{success:!1,createdFiles:i,skippedFiles:c,error:t}}}async function i(e,t){let r=(0,n.join)(t,e),o=(0,n.join)(r,".mock"),i=[],c=[];try{try{await s.promises.access(o),c.push(".mock")}catch{await s.promises.mkdir(o,{recursive:!0}),i.push(".mock")}return{success:!0,createdDirectories:i,skippedDirectories:c}}catch(r){let t=r instanceof Error?r.message:String(r);return console.error(`在为项目 ${e} 创建Mock目录时出错:`,r),{success:!1,createdDirectories:i,skippedDirectories:c,error:t}}}async function c(e,t,r,o){try{let i=r.envs[t];if(!i)return{success:!1,error:`环境 ${t} 不存在于项目 ${e} 配置中`};if(!i.envFileName||!i.proxyKey)return{success:!1,error:`环境 ${t} 缺少 envFileName 或 proxyKey 配置`};let c=(0,n.join)(o,e),a=(0,n.join)(c,i.envFileName),u=i.proxyEnv[i.currentProxy]||i.currentProxy,l=`${i.proxyKey}=${u}`;return await s.promises.writeFile(a,l,"utf-8"),console.log(`已更新环境文件: ${a}, 内容: ${l}`),{success:!0,updatedFile:i.envFileName}}catch(s){let r=s instanceof Error?s.message:String(s);return console.error(`更新项目 ${e} 环境 ${t} 的环境文件时出错:`,s),{success:!1,error:r}}}},6618:function(e){var{g:t,__dirname:r,m:s,e:n}=e},44055:e=>{"use strict";var{g:t,__dirname:r}=e;{e.s({ConcurrentController:()=>t});class t{running=new Set;queue=[];maxConcurrent;constructor(e){this.maxConcurrent=e}async add(e){return new Promise((t,r)=>{this.queue.push(async()=>{try{let r=await e();return t(r),r}catch(e){throw r(e),e}}),this.process()})}async process(){if(this.running.size>=this.maxConcurrent||0===this.queue.length)return;let e=this.queue.shift();if(!e)return;let t=e().finally(()=>{this.running.delete(t),this.process()});this.running.add(t)}async waitForAll(){for(;this.running.size>0||this.queue.length>0;)this.running.size>0&&await Promise.race(this.running),this.process()}getStatus(){return{running:this.running.size,queued:this.queue.length,maxConcurrent:this.maxConcurrent}}}}},77854:e=>{"use strict";var{g:t,__dirname:r}=e;e.s({GET:()=>a});var s=e.i(88941),n=e.i(35692),o=e.i(77159),i=e.i(15079),c=e.i(44055);async function a(){let e=new TextEncoder,t=new ReadableStream({async start(t){try{let r=r=>{"command_log"===r.type&&"string"==typeof r.content&&r.content.length>8192&&(r.content=r.content.slice(0,8192)+"\n[...日志内容已截断...]");let s=`data: ${JSON.stringify(r)}
15
15
 
16
- `;t.enqueue(e.encode(s))},a=async(e,t,s=25)=>{r({type:"project_env_creation",projectName:e,message:`为项目 ${e} 检查/创建环境文件...`,progress:s});let n=await (0,i.createProjectEnvFiles)(e,t,u);if(!n.success)return void r({type:"project_env_failed",projectName:e,message:`为项目 ${e} 创建环境文件失败: ${n.error}`,error:n.error,progress:s+2});{let t=`项目 ${e} 环境文件处理完成。`;n.createdFiles.length>0&&(t+=` 新建: ${n.createdFiles.join(", ")}.`),n.skippedFiles.length>0&&(t+=` 已存在,跳过: ${n.skippedFiles.join(", ")}.`),r({type:"project_env_success",projectName:e,message:t,progress:s+2})}r({type:"project_mock_creation",projectName:e,message:`为项目 ${e} 创建Mock目录结构...`,progress:s+3});let o=await (0,i.createProjectMockDirectories)(e,u);if(o.success){let t=`项目 ${e} Mock目录结构处理完成。`;o.createdDirectories.length>0&&(t+=` 新建: ${o.createdDirectories.join(", ")}.`),o.skippedDirectories.length>0&&(t+=` 已存在,跳过: ${o.skippedDirectories.join(", ")}.`),r({type:"project_mock_success",projectName:e,message:t,progress:s+5})}else r({type:"project_mock_failed",projectName:e,message:`为项目 ${e} 创建Mock目录失败: ${o.error}`,error:o.error,progress:s+5})};r({type:"initialization",message:"开始初始化工作空间...",progress:0});let u=s.WORKSPACE_CONFIG.WORKSPACE_ROOT,l=await (0,o.createWorkspaceDirectory)(u);if(!l.success){r({type:"error",message:l.error,progress:0}),t.close();return}r({type:"workspace_created",message:`工作空间已创建: ${u}`,progress:10});let p=await (0,n.getProjectsConfig)(),g=Object.entries(p),d=g.length;if(0===d){r({type:"error",message:"没有找到需要克隆的项目",progress:10}),t.close();return}r({type:"projects_loaded",message:`找到 ${d} 个项目准备克隆`,projects:g.map(([e])=>e),progress:20});let m={success:!0,totalProjects:d,successCount:0,failedCount:0,failedProjects:[],workspacePath:u,duration:0},h=Date.now(),f=await (0,o.checkMultipleProjectsExist)(u,g.map(([e])=>e)),y=g.filter(([e])=>f[e]),j=g.filter(([e])=>!f[e]);for(let[e,t]of y)r({type:"project_existing",projectName:e,message:`项目 ${e} 已存在,跳过克隆`}),await a(e,t,25);if(m.successCount=y.length,0===j.length){r({type:"completion",message:`所有 ${d} 个项目都已存在,无需克隆`,progress:100,result:{success:!0,totalProjects:d,successCount:d,failedCount:0,failedProjects:[],workspacePath:u,duration:Date.now()-h}}),t.close();return}let C=s.WORKSPACE_CONFIG.MAX_CONCURRENT_CLONES;r({type:"projects_filtered",message:`发现 ${y.length} 个项目已存在,${j.length} 个项目需要克隆(并发数: ${C})`,totalProjects:d,projectsToClone:j.length,existingProjects:y.length,progress:25});let x=new c.ConcurrentController(C),$=new Map,w=j.map(([e,t],n)=>x.add(async()=>{let i=25+70*n/(j.length||1);r({type:"project_start",projectName:e,message:`开始克隆项目: ${e} (${n+1}/${j.length})`,progress:Math.round(i)});try{let n=await (0,o.cloneRepository)({repoUrl:t.repo,targetDir:u,timeout:s.WORKSPACE_CONFIG.GIT_TIMEOUT,onProgress:(t,s)=>{let n=Math.round(i+50*t/(100*(j.length||1)));r({type:"project_progress",projectName:e,progress:Math.min(n,85),message:s})},onError:t=>{r({type:"project_error",projectName:e,error:t,progress:Math.round(i+50/(j.length||1))})},onLog:(e,t,s)=>{r({type:"command_log",projectName:e,logType:t,content:s,timestamp:Date.now()})}});n.status===s.ProjectStatus.SUCCESS?($.set(e,{status:s.ProjectStatus.SUCCESS,result:n}),r({type:"project_success",projectName:e,message:`项目 ${e} 克隆成功`,progress:Math.round(i+55/(j.length||1)),result:n}),await a(e,t,Math.round(i+60/(j.length||1)))):($.set(e,{status:s.ProjectStatus.FAILED,error:n.error,result:n}),r({type:"project_failed",projectName:e,message:`项目 ${e} 克隆失败: ${n.error}`,progress:Math.round(i+70/(j.length||1)),result:n}))}catch(n){let t=n instanceof Error?n.message:String(n);$.set(e,{status:s.ProjectStatus.FAILED,error:t}),r({type:"project_failed",projectName:e,message:`项目 ${e} 克隆异常: ${t}`,error:t,progress:Math.round(i+70/(j.length||1))})}}));for(let[e,t]of(await Promise.all(w),$))t.status===s.ProjectStatus.SUCCESS?m.successCount++:(m.failedCount++,m.failedProjects.push(e));m.duration=Date.now()-h,m.success=0===m.failedCount,r({type:"completion",message:m.success?`所有项目克隆完成!成功 ${m.successCount} 个,耗时 ${Math.round(m.duration/1e3)}s`:`克隆完成,部分失败。成功 ${m.successCount} 个,失败 ${m.failedCount} 个,耗时 ${Math.round(m.duration/1e3)}s`,progress:100,result:m}),t.close()}catch(n){let r=n instanceof Error?n.message:String(n);console.error("工作空间初始化失败:",n);let s=`data: ${JSON.stringify({type:"error",message:`工作空间初始化失败: ${r}`,error:r,progress:0})}
16
+ `;t.enqueue(e.encode(s))},a=async(e,t,s=25)=>{r({type:"project_env_creation",projectName:e,message:`为项目 ${e} 检查/创建环境文件...`,progress:s});let n=await (0,i.createProjectEnvFiles)(e,t,u);if(!n.success)return void r({type:"project_env_failed",projectName:e,message:`为项目 ${e} 创建环境文件失败: ${n.error}`,error:n.error,progress:s+2});{let t=`项目 ${e} 环境文件处理完成。`;n.createdFiles.length>0&&(t+=` 新建: ${n.createdFiles.join(", ")}.`),n.skippedFiles.length>0&&(t+=` 已存在,跳过: ${n.skippedFiles.join(", ")}.`),r({type:"project_env_success",projectName:e,message:t,progress:s+2})}r({type:"project_mock_creation",projectName:e,message:`为项目 ${e} 创建Mock目录结构...`,progress:s+3});let o=await (0,i.createProjectMockDirectories)(e,u);if(o.success){let t=`项目 ${e} Mock目录结构处理完成。`;o.createdDirectories.length>0&&(t+=` 新建: ${o.createdDirectories.join(", ")}.`),o.skippedDirectories.length>0&&(t+=` 已存在,跳过: ${o.skippedDirectories.join(", ")}.`),r({type:"project_mock_success",projectName:e,message:t,progress:s+5})}else r({type:"project_mock_failed",projectName:e,message:`为项目 ${e} 创建Mock目录失败: ${o.error}`,error:o.error,progress:s+5})};r({type:"initialization",message:"开始初始化工作空间...",progress:0});let u=s.WORKSPACE_CONFIG.WORKSPACE_ROOT,l=await (0,o.createWorkspaceDirectory)(u);if(!l.success){r({type:"error",message:l.error,progress:0}),t.close();return}r({type:"workspace_created",message:`工作空间已创建: ${u}`,progress:10});let p=await (0,n.getProjectsConfig)(),g=Object.entries(p),d=g.length;if(0===d){r({type:"error",message:"没有找到需要克隆的项目",progress:10}),t.close();return}r({type:"projects_loaded",message:`找到 ${d} 个项目准备克隆`,projects:g.map(([e])=>e),progress:20});let m={success:!0,totalProjects:d,successCount:0,failedCount:0,failedProjects:[],workspacePath:u,duration:0},h=Date.now(),f=await (0,o.checkMultipleProjectsExist)(u,g.map(([e])=>e)),y=g.filter(([e])=>f[e]),j=g.filter(([e])=>!f[e]);for(let[e,t]of y)r({type:"project_existing",projectName:e,message:`项目 ${e} 已存在,跳过克隆`}),await a(e,t,25);if(m.successCount=y.length,0===j.length){r({type:"completion",message:`所有 ${d} 个项目都已存在,无需克隆`,progress:100,result:{success:!0,totalProjects:d,successCount:d,failedCount:0,failedProjects:[],workspacePath:u,duration:Date.now()-h}}),t.close();return}let C=s.WORKSPACE_CONFIG.MAX_CONCURRENT_CLONES;r({type:"projects_filtered",message:`发现 ${y.length} 个项目已存在,${j.length} 个项目需要克隆(并发数: ${C})`,totalProjects:d,projectsToClone:j.length,existingProjects:y.length,progress:25});let x=new c.ConcurrentController(C),$=new Map,w=j.map(([e,t],n)=>x.add(async()=>{let i=25+70*n/(j.length||1);r({type:"project_start",projectName:e,message:`开始克隆项目: ${e} (${n+1}/${j.length})`,progress:Math.round(i)});try{let n=await (0,o.cloneRepository)({repoUrl:t.repo,projectName:e,targetDir:u,timeout:s.WORKSPACE_CONFIG.GIT_TIMEOUT,onProgress:(t,s)=>{let n=Math.round(i+50*t/(100*(j.length||1)));r({type:"project_progress",projectName:e,progress:Math.min(n,85),message:s})},onError:t=>{r({type:"project_error",projectName:e,error:t,progress:Math.round(i+50/(j.length||1))})},onLog:(e,t,s)=>{r({type:"command_log",projectName:e,logType:t,content:s,timestamp:Date.now()})}});n.status===s.ProjectStatus.SUCCESS?($.set(e,{status:s.ProjectStatus.SUCCESS,result:n}),r({type:"project_success",projectName:e,message:`项目 ${e} 克隆成功`,progress:Math.round(i+55/(j.length||1)),result:n}),await a(e,t,Math.round(i+60/(j.length||1)))):($.set(e,{status:s.ProjectStatus.FAILED,error:n.error,result:n}),r({type:"project_failed",projectName:e,message:`项目 ${e} 克隆失败: ${n.error}`,progress:Math.round(i+70/(j.length||1)),result:n}))}catch(n){let t=n instanceof Error?n.message:String(n);$.set(e,{status:s.ProjectStatus.FAILED,error:t}),r({type:"project_failed",projectName:e,message:`项目 ${e} 克隆异常: ${t}`,error:t,progress:Math.round(i+70/(j.length||1))})}}));for(let[e,t]of(await Promise.all(w),$))t.status===s.ProjectStatus.SUCCESS?m.successCount++:(m.failedCount++,m.failedProjects.push(e));m.duration=Date.now()-h,m.success=0===m.failedCount,r({type:"completion",message:m.success?`所有项目克隆完成!成功 ${m.successCount} 个,耗时 ${Math.round(m.duration/1e3)}s`:`克隆完成,部分失败。成功 ${m.successCount} 个,失败 ${m.failedCount} 个,耗时 ${Math.round(m.duration/1e3)}s`,progress:100,result:m}),t.close()}catch(n){let r=n instanceof Error?n.message:String(n);console.error("工作空间初始化失败:",n);let s=`data: ${JSON.stringify({type:"error",message:`工作空间初始化失败: ${r}`,error:r,progress:0})}
17
17
 
18
18
  `;t.enqueue(e.encode(s)),t.close()}}});return new Response(t,{headers:{"Content-Type":"text/event-stream","Cache-Control":"no-cache",Connection:"keep-alive","Access-Control-Allow-Origin":"*","Access-Control-Allow-Headers":"Cache-Control"}})}},80080:e=>{"use strict";var{g:t,__dirname:r}=e;{e.s({patchFetch:()=>c,routeModule:()=>t,serverHooks:()=>u,workAsyncStorage:()=>r,workUnitAsyncStorage:()=>a});var s=e.i(45746),n=e.i(93828),o=e.i(18250),i=e.i(77854);let t=new s.AppRouteRouteModule({definition:{kind:n.RouteKind.APP_ROUTE,page:"/api/initialize-workspace/route",pathname:"/api/initialize-workspace",filename:"route",bundlePath:""},resolvedPagePath:"[project]/packages/server/src/app/api/initialize-workspace/route.ts",nextConfigOutput:"",userland:i}),{workAsyncStorage:r,workUnitAsyncStorage:a,serverHooks:u}=t;function c(){return(0,o.patchFetch)({workAsyncStorage:r,workUnitAsyncStorage:a})}}}};
19
19