mbxai 0.6.11__tar.gz → 0.6.12__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (25) hide show
  1. {mbxai-0.6.11 → mbxai-0.6.12}/PKG-INFO +1 -1
  2. {mbxai-0.6.11 → mbxai-0.6.12}/pyproject.toml +1 -1
  3. {mbxai-0.6.11 → mbxai-0.6.12}/setup.py +1 -1
  4. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/tools/client.py +28 -91
  5. {mbxai-0.6.11 → mbxai-0.6.12}/uv.lock +7 -7
  6. {mbxai-0.6.11 → mbxai-0.6.12}/.gitignore +0 -0
  7. {mbxai-0.6.11 → mbxai-0.6.12}/LICENSE +0 -0
  8. {mbxai-0.6.11 → mbxai-0.6.12}/README.md +0 -0
  9. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/__init__.py +0 -0
  10. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/core.py +0 -0
  11. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/mcp/__init__.py +0 -0
  12. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/mcp/client.py +0 -0
  13. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/mcp/example.py +0 -0
  14. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/mcp/server.py +0 -0
  15. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/openrouter/__init__.py +0 -0
  16. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/openrouter/client.py +0 -0
  17. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/openrouter/config.py +0 -0
  18. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/openrouter/models.py +0 -0
  19. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/tools/__init__.py +0 -0
  20. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/tools/example.py +0 -0
  21. {mbxai-0.6.11 → mbxai-0.6.12}/src/mbxai/tools/types.py +0 -0
  22. {mbxai-0.6.11 → mbxai-0.6.12}/tests/test_core.py +0 -0
  23. {mbxai-0.6.11 → mbxai-0.6.12}/tests/test_mcp.py +0 -0
  24. {mbxai-0.6.11 → mbxai-0.6.12}/tests/test_openrouter.py +0 -0
  25. {mbxai-0.6.11 → mbxai-0.6.12}/tests/test_tools.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: mbxai
3
- Version: 0.6.11
3
+ Version: 0.6.12
4
4
  Summary: MBX AI SDK
5
5
  Project-URL: Homepage, https://www.mibexx.de
6
6
  Project-URL: Documentation, https://www.mibexx.de
@@ -4,7 +4,7 @@ build-backend = "hatchling.build"
4
4
 
5
5
  [project]
6
6
  name = "mbxai"
7
- version = "0.6.11"
7
+ version = "0.6.12"
8
8
  authors = [
9
9
  { name = "MBX AI" }
10
10
  ]
@@ -2,7 +2,7 @@ from setuptools import setup, find_packages
2
2
 
