mbxai 0.6.17__py3-none-any.whl → 0.6.19__py3-none-any.whl
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.
- mbxai/__init__.py +1 -1
- mbxai/mcp/client.py +60 -43
- mbxai/mcp/server.py +1 -1
- mbxai/tools/__init__.py +2 -1
- mbxai/tools/types.py +75 -64
- {mbxai-0.6.17.dist-info → mbxai-0.6.19.dist-info}/METADATA +2 -1
- {mbxai-0.6.17.dist-info → mbxai-0.6.19.dist-info}/RECORD +9 -9
- {mbxai-0.6.17.dist-info → mbxai-0.6.19.dist-info}/WHEEL +0 -0
- {mbxai-0.6.17.dist-info → mbxai-0.6.19.dist-info}/licenses/LICENSE +0 -0
mbxai/__init__.py
CHANGED
mbxai/mcp/client.py
CHANGED
@@ -4,9 +4,10 @@ from typing import Any, TypeVar, Callable
|
|
4
4
|
import httpx
|
5
5
|
import logging
|
6
6
|
import asyncio
|
7
|
+
import json
|
7
8
|
from pydantic import BaseModel, Field
|
8
9
|
|
9
|
-
from ..tools import ToolClient, Tool
|
10
|
+
from ..tools import ToolClient, Tool, convert_to_strict_schema
|
10
11
|
from ..openrouter import OpenRouterClient
|
11
12
|
|
12
13
|
logger = logging.getLogger(__name__)
|
@@ -21,18 +22,22 @@ class MCPTool(Tool):
|
|
21
22
|
strict: bool = Field(default=True, description="Whether the tool response is strictly validated")
|
22
23
|
input_schema: dict[str, Any] = Field(description="The input schema for the tool")
|
23
24
|
function: Callable[..., Any] | None = Field(default=None, description="The function that implements the tool")
|
25
|
+
_converted_schema: dict[str, Any] | None = None
|
24
26
|
|
25
27
|
def to_openai_function(self) -> dict[str, Any]:
|
26
28
|
"""Convert the tool to an OpenAI function definition."""
|
27
|
-
# Use
|
28
|
-
|
29
|
+
# Use cached schema if available
|
30
|
+
if self._converted_schema is None:
|
31
|
+
# Convert the schema to strict format
|
32
|
+
self._converted_schema = convert_to_strict_schema(self.input_schema, self.strict)
|
33
|
+
logger.debug(f"Converted schema for {self.name}: {json.dumps(self._converted_schema, indent=2)}")
|
29
34
|
|
30
35
|
return {
|
31
36
|
"type": "function",
|
32
37
|
"function": {
|
33
38
|
"name": self.name,
|
34
39
|
"description": self.description,
|
35
|
-
"parameters":
|
40
|
+
"parameters": self._converted_schema
|
36
41
|
}
|
37
42
|
}
|
38
43
|
|
@@ -41,19 +46,6 @@ class MCPTool(Tool):
|
|
41
46
|
if not mcp_schema:
|
42
47
|
return {"type": "object", "properties": {}, "required": []}
|
43
48
|
|
44
|
-
# If schema has a $ref, resolve it
|
45
|
-
if "$ref" in mcp_schema:
|
46
|
-
ref = mcp_schema["$ref"].split("/")[-1]
|
47
|
-
mcp_schema = mcp_schema.get("$defs", {}).get(ref, {})
|
48
|
-
|
49
|
-
# If schema has an input wrapper, unwrap it
|
50
|
-
if "properties" in mcp_schema and "input" in mcp_schema["properties"]:
|
51
|
-
input_schema = mcp_schema["properties"]["input"]
|
52
|
-
if "$ref" in input_schema:
|
53
|
-
ref = input_schema["$ref"].split("/")[-1]
|
54
|
-
input_schema = mcp_schema.get("$defs", {}).get(ref, {})
|
55
|
-
mcp_schema = input_schema
|
56
|
-
|
57
49
|
# Create a new schema object to ensure we have all required fields
|
58
50
|
strict_schema = {
|
59
51
|
"type": "object",
|
@@ -61,31 +53,56 @@ class MCPTool(Tool):
|
|
61
53
|
"required": []
|
62
54
|
}
|
63
55
|
|
64
|
-
#
|
65
|
-
if
|
66
|
-
|
67
|
-
# Create a new property object with required fields
|
68
|
-
new_prop = {
|
69
|
-
"type": prop.get("type", "string"),
|
70
|
-
"description": prop.get("description", f"The {prop_name} parameter")
|
71
|
-
}
|
72
|
-
|
73
|
-
# Copy over any additional fields that might be useful
|
74
|
-
for key, value in prop.items():
|
75
|
-
if key not in new_prop:
|
76
|
-
new_prop[key] = value
|
77
|
-
|
78
|
-
strict_schema["properties"][prop_name] = new_prop
|
79
|
-
|
80
|
-
# Copy over required fields
|
81
|
-
if "required" in mcp_schema:
|
82
|
-
strict_schema["required"] = mcp_schema["required"]
|
56
|
+
# Add additionalProperties: false for strict tools
|
57
|
+
if self.strict:
|
58
|
+
strict_schema["additionalProperties"] = False
|
83
59
|
|
84
|
-
#
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
60
|
+
# Handle input wrapper
|
61
|
+
if "properties" in mcp_schema and "input" in mcp_schema["properties"]:
|
62
|
+
input_schema = mcp_schema["properties"]["input"]
|
63
|
+
|
64
|
+
# If input has a $ref, resolve it
|
65
|
+
if "$ref" in input_schema:
|
66
|
+
ref = input_schema["$ref"].split("/")[-1]
|
67
|
+
input_schema = mcp_schema.get("$defs", {}).get(ref, {})
|
68
|
+
|
69
|
+
# Create the input property schema
|
70
|
+
input_prop_schema = {
|
71
|
+
"type": "object",
|
72
|
+
"properties": {},
|
73
|
+
"required": []
|
74
|
+
}
|
75
|
+
|
76
|
+
# Add additionalProperties: false for input schema
|
77
|
+
if self.strict:
|
78
|
+
input_prop_schema["additionalProperties"] = False
|
79
|
+
|
80
|
+
# Copy over input properties
|
81
|
+
if "properties" in input_schema:
|
82
|
+
for prop_name, prop in input_schema["properties"].items():
|
83
|
+
# Create a new property object with required fields
|
84
|
+
new_prop = {
|
85
|
+
"type": prop.get("type", "string"),
|
86
|
+
"description": prop.get("description", f"The {prop_name} parameter")
|
87
|
+
}
|
88
|
+
|
89
|
+
# Copy over any additional fields that might be useful
|
90
|
+
for key, value in prop.items():
|
91
|
+
if key not in new_prop:
|
92
|
+
new_prop[key] = value
|
93
|
+
|
94
|
+
input_prop_schema["properties"][prop_name] = new_prop
|
95
|
+
|
96
|
+
# Copy over required fields for input schema
|
97
|
+
if "required" in input_schema:
|
98
|
+
input_prop_schema["required"] = input_schema["required"]
|
99
|
+
|
100
|
+
# Add the input property to the main schema
|
101
|
+
strict_schema["properties"]["input"] = input_prop_schema
|
102
|
+
|
103
|
+
# Copy over required fields for main schema
|
104
|
+
if "required" in mcp_schema:
|
105
|
+
strict_schema["required"] = mcp_schema["required"]
|
89
106
|
|
90
107
|
return strict_schema
|
91
108
|
|
@@ -159,7 +176,7 @@ class MCPClient(ToolClient):
|
|
159
176
|
|
160
177
|
# Register each tool
|
161
178
|
for idx, tool_data in enumerate(tools_data):
|
162
|
-
logger.debug(f"Processing tool {idx}: {tool_data}")
|
179
|
+
logger.debug(f"Processing tool {idx}: {json.dumps(tool_data, indent=2)}")
|
163
180
|
|
164
181
|
# Ensure tool_data is a dictionary
|
165
182
|
if not isinstance(tool_data, dict):
|
@@ -181,4 +198,4 @@ class MCPClient(ToolClient):
|
|
181
198
|
logger.info(f"Successfully registered tool: {tool.name}")
|
182
199
|
except Exception as e:
|
183
200
|
logger.error(f"Failed to register tool: {str(e)}")
|
184
|
-
logger.error(f"Tool data that caused the error: {tool_data}")
|
201
|
+
logger.error(f"Tool data that caused the error: {json.dumps(tool_data, indent=2)}")
|
mbxai/mcp/server.py
CHANGED
mbxai/tools/__init__.py
CHANGED
@@ -3,10 +3,11 @@ Tools module for MBX AI.
|
|
3
3
|
"""
|
4
4
|
|
5
5
|
from .client import ToolClient
|
6
|
-
from .types import Tool, ToolCall
|
6
|
+
from .types import Tool, ToolCall, convert_to_strict_schema
|
7
7
|
|
8
8
|
__all__ = [
|
9
9
|
"ToolClient",
|
10
10
|
"Tool",
|
11
11
|
"ToolCall",
|
12
|
+
"convert_to_strict_schema",
|
12
13
|
]
|
mbxai/tools/types.py
CHANGED
@@ -5,65 +5,53 @@ Type definitions for the tools package.
|
|
5
5
|
from typing import Any, Callable
|
6
6
|
from pydantic import BaseModel, Field
|
7
7
|
|
8
|
-
|
9
|
-
"""
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
class Tool(BaseModel):
|
15
|
-
"""A tool that can be used by the model."""
|
16
|
-
name: str
|
17
|
-
description: str
|
18
|
-
function: Callable[..., Any]
|
19
|
-
schema: dict[str, Any]
|
20
|
-
|
21
|
-
def to_openai_function(self) -> dict[str, Any]:
|
22
|
-
"""Convert the tool to an OpenAI function definition."""
|
23
|
-
# Ensure schema is in strict format
|
24
|
-
strict_schema = self._ensure_strict_schema(self.schema)
|
8
|
+
def convert_to_strict_schema(schema: dict[str, Any], strict: bool = True) -> dict[str, Any]:
|
9
|
+
"""Convert a schema to strict format required by OpenAI.
|
10
|
+
|
11
|
+
Args:
|
12
|
+
schema: The input schema to validate and convert
|
13
|
+
strict: Whether to enforce strict validation with additionalProperties: false
|
25
14
|
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
"parameters": strict_schema
|
32
|
-
}
|
33
|
-
}
|
15
|
+
Returns:
|
16
|
+
A schema in strict format
|
17
|
+
"""
|
18
|
+
if not schema:
|
19
|
+
return {"type": "object", "properties": {}, "required": []}
|
34
20
|
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
Returns:
|
42
|
-
A schema in strict format
|
43
|
-
"""
|
44
|
-
# If schema has a $ref, resolve it
|
45
|
-
if "$ref" in schema:
|
46
|
-
ref = schema["$ref"].split("/")[-1]
|
47
|
-
schema = schema.get("$defs", {}).get(ref, {})
|
21
|
+
# Create a new schema object to ensure we have all required fields
|
22
|
+
strict_schema = {
|
23
|
+
"type": "object",
|
24
|
+
"properties": {},
|
25
|
+
"required": []
|
26
|
+
}
|
48
27
|
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
if "$ref" in input_schema:
|
53
|
-
ref = input_schema["$ref"].split("/")[-1]
|
54
|
-
input_schema = schema.get("$defs", {}).get(ref, {})
|
55
|
-
schema = input_schema
|
28
|
+
# Add additionalProperties: false for strict validation
|
29
|
+
if strict:
|
30
|
+
strict_schema["additionalProperties"] = False
|
56
31
|
|
57
|
-
|
58
|
-
|
32
|
+
# Handle input wrapper
|
33
|
+
if "properties" in schema and "input" in schema["properties"]:
|
34
|
+
input_schema = schema["properties"]["input"]
|
35
|
+
|
36
|
+
# If input has a $ref, resolve it
|
37
|
+
if "$ref" in input_schema:
|
38
|
+
ref = input_schema["$ref"].split("/")[-1]
|
39
|
+
input_schema = schema.get("$defs", {}).get(ref, {})
|
40
|
+
|
41
|
+
# Create the input property schema
|
42
|
+
input_prop_schema = {
|
59
43
|
"type": "object",
|
60
44
|
"properties": {},
|
61
45
|
"required": []
|
62
46
|
}
|
63
|
-
|
64
|
-
#
|
65
|
-
if
|
66
|
-
|
47
|
+
|
48
|
+
# Add additionalProperties: false for input schema
|
49
|
+
if strict:
|
50
|
+
input_prop_schema["additionalProperties"] = False
|
51
|
+
|
52
|
+
# Copy over input properties
|
53
|
+
if "properties" in input_schema:
|
54
|
+
for prop_name, prop in input_schema["properties"].items():
|
67
55
|
# Create a new property object with required fields
|
68
56
|
new_prop = {
|
69
57
|
"type": prop.get("type", "string"),
|
@@ -75,21 +63,44 @@ class Tool(BaseModel):
|
|
75
63
|
if key not in new_prop:
|
76
64
|
new_prop[key] = value
|
77
65
|
|
78
|
-
|
79
|
-
|
80
|
-
# Copy over required fields
|
66
|
+
input_prop_schema["properties"][prop_name] = new_prop
|
67
|
+
|
68
|
+
# Copy over required fields for input schema
|
69
|
+
if "required" in input_schema:
|
70
|
+
input_prop_schema["required"] = input_schema["required"]
|
71
|
+
|
72
|
+
# Add the input property to the main schema
|
73
|
+
strict_schema["properties"]["input"] = input_prop_schema
|
74
|
+
|
75
|
+
# Copy over required fields for main schema
|
81
76
|
if "required" in schema:
|
82
77
|
strict_schema["required"] = schema["required"]
|
83
78
|
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
79
|
+
return strict_schema
|
80
|
+
|
81
|
+
class ToolCall(BaseModel):
|
82
|
+
"""A tool call from the model."""
|
83
|
+
id: str
|
84
|
+
name: str
|
85
|
+
arguments: dict[str, Any]
|
89
86
|
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
87
|
+
class Tool(BaseModel):
|
88
|
+
"""A tool that can be used by the model."""
|
89
|
+
name: str
|
90
|
+
description: str
|
91
|
+
function: Callable[..., Any]
|
92
|
+
schema: dict[str, Any]
|
94
93
|
|
95
|
-
|
94
|
+
def to_openai_function(self) -> dict[str, Any]:
|
95
|
+
"""Convert the tool to an OpenAI function definition."""
|
96
|
+
# Ensure schema is in strict format
|
97
|
+
strict_schema = convert_to_strict_schema(self.schema)
|
98
|
+
|
99
|
+
return {
|
100
|
+
"type": "function",
|
101
|
+
"function": {
|
102
|
+
"name": self.name,
|
103
|
+
"description": self.description,
|
104
|
+
"parameters": strict_schema
|
105
|
+
}
|
106
|
+
}
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.4
|
2
2
|
Name: mbxai
|
3
|
-
Version: 0.6.
|
3
|
+
Version: 0.6.19
|
4
4
|
Summary: MBX AI SDK
|
5
5
|
Project-URL: Homepage, https://www.mibexx.de
|
6
6
|
Project-URL: Documentation, https://www.mibexx.de
|
@@ -22,6 +22,7 @@ Requires-Dist: pydantic>=2.9.1
|
|
22
22
|
Requires-Dist: python-multipart>=0.0.20
|
23
23
|
Requires-Dist: sse-starlette>=2.3.4
|
24
24
|
Requires-Dist: starlette>=0.46.2
|
25
|
+
Requires-Dist: typing-inspection<=0.4.0
|
25
26
|
Requires-Dist: uvicorn>=0.34.2
|
26
27
|
Provides-Extra: dev
|
27
28
|
Requires-Dist: black>=24.3.0; extra == 'dev'
|
@@ -1,18 +1,18 @@
|
|
1
|
-
mbxai/__init__.py,sha256=
|
1
|
+
mbxai/__init__.py,sha256=pqQAqM2XXc5pNrLfCSIAv2P-v29FjHrtlZhI1gq_DNY,48
|
2
2
|
mbxai/core.py,sha256=WMvmU9TTa7M_m-qWsUew4xH8Ul6xseCZ2iBCXJTW-Bs,196
|
3
3
|
mbxai/mcp/__init__.py,sha256=_ek9iYdYqW5saKetj4qDci11jxesQDiHPJRpHMKkxgU,175
|
4
|
-
mbxai/mcp/client.py,sha256=
|
4
|
+
mbxai/mcp/client.py,sha256=PpMNZSkmdPKf7artTDLDap-C9GbPjuGmKmr1byF0l6A,8309
|
5
5
|
mbxai/mcp/example.py,sha256=oaol7AvvZnX86JWNz64KvPjab5gg1VjVN3G8eFSzuaE,2350
|
6
|
-
mbxai/mcp/server.py,sha256=
|
6
|
+
mbxai/mcp/server.py,sha256=xFAVMLwi7130DCJ1EdwrkFR4rQTa0O6GzuCsBfQnmM0,3463
|
7
7
|
mbxai/openrouter/__init__.py,sha256=Ito9Qp_B6q-RLGAQcYyTJVWwR2YAZvNqE-HIYXxhtD8,298
|
8
8
|
mbxai/openrouter/client.py,sha256=nusxYObyLAMGQd6J9u2uLfkRXyE5kZmdIGdlU-76HPo,13529
|
9
9
|
mbxai/openrouter/config.py,sha256=Ia93s-auim9Sq71eunVDbn9ET5xX2zusXpV4JBdHAzs,3251
|
10
10
|
mbxai/openrouter/models.py,sha256=b3IjjtZAjeGOf2rLsdnCD1HacjTnS8jmv_ZXorc-KJQ,2604
|
11
|
-
mbxai/tools/__init__.py,sha256=
|
11
|
+
mbxai/tools/__init__.py,sha256=ogxrHvgJ7OR62Lmd5x9Eh5d2C0jqWyQis7Zy3yKpZ78,218
|
12
12
|
mbxai/tools/client.py,sha256=qOf8MiQ8_flPUNiMioOyjeEaV8rN1EBWb98T8qTC3D4,17582
|
13
13
|
mbxai/tools/example.py,sha256=1HgKK39zzUuwFbnp3f0ThyWVfA_8P28PZcTwaUw5K78,2232
|
14
|
-
mbxai/tools/types.py,sha256=
|
15
|
-
mbxai-0.6.
|
16
|
-
mbxai-0.6.
|
17
|
-
mbxai-0.6.
|
18
|
-
mbxai-0.6.
|
14
|
+
mbxai/tools/types.py,sha256=B3ujnPng997FCjQ5j25Da88ASkTDDj5gQpJRzVmdw9Q,3524
|
15
|
+
mbxai-0.6.19.dist-info/METADATA,sha256=k6P9GsIx9wYDS0DVOczSwgUvepDfXxx8BCSTOBnZDpQ,4148
|
16
|
+
mbxai-0.6.19.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
|
17
|
+
mbxai-0.6.19.dist-info/licenses/LICENSE,sha256=hEyhc4FxwYo3NQ40yNgZ7STqwVk-1_XcTXOnAPbGJAw,1069
|
18
|
+
mbxai-0.6.19.dist-info/RECORD,,
|
File without changes
|
File without changes
|