agno 2.2.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 (575) hide show
  1. agno/__init__.py +8 -0
  2. agno/agent/__init__.py +51 -0
  3. agno/agent/agent.py +10405 -0
  4. agno/api/__init__.py +0 -0
  5. agno/api/agent.py +28 -0
  6. agno/api/api.py +40 -0
  7. agno/api/evals.py +22 -0
  8. agno/api/os.py +17 -0
  9. agno/api/routes.py +13 -0
  10. agno/api/schemas/__init__.py +9 -0
  11. agno/api/schemas/agent.py +16 -0
  12. agno/api/schemas/evals.py +16 -0
  13. agno/api/schemas/os.py +14 -0
  14. agno/api/schemas/response.py +6 -0
  15. agno/api/schemas/team.py +16 -0
  16. agno/api/schemas/utils.py +21 -0
  17. agno/api/schemas/workflows.py +16 -0
  18. agno/api/settings.py +53 -0
  19. agno/api/team.py +30 -0
  20. agno/api/workflow.py +28 -0
  21. agno/cloud/aws/base.py +214 -0
  22. agno/cloud/aws/s3/__init__.py +2 -0
  23. agno/cloud/aws/s3/api_client.py +43 -0
  24. agno/cloud/aws/s3/bucket.py +195 -0
  25. agno/cloud/aws/s3/object.py +57 -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 +598 -0
  31. agno/db/dynamo/__init__.py +3 -0
  32. agno/db/dynamo/dynamo.py +2042 -0
  33. agno/db/dynamo/schemas.py +314 -0
  34. agno/db/dynamo/utils.py +743 -0
  35. agno/db/firestore/__init__.py +3 -0
  36. agno/db/firestore/firestore.py +1795 -0
  37. agno/db/firestore/schemas.py +140 -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 +1335 -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 +1160 -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 +1328 -0
  47. agno/db/json/utils.py +230 -0
  48. agno/db/migrations/__init__.py +0 -0
  49. agno/db/migrations/v1_to_v2.py +635 -0
  50. agno/db/mongo/__init__.py +17 -0
  51. agno/db/mongo/async_mongo.py +2026 -0
  52. agno/db/mongo/mongo.py +1982 -0
  53. agno/db/mongo/schemas.py +87 -0
  54. agno/db/mongo/utils.py +259 -0
  55. agno/db/mysql/__init__.py +3 -0
  56. agno/db/mysql/mysql.py +2308 -0
  57. agno/db/mysql/schemas.py +138 -0
  58. agno/db/mysql/utils.py +355 -0
  59. agno/db/postgres/__init__.py +4 -0
  60. agno/db/postgres/async_postgres.py +1927 -0
  61. agno/db/postgres/postgres.py +2260 -0
  62. agno/db/postgres/schemas.py +139 -0
  63. agno/db/postgres/utils.py +442 -0
  64. agno/db/redis/__init__.py +3 -0
  65. agno/db/redis/redis.py +1660 -0
  66. agno/db/redis/schemas.py +123 -0
  67. agno/db/redis/utils.py +346 -0
  68. agno/db/schemas/__init__.py +4 -0
  69. agno/db/schemas/culture.py +120 -0
  70. agno/db/schemas/evals.py +33 -0
  71. agno/db/schemas/knowledge.py +40 -0
  72. agno/db/schemas/memory.py +46 -0
  73. agno/db/schemas/metrics.py +0 -0
  74. agno/db/singlestore/__init__.py +3 -0
  75. agno/db/singlestore/schemas.py +130 -0
  76. agno/db/singlestore/singlestore.py +2272 -0
  77. agno/db/singlestore/utils.py +384 -0
  78. agno/db/sqlite/__init__.py +4 -0
  79. agno/db/sqlite/async_sqlite.py +2293 -0
  80. agno/db/sqlite/schemas.py +133 -0
  81. agno/db/sqlite/sqlite.py +2288 -0
  82. agno/db/sqlite/utils.py +431 -0
  83. agno/db/surrealdb/__init__.py +3 -0
  84. agno/db/surrealdb/metrics.py +292 -0
  85. agno/db/surrealdb/models.py +309 -0
  86. agno/db/surrealdb/queries.py +71 -0
  87. agno/db/surrealdb/surrealdb.py +1353 -0
  88. agno/db/surrealdb/utils.py +147 -0
  89. agno/db/utils.py +116 -0
  90. agno/debug.py +18 -0
  91. agno/eval/__init__.py +14 -0
  92. agno/eval/accuracy.py +834 -0
  93. agno/eval/performance.py +773 -0
  94. agno/eval/reliability.py +306 -0
  95. agno/eval/utils.py +119 -0
  96. agno/exceptions.py +161 -0
  97. agno/filters.py +354 -0
  98. agno/guardrails/__init__.py +6 -0
  99. agno/guardrails/base.py +19 -0
  100. agno/guardrails/openai.py +144 -0
  101. agno/guardrails/pii.py +94 -0
  102. agno/guardrails/prompt_injection.py +52 -0
  103. agno/integrations/__init__.py +0 -0
  104. agno/integrations/discord/__init__.py +3 -0
  105. agno/integrations/discord/client.py +203 -0
  106. agno/knowledge/__init__.py +5 -0
  107. agno/knowledge/chunking/__init__.py +0 -0
  108. agno/knowledge/chunking/agentic.py +79 -0
  109. agno/knowledge/chunking/document.py +91 -0
  110. agno/knowledge/chunking/fixed.py +57 -0
  111. agno/knowledge/chunking/markdown.py +151 -0
  112. agno/knowledge/chunking/recursive.py +63 -0
  113. agno/knowledge/chunking/row.py +39 -0
  114. agno/knowledge/chunking/semantic.py +86 -0
  115. agno/knowledge/chunking/strategy.py +165 -0
  116. agno/knowledge/content.py +74 -0
  117. agno/knowledge/document/__init__.py +5 -0
  118. agno/knowledge/document/base.py +58 -0
  119. agno/knowledge/embedder/__init__.py +5 -0
  120. agno/knowledge/embedder/aws_bedrock.py +343 -0
  121. agno/knowledge/embedder/azure_openai.py +210 -0
  122. agno/knowledge/embedder/base.py +23 -0
  123. agno/knowledge/embedder/cohere.py +323 -0
  124. agno/knowledge/embedder/fastembed.py +62 -0
  125. agno/knowledge/embedder/fireworks.py +13 -0
  126. agno/knowledge/embedder/google.py +258 -0
  127. agno/knowledge/embedder/huggingface.py +94 -0
  128. agno/knowledge/embedder/jina.py +182 -0
  129. agno/knowledge/embedder/langdb.py +22 -0
  130. agno/knowledge/embedder/mistral.py +206 -0
  131. agno/knowledge/embedder/nebius.py +13 -0
  132. agno/knowledge/embedder/ollama.py +154 -0
  133. agno/knowledge/embedder/openai.py +195 -0
  134. agno/knowledge/embedder/sentence_transformer.py +63 -0
  135. agno/knowledge/embedder/together.py +13 -0
  136. agno/knowledge/embedder/vllm.py +262 -0
  137. agno/knowledge/embedder/voyageai.py +165 -0
  138. agno/knowledge/knowledge.py +1988 -0
  139. agno/knowledge/reader/__init__.py +7 -0
  140. agno/knowledge/reader/arxiv_reader.py +81 -0
  141. agno/knowledge/reader/base.py +95 -0
  142. agno/knowledge/reader/csv_reader.py +166 -0
  143. agno/knowledge/reader/docx_reader.py +82 -0
  144. agno/knowledge/reader/field_labeled_csv_reader.py +292 -0
  145. agno/knowledge/reader/firecrawl_reader.py +201 -0
  146. agno/knowledge/reader/json_reader.py +87 -0
  147. agno/knowledge/reader/markdown_reader.py +137 -0
  148. agno/knowledge/reader/pdf_reader.py +431 -0
  149. agno/knowledge/reader/pptx_reader.py +101 -0
  150. agno/knowledge/reader/reader_factory.py +313 -0
  151. agno/knowledge/reader/s3_reader.py +89 -0
  152. agno/knowledge/reader/tavily_reader.py +194 -0
  153. agno/knowledge/reader/text_reader.py +115 -0
  154. agno/knowledge/reader/web_search_reader.py +372 -0
  155. agno/knowledge/reader/website_reader.py +455 -0
  156. agno/knowledge/reader/wikipedia_reader.py +59 -0
  157. agno/knowledge/reader/youtube_reader.py +78 -0
  158. agno/knowledge/remote_content/__init__.py +0 -0
  159. agno/knowledge/remote_content/remote_content.py +88 -0
  160. agno/knowledge/reranker/__init__.py +3 -0
  161. agno/knowledge/reranker/base.py +14 -0
  162. agno/knowledge/reranker/cohere.py +64 -0
  163. agno/knowledge/reranker/infinity.py +195 -0
  164. agno/knowledge/reranker/sentence_transformer.py +54 -0
  165. agno/knowledge/types.py +39 -0
  166. agno/knowledge/utils.py +189 -0
  167. agno/media.py +462 -0
  168. agno/memory/__init__.py +3 -0
  169. agno/memory/manager.py +1327 -0
  170. agno/models/__init__.py +0 -0
  171. agno/models/aimlapi/__init__.py +5 -0
  172. agno/models/aimlapi/aimlapi.py +45 -0
  173. agno/models/anthropic/__init__.py +5 -0
  174. agno/models/anthropic/claude.py +757 -0
  175. agno/models/aws/__init__.py +15 -0
  176. agno/models/aws/bedrock.py +701 -0
  177. agno/models/aws/claude.py +378 -0
  178. agno/models/azure/__init__.py +18 -0
  179. agno/models/azure/ai_foundry.py +485 -0
  180. agno/models/azure/openai_chat.py +131 -0
  181. agno/models/base.py +2175 -0
  182. agno/models/cerebras/__init__.py +12 -0
  183. agno/models/cerebras/cerebras.py +501 -0
  184. agno/models/cerebras/cerebras_openai.py +112 -0
  185. agno/models/cohere/__init__.py +5 -0
  186. agno/models/cohere/chat.py +389 -0
  187. agno/models/cometapi/__init__.py +5 -0
  188. agno/models/cometapi/cometapi.py +57 -0
  189. agno/models/dashscope/__init__.py +5 -0
  190. agno/models/dashscope/dashscope.py +91 -0
  191. agno/models/deepinfra/__init__.py +5 -0
  192. agno/models/deepinfra/deepinfra.py +28 -0
  193. agno/models/deepseek/__init__.py +5 -0
  194. agno/models/deepseek/deepseek.py +61 -0
  195. agno/models/defaults.py +1 -0
  196. agno/models/fireworks/__init__.py +5 -0
  197. agno/models/fireworks/fireworks.py +26 -0
  198. agno/models/google/__init__.py +5 -0
  199. agno/models/google/gemini.py +1085 -0
  200. agno/models/groq/__init__.py +5 -0
  201. agno/models/groq/groq.py +556 -0
  202. agno/models/huggingface/__init__.py +5 -0
  203. agno/models/huggingface/huggingface.py +491 -0
  204. agno/models/ibm/__init__.py +5 -0
  205. agno/models/ibm/watsonx.py +422 -0
  206. agno/models/internlm/__init__.py +3 -0
  207. agno/models/internlm/internlm.py +26 -0
  208. agno/models/langdb/__init__.py +1 -0
  209. agno/models/langdb/langdb.py +48 -0
  210. agno/models/litellm/__init__.py +14 -0
  211. agno/models/litellm/chat.py +468 -0
  212. agno/models/litellm/litellm_openai.py +25 -0
  213. agno/models/llama_cpp/__init__.py +5 -0
  214. agno/models/llama_cpp/llama_cpp.py +22 -0
  215. agno/models/lmstudio/__init__.py +5 -0
  216. agno/models/lmstudio/lmstudio.py +25 -0
  217. agno/models/message.py +434 -0
  218. agno/models/meta/__init__.py +12 -0
  219. agno/models/meta/llama.py +475 -0
  220. agno/models/meta/llama_openai.py +78 -0
  221. agno/models/metrics.py +120 -0
  222. agno/models/mistral/__init__.py +5 -0
  223. agno/models/mistral/mistral.py +432 -0
  224. agno/models/nebius/__init__.py +3 -0
  225. agno/models/nebius/nebius.py +54 -0
  226. agno/models/nexus/__init__.py +3 -0
  227. agno/models/nexus/nexus.py +22 -0
  228. agno/models/nvidia/__init__.py +5 -0
  229. agno/models/nvidia/nvidia.py +28 -0
  230. agno/models/ollama/__init__.py +5 -0
  231. agno/models/ollama/chat.py +441 -0
  232. agno/models/openai/__init__.py +9 -0
  233. agno/models/openai/chat.py +883 -0
  234. agno/models/openai/like.py +27 -0
  235. agno/models/openai/responses.py +1050 -0
  236. agno/models/openrouter/__init__.py +5 -0
  237. agno/models/openrouter/openrouter.py +66 -0
  238. agno/models/perplexity/__init__.py +5 -0
  239. agno/models/perplexity/perplexity.py +187 -0
  240. agno/models/portkey/__init__.py +3 -0
  241. agno/models/portkey/portkey.py +81 -0
  242. agno/models/requesty/__init__.py +5 -0
  243. agno/models/requesty/requesty.py +52 -0
  244. agno/models/response.py +199 -0
  245. agno/models/sambanova/__init__.py +5 -0
  246. agno/models/sambanova/sambanova.py +28 -0
  247. agno/models/siliconflow/__init__.py +5 -0
  248. agno/models/siliconflow/siliconflow.py +25 -0
  249. agno/models/together/__init__.py +5 -0
  250. agno/models/together/together.py +25 -0
  251. agno/models/utils.py +266 -0
  252. agno/models/vercel/__init__.py +3 -0
  253. agno/models/vercel/v0.py +26 -0
  254. agno/models/vertexai/__init__.py +0 -0
  255. agno/models/vertexai/claude.py +70 -0
  256. agno/models/vllm/__init__.py +3 -0
  257. agno/models/vllm/vllm.py +78 -0
  258. agno/models/xai/__init__.py +3 -0
  259. agno/models/xai/xai.py +113 -0
  260. agno/os/__init__.py +3 -0
  261. agno/os/app.py +876 -0
  262. agno/os/auth.py +57 -0
  263. agno/os/config.py +104 -0
  264. agno/os/interfaces/__init__.py +1 -0
  265. agno/os/interfaces/a2a/__init__.py +3 -0
  266. agno/os/interfaces/a2a/a2a.py +42 -0
  267. agno/os/interfaces/a2a/router.py +250 -0
  268. agno/os/interfaces/a2a/utils.py +924 -0
  269. agno/os/interfaces/agui/__init__.py +3 -0
  270. agno/os/interfaces/agui/agui.py +47 -0
  271. agno/os/interfaces/agui/router.py +144 -0
  272. agno/os/interfaces/agui/utils.py +534 -0
  273. agno/os/interfaces/base.py +25 -0
  274. agno/os/interfaces/slack/__init__.py +3 -0
  275. agno/os/interfaces/slack/router.py +148 -0
  276. agno/os/interfaces/slack/security.py +30 -0
  277. agno/os/interfaces/slack/slack.py +47 -0
  278. agno/os/interfaces/whatsapp/__init__.py +3 -0
  279. agno/os/interfaces/whatsapp/router.py +211 -0
  280. agno/os/interfaces/whatsapp/security.py +53 -0
  281. agno/os/interfaces/whatsapp/whatsapp.py +36 -0
  282. agno/os/mcp.py +292 -0
  283. agno/os/middleware/__init__.py +7 -0
  284. agno/os/middleware/jwt.py +233 -0
  285. agno/os/router.py +1763 -0
  286. agno/os/routers/__init__.py +3 -0
  287. agno/os/routers/evals/__init__.py +3 -0
  288. agno/os/routers/evals/evals.py +430 -0
  289. agno/os/routers/evals/schemas.py +142 -0
  290. agno/os/routers/evals/utils.py +162 -0
  291. agno/os/routers/health.py +31 -0
  292. agno/os/routers/home.py +52 -0
  293. agno/os/routers/knowledge/__init__.py +3 -0
  294. agno/os/routers/knowledge/knowledge.py +997 -0
  295. agno/os/routers/knowledge/schemas.py +178 -0
  296. agno/os/routers/memory/__init__.py +3 -0
  297. agno/os/routers/memory/memory.py +515 -0
  298. agno/os/routers/memory/schemas.py +62 -0
  299. agno/os/routers/metrics/__init__.py +3 -0
  300. agno/os/routers/metrics/metrics.py +190 -0
  301. agno/os/routers/metrics/schemas.py +47 -0
  302. agno/os/routers/session/__init__.py +3 -0
  303. agno/os/routers/session/session.py +997 -0
  304. agno/os/schema.py +1055 -0
  305. agno/os/settings.py +43 -0
  306. agno/os/utils.py +630 -0
  307. agno/py.typed +0 -0
  308. agno/reasoning/__init__.py +0 -0
  309. agno/reasoning/anthropic.py +80 -0
  310. agno/reasoning/azure_ai_foundry.py +67 -0
  311. agno/reasoning/deepseek.py +63 -0
  312. agno/reasoning/default.py +97 -0
  313. agno/reasoning/gemini.py +73 -0
  314. agno/reasoning/groq.py +71 -0
  315. agno/reasoning/helpers.py +63 -0
  316. agno/reasoning/ollama.py +67 -0
  317. agno/reasoning/openai.py +86 -0
  318. agno/reasoning/step.py +31 -0
  319. agno/reasoning/vertexai.py +76 -0
  320. agno/run/__init__.py +6 -0
  321. agno/run/agent.py +787 -0
  322. agno/run/base.py +229 -0
  323. agno/run/cancel.py +81 -0
  324. agno/run/messages.py +32 -0
  325. agno/run/team.py +753 -0
  326. agno/run/workflow.py +708 -0
  327. agno/session/__init__.py +10 -0
  328. agno/session/agent.py +295 -0
  329. agno/session/summary.py +265 -0
  330. agno/session/team.py +392 -0
  331. agno/session/workflow.py +205 -0
  332. agno/team/__init__.py +37 -0
  333. agno/team/team.py +8793 -0
  334. agno/tools/__init__.py +10 -0
  335. agno/tools/agentql.py +120 -0
  336. agno/tools/airflow.py +69 -0
  337. agno/tools/api.py +122 -0
  338. agno/tools/apify.py +314 -0
  339. agno/tools/arxiv.py +127 -0
  340. agno/tools/aws_lambda.py +53 -0
  341. agno/tools/aws_ses.py +66 -0
  342. agno/tools/baidusearch.py +89 -0
  343. agno/tools/bitbucket.py +292 -0
  344. agno/tools/brandfetch.py +213 -0
  345. agno/tools/bravesearch.py +106 -0
  346. agno/tools/brightdata.py +367 -0
  347. agno/tools/browserbase.py +209 -0
  348. agno/tools/calcom.py +255 -0
  349. agno/tools/calculator.py +151 -0
  350. agno/tools/cartesia.py +187 -0
  351. agno/tools/clickup.py +244 -0
  352. agno/tools/confluence.py +240 -0
  353. agno/tools/crawl4ai.py +158 -0
  354. agno/tools/csv_toolkit.py +185 -0
  355. agno/tools/dalle.py +110 -0
  356. agno/tools/daytona.py +475 -0
  357. agno/tools/decorator.py +262 -0
  358. agno/tools/desi_vocal.py +108 -0
  359. agno/tools/discord.py +161 -0
  360. agno/tools/docker.py +716 -0
  361. agno/tools/duckdb.py +379 -0
  362. agno/tools/duckduckgo.py +91 -0
  363. agno/tools/e2b.py +703 -0
  364. agno/tools/eleven_labs.py +196 -0
  365. agno/tools/email.py +67 -0
  366. agno/tools/evm.py +129 -0
  367. agno/tools/exa.py +396 -0
  368. agno/tools/fal.py +127 -0
  369. agno/tools/file.py +240 -0
  370. agno/tools/file_generation.py +350 -0
  371. agno/tools/financial_datasets.py +288 -0
  372. agno/tools/firecrawl.py +143 -0
  373. agno/tools/function.py +1187 -0
  374. agno/tools/giphy.py +93 -0
  375. agno/tools/github.py +1760 -0
  376. agno/tools/gmail.py +922 -0
  377. agno/tools/google_bigquery.py +117 -0
  378. agno/tools/google_drive.py +270 -0
  379. agno/tools/google_maps.py +253 -0
  380. agno/tools/googlecalendar.py +674 -0
  381. agno/tools/googlesearch.py +98 -0
  382. agno/tools/googlesheets.py +377 -0
  383. agno/tools/hackernews.py +77 -0
  384. agno/tools/jina.py +101 -0
  385. agno/tools/jira.py +170 -0
  386. agno/tools/knowledge.py +218 -0
  387. agno/tools/linear.py +426 -0
  388. agno/tools/linkup.py +58 -0
  389. agno/tools/local_file_system.py +90 -0
  390. agno/tools/lumalab.py +183 -0
  391. agno/tools/mcp/__init__.py +10 -0
  392. agno/tools/mcp/mcp.py +331 -0
  393. agno/tools/mcp/multi_mcp.py +347 -0
  394. agno/tools/mcp/params.py +24 -0
  395. agno/tools/mcp_toolbox.py +284 -0
  396. agno/tools/mem0.py +193 -0
  397. agno/tools/memori.py +339 -0
  398. agno/tools/memory.py +419 -0
  399. agno/tools/mlx_transcribe.py +139 -0
  400. agno/tools/models/__init__.py +0 -0
  401. agno/tools/models/azure_openai.py +190 -0
  402. agno/tools/models/gemini.py +203 -0
  403. agno/tools/models/groq.py +158 -0
  404. agno/tools/models/morph.py +186 -0
  405. agno/tools/models/nebius.py +124 -0
  406. agno/tools/models_labs.py +195 -0
  407. agno/tools/moviepy_video.py +349 -0
  408. agno/tools/neo4j.py +134 -0
  409. agno/tools/newspaper.py +46 -0
  410. agno/tools/newspaper4k.py +93 -0
  411. agno/tools/notion.py +204 -0
  412. agno/tools/openai.py +202 -0
  413. agno/tools/openbb.py +160 -0
  414. agno/tools/opencv.py +321 -0
  415. agno/tools/openweather.py +233 -0
  416. agno/tools/oxylabs.py +385 -0
  417. agno/tools/pandas.py +102 -0
  418. agno/tools/parallel.py +314 -0
  419. agno/tools/postgres.py +257 -0
  420. agno/tools/pubmed.py +188 -0
  421. agno/tools/python.py +205 -0
  422. agno/tools/reasoning.py +283 -0
  423. agno/tools/reddit.py +467 -0
  424. agno/tools/replicate.py +117 -0
  425. agno/tools/resend.py +62 -0
  426. agno/tools/scrapegraph.py +222 -0
  427. agno/tools/searxng.py +152 -0
  428. agno/tools/serpapi.py +116 -0
  429. agno/tools/serper.py +255 -0
  430. agno/tools/shell.py +53 -0
  431. agno/tools/slack.py +136 -0
  432. agno/tools/sleep.py +20 -0
  433. agno/tools/spider.py +116 -0
  434. agno/tools/sql.py +154 -0
  435. agno/tools/streamlit/__init__.py +0 -0
  436. agno/tools/streamlit/components.py +113 -0
  437. agno/tools/tavily.py +254 -0
  438. agno/tools/telegram.py +48 -0
  439. agno/tools/todoist.py +218 -0
  440. agno/tools/tool_registry.py +1 -0
  441. agno/tools/toolkit.py +146 -0
  442. agno/tools/trafilatura.py +388 -0
  443. agno/tools/trello.py +274 -0
  444. agno/tools/twilio.py +186 -0
  445. agno/tools/user_control_flow.py +78 -0
  446. agno/tools/valyu.py +228 -0
  447. agno/tools/visualization.py +467 -0
  448. agno/tools/webbrowser.py +28 -0
  449. agno/tools/webex.py +76 -0
  450. agno/tools/website.py +54 -0
  451. agno/tools/webtools.py +45 -0
  452. agno/tools/whatsapp.py +286 -0
  453. agno/tools/wikipedia.py +63 -0
  454. agno/tools/workflow.py +278 -0
  455. agno/tools/x.py +335 -0
  456. agno/tools/yfinance.py +257 -0
  457. agno/tools/youtube.py +184 -0
  458. agno/tools/zendesk.py +82 -0
  459. agno/tools/zep.py +454 -0
  460. agno/tools/zoom.py +382 -0
  461. agno/utils/__init__.py +0 -0
  462. agno/utils/agent.py +820 -0
  463. agno/utils/audio.py +49 -0
  464. agno/utils/certs.py +27 -0
  465. agno/utils/code_execution.py +11 -0
  466. agno/utils/common.py +132 -0
  467. agno/utils/dttm.py +13 -0
  468. agno/utils/enum.py +22 -0
  469. agno/utils/env.py +11 -0
  470. agno/utils/events.py +696 -0
  471. agno/utils/format_str.py +16 -0
  472. agno/utils/functions.py +166 -0
  473. agno/utils/gemini.py +426 -0
  474. agno/utils/hooks.py +57 -0
  475. agno/utils/http.py +74 -0
  476. agno/utils/json_schema.py +234 -0
  477. agno/utils/knowledge.py +36 -0
  478. agno/utils/location.py +19 -0
  479. agno/utils/log.py +255 -0
  480. agno/utils/mcp.py +214 -0
  481. agno/utils/media.py +352 -0
  482. agno/utils/merge_dict.py +41 -0
  483. agno/utils/message.py +118 -0
  484. agno/utils/models/__init__.py +0 -0
  485. agno/utils/models/ai_foundry.py +43 -0
  486. agno/utils/models/claude.py +358 -0
  487. agno/utils/models/cohere.py +87 -0
  488. agno/utils/models/llama.py +78 -0
  489. agno/utils/models/mistral.py +98 -0
  490. agno/utils/models/openai_responses.py +140 -0
  491. agno/utils/models/schema_utils.py +153 -0
  492. agno/utils/models/watsonx.py +41 -0
  493. agno/utils/openai.py +257 -0
  494. agno/utils/pickle.py +32 -0
  495. agno/utils/pprint.py +178 -0
  496. agno/utils/print_response/__init__.py +0 -0
  497. agno/utils/print_response/agent.py +842 -0
  498. agno/utils/print_response/team.py +1724 -0
  499. agno/utils/print_response/workflow.py +1668 -0
  500. agno/utils/prompts.py +111 -0
  501. agno/utils/reasoning.py +108 -0
  502. agno/utils/response.py +163 -0
  503. agno/utils/response_iterator.py +17 -0
  504. agno/utils/safe_formatter.py +24 -0
  505. agno/utils/serialize.py +32 -0
  506. agno/utils/shell.py +22 -0
  507. agno/utils/streamlit.py +487 -0
  508. agno/utils/string.py +231 -0
  509. agno/utils/team.py +139 -0
  510. agno/utils/timer.py +41 -0
  511. agno/utils/tools.py +102 -0
  512. agno/utils/web.py +23 -0
  513. agno/utils/whatsapp.py +305 -0
  514. agno/utils/yaml_io.py +25 -0
  515. agno/vectordb/__init__.py +3 -0
  516. agno/vectordb/base.py +127 -0
  517. agno/vectordb/cassandra/__init__.py +5 -0
  518. agno/vectordb/cassandra/cassandra.py +501 -0
  519. agno/vectordb/cassandra/extra_param_mixin.py +11 -0
  520. agno/vectordb/cassandra/index.py +13 -0
  521. agno/vectordb/chroma/__init__.py +5 -0
  522. agno/vectordb/chroma/chromadb.py +929 -0
  523. agno/vectordb/clickhouse/__init__.py +9 -0
  524. agno/vectordb/clickhouse/clickhousedb.py +835 -0
  525. agno/vectordb/clickhouse/index.py +9 -0
  526. agno/vectordb/couchbase/__init__.py +3 -0
  527. agno/vectordb/couchbase/couchbase.py +1442 -0
  528. agno/vectordb/distance.py +7 -0
  529. agno/vectordb/lancedb/__init__.py +6 -0
  530. agno/vectordb/lancedb/lance_db.py +995 -0
  531. agno/vectordb/langchaindb/__init__.py +5 -0
  532. agno/vectordb/langchaindb/langchaindb.py +163 -0
  533. agno/vectordb/lightrag/__init__.py +5 -0
  534. agno/vectordb/lightrag/lightrag.py +388 -0
  535. agno/vectordb/llamaindex/__init__.py +3 -0
  536. agno/vectordb/llamaindex/llamaindexdb.py +166 -0
  537. agno/vectordb/milvus/__init__.py +4 -0
  538. agno/vectordb/milvus/milvus.py +1182 -0
  539. agno/vectordb/mongodb/__init__.py +9 -0
  540. agno/vectordb/mongodb/mongodb.py +1417 -0
  541. agno/vectordb/pgvector/__init__.py +12 -0
  542. agno/vectordb/pgvector/index.py +23 -0
  543. agno/vectordb/pgvector/pgvector.py +1462 -0
  544. agno/vectordb/pineconedb/__init__.py +5 -0
  545. agno/vectordb/pineconedb/pineconedb.py +747 -0
  546. agno/vectordb/qdrant/__init__.py +5 -0
  547. agno/vectordb/qdrant/qdrant.py +1134 -0
  548. agno/vectordb/redis/__init__.py +9 -0
  549. agno/vectordb/redis/redisdb.py +694 -0
  550. agno/vectordb/search.py +7 -0
  551. agno/vectordb/singlestore/__init__.py +10 -0
  552. agno/vectordb/singlestore/index.py +41 -0
  553. agno/vectordb/singlestore/singlestore.py +763 -0
  554. agno/vectordb/surrealdb/__init__.py +3 -0
  555. agno/vectordb/surrealdb/surrealdb.py +699 -0
  556. agno/vectordb/upstashdb/__init__.py +5 -0
  557. agno/vectordb/upstashdb/upstashdb.py +718 -0
  558. agno/vectordb/weaviate/__init__.py +8 -0
  559. agno/vectordb/weaviate/index.py +15 -0
  560. agno/vectordb/weaviate/weaviate.py +1005 -0
  561. agno/workflow/__init__.py +23 -0
  562. agno/workflow/agent.py +299 -0
  563. agno/workflow/condition.py +738 -0
  564. agno/workflow/loop.py +735 -0
  565. agno/workflow/parallel.py +824 -0
  566. agno/workflow/router.py +702 -0
  567. agno/workflow/step.py +1432 -0
  568. agno/workflow/steps.py +592 -0
  569. agno/workflow/types.py +520 -0
  570. agno/workflow/workflow.py +4321 -0
  571. agno-2.2.13.dist-info/METADATA +614 -0
  572. agno-2.2.13.dist-info/RECORD +575 -0
  573. agno-2.2.13.dist-info/WHEEL +5 -0
  574. agno-2.2.13.dist-info/licenses/LICENSE +201 -0
  575. agno-2.2.13.dist-info/top_level.txt +1 -0
