@yxp934/prompt-tree 0.2.0 → 0.3.0
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/.next/standalone/.next/BUILD_ID +1 -1
- package/.next/standalone/.next/app-path-routes-manifest.json +4 -4
- package/.next/standalone/.next/build-manifest.json +2 -2
- package/.next/standalone/.next/server/app/_global-error/page.js +1 -1
- package/.next/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/_global-error.html +2 -2
- package/.next/standalone/.next/server/app/_global-error.rsc +1 -1
- package/.next/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_not-found/page.js +1 -1
- package/.next/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/_not-found.html +1 -1
- package/.next/standalone/.next/server/app/_not-found.rsc +2 -2
- package/.next/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/api/agent/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/chat/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/providers/health/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/providers/models/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/providers/test/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/tools/mcp/call-tool/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/tools/mcp/list-tools/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/tools/mcp/test/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/tools/python/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/api/tools/search/route_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/index.html +2 -2
- package/.next/standalone/.next/server/app/index.rsc +3 -3
- package/.next/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/index.segments/_full.segment.rsc +3 -3
- package/.next/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/index.segments/_index.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/page.js +1 -1
- package/.next/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/settings/page.js +2 -2
- package/.next/standalone/.next/server/app/settings/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/settings.html +1 -1
- package/.next/standalone/.next/server/app/settings.rsc +3 -3
- package/.next/standalone/.next/server/app/settings.segments/_full.segment.rsc +3 -3
- package/.next/standalone/.next/server/app/settings.segments/_head.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/settings.segments/_index.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/settings.segments/_tree.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/settings.segments/settings/__PAGE__.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/settings.segments/settings.segment.rsc +1 -1
- package/.next/standalone/.next/server/app-paths-manifest.json +4 -4
- package/.next/standalone/.next/server/chunks/471.js +4 -4
- package/.next/standalone/.next/server/chunks/689.js +1 -1
- package/.next/standalone/.next/server/chunks/732.js +3 -3
- package/.next/standalone/.next/server/chunks/734.js +1 -1
- package/.next/standalone/.next/server/pages/404.html +1 -1
- package/.next/standalone/.next/server/pages/500.html +2 -2
- package/.next/standalone/.next/static/chunks/{394-1e8c9367852fd6d7.js → 394-dc41566095beafcf.js} +1 -1
- package/.next/standalone/.next/static/chunks/787-12058ee9986bdbfc.js +1 -0
- package/.next/standalone/.next/static/chunks/app/settings/page-76e3d482410f0adc.js +1 -0
- package/.next/standalone/package.json +1 -1
- package/README.md +134 -120
- package/README.zh-CN.md +200 -0
- package/package.json +1 -1
- package/.next/standalone/.next/static/chunks/787-56a48f915165efd3.js +0 -1
- package/.next/standalone/.next/static/chunks/app/settings/page-0fc8a4cbf639025c.js +0 -1
- /package/.next/standalone/.next/static/{wYfac0R-RsurEa801s_qH → i6P3teUAVQJ-hEG3Y6KV0}/_buildManifest.js +0 -0
- /package/.next/standalone/.next/static/{wYfac0R-RsurEa801s_qH → i6P3teUAVQJ-hEG3Y6KV0}/_ssgManifest.js +0 -0
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";exports.id=689,exports.ids=[689],exports.modules={2115:(a,b,c)=>{c.d(b,{d:()=>f,t:()=>e});let d={en:{"common.appName":"Prompt Tree","common.add":"Add","common.cancel":"Cancel","common.clear":"Clear","common.close":"Close","common.copy":"Copy","common.copied":"Copied","common.data":"Data","common.delete":"Delete","common.edit":"Edit","common.hide":"Hide","common.loading":"Loading…","common.new":"New","common.save":"Save","common.selectAll":"Select all","common.settings":"Settings","common.show":"Show","common.tools":"Tools","modal.closeAria":"Close modal","layout.closeSidebarAria":"Close sidebar","layout.closeContextPanelAria":"Close context panel","node.author.assistant":"Assistant","node.author.you":"You","node.author.system":"System","node.author.compressed":"Compressed","node.type.system":"System","node.type.user":"User","node.type.assistant":"Assistant","node.type.compressed":"Compressed","header.menu":"Menu","header.context":"Context","header.fallback.folder":"Folder","header.fallback.loading":"Loading…","header.nodes":"nodes","header.branches":"branches","header.threads":"threads","header.toggleTheme":"Toggle theme","header.theme.light":"Light","header.theme.dark":"Dark","sidebar.tagline":"Dialogue Topology","sidebar.library":"Library","sidebar.newThread":"New Thread","sidebar.newFolder":"New Folder","sidebar.deleteFolderAria":"Delete folder","sidebar.deleteThreadAria":"Delete thread","sidebar.nodeCount":"{count} nodes","sidebar.threadCount":"{count} threads","sidebar.confirmDeleteFolderTitle":"Delete folder?","sidebar.confirmDeleteThreadTitle":"Delete thread?","sidebar.confirmDeleteFolderBody":"Delete folder “{name}”. {threadNote}This action cannot be undone.","sidebar.confirmDeleteFolderThreadNote":"{count} threads will be moved to Home. ","sidebar.confirmDeleteThreadBody":"Delete thread “{title}”. This action cannot be undone.","folder.deleteFolder":"Delete folder","folder.deleteThreadAria":"Delete thread","folder.threadPreviewAria":"Thread canvas preview","folder.empty.selectFolder":"Select a folder to view its threads.","folder.empty.notFound":"Folder not found.","folder.label":"Folder","folder.namePlaceholder":"Folder name (max {count} chars)","folder.newThread":"New Thread","folder.systemPrompt.title":"Unified System Prompt","folder.systemPrompt.collapsedEmpty":"Tap to add a prompt","folder.systemPrompt.placeholder":"This will be the first (system) node of every thread in this folder.","folder.systemPrompt.note":"Applies to all threads in this folder (root system node).","folder.models.title":"Enabled Models","folder.models.allEnabled":"All enabled","folder.models.none":"None","folder.models.selectedCount":"{count} selected","folder.models.noneEnabled":"No models enabled.","folder.models.configureInSettings":"Configure in Settings","folder.models.help":"Click to edit the folder model set.","folder.models.noneSelected":"No models selected for this folder.","folder.models.picker.title":"Folder Enabled Models","folder.models.picker.description":"Pick from globally enabled models.","folder.models.picker.selected":"Selected:","folder.threads.title":"Threads ({count})","folder.threads.empty.title":"No threads yet","folder.threads.empty.description":"Create a thread in this folder. It will inherit the unified system prompt above.","folder.threads.empty.action":"Create Thread","chat.input.placeholder":"Type your message…","chat.input.attach":"Attach","chat.input.send":"Send","chat.input.modelMeta":"Model: {model}{tempPart}","chat.input.modelMetaNone":"Model: —","chat.input.tempPart":" - Temp: {temp}","chat.input.models":"Models","chat.input.enabledModels":"Enabled Models","chat.input.toolUse":"Tool Use","chat.input.tool.webSearch":"Web Search","chat.input.tool.python":"Python","chat.input.tool.pythonSubtitle":"Online code execution","chat.message.retry":"Retry","chat.message.copyResponseAria":"Copy response","chat.message.retryResponseAria":"Retry response","chat.message.toolUseEnabledTitle":"Tool use enabled for this message","chat.tool.args":"Args","chat.tool.error":"Error","chat.tool.result":"Result","chat.tool.unknownError":"Unknown error","chat.tool.status.ok":"ok","chat.tool.status.error":"error","chat.tool.status.running":"running…","chat.thinking":"Thinking…","chat.noConversationLoaded":"No conversation loaded.","chat.modelLabel.count":"{count} models","chat.contextTokens":"Context: {used} / {max}","context.title":"Context Assembly","context.subtitle":"Build your prompt","context.tokenUsage":"Token Usage","context.activeNodes":"Active Nodes","context.clearAll":"Clear all","context.toolBlocks":"Tool Blocks","context.dropzone.title":"Drag nodes here","context.dropzone.subtitle":"Drop to insert into context","context.optimize":"Optimize Context","context.preview.button":"Preview Full Context","context.preview.title":"Full Context Preview","context.preview.building":"Building context…","context.preview.failed":"Failed to build context.","context.removeFromContextAria":"Remove from context","context.card.system":"System Prompt","context.card.user":"User Message","context.card.compressed":"Compressed","context.compress.title":"Compress Nodes","context.compress.selection":"Selection ({count})","context.compress.noneSelected":"No nodes selected.","context.compress.summary":"Summary","context.compress.summaryPlaceholder":"2–3 sentences. You can generate with AI, then edit.","context.compress.language":"Language","context.compress.languagePlaceholder":"zh-CN / en","context.compress.format":"Format","context.compress.formatPlaceholder":"markdown / json","context.compress.role":"Role","context.compress.rolePlaceholder":"expert","context.compress.generate":"Generate with AI","context.compress.confirm":"Compress","context.compress.selected":"Compress Selected ({count})","context.compress.context":"Compress Context ({count})","context.compress.errors.minTwoNodes":"Select at least 2 nodes from the same path to compress.","context.compress.errors.noGaps":"Selection must be a single continuous path (no gaps).","context.compress.errors.noBranches":"Selection must be a single continuous path (no branches).","context.compress.errors.invalidSelection":"Invalid selection","tree.controls.zoomOut":"Zoom out","tree.controls.zoomIn":"Zoom in","tree.controls.fitView":"Fit view","tree.controls.autoLayout":"Auto layout","tree.menu.continueFromHere":"Continue from here","tree.menu.compressBranch":"Compress branch","tree.menu.decompressConfirm":"Decompress this node and restore the full chain?","tree.menu.decompress":"Decompress","tree.menu.addToContext":"Add to Context","tree.menu.editNode":"Edit node","tree.menu.deleteSubtree":"Delete subtree","tree.editor.title":"Edit Node","tree.node.emptySummary":"(empty summary)","tree.node.emptyContent":"(empty)","tree.node.dragToContextAria":"Drag node to context","tree.node.expandCompressedAria":"Expand compressed node","tree.node.collapseCompressedAria":"Collapse compressed node","branches.title":"Branches","branches.activeCount":"{count} active","branches.default.compressed":"Compressed branch","branches.default.type":"{type} branch","settings.nav.configuration":"Configuration","settings.nav.providers":"Model Service","settings.nav.defaultModel":"Default Model","settings.nav.tools":"Tools","settings.nav.general":"General","settings.nav.display":"Display","settings.nav.data":"Data","settings.nav.about":"About","settings.backToHome":"Back to Home","providers.title":"Service Providers","providers.deleteAria":"Delete provider","providers.empty.title":"No providers yet","providers.empty.description":"Click below to add one.","providers.add":"Add provider","providers.dialog.title":"Add provider","providers.dialog.nameLabel":"Provider Name","providers.dialog.namePlaceholder":"OpenAI, Anthropic, DeepSeek…","providerConfig.empty.title":"Select or add a provider","providerConfig.empty.description":"Choose a provider from the middle list to configure.","providerConfig.check.running":"Checking…","providerConfig.check.button":"Test connection","providerConfig.apiKeys.title":"API Key","providerConfig.apiKeys.namePlaceholder":"Key name","providerConfig.apiKeys.primary":"Primary","providerConfig.apiKeys.setPrimary":"Set as primary","providerConfig.apiKeys.add":"Add key","providerConfig.baseUrl.title":"Base URL","providerConfig.baseUrl.preview":"Preview: {url}","providerConfig.models.title":"Models","providerConfig.models.select":"Select models","providerConfig.models.enabled":"Enabled ({count})","providerConfig.models.disabled":"Disabled ({count})","providerConfig.models.streaming":"Streaming","providerConfig.models.empty.title":"No models","providerConfig.models.empty.description":"Click “Select models” to add.","providerConfig.addKey.title":"Add API Key","providerConfig.addKey.nameLabel":"Name","providerConfig.addKey.namePlaceholder":"Primary / backup","providerConfig.addKey.valueLabel":"API Key","errors.connectionFailed":"Connection failed","errors.requestFailedWithStatus":"Request failed ({status})","errors.invalidRequestBody":"Invalid request body","errors.missingApiKey":"Missing API key","errors.invalidBaseUrl":"Invalid Base URL","errors.invalidApiKey":"Invalid API key","errors.requestTimeout":"Request timeout","errors.unknownError":"Unknown error","errors.failedToFetchModels":"Failed to fetch models","errors.missingOpenAIApiKey":"Missing OpenAI API key. Add it in Settings.","errors.noActiveConversationTree":"No active conversation tree loaded.","errors.failedToSendMessage":"Failed to send message","errors.failedToRetryMessage":"Failed to retry message","errors.failedToCompressNodes":"Failed to compress nodes","errors.failedToDecompressNode":"Failed to decompress node","errors.failedToGenerateSuggestion":"Failed to generate suggestion","settings.display.title":"Display","settings.display.description":"Manage theme and language.","settings.display.theme":"Theme","settings.display.theme.light":"Light","settings.display.theme.lightDesc":"Best for daylight environments.","settings.display.theme.dark":"Dark","settings.display.theme.darkDesc":"Best for low-light environments.","settings.display.language":"Language","settings.display.language.en":"English","settings.display.language.enDesc":"Use English UI copy.","settings.display.language.zhCN":"中文","settings.display.language.zhCNDesc":"使用中文界面(保留常用英文术语)。","settings.defaultModel.title":"Model Settings","settings.defaultModel.description":"Select branch models and configure compression + title generation.","settings.defaultModel.enabledModels.title":"Enabled models","settings.defaultModel.enabledModels.description":"When multiple models are selected, sending a message generates parallel branches.","settings.defaultModel.enabledModels.emptyTitle":"No enabled models","settings.defaultModel.enabledModels.emptyDescription":"Enable models first in Model Service.","settings.defaultModel.compressionModel.title":"Compression model","settings.defaultModel.compressionModel.placeholder":"Select a compression model","settings.defaultModel.compressionModel.description":"Used to generate compressed summaries and meta-instruction suggestions.","settings.defaultModel.summaryModel.title":"Title model","settings.defaultModel.summaryModel.placeholder":"Select a title model","settings.defaultModel.summaryModel.description":"Used to generate the conversation title after the first message.","settings.defaultModel.test.title":"Model test","settings.defaultModel.test.latency":"Latency {ms}ms","settings.defaultModel.test.running":"Testing…","settings.defaultModel.test.button":"Test model","settings.defaultModel.test.selectModelFirst":"Select a model first.","settings.defaultModel.test.providerMissing":"Provider not found.","settings.defaultModel.test.success":"Connected","settings.defaultModel.test.failed":"Test failed","settings.general.title":"General","settings.general.description":"Manage default API connection and basic generation settings.","settings.general.api.title":"API","settings.general.api.apiKeyLabel":"API Key","settings.general.api.baseUrlLabel":"Base URL","settings.general.generation.title":"Generation","settings.general.generation.temperature":"Temperature","settings.general.generation.maxTokens":"Max Tokens","settings.general.saveHint":"Changes are saved locally in your browser.","settings.general.saveButton":"Save settings","settings.general.saved":"Saved","settings.data.title":"Data","settings.data.description":"Manage locally stored conversation data and settings.","settings.data.clear.title":"Clear conversations","settings.data.clear.description":"Deletes all conversations stored in IndexedDB.","settings.data.clear.button":"Clear data","settings.data.clear.running":"Clearing…","settings.data.clear.confirm":"Confirm","settings.data.clear.success":"Conversation data cleared. Reloading…","settings.data.clear.failed":"Clear failed","settings.data.reset.title":"Reset settings","settings.data.reset.description":"Clears providers, API keys, and theme settings.","settings.data.reset.button":"Reset settings","settings.data.reset.running":"Resetting…","settings.data.reset.confirm":"Confirm","settings.data.reset.success":"Settings reset. Reloading…","settings.data.reset.failed":"Reset failed","settings.about.title":"About","settings.about.description":"Learn about Prompt Tree version and stack.","settings.about.version.label":"Version","settings.about.version.value":"Prompt Tree v1.0.0","settings.about.framework.label":"Framework","settings.about.framework.value":"Next.js 14 + React 18","settings.about.storage.label":"Storage","settings.about.storage.value":"IndexedDB + LocalStorage","settings.about.ui.label":"UI","settings.about.ui.value":"Zen Serenity","settings.tools.title":"Tools","settings.tools.description":"Configure Web Search (Exa/Tavily), MCP servers, and Python execution.","settings.tools.search.title":"Web Search","settings.tools.search.defaultProvider":"Default Provider","settings.tools.search.searchDepth":"Search Depth","settings.tools.search.maxResults":"Max Results","settings.tools.search.depth.basic":"basic","settings.tools.search.depth.advanced":"advanced","settings.tools.search.missingApiKey":"Set {provider} API Key first.","settings.tools.search.test.title":"Connection test","settings.tools.search.test.latency":"Latency {ms}ms","settings.tools.search.test.running":"Testing…","settings.tools.search.test.button":"Test connection","settings.tools.search.test.success":"Connected","settings.tools.search.test.successWithTitle":"Connected \xb7 {title}","settings.tools.search.test.failed":"Connection test failed","settings.tools.mcp.add":"Add MCP","settings.tools.mcp.description":"Configure MCP via JSON (Streamable HTTP / Legacy SSE / stdio) and set a token per server.","settings.tools.mcp.empty.title":"No MCP Servers","settings.tools.mcp.empty.description":"Click “New” above and paste your MCP config JSON.","settings.tools.mcp.logs":"Logs","settings.tools.mcp.tools":"Tools","settings.tools.mcp.expand":"Expand","settings.tools.mcp.collapse":"Collapse","settings.tools.mcp.noTools":"No tools returned (tools = 0)","settings.tools.mcp.serverNotFound":"MCP server does not exist or was removed.","settings.tools.mcp.invalidTestResponse":"Invalid MCP test response (expected SSE).","settings.tools.mcp.test.button":"Test connection","settings.tools.mcp.test.running":"Testing…","settings.tools.mcp.test.success":"Connected \xb7 {count} tools","settings.tools.mcp.test.failed":"Connection test failed","settings.tools.mcp.test.noResult":"MCP test finished without result.","settings.tools.mcp.newServerName":"New MCP Server","settings.tools.mcp.modal.title":"MCP Server","settings.tools.mcp.modal.titleWithName":"MCP Server \xb7 {name}","settings.tools.mcp.editor.serverId":"Server ID","settings.tools.mcp.editor.name":"Name","settings.tools.mcp.editor.transport":"Transport","settings.tools.mcp.editor.token":"Token","settings.tools.mcp.editor.tokenPlaceholder":"Bearer token (optional)","settings.tools.mcp.editor.configJson":"Config JSON","settings.tools.mcp.editor.configNote":"This JSON is not validated; it is parsed at runtime.","settings.tools.mcp.editor.idRequired":"Server ID is required.","settings.tools.mcp.editor.nameRequired":"Server name is required.","settings.tools.python.title":"Python Execution","settings.tools.python.timeoutSeconds":"Timeout (seconds)","settings.tools.python.maxOutputChars":"Max Output (chars)","settings.tools.python.command":"Python Command","settings.tools.python.note":"For local runs, keep this as python3.","modelSelector.providerFallback":"Provider","modelSelector.models":"Models","modelSelector.searchPlaceholder":"Search model ID or name","modelSelector.fetching":"Fetching models…","modelSelector.fetchFailed":"Failed to fetch models","modelSelector.noMatches":"No matching models","modelSelector.noModels":"No available models","modelSelector.tryOtherQuery":"Try another search query.","modelSelector.configureApiKey":"Configure API Key and refresh.","modelSelector.selectedCount":"Selected {count} models","modelSelector.addSelected":"Add selected","modelSelector.addCount":"Add {count}","modelCategory.all":"All","modelCategory.chat":"Chat","modelCategory.reasoning":"Reasoning","modelCategory.vision":"Vision","modelCategory.embedding":"Embedding","modelCategory.tool":"Tool","modelCategory.other":"Other"},"zh-CN":{"common.appName":"Prompt Tree","common.add":"添加","common.cancel":"取消","common.clear":"清空","common.close":"关闭","common.copy":"复制","common.copied":"已复制","common.data":"数据","common.delete":"删除","common.edit":"编辑","common.hide":"隐藏","common.loading":"加载中…","common.new":"新建","common.save":"保存","common.selectAll":"全选","common.settings":"设置","common.show":"显示","common.tools":"Tools","modal.closeAria":"关闭弹窗","layout.closeSidebarAria":"关闭侧边栏","layout.closeContextPanelAria":"关闭 Context 面板","node.author.assistant":"Assistant","node.author.you":"你","node.author.system":"System","node.author.compressed":"Compressed","node.type.system":"System","node.type.user":"User","node.type.assistant":"Assistant","node.type.compressed":"Compressed","header.menu":"菜单","header.context":"Context","header.fallback.folder":"文件夹","header.fallback.loading":"加载中…","header.nodes":"个节点","header.branches":"条分支","header.threads":"个线程","header.toggleTheme":"切换主题","header.theme.light":"Light","header.theme.dark":"Dark","sidebar.tagline":"Dialogue Topology","sidebar.library":"Library","sidebar.newThread":"New Thread","sidebar.newFolder":"New Folder","sidebar.deleteFolderAria":"删除文件夹","sidebar.deleteThreadAria":"删除线程","sidebar.nodeCount":"{count} 个节点","sidebar.threadCount":"{count} 个线程","sidebar.confirmDeleteFolderTitle":"确认删除文件夹?","sidebar.confirmDeleteThreadTitle":"确认删除线程?","sidebar.confirmDeleteFolderBody":"删除文件夹“{name}”。{threadNote}此操作不可撤销。","sidebar.confirmDeleteFolderThreadNote":"其中的 {count} 个线程将移动到主页。 ","sidebar.confirmDeleteThreadBody":"删除线程“{title}”。此操作不可撤销。","folder.deleteFolder":"删除文件夹","folder.deleteThreadAria":"删除线程","folder.threadPreviewAria":"线程画布预览","folder.empty.selectFolder":"请选择一个文件夹来查看线程。","folder.empty.notFound":"未找到该文件夹。","folder.label":"文件夹","folder.namePlaceholder":"文件夹名称(最多 {count} 个字符)","folder.newThread":"新建线程","folder.systemPrompt.title":"统一 System Prompt","folder.systemPrompt.collapsedEmpty":"点击添加 prompt","folder.systemPrompt.placeholder":"这将作为该文件夹内每个线程的首个(system)node。","folder.systemPrompt.note":"应用于此文件夹内的所有线程(root system node)。","folder.models.title":"启用的 Models","folder.models.allEnabled":"全部启用","folder.models.none":"无","folder.models.selectedCount":"已选择 {count} 个","folder.models.noneEnabled":"暂无启用模型。","folder.models.configureInSettings":"前往设置配置","folder.models.help":"点击编辑文件夹可用模型集合。","folder.models.noneSelected":"此文件夹未选择模型。","folder.models.picker.title":"文件夹启用模型","folder.models.picker.description":"从全局已启用模型中选择。","folder.models.picker.selected":"已选择:","folder.threads.title":"线程({count})","folder.threads.empty.title":"暂无线程","folder.threads.empty.description":"在此文件夹创建一个线程,它会继承上方统一的 system prompt。","folder.threads.empty.action":"创建线程","chat.input.placeholder":"输入消息…","chat.input.attach":"附件","chat.input.send":"发送","chat.input.modelMeta":"Model: {model}{tempPart}","chat.input.modelMetaNone":"Model: —","chat.input.tempPart":" - Temp: {temp}","chat.input.models":"Models","chat.input.enabledModels":"Enabled Models","chat.input.toolUse":"Tool Use","chat.input.tool.webSearch":"Web Search","chat.input.tool.python":"Python","chat.input.tool.pythonSubtitle":"联网 code execution","chat.message.retry":"重试","chat.message.copyResponseAria":"复制回复","chat.message.retryResponseAria":"重试回复","chat.message.toolUseEnabledTitle":"此消息已启用 tool use","chat.tool.args":"Args","chat.tool.error":"错误","chat.tool.result":"结果","chat.tool.unknownError":"未知错误","chat.tool.status.ok":"ok","chat.tool.status.error":"error","chat.tool.status.running":"运行中…","chat.thinking":"思考中…","chat.noConversationLoaded":"未加载对话。","chat.modelLabel.count":"{count} 个模型","chat.contextTokens":"Context: {used} / {max}","context.title":"Context 组装","context.subtitle":"构建你的 prompt","context.tokenUsage":"Token 用量","context.activeNodes":"已选节点","context.clearAll":"全部清除","context.toolBlocks":"Tool Blocks","context.dropzone.title":"将节点拖到这里","context.dropzone.subtitle":"松开以加入 Context","context.optimize":"优化 Context","context.preview.button":"预览完整 Context","context.preview.title":"完整 Context 预览","context.preview.building":"正在构建 Context…","context.preview.failed":"构建 Context 失败。","context.removeFromContextAria":"从 Context 移除","context.card.system":"System Prompt","context.card.user":"User Message","context.card.compressed":"Compressed","context.compress.title":"压缩节点","context.compress.selection":"选择({count})","context.compress.noneSelected":"未选择节点。","context.compress.summary":"摘要","context.compress.summaryPlaceholder":"2–3 句。可先用 AI 生成再编辑。","context.compress.language":"Language","context.compress.languagePlaceholder":"zh-CN / en","context.compress.format":"Format","context.compress.formatPlaceholder":"markdown / json","context.compress.role":"Role","context.compress.rolePlaceholder":"expert","context.compress.generate":"用 AI 生成","context.compress.confirm":"压缩","context.compress.selected":"压缩选中({count})","context.compress.context":"压缩 Context({count})","context.compress.errors.minTwoNodes":"请从同一路径选择至少 2 个节点进行压缩。","context.compress.errors.noGaps":"所选节点必须在同一条连续路径上(不能有断档)。","context.compress.errors.noBranches":"所选节点必须在同一条连续路径上(不能跨分支)。","context.compress.errors.invalidSelection":"无效选择","tree.controls.zoomOut":"缩小","tree.controls.zoomIn":"放大","tree.controls.fitView":"适配视图","tree.controls.autoLayout":"自动布局","tree.menu.continueFromHere":"从此处继续","tree.menu.compressBranch":"压缩分支","tree.menu.decompressConfirm":"解压此节点并恢复完整链路?","tree.menu.decompress":"解压","tree.menu.addToContext":"加入 Context","tree.menu.editNode":"编辑节点","tree.menu.deleteSubtree":"删除子树","tree.editor.title":"编辑节点","tree.node.emptySummary":"(空摘要)","tree.node.emptyContent":"(空)","tree.node.dragToContextAria":"拖动节点到 Context","tree.node.expandCompressedAria":"展开压缩节点","tree.node.collapseCompressedAria":"收起压缩节点","branches.title":"分支","branches.activeCount":"{count} 个","branches.default.compressed":"Compressed 分支","branches.default.type":"{type} 分支","settings.nav.configuration":"Configuration","settings.nav.providers":"模型服务","settings.nav.defaultModel":"默认 Model","settings.nav.tools":"Tools","settings.nav.general":"常规","settings.nav.display":"显示","settings.nav.data":"数据","settings.nav.about":"关于","settings.backToHome":"返回主页","providers.title":"服务提供商","providers.deleteAria":"删除提供商","providers.empty.title":"暂无提供商","providers.empty.description":"点击下方按钮添加。","providers.add":"添加提供商","providers.dialog.title":"添加提供商","providers.dialog.nameLabel":"Provider Name","providers.dialog.namePlaceholder":"OpenAI, Anthropic, DeepSeek…","providerConfig.empty.title":"请选择或添加一个提供商","providerConfig.empty.description":"从中间列表选择一个提供商来配置。","providerConfig.check.running":"检测中…","providerConfig.check.button":"检测连接","providerConfig.apiKeys.title":"API Key","providerConfig.apiKeys.namePlaceholder":"密钥名称","providerConfig.apiKeys.primary":"主密钥","providerConfig.apiKeys.setPrimary":"设为主密钥","providerConfig.apiKeys.add":"添加密钥","providerConfig.baseUrl.title":"Base URL","providerConfig.baseUrl.preview":"预览: {url}","providerConfig.models.title":"Models","providerConfig.models.select":"选择模型","providerConfig.models.enabled":"已启用 ({count})","providerConfig.models.disabled":"未启用 ({count})","providerConfig.models.streaming":"流式","providerConfig.models.empty.title":"暂无模型","providerConfig.models.empty.description":"点击「选择模型」添加。","providerConfig.addKey.title":"添加 API Key","providerConfig.addKey.nameLabel":"名称","providerConfig.addKey.namePlaceholder":"主密钥 / 备用密钥","providerConfig.addKey.valueLabel":"API Key","errors.connectionFailed":"连接失败","errors.requestFailedWithStatus":"请求失败({status})","errors.invalidRequestBody":"无效的请求体","errors.missingApiKey":"缺少 API Key","errors.invalidBaseUrl":"Base URL 无效","errors.invalidApiKey":"API Key 无效","errors.requestTimeout":"请求超时","errors.unknownError":"未知错误","errors.failedToFetchModels":"获取模型列表失败","errors.missingOpenAIApiKey":"缺少 OpenAI API Key,请在设置中添加。","errors.noActiveConversationTree":"未加载对话树。","errors.failedToSendMessage":"发送消息失败","errors.failedToRetryMessage":"重试失败","errors.failedToCompressNodes":"压缩节点失败","errors.failedToDecompressNode":"解压节点失败","errors.failedToGenerateSuggestion":"生成建议失败","settings.display.title":"显示","settings.display.description":"管理主题与语言。","settings.display.theme":"主题","settings.display.theme.light":"浅色","settings.display.theme.lightDesc":"适合白天或自然光环境。","settings.display.theme.dark":"深色","settings.display.theme.darkDesc":"适合夜间或弱光环境。","settings.display.language":"语言","settings.display.language.en":"English","settings.display.language.enDesc":"使用英文界面。","settings.display.language.zhCN":"中文","settings.display.language.zhCNDesc":"使用中文界面(保留常用英文术语)。","settings.defaultModel.title":"模型设置","settings.defaultModel.description":"选择分支模型,并配置压缩与标题生成模型。","settings.defaultModel.enabledModels.title":"已启用模型","settings.defaultModel.enabledModels.description":"选择多个模型后,发送消息将并行生成多个分支回复。","settings.defaultModel.enabledModels.emptyTitle":"暂无启用模型","settings.defaultModel.enabledModels.emptyDescription":"请先在模型服务中启用模型。","settings.defaultModel.compressionModel.title":"压缩模型","settings.defaultModel.compressionModel.placeholder":"请选择压缩模型","settings.defaultModel.compressionModel.description":"用于生成压缩节点的摘要与元指令建议。","settings.defaultModel.summaryModel.title":"标题模型","settings.defaultModel.summaryModel.placeholder":"请选择标题模型","settings.defaultModel.summaryModel.description":"用于在首条消息后生成对话标题。","settings.defaultModel.test.title":"模型测试","settings.defaultModel.test.latency":"响应时间 {ms}ms","settings.defaultModel.test.running":"测试中…","settings.defaultModel.test.button":"测试模型","settings.defaultModel.test.selectModelFirst":"请先选择模型","settings.defaultModel.test.providerMissing":"模型服务商不存在","settings.defaultModel.test.success":"连接成功","settings.defaultModel.test.failed":"测试失败","settings.general.title":"常规","settings.general.description":"管理默认 API 连接和基础生成参数。","settings.general.api.title":"API 连接","settings.general.api.apiKeyLabel":"API Key","settings.general.api.baseUrlLabel":"Base URL","settings.general.generation.title":"生成参数","settings.general.generation.temperature":"Temperature","settings.general.generation.maxTokens":"Max Tokens","settings.general.saveHint":"修改将保存到本地浏览器","settings.general.saveButton":"保存设置","settings.general.saved":"设置已保存","settings.data.title":"数据","settings.data.description":"管理本地存储的对话数据与设置。","settings.data.clear.title":"清除对话数据","settings.data.clear.description":"删除 IndexedDB 中保存的全部对话记录。","settings.data.clear.button":"清除数据","settings.data.clear.running":"清除中…","settings.data.clear.confirm":"确认清除","settings.data.clear.success":"对话数据已清除,正在刷新…","settings.data.clear.failed":"清除失败","settings.data.reset.title":"重置设置","settings.data.reset.description":"清空提供商配置、API Key 与主题设置。","settings.data.reset.button":"重置设置","settings.data.reset.running":"重置中…","settings.data.reset.confirm":"确认重置","settings.data.reset.success":"设置已重置,正在刷新…","settings.data.reset.failed":"重置失败","settings.about.title":"关于","settings.about.description":"了解 Prompt Tree 的版本与技术栈。","settings.about.version.label":"版本","settings.about.version.value":"Prompt Tree v1.0.0","settings.about.framework.label":"框架","settings.about.framework.value":"Next.js 14 + React 18","settings.about.storage.label":"数据存储","settings.about.storage.value":"IndexedDB + LocalStorage","settings.about.ui.label":"界面风格","settings.about.ui.value":"Zen Serenity","settings.tools.title":"工具","settings.tools.description":"配置 Web Search(Exa/Tavily)、MCP servers 与 Python 执行参数。","settings.tools.search.title":"Web Search","settings.tools.search.defaultProvider":"默认 Provider","settings.tools.search.searchDepth":"Search Depth","settings.tools.search.maxResults":"Max Results","settings.tools.search.depth.basic":"basic","settings.tools.search.depth.advanced":"advanced","settings.tools.search.missingApiKey":"请先配置 {provider} API Key。","settings.tools.search.test.title":"连接测试","settings.tools.search.test.latency":"响应时间 {ms}ms","settings.tools.search.test.running":"测试中…","settings.tools.search.test.button":"测试连接","settings.tools.search.test.success":"连接成功","settings.tools.search.test.successWithTitle":"连接成功 \xb7 {title}","settings.tools.search.test.failed":"连接测试失败","settings.tools.mcp.add":"添加 MCP","settings.tools.mcp.description":"使用 JSON 配置 MCP(Streamable HTTP / Legacy SSE / stdio),并为每个 server 设置 token。","settings.tools.mcp.empty.title":"暂无 MCP Servers","settings.tools.mcp.empty.description":"点击上方「新建」,粘贴你的 MCP 配置 JSON。","settings.tools.mcp.logs":"日志","settings.tools.mcp.tools":"可用工具","settings.tools.mcp.expand":"展开","settings.tools.mcp.collapse":"收起","settings.tools.mcp.noTools":"未返回工具列表(tools = 0)","settings.tools.mcp.serverNotFound":"MCP Server 不存在或已删除。","settings.tools.mcp.invalidTestResponse":"MCP 测试响应无效(需要 SSE)。","settings.tools.mcp.test.button":"测试连接","settings.tools.mcp.test.running":"测试中…","settings.tools.mcp.test.success":"连接成功 \xb7 {count} tools","settings.tools.mcp.test.failed":"测试连接失败","settings.tools.mcp.test.noResult":"MCP 测试结束但未返回结果。","settings.tools.mcp.newServerName":"New MCP Server","settings.tools.mcp.modal.title":"MCP Server","settings.tools.mcp.modal.titleWithName":"MCP Server \xb7 {name}","settings.tools.mcp.editor.serverId":"Server ID","settings.tools.mcp.editor.name":"Name","settings.tools.mcp.editor.transport":"Transport","settings.tools.mcp.editor.token":"Token","settings.tools.mcp.editor.tokenPlaceholder":"Bearer token(可选)","settings.tools.mcp.editor.configJson":"Config JSON","settings.tools.mcp.editor.configNote":"该 JSON 不做校验,运行时直接解析。","settings.tools.mcp.editor.idRequired":"Server ID 不能为空。","settings.tools.mcp.editor.nameRequired":"Server name 不能为空。","settings.tools.python.title":"Python 执行","settings.tools.python.timeoutSeconds":"超时(秒)","settings.tools.python.maxOutputChars":"最大输出(字符)","settings.tools.python.command":"Python 命令","settings.tools.python.note":"本地运行建议保持为 python3。","modelSelector.providerFallback":"提供商","modelSelector.models":"模型","modelSelector.searchPlaceholder":"搜索 model ID 或名称","modelSelector.fetching":"正在获取模型列表…","modelSelector.fetchFailed":"获取模型列表失败","modelSelector.noMatches":"未找到匹配的模型","modelSelector.noModels":"暂无可用模型","modelSelector.tryOtherQuery":"尝试其他搜索关键词。","modelSelector.configureApiKey":"请先配置 API Key 并刷新。","modelSelector.selectedCount":"已选择 {count} 个模型","modelSelector.addSelected":"添加选中","modelSelector.addCount":"添加 {count}","modelCategory.all":"全部","modelCategory.chat":"对话","modelCategory.reasoning":"推理","modelCategory.vision":"视觉","modelCategory.embedding":"嵌入","modelCategory.tool":"工具","modelCategory.other":"其他"}};function e(a,b,c){var e;return e=(d[a]??d.en)[b]??d.en[b],c?e.replaceAll(/\{(\w+)\}/g,(a,b)=>{let d=c[b];return null==d?"":String(d)}):e}function f(a){return Object.prototype.hasOwnProperty.call(d.en,a)}},2815:(a,b,c)=>{c.d(b,{a:()=>f});var d=c(8249);c(7484);var e=c(4995);function f({open:a,title:b,children:c,onClose:f}){let g=(0,e.k)();return a?(0,d.jsxs)("div",{className:"fixed inset-0 z-50 flex items-center justify-center p-6",children:[(0,d.jsx)("button",{className:"absolute inset-0 bg-ink/30","aria-label":g("modal.closeAria"),onClick:f}),(0,d.jsxs)("div",{className:"relative w-full max-w-[520px] rounded-2xl border border-parchment bg-cream p-6 shadow-[0_22px_60px_rgba(35,31,28,0.18)]",children:[(0,d.jsx)("div",{className:"mb-4 font-display text-[1.15rem] text-ink",children:b}),c]})]}):null}},4995:(a,b,c)=>{c.d(b,{k:()=>g});var d=c(7484),e=c(3006),f=c(2115);function g(){let a=(0,e.CU)(a=>a.locale);return(0,d.useMemo)(()=>(b,c)=>(0,f.t)(a,b,c),[a])}},8655:(a,b,c)=>{function d(a){return`${a.providerId}:${a.modelId}`}function e(a){let b=[];for(let c of a)for(let a of c.models.filter(a=>a.enabled)){let d=c.name?`${c.name} \xb7 ${a.id}`:a.id;b.push({providerId:c.id,providerName:c.name,modelId:a.id,label:d})}return b.sort((a,b)=>{let c=a.providerName.localeCompare(b.providerName);return 0!==c?c:a.modelId.localeCompare(b.modelId)}),b}c.d(b,{R:()=>d,n:()=>e})}};
|
|
1
|
+
"use strict";exports.id=689,exports.ids=[689],exports.modules={2115:(a,b,c)=>{c.d(b,{d:()=>f,t:()=>e});let d={en:{"common.appName":"Prompt Tree","common.add":"Add","common.cancel":"Cancel","common.clear":"Clear","common.close":"Close","common.copy":"Copy","common.copied":"Copied","common.data":"Data","common.delete":"Delete","common.edit":"Edit","common.hide":"Hide","common.loading":"Loading…","common.new":"New","common.save":"Save","common.selectAll":"Select all","common.settings":"Settings","common.show":"Show","common.tools":"Tools","modal.closeAria":"Close modal","layout.closeSidebarAria":"Close sidebar","layout.closeContextPanelAria":"Close context panel","node.author.assistant":"Assistant","node.author.you":"You","node.author.system":"System","node.author.compressed":"Compressed","node.type.system":"System","node.type.user":"User","node.type.assistant":"Assistant","node.type.compressed":"Compressed","header.menu":"Menu","header.context":"Context","header.fallback.folder":"Folder","header.fallback.loading":"Loading…","header.nodes":"nodes","header.branches":"branches","header.threads":"threads","header.toggleTheme":"Toggle theme","header.theme.light":"Light","header.theme.dark":"Dark","sidebar.tagline":"Dialogue Topology","sidebar.library":"Library","sidebar.newThread":"New Thread","sidebar.newFolder":"New Folder","sidebar.deleteFolderAria":"Delete folder","sidebar.deleteThreadAria":"Delete thread","sidebar.nodeCount":"{count} nodes","sidebar.threadCount":"{count} threads","sidebar.confirmDeleteFolderTitle":"Delete folder?","sidebar.confirmDeleteThreadTitle":"Delete thread?","sidebar.confirmDeleteFolderBody":"Delete folder “{name}”. {threadNote}This action cannot be undone.","sidebar.confirmDeleteFolderThreadNote":"{count} threads will be moved to Home. ","sidebar.confirmDeleteThreadBody":"Delete thread “{title}”. This action cannot be undone.","folder.deleteFolder":"Delete folder","folder.deleteThreadAria":"Delete thread","folder.threadPreviewAria":"Thread canvas preview","folder.empty.selectFolder":"Select a folder to view its threads.","folder.empty.notFound":"Folder not found.","folder.label":"Folder","folder.namePlaceholder":"Folder name (max {count} chars)","folder.newThread":"New Thread","folder.systemPrompt.title":"Unified System Prompt","folder.systemPrompt.collapsedEmpty":"Tap to add a prompt","folder.systemPrompt.placeholder":"This will be the first (system) node of every thread in this folder.","folder.systemPrompt.note":"Applies to all threads in this folder (root system node).","folder.models.title":"Enabled Models","folder.models.allEnabled":"All enabled","folder.models.none":"None","folder.models.selectedCount":"{count} selected","folder.models.noneEnabled":"No models enabled.","folder.models.configureInSettings":"Configure in Settings","folder.models.help":"Click to edit the folder model set.","folder.models.noneSelected":"No models selected for this folder.","folder.models.picker.title":"Folder Enabled Models","folder.models.picker.description":"Pick from globally enabled models.","folder.models.picker.selected":"Selected:","folder.threads.title":"Threads ({count})","folder.threads.empty.title":"No threads yet","folder.threads.empty.description":"Create a thread in this folder. It will inherit the unified system prompt above.","folder.threads.empty.action":"Create Thread","chat.input.placeholder":"Type your message…","chat.input.attach":"Attach","chat.input.send":"Send","chat.input.modelMeta":"Model: {model}{tempPart}","chat.input.modelMetaNone":"Model: —","chat.input.tempPart":" - Temp: {temp}","chat.input.models":"Models","chat.input.enabledModels":"Enabled Models","chat.input.toolUse":"Tool Use","chat.input.tool.webSearch":"Web Search","chat.input.tool.python":"Python","chat.input.tool.pythonSubtitle":"Online code execution","chat.message.retry":"Retry","chat.message.copyResponseAria":"Copy response","chat.message.retryResponseAria":"Retry response","chat.message.toolUseEnabledTitle":"Tool use enabled for this message","chat.tool.args":"Args","chat.tool.error":"Error","chat.tool.result":"Result","chat.tool.unknownError":"Unknown error","chat.tool.status.ok":"ok","chat.tool.status.error":"error","chat.tool.status.running":"running…","chat.thinking":"Thinking…","chat.noConversationLoaded":"No conversation loaded.","chat.modelLabel.count":"{count} models","chat.contextTokens":"Context: {used} / {max}","context.title":"Context Assembly","context.subtitle":"Build your prompt","context.tokenUsage":"Token Usage","context.activeNodes":"Active Nodes","context.clearAll":"Clear all","context.toolBlocks":"Tool Blocks","context.dropzone.title":"Drag nodes here","context.dropzone.subtitle":"Drop to insert into context","context.optimize":"Optimize Context","context.preview.button":"Preview Full Context","context.preview.title":"Full Context Preview","context.preview.building":"Building context…","context.preview.failed":"Failed to build context.","context.removeFromContextAria":"Remove from context","context.card.system":"System Prompt","context.card.user":"User Message","context.card.compressed":"Compressed","context.compress.title":"Compress Nodes","context.compress.selection":"Selection ({count})","context.compress.noneSelected":"No nodes selected.","context.compress.summary":"Summary","context.compress.summaryPlaceholder":"2–3 sentences. You can generate with AI, then edit.","context.compress.language":"Language","context.compress.languagePlaceholder":"zh-CN / en","context.compress.format":"Format","context.compress.formatPlaceholder":"markdown / json","context.compress.role":"Role","context.compress.rolePlaceholder":"expert","context.compress.generate":"Generate with AI","context.compress.confirm":"Compress","context.compress.selected":"Compress Selected ({count})","context.compress.context":"Compress Context ({count})","context.compress.errors.minTwoNodes":"Select at least 2 nodes from the same path to compress.","context.compress.errors.noGaps":"Selection must be a single continuous path (no gaps).","context.compress.errors.noBranches":"Selection must be a single continuous path (no branches).","context.compress.errors.invalidSelection":"Invalid selection","tree.controls.zoomOut":"Zoom out","tree.controls.zoomIn":"Zoom in","tree.controls.fitView":"Fit view","tree.controls.autoLayout":"Auto layout","tree.menu.continueFromHere":"Continue from here","tree.menu.compressBranch":"Compress branch","tree.menu.decompressConfirm":"Decompress this node and restore the full chain?","tree.menu.decompress":"Decompress","tree.menu.addToContext":"Add to Context","tree.menu.editNode":"Edit node","tree.menu.deleteSubtree":"Delete subtree","tree.editor.title":"Edit Node","tree.node.emptySummary":"(empty summary)","tree.node.emptyContent":"(empty)","tree.node.dragToContextAria":"Drag node to context","tree.node.expandCompressedAria":"Expand compressed node","tree.node.collapseCompressedAria":"Collapse compressed node","branches.title":"Branches","branches.activeCount":"{count} active","branches.default.compressed":"Compressed branch","branches.default.type":"{type} branch","settings.nav.configuration":"Configuration","settings.nav.providers":"Model Service","settings.nav.defaultModel":"Default Model","settings.nav.tools":"Tools","settings.nav.general":"Participation","settings.nav.display":"Display","settings.nav.data":"Data","settings.nav.about":"About","settings.backToHome":"Back to Home","providers.title":"Service Providers","providers.deleteAria":"Delete provider","providers.empty.title":"No providers yet","providers.empty.description":"Click below to add one.","providers.add":"Add provider","providers.dialog.title":"Add provider","providers.dialog.nameLabel":"Provider Name","providers.dialog.namePlaceholder":"OpenAI, Anthropic, DeepSeek…","providerConfig.empty.title":"Select or add a provider","providerConfig.empty.description":"Choose a provider from the middle list to configure.","providerConfig.check.running":"Checking…","providerConfig.check.button":"Test connection","providerConfig.apiKeys.title":"API Key","providerConfig.apiKeys.namePlaceholder":"Key name","providerConfig.apiKeys.primary":"Primary","providerConfig.apiKeys.setPrimary":"Set as primary","providerConfig.apiKeys.add":"Add key","providerConfig.baseUrl.title":"Base URL","providerConfig.baseUrl.preview":"Preview: {url}","providerConfig.models.title":"Models","providerConfig.models.select":"Select models","providerConfig.models.enabled":"Enabled ({count})","providerConfig.models.disabled":"Disabled ({count})","providerConfig.models.streaming":"Streaming","providerConfig.models.empty.title":"No models","providerConfig.models.empty.description":"Click “Select models” to add.","providerConfig.addKey.title":"Add API Key","providerConfig.addKey.nameLabel":"Name","providerConfig.addKey.namePlaceholder":"Primary / backup","providerConfig.addKey.valueLabel":"API Key","errors.connectionFailed":"Connection failed","errors.requestFailedWithStatus":"Request failed ({status})","errors.invalidRequestBody":"Invalid request body","errors.missingApiKey":"Missing API key","errors.invalidBaseUrl":"Invalid Base URL","errors.invalidApiKey":"Invalid API key","errors.requestTimeout":"Request timeout","errors.unknownError":"Unknown error","errors.failedToFetchModels":"Failed to fetch models","errors.missingOpenAIApiKey":"Missing OpenAI API key. Add it in Settings.","errors.noActiveConversationTree":"No active conversation tree loaded.","errors.failedToSendMessage":"Failed to send message","errors.failedToRetryMessage":"Failed to retry message","errors.failedToCompressNodes":"Failed to compress nodes","errors.failedToDecompressNode":"Failed to decompress node","errors.failedToGenerateSuggestion":"Failed to generate suggestion","settings.display.title":"Display","settings.display.description":"Manage theme and language.","settings.display.theme":"Theme","settings.display.theme.light":"Light","settings.display.theme.lightDesc":"Best for daylight environments.","settings.display.theme.dark":"Dark","settings.display.theme.darkDesc":"Best for low-light environments.","settings.display.language":"Language","settings.display.language.en":"English","settings.display.language.enDesc":"Use English UI copy.","settings.display.language.zhCN":"中文","settings.display.language.zhCNDesc":"使用中文界面(保留常用英文术语)。","settings.defaultModel.title":"Model Settings","settings.defaultModel.description":"Select branch models and configure compression + title generation.","settings.defaultModel.enabledModels.title":"Enabled models","settings.defaultModel.enabledModels.description":"When multiple models are selected, sending a message generates parallel branches.","settings.defaultModel.enabledModels.emptyTitle":"No enabled models","settings.defaultModel.enabledModels.emptyDescription":"Enable models first in Model Service.","settings.defaultModel.compressionModel.title":"Compression model","settings.defaultModel.compressionModel.placeholder":"Select a compression model","settings.defaultModel.compressionModel.description":"Used to generate compressed summaries and meta-instruction suggestions.","settings.defaultModel.summaryModel.title":"Title model","settings.defaultModel.summaryModel.placeholder":"Select a title model","settings.defaultModel.summaryModel.description":"Used to generate the conversation title after the first message.","settings.defaultModel.test.title":"Model test","settings.defaultModel.test.latency":"Latency {ms}ms","settings.defaultModel.test.running":"Testing…","settings.defaultModel.test.button":"Test model","settings.defaultModel.test.selectModelFirst":"Select a model first.","settings.defaultModel.test.providerMissing":"Provider not found.","settings.defaultModel.test.success":"Connected","settings.defaultModel.test.failed":"Test failed","settings.general.title":"Participation","settings.general.description":"Manage generation parameters.","settings.general.api.title":"API","settings.general.api.apiKeyLabel":"API Key","settings.general.api.baseUrlLabel":"Base URL","settings.general.generation.title":"Parameters","settings.general.generation.temperature":"Temperature","settings.general.generation.maxTokens":"Max Tokens","settings.general.saveHint":"Changes are saved locally in your browser.","settings.general.saveButton":"Save settings","settings.general.saved":"Saved","settings.data.title":"Data","settings.data.description":"Manage locally stored conversation data and settings.","settings.data.clear.title":"Clear conversations","settings.data.clear.description":"Deletes all conversations stored in IndexedDB.","settings.data.clear.button":"Clear data","settings.data.clear.running":"Clearing…","settings.data.clear.confirm":"Confirm","settings.data.clear.success":"Conversation data cleared. Reloading…","settings.data.clear.failed":"Clear failed","settings.data.reset.title":"Reset settings","settings.data.reset.description":"Clears providers, API keys, and theme settings.","settings.data.reset.button":"Reset settings","settings.data.reset.running":"Resetting…","settings.data.reset.confirm":"Confirm","settings.data.reset.success":"Settings reset. Reloading…","settings.data.reset.failed":"Reset failed","settings.about.title":"About","settings.about.description":"Learn about Prompt Tree version and stack.","settings.about.version.label":"Version","settings.about.version.value":"Prompt Tree v0.3.0","settings.about.framework.label":"Framework","settings.about.framework.value":"Next.js 14 + React 18","settings.about.storage.label":"Storage","settings.about.storage.value":"IndexedDB + LocalStorage","settings.about.contact.email":"Email","settings.about.contact.wechat":"WeChat","settings.tools.title":"Tools","settings.tools.description":"Configure Web Search (Exa/Tavily), MCP servers, and Python execution.","settings.tools.search.title":"Web Search","settings.tools.search.defaultProvider":"Default Provider","settings.tools.search.searchDepth":"Search Depth","settings.tools.search.maxResults":"Max Results","settings.tools.search.depth.basic":"basic","settings.tools.search.depth.advanced":"advanced","settings.tools.search.missingApiKey":"Set {provider} API Key first.","settings.tools.search.test.title":"Connection test","settings.tools.search.test.latency":"Latency {ms}ms","settings.tools.search.test.running":"Testing…","settings.tools.search.test.button":"Test connection","settings.tools.search.test.success":"Connected","settings.tools.search.test.successWithTitle":"Connected \xb7 {title}","settings.tools.search.test.failed":"Connection test failed","settings.tools.mcp.add":"Add MCP","settings.tools.mcp.description":"Configure MCP via JSON (Streamable HTTP / Legacy SSE / stdio) and set a token per server.","settings.tools.mcp.empty.title":"No MCP Servers","settings.tools.mcp.empty.description":"Click “New” above and paste your MCP config JSON.","settings.tools.mcp.logs":"Logs","settings.tools.mcp.tools":"Tools","settings.tools.mcp.expand":"Expand","settings.tools.mcp.collapse":"Collapse","settings.tools.mcp.noTools":"No tools returned (tools = 0)","settings.tools.mcp.serverNotFound":"MCP server does not exist or was removed.","settings.tools.mcp.invalidTestResponse":"Invalid MCP test response (expected SSE).","settings.tools.mcp.test.button":"Test connection","settings.tools.mcp.test.running":"Testing…","settings.tools.mcp.test.success":"Connected \xb7 {count} tools","settings.tools.mcp.test.failed":"Connection test failed","settings.tools.mcp.test.noResult":"MCP test finished without result.","settings.tools.mcp.newServerName":"New MCP Server","settings.tools.mcp.modal.title":"MCP Server","settings.tools.mcp.modal.titleWithName":"MCP Server \xb7 {name}","settings.tools.mcp.editor.serverId":"Server ID","settings.tools.mcp.editor.name":"Name","settings.tools.mcp.editor.transport":"Transport","settings.tools.mcp.editor.token":"Token","settings.tools.mcp.editor.tokenPlaceholder":"Bearer token (optional)","settings.tools.mcp.editor.configJson":"Config JSON","settings.tools.mcp.editor.configNote":"This JSON is not validated; it is parsed at runtime.","settings.tools.mcp.editor.idRequired":"Server ID is required.","settings.tools.mcp.editor.nameRequired":"Server name is required.","settings.tools.python.title":"Python Execution","settings.tools.python.timeoutSeconds":"Timeout (seconds)","settings.tools.python.maxOutputChars":"Max Output (chars)","settings.tools.python.command":"Python Command","settings.tools.python.note":"For local runs, keep this as python3.","modelSelector.providerFallback":"Provider","modelSelector.models":"Models","modelSelector.searchPlaceholder":"Search model ID or name","modelSelector.fetching":"Fetching models…","modelSelector.fetchFailed":"Failed to fetch models","modelSelector.noMatches":"No matching models","modelSelector.noModels":"No available models","modelSelector.tryOtherQuery":"Try another search query.","modelSelector.configureApiKey":"Configure API Key and refresh.","modelSelector.selectedCount":"Selected {count} models","modelSelector.addSelected":"Add selected","modelSelector.addCount":"Add {count}","modelCategory.all":"All","modelCategory.chat":"Chat","modelCategory.reasoning":"Reasoning","modelCategory.vision":"Vision","modelCategory.embedding":"Embedding","modelCategory.tool":"Tool","modelCategory.other":"Other"},"zh-CN":{"common.appName":"Prompt Tree","common.add":"添加","common.cancel":"取消","common.clear":"清空","common.close":"关闭","common.copy":"复制","common.copied":"已复制","common.data":"数据","common.delete":"删除","common.edit":"编辑","common.hide":"隐藏","common.loading":"加载中…","common.new":"新建","common.save":"保存","common.selectAll":"全选","common.settings":"设置","common.show":"显示","common.tools":"Tools","modal.closeAria":"关闭弹窗","layout.closeSidebarAria":"关闭侧边栏","layout.closeContextPanelAria":"关闭 Context 面板","node.author.assistant":"Assistant","node.author.you":"你","node.author.system":"System","node.author.compressed":"Compressed","node.type.system":"System","node.type.user":"User","node.type.assistant":"Assistant","node.type.compressed":"Compressed","header.menu":"菜单","header.context":"Context","header.fallback.folder":"文件夹","header.fallback.loading":"加载中…","header.nodes":"个节点","header.branches":"条分支","header.threads":"个线程","header.toggleTheme":"切换主题","header.theme.light":"Light","header.theme.dark":"Dark","sidebar.tagline":"Dialogue Topology","sidebar.library":"Library","sidebar.newThread":"New Thread","sidebar.newFolder":"New Folder","sidebar.deleteFolderAria":"删除文件夹","sidebar.deleteThreadAria":"删除线程","sidebar.nodeCount":"{count} 个节点","sidebar.threadCount":"{count} 个线程","sidebar.confirmDeleteFolderTitle":"确认删除文件夹?","sidebar.confirmDeleteThreadTitle":"确认删除线程?","sidebar.confirmDeleteFolderBody":"删除文件夹“{name}”。{threadNote}此操作不可撤销。","sidebar.confirmDeleteFolderThreadNote":"其中的 {count} 个线程将移动到主页。 ","sidebar.confirmDeleteThreadBody":"删除线程“{title}”。此操作不可撤销。","folder.deleteFolder":"删除文件夹","folder.deleteThreadAria":"删除线程","folder.threadPreviewAria":"线程画布预览","folder.empty.selectFolder":"请选择一个文件夹来查看线程。","folder.empty.notFound":"未找到该文件夹。","folder.label":"文件夹","folder.namePlaceholder":"文件夹名称(最多 {count} 个字符)","folder.newThread":"新建线程","folder.systemPrompt.title":"统一 System Prompt","folder.systemPrompt.collapsedEmpty":"点击添加 prompt","folder.systemPrompt.placeholder":"这将作为该文件夹内每个线程的首个(system)node。","folder.systemPrompt.note":"应用于此文件夹内的所有线程(root system node)。","folder.models.title":"启用的 Models","folder.models.allEnabled":"全部启用","folder.models.none":"无","folder.models.selectedCount":"已选择 {count} 个","folder.models.noneEnabled":"暂无启用模型。","folder.models.configureInSettings":"前往设置配置","folder.models.help":"点击编辑文件夹可用模型集合。","folder.models.noneSelected":"此文件夹未选择模型。","folder.models.picker.title":"文件夹启用模型","folder.models.picker.description":"从全局已启用模型中选择。","folder.models.picker.selected":"已选择:","folder.threads.title":"线程({count})","folder.threads.empty.title":"暂无线程","folder.threads.empty.description":"在此文件夹创建一个线程,它会继承上方统一的 system prompt。","folder.threads.empty.action":"创建线程","chat.input.placeholder":"输入消息…","chat.input.attach":"附件","chat.input.send":"发送","chat.input.modelMeta":"Model: {model}{tempPart}","chat.input.modelMetaNone":"Model: —","chat.input.tempPart":" - Temp: {temp}","chat.input.models":"Models","chat.input.enabledModels":"Enabled Models","chat.input.toolUse":"Tool Use","chat.input.tool.webSearch":"Web Search","chat.input.tool.python":"Python","chat.input.tool.pythonSubtitle":"联网 code execution","chat.message.retry":"重试","chat.message.copyResponseAria":"复制回复","chat.message.retryResponseAria":"重试回复","chat.message.toolUseEnabledTitle":"此消息已启用 tool use","chat.tool.args":"Args","chat.tool.error":"错误","chat.tool.result":"结果","chat.tool.unknownError":"未知错误","chat.tool.status.ok":"ok","chat.tool.status.error":"error","chat.tool.status.running":"运行中…","chat.thinking":"思考中…","chat.noConversationLoaded":"未加载对话。","chat.modelLabel.count":"{count} 个模型","chat.contextTokens":"Context: {used} / {max}","context.title":"Context 组装","context.subtitle":"构建你的 prompt","context.tokenUsage":"Token 用量","context.activeNodes":"已选节点","context.clearAll":"全部清除","context.toolBlocks":"Tool Blocks","context.dropzone.title":"将节点拖到这里","context.dropzone.subtitle":"松开以加入 Context","context.optimize":"优化 Context","context.preview.button":"预览完整 Context","context.preview.title":"完整 Context 预览","context.preview.building":"正在构建 Context…","context.preview.failed":"构建 Context 失败。","context.removeFromContextAria":"从 Context 移除","context.card.system":"System Prompt","context.card.user":"User Message","context.card.compressed":"Compressed","context.compress.title":"压缩节点","context.compress.selection":"选择({count})","context.compress.noneSelected":"未选择节点。","context.compress.summary":"摘要","context.compress.summaryPlaceholder":"2–3 句。可先用 AI 生成再编辑。","context.compress.language":"Language","context.compress.languagePlaceholder":"zh-CN / en","context.compress.format":"Format","context.compress.formatPlaceholder":"markdown / json","context.compress.role":"Role","context.compress.rolePlaceholder":"expert","context.compress.generate":"用 AI 生成","context.compress.confirm":"压缩","context.compress.selected":"压缩选中({count})","context.compress.context":"压缩 Context({count})","context.compress.errors.minTwoNodes":"请从同一路径选择至少 2 个节点进行压缩。","context.compress.errors.noGaps":"所选节点必须在同一条连续路径上(不能有断档)。","context.compress.errors.noBranches":"所选节点必须在同一条连续路径上(不能跨分支)。","context.compress.errors.invalidSelection":"无效选择","tree.controls.zoomOut":"缩小","tree.controls.zoomIn":"放大","tree.controls.fitView":"适配视图","tree.controls.autoLayout":"自动布局","tree.menu.continueFromHere":"从此处继续","tree.menu.compressBranch":"压缩分支","tree.menu.decompressConfirm":"解压此节点并恢复完整链路?","tree.menu.decompress":"解压","tree.menu.addToContext":"加入 Context","tree.menu.editNode":"编辑节点","tree.menu.deleteSubtree":"删除子树","tree.editor.title":"编辑节点","tree.node.emptySummary":"(空摘要)","tree.node.emptyContent":"(空)","tree.node.dragToContextAria":"拖动节点到 Context","tree.node.expandCompressedAria":"展开压缩节点","tree.node.collapseCompressedAria":"收起压缩节点","branches.title":"分支","branches.activeCount":"{count} 个","branches.default.compressed":"Compressed 分支","branches.default.type":"{type} 分支","settings.nav.configuration":"Configuration","settings.nav.providers":"模型服务","settings.nav.defaultModel":"默认 Model","settings.nav.tools":"Tools","settings.nav.general":"参与","settings.nav.display":"显示","settings.nav.data":"数据","settings.nav.about":"关于","settings.backToHome":"返回主页","providers.title":"服务提供商","providers.deleteAria":"删除提供商","providers.empty.title":"暂无提供商","providers.empty.description":"点击下方按钮添加。","providers.add":"添加提供商","providers.dialog.title":"添加提供商","providers.dialog.nameLabel":"Provider Name","providers.dialog.namePlaceholder":"OpenAI, Anthropic, DeepSeek…","providerConfig.empty.title":"请选择或添加一个提供商","providerConfig.empty.description":"从中间列表选择一个提供商来配置。","providerConfig.check.running":"检测中…","providerConfig.check.button":"检测连接","providerConfig.apiKeys.title":"API Key","providerConfig.apiKeys.namePlaceholder":"密钥名称","providerConfig.apiKeys.primary":"主密钥","providerConfig.apiKeys.setPrimary":"设为主密钥","providerConfig.apiKeys.add":"添加密钥","providerConfig.baseUrl.title":"Base URL","providerConfig.baseUrl.preview":"预览: {url}","providerConfig.models.title":"Models","providerConfig.models.select":"选择模型","providerConfig.models.enabled":"已启用 ({count})","providerConfig.models.disabled":"未启用 ({count})","providerConfig.models.streaming":"流式","providerConfig.models.empty.title":"暂无模型","providerConfig.models.empty.description":"点击「选择模型」添加。","providerConfig.addKey.title":"添加 API Key","providerConfig.addKey.nameLabel":"名称","providerConfig.addKey.namePlaceholder":"主密钥 / 备用密钥","providerConfig.addKey.valueLabel":"API Key","errors.connectionFailed":"连接失败","errors.requestFailedWithStatus":"请求失败({status})","errors.invalidRequestBody":"无效的请求体","errors.missingApiKey":"缺少 API Key","errors.invalidBaseUrl":"Base URL 无效","errors.invalidApiKey":"API Key 无效","errors.requestTimeout":"请求超时","errors.unknownError":"未知错误","errors.failedToFetchModels":"获取模型列表失败","errors.missingOpenAIApiKey":"缺少 OpenAI API Key,请在设置中添加。","errors.noActiveConversationTree":"未加载对话树。","errors.failedToSendMessage":"发送消息失败","errors.failedToRetryMessage":"重试失败","errors.failedToCompressNodes":"压缩节点失败","errors.failedToDecompressNode":"解压节点失败","errors.failedToGenerateSuggestion":"生成建议失败","settings.display.title":"显示","settings.display.description":"管理主题与语言。","settings.display.theme":"主题","settings.display.theme.light":"浅色","settings.display.theme.lightDesc":"适合白天或自然光环境。","settings.display.theme.dark":"深色","settings.display.theme.darkDesc":"适合夜间或弱光环境。","settings.display.language":"语言","settings.display.language.en":"English","settings.display.language.enDesc":"使用英文界面。","settings.display.language.zhCN":"中文","settings.display.language.zhCNDesc":"使用中文界面(保留常用英文术语)。","settings.defaultModel.title":"模型设置","settings.defaultModel.description":"选择分支模型,并配置压缩与标题生成模型。","settings.defaultModel.enabledModels.title":"已启用模型","settings.defaultModel.enabledModels.description":"选择多个模型后,发送消息将并行生成多个分支回复。","settings.defaultModel.enabledModels.emptyTitle":"暂无启用模型","settings.defaultModel.enabledModels.emptyDescription":"请先在模型服务中启用模型。","settings.defaultModel.compressionModel.title":"压缩模型","settings.defaultModel.compressionModel.placeholder":"请选择压缩模型","settings.defaultModel.compressionModel.description":"用于生成压缩节点的摘要与元指令建议。","settings.defaultModel.summaryModel.title":"标题模型","settings.defaultModel.summaryModel.placeholder":"请选择标题模型","settings.defaultModel.summaryModel.description":"用于在首条消息后生成对话标题。","settings.defaultModel.test.title":"模型测试","settings.defaultModel.test.latency":"响应时间 {ms}ms","settings.defaultModel.test.running":"测试中…","settings.defaultModel.test.button":"测试模型","settings.defaultModel.test.selectModelFirst":"请先选择模型","settings.defaultModel.test.providerMissing":"模型服务商不存在","settings.defaultModel.test.success":"连接成功","settings.defaultModel.test.failed":"测试失败","settings.general.title":"参与","settings.general.description":"管理生成参数。","settings.general.api.title":"API 连接","settings.general.api.apiKeyLabel":"API Key","settings.general.api.baseUrlLabel":"Base URL","settings.general.generation.title":"生成参数","settings.general.generation.temperature":"Temperature","settings.general.generation.maxTokens":"Max Tokens","settings.general.saveHint":"修改将保存到本地浏览器","settings.general.saveButton":"保存设置","settings.general.saved":"设置已保存","settings.data.title":"数据","settings.data.description":"管理本地存储的对话数据与设置。","settings.data.clear.title":"清除对话数据","settings.data.clear.description":"删除 IndexedDB 中保存的全部对话记录。","settings.data.clear.button":"清除数据","settings.data.clear.running":"清除中…","settings.data.clear.confirm":"确认清除","settings.data.clear.success":"对话数据已清除,正在刷新…","settings.data.clear.failed":"清除失败","settings.data.reset.title":"重置设置","settings.data.reset.description":"清空提供商配置、API Key 与主题设置。","settings.data.reset.button":"重置设置","settings.data.reset.running":"重置中…","settings.data.reset.confirm":"确认重置","settings.data.reset.success":"设置已重置,正在刷新…","settings.data.reset.failed":"重置失败","settings.about.title":"关于","settings.about.description":"了解 Prompt Tree 的版本与技术栈。","settings.about.version.label":"版本","settings.about.version.value":"Prompt Tree v0.3.0","settings.about.framework.label":"框架","settings.about.framework.value":"Next.js 14 + React 18","settings.about.storage.label":"数据存储","settings.about.storage.value":"IndexedDB + LocalStorage","settings.about.contact.email":"Email","settings.about.contact.wechat":"WeChat","settings.tools.title":"工具","settings.tools.description":"配置 Web Search(Exa/Tavily)、MCP servers 与 Python 执行参数。","settings.tools.search.title":"Web Search","settings.tools.search.defaultProvider":"默认 Provider","settings.tools.search.searchDepth":"Search Depth","settings.tools.search.maxResults":"Max Results","settings.tools.search.depth.basic":"basic","settings.tools.search.depth.advanced":"advanced","settings.tools.search.missingApiKey":"请先配置 {provider} API Key。","settings.tools.search.test.title":"连接测试","settings.tools.search.test.latency":"响应时间 {ms}ms","settings.tools.search.test.running":"测试中…","settings.tools.search.test.button":"测试连接","settings.tools.search.test.success":"连接成功","settings.tools.search.test.successWithTitle":"连接成功 \xb7 {title}","settings.tools.search.test.failed":"连接测试失败","settings.tools.mcp.add":"添加 MCP","settings.tools.mcp.description":"使用 JSON 配置 MCP(Streamable HTTP / Legacy SSE / stdio),并为每个 server 设置 token。","settings.tools.mcp.empty.title":"暂无 MCP Servers","settings.tools.mcp.empty.description":"点击上方「新建」,粘贴你的 MCP 配置 JSON。","settings.tools.mcp.logs":"日志","settings.tools.mcp.tools":"可用工具","settings.tools.mcp.expand":"展开","settings.tools.mcp.collapse":"收起","settings.tools.mcp.noTools":"未返回工具列表(tools = 0)","settings.tools.mcp.serverNotFound":"MCP Server 不存在或已删除。","settings.tools.mcp.invalidTestResponse":"MCP 测试响应无效(需要 SSE)。","settings.tools.mcp.test.button":"测试连接","settings.tools.mcp.test.running":"测试中…","settings.tools.mcp.test.success":"连接成功 \xb7 {count} tools","settings.tools.mcp.test.failed":"测试连接失败","settings.tools.mcp.test.noResult":"MCP 测试结束但未返回结果。","settings.tools.mcp.newServerName":"New MCP Server","settings.tools.mcp.modal.title":"MCP Server","settings.tools.mcp.modal.titleWithName":"MCP Server \xb7 {name}","settings.tools.mcp.editor.serverId":"Server ID","settings.tools.mcp.editor.name":"Name","settings.tools.mcp.editor.transport":"Transport","settings.tools.mcp.editor.token":"Token","settings.tools.mcp.editor.tokenPlaceholder":"Bearer token(可选)","settings.tools.mcp.editor.configJson":"Config JSON","settings.tools.mcp.editor.configNote":"该 JSON 不做校验,运行时直接解析。","settings.tools.mcp.editor.idRequired":"Server ID 不能为空。","settings.tools.mcp.editor.nameRequired":"Server name 不能为空。","settings.tools.python.title":"Python 执行","settings.tools.python.timeoutSeconds":"超时(秒)","settings.tools.python.maxOutputChars":"最大输出(字符)","settings.tools.python.command":"Python 命令","settings.tools.python.note":"本地运行建议保持为 python3。","modelSelector.providerFallback":"提供商","modelSelector.models":"模型","modelSelector.searchPlaceholder":"搜索 model ID 或名称","modelSelector.fetching":"正在获取模型列表…","modelSelector.fetchFailed":"获取模型列表失败","modelSelector.noMatches":"未找到匹配的模型","modelSelector.noModels":"暂无可用模型","modelSelector.tryOtherQuery":"尝试其他搜索关键词。","modelSelector.configureApiKey":"请先配置 API Key 并刷新。","modelSelector.selectedCount":"已选择 {count} 个模型","modelSelector.addSelected":"添加选中","modelSelector.addCount":"添加 {count}","modelCategory.all":"全部","modelCategory.chat":"对话","modelCategory.reasoning":"推理","modelCategory.vision":"视觉","modelCategory.embedding":"嵌入","modelCategory.tool":"工具","modelCategory.other":"其他"}};function e(a,b,c){var e;return e=(d[a]??d.en)[b]??d.en[b],c?e.replaceAll(/\{(\w+)\}/g,(a,b)=>{let d=c[b];return null==d?"":String(d)}):e}function f(a){return Object.prototype.hasOwnProperty.call(d.en,a)}},2815:(a,b,c)=>{c.d(b,{a:()=>f});var d=c(8249);c(7484);var e=c(4995);function f({open:a,title:b,children:c,onClose:f}){let g=(0,e.k)();return a?(0,d.jsxs)("div",{className:"fixed inset-0 z-50 flex items-center justify-center p-6",children:[(0,d.jsx)("button",{className:"absolute inset-0 bg-ink/30","aria-label":g("modal.closeAria"),onClick:f}),(0,d.jsxs)("div",{className:"relative w-full max-w-[520px] rounded-2xl border border-parchment bg-cream p-6 shadow-[0_22px_60px_rgba(35,31,28,0.18)]",children:[(0,d.jsx)("div",{className:"mb-4 font-display text-[1.15rem] text-ink",children:b}),c]})]}):null}},4995:(a,b,c)=>{c.d(b,{k:()=>g});var d=c(7484),e=c(3006),f=c(2115);function g(){let a=(0,e.CU)(a=>a.locale);return(0,d.useMemo)(()=>(b,c)=>(0,f.t)(a,b,c),[a])}},8655:(a,b,c)=>{function d(a){return`${a.providerId}:${a.modelId}`}function e(a){let b=[];for(let c of a)for(let a of c.models.filter(a=>a.enabled)){let d=c.name?`${c.name} \xb7 ${a.id}`:a.id;b.push({providerId:c.id,providerName:c.name,modelId:a.id,label:d})}return b.sort((a,b)=>{let c=a.providerName.localeCompare(b.providerName);return 0!==c?c:a.modelId.localeCompare(b.modelId)}),b}c.d(b,{R:()=>d,n:()=>e})}};
|
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
exports.id=732,exports.ids=[732],exports.modules={318:(a,b,c)=>{"use strict";function d(){return null}function e(a){}c.d(b,{YA:()=>e,sK:()=>d})},
|
|
1
|
+
exports.id=732,exports.ids=[732],exports.modules={318:(a,b,c)=>{"use strict";function d(){return null}function e(a){}c.d(b,{YA:()=>e,sK:()=>d})},518:(a,b,c)=>{"use strict";c.d(b,{MR:()=>i,xA:()=>g});var d=c(2771);let e=null;function f(){if("u"<typeof indexedDB)throw Error("IndexedDB is not available in this environment.")}function g(){return e||(f(),e=new Promise((a,b)=>{let c=indexedDB.open(d.w.name,d.w.version);c.onerror=()=>b(c.error),c.onsuccess=()=>{let b=c.result;b.onversionchange=()=>{b.close(),e=null},a(b)},c.onupgradeneeded=()=>{let a=c.result,b=c.transaction;if(!b)throw Error("Missing upgrade transaction for IndexedDB.");for(let c of Object.keys(d.w.stores)){let e=d.w.stores[c],f=a.objectStoreNames.contains(e.name)?b.objectStore(e.name):a.createObjectStore(e.name,{keyPath:e.keyPath});for(let[a,b]of Object.entries(e.indexes))f.indexNames.contains(a)||f.createIndex(a,b.keyPath,b.options)}}})),e}async function h(){e&&((await e).close(),e=null)}async function i(){f(),await h(),await new Promise((a,b)=>{let c=indexedDB.deleteDatabase(d.w.name);c.onerror=()=>b(c.error),c.onsuccess=()=>a(),c.onblocked=()=>b(Error("deleteDatabase() was blocked."))})}},768:(a,b,c)=>{"use strict";c.d(b,{default:()=>d});let d=(0,c(7943).registerClientReference)(function(){throw Error("Attempted to call the default export of \"/Users/yxp/Documents/new-chat/ai-chat-client/src/components/layout/ClientGlobals.tsx\" from the server, but it's on the client. It's not possible to invoke a client function from the server, it can only be rendered as a Component or passed to props of a Client Component.")},"/Users/yxp/Documents/new-chat/ai-chat-client/src/components/layout/ClientGlobals.tsx","default")},1135:()=>{},1879:(a,b,c)=>{"use strict";function d(){return[]}function e(a){let b=[],c=b.findIndex(b=>b.id===a.id),d={...a,updatedAt:Date.now()};c>=0?b[c]=d:b.push(d)}function f(a){[].filter(b=>b.id!==a)}function g(){return{}}function h(a){}function i(){}function j(){}c.d(b,{BZ:()=>e,D_:()=>g,Ny:()=>i,Tm:()=>h,_5:()=>d,h1:()=>f,k8:()=>j})},2771:(a,b,c)=>{"use strict";c.d(b,{w:()=>d});let d={name:"AIChatClientDB",version:2,stores:{nodes:{name:"nodes",keyPath:"id",indexes:{parentId:{keyPath:"parentId",options:{unique:!1}},type:{keyPath:"type",options:{unique:!1}},createdAt:{keyPath:"createdAt",options:{unique:!1}}}},trees:{name:"trees",keyPath:"id",indexes:{rootId:{keyPath:"rootId",options:{unique:!1}},folderId:{keyPath:"folderId",options:{unique:!1}},updatedAt:{keyPath:"updatedAt",options:{unique:!1}}}},folders:{name:"folders",keyPath:"id",indexes:{createdAt:{keyPath:"createdAt",options:{unique:!1}},updatedAt:{keyPath:"updatedAt",options:{unique:!1}}}},contextBoxes:{name:"contextBoxes",keyPath:"id",indexes:{createdAt:{keyPath:"createdAt",options:{unique:!1}}}}}}},3006:(a,b,c)=>{"use strict";let d;c.d(b,{jL:()=>N,CU:()=>O});var e=c(9517),f=c(4549),g=c(3763),h=c(8859);function i(a){let b=new Set,c=[];for(let d of a)b.has(d)||(b.add(d),c.push(d));return c}function j(a,b){let c=i(a).filter(a=>b.has(a));if(c.length<2)throw Error("Select at least 2 nodes from the same path to compress.");let d=new Set(c),e=[];for(let a of d){let c=b.get(a);c&&(c.parentId&&d.has(c.parentId)||e.push(a))}if(1!==e.length)throw Error("Selection must be a single continuous path (no branches).");let f=e[0],g=[],h=new Set,j=f;for(;j;){if(h.has(j))throw Error("Selection contains a cycle.");h.add(j),g.push(j);let a=c.filter(a=>b.get(a)?.parentId===j);if(a.length>1)throw Error("Selection contains branches; pick a linear chain.");let d=a[0];if(!d)break;j=d}if(g.length!==d.size)throw Error("Selection must be a single continuous path (no gaps).");return g}function k(a){let b=a.find(a=>a.content.trim())?.content.trim()??"",c=a.slice().reverse().find(a=>a.content.trim())?.content.trim()??"";if(!b&&!c)return"Compressed conversation segment.";if(b===c)return b.slice(0,240);let d=`${b}
|
|
2
2
|
...
|
|
3
3
|
${c}`.trim();return d.length>320?`${d.slice(0,320)}...`:d}class l{constructor(a=new h.w){this.nodeService=a}buildCompressionPrompt(a){return['You are a helpful assistant designed to output JSON.\nTask: Compress the conversation segment into a reusable summary and extract any implicit meta instructions.\nReturn ONLY valid JSON with this shape:\n{"summary":"...","metaInstructions":{"language":"zh-CN","format":"markdown","role":"expert"}}\nRules:\n- summary: 2-3 short sentences, capture goals, decisions, constraints.\n- metaInstructions: include keys only when clearly implied by the conversation (language/format/role).\n- Do not wrap the JSON in markdown fences.\n\nConversation:',a.map(a=>{let b=a.type===g.Z.USER?"user":a.type===g.Z.ASSISTANT?"assistant":"system";return`${b}: ${a.content}`}).join("\n\n")].join("\n")}extractMetaInstructions(a){let b,c=a.map(a=>a.content).join("\n\n"),d={},e=(b=c.replace(/\s+/g,""))?(b.match(/[\u4e00-\u9fff]/g)||[]).length/b.length>.2?"zh-CN":"en":void 0;e&&(d.language=e);let f=/\bmarkdown\b/i.test(c)||/用\s*markdown|以\s*markdown/i.test(c)?"markdown":/\bjson\b/i.test(c)||/返回\s*json|以\s*json/i.test(c)?"json":c.includes("```")?"markdown":void 0;f&&(d.format=f);let g=/\bexpert\b/i.test(c)||/专家/.test(c)?"expert":/\bmentor\b/i.test(c)||/导师/.test(c)?"mentor":void 0;return g&&(d.role=g),d}async generateSuggestion(a,b,c){let d,e=this.extractMetaInstructions(b),f={messages:[{role:"user",content:this.buildCompressionPrompt(b)}],model:c?.model,temperature:c?.temperature??.2,maxTokens:c?.maxTokens??512,responseFormat:c?.responseFormat??{type:"json_object"}};try{d=await a.chat(f)}catch{let b={...f};delete b.responseFormat,d=await a.chat(b)}let g=function(a){let b=function(a){let b=a.trim();if(!b)return null;try{return JSON.parse(b),b}catch{}let c=b.match(/```(?:json)?\s*([\s\S]*?)\s*```/i);if(c?.[1]){let a=c[1].trim();if(a)return a}let d=b.indexOf("{"),e=b.lastIndexOf("}");return -1!==d&&-1!==e&&e>d?b.slice(d,e+1):null}(a);if(!b)throw Error("Model returned empty JSON payload.");try{return JSON.parse(b)}catch(b){let a=b instanceof Error?b.message:"Unknown JSON parse error";throw Error(`Failed to parse JSON response: ${a}`)}}(d),h="object"==typeof g&&null!==g?g:{},i="string"==typeof h.summary?h.summary.trim():"",j=function(a){if("object"!=typeof a||null===a)return{};let b={};for(let[c,d]of Object.entries(a))null!=d&&"string"==typeof d&&d.trim()&&(b[c]=d.trim());return b}(h.metaInstructions);return{summary:i||k(b),metaInstructions:{...e,...j}}}async compress(a,b){let c=i(a),d=(await Promise.all(c.map(a=>this.nodeService.read(a)))).filter(a=>!!a);if(d.length!==c.length)throw Error("Some selected nodes are missing.");if(d.some(a=>a.type===g.Z.COMPRESSED))throw Error("Cannot compress an already compressed node.");let e=new Map(d.map(a=>[a.id,a])),f=j(c,e),h=f.map(a=>e.get(a)),l=h[0],m=h[h.length-1];if(!l.parentId)throw Error("Cannot compress the root node.");let n=b?.summary?.trim()||k(h),o=b?.metaInstructions??this.extractMetaInstructions(h),p=b?.collapsed??!0,q=await this.nodeService.create({type:g.Z.COMPRESSED,parentId:m.id,content:"",summary:n,position:m.position??l.position,metadata:{tags:["compressed"],metaInstructions:o,compressedNodeIds:f,collapsed:p}});for(let a of(await this.nodeService.getChildren(m.id)))a.id!==q.id&&(f.includes(a.id)||await this.nodeService.update(a.id,{parentId:q.id}));return q}async decompress(a){let b=await this.nodeService.read(a);if(!b)throw Error(`Node ${a} not found`);if(b.type!==g.Z.COMPRESSED)throw Error("Node is not a compressed node.");let c=b.metadata.compressedNodeIds??[];if(0===c.length)throw Error("Compressed node has no compressedNodeIds.");let d=(await Promise.all(c.map(a=>this.nodeService.read(a)))).filter(a=>!!a);if(d.length!==c.length)throw Error("Some nodes referenced by the compressed node are missing.");let e=new Map(d.map(a=>[a.id,a])),f=j(c,e).map(a=>e.get(a)),h=f[f.length-1];for(let a of(await this.nodeService.getChildren(b.id)))await this.nodeService.update(a.id,{parentId:h.id});return await this.nodeService.delete(b.id),f}}var m=c(518),n=c(7094),o=c(2771);class p{async read(a){let b=(await (0,m.xA)()).transaction([o.w.stores.contextBoxes.name],"readonly"),c=b.objectStore(o.w.stores.contextBoxes.name),d=await (0,n.k7)(c.get(a));return await (0,n.PE)(b),d??null}async put(a){let b=(await (0,m.xA)()).transaction([o.w.stores.contextBoxes.name],"readwrite");return b.objectStore(o.w.stores.contextBoxes.name).put(a),await (0,n.PE)(b),a}async update(a,b){let c=await this.read(a);if(!c)throw Error(`ContextBox ${a} not found`);let d={...c,...b,id:a};return await this.put(d),d}async delete(a){let b=(await (0,m.xA)()).transaction([o.w.stores.contextBoxes.name],"readwrite");b.objectStore(o.w.stores.contextBoxes.name).delete(a),await (0,n.PE)(b)}}var q=c(7558);function r(a,b){let c=Array.from(a);return c.length<=b?a:c.slice(0,b).join("")}class s{async create(a){let b=Date.now(),c=a?.trim()??"",d={id:(0,q.l)(),name:r(c||"Folder",6),systemPrompt:"You are Prompt Tree, a helpful assistant.",enabledModels:null,createdAt:b,updatedAt:b},e=(await (0,m.xA)()).transaction([o.w.stores.folders.name],"readwrite");return e.objectStore(o.w.stores.folders.name).put(d),await (0,n.PE)(e),d}async read(a){let b=(await (0,m.xA)()).transaction([o.w.stores.folders.name],"readonly"),c=b.objectStore(o.w.stores.folders.name),d=await (0,n.k7)(c.get(a));return await (0,n.PE)(b),d??null}async list(){let a=(await (0,m.xA)()).transaction([o.w.stores.folders.name],"readonly"),b=a.objectStore(o.w.stores.folders.name),c=await (0,n.k7)(b.getAll());return await (0,n.PE)(a),(c??[]).slice().sort((a,b)=>b.updatedAt-a.updatedAt)}async updateName(a,b){let c=await this.read(a);if(!c)throw Error(`Folder ${a} not found`);let d={...c,name:r(b.trim()||c.name,6),updatedAt:Date.now()},e=(await (0,m.xA)()).transaction([o.w.stores.folders.name],"readwrite");return e.objectStore(o.w.stores.folders.name).put(d),await (0,n.PE)(e),d}async updateSystemPrompt(a,b){let c=await this.read(a);if(!c)throw Error(`Folder ${a} not found`);let d={...c,systemPrompt:b,updatedAt:Date.now()},e=(await (0,m.xA)()).transaction([o.w.stores.folders.name],"readwrite");return e.objectStore(o.w.stores.folders.name).put(d),await (0,n.PE)(e),d}async updateEnabledModels(a,b){let c=await this.read(a);if(!c)throw Error(`Folder ${a} not found`);let d={...c,enabledModels:b,updatedAt:Date.now()},e=(await (0,m.xA)()).transaction([o.w.stores.folders.name],"readwrite");return e.objectStore(o.w.stores.folders.name).put(d),await (0,n.PE)(e),d}async touch(a){let b=await this.read(a);if(!b)throw Error(`Folder ${a} not found`);let c={...b,updatedAt:Date.now()},d=(await (0,m.xA)()).transaction([o.w.stores.folders.name],"readwrite");return d.objectStore(o.w.stores.folders.name).put(c),await (0,n.PE)(d),c}async delete(a){let b=(await (0,m.xA)()).transaction([o.w.stores.folders.name],"readwrite");b.objectStore(o.w.stores.folders.name).delete(a),await (0,n.PE)(b)}}class t{async run(a){let b=await fetch("/api/agent",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(a)});if(!b.ok){let a=await b.text().catch(()=>"");throw Error(`Agent request failed (${b.status}): ${a}`)}let c=b.headers.get("content-type")??"";if(!b.body||c.includes("application/json")){let a=await b.json().catch(()=>null);if("object"==typeof a&&null!==a&&"content"in a&&"string"==typeof a.content)return{content:a.content};throw Error("Invalid agent response payload.")}let d=b.body.getReader(),e=new TextDecoder("utf-8"),f="",g="",h=!1;for(;!h;){let b=await d.read();if(b.done)break;let c=(f+=e.decode(b.value,{stream:!0})).split(/\r?\n\r?\n/);for(let b of(f=c.pop()??"",c)){for(let c of b.split(/\r?\n/)){if(!c.startsWith("data:"))continue;let b=c.replace(/^data:\s*/,"");if(b){if("[DONE]"===b){h=!0;break}try{let c=JSON.parse(b);a.onEvent?.(c),"assistant_delta"===c.type?g+=c.delta:"assistant_final"===c.type&&(g=c.content)}catch{}}}if(h)break}}return{content:g}}}var u=c(318),v=c(7440);class w{async chat(a){let{apiKey:b,baseUrl:c,onToken:d,stream:e,...f}=a,g=b??(0,u.sK)();if(!g)throw Error("Missing OpenAI API key. Add it in Settings.");let h=c??(0,v.LB)(),i=await fetch("/api/chat",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({apiKey:g,baseUrl:h,...f,stream:!!e})});if(!i.ok){let a=await i.text().catch(()=>"");throw Error(`LLM request failed (${i.status}): ${a}`)}if(e){let a=i.headers.get("content-type")??"";if(!i.body||a.includes("application/json")){let a=await i.json();if("object"==typeof a&&null!==a&&"content"in a&&"string"==typeof a.content)return a.content;throw Error("Invalid LLM response payload.")}let b=i.body.getReader(),c=new TextDecoder("utf-8"),e="",f="",g=!1;for(;!g;){let a=await b.read();if(a.done)break;let h=(e+=c.decode(a.value,{stream:!0})).split(/\r?\n\r?\n/);for(let a of(e=h.pop()??"",h)){for(let b of a.split(/\r?\n/)){if(!b.startsWith("data:"))continue;let a=b.replace(/^data:\s*/,"");if(a){if("[DONE]"===a){g=!0;break}try{let b=JSON.parse(a),c=b.choices?.[0]?.delta?.content??"";c&&(f+=c,d?.(c))}catch{}}}if(g)break}}return f}let j=await i.json();if("object"==typeof j&&null!==j&&"content"in j&&"string"==typeof j.content)return j.content;throw Error("Invalid LLM response payload.")}}class x{async create(a){let b=Date.now(),c=a?.systemPrompt??"You are Prompt Tree, a helpful assistant.",d=await this.nodeService.create({id:(0,q.l)(),type:g.Z.SYSTEM,createdAt:b,updatedAt:b,parentId:null,content:c,metadata:{tags:[],metaInstructions:{}}}),e={id:(0,q.l)(),rootId:d.id,title:a?.title?.trim()?a.title.trim():"New Chat",folderId:a?.folderId??null,createdAt:b,updatedAt:b},f={id:e.id,nodeIds:[d.id],totalTokens:d.tokenCount,maxTokens:8192,createdAt:b},h=(await (0,m.xA)()).transaction([o.w.stores.trees.name,o.w.stores.contextBoxes.name],"readwrite");return h.objectStore(o.w.stores.trees.name).put(e),h.objectStore(o.w.stores.contextBoxes.name).put(f),await (0,n.PE)(h),e}async read(a){let b=(await (0,m.xA)()).transaction([o.w.stores.trees.name],"readonly"),c=b.objectStore(o.w.stores.trees.name),d=await (0,n.k7)(c.get(a));return await (0,n.PE)(b),d??null}async list(){let a=(await (0,m.xA)()).transaction([o.w.stores.trees.name],"readonly"),b=a.objectStore(o.w.stores.trees.name),c=await (0,n.k7)(b.getAll());return await (0,n.PE)(a),(c??[]).slice().sort((a,b)=>b.updatedAt-a.updatedAt)}async updateTitle(a,b){let c=await this.read(a);if(!c)throw Error(`Tree ${a} not found`);let d={...c,title:b.trim()||c.title,updatedAt:Date.now()},e=(await (0,m.xA)()).transaction([o.w.stores.trees.name],"readwrite");return e.objectStore(o.w.stores.trees.name).put(d),await (0,n.PE)(e),d}async updateFolderId(a,b){let c=await this.read(a);if(!c)throw Error(`Tree ${a} not found`);let d={...c,folderId:b??null},e=(await (0,m.xA)()).transaction([o.w.stores.trees.name],"readwrite");return e.objectStore(o.w.stores.trees.name).put(d),await (0,n.PE)(e),d}async touch(a){let b=await this.read(a);if(!b)throw Error(`Tree ${a} not found`);let c={...b,updatedAt:Date.now()},d=(await (0,m.xA)()).transaction([o.w.stores.trees.name],"readwrite");return d.objectStore(o.w.stores.trees.name).put(c),await (0,n.PE)(d),c}async delete(a){let b=await this.read(a);if(!b)return;await this.nodeService.delete(b.rootId);let c=(await (0,m.xA)()).transaction([o.w.stores.trees.name,o.w.stores.contextBoxes.name],"readwrite");c.objectStore(o.w.stores.trees.name).delete(a),c.objectStore(o.w.stores.contextBoxes.name).delete(a),await (0,n.PE)(c)}async updateRootSystemPrompt(a,b){let c=await this.read(a);if(!c)throw Error(`Tree ${a} not found`);let d=await this.nodeService.read(c.rootId),e=await this.nodeService.update(c.rootId,{type:g.Z.SYSTEM,content:b}),f=(await (0,m.xA)()).transaction([o.w.stores.contextBoxes.name],"readwrite"),h=f.objectStore(o.w.stores.contextBoxes.name),i=await (0,n.k7)(h.get(c.id));if(i&&i.nodeIds.includes(c.rootId)){let a=e.tokenCount-(d?.tokenCount??0);h.put({...i,totalTokens:(i.totalTokens??0)+a})}await (0,n.PE)(f)}async loadTreeNodes(a){let b=await this.read(a);if(!b)throw Error(`Tree ${a} not found`);let c=(await (0,m.xA)()).transaction([o.w.stores.nodes.name],"readonly"),d=c.objectStore(o.w.stores.nodes.name),e=await (0,n.k7)(d.getAll());await (0,n.PE)(c);let f=new Map;for(let a of e??[]){let b=a.parentId??null,c=f.get(b);c?c.push(a):f.set(b,[a])}for(let a of f.values())a.sort((a,b)=>a.createdAt-b.createdAt);let g=[],h=[b.rootId],i=new Set,j=new Map;for(let a of e??[])j.set(a.id,a);for(;h.length;){let a=h.shift();if(!a||i.has(a))continue;i.add(a);let b=j.get(a);if(b)for(let c of(g.push(b),f.get(a)??[]))h.push(c.id)}return{tree:b,nodes:g}}constructor(){this.nodeService=new h.w}}var y=c(8729),z=c(8033);function A(a,b){let c=0;for(let d of a)c+=b.get(d)?.tokenCount??0;return c}var B=c(5951),C=c(7872);function D(a){switch(a.type){case g.Z.SYSTEM:return{role:"system",content:a.content};case g.Z.USER:return{role:"user",content:a.content};case g.Z.ASSISTANT:return{role:"assistant",content:a.content};case g.Z.COMPRESSED:return{role:"system",content:a.summary?`[Compressed]
|
|
4
|
-
${a.summary}`:a.content}}}function E(a){}var F=c(1879),G=c(6684);let H={search:{provider:"tavily",exaApiKey:"",tavilyApiKey:"",maxResults:5,searchDepth:"basic"},mcp:{servers:[]},python:{timeoutMs:15e3,maxOutputChars:2e4,pythonCommand:"python3"}};function I(a){}function J(a){}function K(a){let b=[],c=new Set;for(let d of a){let a="string"==typeof d?d.trim():"";if(!a)continue;let e="web_search"===a||"python"===a||"mcp"===a?a:a.startsWith("mcp:")&&a.slice(4).trim()?`mcp:${a.slice(4).trim()}`:null;e&&(c.has(e)||(c.add(e),b.push(e)))}return b.includes("mcp")?b.filter(a=>"web_search"===a||"python"===a||"mcp"===a):b}function L(a,b){if(!a.includes("mcp"))return a;let c=b.mcp.servers.map(a=>`mcp:${a.id}`);return K([...a.filter(a=>"mcp"!==a),...c])}function M(a,b){let c=new Set(b.mcp.servers.map(a=>a.id));return a.filter(a=>{if("web_search"===a||"python"===a||"mcp"===a)return!0;if(!a.startsWith("mcp:"))return!1;let b=a.slice(4).trim();return!!(b&&c.has(b))})}let N=(d={nodeService:(void 0)??new h.w,treeService:(void 0)??new x,folderService:(void 0)??new s,contextBoxService:(void 0)??new p,llmService:(void 0)??new w,agentService:(void 0)??new t,compressionService:(void 0)??new l},(0,f.y)()((a,b,...c)=>{let e;return{initialized:!1,isLoading:!1,error:null,initialize:async()=>{if(!b().initialized){a({isLoading:!0,error:null});try{let[c,e]=await Promise.all([d.folderService.list(),d.treeService.list()]),f=e;0===f.length&&(f=[await d.treeService.create()]),a({trees:new Map(f.map(a=>[a.id,a])),folders:new Map(c.map(a=>[a.id,a]))}),await b().loadTree(f[0].id),a({initialized:!0})}catch(b){a({error:b instanceof Error?b.message:"Failed to initialize store"})}finally{a({isLoading:!1})}}},...((a,b)=>({nodes:new Map,activeNodeId:null,selectedNodeIds:[],getNodes:()=>b().nodes,getNode:a=>b().nodes.get(a),getActiveNode:()=>{let a=b().activeNodeId;return a?b().nodes.get(a)??null:null},getSelectedNodes:()=>b().selectedNodeIds.map(a=>b().nodes.get(a)).filter(a=>!!a),createNode:async b=>{let c=await d.nodeService.create(b);return a(a=>{let b=new Map(a.nodes);return b.set(c.id,c),{nodes:b}}),c},updateNode:async(c,e)=>{let f=b().nodes.get(c);f&&a(a=>{let b=new Map(a.nodes);return b.set(c,{...f,...e,id:c}),{nodes:b}});let g=await d.nodeService.update(c,e);return a(a=>{let b=new Map(a.nodes);return b.set(g.id,g),{nodes:b}}),g},deleteNode:async c=>{await d.nodeService.delete(c);let e=b().currentTreeId;e?await b().loadTree(e):a(a=>{let b=new Map(a.nodes);return b.delete(c),{nodes:b}})},setActiveNode:c=>{a({activeNodeId:c}),b().syncContextToNode(c),b().syncToolsToNode(c)},toggleNodeSelection:b=>a(a=>({selectedNodeIds:a.selectedNodeIds.includes(b)?a.selectedNodeIds.filter(a=>a!==b):[...a.selectedNodeIds,b]})),setSelectedNodeIds:b=>a(a=>!function(a,b){if(a===b)return!0;if(a.length!==b.length)return!1;if(0===a.length)return!0;let c=new Set(a);if(c.size!==a.length)return!1;for(let a of b)if(!c.has(a))return!1;return!0}(a.selectedNodeIds,b)?{selectedNodeIds:b}:a),clearSelection:()=>a(a=>0===a.selectedNodeIds.length?a:{selectedNodeIds:[]}),getNodePath:a=>d.nodeService.getPath(a),getChildren:a=>d.nodeService.getChildren(a)}))(a,b,...c),...((a,b)=>({trees:new Map,currentTreeId:null,getCurrentTree:()=>{let a=b().currentTreeId;return a?b().trees.get(a)??null:null},createTree:async c=>{a({isLoading:!0,error:null});try{let e=await d.treeService.create({title:c});return a(a=>{let b=new Map(a.trees);return b.set(e.id,e),{trees:b}}),await b().loadTree(e.id),e.id}catch(b){throw a({error:b instanceof Error?b.message:"Failed to create tree"}),b}finally{a({isLoading:!1})}},createTreeInFolder:async(c,e)=>{a({isLoading:!0,error:null});try{let f=b().folders.get(c)??await d.folderService.read(c);if(!f)throw Error(`Folder ${c} not found`);let g=await d.treeService.create({title:e,folderId:c,systemPrompt:f.systemPrompt});return a(a=>{let b=new Map(a.trees);return b.set(g.id,g),{trees:b}}),await b().loadTree(g.id),g.id}catch(b){throw a({error:b instanceof Error?b.message:"Failed to create tree in folder"}),b}finally{a({isLoading:!1})}},loadTree:async b=>{a({isLoading:!0,error:null});try{let{tree:c,nodes:e}=await d.treeService.loadTreeNodes(b),f=new Map;for(let a of e)f.set(a.id,a);let g=function(a,b){let c=b,d=-1/0;for(let b of a)b.createdAt>d&&(c=b.id,d=b.createdAt);return c}(e,c.rootId);a(a=>{let b=new Map(a.trees);return b.set(c.id,c),{trees:b,currentTreeId:c.id,currentView:"tree",currentFolderId:c.folderId??null,nodes:f,activeNodeId:g,selectedNodeIds:[]}});let h=await d.contextBoxService.read(c.id);if(!h){let a=f.get(c.rootId),b=Date.now(),e={id:c.id,nodeIds:a?[a.id]:[],totalTokens:a?.tokenCount??0,maxTokens:8192,createdAt:b};h=await d.contextBoxService.put(e)}let i=h.nodeIds.filter(a=>f.has(a));i.length!==h.nodeIds.length&&(h=await d.contextBoxService.put({...h,nodeIds:i,totalTokens:function(a,b){let c=0;for(let d of a)c+=b.get(d)?.tokenCount??0;return c}(i,f)})),a({contextBox:h})}catch(b){a({error:b instanceof Error?b.message:"Failed to load tree"})}finally{a({isLoading:!1})}},deleteTree:async c=>{a({isLoading:!0,error:null});try{await d.treeService.delete(c);let e=new Map(b().trees);if(e.delete(c),a({trees:e}),b().currentTreeId!==c)return;let f=e.keys().next().value;if(f)return void await b().loadTree(f);let g=await d.treeService.create(),h=new Map(e);h.set(g.id,g),a({trees:h}),await b().loadTree(g.id)}catch(b){a({error:b instanceof Error?b.message:"Failed to delete tree"})}finally{a({isLoading:!1})}},updateTreeTitle:async(b,c)=>{a({isLoading:!0,error:null});try{let e=await d.treeService.updateTitle(b,c);a(a=>{let b=new Map(a.trees);return b.set(e.id,e),{trees:b}})}catch(b){a({error:b instanceof Error?b.message:"Failed to update title"})}finally{a({isLoading:!1})}}}))(a,b,...c),...((a,b)=>({folders:new Map,currentFolderId:null,currentView:"tree",getCurrentFolder:()=>{let a=b().currentFolderId;return a?b().folders.get(a)??null:null},createFolder:async b=>{a({isLoading:!0,error:null});try{let c=await d.folderService.create(b);return a(a=>{let b=new Map(a.folders);return b.set(c.id,c),{folders:b,currentView:"folder",currentFolderId:c.id,currentTreeId:null,nodes:new Map,activeNodeId:null,selectedNodeIds:[],contextBox:null}}),c.id}catch(b){throw a({error:b instanceof Error?b.message:"Failed to create folder"}),b}finally{a({isLoading:!1})}},loadFolder:b=>{a({currentView:"folder",currentFolderId:b,currentTreeId:null,nodes:new Map,activeNodeId:null,selectedNodeIds:[],contextBox:null})},updateFolderName:async(b,c)=>{a({isLoading:!0,error:null});try{let e=await d.folderService.updateName(b,c);a(a=>{let b=new Map(a.folders);return b.set(e.id,e),{folders:b}})}catch(b){a({error:b instanceof Error?b.message:"Failed to update folder name"})}finally{a({isLoading:!1})}},updateFolderSystemPrompt:async(c,e)=>{a({isLoading:!0,error:null});try{let f=await d.folderService.updateSystemPrompt(c,e);a(a=>{let b=new Map(a.folders);return b.set(f.id,f),{folders:b}});let g=Array.from(b().trees.values()).filter(a=>(a.folderId??null)===c),h=await Promise.all(g.map(async a=>(await d.treeService.updateRootSystemPrompt(a.id,f.systemPrompt),d.treeService.touch(a.id))));h.length>0&&a(a=>{let b=new Map(a.trees);for(let a of h)b.set(a.id,a);return{trees:b}})}catch(b){a({error:b instanceof Error?b.message:"Failed to update folder system prompt"})}finally{a({isLoading:!1})}},updateFolderEnabledModels:async(b,c)=>{a({isLoading:!0,error:null});try{let e=await d.folderService.updateEnabledModels(b,c);a(a=>{let b=new Map(a.folders);return b.set(e.id,e),{folders:b}})}catch(b){a({error:b instanceof Error?b.message:"Failed to update folder enabled models"})}finally{a({isLoading:!1})}},deleteFolder:async c=>{a({isLoading:!0,error:null});try{let e=Array.from(b().trees.values()).filter(a=>(a.folderId??null)===c),f=await Promise.all(e.map(a=>d.treeService.updateFolderId(a.id,null)));if(await d.folderService.delete(c),a(a=>{let b=new Map(a.folders);b.delete(c);let d=a.currentFolderId===c,e=new Map(a.trees);for(let a of f)e.set(a.id,a);return{folders:b,trees:e,currentFolderId:d?null:a.currentFolderId,currentView:d?"tree":a.currentView}}),"tree"===b().currentView&&null==b().currentTreeId){let a=Array.from(b().trees.values()).sort((a,b)=>b.updatedAt-a.updatedAt)[0];a&&await b().loadTree(a.id)}}catch(b){a({error:b instanceof Error?b.message:"Failed to delete folder"})}finally{a({isLoading:!1})}}}))(a,b,...c),...((a,b)=>({contextBox:null,getContextBox:()=>b().contextBox,addToContext:async(c,e)=>{let f=b().contextBox;if(!f||f.nodeIds.includes(c))return;let g=b().nodes,h=g.get(c)??await d.nodeService.read(c);if(!h)return;let i=g.has(c)?g:new Map(g).set(c,h);i!==g&&a({nodes:i});let j=f.nodeIds.slice(),k="number"==typeof e&&Number.isFinite(e)?Math.max(0,Math.min(e,j.length)):j.length;j.splice(k,0,c);let l=A(j,i),m={...f,nodeIds:j,totalTokens:l};a({contextBox:m}),await d.contextBoxService.put(m)},removeFromContext:c=>{let e=b().contextBox;if(!e)return;let f=e.nodeIds.filter(a=>a!==c),g=A(f,b().nodes),h={...e,nodeIds:f,totalTokens:g};a({contextBox:h}),d.contextBoxService.put(h)},clearContext:()=>{let c=b().contextBox;if(!c)return;let e={...c,nodeIds:[],totalTokens:0};a({contextBox:e}),d.contextBoxService.put(e)},reorderContext:c=>{let e=b().contextBox;if(!e)return;let f=Array.from(new Set(c)),g=A(f,b().nodes),h={...e,nodeIds:f,totalTokens:g};a({contextBox:h}),d.contextBoxService.put(h)},syncContextToNode:async c=>{let e=b().contextBox;if(!e)return;let f=b().nodes;if(!f.has(c))return;let h=(0,y.VK)(f,c);if(0===h.length)return;let i=new Set(h),j=new Set;for(let a of h){let b=f.get(a);if(b&&b.type===g.Z.COMPRESSED&&b.metadata.collapsed)for(let a of b.metadata.compressedNodeIds??[])i.has(a)&&j.add(a)}let k=h.filter(a=>!j.has(a)),l=[],m=new Set;for(let a of k)!(!a||m.has(a))&&f.has(a)&&(m.add(a),l.push(a));if(l.length===e.nodeIds.length&&l.every((a,b)=>a===e.nodeIds[b]))return;let n=A(l,f),o={...e,nodeIds:l,totalTokens:n};a({contextBox:o}),await d.contextBoxService.put(o)},buildContextContent:async()=>{let a=b().contextBox;if(!a)return"";let c=(0,z.d)(b().draftToolUses??[],b().toolSettings),d=!1,e=[];for(let f of a.nodeIds){let a=b().nodes.get(f);if(!a)continue;if(!d&&a.type===g.Z.SYSTEM){for(let b of(e.push(`System: ${a.content}`),d=!0,c))e.push(`System: ${b.content}`);continue}if(a.type===g.Z.COMPRESSED){e.push(`[Compressed Context: ${a.summary??""}]`);continue}let h=a.type===g.Z.USER?"User":a.type===g.Z.ASSISTANT?"Assistant":"System";e.push(`${h}: ${a.content}`)}if(!d)for(let a of c)e.unshift(`System: ${a.content}`);return e.join("\n\n")}}))(a,b,...c),...(a=>({sidebarOpen:!0,theme:"light",locale:"en",compressionOpen:!1,hydrateUiFromStorage:()=>a(a=>({theme:a.theme,locale:a.locale})),toggleSidebar:()=>a(a=>({sidebarOpen:!a.sidebarOpen})),setSidebarOpen:b=>a({sidebarOpen:b}),setTheme:b=>a(()=>(E(b),{theme:b})),setLocale:b=>a(()=>({locale:b})),toggleTheme:()=>a(a=>{let b="light"===a.theme?"dark":"light";return E(b),{theme:b}}),contextPanelOpen:!0,toggleContextPanel:()=>a(a=>({contextPanelOpen:!a.contextPanelOpen})),setContextPanelOpen:b=>a({contextPanelOpen:b}),openCompression:()=>a({compressionOpen:!0}),closeCompression:()=>a({compressionOpen:!1})}))(a,b,...c),...(e=B.ES,(a,b)=>{let c=a=>{let c=b().model;if(!a)return{modelId:c,modelName:c};let d=b().providers.find(b=>b.id===a.providerId);if(!d)throw Error("Selected model provider is missing.");let e=(0,C.aA)(d);if(!e)throw Error("Selected model is missing API key.");let f=d.name?`${d.name} \xb7 ${a.modelId}`:a.modelId;return{modelId:a.modelId,modelName:f,providerId:d.id,providerName:d.name,apiKey:e.value,baseUrl:d.baseUrl}},f=a=>"new chat"===a.trim().toLowerCase(),h=(a,c)=>{if(!a)return!1;let d=b().providers.find(b=>b.id===a);if(!d)return!1;let e=d.models.find(a=>a.id===c);return e?.supportsStreaming??!1},i=(b,c)=>{a(a=>{let d=new Map(a.nodes),e=d.get(b);return e?(d.set(b,{...e,content:c}),{nodes:d}):a})},j=(a,b,c)=>{let d,e=(0,z.d)(b,c);if(0===e.length)return a;let f=e.map(a=>({role:"system",content:a.content})),g=a.slice(),h=-1===(d=g.findIndex(a=>"system"!==a.role))?g.length:d;return g.splice(h,0,...f),g},k=async(c,e,f,h,k)=>{if(h.length>0){let l=await d.nodeService.create({type:g.Z.ASSISTANT,parentId:e,content:"",metadata:{tags:[],metaInstructions:{},modelId:c.modelId,modelName:c.modelName,providerId:c.providerId,providerName:c.providerName,toolLogs:[]}});a(a=>{let b=new Map(a.nodes);return b.set(l.id,l),{nodes:b}});let m="",n=[],o=null,p=0,q=(b=!1)=>{let c=Date.now();if(b||!(c-p<50)){var d,e;p=c,i(l.id,m),d=l.id,e=n,a(a=>{let b=new Map(a.nodes),c=b.get(d);return c?(b.set(d,{...c,metadata:{...c.metadata,toolLogs:e}}),{nodes:b}):a})}},r=a=>{let b=n.findIndex(b=>b.id===a.id);b>=0?(n=n.slice())[b]=a:n=[...n,a]};try{let a=j(f,h,k),e=c.apiKey??(0,u.sK)();if(!e)throw Error("errors.missingOpenAIApiKey");let g=c.baseUrl??(0,v.LB)();if(await d.agentService.run({apiKey:e,baseUrl:g,headers:c.headers,timeout:c.timeout,model:c.modelId,temperature:b().temperature,maxTokens:b().maxTokens,messages:a,toolUses:h,toolSettings:k,stream:!!c.supportsStreaming,onEvent:a=>{if("assistant_delta"===a.type){m+=a.delta,q();return}if("assistant_final"===a.type){m=a.content,q(!0);return}if("tool_call"===a.type){r({id:a.call.id,tool:a.call.name,args:a.call.arguments,status:"running",startedAt:Date.now()}),q();return}if("tool_result"===a.type){let b=n.find(b=>b.id===a.callId)??{id:a.callId,tool:a.name,args:{},status:"running",startedAt:Date.now()};r({...b,status:"success",endedAt:Date.now(),result:a.result}),q();return}if("tool_error"===a.type){let b=n.find(b=>b.id===a.callId)??{id:a.callId,tool:a.name,args:{},status:"running",startedAt:Date.now()};r({...b,status:"error",endedAt:Date.now(),error:a.error}),q();return}"error"===a.type&&(o=a.message)}}),o)throw Error(o)}catch(b){throw m||(await d.nodeService.delete(l.id),a(a=>{let b=new Map(a.nodes);return b.delete(l.id),{nodes:b}})),b}q(!0);let s=await d.nodeService.update(l.id,{content:m,metadata:{toolLogs:n}});return a(a=>{let b=new Map(a.nodes);return b.set(s.id,s),{nodes:b}}),s}if(c.supportsStreaming){let l=await d.nodeService.create({type:g.Z.ASSISTANT,parentId:e,content:"",metadata:{tags:[],metaInstructions:{},modelId:c.modelId,modelName:c.modelName,providerId:c.providerId,providerName:c.providerName}});a(a=>{let b=new Map(a.nodes);return b.set(l.id,l),{nodes:b}});let m="",n=0,o=(a=!1)=>{let b=Date.now();(a||!(b-n<50))&&(n=b,i(l.id,m))};try{await d.llmService.chat({messages:j(f,h,k),model:c.modelId,temperature:b().temperature,maxTokens:b().maxTokens,apiKey:c.apiKey,baseUrl:c.baseUrl,stream:!0,onToken:a=>{m+=a,o()}})}catch(b){throw m||(await d.nodeService.delete(l.id),a(a=>{let b=new Map(a.nodes);return b.delete(l.id),{nodes:b}})),b}o(!0);let p=await d.nodeService.update(l.id,{content:m});return a(a=>{let b=new Map(a.nodes);return b.set(p.id,p),{nodes:b}}),p}let l=await d.llmService.chat({messages:j(f,h,k),model:c.modelId,temperature:b().temperature,maxTokens:b().maxTokens,apiKey:c.apiKey,baseUrl:c.baseUrl});return d.nodeService.create({type:g.Z.ASSISTANT,parentId:e,content:l,metadata:{tags:[],metaInstructions:{},modelId:c.modelId,modelName:c.modelName,providerId:c.providerId,providerName:c.providerName}})};return{isSending:!1,llmError:null,isCompressing:!1,compressionError:null,model:e.model,temperature:e.temperature,maxTokens:e.maxTokens,selectedModels:e.selectedModels,compressionModel:e.compressionModel,summaryModel:e.summaryModel,hydrateLLMSettingsFromStorage:()=>{let b=(0,B.sO)();b&&a({model:b.model,temperature:b.temperature,maxTokens:b.maxTokens,selectedModels:b.selectedModels,compressionModel:b.compressionModel,summaryModel:b.summaryModel})},setLLMSettings:b=>a(a=>{let c=(0,B.ci)({model:Object.prototype.hasOwnProperty.call(b,"model")?b.model:a.model,temperature:Object.prototype.hasOwnProperty.call(b,"temperature")?b.temperature:a.temperature,maxTokens:Object.prototype.hasOwnProperty.call(b,"maxTokens")?b.maxTokens:a.maxTokens,selectedModels:Object.prototype.hasOwnProperty.call(b,"selectedModels")?b.selectedModels:a.selectedModels,compressionModel:Object.prototype.hasOwnProperty.call(b,"compressionModel")?b.compressionModel:a.compressionModel,summaryModel:Object.prototype.hasOwnProperty.call(b,"summaryModel")?b.summaryModel:a.summaryModel},B.ES);return(0,B.g8)(c),c}),setSelectedModels:b=>a(a=>{let c=(0,B.ci)({model:a.model,temperature:a.temperature,maxTokens:a.maxTokens,selectedModels:b,compressionModel:a.compressionModel,summaryModel:a.summaryModel},B.ES);return(0,B.g8)(c),{selectedModels:c.selectedModels}}),sendMessage:async(c,e)=>{let i=c.trim();if(!i)throw Error("Message is empty.");let j=b().getCurrentTree();if(!j)throw Error("errors.noActiveConversationTree");a({isSending:!0,llmError:null});try{let c=b().draftToolUses??[],l=b().toolSettings,m=b().activeNodeId??j.rootId,n=await d.nodeService.create({type:g.Z.USER,parentId:m,content:i,metadata:{toolUses:c}});a(a=>{let b=new Map(a.nodes);return b.set(n.id,n),{nodes:b,activeNodeId:n.id}}),await b().addToContext(n.id),f(j.title)&&(async()=>{try{var c;let e=(c=await b().generateSummary(i),Array.from(c.replace(/[\r\n]+/g," ").trim().replace(/[,,。.!!??;;::"“”'‘’、()[\]{}]/g,"").replace(/\s+/g,"")).slice(0,6).join(""));if(!e)return;let g=b().getCurrentTree();if(!g||g.id!==j.id||!f(g.title))return;let h=await d.treeService.updateTitle(j.id,e);a(a=>{let b=new Map(a.trees);return b.set(h.id,h),{trees:b}})}catch{}})();let o=e?.length?e:b().contextBox?.nodeIds?.length?b().contextBox.nodeIds:null,p=o?(await Promise.all(o.map(async a=>b().nodes.get(a)??d.nodeService.read(a)))).filter(a=>!!a):await d.nodeService.getPath(n.id),q=[];for(let a of p){let b=D(a);b&&q.push(b)}let r=q[q.length-1];r&&"user"===r.role&&r.content===i||q.push({role:"user",content:i});let s=b().selectedModels,t=s.map(a=>(a=>{let c=b().providers.find(b=>b.id===a.providerId);if(!c)return null;let d=(0,C.aA)(c);if(!d)return null;let e=c.name?`${c.name} \xb7 ${a.modelId}`:a.modelId;return{modelId:a.modelId,modelName:e,providerId:c.id,providerName:c.name,apiKey:d.value,baseUrl:c.baseUrl,headers:c.headers,timeout:c.timeout,supportsStreaming:h(c.id,a.modelId)}})(a)).filter(a=>!!a);if(s.length>0&&0===t.length)throw Error("Selected models are missing API keys or providers.");if(0===t.length){let a=b().model;t.push({modelId:a,modelName:a,supportsStreaming:!1})}let u=await Promise.allSettled(t.map(async a=>{let b=await k(a,n.id,q,c,l);return{request:a,node:b}})),v=[],w=[];for(let[a,b]of u.entries()){let c=t[a];if("rejected"===b.status){let a=b.reason instanceof Error?b.reason.message:"errors.failedToSendMessage";w.push(`${c.modelName}: ${a}`);continue}v.push(b.value.node)}if(0===v.length)throw Error(w[0]??"errors.failedToSendMessage");for(let c of(a(a=>{let b=new Map(a.nodes);for(let a of v)b.set(a.id,a);return{nodes:b,activeNodeId:v[v.length-1]?.id??n.id}}),v))await b().addToContext(c.id);if(w.length>0&&a({llmError:w[0]}),b().currentTreeId){let c=await d.treeService.touch(b().currentTreeId);a(a=>{let b=new Map(a.trees);return b.set(c.id,c),{trees:b}})}return v[v.length-1]}catch(b){throw a({llmError:b instanceof Error?b.message:"errors.failedToSendMessage"}),b}finally{a({isSending:!1})}},retryAssistant:async c=>{if(!b().getCurrentTree())throw Error("errors.noActiveConversationTree");let e=b().nodes.get(c)??await d.nodeService.read(c);if(!e||e.type!==g.Z.ASSISTANT)throw Error("Selected node is not an assistant message.");let f=e.parentId;if(!f)throw Error("Assistant message has no parent user node.");let i=b().nodes.get(f)??await d.nodeService.read(f);if(!i||i.type!==g.Z.USER)throw Error("Parent user node not found.");a({isSending:!0,llmError:null});try{let c=i.metadata.toolUses??[],f=b().toolSettings,g=b().contextBox?.nodeIds?.length?b().contextBox.nodeIds:null,j=g?(await Promise.all(g.map(async a=>b().nodes.get(a)??d.nodeService.read(a)))).filter(a=>!!a):await d.nodeService.getPath(i.id),l=[];for(let a of j){let b=D(a);b&&l.push(b)}let m=l[l.length-1];m&&"user"===m.role&&m.content===i.content||l.push({role:"user",content:i.content});let n=(()=>{let a=e.metadata.modelId??b().model,c=e.metadata.providerId;if(!c)return{modelId:a,modelName:e.metadata.modelName??a,supportsStreaming:!1};let d=b().providers.find(a=>a.id===c);if(!d)throw Error("Selected model provider is missing.");let f=(0,C.aA)(d);if(!f)throw Error("Selected model is missing API key.");let g=e.metadata.modelName??(d.name?`${d.name} \xb7 ${a}`:a);return{modelId:a,modelName:g,providerId:d.id,providerName:d.name,apiKey:f.value,baseUrl:d.baseUrl,headers:d.headers,timeout:d.timeout,supportsStreaming:h(d.id,a)}})(),o=await k(n,i.id,l,c,f);if(a(a=>{let b=new Map(a.nodes);return b.set(o.id,o),{nodes:b,activeNodeId:o.id}}),await b().addToContext(o.id),b().currentTreeId){let c=await d.treeService.touch(b().currentTreeId);a(a=>{let b=new Map(a.trees);return b.set(c.id,c),{trees:b}})}return o}catch(b){throw a({llmError:b instanceof Error?b.message:"errors.failedToRetryMessage"}),b}finally{a({isSending:!1})}},compressNodes:async(c,e)=>{let f=b().currentTreeId;if(!f)throw Error("errors.noActiveConversationTree");a({isCompressing:!0,compressionError:null});try{let g=await d.compressionService.compress(c,{summary:e?.summary,metaInstructions:e?.metaInstructions});await d.treeService.touch(f),await b().loadTree(f);let h=b().contextBox;if(h){let e=new Set(c),f=[],i=!1;for(let a of h.nodeIds){if(e.has(a)){i||(f.push(g.id),i=!0);continue}f.push(a)}let j=[],k=new Set;for(let a of f)!(!a||k.has(a))&&b().nodes.has(a)&&(k.add(a),j.push(a));if(i){let c=j.reduce((a,c)=>a+(b().nodes.get(c)?.tokenCount??0),0),e={...h,nodeIds:j,totalTokens:c};a({contextBox:e}),await d.contextBoxService.put(e)}}return g}catch(b){throw a({compressionError:b instanceof Error?b.message:"errors.failedToCompressNodes"}),b}finally{a({isCompressing:!1})}},decompressNode:async c=>{let e=b().currentTreeId;if(!e)throw Error("errors.noActiveConversationTree");let f=b().contextBox?.nodeIds??[];a({isCompressing:!0,compressionError:null});try{let g=await d.compressionService.decompress(c),h=g.map(a=>a.id);await d.treeService.touch(e),await b().loadTree(e);let i=b().contextBox;if(i&&f.includes(c)){let e=[];for(let a of f){if(a===c){e.push(...h);continue}e.push(a)}let g=[],j=new Set;for(let a of e)!(!a||j.has(a))&&b().nodes.has(a)&&(j.add(a),g.push(a));let k=g.reduce((a,c)=>a+(b().nodes.get(c)?.tokenCount??0),0),l={...i,nodeIds:g,totalTokens:k};a({contextBox:l}),await d.contextBoxService.put(l)}return g}catch(b){throw a({compressionError:b instanceof Error?b.message:"errors.failedToDecompressNode"}),b}finally{a({isCompressing:!1})}},generateCompressionSuggestion:async e=>{a({isCompressing:!0,compressionError:null});try{let a=await Promise.all(e.map(async a=>b().nodes.get(a)??d.nodeService.read(a))).then(a=>a.filter(a=>!!a)),f=c(b().compressionModel);return await d.compressionService.generateSuggestion(d.llmService,a,{model:f.modelId,temperature:.2,maxTokens:512,responseFormat:{type:"json_object"},apiKey:f.apiKey,baseUrl:f.baseUrl})}catch(b){throw a({compressionError:b instanceof Error?b.message:"errors.failedToGenerateSuggestion"}),b}finally{a({isCompressing:!1})}},generateSummary:async a=>{let e="zh-CN"===b().locale?["你是一个标题生成器。\n请根据下面的内容生成不超过 6 个汉字的主题标题。\n只输出标题本身,不要解释,不要标点。\n",a].join("\n"):["You are a title generator.\nBased on the content below, generate a concise topic title in 2–6 words.\nOutput only the title. No explanation. No quotes. No punctuation.\n",a].join("\n"),f=c(b().summaryModel??b().compressionModel??null);return(await d.llmService.chat({messages:[{role:"user",content:e}],model:f.modelId,temperature:.2,maxTokens:64,apiKey:f.apiKey,baseUrl:f.baseUrl})).trim()}}})(a,b,...c),...((a,b)=>({providers:[],selectedProviderId:null,modelSelector:{open:!1,providerId:null,searchQuery:"",activeTab:"all",fetchedModels:[],isLoading:!1,error:null},healthChecks:{},loadProviders:()=>{let b=(0,F._5)(),c=(0,F.D_)();a(a=>{let d=a.selectedProviderId&&b.some(b=>b.id===a.selectedProviderId)?a.selectedProviderId:b.length>0?b[0].id:null;return{providers:b,healthChecks:c,selectedProviderId:d}})},addProvider:b=>{let c=(0,C.DA)(b);return a(a=>({providers:[...a.providers,c],selectedProviderId:c.id})),(0,F.BZ)(c),c},updateProvider:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,...d,updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},deleteProvider:c=>{a(a=>{let b=a.providers.filter(a=>a.id!==c),d=a.selectedProviderId===c?b.length>0?b[0].id:null:a.selectedProviderId;return{providers:b,selectedProviderId:d}}),(0,F.h1)(c);let d={...b().healthChecks};delete d[c],a({healthChecks:d}),(0,F.Tm)(d)},selectProvider:b=>{a({selectedProviderId:b})},toggleProviderEnabled:c=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,enabled:!a.enabled,updatedAt:Date.now()}:a)}));let d=b().providers.find(a=>a.id===c);d&&(0,F.BZ)(d)},addApiKey:(c,d,e)=>{let f=(0,C.Iq)(d,e);a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:[...a.apiKeys,f],updatedAt:Date.now()}:a)}));let g=b().providers.find(a=>a.id===c);g&&(0,F.BZ)(g)},updateApiKey:(c,d,e)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:a.apiKeys.map(a=>a.id===d?{...a,...e}:a),updatedAt:Date.now()}:a)}));let f=b().providers.find(a=>a.id===c);f&&(0,F.BZ)(f)},deleteApiKey:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:a.apiKeys.filter(a=>a.id!==d),updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},setPrimaryApiKey:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:a.apiKeys.map(a=>({...a,isPrimary:a.id===d})),updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},addModel:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,models:[...a.models,d],updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},removeModel:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,models:a.models.filter(a=>a.id!==d),updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},toggleModelEnabled:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,models:a.models.map(a=>a.id===d?{...a,enabled:!a.enabled}:a),updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},updateModel:(c,d,e)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,models:a.models.map(a=>a.id===d?{...a,...e}:a),updatedAt:Date.now()}:a)}));let f=b().providers.find(a=>a.id===c);f&&(0,F.BZ)(f)},openModelSelector:b=>{a({modelSelector:{open:!0,providerId:b,searchQuery:"",activeTab:"all",fetchedModels:[],isLoading:!1,error:null}})},closeModelSelector:()=>{a({modelSelector:{open:!1,providerId:null,searchQuery:"",activeTab:"all",fetchedModels:[],isLoading:!1,error:null}})},setModelSelectorSearch:b=>{a(a=>({modelSelector:{...a.modelSelector,searchQuery:b}}))},setModelSelectorTab:b=>{a(a=>({modelSelector:{...a.modelSelector,activeTab:b}}))},fetchModelsForSelector:async c=>{let d=b().providers.find(a=>a.id===c);if(!d)return;let e=(0,C.aA)(d);if(!e)return void a(a=>({modelSelector:{...a.modelSelector,error:"errors.missingApiKey"}}));a(a=>({modelSelector:{...a.modelSelector,isLoading:!0,error:null}}));try{let b=await (0,G.QF)(e.value,d.baseUrl,{headers:d.headers,timeout:d.timeout});a(a=>({modelSelector:{...a.modelSelector,fetchedModels:b.models,isLoading:!1,error:b.error??null}}))}catch(c){let b=c instanceof Error?c.message:"errors.failedToFetchModels";a(a=>({modelSelector:{...a.modelSelector,isLoading:!1,error:b}}))}},addFetchedModels:c=>{let{modelSelector:d}=b(),e=d.providerId;if(!e)return;let f=b().providers.find(a=>a.id===e);if(!f)return;let g=new Set(f.models.map(a=>a.id)),h=d.fetchedModels.filter(a=>c.includes(a.id)&&!g.has(a.id));a(a=>({providers:a.providers.map(a=>a.id===e?{...a,models:[...a.models,...h],updatedAt:Date.now()}:a)}));let i=b().providers.find(a=>a.id===e);i&&(0,F.BZ)(i)},checkProviderHealth:async(c,d)=>{let e=b().providers.find(a=>a.id===c);if(!e)throw Error("Provider not found");let f=await (0,G._C)(e,d);a(a=>({healthChecks:{...a.healthChecks,[c]:f}})),a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:a.apiKeys.map(a=>{let b=f.keyResults.find(b=>b.keyId===a.id);return b?{...a,healthStatus:b.status,lastChecked:f.checkedAt}:a}),updatedAt:Date.now()}:a)}));let g=b().providers.find(a=>a.id===c);g&&(0,F.BZ)(g);let h={...b().healthChecks,[c]:f};return(0,F.Tm)(h),f},checkAllProviders:async()=>{let{providers:a}=b(),c={};for(let d of a)d.enabled&&d.apiKeys.length>0&&(c[d.id]=await b().checkProviderHealth(d.id));return c},getProvider:a=>b().providers.find(b=>b.id===a)||null,getSelectedProvider:()=>{let{selectedProviderId:a,providers:c}=b();return a&&c.find(b=>b.id===a)||null}}))(a,b,...c),...((a,b)=>({toolSettings:H,draftToolUses:[],hydrateToolsFromStorage:()=>{let b=M(L(K([]),H),H);a({toolSettings:H,draftToolUses:b})},setToolSettings:b=>{a(a=>{let c=M(L(a.draftToolUses,b),b);return J(c),{toolSettings:b,draftToolUses:c}})},setDraftToolUses:c=>{let d=b().toolSettings,e=M(L(K(c),d),d);a({draftToolUses:e})},toggleDraftToolUse:a=>{let c=b().draftToolUses,d=c.includes(a)?c.filter(b=>b!==a):[...c,a];b().setDraftToolUses(d)},upsertMcpServer:b=>{let c=Date.now(),d=b.id.trim();a(a=>{let e=a.toolSettings,f=e.mcp.servers.slice(),g=f.findIndex(a=>a.id===d),h={...b,id:d,createdAt:g>=0?f[g].createdAt:c,updatedAt:c};g>=0?f[g]=h:f.push(h);let i={...e,mcp:{...e.mcp,servers:f.sort((a,b)=>b.updatedAt-a.updatedAt)}};return I(i),{toolSettings:i}})},deleteMcpServer:b=>{a(a=>{let c=a.toolSettings,d={...c,mcp:{...c.mcp,servers:c.mcp.servers.filter(a=>a.id!==b)}},e=M(a.draftToolUses,d);return I(d),J(e),{toolSettings:d,draftToolUses:e}})},syncToolsToNode:a=>{let c=b().nodes.get(a);if(c){if(c.type===g.Z.USER){let a=c.metadata.toolUses??[];b().setDraftToolUses(a);return}if(c.type===g.Z.ASSISTANT){let a=c.parentId;if(!a)return;let d=b().nodes.get(a);if(!d||d.type!==g.Z.USER)return;let e=d.metadata.toolUses??[];b().setDraftToolUses(e)}}}}))(a,b,...c)}}));function O(a){return(0,e.P)(N,a)}},3412:(a,b,c)=>{Promise.resolve().then(c.t.bind(c,1921,23)),Promise.resolve().then(c.t.bind(c,8059,23)),Promise.resolve().then(c.t.bind(c,4342,23)),Promise.resolve().then(c.t.bind(c,2265,23)),Promise.resolve().then(c.t.bind(c,5421,23)),Promise.resolve().then(c.t.bind(c,1335,23)),Promise.resolve().then(c.t.bind(c,664,23)),Promise.resolve().then(c.bind(c,4661))},3763:(a,b,c)=>{"use strict";c.d(b,{Z:()=>e});var d,e=((d={}).USER="user",d.ASSISTANT="assistant",d.SYSTEM="system",d.COMPRESSED="compressed",d)},4852:(a,b,c)=>{Promise.resolve().then(c.t.bind(c,5547,23)),Promise.resolve().then(c.t.bind(c,5098,23)),Promise.resolve().then(c.t.bind(c,7644,23)),Promise.resolve().then(c.t.bind(c,3859,23)),Promise.resolve().then(c.t.bind(c,8099,23)),Promise.resolve().then(c.t.bind(c,6237,23)),Promise.resolve().then(c.t.bind(c,8562,23)),Promise.resolve().then(c.t.bind(c,6675,23))},5301:(a,b,c)=>{Promise.resolve().then(c.t.bind(c,5547,23))},5723:(a,b,c)=>{"use strict";c.d(b,{default:()=>i});var d=c(8249);c(7484);var e=c(3006);function f(){return(0,e.CU)(a=>a.theme),null}function g(){return(0,e.CU)(a=>a.locale),null}function h(){return null}function i(){return(0,d.jsxs)(d.Fragment,{children:[(0,d.jsx)(h,{}),(0,d.jsx)(f,{}),(0,d.jsx)(g,{})]})}},5951:(a,b,c)=>{"use strict";c.d(b,{ES:()=>d,ci:()=>f,g8:()=>h,sO:()=>g});let d={model:"gpt-4o-mini",temperature:.7,maxTokens:1024,selectedModels:[],compressionModel:null,summaryModel:null};function e(a){if(!a||"object"!=typeof a)return null;let{providerId:b,modelId:c}=a;return"string"!=typeof b||"string"!=typeof c?null:{providerId:b,modelId:c}}function f(a,b=d){let c="string"==typeof a.model&&a.model.trim()?a.model.trim():b.model,g="number"==typeof a.temperature&&Number.isFinite(a.temperature)?Math.min(2,Math.max(0,a.temperature)):b.temperature,h="number"==typeof a.maxTokens&&Number.isFinite(a.maxTokens)?Math.max(1,Math.round(a.maxTokens)):b.maxTokens,i=function(a,b){if(!Array.isArray(a))return b;let c=new Set,d=[];for(let b of a){if(!b||"object"!=typeof b)continue;let{providerId:a,modelId:e}=b;if("string"!=typeof a||"string"!=typeof e)continue;let f=`${a}:${e}`;c.has(f)||(c.add(f),d.push({providerId:a,modelId:e}))}return d}(a.selectedModels,b.selectedModels);return{model:c,temperature:g,maxTokens:h,selectedModels:i,compressionModel:Object.prototype.hasOwnProperty.call(a,"compressionModel")?e(a.compressionModel):b.compressionModel,summaryModel:Object.prototype.hasOwnProperty.call(a,"summaryModel")?e(a.summaryModel):b.summaryModel}}function g(){return null}function h(a){}},6239:(a,b,c)=>{"use strict";function d(a){let b=a.trim();if(!b)return 0;let c=(b.match(/[\u3040-\u30ff\u3400-\u4dbf\u4e00-\u9fff\uf900-\ufaff]/g)??[]).length;return Math.ceil((b.length-c)/4+c/2)}c.d(b,{b:()=>d})},6487:()=>{},6537:(a,b,c)=>{"use strict";c.r(b),c.d(b,{default:()=>g,metadata:()=>f});var d=c(5735);c(1135);var e=c(768);let f={title:"Prompt Tree - AI Dialogue Topology",description:"A visual dialogue tree interface for AI conversations with context management and token optimization.",keywords:["AI","dialogue","chat","context","topology","tree"],authors:[{name:"Prompt Tree Team"}],icons:{icon:"/icon.svg",apple:"/icon.svg"}};function g({children:a}){return(0,d.jsx)("html",{lang:"en",children:(0,d.jsxs)("body",{className:"antialiased",children:[(0,d.jsx)(e.default,{}),a]})})}},6684:(a,b,c)=>{"use strict";function d(a){let b=a.trim().replace(/\/+$/,"");return b.endsWith("/chat/completions")&&(b=b.replace(/\/chat\/completions$/,"")),b.endsWith("/models")&&(b=b.replace(/\/models$/,"")),!b.includes("/v1")&&!b.includes("/api")&&b.includes("api.openai.com")&&(b=`${b}/v1`),b}async function e(a,b,c){let d=await fetch(a,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(b),signal:c});if(!d.ok){let a=await d.text().catch(()=>"");throw Error(`Request failed (${d.status}): ${a||d.statusText}`)}return await d.json()}function f(a,b,c){return{apiKey:a,baseUrl:d(b),headers:c?.headers,timeout:c?.timeout}}async function g(a,b,c){try{let d=await e("/api/providers/models",f(a,b,c),c?.signal);if(d.error)return{models:[],error:d.error};let g=(d.models??[]).filter(a=>a&&"string"==typeof a.id).map(a=>{var b,c;let d;return{id:a.id,name:a.id,enabled:!1,category:(b=a.id,c=a.object,(d=b.toLowerCase()).includes("vision")||d.includes("image")||d.includes("gpt-4o")||d.includes("claude-3")?"vision":d.includes("embed")?"embedding":d.includes("reason")||d.includes("o1")||d.includes("r1")?"reasoning":d.includes("tool")?"tool":"embedding"===c?"embedding":"chat")}});return g.sort((a,b)=>a.id.localeCompare(b.id)),{models:g}}catch(a){if(a instanceof Error)return{models:[],error:a.message};return{models:[],error:"errors.unknownError"}}}async function h(a,b){try{return await e("/api/providers/health",{providerId:a.id,baseUrl:d(a.baseUrl),apiKeys:a.apiKeys.map(a=>({id:a.id,value:a.value,isPrimary:a.isPrimary})),headers:a.headers,timeout:a.timeout},b)}catch(c){let b=c instanceof Error?c.message:"errors.connectionFailed";return{providerId:a.id,status:"error",keyResults:a.apiKeys.map(a=>({keyId:a.id,status:"error",error:b})),checkedAt:Date.now()}}}async function i(a,b,c,d){try{return await e("/api/providers/test",{...f(a,b,d),model:c,prompt:d?.prompt},d?.signal)}catch(a){return{status:"error",error:a instanceof Error?a.message:"errors.connectionFailed"}}}c.d(b,{DT:()=>i,QF:()=>g,_C:()=>h,qT:()=>d})},7094:(a,b,c)=>{"use strict";function d(a){return new Promise((b,c)=>{a.onsuccess=()=>b(a.result),a.onerror=()=>c(a.error)})}function e(a){return new Promise((b,c)=>{a.oncomplete=()=>b(),a.onerror=()=>c(a.error),a.onabort=()=>c(a.error??Error("IndexedDB transaction aborted."))})}c.d(b,{PE:()=>e,k7:()=>d})},7157:(a,b,c)=>{Promise.resolve().then(c.t.bind(c,1921,23))},7440:(a,b,c)=>{"use strict";c.d(b,{$t:()=>f,LB:()=>e,rd:()=>d});let d="https://api.openai.com/v1";function e(){return d}function f(a){}},7558:(a,b,c)=>{"use strict";c.d(b,{l:()=>e});var d=c(4579);function e(){return"u">typeof crypto&&"randomUUID"in crypto?crypto.randomUUID():(0,d.A)()}},7872:(a,b,c)=>{"use strict";function d(a){let b=Date.now();return{id:`provider_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,name:a,baseUrl:"https://api.openai.com/v1",apiKeys:[],models:[],enabled:!0,timeout:3e4,createdAt:b,updatedAt:b}}function e(a,b){return{id:`key_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,value:a.trim(),name:b||`Key ${new Date().toLocaleDateString()}`,isPrimary:!1,healthStatus:"unknown"}}function f(a){let b=a.apiKeys.find(a=>a.isPrimary);return b||a.apiKeys[0]||null}c.d(b,{DA:()=>d,Iq:()=>e,aA:()=>f})},8033:(a,b,c)=>{"use strict";function d(a,b){let c=[],d=new Set(a);if(d.has("web_search")){let a="exa"===b.search.provider?"Exa":"Tavily";c.push({id:"web_search",title:"Tool Use: Web Search",content:`You can search the web for up-to-date information.
|
|
4
|
+
${a.summary}`:a.content}}}function E(a){}var F=c(1879),G=c(6684);let H={search:{provider:"tavily",exaApiKey:"",tavilyApiKey:"",maxResults:5,searchDepth:"basic"},mcp:{servers:[]},python:{timeoutMs:15e3,maxOutputChars:2e4,pythonCommand:"python3"}};function I(a){}function J(a){}function K(a){let b=[],c=new Set;for(let d of a){let a="string"==typeof d?d.trim():"";if(!a)continue;let e="web_search"===a||"python"===a||"mcp"===a?a:a.startsWith("mcp:")&&a.slice(4).trim()?`mcp:${a.slice(4).trim()}`:null;e&&(c.has(e)||(c.add(e),b.push(e)))}return b.includes("mcp")?b.filter(a=>"web_search"===a||"python"===a||"mcp"===a):b}function L(a,b){if(!a.includes("mcp"))return a;let c=b.mcp.servers.map(a=>`mcp:${a.id}`);return K([...a.filter(a=>"mcp"!==a),...c])}function M(a,b){let c=new Set(b.mcp.servers.map(a=>a.id));return a.filter(a=>{if("web_search"===a||"python"===a||"mcp"===a)return!0;if(!a.startsWith("mcp:"))return!1;let b=a.slice(4).trim();return!!(b&&c.has(b))})}let N=(d={nodeService:(void 0)??new h.w,treeService:(void 0)??new x,folderService:(void 0)??new s,contextBoxService:(void 0)??new p,llmService:(void 0)??new w,agentService:(void 0)??new t,compressionService:(void 0)??new l},(0,f.y)()((a,b,...c)=>{let e;return{initialized:!1,isLoading:!1,error:null,initialize:async()=>{if(!b().initialized){a({isLoading:!0,error:null});try{let[c,e]=await Promise.all([d.folderService.list(),d.treeService.list()]),f=e;0===f.length&&(f=[await d.treeService.create()]),a({trees:new Map(f.map(a=>[a.id,a])),folders:new Map(c.map(a=>[a.id,a]))}),await b().loadTree(f[0].id),a({initialized:!0})}catch(b){a({error:b instanceof Error?b.message:"Failed to initialize store"})}finally{a({isLoading:!1})}}},...((a,b)=>({nodes:new Map,activeNodeId:null,selectedNodeIds:[],getNodes:()=>b().nodes,getNode:a=>b().nodes.get(a),getActiveNode:()=>{let a=b().activeNodeId;return a?b().nodes.get(a)??null:null},getSelectedNodes:()=>b().selectedNodeIds.map(a=>b().nodes.get(a)).filter(a=>!!a),createNode:async b=>{let c=await d.nodeService.create(b);return a(a=>{let b=new Map(a.nodes);return b.set(c.id,c),{nodes:b}}),c},updateNode:async(c,e)=>{let f=b().nodes.get(c);f&&a(a=>{let b=new Map(a.nodes);return b.set(c,{...f,...e,id:c}),{nodes:b}});let g=await d.nodeService.update(c,e);return a(a=>{let b=new Map(a.nodes);return b.set(g.id,g),{nodes:b}}),g},deleteNode:async c=>{await d.nodeService.delete(c);let e=b().currentTreeId;e?await b().loadTree(e):a(a=>{let b=new Map(a.nodes);return b.delete(c),{nodes:b}})},setActiveNode:c=>{a({activeNodeId:c}),b().syncContextToNode(c),b().syncToolsToNode(c)},toggleNodeSelection:b=>a(a=>({selectedNodeIds:a.selectedNodeIds.includes(b)?a.selectedNodeIds.filter(a=>a!==b):[...a.selectedNodeIds,b]})),setSelectedNodeIds:b=>a(a=>!function(a,b){if(a===b)return!0;if(a.length!==b.length)return!1;if(0===a.length)return!0;let c=new Set(a);if(c.size!==a.length)return!1;for(let a of b)if(!c.has(a))return!1;return!0}(a.selectedNodeIds,b)?{selectedNodeIds:b}:a),clearSelection:()=>a(a=>0===a.selectedNodeIds.length?a:{selectedNodeIds:[]}),getNodePath:a=>d.nodeService.getPath(a),getChildren:a=>d.nodeService.getChildren(a)}))(a,b,...c),...((a,b)=>({trees:new Map,currentTreeId:null,getCurrentTree:()=>{let a=b().currentTreeId;return a?b().trees.get(a)??null:null},createTree:async c=>{a({isLoading:!0,error:null});try{let e=await d.treeService.create({title:c});return a(a=>{let b=new Map(a.trees);return b.set(e.id,e),{trees:b}}),await b().loadTree(e.id),e.id}catch(b){throw a({error:b instanceof Error?b.message:"Failed to create tree"}),b}finally{a({isLoading:!1})}},createTreeInFolder:async(c,e)=>{a({isLoading:!0,error:null});try{let f=b().folders.get(c)??await d.folderService.read(c);if(!f)throw Error(`Folder ${c} not found`);let g=await d.treeService.create({title:e,folderId:c,systemPrompt:f.systemPrompt});return a(a=>{let b=new Map(a.trees);return b.set(g.id,g),{trees:b}}),await b().loadTree(g.id),g.id}catch(b){throw a({error:b instanceof Error?b.message:"Failed to create tree in folder"}),b}finally{a({isLoading:!1})}},loadTree:async b=>{a({isLoading:!0,error:null});try{let{tree:c,nodes:e}=await d.treeService.loadTreeNodes(b),f=new Map;for(let a of e)f.set(a.id,a);let g=function(a,b){let c=b,d=-1/0;for(let b of a)b.createdAt>d&&(c=b.id,d=b.createdAt);return c}(e,c.rootId);a(a=>{let b=new Map(a.trees);return b.set(c.id,c),{trees:b,currentTreeId:c.id,currentView:"tree",currentFolderId:c.folderId??null,nodes:f,activeNodeId:g,selectedNodeIds:[]}});let h=await d.contextBoxService.read(c.id);if(!h){let a=f.get(c.rootId),b=Date.now(),e={id:c.id,nodeIds:a?[a.id]:[],totalTokens:a?.tokenCount??0,maxTokens:8192,createdAt:b};h=await d.contextBoxService.put(e)}let i=h.nodeIds.filter(a=>f.has(a));i.length!==h.nodeIds.length&&(h=await d.contextBoxService.put({...h,nodeIds:i,totalTokens:function(a,b){let c=0;for(let d of a)c+=b.get(d)?.tokenCount??0;return c}(i,f)})),a({contextBox:h})}catch(b){a({error:b instanceof Error?b.message:"Failed to load tree"})}finally{a({isLoading:!1})}},deleteTree:async c=>{a({isLoading:!0,error:null});try{await d.treeService.delete(c);let e=new Map(b().trees);if(e.delete(c),a({trees:e}),b().currentTreeId!==c)return;let f=e.keys().next().value;if(f)return void await b().loadTree(f);let g=await d.treeService.create(),h=new Map(e);h.set(g.id,g),a({trees:h}),await b().loadTree(g.id)}catch(b){a({error:b instanceof Error?b.message:"Failed to delete tree"})}finally{a({isLoading:!1})}},updateTreeTitle:async(b,c)=>{a({isLoading:!0,error:null});try{let e=await d.treeService.updateTitle(b,c);a(a=>{let b=new Map(a.trees);return b.set(e.id,e),{trees:b}})}catch(b){a({error:b instanceof Error?b.message:"Failed to update title"})}finally{a({isLoading:!1})}}}))(a,b,...c),...((a,b)=>({folders:new Map,currentFolderId:null,currentView:"tree",getCurrentFolder:()=>{let a=b().currentFolderId;return a?b().folders.get(a)??null:null},createFolder:async b=>{a({isLoading:!0,error:null});try{let c=await d.folderService.create(b);return a(a=>{let b=new Map(a.folders);return b.set(c.id,c),{folders:b,currentView:"folder",currentFolderId:c.id,currentTreeId:null,nodes:new Map,activeNodeId:null,selectedNodeIds:[],contextBox:null}}),c.id}catch(b){throw a({error:b instanceof Error?b.message:"Failed to create folder"}),b}finally{a({isLoading:!1})}},loadFolder:b=>{a({currentView:"folder",currentFolderId:b,currentTreeId:null,nodes:new Map,activeNodeId:null,selectedNodeIds:[],contextBox:null})},updateFolderName:async(b,c)=>{a({isLoading:!0,error:null});try{let e=await d.folderService.updateName(b,c);a(a=>{let b=new Map(a.folders);return b.set(e.id,e),{folders:b}})}catch(b){a({error:b instanceof Error?b.message:"Failed to update folder name"})}finally{a({isLoading:!1})}},updateFolderSystemPrompt:async(c,e)=>{a({isLoading:!0,error:null});try{let f=await d.folderService.updateSystemPrompt(c,e);a(a=>{let b=new Map(a.folders);return b.set(f.id,f),{folders:b}});let g=Array.from(b().trees.values()).filter(a=>(a.folderId??null)===c),h=await Promise.all(g.map(async a=>(await d.treeService.updateRootSystemPrompt(a.id,f.systemPrompt),d.treeService.touch(a.id))));h.length>0&&a(a=>{let b=new Map(a.trees);for(let a of h)b.set(a.id,a);return{trees:b}})}catch(b){a({error:b instanceof Error?b.message:"Failed to update folder system prompt"})}finally{a({isLoading:!1})}},updateFolderEnabledModels:async(b,c)=>{a({isLoading:!0,error:null});try{let e=await d.folderService.updateEnabledModels(b,c);a(a=>{let b=new Map(a.folders);return b.set(e.id,e),{folders:b}})}catch(b){a({error:b instanceof Error?b.message:"Failed to update folder enabled models"})}finally{a({isLoading:!1})}},deleteFolder:async c=>{a({isLoading:!0,error:null});try{let e=Array.from(b().trees.values()).filter(a=>(a.folderId??null)===c),f=await Promise.all(e.map(a=>d.treeService.updateFolderId(a.id,null)));if(await d.folderService.delete(c),a(a=>{let b=new Map(a.folders);b.delete(c);let d=a.currentFolderId===c,e=new Map(a.trees);for(let a of f)e.set(a.id,a);return{folders:b,trees:e,currentFolderId:d?null:a.currentFolderId,currentView:d?"tree":a.currentView}}),"tree"===b().currentView&&null==b().currentTreeId){let a=Array.from(b().trees.values()).sort((a,b)=>b.updatedAt-a.updatedAt)[0];a&&await b().loadTree(a.id)}}catch(b){a({error:b instanceof Error?b.message:"Failed to delete folder"})}finally{a({isLoading:!1})}}}))(a,b,...c),...((a,b)=>({contextBox:null,getContextBox:()=>b().contextBox,addToContext:async(c,e)=>{let f=b().contextBox;if(!f||f.nodeIds.includes(c))return;let g=b().nodes,h=g.get(c)??await d.nodeService.read(c);if(!h)return;let i=g.has(c)?g:new Map(g).set(c,h);i!==g&&a({nodes:i});let j=f.nodeIds.slice(),k="number"==typeof e&&Number.isFinite(e)?Math.max(0,Math.min(e,j.length)):j.length;j.splice(k,0,c);let l=A(j,i),m={...f,nodeIds:j,totalTokens:l};a({contextBox:m}),await d.contextBoxService.put(m)},removeFromContext:c=>{let e=b().contextBox;if(!e)return;let f=e.nodeIds.filter(a=>a!==c),g=A(f,b().nodes),h={...e,nodeIds:f,totalTokens:g};a({contextBox:h}),d.contextBoxService.put(h)},clearContext:()=>{let c=b().contextBox;if(!c)return;let e={...c,nodeIds:[],totalTokens:0};a({contextBox:e}),d.contextBoxService.put(e)},reorderContext:c=>{let e=b().contextBox;if(!e)return;let f=Array.from(new Set(c)),g=A(f,b().nodes),h={...e,nodeIds:f,totalTokens:g};a({contextBox:h}),d.contextBoxService.put(h)},syncContextToNode:async c=>{let e=b().contextBox;if(!e)return;let f=b().nodes;if(!f.has(c))return;let h=(0,y.VK)(f,c);if(0===h.length)return;let i=new Set(h),j=new Set;for(let a of h){let b=f.get(a);if(b&&b.type===g.Z.COMPRESSED&&b.metadata.collapsed)for(let a of b.metadata.compressedNodeIds??[])i.has(a)&&j.add(a)}let k=h.filter(a=>!j.has(a)),l=[],m=new Set;for(let a of k)!(!a||m.has(a))&&f.has(a)&&(m.add(a),l.push(a));if(l.length===e.nodeIds.length&&l.every((a,b)=>a===e.nodeIds[b]))return;let n=A(l,f),o={...e,nodeIds:l,totalTokens:n};a({contextBox:o}),await d.contextBoxService.put(o)},buildContextContent:async()=>{let a=b().contextBox;if(!a)return"";let c=(0,z.d)(b().draftToolUses??[],b().toolSettings),d=!1,e=[];for(let f of a.nodeIds){let a=b().nodes.get(f);if(!a)continue;if(!d&&a.type===g.Z.SYSTEM){for(let b of(e.push(`System: ${a.content}`),d=!0,c))e.push(`System: ${b.content}`);continue}if(a.type===g.Z.COMPRESSED){e.push(`[Compressed Context: ${a.summary??""}]`);continue}let h=a.type===g.Z.USER?"User":a.type===g.Z.ASSISTANT?"Assistant":"System";e.push(`${h}: ${a.content}`)}if(!d)for(let a of c)e.unshift(`System: ${a.content}`);return e.join("\n\n")}}))(a,b,...c),...(a=>({sidebarOpen:!0,theme:"light",locale:"en",compressionOpen:!1,hydrateUiFromStorage:()=>a(a=>({theme:a.theme,locale:a.locale})),toggleSidebar:()=>a(a=>({sidebarOpen:!a.sidebarOpen})),setSidebarOpen:b=>a({sidebarOpen:b}),setTheme:b=>a(()=>(E(b),{theme:b})),setLocale:b=>a(()=>({locale:b})),toggleTheme:()=>a(a=>{let b="light"===a.theme?"dark":"light";return E(b),{theme:b}}),contextPanelOpen:!0,toggleContextPanel:()=>a(a=>({contextPanelOpen:!a.contextPanelOpen})),setContextPanelOpen:b=>a({contextPanelOpen:b}),openCompression:()=>a({compressionOpen:!0}),closeCompression:()=>a({compressionOpen:!1})}))(a,b,...c),...(e=B.ES,(a,b)=>{let c=a=>{let c=b().model;if(!a)return{modelId:c,modelName:c};let d=b().providers.find(b=>b.id===a.providerId);if(!d)throw Error("Selected model provider is missing.");let e=(0,C.aA)(d);if(!e)throw Error("Selected model is missing API key.");let f=d.name?`${d.name} \xb7 ${a.modelId}`:a.modelId;return{modelId:a.modelId,modelName:f,providerId:d.id,providerName:d.name,apiKey:e.value,baseUrl:d.baseUrl}},f=a=>"new chat"===a.trim().toLowerCase(),h=(a,c)=>{if(!a)return!1;let d=b().providers.find(b=>b.id===a);if(!d)return!1;let e=d.models.find(a=>a.id===c);return e?.supportsStreaming??!1},i=(b,c)=>{a(a=>{let d=new Map(a.nodes),e=d.get(b);return e?(d.set(b,{...e,content:c}),{nodes:d}):a})},j=(a,b,c)=>{let d,e=(0,z.d)(b,c);if(0===e.length)return a;let f=e.map(a=>({role:"system",content:a.content})),g=a.slice(),h=-1===(d=g.findIndex(a=>"system"!==a.role))?g.length:d;return g.splice(h,0,...f),g},k=async(c,e,f,h,k)=>{if(h.length>0){let l=await d.nodeService.create({type:g.Z.ASSISTANT,parentId:e,content:"",metadata:{tags:[],metaInstructions:{},modelId:c.modelId,modelName:c.modelName,providerId:c.providerId,providerName:c.providerName,toolLogs:[]}});a(a=>{let b=new Map(a.nodes);return b.set(l.id,l),{nodes:b}});let m="",n=[],o=null,p=0,q=(b=!1)=>{let c=Date.now();if(b||!(c-p<50)){var d,e;p=c,i(l.id,m),d=l.id,e=n,a(a=>{let b=new Map(a.nodes),c=b.get(d);return c?(b.set(d,{...c,metadata:{...c.metadata,toolLogs:e}}),{nodes:b}):a})}},r=a=>{let b=n.findIndex(b=>b.id===a.id);b>=0?(n=n.slice())[b]=a:n=[...n,a]};try{let a=j(f,h,k),e=c.apiKey??(0,u.sK)();if(!e)throw Error("errors.missingOpenAIApiKey");let g=c.baseUrl??(0,v.LB)();if(await d.agentService.run({apiKey:e,baseUrl:g,headers:c.headers,timeout:c.timeout,model:c.modelId,temperature:b().temperature,maxTokens:b().maxTokens,messages:a,toolUses:h,toolSettings:k,stream:!!c.supportsStreaming,onEvent:a=>{if("assistant_delta"===a.type){m+=a.delta,q();return}if("assistant_final"===a.type){m=a.content,q(!0);return}if("tool_call"===a.type){r({id:a.call.id,tool:a.call.name,args:a.call.arguments,status:"running",startedAt:Date.now()}),q();return}if("tool_result"===a.type){let b=n.find(b=>b.id===a.callId)??{id:a.callId,tool:a.name,args:{},status:"running",startedAt:Date.now()};r({...b,status:"success",endedAt:Date.now(),result:a.result}),q();return}if("tool_error"===a.type){let b=n.find(b=>b.id===a.callId)??{id:a.callId,tool:a.name,args:{},status:"running",startedAt:Date.now()};r({...b,status:"error",endedAt:Date.now(),error:a.error}),q();return}"error"===a.type&&(o=a.message)}}),o)throw Error(o)}catch(b){throw m||(await d.nodeService.delete(l.id),a(a=>{let b=new Map(a.nodes);return b.delete(l.id),{nodes:b}})),b}q(!0);let s=await d.nodeService.update(l.id,{content:m,metadata:{toolLogs:n}});return a(a=>{let b=new Map(a.nodes);return b.set(s.id,s),{nodes:b}}),s}if(c.supportsStreaming){let l=await d.nodeService.create({type:g.Z.ASSISTANT,parentId:e,content:"",metadata:{tags:[],metaInstructions:{},modelId:c.modelId,modelName:c.modelName,providerId:c.providerId,providerName:c.providerName}});a(a=>{let b=new Map(a.nodes);return b.set(l.id,l),{nodes:b}});let m="",n=0,o=(a=!1)=>{let b=Date.now();(a||!(b-n<50))&&(n=b,i(l.id,m))};try{await d.llmService.chat({messages:j(f,h,k),model:c.modelId,temperature:b().temperature,maxTokens:b().maxTokens,apiKey:c.apiKey,baseUrl:c.baseUrl,stream:!0,onToken:a=>{m+=a,o()}})}catch(b){throw m||(await d.nodeService.delete(l.id),a(a=>{let b=new Map(a.nodes);return b.delete(l.id),{nodes:b}})),b}o(!0);let p=await d.nodeService.update(l.id,{content:m});return a(a=>{let b=new Map(a.nodes);return b.set(p.id,p),{nodes:b}}),p}let l=await d.llmService.chat({messages:j(f,h,k),model:c.modelId,temperature:b().temperature,maxTokens:b().maxTokens,apiKey:c.apiKey,baseUrl:c.baseUrl});return d.nodeService.create({type:g.Z.ASSISTANT,parentId:e,content:l,metadata:{tags:[],metaInstructions:{},modelId:c.modelId,modelName:c.modelName,providerId:c.providerId,providerName:c.providerName}})};return{isSending:!1,llmError:null,isCompressing:!1,compressionError:null,model:e.model,temperature:e.temperature,maxTokens:e.maxTokens,selectedModels:e.selectedModels,compressionModel:e.compressionModel,summaryModel:e.summaryModel,hydrateLLMSettingsFromStorage:()=>{let b=(0,B.sO)();b&&a({model:b.model,temperature:b.temperature,maxTokens:b.maxTokens,selectedModels:b.selectedModels,compressionModel:b.compressionModel,summaryModel:b.summaryModel})},setLLMSettings:b=>a(a=>{let c=(0,B.ci)({model:Object.prototype.hasOwnProperty.call(b,"model")?b.model:a.model,temperature:Object.prototype.hasOwnProperty.call(b,"temperature")?b.temperature:a.temperature,maxTokens:Object.prototype.hasOwnProperty.call(b,"maxTokens")?b.maxTokens:a.maxTokens,selectedModels:Object.prototype.hasOwnProperty.call(b,"selectedModels")?b.selectedModels:a.selectedModels,compressionModel:Object.prototype.hasOwnProperty.call(b,"compressionModel")?b.compressionModel:a.compressionModel,summaryModel:Object.prototype.hasOwnProperty.call(b,"summaryModel")?b.summaryModel:a.summaryModel},B.ES);return(0,B.g8)(c),c}),setSelectedModels:b=>a(a=>{let c=(0,B.ci)({model:a.model,temperature:a.temperature,maxTokens:a.maxTokens,selectedModels:b,compressionModel:a.compressionModel,summaryModel:a.summaryModel},B.ES);return(0,B.g8)(c),{selectedModels:c.selectedModels}}),sendMessage:async(c,e)=>{let i=c.trim();if(!i)throw Error("Message is empty.");let j=b().getCurrentTree();if(!j)throw Error("errors.noActiveConversationTree");a({isSending:!0,llmError:null});try{let c=b().draftToolUses??[],l=b().toolSettings,m=b().activeNodeId??j.rootId,n=await d.nodeService.create({type:g.Z.USER,parentId:m,content:i,metadata:{toolUses:c}});a(a=>{let b=new Map(a.nodes);return b.set(n.id,n),{nodes:b,activeNodeId:n.id}}),await b().addToContext(n.id),f(j.title)&&(async()=>{try{var c;let e=(c=await b().generateSummary(i),Array.from(c.replace(/[\r\n]+/g," ").trim().replace(/[,,。.!!??;;::"“”'‘’、()[\]{}]/g,"").replace(/\s+/g,"")).slice(0,6).join(""));if(!e)return;let g=b().getCurrentTree();if(!g||g.id!==j.id||!f(g.title))return;let h=await d.treeService.updateTitle(j.id,e);a(a=>{let b=new Map(a.trees);return b.set(h.id,h),{trees:b}})}catch{}})();let o=e?.length?e:b().contextBox?.nodeIds?.length?b().contextBox.nodeIds:null,p=o?(await Promise.all(o.map(async a=>b().nodes.get(a)??d.nodeService.read(a)))).filter(a=>!!a):await d.nodeService.getPath(n.id),q=[];for(let a of p){let b=D(a);b&&q.push(b)}let r=q[q.length-1];r&&"user"===r.role&&r.content===i||q.push({role:"user",content:i});let s=b().selectedModels,t=s.map(a=>(a=>{let c=b().providers.find(b=>b.id===a.providerId);if(!c)return null;let d=(0,C.aA)(c);if(!d)return null;let e=c.name?`${c.name} \xb7 ${a.modelId}`:a.modelId;return{modelId:a.modelId,modelName:e,providerId:c.id,providerName:c.name,apiKey:d.value,baseUrl:c.baseUrl,headers:c.headers,timeout:c.timeout,supportsStreaming:h(c.id,a.modelId)}})(a)).filter(a=>!!a);if(s.length>0&&0===t.length)throw Error("Selected models are missing API keys or providers.");if(0===t.length){let a=b().model;t.push({modelId:a,modelName:a,supportsStreaming:!1})}let u=await Promise.allSettled(t.map(async a=>{let b=await k(a,n.id,q,c,l);return{request:a,node:b}})),v=[],w=[];for(let[a,b]of u.entries()){let c=t[a];if("rejected"===b.status){let a=b.reason instanceof Error?b.reason.message:"errors.failedToSendMessage";w.push(`${c.modelName}: ${a}`);continue}v.push(b.value.node)}if(0===v.length)throw Error(w[0]??"errors.failedToSendMessage");for(let c of(a(a=>{let b=new Map(a.nodes);for(let a of v)b.set(a.id,a);return{nodes:b,activeNodeId:v[v.length-1]?.id??n.id}}),v))await b().addToContext(c.id);if(w.length>0&&a({llmError:w[0]}),b().currentTreeId){let c=await d.treeService.touch(b().currentTreeId);a(a=>{let b=new Map(a.trees);return b.set(c.id,c),{trees:b}})}return v[v.length-1]}catch(b){throw a({llmError:b instanceof Error?b.message:"errors.failedToSendMessage"}),b}finally{a({isSending:!1})}},retryAssistant:async c=>{if(!b().getCurrentTree())throw Error("errors.noActiveConversationTree");let e=b().nodes.get(c)??await d.nodeService.read(c);if(!e||e.type!==g.Z.ASSISTANT)throw Error("Selected node is not an assistant message.");let f=e.parentId;if(!f)throw Error("Assistant message has no parent user node.");let i=b().nodes.get(f)??await d.nodeService.read(f);if(!i||i.type!==g.Z.USER)throw Error("Parent user node not found.");a({isSending:!0,llmError:null});try{let c=i.metadata.toolUses??[],f=b().toolSettings,g=b().contextBox?.nodeIds?.length?b().contextBox.nodeIds:null,j=g?(await Promise.all(g.map(async a=>b().nodes.get(a)??d.nodeService.read(a)))).filter(a=>!!a):await d.nodeService.getPath(i.id),l=[];for(let a of j){let b=D(a);b&&l.push(b)}let m=l[l.length-1];m&&"user"===m.role&&m.content===i.content||l.push({role:"user",content:i.content});let n=(()=>{let a=e.metadata.modelId??b().model,c=e.metadata.providerId;if(!c)return{modelId:a,modelName:e.metadata.modelName??a,supportsStreaming:!1};let d=b().providers.find(a=>a.id===c);if(!d)throw Error("Selected model provider is missing.");let f=(0,C.aA)(d);if(!f)throw Error("Selected model is missing API key.");let g=e.metadata.modelName??(d.name?`${d.name} \xb7 ${a}`:a);return{modelId:a,modelName:g,providerId:d.id,providerName:d.name,apiKey:f.value,baseUrl:d.baseUrl,headers:d.headers,timeout:d.timeout,supportsStreaming:h(d.id,a)}})(),o=await k(n,i.id,l,c,f);if(a(a=>{let b=new Map(a.nodes);return b.set(o.id,o),{nodes:b,activeNodeId:o.id}}),await b().addToContext(o.id),b().currentTreeId){let c=await d.treeService.touch(b().currentTreeId);a(a=>{let b=new Map(a.trees);return b.set(c.id,c),{trees:b}})}return o}catch(b){throw a({llmError:b instanceof Error?b.message:"errors.failedToRetryMessage"}),b}finally{a({isSending:!1})}},compressNodes:async(c,e)=>{let f=b().currentTreeId;if(!f)throw Error("errors.noActiveConversationTree");a({isCompressing:!0,compressionError:null});try{let g=await d.compressionService.compress(c,{summary:e?.summary,metaInstructions:e?.metaInstructions});await d.treeService.touch(f),await b().loadTree(f);let h=b().contextBox;if(h){let e=new Set(c),f=[],i=!1;for(let a of h.nodeIds){if(e.has(a)){i||(f.push(g.id),i=!0);continue}f.push(a)}let j=[],k=new Set;for(let a of f)!(!a||k.has(a))&&b().nodes.has(a)&&(k.add(a),j.push(a));if(i){let c=j.reduce((a,c)=>a+(b().nodes.get(c)?.tokenCount??0),0),e={...h,nodeIds:j,totalTokens:c};a({contextBox:e}),await d.contextBoxService.put(e)}}return g}catch(b){throw a({compressionError:b instanceof Error?b.message:"errors.failedToCompressNodes"}),b}finally{a({isCompressing:!1})}},decompressNode:async c=>{let e=b().currentTreeId;if(!e)throw Error("errors.noActiveConversationTree");let f=b().contextBox?.nodeIds??[];a({isCompressing:!0,compressionError:null});try{let g=await d.compressionService.decompress(c),h=g.map(a=>a.id);await d.treeService.touch(e),await b().loadTree(e);let i=b().contextBox;if(i&&f.includes(c)){let e=[];for(let a of f){if(a===c){e.push(...h);continue}e.push(a)}let g=[],j=new Set;for(let a of e)!(!a||j.has(a))&&b().nodes.has(a)&&(j.add(a),g.push(a));let k=g.reduce((a,c)=>a+(b().nodes.get(c)?.tokenCount??0),0),l={...i,nodeIds:g,totalTokens:k};a({contextBox:l}),await d.contextBoxService.put(l)}return g}catch(b){throw a({compressionError:b instanceof Error?b.message:"errors.failedToDecompressNode"}),b}finally{a({isCompressing:!1})}},generateCompressionSuggestion:async e=>{a({isCompressing:!0,compressionError:null});try{let a=await Promise.all(e.map(async a=>b().nodes.get(a)??d.nodeService.read(a))).then(a=>a.filter(a=>!!a)),f=c(b().compressionModel);return await d.compressionService.generateSuggestion(d.llmService,a,{model:f.modelId,temperature:.2,maxTokens:512,responseFormat:{type:"json_object"},apiKey:f.apiKey,baseUrl:f.baseUrl})}catch(b){throw a({compressionError:b instanceof Error?b.message:"errors.failedToGenerateSuggestion"}),b}finally{a({isCompressing:!1})}},generateSummary:async a=>{let e="zh-CN"===b().locale?["你是一个标题生成器。\n请根据下面的内容生成不超过 6 个汉字的主题标题。\n只输出标题本身,不要解释,不要标点。\n",a].join("\n"):["You are a title generator.\nBased on the content below, generate a concise topic title in 2–6 words.\nOutput only the title. No explanation. No quotes. No punctuation.\n",a].join("\n"),f=c(b().summaryModel??b().compressionModel??null);return(await d.llmService.chat({messages:[{role:"user",content:e}],model:f.modelId,temperature:.2,maxTokens:64,apiKey:f.apiKey,baseUrl:f.baseUrl})).trim()}}})(a,b,...c),...((a,b)=>({providers:[],selectedProviderId:null,modelSelector:{open:!1,providerId:null,searchQuery:"",activeTab:"all",fetchedModels:[],isLoading:!1,error:null},healthChecks:{},loadProviders:()=>{let b=(0,F._5)(),c=(0,F.D_)();a(a=>{let d=a.selectedProviderId&&b.some(b=>b.id===a.selectedProviderId)?a.selectedProviderId:b.length>0?b[0].id:null;return{providers:b,healthChecks:c,selectedProviderId:d}})},addProvider:b=>{let c=(0,C.DA)(b);return a(a=>({providers:[...a.providers,c],selectedProviderId:c.id})),(0,F.BZ)(c),c},updateProvider:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,...d,updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},deleteProvider:c=>{a(a=>{let b=a.providers.filter(a=>a.id!==c),d=a.selectedProviderId===c?b.length>0?b[0].id:null:a.selectedProviderId;return{providers:b,selectedProviderId:d}}),(0,F.h1)(c);let d={...b().healthChecks};delete d[c],a({healthChecks:d}),(0,F.Tm)(d)},selectProvider:b=>{a({selectedProviderId:b})},toggleProviderEnabled:c=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,enabled:!a.enabled,updatedAt:Date.now()}:a)}));let d=b().providers.find(a=>a.id===c);d&&(0,F.BZ)(d)},addApiKey:(c,d,e)=>{let f=(0,C.Iq)(d,e);a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:[...a.apiKeys,f],updatedAt:Date.now()}:a)}));let g=b().providers.find(a=>a.id===c);g&&(0,F.BZ)(g)},updateApiKey:(c,d,e)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:a.apiKeys.map(a=>a.id===d?{...a,...e}:a),updatedAt:Date.now()}:a)}));let f=b().providers.find(a=>a.id===c);f&&(0,F.BZ)(f)},deleteApiKey:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:a.apiKeys.filter(a=>a.id!==d),updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},setPrimaryApiKey:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:a.apiKeys.map(a=>({...a,isPrimary:a.id===d})),updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},addModel:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,models:[...a.models,d],updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},removeModel:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,models:a.models.filter(a=>a.id!==d),updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},toggleModelEnabled:(c,d)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,models:a.models.map(a=>a.id===d?{...a,enabled:!a.enabled}:a),updatedAt:Date.now()}:a)}));let e=b().providers.find(a=>a.id===c);e&&(0,F.BZ)(e)},updateModel:(c,d,e)=>{a(a=>({providers:a.providers.map(a=>a.id===c?{...a,models:a.models.map(a=>a.id===d?{...a,...e}:a),updatedAt:Date.now()}:a)}));let f=b().providers.find(a=>a.id===c);f&&(0,F.BZ)(f)},openModelSelector:b=>{a({modelSelector:{open:!0,providerId:b,searchQuery:"",activeTab:"all",fetchedModels:[],isLoading:!1,error:null}})},closeModelSelector:()=>{a({modelSelector:{open:!1,providerId:null,searchQuery:"",activeTab:"all",fetchedModels:[],isLoading:!1,error:null}})},setModelSelectorSearch:b=>{a(a=>({modelSelector:{...a.modelSelector,searchQuery:b}}))},setModelSelectorTab:b=>{a(a=>({modelSelector:{...a.modelSelector,activeTab:b}}))},fetchModelsForSelector:async c=>{let d=b().providers.find(a=>a.id===c);if(!d)return;let e=(0,C.aA)(d);if(!e)return void a(a=>({modelSelector:{...a.modelSelector,error:"errors.missingApiKey"}}));a(a=>({modelSelector:{...a.modelSelector,isLoading:!0,error:null}}));try{let b=await (0,G.QF)(e.value,d.baseUrl,{headers:d.headers,timeout:d.timeout});a(a=>({modelSelector:{...a.modelSelector,fetchedModels:b.models,isLoading:!1,error:b.error??null}}))}catch(c){let b=c instanceof Error?c.message:"errors.failedToFetchModels";a(a=>({modelSelector:{...a.modelSelector,isLoading:!1,error:b}}))}},addFetchedModels:c=>{let{modelSelector:d}=b(),e=d.providerId;if(!e)return;let f=b().providers.find(a=>a.id===e);if(!f)return;let g=new Set(f.models.map(a=>a.id)),h=d.fetchedModels.filter(a=>c.includes(a.id)&&!g.has(a.id));a(a=>({providers:a.providers.map(a=>a.id===e?{...a,models:[...a.models,...h],updatedAt:Date.now()}:a)}));let i=b().providers.find(a=>a.id===e);i&&(0,F.BZ)(i)},checkProviderHealth:async(c,d)=>{let e=b().providers.find(a=>a.id===c);if(!e)throw Error("Provider not found");let f=await (0,G._C)(e,d);a(a=>({healthChecks:{...a.healthChecks,[c]:f}})),a(a=>({providers:a.providers.map(a=>a.id===c?{...a,apiKeys:a.apiKeys.map(a=>{let b=f.keyResults.find(b=>b.keyId===a.id);return b?{...a,healthStatus:b.status,lastChecked:f.checkedAt}:a}),updatedAt:Date.now()}:a)}));let g=b().providers.find(a=>a.id===c);g&&(0,F.BZ)(g);let h={...b().healthChecks,[c]:f};return(0,F.Tm)(h),f},checkAllProviders:async()=>{let{providers:a}=b(),c={};for(let d of a)d.enabled&&d.apiKeys.length>0&&(c[d.id]=await b().checkProviderHealth(d.id));return c},getProvider:a=>b().providers.find(b=>b.id===a)||null,getSelectedProvider:()=>{let{selectedProviderId:a,providers:c}=b();return a&&c.find(b=>b.id===a)||null}}))(a,b,...c),...((a,b)=>({toolSettings:H,draftToolUses:[],hydrateToolsFromStorage:()=>{let b=M(L(K([]),H),H);a({toolSettings:H,draftToolUses:b})},setToolSettings:b=>{a(a=>{let c=M(L(a.draftToolUses,b),b);return J(c),{toolSettings:b,draftToolUses:c}})},setDraftToolUses:c=>{let d=b().toolSettings,e=M(L(K(c),d),d);a({draftToolUses:e})},toggleDraftToolUse:a=>{let c=b().draftToolUses,d=c.includes(a)?c.filter(b=>b!==a):[...c,a];b().setDraftToolUses(d)},upsertMcpServer:b=>{let c=Date.now(),d=b.id.trim();a(a=>{let e=a.toolSettings,f=e.mcp.servers.slice(),g=f.findIndex(a=>a.id===d),h={...b,id:d,createdAt:g>=0?f[g].createdAt:c,updatedAt:c};g>=0?f[g]=h:f.push(h);let i={...e,mcp:{...e.mcp,servers:f.sort((a,b)=>b.updatedAt-a.updatedAt)}};return I(i),{toolSettings:i}})},deleteMcpServer:b=>{a(a=>{let c=a.toolSettings,d={...c,mcp:{...c.mcp,servers:c.mcp.servers.filter(a=>a.id!==b)}},e=M(a.draftToolUses,d);return I(d),J(e),{toolSettings:d,draftToolUses:e}})},syncToolsToNode:a=>{let c=b().nodes.get(a);if(c){if(c.type===g.Z.USER){let a=c.metadata.toolUses??[];b().setDraftToolUses(a);return}if(c.type===g.Z.ASSISTANT){let a=c.parentId;if(!a)return;let d=b().nodes.get(a);if(!d||d.type!==g.Z.USER)return;let e=d.metadata.toolUses??[];b().setDraftToolUses(e)}}}}))(a,b,...c)}}));function O(a){return(0,e.P)(N,a)}},3412:(a,b,c)=>{Promise.resolve().then(c.t.bind(c,1921,23)),Promise.resolve().then(c.t.bind(c,440,23)),Promise.resolve().then(c.t.bind(c,4342,23)),Promise.resolve().then(c.t.bind(c,2265,23)),Promise.resolve().then(c.t.bind(c,5421,23)),Promise.resolve().then(c.t.bind(c,1335,23)),Promise.resolve().then(c.t.bind(c,664,23)),Promise.resolve().then(c.bind(c,4661))},3763:(a,b,c)=>{"use strict";c.d(b,{Z:()=>e});var d,e=((d={}).USER="user",d.ASSISTANT="assistant",d.SYSTEM="system",d.COMPRESSED="compressed",d)},4852:(a,b,c)=>{Promise.resolve().then(c.t.bind(c,5547,23)),Promise.resolve().then(c.t.bind(c,5098,23)),Promise.resolve().then(c.t.bind(c,7644,23)),Promise.resolve().then(c.t.bind(c,3859,23)),Promise.resolve().then(c.t.bind(c,8099,23)),Promise.resolve().then(c.t.bind(c,6237,23)),Promise.resolve().then(c.t.bind(c,8562,23)),Promise.resolve().then(c.t.bind(c,6675,23))},5301:(a,b,c)=>{Promise.resolve().then(c.t.bind(c,5547,23))},5723:(a,b,c)=>{"use strict";c.d(b,{default:()=>i});var d=c(8249);c(7484);var e=c(3006);function f(){return(0,e.CU)(a=>a.theme),null}function g(){return(0,e.CU)(a=>a.locale),null}function h(){return null}function i(){return(0,d.jsxs)(d.Fragment,{children:[(0,d.jsx)(h,{}),(0,d.jsx)(f,{}),(0,d.jsx)(g,{})]})}},5951:(a,b,c)=>{"use strict";c.d(b,{ES:()=>d,ci:()=>f,g8:()=>h,sO:()=>g});let d={model:"gpt-4o-mini",temperature:.7,maxTokens:1024,selectedModels:[],compressionModel:null,summaryModel:null};function e(a){if(!a||"object"!=typeof a)return null;let{providerId:b,modelId:c}=a;return"string"!=typeof b||"string"!=typeof c?null:{providerId:b,modelId:c}}function f(a,b=d){let c="string"==typeof a.model&&a.model.trim()?a.model.trim():b.model,g="number"==typeof a.temperature&&Number.isFinite(a.temperature)?Math.min(2,Math.max(0,a.temperature)):b.temperature,h="number"==typeof a.maxTokens&&Number.isFinite(a.maxTokens)?Math.max(1,Math.round(a.maxTokens)):b.maxTokens,i=function(a,b){if(!Array.isArray(a))return b;let c=new Set,d=[];for(let b of a){if(!b||"object"!=typeof b)continue;let{providerId:a,modelId:e}=b;if("string"!=typeof a||"string"!=typeof e)continue;let f=`${a}:${e}`;c.has(f)||(c.add(f),d.push({providerId:a,modelId:e}))}return d}(a.selectedModels,b.selectedModels);return{model:c,temperature:g,maxTokens:h,selectedModels:i,compressionModel:Object.prototype.hasOwnProperty.call(a,"compressionModel")?e(a.compressionModel):b.compressionModel,summaryModel:Object.prototype.hasOwnProperty.call(a,"summaryModel")?e(a.summaryModel):b.summaryModel}}function g(){return null}function h(a){}},6239:(a,b,c)=>{"use strict";function d(a){let b=a.trim();if(!b)return 0;let c=(b.match(/[\u3040-\u30ff\u3400-\u4dbf\u4e00-\u9fff\uf900-\ufaff]/g)??[]).length;return Math.ceil((b.length-c)/4+c/2)}c.d(b,{b:()=>d})},6487:()=>{},6537:(a,b,c)=>{"use strict";c.r(b),c.d(b,{default:()=>g,metadata:()=>f});var d=c(5735);c(1135);var e=c(768);let f={title:"Prompt Tree - AI Dialogue Topology",description:"A visual dialogue tree interface for AI conversations with context management and token optimization.",keywords:["AI","dialogue","chat","context","topology","tree"],authors:[{name:"Prompt Tree Team"}],icons:{icon:"/icon.svg",apple:"/icon.svg"}};function g({children:a}){return(0,d.jsx)("html",{lang:"en",children:(0,d.jsxs)("body",{className:"antialiased",children:[(0,d.jsx)(e.default,{}),a]})})}},6684:(a,b,c)=>{"use strict";function d(a){let b=a.trim().replace(/\/+$/,"");return b.endsWith("/chat/completions")&&(b=b.replace(/\/chat\/completions$/,"")),b.endsWith("/models")&&(b=b.replace(/\/models$/,"")),!b.includes("/v1")&&!b.includes("/api")&&b.includes("api.openai.com")&&(b=`${b}/v1`),b}async function e(a,b,c){let d=await fetch(a,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(b),signal:c});if(!d.ok){let a=await d.text().catch(()=>"");throw Error(`Request failed (${d.status}): ${a||d.statusText}`)}return await d.json()}function f(a,b,c){return{apiKey:a,baseUrl:d(b),headers:c?.headers,timeout:c?.timeout}}async function g(a,b,c){try{let d=await e("/api/providers/models",f(a,b,c),c?.signal);if(d.error)return{models:[],error:d.error};let g=(d.models??[]).filter(a=>a&&"string"==typeof a.id).map(a=>{var b,c;let d;return{id:a.id,name:a.id,enabled:!1,category:(b=a.id,c=a.object,(d=b.toLowerCase()).includes("vision")||d.includes("image")||d.includes("gpt-4o")||d.includes("claude-3")?"vision":d.includes("embed")?"embedding":d.includes("reason")||d.includes("o1")||d.includes("r1")?"reasoning":d.includes("tool")?"tool":"embedding"===c?"embedding":"chat")}});return g.sort((a,b)=>a.id.localeCompare(b.id)),{models:g}}catch(a){if(a instanceof Error)return{models:[],error:a.message};return{models:[],error:"errors.unknownError"}}}async function h(a,b){try{return await e("/api/providers/health",{providerId:a.id,baseUrl:d(a.baseUrl),apiKeys:a.apiKeys.map(a=>({id:a.id,value:a.value,isPrimary:a.isPrimary})),headers:a.headers,timeout:a.timeout},b)}catch(c){let b=c instanceof Error?c.message:"errors.connectionFailed";return{providerId:a.id,status:"error",keyResults:a.apiKeys.map(a=>({keyId:a.id,status:"error",error:b})),checkedAt:Date.now()}}}async function i(a,b,c,d){try{return await e("/api/providers/test",{...f(a,b,d),model:c,prompt:d?.prompt},d?.signal)}catch(a){return{status:"error",error:a instanceof Error?a.message:"errors.connectionFailed"}}}c.d(b,{DT:()=>i,QF:()=>g,_C:()=>h,qT:()=>d})},7094:(a,b,c)=>{"use strict";function d(a){return new Promise((b,c)=>{a.onsuccess=()=>b(a.result),a.onerror=()=>c(a.error)})}function e(a){return new Promise((b,c)=>{a.oncomplete=()=>b(),a.onerror=()=>c(a.error),a.onabort=()=>c(a.error??Error("IndexedDB transaction aborted."))})}c.d(b,{PE:()=>e,k7:()=>d})},7157:(a,b,c)=>{Promise.resolve().then(c.t.bind(c,1921,23))},7440:(a,b,c)=>{"use strict";function d(){return"https://api.openai.com/v1"}function e(a){}c.d(b,{$t:()=>e,LB:()=>d})},7558:(a,b,c)=>{"use strict";c.d(b,{l:()=>e});var d=c(4579);function e(){return"u">typeof crypto&&"randomUUID"in crypto?crypto.randomUUID():(0,d.A)()}},7872:(a,b,c)=>{"use strict";function d(a){let b=Date.now();return{id:`provider_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,name:a,baseUrl:"https://api.openai.com/v1",apiKeys:[],models:[],enabled:!0,timeout:3e4,createdAt:b,updatedAt:b}}function e(a,b){return{id:`key_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,value:a.trim(),name:b||`Key ${new Date().toLocaleDateString()}`,isPrimary:!1,healthStatus:"unknown"}}function f(a){let b=a.apiKeys.find(a=>a.isPrimary);return b||a.apiKeys[0]||null}c.d(b,{DA:()=>d,Iq:()=>e,aA:()=>f})},8033:(a,b,c)=>{"use strict";function d(a,b){let c=[],d=new Set(a);if(d.has("web_search")){let a="exa"===b.search.provider?"Exa":"Tavily";c.push({id:"web_search",title:"Tool Use: Web Search",content:`You can search the web for up-to-date information.
|
|
5
5
|
|
|
6
6
|
How to use:
|
|
7
7
|
- Call \`web_search\` with a natural language \`query\`.
|
|
@@ -27,4 +27,4 @@ How to use:
|
|
|
27
27
|
|
|
28
28
|
Limits:
|
|
29
29
|
- Timeout: ${Math.round(b.python.timeoutMs/1e3)}s
|
|
30
|
-
- Max output: ${b.python.maxOutputChars} chars (stdout+stderr)`}),c}c.d(b,{d:()=>d})},8283:(a,b,c)=>{Promise.resolve().then(c.bind(c,768))},8335:()=>{},8729:(a,b,c)=>{"use strict";c.d(b,{$u:()=>j,VK:()=>h,bz:()=>k,qi:()=>i});var d=c(3763);let e={xSpacing:280,ySpacing:140};function f(a){let b=Array.from(a),c=new Map(b.map(a=>[a.id,a])),e=new Map;for(let a of b){if(!a.parentId)continue;let b=e.get(a.parentId);b?b.push(a):e.set(a.parentId,[a])}let f=new Set,g=new Map,h=new Map;for(let a of b){if(a.type!==d.Z.COMPRESSED||!a.metadata?.collapsed)continue;let b=a.metadata.compressedNodeIds??[];if(0===b.length)continue;let i=new Set(b);for(let c of b)f.add(c),g.set(c,a.id);let j=new Set,k=[...b];for(;k.length>0;){let b=k.pop();if(!(!b||j.has(b)))for(let c of(j.add(b),e.get(b)??[]))c.id!==a.id&&(f.has(c.id)||(f.add(c.id),g.set(c.id,a.id)),k.push(c.id))}let l=b.find(a=>{let b=c.get(a);return!!b&&(!b.parentId||!i.has(b.parentId))}),m=l?c.get(l)??null:null;h.set(a.id,m?.parentId??null)}return b.filter(a=>!f.has(a.id)).map(a=>{let b=a.parentId??null,c=b;return(h.has(a.id)?c=h.get(a.id)??null:c&&g.has(c)&&(c=g.get(c)??null),c===b)?a:{...a,parentId:c}})}function g(a){let b=new Map;for(let c of a)b.set(c.id,c);let c=new Map;for(let a of b.values()){if(!a.parentId)continue;let b=c.get(a.parentId);b?b.push(a):c.set(a.parentId,[a])}for(let a of c.values())a.sort((a,b)=>a.createdAt-b.createdAt);return{byId:b,childrenByParent:c}}function h(a,b){let c=[],d=new Set,e=a.get(b)??null;for(;e&&!d.has(e.id)&&(d.add(e.id),c.unshift(e.id),e.parentId);)e=a.get(e.parentId)??null;return c}function i(a,b){let{byId:c,childrenByParent:d}=g(f(a));if(!c.has(b))return 0;let e=new Set,h=a=>{if(e.has(a))return 0;e.add(a);let b=d.get(a)??[];if(0===b.length)return 1;let c=0;for(let a of b)c+=h(a.id);return c};return h(b)}function j(a,b,c){let{byId:d,childrenByParent:h}=g(f(a));if(!d.has(b))return new Map;let{xSpacing:i,ySpacing:j}={...e,...c},k=new Map,l=0,m=(a,b)=>{if(!d.get(a))return 0;let c=h.get(a)??[],e=b*i;if(0===c.length){let b=l;return l+=j,k.set(a,{x:e,y:b}),b}let f=[];for(let a of c)f.push(m(a.id,b+1));let g=0===f.length?l:f.reduce((a,b)=>a+b,0)/f.length;return k.set(a,{x:e,y:g}),g};return m(b,0),k}function k({nodes:a,rootId:b,activeNodeId:c,selectedNodeIds:d,forceAutoLayout:k=!1,layout:l,onToggleCollapse:m}){let n=f(a),{byId:o,childrenByParent:p}=g(n),q=i(n,b),r=new Set(c?h(o,c):[]),s=new Set(d??[]),t=k||Array.from(o.values()).some(a=>!a.position)?j(o.values(),b,l):new Map,u=[],v=new Map,w=[b];v.set(b,0);let x=new Set;for(;w.length;){let a=w.shift();if(!a||x.has(a))continue;x.add(a);let b=o.get(a);if(!b)continue;u.push(b);let c=(v.get(a)??0)+1;for(let b of p.get(a)??[])v.has(b.id)||v.set(b.id,c),w.push(b.id)}let y=u.map(a=>{let b=v.get(a.id)??0,d=k?t.get(a.id)??{x:b*e.xSpacing,y:0}:a.position??t.get(a.id)??{x:b*e.xSpacing,y:0};return{id:a.id,type:"treeNode",position:d,draggable:!0,selectable:!0,selected:s.has(a.id),data:{node:a,depth:b,isActive:c===a.id,isSelected:s.has(a.id),isInActivePath:r.has(a.id),onToggleCollapse:m}}}),z=[];for(let a of u)a.parentId&&o.has(a.parentId)&&z.push({id:`${a.parentId}-${a.id}`,source:a.parentId,target:a.id,type:"treeEdge",data:{isInActivePath:r.has(a.id)}});return{nodes:y,edges:z,branchCount:q}}},8859:(a,b,c)=>{"use strict";c.d(b,{w:()=>k});var d=c(518),e=c(7094),f=c(2771),g=c(6239),h=c(7558),i=c(3763);function j(a){return{tags:a?.tags??[],metaInstructions:a?.metaInstructions??{},compressedNodeIds:a?.compressedNodeIds,collapsed:a?.collapsed,branchLabel:a?.branchLabel,modelId:a?.modelId,modelName:a?.modelName,providerId:a?.providerId,providerName:a?.providerName,toolUses:a?.toolUses,toolLogs:a?.toolLogs}}class k{async create(a){let b=Date.now(),c=a.createdAt??b,k=a.updatedAt??c,l=a.type??i.Z.USER,m=a.content??"",n=a.summary,o=l===i.Z.COMPRESSED?n??m:m,p={id:a.id??(0,h.l)(),type:l,createdAt:c,updatedAt:k,parentId:a.parentId??null,content:m,summary:n,metadata:j(a.metadata),tokenCount:a.tokenCount??(0,g.b)(o),position:a.position,style:a.style},q=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readwrite");return q.objectStore(f.w.stores.nodes.name).put(p),await (0,e.PE)(q),p}async read(a){let b=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readonly"),c=b.objectStore(f.w.stores.nodes.name),g=await (0,e.k7)(c.get(a));return await (0,e.PE)(b),g??null}async update(a,b){let c=await this.read(a);if(!c)throw Error(`Node ${a} not found`);let h=b.type??c.type,j=Object.prototype.hasOwnProperty.call(b,"content")?b.content??"":c.content,k=Object.prototype.hasOwnProperty.call(b,"summary")?b.summary:c.summary,l=h===i.Z.COMPRESSED?k??j:j,m={...c,...b,id:a,type:h,content:j,summary:k,metadata:b.metadata?{...c.metadata,...b.metadata,metaInstructions:{...c.metadata.metaInstructions,...b.metadata.metaInstructions}}:c.metadata,tokenCount:b.tokenCount??(Object.prototype.hasOwnProperty.call(b,"content")||Object.prototype.hasOwnProperty.call(b,"summary")||Object.prototype.hasOwnProperty.call(b,"type")?(0,g.b)(l):c.tokenCount),updatedAt:Date.now()},n=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readwrite");return n.objectStore(f.w.stores.nodes.name).put(m),await (0,e.PE)(n),m}async delete(a){for(let b of(await this.getChildren(a)))await this.delete(b.id);let b=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readwrite");b.objectStore(f.w.stores.nodes.name).delete(a),await (0,e.PE)(b)}async getChildren(a){let b=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readonly"),c=b.objectStore(f.w.stores.nodes.name).index("parentId"),g=await (0,e.k7)(c.getAll(a));return await (0,e.PE)(b),(g??[]).slice().sort((a,b)=>a.createdAt-b.createdAt)}async getPath(a){let b=[],c=await this.read(a);for(;c&&(b.unshift(c),c.parentId);)c=await this.read(c.parentId);return b}async search(a){let b=a.trim().toLowerCase();if(!b)return[];let c=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readonly"),g=c.objectStore(f.w.stores.nodes.name),h=await (0,e.k7)(g.getAll());return await (0,e.PE)(c),(h??[]).filter(a=>{let c=a.content.toLowerCase().includes(b),d=a.metadata.tags.some(a=>a.toLowerCase().includes(b));return c||d})}async batchCreate(a){let b=Date.now(),c=[],k=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readwrite"),l=k.objectStore(f.w.stores.nodes.name);for(let d of a){let a=d.createdAt??b,e=d.updatedAt??a,f=d.type??i.Z.USER,k=d.content??"",m=d.summary,n=f===i.Z.COMPRESSED?m??k:k,o={id:d.id??(0,h.l)(),type:f,createdAt:a,updatedAt:e,parentId:d.parentId??null,content:k,summary:m,metadata:j(d.metadata),tokenCount:d.tokenCount??(0,g.b)(n),position:d.position,style:d.style};c.push(o),l.put(o)}return await (0,e.PE)(k),c}}},8899:(a,b,c)=>{Promise.resolve().then(c.bind(c,5723))}};
|
|
30
|
+
- Max output: ${b.python.maxOutputChars} chars (stdout+stderr)`}),c}c.d(b,{d:()=>d})},8059:(a,b,c)=>{"use strict";c.r(b),c.d(b,{default:()=>e});var d=c(8868);let e=async a=>[{type:"image/x-icon",sizes:"16x16",url:(0,d.fillMetadataSegment)(".",await a.params,"favicon.ico")+"?603d046c9a6fdfbb"}]},8283:(a,b,c)=>{Promise.resolve().then(c.bind(c,768))},8335:()=>{},8729:(a,b,c)=>{"use strict";c.d(b,{$u:()=>j,VK:()=>h,bz:()=>k,qi:()=>i});var d=c(3763);let e={xSpacing:280,ySpacing:140};function f(a){let b=Array.from(a),c=new Map(b.map(a=>[a.id,a])),e=new Map;for(let a of b){if(!a.parentId)continue;let b=e.get(a.parentId);b?b.push(a):e.set(a.parentId,[a])}let f=new Set,g=new Map,h=new Map;for(let a of b){if(a.type!==d.Z.COMPRESSED||!a.metadata?.collapsed)continue;let b=a.metadata.compressedNodeIds??[];if(0===b.length)continue;let i=new Set(b);for(let c of b)f.add(c),g.set(c,a.id);let j=new Set,k=[...b];for(;k.length>0;){let b=k.pop();if(!(!b||j.has(b)))for(let c of(j.add(b),e.get(b)??[]))c.id!==a.id&&(f.has(c.id)||(f.add(c.id),g.set(c.id,a.id)),k.push(c.id))}let l=b.find(a=>{let b=c.get(a);return!!b&&(!b.parentId||!i.has(b.parentId))}),m=l?c.get(l)??null:null;h.set(a.id,m?.parentId??null)}return b.filter(a=>!f.has(a.id)).map(a=>{let b=a.parentId??null,c=b;return(h.has(a.id)?c=h.get(a.id)??null:c&&g.has(c)&&(c=g.get(c)??null),c===b)?a:{...a,parentId:c}})}function g(a){let b=new Map;for(let c of a)b.set(c.id,c);let c=new Map;for(let a of b.values()){if(!a.parentId)continue;let b=c.get(a.parentId);b?b.push(a):c.set(a.parentId,[a])}for(let a of c.values())a.sort((a,b)=>a.createdAt-b.createdAt);return{byId:b,childrenByParent:c}}function h(a,b){let c=[],d=new Set,e=a.get(b)??null;for(;e&&!d.has(e.id)&&(d.add(e.id),c.unshift(e.id),e.parentId);)e=a.get(e.parentId)??null;return c}function i(a,b){let{byId:c,childrenByParent:d}=g(f(a));if(!c.has(b))return 0;let e=new Set,h=a=>{if(e.has(a))return 0;e.add(a);let b=d.get(a)??[];if(0===b.length)return 1;let c=0;for(let a of b)c+=h(a.id);return c};return h(b)}function j(a,b,c){let{byId:d,childrenByParent:h}=g(f(a));if(!d.has(b))return new Map;let{xSpacing:i,ySpacing:j}={...e,...c},k=new Map,l=0,m=(a,b)=>{if(!d.get(a))return 0;let c=h.get(a)??[],e=b*i;if(0===c.length){let b=l;return l+=j,k.set(a,{x:e,y:b}),b}let f=[];for(let a of c)f.push(m(a.id,b+1));let g=0===f.length?l:f.reduce((a,b)=>a+b,0)/f.length;return k.set(a,{x:e,y:g}),g};return m(b,0),k}function k({nodes:a,rootId:b,activeNodeId:c,selectedNodeIds:d,forceAutoLayout:k=!1,layout:l,onToggleCollapse:m}){let n=f(a),{byId:o,childrenByParent:p}=g(n),q=i(n,b),r=new Set(c?h(o,c):[]),s=new Set(d??[]),t=k||Array.from(o.values()).some(a=>!a.position)?j(o.values(),b,l):new Map,u=[],v=new Map,w=[b];v.set(b,0);let x=new Set;for(;w.length;){let a=w.shift();if(!a||x.has(a))continue;x.add(a);let b=o.get(a);if(!b)continue;u.push(b);let c=(v.get(a)??0)+1;for(let b of p.get(a)??[])v.has(b.id)||v.set(b.id,c),w.push(b.id)}let y=u.map(a=>{let b=v.get(a.id)??0,d=k?t.get(a.id)??{x:b*e.xSpacing,y:0}:a.position??t.get(a.id)??{x:b*e.xSpacing,y:0};return{id:a.id,type:"treeNode",position:d,draggable:!0,selectable:!0,selected:s.has(a.id),data:{node:a,depth:b,isActive:c===a.id,isSelected:s.has(a.id),isInActivePath:r.has(a.id),onToggleCollapse:m}}}),z=[];for(let a of u)a.parentId&&o.has(a.parentId)&&z.push({id:`${a.parentId}-${a.id}`,source:a.parentId,target:a.id,type:"treeEdge",data:{isInActivePath:r.has(a.id)}});return{nodes:y,edges:z,branchCount:q}}},8859:(a,b,c)=>{"use strict";c.d(b,{w:()=>k});var d=c(518),e=c(7094),f=c(2771),g=c(6239),h=c(7558),i=c(3763);function j(a){return{tags:a?.tags??[],metaInstructions:a?.metaInstructions??{},compressedNodeIds:a?.compressedNodeIds,collapsed:a?.collapsed,branchLabel:a?.branchLabel,modelId:a?.modelId,modelName:a?.modelName,providerId:a?.providerId,providerName:a?.providerName,toolUses:a?.toolUses,toolLogs:a?.toolLogs}}class k{async create(a){let b=Date.now(),c=a.createdAt??b,k=a.updatedAt??c,l=a.type??i.Z.USER,m=a.content??"",n=a.summary,o=l===i.Z.COMPRESSED?n??m:m,p={id:a.id??(0,h.l)(),type:l,createdAt:c,updatedAt:k,parentId:a.parentId??null,content:m,summary:n,metadata:j(a.metadata),tokenCount:a.tokenCount??(0,g.b)(o),position:a.position,style:a.style},q=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readwrite");return q.objectStore(f.w.stores.nodes.name).put(p),await (0,e.PE)(q),p}async read(a){let b=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readonly"),c=b.objectStore(f.w.stores.nodes.name),g=await (0,e.k7)(c.get(a));return await (0,e.PE)(b),g??null}async update(a,b){let c=await this.read(a);if(!c)throw Error(`Node ${a} not found`);let h=b.type??c.type,j=Object.prototype.hasOwnProperty.call(b,"content")?b.content??"":c.content,k=Object.prototype.hasOwnProperty.call(b,"summary")?b.summary:c.summary,l=h===i.Z.COMPRESSED?k??j:j,m={...c,...b,id:a,type:h,content:j,summary:k,metadata:b.metadata?{...c.metadata,...b.metadata,metaInstructions:{...c.metadata.metaInstructions,...b.metadata.metaInstructions}}:c.metadata,tokenCount:b.tokenCount??(Object.prototype.hasOwnProperty.call(b,"content")||Object.prototype.hasOwnProperty.call(b,"summary")||Object.prototype.hasOwnProperty.call(b,"type")?(0,g.b)(l):c.tokenCount),updatedAt:Date.now()},n=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readwrite");return n.objectStore(f.w.stores.nodes.name).put(m),await (0,e.PE)(n),m}async delete(a){for(let b of(await this.getChildren(a)))await this.delete(b.id);let b=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readwrite");b.objectStore(f.w.stores.nodes.name).delete(a),await (0,e.PE)(b)}async getChildren(a){let b=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readonly"),c=b.objectStore(f.w.stores.nodes.name).index("parentId"),g=await (0,e.k7)(c.getAll(a));return await (0,e.PE)(b),(g??[]).slice().sort((a,b)=>a.createdAt-b.createdAt)}async getPath(a){let b=[],c=await this.read(a);for(;c&&(b.unshift(c),c.parentId);)c=await this.read(c.parentId);return b}async search(a){let b=a.trim().toLowerCase();if(!b)return[];let c=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readonly"),g=c.objectStore(f.w.stores.nodes.name),h=await (0,e.k7)(g.getAll());return await (0,e.PE)(c),(h??[]).filter(a=>{let c=a.content.toLowerCase().includes(b),d=a.metadata.tags.some(a=>a.toLowerCase().includes(b));return c||d})}async batchCreate(a){let b=Date.now(),c=[],k=(await (0,d.xA)()).transaction([f.w.stores.nodes.name],"readwrite"),l=k.objectStore(f.w.stores.nodes.name);for(let d of a){let a=d.createdAt??b,e=d.updatedAt??a,f=d.type??i.Z.USER,k=d.content??"",m=d.summary,n=f===i.Z.COMPRESSED?m??k:k,o={id:d.id??(0,h.l)(),type:f,createdAt:a,updatedAt:e,parentId:d.parentId??null,content:k,summary:m,metadata:j(d.metadata),tokenCount:d.tokenCount??(0,g.b)(n),position:d.position,style:d.style};c.push(o),l.put(o)}return await (0,e.PE)(k),c}}},8899:(a,b,c)=>{Promise.resolve().then(c.bind(c,5723))}};
|