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
@@ -0,0 +1,115 @@
1
+ import asyncio
2
+ import uuid
3
+ from pathlib import Path
4
+ from typing import IO, Any, List, Optional, Union
5
+
6
+ from agno.knowledge.chunking.fixed import FixedSizeChunking
7
+ from agno.knowledge.chunking.strategy import ChunkingStrategy, ChunkingStrategyType
8
+ from agno.knowledge.document.base import Document
9
+ from agno.knowledge.reader.base import Reader
10
+ from agno.knowledge.types import ContentType
11
+ from agno.utils.log import log_info, logger
12
+
13
+
14
+ class TextReader(Reader):
15
+ """Reader for Text files"""
16
+
17
+ def __init__(self, chunking_strategy: Optional[ChunkingStrategy] = FixedSizeChunking(), **kwargs):
18
+ super().__init__(chunking_strategy=chunking_strategy, **kwargs)
19
+
20
+ @classmethod
21
+ def get_supported_chunking_strategies(self) -> List[ChunkingStrategyType]:
22
+ """Get the list of supported chunking strategies for Text readers."""
23
+ return [
24
+ ChunkingStrategyType.FIXED_SIZE_CHUNKER,
25
+ ChunkingStrategyType.AGENTIC_CHUNKER,
26
+ ChunkingStrategyType.DOCUMENT_CHUNKER,
27
+ ChunkingStrategyType.RECURSIVE_CHUNKER,
28
+ ChunkingStrategyType.SEMANTIC_CHUNKER,
29
+ ]
30
+
31
+ @classmethod
32
+ def get_supported_content_types(self) -> List[ContentType]:
33
+ return [ContentType.TXT]
34
+
35
+ def read(self, file: Union[Path, IO[Any]], name: Optional[str] = None) -> List[Document]:
36
+ try:
37
+ if isinstance(file, Path):
38
+ if not file.exists():
39
+ raise FileNotFoundError(f"Could not find file: {file}")
40
+ log_info(f"Reading: {file}")
41
+ file_name = name or file.stem
42
+ file_contents = file.read_text(self.encoding or "utf-8")
43
+ else:
44
+ file_name = name or file.name.split(".")[0]
45
+ log_info(f"Reading uploaded file: {file_name}")
46
+ file.seek(0)
47
+ file_contents = file.read().decode(self.encoding or "utf-8")
48
+
49
+ documents = [
50
+ Document(
51
+ name=file_name,
52
+ id=str(uuid.uuid4()),
53
+ content=file_contents,
54
+ )
55
+ ]
56
+ if self.chunk:
57
+ chunked_documents = []
58
+ for document in documents:
59
+ chunked_documents.extend(self.chunk_document(document))
60
+ return chunked_documents
61
+ return documents
62
+ except Exception as e:
63
+ logger.error(f"Error reading: {file}: {e}")
64
+ return []
65
+
66
+ async def async_read(self, file: Union[Path, IO[Any]], name: Optional[str] = None) -> List[Document]:
67
+ try:
68
+ if isinstance(file, Path):
69
+ if not file.exists():
70
+ raise FileNotFoundError(f"Could not find file: {file}")
71
+
72
+ log_info(f"Reading asynchronously: {file}")
73
+ file_name = name or file.stem
74
+
75
+ try:
76
+ import aiofiles
77
+
78
+ async with aiofiles.open(file, "r", encoding=self.encoding or "utf-8") as f:
79
+ file_contents = await f.read()
80
+ except ImportError:
81
+ logger.warning("aiofiles not installed, using synchronous file I/O")
82
+ file_contents = file.read_text(self.encoding or "utf-8")
83
+ else:
84
+ log_info(f"Reading uploaded file asynchronously: {file.name}")
85
+ file_name = name or file.name.split(".")[0]
86
+ file.seek(0)
87
+ file_contents = file.read().decode(self.encoding or "utf-8")
88
+
89
+ document = Document(
90
+ name=file_name,
91
+ id=str(uuid.uuid4()),
92
+ content=file_contents,
93
+ )
94
+
95
+ if self.chunk:
96
+ return await self._async_chunk_document(document)
97
+ return [document]
98
+ except Exception as e:
99
+ logger.error(f"Error reading asynchronously: {file}: {e}")
100
+ return []
101
+
102
+ async def _async_chunk_document(self, document: Document) -> List[Document]:
103
+ if not self.chunk or not document:
104
+ return [document]
105
+
106
+ async def process_chunk(chunk_doc: Document) -> Document:
107
+ return chunk_doc
108
+
109
+ chunked_documents = self.chunk_document(document)
110
+
111
+ if not chunked_documents:
112
+ return [document]
113
+
114
+ tasks = [process_chunk(chunk_doc) for chunk_doc in chunked_documents]
115
+ return await asyncio.gather(*tasks)
@@ -0,0 +1,372 @@
1
+ import asyncio
2
+ import random
3
+ import time
4
+ from dataclasses import dataclass, field
5
+ from typing import Dict, List, Literal, Optional, Set
6
+ from urllib.parse import urlparse
7
+
8
+ import httpx
9
+
10
+ from agno.knowledge.chunking.semantic import SemanticChunking
11
+ from agno.knowledge.chunking.strategy import ChunkingStrategy, ChunkingStrategyType
12
+ from agno.knowledge.document.base import Document
13
+ from agno.knowledge.reader.base import Reader
14
+ from agno.knowledge.types import ContentType
15
+ from agno.utils.log import log_debug, logger
16
+
17
+ try:
18
+ from bs4 import BeautifulSoup, Tag # noqa: F401
19
+ except ImportError:
20
+ raise ImportError("The `bs4` package is not installed. Please install it via `pip install beautifulsoup4`.")
21
+
22
+ try:
23
+ from ddgs import DDGS
24
+ except ImportError:
25
+ raise ImportError("The `ddgs` package is not installed. Please install it via `pip install ddgs`.")
26
+
27
+
28
+ @dataclass
29
+ class WebSearchReader(Reader):
30
+ """Reader that uses web search to find content for a given query"""
31
+
32
+ search_timeout: int = 10
33
+
34
+ request_timeout: int = 30
35
+ delay_between_requests: float = 2.0 # Increased default delay
36
+ max_retries: int = 3
37
+ user_agent: str = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36"
38
+
39
+ # Search engine configuration
40
+ search_engine: Literal["duckduckgo", "google"] = "duckduckgo"
41
+ search_delay: float = 3.0 # Delay between search requests
42
+ max_search_retries: int = 2 # Retries for search operations
43
+
44
+ # Rate limiting
45
+ rate_limit_delay: float = 5.0 # Delay when rate limited
46
+ exponential_backoff: bool = True
47
+
48
+ # Internal state
49
+ _visited_urls: Set[str] = field(default_factory=set)
50
+ _last_search_time: float = field(default=0.0, init=False)
51
+
52
+ # Override default chunking strategy
53
+ chunking_strategy: Optional[ChunkingStrategy] = SemanticChunking()
54
+
55
+ @classmethod
56
+ def get_supported_chunking_strategies(self) -> List[ChunkingStrategyType]:
57
+ """Get the list of supported chunking strategies for Web Search readers."""
58
+ return [
59
+ ChunkingStrategyType.AGENTIC_CHUNKER,
60
+ ChunkingStrategyType.DOCUMENT_CHUNKER,
61
+ ChunkingStrategyType.RECURSIVE_CHUNKER,
62
+ ChunkingStrategyType.SEMANTIC_CHUNKER,
63
+ ChunkingStrategyType.FIXED_SIZE_CHUNKER,
64
+ ]
65
+
66
+ @classmethod
67
+ def get_supported_content_types(self) -> List[ContentType]:
68
+ return [ContentType.TOPIC]
69
+
70
+ def _respect_rate_limits(self):
71
+ """Ensure we don't exceed rate limits"""
72
+ current_time = time.time()
73
+ time_since_last_search = current_time - self._last_search_time
74
+
75
+ if time_since_last_search < self.search_delay:
76
+ sleep_time = self.search_delay - time_since_last_search
77
+ log_debug(f"Rate limiting: sleeping for {sleep_time:.2f} seconds")
78
+ time.sleep(sleep_time)
79
+
80
+ self._last_search_time = time.time()
81
+
82
+ def _perform_duckduckgo_search(self, query: str) -> List[Dict[str, str]]:
83
+ """Perform web search using DuckDuckGo with rate limiting"""
84
+ log_debug(f"Performing DuckDuckGo search for: {query}")
85
+
86
+ for attempt in range(self.max_search_retries):
87
+ try:
88
+ self._respect_rate_limits()
89
+
90
+ ddgs = DDGS(timeout=self.search_timeout)
91
+ search_results = ddgs.text(query=query, max_results=self.max_results)
92
+
93
+ # Convert to list and extract relevant fields
94
+ results = []
95
+ for result in search_results:
96
+ results.append(
97
+ {
98
+ "title": result.get("title", ""),
99
+ "url": result.get("href", ""),
100
+ "description": result.get("body", ""),
101
+ }
102
+ )
103
+
104
+ log_debug(f"Found {len(results)} search results")
105
+ return results
106
+
107
+ except Exception as e:
108
+ logger.warning(f"DuckDuckGo search attempt {attempt + 1} failed: {e}")
109
+ if "rate limit" in str(e).lower() or "429" in str(e):
110
+ # Rate limited - wait longer
111
+ wait_time = (
112
+ self.rate_limit_delay * (2**attempt) if self.exponential_backoff else self.rate_limit_delay
113
+ )
114
+ logger.info(f"Rate limited, waiting {wait_time} seconds before retry")
115
+ time.sleep(wait_time)
116
+ elif attempt < self.max_search_retries - 1:
117
+ # Other error - shorter wait
118
+ time.sleep(self.search_delay)
119
+ else:
120
+ logger.error(f"All DuckDuckGo search attempts failed: {e}")
121
+ return []
122
+ return []
123
+
124
+ def _perform_google_search(self, query: str) -> List[Dict[str, str]]:
125
+ """Perform web search using Google (requires googlesearch-python)"""
126
+ log_debug(f"Performing Google search for: {query}")
127
+
128
+ try:
129
+ from googlesearch import search
130
+ except ImportError:
131
+ logger.error("Google search requires 'googlesearch-python'. Install with: pip install googlesearch-python")
132
+ return []
133
+
134
+ for attempt in range(self.max_search_retries):
135
+ try:
136
+ self._respect_rate_limits()
137
+
138
+ results = []
139
+ # Use the basic search function without unsupported parameters
140
+ # The googlesearch-python library's search function only accepts basic parameters
141
+ search_results = search(query)
142
+
143
+ # Convert iterator to list and limit results
144
+ result_list = list(search_results)[: self.max_results]
145
+
146
+ for result in result_list:
147
+ # The search function returns URLs as strings
148
+ results.append(
149
+ {
150
+ "title": "", # Google search doesn't provide titles directly
151
+ "url": result,
152
+ "description": "", # Google search doesn't provide descriptions directly
153
+ }
154
+ )
155
+
156
+ log_debug(f"Found {len(results)} Google search results")
157
+ return results
158
+
159
+ except Exception as e:
160
+ logger.warning(f"Google search attempt {attempt + 1} failed: {e}")
161
+ if attempt < self.max_search_retries - 1:
162
+ time.sleep(self.search_delay)
163
+ else:
164
+ logger.error(f"All Google search attempts failed: {e}")
165
+ return []
166
+
167
+ return []
168
+
169
+ def _perform_web_search(self, query: str) -> List[Dict[str, str]]:
170
+ """Perform web search using the configured search engine"""
171
+ if self.search_engine == "duckduckgo":
172
+ return self._perform_duckduckgo_search(query)
173
+ elif self.search_engine == "google":
174
+ return self._perform_google_search(query)
175
+ else:
176
+ logger.error(f"Unsupported search engine: {self.search_engine}")
177
+ return []
178
+
179
+ def _is_valid_url(self, url: str) -> bool:
180
+ """Check if URL is valid and not already visited"""
181
+ try:
182
+ parsed = urlparse(url)
183
+ return bool(parsed.scheme in ["http", "https"] and parsed.netloc and url not in self._visited_urls)
184
+ except Exception:
185
+ return False
186
+
187
+ def _extract_text_from_html(self, html_content: str, url: str) -> str:
188
+ """Extract clean text content from HTML"""
189
+ try:
190
+ soup = BeautifulSoup(html_content, "html.parser")
191
+
192
+ # Remove script and style elements
193
+ for script in soup(["script", "style"]):
194
+ script.decompose()
195
+
196
+ # Get text content
197
+ text = soup.get_text()
198
+
199
+ # Clean up whitespace
200
+ lines = (line.strip() for line in text.splitlines())
201
+ chunks = (phrase.strip() for line in lines for phrase in line.split(" "))
202
+ text = " ".join(chunk for chunk in chunks if chunk)
203
+
204
+ return text
205
+
206
+ except Exception as e:
207
+ logger.warning(f"Error extracting text from {url}: {e}")
208
+ return html_content
209
+
210
+ def _fetch_url_content(self, url: str) -> Optional[str]:
211
+ """Fetch content from a URL with retry logic"""
212
+ headers = {"User-Agent": self.user_agent}
213
+
214
+ for attempt in range(self.max_retries):
215
+ try:
216
+ response = httpx.get(url, headers=headers, timeout=self.request_timeout, follow_redirects=True)
217
+ response.raise_for_status()
218
+
219
+ # Check if it's HTML content
220
+ content_type = response.headers.get("content-type", "").lower()
221
+ if "text/html" in content_type:
222
+ return self._extract_text_from_html(response.text, url)
223
+ else:
224
+ # For non-HTML content, return as-is
225
+ return response.text
226
+
227
+ except Exception as e:
228
+ logger.warning(f"Attempt {attempt + 1} failed for {url}: {e}")
229
+ if attempt < self.max_retries - 1:
230
+ time.sleep(random.uniform(1, 3)) # Random delay between retries
231
+ continue
232
+
233
+ logger.error(f"Failed to fetch content from {url} after {self.max_retries} attempts")
234
+ return None
235
+
236
+ def _create_document_from_url(self, url: str, content: str, search_result: Dict[str, str]) -> Document:
237
+ """Create a Document object from URL content and search result metadata"""
238
+ # Use the URL as the document ID
239
+ doc_id = url
240
+
241
+ # Use the search result title as the document name, fallback to URL
242
+ doc_name = search_result.get("title", urlparse(url).netloc)
243
+
244
+ # Create metadata with search information
245
+ meta_data = {
246
+ "url": url,
247
+ "search_title": search_result.get("title", ""),
248
+ "search_description": search_result.get("description", ""),
249
+ "source": "web_search",
250
+ "search_engine": self.search_engine,
251
+ }
252
+
253
+ return Document(id=doc_id, name=doc_name, content=content, meta_data=meta_data)
254
+
255
+ def read(self, query: str) -> List[Document]:
256
+ """Read content for a given query by performing web search and fetching content"""
257
+ if not query:
258
+ raise ValueError("Query cannot be empty")
259
+
260
+ log_debug(f"Starting web search reader for query: {query}")
261
+
262
+ # Perform web search
263
+ search_results = self._perform_web_search(query)
264
+ if not search_results:
265
+ logger.warning(f"No search results found for query: {query}")
266
+ return []
267
+
268
+ documents: List[Document] = []
269
+
270
+ for result in search_results:
271
+ url = result.get("url", "")
272
+
273
+ # Skip if URL is invalid or already visited
274
+ if not self._is_valid_url(url):
275
+ continue
276
+
277
+ # Mark URL as visited
278
+ self._visited_urls.add(url)
279
+
280
+ # Add delay between requests to be respectful
281
+ if len(documents) > 0:
282
+ time.sleep(self.delay_between_requests)
283
+
284
+ # Fetch content from URL
285
+ content = self._fetch_url_content(url)
286
+ if content is None:
287
+ continue
288
+
289
+ # Create document
290
+ document = self._create_document_from_url(url, content, result)
291
+
292
+ # Apply chunking if enabled
293
+ if self.chunk:
294
+ chunked_docs = self.chunk_document(document)
295
+ documents.extend(chunked_docs)
296
+ else:
297
+ documents.append(document)
298
+
299
+ # Stop if we've reached max_results
300
+ if len(documents) >= self.max_results:
301
+ break
302
+
303
+ log_debug(f"Created {len(documents)} documents from web search")
304
+ return documents
305
+
306
+ async def async_read(self, query: str) -> List[Document]:
307
+ """Asynchronously read content for a given query"""
308
+ if not query:
309
+ raise ValueError("Query cannot be empty")
310
+
311
+ log_debug(f"Starting async web search reader for query: {query}")
312
+
313
+ # Perform web search (synchronous operation)
314
+ search_results = self._perform_web_search(query)
315
+ if not search_results:
316
+ logger.warning(f"No search results found for query: {query}")
317
+ return []
318
+
319
+ # Create tasks for fetching content from each URL
320
+ async def fetch_url_async(result: Dict[str, str]) -> Optional[Document]:
321
+ url = result.get("url", "")
322
+
323
+ # Skip if URL is invalid or already visited
324
+ if not self._is_valid_url(url):
325
+ return None
326
+
327
+ # Mark URL as visited
328
+ self._visited_urls.add(url)
329
+
330
+ try:
331
+ headers = {"User-Agent": self.user_agent}
332
+ async with httpx.AsyncClient(timeout=self.request_timeout) as client:
333
+ response = await client.get(url, headers=headers, follow_redirects=True)
334
+ response.raise_for_status()
335
+
336
+ content_type = response.headers.get("content-type", "").lower()
337
+ if "text/html" in content_type:
338
+ content = self._extract_text_from_html(response.text, url)
339
+ else:
340
+ content = response.text
341
+
342
+ document = self._create_document_from_url(url, content, result)
343
+ return document
344
+
345
+ except Exception as e:
346
+ logger.warning(f"Error fetching {url}: {e}")
347
+ return None
348
+
349
+ # Create tasks for all URLs
350
+ tasks = [fetch_url_async(result) for result in search_results]
351
+
352
+ # Execute all tasks concurrently with delays
353
+ documents = []
354
+ for i, task in enumerate(tasks):
355
+ if i > 0: # Add delay between requests (except for the first one)
356
+ await asyncio.sleep(self.delay_between_requests)
357
+
358
+ doc = await task
359
+ if doc is not None:
360
+ # Apply chunking if enabled
361
+ if self.chunk:
362
+ chunked_docs = await self.chunk_documents_async([doc])
363
+ documents.extend(chunked_docs)
364
+ else:
365
+ documents.append(doc)
366
+
367
+ # Stop if we've reached max_results
368
+ if len(documents) >= self.max_results:
369
+ break
370
+
371
+ log_debug(f"Created {len(documents)} documents from async web search")
372
+ return documents