agno 0.1.2__py3-none-any.whl → 2.3.13__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.
- agno/__init__.py +8 -0
- agno/agent/__init__.py +44 -5
- agno/agent/agent.py +10531 -2975
- agno/api/agent.py +14 -53
- agno/api/api.py +7 -46
- agno/api/evals.py +22 -0
- agno/api/os.py +17 -0
- agno/api/routes.py +6 -25
- agno/api/schemas/__init__.py +9 -0
- agno/api/schemas/agent.py +6 -9
- agno/api/schemas/evals.py +16 -0
- agno/api/schemas/os.py +14 -0
- agno/api/schemas/team.py +10 -10
- agno/api/schemas/utils.py +21 -0
- agno/api/schemas/workflows.py +16 -0
- agno/api/settings.py +53 -0
- agno/api/team.py +22 -26
- agno/api/workflow.py +28 -0
- agno/cloud/aws/base.py +214 -0
- agno/cloud/aws/s3/__init__.py +2 -0
- agno/cloud/aws/s3/api_client.py +43 -0
- agno/cloud/aws/s3/bucket.py +195 -0
- agno/cloud/aws/s3/object.py +57 -0
- agno/compression/__init__.py +3 -0
- agno/compression/manager.py +247 -0
- agno/culture/__init__.py +3 -0
- agno/culture/manager.py +956 -0
- agno/db/__init__.py +24 -0
- agno/db/async_postgres/__init__.py +3 -0
- agno/db/base.py +946 -0
- agno/db/dynamo/__init__.py +3 -0
- agno/db/dynamo/dynamo.py +2781 -0
- agno/db/dynamo/schemas.py +442 -0
- agno/db/dynamo/utils.py +743 -0
- agno/db/firestore/__init__.py +3 -0
- agno/db/firestore/firestore.py +2379 -0
- agno/db/firestore/schemas.py +181 -0
- agno/db/firestore/utils.py +376 -0
- agno/db/gcs_json/__init__.py +3 -0
- agno/db/gcs_json/gcs_json_db.py +1791 -0
- agno/db/gcs_json/utils.py +228 -0
- agno/db/in_memory/__init__.py +3 -0
- agno/db/in_memory/in_memory_db.py +1312 -0
- agno/db/in_memory/utils.py +230 -0
- agno/db/json/__init__.py +3 -0
- agno/db/json/json_db.py +1777 -0
- agno/db/json/utils.py +230 -0
- agno/db/migrations/manager.py +199 -0
- agno/db/migrations/v1_to_v2.py +635 -0
- agno/db/migrations/versions/v2_3_0.py +938 -0
- agno/db/mongo/__init__.py +17 -0
- agno/db/mongo/async_mongo.py +2760 -0
- agno/db/mongo/mongo.py +2597 -0
- agno/db/mongo/schemas.py +119 -0
- agno/db/mongo/utils.py +276 -0
- agno/db/mysql/__init__.py +4 -0
- agno/db/mysql/async_mysql.py +2912 -0
- agno/db/mysql/mysql.py +2923 -0
- agno/db/mysql/schemas.py +186 -0
- agno/db/mysql/utils.py +488 -0
- agno/db/postgres/__init__.py +4 -0
- agno/db/postgres/async_postgres.py +2579 -0
- agno/db/postgres/postgres.py +2870 -0
- agno/db/postgres/schemas.py +187 -0
- agno/db/postgres/utils.py +442 -0
- agno/db/redis/__init__.py +3 -0
- agno/db/redis/redis.py +2141 -0
- agno/db/redis/schemas.py +159 -0
- agno/db/redis/utils.py +346 -0
- agno/db/schemas/__init__.py +4 -0
- agno/db/schemas/culture.py +120 -0
- agno/db/schemas/evals.py +34 -0
- agno/db/schemas/knowledge.py +40 -0
- agno/db/schemas/memory.py +61 -0
- agno/db/singlestore/__init__.py +3 -0
- agno/db/singlestore/schemas.py +179 -0
- agno/db/singlestore/singlestore.py +2877 -0
- agno/db/singlestore/utils.py +384 -0
- agno/db/sqlite/__init__.py +4 -0
- agno/db/sqlite/async_sqlite.py +2911 -0
- agno/db/sqlite/schemas.py +181 -0
- agno/db/sqlite/sqlite.py +2908 -0
- agno/db/sqlite/utils.py +429 -0
- agno/db/surrealdb/__init__.py +3 -0
- agno/db/surrealdb/metrics.py +292 -0
- agno/db/surrealdb/models.py +334 -0
- agno/db/surrealdb/queries.py +71 -0
- agno/db/surrealdb/surrealdb.py +1908 -0
- agno/db/surrealdb/utils.py +147 -0
- agno/db/utils.py +118 -0
- agno/eval/__init__.py +24 -0
- agno/eval/accuracy.py +666 -276
- agno/eval/agent_as_judge.py +861 -0
- agno/eval/base.py +29 -0
- agno/eval/performance.py +779 -0
- agno/eval/reliability.py +241 -62
- agno/eval/utils.py +120 -0
- agno/exceptions.py +143 -1
- agno/filters.py +354 -0
- agno/guardrails/__init__.py +6 -0
- agno/guardrails/base.py +19 -0
- agno/guardrails/openai.py +144 -0
- agno/guardrails/pii.py +94 -0
- agno/guardrails/prompt_injection.py +52 -0
- agno/hooks/__init__.py +3 -0
- agno/hooks/decorator.py +164 -0
- agno/integrations/discord/__init__.py +3 -0
- agno/integrations/discord/client.py +203 -0
- agno/knowledge/__init__.py +5 -1
- agno/{document → knowledge}/chunking/agentic.py +22 -14
- agno/{document → knowledge}/chunking/document.py +2 -2
- agno/{document → knowledge}/chunking/fixed.py +7 -6
- agno/knowledge/chunking/markdown.py +151 -0
- agno/{document → knowledge}/chunking/recursive.py +15 -3
- agno/knowledge/chunking/row.py +39 -0
- agno/knowledge/chunking/semantic.py +91 -0
- agno/knowledge/chunking/strategy.py +165 -0
- agno/knowledge/content.py +74 -0
- agno/knowledge/document/__init__.py +5 -0
- agno/{document → knowledge/document}/base.py +12 -2
- agno/knowledge/embedder/__init__.py +5 -0
- agno/knowledge/embedder/aws_bedrock.py +343 -0
- agno/knowledge/embedder/azure_openai.py +210 -0
- agno/{embedder → knowledge/embedder}/base.py +8 -0
- agno/knowledge/embedder/cohere.py +323 -0
- agno/knowledge/embedder/fastembed.py +62 -0
- agno/{embedder → knowledge/embedder}/fireworks.py +1 -1
- agno/knowledge/embedder/google.py +258 -0
- agno/knowledge/embedder/huggingface.py +94 -0
- agno/knowledge/embedder/jina.py +182 -0
- agno/knowledge/embedder/langdb.py +22 -0
- agno/knowledge/embedder/mistral.py +206 -0
- agno/knowledge/embedder/nebius.py +13 -0
- agno/knowledge/embedder/ollama.py +154 -0
- agno/knowledge/embedder/openai.py +195 -0
- agno/knowledge/embedder/sentence_transformer.py +63 -0
- agno/{embedder → knowledge/embedder}/together.py +1 -1
- agno/knowledge/embedder/vllm.py +262 -0
- agno/knowledge/embedder/voyageai.py +165 -0
- agno/knowledge/knowledge.py +3006 -0
- agno/knowledge/reader/__init__.py +7 -0
- agno/knowledge/reader/arxiv_reader.py +81 -0
- agno/knowledge/reader/base.py +95 -0
- agno/knowledge/reader/csv_reader.py +164 -0
- agno/knowledge/reader/docx_reader.py +82 -0
- agno/knowledge/reader/field_labeled_csv_reader.py +290 -0
- agno/knowledge/reader/firecrawl_reader.py +201 -0
- agno/knowledge/reader/json_reader.py +88 -0
- agno/knowledge/reader/markdown_reader.py +137 -0
- agno/knowledge/reader/pdf_reader.py +431 -0
- agno/knowledge/reader/pptx_reader.py +101 -0
- agno/knowledge/reader/reader_factory.py +313 -0
- agno/knowledge/reader/s3_reader.py +89 -0
- agno/knowledge/reader/tavily_reader.py +193 -0
- agno/knowledge/reader/text_reader.py +127 -0
- agno/knowledge/reader/web_search_reader.py +325 -0
- agno/knowledge/reader/website_reader.py +455 -0
- agno/knowledge/reader/wikipedia_reader.py +91 -0
- agno/knowledge/reader/youtube_reader.py +78 -0
- agno/knowledge/remote_content/remote_content.py +88 -0
- agno/knowledge/reranker/__init__.py +3 -0
- agno/{reranker → knowledge/reranker}/base.py +1 -1
- agno/{reranker → knowledge/reranker}/cohere.py +2 -2
- agno/knowledge/reranker/infinity.py +195 -0
- agno/knowledge/reranker/sentence_transformer.py +54 -0
- agno/knowledge/types.py +39 -0
- agno/knowledge/utils.py +234 -0
- agno/media.py +439 -95
- agno/memory/__init__.py +16 -3
- agno/memory/manager.py +1474 -123
- agno/memory/strategies/__init__.py +15 -0
- agno/memory/strategies/base.py +66 -0
- agno/memory/strategies/summarize.py +196 -0
- agno/memory/strategies/types.py +37 -0
- agno/models/aimlapi/__init__.py +5 -0
- agno/models/aimlapi/aimlapi.py +62 -0
- agno/models/anthropic/__init__.py +4 -0
- agno/models/anthropic/claude.py +960 -496
- agno/models/aws/__init__.py +15 -0
- agno/models/aws/bedrock.py +686 -451
- agno/models/aws/claude.py +190 -183
- agno/models/azure/__init__.py +18 -1
- agno/models/azure/ai_foundry.py +489 -0
- agno/models/azure/openai_chat.py +89 -40
- agno/models/base.py +2477 -550
- agno/models/cerebras/__init__.py +12 -0
- agno/models/cerebras/cerebras.py +565 -0
- agno/models/cerebras/cerebras_openai.py +131 -0
- agno/models/cohere/__init__.py +4 -0
- agno/models/cohere/chat.py +306 -492
- agno/models/cometapi/__init__.py +5 -0
- agno/models/cometapi/cometapi.py +74 -0
- agno/models/dashscope/__init__.py +5 -0
- agno/models/dashscope/dashscope.py +90 -0
- agno/models/deepinfra/__init__.py +5 -0
- agno/models/deepinfra/deepinfra.py +45 -0
- agno/models/deepseek/__init__.py +4 -0
- agno/models/deepseek/deepseek.py +110 -9
- agno/models/fireworks/__init__.py +4 -0
- agno/models/fireworks/fireworks.py +19 -22
- agno/models/google/__init__.py +3 -7
- agno/models/google/gemini.py +1717 -662
- agno/models/google/utils.py +22 -0
- agno/models/groq/__init__.py +4 -0
- agno/models/groq/groq.py +391 -666
- agno/models/huggingface/__init__.py +4 -0
- agno/models/huggingface/huggingface.py +266 -538
- agno/models/ibm/__init__.py +5 -0
- agno/models/ibm/watsonx.py +432 -0
- agno/models/internlm/__init__.py +3 -0
- agno/models/internlm/internlm.py +20 -3
- agno/models/langdb/__init__.py +1 -0
- agno/models/langdb/langdb.py +60 -0
- agno/models/litellm/__init__.py +14 -0
- agno/models/litellm/chat.py +503 -0
- agno/models/litellm/litellm_openai.py +42 -0
- agno/models/llama_cpp/__init__.py +5 -0
- agno/models/llama_cpp/llama_cpp.py +22 -0
- agno/models/lmstudio/__init__.py +5 -0
- agno/models/lmstudio/lmstudio.py +25 -0
- agno/models/message.py +361 -39
- agno/models/meta/__init__.py +12 -0
- agno/models/meta/llama.py +502 -0
- agno/models/meta/llama_openai.py +79 -0
- agno/models/metrics.py +120 -0
- agno/models/mistral/__init__.py +4 -0
- agno/models/mistral/mistral.py +293 -393
- agno/models/nebius/__init__.py +3 -0
- agno/models/nebius/nebius.py +53 -0
- agno/models/nexus/__init__.py +3 -0
- agno/models/nexus/nexus.py +22 -0
- agno/models/nvidia/__init__.py +4 -0
- agno/models/nvidia/nvidia.py +22 -3
- agno/models/ollama/__init__.py +4 -2
- agno/models/ollama/chat.py +257 -492
- agno/models/openai/__init__.py +7 -0
- agno/models/openai/chat.py +725 -770
- agno/models/openai/like.py +16 -2
- agno/models/openai/responses.py +1121 -0
- agno/models/openrouter/__init__.py +4 -0
- agno/models/openrouter/openrouter.py +62 -5
- agno/models/perplexity/__init__.py +5 -0
- agno/models/perplexity/perplexity.py +203 -0
- agno/models/portkey/__init__.py +3 -0
- agno/models/portkey/portkey.py +82 -0
- agno/models/requesty/__init__.py +5 -0
- agno/models/requesty/requesty.py +69 -0
- agno/models/response.py +177 -7
- agno/models/sambanova/__init__.py +4 -0
- agno/models/sambanova/sambanova.py +23 -4
- agno/models/siliconflow/__init__.py +5 -0
- agno/models/siliconflow/siliconflow.py +42 -0
- agno/models/together/__init__.py +4 -0
- agno/models/together/together.py +21 -164
- agno/models/utils.py +266 -0
- agno/models/vercel/__init__.py +3 -0
- agno/models/vercel/v0.py +43 -0
- agno/models/vertexai/__init__.py +0 -1
- agno/models/vertexai/claude.py +190 -0
- agno/models/vllm/__init__.py +3 -0
- agno/models/vllm/vllm.py +83 -0
- agno/models/xai/__init__.py +2 -0
- agno/models/xai/xai.py +111 -7
- agno/os/__init__.py +3 -0
- agno/os/app.py +1027 -0
- agno/os/auth.py +244 -0
- agno/os/config.py +126 -0
- agno/os/interfaces/__init__.py +1 -0
- agno/os/interfaces/a2a/__init__.py +3 -0
- agno/os/interfaces/a2a/a2a.py +42 -0
- agno/os/interfaces/a2a/router.py +249 -0
- agno/os/interfaces/a2a/utils.py +924 -0
- agno/os/interfaces/agui/__init__.py +3 -0
- agno/os/interfaces/agui/agui.py +47 -0
- agno/os/interfaces/agui/router.py +147 -0
- agno/os/interfaces/agui/utils.py +574 -0
- agno/os/interfaces/base.py +25 -0
- agno/os/interfaces/slack/__init__.py +3 -0
- agno/os/interfaces/slack/router.py +148 -0
- agno/os/interfaces/slack/security.py +30 -0
- agno/os/interfaces/slack/slack.py +47 -0
- agno/os/interfaces/whatsapp/__init__.py +3 -0
- agno/os/interfaces/whatsapp/router.py +210 -0
- agno/os/interfaces/whatsapp/security.py +55 -0
- agno/os/interfaces/whatsapp/whatsapp.py +36 -0
- agno/os/mcp.py +293 -0
- agno/os/middleware/__init__.py +9 -0
- agno/os/middleware/jwt.py +797 -0
- agno/os/router.py +258 -0
- agno/os/routers/__init__.py +3 -0
- agno/os/routers/agents/__init__.py +3 -0
- agno/os/routers/agents/router.py +599 -0
- agno/os/routers/agents/schema.py +261 -0
- agno/os/routers/evals/__init__.py +3 -0
- agno/os/routers/evals/evals.py +450 -0
- agno/os/routers/evals/schemas.py +174 -0
- agno/os/routers/evals/utils.py +231 -0
- agno/os/routers/health.py +31 -0
- agno/os/routers/home.py +52 -0
- agno/os/routers/knowledge/__init__.py +3 -0
- agno/os/routers/knowledge/knowledge.py +1008 -0
- agno/os/routers/knowledge/schemas.py +178 -0
- agno/os/routers/memory/__init__.py +3 -0
- agno/os/routers/memory/memory.py +661 -0
- agno/os/routers/memory/schemas.py +88 -0
- agno/os/routers/metrics/__init__.py +3 -0
- agno/os/routers/metrics/metrics.py +190 -0
- agno/os/routers/metrics/schemas.py +47 -0
- agno/os/routers/session/__init__.py +3 -0
- agno/os/routers/session/session.py +997 -0
- agno/os/routers/teams/__init__.py +3 -0
- agno/os/routers/teams/router.py +512 -0
- agno/os/routers/teams/schema.py +257 -0
- agno/os/routers/traces/__init__.py +3 -0
- agno/os/routers/traces/schemas.py +414 -0
- agno/os/routers/traces/traces.py +499 -0
- agno/os/routers/workflows/__init__.py +3 -0
- agno/os/routers/workflows/router.py +624 -0
- agno/os/routers/workflows/schema.py +75 -0
- agno/os/schema.py +534 -0
- agno/os/scopes.py +469 -0
- agno/{playground → os}/settings.py +7 -15
- agno/os/utils.py +973 -0
- agno/reasoning/anthropic.py +80 -0
- agno/reasoning/azure_ai_foundry.py +67 -0
- agno/reasoning/deepseek.py +63 -0
- agno/reasoning/default.py +97 -0
- agno/reasoning/gemini.py +73 -0
- agno/reasoning/groq.py +71 -0
- agno/reasoning/helpers.py +24 -1
- agno/reasoning/ollama.py +67 -0
- agno/reasoning/openai.py +86 -0
- agno/reasoning/step.py +2 -1
- agno/reasoning/vertexai.py +76 -0
- agno/run/__init__.py +6 -0
- agno/run/agent.py +822 -0
- agno/run/base.py +247 -0
- agno/run/cancel.py +81 -0
- agno/run/requirement.py +181 -0
- agno/run/team.py +767 -0
- agno/run/workflow.py +708 -0
- agno/session/__init__.py +10 -0
- agno/session/agent.py +260 -0
- agno/session/summary.py +265 -0
- agno/session/team.py +342 -0
- agno/session/workflow.py +501 -0
- agno/table.py +10 -0
- agno/team/__init__.py +37 -0
- agno/team/team.py +9536 -0
- agno/tools/__init__.py +7 -0
- agno/tools/agentql.py +120 -0
- agno/tools/airflow.py +22 -12
- agno/tools/api.py +122 -0
- agno/tools/apify.py +276 -83
- agno/tools/{arxiv_toolkit.py → arxiv.py} +20 -12
- agno/tools/aws_lambda.py +28 -7
- agno/tools/aws_ses.py +66 -0
- agno/tools/baidusearch.py +11 -4
- agno/tools/bitbucket.py +292 -0
- agno/tools/brandfetch.py +213 -0
- agno/tools/bravesearch.py +106 -0
- agno/tools/brightdata.py +367 -0
- agno/tools/browserbase.py +209 -0
- agno/tools/calcom.py +32 -23
- agno/tools/calculator.py +24 -37
- agno/tools/cartesia.py +187 -0
- agno/tools/{clickup_tool.py → clickup.py} +17 -28
- agno/tools/confluence.py +91 -26
- agno/tools/crawl4ai.py +139 -43
- agno/tools/csv_toolkit.py +28 -22
- agno/tools/dalle.py +36 -22
- agno/tools/daytona.py +475 -0
- agno/tools/decorator.py +169 -14
- agno/tools/desi_vocal.py +23 -11
- agno/tools/discord.py +32 -29
- agno/tools/docker.py +716 -0
- agno/tools/duckdb.py +76 -81
- agno/tools/duckduckgo.py +43 -40
- agno/tools/e2b.py +703 -0
- agno/tools/eleven_labs.py +65 -54
- agno/tools/email.py +13 -5
- agno/tools/evm.py +129 -0
- agno/tools/exa.py +324 -42
- agno/tools/fal.py +39 -35
- agno/tools/file.py +196 -30
- agno/tools/file_generation.py +356 -0
- agno/tools/financial_datasets.py +288 -0
- agno/tools/firecrawl.py +108 -33
- agno/tools/function.py +960 -122
- agno/tools/giphy.py +34 -12
- agno/tools/github.py +1294 -97
- agno/tools/gmail.py +922 -0
- agno/tools/google_bigquery.py +117 -0
- agno/tools/google_drive.py +271 -0
- agno/tools/google_maps.py +253 -0
- agno/tools/googlecalendar.py +607 -107
- agno/tools/googlesheets.py +377 -0
- agno/tools/hackernews.py +20 -12
- agno/tools/jina.py +24 -14
- agno/tools/jira.py +48 -19
- agno/tools/knowledge.py +218 -0
- agno/tools/linear.py +82 -43
- agno/tools/linkup.py +58 -0
- agno/tools/local_file_system.py +15 -7
- agno/tools/lumalab.py +41 -26
- agno/tools/mcp/__init__.py +10 -0
- agno/tools/mcp/mcp.py +331 -0
- agno/tools/mcp/multi_mcp.py +347 -0
- agno/tools/mcp/params.py +24 -0
- agno/tools/mcp_toolbox.py +284 -0
- agno/tools/mem0.py +193 -0
- agno/tools/memory.py +419 -0
- agno/tools/mlx_transcribe.py +11 -9
- agno/tools/models/azure_openai.py +190 -0
- agno/tools/models/gemini.py +203 -0
- agno/tools/models/groq.py +158 -0
- agno/tools/models/morph.py +186 -0
- agno/tools/models/nebius.py +124 -0
- agno/tools/models_labs.py +163 -82
- agno/tools/moviepy_video.py +18 -13
- agno/tools/nano_banana.py +151 -0
- agno/tools/neo4j.py +134 -0
- agno/tools/newspaper.py +15 -4
- agno/tools/newspaper4k.py +19 -6
- agno/tools/notion.py +204 -0
- agno/tools/openai.py +181 -17
- agno/tools/openbb.py +27 -20
- agno/tools/opencv.py +321 -0
- agno/tools/openweather.py +233 -0
- agno/tools/oxylabs.py +385 -0
- agno/tools/pandas.py +25 -15
- agno/tools/parallel.py +314 -0
- agno/tools/postgres.py +238 -185
- agno/tools/pubmed.py +125 -13
- agno/tools/python.py +48 -35
- agno/tools/reasoning.py +283 -0
- agno/tools/reddit.py +207 -29
- agno/tools/redshift.py +406 -0
- agno/tools/replicate.py +69 -26
- agno/tools/resend.py +11 -6
- agno/tools/scrapegraph.py +179 -19
- agno/tools/searxng.py +23 -31
- agno/tools/serpapi.py +15 -10
- agno/tools/serper.py +255 -0
- agno/tools/shell.py +23 -12
- agno/tools/shopify.py +1519 -0
- agno/tools/slack.py +56 -14
- agno/tools/sleep.py +8 -6
- agno/tools/spider.py +35 -11
- agno/tools/spotify.py +919 -0
- agno/tools/sql.py +34 -19
- agno/tools/tavily.py +158 -8
- agno/tools/telegram.py +18 -8
- agno/tools/todoist.py +218 -0
- agno/tools/toolkit.py +134 -9
- agno/tools/trafilatura.py +388 -0
- agno/tools/trello.py +25 -28
- agno/tools/twilio.py +18 -9
- agno/tools/user_control_flow.py +78 -0
- agno/tools/valyu.py +228 -0
- agno/tools/visualization.py +467 -0
- agno/tools/webbrowser.py +28 -0
- agno/tools/webex.py +76 -0
- agno/tools/website.py +23 -19
- agno/tools/webtools.py +45 -0
- agno/tools/whatsapp.py +286 -0
- agno/tools/wikipedia.py +28 -19
- agno/tools/workflow.py +285 -0
- agno/tools/{twitter.py → x.py} +142 -46
- agno/tools/yfinance.py +41 -39
- agno/tools/youtube.py +34 -17
- agno/tools/zendesk.py +15 -5
- agno/tools/zep.py +454 -0
- agno/tools/zoom.py +86 -37
- agno/tracing/__init__.py +12 -0
- agno/tracing/exporter.py +157 -0
- agno/tracing/schemas.py +276 -0
- agno/tracing/setup.py +111 -0
- agno/utils/agent.py +938 -0
- agno/utils/audio.py +37 -1
- agno/utils/certs.py +27 -0
- agno/utils/code_execution.py +11 -0
- agno/utils/common.py +103 -20
- agno/utils/cryptography.py +22 -0
- agno/utils/dttm.py +33 -0
- agno/utils/events.py +700 -0
- agno/utils/functions.py +107 -37
- agno/utils/gemini.py +426 -0
- agno/utils/hooks.py +171 -0
- agno/utils/http.py +185 -0
- agno/utils/json_schema.py +159 -37
- agno/utils/knowledge.py +36 -0
- agno/utils/location.py +19 -0
- agno/utils/log.py +221 -8
- agno/utils/mcp.py +214 -0
- agno/utils/media.py +335 -14
- agno/utils/merge_dict.py +22 -1
- agno/utils/message.py +77 -2
- agno/utils/models/ai_foundry.py +50 -0
- agno/utils/models/claude.py +373 -0
- agno/utils/models/cohere.py +94 -0
- agno/utils/models/llama.py +85 -0
- agno/utils/models/mistral.py +100 -0
- agno/utils/models/openai_responses.py +140 -0
- agno/utils/models/schema_utils.py +153 -0
- agno/utils/models/watsonx.py +41 -0
- agno/utils/openai.py +257 -0
- agno/utils/pickle.py +1 -1
- agno/utils/pprint.py +124 -8
- agno/utils/print_response/agent.py +930 -0
- agno/utils/print_response/team.py +1914 -0
- agno/utils/print_response/workflow.py +1668 -0
- agno/utils/prompts.py +111 -0
- agno/utils/reasoning.py +108 -0
- agno/utils/response.py +163 -0
- agno/utils/serialize.py +32 -0
- agno/utils/shell.py +4 -4
- agno/utils/streamlit.py +487 -0
- agno/utils/string.py +204 -51
- agno/utils/team.py +139 -0
- agno/utils/timer.py +9 -2
- agno/utils/tokens.py +657 -0
- agno/utils/tools.py +19 -1
- agno/utils/whatsapp.py +305 -0
- agno/utils/yaml_io.py +3 -3
- agno/vectordb/__init__.py +2 -0
- agno/vectordb/base.py +87 -9
- agno/vectordb/cassandra/__init__.py +5 -1
- agno/vectordb/cassandra/cassandra.py +383 -27
- agno/vectordb/chroma/__init__.py +4 -0
- agno/vectordb/chroma/chromadb.py +748 -83
- agno/vectordb/clickhouse/__init__.py +7 -1
- agno/vectordb/clickhouse/clickhousedb.py +554 -53
- agno/vectordb/couchbase/__init__.py +3 -0
- agno/vectordb/couchbase/couchbase.py +1446 -0
- agno/vectordb/lancedb/__init__.py +5 -0
- agno/vectordb/lancedb/lance_db.py +730 -98
- agno/vectordb/langchaindb/__init__.py +5 -0
- agno/vectordb/langchaindb/langchaindb.py +163 -0
- agno/vectordb/lightrag/__init__.py +5 -0
- agno/vectordb/lightrag/lightrag.py +388 -0
- agno/vectordb/llamaindex/__init__.py +3 -0
- agno/vectordb/llamaindex/llamaindexdb.py +166 -0
- agno/vectordb/milvus/__init__.py +3 -0
- agno/vectordb/milvus/milvus.py +966 -78
- agno/vectordb/mongodb/__init__.py +9 -1
- agno/vectordb/mongodb/mongodb.py +1175 -172
- agno/vectordb/pgvector/__init__.py +8 -0
- agno/vectordb/pgvector/pgvector.py +599 -115
- agno/vectordb/pineconedb/__init__.py +5 -1
- agno/vectordb/pineconedb/pineconedb.py +406 -43
- agno/vectordb/qdrant/__init__.py +4 -0
- agno/vectordb/qdrant/qdrant.py +914 -61
- agno/vectordb/redis/__init__.py +9 -0
- agno/vectordb/redis/redisdb.py +682 -0
- agno/vectordb/singlestore/__init__.py +8 -1
- agno/vectordb/singlestore/singlestore.py +771 -0
- agno/vectordb/surrealdb/__init__.py +3 -0
- agno/vectordb/surrealdb/surrealdb.py +663 -0
- agno/vectordb/upstashdb/__init__.py +5 -0
- agno/vectordb/upstashdb/upstashdb.py +718 -0
- agno/vectordb/weaviate/__init__.py +8 -0
- agno/vectordb/weaviate/index.py +15 -0
- agno/vectordb/weaviate/weaviate.py +1009 -0
- agno/workflow/__init__.py +23 -1
- agno/workflow/agent.py +299 -0
- agno/workflow/condition.py +759 -0
- agno/workflow/loop.py +756 -0
- agno/workflow/parallel.py +853 -0
- agno/workflow/router.py +723 -0
- agno/workflow/step.py +1564 -0
- agno/workflow/steps.py +613 -0
- agno/workflow/types.py +556 -0
- agno/workflow/workflow.py +4327 -514
- agno-2.3.13.dist-info/METADATA +639 -0
- agno-2.3.13.dist-info/RECORD +613 -0
- {agno-0.1.2.dist-info → agno-2.3.13.dist-info}/WHEEL +1 -1
- agno-2.3.13.dist-info/licenses/LICENSE +201 -0
- agno/api/playground.py +0 -91
- agno/api/schemas/playground.py +0 -22
- agno/api/schemas/user.py +0 -22
- agno/api/schemas/workspace.py +0 -46
- agno/api/user.py +0 -160
- agno/api/workspace.py +0 -151
- agno/cli/auth_server.py +0 -118
- agno/cli/config.py +0 -275
- agno/cli/console.py +0 -88
- agno/cli/credentials.py +0 -23
- agno/cli/entrypoint.py +0 -571
- agno/cli/operator.py +0 -355
- agno/cli/settings.py +0 -85
- agno/cli/ws/ws_cli.py +0 -817
- agno/constants.py +0 -13
- agno/document/__init__.py +0 -1
- agno/document/chunking/semantic.py +0 -47
- agno/document/chunking/strategy.py +0 -31
- agno/document/reader/__init__.py +0 -1
- agno/document/reader/arxiv_reader.py +0 -41
- agno/document/reader/base.py +0 -22
- agno/document/reader/csv_reader.py +0 -84
- agno/document/reader/docx_reader.py +0 -46
- agno/document/reader/firecrawl_reader.py +0 -99
- agno/document/reader/json_reader.py +0 -43
- agno/document/reader/pdf_reader.py +0 -219
- agno/document/reader/s3/pdf_reader.py +0 -46
- agno/document/reader/s3/text_reader.py +0 -51
- agno/document/reader/text_reader.py +0 -41
- agno/document/reader/website_reader.py +0 -175
- agno/document/reader/youtube_reader.py +0 -50
- agno/embedder/__init__.py +0 -1
- agno/embedder/azure_openai.py +0 -86
- agno/embedder/cohere.py +0 -72
- agno/embedder/fastembed.py +0 -37
- agno/embedder/google.py +0 -73
- agno/embedder/huggingface.py +0 -54
- agno/embedder/mistral.py +0 -80
- agno/embedder/ollama.py +0 -57
- agno/embedder/openai.py +0 -74
- agno/embedder/sentence_transformer.py +0 -38
- agno/embedder/voyageai.py +0 -64
- agno/eval/perf.py +0 -201
- agno/file/__init__.py +0 -1
- agno/file/file.py +0 -16
- agno/file/local/csv.py +0 -32
- agno/file/local/txt.py +0 -19
- agno/infra/app.py +0 -240
- agno/infra/base.py +0 -144
- agno/infra/context.py +0 -20
- agno/infra/db_app.py +0 -52
- agno/infra/resource.py +0 -205
- agno/infra/resources.py +0 -55
- agno/knowledge/agent.py +0 -230
- agno/knowledge/arxiv.py +0 -22
- agno/knowledge/combined.py +0 -22
- agno/knowledge/csv.py +0 -28
- agno/knowledge/csv_url.py +0 -19
- agno/knowledge/document.py +0 -20
- agno/knowledge/docx.py +0 -30
- agno/knowledge/json.py +0 -28
- agno/knowledge/langchain.py +0 -71
- agno/knowledge/llamaindex.py +0 -66
- agno/knowledge/pdf.py +0 -28
- agno/knowledge/pdf_url.py +0 -26
- agno/knowledge/s3/base.py +0 -60
- agno/knowledge/s3/pdf.py +0 -21
- agno/knowledge/s3/text.py +0 -23
- agno/knowledge/text.py +0 -30
- agno/knowledge/website.py +0 -88
- agno/knowledge/wikipedia.py +0 -31
- agno/knowledge/youtube.py +0 -22
- agno/memory/agent.py +0 -392
- agno/memory/classifier.py +0 -104
- agno/memory/db/__init__.py +0 -1
- agno/memory/db/base.py +0 -42
- agno/memory/db/mongodb.py +0 -189
- agno/memory/db/postgres.py +0 -203
- agno/memory/db/sqlite.py +0 -193
- agno/memory/memory.py +0 -15
- agno/memory/row.py +0 -36
- agno/memory/summarizer.py +0 -192
- agno/memory/summary.py +0 -19
- agno/memory/workflow.py +0 -38
- agno/models/google/gemini_openai.py +0 -26
- agno/models/ollama/hermes.py +0 -221
- agno/models/ollama/tools.py +0 -362
- agno/models/vertexai/gemini.py +0 -595
- agno/playground/__init__.py +0 -3
- agno/playground/async_router.py +0 -421
- agno/playground/deploy.py +0 -249
- agno/playground/operator.py +0 -92
- agno/playground/playground.py +0 -91
- agno/playground/schemas.py +0 -76
- agno/playground/serve.py +0 -55
- agno/playground/sync_router.py +0 -405
- agno/reasoning/agent.py +0 -68
- agno/run/response.py +0 -112
- agno/storage/agent/__init__.py +0 -0
- agno/storage/agent/base.py +0 -38
- agno/storage/agent/dynamodb.py +0 -350
- agno/storage/agent/json.py +0 -92
- agno/storage/agent/mongodb.py +0 -228
- agno/storage/agent/postgres.py +0 -367
- agno/storage/agent/session.py +0 -79
- agno/storage/agent/singlestore.py +0 -303
- agno/storage/agent/sqlite.py +0 -357
- agno/storage/agent/yaml.py +0 -93
- agno/storage/workflow/__init__.py +0 -0
- agno/storage/workflow/base.py +0 -40
- agno/storage/workflow/mongodb.py +0 -233
- agno/storage/workflow/postgres.py +0 -366
- agno/storage/workflow/session.py +0 -60
- agno/storage/workflow/sqlite.py +0 -359
- agno/tools/googlesearch.py +0 -88
- agno/utils/defaults.py +0 -57
- agno/utils/filesystem.py +0 -39
- agno/utils/git.py +0 -52
- agno/utils/json_io.py +0 -30
- agno/utils/load_env.py +0 -19
- agno/utils/py_io.py +0 -19
- agno/utils/pyproject.py +0 -18
- agno/utils/resource_filter.py +0 -31
- agno/vectordb/singlestore/s2vectordb.py +0 -390
- agno/vectordb/singlestore/s2vectordb2.py +0 -355
- agno/workspace/__init__.py +0 -0
- agno/workspace/config.py +0 -325
- agno/workspace/enums.py +0 -6
- agno/workspace/helpers.py +0 -48
- agno/workspace/operator.py +0 -758
- agno/workspace/settings.py +0 -63
- agno-0.1.2.dist-info/LICENSE +0 -375
- agno-0.1.2.dist-info/METADATA +0 -502
- agno-0.1.2.dist-info/RECORD +0 -352
- agno-0.1.2.dist-info/entry_points.txt +0 -3
- /agno/{cli → db/migrations}/__init__.py +0 -0
- /agno/{cli/ws → db/migrations/versions}/__init__.py +0 -0
- /agno/{document/chunking/__init__.py → db/schemas/metrics.py} +0 -0
- /agno/{document/reader/s3 → integrations}/__init__.py +0 -0
- /agno/{file/local → knowledge/chunking}/__init__.py +0 -0
- /agno/{infra → knowledge/remote_content}/__init__.py +0 -0
- /agno/{knowledge/s3 → tools/models}/__init__.py +0 -0
- /agno/{reranker → utils/models}/__init__.py +0 -0
- /agno/{storage → utils/print_response}/__init__.py +0 -0
- {agno-0.1.2.dist-info → agno-2.3.13.dist-info}/top_level.txt +0 -0
agno/models/ollama/hermes.py
DELETED
|
@@ -1,221 +0,0 @@
|
|
|
1
|
-
import json
|
|
2
|
-
from dataclasses import dataclass, field
|
|
3
|
-
from typing import Any, Dict, Iterator, List, Mapping, Optional, Tuple
|
|
4
|
-
|
|
5
|
-
from agno.models.message import Message
|
|
6
|
-
from agno.models.ollama.chat import Metrics, Ollama
|
|
7
|
-
from agno.models.response import ModelResponse
|
|
8
|
-
from agno.utils.log import logger
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
@dataclass
|
|
12
|
-
class MessageData:
|
|
13
|
-
response_role: Optional[str] = None
|
|
14
|
-
response_message: Optional[Dict[str, Any]] = None
|
|
15
|
-
response_content: Any = ""
|
|
16
|
-
response_content_chunk: str = ""
|
|
17
|
-
tool_calls: List[Dict[str, Any]] = field(default_factory=list)
|
|
18
|
-
tool_call_blocks: Any = field(default_factory=list)
|
|
19
|
-
tool_call_chunk: str = ""
|
|
20
|
-
in_tool_call: bool = False
|
|
21
|
-
end_tool_call: bool = False
|
|
22
|
-
response_usage: Optional[Mapping[str, Any]] = None
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
@dataclass
|
|
26
|
-
class OllamaHermes(Ollama):
|
|
27
|
-
"""
|
|
28
|
-
A class for interacting with the OllamaHermes model via Ollama. This is a subclass of the Ollama model,
|
|
29
|
-
which customizes tool call streaming for the hermes3 model.
|
|
30
|
-
"""
|
|
31
|
-
|
|
32
|
-
id: str = "hermes3"
|
|
33
|
-
name: str = "OllamaHermes"
|
|
34
|
-
provider: str = "Ollama"
|
|
35
|
-
|
|
36
|
-
def handle_tool_call_chunk(self, content, tool_call_buffer, message_data) -> Tuple[str, bool]:
|
|
37
|
-
"""
|
|
38
|
-
Handle a tool call chunk for response stream.
|
|
39
|
-
|
|
40
|
-
Args:
|
|
41
|
-
content: The content of the tool call.
|
|
42
|
-
tool_call_buffer: The tool call buffer.
|
|
43
|
-
message_data: The message data.
|
|
44
|
-
|
|
45
|
-
Returns:
|
|
46
|
-
Tuple[str, bool]: The tool call buffer and a boolean indicating if the tool call is complete.
|
|
47
|
-
"""
|
|
48
|
-
if content != "</tool_call>":
|
|
49
|
-
tool_call_buffer += content
|
|
50
|
-
|
|
51
|
-
if message_data.end_tool_call:
|
|
52
|
-
try:
|
|
53
|
-
tool_call_data = json.loads(tool_call_buffer)
|
|
54
|
-
message_data.tool_call_blocks.append(tool_call_data)
|
|
55
|
-
message_data.end_tool_call = False
|
|
56
|
-
except json.JSONDecodeError:
|
|
57
|
-
logger.error("Failed to parse tool call JSON.")
|
|
58
|
-
return "", False
|
|
59
|
-
|
|
60
|
-
return tool_call_buffer, True
|
|
61
|
-
|
|
62
|
-
def _format_tool_calls(self, message_data: MessageData) -> MessageData:
|
|
63
|
-
if message_data.tool_call_blocks is not None:
|
|
64
|
-
for block in message_data.tool_call_blocks:
|
|
65
|
-
tool_name = block.get("name")
|
|
66
|
-
tool_args = block.get("arguments")
|
|
67
|
-
|
|
68
|
-
function_def = {
|
|
69
|
-
"name": tool_name,
|
|
70
|
-
"arguments": json.dumps(tool_args) if tool_args is not None else None,
|
|
71
|
-
}
|
|
72
|
-
message_data.tool_calls.append({"type": "function", "function": function_def})
|
|
73
|
-
return message_data
|
|
74
|
-
|
|
75
|
-
def response_stream(self, messages: List[Message]) -> Iterator[ModelResponse]:
|
|
76
|
-
"""
|
|
77
|
-
Generate a streaming response from Ollama.
|
|
78
|
-
|
|
79
|
-
Args:
|
|
80
|
-
messages (List[Message]): A list of messages.
|
|
81
|
-
|
|
82
|
-
Returns:
|
|
83
|
-
Iterator[ModelResponse]: An iterator of the model responses.
|
|
84
|
-
"""
|
|
85
|
-
logger.debug("---------- Ollama OllamaHermes Response Start ----------")
|
|
86
|
-
self._log_messages(messages)
|
|
87
|
-
message_data = MessageData()
|
|
88
|
-
metrics: Metrics = Metrics()
|
|
89
|
-
|
|
90
|
-
# -*- Generate response
|
|
91
|
-
metrics.start_response_timer()
|
|
92
|
-
for response in self.invoke_stream(messages=messages):
|
|
93
|
-
message_data.response_message = response.get("message", {})
|
|
94
|
-
if message_data.response_message:
|
|
95
|
-
metrics.output_tokens += 1
|
|
96
|
-
if metrics.output_tokens == 1:
|
|
97
|
-
metrics.time_to_first_token = metrics.response_timer.elapsed
|
|
98
|
-
|
|
99
|
-
message_data.response_content_chunk = message_data.response_message.get("content", "").strip("`")
|
|
100
|
-
|
|
101
|
-
if message_data.response_content_chunk:
|
|
102
|
-
if message_data.response_content_chunk.strip().startswith("</tool_call>"):
|
|
103
|
-
message_data.end_tool_call = True
|
|
104
|
-
if message_data.in_tool_call:
|
|
105
|
-
message_data.tool_call_chunk, message_data.in_tool_call = self.handle_tool_call_chunk(
|
|
106
|
-
message_data.response_content_chunk, message_data.tool_call_chunk, message_data
|
|
107
|
-
)
|
|
108
|
-
elif message_data.response_content_chunk.strip().startswith("<tool_call>"):
|
|
109
|
-
message_data.in_tool_call = True
|
|
110
|
-
else:
|
|
111
|
-
yield ModelResponse(content=message_data.response_content_chunk)
|
|
112
|
-
message_data.response_content += message_data.response_content_chunk
|
|
113
|
-
|
|
114
|
-
if response.get("done"):
|
|
115
|
-
message_data.response_usage = response
|
|
116
|
-
metrics.stop_response_timer()
|
|
117
|
-
|
|
118
|
-
# Format tool calls
|
|
119
|
-
message_data = self._format_tool_calls(message_data)
|
|
120
|
-
|
|
121
|
-
# -*- Create assistant message
|
|
122
|
-
assistant_message = Message(role="assistant", content=message_data.response_content)
|
|
123
|
-
|
|
124
|
-
if len(message_data.tool_calls) > 0:
|
|
125
|
-
assistant_message.tool_calls = message_data.tool_calls
|
|
126
|
-
|
|
127
|
-
# -*- Update usage metrics
|
|
128
|
-
self.update_usage_metrics(
|
|
129
|
-
assistant_message=assistant_message, metrics=metrics, response=message_data.response_usage
|
|
130
|
-
)
|
|
131
|
-
|
|
132
|
-
# -*- Add assistant message to messages
|
|
133
|
-
messages.append(assistant_message)
|
|
134
|
-
|
|
135
|
-
# -*- Log response and metrics
|
|
136
|
-
assistant_message.log()
|
|
137
|
-
metrics.log()
|
|
138
|
-
|
|
139
|
-
# -*- Handle tool calls
|
|
140
|
-
if assistant_message.tool_calls is not None and len(assistant_message.tool_calls) > 0:
|
|
141
|
-
yield from self.handle_stream_tool_calls(assistant_message, messages)
|
|
142
|
-
yield from self.handle_post_tool_call_messages_stream(messages=messages)
|
|
143
|
-
logger.debug("---------- Ollama OllamaHermes Response End ----------")
|
|
144
|
-
|
|
145
|
-
async def aresponse_stream(self, messages: List[Message]) -> Any:
|
|
146
|
-
"""
|
|
147
|
-
Generate an asynchronous streaming response from Ollama.
|
|
148
|
-
|
|
149
|
-
Args:
|
|
150
|
-
messages (List[Message]): A list of messages.
|
|
151
|
-
|
|
152
|
-
Returns:
|
|
153
|
-
Any: An asynchronous iterator of the model responses.
|
|
154
|
-
"""
|
|
155
|
-
logger.debug("---------- Ollama OllamaHermes Async Response Start ----------")
|
|
156
|
-
self._log_messages(messages)
|
|
157
|
-
message_data = MessageData()
|
|
158
|
-
metrics: Metrics = Metrics()
|
|
159
|
-
|
|
160
|
-
# -*- Generate response
|
|
161
|
-
metrics.start_response_timer()
|
|
162
|
-
async for response in self.ainvoke_stream(messages=messages):
|
|
163
|
-
message_data.response_message = response.get("message", {})
|
|
164
|
-
if message_data.response_message:
|
|
165
|
-
metrics.output_tokens += 1
|
|
166
|
-
if metrics.output_tokens == 1:
|
|
167
|
-
metrics.time_to_first_token = metrics.response_timer.elapsed
|
|
168
|
-
|
|
169
|
-
message_data.response_content_chunk = message_data.response_message.get("content", "").strip("`")
|
|
170
|
-
message_data.response_content_chunk = message_data.response_message.get("content", "").strip(
|
|
171
|
-
"<|end_of_text|>"
|
|
172
|
-
)
|
|
173
|
-
message_data.response_content_chunk = message_data.response_message.get("content", "").strip(
|
|
174
|
-
"<|begin_of_text|>"
|
|
175
|
-
)
|
|
176
|
-
|
|
177
|
-
if message_data.response_content_chunk:
|
|
178
|
-
if message_data.response_content_chunk.strip().startswith("</tool_call>"):
|
|
179
|
-
message_data.end_tool_call = True
|
|
180
|
-
if message_data.in_tool_call:
|
|
181
|
-
message_data.tool_call_chunk, message_data.in_tool_call = self.handle_tool_call_chunk(
|
|
182
|
-
message_data.response_content_chunk, message_data.tool_call_chunk, message_data
|
|
183
|
-
)
|
|
184
|
-
elif message_data.response_content_chunk.strip().startswith("<tool_call>"):
|
|
185
|
-
message_data.in_tool_call = True
|
|
186
|
-
else:
|
|
187
|
-
yield ModelResponse(content=message_data.response_content_chunk)
|
|
188
|
-
message_data.response_content += message_data.response_content_chunk
|
|
189
|
-
|
|
190
|
-
if response.get("done"):
|
|
191
|
-
message_data.response_usage = response
|
|
192
|
-
metrics.stop_response_timer()
|
|
193
|
-
|
|
194
|
-
# Format tool calls
|
|
195
|
-
message_data = self._format_tool_calls(message_data)
|
|
196
|
-
|
|
197
|
-
# -*- Create assistant message
|
|
198
|
-
assistant_message = Message(role="assistant", content=message_data.response_content)
|
|
199
|
-
|
|
200
|
-
if len(message_data.tool_calls) > 0:
|
|
201
|
-
assistant_message.tool_calls = message_data.tool_calls
|
|
202
|
-
|
|
203
|
-
# -*- Update usage metrics
|
|
204
|
-
self.update_usage_metrics(
|
|
205
|
-
assistant_message=assistant_message, metrics=metrics, response=message_data.response_usage
|
|
206
|
-
)
|
|
207
|
-
|
|
208
|
-
# -*- Add assistant message to messages
|
|
209
|
-
messages.append(assistant_message)
|
|
210
|
-
|
|
211
|
-
# -*- Log response and metrics
|
|
212
|
-
assistant_message.log()
|
|
213
|
-
metrics.log()
|
|
214
|
-
|
|
215
|
-
# -*- Handle tool calls
|
|
216
|
-
if assistant_message.tool_calls is not None and len(assistant_message.tool_calls) > 0:
|
|
217
|
-
for tool_call_response in self.handle_stream_tool_calls(assistant_message, messages):
|
|
218
|
-
yield tool_call_response
|
|
219
|
-
async for post_tool_call_response in self.ahandle_post_tool_call_messages_stream(messages=messages):
|
|
220
|
-
yield post_tool_call_response
|
|
221
|
-
logger.debug("---------- Ollama OllamaHermes Async Response End ----------")
|
agno/models/ollama/tools.py
DELETED
|
@@ -1,362 +0,0 @@
|
|
|
1
|
-
import json
|
|
2
|
-
from dataclasses import dataclass, field
|
|
3
|
-
from textwrap import dedent
|
|
4
|
-
from typing import Any, Dict, Iterator, List, Mapping, Optional
|
|
5
|
-
|
|
6
|
-
from agno.models.message import Message
|
|
7
|
-
from agno.models.ollama.chat import Metrics, Ollama
|
|
8
|
-
from agno.models.response import ModelResponse
|
|
9
|
-
from agno.utils.log import logger
|
|
10
|
-
from agno.utils.tools import (
|
|
11
|
-
extract_tool_call_from_string,
|
|
12
|
-
remove_tool_calls_from_string,
|
|
13
|
-
)
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
@dataclass
|
|
17
|
-
class MessageData:
|
|
18
|
-
response_role: Optional[str] = None
|
|
19
|
-
response_message: Optional[Dict[str, Any]] = None
|
|
20
|
-
response_content: Any = ""
|
|
21
|
-
response_content_chunk: str = ""
|
|
22
|
-
tool_calls: List[Dict[str, Any]] = field(default_factory=list)
|
|
23
|
-
response_usage: Optional[Mapping[str, Any]] = None
|
|
24
|
-
response_is_tool_call = False
|
|
25
|
-
is_closing_tool_call_tag = False
|
|
26
|
-
tool_calls_counter = 0
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
@dataclass
|
|
30
|
-
class OllamaTools(Ollama):
|
|
31
|
-
"""
|
|
32
|
-
An Ollama class that uses XML tags for tool calls.
|
|
33
|
-
|
|
34
|
-
For more information, see: https://github.com/ollama/ollama/blob/main/docs/api.md
|
|
35
|
-
"""
|
|
36
|
-
|
|
37
|
-
id: str = "llama3.2"
|
|
38
|
-
name: str = "OllamaTools"
|
|
39
|
-
provider: str = "Ollama"
|
|
40
|
-
|
|
41
|
-
@property
|
|
42
|
-
def request_kwargs(self) -> Dict[str, Any]:
|
|
43
|
-
"""
|
|
44
|
-
Returns keyword arguments for API requests.
|
|
45
|
-
|
|
46
|
-
Returns:
|
|
47
|
-
Dict[str, Any]: The API kwargs for the model.
|
|
48
|
-
"""
|
|
49
|
-
request_params: Dict[str, Any] = {}
|
|
50
|
-
if self.format is not None:
|
|
51
|
-
request_params["format"] = self.format
|
|
52
|
-
if self.options is not None:
|
|
53
|
-
request_params["options"] = self.options
|
|
54
|
-
if self.keep_alive is not None:
|
|
55
|
-
request_params["keep_alive"] = self.keep_alive
|
|
56
|
-
if self.request_params is not None:
|
|
57
|
-
request_params.update(self.request_params)
|
|
58
|
-
return request_params
|
|
59
|
-
|
|
60
|
-
def create_assistant_message(self, response: Mapping[str, Any], metrics: Metrics) -> Message:
|
|
61
|
-
"""
|
|
62
|
-
Create an assistant message from the response.
|
|
63
|
-
|
|
64
|
-
Args:
|
|
65
|
-
response: The response from Ollama.
|
|
66
|
-
metrics: The metrics for this response.
|
|
67
|
-
|
|
68
|
-
Returns:
|
|
69
|
-
Message: The assistant message.
|
|
70
|
-
"""
|
|
71
|
-
message_data = MessageData()
|
|
72
|
-
|
|
73
|
-
message_data.response_message = response.get("message")
|
|
74
|
-
if message_data.response_message:
|
|
75
|
-
message_data.response_content = message_data.response_message.get("content")
|
|
76
|
-
message_data.response_role = message_data.response_message.get("role")
|
|
77
|
-
|
|
78
|
-
assistant_message = Message(
|
|
79
|
-
role=message_data.response_role or "assistant",
|
|
80
|
-
content=message_data.response_content,
|
|
81
|
-
)
|
|
82
|
-
# -*- Check if the response contains a tool call
|
|
83
|
-
try:
|
|
84
|
-
if message_data.response_content is not None:
|
|
85
|
-
if "<tool_call>" in message_data.response_content and "</tool_call>" in message_data.response_content:
|
|
86
|
-
# Break the response into tool calls
|
|
87
|
-
tool_call_responses = message_data.response_content.split("</tool_call>")
|
|
88
|
-
for tool_call_response in tool_call_responses:
|
|
89
|
-
# Add back the closing tag if this is not the last tool call
|
|
90
|
-
if tool_call_response != tool_call_responses[-1]:
|
|
91
|
-
tool_call_response += "</tool_call>"
|
|
92
|
-
|
|
93
|
-
if "<tool_call>" in tool_call_response and "</tool_call>" in tool_call_response:
|
|
94
|
-
# Extract tool call string from response
|
|
95
|
-
tool_call_content = extract_tool_call_from_string(tool_call_response)
|
|
96
|
-
# Convert the extracted string to a dictionary
|
|
97
|
-
try:
|
|
98
|
-
tool_call_dict = json.loads(tool_call_content)
|
|
99
|
-
except json.JSONDecodeError:
|
|
100
|
-
raise ValueError(f"Could not parse tool call from: {tool_call_content}")
|
|
101
|
-
|
|
102
|
-
tool_call_name = tool_call_dict.get("name")
|
|
103
|
-
tool_call_args = tool_call_dict.get("arguments")
|
|
104
|
-
function_def = {"name": tool_call_name}
|
|
105
|
-
if tool_call_args is not None:
|
|
106
|
-
function_def["arguments"] = json.dumps(tool_call_args)
|
|
107
|
-
message_data.tool_calls.append(
|
|
108
|
-
{
|
|
109
|
-
"type": "function",
|
|
110
|
-
"function": function_def,
|
|
111
|
-
}
|
|
112
|
-
)
|
|
113
|
-
except Exception as e:
|
|
114
|
-
logger.warning(e)
|
|
115
|
-
pass
|
|
116
|
-
|
|
117
|
-
if message_data.tool_calls is not None:
|
|
118
|
-
assistant_message.tool_calls = message_data.tool_calls
|
|
119
|
-
|
|
120
|
-
# -*- Update metrics
|
|
121
|
-
self.update_usage_metrics(assistant_message=assistant_message, metrics=metrics, response=response)
|
|
122
|
-
return assistant_message
|
|
123
|
-
|
|
124
|
-
def format_function_call_results(self, function_call_results: List[Message], messages: List[Message]) -> None:
|
|
125
|
-
"""
|
|
126
|
-
Format the function call results and append them to the messages.
|
|
127
|
-
|
|
128
|
-
Args:
|
|
129
|
-
function_call_results (List[Message]): The list of function call results.
|
|
130
|
-
messages (List[Message]): The list of messages.
|
|
131
|
-
"""
|
|
132
|
-
if len(function_call_results) > 0:
|
|
133
|
-
for _fc_message in function_call_results:
|
|
134
|
-
_fc_message.content = (
|
|
135
|
-
"<tool_response>\n"
|
|
136
|
-
+ json.dumps({"name": _fc_message.tool_name, "content": _fc_message.content})
|
|
137
|
-
+ "\n</tool_response>"
|
|
138
|
-
)
|
|
139
|
-
messages.append(_fc_message)
|
|
140
|
-
|
|
141
|
-
def handle_tool_calls(
|
|
142
|
-
self,
|
|
143
|
-
assistant_message: Message,
|
|
144
|
-
messages: List[Message],
|
|
145
|
-
model_response: ModelResponse,
|
|
146
|
-
) -> Optional[ModelResponse]:
|
|
147
|
-
"""
|
|
148
|
-
Handle tool calls in the assistant message.
|
|
149
|
-
|
|
150
|
-
Args:
|
|
151
|
-
assistant_message (Message): The assistant message.
|
|
152
|
-
messages (List[Message]): The list of messages.
|
|
153
|
-
model_response (ModelResponse): The model response.
|
|
154
|
-
|
|
155
|
-
Returns:
|
|
156
|
-
Optional[ModelResponse]: The model response.
|
|
157
|
-
"""
|
|
158
|
-
if assistant_message.tool_calls is not None and len(assistant_message.tool_calls) > 0:
|
|
159
|
-
model_response.content = str(remove_tool_calls_from_string(assistant_message.get_content_string()))
|
|
160
|
-
model_response.content += "\n\n"
|
|
161
|
-
function_calls_to_run = self._get_function_calls_to_run(assistant_message, messages)
|
|
162
|
-
function_call_results: List[Message] = []
|
|
163
|
-
|
|
164
|
-
if self.show_tool_calls:
|
|
165
|
-
if len(function_calls_to_run) == 1:
|
|
166
|
-
model_response.content += f" - Running: {function_calls_to_run[0].get_call_str()}\n\n"
|
|
167
|
-
elif len(function_calls_to_run) > 1:
|
|
168
|
-
model_response.content += "Running:"
|
|
169
|
-
for _f in function_calls_to_run:
|
|
170
|
-
model_response.content += f"\n - {_f.get_call_str()}"
|
|
171
|
-
model_response.content += "\n\n"
|
|
172
|
-
|
|
173
|
-
for _ in self.run_function_calls(
|
|
174
|
-
function_calls=function_calls_to_run,
|
|
175
|
-
function_call_results=function_call_results,
|
|
176
|
-
):
|
|
177
|
-
pass
|
|
178
|
-
|
|
179
|
-
self.format_function_call_results(function_call_results, messages)
|
|
180
|
-
|
|
181
|
-
return model_response
|
|
182
|
-
return None
|
|
183
|
-
|
|
184
|
-
def response_stream(self, messages: List[Message]) -> Iterator[ModelResponse]:
|
|
185
|
-
"""
|
|
186
|
-
Generate a streaming response from OllamaTools.
|
|
187
|
-
|
|
188
|
-
Args:
|
|
189
|
-
messages (List[Message]): A list of messages.
|
|
190
|
-
|
|
191
|
-
Returns:
|
|
192
|
-
Iterator[ModelResponse]: An iterator of the model responses.
|
|
193
|
-
"""
|
|
194
|
-
logger.debug("---------- Ollama Response Start ----------")
|
|
195
|
-
self._log_messages(messages)
|
|
196
|
-
message_data = MessageData()
|
|
197
|
-
metrics: Metrics = Metrics()
|
|
198
|
-
|
|
199
|
-
# -*- Generate response
|
|
200
|
-
metrics.start_response_timer()
|
|
201
|
-
for response in self.invoke_stream(messages=messages):
|
|
202
|
-
# Parse response
|
|
203
|
-
message_data.response_message = response.get("message", {})
|
|
204
|
-
if message_data.response_message:
|
|
205
|
-
metrics.output_tokens += 1
|
|
206
|
-
if metrics.output_tokens == 1:
|
|
207
|
-
metrics.time_to_first_token = metrics.response_timer.elapsed
|
|
208
|
-
|
|
209
|
-
if message_data.response_message:
|
|
210
|
-
message_data.response_content_chunk = message_data.response_message.get("content", "")
|
|
211
|
-
|
|
212
|
-
# Add response content to assistant message
|
|
213
|
-
if message_data.response_content_chunk is not None:
|
|
214
|
-
message_data.response_content += message_data.response_content_chunk
|
|
215
|
-
|
|
216
|
-
# Detect if response is a tool call
|
|
217
|
-
# If the response is a tool call, it will start a <tool token
|
|
218
|
-
if not message_data.response_is_tool_call and "<tool" in message_data.response_content_chunk:
|
|
219
|
-
message_data.response_is_tool_call = True
|
|
220
|
-
# logger.debug(f"Response is tool call: {message_data.response_is_tool_call}")
|
|
221
|
-
|
|
222
|
-
# If response is a tool call, count the number of tool calls
|
|
223
|
-
if message_data.response_is_tool_call:
|
|
224
|
-
# If the response is an opening tool call tag, increment the tool call counter
|
|
225
|
-
if "<tool" in message_data.response_content_chunk:
|
|
226
|
-
message_data.tool_calls_counter += 1
|
|
227
|
-
|
|
228
|
-
# If the response is a closing tool call tag, decrement the tool call counter
|
|
229
|
-
if message_data.response_content.strip().endswith("</tool_call>"):
|
|
230
|
-
message_data.tool_calls_counter -= 1
|
|
231
|
-
|
|
232
|
-
# If the response is a closing tool call tag and the tool call counter is 0,
|
|
233
|
-
# tool call response is complete
|
|
234
|
-
if message_data.tool_calls_counter == 0 and message_data.response_content_chunk.strip().endswith(">"):
|
|
235
|
-
message_data.response_is_tool_call = False
|
|
236
|
-
# logger.debug(f"Response is tool call: {message_data.response_is_tool_call}")
|
|
237
|
-
message_data.is_closing_tool_call_tag = True
|
|
238
|
-
|
|
239
|
-
# Yield content if not a tool call and content is not None
|
|
240
|
-
if not message_data.response_is_tool_call and message_data.response_content_chunk is not None:
|
|
241
|
-
if message_data.is_closing_tool_call_tag and message_data.response_content_chunk.strip().endswith(">"):
|
|
242
|
-
message_data.is_closing_tool_call_tag = False
|
|
243
|
-
continue
|
|
244
|
-
|
|
245
|
-
yield ModelResponse(content=message_data.response_content_chunk)
|
|
246
|
-
|
|
247
|
-
if response.get("done"):
|
|
248
|
-
message_data.response_usage = response
|
|
249
|
-
metrics.stop_response_timer()
|
|
250
|
-
|
|
251
|
-
# -*- Create assistant message
|
|
252
|
-
assistant_message = Message(role="assistant", content=message_data.response_content)
|
|
253
|
-
|
|
254
|
-
# -*- Parse tool calls from the assistant message content
|
|
255
|
-
try:
|
|
256
|
-
if "<tool_call>" in message_data.response_content and "</tool_call>" in message_data.response_content:
|
|
257
|
-
# Break the response into tool calls
|
|
258
|
-
tool_call_responses = message_data.response_content.split("</tool_call>")
|
|
259
|
-
for tool_call_response in tool_call_responses:
|
|
260
|
-
# Add back the closing tag if this is not the last tool call
|
|
261
|
-
if tool_call_response != tool_call_responses[-1]:
|
|
262
|
-
tool_call_response += "</tool_call>"
|
|
263
|
-
|
|
264
|
-
if "<tool_call>" in tool_call_response and "</tool_call>" in tool_call_response:
|
|
265
|
-
# Extract tool call string from response
|
|
266
|
-
tool_call_content = extract_tool_call_from_string(tool_call_response)
|
|
267
|
-
# Convert the extracted string to a dictionary
|
|
268
|
-
try:
|
|
269
|
-
tool_call_dict = json.loads(tool_call_content)
|
|
270
|
-
except json.JSONDecodeError:
|
|
271
|
-
raise ValueError(f"Could not parse tool call from: {tool_call_content}")
|
|
272
|
-
|
|
273
|
-
tool_call_name = tool_call_dict.get("name")
|
|
274
|
-
tool_call_args = tool_call_dict.get("arguments")
|
|
275
|
-
function_def = {"name": tool_call_name}
|
|
276
|
-
if tool_call_args is not None:
|
|
277
|
-
function_def["arguments"] = json.dumps(tool_call_args)
|
|
278
|
-
message_data.tool_calls.append(
|
|
279
|
-
{
|
|
280
|
-
"type": "function",
|
|
281
|
-
"function": function_def,
|
|
282
|
-
}
|
|
283
|
-
)
|
|
284
|
-
|
|
285
|
-
except Exception as e:
|
|
286
|
-
yield ModelResponse(content=f"Error parsing tool call: {e}")
|
|
287
|
-
logger.warning(e)
|
|
288
|
-
pass
|
|
289
|
-
|
|
290
|
-
if len(message_data.tool_calls) > 0:
|
|
291
|
-
assistant_message.tool_calls = message_data.tool_calls
|
|
292
|
-
|
|
293
|
-
# -*- Update usage metrics
|
|
294
|
-
self.update_usage_metrics(
|
|
295
|
-
assistant_message=assistant_message, metrics=metrics, response=message_data.response_usage
|
|
296
|
-
)
|
|
297
|
-
|
|
298
|
-
# -*- Add assistant message to messages
|
|
299
|
-
messages.append(assistant_message)
|
|
300
|
-
|
|
301
|
-
# -*- Log response and metrics
|
|
302
|
-
assistant_message.log()
|
|
303
|
-
metrics.log()
|
|
304
|
-
|
|
305
|
-
# -*- Handle tool calls
|
|
306
|
-
if assistant_message.tool_calls is not None and len(assistant_message.tool_calls) > 0:
|
|
307
|
-
yield from self.handle_stream_tool_calls(assistant_message, messages)
|
|
308
|
-
yield from self.handle_post_tool_call_messages_stream(messages=messages)
|
|
309
|
-
logger.debug("---------- Ollama Response End ----------")
|
|
310
|
-
|
|
311
|
-
def get_instructions_to_generate_tool_calls(self) -> List[str]:
|
|
312
|
-
if self._functions is not None:
|
|
313
|
-
return [
|
|
314
|
-
"At the very first turn you don't have <tool_results> so you shouldn't not make up the results.",
|
|
315
|
-
"To respond to the users message, you can use only one tool at a time.",
|
|
316
|
-
"When using a tool, only respond with the tool call. Nothing else. Do not add any additional notes, explanations or white space.",
|
|
317
|
-
"Do not stop calling functions until the task has been accomplished or you've reached max iteration of 10.",
|
|
318
|
-
]
|
|
319
|
-
return []
|
|
320
|
-
|
|
321
|
-
def get_tool_call_prompt(self) -> Optional[str]:
|
|
322
|
-
if self._functions is not None and len(self._functions) > 0:
|
|
323
|
-
tool_call_prompt = dedent(
|
|
324
|
-
"""\
|
|
325
|
-
You are a function calling AI model with self-recursion.
|
|
326
|
-
You are provided with function signatures within <tools></tools> XML tags.
|
|
327
|
-
You may use agentic frameworks for reasoning and planning to help with user query.
|
|
328
|
-
Please call a function and wait for function results to be provided to you in the next iteration.
|
|
329
|
-
Don't make assumptions about what values to plug into functions.
|
|
330
|
-
When you call a function, don't add any additional notes, explanations or white space.
|
|
331
|
-
Once you have called a function, results will be provided to you within <tool_response></tool_response> XML tags.
|
|
332
|
-
Do not make assumptions about tool results if <tool_response> XML tags are not present since the function is not yet executed.
|
|
333
|
-
Analyze the results once you get them and call another function if needed.
|
|
334
|
-
Your final response should directly answer the user query with an analysis or summary of the results of function calls.
|
|
335
|
-
"""
|
|
336
|
-
)
|
|
337
|
-
tool_call_prompt += "\nHere are the available tools:"
|
|
338
|
-
tool_call_prompt += "\n<tools>\n"
|
|
339
|
-
tool_definitions: List[str] = []
|
|
340
|
-
for _f_name, _function in self._functions.items():
|
|
341
|
-
_function_def = _function.get_definition_for_prompt()
|
|
342
|
-
if _function_def:
|
|
343
|
-
tool_definitions.append(_function_def)
|
|
344
|
-
tool_call_prompt += "\n".join(tool_definitions)
|
|
345
|
-
tool_call_prompt += "\n</tools>\n\n"
|
|
346
|
-
tool_call_prompt += dedent(
|
|
347
|
-
"""\
|
|
348
|
-
Use the following pydantic model json schema for each tool call you will make: {'title': 'FunctionCall', 'type': 'object', 'properties': {'arguments': {'title': 'Arguments', 'type': 'object'}, 'name': {'title': 'Name', 'type': 'string'}}, 'required': ['arguments', 'name']}
|
|
349
|
-
For each function call return a json object with function name and arguments within <tool_call></tool_call> XML tags as follows:
|
|
350
|
-
<tool_call>
|
|
351
|
-
{"arguments": <args-dict>, "name": <function-name>}
|
|
352
|
-
</tool_call>\n
|
|
353
|
-
"""
|
|
354
|
-
)
|
|
355
|
-
return tool_call_prompt
|
|
356
|
-
return None
|
|
357
|
-
|
|
358
|
-
def get_system_message_for_model(self) -> Optional[str]:
|
|
359
|
-
return self.get_tool_call_prompt()
|
|
360
|
-
|
|
361
|
-
def get_instructions_for_model(self) -> Optional[List[str]]:
|
|
362
|
-
return self.get_instructions_to_generate_tool_calls()
|