agno/tools/sql.py ADDED
@@ -0,0 +1,154 @@
1
+ import json
2
+ from typing import Any, Dict, List, Optional
3
+
4
+ from agno.tools import Toolkit
5
+ from agno.utils.log import log_debug, logger
6
+
7
+ try:
8
+ from sqlalchemy import Engine, create_engine
9
+ from sqlalchemy.inspection import inspect
10
+ from sqlalchemy.orm import Session, sessionmaker
11
+ from sqlalchemy.sql.expression import text
12
+ except ImportError:
13
+ raise ImportError("`sqlalchemy` not installed")
14
+
15
+
16
+ class SQLTools(Toolkit):
17
+ def __init__(
18
+ self,
19
+ db_url: Optional[str] = None,
20
+ db_engine: Optional[Engine] = None,
21
+ user: Optional[str] = None,
22
+ password: Optional[str] = None,
23
+ host: Optional[str] = None,
24
+ port: Optional[int] = None,
25
+ schema: Optional[str] = None,
26
+ dialect: Optional[str] = None,
27
+ tables: Optional[Dict[str, Any]] = None,
28
+ enable_list_tables: bool = True,
29
+ enable_describe_table: bool = True,
30
+ enable_run_sql_query: bool = True,
31
+ all: bool = False,
32
+ **kwargs,
33
+ ):
34
+ # Get the database engine
35
+ _engine: Optional[Engine] = db_engine
36
+ if _engine is None and db_url is not None:
37
+ _engine = create_engine(db_url)
38
+ elif user and password and host and port and dialect:
39
+ if schema is not None:
40
+ _engine = create_engine(f"{dialect}://{user}:{password}@{host}:{port}/{schema}")
41
+ else:
42
+ _engine = create_engine(f"{dialect}://{user}:{password}@{host}:{port}")
43
+
44
+ if _engine is None:
45
+ raise ValueError("Could not build the database connection")
46
+
47
+ # Database connection
48
+ self.db_engine: Engine = _engine
49
+ self.Session: sessionmaker[Session] = sessionmaker(bind=self.db_engine)
50
+
51
+ self.schema = schema
52
+
53
+ # Tables this toolkit can access
54
+ self.tables: Optional[Dict[str, Any]] = tables
55
+
56
+ tools: List[Any] = []
57
+ if enable_list_tables or all:
58
+ tools.append(self.list_tables)
59
+ if enable_describe_table or all:
60
+ tools.append(self.describe_table)
61
+ if enable_run_sql_query or all:
62
+ tools.append(self.run_sql_query)
63
+
64
+ super().__init__(name="sql_tools", tools=tools, **kwargs)
65
+
66
+ def list_tables(self) -> str:
67
+ """Use this function to get a list of table names in the database.
68
+
69
+ Returns:
70
+ str: list of tables in the database.
71
+ """
72
+ if self.tables is not None:
73
+ return json.dumps(self.tables)
74
+
75
+ try:
76
+ log_debug("listing tables in the database")
77
+ inspector = inspect(self.db_engine)
78
+ if self.schema:
79
+ table_names = inspector.get_table_names(schema=self.schema)
80
+ else:
81
+ table_names = inspector.get_table_names()
82
+ log_debug(f"table_names: {table_names}")
83
+ return json.dumps(table_names)
84
+ except Exception as e:
85
+ logger.error(f"Error getting tables: {e}")
86
+ return f"Error getting tables: {e}"
87
+
88
+ def describe_table(self, table_name: str) -> str:
89
+ """Use this function to describe a table.
90
+
91
+ Args:
92
+ table_name (str): The name of the table to get the schema for.
93
+
94
+ Returns:
95
+ str: schema of a table
96
+ """
97
+
98
+ try:
99
+ log_debug(f"Describing table: {table_name}")
100
+ inspector = inspect(self.db_engine)
101
+ table_schema = inspector.get_columns(table_name, schema=self.schema)
102
+ return json.dumps(
103
+ [
104
+ {"name": column["name"], "type": str(column["type"]), "nullable": column["nullable"]}
105
+ for column in table_schema
106
+ ]
107
+ )
108
+ except Exception as e:
109
+ logger.error(f"Error getting table schema: {e}")
110
+ return f"Error getting table schema: {e}"
111
+
112
+ def run_sql_query(self, query: str, limit: Optional[int] = 10) -> str:
113
+ """Use this function to run a SQL query and return the result.
114
+
115
+ Args:
116
+ query (str): The query to run.
117
+ limit (int, optional): The number of rows to return. Defaults to 10. Use `None` to show all results.
118
+ Returns:
119
+ str: Result of the SQL query.
120
+ Notes:
121
+ - The result may be empty if the query does not return any data.
122
+ """
123
+
124
+ try:
125
+ return json.dumps(self.run_sql(sql=query, limit=limit), default=str)
126
+ except Exception as e:
127
+ logger.error(f"Error running query: {e}")
128
+ return f"Error running query: {e}"
129
+
130
+ def run_sql(self, sql: str, limit: Optional[int] = None) -> List[dict]:
131
+ """Internal function to run a sql query.
132
+
133
+ Args:
134
+ sql (str): The sql query to run.
135
+ limit (int, optional): The number of rows to return. Defaults to None.
136
+
137
+ Returns:
138
+ List[dict]: The result of the query.
139
+ """
140
+ log_debug(f"Running sql |\n{sql}")
141
+
142
+ with self.Session() as sess, sess.begin():
143
+ result = sess.execute(text(sql))
144
+
145
+ # Check if the operation has returned rows.
146
+ try:
147
+ if limit:
148
+ rows = result.fetchmany(limit)
149
+ else:
150
+ rows = result.fetchall()
151
+ return [row._asdict() for row in rows]
152
+ except Exception as e:
153
+ logger.error(f"Error while executing SQL: {e}")
154
+ return []
File without changes
@@ -0,0 +1,113 @@
1
+ from os import environ, getenv
2
+ from typing import Optional
3
+
4
+ try:
5
+ import streamlit as st
6
+ except ImportError:
7
+ raise ImportError("`streamlit` library not installed. Please install using `pip install streamlit`")
8
+
9
+
10
+ def get_username_sidebar() -> Optional[str]:
11
+ """Sidebar component to get username"""
12
+
13
+ # Get username from user if not in session state
14
+ if "username" not in st.session_state:
15
+ username_input_container = st.sidebar.empty()
16
+ username = username_input_container.text_input(":technologist: Enter username")
17
+ if username != "":
18
+ st.session_state["username"] = username
19
+ username_input_container.empty()
20
+
21
+ # Get username from session state
22
+ username = st.session_state.get("username") # type: ignore
23
+ return username
24
+
25
+
26
+ def reload_button_sidebar(text: str = "Reload Session", **kwargs) -> None:
27
+ """Sidebar component to show reload button"""
28
+
29
+ if st.sidebar.button(text, **kwargs):
30
+ st.session_state.clear()
31
+ st.rerun()
32
+
33
+
34
+ def check_password(password_env_var: str = "APP_PASSWORD") -> bool:
35
+ """Component to check if a password entered by the user is correct.
36
+ To use this component, set the environment variable `APP_PASSWORD`.
37
+
38
+ Args:
39
+ password_env_var (str, optional): The environment variable to use for the password. Defaults to "APP_PASSWORD".
40
+
41
+ Returns:
42
+ bool: `True` if the user had the correct password.
43
+ """
44
+
45
+ app_password = getenv(password_env_var)
46
+ if app_password is None:
47
+ return True
48
+
49
+ def check_first_run_password():
50
+ """Checks whether a password entered on the first run is correct."""
51
+
52
+ if "first_run_password" in st.session_state:
53
+ password_to_check = st.session_state["first_run_password"]
54
+ if password_to_check == app_password:
55
+ st.session_state["password_correct"] = True
56
+ # don't store password
57
+ del st.session_state["first_run_password"]
58
+ else:
59
+ st.session_state["password_correct"] = False
60
+
61
+ def check_updated_password():
62
+ """Checks whether an updated password is correct."""
63
+
64
+ if "updated_password" in st.session_state:
65
+ password_to_check = st.session_state["updated_password"]
66
+ if password_to_check == app_password:
67
+ st.session_state["password_correct"] = True
68
+ # don't store password
69
+ del st.session_state["updated_password"]
70
+ else:
71
+ st.session_state["password_correct"] = False
72
+
73
+ # First run, show input for password.
74
+ if "password_correct" not in st.session_state:
75
+ st.text_input(
76
+ "Password",
77
+ type="password",
78
+ on_change=check_first_run_password,
79
+ key="first_run_password",
80
+ )
81
+ return False
82
+ # Password incorrect, show input for updated password + error.
83
+ elif not st.session_state["password_correct"]:
84
+ st.text_input(
85
+ "Password",
86
+ type="password",
87
+ on_change=check_updated_password,
88
+ key="updated_password",
89
+ )
90
+ st.error("😕 Password incorrect")
91
+ return False
92
+ # Password correct.
93
+ else:
94
+ return True
95
+
96
+
97
+ def get_openai_key_sidebar() -> Optional[str]:
98
+ """Sidebar component to get OpenAI API key"""
99
+
100
+ # Get OpenAI API key from environment variable
101
+ openai_key: Optional[str] = getenv("OPENAI_API_KEY")
102
+ # If not found, get it from user input
103
+ if openai_key is None or openai_key == "" or openai_key == "sk-***":
104
+ api_key = st.sidebar.text_input("OpenAI API key", placeholder="sk-***", key="api_key")
105
+ if api_key != "sk-***" or api_key != "" or api_key is not None:
106
+ openai_key = api_key
107
+
108
+ # Store it in session state and environment variable
109
+ if openai_key is not None and openai_key != "":
110
+ st.session_state["OPENAI_API_KEY"] = openai_key
111
+ environ["OPENAI_API_KEY"] = openai_key
112
+
113
+ return openai_key
agno/tools/tavily.py ADDED
@@ -0,0 +1,254 @@
1
+ import json
2
+ from os import getenv
3
+ from typing import Any, Dict, List, Literal, Optional
4
+
5
+ from agno.tools import Toolkit
6
+ from agno.utils.log import logger
7
+
8
+ try:
9
+ from tavily import TavilyClient
10
+ except ImportError:
11
+ raise ImportError("`tavily-python` not installed. Please install using `pip install tavily-python`")
12
+
13
+
14
+ class TavilyTools(Toolkit):
15
+ def __init__(
16
+ self,
17
+ api_key: Optional[str] = None,
18
+ enable_search: bool = True,
19
+ enable_search_context: bool = False,
20
+ enable_extract: bool = False,
21
+ all: bool = False,
22
+ max_tokens: int = 6000,
23
+ include_answer: bool = True,
24
+ search_depth: Literal["basic", "advanced"] = "advanced",
25
+ extract_depth: Literal["basic", "advanced"] = "basic",
26
+ include_images: bool = False,
27
+ include_favicon: bool = False,
28
+ extract_timeout: Optional[int] = None,
29
+ extract_format: Literal["markdown", "text"] = "markdown",
30
+ format: Literal["json", "markdown"] = "markdown",
31
+ **kwargs,
32
+ ):
33
+ """Initialize TavilyTools with search and extract capabilities.
34
+
35
+ Args:
36
+ api_key: Tavily API key. If not provided, will use TAVILY_API_KEY env var.
37
+ enable_search: Enable web search functionality. Defaults to True.
38
+ enable_search_context: Use search context mode instead of regular search. Defaults to False.
39
+ enable_extract: Enable URL content extraction functionality. Defaults to False.
40
+ all: Enable all available tools. Defaults to False.
41
+ max_tokens: Maximum tokens for search results. Defaults to 6000.
42
+ include_answer: Include AI-generated answer in search results. Defaults to True.
43
+ search_depth: Search depth level - basic (1 credit) or advanced (2 credits). Defaults to "advanced".
44
+ extract_depth: Extract depth level - basic (1 credit/5 URLs) or advanced (2 credits/5 URLs). Defaults to "basic".
45
+ include_images: Include images in extracted content. Defaults to False.
46
+ include_favicon: Include favicon in extracted content. Defaults to False.
47
+ extract_timeout: Timeout in seconds for extraction requests. Defaults to None.
48
+ extract_format: Output format for extracted content - markdown or text. Defaults to "markdown".
49
+ format: Output format for search results - json or markdown. Defaults to "markdown".
50
+ **kwargs: Additional arguments passed to Toolkit.
51
+ """
52
+ self.api_key = api_key or getenv("TAVILY_API_KEY")
53
+ if not self.api_key:
54
+ logger.error("TAVILY_API_KEY not provided")
55
+
56
+ self.client: TavilyClient = TavilyClient(api_key=self.api_key)
57
+ self.search_depth: Literal["basic", "advanced"] = search_depth
58
+ self.extract_depth: Literal["basic", "advanced"] = extract_depth
59
+ self.max_tokens: int = max_tokens
60
+ self.include_answer: bool = include_answer
61
+ self.include_images: bool = include_images
62
+ self.include_favicon: bool = include_favicon
63
+ self.extract_timeout: Optional[int] = extract_timeout
64
+ self.extract_format: Literal["markdown", "text"] = extract_format
65
+ self.format: Literal["json", "markdown"] = format
66
+
67
+ tools: List[Any] = []
68
+
69
+ if enable_search or all:
70
+ if enable_search_context:
71
+ tools.append(self.web_search_with_tavily)
72
+ else:
73
+ tools.append(self.web_search_using_tavily)
74
+
75
+ if enable_extract or all:
76
+ tools.append(self.extract_url_content)
77
+
78
+ super().__init__(name="tavily_tools", tools=tools, **kwargs)
79
+
80
+ def web_search_using_tavily(self, query: str, max_results: int = 5) -> str:
81
+ """Use this function to search the web for a given query.
82
+ This function uses the Tavily API to provide realtime online information about the query.
83
+
84
+ Args:
85
+ query (str): Query to search for.
86
+ max_results (int): Maximum number of results to return. Defaults to 5.
87
+
88
+ Returns:
89
+ str: JSON string of results related to the query.
90
+ """
91
+
92
+ response = self.client.search(
93
+ query=query, search_depth=self.search_depth, include_answer=self.include_answer, max_results=max_results
94
+ )
95
+
96
+ clean_response: Dict[str, Any] = {"query": query}
97
+ if "answer" in response:
98
+ clean_response["answer"] = response["answer"]
99
+
100
+ clean_results = []
101
+ current_token_count = len(json.dumps(clean_response))
102
+ for result in response.get("results", []):
103
+ _result = {
104
+ "title": result["title"],
105
+ "url": result["url"],
106
+ "content": result["content"],
107
+ "score": result["score"],
108
+ }
109
+ current_token_count += len(json.dumps(_result))
110
+ if current_token_count > self.max_tokens:
111
+ break
112
+ clean_results.append(_result)
113
+ clean_response["results"] = clean_results
114
+
115
+ if self.format == "json":
116
+ return json.dumps(clean_response) if clean_response else "No results found."
117
+ elif self.format == "markdown":
118
+ _markdown = ""
119
+ _markdown += f"# {query}\n\n"
120
+ if "answer" in clean_response:
121
+ _markdown += "### Summary\n"
122
+ _markdown += f"{clean_response.get('answer')}\n\n"
123
+ for result in clean_response["results"]:
124
+ _markdown += f"### [{result['title']}]({result['url']})\n"
125
+ _markdown += f"{result['content']}\n\n"
126
+ return _markdown
127
+
128
+ def web_search_with_tavily(self, query: str) -> str:
129
+ """Use this function to search the web for a given query.
130
+ This function uses the Tavily API to provide realtime online information about the query.
131
+
132
+ Args:
133
+ query (str): Query to search for.
134
+
135
+ Returns:
136
+ str: JSON string of results related to the query.
137
+ """
138
+
139
+ return self.client.get_search_context(
140
+ query=query, search_depth=self.search_depth, max_tokens=self.max_tokens, include_answer=self.include_answer
141
+ )
142
+
143
+ def extract_url_content(self, urls: str) -> str:
144
+ """Extract content from one or more URLs using Tavily's Extract API.
145
+ This function retrieves the main content from web pages in markdown or text format.
146
+
147
+ Args:
148
+ urls (str): Single URL or multiple comma-separated URLs to extract content from.
149
+ Example: "https://example.com" or "https://example.com,https://another.com"
150
+
151
+ Returns:
152
+ str: Extracted content in the specified format (markdown or text).
153
+ For multiple URLs, returns combined content with URL headers.
154
+ Failed extractions are noted in the output.
155
+ """
156
+ # Parse URLs - handle both single and comma-separated multiple URLs
157
+ url_list = [url.strip() for url in urls.split(",") if url.strip()]
158
+
159
+ if not url_list:
160
+ return "Error: No valid URLs provided."
161
+
162
+ try:
163
+ # Prepare extract parameters
164
+ extract_params: Dict[str, Any] = {
165
+ "urls": url_list,
166
+ "depth": self.extract_depth,
167
+ }
168
+
169
+ # Add optional parameters if specified
170
+ if self.include_images:
171
+ extract_params["include_images"] = True
172
+ if self.include_favicon:
173
+ extract_params["include_favicon"] = True
174
+ if self.extract_timeout is not None:
175
+ extract_params["timeout"] = self.extract_timeout
176
+
177
+ # Call Tavily Extract API
178
+ response = self.client.extract(**extract_params)
179
+
180
+ # Process response based on format preference
181
+ if not response or "results" not in response:
182
+ return "Error: No content could be extracted from the provided URL(s)."
183
+
184
+ results = response.get("results", [])
185
+ if not results:
186
+ return "Error: No content could be extracted from the provided URL(s)."
187
+
188
+ # Format output
189
+ if self.extract_format == "markdown":
190
+ return self._format_extract_markdown(results)
191
+ elif self.extract_format == "text":
192
+ return self._format_extract_text(results)
193
+ else:
194
+ # Fallback to JSON if format is unrecognized
195
+ return json.dumps(results, indent=2)
196
+
197
+ except Exception as e:
198
+ logger.error(f"Error extracting content from URLs: {e}")
199
+ return f"Error extracting content: {str(e)}"
200
+
201
+ def _format_extract_markdown(self, results: List[Dict[str, Any]]) -> str:
202
+ """Format extraction results as markdown.
203
+
204
+ Args:
205
+ results: List of extraction result dictionaries from Tavily API.
206
+
207
+ Returns:
208
+ str: Formatted markdown content.
209
+ """
210
+ output = []
211
+
212
+ for result in results:
213
+ url = result.get("url", "Unknown URL")
214
+ raw_content = result.get("raw_content", "")
215
+ failed_reason = result.get("failed_reason")
216
+
217
+ if failed_reason:
218
+ output.append(f"## {url}\n\n **Extraction Failed**: {failed_reason}\n\n")
219
+ elif raw_content:
220
+ output.append(f"## {url}\n\n{raw_content}\n\n")
221
+ else:
222
+ output.append(f"## {url}\n\n*No content available*\n\n")
223
+
224
+ return "".join(output) if output else "No content extracted."
225
+
226
+ def _format_extract_text(self, results: List[Dict[str, Any]]) -> str:
227
+ """Format extraction results as plain text.
228
+
229
+ Args:
230
+ results: List of extraction result dictionaries from Tavily API.
231
+
232
+ Returns:
233
+ str: Formatted plain text content.
234
+ """
235
+ output = []
236
+
237
+ for result in results:
238
+ url = result.get("url", "Unknown URL")
239
+ raw_content = result.get("raw_content", "")
240
+ failed_reason = result.get("failed_reason")
241
+
242
+ output.append(f"URL: {url}")
243
+ output.append("-" * 80)
244
+
245
+ if failed_reason:
246
+ output.append(f"EXTRACTION FAILED: {failed_reason}")
247
+ elif raw_content:
248
+ output.append(raw_content)
249
+ else:
250
+ output.append("No content available")
251
+
252
+ output.append("\n")
253
+
254
+ return "\n".join(output) if output else "No content extracted."
agno/tools/telegram.py ADDED
@@ -0,0 +1,48 @@
1
+ from os import getenv
2
+ from typing import Any, List, Optional, Union
3
+
4
+ import httpx
5
+
6
+ from agno.tools import Toolkit
7
+ from agno.utils.log import log_debug, logger
8
+
9
+
10
+ class TelegramTools(Toolkit):
11
+ base_url = "https://api.telegram.org"
12
+
13
+ def __init__(
14
+ self,
15
+ chat_id: Union[str, int],
16
+ token: Optional[str] = None,
17
+ enable_send_message: bool = True,
18
+ all: bool = False,
19
+ **kwargs,
20
+ ):
21
+ self.token = token or getenv("TELEGRAM_TOKEN")
22
+ if not self.token:
23
+ logger.error("TELEGRAM_TOKEN not set. Please set the TELEGRAM_TOKEN environment variable.")
24
+
25
+ self.chat_id = chat_id
26
+
27
+ tools: List[Any] = []
28
+ if all or enable_send_message:
29
+ tools.append(self.send_message)
30
+
31
+ super().__init__(name="telegram", tools=tools, **kwargs)
32
+
33
+ def _call_post_method(self, method, *args, **kwargs):
34
+ return httpx.post(f"{self.base_url}/bot{self.token}/{method}", *args, **kwargs)
35
+
36
+ def send_message(self, message: str) -> str:
37
+ """This function sends a message to the chat ID.
38
+
39
+ :param message: The message to send.
40
+ :return: The response from the API.
41
+ """
42
+ log_debug(f"Sending telegram message: {message}")
43
+ response = self._call_post_method("sendMessage", json={"chat_id": self.chat_id, "text": message})
44
+ try:
45
+ response.raise_for_status()
46
+ return response.text
47
+ except httpx.HTTPStatusError as e:
48
+ return f"An error occurred: {e}"