3
3
  setup(
4
4
  name="mbxai",
5
- version="0.6.11",
5
+ version="0.6.12",
6
6
  author="MBX AI",
7
7
  description="MBX AI SDK",
8
8
  long_description=open("README.md").read(),
@@ -327,99 +327,36 @@ class ToolClient:
327
327
  Returns:
328
328
  The parsed response from the model
329
329
  """
330
- tools = [tool.to_openai_function() for tool in self._tools.values()]
331
-
332
- if tools:
333
- logger.info(f"Available tools: {[tool['function']['name'] for tool in tools]}")
334
- kwargs["tools"] = tools
335
- kwargs["tool_choice"] = "auto"
336
-
337
- while True:
338
- # Log messages before sending to OpenRouter
339
- self._log_messages(messages)
340
-
341
- # Get the model's response
342
- response = self._client.chat_completion_parse(
343
- messages=messages,
344
- response_format=response_format,
345
- model=model,
346
- stream=stream,
347
- **kwargs,
348
- )
330
+ # First, use chat to handle any tool calls
331
+ chat_response = await self.chat(
332
+ messages=messages,
333
+ model=model,
334
+ stream=stream,
335
+ **kwargs,
336
+ )
349
337
 
350
- if stream:
351
- return response
338
+ if stream:
339
+ return chat_response
352
340
 
353
- message = response.choices[0].message
354
- # Add the assistant's message with tool calls
355
- assistant_message = {
341
+ # Get the final message after all tool calls are handled
342
+ final_message = chat_response.choices[0].message
343
+
344
+ # Create a new message list with just the final response
345
+ parse_messages = [
346
+ *messages, # Include original context
347
+ {
356
348
  "role": "assistant",
357
- "content": message.content or None, # Ensure content is None if empty
349
+ "content": final_message.content,
358
350
  }
359
- if message.tool_calls:
360
- assistant_message["tool_calls"] = [
361
- {
362
- "id": tool_call.id,
363
- "type": "function",
364
- "function": {
365
- "name": tool_call.function.name,
366
- "arguments": tool_call.function.arguments,
367
- },
368
- }
369
- for tool_call in message.tool_calls
370
- ]
371
- messages.append(assistant_message)
372
- logger.info(f"Message count: {len(messages)}, Added assistant message with tool calls: {[tc.function.name for tc in message.tool_calls] if message.tool_calls else None}")
373
-
374
- # If there are no tool calls, we're done
375
- if not message.tool_calls:
376
- return response
351
+ ]
377
352
 
378
- # Process all tool calls
379
- tool_responses = []
380
- for tool_call in message.tool_calls:
381
- tool = self._tools.get(tool_call.function.name)
382
- if not tool:
383
- raise ValueError(f"Unknown tool: {tool_call.function.name}")
384
-
385
- # Parse arguments if they're a string
386
- arguments = tool_call.function.arguments
387
- if isinstance(arguments, str):
388
- try:
389
- arguments = json.loads(arguments)
390
- except json.JSONDecodeError as e:
391
- logger.error(f"Failed to parse tool arguments: {e}")
392
- raise ValueError(f"Invalid tool arguments format: {arguments}")
393
-
394
- # Call the tool
395
- logger.info(f"Calling tool: {tool.name} with args: {self._truncate_dict(arguments)}")
396
- if inspect.iscoroutinefunction(tool.function):
397
- result = await tool.function(**arguments)
398
- else:
399
- result = tool.function(**arguments)
400
-
401
- # Convert result to JSON string if it's not already
402
- if not isinstance(result, str):
403
- result = json.dumps(result)
404
-
405
- # Create the tool response
406
- tool_response = {
407
- "role": "tool",
408
- "tool_call_id": tool_call.id,
409
- "content": result,
410
- }
411
- tool_responses.append(tool_response)
412
- logger.info(f"Created tool response for call ID {tool_call.id}")
413
-
414
- # Add all tool responses to the messages
415
- messages.extend(tool_responses)
416
- logger.info(f"Message count: {len(messages)}, Added {len(tool_responses)} tool responses to messages")
417
-
418
- # Validate the message sequence
419
- self._validate_message_sequence(messages, validate_responses=True)
420
-
421
- # Log the messages we're about to send
422
- self._log_messages(messages, validate_responses=False)
423
-
424
- # Continue the loop to get the next response
425
- continue
353
+ # Make a final parse request with the structured output format
354
+ parse_kwargs = kwargs.copy()
355
+ parse_kwargs["response_format"] = response_format
356
+
357
+ return self._client.chat_completion_parse(
358
+ messages=parse_messages,
359
+ model=model,
360
+ stream=stream,
361
+ **parse_kwargs,
362
+ )
@@ -292,11 +292,11 @@ wheels = [
292
292
 
293
293
  [[package]]
294
294
  name = "httpx-sse"
295
- version = "0.6.11"
295
+ version = "0.6.12"
296
296
  source = { registry = "https://pypi.org/simple" }
297
- sdist = { url = "https://files.pythonhosted.org/packages/4c/60/8f4281fa9bbf3c8034fd54c0e7412e66edbab6bc74c4996bd616f8d0406e/httpx-sse-0.6.11.tar.gz", hash = "sha256:1e81a3a3070ce322add1d3529ed42eb5f70817f45ed6ec915ab753f961139721", size = 12624 }
297
+ sdist = { url = "https://files.pythonhosted.org/packages/4c/60/8f4281fa9bbf3c8034fd54c0e7412e66edbab6bc74c4996bd616f8d0406e/httpx-sse-0.6.12.tar.gz", hash = "sha256:1e81a3a3070ce322add1d3529ed42eb5f70817f45ed6ec915ab753f961139721", size = 12624 }
298
298
  wheels = [
299
- { url = "https://files.pythonhosted.org/packages/e1/9b/a181f281f65d776426002f330c31849b86b31fc9d848db62e16f03ff739f/httpx_sse-0.6.11-py3-none-any.whl", hash = "sha256:f329af6eae57eaa2bdfd962b42524764af68075ea87370a2de920af5341e318f", size = 7819 },
299
+ { url = "https://files.pythonhosted.org/packages/e1/9b/a181f281f65d776426002f330c31849b86b31fc9d848db62e16f03ff739f/httpx_sse-0.6.12-py3-none-any.whl", hash = "sha256:f329af6eae57eaa2bdfd962b42524764af68075ea87370a2de920af5341e318f", size = 7819 },
300
300
  ]
301
301
 
302
302
  [[package]]
@@ -446,7 +446,7 @@ wheels = [
446
446
 
447
447
  [[package]]
448
448
  name = "mbxai"
449
- version = "0.6.11"
449
+ version = "0.6.12"
450
450
  source = { editable = "." }
451
451
  dependencies = [
452
452
  { name = "fastapi" },
@@ -980,14 +980,14 @@ wheels = [
980
980
 
981
981
  [[package]]
982
982
  name = "typing-inspection"
983
- version = "0.6.11"
983
+ version = "0.6.12"
984
984
  source = { registry = "https://pypi.org/simple" }
985
985
  dependencies = [
986
986
  { name = "typing-extensions" },
987
987
  ]
988
- sdist = { url = "https://files.pythonhosted.org/packages/82/5c/e6082df02e215b846b4b8c0b887a64d7d08ffaba30605502639d44c06b82/typing_inspection-0.6.11.tar.gz", hash = "sha256:9765c87de36671694a67904bf2c96e395be9c6439bb6c87b5142569dcdd65122", size = 76222 }
988
+ sdist = { url = "https://files.pythonhosted.org/packages/82/5c/e6082df02e215b846b4b8c0b887a64d7d08ffaba30605502639d44c06b82/typing_inspection-0.6.12.tar.gz", hash = "sha256:9765c87de36671694a67904bf2c96e395be9c6439bb6c87b5142569dcdd65122", size = 76222 }
989
989
  wheels = [
990
- { url = "https://files.pythonhosted.org/packages/31/08/aa4fdfb71f7de5176385bd9e90852eaf6b5d622735020ad600f2bab54385/typing_inspection-0.6.11-py3-none-any.whl", hash = "sha256:50e72559fcd2a6367a19f7a7e610e6afcb9fac940c650290eed893d61386832f", size = 14125 },
990
+ { url = "https://files.pythonhosted.org/packages/31/08/aa4fdfb71f7de5176385bd9e90852eaf6b5d622735020ad600f2bab54385/typing_inspection-0.6.12-py3-none-any.whl", hash = "sha256:50e72559fcd2a6367a19f7a7e610e6afcb9fac940c650290eed893d61386832f", size = 14125 },
991
991
  ]
992
992
 
993
993
  [[package]]
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes