mirascope 1.25.7__py3-none-any.whl → 2.0.0a0__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.
- mirascope/__init__.py +3 -59
- mirascope/graphs/__init__.py +22 -0
- mirascope/{experimental/graphs → graphs}/finite_state_machine.py +70 -159
- mirascope/llm/__init__.py +206 -16
- mirascope/llm/agents/__init__.py +15 -0
- mirascope/llm/agents/agent.py +97 -0
- mirascope/llm/agents/agent_template.py +45 -0
- mirascope/llm/agents/decorator.py +176 -0
- mirascope/llm/calls/__init__.py +16 -0
- mirascope/llm/calls/base_call.py +33 -0
- mirascope/llm/calls/calls.py +315 -0
- mirascope/llm/calls/decorator.py +255 -0
- mirascope/llm/clients/__init__.py +34 -0
- mirascope/llm/clients/anthropic/__init__.py +11 -0
- mirascope/llm/clients/anthropic/_utils/__init__.py +13 -0
- mirascope/llm/clients/anthropic/_utils/decode.py +244 -0
- mirascope/llm/clients/anthropic/_utils/encode.py +243 -0
- mirascope/llm/clients/anthropic/clients.py +819 -0
- mirascope/llm/clients/anthropic/model_ids.py +8 -0
- mirascope/llm/clients/base/__init__.py +15 -0
- mirascope/llm/clients/base/_utils.py +192 -0
- mirascope/llm/clients/base/client.py +1256 -0
- mirascope/llm/clients/base/kwargs.py +12 -0
- mirascope/llm/clients/base/params.py +93 -0
- mirascope/llm/clients/google/__init__.py +6 -0
- mirascope/llm/clients/google/_utils/__init__.py +13 -0
- mirascope/llm/clients/google/_utils/decode.py +231 -0
- mirascope/llm/clients/google/_utils/encode.py +279 -0
- mirascope/llm/clients/google/clients.py +853 -0
- mirascope/llm/clients/google/message.py +7 -0
- mirascope/llm/clients/google/model_ids.py +15 -0
- mirascope/llm/clients/openai/__init__.py +25 -0
- mirascope/llm/clients/openai/completions/__init__.py +9 -0
- mirascope/llm/clients/openai/completions/_utils/__init__.py +13 -0
- mirascope/llm/clients/openai/completions/_utils/decode.py +187 -0
- mirascope/llm/clients/openai/completions/_utils/encode.py +358 -0
- mirascope/llm/clients/openai/completions/_utils/model_features.py +81 -0
- mirascope/llm/clients/openai/completions/clients.py +833 -0
- mirascope/llm/clients/openai/completions/model_ids.py +8 -0
- mirascope/llm/clients/openai/responses/__init__.py +9 -0
- mirascope/llm/clients/openai/responses/_utils/__init__.py +13 -0
- mirascope/llm/clients/openai/responses/_utils/decode.py +194 -0
- mirascope/llm/clients/openai/responses/_utils/encode.py +333 -0
- mirascope/llm/clients/openai/responses/_utils/model_features.py +87 -0
- mirascope/llm/clients/openai/responses/clients.py +832 -0
- mirascope/llm/clients/openai/responses/model_ids.py +8 -0
- mirascope/llm/clients/openai/shared/__init__.py +7 -0
- mirascope/llm/clients/openai/shared/_utils.py +55 -0
- mirascope/llm/clients/providers.py +175 -0
- mirascope/llm/content/__init__.py +70 -0
- mirascope/llm/content/audio.py +173 -0
- mirascope/llm/content/document.py +94 -0
- mirascope/llm/content/image.py +206 -0
- mirascope/llm/content/text.py +47 -0
- mirascope/llm/content/thought.py +58 -0
- mirascope/llm/content/tool_call.py +63 -0
- mirascope/llm/content/tool_output.py +26 -0
- mirascope/llm/context/__init__.py +6 -0
- mirascope/llm/context/_utils.py +28 -0
- mirascope/llm/context/context.py +24 -0
- mirascope/llm/exceptions.py +105 -0
- mirascope/llm/formatting/__init__.py +22 -0
- mirascope/llm/formatting/_utils.py +74 -0
- mirascope/llm/formatting/format.py +104 -0
- mirascope/llm/formatting/from_call_args.py +30 -0
- mirascope/llm/formatting/partial.py +58 -0
- mirascope/llm/formatting/types.py +109 -0
- mirascope/llm/mcp/__init__.py +5 -0
- mirascope/llm/mcp/client.py +118 -0
- mirascope/llm/messages/__init__.py +32 -0
- mirascope/llm/messages/message.py +182 -0
- mirascope/llm/models/__init__.py +16 -0
- mirascope/llm/models/models.py +1243 -0
- mirascope/llm/prompts/__init__.py +33 -0
- mirascope/llm/prompts/_utils.py +60 -0
- mirascope/llm/prompts/decorator.py +286 -0
- mirascope/llm/prompts/protocols.py +99 -0
- mirascope/llm/responses/__init__.py +57 -0
- mirascope/llm/responses/_utils.py +56 -0
- mirascope/llm/responses/base_response.py +91 -0
- mirascope/llm/responses/base_stream_response.py +697 -0
- mirascope/llm/responses/finish_reason.py +27 -0
- mirascope/llm/responses/response.py +345 -0
- mirascope/llm/responses/root_response.py +177 -0
- mirascope/llm/responses/stream_response.py +572 -0
- mirascope/llm/responses/streams.py +363 -0
- mirascope/llm/tools/__init__.py +40 -0
- mirascope/llm/tools/_utils.py +25 -0
- mirascope/llm/tools/decorator.py +175 -0
- mirascope/llm/tools/protocols.py +96 -0
- mirascope/llm/tools/tool_schema.py +246 -0
- mirascope/llm/tools/toolkit.py +152 -0
- mirascope/llm/tools/tools.py +169 -0
- mirascope/llm/types/__init__.py +22 -0
- mirascope/llm/types/dataclass.py +9 -0
- mirascope/llm/types/jsonable.py +44 -0
- mirascope/llm/types/type_vars.py +19 -0
- mirascope-2.0.0a0.dist-info/METADATA +117 -0
- mirascope-2.0.0a0.dist-info/RECORD +101 -0
- mirascope/beta/__init__.py +0 -3
- mirascope/beta/openai/__init__.py +0 -17
- mirascope/beta/openai/realtime/__init__.py +0 -13
- mirascope/beta/openai/realtime/_utils/__init__.py +0 -3
- mirascope/beta/openai/realtime/_utils/_audio.py +0 -74
- mirascope/beta/openai/realtime/_utils/_protocols.py +0 -50
- mirascope/beta/openai/realtime/realtime.py +0 -500
- mirascope/beta/openai/realtime/recording.py +0 -98
- mirascope/beta/openai/realtime/tool.py +0 -113
- mirascope/beta/rag/__init__.py +0 -24
- mirascope/beta/rag/base/__init__.py +0 -22
- mirascope/beta/rag/base/chunkers/__init__.py +0 -2
- mirascope/beta/rag/base/chunkers/base_chunker.py +0 -37
- mirascope/beta/rag/base/chunkers/text_chunker.py +0 -33
- mirascope/beta/rag/base/config.py +0 -8
- mirascope/beta/rag/base/document.py +0 -11
- mirascope/beta/rag/base/embedders.py +0 -35
- mirascope/beta/rag/base/embedding_params.py +0 -18
- mirascope/beta/rag/base/embedding_response.py +0 -30
- mirascope/beta/rag/base/query_results.py +0 -7
- mirascope/beta/rag/base/vectorstore_params.py +0 -18
- mirascope/beta/rag/base/vectorstores.py +0 -37
- mirascope/beta/rag/chroma/__init__.py +0 -11
- mirascope/beta/rag/chroma/types.py +0 -62
- mirascope/beta/rag/chroma/vectorstores.py +0 -121
- mirascope/beta/rag/cohere/__init__.py +0 -11
- mirascope/beta/rag/cohere/embedders.py +0 -87
- mirascope/beta/rag/cohere/embedding_params.py +0 -29
- mirascope/beta/rag/cohere/embedding_response.py +0 -29
- mirascope/beta/rag/cohere/py.typed +0 -0
- mirascope/beta/rag/openai/__init__.py +0 -11
- mirascope/beta/rag/openai/embedders.py +0 -144
- mirascope/beta/rag/openai/embedding_params.py +0 -18
- mirascope/beta/rag/openai/embedding_response.py +0 -14
- mirascope/beta/rag/openai/py.typed +0 -0
- mirascope/beta/rag/pinecone/__init__.py +0 -19
- mirascope/beta/rag/pinecone/types.py +0 -143
- mirascope/beta/rag/pinecone/vectorstores.py +0 -148
- mirascope/beta/rag/weaviate/__init__.py +0 -6
- mirascope/beta/rag/weaviate/types.py +0 -92
- mirascope/beta/rag/weaviate/vectorstores.py +0 -103
- mirascope/core/__init__.py +0 -109
- mirascope/core/anthropic/__init__.py +0 -31
- mirascope/core/anthropic/_call.py +0 -67
- mirascope/core/anthropic/_call_kwargs.py +0 -13
- mirascope/core/anthropic/_thinking.py +0 -70
- mirascope/core/anthropic/_utils/__init__.py +0 -16
- mirascope/core/anthropic/_utils/_convert_common_call_params.py +0 -25
- mirascope/core/anthropic/_utils/_convert_finish_reason_to_common_finish_reasons.py +0 -21
- mirascope/core/anthropic/_utils/_convert_message_params.py +0 -102
- mirascope/core/anthropic/_utils/_get_json_output.py +0 -31
- mirascope/core/anthropic/_utils/_handle_stream.py +0 -113
- mirascope/core/anthropic/_utils/_message_param_converter.py +0 -154
- mirascope/core/anthropic/_utils/_setup_call.py +0 -146
- mirascope/core/anthropic/call_params.py +0 -44
- mirascope/core/anthropic/call_response.py +0 -226
- mirascope/core/anthropic/call_response_chunk.py +0 -152
- mirascope/core/anthropic/dynamic_config.py +0 -40
- mirascope/core/anthropic/py.typed +0 -0
- mirascope/core/anthropic/stream.py +0 -204
- mirascope/core/anthropic/tool.py +0 -101
- mirascope/core/azure/__init__.py +0 -31
- mirascope/core/azure/_call.py +0 -67
- mirascope/core/azure/_call_kwargs.py +0 -13
- mirascope/core/azure/_utils/__init__.py +0 -14
- mirascope/core/azure/_utils/_convert_common_call_params.py +0 -26
- mirascope/core/azure/_utils/_convert_finish_reason_to_common_finish_reasons.py +0 -21
- mirascope/core/azure/_utils/_convert_message_params.py +0 -121
- mirascope/core/azure/_utils/_get_credential.py +0 -33
- mirascope/core/azure/_utils/_get_json_output.py +0 -27
- mirascope/core/azure/_utils/_handle_stream.py +0 -130
- mirascope/core/azure/_utils/_message_param_converter.py +0 -117
- mirascope/core/azure/_utils/_setup_call.py +0 -183
- mirascope/core/azure/call_params.py +0 -59
- mirascope/core/azure/call_response.py +0 -215
- mirascope/core/azure/call_response_chunk.py +0 -105
- mirascope/core/azure/dynamic_config.py +0 -30
- mirascope/core/azure/py.typed +0 -0
- mirascope/core/azure/stream.py +0 -147
- mirascope/core/azure/tool.py +0 -93
- mirascope/core/base/__init__.py +0 -86
- mirascope/core/base/_call_factory.py +0 -256
- mirascope/core/base/_create.py +0 -253
- mirascope/core/base/_extract.py +0 -175
- mirascope/core/base/_extract_with_tools.py +0 -189
- mirascope/core/base/_partial.py +0 -95
- mirascope/core/base/_utils/__init__.py +0 -92
- mirascope/core/base/_utils/_base_message_param_converter.py +0 -22
- mirascope/core/base/_utils/_base_type.py +0 -26
- mirascope/core/base/_utils/_convert_base_model_to_base_tool.py +0 -48
- mirascope/core/base/_utils/_convert_base_type_to_base_tool.py +0 -24
- mirascope/core/base/_utils/_convert_function_to_base_tool.py +0 -139
- mirascope/core/base/_utils/_convert_messages_to_message_params.py +0 -178
- mirascope/core/base/_utils/_convert_provider_finish_reason_to_finish_reason.py +0 -20
- mirascope/core/base/_utils/_default_tool_docstring.py +0 -6
- mirascope/core/base/_utils/_extract_tool_return.py +0 -42
- mirascope/core/base/_utils/_fn_is_async.py +0 -24
- mirascope/core/base/_utils/_format_template.py +0 -32
- mirascope/core/base/_utils/_get_audio_type.py +0 -18
- mirascope/core/base/_utils/_get_common_usage.py +0 -20
- mirascope/core/base/_utils/_get_create_fn_or_async_create_fn.py +0 -137
- mirascope/core/base/_utils/_get_document_type.py +0 -7
- mirascope/core/base/_utils/_get_dynamic_configuration.py +0 -69
- mirascope/core/base/_utils/_get_fields_from_call_args.py +0 -34
- mirascope/core/base/_utils/_get_fn_args.py +0 -23
- mirascope/core/base/_utils/_get_image_dimensions.py +0 -39
- mirascope/core/base/_utils/_get_image_type.py +0 -26
- mirascope/core/base/_utils/_get_metadata.py +0 -17
- mirascope/core/base/_utils/_get_possible_user_message_param.py +0 -21
- mirascope/core/base/_utils/_get_prompt_template.py +0 -28
- mirascope/core/base/_utils/_get_template_values.py +0 -51
- mirascope/core/base/_utils/_get_template_variables.py +0 -38
- mirascope/core/base/_utils/_get_unsupported_tool_config_keys.py +0 -10
- mirascope/core/base/_utils/_is_prompt_template.py +0 -24
- mirascope/core/base/_utils/_json_mode_content.py +0 -17
- mirascope/core/base/_utils/_messages_decorator.py +0 -121
- mirascope/core/base/_utils/_parse_content_template.py +0 -323
- mirascope/core/base/_utils/_parse_prompt_messages.py +0 -63
- mirascope/core/base/_utils/_pil_image_to_bytes.py +0 -13
- mirascope/core/base/_utils/_protocols.py +0 -901
- mirascope/core/base/_utils/_setup_call.py +0 -79
- mirascope/core/base/_utils/_setup_extract_tool.py +0 -30
- mirascope/core/base/call_kwargs.py +0 -13
- mirascope/core/base/call_params.py +0 -36
- mirascope/core/base/call_response.py +0 -338
- mirascope/core/base/call_response_chunk.py +0 -130
- mirascope/core/base/dynamic_config.py +0 -82
- mirascope/core/base/from_call_args.py +0 -30
- mirascope/core/base/merge_decorators.py +0 -59
- mirascope/core/base/message_param.py +0 -175
- mirascope/core/base/messages.py +0 -116
- mirascope/core/base/metadata.py +0 -13
- mirascope/core/base/prompt.py +0 -497
- mirascope/core/base/response_model_config_dict.py +0 -9
- mirascope/core/base/stream.py +0 -479
- mirascope/core/base/stream_config.py +0 -11
- mirascope/core/base/structured_stream.py +0 -296
- mirascope/core/base/tool.py +0 -214
- mirascope/core/base/toolkit.py +0 -176
- mirascope/core/base/types.py +0 -344
- mirascope/core/bedrock/__init__.py +0 -34
- mirascope/core/bedrock/_call.py +0 -68
- mirascope/core/bedrock/_call_kwargs.py +0 -12
- mirascope/core/bedrock/_types.py +0 -104
- mirascope/core/bedrock/_utils/__init__.py +0 -14
- mirascope/core/bedrock/_utils/_convert_common_call_params.py +0 -39
- mirascope/core/bedrock/_utils/_convert_finish_reason_to_common_finish_reasons.py +0 -23
- mirascope/core/bedrock/_utils/_convert_message_params.py +0 -111
- mirascope/core/bedrock/_utils/_get_json_output.py +0 -30
- mirascope/core/bedrock/_utils/_handle_stream.py +0 -104
- mirascope/core/bedrock/_utils/_message_param_converter.py +0 -172
- mirascope/core/bedrock/_utils/_setup_call.py +0 -258
- mirascope/core/bedrock/call_params.py +0 -38
- mirascope/core/bedrock/call_response.py +0 -248
- mirascope/core/bedrock/call_response_chunk.py +0 -111
- mirascope/core/bedrock/dynamic_config.py +0 -37
- mirascope/core/bedrock/py.typed +0 -0
- mirascope/core/bedrock/stream.py +0 -154
- mirascope/core/bedrock/tool.py +0 -100
- mirascope/core/cohere/__init__.py +0 -30
- mirascope/core/cohere/_call.py +0 -67
- mirascope/core/cohere/_call_kwargs.py +0 -11
- mirascope/core/cohere/_types.py +0 -20
- mirascope/core/cohere/_utils/__init__.py +0 -14
- mirascope/core/cohere/_utils/_convert_common_call_params.py +0 -26
- mirascope/core/cohere/_utils/_convert_finish_reason_to_common_finish_reasons.py +0 -24
- mirascope/core/cohere/_utils/_convert_message_params.py +0 -32
- mirascope/core/cohere/_utils/_get_json_output.py +0 -30
- mirascope/core/cohere/_utils/_handle_stream.py +0 -35
- mirascope/core/cohere/_utils/_message_param_converter.py +0 -54
- mirascope/core/cohere/_utils/_setup_call.py +0 -150
- mirascope/core/cohere/call_params.py +0 -62
- mirascope/core/cohere/call_response.py +0 -205
- mirascope/core/cohere/call_response_chunk.py +0 -125
- mirascope/core/cohere/dynamic_config.py +0 -32
- mirascope/core/cohere/py.typed +0 -0
- mirascope/core/cohere/stream.py +0 -113
- mirascope/core/cohere/tool.py +0 -93
- mirascope/core/costs/__init__.py +0 -5
- mirascope/core/costs/_anthropic_calculate_cost.py +0 -219
- mirascope/core/costs/_azure_calculate_cost.py +0 -11
- mirascope/core/costs/_bedrock_calculate_cost.py +0 -15
- mirascope/core/costs/_cohere_calculate_cost.py +0 -44
- mirascope/core/costs/_gemini_calculate_cost.py +0 -67
- mirascope/core/costs/_google_calculate_cost.py +0 -427
- mirascope/core/costs/_groq_calculate_cost.py +0 -156
- mirascope/core/costs/_litellm_calculate_cost.py +0 -11
- mirascope/core/costs/_mistral_calculate_cost.py +0 -64
- mirascope/core/costs/_openai_calculate_cost.py +0 -416
- mirascope/core/costs/_vertex_calculate_cost.py +0 -67
- mirascope/core/costs/_xai_calculate_cost.py +0 -104
- mirascope/core/costs/calculate_cost.py +0 -86
- mirascope/core/gemini/__init__.py +0 -40
- mirascope/core/gemini/_call.py +0 -67
- mirascope/core/gemini/_call_kwargs.py +0 -12
- mirascope/core/gemini/_utils/__init__.py +0 -14
- mirascope/core/gemini/_utils/_convert_common_call_params.py +0 -39
- mirascope/core/gemini/_utils/_convert_finish_reason_to_common_finish_reasons.py +0 -23
- mirascope/core/gemini/_utils/_convert_message_params.py +0 -156
- mirascope/core/gemini/_utils/_get_json_output.py +0 -35
- mirascope/core/gemini/_utils/_handle_stream.py +0 -33
- mirascope/core/gemini/_utils/_message_param_converter.py +0 -209
- mirascope/core/gemini/_utils/_setup_call.py +0 -149
- mirascope/core/gemini/call_params.py +0 -52
- mirascope/core/gemini/call_response.py +0 -216
- mirascope/core/gemini/call_response_chunk.py +0 -100
- mirascope/core/gemini/dynamic_config.py +0 -26
- mirascope/core/gemini/stream.py +0 -120
- mirascope/core/gemini/tool.py +0 -104
- mirascope/core/google/__init__.py +0 -29
- mirascope/core/google/_call.py +0 -67
- mirascope/core/google/_call_kwargs.py +0 -13
- mirascope/core/google/_utils/__init__.py +0 -14
- mirascope/core/google/_utils/_convert_common_call_params.py +0 -38
- mirascope/core/google/_utils/_convert_finish_reason_to_common_finish_reasons.py +0 -27
- mirascope/core/google/_utils/_convert_message_params.py +0 -297
- mirascope/core/google/_utils/_get_json_output.py +0 -37
- mirascope/core/google/_utils/_handle_stream.py +0 -58
- mirascope/core/google/_utils/_message_param_converter.py +0 -200
- mirascope/core/google/_utils/_setup_call.py +0 -201
- mirascope/core/google/_utils/_validate_media_type.py +0 -58
- mirascope/core/google/call_params.py +0 -22
- mirascope/core/google/call_response.py +0 -255
- mirascope/core/google/call_response_chunk.py +0 -135
- mirascope/core/google/dynamic_config.py +0 -26
- mirascope/core/google/stream.py +0 -199
- mirascope/core/google/tool.py +0 -146
- mirascope/core/groq/__init__.py +0 -30
- mirascope/core/groq/_call.py +0 -67
- mirascope/core/groq/_call_kwargs.py +0 -13
- mirascope/core/groq/_utils/__init__.py +0 -14
- mirascope/core/groq/_utils/_convert_common_call_params.py +0 -26
- mirascope/core/groq/_utils/_convert_message_params.py +0 -112
- mirascope/core/groq/_utils/_get_json_output.py +0 -27
- mirascope/core/groq/_utils/_handle_stream.py +0 -123
- mirascope/core/groq/_utils/_message_param_converter.py +0 -89
- mirascope/core/groq/_utils/_setup_call.py +0 -132
- mirascope/core/groq/call_params.py +0 -52
- mirascope/core/groq/call_response.py +0 -213
- mirascope/core/groq/call_response_chunk.py +0 -104
- mirascope/core/groq/dynamic_config.py +0 -29
- mirascope/core/groq/py.typed +0 -0
- mirascope/core/groq/stream.py +0 -135
- mirascope/core/groq/tool.py +0 -80
- mirascope/core/litellm/__init__.py +0 -28
- mirascope/core/litellm/_call.py +0 -67
- mirascope/core/litellm/_utils/__init__.py +0 -5
- mirascope/core/litellm/_utils/_setup_call.py +0 -109
- mirascope/core/litellm/call_params.py +0 -10
- mirascope/core/litellm/call_response.py +0 -24
- mirascope/core/litellm/call_response_chunk.py +0 -14
- mirascope/core/litellm/dynamic_config.py +0 -8
- mirascope/core/litellm/py.typed +0 -0
- mirascope/core/litellm/stream.py +0 -86
- mirascope/core/litellm/tool.py +0 -13
- mirascope/core/mistral/__init__.py +0 -36
- mirascope/core/mistral/_call.py +0 -65
- mirascope/core/mistral/_call_kwargs.py +0 -19
- mirascope/core/mistral/_utils/__init__.py +0 -14
- mirascope/core/mistral/_utils/_convert_common_call_params.py +0 -24
- mirascope/core/mistral/_utils/_convert_finish_reason_to_common_finish_reasons.py +0 -22
- mirascope/core/mistral/_utils/_convert_message_params.py +0 -122
- mirascope/core/mistral/_utils/_get_json_output.py +0 -34
- mirascope/core/mistral/_utils/_handle_stream.py +0 -139
- mirascope/core/mistral/_utils/_message_param_converter.py +0 -176
- mirascope/core/mistral/_utils/_setup_call.py +0 -164
- mirascope/core/mistral/call_params.py +0 -36
- mirascope/core/mistral/call_response.py +0 -205
- mirascope/core/mistral/call_response_chunk.py +0 -105
- mirascope/core/mistral/dynamic_config.py +0 -33
- mirascope/core/mistral/py.typed +0 -0
- mirascope/core/mistral/stream.py +0 -120
- mirascope/core/mistral/tool.py +0 -81
- mirascope/core/openai/__init__.py +0 -31
- mirascope/core/openai/_call.py +0 -67
- mirascope/core/openai/_call_kwargs.py +0 -13
- mirascope/core/openai/_utils/__init__.py +0 -14
- mirascope/core/openai/_utils/_convert_common_call_params.py +0 -26
- mirascope/core/openai/_utils/_convert_message_params.py +0 -148
- mirascope/core/openai/_utils/_get_json_output.py +0 -31
- mirascope/core/openai/_utils/_handle_stream.py +0 -138
- mirascope/core/openai/_utils/_message_param_converter.py +0 -105
- mirascope/core/openai/_utils/_setup_call.py +0 -155
- mirascope/core/openai/call_params.py +0 -92
- mirascope/core/openai/call_response.py +0 -273
- mirascope/core/openai/call_response_chunk.py +0 -139
- mirascope/core/openai/dynamic_config.py +0 -34
- mirascope/core/openai/py.typed +0 -0
- mirascope/core/openai/stream.py +0 -185
- mirascope/core/openai/tool.py +0 -101
- mirascope/core/py.typed +0 -0
- mirascope/core/vertex/__init__.py +0 -45
- mirascope/core/vertex/_call.py +0 -62
- mirascope/core/vertex/_call_kwargs.py +0 -12
- mirascope/core/vertex/_utils/__init__.py +0 -14
- mirascope/core/vertex/_utils/_convert_common_call_params.py +0 -37
- mirascope/core/vertex/_utils/_convert_finish_reason_to_common_finish_reasons.py +0 -23
- mirascope/core/vertex/_utils/_convert_message_params.py +0 -171
- mirascope/core/vertex/_utils/_get_json_output.py +0 -36
- mirascope/core/vertex/_utils/_handle_stream.py +0 -33
- mirascope/core/vertex/_utils/_message_param_converter.py +0 -133
- mirascope/core/vertex/_utils/_setup_call.py +0 -160
- mirascope/core/vertex/call_params.py +0 -24
- mirascope/core/vertex/call_response.py +0 -206
- mirascope/core/vertex/call_response_chunk.py +0 -99
- mirascope/core/vertex/dynamic_config.py +0 -28
- mirascope/core/vertex/stream.py +0 -119
- mirascope/core/vertex/tool.py +0 -101
- mirascope/core/xai/__init__.py +0 -28
- mirascope/core/xai/_call.py +0 -67
- mirascope/core/xai/_utils/__init__.py +0 -5
- mirascope/core/xai/_utils/_setup_call.py +0 -113
- mirascope/core/xai/call_params.py +0 -10
- mirascope/core/xai/call_response.py +0 -16
- mirascope/core/xai/call_response_chunk.py +0 -14
- mirascope/core/xai/dynamic_config.py +0 -8
- mirascope/core/xai/py.typed +0 -0
- mirascope/core/xai/stream.py +0 -57
- mirascope/core/xai/tool.py +0 -13
- mirascope/experimental/graphs/__init__.py +0 -5
- mirascope/integrations/__init__.py +0 -16
- mirascope/integrations/_middleware_factory.py +0 -403
- mirascope/integrations/langfuse/__init__.py +0 -3
- mirascope/integrations/langfuse/_utils.py +0 -114
- mirascope/integrations/langfuse/_with_langfuse.py +0 -70
- mirascope/integrations/logfire/__init__.py +0 -3
- mirascope/integrations/logfire/_utils.py +0 -225
- mirascope/integrations/logfire/_with_logfire.py +0 -63
- mirascope/integrations/otel/__init__.py +0 -10
- mirascope/integrations/otel/_utils.py +0 -270
- mirascope/integrations/otel/_with_hyperdx.py +0 -60
- mirascope/integrations/otel/_with_otel.py +0 -59
- mirascope/integrations/tenacity.py +0 -14
- mirascope/llm/_call.py +0 -401
- mirascope/llm/_context.py +0 -384
- mirascope/llm/_override.py +0 -3639
- mirascope/llm/_protocols.py +0 -500
- mirascope/llm/_response_metaclass.py +0 -31
- mirascope/llm/call_response.py +0 -158
- mirascope/llm/call_response_chunk.py +0 -66
- mirascope/llm/stream.py +0 -162
- mirascope/llm/tool.py +0 -64
- mirascope/mcp/__init__.py +0 -7
- mirascope/mcp/_utils.py +0 -288
- mirascope/mcp/client.py +0 -167
- mirascope/mcp/server.py +0 -356
- mirascope/mcp/tools.py +0 -110
- mirascope/py.typed +0 -0
- mirascope/retries/__init__.py +0 -11
- mirascope/retries/fallback.py +0 -131
- mirascope/retries/tenacity.py +0 -50
- mirascope/tools/__init__.py +0 -37
- mirascope/tools/base.py +0 -98
- mirascope/tools/system/__init__.py +0 -0
- mirascope/tools/system/_docker_operation.py +0 -166
- mirascope/tools/system/_file_system.py +0 -267
- mirascope/tools/web/__init__.py +0 -0
- mirascope/tools/web/_duckduckgo.py +0 -111
- mirascope/tools/web/_httpx.py +0 -125
- mirascope/tools/web/_parse_url_content.py +0 -94
- mirascope/tools/web/_requests.py +0 -54
- mirascope/v0/__init__.py +0 -43
- mirascope/v0/anthropic.py +0 -54
- mirascope/v0/base/__init__.py +0 -12
- mirascope/v0/base/calls.py +0 -118
- mirascope/v0/base/extractors.py +0 -122
- mirascope/v0/base/ops_utils.py +0 -207
- mirascope/v0/base/prompts.py +0 -48
- mirascope/v0/base/types.py +0 -14
- mirascope/v0/base/utils.py +0 -21
- mirascope/v0/openai.py +0 -54
- mirascope-1.25.7.dist-info/METADATA +0 -169
- mirascope-1.25.7.dist-info/RECORD +0 -378
- {mirascope-1.25.7.dist-info → mirascope-2.0.0a0.dist-info}/WHEEL +0 -0
- {mirascope-1.25.7.dist-info → mirascope-2.0.0a0.dist-info}/licenses/LICENSE +0 -0
|
@@ -1,17 +0,0 @@
|
|
|
1
|
-
"""Utility for pulling metadata from a call and merging with any dynamic metadata."""
|
|
2
|
-
|
|
3
|
-
from collections.abc import Callable
|
|
4
|
-
|
|
5
|
-
from pydantic import BaseModel
|
|
6
|
-
|
|
7
|
-
from ..dynamic_config import BaseDynamicConfig
|
|
8
|
-
from ..metadata import Metadata
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
def get_metadata(
|
|
12
|
-
fn: Callable | BaseModel, dynamic_config: BaseDynamicConfig
|
|
13
|
-
) -> Metadata:
|
|
14
|
-
"""Get the metadata from the function and merge with any dynamic metadata."""
|
|
15
|
-
if dynamic_config and "metadata" in dynamic_config:
|
|
16
|
-
return dynamic_config["metadata"]
|
|
17
|
-
return getattr(fn, "_metadata", Metadata())
|
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
"""Utility for getting the possible most recent user message."""
|
|
2
|
-
|
|
3
|
-
from typing import TypeVar
|
|
4
|
-
|
|
5
|
-
_T = TypeVar("_T")
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
def get_possible_user_message_param(messages: list[_T]) -> _T | None:
|
|
9
|
-
"""Get the possible most recent user message."""
|
|
10
|
-
if not messages:
|
|
11
|
-
return None
|
|
12
|
-
most_recent_message = messages[-1]
|
|
13
|
-
if (
|
|
14
|
-
isinstance(most_recent_message, dict)
|
|
15
|
-
and "role" in most_recent_message
|
|
16
|
-
and most_recent_message["role"] == "user"
|
|
17
|
-
):
|
|
18
|
-
return most_recent_message
|
|
19
|
-
if hasattr(most_recent_message, "role") and most_recent_message.role == "user": # pyright: ignore [reportAttributeAccessIssue]
|
|
20
|
-
return most_recent_message
|
|
21
|
-
return None
|
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
"""Utility for pulling the `prompt_template` from a call or `BasePrompt`."""
|
|
2
|
-
|
|
3
|
-
import os
|
|
4
|
-
from collections.abc import Callable
|
|
5
|
-
from textwrap import dedent
|
|
6
|
-
|
|
7
|
-
from pydantic import BaseModel
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
def get_prompt_template(fn: Callable | BaseModel) -> str:
|
|
11
|
-
"""Get the metadata from the function and merge with any dynamic metadata."""
|
|
12
|
-
|
|
13
|
-
prompt_template = getattr(fn, "prompt_template", None) or getattr(
|
|
14
|
-
fn, "_prompt_template", None
|
|
15
|
-
)
|
|
16
|
-
if prompt_template:
|
|
17
|
-
return prompt_template
|
|
18
|
-
|
|
19
|
-
docstring_prompt_enabled = os.getenv("MIRASCOPE_DOCSTRING_PROMPT_TEMPLATE")
|
|
20
|
-
doc = fn.__doc__
|
|
21
|
-
if not doc:
|
|
22
|
-
raise ValueError("No prompt template set!")
|
|
23
|
-
if docstring_prompt_enabled != "ENABLED":
|
|
24
|
-
raise ValueError(
|
|
25
|
-
"You must explicitly enable docstring prompt templates by setting "
|
|
26
|
-
"`MIRASCOPE_DOCSTRING_PROMPT_TEMPLATE=ENABLED` in your environment."
|
|
27
|
-
)
|
|
28
|
-
return dedent(doc).strip()
|
|
@@ -1,51 +0,0 @@
|
|
|
1
|
-
"""This module contains the `get_template_values` function."""
|
|
2
|
-
|
|
3
|
-
from typing import Any
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
def get_template_values(
|
|
7
|
-
template_variables: list[tuple[str, str | None]], attrs: dict[str, Any]
|
|
8
|
-
) -> dict[str, Any]:
|
|
9
|
-
"""Returns the values of the given `template_variables` from the provided `attrs`.
|
|
10
|
-
|
|
11
|
-
Args:
|
|
12
|
-
template_variables: The variables to extract from the `attrs`.
|
|
13
|
-
attrs: The attributes to extract the variables from.
|
|
14
|
-
|
|
15
|
-
Returns:
|
|
16
|
-
The values of the template variables.
|
|
17
|
-
"""
|
|
18
|
-
values = {}
|
|
19
|
-
if "self" in attrs:
|
|
20
|
-
values["self"] = attrs.get("self")
|
|
21
|
-
for var, format_spec in template_variables:
|
|
22
|
-
if var.startswith("self"):
|
|
23
|
-
values["self"] = attrs.get("self")
|
|
24
|
-
elif "." in var:
|
|
25
|
-
var = var.split(".")[0]
|
|
26
|
-
values[var] = attrs.get(var)
|
|
27
|
-
continue
|
|
28
|
-
elif format_spec in ["list", "lists"]:
|
|
29
|
-
value = attrs[var]
|
|
30
|
-
if format_spec == "list":
|
|
31
|
-
if not isinstance(value, list):
|
|
32
|
-
raise ValueError(
|
|
33
|
-
f"Template variable '{var}' must be a list when using the "
|
|
34
|
-
"'list' format spec."
|
|
35
|
-
)
|
|
36
|
-
values[var] = "\n".join([str(item) for item in attrs[var]])
|
|
37
|
-
else:
|
|
38
|
-
if not isinstance(value, list) or (
|
|
39
|
-
value
|
|
40
|
-
and not all(isinstance(item, list | tuple | set) for item in value)
|
|
41
|
-
):
|
|
42
|
-
raise ValueError(
|
|
43
|
-
f"Template variable '{var}' must be a list of lists when using "
|
|
44
|
-
"the 'lists' format spec."
|
|
45
|
-
)
|
|
46
|
-
values[var] = "\n\n".join(
|
|
47
|
-
["\n".join([str(subitem) for subitem in item]) for item in value]
|
|
48
|
-
)
|
|
49
|
-
else:
|
|
50
|
-
values[var] = attrs[var] if attrs[var] is not None else ""
|
|
51
|
-
return values
|
|
@@ -1,38 +0,0 @@
|
|
|
1
|
-
"""This module provides a function to get the variables in a template string."""
|
|
2
|
-
|
|
3
|
-
from string import Formatter
|
|
4
|
-
from typing import Literal, overload
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
@overload
|
|
8
|
-
def get_template_variables(
|
|
9
|
-
template: str, include_format_spec: Literal[True]
|
|
10
|
-
) -> list[tuple[str, str | None]]: ...
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
@overload
|
|
14
|
-
def get_template_variables(
|
|
15
|
-
template: str, include_format_spec: Literal[False]
|
|
16
|
-
) -> list[str]: ...
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
def get_template_variables(
|
|
20
|
-
template: str, include_format_spec: bool
|
|
21
|
-
) -> list[str] | list[tuple[str, str | None]]:
|
|
22
|
-
"""Returns the variables in the given template string.
|
|
23
|
-
|
|
24
|
-
Args:
|
|
25
|
-
template: The template string to parse.
|
|
26
|
-
include_format_spec: A boolean indicating whether to include format specifications.
|
|
27
|
-
|
|
28
|
-
Returns:
|
|
29
|
-
The variables in the template string.
|
|
30
|
-
"""
|
|
31
|
-
if include_format_spec:
|
|
32
|
-
return [
|
|
33
|
-
(var, format_spec)
|
|
34
|
-
for _, var, format_spec, _ in Formatter().parse(template)
|
|
35
|
-
if var
|
|
36
|
-
]
|
|
37
|
-
else:
|
|
38
|
-
return [var for _, var, _, _ in Formatter().parse(template) if var]
|
|
@@ -1,10 +0,0 @@
|
|
|
1
|
-
"""Utility for getting provider-specific valid tool configuration keys."""
|
|
2
|
-
|
|
3
|
-
from ..tool import ToolConfig
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
def get_unsupported_tool_config_keys(
|
|
7
|
-
tool_config: ToolConfig, provider_config_type: type[ToolConfig]
|
|
8
|
-
) -> set[str]:
|
|
9
|
-
"""Returns the list of valid tool configuration keys for the specific provider."""
|
|
10
|
-
return set(tool_config) - set(provider_config_type.__annotations__.keys())
|
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
"""Utility for determining if a prompt template has been decorated."""
|
|
2
|
-
|
|
3
|
-
from collections.abc import Awaitable, Callable
|
|
4
|
-
from typing import ParamSpec
|
|
5
|
-
|
|
6
|
-
from typing_extensions import TypeIs
|
|
7
|
-
|
|
8
|
-
from ..dynamic_config import BaseDynamicConfig
|
|
9
|
-
from ..messages import Messages
|
|
10
|
-
|
|
11
|
-
_P = ParamSpec("_P")
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
def is_prompt_template(
|
|
15
|
-
fn: Callable[..., BaseDynamicConfig | Messages.Type]
|
|
16
|
-
| Callable[..., Awaitable[BaseDynamicConfig | Messages.Type]],
|
|
17
|
-
) -> TypeIs[
|
|
18
|
-
Callable[..., BaseDynamicConfig]
|
|
19
|
-
| Callable[
|
|
20
|
-
...,
|
|
21
|
-
Awaitable[BaseDynamicConfig],
|
|
22
|
-
]
|
|
23
|
-
]:
|
|
24
|
-
return hasattr(fn, "__mirascope_prompt_template__")
|
|
@@ -1,17 +0,0 @@
|
|
|
1
|
-
"""A function generating content to request JSON mode from models without it."""
|
|
2
|
-
|
|
3
|
-
import json
|
|
4
|
-
|
|
5
|
-
from pydantic import BaseModel
|
|
6
|
-
|
|
7
|
-
from ..tool import GenerateJsonSchemaNoTitles
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
def json_mode_content(tool_type: type[BaseModel] | None) -> str:
|
|
11
|
-
"""Returns the content to request JSON mode from models without it."""
|
|
12
|
-
if not tool_type:
|
|
13
|
-
return "\n\nFor your final response, output ONLY a valid JSON dict that adheres to the schema"
|
|
14
|
-
return f"""
|
|
15
|
-
|
|
16
|
-
For your final response, output ONLY a valid JSON dict (NOT THE SCHEMA) from the content that adheres to this schema:
|
|
17
|
-
{json.dumps(tool_type.model_json_schema(schema_generator=GenerateJsonSchemaNoTitles), indent=2)}"""
|
|
@@ -1,121 +0,0 @@
|
|
|
1
|
-
from collections.abc import Awaitable, Callable, Sequence
|
|
2
|
-
from functools import wraps
|
|
3
|
-
from typing import ParamSpec, Protocol, TypeAlias, TypeVar, overload
|
|
4
|
-
|
|
5
|
-
from typing_extensions import TypeIs
|
|
6
|
-
|
|
7
|
-
from ..dynamic_config import BaseDynamicConfig
|
|
8
|
-
from ..message_param import BaseMessageParam
|
|
9
|
-
from ..messages import Messages
|
|
10
|
-
from ._convert_messages_to_message_params import (
|
|
11
|
-
convert_messages_to_message_params,
|
|
12
|
-
)
|
|
13
|
-
from ._fn_is_async import fn_is_async
|
|
14
|
-
|
|
15
|
-
_P = ParamSpec("_P")
|
|
16
|
-
|
|
17
|
-
_MessageFuncReturnTypes: TypeAlias = BaseDynamicConfig | Messages.Type
|
|
18
|
-
_MessageFuncReturnT = TypeVar(
|
|
19
|
-
"_MessageFuncReturnT", bound=_MessageFuncReturnTypes, contravariant=True
|
|
20
|
-
)
|
|
21
|
-
|
|
22
|
-
MessagesSyncFunction: TypeAlias = Callable[_P, _MessageFuncReturnT]
|
|
23
|
-
MessagesAsyncFunction: TypeAlias = Callable[_P, Awaitable[_MessageFuncReturnT]]
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
def _is_messages_type(value: object) -> TypeIs[Messages.Type]:
|
|
27
|
-
return isinstance(
|
|
28
|
-
value,
|
|
29
|
-
str | Sequence | list | BaseMessageParam,
|
|
30
|
-
)
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
class MessagesDecorator(Protocol):
|
|
34
|
-
@overload
|
|
35
|
-
def __call__(
|
|
36
|
-
self,
|
|
37
|
-
messages_fn: Callable[_P, Messages.Type],
|
|
38
|
-
) -> Callable[_P, list[BaseMessageParam]]: ...
|
|
39
|
-
|
|
40
|
-
@overload
|
|
41
|
-
def __call__(
|
|
42
|
-
self,
|
|
43
|
-
messages_fn: Callable[_P, BaseDynamicConfig],
|
|
44
|
-
) -> Callable[_P, BaseDynamicConfig]: ...
|
|
45
|
-
|
|
46
|
-
@overload
|
|
47
|
-
def __call__(
|
|
48
|
-
self,
|
|
49
|
-
messages_fn: Callable[_P, Awaitable[Messages.Type]],
|
|
50
|
-
) -> Callable[_P, Awaitable[list[BaseMessageParam]]]: ...
|
|
51
|
-
|
|
52
|
-
@overload
|
|
53
|
-
def __call__(
|
|
54
|
-
self,
|
|
55
|
-
messages_fn: Callable[_P, Awaitable[BaseDynamicConfig]],
|
|
56
|
-
) -> Callable[_P, Awaitable[BaseDynamicConfig]]: ...
|
|
57
|
-
|
|
58
|
-
def __call__(
|
|
59
|
-
self,
|
|
60
|
-
messages_fn: MessagesSyncFunction[_P, _MessageFuncReturnT]
|
|
61
|
-
| MessagesAsyncFunction[_P, _MessageFuncReturnT],
|
|
62
|
-
) -> (
|
|
63
|
-
Callable[_P, list[BaseMessageParam] | BaseDynamicConfig]
|
|
64
|
-
| Callable[_P, Awaitable[list[BaseMessageParam] | BaseDynamicConfig]]
|
|
65
|
-
): ...
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
def messages_decorator() -> MessagesDecorator:
|
|
69
|
-
@overload
|
|
70
|
-
def inner(
|
|
71
|
-
messages_fn: Callable[_P, Messages.Type],
|
|
72
|
-
) -> Callable[_P, list[BaseMessageParam]]: ...
|
|
73
|
-
|
|
74
|
-
@overload
|
|
75
|
-
def inner(
|
|
76
|
-
messages_fn: Callable[_P, BaseDynamicConfig],
|
|
77
|
-
) -> Callable[_P, BaseDynamicConfig]: ...
|
|
78
|
-
|
|
79
|
-
@overload
|
|
80
|
-
def inner(
|
|
81
|
-
messages_fn: Callable[_P, Awaitable[Messages.Type]],
|
|
82
|
-
) -> Callable[_P, Awaitable[list[BaseMessageParam]]]: ...
|
|
83
|
-
|
|
84
|
-
@overload
|
|
85
|
-
def inner(
|
|
86
|
-
messages_fn: Callable[_P, Awaitable[BaseDynamicConfig]],
|
|
87
|
-
) -> Callable[_P, Awaitable[BaseDynamicConfig]]: ...
|
|
88
|
-
|
|
89
|
-
def inner(
|
|
90
|
-
messages_fn: MessagesSyncFunction[_P, _MessageFuncReturnT]
|
|
91
|
-
| MessagesAsyncFunction[_P, _MessageFuncReturnT],
|
|
92
|
-
) -> (
|
|
93
|
-
Callable[_P, Awaitable[list[BaseMessageParam] | BaseDynamicConfig]]
|
|
94
|
-
| Callable[_P, list[BaseMessageParam] | BaseDynamicConfig]
|
|
95
|
-
):
|
|
96
|
-
if fn_is_async(messages_fn):
|
|
97
|
-
|
|
98
|
-
@wraps(messages_fn)
|
|
99
|
-
async def get_base_message_params_async(
|
|
100
|
-
*args: _P.args, **kwargs: _P.kwargs
|
|
101
|
-
) -> list[BaseMessageParam] | BaseDynamicConfig:
|
|
102
|
-
raw_messages = await messages_fn(*args, **kwargs)
|
|
103
|
-
if _is_messages_type(raw_messages):
|
|
104
|
-
return convert_messages_to_message_params(raw_messages)
|
|
105
|
-
return raw_messages
|
|
106
|
-
|
|
107
|
-
return get_base_message_params_async
|
|
108
|
-
else:
|
|
109
|
-
|
|
110
|
-
@wraps(messages_fn)
|
|
111
|
-
def get_base_message_params(
|
|
112
|
-
*args: _P.args, **kwargs: _P.kwargs
|
|
113
|
-
) -> list[BaseMessageParam] | BaseDynamicConfig:
|
|
114
|
-
raw_messages = messages_fn(*args, **kwargs)
|
|
115
|
-
if _is_messages_type(raw_messages):
|
|
116
|
-
return convert_messages_to_message_params(raw_messages)
|
|
117
|
-
return raw_messages
|
|
118
|
-
|
|
119
|
-
return get_base_message_params
|
|
120
|
-
|
|
121
|
-
return inner
|
|
@@ -1,323 +0,0 @@
|
|
|
1
|
-
"""This module provides a function to parse content parts from a prompt template."""
|
|
2
|
-
|
|
3
|
-
import re
|
|
4
|
-
import urllib.request
|
|
5
|
-
from functools import reduce
|
|
6
|
-
from typing import Any, Literal, cast
|
|
7
|
-
|
|
8
|
-
from typing_extensions import TypedDict
|
|
9
|
-
|
|
10
|
-
from ..message_param import (
|
|
11
|
-
AudioPart,
|
|
12
|
-
AudioURLPart,
|
|
13
|
-
BaseMessageParam,
|
|
14
|
-
CacheControlPart,
|
|
15
|
-
DocumentPart,
|
|
16
|
-
ImagePart,
|
|
17
|
-
ImageURLPart,
|
|
18
|
-
TextPart,
|
|
19
|
-
)
|
|
20
|
-
from ..types import Image, has_pil_module
|
|
21
|
-
from ._format_template import format_template
|
|
22
|
-
from ._get_audio_type import get_audio_type
|
|
23
|
-
from ._get_document_type import get_document_type
|
|
24
|
-
from ._get_image_type import get_image_type
|
|
25
|
-
from ._pil_image_to_bytes import pil_image_to_bytes
|
|
26
|
-
|
|
27
|
-
_PartType = Literal[
|
|
28
|
-
"image",
|
|
29
|
-
"images",
|
|
30
|
-
"audio",
|
|
31
|
-
"audios",
|
|
32
|
-
"text",
|
|
33
|
-
"texts",
|
|
34
|
-
"document",
|
|
35
|
-
"documents",
|
|
36
|
-
"cache_control",
|
|
37
|
-
"part",
|
|
38
|
-
"parts",
|
|
39
|
-
]
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
class _Part(TypedDict):
|
|
43
|
-
template: str
|
|
44
|
-
type: _PartType
|
|
45
|
-
options: dict[str, str] | None
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
def _cleanup_text_preserve_newlines(text: str) -> str:
|
|
49
|
-
def clean_line(line: str) -> str:
|
|
50
|
-
# Remove '\r' to handle Windows-style line breaks
|
|
51
|
-
line = line.replace("\r", "")
|
|
52
|
-
|
|
53
|
-
# Remove zero-width spaces (common examples)
|
|
54
|
-
zero_width_spaces = ["\u200b", "\u200c", "\u200d", "\ufeff"]
|
|
55
|
-
for zw in zero_width_spaces:
|
|
56
|
-
line = line.replace(zw, "")
|
|
57
|
-
|
|
58
|
-
# Remove all fullwidth spaces (u3000) anywhere in the line
|
|
59
|
-
line = line.replace("\u3000", "")
|
|
60
|
-
|
|
61
|
-
# Strip leading/trailing half-width spaces (but keep tabs/newlines)
|
|
62
|
-
line = line.strip(" ")
|
|
63
|
-
|
|
64
|
-
# Remove leading tabs
|
|
65
|
-
line = line.lstrip("\t")
|
|
66
|
-
# Remove trailing tabs
|
|
67
|
-
line = line.rstrip("\t")
|
|
68
|
-
|
|
69
|
-
return line
|
|
70
|
-
|
|
71
|
-
# Split lines but preserve the newline at the end of each line
|
|
72
|
-
lines = text.splitlines(keepends=True)
|
|
73
|
-
|
|
74
|
-
# Use `map` to apply `clean_line` to each line
|
|
75
|
-
cleaned_lines = map(clean_line, lines)
|
|
76
|
-
|
|
77
|
-
# Use `reduce` to concatenate all lines into a single string
|
|
78
|
-
return reduce(lambda acc, x: acc + x, cleaned_lines, "")
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
def _parse_parts(template: str) -> list[_Part]:
|
|
82
|
-
# \{ and \} match the literal curly braces.
|
|
83
|
-
#
|
|
84
|
-
# ([^:{}]*) captures content before the colon that are not { or } or :.
|
|
85
|
-
# This can be empty when using `cache_control`.
|
|
86
|
-
#
|
|
87
|
-
# : matches the literal colon separating the type from the content.
|
|
88
|
-
#
|
|
89
|
-
# (image|images|...) captures the supported special type after the colon.
|
|
90
|
-
#
|
|
91
|
-
# (?:\(([^)]*)\))? captures the optional additional options in parentheses.
|
|
92
|
-
pattern = r"\{([^:{}]*):(image|images|audio|audios|document|documents|text|texts|cache_control|part|parts)(?:\(([^)]*)\))?\}"
|
|
93
|
-
split = re.split(pattern, template)
|
|
94
|
-
parts: list[_Part] = []
|
|
95
|
-
for i in range(0, len(split), 4):
|
|
96
|
-
if split[i]:
|
|
97
|
-
parts.append(
|
|
98
|
-
_Part(
|
|
99
|
-
template=_cleanup_text_preserve_newlines(split[i]),
|
|
100
|
-
type="text",
|
|
101
|
-
options=None,
|
|
102
|
-
)
|
|
103
|
-
)
|
|
104
|
-
if i + 3 < len(split):
|
|
105
|
-
special_content = split[i + 1]
|
|
106
|
-
special_type = cast(_PartType, split[i + 2])
|
|
107
|
-
special_options = split[i + 3]
|
|
108
|
-
if special_options is not None:
|
|
109
|
-
options: dict[str, str] = {}
|
|
110
|
-
for option in special_options.split(","):
|
|
111
|
-
key, value = option.split("=")
|
|
112
|
-
options[key] = value
|
|
113
|
-
special_options = options
|
|
114
|
-
parts.append(
|
|
115
|
-
_Part(
|
|
116
|
-
template=special_content, type=special_type, options=special_options
|
|
117
|
-
)
|
|
118
|
-
)
|
|
119
|
-
return parts
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
def _load_media(source: str | bytes) -> bytes:
|
|
123
|
-
try:
|
|
124
|
-
# Some typing weirdness here where checking `isinstance(source, bytes)` results
|
|
125
|
-
# in a type hint of `str | bytearray | memoryview` for source in the else.
|
|
126
|
-
if isinstance(source, bytes | bytearray | memoryview):
|
|
127
|
-
data = source
|
|
128
|
-
elif source.startswith(("http://", "https://", "data:", "file://")):
|
|
129
|
-
headers = {'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) Mirascope'}
|
|
130
|
-
req = urllib.request.Request(source, headers=headers)
|
|
131
|
-
with urllib.request.urlopen(req) as response:
|
|
132
|
-
data = response.read()
|
|
133
|
-
else:
|
|
134
|
-
with open(source, "rb") as f:
|
|
135
|
-
data = f.read()
|
|
136
|
-
return data
|
|
137
|
-
except Exception as e: # pragma: no cover
|
|
138
|
-
raise ValueError(
|
|
139
|
-
f"Failed to load or encode data from {source}"
|
|
140
|
-
) from e # pragma: no cover
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
def _construct_image_part(
|
|
144
|
-
source: str | bytes | Image.Image, options: dict[str, str] | None
|
|
145
|
-
) -> ImagePart | ImageURLPart:
|
|
146
|
-
detail = None
|
|
147
|
-
if options:
|
|
148
|
-
detail = options.get("detail", None)
|
|
149
|
-
if isinstance(source, str) and source.startswith(("http://", "https://", "gs://")):
|
|
150
|
-
return ImageURLPart(type="image_url", url=source, detail=detail)
|
|
151
|
-
if isinstance(source, Image.Image):
|
|
152
|
-
image = pil_image_to_bytes(source)
|
|
153
|
-
media_type = (
|
|
154
|
-
Image.MIME[source.format]
|
|
155
|
-
if has_pil_module and source.format
|
|
156
|
-
else "image/unknown"
|
|
157
|
-
)
|
|
158
|
-
else:
|
|
159
|
-
image = _load_media(source)
|
|
160
|
-
media_type = f"image/{get_image_type(image)}"
|
|
161
|
-
return ImagePart(
|
|
162
|
-
type="image",
|
|
163
|
-
media_type=media_type,
|
|
164
|
-
image=image,
|
|
165
|
-
detail=detail,
|
|
166
|
-
)
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
def _construct_audio_part(source: str | bytes) -> AudioPart | AudioURLPart:
|
|
170
|
-
# Note: audio does not currently support additional options, at least for now.
|
|
171
|
-
if isinstance(source, str) and source.startswith(("http://", "https://", "gs://")):
|
|
172
|
-
return AudioURLPart(type="audio_url", url=source)
|
|
173
|
-
audio = _load_media(source)
|
|
174
|
-
return AudioPart(
|
|
175
|
-
type="audio", media_type=f"audio/{get_audio_type(audio)}", audio=audio
|
|
176
|
-
)
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
def _construct_document_part(source: str | bytes) -> DocumentPart:
|
|
180
|
-
document = _load_media(source)
|
|
181
|
-
return DocumentPart(
|
|
182
|
-
type="document",
|
|
183
|
-
media_type=f"application/{get_document_type(document)}",
|
|
184
|
-
document=document,
|
|
185
|
-
)
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
def _construct_parts(
|
|
189
|
-
part: _Part, attrs: dict[str, Any]
|
|
190
|
-
) -> list[
|
|
191
|
-
TextPart
|
|
192
|
-
| ImagePart
|
|
193
|
-
| ImageURLPart
|
|
194
|
-
| AudioPart
|
|
195
|
-
| AudioURLPart
|
|
196
|
-
| CacheControlPart
|
|
197
|
-
| DocumentPart
|
|
198
|
-
]:
|
|
199
|
-
if part["type"] in "image":
|
|
200
|
-
source = attrs[part["template"]]
|
|
201
|
-
return [_construct_image_part(source, part["options"])] if source else []
|
|
202
|
-
elif part["type"] == "images":
|
|
203
|
-
sources = attrs[part["template"]]
|
|
204
|
-
if not isinstance(sources, list):
|
|
205
|
-
raise ValueError(
|
|
206
|
-
f"When using 'images' template, '{part['template']}' must be a list."
|
|
207
|
-
)
|
|
208
|
-
return (
|
|
209
|
-
[_construct_image_part(source, part["options"]) for source in sources]
|
|
210
|
-
if sources
|
|
211
|
-
else []
|
|
212
|
-
)
|
|
213
|
-
elif part["type"] == "audio":
|
|
214
|
-
source = attrs[part["template"]]
|
|
215
|
-
return [_construct_audio_part(source)] if source else []
|
|
216
|
-
elif part["type"] == "audios":
|
|
217
|
-
sources = attrs[part["template"]]
|
|
218
|
-
if not isinstance(sources, list):
|
|
219
|
-
raise ValueError(
|
|
220
|
-
f"When using 'audios' template, '{part['template']}' must be a list."
|
|
221
|
-
)
|
|
222
|
-
return [_construct_audio_part(source) for source in sources] if sources else []
|
|
223
|
-
elif part["type"] == "document":
|
|
224
|
-
source = attrs[part["template"]]
|
|
225
|
-
return [_construct_document_part(source)] if source else []
|
|
226
|
-
elif part["type"] == "documents":
|
|
227
|
-
sources = attrs[part["template"]]
|
|
228
|
-
if not isinstance(sources, list):
|
|
229
|
-
raise ValueError(
|
|
230
|
-
f"When using 'documents' template, '{part['template']}' must be a list."
|
|
231
|
-
)
|
|
232
|
-
return (
|
|
233
|
-
[_construct_document_part(source) for source in sources] if sources else []
|
|
234
|
-
)
|
|
235
|
-
elif part["type"] == "cache_control":
|
|
236
|
-
return [
|
|
237
|
-
CacheControlPart(
|
|
238
|
-
type="cache_control",
|
|
239
|
-
cache_type=part["options"].get("type", "ephemeral")
|
|
240
|
-
if part["options"]
|
|
241
|
-
else "ephemeral",
|
|
242
|
-
)
|
|
243
|
-
]
|
|
244
|
-
elif part["type"] == "part":
|
|
245
|
-
source = attrs[part["template"]]
|
|
246
|
-
if not isinstance(
|
|
247
|
-
source,
|
|
248
|
-
TextPart
|
|
249
|
-
| ImagePart
|
|
250
|
-
| ImageURLPart
|
|
251
|
-
| AudioPart
|
|
252
|
-
| AudioURLPart
|
|
253
|
-
| CacheControlPart
|
|
254
|
-
| DocumentPart,
|
|
255
|
-
):
|
|
256
|
-
raise ValueError(
|
|
257
|
-
f"When using 'part' template, '{part['template']}' must be a valid content part."
|
|
258
|
-
)
|
|
259
|
-
return [source] if source else []
|
|
260
|
-
elif part["type"] == "parts":
|
|
261
|
-
sources = attrs[part["template"]]
|
|
262
|
-
if not isinstance(sources, list):
|
|
263
|
-
raise ValueError(
|
|
264
|
-
f"When using 'parts' template, '{part['template']}' must be a list."
|
|
265
|
-
)
|
|
266
|
-
|
|
267
|
-
# validate each part is a valid content part
|
|
268
|
-
for source in sources:
|
|
269
|
-
if not isinstance(
|
|
270
|
-
source,
|
|
271
|
-
TextPart
|
|
272
|
-
| ImagePart
|
|
273
|
-
| ImageURLPart
|
|
274
|
-
| AudioPart
|
|
275
|
-
| AudioURLPart
|
|
276
|
-
| CacheControlPart
|
|
277
|
-
| DocumentPart,
|
|
278
|
-
):
|
|
279
|
-
raise ValueError(
|
|
280
|
-
f"When using 'parts' template, '{part['template']}' must be a list of valid content parts."
|
|
281
|
-
)
|
|
282
|
-
return sources if sources else []
|
|
283
|
-
elif part["type"] == "texts":
|
|
284
|
-
sources = attrs[part["template"]]
|
|
285
|
-
if not isinstance(sources, list):
|
|
286
|
-
raise ValueError(
|
|
287
|
-
f"When using 'texts' template, '{part['template']}' must be a list."
|
|
288
|
-
)
|
|
289
|
-
return (
|
|
290
|
-
[TextPart(type="text", text=source) for source in sources]
|
|
291
|
-
if sources
|
|
292
|
-
else []
|
|
293
|
-
)
|
|
294
|
-
else: # text type
|
|
295
|
-
text = part["template"]
|
|
296
|
-
if text in attrs:
|
|
297
|
-
source = attrs[text]
|
|
298
|
-
return [TextPart(type="text", text=source)]
|
|
299
|
-
formatted_template = format_template(part["template"], attrs, strip=False)
|
|
300
|
-
if not formatted_template:
|
|
301
|
-
return []
|
|
302
|
-
return [TextPart(type="text", text=formatted_template)]
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
def parse_content_template(
|
|
306
|
-
role: str, template: str, attrs: dict[str, Any]
|
|
307
|
-
) -> BaseMessageParam | None:
|
|
308
|
-
"""Returns the content template parsed and formatted as a message parameter."""
|
|
309
|
-
if not template:
|
|
310
|
-
return None
|
|
311
|
-
|
|
312
|
-
parts = [
|
|
313
|
-
item
|
|
314
|
-
for part in _parse_parts(template.strip())
|
|
315
|
-
for item in _construct_parts(part, attrs)
|
|
316
|
-
]
|
|
317
|
-
|
|
318
|
-
if not parts:
|
|
319
|
-
return None
|
|
320
|
-
|
|
321
|
-
if len(parts) == 1 and parts[0].type == "text":
|
|
322
|
-
return BaseMessageParam(role=role, content=parts[0].text)
|
|
323
|
-
return BaseMessageParam(role=role, content=parts)
|