Merge remote-tracking branch 'origin/main' into v2

This commit is contained in:
MyPrototypeWhat 2025-12-02 18:35:27 +08:00
commit b4d613972f
9 changed files with 200 additions and 32 deletions

View File

@ -166,7 +166,7 @@
"@langchain/core": "patch:@langchain/core@npm%3A1.0.2#~/.yarn/patches/@langchain-core-npm-1.0.2-183ef83fe4.patch",
"@langchain/openai": "patch:@langchain/openai@npm%3A1.0.0#~/.yarn/patches/@langchain-openai-npm-1.0.0-474d0ad9d4.patch",
"@mistralai/mistralai": "^1.7.5",
"@modelcontextprotocol/sdk": "^1.17.5",
"@modelcontextprotocol/sdk": "^1.23.0",
"@mozilla/readability": "^0.6.0",
"@notionhq/client": "^2.2.15",
"@openrouter/ai-sdk-provider": "^1.2.8",
@ -212,6 +212,7 @@
"@types/content-type": "^1.1.9",
"@types/cors": "^2.8.19",
"@types/diff": "^7",
"@types/dotenv": "^8.2.3",
"@types/express": "^5",
"@types/fs-extra": "^11",
"@types/he": "^1",

View File

@ -13,6 +13,7 @@ import { TraceMethod, withSpanFunc } from '@mcp-trace/trace-core'
import { Client } from '@modelcontextprotocol/sdk/client/index.js'
import type { SSEClientTransportOptions } from '@modelcontextprotocol/sdk/client/sse.js'
import { SSEClientTransport } from '@modelcontextprotocol/sdk/client/sse.js'
import type { StdioServerParameters } from '@modelcontextprotocol/sdk/client/stdio.js'
import { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js'
import {
StreamableHTTPClientTransport,
@ -43,11 +44,14 @@ import {
type MCPPrompt,
type MCPResource,
type MCPServer,
type MCPTool
type MCPTool,
MCPToolInputSchema,
MCPToolOutputSchema
} from '@types'
import { app, net } from 'electron'
import { EventEmitter } from 'events'
import { v4 as uuidv4 } from 'uuid'
import * as z from 'zod'
import DxtService from './DxtService'
import { CallBackServer } from './mcp/oauth/callback'
@ -343,7 +347,7 @@ class McpService {
removeEnvProxy(loginShellEnv)
}
const transportOptions: any = {
const transportOptions: StdioServerParameters = {
command: cmd,
args,
env: {
@ -620,6 +624,8 @@ class McpService {
tools.map((tool: SDKTool) => {
const serverTool: MCPTool = {
...tool,
inputSchema: z.parse(MCPToolInputSchema, tool.inputSchema),
outputSchema: tool.outputSchema ? z.parse(MCPToolOutputSchema, tool.outputSchema) : undefined,
id: buildFunctionCallToolName(server.name, tool.name, server.id),
serverId: server.id,
serverName: server.name,

View File

@ -177,8 +177,12 @@ export async function buildStreamTextParams(
let headers: Record<string, string | undefined> = options.requestOptions?.headers ?? {}
if (isAnthropicModel(model) && !isAwsBedrockProvider(provider)) {
const newBetaHeaders = { 'anthropic-beta': addAnthropicHeaders(assistant, model).join(',') }
headers = combineHeaders(headers, newBetaHeaders)
const betaHeaders = addAnthropicHeaders(assistant, model)
// Only add the anthropic-beta header if there are actual beta headers to include
if (betaHeaders.length > 0) {
const newBetaHeaders = { 'anthropic-beta': betaHeaders.join(',') }
headers = combineHeaders(headers, newBetaHeaders)
}
}
// 构建基础参数

View File

@ -9,6 +9,7 @@ import { useMCPServer, useMCPServers } from '@renderer/hooks/useMCPServers'
import { useMCPServerTrust } from '@renderer/hooks/useMCPServerTrust'
import MCPDescription from '@renderer/pages/settings/MCPSettings/McpDescription'
import type { MCPPrompt, MCPResource, MCPServer, MCPTool } from '@renderer/types'
import { parseKeyValueString } from '@renderer/utils/env'
import { formatMcpError } from '@renderer/utils/error'
import type { TabsProps } from 'antd'
import { Badge, Form, Input, Radio, Select, Tabs } from 'antd'
@ -65,21 +66,6 @@ const PipRegistry: Registry[] = [
type TabKey = 'settings' | 'description' | 'tools' | 'prompts' | 'resources'
const parseKeyValueString = (str: string): Record<string, string> => {
const result: Record<string, string> = {}
str.split('\n').forEach((line) => {
if (line.trim()) {
const [key, ...value] = line.split('=')
const formatValue = value.join('=').trim()
const formatKey = key.trim()
if (formatKey && formatValue) {
result[formatKey] = formatValue
}
}
})
return result
}
const McpSettings: React.FC = () => {
const { t } = useTranslation()
const { serverId } = useParams<{ serverId: string }>()

View File

@ -34,6 +34,15 @@ export const MCPToolInputSchema = z
required: z.array(z.string()).optional()
})
.loose()
.transform((schema) => {
if (!schema.properties) {
schema.properties = {}
}
if (!schema.required) {
schema.required = []
}
return schema
})
export interface BuiltinTool extends BaseTool {
inputSchema: z.infer<typeof MCPToolInputSchema>

View File

@ -0,0 +1,98 @@
import { describe, expect, it } from 'vitest'
import { parseKeyValueString } from '../env'
describe('parseKeyValueString', () => {
it('should parse empty string', () => {
expect(parseKeyValueString('')).toEqual({})
})
it('should parse single key-value pair', () => {
expect(parseKeyValueString('KEY=value')).toEqual({ KEY: 'value' })
})
it('should parse multiple key-value pairs', () => {
const input = `KEY1=value1
KEY2=value2
KEY3=value3`
expect(parseKeyValueString(input)).toEqual({
KEY1: 'value1',
KEY2: 'value2',
KEY3: 'value3'
})
})
it('should handle quoted values', () => {
expect(parseKeyValueString('KEY="quoted value"')).toEqual({ KEY: 'quoted value' })
})
it('should handle single quoted values', () => {
expect(parseKeyValueString("KEY='single quoted'")).toEqual({ KEY: 'single quoted' })
})
it('should handle values with equals signs', () => {
expect(parseKeyValueString('URL=https://example.com?param=value')).toEqual({
URL: 'https://example.com?param=value'
})
})
it('should handle empty values', () => {
expect(parseKeyValueString('KEY=')).toEqual({ KEY: '' })
})
it('should handle comments', () => {
const input = `KEY=value
# This is a comment
ANOTHER_KEY=another_value`
expect(parseKeyValueString(input)).toEqual({
KEY: 'value',
ANOTHER_KEY: 'another_value'
})
})
it('should handle whitespace around key-value pairs', () => {
expect(parseKeyValueString(' KEY=value \n ANOTHER=another ')).toEqual({
KEY: 'value',
ANOTHER: 'another'
})
})
it('should handle special characters in values', () => {
expect(parseKeyValueString('KEY=value with spaces & symbols!')).toEqual({
KEY: 'value with spaces & symbols!'
})
})
it('should handle multiline values', () => {
const input = `KEY="value
with
multiple
lines"`
expect(parseKeyValueString(input)).toEqual({
KEY: 'value\nwith\nmultiple\nlines'
})
})
it('should handle invalid lines gracefully', () => {
const input = `KEY=value
invalid line without equals
ANOTHER_KEY=another_value`
expect(parseKeyValueString(input)).toEqual({
KEY: 'value',
ANOTHER_KEY: 'another_value'
})
})
it('should handle duplicate keys (last one wins)', () => {
const input = `KEY=first
KEY=second
KEY=third`
expect(parseKeyValueString(input)).toEqual({ KEY: 'third' })
})
it('should handle keys and values with special characters', () => {
expect(parseKeyValueString('API-URL_123=https://api.example.com/v1/users')).toEqual({
'API-URL_123': 'https://api.example.com/v1/users'
})
})
})

View File

@ -0,0 +1,5 @@
import { parse } from 'dotenv'
export const parseKeyValueString = (str: string): Record<string, string> => {
return parse(str)
}

View File

@ -136,7 +136,10 @@ export async function callMCPTool(
topicId?: string,
modelName?: string
): Promise<MCPCallToolResponse> {
logger.info(`Calling Tool: ${toolResponse.tool.serverName} ${toolResponse.tool.name}`, toolResponse.tool)
logger.info(
`Calling Tool: ${toolResponse.id} ${toolResponse.tool.serverName} ${toolResponse.tool.name}`,
toolResponse.tool
)
try {
const server = getMcpServerByTool(toolResponse.tool)

View File

@ -6430,11 +6430,12 @@ __metadata:
languageName: node
linkType: hard
"@modelcontextprotocol/sdk@npm:^1.17.5":
version: 1.17.5
resolution: "@modelcontextprotocol/sdk@npm:1.17.5"
"@modelcontextprotocol/sdk@npm:^1.23.0":
version: 1.23.0
resolution: "@modelcontextprotocol/sdk@npm:1.23.0"
dependencies:
ajv: "npm:^6.12.6"
ajv: "npm:^8.17.1"
ajv-formats: "npm:^3.0.1"
content-type: "npm:^1.0.5"
cors: "npm:^2.8.5"
cross-spawn: "npm:^7.0.5"
@ -6444,9 +6445,17 @@ __metadata:
express-rate-limit: "npm:^7.5.0"
pkce-challenge: "npm:^5.0.0"
raw-body: "npm:^3.0.0"
zod: "npm:^3.23.8"
zod-to-json-schema: "npm:^3.24.1"
checksum: 10c0/182b92b5e7c07da428fd23c6de22021c4f9a91f799c02a8ef15def07e4f9361d0fc22303548658fec2a700623535fd44a9dc4d010fb5d803a8f80e3c6c64a45e
zod: "npm:^3.25 || ^4.0"
zod-to-json-schema: "npm:^3.25.0"
peerDependencies:
"@cfworker/json-schema": ^4.1.1
zod: ^3.25 || ^4.0
peerDependenciesMeta:
"@cfworker/json-schema":
optional: true
zod:
optional: false
checksum: 10c0/b0291f921ad9bda06bbf1a61b1bb61ceca1173da5d74d39a411c40428d6ca50a95f0de3a1631f25a44b439220b15c30c1306600bf48bef665ab7ad118d528260
languageName: node
linkType: hard
@ -12196,6 +12205,15 @@ __metadata:
languageName: node
linkType: hard
"@types/dotenv@npm:^8.2.3":
version: 8.2.3
resolution: "@types/dotenv@npm:8.2.3"
dependencies:
dotenv: "npm:*"
checksum: 10c0/af9178da617959cddc8259aaa3f16c474523ead469f4a03490de2f2d1cafc8615c5d0d1ed3fad837096218126421c38cd46b4065548bb5aee3cc002c518b69f7
languageName: node
linkType: hard
"@types/estree-jsx@npm:^1.0.0":
version: 1.0.5
resolution: "@types/estree-jsx@npm:1.0.5"
@ -13857,7 +13875,7 @@ __metadata:
"@libsql/client": "npm:0.14.0"
"@libsql/win32-x64-msvc": "npm:^0.4.7"
"@mistralai/mistralai": "npm:^1.7.5"
"@modelcontextprotocol/sdk": "npm:^1.17.5"
"@modelcontextprotocol/sdk": "npm:^1.23.0"
"@mozilla/readability": "npm:^0.6.0"
"@napi-rs/system-ocr": "patch:@napi-rs/system-ocr@npm%3A1.0.2#~/.yarn/patches/@napi-rs-system-ocr-npm-1.0.2-59e7a78e8b.patch"
"@notionhq/client": "npm:^2.2.15"
@ -13906,6 +13924,7 @@ __metadata:
"@types/content-type": "npm:^1.1.9"
"@types/cors": "npm:^2.8.19"
"@types/diff": "npm:^7"
"@types/dotenv": "npm:^8.2.3"
"@types/express": "npm:^5"
"@types/fs-extra": "npm:^11"
"@types/he": "npm:^1"
@ -14208,6 +14227,20 @@ __metadata:
languageName: node
linkType: hard
"ajv-formats@npm:^3.0.1":
version: 3.0.1
resolution: "ajv-formats@npm:3.0.1"
dependencies:
ajv: "npm:^8.0.0"
peerDependencies:
ajv: ^8.0.0
peerDependenciesMeta:
ajv:
optional: true
checksum: 10c0/168d6bca1ea9f163b41c8147bae537e67bd963357a5488a1eaf3abe8baa8eec806d4e45f15b10767e6020679315c7e1e5e6803088dfb84efa2b4e9353b83dd0a
languageName: node
linkType: hard
"ajv-keywords@npm:^3.4.1":
version: 3.5.2
resolution: "ajv-keywords@npm:3.5.2"
@ -14217,7 +14250,7 @@ __metadata:
languageName: node
linkType: hard
"ajv@npm:^6.10.0, ajv@npm:^6.12.0, ajv@npm:^6.12.4, ajv@npm:^6.12.6":
"ajv@npm:^6.10.0, ajv@npm:^6.12.0, ajv@npm:^6.12.4":
version: 6.12.6
resolution: "ajv@npm:6.12.6"
dependencies:
@ -14229,7 +14262,7 @@ __metadata:
languageName: node
linkType: hard
"ajv@npm:^8.0.0, ajv@npm:^8.6.3":
"ajv@npm:^8.0.0, ajv@npm:^8.17.1, ajv@npm:^8.6.3":
version: 8.17.1
resolution: "ajv@npm:8.17.1"
dependencies:
@ -17407,6 +17440,13 @@ __metadata:
languageName: node
linkType: hard
"dotenv@npm:*":
version: 17.2.3
resolution: "dotenv@npm:17.2.3"
checksum: 10c0/c884403209f713214a1b64d4d1defa4934c2aa5b0002f5a670ae298a51e3c3ad3ba79dfee2f8df49f01ae74290fcd9acdb1ab1d09c7bfb42b539036108bb2ba0
languageName: node
linkType: hard
"dotenv@npm:^16.1.4, dotenv@npm:^16.3.0, dotenv@npm:^16.3.1, dotenv@npm:^16.4.5":
version: 16.6.1
resolution: "dotenv@npm:16.6.1"
@ -30758,6 +30798,15 @@ __metadata:
languageName: node
linkType: hard
"zod-to-json-schema@npm:^3.25.0":
version: 3.25.0
resolution: "zod-to-json-schema@npm:3.25.0"
peerDependencies:
zod: ^3.25 || ^4
checksum: 10c0/2d2cf6ca49752bf3dc5fb37bc8f275eddbbc4020e7958d9c198ea88cd197a5f527459118188a0081b889da6a6474d64c4134cd60951fa70178c125138761c680
languageName: node
linkType: hard
"zod-validation-error@npm:^3.4.0":
version: 3.4.0
resolution: "zod-validation-error@npm:3.4.0"
@ -30767,13 +30816,20 @@ __metadata:
languageName: node
linkType: hard
"zod@npm:^3.22.4, zod@npm:^3.23.8, zod@npm:^3.24.1":
"zod@npm:^3.22.4, zod@npm:^3.24.1":
version: 3.25.56
resolution: "zod@npm:3.25.56"
checksum: 10c0/3800f01d4b1df932b91354eb1e648f69cc7e5561549e6d2bf83827d930a5f33bbf92926099445f6fc1ebb64ca9c6513ef9ae5e5409cfef6325f354bcf6fc9a24
languageName: node
linkType: hard
"zod@npm:^3.25 || ^4.0":
version: 4.1.13
resolution: "zod@npm:4.1.13"
checksum: 10c0/d7e74e82dba81a91ffc3239cd85bc034abe193a28f7087a94ab258a3e48e9a7ca4141920cac979a0d781495b48fc547777394149f26be04c3dc642f58bbc3941
languageName: node
linkType: hard
"zod@npm:^3.25.0 || ^4.0.0, zod@npm:^3.25.76 || ^4":
version: 4.1.12
resolution: "zod@npm:4.1.12"