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/knowledge/s3/base.py
DELETED
|
@@ -1,60 +0,0 @@
|
|
|
1
|
-
from typing import Iterator, List, Optional
|
|
2
|
-
|
|
3
|
-
from agno.aws.resource.s3.bucket import S3Bucket # type: ignore
|
|
4
|
-
from agno.aws.resource.s3.object import S3Object # type: ignore
|
|
5
|
-
from agno.document import Document
|
|
6
|
-
from agno.knowledge.agent import AgentKnowledge
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
class S3KnowledgeBase(AgentKnowledge):
|
|
10
|
-
# Provide either bucket or bucket_name
|
|
11
|
-
bucket: Optional[S3Bucket] = None
|
|
12
|
-
bucket_name: Optional[str] = None
|
|
13
|
-
|
|
14
|
-
# Provide either object or key
|
|
15
|
-
key: Optional[str] = None
|
|
16
|
-
object: Optional[S3Object] = None
|
|
17
|
-
|
|
18
|
-
# Filter objects by prefix
|
|
19
|
-
# Ignored if object or key is provided
|
|
20
|
-
prefix: Optional[str] = None
|
|
21
|
-
|
|
22
|
-
@property
|
|
23
|
-
def document_lists(self) -> Iterator[List[Document]]:
|
|
24
|
-
raise NotImplementedError
|
|
25
|
-
|
|
26
|
-
@property
|
|
27
|
-
def s3_objects(self) -> List[S3Object]:
|
|
28
|
-
"""Iterate over PDFs in a s3 bucket and yield lists of documents.
|
|
29
|
-
Each object yielded by the iterator is a list of documents.
|
|
30
|
-
|
|
31
|
-
Returns:
|
|
32
|
-
Iterator[List[Document]]: Iterator yielding list of documents
|
|
33
|
-
"""
|
|
34
|
-
|
|
35
|
-
s3_objects_to_read: List[S3Object] = []
|
|
36
|
-
|
|
37
|
-
if self.bucket is None and self.bucket_name is None:
|
|
38
|
-
raise ValueError("No bucket or bucket_name provided")
|
|
39
|
-
|
|
40
|
-
if self.bucket is not None and self.bucket_name is not None:
|
|
41
|
-
raise ValueError("Provide either bucket or bucket_name")
|
|
42
|
-
|
|
43
|
-
if self.object is not None and self.key is not None:
|
|
44
|
-
raise ValueError("Provide either object or key")
|
|
45
|
-
|
|
46
|
-
if self.bucket_name is not None:
|
|
47
|
-
self.bucket = S3Bucket(name=self.bucket_name)
|
|
48
|
-
|
|
49
|
-
if self.bucket is not None:
|
|
50
|
-
if self.key is not None:
|
|
51
|
-
_object = S3Object(bucket_name=self.bucket.name, name=self.key)
|
|
52
|
-
s3_objects_to_read.append(_object)
|
|
53
|
-
elif self.object is not None:
|
|
54
|
-
s3_objects_to_read.append(self.object)
|
|
55
|
-
elif self.prefix is not None:
|
|
56
|
-
s3_objects_to_read.extend(self.bucket.get_objects(prefix=self.prefix))
|
|
57
|
-
else:
|
|
58
|
-
s3_objects_to_read.extend(self.bucket.get_objects())
|
|
59
|
-
|
|
60
|
-
return s3_objects_to_read
|
agno/knowledge/s3/pdf.py
DELETED
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
from typing import Iterator, List
|
|
2
|
-
|
|
3
|
-
from agno.document import Document
|
|
4
|
-
from agno.document.reader.s3.pdf_reader import S3PDFReader
|
|
5
|
-
from agno.knowledge.s3.base import S3KnowledgeBase
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
class S3PDFKnowledgeBase(S3KnowledgeBase):
|
|
9
|
-
reader: S3PDFReader = S3PDFReader()
|
|
10
|
-
|
|
11
|
-
@property
|
|
12
|
-
def document_lists(self) -> Iterator[List[Document]]:
|
|
13
|
-
"""Iterate over PDFs in a s3 bucket and yield lists of documents.
|
|
14
|
-
Each object yielded by the iterator is a list of documents.
|
|
15
|
-
|
|
16
|
-
Returns:
|
|
17
|
-
Iterator[List[Document]]: Iterator yielding list of documents
|
|
18
|
-
"""
|
|
19
|
-
for s3_object in self.s3_objects:
|
|
20
|
-
if s3_object.name.endswith(".pdf"):
|
|
21
|
-
yield self.reader.read(s3_object=s3_object)
|
agno/knowledge/s3/text.py
DELETED
|
@@ -1,23 +0,0 @@
|
|
|
1
|
-
from typing import Iterator, List
|
|
2
|
-
|
|
3
|
-
from agno.document import Document
|
|
4
|
-
from agno.document.reader.s3.text_reader import S3TextReader
|
|
5
|
-
from agno.knowledge.s3.base import S3KnowledgeBase
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
class S3TextKnowledgeBase(S3KnowledgeBase):
|
|
9
|
-
formats: List[str] = [".doc", ".docx"]
|
|
10
|
-
reader: S3TextReader = S3TextReader()
|
|
11
|
-
|
|
12
|
-
@property
|
|
13
|
-
def document_lists(self) -> Iterator[List[Document]]:
|
|
14
|
-
"""Iterate over text files in a s3 bucket and yield lists of documents.
|
|
15
|
-
Each object yielded by the iterator is a list of documents.
|
|
16
|
-
|
|
17
|
-
Returns:
|
|
18
|
-
Iterator[List[Document]]: Iterator yielding list of documents
|
|
19
|
-
"""
|
|
20
|
-
|
|
21
|
-
for s3_object in self.s3_objects:
|
|
22
|
-
if s3_object.name.endswith(tuple(self.formats)):
|
|
23
|
-
yield self.reader.read(s3_object=s3_object)
|
agno/knowledge/text.py
DELETED
|
@@ -1,30 +0,0 @@
|
|
|
1
|
-
from pathlib import Path
|
|
2
|
-
from typing import Iterator, List, Union
|
|
3
|
-
|
|
4
|
-
from agno.document import Document
|
|
5
|
-
from agno.document.reader.text_reader import TextReader
|
|
6
|
-
from agno.knowledge.agent import AgentKnowledge
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
class TextKnowledgeBase(AgentKnowledge):
|
|
10
|
-
path: Union[str, Path]
|
|
11
|
-
formats: List[str] = [".txt"]
|
|
12
|
-
reader: TextReader = TextReader()
|
|
13
|
-
|
|
14
|
-
@property
|
|
15
|
-
def document_lists(self) -> Iterator[List[Document]]:
|
|
16
|
-
"""Iterate over text files and yield lists of documents.
|
|
17
|
-
Each object yielded by the iterator is a list of documents.
|
|
18
|
-
|
|
19
|
-
Returns:
|
|
20
|
-
Iterator[List[Document]]: Iterator yielding list of documents
|
|
21
|
-
"""
|
|
22
|
-
|
|
23
|
-
_file_path: Path = Path(self.path) if isinstance(self.path, str) else self.path
|
|
24
|
-
|
|
25
|
-
if _file_path.exists() and _file_path.is_dir():
|
|
26
|
-
for _file in _file_path.glob("**/*"):
|
|
27
|
-
if _file.suffix in self.formats:
|
|
28
|
-
yield self.reader.read(file=_file)
|
|
29
|
-
elif _file_path.exists() and _file_path.is_file() and _file_path.suffix in self.formats:
|
|
30
|
-
yield self.reader.read(file=_file_path)
|
agno/knowledge/website.py
DELETED
|
@@ -1,88 +0,0 @@
|
|
|
1
|
-
from typing import Any, Dict, Iterator, List, Optional
|
|
2
|
-
|
|
3
|
-
from pydantic import model_validator
|
|
4
|
-
|
|
5
|
-
from agno.document import Document
|
|
6
|
-
from agno.document.reader.website_reader import WebsiteReader
|
|
7
|
-
from agno.knowledge.agent import AgentKnowledge
|
|
8
|
-
from agno.utils.log import logger
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
class WebsiteKnowledgeBase(AgentKnowledge):
|
|
12
|
-
urls: List[str] = []
|
|
13
|
-
reader: Optional[WebsiteReader] = None
|
|
14
|
-
|
|
15
|
-
# WebsiteReader parameters
|
|
16
|
-
max_depth: int = 3
|
|
17
|
-
max_links: int = 10
|
|
18
|
-
|
|
19
|
-
@model_validator(mode="after")
|
|
20
|
-
def set_reader(self) -> "WebsiteKnowledgeBase":
|
|
21
|
-
if self.reader is None:
|
|
22
|
-
self.reader = WebsiteReader(max_depth=self.max_depth, max_links=self.max_links)
|
|
23
|
-
return self
|
|
24
|
-
|
|
25
|
-
@property
|
|
26
|
-
def document_lists(self) -> Iterator[List[Document]]:
|
|
27
|
-
"""Iterate over urls and yield lists of documents.
|
|
28
|
-
Each object yielded by the iterator is a list of documents.
|
|
29
|
-
|
|
30
|
-
Returns:
|
|
31
|
-
Iterator[List[Document]]: Iterator yielding list of documents
|
|
32
|
-
"""
|
|
33
|
-
if self.reader is not None:
|
|
34
|
-
for _url in self.urls:
|
|
35
|
-
yield self.reader.read(url=_url)
|
|
36
|
-
|
|
37
|
-
def load(
|
|
38
|
-
self,
|
|
39
|
-
recreate: bool = False,
|
|
40
|
-
upsert: bool = True,
|
|
41
|
-
skip_existing: bool = True,
|
|
42
|
-
filters: Optional[Dict[str, Any]] = None,
|
|
43
|
-
) -> None:
|
|
44
|
-
"""Load the website contents to the vector db"""
|
|
45
|
-
|
|
46
|
-
if self.vector_db is None:
|
|
47
|
-
logger.warning("No vector db provided")
|
|
48
|
-
return
|
|
49
|
-
|
|
50
|
-
if self.reader is None:
|
|
51
|
-
logger.warning("No reader provided")
|
|
52
|
-
return
|
|
53
|
-
|
|
54
|
-
if recreate:
|
|
55
|
-
logger.debug("Dropping collection")
|
|
56
|
-
self.vector_db.drop()
|
|
57
|
-
|
|
58
|
-
logger.debug("Creating collection")
|
|
59
|
-
self.vector_db.create()
|
|
60
|
-
|
|
61
|
-
logger.info("Loading knowledge base")
|
|
62
|
-
num_documents = 0
|
|
63
|
-
|
|
64
|
-
# Given that the crawler needs to parse the URL before existence can be checked
|
|
65
|
-
# We check if the website url exists in the vector db if recreate is False
|
|
66
|
-
urls_to_read = self.urls.copy()
|
|
67
|
-
if not recreate:
|
|
68
|
-
for url in urls_to_read:
|
|
69
|
-
logger.debug(f"Checking if {url} exists in the vector db")
|
|
70
|
-
if self.vector_db.name_exists(name=url):
|
|
71
|
-
logger.debug(f"Skipping {url} as it exists in the vector db")
|
|
72
|
-
urls_to_read.remove(url)
|
|
73
|
-
|
|
74
|
-
for url in urls_to_read:
|
|
75
|
-
document_list = self.reader.read(url=url)
|
|
76
|
-
# Filter out documents which already exist in the vector db
|
|
77
|
-
if not recreate:
|
|
78
|
-
document_list = [document for document in document_list if not self.vector_db.doc_exists(document)]
|
|
79
|
-
if upsert and self.vector_db.upsert_available():
|
|
80
|
-
self.vector_db.upsert(documents=document_list, filters=filters)
|
|
81
|
-
else:
|
|
82
|
-
self.vector_db.insert(documents=document_list, filters=filters)
|
|
83
|
-
num_documents += len(document_list)
|
|
84
|
-
logger.info(f"Loaded {num_documents} documents to knowledge base")
|
|
85
|
-
|
|
86
|
-
if self.optimize_on is not None and num_documents > self.optimize_on:
|
|
87
|
-
logger.debug("Optimizing Vector DB")
|
|
88
|
-
self.vector_db.optimize()
|
agno/knowledge/wikipedia.py
DELETED
|
@@ -1,31 +0,0 @@
|
|
|
1
|
-
from typing import Iterator, List
|
|
2
|
-
|
|
3
|
-
from agno.document import Document
|
|
4
|
-
from agno.knowledge.agent import AgentKnowledge
|
|
5
|
-
|
|
6
|
-
try:
|
|
7
|
-
import wikipedia # noqa: F401
|
|
8
|
-
except ImportError:
|
|
9
|
-
raise ImportError("The `wikipedia` package is not installed. Please install it via `pip install wikipedia`.")
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
class WikipediaKnowledgeBase(AgentKnowledge):
|
|
13
|
-
topics: List[str] = []
|
|
14
|
-
|
|
15
|
-
@property
|
|
16
|
-
def document_lists(self) -> Iterator[List[Document]]:
|
|
17
|
-
"""Iterate over urls and yield lists of documents.
|
|
18
|
-
Each object yielded by the iterator is a list of documents.
|
|
19
|
-
|
|
20
|
-
Returns:
|
|
21
|
-
Iterator[List[Document]]: Iterator yielding list of documents
|
|
22
|
-
"""
|
|
23
|
-
|
|
24
|
-
for topic in self.topics:
|
|
25
|
-
yield [
|
|
26
|
-
Document(
|
|
27
|
-
name=topic,
|
|
28
|
-
meta_data={"topic": topic},
|
|
29
|
-
content=wikipedia.summary(topic),
|
|
30
|
-
)
|
|
31
|
-
]
|
agno/knowledge/youtube.py
DELETED
|
@@ -1,22 +0,0 @@
|
|
|
1
|
-
from typing import Iterator, List
|
|
2
|
-
|
|
3
|
-
from agno.document import Document
|
|
4
|
-
from agno.document.reader.youtube_reader import YouTubeReader
|
|
5
|
-
from agno.knowledge.agent import AgentKnowledge
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
class YouTubeKnowledgeBase(AgentKnowledge):
|
|
9
|
-
urls: List[str] = []
|
|
10
|
-
reader: YouTubeReader = YouTubeReader()
|
|
11
|
-
|
|
12
|
-
@property
|
|
13
|
-
def document_lists(self) -> Iterator[List[Document]]:
|
|
14
|
-
"""Iterate over YouTube URLs and yield lists of documents.
|
|
15
|
-
Each object yielded by the iterator is a list of documents.
|
|
16
|
-
|
|
17
|
-
Returns:
|
|
18
|
-
Iterator[List[Document]]: Iterator yielding list of documents
|
|
19
|
-
"""
|
|
20
|
-
|
|
21
|
-
for url in self.urls:
|
|
22
|
-
yield self.reader.read(video_url=url)
|
agno/memory/agent.py
DELETED
|
@@ -1,392 +0,0 @@
|
|
|
1
|
-
from __future__ import annotations
|
|
2
|
-
|
|
3
|
-
from enum import Enum
|
|
4
|
-
from typing import Any, Dict, List, Optional, Tuple
|
|
5
|
-
|
|
6
|
-
from pydantic import BaseModel, ConfigDict
|
|
7
|
-
|
|
8
|
-
from agno.memory.classifier import MemoryClassifier
|
|
9
|
-
from agno.memory.db import MemoryDb
|
|
10
|
-
from agno.memory.manager import MemoryManager
|
|
11
|
-
from agno.memory.memory import Memory
|
|
12
|
-
from agno.memory.summarizer import MemorySummarizer
|
|
13
|
-
from agno.memory.summary import SessionSummary
|
|
14
|
-
from agno.models.message import Message
|
|
15
|
-
from agno.run.response import RunResponse
|
|
16
|
-
from agno.utils.log import logger
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
class AgentRun(BaseModel):
|
|
20
|
-
message: Optional[Message] = None
|
|
21
|
-
messages: Optional[List[Message]] = None
|
|
22
|
-
response: Optional[RunResponse] = None
|
|
23
|
-
|
|
24
|
-
model_config = ConfigDict(arbitrary_types_allowed=True)
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
class MemoryRetrieval(str, Enum):
|
|
28
|
-
last_n = "last_n"
|
|
29
|
-
first_n = "first_n"
|
|
30
|
-
semantic = "semantic"
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
class AgentMemory(BaseModel):
|
|
34
|
-
# Runs between the user and agent
|
|
35
|
-
runs: List[AgentRun] = []
|
|
36
|
-
# List of messages sent to the model
|
|
37
|
-
messages: List[Message] = []
|
|
38
|
-
update_system_message_on_change: bool = False
|
|
39
|
-
|
|
40
|
-
# Summary of the session
|
|
41
|
-
summary: Optional[SessionSummary] = None
|
|
42
|
-
# Create and store session summaries
|
|
43
|
-
create_session_summary: bool = False
|
|
44
|
-
# Update session summaries after each run
|
|
45
|
-
update_session_summary_after_run: bool = True
|
|
46
|
-
# Summarizer to generate session summaries
|
|
47
|
-
summarizer: Optional[MemorySummarizer] = None
|
|
48
|
-
|
|
49
|
-
# Create and store personalized memories for this user
|
|
50
|
-
create_user_memories: bool = False
|
|
51
|
-
# Update memories for the user after each run
|
|
52
|
-
update_user_memories_after_run: bool = True
|
|
53
|
-
|
|
54
|
-
# MemoryDb to store personalized memories
|
|
55
|
-
db: Optional[MemoryDb] = None
|
|
56
|
-
# User ID for the personalized memories
|
|
57
|
-
user_id: Optional[str] = None
|
|
58
|
-
retrieval: MemoryRetrieval = MemoryRetrieval.last_n
|
|
59
|
-
memories: Optional[List[Memory]] = None
|
|
60
|
-
num_memories: Optional[int] = None
|
|
61
|
-
classifier: Optional[MemoryClassifier] = None
|
|
62
|
-
manager: Optional[MemoryManager] = None
|
|
63
|
-
|
|
64
|
-
# True when memory is being updated
|
|
65
|
-
updating_memory: bool = False
|
|
66
|
-
|
|
67
|
-
model_config = ConfigDict(arbitrary_types_allowed=True)
|
|
68
|
-
|
|
69
|
-
def to_dict(self) -> Dict[str, Any]:
|
|
70
|
-
_memory_dict = self.model_dump(
|
|
71
|
-
exclude_none=True,
|
|
72
|
-
include={
|
|
73
|
-
"runs",
|
|
74
|
-
"messages",
|
|
75
|
-
"update_system_message_on_change",
|
|
76
|
-
"create_session_summary",
|
|
77
|
-
"update_session_summary_after_run",
|
|
78
|
-
"create_user_memories",
|
|
79
|
-
"update_user_memories_after_run",
|
|
80
|
-
"user_id",
|
|
81
|
-
"num_memories",
|
|
82
|
-
},
|
|
83
|
-
)
|
|
84
|
-
# Add summary if it exists
|
|
85
|
-
if self.summary is not None:
|
|
86
|
-
_memory_dict["summary"] = self.summary.to_dict()
|
|
87
|
-
# Add memories if they exist
|
|
88
|
-
if self.memories is not None:
|
|
89
|
-
_memory_dict["memories"] = [memory.to_dict() for memory in self.memories]
|
|
90
|
-
return _memory_dict
|
|
91
|
-
|
|
92
|
-
def add_run(self, agent_run: AgentRun) -> None:
|
|
93
|
-
"""Adds an AgentRun to the runs list."""
|
|
94
|
-
self.runs.append(agent_run)
|
|
95
|
-
logger.debug("Added AgentRun to AgentMemory")
|
|
96
|
-
|
|
97
|
-
def add_system_message(self, message: Message, system_message_role: str = "system") -> None:
|
|
98
|
-
"""Add the system messages to the messages list"""
|
|
99
|
-
# If this is the first run in the session, add the system message to the messages list
|
|
100
|
-
if len(self.messages) == 0:
|
|
101
|
-
if message is not None:
|
|
102
|
-
self.messages.append(message)
|
|
103
|
-
# If there are messages in the memory, check if the system message is already in the memory
|
|
104
|
-
# If it is not, add the system message to the messages list
|
|
105
|
-
# If it is, update the system message if content has changed and update_system_message_on_change is True
|
|
106
|
-
else:
|
|
107
|
-
system_message_index = next((i for i, m in enumerate(self.messages) if m.role == system_message_role), None)
|
|
108
|
-
# Update the system message in memory if content has changed
|
|
109
|
-
if system_message_index is not None:
|
|
110
|
-
if (
|
|
111
|
-
self.messages[system_message_index].content != message.content
|
|
112
|
-
and self.update_system_message_on_change
|
|
113
|
-
):
|
|
114
|
-
logger.info("Updating system message in memory with new content")
|
|
115
|
-
self.messages[system_message_index] = message
|
|
116
|
-
else:
|
|
117
|
-
# Add the system message to the messages list
|
|
118
|
-
self.messages.insert(0, message)
|
|
119
|
-
|
|
120
|
-
def add_message(self, message: Message) -> None:
|
|
121
|
-
"""Add a Message to the messages list."""
|
|
122
|
-
self.messages.append(message)
|
|
123
|
-
logger.debug("Added Message to AgentMemory")
|
|
124
|
-
|
|
125
|
-
def add_messages(self, messages: List[Message]) -> None:
|
|
126
|
-
"""Add a list of messages to the messages list."""
|
|
127
|
-
self.messages.extend(messages)
|
|
128
|
-
logger.debug(f"Added {len(messages)} Messages to AgentMemory")
|
|
129
|
-
|
|
130
|
-
def get_messages(self) -> List[Dict[str, Any]]:
|
|
131
|
-
"""Returns the messages list as a list of dictionaries."""
|
|
132
|
-
return [message.model_dump() for message in self.messages]
|
|
133
|
-
|
|
134
|
-
def get_messages_from_last_n_runs(
|
|
135
|
-
self, last_n: Optional[int] = None, skip_role: Optional[str] = None
|
|
136
|
-
) -> List[Message]:
|
|
137
|
-
"""Returns the messages from the last_n runs
|
|
138
|
-
|
|
139
|
-
Args:
|
|
140
|
-
last_n: The number of runs to return from the end of the conversation.
|
|
141
|
-
skip_role: Skip messages with this role.
|
|
142
|
-
|
|
143
|
-
Returns:
|
|
144
|
-
A list of Messages in the last_n runs.
|
|
145
|
-
"""
|
|
146
|
-
if last_n is None:
|
|
147
|
-
logger.debug("Getting messages from all previous runs")
|
|
148
|
-
messages_from_all_history = []
|
|
149
|
-
for prev_run in self.runs:
|
|
150
|
-
if prev_run.response and prev_run.response.messages:
|
|
151
|
-
if skip_role:
|
|
152
|
-
prev_run_messages = [m for m in prev_run.response.messages if m.role != skip_role]
|
|
153
|
-
else:
|
|
154
|
-
prev_run_messages = prev_run.response.messages
|
|
155
|
-
messages_from_all_history.extend(prev_run_messages)
|
|
156
|
-
logger.debug(f"Messages from previous runs: {len(messages_from_all_history)}")
|
|
157
|
-
return messages_from_all_history
|
|
158
|
-
|
|
159
|
-
logger.debug(f"Getting messages from last {last_n} runs")
|
|
160
|
-
messages_from_last_n_history = []
|
|
161
|
-
for prev_run in self.runs[-last_n:]:
|
|
162
|
-
if prev_run.response and prev_run.response.messages:
|
|
163
|
-
if skip_role:
|
|
164
|
-
prev_run_messages = [m for m in prev_run.response.messages if m.role != skip_role]
|
|
165
|
-
else:
|
|
166
|
-
prev_run_messages = prev_run.response.messages
|
|
167
|
-
messages_from_last_n_history.extend(prev_run_messages)
|
|
168
|
-
logger.debug(f"Messages from last {last_n} runs: {len(messages_from_last_n_history)}")
|
|
169
|
-
return messages_from_last_n_history
|
|
170
|
-
|
|
171
|
-
def get_message_pairs(
|
|
172
|
-
self, user_role: str = "user", assistant_role: Optional[List[str]] = None
|
|
173
|
-
) -> List[Tuple[Message, Message]]:
|
|
174
|
-
"""Returns a list of tuples of (user message, assistant response)."""
|
|
175
|
-
|
|
176
|
-
if assistant_role is None:
|
|
177
|
-
assistant_role = ["assistant", "model", "CHATBOT"]
|
|
178
|
-
|
|
179
|
-
runs_as_message_pairs: List[Tuple[Message, Message]] = []
|
|
180
|
-
for run in self.runs:
|
|
181
|
-
if run.response and run.response.messages:
|
|
182
|
-
user_messages_from_run = None
|
|
183
|
-
assistant_messages_from_run = None
|
|
184
|
-
|
|
185
|
-
# Start from the beginning to look for the user message
|
|
186
|
-
for message in run.response.messages:
|
|
187
|
-
if message.role == user_role:
|
|
188
|
-
user_messages_from_run = message
|
|
189
|
-
break
|
|
190
|
-
|
|
191
|
-
# Start from the end to look for the assistant response
|
|
192
|
-
for message in run.response.messages[::-1]:
|
|
193
|
-
if message.role in assistant_role:
|
|
194
|
-
assistant_messages_from_run = message
|
|
195
|
-
break
|
|
196
|
-
|
|
197
|
-
if user_messages_from_run and assistant_messages_from_run:
|
|
198
|
-
runs_as_message_pairs.append((user_messages_from_run, assistant_messages_from_run))
|
|
199
|
-
return runs_as_message_pairs
|
|
200
|
-
|
|
201
|
-
def get_tool_calls(self, num_calls: Optional[int] = None) -> List[Dict[str, Any]]:
|
|
202
|
-
"""Returns a list of tool calls from the messages"""
|
|
203
|
-
|
|
204
|
-
tool_calls = []
|
|
205
|
-
for message in self.messages[::-1]:
|
|
206
|
-
if message.tool_calls:
|
|
207
|
-
for tool_call in message.tool_calls:
|
|
208
|
-
tool_calls.append(tool_call)
|
|
209
|
-
if num_calls and len(tool_calls) >= num_calls:
|
|
210
|
-
return tool_calls
|
|
211
|
-
return tool_calls
|
|
212
|
-
|
|
213
|
-
def load_user_memories(self) -> None:
|
|
214
|
-
"""Load memories from memory db for this user."""
|
|
215
|
-
|
|
216
|
-
if self.db is None:
|
|
217
|
-
return
|
|
218
|
-
|
|
219
|
-
try:
|
|
220
|
-
if self.retrieval in (MemoryRetrieval.last_n, MemoryRetrieval.first_n):
|
|
221
|
-
memory_rows = self.db.read_memories(
|
|
222
|
-
user_id=self.user_id,
|
|
223
|
-
limit=self.num_memories,
|
|
224
|
-
sort="asc" if self.retrieval == MemoryRetrieval.first_n else "desc",
|
|
225
|
-
)
|
|
226
|
-
else:
|
|
227
|
-
raise NotImplementedError("Semantic retrieval not yet supported.")
|
|
228
|
-
except Exception as e:
|
|
229
|
-
logger.debug(f"Error reading memory: {e}")
|
|
230
|
-
return
|
|
231
|
-
|
|
232
|
-
# Clear the existing memories
|
|
233
|
-
self.memories = []
|
|
234
|
-
|
|
235
|
-
# No memories to load
|
|
236
|
-
if memory_rows is None or len(memory_rows) == 0:
|
|
237
|
-
return
|
|
238
|
-
|
|
239
|
-
for row in memory_rows:
|
|
240
|
-
try:
|
|
241
|
-
self.memories.append(Memory.model_validate(row.memory))
|
|
242
|
-
except Exception as e:
|
|
243
|
-
logger.warning(f"Error loading memory: {e}")
|
|
244
|
-
continue
|
|
245
|
-
|
|
246
|
-
def should_update_memory(self, input: str) -> bool:
|
|
247
|
-
"""Determines if a message should be added to the memory db."""
|
|
248
|
-
from agno.memory.classifier import MemoryClassifier
|
|
249
|
-
|
|
250
|
-
if self.classifier is None:
|
|
251
|
-
self.classifier = MemoryClassifier()
|
|
252
|
-
|
|
253
|
-
self.classifier.existing_memories = self.memories
|
|
254
|
-
classifier_response = self.classifier.run(input)
|
|
255
|
-
if classifier_response == "yes":
|
|
256
|
-
return True
|
|
257
|
-
return False
|
|
258
|
-
|
|
259
|
-
async def ashould_update_memory(self, input: str) -> bool:
|
|
260
|
-
"""Determines if a message should be added to the memory db."""
|
|
261
|
-
from agno.memory.classifier import MemoryClassifier
|
|
262
|
-
|
|
263
|
-
if self.classifier is None:
|
|
264
|
-
self.classifier = MemoryClassifier()
|
|
265
|
-
|
|
266
|
-
self.classifier.existing_memories = self.memories
|
|
267
|
-
classifier_response = await self.classifier.arun(input)
|
|
268
|
-
if classifier_response == "yes":
|
|
269
|
-
return True
|
|
270
|
-
return False
|
|
271
|
-
|
|
272
|
-
def update_memory(self, input: str, force: bool = False) -> Optional[str]:
|
|
273
|
-
"""Creates a memory from a message and adds it to the memory db."""
|
|
274
|
-
from agno.memory.manager import MemoryManager
|
|
275
|
-
|
|
276
|
-
if input is None or not isinstance(input, str):
|
|
277
|
-
return "Invalid message content"
|
|
278
|
-
|
|
279
|
-
if self.db is None:
|
|
280
|
-
logger.warning("MemoryDb not provided.")
|
|
281
|
-
return "Please provide a db to store memories"
|
|
282
|
-
|
|
283
|
-
self.updating_memory = True
|
|
284
|
-
|
|
285
|
-
# Check if this user message should be added to long term memory
|
|
286
|
-
should_update_memory = force or self.should_update_memory(input=input)
|
|
287
|
-
logger.debug(f"Update memory: {should_update_memory}")
|
|
288
|
-
|
|
289
|
-
if not should_update_memory:
|
|
290
|
-
logger.debug("Memory update not required")
|
|
291
|
-
return "Memory update not required"
|
|
292
|
-
|
|
293
|
-
if self.manager is None:
|
|
294
|
-
self.manager = MemoryManager(user_id=self.user_id, db=self.db)
|
|
295
|
-
|
|
296
|
-
else:
|
|
297
|
-
self.manager.db = self.db
|
|
298
|
-
self.manager.user_id = self.user_id
|
|
299
|
-
|
|
300
|
-
response = self.manager.run(input)
|
|
301
|
-
self.load_user_memories()
|
|
302
|
-
self.updating_memory = False
|
|
303
|
-
return response
|
|
304
|
-
|
|
305
|
-
async def aupdate_memory(self, input: str, force: bool = False) -> Optional[str]:
|
|
306
|
-
"""Creates a memory from a message and adds it to the memory db."""
|
|
307
|
-
from agno.memory.manager import MemoryManager
|
|
308
|
-
|
|
309
|
-
if input is None or not isinstance(input, str):
|
|
310
|
-
return "Invalid message content"
|
|
311
|
-
|
|
312
|
-
if self.db is None:
|
|
313
|
-
logger.warning("MemoryDb not provided.")
|
|
314
|
-
return "Please provide a db to store memories"
|
|
315
|
-
|
|
316
|
-
self.updating_memory = True
|
|
317
|
-
|
|
318
|
-
# Check if this user message should be added to long term memory
|
|
319
|
-
should_update_memory = force or await self.ashould_update_memory(input=input)
|
|
320
|
-
logger.debug(f"Async update memory: {should_update_memory}")
|
|
321
|
-
|
|
322
|
-
if not should_update_memory:
|
|
323
|
-
logger.debug("Memory update not required")
|
|
324
|
-
return "Memory update not required"
|
|
325
|
-
|
|
326
|
-
if self.manager is None:
|
|
327
|
-
self.manager = MemoryManager(user_id=self.user_id, db=self.db)
|
|
328
|
-
|
|
329
|
-
else:
|
|
330
|
-
self.manager.db = self.db
|
|
331
|
-
self.manager.user_id = self.user_id
|
|
332
|
-
|
|
333
|
-
response = await self.manager.arun(input)
|
|
334
|
-
self.load_user_memories()
|
|
335
|
-
self.updating_memory = False
|
|
336
|
-
return response
|
|
337
|
-
|
|
338
|
-
def update_summary(self) -> Optional[SessionSummary]:
|
|
339
|
-
"""Creates a summary of the session"""
|
|
340
|
-
from agno.memory.summarizer import MemorySummarizer
|
|
341
|
-
|
|
342
|
-
self.updating_memory = True
|
|
343
|
-
|
|
344
|
-
if self.summarizer is None:
|
|
345
|
-
self.summarizer = MemorySummarizer()
|
|
346
|
-
|
|
347
|
-
self.summary = self.summarizer.run(self.get_message_pairs())
|
|
348
|
-
self.updating_memory = False
|
|
349
|
-
return self.summary
|
|
350
|
-
|
|
351
|
-
async def aupdate_summary(self) -> Optional[SessionSummary]:
|
|
352
|
-
"""Creates a summary of the session"""
|
|
353
|
-
from agno.memory.summarizer import MemorySummarizer
|
|
354
|
-
|
|
355
|
-
self.updating_memory = True
|
|
356
|
-
|
|
357
|
-
if self.summarizer is None:
|
|
358
|
-
self.summarizer = MemorySummarizer()
|
|
359
|
-
|
|
360
|
-
self.summary = await self.summarizer.arun(self.get_message_pairs())
|
|
361
|
-
self.updating_memory = False
|
|
362
|
-
return self.summary
|
|
363
|
-
|
|
364
|
-
def clear(self) -> None:
|
|
365
|
-
"""Clear the AgentMemory"""
|
|
366
|
-
|
|
367
|
-
self.runs = []
|
|
368
|
-
self.messages = []
|
|
369
|
-
self.summary = None
|
|
370
|
-
self.memories = None
|
|
371
|
-
|
|
372
|
-
def deep_copy(self) -> "AgentMemory":
|
|
373
|
-
from copy import deepcopy
|
|
374
|
-
|
|
375
|
-
# Create a shallow copy of the object
|
|
376
|
-
copied_obj = self.__class__(**self.to_dict())
|
|
377
|
-
|
|
378
|
-
# Manually deepcopy fields that are known to be safe
|
|
379
|
-
for field_name, field_value in self.__dict__.items():
|
|
380
|
-
if field_name not in ["db", "classifier", "manager", "summarizer"]:
|
|
381
|
-
try:
|
|
382
|
-
setattr(copied_obj, field_name, deepcopy(field_value))
|
|
383
|
-
except Exception as e:
|
|
384
|
-
logger.warning(f"Failed to deepcopy field: {field_name} - {e}")
|
|
385
|
-
setattr(copied_obj, field_name, field_value)
|
|
386
|
-
|
|
387
|
-
copied_obj.db = self.db
|
|
388
|
-
copied_obj.classifier = self.classifier
|
|
389
|
-
copied_obj.manager = self.manager
|
|
390
|
-
copied_obj.summarizer = self.summarizer
|
|
391
|
-
|
|
392
|
-
return copied_obj
|