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,138 @@
1
+ """Table schemas and related utils used by the MySQLDb class"""
2
+
3
+ from typing import Any
4
+
5
+ try:
6
+ from sqlalchemy.types import JSON, BigInteger, Boolean, Date, String, Text
7
+ except ImportError:
8
+ raise ImportError("`sqlalchemy` not installed. Please install it using `pip install sqlalchemy`")
9
+
10
+ SESSION_TABLE_SCHEMA = {
11
+ "session_id": {"type": lambda: String(128), "nullable": False},
12
+ "session_type": {"type": lambda: String(20), "nullable": False, "index": True},
13
+ "agent_id": {"type": lambda: String(128), "nullable": True},
14
+ "team_id": {"type": lambda: String(128), "nullable": True},
15
+ "workflow_id": {"type": lambda: String(128), "nullable": True},
16
+ "user_id": {"type": lambda: String(128), "nullable": True},
17
+ "session_data": {"type": JSON, "nullable": True},
18
+ "agent_data": {"type": JSON, "nullable": True},
19
+ "team_data": {"type": JSON, "nullable": True},
20
+ "workflow_data": {"type": JSON, "nullable": True},
21
+ "metadata": {"type": JSON, "nullable": True},
22
+ "runs": {"type": JSON, "nullable": True},
23
+ "summary": {"type": JSON, "nullable": True},
24
+ "created_at": {"type": BigInteger, "nullable": False, "index": True},
25
+ "updated_at": {"type": BigInteger, "nullable": True},
26
+ "_unique_constraints": [
27
+ {
28
+ "name": "uq_session_id",
29
+ "columns": ["session_id"],
30
+ },
31
+ ],
32
+ }
33
+
34
+ USER_MEMORY_TABLE_SCHEMA = {
35
+ "memory_id": {"type": lambda: String(128), "primary_key": True, "nullable": False},
36
+ "memory": {"type": JSON, "nullable": False},
37
+ "input": {"type": Text, "nullable": True},
38
+ "agent_id": {"type": lambda: String(128), "nullable": True},
39
+ "team_id": {"type": lambda: String(128), "nullable": True},
40
+ "user_id": {"type": lambda: String(128), "nullable": True, "index": True},
41
+ "topics": {"type": JSON, "nullable": True},
42
+ "updated_at": {"type": BigInteger, "nullable": True, "index": True},
43
+ }
44
+
45
+ EVAL_TABLE_SCHEMA = {
46
+ "run_id": {"type": lambda: String(128), "primary_key": True, "nullable": False},
47
+ "eval_type": {"type": lambda: String(50), "nullable": False},
48
+ "eval_data": {"type": JSON, "nullable": False},
49
+ "eval_input": {"type": JSON, "nullable": False},
50
+ "name": {"type": lambda: String(255), "nullable": True},
51
+ "agent_id": {"type": lambda: String(128), "nullable": True},
52
+ "team_id": {"type": lambda: String(128), "nullable": True},
53
+ "workflow_id": {"type": lambda: String(128), "nullable": True},
54
+ "model_id": {"type": lambda: String(128), "nullable": True},
55
+ "model_provider": {"type": lambda: String(128), "nullable": True},
56
+ "evaluated_component_name": {"type": lambda: String(255), "nullable": True},
57
+ "created_at": {"type": BigInteger, "nullable": False, "index": True},
58
+ "updated_at": {"type": BigInteger, "nullable": True},
59
+ }
60
+
61
+ KNOWLEDGE_TABLE_SCHEMA = {
62
+ "id": {"type": lambda: String(128), "primary_key": True, "nullable": False},
63
+ "name": {"type": lambda: String(255), "nullable": False},
64
+ "description": {"type": Text, "nullable": False},
65
+ "metadata": {"type": JSON, "nullable": True},
66
+ "type": {"type": lambda: String(50), "nullable": True},
67
+ "size": {"type": BigInteger, "nullable": True},
68
+ "linked_to": {"type": lambda: String(128), "nullable": True},
69
+ "access_count": {"type": BigInteger, "nullable": True},
70
+ "created_at": {"type": BigInteger, "nullable": True},
71
+ "updated_at": {"type": BigInteger, "nullable": True},
72
+ "status": {"type": lambda: String(50), "nullable": True},
73
+ "status_message": {"type": Text, "nullable": True},
74
+ "external_id": {"type": lambda: String(128), "nullable": True},
75
+ }
76
+
77
+ METRICS_TABLE_SCHEMA = {
78
+ "id": {"type": lambda: String(128), "primary_key": True, "nullable": False},
79
+ "agent_runs_count": {"type": BigInteger, "nullable": False},
80
+ "team_runs_count": {"type": BigInteger, "nullable": False},
81
+ "workflow_runs_count": {"type": BigInteger, "nullable": False},
82
+ "agent_sessions_count": {"type": BigInteger, "nullable": False},
83
+ "team_sessions_count": {"type": BigInteger, "nullable": False},
84
+ "workflow_sessions_count": {"type": BigInteger, "nullable": False},
85
+ "users_count": {"type": BigInteger, "nullable": False},
86
+ "token_metrics": {"type": JSON, "nullable": False},
87
+ "model_metrics": {"type": JSON, "nullable": False},
88
+ "date": {"type": Date, "nullable": False, "index": True},
89
+ "aggregation_period": {"type": lambda: String(20), "nullable": False},
90
+ "created_at": {"type": BigInteger, "nullable": False},
91
+ "updated_at": {"type": BigInteger, "nullable": True},
92
+ "completed": {"type": Boolean, "nullable": False},
93
+ "_unique_constraints": [
94
+ {
95
+ "name": "uq_metrics_date_period",
96
+ "columns": ["date", "aggregation_period"],
97
+ }
98
+ ],
99
+ }
100
+
101
+ CULTURAL_KNOWLEDGE_TABLE_SCHEMA = {
102
+ "id": {"type": lambda: String(128), "primary_key": True, "nullable": False},
103
+ "name": {"type": lambda: String(255), "nullable": False, "index": True},
104
+ "summary": {"type": Text, "nullable": True},
105
+ "content": {"type": JSON, "nullable": True},
106
+ "metadata": {"type": JSON, "nullable": True},
107
+ "input": {"type": Text, "nullable": True},
108
+ "created_at": {"type": BigInteger, "nullable": True},
109
+ "updated_at": {"type": BigInteger, "nullable": True},
110
+ "agent_id": {"type": lambda: String(128), "nullable": True},
111
+ "team_id": {"type": lambda: String(128), "nullable": True},
112
+ }
113
+
114
+
115
+ def get_table_schema_definition(table_type: str) -> dict[str, Any]:
116
+ """
117
+ Get the expected schema definition for the given table.
118
+
119
+ Args:
120
+ table_type (str): The type of table to get the schema for.
121
+
122
+ Returns:
123
+ Dict[str, Any]: Dictionary containing column definitions for the table
124
+ """
125
+ schemas = {
126
+ "sessions": SESSION_TABLE_SCHEMA,
127
+ "evals": EVAL_TABLE_SCHEMA,
128
+ "metrics": METRICS_TABLE_SCHEMA,
129
+ "memories": USER_MEMORY_TABLE_SCHEMA,
130
+ "knowledge": KNOWLEDGE_TABLE_SCHEMA,
131
+ "culture": CULTURAL_KNOWLEDGE_TABLE_SCHEMA,
132
+ }
133
+
134
+ schema = schemas.get(table_type, {})
135
+ if not schema:
136
+ raise ValueError(f"Unknown table type: {table_type}")
137
+
138
+ return schema # type: ignore[return-value]
agno/db/mysql/utils.py ADDED
@@ -0,0 +1,355 @@
1
+ """Utility functions for the MySQL database class."""
2
+
3
+ import time
4
+ from datetime import date, datetime, timedelta, timezone
5
+ from typing import Any, Dict, List, Optional
6
+ from uuid import uuid4
7
+
8
+ from sqlalchemy import Engine
9
+
10
+ from agno.db.mysql.schemas import get_table_schema_definition
11
+ from agno.db.schemas.culture import CulturalKnowledge
12
+ from agno.utils.log import log_debug, log_error, log_warning
13
+
14
+ try:
15
+ from sqlalchemy import Table
16
+ from sqlalchemy.dialects import mysql
17
+ from sqlalchemy.inspection import inspect
18
+ from sqlalchemy.orm import Session
19
+ from sqlalchemy.sql.expression import text
20
+ except ImportError:
21
+ raise ImportError("`sqlalchemy` not installed. Please install it using `pip install sqlalchemy`")
22
+
23
+
24
+ # -- DB util methods --
25
+ def apply_sorting(stmt, table: Table, sort_by: Optional[str] = None, sort_order: Optional[str] = None):
26
+ """Apply sorting to the given SQLAlchemy statement.
27
+
28
+ Args:
29
+ stmt: The SQLAlchemy statement to modify
30
+ table: The table being queried
31
+ sort_by: The field to sort by
32
+ sort_order: The sort order ('asc' or 'desc')
33
+
34
+ Returns:
35
+ The modified statement with sorting applied
36
+ """
37
+ if sort_by is None:
38
+ return stmt
39
+
40
+ if not hasattr(table.c, sort_by):
41
+ log_debug(f"Invalid sort field: '{sort_by}'. Will not apply any sorting.")
42
+ return stmt
43
+
44
+ # Apply the given sorting
45
+ sort_column = getattr(table.c, sort_by)
46
+ if sort_order and sort_order == "asc":
47
+ return stmt.order_by(sort_column.asc())
48
+ else:
49
+ return stmt.order_by(sort_column.desc())
50
+
51
+
52
+ def create_schema(session: Session, db_schema: str) -> None:
53
+ """Create the database schema if it doesn't exist.
54
+
55
+ Args:
56
+ session: The SQLAlchemy session to use
57
+ db_schema (str): The definition of the database schema to create
58
+ """
59
+ try:
60
+ log_debug(f"Creating database if not exists: {db_schema}")
61
+ # MySQL uses CREATE DATABASE instead of CREATE SCHEMA
62
+ session.execute(text(f"CREATE DATABASE IF NOT EXISTS {db_schema};"))
63
+ except Exception as e:
64
+ log_warning(f"Could not create database {db_schema}: {e}")
65
+
66
+
67
+ def is_table_available(session: Session, table_name: str, db_schema: str) -> bool:
68
+ """
69
+ Check if a table with the given name exists in the given schema.
70
+
71
+ Returns:
72
+ bool: True if the table exists, False otherwise.
73
+ """
74
+ try:
75
+ exists_query = text(
76
+ "SELECT 1 FROM information_schema.tables WHERE table_schema = :schema AND table_name = :table"
77
+ )
78
+ exists = session.execute(exists_query, {"schema": db_schema, "table": table_name}).scalar() is not None
79
+ if not exists:
80
+ log_debug(f"Table {db_schema}.{table_name} {'exists' if exists else 'does not exist'}")
81
+
82
+ return exists
83
+
84
+ except Exception as e:
85
+ log_error(f"Error checking if table exists: {e}")
86
+ return False
87
+
88
+
89
+ def is_valid_table(db_engine: Engine, table_name: str, table_type: str, db_schema: str) -> bool:
90
+ """
91
+ Check if the existing table has the expected column names.
92
+
93
+ Args:
94
+ table_name (str): Name of the table to validate
95
+ schema (str): Database schema name
96
+
97
+ Returns:
98
+ bool: True if table has all expected columns, False otherwise
99
+ """
100
+ try:
101
+ expected_table_schema = get_table_schema_definition(table_type)
102
+ expected_columns = {col_name for col_name in expected_table_schema.keys() if not col_name.startswith("_")}
103
+
104
+ # Get existing columns
105
+ inspector = inspect(db_engine)
106
+ existing_columns_info = inspector.get_columns(table_name, schema=db_schema)
107
+ existing_columns = set(col["name"] for col in existing_columns_info)
108
+
109
+ # Check if all expected columns exist
110
+ missing_columns = expected_columns - existing_columns
111
+ if missing_columns:
112
+ log_warning(f"Missing columns {missing_columns} in table {db_schema}.{table_name}")
113
+ return False
114
+
115
+ return True
116
+ except Exception as e:
117
+ log_error(f"Error validating table schema for {db_schema}.{table_name}: {e}")
118
+ return False
119
+
120
+
121
+ # -- Metrics util methods --
122
+ def bulk_upsert_metrics(session: Session, table: Table, metrics_records: list[dict]) -> list[dict]:
123
+ """Bulk upsert metrics into the database.
124
+
125
+ Args:
126
+ table (Table): The table to upsert into.
127
+ metrics_records (list[dict]): The metrics records to upsert.
128
+
129
+ Returns:
130
+ list[dict]: The upserted metrics records.
131
+ """
132
+ if not metrics_records:
133
+ return []
134
+
135
+ results = []
136
+
137
+ # MySQL doesn't support returning in the same way as PostgreSQL
138
+ # We'll need to insert/update and then fetch the records
139
+ for record in metrics_records:
140
+ stmt = mysql.insert(table).values(record)
141
+
142
+ # Columns to update in case of conflict
143
+ update_dict = {
144
+ col.name: record.get(col.name)
145
+ for col in table.columns
146
+ if col.name not in ["id", "date", "created_at", "aggregation_period"] and col.name in record
147
+ }
148
+
149
+ stmt = stmt.on_duplicate_key_update(**update_dict)
150
+ session.execute(stmt)
151
+
152
+ session.commit()
153
+
154
+ # Fetch the updated records
155
+ from sqlalchemy import and_, select
156
+
157
+ for record in metrics_records:
158
+ select_stmt = select(table).where(
159
+ and_(table.c.date == record["date"], table.c.aggregation_period == record["aggregation_period"])
160
+ )
161
+ result = session.execute(select_stmt).fetchone()
162
+ if result:
163
+ results.append(result._mapping)
164
+
165
+ return results # type: ignore
166
+
167
+
168
+ def calculate_date_metrics(date_to_process: date, sessions_data: dict) -> dict:
169
+ """Calculate metrics for the given single date.
170
+
171
+ Args:
172
+ date_to_process (date): The date to calculate metrics for.
173
+ sessions_data (dict): The sessions data to calculate metrics for.
174
+
175
+ Returns:
176
+ dict: The calculated metrics.
177
+ """
178
+ metrics = {
179
+ "users_count": 0,
180
+ "agent_sessions_count": 0,
181
+ "team_sessions_count": 0,
182
+ "workflow_sessions_count": 0,
183
+ "agent_runs_count": 0,
184
+ "team_runs_count": 0,
185
+ "workflow_runs_count": 0,
186
+ }
187
+ token_metrics = {
188
+ "input_tokens": 0,
189
+ "output_tokens": 0,
190
+ "total_tokens": 0,
191
+ "audio_total_tokens": 0,
192
+ "audio_input_tokens": 0,
193
+ "audio_output_tokens": 0,
194
+ "cache_read_tokens": 0,
195
+ "cache_write_tokens": 0,
196
+ "reasoning_tokens": 0,
197
+ }
198
+ model_counts: Dict[str, int] = {}
199
+
200
+ session_types = [
201
+ ("agent", "agent_sessions_count", "agent_runs_count"),
202
+ ("team", "team_sessions_count", "team_runs_count"),
203
+ ("workflow", "workflow_sessions_count", "workflow_runs_count"),
204
+ ]
205
+ all_user_ids = set()
206
+
207
+ for session_type, sessions_count_key, runs_count_key in session_types:
208
+ sessions = sessions_data.get(session_type, []) or []
209
+ metrics[sessions_count_key] = len(sessions)
210
+
211
+ for session in sessions:
212
+ if session.get("user_id"):
213
+ all_user_ids.add(session["user_id"])
214
+ metrics[runs_count_key] += len(session.get("runs", []))
215
+ if runs := session.get("runs", []):
216
+ for run in runs:
217
+ if model_id := run.get("model"):
218
+ model_provider = run.get("model_provider", "")
219
+ model_counts[f"{model_id}:{model_provider}"] = (
220
+ model_counts.get(f"{model_id}:{model_provider}", 0) + 1
221
+ )
222
+
223
+ session_metrics = session.get("session_data", {}).get("session_metrics", {})
224
+ for field in token_metrics:
225
+ token_metrics[field] += session_metrics.get(field, 0)
226
+
227
+ model_metrics = []
228
+ for model, count in model_counts.items():
229
+ model_id, model_provider = model.rsplit(":", 1)
230
+ model_metrics.append({"model_id": model_id, "model_provider": model_provider, "count": count})
231
+
232
+ metrics["users_count"] = len(all_user_ids)
233
+ current_time = int(time.time())
234
+
235
+ return {
236
+ "id": str(uuid4()),
237
+ "date": date_to_process,
238
+ "completed": date_to_process < datetime.now(timezone.utc).date(),
239
+ "token_metrics": token_metrics,
240
+ "model_metrics": model_metrics,
241
+ "created_at": current_time,
242
+ "updated_at": current_time,
243
+ "aggregation_period": "daily",
244
+ **metrics,
245
+ }
246
+
247
+
248
+ def fetch_all_sessions_data(
249
+ sessions: List[Dict[str, Any]], dates_to_process: list[date], start_timestamp: int
250
+ ) -> Optional[dict]:
251
+ """Return all session data for the given dates, for all session types.
252
+
253
+ Args:
254
+ dates_to_process (list[date]): The dates to fetch session data for.
255
+
256
+ Returns:
257
+ dict: A dictionary with dates as keys and session data as values, for all session types.
258
+
259
+ Example:
260
+ {
261
+ "2000-01-01": {
262
+ "agent": [<session1>, <session2>, ...],
263
+ "team": [...],
264
+ "workflow": [...],
265
+ }
266
+ }
267
+ """
268
+ if not dates_to_process:
269
+ return None
270
+
271
+ all_sessions_data: Dict[str, Dict[str, List[Dict[str, Any]]]] = {
272
+ date_to_process.isoformat(): {"agent": [], "team": [], "workflow": []} for date_to_process in dates_to_process
273
+ }
274
+
275
+ for session in sessions:
276
+ session_date = (
277
+ datetime.fromtimestamp(session.get("created_at", start_timestamp), tz=timezone.utc).date().isoformat()
278
+ )
279
+ if session_date in all_sessions_data:
280
+ all_sessions_data[session_date][session["session_type"]].append(session)
281
+
282
+ return all_sessions_data
283
+
284
+
285
+ def get_dates_to_calculate_metrics_for(starting_date: date) -> list[date]:
286
+ """Return the list of dates to calculate metrics for.
287
+
288
+ Args:
289
+ starting_date (date): The starting date to calculate metrics for.
290
+
291
+ Returns:
292
+ list[date]: The list of dates to calculate metrics for.
293
+ """
294
+ today = datetime.now(timezone.utc).date()
295
+ days_diff = (today - starting_date).days + 1
296
+ if days_diff <= 0:
297
+ return []
298
+ return [starting_date + timedelta(days=x) for x in range(days_diff)]
299
+
300
+
301
+ # -- Cultural Knowledge util methods --
302
+ def serialize_cultural_knowledge_for_db(cultural_knowledge: CulturalKnowledge) -> Dict[str, Any]:
303
+ """Serialize a CulturalKnowledge object for database storage.
304
+
305
+ Converts the model's separate content, categories, and notes fields
306
+ into a single JSON dict for the database content column.
307
+
308
+ Args:
309
+ cultural_knowledge (CulturalKnowledge): The cultural knowledge object to serialize.
310
+
311
+ Returns:
312
+ Dict[str, Any]: A dictionary with the content field as JSON containing content, categories, and notes.
313
+ """
314
+ content_dict: Dict[str, Any] = {}
315
+ if cultural_knowledge.content is not None:
316
+ content_dict["content"] = cultural_knowledge.content
317
+ if cultural_knowledge.categories is not None:
318
+ content_dict["categories"] = cultural_knowledge.categories
319
+ if cultural_knowledge.notes is not None:
320
+ content_dict["notes"] = cultural_knowledge.notes
321
+
322
+ return content_dict if content_dict else {}
323
+
324
+
325
+ def deserialize_cultural_knowledge_from_db(db_row: Dict[str, Any]) -> CulturalKnowledge:
326
+ """Deserialize a database row to a CulturalKnowledge object.
327
+
328
+ The database stores content as a JSON dict containing content, categories, and notes.
329
+ This method extracts those fields and converts them back to the model format.
330
+
331
+ Args:
332
+ db_row (Dict[str, Any]): The database row as a dictionary.
333
+
334
+ Returns:
335
+ CulturalKnowledge: The cultural knowledge object.
336
+ """
337
+ # Extract content, categories, and notes from the JSON content field
338
+ content_json = db_row.get("content", {}) or {}
339
+
340
+ return CulturalKnowledge.from_dict(
341
+ {
342
+ "id": db_row.get("id"),
343
+ "name": db_row.get("name"),
344
+ "summary": db_row.get("summary"),
345
+ "content": content_json.get("content"),
346
+ "categories": content_json.get("categories"),
347
+ "notes": content_json.get("notes"),
348
+ "metadata": db_row.get("metadata"),
349
+ "input": db_row.get("input"),
350
+ "created_at": db_row.get("created_at"),
351
+ "updated_at": db_row.get("updated_at"),
352
+ "agent_id": db_row.get("agent_id"),
353
+ "team_id": db_row.get("team_id"),
354
+ }
355
+ )
@@ -0,0 +1,4 @@
1
+ from agno.db.postgres.async_postgres import AsyncPostgresDb
2
+ from agno.db.postgres.postgres import PostgresDb
3
+
4
+ __all__ = ["PostgresDb", "AsyncPostgresDb"]