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.
Files changed (723) hide show
  1. agno/__init__.py +8 -0
  2. agno/agent/__init__.py +44 -5
  3. agno/agent/agent.py +10531 -2975
  4. agno/api/agent.py +14 -53
  5. agno/api/api.py +7 -46
  6. agno/api/evals.py +22 -0
  7. agno/api/os.py +17 -0
  8. agno/api/routes.py +6 -25
  9. agno/api/schemas/__init__.py +9 -0
  10. agno/api/schemas/agent.py +6 -9
  11. agno/api/schemas/evals.py +16 -0
  12. agno/api/schemas/os.py +14 -0
  13. agno/api/schemas/team.py +10 -10
  14. agno/api/schemas/utils.py +21 -0
  15. agno/api/schemas/workflows.py +16 -0
  16. agno/api/settings.py +53 -0
  17. agno/api/team.py +22 -26
  18. agno/api/workflow.py +28 -0
  19. agno/cloud/aws/base.py +214 -0
  20. agno/cloud/aws/s3/__init__.py +2 -0
  21. agno/cloud/aws/s3/api_client.py +43 -0
  22. agno/cloud/aws/s3/bucket.py +195 -0
  23. agno/cloud/aws/s3/object.py +57 -0
  24. agno/compression/__init__.py +3 -0
  25. agno/compression/manager.py +247 -0
  26. agno/culture/__init__.py +3 -0
  27. agno/culture/manager.py +956 -0
  28. agno/db/__init__.py +24 -0
  29. agno/db/async_postgres/__init__.py +3 -0
  30. agno/db/base.py +946 -0
  31. agno/db/dynamo/__init__.py +3 -0
  32. agno/db/dynamo/dynamo.py +2781 -0
  33. agno/db/dynamo/schemas.py +442 -0
  34. agno/db/dynamo/utils.py +743 -0
  35. agno/db/firestore/__init__.py +3 -0
  36. agno/db/firestore/firestore.py +2379 -0
  37. agno/db/firestore/schemas.py +181 -0
  38. agno/db/firestore/utils.py +376 -0
  39. agno/db/gcs_json/__init__.py +3 -0
  40. agno/db/gcs_json/gcs_json_db.py +1791 -0
  41. agno/db/gcs_json/utils.py +228 -0
  42. agno/db/in_memory/__init__.py +3 -0
  43. agno/db/in_memory/in_memory_db.py +1312 -0
  44. agno/db/in_memory/utils.py +230 -0
  45. agno/db/json/__init__.py +3 -0
  46. agno/db/json/json_db.py +1777 -0
  47. agno/db/json/utils.py +230 -0
  48. agno/db/migrations/manager.py +199 -0
  49. agno/db/migrations/v1_to_v2.py +635 -0
  50. agno/db/migrations/versions/v2_3_0.py +938 -0
  51. agno/db/mongo/__init__.py +17 -0
  52. agno/db/mongo/async_mongo.py +2760 -0
  53. agno/db/mongo/mongo.py +2597 -0
  54. agno/db/mongo/schemas.py +119 -0
  55. agno/db/mongo/utils.py +276 -0
  56. agno/db/mysql/__init__.py +4 -0
  57. agno/db/mysql/async_mysql.py +2912 -0
  58. agno/db/mysql/mysql.py +2923 -0
  59. agno/db/mysql/schemas.py +186 -0
  60. agno/db/mysql/utils.py +488 -0
  61. agno/db/postgres/__init__.py +4 -0
  62. agno/db/postgres/async_postgres.py +2579 -0
  63. agno/db/postgres/postgres.py +2870 -0
  64. agno/db/postgres/schemas.py +187 -0
  65. agno/db/postgres/utils.py +442 -0
  66. agno/db/redis/__init__.py +3 -0
  67. agno/db/redis/redis.py +2141 -0
  68. agno/db/redis/schemas.py +159 -0
  69. agno/db/redis/utils.py +346 -0
  70. agno/db/schemas/__init__.py +4 -0
  71. agno/db/schemas/culture.py +120 -0
  72. agno/db/schemas/evals.py +34 -0
  73. agno/db/schemas/knowledge.py +40 -0
  74. agno/db/schemas/memory.py +61 -0
  75. agno/db/singlestore/__init__.py +3 -0
  76. agno/db/singlestore/schemas.py +179 -0
  77. agno/db/singlestore/singlestore.py +2877 -0
  78. agno/db/singlestore/utils.py +384 -0
  79. agno/db/sqlite/__init__.py +4 -0
  80. agno/db/sqlite/async_sqlite.py +2911 -0
  81. agno/db/sqlite/schemas.py +181 -0
  82. agno/db/sqlite/sqlite.py +2908 -0
  83. agno/db/sqlite/utils.py +429 -0
  84. agno/db/surrealdb/__init__.py +3 -0
  85. agno/db/surrealdb/metrics.py +292 -0
  86. agno/db/surrealdb/models.py +334 -0
  87. agno/db/surrealdb/queries.py +71 -0
  88. agno/db/surrealdb/surrealdb.py +1908 -0
  89. agno/db/surrealdb/utils.py +147 -0
  90. agno/db/utils.py +118 -0
  91. agno/eval/__init__.py +24 -0
  92. agno/eval/accuracy.py +666 -276
  93. agno/eval/agent_as_judge.py +861 -0
  94. agno/eval/base.py +29 -0
  95. agno/eval/performance.py +779 -0
  96. agno/eval/reliability.py +241 -62
  97. agno/eval/utils.py +120 -0
  98. agno/exceptions.py +143 -1
  99. agno/filters.py +354 -0
  100. agno/guardrails/__init__.py +6 -0
  101. agno/guardrails/base.py +19 -0
  102. agno/guardrails/openai.py +144 -0
  103. agno/guardrails/pii.py +94 -0
  104. agno/guardrails/prompt_injection.py +52 -0
  105. agno/hooks/__init__.py +3 -0
  106. agno/hooks/decorator.py +164 -0
  107. agno/integrations/discord/__init__.py +3 -0
  108. agno/integrations/discord/client.py +203 -0
  109. agno/knowledge/__init__.py +5 -1
  110. agno/{document → knowledge}/chunking/agentic.py +22 -14
  111. agno/{document → knowledge}/chunking/document.py +2 -2
  112. agno/{document → knowledge}/chunking/fixed.py +7 -6
  113. agno/knowledge/chunking/markdown.py +151 -0
  114. agno/{document → knowledge}/chunking/recursive.py +15 -3
  115. agno/knowledge/chunking/row.py +39 -0
  116. agno/knowledge/chunking/semantic.py +91 -0
  117. agno/knowledge/chunking/strategy.py +165 -0
  118. agno/knowledge/content.py +74 -0
  119. agno/knowledge/document/__init__.py +5 -0
  120. agno/{document → knowledge/document}/base.py +12 -2
  121. agno/knowledge/embedder/__init__.py +5 -0
  122. agno/knowledge/embedder/aws_bedrock.py +343 -0
  123. agno/knowledge/embedder/azure_openai.py +210 -0
  124. agno/{embedder → knowledge/embedder}/base.py +8 -0
  125. agno/knowledge/embedder/cohere.py +323 -0
  126. agno/knowledge/embedder/fastembed.py +62 -0
  127. agno/{embedder → knowledge/embedder}/fireworks.py +1 -1
  128. agno/knowledge/embedder/google.py +258 -0
  129. agno/knowledge/embedder/huggingface.py +94 -0
  130. agno/knowledge/embedder/jina.py +182 -0
  131. agno/knowledge/embedder/langdb.py +22 -0
  132. agno/knowledge/embedder/mistral.py +206 -0
  133. agno/knowledge/embedder/nebius.py +13 -0
  134. agno/knowledge/embedder/ollama.py +154 -0
  135. agno/knowledge/embedder/openai.py +195 -0
  136. agno/knowledge/embedder/sentence_transformer.py +63 -0
  137. agno/{embedder → knowledge/embedder}/together.py +1 -1
  138. agno/knowledge/embedder/vllm.py +262 -0
  139. agno/knowledge/embedder/voyageai.py +165 -0
  140. agno/knowledge/knowledge.py +3006 -0
  141. agno/knowledge/reader/__init__.py +7 -0
  142. agno/knowledge/reader/arxiv_reader.py +81 -0
  143. agno/knowledge/reader/base.py +95 -0
  144. agno/knowledge/reader/csv_reader.py +164 -0
  145. agno/knowledge/reader/docx_reader.py +82 -0
  146. agno/knowledge/reader/field_labeled_csv_reader.py +290 -0
  147. agno/knowledge/reader/firecrawl_reader.py +201 -0
  148. agno/knowledge/reader/json_reader.py +88 -0
  149. agno/knowledge/reader/markdown_reader.py +137 -0
  150. agno/knowledge/reader/pdf_reader.py +431 -0
  151. agno/knowledge/reader/pptx_reader.py +101 -0
  152. agno/knowledge/reader/reader_factory.py +313 -0
  153. agno/knowledge/reader/s3_reader.py +89 -0
  154. agno/knowledge/reader/tavily_reader.py +193 -0
  155. agno/knowledge/reader/text_reader.py +127 -0
  156. agno/knowledge/reader/web_search_reader.py +325 -0
  157. agno/knowledge/reader/website_reader.py +455 -0
  158. agno/knowledge/reader/wikipedia_reader.py +91 -0
  159. agno/knowledge/reader/youtube_reader.py +78 -0
  160. agno/knowledge/remote_content/remote_content.py +88 -0
  161. agno/knowledge/reranker/__init__.py +3 -0
  162. agno/{reranker → knowledge/reranker}/base.py +1 -1
  163. agno/{reranker → knowledge/reranker}/cohere.py +2 -2
  164. agno/knowledge/reranker/infinity.py +195 -0
  165. agno/knowledge/reranker/sentence_transformer.py +54 -0
  166. agno/knowledge/types.py +39 -0
  167. agno/knowledge/utils.py +234 -0
  168. agno/media.py +439 -95
  169. agno/memory/__init__.py +16 -3
  170. agno/memory/manager.py +1474 -123
  171. agno/memory/strategies/__init__.py +15 -0
  172. agno/memory/strategies/base.py +66 -0
  173. agno/memory/strategies/summarize.py +196 -0
  174. agno/memory/strategies/types.py +37 -0
  175. agno/models/aimlapi/__init__.py +5 -0
  176. agno/models/aimlapi/aimlapi.py +62 -0
  177. agno/models/anthropic/__init__.py +4 -0
  178. agno/models/anthropic/claude.py +960 -496
  179. agno/models/aws/__init__.py +15 -0
  180. agno/models/aws/bedrock.py +686 -451
  181. agno/models/aws/claude.py +190 -183
  182. agno/models/azure/__init__.py +18 -1
  183. agno/models/azure/ai_foundry.py +489 -0
  184. agno/models/azure/openai_chat.py +89 -40
  185. agno/models/base.py +2477 -550
  186. agno/models/cerebras/__init__.py +12 -0
  187. agno/models/cerebras/cerebras.py +565 -0
  188. agno/models/cerebras/cerebras_openai.py +131 -0
  189. agno/models/cohere/__init__.py +4 -0
  190. agno/models/cohere/chat.py +306 -492
  191. agno/models/cometapi/__init__.py +5 -0
  192. agno/models/cometapi/cometapi.py +74 -0
  193. agno/models/dashscope/__init__.py +5 -0
  194. agno/models/dashscope/dashscope.py +90 -0
  195. agno/models/deepinfra/__init__.py +5 -0
  196. agno/models/deepinfra/deepinfra.py +45 -0
  197. agno/models/deepseek/__init__.py +4 -0
  198. agno/models/deepseek/deepseek.py +110 -9
  199. agno/models/fireworks/__init__.py +4 -0
  200. agno/models/fireworks/fireworks.py +19 -22
  201. agno/models/google/__init__.py +3 -7
  202. agno/models/google/gemini.py +1717 -662
  203. agno/models/google/utils.py +22 -0
  204. agno/models/groq/__init__.py +4 -0
  205. agno/models/groq/groq.py +391 -666
  206. agno/models/huggingface/__init__.py +4 -0
  207. agno/models/huggingface/huggingface.py +266 -538
  208. agno/models/ibm/__init__.py +5 -0
  209. agno/models/ibm/watsonx.py +432 -0
  210. agno/models/internlm/__init__.py +3 -0
  211. agno/models/internlm/internlm.py +20 -3
  212. agno/models/langdb/__init__.py +1 -0
  213. agno/models/langdb/langdb.py +60 -0
  214. agno/models/litellm/__init__.py +14 -0
  215. agno/models/litellm/chat.py +503 -0
  216. agno/models/litellm/litellm_openai.py +42 -0
  217. agno/models/llama_cpp/__init__.py +5 -0
  218. agno/models/llama_cpp/llama_cpp.py +22 -0
  219. agno/models/lmstudio/__init__.py +5 -0
  220. agno/models/lmstudio/lmstudio.py +25 -0
  221. agno/models/message.py +361 -39
  222. agno/models/meta/__init__.py +12 -0
  223. agno/models/meta/llama.py +502 -0
  224. agno/models/meta/llama_openai.py +79 -0
  225. agno/models/metrics.py +120 -0
  226. agno/models/mistral/__init__.py +4 -0
  227. agno/models/mistral/mistral.py +293 -393
  228. agno/models/nebius/__init__.py +3 -0
  229. agno/models/nebius/nebius.py +53 -0
  230. agno/models/nexus/__init__.py +3 -0
  231. agno/models/nexus/nexus.py +22 -0
  232. agno/models/nvidia/__init__.py +4 -0
  233. agno/models/nvidia/nvidia.py +22 -3
  234. agno/models/ollama/__init__.py +4 -2
  235. agno/models/ollama/chat.py +257 -492
  236. agno/models/openai/__init__.py +7 -0
  237. agno/models/openai/chat.py +725 -770
  238. agno/models/openai/like.py +16 -2
  239. agno/models/openai/responses.py +1121 -0
  240. agno/models/openrouter/__init__.py +4 -0
  241. agno/models/openrouter/openrouter.py +62 -5
  242. agno/models/perplexity/__init__.py +5 -0
  243. agno/models/perplexity/perplexity.py +203 -0
  244. agno/models/portkey/__init__.py +3 -0
  245. agno/models/portkey/portkey.py +82 -0
  246. agno/models/requesty/__init__.py +5 -0
  247. agno/models/requesty/requesty.py +69 -0
  248. agno/models/response.py +177 -7
  249. agno/models/sambanova/__init__.py +4 -0
  250. agno/models/sambanova/sambanova.py +23 -4
  251. agno/models/siliconflow/__init__.py +5 -0
  252. agno/models/siliconflow/siliconflow.py +42 -0
  253. agno/models/together/__init__.py +4 -0
  254. agno/models/together/together.py +21 -164
  255. agno/models/utils.py +266 -0
  256. agno/models/vercel/__init__.py +3 -0
  257. agno/models/vercel/v0.py +43 -0
  258. agno/models/vertexai/__init__.py +0 -1
  259. agno/models/vertexai/claude.py +190 -0
  260. agno/models/vllm/__init__.py +3 -0
  261. agno/models/vllm/vllm.py +83 -0
  262. agno/models/xai/__init__.py +2 -0
  263. agno/models/xai/xai.py +111 -7
  264. agno/os/__init__.py +3 -0
  265. agno/os/app.py +1027 -0
  266. agno/os/auth.py +244 -0
  267. agno/os/config.py +126 -0
  268. agno/os/interfaces/__init__.py +1 -0
  269. agno/os/interfaces/a2a/__init__.py +3 -0
  270. agno/os/interfaces/a2a/a2a.py +42 -0
  271. agno/os/interfaces/a2a/router.py +249 -0
  272. agno/os/interfaces/a2a/utils.py +924 -0
  273. agno/os/interfaces/agui/__init__.py +3 -0
  274. agno/os/interfaces/agui/agui.py +47 -0
  275. agno/os/interfaces/agui/router.py +147 -0
  276. agno/os/interfaces/agui/utils.py +574 -0
  277. agno/os/interfaces/base.py +25 -0
  278. agno/os/interfaces/slack/__init__.py +3 -0
  279. agno/os/interfaces/slack/router.py +148 -0
  280. agno/os/interfaces/slack/security.py +30 -0
  281. agno/os/interfaces/slack/slack.py +47 -0
  282. agno/os/interfaces/whatsapp/__init__.py +3 -0
  283. agno/os/interfaces/whatsapp/router.py +210 -0
  284. agno/os/interfaces/whatsapp/security.py +55 -0
  285. agno/os/interfaces/whatsapp/whatsapp.py +36 -0
  286. agno/os/mcp.py +293 -0
  287. agno/os/middleware/__init__.py +9 -0
  288. agno/os/middleware/jwt.py +797 -0
  289. agno/os/router.py +258 -0
  290. agno/os/routers/__init__.py +3 -0
  291. agno/os/routers/agents/__init__.py +3 -0
  292. agno/os/routers/agents/router.py +599 -0
  293. agno/os/routers/agents/schema.py +261 -0
  294. agno/os/routers/evals/__init__.py +3 -0
  295. agno/os/routers/evals/evals.py +450 -0
  296. agno/os/routers/evals/schemas.py +174 -0
  297. agno/os/routers/evals/utils.py +231 -0
  298. agno/os/routers/health.py +31 -0
  299. agno/os/routers/home.py +52 -0
  300. agno/os/routers/knowledge/__init__.py +3 -0
  301. agno/os/routers/knowledge/knowledge.py +1008 -0
  302. agno/os/routers/knowledge/schemas.py +178 -0
  303. agno/os/routers/memory/__init__.py +3 -0
  304. agno/os/routers/memory/memory.py +661 -0
  305. agno/os/routers/memory/schemas.py +88 -0
  306. agno/os/routers/metrics/__init__.py +3 -0
  307. agno/os/routers/metrics/metrics.py +190 -0
  308. agno/os/routers/metrics/schemas.py +47 -0
  309. agno/os/routers/session/__init__.py +3 -0
  310. agno/os/routers/session/session.py +997 -0
  311. agno/os/routers/teams/__init__.py +3 -0
  312. agno/os/routers/teams/router.py +512 -0
  313. agno/os/routers/teams/schema.py +257 -0
  314. agno/os/routers/traces/__init__.py +3 -0
  315. agno/os/routers/traces/schemas.py +414 -0
  316. agno/os/routers/traces/traces.py +499 -0
  317. agno/os/routers/workflows/__init__.py +3 -0
  318. agno/os/routers/workflows/router.py +624 -0
  319. agno/os/routers/workflows/schema.py +75 -0
  320. agno/os/schema.py +534 -0
  321. agno/os/scopes.py +469 -0
  322. agno/{playground → os}/settings.py +7 -15
  323. agno/os/utils.py +973 -0
  324. agno/reasoning/anthropic.py +80 -0
  325. agno/reasoning/azure_ai_foundry.py +67 -0
  326. agno/reasoning/deepseek.py +63 -0
  327. agno/reasoning/default.py +97 -0
  328. agno/reasoning/gemini.py +73 -0
  329. agno/reasoning/groq.py +71 -0
  330. agno/reasoning/helpers.py +24 -1
  331. agno/reasoning/ollama.py +67 -0
  332. agno/reasoning/openai.py +86 -0
  333. agno/reasoning/step.py +2 -1
  334. agno/reasoning/vertexai.py +76 -0
  335. agno/run/__init__.py +6 -0
  336. agno/run/agent.py +822 -0
  337. agno/run/base.py +247 -0
  338. agno/run/cancel.py +81 -0
  339. agno/run/requirement.py +181 -0
  340. agno/run/team.py +767 -0
  341. agno/run/workflow.py +708 -0
  342. agno/session/__init__.py +10 -0
  343. agno/session/agent.py +260 -0
  344. agno/session/summary.py +265 -0
  345. agno/session/team.py +342 -0
  346. agno/session/workflow.py +501 -0
  347. agno/table.py +10 -0
  348. agno/team/__init__.py +37 -0
  349. agno/team/team.py +9536 -0
  350. agno/tools/__init__.py +7 -0
  351. agno/tools/agentql.py +120 -0
  352. agno/tools/airflow.py +22 -12
  353. agno/tools/api.py +122 -0
  354. agno/tools/apify.py +276 -83
  355. agno/tools/{arxiv_toolkit.py → arxiv.py} +20 -12
  356. agno/tools/aws_lambda.py +28 -7
  357. agno/tools/aws_ses.py +66 -0
  358. agno/tools/baidusearch.py +11 -4
  359. agno/tools/bitbucket.py +292 -0
  360. agno/tools/brandfetch.py +213 -0
  361. agno/tools/bravesearch.py +106 -0
  362. agno/tools/brightdata.py +367 -0
  363. agno/tools/browserbase.py +209 -0
  364. agno/tools/calcom.py +32 -23
  365. agno/tools/calculator.py +24 -37
  366. agno/tools/cartesia.py +187 -0
  367. agno/tools/{clickup_tool.py → clickup.py} +17 -28
  368. agno/tools/confluence.py +91 -26
  369. agno/tools/crawl4ai.py +139 -43
  370. agno/tools/csv_toolkit.py +28 -22
  371. agno/tools/dalle.py +36 -22
  372. agno/tools/daytona.py +475 -0
  373. agno/tools/decorator.py +169 -14
  374. agno/tools/desi_vocal.py +23 -11
  375. agno/tools/discord.py +32 -29
  376. agno/tools/docker.py +716 -0
  377. agno/tools/duckdb.py +76 -81
  378. agno/tools/duckduckgo.py +43 -40
  379. agno/tools/e2b.py +703 -0
  380. agno/tools/eleven_labs.py +65 -54
  381. agno/tools/email.py +13 -5
  382. agno/tools/evm.py +129 -0
  383. agno/tools/exa.py +324 -42
  384. agno/tools/fal.py +39 -35
  385. agno/tools/file.py +196 -30
  386. agno/tools/file_generation.py +356 -0
  387. agno/tools/financial_datasets.py +288 -0
  388. agno/tools/firecrawl.py +108 -33
  389. agno/tools/function.py +960 -122
  390. agno/tools/giphy.py +34 -12
  391. agno/tools/github.py +1294 -97
  392. agno/tools/gmail.py +922 -0
  393. agno/tools/google_bigquery.py +117 -0
  394. agno/tools/google_drive.py +271 -0
  395. agno/tools/google_maps.py +253 -0
  396. agno/tools/googlecalendar.py +607 -107
  397. agno/tools/googlesheets.py +377 -0
  398. agno/tools/hackernews.py +20 -12
  399. agno/tools/jina.py +24 -14
  400. agno/tools/jira.py +48 -19
  401. agno/tools/knowledge.py +218 -0
  402. agno/tools/linear.py +82 -43
  403. agno/tools/linkup.py +58 -0
  404. agno/tools/local_file_system.py +15 -7
  405. agno/tools/lumalab.py +41 -26
  406. agno/tools/mcp/__init__.py +10 -0
  407. agno/tools/mcp/mcp.py +331 -0
  408. agno/tools/mcp/multi_mcp.py +347 -0
  409. agno/tools/mcp/params.py +24 -0
  410. agno/tools/mcp_toolbox.py +284 -0
  411. agno/tools/mem0.py +193 -0
  412. agno/tools/memory.py +419 -0
  413. agno/tools/mlx_transcribe.py +11 -9
  414. agno/tools/models/azure_openai.py +190 -0
  415. agno/tools/models/gemini.py +203 -0
  416. agno/tools/models/groq.py +158 -0
  417. agno/tools/models/morph.py +186 -0
  418. agno/tools/models/nebius.py +124 -0
  419. agno/tools/models_labs.py +163 -82
  420. agno/tools/moviepy_video.py +18 -13
  421. agno/tools/nano_banana.py +151 -0
  422. agno/tools/neo4j.py +134 -0
  423. agno/tools/newspaper.py +15 -4
  424. agno/tools/newspaper4k.py +19 -6
  425. agno/tools/notion.py +204 -0
  426. agno/tools/openai.py +181 -17
  427. agno/tools/openbb.py +27 -20
  428. agno/tools/opencv.py +321 -0
  429. agno/tools/openweather.py +233 -0
  430. agno/tools/oxylabs.py +385 -0
  431. agno/tools/pandas.py +25 -15
  432. agno/tools/parallel.py +314 -0
  433. agno/tools/postgres.py +238 -185
  434. agno/tools/pubmed.py +125 -13
  435. agno/tools/python.py +48 -35
  436. agno/tools/reasoning.py +283 -0
  437. agno/tools/reddit.py +207 -29
  438. agno/tools/redshift.py +406 -0
  439. agno/tools/replicate.py +69 -26
  440. agno/tools/resend.py +11 -6
  441. agno/tools/scrapegraph.py +179 -19
  442. agno/tools/searxng.py +23 -31
  443. agno/tools/serpapi.py +15 -10
  444. agno/tools/serper.py +255 -0
  445. agno/tools/shell.py +23 -12
  446. agno/tools/shopify.py +1519 -0
  447. agno/tools/slack.py +56 -14
  448. agno/tools/sleep.py +8 -6
  449. agno/tools/spider.py +35 -11
  450. agno/tools/spotify.py +919 -0
  451. agno/tools/sql.py +34 -19
  452. agno/tools/tavily.py +158 -8
  453. agno/tools/telegram.py +18 -8
  454. agno/tools/todoist.py +218 -0
  455. agno/tools/toolkit.py +134 -9
  456. agno/tools/trafilatura.py +388 -0
  457. agno/tools/trello.py +25 -28
  458. agno/tools/twilio.py +18 -9
  459. agno/tools/user_control_flow.py +78 -0
  460. agno/tools/valyu.py +228 -0
  461. agno/tools/visualization.py +467 -0
  462. agno/tools/webbrowser.py +28 -0
  463. agno/tools/webex.py +76 -0
  464. agno/tools/website.py +23 -19
  465. agno/tools/webtools.py +45 -0
  466. agno/tools/whatsapp.py +286 -0
  467. agno/tools/wikipedia.py +28 -19
  468. agno/tools/workflow.py +285 -0
  469. agno/tools/{twitter.py → x.py} +142 -46
  470. agno/tools/yfinance.py +41 -39
  471. agno/tools/youtube.py +34 -17
  472. agno/tools/zendesk.py +15 -5
  473. agno/tools/zep.py +454 -0
  474. agno/tools/zoom.py +86 -37
  475. agno/tracing/__init__.py +12 -0
  476. agno/tracing/exporter.py +157 -0
  477. agno/tracing/schemas.py +276 -0
  478. agno/tracing/setup.py +111 -0
  479. agno/utils/agent.py +938 -0
  480. agno/utils/audio.py +37 -1
  481. agno/utils/certs.py +27 -0
  482. agno/utils/code_execution.py +11 -0
  483. agno/utils/common.py +103 -20
  484. agno/utils/cryptography.py +22 -0
  485. agno/utils/dttm.py +33 -0
  486. agno/utils/events.py +700 -0
  487. agno/utils/functions.py +107 -37
  488. agno/utils/gemini.py +426 -0
  489. agno/utils/hooks.py +171 -0
  490. agno/utils/http.py +185 -0
  491. agno/utils/json_schema.py +159 -37
  492. agno/utils/knowledge.py +36 -0
  493. agno/utils/location.py +19 -0
  494. agno/utils/log.py +221 -8
  495. agno/utils/mcp.py +214 -0
  496. agno/utils/media.py +335 -14
  497. agno/utils/merge_dict.py +22 -1
  498. agno/utils/message.py +77 -2
  499. agno/utils/models/ai_foundry.py +50 -0
  500. agno/utils/models/claude.py +373 -0
  501. agno/utils/models/cohere.py +94 -0
  502. agno/utils/models/llama.py +85 -0
  503. agno/utils/models/mistral.py +100 -0
  504. agno/utils/models/openai_responses.py +140 -0
  505. agno/utils/models/schema_utils.py +153 -0
  506. agno/utils/models/watsonx.py +41 -0
  507. agno/utils/openai.py +257 -0
  508. agno/utils/pickle.py +1 -1
  509. agno/utils/pprint.py +124 -8
  510. agno/utils/print_response/agent.py +930 -0
  511. agno/utils/print_response/team.py +1914 -0
  512. agno/utils/print_response/workflow.py +1668 -0
  513. agno/utils/prompts.py +111 -0
  514. agno/utils/reasoning.py +108 -0
  515. agno/utils/response.py +163 -0
  516. agno/utils/serialize.py +32 -0
  517. agno/utils/shell.py +4 -4
  518. agno/utils/streamlit.py +487 -0
  519. agno/utils/string.py +204 -51
  520. agno/utils/team.py +139 -0
  521. agno/utils/timer.py +9 -2
  522. agno/utils/tokens.py +657 -0
  523. agno/utils/tools.py +19 -1
  524. agno/utils/whatsapp.py +305 -0
  525. agno/utils/yaml_io.py +3 -3
  526. agno/vectordb/__init__.py +2 -0
  527. agno/vectordb/base.py +87 -9
  528. agno/vectordb/cassandra/__init__.py +5 -1
  529. agno/vectordb/cassandra/cassandra.py +383 -27
  530. agno/vectordb/chroma/__init__.py +4 -0
  531. agno/vectordb/chroma/chromadb.py +748 -83
  532. agno/vectordb/clickhouse/__init__.py +7 -1
  533. agno/vectordb/clickhouse/clickhousedb.py +554 -53
  534. agno/vectordb/couchbase/__init__.py +3 -0
  535. agno/vectordb/couchbase/couchbase.py +1446 -0
  536. agno/vectordb/lancedb/__init__.py +5 -0
  537. agno/vectordb/lancedb/lance_db.py +730 -98
  538. agno/vectordb/langchaindb/__init__.py +5 -0
  539. agno/vectordb/langchaindb/langchaindb.py +163 -0
  540. agno/vectordb/lightrag/__init__.py +5 -0
  541. agno/vectordb/lightrag/lightrag.py +388 -0
  542. agno/vectordb/llamaindex/__init__.py +3 -0
  543. agno/vectordb/llamaindex/llamaindexdb.py +166 -0
  544. agno/vectordb/milvus/__init__.py +3 -0
  545. agno/vectordb/milvus/milvus.py +966 -78
  546. agno/vectordb/mongodb/__init__.py +9 -1
  547. agno/vectordb/mongodb/mongodb.py +1175 -172
  548. agno/vectordb/pgvector/__init__.py +8 -0
  549. agno/vectordb/pgvector/pgvector.py +599 -115
  550. agno/vectordb/pineconedb/__init__.py +5 -1
  551. agno/vectordb/pineconedb/pineconedb.py +406 -43
  552. agno/vectordb/qdrant/__init__.py +4 -0
  553. agno/vectordb/qdrant/qdrant.py +914 -61
  554. agno/vectordb/redis/__init__.py +9 -0
  555. agno/vectordb/redis/redisdb.py +682 -0
  556. agno/vectordb/singlestore/__init__.py +8 -1
  557. agno/vectordb/singlestore/singlestore.py +771 -0
  558. agno/vectordb/surrealdb/__init__.py +3 -0
  559. agno/vectordb/surrealdb/surrealdb.py +663 -0
  560. agno/vectordb/upstashdb/__init__.py +5 -0
  561. agno/vectordb/upstashdb/upstashdb.py +718 -0
  562. agno/vectordb/weaviate/__init__.py +8 -0
  563. agno/vectordb/weaviate/index.py +15 -0
  564. agno/vectordb/weaviate/weaviate.py +1009 -0
  565. agno/workflow/__init__.py +23 -1
  566. agno/workflow/agent.py +299 -0
  567. agno/workflow/condition.py +759 -0
  568. agno/workflow/loop.py +756 -0
  569. agno/workflow/parallel.py +853 -0
  570. agno/workflow/router.py +723 -0
  571. agno/workflow/step.py +1564 -0
  572. agno/workflow/steps.py +613 -0
  573. agno/workflow/types.py +556 -0
  574. agno/workflow/workflow.py +4327 -514
  575. agno-2.3.13.dist-info/METADATA +639 -0
  576. agno-2.3.13.dist-info/RECORD +613 -0
  577. {agno-0.1.2.dist-info → agno-2.3.13.dist-info}/WHEEL +1 -1
  578. agno-2.3.13.dist-info/licenses/LICENSE +201 -0
  579. agno/api/playground.py +0 -91
  580. agno/api/schemas/playground.py +0 -22
  581. agno/api/schemas/user.py +0 -22
  582. agno/api/schemas/workspace.py +0 -46
  583. agno/api/user.py +0 -160
  584. agno/api/workspace.py +0 -151
  585. agno/cli/auth_server.py +0 -118
  586. agno/cli/config.py +0 -275
  587. agno/cli/console.py +0 -88
  588. agno/cli/credentials.py +0 -23
  589. agno/cli/entrypoint.py +0 -571
  590. agno/cli/operator.py +0 -355
  591. agno/cli/settings.py +0 -85
  592. agno/cli/ws/ws_cli.py +0 -817
  593. agno/constants.py +0 -13
  594. agno/document/__init__.py +0 -1
  595. agno/document/chunking/semantic.py +0 -47
  596. agno/document/chunking/strategy.py +0 -31
  597. agno/document/reader/__init__.py +0 -1
  598. agno/document/reader/arxiv_reader.py +0 -41
  599. agno/document/reader/base.py +0 -22
  600. agno/document/reader/csv_reader.py +0 -84
  601. agno/document/reader/docx_reader.py +0 -46
  602. agno/document/reader/firecrawl_reader.py +0 -99
  603. agno/document/reader/json_reader.py +0 -43
  604. agno/document/reader/pdf_reader.py +0 -219
  605. agno/document/reader/s3/pdf_reader.py +0 -46
  606. agno/document/reader/s3/text_reader.py +0 -51
  607. agno/document/reader/text_reader.py +0 -41
  608. agno/document/reader/website_reader.py +0 -175
  609. agno/document/reader/youtube_reader.py +0 -50
  610. agno/embedder/__init__.py +0 -1
  611. agno/embedder/azure_openai.py +0 -86
  612. agno/embedder/cohere.py +0 -72
  613. agno/embedder/fastembed.py +0 -37
  614. agno/embedder/google.py +0 -73
  615. agno/embedder/huggingface.py +0 -54
  616. agno/embedder/mistral.py +0 -80
  617. agno/embedder/ollama.py +0 -57
  618. agno/embedder/openai.py +0 -74
  619. agno/embedder/sentence_transformer.py +0 -38
  620. agno/embedder/voyageai.py +0 -64
  621. agno/eval/perf.py +0 -201
  622. agno/file/__init__.py +0 -1
  623. agno/file/file.py +0 -16
  624. agno/file/local/csv.py +0 -32
  625. agno/file/local/txt.py +0 -19
  626. agno/infra/app.py +0 -240
  627. agno/infra/base.py +0 -144
  628. agno/infra/context.py +0 -20
  629. agno/infra/db_app.py +0 -52
  630. agno/infra/resource.py +0 -205
  631. agno/infra/resources.py +0 -55
  632. agno/knowledge/agent.py +0 -230
  633. agno/knowledge/arxiv.py +0 -22
  634. agno/knowledge/combined.py +0 -22
  635. agno/knowledge/csv.py +0 -28
  636. agno/knowledge/csv_url.py +0 -19
  637. agno/knowledge/document.py +0 -20
  638. agno/knowledge/docx.py +0 -30
  639. agno/knowledge/json.py +0 -28
  640. agno/knowledge/langchain.py +0 -71
  641. agno/knowledge/llamaindex.py +0 -66
  642. agno/knowledge/pdf.py +0 -28
  643. agno/knowledge/pdf_url.py +0 -26
  644. agno/knowledge/s3/base.py +0 -60
  645. agno/knowledge/s3/pdf.py +0 -21
  646. agno/knowledge/s3/text.py +0 -23
  647. agno/knowledge/text.py +0 -30
  648. agno/knowledge/website.py +0 -88
  649. agno/knowledge/wikipedia.py +0 -31
  650. agno/knowledge/youtube.py +0 -22
  651. agno/memory/agent.py +0 -392
  652. agno/memory/classifier.py +0 -104
  653. agno/memory/db/__init__.py +0 -1
  654. agno/memory/db/base.py +0 -42
  655. agno/memory/db/mongodb.py +0 -189
  656. agno/memory/db/postgres.py +0 -203
  657. agno/memory/db/sqlite.py +0 -193
  658. agno/memory/memory.py +0 -15
  659. agno/memory/row.py +0 -36
  660. agno/memory/summarizer.py +0 -192
  661. agno/memory/summary.py +0 -19
  662. agno/memory/workflow.py +0 -38
  663. agno/models/google/gemini_openai.py +0 -26
  664. agno/models/ollama/hermes.py +0 -221
  665. agno/models/ollama/tools.py +0 -362
  666. agno/models/vertexai/gemini.py +0 -595
  667. agno/playground/__init__.py +0 -3
  668. agno/playground/async_router.py +0 -421
  669. agno/playground/deploy.py +0 -249
  670. agno/playground/operator.py +0 -92
  671. agno/playground/playground.py +0 -91
  672. agno/playground/schemas.py +0 -76
  673. agno/playground/serve.py +0 -55
  674. agno/playground/sync_router.py +0 -405
  675. agno/reasoning/agent.py +0 -68
  676. agno/run/response.py +0 -112
  677. agno/storage/agent/__init__.py +0 -0
  678. agno/storage/agent/base.py +0 -38
  679. agno/storage/agent/dynamodb.py +0 -350
  680. agno/storage/agent/json.py +0 -92
  681. agno/storage/agent/mongodb.py +0 -228
  682. agno/storage/agent/postgres.py +0 -367
  683. agno/storage/agent/session.py +0 -79
  684. agno/storage/agent/singlestore.py +0 -303
  685. agno/storage/agent/sqlite.py +0 -357
  686. agno/storage/agent/yaml.py +0 -93
  687. agno/storage/workflow/__init__.py +0 -0
  688. agno/storage/workflow/base.py +0 -40
  689. agno/storage/workflow/mongodb.py +0 -233
  690. agno/storage/workflow/postgres.py +0 -366
  691. agno/storage/workflow/session.py +0 -60
  692. agno/storage/workflow/sqlite.py +0 -359
  693. agno/tools/googlesearch.py +0 -88
  694. agno/utils/defaults.py +0 -57
  695. agno/utils/filesystem.py +0 -39
  696. agno/utils/git.py +0 -52
  697. agno/utils/json_io.py +0 -30
  698. agno/utils/load_env.py +0 -19
  699. agno/utils/py_io.py +0 -19
  700. agno/utils/pyproject.py +0 -18
  701. agno/utils/resource_filter.py +0 -31
  702. agno/vectordb/singlestore/s2vectordb.py +0 -390
  703. agno/vectordb/singlestore/s2vectordb2.py +0 -355
  704. agno/workspace/__init__.py +0 -0
  705. agno/workspace/config.py +0 -325
  706. agno/workspace/enums.py +0 -6
  707. agno/workspace/helpers.py +0 -48
  708. agno/workspace/operator.py +0 -758
  709. agno/workspace/settings.py +0 -63
  710. agno-0.1.2.dist-info/LICENSE +0 -375
  711. agno-0.1.2.dist-info/METADATA +0 -502
  712. agno-0.1.2.dist-info/RECORD +0 -352
  713. agno-0.1.2.dist-info/entry_points.txt +0 -3
  714. /agno/{cli → db/migrations}/__init__.py +0 -0
  715. /agno/{cli/ws → db/migrations/versions}/__init__.py +0 -0
  716. /agno/{document/chunking/__init__.py → db/schemas/metrics.py} +0 -0
  717. /agno/{document/reader/s3 → integrations}/__init__.py +0 -0
  718. /agno/{file/local → knowledge/chunking}/__init__.py +0 -0
  719. /agno/{infra → knowledge/remote_content}/__init__.py +0 -0
  720. /agno/{knowledge/s3 → tools/models}/__init__.py +0 -0
  721. /agno/{reranker → utils/models}/__init__.py +0 -0
  722. /agno/{storage → utils/print_response}/__init__.py +0 -0
  723. {agno-0.1.2.dist-info → agno-2.3.13.dist-info}/top_level.txt +0 -0
@@ -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 ----------")
@@ -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()