@chatbi-v/mocks 2.1.0 → 2.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.js +846 -85
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +789 -15
- package/dist/index.mjs.map +1 -1
- package/package.json +4 -4
- package/dist/adapter.d.ts +0 -48
- package/dist/adapter.js +0 -308
- package/dist/adapter.js.map +0 -1
- package/dist/generator.d.ts +0 -68
- package/dist/generator.js +0 -150
- package/dist/generator.js.map +0 -1
- package/dist/index.cjs +0 -25
- package/dist/index.cjs.map +0 -1
- package/dist/index.d.ts +0 -89
- package/dist/interceptor.d.ts +0 -10
- package/dist/interceptor.js +0 -80
- package/dist/interceptor.js.map +0 -1
- package/dist/strategies.d.ts +0 -20
- package/dist/strategies.js +0 -137
- package/dist/strategies.js.map +0 -1
- package/dist/types.d.ts +0 -77
- package/dist/types.js +0 -8
- package/dist/types.js.map +0 -1
- package/dist/utils.d.ts +0 -19
- package/dist/utils.js +0 -108
- package/dist/utils.js.map +0 -1
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/adapter.ts","../src/strategies.ts","../src/generator.ts","../src/utils.ts","../src/interceptor.ts","../src/types.ts"],"sourcesContent":["/**\n * @file index.ts\n * @description Mock 模块入口,导出模拟数据适配器、生成器及静态测试数据\n * @author ChatBI Team\n */\n\nimport { dateUtils } from '@chatbi-v/core';\n\n/**\n * Mock 数据模块\n * @description 提供前端开发所需的模拟数据适配器和工具\n */\n\nexport * from './adapter';\nexport * from './generator';\nexport * from './interceptor';\nexport * from './strategies';\nexport * from './types';\nexport * from './utils';\n\n/**\n * 模拟用户数据\n */\nexport const MOCK_USER = {\n id: 'u_001',\n name: 'Admin User',\n avatar: 'https://api.dicebear.com/7.x/avataaars/svg?seed=Admin',\n role: 'admin',\n};\n\n// 保留原有的静态数据导出,以兼容可能直接引用的地方,但建议使用 adapter 自动生成\nexport const MOCK_SESSIONS = [\n { id: 's_001', title: '2024 Q1 销售分析', date: '2024-03-15', lastMessage: '好的,正在为您分析 Q1 销售数据...' },\n { id: 's_002', title: '用户增长趋势', date: '2024-03-10', lastMessage: '用户增长曲线显示...' },\n { id: 's_003', title: '竞品分析报告', date: '2024-03-08', lastMessage: '主要竞品 A 的市场份额...' },\n { id: 's_004', title: '营销活动复盘', date: '2024-03-05', lastMessage: 'ROI 提升了 15%...' },\n { id: 's_005', title: '财务报表审计', date: '2024-03-01', lastMessage: '请确认以下财务指标...' },\n];\n\nexport const MOCK_FAVORITES = [\n { id: 'fav-1', title: '季度销售总结', date: '2025/12/15' },\n { id: 'fav-2', title: '年度预算规划', date: '2025/12/10' },\n { id: 'fav-3', title: '核心指标监控', date: '2025/11/20' },\n];\n\nexport const MOCK_SCENES = [\n {\n id: '1',\n name: '销售数据分析',\n description: '全方位分析销售业绩,包括区域、产品、时间维度',\n code: 'SALES_001',\n creator: '管理员',\n createTime: '2023-01-01',\n tableCount: 5,\n kbCount: 2,\n qaCount: 10,\n cover: 'https://images.unsplash.com/photo-1551288049-bebda4e38f71?w=800&q=80',\n },\n {\n id: '2',\n name: '人力资源概览',\n description: '员工入职、离职、绩效分布与培训情况分析',\n code: 'HR_001',\n creator: 'HR 经理',\n createTime: '2023-02-15',\n tableCount: 3,\n kbCount: 1,\n qaCount: 5,\n cover: 'https://images.unsplash.com/photo-1460925895917-afdab827c52f?w=800&q=80',\n },\n {\n id: '3',\n name: '供应链优化',\n description: '库存周转、物流成本与供应商绩效评估',\n code: 'SCM_001',\n creator: '供应链总监',\n createTime: '2023-03-20',\n tableCount: 8,\n kbCount: 0,\n qaCount: 15,\n cover: 'https://images.unsplash.com/photo-1586528116311-ad8dd3c8310d?w=800&q=80',\n },\n];\n\nexport const generateMockMessages = (sessionId: string) => [\n {\n key: `msg_${sessionId}_1`,\n role: 'assistant',\n content: `您好!我是您的智能助手。当前会话 ID: ${sessionId}。请问有什么可以帮您?`,\n time: dateUtils.dayjs(dateUtils.now() - 10000).toISOString(),\n },\n];\n\nexport const mockData = {\n user: MOCK_USER,\n sessions: MOCK_SESSIONS,\n favorites: MOCK_FAVORITES,\n scenes: MOCK_SCENES,\n favoritesList: MOCK_FAVORITES, // Alias for compatibility if needed\n};\n","import {\n ApiAdapter,\n ApiEndpointConfig,\n ApiRequestConfig,\n createLogger,\n dateUtils,\n StreamCallbacks,\n} from '@chatbi-v/core';\nimport Mock from 'mockjs';\n\nimport { ChatStreamStrategy, HistoryStreamStrategy, StrategyFactory } from './strategies';\nimport { MockConfig, MockGeneratorStrategy } from './types';\nimport { sleep } from './utils';\n\nconst logger = createLogger('MockAdapter');\n\n/**\n * 基于 Mock.js 的请求适配器\n * @description 根据 API 配置中的 responseSchema 自动生成 Mock 数据,支持普通 JSON 和流式 SSE 响应\n */\nexport class MockAdapter implements ApiAdapter {\n private delay: number;\n // 遗留策略(为了兼容旧版代码)\n private static strategies: Record<string, MockGeneratorStrategy> = {\n chat_stream: new ChatStreamStrategy(),\n history_stream: new HistoryStreamStrategy(),\n };\n\n /**\n * 构造函数\n * @param delay 全局模拟延迟时间(毫秒),默认 300ms\n */\n constructor(delay: number = 300) {\n this.delay = delay;\n }\n\n /**\n * 注册自定义 Mock 生成策略\n * @param key 策略唯一标识符 (e.g., 'chat_stream', 'history_stream')\n * @param strategy 实现了 MockGeneratorStrategy 接口的策略实例\n */\n static registerStrategy(key: string, strategy: MockGeneratorStrategy) {\n this.strategies[key] = strategy;\n }\n\n /**\n * 处理普通 HTTP 请求(非流式)\n * @param config 请求配置对象\n * @param endpointConfig API 端点配置,包含 responseSchema\n * @returns Promise 返回模拟的响应数据\n */\n async request<T = any>(config: ApiRequestConfig, endpointConfig?: ApiEndpointConfig): Promise<T> {\n // 优先使用接口定义的 delay,否则使用全局默认 delay\n const delay = (endpointConfig as any)?.delay ?? this.delay;\n\n return new Promise((resolve, reject) => {\n setTimeout(() => {\n try {\n if (!endpointConfig || !endpointConfig.responseSchema) {\n logger.warn(`未找到响应架构配置: ${config.url}`);\n resolve({} as T);\n return;\n }\n\n let schema = endpointConfig.responseSchema;\n let mockData;\n\n // 支持函数式 schema,允许根据请求参数动态生成 schema\n if (typeof schema === 'function') {\n schema = schema(config);\n }\n\n // 构造有效配置对象,合并 endpointConfig 顶层属性\n const effectiveConfig = {\n type: (endpointConfig as any).type,\n status: (endpointConfig as any).status || 200,\n pageEvent: (endpointConfig as any).pageEvent,\n responseSchema: schema,\n ...(typeof schema === 'object' ? schema : {}),\n };\n\n // 0. 处理非 200 状态码\n if (effectiveConfig.status !== 200) {\n const error: any = new Error(`Request failed with status code ${effectiveConfig.status}`);\n error.response = {\n status: effectiveConfig.status,\n data: Mock.mock(schema),\n headers: {},\n };\n reject(error);\n return;\n }\n\n // 1. 字符串 Schema:直接作为结果返回(极少情况)\n if (typeof schema === 'string') {\n mockData = schema;\n }\n // 2. 遗留的高级 Schema (通过 _type 字段识别)\n else if (this.isLegacySchema(schema)) {\n const strategy = MockAdapter.strategies[schema._type];\n if (strategy) {\n const chunks = strategy.generate(schema);\n // 将流式块合并为单个字符串,并清理 SSE 格式标记\n mockData = chunks\n .join('')\n .replace(/event: data\\ndata: /g, '')\n .replace(/\\n\\n/g, '');\n try {\n mockData = JSON.parse(mockData);\n } catch (e) {\n // 忽略解析错误,保持原始数据格式\n }\n } else {\n logger.warn(`未找到对应的策略类型: ${schema._type}`);\n mockData = {};\n }\n }\n // 3. 通用策略模式\n else {\n const type = effectiveConfig.type || 'json';\n const strategy = StrategyFactory.getStrategy(type);\n mockData = strategy.process(effectiveConfig as MockConfig, config.params || config.data);\n }\n\n logger.info(`Request: ${config.method} ${config.url}`, config.data || config.params);\n logger.info(`Response:`, mockData);\n\n resolve(mockData as T);\n } catch (error) {\n reject(error);\n }\n }, delay);\n });\n }\n\n /**\n * 处理流式请求 (SSE)\n * @param config 请求配置对象\n * @param callbacks 流式回调函数集合 (onMessage, onFinish, onError)\n * @param endpointConfig API 端点配置,包含 responseSchema\n */\n async stream(\n config: ApiRequestConfig,\n callbacks: StreamCallbacks,\n endpointConfig?: ApiEndpointConfig,\n ): Promise<void> {\n const { onMessage, onFinish, onError } = callbacks;\n const signal = config.signal;\n\n // 优先使用接口定义的 delay,否则使用全局默认 delay\n const delay = (endpointConfig as any)?.delay ?? this.delay;\n\n return new Promise<void>((resolve, reject) => {\n // 如果请求已被取消,直接结束\n if (signal && signal.aborted) {\n if (onFinish) onFinish();\n resolve();\n return;\n }\n\n setTimeout(async () => {\n try {\n if (!endpointConfig || !endpointConfig.responseSchema) {\n logger.warn(`未找到流式响应架构: ${config.url}`);\n if (onFinish) onFinish();\n resolve();\n return;\n }\n\n const requestStart = dateUtils.now();\n logger.info(`[SSE Start] Request: ${config.method} ${config.url}`, {\n params: config.data || config.params,\n time: dateUtils.dayjs().toISOString(),\n });\n\n let schema = endpointConfig.responseSchema;\n\n // 支持函数式 schema\n if (typeof schema === 'function') {\n schema = schema(config);\n }\n\n // 构造有效配置对象,合并 endpointConfig 顶层属性\n const effectiveConfig = {\n type: (endpointConfig as any).type,\n status: (endpointConfig as any).status || 200,\n pageEvent: (endpointConfig as any).pageEvent,\n responseSchema: schema,\n ...(typeof schema === 'object' ? schema : {}),\n };\n\n // 0. 处理非 200 状态码\n if (effectiveConfig.status !== 200) {\n const response = {\n status: effectiveConfig.status,\n data: Mock.mock(schema),\n headers: {},\n };\n\n // 触发 onResponse 回调\n if (callbacks.onResponse) {\n const hijacked = await callbacks.onResponse(response);\n if (hijacked) {\n resolve();\n return;\n }\n }\n\n const error: any = new Error(`Stream request failed with status code ${effectiveConfig.status}`);\n error.response = response;\n logger.error(`[SSE Error] Request failed with status ${effectiveConfig.status}`, error);\n if (onError) onError(error);\n // 保持与 request 同样的处理逻辑,抛出异常以便顶层拦截\n reject(error);\n return;\n }\n\n // 触发 onResponse 回调 (200 场景)\n if (callbacks.onResponse) {\n const hijacked = await callbacks.onResponse({\n status: 200,\n data: null,\n headers: {},\n });\n if (hijacked) {\n resolve();\n return;\n }\n }\n\n // 1. 新版 Schema 策略 (sse, sse-page)\n if (this.isNewConfig(effectiveConfig)) {\n const type = effectiveConfig.type || 'sse';\n const strategy = StrategyFactory.getStrategy(type);\n\n // 策略处理返回 MockEvent 数组\n const events = strategy.process(\n effectiveConfig as MockConfig,\n config.params || config.data,\n );\n\n if (Array.isArray(events)) {\n const startTime = dateUtils.now();\n let eventCount = 0;\n\n logger.info(\n `[SSE Processing] Generated ${events.length} events for ${type} strategy`,\n );\n\n for (const event of events) {\n if (signal && signal.aborted) {\n logger.info(\n `[SSE Abort] Stream aborted by user after ${dateUtils.now() - requestStart}ms`,\n );\n break;\n }\n\n // 计算延迟\n const delay =\n typeof event.delay === 'number'\n ? event.delay\n : parseInt(String(event.delay || 0));\n const elapsed = dateUtils.now() - startTime;\n const remaining = Math.max(0, delay - elapsed);\n\n if (remaining > 0) {\n await sleep(remaining);\n }\n\n if (signal && signal.aborted) break;\n\n // 构造 SSE 格式数据块\n const chunk = `event: ${event.event}\\ndata: ${JSON.stringify(event.data)}\\n\\n`;\n\n // 记录关键事件日志\n if (['error', 'todos', 'page'].includes(event.event)) {\n logger.info(`[SSE Event] Emitting special event: ${event.event}`, event.data);\n } else if (eventCount === 0 || eventCount === events.length - 1) {\n logger.info(\n `[SSE Event] Emitting ${eventCount === 0 ? 'first' : 'last'} data event`,\n { preview: JSON.stringify(event.data).slice(0, 50) + '...' },\n );\n }\n if (onMessage) onMessage(chunk);\n eventCount++;\n }\n }\n }\n // 2. 遗留策略处理\n else {\n let chunks: string[] = [];\n\n if (typeof schema === 'string') {\n chunks = schema.split('\\n\\n').map((chunk) => chunk + '\\n\\n');\n } else if (this.isLegacySchema(schema)) {\n chunks = this.generateAdvancedChunks(schema);\n } else {\n const mockData = Mock.mock(schema);\n chunks = [`data: ${JSON.stringify(mockData)}\\n\\n`];\n }\n\n for (const chunk of chunks) {\n if (signal && signal.aborted) {\n logger.info('[SSE Abort] Stream aborted by user');\n break;\n }\n\n if (chunk.trim()) {\n if (onMessage) onMessage(chunk);\n await new Promise((r) => setTimeout(r, 200));\n }\n }\n }\n\n if (!signal || !signal.aborted) {\n logger.info(\n `[SSE Complete] Stream finished successfully in ${dateUtils.now() - requestStart}ms`,\n );\n if (onFinish) onFinish();\n }\n resolve();\n } catch (error) {\n logger.error(`[SSE Error] Stream processing failed`, error);\n if (onError) onError(error);\n resolve();\n }\n }, delay);\n });\n }\n\n /**\n * 判断是否为遗留的 schema 格式\n */\n private isLegacySchema(schema: any): boolean {\n return schema && typeof schema === 'object' && '_type' in schema;\n }\n\n /**\n * 判断是否为新版配置格式\n * @description 检查配置对象中是否包含有效的 type 字段\n */\n private isNewConfig(config: any): boolean {\n return (\n config &&\n typeof config === 'object' &&\n (['json', 'sse', 'sse-page'].includes(config.type) || 'responseSchema' in config) // 备用检查,如果 schema 内部定义了结构\n );\n }\n\n /**\n * 生成遗留的高级 schema 数据块\n */\n private generateAdvancedChunks(schema: any): string[] {\n const strategy = MockAdapter.strategies[schema._type];\n if (strategy) {\n return strategy.generate(schema);\n }\n logger.warn(`未找到对应的策略类型: ${schema._type}`);\n return [];\n }\n}\n","import Mock from 'mockjs';\n\nimport { MockResponseGenerator } from './generator';\nimport { \n ChatStreamConfig,\n HistoryStreamConfig,\n JsonMockConfig, \n MockConfig, \n MockEvent,\n MockGeneratorStrategy, // Legacy\n MockStrategy, \n SseMockConfig, \n SsePageMockConfig} from './types';\nimport { flatEvents, processTemplate } from './utils';\n\n// === Legacy Strategies (Preserved) ===\n\nexport class ChatStreamStrategy implements MockGeneratorStrategy {\n generate(schema: any): string[] {\n const config = schema as ChatStreamConfig;\n const defaultName = config.agentName || 'BI助手';\n const generator = new MockResponseGenerator(defaultName);\n \n const mockData = config.data ? Mock.mock(config.data) : {};\n \n // 1. Init Plan\n if (config.plan) {\n generator.initPlan(config.plan);\n }\n\n // 2. Execute Timeline\n if (config.timeline) {\n config.timeline.forEach(step => {\n if (step.planIndex !== undefined) {\n generator.updatePlanStatus(step.planIndex);\n }\n if (step.log) {\n const logContent = typeof step.log === 'string' ? step.log : step.log.content;\n const logLevel = typeof step.log === 'object' ? step.log.level : 'info';\n const logAgent = typeof step.log === 'object' ? step.log.agent : undefined;\n generator.addLog(logContent, logLevel, logAgent);\n }\n });\n }\n\n // 3. Content\n let content;\n if (typeof config.content === 'function') {\n content = config.content(mockData);\n } else {\n content = config.content;\n }\n\n if (typeof content === 'object' && content !== null) {\n generator.emitA2UI(content as Record<string, any>);\n } else {\n generator.streamContent(content as string);\n }\n\n // 4. Complete\n generator.completePlan();\n generator.finish();\n\n return generator.toString().split('\\n\\n').map(chunk => chunk + '\\n\\n');\n }\n}\n\nexport class HistoryStreamStrategy implements MockGeneratorStrategy {\n generate(schema: any): string[] {\n const config = schema as HistoryStreamConfig;\n const generator = new MockResponseGenerator('BI助手');\n \n const historyMock = Mock.mock(config.template);\n const fullHistory = [...(config.prepend || []), ...(historyMock.list || historyMock)];\n generator.emitHistory(fullHistory);\n \n return generator.toString().split('\\n\\n').map(chunk => chunk + '\\n\\n');\n }\n}\n\n// === New Schema Strategies ===\n\nexport class JsonStrategy implements MockStrategy {\n process(config: MockConfig, _requestParams?: any): any {\n const jsonConfig = config as JsonMockConfig;\n if (!jsonConfig.responseSchema) return {};\n return Mock.mock(jsonConfig.responseSchema);\n }\n}\n\nexport class SseStrategy implements MockStrategy {\n process(config: MockConfig, requestParams: any = {}): MockEvent[] {\n const sseConfig = config as SseMockConfig;\n if (!sseConfig.responseSchema) return [];\n \n // Use flatEvents to process the schema with MockJS and sort by delay\n return flatEvents(sseConfig.responseSchema, requestParams);\n }\n}\n\nexport class SsePageStrategy implements MockStrategy {\n process(config: MockConfig, requestParams: any = {}): MockEvent[] {\n const ssePageConfig = config as SsePageMockConfig;\n if (!ssePageConfig.responseSchema) return [];\n\n // 1. Generate base events\n const events = flatEvents(ssePageConfig.responseSchema, requestParams);\n\n // 2. Generate page event if exists\n if (ssePageConfig.pageEvent) {\n const context = {\n $query: requestParams, // Backward compatibility\n $body: requestParams,\n $param: requestParams?.param || requestParams\n };\n\n Mock.Random.extend({ \n $query: () => context.$query,\n $body: () => context.$body,\n $param: () => context.$param\n });\n\n // First let Mock.js process standard templates (like @integer)\n let pageEvent = Mock.mock(ssePageConfig.pageEvent);\n // Then process custom {{}} templates\n pageEvent = processTemplate(pageEvent, context);\n \n // Calculate max delay from existing events to ensure page event is last\n let maxDelay = 0;\n if (events.length > 0) {\n const lastEvent = events[events.length - 1];\n maxDelay = typeof lastEvent.delay === 'number' ? lastEvent.delay : parseInt(String(lastEvent.delay || 0));\n }\n \n // Set page event delay to be slightly after the last event\n const pageDelay = typeof pageEvent.delay === 'number' ? pageEvent.delay : parseInt(String(pageEvent.delay || 0));\n if (pageDelay <= maxDelay) {\n pageEvent.delay = maxDelay + 20; // Add small buffer\n }\n\n events.push(pageEvent);\n }\n \n // Re-sort in case pageEvent has specific delay\n const es = events.sort((a, b) => {\n const delayA = typeof a.delay === 'number' ? a.delay : parseInt(String(a.delay || 0));\n const delayB = typeof b.delay === 'number' ? b.delay : parseInt(String(b.delay || 0));\n return delayA - delayB;\n });\n console.log(es)\n return es;\n }\n}\n\n// Factory for new strategies\nexport const StrategyFactory = {\n getStrategy(type: string = 'json'): MockStrategy {\n switch (type) {\n case 'sse':\n return new SseStrategy();\n case 'sse-page':\n return new SsePageStrategy();\n case 'json':\n default:\n return new JsonStrategy();\n }\n }\n};\n","import { dateUtils } from '@chatbi-v/core';\n\n/**\n * Mock 响应生成器\n * @description 用于动态生成流式响应数据,支持阶段流转、日志插入和内容分块\n */\nexport class MockResponseGenerator {\n private events: string[] = [];\n private sessionId: string;\n private agentName: string;\n private currentTodos: { content: string; status: 'PENDING' | 'IN_PROGRESS' | 'COMPLETED' }[] = [];\n\n constructor(agentName: string = 'Assistant', sessionId?: string) {\n this.sessionId = sessionId || ('conv_' + dateUtils.now());\n this.agentName = agentName;\n }\n\n /**\n * 生成历史消息流\n */\n emitHistory(history: { role: string; content: string; createTime: string; todos?: any[] }[]) {\n history.forEach(msg => {\n const data = {\n content: msg.content,\n sessionId: this.sessionId,\n role: msg.role,\n completed: true,\n agentName: this.agentName,\n createTime: msg.createTime\n };\n this.pushEvent('data', data);\n\n if (msg.todos) {\n this.pushEvent('todos', { items: msg.todos });\n }\n });\n return this;\n }\n\n /**\n * 初始化执行计划\n * @param steps 计划步骤列表\n */\n initPlan(steps: string[]) {\n this.currentTodos = steps.map(step => ({\n content: step,\n status: 'PENDING'\n }));\n this.pushTodos();\n return this;\n }\n\n /**\n * 更新执行计划状态\n * @param activeIndex 当前正在进行的步骤索引\n */\n updatePlanStatus(activeIndex: number) {\n this.currentTodos = this.currentTodos.map((todo, index) => {\n if (index < activeIndex) {\n return { ...todo, status: 'COMPLETED' };\n } else if (index === activeIndex) {\n return { ...todo, status: 'IN_PROGRESS' };\n } else {\n return { ...todo, status: 'PENDING' };\n }\n });\n this.pushTodos();\n return this;\n }\n\n /**\n * 标记所有计划为完成\n */\n completePlan() {\n this.currentTodos = this.currentTodos.map(todo => ({ ...todo, status: 'COMPLETED' }));\n this.pushTodos();\n return this;\n }\n\n /**\n * 添加日志\n * @param content 日志内容\n * @param type 日志类型\n * @param agentName 可选的 Agent 名称\n */\n addLog(content: string, type: 'info' | 'warning' | 'error' = 'info', agentName?: string) {\n this.pushEvent('log', {\n type,\n content,\n agentName: agentName || this.agentName\n });\n return this;\n }\n\n /**\n * 添加系统错误事件\n * @param errorCode 错误码\n * @param errorMessage 错误信息\n */\n addError(errorCode: string, errorMessage: string) {\n this.pushEvent('error', {\n errorCode,\n errorMessage\n });\n return this;\n }\n\n /**\n * 添加流式内容块\n * @param content 完整内容\n * @param chunkSize 分块大小\n */\n streamContent(content: string, chunkSize: number = 20) {\n for (let i = 0; i < content.length; i += chunkSize) {\n const chunk = content.slice(i, i + chunkSize);\n this.pushEvent('data', {\n content: chunk,\n sessionId: this.sessionId,\n completed: false,\n agentName: this.agentName\n });\n }\n return this;\n }\n\n /**\n * 结束流\n */\n finish() {\n this.pushEvent('data', {\n content: '',\n sessionId: this.sessionId,\n completed: true,\n agentName: this.agentName\n });\n return this;\n }\n\n /**\n * 生成 A2UI 响应\n * @param component A2UI 组件配置对象\n */\n emitA2UI(component: Record<string, any>) {\n const content = JSON.stringify(component, null, 2);\n return this.streamContent(content);\n }\n\n /**\n * 生成最终的响应字符串\n */\n toString() {\n return this.events.join('\\n');\n }\n\n private pushTodos() {\n this.pushEvent('todos', { items: this.currentTodos });\n }\n\n private pushEvent(type: string, data: any) {\n this.events.push(`event: ${type}\\ndata: ${JSON.stringify(data)}\\n`);\n }\n}\n","import dayjs from 'dayjs';\nimport Mock from 'mockjs';\n\nimport { MockEvent } from './types';\n\nexport const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));\n\n/**\n * Process string templates like \"{{$query.pageNo * 10}}\" in object values.\n * Supports $query (for backward compatibility), $body, and $param.\n */\nexport function processTemplate(data: any, context: Record<string, any>): any {\n if (typeof data === 'string') {\n // Check for {{ ... }} pattern\n if (/^\\{\\{.*\\}\\}$/.test(data)) {\n const expression = data.slice(2, -2).trim();\n try {\n // Safe evaluation using Function with context keys\n const keys = Object.keys(context);\n const values = Object.values(context);\n const fn = new Function(...keys, `return ${expression}`);\n return fn(...values);\n } catch (e) {\n console.warn(`[Mock] Failed to evaluate template: ${data}`, e);\n return data;\n }\n }\n return data;\n }\n \n if (Array.isArray(data)) {\n return data.map(item => processTemplate(item, context));\n }\n \n if (typeof data === 'object' && data !== null) {\n const result: Record<string, any> = {};\n for (const key in data) {\n result[key] = processTemplate(data[key], context);\n }\n return result;\n }\n \n return data;\n}\n\n/**\n * Flatten eventsSchema into a sorted array of events.\n * \n * Supports Mock.js array generation rules in keys, e.g.:\n * { 'data|8-12': [{ event: 'data', ... }] } \n * -> generates 8 to 12 data events.\n */\nexport function flatEvents(\n eventsSchema: Record<string, MockEvent[]>, \n requestData: any = {}\n): MockEvent[] {\n // Construct context for template processing\n const context: Record<string, any> = {\n $query: requestData, // Backward compatibility\n $body: requestData,\n $param: requestData?.param || requestData\n };\n\n // Extend Mock.Random to support templates\n Mock.Random.extend({ \n $query: () => context.$query,\n $body: () => context.$body,\n $param: () => context.$param\n });\n \n // Generate data using Mock.mock\n const compiled = Mock.mock(eventsSchema) as Record<string, MockEvent[]>;\n \n // Flatten and sort by delay\n return Object.values(compiled)\n .flat()\n .map(event => processTemplate(event, context))\n .sort((a, b) => {\n const delayA = typeof a.delay === 'number' ? a.delay : parseInt(String(a.delay || 0));\n const delayB = typeof b.delay === 'number' ? b.delay : parseInt(String(b.delay || 0));\n return delayA - delayB;\n });\n}\n\n/**\n * Convert an array of events into a ReadableStream (for SSE).\n */\nexport function eventsToStream(events: MockEvent[]): ReadableStream {\n const encoder = new TextEncoder();\n \n return new ReadableStream({\n start(ctrl) {\n // Check if events is empty\n if (!events || events.length === 0) {\n ctrl.close();\n return;\n }\n\n (async () => {\n try {\n // We assume events are sorted by their `delay` property which represents \n // \"absolute time from start\".\n const startTime = dayjs().valueOf();\n \n for (const eventItem of events) {\n const delay = typeof eventItem.delay === 'number' ? eventItem.delay : parseInt(String(eventItem.delay || 0));\n \n // Calculate how much time passed since start\n const elapsed = dayjs().valueOf() - startTime;\n const remaining = Math.max(0, delay - elapsed);\n \n if (remaining > 0) {\n await sleep(remaining);\n }\n \n const payload = `event: ${eventItem.event}\\ndata: ${JSON.stringify(eventItem.data)}\\n\\n`;\n ctrl.enqueue(encoder.encode(payload));\n }\n ctrl.close();\n } catch (e) {\n ctrl.error(e);\n }\n })();\n },\n });\n}\n","import Mock from 'mockjs';\n\nimport { StrategyFactory } from './strategies';\nimport { JsonMockConfig, MockConfig, MockSchema, SseMockConfig, SsePageMockConfig } from './types';\nimport { eventsToStream, sleep } from './utils';\n\ndeclare global {\n interface Window {\n _originalFetch: typeof fetch;\n _originalEventSource: typeof EventSource;\n }\n}\n\nexport function installFetchMock(schema: MockSchema) {\n if (!window._originalFetch) {\n window._originalFetch = window.fetch;\n }\n\n window.fetch = async (input: RequestInfo | URL, init: RequestInit = {}) => {\n const url = typeof input === 'string' ? input : input instanceof URL ? input.toString() : input.url;\n const method = (init.method || 'GET').toUpperCase();\n \n // Find matching rule\n const rules = Object.values(schema);\n const rule = rules.find((v) => {\n if (!v.url) return false;\n // Convert route params :id to regex \\w+\n // We might want to be more specific, e.g., ([^/]+)\n const pattern = v.url.replace(/:[a-zA-Z0-9_]+/g, '([^/]+)');\n const regex = new RegExp(`^${pattern}$`);\n \n // Check if URL matches (ignoring query params for the match, but we might need them later)\n const [path] = url.split('?');\n // Match method if specified, default to GET match? or just match URL?\n // Usually method matters.\n const ruleMethod = (v.method || 'GET').toUpperCase();\n return regex.test(path) && ruleMethod === method;\n });\n\n if (!rule) {\n return window._originalFetch(input, init);\n }\n\n console.log(`[Mock] Intercepted ${method} ${url}`, rule);\n\n // Handle non-200 status code immediately\n const status = rule.status || 200;\n if (status !== 200) {\n await sleep(rule.delay || 0);\n return new Response(JSON.stringify({ \n message: `Mock error status ${status}`,\n code: status \n }), {\n status,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n await sleep(rule.delay || 0);\n\n // Determine type (default 'json' if not present)\n const type = rule.type || 'json';\n const strategy = StrategyFactory.getStrategy(type);\n \n // Parse query params for strategies that need them\n const urlObj = new URL(url, window.location.origin);\n const query = Object.fromEntries(urlObj.searchParams);\n \n const result = strategy.process(rule as MockConfig, query);\n\n /* 1. JSON Response */\n if (type === 'json') {\n return new Response(JSON.stringify(result), {\n status,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n /* 2. SSE / SSE-Page Response */\n if (type === 'sse' || type === 'sse-page') {\n const events = result; // Strategy returns MockEvent[]\n const stream = eventsToStream(events);\n return new Response(stream, { \n status, \n headers: { 'Content-Type': 'text/event-stream' } \n });\n }\n\n return window._originalFetch(input, init);\n };\n}\n\nexport function installSSEMock() {\n // Placeholder for EventSource mocking if needed.\n // Currently, we rely on fetch interception.\n // Many modern apps use fetch for SSE (POST requests, headers, etc.)\n}\n\nexport function setupMock(schema: MockSchema) {\n if (typeof window !== 'undefined') {\n installFetchMock(schema);\n installSSEMock();\n }\n}\n","\nimport { ApiRequestConfig } from '@chatbi-v/core';\n\n// === Base Types ===\n\nexport type MockMethod = 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH';\nexport type MockType = 'json' | 'sse' | 'sse-page';\n\nexport interface MockEvent {\n event: string;\n data: any;\n /** Delay in ms, or a mockjs template string like '@increment(100)' */\n delay?: number | string;\n}\n\n// === Unified Strategy Interface ===\n\nexport interface MockStrategy {\n /**\n * Process the schema and return the result.\n * For JSON: returns the mocked object.\n * For SSE: returns an array of formatted event strings (chunks).\n */\n process(config: MockConfig, requestParams?: any): any;\n}\n\n// === Mock Configuration Types (Matching chat.mock.ts) ===\n\nexport interface BaseMockConfig {\n /** Optional: URL pattern if used in global interceptor */\n url?: string;\n /** Optional: Method if used in global interceptor */\n method?: MockMethod;\n /** Global delay for the request */\n delay?: number;\n /** Optional: HTTP status code (default: 200) */\n status?: number;\n /** Mock type (default: 'json') */\n type?: MockType;\n}\n\nexport interface JsonMockConfig extends BaseMockConfig {\n type?: 'json';\n /** The MockJS template for the response body */\n responseSchema: any;\n}\n\nexport interface SseMockConfig extends BaseMockConfig {\n type: 'sse';\n /** \n * Dictionary of event templates. \n * Key can be a mockjs template like 'data|3-5' \n */\n responseSchema: Record<string, MockEvent[]>;\n}\n\nexport interface SsePageMockConfig extends BaseMockConfig {\n type: 'sse-page';\n /** Dictionary of event templates */\n responseSchema: Record<string, MockEvent[]>;\n /** Special event for pagination metadata */\n pageEvent: MockEvent;\n}\n\nexport type MockConfig = JsonMockConfig | SseMockConfig | SsePageMockConfig;\n\nexport type MockSchema = Record<string, MockConfig>;\n\n// === Legacy Support (Keep these for backward compatibility if needed) ===\n\nexport interface ChatStreamConfig {\n _type: 'chat_stream';\n agentName?: string;\n plan?: string[];\n data?: Record<string, any>;\n timeline?: Array<{\n log?: string | { content: string; level?: 'info' | 'warning' | 'error'; agent?: string };\n planIndex?: number;\n delay?: number;\n }>;\n content: string | Record<string, any> | ((data: any) => string | Record<string, any>);\n}\n\nexport interface HistoryStreamConfig {\n _type: 'history_stream';\n template: Record<string, any>;\n prepend?: any[];\n}\n\nexport type AdvancedMockSchema = ChatStreamConfig | HistoryStreamConfig;\n\nexport interface MockGeneratorStrategy {\n generate(schema: any): string[];\n}\n\n// === Helper Functions (Restored) ===\n\nexport function createChatStream(config: Omit<ChatStreamConfig, '_type'>): ChatStreamConfig {\n return { _type: 'chat_stream', ...config };\n}\n\nexport function createHistoryStream(config: Omit<HistoryStreamConfig, '_type'>): HistoryStreamConfig {\n return { _type: 'history_stream', ...config };\n}\n"],"mappings":"AAMA,OAAS,aAAAA,MAAiB,iBCN1B,OAIE,gBAAAC,EACA,aAAAC,MAEK,iBACP,OAAOC,MAAU,SCRjB,OAAOC,MAAU,SCAjB,OAAS,aAAAC,MAAiB,iBAMnB,IAAMC,EAAN,KAA4B,CACzB,OAAmB,CAAC,EACpB,UACA,UACA,aAAuF,CAAC,EAEhG,YAAYC,EAAoB,YAAaC,EAAoB,CAC/D,KAAK,UAAYA,GAAc,QAAUH,EAAU,IAAI,EACvD,KAAK,UAAYE,CACnB,CAKA,YAAYE,EAAiF,CAC3F,OAAAA,EAAQ,QAAQC,GAAO,CACrB,IAAMC,EAAO,CACX,QAASD,EAAI,QACb,UAAW,KAAK,UAChB,KAAMA,EAAI,KACV,UAAW,GACX,UAAW,KAAK,UAChB,WAAYA,EAAI,UAClB,EACA,KAAK,UAAU,OAAQC,CAAI,EAEvBD,EAAI,OACN,KAAK,UAAU,QAAS,CAAE,MAAOA,EAAI,KAAM,CAAC,CAEhD,CAAC,EACM,IACT,CAMA,SAASE,EAAiB,CACxB,YAAK,aAAeA,EAAM,IAAIC,IAAS,CACrC,QAASA,EACT,OAAQ,SACV,EAAE,EACF,KAAK,UAAU,EACR,IACT,CAMA,iBAAiBC,EAAqB,CACpC,YAAK,aAAe,KAAK,aAAa,IAAI,CAACC,EAAMC,IAC3CA,EAAQF,EACH,CAAE,GAAGC,EAAM,OAAQ,WAAY,EAC7BC,IAAUF,EACZ,CAAE,GAAGC,EAAM,OAAQ,aAAc,EAEjC,CAAE,GAAGA,EAAM,OAAQ,SAAU,CAEvC,EACD,KAAK,UAAU,EACR,IACT,CAKA,cAAe,CACb,YAAK,aAAe,KAAK,aAAa,IAAIA,IAAS,CAAE,GAAGA,EAAM,OAAQ,WAAY,EAAE,EACpF,KAAK,UAAU,EACR,IACT,CAQA,OAAOE,EAAiBC,EAAqC,OAAQX,EAAoB,CACvF,YAAK,UAAU,MAAO,CACpB,KAAAW,EACA,QAAAD,EACA,UAAWV,GAAa,KAAK,SAC/B,CAAC,EACM,IACT,CAOA,SAASY,EAAmBC,EAAsB,CAChD,YAAK,UAAU,QAAS,CACtB,UAAAD,EACA,aAAAC,CACF,CAAC,EACM,IACT,CAOA,cAAcH,EAAiBI,EAAoB,GAAI,CACrD,QAASC,EAAI,EAAGA,EAAIL,EAAQ,OAAQK,GAAKD,EAAW,CAClD,IAAME,EAAQN,EAAQ,MAAMK,EAAGA,EAAID,CAAS,EAC5C,KAAK,UAAU,OAAQ,CACrB,QAASE,EACT,UAAW,KAAK,UAChB,UAAW,GACX,UAAW,KAAK,SAClB,CAAC,CACH,CACA,OAAO,IACT,CAKA,QAAS,CACP,YAAK,UAAU,OAAQ,CACrB,QAAS,GACT,UAAW,KAAK,UAChB,UAAW,GACX,UAAW,KAAK,SAClB,CAAC,EACM,IACT,CAMA,SAASC,EAAgC,CACvC,IAAMP,EAAU,KAAK,UAAUO,EAAW,KAAM,CAAC,EACjD,OAAO,KAAK,cAAcP,CAAO,CACnC,CAKA,UAAW,CACT,OAAO,KAAK,OAAO,KAAK;AAAA,CAAI,CAC9B,CAEQ,WAAY,CAClB,KAAK,UAAU,QAAS,CAAE,MAAO,KAAK,YAAa,CAAC,CACtD,CAEQ,UAAUC,EAAcP,EAAW,CACzC,KAAK,OAAO,KAAK,UAAUO,CAAI;AAAA,QAAW,KAAK,UAAUP,CAAI,CAAC;AAAA,CAAI,CACpE,CACF,ECjKA,OAAOc,MAAW,QAClB,OAAOC,MAAU,SAIV,IAAMC,EAASC,GAAe,IAAI,QAASC,GAAM,WAAWA,EAAGD,CAAE,CAAC,EAMlE,SAASE,EAAgBC,EAAWC,EAAmC,CAC5E,GAAI,OAAOD,GAAS,SAAU,CAE5B,GAAI,eAAe,KAAKA,CAAI,EAAG,CAC7B,IAAME,EAAaF,EAAK,MAAM,EAAG,EAAE,EAAE,KAAK,EAC1C,GAAI,CAEF,IAAMG,EAAO,OAAO,KAAKF,CAAO,EAC1BG,EAAS,OAAO,OAAOH,CAAO,EAEpC,OADW,IAAI,SAAS,GAAGE,EAAM,UAAUD,CAAU,EAAE,EAC7C,GAAGE,CAAM,CACrB,OAASC,EAAG,CACV,eAAQ,KAAK,uCAAuCL,CAAI,GAAIK,CAAC,EACtDL,CACT,CACF,CACA,OAAOA,CACT,CAEA,GAAI,MAAM,QAAQA,CAAI,EACpB,OAAOA,EAAK,IAAIM,GAAQP,EAAgBO,EAAML,CAAO,CAAC,EAGxD,GAAI,OAAOD,GAAS,UAAYA,IAAS,KAAM,CAC7C,IAAMO,EAA8B,CAAC,EACrC,QAAWC,KAAOR,EAChBO,EAAOC,CAAG,EAAIT,EAAgBC,EAAKQ,CAAG,EAAGP,CAAO,EAElD,OAAOM,CACT,CAEA,OAAOP,CACT,CASO,SAASS,EACdC,EACAC,EAAmB,CAAC,EACP,CAEb,IAAMV,EAA+B,CACnC,OAAQU,EACR,MAAOA,EACP,OAAQA,GAAa,OAASA,CAChC,EAGAhB,EAAK,OAAO,OAAO,CACjB,OAAQ,IAAMM,EAAQ,OACtB,MAAO,IAAMA,EAAQ,MACrB,OAAQ,IAAMA,EAAQ,MACxB,CAAC,EAGD,IAAMW,EAAWjB,EAAK,KAAKe,CAAY,EAGvC,OAAO,OAAO,OAAOE,CAAQ,EAC1B,KAAK,EACL,IAAIC,GAASd,EAAgBc,EAAOZ,CAAO,CAAC,EAC5C,KAAK,CAACa,EAAGC,IAAM,CACZ,IAAMC,EAAS,OAAOF,EAAE,OAAU,SAAWA,EAAE,MAAQ,SAAS,OAAOA,EAAE,OAAS,CAAC,CAAC,EAC9EG,EAAS,OAAOF,EAAE,OAAU,SAAWA,EAAE,MAAQ,SAAS,OAAOA,EAAE,OAAS,CAAC,CAAC,EACpF,OAAOC,EAASC,CACpB,CAAC,CACL,CAKO,SAASC,EAAeC,EAAqC,CAClE,IAAMC,EAAU,IAAI,YAEpB,OAAO,IAAI,eAAe,CACxB,MAAMC,EAAM,CAEV,GAAI,CAACF,GAAUA,EAAO,SAAW,EAAG,CAClCE,EAAK,MAAM,EACX,MACF,EAEC,SAAY,CACX,GAAI,CAGA,IAAMC,EAAY5B,EAAM,EAAE,QAAQ,EAElC,QAAW6B,KAAaJ,EAAQ,CAC5B,IAAMK,EAAQ,OAAOD,EAAU,OAAU,SAAWA,EAAU,MAAQ,SAAS,OAAOA,EAAU,OAAS,CAAC,CAAC,EAGrGE,EAAU/B,EAAM,EAAE,QAAQ,EAAI4B,EAC9BI,EAAY,KAAK,IAAI,EAAGF,EAAQC,CAAO,EAEzCC,EAAY,GACZ,MAAM9B,EAAM8B,CAAS,EAGzB,IAAMC,EAAU,UAAUJ,EAAU,KAAK;AAAA,QAAW,KAAK,UAAUA,EAAU,IAAI,CAAC;AAAA;AAAA,EAClFF,EAAK,QAAQD,EAAQ,OAAOO,CAAO,CAAC,CACxC,CACAN,EAAK,MAAM,CACf,OAAShB,EAAG,CACRgB,EAAK,MAAMhB,CAAC,CAChB,CACF,GAAG,CACL,CACF,CAAC,CACH,CF5GO,IAAMuB,EAAN,KAA0D,CAC/D,SAASC,EAAuB,CAC9B,IAAMC,EAASD,EACTE,EAAcD,EAAO,WAAa,iBAClCE,EAAY,IAAIC,EAAsBF,CAAW,EAEjDG,EAAWJ,EAAO,KAAOK,EAAK,KAAKL,EAAO,IAAI,EAAI,CAAC,EAGrDA,EAAO,MACTE,EAAU,SAASF,EAAO,IAAI,EAI5BA,EAAO,UACTA,EAAO,SAAS,QAAQM,GAAQ,CAI9B,GAHIA,EAAK,YAAc,QACrBJ,EAAU,iBAAiBI,EAAK,SAAS,EAEvCA,EAAK,IAAK,CACZ,IAAMC,EAAa,OAAOD,EAAK,KAAQ,SAAWA,EAAK,IAAMA,EAAK,IAAI,QAChEE,EAAW,OAAOF,EAAK,KAAQ,SAAWA,EAAK,IAAI,MAAQ,OAC3DG,EAAW,OAAOH,EAAK,KAAQ,SAAWA,EAAK,IAAI,MAAQ,OACjEJ,EAAU,OAAOK,EAAYC,EAAUC,CAAQ,CACjD,CACF,CAAC,EAIH,IAAIC,EACJ,OAAI,OAAOV,EAAO,SAAY,WAC5BU,EAAUV,EAAO,QAAQI,CAAQ,EAEjCM,EAAUV,EAAO,QAGf,OAAOU,GAAY,UAAYA,IAAY,KAC5CR,EAAU,SAASQ,CAA8B,EAEjDR,EAAU,cAAcQ,CAAiB,EAI5CR,EAAU,aAAa,EACvBA,EAAU,OAAO,EAEVA,EAAU,SAAS,EAAE,MAAM;AAAA;AAAA,CAAM,EAAE,IAAIS,GAASA,EAAQ;AAAA;AAAA,CAAM,CACvE,CACF,EAEaC,EAAN,KAA6D,CAClE,SAASb,EAAuB,CAC9B,IAAMC,EAASD,EACTG,EAAY,IAAIC,EAAsB,gBAAM,EAE5CU,EAAcR,EAAK,KAAKL,EAAO,QAAQ,EACvCc,EAAc,CAAC,GAAId,EAAO,SAAW,CAAC,EAAI,GAAIa,EAAY,MAAQA,CAAY,EACpF,OAAAX,EAAU,YAAYY,CAAW,EAE1BZ,EAAU,SAAS,EAAE,MAAM;AAAA;AAAA,CAAM,EAAE,IAAIS,GAASA,EAAQ;AAAA;AAAA,CAAM,CACvE,CACF,EAIaI,EAAN,KAA2C,CAChD,QAAQf,EAAoBgB,EAA2B,CACrD,IAAMC,EAAajB,EACnB,OAAKiB,EAAW,eACTZ,EAAK,KAAKY,EAAW,cAAc,EADH,CAAC,CAE1C,CACF,EAEaC,EAAN,KAA0C,CAC/C,QAAQlB,EAAoBmB,EAAqB,CAAC,EAAgB,CAChE,IAAMC,EAAYpB,EAClB,OAAKoB,EAAU,eAGRC,EAAWD,EAAU,eAAgBD,CAAa,EAHnB,CAAC,CAIzC,CACF,EAEaG,EAAN,KAA8C,CACnD,QAAQtB,EAAoBmB,EAAqB,CAAC,EAAgB,CAChE,IAAMI,EAAgBvB,EACtB,GAAI,CAACuB,EAAc,eAAgB,MAAO,CAAC,EAG3C,IAAMC,EAASH,EAAWE,EAAc,eAAgBJ,CAAa,EAGrE,GAAII,EAAc,UAAW,CAC3B,IAAME,EAAU,CACd,OAAQN,EACR,MAAOA,EACP,OAAQA,GAAe,OAASA,CAClC,EAEAd,EAAK,OAAO,OAAO,CACjB,OAAQ,IAAMoB,EAAQ,OACtB,MAAO,IAAMA,EAAQ,MACrB,OAAQ,IAAMA,EAAQ,MACxB,CAAC,EAGD,IAAIC,EAAYrB,EAAK,KAAKkB,EAAc,SAAS,EAEjDG,EAAYC,EAAgBD,EAAWD,CAAO,EAG9C,IAAIG,EAAW,EACf,GAAIJ,EAAO,OAAS,EAAG,CACrB,IAAMK,EAAYL,EAAOA,EAAO,OAAS,CAAC,EAC1CI,EAAW,OAAOC,EAAU,OAAU,SAAWA,EAAU,MAAQ,SAAS,OAAOA,EAAU,OAAS,CAAC,CAAC,CAC1G,EAGkB,OAAOH,EAAU,OAAU,SAAWA,EAAU,MAAQ,SAAS,OAAOA,EAAU,OAAS,CAAC,CAAC,IAC9FE,IACfF,EAAU,MAAQE,EAAW,IAG/BJ,EAAO,KAAKE,CAAS,CACvB,CAGA,IAAMI,EAAKN,EAAO,KAAK,CAACO,EAAGC,IAAM,CAC7B,IAAMC,EAAS,OAAOF,EAAE,OAAU,SAAWA,EAAE,MAAQ,SAAS,OAAOA,EAAE,OAAS,CAAC,CAAC,EAC9EG,EAAS,OAAOF,EAAE,OAAU,SAAWA,EAAE,MAAQ,SAAS,OAAOA,EAAE,OAAS,CAAC,CAAC,EACpF,OAAOC,EAASC,CACpB,CAAC,EACD,eAAQ,IAAIJ,CAAE,EACPA,CACT,CACF,EAGaK,EAAkB,CAC7B,YAAYC,EAAe,OAAsB,CAC/C,OAAQA,EAAM,CACZ,IAAK,MACH,OAAO,IAAIlB,EACb,IAAK,WACH,OAAO,IAAII,EAEb,QACE,OAAO,IAAIP,CACf,CACF,CACF,EDzJA,IAAMsB,EAASC,EAAa,aAAa,EAM5BC,EAAN,MAAMC,CAAkC,CACrC,MAER,OAAe,WAAoD,CACjE,YAAa,IAAIC,EACjB,eAAgB,IAAIC,CACtB,EAMA,YAAYC,EAAgB,IAAK,CAC/B,KAAK,MAAQA,CACf,CAOA,OAAO,iBAAiBC,EAAaC,EAAiC,CACpE,KAAK,WAAWD,CAAG,EAAIC,CACzB,CAQA,MAAM,QAAiBC,EAA0BC,EAAgD,CAE/F,IAAMJ,EAASI,GAAwB,OAAS,KAAK,MAErD,OAAO,IAAI,QAAQ,CAACC,EAASC,IAAW,CACtC,WAAW,IAAM,CACf,GAAI,CACF,GAAI,CAACF,GAAkB,CAACA,EAAe,eAAgB,CACrDV,EAAO,KAAK,2DAAcS,EAAO,GAAG,EAAE,EACtCE,EAAQ,CAAC,CAAM,EACf,MACF,CAEA,IAAIE,EAASH,EAAe,eACxBI,EAGA,OAAOD,GAAW,aACpBA,EAASA,EAAOJ,CAAM,GAIxB,IAAMM,EAAkB,CACtB,KAAOL,EAAuB,KAC9B,OAASA,EAAuB,QAAU,IAC1C,UAAYA,EAAuB,UACnC,eAAgBG,EAChB,GAAI,OAAOA,GAAW,SAAWA,EAAS,CAAC,CAC7C,EAGA,GAAIE,EAAgB,SAAW,IAAK,CAClC,IAAMC,EAAa,IAAI,MAAM,mCAAmCD,EAAgB,MAAM,EAAE,EACxFC,EAAM,SAAW,CACf,OAAQD,EAAgB,OACxB,KAAME,EAAK,KAAKJ,CAAM,EACtB,QAAS,CAAC,CACZ,EACAD,EAAOI,CAAK,EACZ,MACF,CAGA,GAAI,OAAOH,GAAW,SACpBC,EAAWD,UAGJ,KAAK,eAAeA,CAAM,EAAG,CACpC,IAAML,EAAWL,EAAY,WAAWU,EAAO,KAAK,EACpD,GAAIL,EAAU,CAGZM,EAFeN,EAAS,SAASK,CAAM,EAGpC,KAAK,EAAE,EACP,QAAQ,uBAAwB,EAAE,EAClC,QAAQ,QAAS,EAAE,EACtB,GAAI,CACFC,EAAW,KAAK,MAAMA,CAAQ,CAChC,MAAY,CAEZ,CACF,MACEd,EAAO,KAAK,iEAAea,EAAO,KAAK,EAAE,EACzCC,EAAW,CAAC,CAEhB,KAEK,CACH,IAAMI,EAAOH,EAAgB,MAAQ,OAErCD,EADiBK,EAAgB,YAAYD,CAAI,EAC7B,QAAQH,EAA+BN,EAAO,QAAUA,EAAO,IAAI,CACzF,CAEAT,EAAO,KAAK,YAAYS,EAAO,MAAM,IAAIA,EAAO,GAAG,GAAIA,EAAO,MAAQA,EAAO,MAAM,EACnFT,EAAO,KAAK,YAAac,CAAQ,EAEjCH,EAAQG,CAAa,CACvB,OAASE,EAAO,CACdJ,EAAOI,CAAK,CACd,CACF,EAAGV,CAAK,CACV,CAAC,CACH,CAQA,MAAM,OACJG,EACAW,EACAV,EACe,CACf,GAAM,CAAE,UAAAW,EAAW,SAAAC,EAAU,QAAAC,CAAQ,EAAIH,EACnCI,EAASf,EAAO,OAGhBH,EAASI,GAAwB,OAAS,KAAK,MAErD,OAAO,IAAI,QAAc,CAACC,EAASC,IAAW,CAE5C,GAAIY,GAAUA,EAAO,QAAS,CACxBF,GAAUA,EAAS,EACvBX,EAAQ,EACR,MACF,CAEA,WAAW,SAAY,CACrB,GAAI,CACF,GAAI,CAACD,GAAkB,CAACA,EAAe,eAAgB,CACrDV,EAAO,KAAK,2DAAcS,EAAO,GAAG,EAAE,EAClCa,GAAUA,EAAS,EACvBX,EAAQ,EACR,MACF,CAEA,IAAMc,EAAeC,EAAU,IAAI,EACnC1B,EAAO,KAAK,wBAAwBS,EAAO,MAAM,IAAIA,EAAO,GAAG,GAAI,CACjE,OAAQA,EAAO,MAAQA,EAAO,OAC9B,KAAMiB,EAAU,MAAM,EAAE,YAAY,CACtC,CAAC,EAED,IAAIb,EAASH,EAAe,eAGxB,OAAOG,GAAW,aACpBA,EAASA,EAAOJ,CAAM,GAIxB,IAAMM,EAAkB,CACtB,KAAOL,EAAuB,KAC9B,OAASA,EAAuB,QAAU,IAC1C,UAAYA,EAAuB,UACnC,eAAgBG,EAChB,GAAI,OAAOA,GAAW,SAAWA,EAAS,CAAC,CAC7C,EAGA,GAAIE,EAAgB,SAAW,IAAK,CAClC,IAAMY,EAAW,CACf,OAAQZ,EAAgB,OACxB,KAAME,EAAK,KAAKJ,CAAM,EACtB,QAAS,CAAC,CACZ,EAGA,GAAIO,EAAU,YACK,MAAMA,EAAU,WAAWO,CAAQ,EACtC,CACZhB,EAAQ,EACR,MACF,CAGF,IAAMK,EAAa,IAAI,MAAM,0CAA0CD,EAAgB,MAAM,EAAE,EAC/FC,EAAM,SAAWW,EACjB3B,EAAO,MAAM,0CAA0Ce,EAAgB,MAAM,GAAIC,CAAK,EAClFO,GAASA,EAAQP,CAAK,EAE1BJ,EAAOI,CAAK,EACZ,MACF,CAGA,GAAII,EAAU,YACK,MAAMA,EAAU,WAAW,CAC1C,OAAQ,IACR,KAAM,KACN,QAAS,CAAC,CACZ,CAAC,EACa,CACZT,EAAQ,EACR,MACF,CAIF,GAAI,KAAK,YAAYI,CAAe,EAAG,CACrC,IAAMG,EAAOH,EAAgB,MAAQ,MAI/Ba,EAHWT,EAAgB,YAAYD,CAAI,EAGzB,QACtBH,EACAN,EAAO,QAAUA,EAAO,IAC1B,EAEA,GAAI,MAAM,QAAQmB,CAAM,EAAG,CACzB,IAAMC,EAAYH,EAAU,IAAI,EAC5BI,EAAa,EAEjB9B,EAAO,KACL,8BAA8B4B,EAAO,MAAM,eAAeV,CAAI,WAChE,EAEA,QAAWa,KAASH,EAAQ,CAC1B,GAAIJ,GAAUA,EAAO,QAAS,CAC5BxB,EAAO,KACL,4CAA4C0B,EAAU,IAAI,EAAID,CAAY,IAC5E,EACA,KACF,CAGA,IAAMnB,EACJ,OAAOyB,EAAM,OAAU,SACnBA,EAAM,MACN,SAAS,OAAOA,EAAM,OAAS,CAAC,CAAC,EACjCC,EAAUN,EAAU,IAAI,EAAIG,EAC5BI,EAAY,KAAK,IAAI,EAAG3B,EAAQ0B,CAAO,EAM7C,GAJIC,EAAY,GACd,MAAMC,EAAMD,CAAS,EAGnBT,GAAUA,EAAO,QAAS,MAG9B,IAAMW,EAAQ,UAAUJ,EAAM,KAAK;AAAA,QAAW,KAAK,UAAUA,EAAM,IAAI,CAAC;AAAA;AAAA,EAGpE,CAAC,QAAS,QAAS,MAAM,EAAE,SAASA,EAAM,KAAK,EACjD/B,EAAO,KAAK,uCAAuC+B,EAAM,KAAK,GAAIA,EAAM,IAAI,GACnED,IAAe,GAAKA,IAAeF,EAAO,OAAS,IAC5D5B,EAAO,KACL,wBAAwB8B,IAAe,EAAI,QAAU,MAAM,cAC3D,CAAE,QAAS,KAAK,UAAUC,EAAM,IAAI,EAAE,MAAM,EAAG,EAAE,EAAI,KAAM,CAC7D,EAEEV,GAAWA,EAAUc,CAAK,EAC9BL,GACF,CACF,CACF,KAEK,CACH,IAAIM,EAAmB,CAAC,EAExB,GAAI,OAAOvB,GAAW,SACpBuB,EAASvB,EAAO,MAAM;AAAA;AAAA,CAAM,EAAE,IAAKsB,GAAUA,EAAQ;AAAA;AAAA,CAAM,UAClD,KAAK,eAAetB,CAAM,EACnCuB,EAAS,KAAK,uBAAuBvB,CAAM,MACtC,CACL,IAAMC,EAAWG,EAAK,KAAKJ,CAAM,EACjCuB,EAAS,CAAC,SAAS,KAAK,UAAUtB,CAAQ,CAAC;AAAA;AAAA,CAAM,CACnD,CAEA,QAAWqB,KAASC,EAAQ,CAC1B,GAAIZ,GAAUA,EAAO,QAAS,CAC5BxB,EAAO,KAAK,oCAAoC,EAChD,KACF,CAEImC,EAAM,KAAK,IACTd,GAAWA,EAAUc,CAAK,EAC9B,MAAM,IAAI,QAASE,GAAM,WAAWA,EAAG,GAAG,CAAC,EAE/C,CACF,EAEI,CAACb,GAAU,CAACA,EAAO,WACrBxB,EAAO,KACL,kDAAkD0B,EAAU,IAAI,EAAID,CAAY,IAClF,EACIH,GAAUA,EAAS,GAEzBX,EAAQ,CACV,OAASK,EAAO,CACdhB,EAAO,MAAM,uCAAwCgB,CAAK,EACtDO,GAASA,EAAQP,CAAK,EAC1BL,EAAQ,CACV,CACF,EAAGL,CAAK,CACV,CAAC,CACH,CAKQ,eAAeO,EAAsB,CAC3C,OAAOA,GAAU,OAAOA,GAAW,UAAY,UAAWA,CAC5D,CAMQ,YAAYJ,EAAsB,CACxC,OACEA,GACA,OAAOA,GAAW,WACjB,CAAC,OAAQ,MAAO,UAAU,EAAE,SAASA,EAAO,IAAI,GAAK,mBAAoBA,EAE9E,CAKQ,uBAAuBI,EAAuB,CACpD,IAAML,EAAWL,EAAY,WAAWU,EAAO,KAAK,EACpD,OAAIL,EACKA,EAAS,SAASK,CAAM,GAEjCb,EAAO,KAAK,iEAAea,EAAO,KAAK,EAAE,EAClC,CAAC,EACV,CACF,EI3VO,SAASyB,EAAiBC,EAAoB,CAC9C,OAAO,iBACV,OAAO,eAAiB,OAAO,OAGjC,OAAO,MAAQ,MAAOC,EAA0BC,EAAoB,CAAC,IAAM,CACzE,IAAMC,EAAM,OAAOF,GAAU,SAAWA,EAAQA,aAAiB,IAAMA,EAAM,SAAS,EAAIA,EAAM,IAC1FG,GAAUF,EAAK,QAAU,OAAO,YAAY,EAI5CG,EADQ,OAAO,OAAOL,CAAM,EACf,KAAMM,GAAM,CAC3B,GAAI,CAACA,EAAE,IAAK,MAAO,GAGnB,IAAMC,EAAUD,EAAE,IAAI,QAAQ,kBAAmB,SAAS,EACpDE,EAAQ,IAAI,OAAO,IAAID,CAAO,GAAG,EAGjC,CAACE,CAAI,EAAIN,EAAI,MAAM,GAAG,EAGtBO,GAAcJ,EAAE,QAAU,OAAO,YAAY,EACnD,OAAOE,EAAM,KAAKC,CAAI,GAAKC,IAAeN,CAC9C,CAAC,EAED,GAAI,CAACC,EACH,OAAO,OAAO,eAAeJ,EAAOC,CAAI,EAG1C,QAAQ,IAAI,sBAAsBE,CAAM,IAAID,CAAG,GAAIE,CAAI,EAGvD,IAAMM,EAASN,EAAK,QAAU,IAC9B,GAAIM,IAAW,IACb,aAAMC,EAAMP,EAAK,OAAS,CAAC,EACpB,IAAI,SAAS,KAAK,UAAU,CACjC,QAAS,qBAAqBM,CAAM,GACpC,KAAMA,CACR,CAAC,EAAG,CACF,OAAAA,EACA,QAAS,CAAE,eAAgB,kBAAmB,CAChD,CAAC,EAGH,MAAMC,EAAMP,EAAK,OAAS,CAAC,EAG3B,IAAMQ,EAAOR,EAAK,MAAQ,OACpBS,EAAWC,EAAgB,YAAYF,CAAI,EAG3CG,EAAS,IAAI,IAAIb,EAAK,OAAO,SAAS,MAAM,EAC5Cc,EAAQ,OAAO,YAAYD,EAAO,YAAY,EAE9CE,EAASJ,EAAS,QAAQT,EAAoBY,CAAK,EAGzD,GAAIJ,IAAS,OACX,OAAO,IAAI,SAAS,KAAK,UAAUK,CAAM,EAAG,CAC1C,OAAAP,EACA,QAAS,CAAE,eAAgB,kBAAmB,CAChD,CAAC,EAIH,GAAIE,IAAS,OAASA,IAAS,WAAY,CAEzC,IAAMM,EAASC,EADAF,CACqB,EACpC,OAAO,IAAI,SAASC,EAAQ,CAC1B,OAAAR,EACA,QAAS,CAAE,eAAgB,mBAAoB,CACjD,CAAC,CACH,CAEA,OAAO,OAAO,eAAeV,EAAOC,CAAI,CAC1C,CACF,CAEO,SAASmB,IAAiB,CAIjC,CAEO,SAASC,GAAUtB,EAAoB,CACxC,OAAO,OAAW,MACpBD,EAAiBC,CAAM,EACvB,OAEJ,CCNO,SAASuB,GAAiBC,EAA2D,CAC1F,MAAO,CAAE,MAAO,cAAe,GAAGA,CAAO,CAC3C,CAEO,SAASC,GAAoBD,EAAiE,CACnG,MAAO,CAAE,MAAO,iBAAkB,GAAGA,CAAO,CAC9C,CNhFO,IAAME,EAAY,CACvB,GAAI,QACJ,KAAM,aACN,OAAQ,wDACR,KAAM,OACR,EAGaC,EAAgB,CAC3B,CAAE,GAAI,QAAS,MAAO,mCAAgB,KAAM,aAAc,YAAa,uFAAuB,EAC9F,CAAE,GAAI,QAAS,MAAO,uCAAU,KAAM,aAAc,YAAa,qDAAc,EAC/E,CAAE,GAAI,QAAS,MAAO,uCAAU,KAAM,aAAc,YAAa,8DAAkB,EACnF,CAAE,GAAI,QAAS,MAAO,uCAAU,KAAM,aAAc,YAAa,+BAAiB,EAClF,CAAE,GAAI,QAAS,MAAO,uCAAU,KAAM,aAAc,YAAa,2DAAe,CAClF,EAEaC,EAAiB,CAC5B,CAAE,GAAI,QAAS,MAAO,uCAAU,KAAM,YAAa,EACnD,CAAE,GAAI,QAAS,MAAO,uCAAU,KAAM,YAAa,EACnD,CAAE,GAAI,QAAS,MAAO,uCAAU,KAAM,YAAa,CACrD,EAEaC,EAAc,CACzB,CACE,GAAI,IACJ,KAAM,uCACN,YAAa,uIACb,KAAM,YACN,QAAS,qBACT,WAAY,aACZ,WAAY,EACZ,QAAS,EACT,QAAS,GACT,MAAO,sEACT,EACA,CACE,GAAI,IACJ,KAAM,uCACN,YAAa,qHACb,KAAM,SACN,QAAS,kBACT,WAAY,aACZ,WAAY,EACZ,QAAS,EACT,QAAS,EACT,MAAO,yEACT,EACA,CACE,GAAI,IACJ,KAAM,iCACN,YAAa,yGACb,KAAM,UACN,QAAS,iCACT,WAAY,aACZ,WAAY,EACZ,QAAS,EACT,QAAS,GACT,MAAO,yEACT,CACF,EAEaC,GAAwBC,GAAsB,CACzD,CACE,IAAK,OAAOA,CAAS,KACrB,KAAM,YACN,QAAS,wGAAwBA,CAAS,qEAC1C,KAAMC,EAAU,MAAMA,EAAU,IAAI,EAAI,GAAK,EAAE,YAAY,CAC7D,CACF,EAEaC,GAAW,CACtB,KAAMP,EACN,SAAUC,EACV,UAAWC,EACX,OAAQC,EACR,cAAeD,CACjB","names":["dateUtils","createLogger","dateUtils","Mock","Mock","dateUtils","MockResponseGenerator","agentName","sessionId","history","msg","data","steps","step","activeIndex","todo","index","content","type","errorCode","errorMessage","chunkSize","i","chunk","component","dayjs","Mock","sleep","ms","r","processTemplate","data","context","expression","keys","values","e","item","result","key","flatEvents","eventsSchema","requestData","compiled","event","a","b","delayA","delayB","eventsToStream","events","encoder","ctrl","startTime","eventItem","delay","elapsed","remaining","payload","ChatStreamStrategy","schema","config","defaultName","generator","MockResponseGenerator","mockData","Mock","step","logContent","logLevel","logAgent","content","chunk","HistoryStreamStrategy","historyMock","fullHistory","JsonStrategy","_requestParams","jsonConfig","SseStrategy","requestParams","sseConfig","flatEvents","SsePageStrategy","ssePageConfig","events","context","pageEvent","processTemplate","maxDelay","lastEvent","es","a","b","delayA","delayB","StrategyFactory","type","logger","createLogger","MockAdapter","_MockAdapter","ChatStreamStrategy","HistoryStreamStrategy","delay","key","strategy","config","endpointConfig","resolve","reject","schema","mockData","effectiveConfig","error","Mock","type","StrategyFactory","callbacks","onMessage","onFinish","onError","signal","requestStart","dateUtils","response","events","startTime","eventCount","event","elapsed","remaining","sleep","chunk","chunks","r","installFetchMock","schema","input","init","url","method","rule","v","pattern","regex","path","ruleMethod","status","sleep","type","strategy","StrategyFactory","urlObj","query","result","stream","eventsToStream","installSSEMock","setupMock","createChatStream","config","createHistoryStream","MOCK_USER","MOCK_SESSIONS","MOCK_FAVORITES","MOCK_SCENES","generateMockMessages","sessionId","dateUtils","mockData"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/adapter.ts","../src/strategies.ts","../src/generator.ts","../src/utils.ts","../src/interceptor.ts","../src/types.ts"],"sourcesContent":["/**\n * @file index.ts\n * @description Mock 模块入口,导出模拟数据适配器、生成器及静态测试数据\n * @author ChatBI Team\n */\n\nimport { dateUtils } from '@chatbi-v/core';\n\n/**\n * Mock 数据模块\n * @description 提供前端开发所需的模拟数据适配器和工具\n */\n\nexport * from './adapter';\nexport * from './generator';\nexport * from './interceptor';\nexport * from './strategies';\nexport * from './types';\nexport * from './utils';\n\n/**\n * 模拟用户数据\n */\nexport const MOCK_USER = {\n id: 'u_001',\n name: 'Admin User',\n avatar: 'https://api.dicebear.com/7.x/avataaars/svg?seed=Admin',\n role: 'admin',\n};\n\n// 保留原有的静态数据导出,以兼容可能直接引用的地方,但建议使用 adapter 自动生成\nexport const MOCK_SESSIONS = [\n { id: 's_001', title: '2024 Q1 销售分析', date: '2024-03-15', lastMessage: '好的,正在为您分析 Q1 销售数据...' },\n { id: 's_002', title: '用户增长趋势', date: '2024-03-10', lastMessage: '用户增长曲线显示...' },\n { id: 's_003', title: '竞品分析报告', date: '2024-03-08', lastMessage: '主要竞品 A 的市场份额...' },\n { id: 's_004', title: '营销活动复盘', date: '2024-03-05', lastMessage: 'ROI 提升了 15%...' },\n { id: 's_005', title: '财务报表审计', date: '2024-03-01', lastMessage: '请确认以下财务指标...' },\n];\n\nexport const MOCK_FAVORITES = [\n { id: 'fav-1', title: '季度销售总结', date: '2025/12/15' },\n { id: 'fav-2', title: '年度预算规划', date: '2025/12/10' },\n { id: 'fav-3', title: '核心指标监控', date: '2025/11/20' },\n];\n\nexport const MOCK_SCENES = [\n {\n id: '1',\n name: '销售数据分析',\n description: '全方位分析销售业绩,包括区域、产品、时间维度',\n code: 'SALES_001',\n creator: '管理员',\n createTime: '2023-01-01',\n tableCount: 5,\n kbCount: 2,\n qaCount: 10,\n cover: 'https://images.unsplash.com/photo-1551288049-bebda4e38f71?w=800&q=80',\n },\n {\n id: '2',\n name: '人力资源概览',\n description: '员工入职、离职、绩效分布与培训情况分析',\n code: 'HR_001',\n creator: 'HR 经理',\n createTime: '2023-02-15',\n tableCount: 3,\n kbCount: 1,\n qaCount: 5,\n cover: 'https://images.unsplash.com/photo-1460925895917-afdab827c52f?w=800&q=80',\n },\n {\n id: '3',\n name: '供应链优化',\n description: '库存周转、物流成本与供应商绩效评估',\n code: 'SCM_001',\n creator: '供应链总监',\n createTime: '2023-03-20',\n tableCount: 8,\n kbCount: 0,\n qaCount: 15,\n cover: 'https://images.unsplash.com/photo-1586528116311-ad8dd3c8310d?w=800&q=80',\n },\n];\n\nexport const generateMockMessages = (sessionId: string) => [\n {\n key: `msg_${sessionId}_1`,\n role: 'assistant',\n content: `您好!我是您的智能助手。当前会话 ID: ${sessionId}。请问有什么可以帮您?`,\n time: dateUtils.dayjs(dateUtils.now() - 10000).toISOString(),\n },\n];\n\nexport const mockData = {\n user: MOCK_USER,\n sessions: MOCK_SESSIONS,\n favorites: MOCK_FAVORITES,\n scenes: MOCK_SCENES,\n favoritesList: MOCK_FAVORITES, // Alias for compatibility if needed\n};\n","import {\n ApiAdapter,\n ApiEndpointConfig,\n ApiRequestConfig,\n createLogger,\n dateUtils,\n StreamCallbacks,\n} from '@chatbi-v/core';\nimport Mock from 'mockjs';\n\nimport { ChatStreamStrategy, HistoryStreamStrategy, StrategyFactory } from './strategies';\nimport { MockConfig, MockGeneratorStrategy } from './types';\nimport { sleep } from './utils';\n\nconst logger = createLogger('MockAdapter');\n\n/**\n * 基于 Mock.js 的请求适配器\n * @description 根据 API 配置中的 responseSchema 自动生成 Mock 数据,支持普通 JSON 和流式 SSE 响应\n */\nexport class MockAdapter implements ApiAdapter {\n private delay: number;\n // 遗留策略(为了兼容旧版代码)\n private static strategies: Record<string, MockGeneratorStrategy> = {\n chat_stream: new ChatStreamStrategy(),\n history_stream: new HistoryStreamStrategy(),\n };\n\n /**\n * 构造函数\n * @param delay 全局模拟延迟时间(毫秒),默认 300ms\n */\n constructor(delay: number = 300) {\n this.delay = delay;\n }\n\n /**\n * 注册自定义 Mock 生成策略\n * @param key 策略唯一标识符 (e.g., 'chat_stream', 'history_stream')\n * @param strategy 实现了 MockGeneratorStrategy 接口的策略实例\n */\n static registerStrategy(key: string, strategy: MockGeneratorStrategy) {\n this.strategies[key] = strategy;\n }\n\n /**\n * 处理普通 HTTP 请求(非流式)\n * @param config 请求配置对象\n * @param endpointConfig API 端点配置,包含 responseSchema\n * @returns Promise 返回模拟的响应数据\n */\n async request<T = any>(config: ApiRequestConfig, endpointConfig?: ApiEndpointConfig): Promise<T> {\n // 优先使用接口定义的 delay,否则使用全局默认 delay\n const delay = (endpointConfig as any)?.delay ?? this.delay;\n\n return new Promise((resolve, reject) => {\n setTimeout(() => {\n try {\n if (!endpointConfig || !endpointConfig.responseSchema) {\n logger.warn(`未找到响应架构配置: ${config.url}`);\n resolve({} as T);\n return;\n }\n\n let schema = endpointConfig.responseSchema;\n let mockData;\n\n // 支持函数式 schema,允许根据请求参数动态生成 schema\n if (typeof schema === 'function') {\n schema = schema(config);\n }\n\n // 构造有效配置对象,合并 endpointConfig 顶层属性\n const effectiveConfig = {\n type: (endpointConfig as any).type,\n status: (endpointConfig as any).status || 200,\n pageEvent: (endpointConfig as any).pageEvent,\n responseSchema: schema,\n ...(typeof schema === 'object' ? schema : {}),\n };\n\n // 0. 处理非 200 状态码\n if (effectiveConfig.status !== 200) {\n const error: any = new Error(`Request failed with status code ${effectiveConfig.status}`);\n error.response = {\n status: effectiveConfig.status,\n data: Mock.mock(schema),\n headers: {},\n };\n reject(error);\n return;\n }\n\n // 1. 字符串 Schema:直接作为结果返回(极少情况)\n if (typeof schema === 'string') {\n mockData = schema;\n }\n // 2. 遗留的高级 Schema (通过 _type 字段识别)\n else if (this.isLegacySchema(schema)) {\n const strategy = MockAdapter.strategies[schema._type];\n if (strategy) {\n const chunks = strategy.generate(schema);\n // 将流式块合并为单个字符串,并清理 SSE 格式标记\n mockData = chunks\n .join('')\n .replace(/event: data\\ndata: /g, '')\n .replace(/\\n\\n/g, '');\n try {\n mockData = JSON.parse(mockData);\n } catch (e) {\n // 忽略解析错误,保持原始数据格式\n }\n } else {\n logger.warn(`未找到对应的策略类型: ${schema._type}`);\n mockData = {};\n }\n }\n // 3. 通用策略模式\n else {\n const type = effectiveConfig.type || 'json';\n const strategy = StrategyFactory.getStrategy(type);\n mockData = strategy.process(effectiveConfig as MockConfig, config.params || config.data);\n }\n\n logger.info(`Request: ${config.method} ${config.url}`, config.data || config.params);\n logger.info(`Response:`, mockData);\n\n resolve(mockData as T);\n } catch (error) {\n reject(error);\n }\n }, delay);\n });\n }\n\n /**\n * 处理流式请求 (SSE)\n * @param config 请求配置对象\n * @param callbacks 流式回调函数集合 (onMessage, onFinish, onError)\n * @param endpointConfig API 端点配置,包含 responseSchema\n */\n async stream(\n config: ApiRequestConfig,\n callbacks: StreamCallbacks,\n endpointConfig?: ApiEndpointConfig,\n ): Promise<void> {\n const { onMessage, onFinish, onError } = callbacks;\n const signal = config.signal;\n\n // 优先使用接口定义的 delay,否则使用全局默认 delay\n const delay = (endpointConfig as any)?.delay ?? this.delay;\n\n return new Promise<void>((resolve, reject) => {\n // 如果请求已被取消,直接结束\n if (signal && signal.aborted) {\n if (onFinish) onFinish();\n resolve();\n return;\n }\n\n setTimeout(async () => {\n try {\n if (!endpointConfig || !endpointConfig.responseSchema) {\n logger.warn(`未找到流式响应架构: ${config.url}`);\n if (onFinish) onFinish();\n resolve();\n return;\n }\n\n const requestStart = dateUtils.now();\n logger.info(`[SSE Start] Request: ${config.method} ${config.url}`, {\n params: config.data || config.params,\n time: dateUtils.dayjs().toISOString(),\n });\n\n let schema = endpointConfig.responseSchema;\n\n // 支持函数式 schema\n if (typeof schema === 'function') {\n schema = schema(config);\n }\n\n // 构造有效配置对象,合并 endpointConfig 顶层属性\n const effectiveConfig = {\n type: (endpointConfig as any).type,\n status: (endpointConfig as any).status || 200,\n pageEvent: (endpointConfig as any).pageEvent,\n responseSchema: schema,\n ...(typeof schema === 'object' ? schema : {}),\n };\n\n // 0. 处理非 200 状态码\n if (effectiveConfig.status !== 200) {\n const response = {\n status: effectiveConfig.status,\n data: Mock.mock(schema),\n headers: {},\n };\n\n // 触发 onResponse 回调\n if (callbacks.onResponse) {\n const hijacked = await callbacks.onResponse(response);\n if (hijacked) {\n resolve();\n return;\n }\n }\n\n const error: any = new Error(`Stream request failed with status code ${effectiveConfig.status}`);\n error.response = response;\n logger.error(`[SSE Error] Request failed with status ${effectiveConfig.status}`, error);\n if (onError) onError(error);\n // 保持与 request 同样的处理逻辑,抛出异常以便顶层拦截\n reject(error);\n return;\n }\n\n // 触发 onResponse 回调 (200 场景)\n if (callbacks.onResponse) {\n const hijacked = await callbacks.onResponse({\n status: 200,\n data: null,\n headers: {},\n });\n if (hijacked) {\n resolve();\n return;\n }\n }\n\n // 1. 新版 Schema 策略 (sse, sse-page)\n if (this.isNewConfig(effectiveConfig)) {\n const type = effectiveConfig.type || 'sse';\n const strategy = StrategyFactory.getStrategy(type);\n\n // 策略处理返回 MockEvent 数组\n const events = strategy.process(\n effectiveConfig as MockConfig,\n config.params || config.data,\n );\n\n if (Array.isArray(events)) {\n const startTime = dateUtils.now();\n let eventCount = 0;\n\n logger.info(\n `[SSE Processing] Generated ${events.length} events for ${type} strategy`,\n );\n\n for (const event of events) {\n if (signal && signal.aborted) {\n logger.info(\n `[SSE Abort] Stream aborted by user after ${dateUtils.now() - requestStart}ms`,\n );\n break;\n }\n\n // 计算延迟\n const delay =\n typeof event.delay === 'number'\n ? event.delay\n : parseInt(String(event.delay || 0));\n const elapsed = dateUtils.now() - startTime;\n const remaining = Math.max(0, delay - elapsed);\n\n if (remaining > 0) {\n await sleep(remaining);\n }\n\n if (signal && signal.aborted) break;\n\n // 构造 SSE 格式数据块\n const chunk = `event: ${event.event}\\ndata: ${JSON.stringify(event.data)}\\n\\n`;\n\n // 记录关键事件日志\n if (['error', 'todos', 'page'].includes(event.event)) {\n logger.info(`[SSE Event] Emitting special event: ${event.event}`, event.data);\n } else if (eventCount === 0 || eventCount === events.length - 1) {\n logger.info(\n `[SSE Event] Emitting ${eventCount === 0 ? 'first' : 'last'} data event`,\n { preview: JSON.stringify(event.data).slice(0, 50) + '...' },\n );\n }\n if (onMessage) onMessage(chunk);\n eventCount++;\n }\n }\n }\n // 2. 遗留策略处理\n else {\n let chunks: string[] = [];\n\n if (typeof schema === 'string') {\n chunks = schema.split('\\n\\n').map((chunk) => chunk + '\\n\\n');\n } else if (this.isLegacySchema(schema)) {\n chunks = this.generateAdvancedChunks(schema);\n } else {\n const mockData = Mock.mock(schema);\n chunks = [`data: ${JSON.stringify(mockData)}\\n\\n`];\n }\n\n for (const chunk of chunks) {\n if (signal && signal.aborted) {\n logger.info('[SSE Abort] Stream aborted by user');\n break;\n }\n\n if (chunk.trim()) {\n if (onMessage) onMessage(chunk);\n await new Promise((r) => setTimeout(r, 200));\n }\n }\n }\n\n if (!signal || !signal.aborted) {\n logger.info(\n `[SSE Complete] Stream finished successfully in ${dateUtils.now() - requestStart}ms`,\n );\n if (onFinish) onFinish();\n }\n resolve();\n } catch (error) {\n logger.error(`[SSE Error] Stream processing failed`, error);\n if (onError) onError(error);\n resolve();\n }\n }, delay);\n });\n }\n\n /**\n * 判断是否为遗留的 schema 格式\n */\n private isLegacySchema(schema: any): boolean {\n return schema && typeof schema === 'object' && '_type' in schema;\n }\n\n /**\n * 判断是否为新版配置格式\n * @description 检查配置对象中是否包含有效的 type 字段\n */\n private isNewConfig(config: any): boolean {\n return (\n config &&\n typeof config === 'object' &&\n (['json', 'sse', 'sse-page'].includes(config.type) || 'responseSchema' in config) // 备用检查,如果 schema 内部定义了结构\n );\n }\n\n /**\n * 生成遗留的高级 schema 数据块\n */\n private generateAdvancedChunks(schema: any): string[] {\n const strategy = MockAdapter.strategies[schema._type];\n if (strategy) {\n return strategy.generate(schema);\n }\n logger.warn(`未找到对应的策略类型: ${schema._type}`);\n return [];\n }\n}\n","import Mock from 'mockjs';\n\nimport { MockResponseGenerator } from './generator';\nimport { \n ChatStreamConfig,\n HistoryStreamConfig,\n JsonMockConfig, \n MockConfig, \n MockEvent,\n MockGeneratorStrategy, // Legacy\n MockStrategy, \n SseMockConfig, \n SsePageMockConfig} from './types';\nimport { flatEvents, processTemplate } from './utils';\n\n// === Legacy Strategies (Preserved) ===\n\nexport class ChatStreamStrategy implements MockGeneratorStrategy {\n generate(schema: any): string[] {\n const config = schema as ChatStreamConfig;\n const defaultName = config.agentName || 'BI助手';\n const generator = new MockResponseGenerator(defaultName);\n \n const mockData = config.data ? Mock.mock(config.data) : {};\n \n // 1. Init Plan\n if (config.plan) {\n generator.initPlan(config.plan);\n }\n\n // 2. Execute Timeline\n if (config.timeline) {\n config.timeline.forEach(step => {\n if (step.planIndex !== undefined) {\n generator.updatePlanStatus(step.planIndex);\n }\n if (step.log) {\n const logContent = typeof step.log === 'string' ? step.log : step.log.content;\n const logLevel = typeof step.log === 'object' ? step.log.level : 'info';\n const logAgent = typeof step.log === 'object' ? step.log.agent : undefined;\n generator.addLog(logContent, logLevel, logAgent);\n }\n });\n }\n\n // 3. Content\n let content;\n if (typeof config.content === 'function') {\n content = config.content(mockData);\n } else {\n content = config.content;\n }\n\n if (typeof content === 'object' && content !== null) {\n generator.emitA2UI(content as Record<string, any>);\n } else {\n generator.streamContent(content as string);\n }\n\n // 4. Complete\n generator.completePlan();\n generator.finish();\n\n return generator.toString().split('\\n\\n').map(chunk => chunk + '\\n\\n');\n }\n}\n\nexport class HistoryStreamStrategy implements MockGeneratorStrategy {\n generate(schema: any): string[] {\n const config = schema as HistoryStreamConfig;\n const generator = new MockResponseGenerator('BI助手');\n \n const historyMock = Mock.mock(config.template);\n const fullHistory = [...(config.prepend || []), ...(historyMock.list || historyMock)];\n generator.emitHistory(fullHistory);\n \n return generator.toString().split('\\n\\n').map(chunk => chunk + '\\n\\n');\n }\n}\n\n// === New Schema Strategies ===\n\nexport class JsonStrategy implements MockStrategy {\n process(config: MockConfig, _requestParams?: any): any {\n const jsonConfig = config as JsonMockConfig;\n if (!jsonConfig.responseSchema) return {};\n return Mock.mock(jsonConfig.responseSchema);\n }\n}\n\nexport class SseStrategy implements MockStrategy {\n process(config: MockConfig, requestParams: any = {}): MockEvent[] {\n const sseConfig = config as SseMockConfig;\n if (!sseConfig.responseSchema) return [];\n \n // Use flatEvents to process the schema with MockJS and sort by delay\n return flatEvents(sseConfig.responseSchema, requestParams);\n }\n}\n\nexport class SsePageStrategy implements MockStrategy {\n process(config: MockConfig, requestParams: any = {}): MockEvent[] {\n const ssePageConfig = config as SsePageMockConfig;\n if (!ssePageConfig.responseSchema) return [];\n\n // 1. Generate base events\n const events = flatEvents(ssePageConfig.responseSchema, requestParams);\n\n // 2. Generate page event if exists\n if (ssePageConfig.pageEvent) {\n const context = {\n $query: requestParams, // Backward compatibility\n $body: requestParams,\n $param: requestParams?.param || requestParams\n };\n\n Mock.Random.extend({ \n $query: () => context.$query,\n $body: () => context.$body,\n $param: () => context.$param\n });\n\n // First let Mock.js process standard templates (like @integer)\n let pageEvent = Mock.mock(ssePageConfig.pageEvent);\n // Then process custom {{}} templates\n pageEvent = processTemplate(pageEvent, context);\n \n // Calculate max delay from existing events to ensure page event is last\n let maxDelay = 0;\n if (events.length > 0) {\n const lastEvent = events[events.length - 1];\n maxDelay = typeof lastEvent.delay === 'number' ? lastEvent.delay : parseInt(String(lastEvent.delay || 0));\n }\n \n // Set page event delay to be slightly after the last event\n const pageDelay = typeof pageEvent.delay === 'number' ? pageEvent.delay : parseInt(String(pageEvent.delay || 0));\n if (pageDelay <= maxDelay) {\n pageEvent.delay = maxDelay + 20; // Add small buffer\n }\n\n events.push(pageEvent);\n }\n \n // Re-sort in case pageEvent has specific delay\n const es = events.sort((a, b) => {\n const delayA = typeof a.delay === 'number' ? a.delay : parseInt(String(a.delay || 0));\n const delayB = typeof b.delay === 'number' ? b.delay : parseInt(String(b.delay || 0));\n return delayA - delayB;\n });\n return es;\n }\n}\n\n// Factory for new strategies\nexport const StrategyFactory = {\n getStrategy(type: string = 'json'): MockStrategy {\n switch (type) {\n case 'sse':\n return new SseStrategy();\n case 'sse-page':\n return new SsePageStrategy();\n case 'json':\n default:\n return new JsonStrategy();\n }\n }\n};\n","import { dateUtils } from '@chatbi-v/core';\n\n/**\n * Mock 响应生成器\n * @description 用于动态生成流式响应数据,支持阶段流转、日志插入和内容分块\n */\nexport class MockResponseGenerator {\n private events: string[] = [];\n private sessionId: string;\n private agentName: string;\n private currentTodos: { content: string; status: 'PENDING' | 'IN_PROGRESS' | 'COMPLETED' }[] = [];\n\n constructor(agentName: string = 'Assistant', sessionId?: string) {\n this.sessionId = sessionId || ('conv_' + dateUtils.now());\n this.agentName = agentName;\n }\n\n /**\n * 生成历史消息流\n */\n emitHistory(history: { role: string; content: string; createTime: string; todos?: any[] }[]) {\n history.forEach(msg => {\n const data = {\n content: msg.content,\n sessionId: this.sessionId,\n role: msg.role,\n completed: true,\n agentName: this.agentName,\n createTime: msg.createTime\n };\n this.pushEvent('data', data);\n\n if (msg.todos) {\n this.pushEvent('todos', { items: msg.todos });\n }\n });\n return this;\n }\n\n /**\n * 初始化执行计划\n * @param steps 计划步骤列表\n */\n initPlan(steps: string[]) {\n this.currentTodos = steps.map(step => ({\n content: step,\n status: 'PENDING'\n }));\n this.pushTodos();\n return this;\n }\n\n /**\n * 更新执行计划状态\n * @param activeIndex 当前正在进行的步骤索引\n */\n updatePlanStatus(activeIndex: number) {\n this.currentTodos = this.currentTodos.map((todo, index) => {\n if (index < activeIndex) {\n return { ...todo, status: 'COMPLETED' };\n } else if (index === activeIndex) {\n return { ...todo, status: 'IN_PROGRESS' };\n } else {\n return { ...todo, status: 'PENDING' };\n }\n });\n this.pushTodos();\n return this;\n }\n\n /**\n * 标记所有计划为完成\n */\n completePlan() {\n this.currentTodos = this.currentTodos.map(todo => ({ ...todo, status: 'COMPLETED' }));\n this.pushTodos();\n return this;\n }\n\n /**\n * 添加日志\n * @param content 日志内容\n * @param type 日志类型\n * @param agentName 可选的 Agent 名称\n */\n addLog(content: string, type: 'info' | 'warning' | 'error' = 'info', agentName?: string) {\n this.pushEvent('log', {\n type,\n content,\n agentName: agentName || this.agentName\n });\n return this;\n }\n\n /**\n * 添加系统错误事件\n * @param errorCode 错误码\n * @param errorMessage 错误信息\n */\n addError(errorCode: string, errorMessage: string) {\n this.pushEvent('error', {\n errorCode,\n errorMessage\n });\n return this;\n }\n\n /**\n * 添加流式内容块\n * @param content 完整内容\n * @param chunkSize 分块大小\n */\n streamContent(content: string, chunkSize: number = 20) {\n for (let i = 0; i < content.length; i += chunkSize) {\n const chunk = content.slice(i, i + chunkSize);\n this.pushEvent('data', {\n content: chunk,\n sessionId: this.sessionId,\n completed: false,\n agentName: this.agentName\n });\n }\n return this;\n }\n\n /**\n * 结束流\n */\n finish() {\n this.pushEvent('data', {\n content: '',\n sessionId: this.sessionId,\n completed: true,\n agentName: this.agentName\n });\n return this;\n }\n\n /**\n * 生成 A2UI 响应\n * @param component A2UI 组件配置对象\n */\n emitA2UI(component: Record<string, any>) {\n const content = JSON.stringify(component, null, 2);\n return this.streamContent(content);\n }\n\n /**\n * 生成最终的响应字符串\n */\n toString() {\n return this.events.join('\\n');\n }\n\n private pushTodos() {\n this.pushEvent('todos', { items: this.currentTodos });\n }\n\n private pushEvent(type: string, data: any) {\n this.events.push(`event: ${type}\\ndata: ${JSON.stringify(data)}\\n`);\n }\n}\n","import dayjs from 'dayjs';\nimport Mock from 'mockjs';\n\nimport { MockEvent } from './types';\n\nexport const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));\n\n/**\n * Process string templates like \"{{$query.pageNo * 10}}\" in object values.\n * Supports $query (for backward compatibility), $body, and $param.\n */\nexport function processTemplate(data: any, context: Record<string, any>): any {\n if (typeof data === 'string') {\n // Check for {{ ... }} pattern\n if (/^\\{\\{.*\\}\\}$/.test(data)) {\n const expression = data.slice(2, -2).trim();\n try {\n // Safe evaluation using Function with context keys\n const keys = Object.keys(context);\n const values = Object.values(context);\n const fn = new Function(...keys, `return ${expression}`);\n return fn(...values);\n } catch (e) {\n console.warn(`[Mock] Failed to evaluate template: ${data}`, e);\n return data;\n }\n }\n return data;\n }\n \n if (Array.isArray(data)) {\n return data.map(item => processTemplate(item, context));\n }\n \n if (typeof data === 'object' && data !== null) {\n const result: Record<string, any> = {};\n for (const key in data) {\n result[key] = processTemplate(data[key], context);\n }\n return result;\n }\n \n return data;\n}\n\n/**\n * Flatten eventsSchema into a sorted array of events.\n * \n * Supports Mock.js array generation rules in keys, e.g.:\n * { 'data|8-12': [{ event: 'data', ... }] } \n * -> generates 8 to 12 data events.\n */\nexport function flatEvents(\n eventsSchema: Record<string, MockEvent[]>, \n requestData: any = {}\n): MockEvent[] {\n // Construct context for template processing\n const context: Record<string, any> = {\n $query: requestData, // Backward compatibility\n $body: requestData,\n $param: requestData?.param || requestData\n };\n\n // Extend Mock.Random to support templates\n Mock.Random.extend({ \n $query: () => context.$query,\n $body: () => context.$body,\n $param: () => context.$param\n });\n \n // Generate data using Mock.mock\n const compiled = Mock.mock(eventsSchema) as Record<string, MockEvent[]>;\n \n // Flatten and sort by delay\n return Object.values(compiled)\n .flat()\n .map(event => processTemplate(event, context))\n .sort((a, b) => {\n const delayA = typeof a.delay === 'number' ? a.delay : parseInt(String(a.delay || 0));\n const delayB = typeof b.delay === 'number' ? b.delay : parseInt(String(b.delay || 0));\n return delayA - delayB;\n });\n}\n\n/**\n * Convert an array of events into a ReadableStream (for SSE).\n */\nexport function eventsToStream(events: MockEvent[]): ReadableStream {\n const encoder = new TextEncoder();\n \n return new ReadableStream({\n start(ctrl) {\n // Check if events is empty\n if (!events || events.length === 0) {\n ctrl.close();\n return;\n }\n\n (async () => {\n try {\n // We assume events are sorted by their `delay` property which represents \n // \"absolute time from start\".\n const startTime = dayjs().valueOf();\n \n for (const eventItem of events) {\n const delay = typeof eventItem.delay === 'number' ? eventItem.delay : parseInt(String(eventItem.delay || 0));\n \n // Calculate how much time passed since start\n const elapsed = dayjs().valueOf() - startTime;\n const remaining = Math.max(0, delay - elapsed);\n \n if (remaining > 0) {\n await sleep(remaining);\n }\n \n const payload = `event: ${eventItem.event}\\ndata: ${JSON.stringify(eventItem.data)}\\n\\n`;\n ctrl.enqueue(encoder.encode(payload));\n }\n ctrl.close();\n } catch (e) {\n ctrl.error(e);\n }\n })();\n },\n });\n}\n","import Mock from 'mockjs';\n\nimport { StrategyFactory } from './strategies';\nimport { JsonMockConfig, MockConfig, MockSchema, SseMockConfig, SsePageMockConfig } from './types';\nimport { eventsToStream, sleep } from './utils';\n\ndeclare global {\n interface Window {\n _originalFetch: typeof fetch;\n _originalEventSource: typeof EventSource;\n }\n}\n\nexport function installFetchMock(schema: MockSchema) {\n if (!window._originalFetch) {\n window._originalFetch = window.fetch;\n }\n\n window.fetch = async (input: RequestInfo | URL, init: RequestInit = {}) => {\n const url = typeof input === 'string' ? input : input instanceof URL ? input.toString() : input.url;\n const method = (init.method || 'GET').toUpperCase();\n \n // Find matching rule\n const rules = Object.values(schema);\n const rule = rules.find((v) => {\n if (!v.url) return false;\n // Convert route params :id to regex \\w+\n // We might want to be more specific, e.g., ([^/]+)\n const pattern = v.url.replace(/:[a-zA-Z0-9_]+/g, '([^/]+)');\n const regex = new RegExp(`^${pattern}$`);\n \n // Check if URL matches (ignoring query params for the match, but we might need them later)\n const [path] = url.split('?');\n // Match method if specified, default to GET match? or just match URL?\n // Usually method matters.\n const ruleMethod = (v.method || 'GET').toUpperCase();\n return regex.test(path) && ruleMethod === method;\n });\n\n if (!rule) {\n return window._originalFetch(input, init);\n }\n\n console.log(`[Mock] Intercepted ${method} ${url}`, rule);\n\n // Handle non-200 status code immediately\n const status = rule.status || 200;\n if (status !== 200) {\n await sleep(rule.delay || 0);\n return new Response(JSON.stringify({ \n message: `Mock error status ${status}`,\n code: status \n }), {\n status,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n await sleep(rule.delay || 0);\n\n // Determine type (default 'json' if not present)\n const type = rule.type || 'json';\n const strategy = StrategyFactory.getStrategy(type);\n \n // Parse query params for strategies that need them\n const urlObj = new URL(url, window.location.origin);\n const query = Object.fromEntries(urlObj.searchParams);\n \n const result = strategy.process(rule as MockConfig, query);\n\n /* 1. JSON Response */\n if (type === 'json') {\n return new Response(JSON.stringify(result), {\n status,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n /* 2. SSE / SSE-Page Response */\n if (type === 'sse' || type === 'sse-page') {\n const events = result; // Strategy returns MockEvent[]\n const stream = eventsToStream(events);\n return new Response(stream, { \n status, \n headers: { 'Content-Type': 'text/event-stream' } \n });\n }\n\n return window._originalFetch(input, init);\n };\n}\n\nexport function installSSEMock() {\n // Placeholder for EventSource mocking if needed.\n // Currently, we rely on fetch interception.\n // Many modern apps use fetch for SSE (POST requests, headers, etc.)\n}\n\nexport function setupMock(schema: MockSchema) {\n if (typeof window !== 'undefined') {\n installFetchMock(schema);\n installSSEMock();\n }\n}\n","\nimport { ApiRequestConfig } from '@chatbi-v/core';\n\n// === Base Types ===\n\nexport type MockMethod = 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH';\nexport type MockType = 'json' | 'sse' | 'sse-page';\n\nexport interface MockEvent {\n event: string;\n data: any;\n /** Delay in ms, or a mockjs template string like '@increment(100)' */\n delay?: number | string;\n}\n\n// === Unified Strategy Interface ===\n\nexport interface MockStrategy {\n /**\n * Process the schema and return the result.\n * For JSON: returns the mocked object.\n * For SSE: returns an array of formatted event strings (chunks).\n */\n process(config: MockConfig, requestParams?: any): any;\n}\n\n// === Mock Configuration Types (Matching chat.mock.ts) ===\n\nexport interface BaseMockConfig {\n /** Optional: URL pattern if used in global interceptor */\n url?: string;\n /** Optional: Method if used in global interceptor */\n method?: MockMethod;\n /** Global delay for the request */\n delay?: number;\n /** Optional: HTTP status code (default: 200) */\n status?: number;\n /** Mock type (default: 'json') */\n type?: MockType;\n}\n\nexport interface JsonMockConfig extends BaseMockConfig {\n type?: 'json';\n /** The MockJS template for the response body */\n responseSchema: any;\n}\n\nexport interface SseMockConfig extends BaseMockConfig {\n type: 'sse';\n /** \n * Dictionary of event templates. \n * Key can be a mockjs template like 'data|3-5' \n */\n responseSchema: Record<string, MockEvent[]>;\n}\n\nexport interface SsePageMockConfig extends BaseMockConfig {\n type: 'sse-page';\n /** Dictionary of event templates */\n responseSchema: Record<string, MockEvent[]>;\n /** Special event for pagination metadata */\n pageEvent: MockEvent;\n}\n\nexport type MockConfig = JsonMockConfig | SseMockConfig | SsePageMockConfig;\n\nexport type MockSchema = Record<string, MockConfig>;\n\n// === Legacy Support (Keep these for backward compatibility if needed) ===\n\nexport interface ChatStreamConfig {\n _type: 'chat_stream';\n agentName?: string;\n plan?: string[];\n data?: Record<string, any>;\n timeline?: Array<{\n log?: string | { content: string; level?: 'info' | 'warning' | 'error'; agent?: string };\n planIndex?: number;\n delay?: number;\n }>;\n content: string | Record<string, any> | ((data: any) => string | Record<string, any>);\n}\n\nexport interface HistoryStreamConfig {\n _type: 'history_stream';\n template: Record<string, any>;\n prepend?: any[];\n}\n\nexport type AdvancedMockSchema = ChatStreamConfig | HistoryStreamConfig;\n\nexport interface MockGeneratorStrategy {\n generate(schema: any): string[];\n}\n\n// === Helper Functions (Restored) ===\n\nexport function createChatStream(config: Omit<ChatStreamConfig, '_type'>): ChatStreamConfig {\n return { _type: 'chat_stream', ...config };\n}\n\nexport function createHistoryStream(config: Omit<HistoryStreamConfig, '_type'>): HistoryStreamConfig {\n return { _type: 'history_stream', ...config };\n}\n"],"mappings":"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","names":["dateUtils","dateUtils","Mock","Mock","mockData","Mock","mockData","Mock","dateUtils","delay","dateUtils"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@chatbi-v/mocks",
|
|
3
|
-
"version": "2.1.
|
|
3
|
+
"version": "2.1.2",
|
|
4
4
|
"main": "dist/index.cjs",
|
|
5
5
|
"files": [
|
|
6
6
|
"dist"
|
|
@@ -21,13 +21,13 @@
|
|
|
21
21
|
"mockjs": "^1.1.0",
|
|
22
22
|
"dayjs": "^1.11.10",
|
|
23
23
|
"react": ">=18.0.0",
|
|
24
|
-
"@chatbi-v/core": "2.1.
|
|
24
|
+
"@chatbi-v/core": "2.1.2"
|
|
25
25
|
},
|
|
26
26
|
"devDependencies": {
|
|
27
27
|
"@types/mockjs": "^1.0.10",
|
|
28
28
|
"tsup": "^8.5.1",
|
|
29
|
-
"@chatbi-v/cli": "2.1.
|
|
30
|
-
"@chatbi-v/
|
|
29
|
+
"@chatbi-v/cli": "2.1.2",
|
|
30
|
+
"@chatbi-v/config": "2.1.2"
|
|
31
31
|
},
|
|
32
32
|
"scripts": {
|
|
33
33
|
"build": "chatbi-cli build",
|
package/dist/adapter.d.ts
DELETED
|
@@ -1,48 +0,0 @@
|
|
|
1
|
-
import { ApiAdapter, ApiEndpointConfig, ApiRequestConfig, StreamCallbacks } from '@chatbi-v/core';
|
|
2
|
-
import { MockGeneratorStrategy } from './types';
|
|
3
|
-
/**
|
|
4
|
-
* 基于 Mock.js 的请求适配器
|
|
5
|
-
* @description 根据 API 配置中的 responseSchema 自动生成 Mock 数据,支持普通 JSON 和流式 SSE 响应
|
|
6
|
-
*/
|
|
7
|
-
export declare class MockAdapter implements ApiAdapter {
|
|
8
|
-
private delay;
|
|
9
|
-
private static strategies;
|
|
10
|
-
/**
|
|
11
|
-
* 构造函数
|
|
12
|
-
* @param delay 全局模拟延迟时间(毫秒),默认 300ms
|
|
13
|
-
*/
|
|
14
|
-
constructor(delay?: number);
|
|
15
|
-
/**
|
|
16
|
-
* 注册自定义 Mock 生成策略
|
|
17
|
-
* @param key 策略唯一标识符 (e.g., 'chat_stream', 'history_stream')
|
|
18
|
-
* @param strategy 实现了 MockGeneratorStrategy 接口的策略实例
|
|
19
|
-
*/
|
|
20
|
-
static registerStrategy(key: string, strategy: MockGeneratorStrategy): void;
|
|
21
|
-
/**
|
|
22
|
-
* 处理普通 HTTP 请求(非流式)
|
|
23
|
-
* @param config 请求配置对象
|
|
24
|
-
* @param endpointConfig API 端点配置,包含 responseSchema
|
|
25
|
-
* @returns Promise 返回模拟的响应数据
|
|
26
|
-
*/
|
|
27
|
-
request<T = any>(config: ApiRequestConfig, endpointConfig?: ApiEndpointConfig): Promise<T>;
|
|
28
|
-
/**
|
|
29
|
-
* 处理流式请求 (SSE)
|
|
30
|
-
* @param config 请求配置对象
|
|
31
|
-
* @param callbacks 流式回调函数集合 (onMessage, onFinish, onError)
|
|
32
|
-
* @param endpointConfig API 端点配置,包含 responseSchema
|
|
33
|
-
*/
|
|
34
|
-
stream(config: ApiRequestConfig, callbacks: StreamCallbacks, endpointConfig?: ApiEndpointConfig): Promise<void>;
|
|
35
|
-
/**
|
|
36
|
-
* 判断是否为遗留的 schema 格式
|
|
37
|
-
*/
|
|
38
|
-
private isLegacySchema;
|
|
39
|
-
/**
|
|
40
|
-
* 判断是否为新版配置格式
|
|
41
|
-
* @description 检查配置对象中是否包含有效的 type 字段
|
|
42
|
-
*/
|
|
43
|
-
private isNewConfig;
|
|
44
|
-
/**
|
|
45
|
-
* 生成遗留的高级 schema 数据块
|
|
46
|
-
*/
|
|
47
|
-
private generateAdvancedChunks;
|
|
48
|
-
}
|
package/dist/adapter.js
DELETED
|
@@ -1,308 +0,0 @@
|
|
|
1
|
-
import { createLogger, dateUtils, } from '@chatbi-v/core';
|
|
2
|
-
import Mock from 'mockjs';
|
|
3
|
-
import { ChatStreamStrategy, HistoryStreamStrategy, StrategyFactory } from './strategies';
|
|
4
|
-
import { sleep } from './utils';
|
|
5
|
-
const logger = createLogger('MockAdapter');
|
|
6
|
-
/**
|
|
7
|
-
* 基于 Mock.js 的请求适配器
|
|
8
|
-
* @description 根据 API 配置中的 responseSchema 自动生成 Mock 数据,支持普通 JSON 和流式 SSE 响应
|
|
9
|
-
*/
|
|
10
|
-
export class MockAdapter {
|
|
11
|
-
delay;
|
|
12
|
-
// 遗留策略(为了兼容旧版代码)
|
|
13
|
-
static strategies = {
|
|
14
|
-
chat_stream: new ChatStreamStrategy(),
|
|
15
|
-
history_stream: new HistoryStreamStrategy(),
|
|
16
|
-
};
|
|
17
|
-
/**
|
|
18
|
-
* 构造函数
|
|
19
|
-
* @param delay 全局模拟延迟时间(毫秒),默认 300ms
|
|
20
|
-
*/
|
|
21
|
-
constructor(delay = 300) {
|
|
22
|
-
this.delay = delay;
|
|
23
|
-
}
|
|
24
|
-
/**
|
|
25
|
-
* 注册自定义 Mock 生成策略
|
|
26
|
-
* @param key 策略唯一标识符 (e.g., 'chat_stream', 'history_stream')
|
|
27
|
-
* @param strategy 实现了 MockGeneratorStrategy 接口的策略实例
|
|
28
|
-
*/
|
|
29
|
-
static registerStrategy(key, strategy) {
|
|
30
|
-
this.strategies[key] = strategy;
|
|
31
|
-
}
|
|
32
|
-
/**
|
|
33
|
-
* 处理普通 HTTP 请求(非流式)
|
|
34
|
-
* @param config 请求配置对象
|
|
35
|
-
* @param endpointConfig API 端点配置,包含 responseSchema
|
|
36
|
-
* @returns Promise 返回模拟的响应数据
|
|
37
|
-
*/
|
|
38
|
-
async request(config, endpointConfig) {
|
|
39
|
-
// 优先使用接口定义的 delay,否则使用全局默认 delay
|
|
40
|
-
const delay = endpointConfig?.delay ?? this.delay;
|
|
41
|
-
return new Promise((resolve, reject) => {
|
|
42
|
-
setTimeout(() => {
|
|
43
|
-
try {
|
|
44
|
-
if (!endpointConfig || !endpointConfig.responseSchema) {
|
|
45
|
-
logger.warn(`未找到响应架构配置: ${config.url}`);
|
|
46
|
-
resolve({});
|
|
47
|
-
return;
|
|
48
|
-
}
|
|
49
|
-
let schema = endpointConfig.responseSchema;
|
|
50
|
-
let mockData;
|
|
51
|
-
// 支持函数式 schema,允许根据请求参数动态生成 schema
|
|
52
|
-
if (typeof schema === 'function') {
|
|
53
|
-
schema = schema(config);
|
|
54
|
-
}
|
|
55
|
-
// 构造有效配置对象,合并 endpointConfig 顶层属性
|
|
56
|
-
const effectiveConfig = {
|
|
57
|
-
type: endpointConfig.type,
|
|
58
|
-
status: endpointConfig.status || 200,
|
|
59
|
-
pageEvent: endpointConfig.pageEvent,
|
|
60
|
-
responseSchema: schema,
|
|
61
|
-
...(typeof schema === 'object' ? schema : {}),
|
|
62
|
-
};
|
|
63
|
-
// 0. 处理非 200 状态码
|
|
64
|
-
if (effectiveConfig.status !== 200) {
|
|
65
|
-
const error = new Error(`Request failed with status code ${effectiveConfig.status}`);
|
|
66
|
-
error.response = {
|
|
67
|
-
status: effectiveConfig.status,
|
|
68
|
-
data: Mock.mock(schema),
|
|
69
|
-
headers: {},
|
|
70
|
-
};
|
|
71
|
-
reject(error);
|
|
72
|
-
return;
|
|
73
|
-
}
|
|
74
|
-
// 1. 字符串 Schema:直接作为结果返回(极少情况)
|
|
75
|
-
if (typeof schema === 'string') {
|
|
76
|
-
mockData = schema;
|
|
77
|
-
}
|
|
78
|
-
// 2. 遗留的高级 Schema (通过 _type 字段识别)
|
|
79
|
-
else if (this.isLegacySchema(schema)) {
|
|
80
|
-
const strategy = MockAdapter.strategies[schema._type];
|
|
81
|
-
if (strategy) {
|
|
82
|
-
const chunks = strategy.generate(schema);
|
|
83
|
-
// 将流式块合并为单个字符串,并清理 SSE 格式标记
|
|
84
|
-
mockData = chunks
|
|
85
|
-
.join('')
|
|
86
|
-
.replace(/event: data\ndata: /g, '')
|
|
87
|
-
.replace(/\n\n/g, '');
|
|
88
|
-
try {
|
|
89
|
-
mockData = JSON.parse(mockData);
|
|
90
|
-
}
|
|
91
|
-
catch (e) {
|
|
92
|
-
// 忽略解析错误,保持原始数据格式
|
|
93
|
-
}
|
|
94
|
-
}
|
|
95
|
-
else {
|
|
96
|
-
logger.warn(`未找到对应的策略类型: ${schema._type}`);
|
|
97
|
-
mockData = {};
|
|
98
|
-
}
|
|
99
|
-
}
|
|
100
|
-
// 3. 通用策略模式
|
|
101
|
-
else {
|
|
102
|
-
const type = effectiveConfig.type || 'json';
|
|
103
|
-
const strategy = StrategyFactory.getStrategy(type);
|
|
104
|
-
mockData = strategy.process(effectiveConfig, config.params || config.data);
|
|
105
|
-
}
|
|
106
|
-
logger.info(`Request: ${config.method} ${config.url}`, config.data || config.params);
|
|
107
|
-
logger.info(`Response:`, mockData);
|
|
108
|
-
resolve(mockData);
|
|
109
|
-
}
|
|
110
|
-
catch (error) {
|
|
111
|
-
reject(error);
|
|
112
|
-
}
|
|
113
|
-
}, delay);
|
|
114
|
-
});
|
|
115
|
-
}
|
|
116
|
-
/**
|
|
117
|
-
* 处理流式请求 (SSE)
|
|
118
|
-
* @param config 请求配置对象
|
|
119
|
-
* @param callbacks 流式回调函数集合 (onMessage, onFinish, onError)
|
|
120
|
-
* @param endpointConfig API 端点配置,包含 responseSchema
|
|
121
|
-
*/
|
|
122
|
-
async stream(config, callbacks, endpointConfig) {
|
|
123
|
-
const { onMessage, onFinish, onError } = callbacks;
|
|
124
|
-
const signal = config.signal;
|
|
125
|
-
// 优先使用接口定义的 delay,否则使用全局默认 delay
|
|
126
|
-
const delay = endpointConfig?.delay ?? this.delay;
|
|
127
|
-
return new Promise((resolve, reject) => {
|
|
128
|
-
// 如果请求已被取消,直接结束
|
|
129
|
-
if (signal && signal.aborted) {
|
|
130
|
-
if (onFinish)
|
|
131
|
-
onFinish();
|
|
132
|
-
resolve();
|
|
133
|
-
return;
|
|
134
|
-
}
|
|
135
|
-
setTimeout(async () => {
|
|
136
|
-
try {
|
|
137
|
-
if (!endpointConfig || !endpointConfig.responseSchema) {
|
|
138
|
-
logger.warn(`未找到流式响应架构: ${config.url}`);
|
|
139
|
-
if (onFinish)
|
|
140
|
-
onFinish();
|
|
141
|
-
resolve();
|
|
142
|
-
return;
|
|
143
|
-
}
|
|
144
|
-
const requestStart = dateUtils.now();
|
|
145
|
-
logger.info(`[SSE Start] Request: ${config.method} ${config.url}`, {
|
|
146
|
-
params: config.data || config.params,
|
|
147
|
-
time: dateUtils.dayjs().toISOString(),
|
|
148
|
-
});
|
|
149
|
-
let schema = endpointConfig.responseSchema;
|
|
150
|
-
// 支持函数式 schema
|
|
151
|
-
if (typeof schema === 'function') {
|
|
152
|
-
schema = schema(config);
|
|
153
|
-
}
|
|
154
|
-
// 构造有效配置对象,合并 endpointConfig 顶层属性
|
|
155
|
-
const effectiveConfig = {
|
|
156
|
-
type: endpointConfig.type,
|
|
157
|
-
status: endpointConfig.status || 200,
|
|
158
|
-
pageEvent: endpointConfig.pageEvent,
|
|
159
|
-
responseSchema: schema,
|
|
160
|
-
...(typeof schema === 'object' ? schema : {}),
|
|
161
|
-
};
|
|
162
|
-
// 0. 处理非 200 状态码
|
|
163
|
-
if (effectiveConfig.status !== 200) {
|
|
164
|
-
const response = {
|
|
165
|
-
status: effectiveConfig.status,
|
|
166
|
-
data: Mock.mock(schema),
|
|
167
|
-
headers: {},
|
|
168
|
-
};
|
|
169
|
-
// 触发 onResponse 回调
|
|
170
|
-
if (callbacks.onResponse) {
|
|
171
|
-
const hijacked = await callbacks.onResponse(response);
|
|
172
|
-
if (hijacked) {
|
|
173
|
-
resolve();
|
|
174
|
-
return;
|
|
175
|
-
}
|
|
176
|
-
}
|
|
177
|
-
const error = new Error(`Stream request failed with status code ${effectiveConfig.status}`);
|
|
178
|
-
error.response = response;
|
|
179
|
-
logger.error(`[SSE Error] Request failed with status ${effectiveConfig.status}`, error);
|
|
180
|
-
if (onError)
|
|
181
|
-
onError(error);
|
|
182
|
-
// 保持与 request 同样的处理逻辑,抛出异常以便顶层拦截
|
|
183
|
-
reject(error);
|
|
184
|
-
return;
|
|
185
|
-
}
|
|
186
|
-
// 触发 onResponse 回调 (200 场景)
|
|
187
|
-
if (callbacks.onResponse) {
|
|
188
|
-
const hijacked = await callbacks.onResponse({
|
|
189
|
-
status: 200,
|
|
190
|
-
data: null,
|
|
191
|
-
headers: {},
|
|
192
|
-
});
|
|
193
|
-
if (hijacked) {
|
|
194
|
-
resolve();
|
|
195
|
-
return;
|
|
196
|
-
}
|
|
197
|
-
}
|
|
198
|
-
// 1. 新版 Schema 策略 (sse, sse-page)
|
|
199
|
-
if (this.isNewConfig(effectiveConfig)) {
|
|
200
|
-
const type = effectiveConfig.type || 'sse';
|
|
201
|
-
const strategy = StrategyFactory.getStrategy(type);
|
|
202
|
-
// 策略处理返回 MockEvent 数组
|
|
203
|
-
const events = strategy.process(effectiveConfig, config.params || config.data);
|
|
204
|
-
if (Array.isArray(events)) {
|
|
205
|
-
const startTime = dateUtils.now();
|
|
206
|
-
let eventCount = 0;
|
|
207
|
-
logger.info(`[SSE Processing] Generated ${events.length} events for ${type} strategy`);
|
|
208
|
-
for (const event of events) {
|
|
209
|
-
if (signal && signal.aborted) {
|
|
210
|
-
logger.info(`[SSE Abort] Stream aborted by user after ${dateUtils.now() - requestStart}ms`);
|
|
211
|
-
break;
|
|
212
|
-
}
|
|
213
|
-
// 计算延迟
|
|
214
|
-
const delay = typeof event.delay === 'number'
|
|
215
|
-
? event.delay
|
|
216
|
-
: parseInt(String(event.delay || 0));
|
|
217
|
-
const elapsed = dateUtils.now() - startTime;
|
|
218
|
-
const remaining = Math.max(0, delay - elapsed);
|
|
219
|
-
if (remaining > 0) {
|
|
220
|
-
await sleep(remaining);
|
|
221
|
-
}
|
|
222
|
-
if (signal && signal.aborted)
|
|
223
|
-
break;
|
|
224
|
-
// 构造 SSE 格式数据块
|
|
225
|
-
const chunk = `event: ${event.event}\ndata: ${JSON.stringify(event.data)}\n\n`;
|
|
226
|
-
// 记录关键事件日志
|
|
227
|
-
if (['error', 'todos', 'page'].includes(event.event)) {
|
|
228
|
-
logger.info(`[SSE Event] Emitting special event: ${event.event}`, event.data);
|
|
229
|
-
}
|
|
230
|
-
else if (eventCount === 0 || eventCount === events.length - 1) {
|
|
231
|
-
logger.info(`[SSE Event] Emitting ${eventCount === 0 ? 'first' : 'last'} data event`, { preview: JSON.stringify(event.data).slice(0, 50) + '...' });
|
|
232
|
-
}
|
|
233
|
-
if (onMessage)
|
|
234
|
-
onMessage(chunk);
|
|
235
|
-
eventCount++;
|
|
236
|
-
}
|
|
237
|
-
}
|
|
238
|
-
}
|
|
239
|
-
// 2. 遗留策略处理
|
|
240
|
-
else {
|
|
241
|
-
let chunks = [];
|
|
242
|
-
if (typeof schema === 'string') {
|
|
243
|
-
chunks = schema.split('\n\n').map((chunk) => chunk + '\n\n');
|
|
244
|
-
}
|
|
245
|
-
else if (this.isLegacySchema(schema)) {
|
|
246
|
-
chunks = this.generateAdvancedChunks(schema);
|
|
247
|
-
}
|
|
248
|
-
else {
|
|
249
|
-
const mockData = Mock.mock(schema);
|
|
250
|
-
chunks = [`data: ${JSON.stringify(mockData)}\n\n`];
|
|
251
|
-
}
|
|
252
|
-
for (const chunk of chunks) {
|
|
253
|
-
if (signal && signal.aborted) {
|
|
254
|
-
logger.info('[SSE Abort] Stream aborted by user');
|
|
255
|
-
break;
|
|
256
|
-
}
|
|
257
|
-
if (chunk.trim()) {
|
|
258
|
-
if (onMessage)
|
|
259
|
-
onMessage(chunk);
|
|
260
|
-
await new Promise((r) => setTimeout(r, 200));
|
|
261
|
-
}
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
if (!signal || !signal.aborted) {
|
|
265
|
-
logger.info(`[SSE Complete] Stream finished successfully in ${dateUtils.now() - requestStart}ms`);
|
|
266
|
-
if (onFinish)
|
|
267
|
-
onFinish();
|
|
268
|
-
}
|
|
269
|
-
resolve();
|
|
270
|
-
}
|
|
271
|
-
catch (error) {
|
|
272
|
-
logger.error(`[SSE Error] Stream processing failed`, error);
|
|
273
|
-
if (onError)
|
|
274
|
-
onError(error);
|
|
275
|
-
resolve();
|
|
276
|
-
}
|
|
277
|
-
}, delay);
|
|
278
|
-
});
|
|
279
|
-
}
|
|
280
|
-
/**
|
|
281
|
-
* 判断是否为遗留的 schema 格式
|
|
282
|
-
*/
|
|
283
|
-
isLegacySchema(schema) {
|
|
284
|
-
return schema && typeof schema === 'object' && '_type' in schema;
|
|
285
|
-
}
|
|
286
|
-
/**
|
|
287
|
-
* 判断是否为新版配置格式
|
|
288
|
-
* @description 检查配置对象中是否包含有效的 type 字段
|
|
289
|
-
*/
|
|
290
|
-
isNewConfig(config) {
|
|
291
|
-
return (config &&
|
|
292
|
-
typeof config === 'object' &&
|
|
293
|
-
(['json', 'sse', 'sse-page'].includes(config.type) || 'responseSchema' in config) // 备用检查,如果 schema 内部定义了结构
|
|
294
|
-
);
|
|
295
|
-
}
|
|
296
|
-
/**
|
|
297
|
-
* 生成遗留的高级 schema 数据块
|
|
298
|
-
*/
|
|
299
|
-
generateAdvancedChunks(schema) {
|
|
300
|
-
const strategy = MockAdapter.strategies[schema._type];
|
|
301
|
-
if (strategy) {
|
|
302
|
-
return strategy.generate(schema);
|
|
303
|
-
}
|
|
304
|
-
logger.warn(`未找到对应的策略类型: ${schema._type}`);
|
|
305
|
-
return [];
|
|
306
|
-
}
|
|
307
|
-
}
|
|
308
|
-
//# sourceMappingURL=adapter.js.map
|