agno 2.0.1__py3-none-any.whl → 2.3.0__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 (314) hide show
  1. agno/agent/agent.py +6015 -2823
  2. agno/api/api.py +2 -0
  3. agno/api/os.py +1 -1
  4. agno/culture/__init__.py +3 -0
  5. agno/culture/manager.py +956 -0
  6. agno/db/async_postgres/__init__.py +3 -0
  7. agno/db/base.py +385 -6
  8. agno/db/dynamo/dynamo.py +388 -81
  9. agno/db/dynamo/schemas.py +47 -10
  10. agno/db/dynamo/utils.py +63 -4
  11. agno/db/firestore/firestore.py +435 -64
  12. agno/db/firestore/schemas.py +11 -0
  13. agno/db/firestore/utils.py +102 -4
  14. agno/db/gcs_json/gcs_json_db.py +384 -42
  15. agno/db/gcs_json/utils.py +60 -26
  16. agno/db/in_memory/in_memory_db.py +351 -66
  17. agno/db/in_memory/utils.py +60 -2
  18. agno/db/json/json_db.py +339 -48
  19. agno/db/json/utils.py +60 -26
  20. agno/db/migrations/manager.py +199 -0
  21. agno/db/migrations/v1_to_v2.py +510 -37
  22. agno/db/migrations/versions/__init__.py +0 -0
  23. agno/db/migrations/versions/v2_3_0.py +938 -0
  24. agno/db/mongo/__init__.py +15 -1
  25. agno/db/mongo/async_mongo.py +2036 -0
  26. agno/db/mongo/mongo.py +653 -76
  27. agno/db/mongo/schemas.py +13 -0
  28. agno/db/mongo/utils.py +80 -8
  29. agno/db/mysql/mysql.py +687 -25
  30. agno/db/mysql/schemas.py +61 -37
  31. agno/db/mysql/utils.py +60 -2
  32. agno/db/postgres/__init__.py +2 -1
  33. agno/db/postgres/async_postgres.py +2001 -0
  34. agno/db/postgres/postgres.py +676 -57
  35. agno/db/postgres/schemas.py +43 -18
  36. agno/db/postgres/utils.py +164 -2
  37. agno/db/redis/redis.py +344 -38
  38. agno/db/redis/schemas.py +18 -0
  39. agno/db/redis/utils.py +60 -2
  40. agno/db/schemas/__init__.py +2 -1
  41. agno/db/schemas/culture.py +120 -0
  42. agno/db/schemas/memory.py +13 -0
  43. agno/db/singlestore/schemas.py +26 -1
  44. agno/db/singlestore/singlestore.py +687 -53
  45. agno/db/singlestore/utils.py +60 -2
  46. agno/db/sqlite/__init__.py +2 -1
  47. agno/db/sqlite/async_sqlite.py +2371 -0
  48. agno/db/sqlite/schemas.py +24 -0
  49. agno/db/sqlite/sqlite.py +774 -85
  50. agno/db/sqlite/utils.py +168 -5
  51. agno/db/surrealdb/__init__.py +3 -0
  52. agno/db/surrealdb/metrics.py +292 -0
  53. agno/db/surrealdb/models.py +309 -0
  54. agno/db/surrealdb/queries.py +71 -0
  55. agno/db/surrealdb/surrealdb.py +1361 -0
  56. agno/db/surrealdb/utils.py +147 -0
  57. agno/db/utils.py +50 -22
  58. agno/eval/accuracy.py +50 -43
  59. agno/eval/performance.py +6 -3
  60. agno/eval/reliability.py +6 -3
  61. agno/eval/utils.py +33 -16
  62. agno/exceptions.py +68 -1
  63. agno/filters.py +354 -0
  64. agno/guardrails/__init__.py +6 -0
  65. agno/guardrails/base.py +19 -0
  66. agno/guardrails/openai.py +144 -0
  67. agno/guardrails/pii.py +94 -0
  68. agno/guardrails/prompt_injection.py +52 -0
  69. agno/integrations/discord/client.py +1 -0
  70. agno/knowledge/chunking/agentic.py +13 -10
  71. agno/knowledge/chunking/fixed.py +1 -1
  72. agno/knowledge/chunking/semantic.py +40 -8
  73. agno/knowledge/chunking/strategy.py +59 -15
  74. agno/knowledge/embedder/aws_bedrock.py +9 -4
  75. agno/knowledge/embedder/azure_openai.py +54 -0
  76. agno/knowledge/embedder/base.py +2 -0
  77. agno/knowledge/embedder/cohere.py +184 -5
  78. agno/knowledge/embedder/fastembed.py +1 -1
  79. agno/knowledge/embedder/google.py +79 -1
  80. agno/knowledge/embedder/huggingface.py +9 -4
  81. agno/knowledge/embedder/jina.py +63 -0
  82. agno/knowledge/embedder/mistral.py +78 -11
  83. agno/knowledge/embedder/nebius.py +1 -1
  84. agno/knowledge/embedder/ollama.py +13 -0
  85. agno/knowledge/embedder/openai.py +37 -65
  86. agno/knowledge/embedder/sentence_transformer.py +8 -4
  87. agno/knowledge/embedder/vllm.py +262 -0
  88. agno/knowledge/embedder/voyageai.py +69 -16
  89. agno/knowledge/knowledge.py +594 -186
  90. agno/knowledge/reader/base.py +9 -2
  91. agno/knowledge/reader/csv_reader.py +8 -10
  92. agno/knowledge/reader/docx_reader.py +5 -6
  93. agno/knowledge/reader/field_labeled_csv_reader.py +290 -0
  94. agno/knowledge/reader/json_reader.py +6 -5
  95. agno/knowledge/reader/markdown_reader.py +13 -13
  96. agno/knowledge/reader/pdf_reader.py +43 -68
  97. agno/knowledge/reader/pptx_reader.py +101 -0
  98. agno/knowledge/reader/reader_factory.py +51 -6
  99. agno/knowledge/reader/s3_reader.py +3 -15
  100. agno/knowledge/reader/tavily_reader.py +194 -0
  101. agno/knowledge/reader/text_reader.py +13 -13
  102. agno/knowledge/reader/web_search_reader.py +2 -43
  103. agno/knowledge/reader/website_reader.py +43 -25
  104. agno/knowledge/reranker/__init__.py +2 -8
  105. agno/knowledge/types.py +9 -0
  106. agno/knowledge/utils.py +20 -0
  107. agno/media.py +72 -0
  108. agno/memory/manager.py +336 -82
  109. agno/models/aimlapi/aimlapi.py +2 -2
  110. agno/models/anthropic/claude.py +183 -37
  111. agno/models/aws/bedrock.py +52 -112
  112. agno/models/aws/claude.py +33 -1
  113. agno/models/azure/ai_foundry.py +33 -15
  114. agno/models/azure/openai_chat.py +25 -8
  115. agno/models/base.py +999 -519
  116. agno/models/cerebras/cerebras.py +19 -13
  117. agno/models/cerebras/cerebras_openai.py +8 -5
  118. agno/models/cohere/chat.py +27 -1
  119. agno/models/cometapi/__init__.py +5 -0
  120. agno/models/cometapi/cometapi.py +57 -0
  121. agno/models/dashscope/dashscope.py +1 -0
  122. agno/models/deepinfra/deepinfra.py +2 -2
  123. agno/models/deepseek/deepseek.py +2 -2
  124. agno/models/fireworks/fireworks.py +2 -2
  125. agno/models/google/gemini.py +103 -31
  126. agno/models/groq/groq.py +28 -11
  127. agno/models/huggingface/huggingface.py +2 -1
  128. agno/models/internlm/internlm.py +2 -2
  129. agno/models/langdb/langdb.py +4 -4
  130. agno/models/litellm/chat.py +18 -1
  131. agno/models/litellm/litellm_openai.py +2 -2
  132. agno/models/llama_cpp/__init__.py +5 -0
  133. agno/models/llama_cpp/llama_cpp.py +22 -0
  134. agno/models/message.py +139 -0
  135. agno/models/meta/llama.py +27 -10
  136. agno/models/meta/llama_openai.py +5 -17
  137. agno/models/nebius/nebius.py +6 -6
  138. agno/models/nexus/__init__.py +3 -0
  139. agno/models/nexus/nexus.py +22 -0
  140. agno/models/nvidia/nvidia.py +2 -2
  141. agno/models/ollama/chat.py +59 -5
  142. agno/models/openai/chat.py +69 -29
  143. agno/models/openai/responses.py +103 -106
  144. agno/models/openrouter/openrouter.py +41 -3
  145. agno/models/perplexity/perplexity.py +4 -5
  146. agno/models/portkey/portkey.py +3 -3
  147. agno/models/requesty/__init__.py +5 -0
  148. agno/models/requesty/requesty.py +52 -0
  149. agno/models/response.py +77 -1
  150. agno/models/sambanova/sambanova.py +2 -2
  151. agno/models/siliconflow/__init__.py +5 -0
  152. agno/models/siliconflow/siliconflow.py +25 -0
  153. agno/models/together/together.py +2 -2
  154. agno/models/utils.py +254 -8
  155. agno/models/vercel/v0.py +2 -2
  156. agno/models/vertexai/__init__.py +0 -0
  157. agno/models/vertexai/claude.py +96 -0
  158. agno/models/vllm/vllm.py +1 -0
  159. agno/models/xai/xai.py +3 -2
  160. agno/os/app.py +543 -178
  161. agno/os/auth.py +24 -14
  162. agno/os/config.py +1 -0
  163. agno/os/interfaces/__init__.py +1 -0
  164. agno/os/interfaces/a2a/__init__.py +3 -0
  165. agno/os/interfaces/a2a/a2a.py +42 -0
  166. agno/os/interfaces/a2a/router.py +250 -0
  167. agno/os/interfaces/a2a/utils.py +924 -0
  168. agno/os/interfaces/agui/agui.py +23 -7
  169. agno/os/interfaces/agui/router.py +27 -3
  170. agno/os/interfaces/agui/utils.py +242 -142
  171. agno/os/interfaces/base.py +6 -2
  172. agno/os/interfaces/slack/router.py +81 -23
  173. agno/os/interfaces/slack/slack.py +29 -14
  174. agno/os/interfaces/whatsapp/router.py +11 -4
  175. agno/os/interfaces/whatsapp/whatsapp.py +14 -7
  176. agno/os/mcp.py +111 -54
  177. agno/os/middleware/__init__.py +7 -0
  178. agno/os/middleware/jwt.py +233 -0
  179. agno/os/router.py +556 -139
  180. agno/os/routers/evals/evals.py +71 -34
  181. agno/os/routers/evals/schemas.py +31 -31
  182. agno/os/routers/evals/utils.py +6 -5
  183. agno/os/routers/health.py +31 -0
  184. agno/os/routers/home.py +52 -0
  185. agno/os/routers/knowledge/knowledge.py +185 -38
  186. agno/os/routers/knowledge/schemas.py +82 -22
  187. agno/os/routers/memory/memory.py +158 -53
  188. agno/os/routers/memory/schemas.py +20 -16
  189. agno/os/routers/metrics/metrics.py +20 -8
  190. agno/os/routers/metrics/schemas.py +16 -16
  191. agno/os/routers/session/session.py +499 -38
  192. agno/os/schema.py +308 -198
  193. agno/os/utils.py +401 -41
  194. agno/reasoning/anthropic.py +80 -0
  195. agno/reasoning/azure_ai_foundry.py +2 -2
  196. agno/reasoning/deepseek.py +2 -2
  197. agno/reasoning/default.py +3 -1
  198. agno/reasoning/gemini.py +73 -0
  199. agno/reasoning/groq.py +2 -2
  200. agno/reasoning/ollama.py +2 -2
  201. agno/reasoning/openai.py +7 -2
  202. agno/reasoning/vertexai.py +76 -0
  203. agno/run/__init__.py +6 -0
  204. agno/run/agent.py +248 -94
  205. agno/run/base.py +44 -5
  206. agno/run/team.py +238 -97
  207. agno/run/workflow.py +144 -33
  208. agno/session/agent.py +105 -89
  209. agno/session/summary.py +65 -25
  210. agno/session/team.py +176 -96
  211. agno/session/workflow.py +406 -40
  212. agno/team/team.py +3854 -1610
  213. agno/tools/dalle.py +2 -4
  214. agno/tools/decorator.py +4 -2
  215. agno/tools/duckduckgo.py +15 -11
  216. agno/tools/e2b.py +14 -7
  217. agno/tools/eleven_labs.py +23 -25
  218. agno/tools/exa.py +21 -16
  219. agno/tools/file.py +153 -23
  220. agno/tools/file_generation.py +350 -0
  221. agno/tools/firecrawl.py +4 -4
  222. agno/tools/function.py +250 -30
  223. agno/tools/gmail.py +238 -14
  224. agno/tools/google_drive.py +270 -0
  225. agno/tools/googlecalendar.py +36 -8
  226. agno/tools/googlesheets.py +20 -5
  227. agno/tools/jira.py +20 -0
  228. agno/tools/knowledge.py +3 -3
  229. agno/tools/mcp/__init__.py +10 -0
  230. agno/tools/mcp/mcp.py +331 -0
  231. agno/tools/mcp/multi_mcp.py +347 -0
  232. agno/tools/mcp/params.py +24 -0
  233. agno/tools/mcp_toolbox.py +284 -0
  234. agno/tools/mem0.py +11 -17
  235. agno/tools/memori.py +1 -53
  236. agno/tools/memory.py +419 -0
  237. agno/tools/models/nebius.py +5 -5
  238. agno/tools/models_labs.py +20 -10
  239. agno/tools/notion.py +204 -0
  240. agno/tools/parallel.py +314 -0
  241. agno/tools/scrapegraph.py +58 -31
  242. agno/tools/searxng.py +2 -2
  243. agno/tools/serper.py +2 -2
  244. agno/tools/slack.py +18 -3
  245. agno/tools/spider.py +2 -2
  246. agno/tools/tavily.py +146 -0
  247. agno/tools/whatsapp.py +1 -1
  248. agno/tools/workflow.py +278 -0
  249. agno/tools/yfinance.py +12 -11
  250. agno/utils/agent.py +820 -0
  251. agno/utils/audio.py +27 -0
  252. agno/utils/common.py +90 -1
  253. agno/utils/events.py +217 -2
  254. agno/utils/gemini.py +180 -22
  255. agno/utils/hooks.py +57 -0
  256. agno/utils/http.py +111 -0
  257. agno/utils/knowledge.py +12 -5
  258. agno/utils/log.py +1 -0
  259. agno/utils/mcp.py +92 -2
  260. agno/utils/media.py +188 -10
  261. agno/utils/merge_dict.py +22 -1
  262. agno/utils/message.py +60 -0
  263. agno/utils/models/claude.py +40 -11
  264. agno/utils/print_response/agent.py +105 -21
  265. agno/utils/print_response/team.py +103 -38
  266. agno/utils/print_response/workflow.py +251 -34
  267. agno/utils/reasoning.py +22 -1
  268. agno/utils/serialize.py +32 -0
  269. agno/utils/streamlit.py +16 -10
  270. agno/utils/string.py +41 -0
  271. agno/utils/team.py +98 -9
  272. agno/utils/tools.py +1 -1
  273. agno/vectordb/base.py +23 -4
  274. agno/vectordb/cassandra/cassandra.py +65 -9
  275. agno/vectordb/chroma/chromadb.py +182 -38
  276. agno/vectordb/clickhouse/clickhousedb.py +64 -11
  277. agno/vectordb/couchbase/couchbase.py +105 -10
  278. agno/vectordb/lancedb/lance_db.py +124 -133
  279. agno/vectordb/langchaindb/langchaindb.py +25 -7
  280. agno/vectordb/lightrag/lightrag.py +17 -3
  281. agno/vectordb/llamaindex/__init__.py +3 -0
  282. agno/vectordb/llamaindex/llamaindexdb.py +46 -7
  283. agno/vectordb/milvus/milvus.py +126 -9
  284. agno/vectordb/mongodb/__init__.py +7 -1
  285. agno/vectordb/mongodb/mongodb.py +112 -7
  286. agno/vectordb/pgvector/pgvector.py +142 -21
  287. agno/vectordb/pineconedb/pineconedb.py +80 -8
  288. agno/vectordb/qdrant/qdrant.py +125 -39
  289. agno/vectordb/redis/__init__.py +9 -0
  290. agno/vectordb/redis/redisdb.py +694 -0
  291. agno/vectordb/singlestore/singlestore.py +111 -25
  292. agno/vectordb/surrealdb/surrealdb.py +31 -5
  293. agno/vectordb/upstashdb/upstashdb.py +76 -8
  294. agno/vectordb/weaviate/weaviate.py +86 -15
  295. agno/workflow/__init__.py +2 -0
  296. agno/workflow/agent.py +299 -0
  297. agno/workflow/condition.py +112 -18
  298. agno/workflow/loop.py +69 -10
  299. agno/workflow/parallel.py +266 -118
  300. agno/workflow/router.py +110 -17
  301. agno/workflow/step.py +638 -129
  302. agno/workflow/steps.py +65 -6
  303. agno/workflow/types.py +61 -23
  304. agno/workflow/workflow.py +2085 -272
  305. {agno-2.0.1.dist-info → agno-2.3.0.dist-info}/METADATA +182 -58
  306. agno-2.3.0.dist-info/RECORD +577 -0
  307. agno/knowledge/reader/url_reader.py +0 -128
  308. agno/tools/googlesearch.py +0 -98
  309. agno/tools/mcp.py +0 -610
  310. agno/utils/models/aws_claude.py +0 -170
  311. agno-2.0.1.dist-info/RECORD +0 -515
  312. {agno-2.0.1.dist-info → agno-2.3.0.dist-info}/WHEEL +0 -0
  313. {agno-2.0.1.dist-info → agno-2.3.0.dist-info}/licenses/LICENSE +0 -0
  314. {agno-2.0.1.dist-info → agno-2.3.0.dist-info}/top_level.txt +0 -0
agno/db/sqlite/utils.py CHANGED
@@ -4,6 +4,9 @@ from datetime import date, datetime, timedelta, timezone
4
4
  from typing import Any, Dict, List, Optional
5
5
  from uuid import uuid4
6
6
 
7
+ from sqlalchemy.ext.asyncio import AsyncEngine, AsyncSession
8
+
9
+ from agno.db.schemas.culture import CulturalKnowledge
7
10
  from agno.db.sqlite.schemas import get_table_schema_definition
8
11
  from agno.utils.log import log_debug, log_error, log_warning
9
12
 
@@ -49,6 +52,7 @@ def is_table_available(session: Session, table_name: str, db_schema: Optional[st
49
52
  """
50
53
  Check if a table with the given name exists.
51
54
  Note: db_schema parameter is ignored in SQLite but kept for API compatibility.
55
+
52
56
  Returns:
53
57
  bool: True if the table exists, False otherwise.
54
58
  """
@@ -64,6 +68,25 @@ def is_table_available(session: Session, table_name: str, db_schema: Optional[st
64
68
  return False
65
69
 
66
70
 
71
+ async def ais_table_available(session: AsyncSession, table_name: str, db_schema: Optional[str] = None) -> bool:
72
+ """
73
+ Check if a table with the given name exists.
74
+ Note: db_schema parameter is ignored in SQLite but kept for API compatibility.
75
+
76
+ Returns:
77
+ bool: True if the table exists, False otherwise.
78
+ """
79
+ try:
80
+ exists_query = text("SELECT 1 FROM sqlite_master WHERE type = 'table' AND name = :table")
81
+ exists = (await session.execute(exists_query, {"table": table_name})).scalar() is not None
82
+ if not exists:
83
+ log_debug(f"Table {table_name} {'exists' if exists else 'does not exist'}")
84
+ return exists
85
+ except Exception as e:
86
+ log_error(f"Error checking if table exists: {e}")
87
+ return False
88
+
89
+
67
90
  def is_valid_table(db_engine: Engine, table_name: str, table_type: str, db_schema: Optional[str] = None) -> bool:
68
91
  """
69
92
  Check if the existing table has the expected column names.
@@ -97,6 +120,47 @@ def is_valid_table(db_engine: Engine, table_name: str, table_type: str, db_schem
97
120
  return False
98
121
 
99
122
 
123
+ async def ais_valid_table(
124
+ db_engine: AsyncEngine, table_name: str, table_type: str, db_schema: Optional[str] = None
125
+ ) -> bool:
126
+ """
127
+ Check if the existing table has the expected column names.
128
+ Note: db_schema parameter is ignored in SQLite but kept for API compatibility.
129
+ Args:
130
+ db_engine (Engine): Database engine
131
+ table_name (str): Name of the table to validate
132
+ table_type (str): Type of table to get expected schema
133
+ db_schema (Optional[str]): Database schema name (ignored in SQLite)
134
+ Returns:
135
+ bool: True if table has all expected columns, False otherwise
136
+ """
137
+ try:
138
+ expected_table_schema = get_table_schema_definition(table_type)
139
+ expected_columns = {col_name for col_name in expected_table_schema.keys() if not col_name.startswith("_")}
140
+
141
+ # Get existing columns from the async engine
142
+ async with db_engine.connect() as conn:
143
+ existing_columns = await conn.run_sync(_get_table_columns, table_name)
144
+
145
+ missing_columns = expected_columns - existing_columns
146
+ if missing_columns:
147
+ log_warning(f"Missing columns {missing_columns} in table {table_name}")
148
+ return False
149
+
150
+ return True
151
+
152
+ except Exception as e:
153
+ log_error(f"Error validating table schema for {table_name}: {e}")
154
+ return False
155
+
156
+
157
+ def _get_table_columns(conn, table_name: str) -> set[str]:
158
+ """Helper function to get table columns using sync inspector."""
159
+ inspector = inspect(conn)
160
+ columns_info = inspector.get_columns(table_name)
161
+ return {col["name"] for col in columns_info}
162
+
163
+
100
164
  # -- Metrics util methods --
101
165
 
102
166
 
@@ -133,6 +197,39 @@ def bulk_upsert_metrics(session: Session, table: Table, metrics_records: list[di
133
197
  return results # type: ignore
134
198
 
135
199
 
200
+ async def abulk_upsert_metrics(session: AsyncSession, table: Table, metrics_records: list[dict]) -> list[dict]:
201
+ """Bulk upsert metrics into the database.
202
+
203
+ Args:
204
+ table (Table): The table to upsert into.
205
+ metrics_records (list[dict]): The metrics records to upsert.
206
+
207
+ Returns:
208
+ list[dict]: The upserted metrics records.
209
+ """
210
+ if not metrics_records:
211
+ return []
212
+
213
+ results = []
214
+ stmt = sqlite.insert(table)
215
+
216
+ # Columns to update in case of conflict
217
+ update_columns = {
218
+ col.name: stmt.excluded[col.name]
219
+ for col in table.columns
220
+ if col.name not in ["id", "date", "created_at", "aggregation_period"]
221
+ }
222
+
223
+ stmt = stmt.on_conflict_do_update(index_elements=["date", "aggregation_period"], set_=update_columns).returning( # type: ignore
224
+ table
225
+ )
226
+ result = await session.execute(stmt, metrics_records)
227
+ results = [dict(row._mapping) for row in result.fetchall()]
228
+ await session.commit()
229
+
230
+ return results # type: ignore
231
+
232
+
136
233
  def calculate_date_metrics(date_to_process: date, sessions_data: dict) -> dict:
137
234
  """Calculate metrics for the given single date.
138
235
 
@@ -173,15 +270,17 @@ def calculate_date_metrics(date_to_process: date, sessions_data: dict) -> dict:
173
270
  all_user_ids = set()
174
271
 
175
272
  for session_type, sessions_count_key, runs_count_key in session_types:
176
- sessions = sessions_data.get(session_type, [])
273
+ sessions = sessions_data.get(session_type, []) or []
177
274
  metrics[sessions_count_key] = len(sessions)
178
275
 
179
276
  for session in sessions:
180
277
  if session.get("user_id"):
181
278
  all_user_ids.add(session["user_id"])
182
- metrics[runs_count_key] += len(session.get("runs", []))
279
+
280
+ # Parse runs from JSON string
183
281
  if runs := session.get("runs", []):
184
- runs = json.loads(runs)
282
+ runs = json.loads(runs) if isinstance(runs, str) else runs
283
+ metrics[runs_count_key] += len(runs)
185
284
  for run in runs:
186
285
  if model_id := run.get("model"):
187
286
  model_provider = run.get("model_provider", "")
@@ -189,14 +288,17 @@ def calculate_date_metrics(date_to_process: date, sessions_data: dict) -> dict:
189
288
  model_counts.get(f"{model_id}:{model_provider}", 0) + 1
190
289
  )
191
290
 
192
- session_data = json.loads(session.get("session_data", {}))
291
+ # Parse session_data from JSON string
292
+ session_data = session.get("session_data", {})
293
+ if isinstance(session_data, str):
294
+ session_data = json.loads(session_data)
193
295
  session_metrics = session_data.get("session_metrics", {})
194
296
  for field in token_metrics:
195
297
  token_metrics[field] += session_metrics.get(field, 0)
196
298
 
197
299
  model_metrics = []
198
300
  for model, count in model_counts.items():
199
- model_id, model_provider = model.split(":")
301
+ model_id, model_provider = model.rsplit(":", 1)
200
302
  model_metrics.append({"model_id": model_id, "model_provider": model_provider, "count": count})
201
303
 
202
304
  metrics["users_count"] = len(all_user_ids)
@@ -266,3 +368,64 @@ def get_dates_to_calculate_metrics_for(starting_date: date) -> list[date]:
266
368
  if days_diff <= 0:
267
369
  return []
268
370
  return [starting_date + timedelta(days=x) for x in range(days_diff)]
371
+
372
+
373
+ # -- Cultural Knowledge util methods --
374
+ def serialize_cultural_knowledge_for_db(cultural_knowledge: CulturalKnowledge) -> str:
375
+ """Serialize a CulturalKnowledge object for database storage.
376
+
377
+ Converts the model's separate content, categories, and notes fields
378
+ into a single JSON string for the database content column.
379
+ SQLite requires JSON to be stored as strings.
380
+
381
+ Args:
382
+ cultural_knowledge (CulturalKnowledge): The cultural knowledge object to serialize.
383
+
384
+ Returns:
385
+ str: A JSON string containing content, categories, and notes.
386
+ """
387
+ content_dict: Dict[str, Any] = {}
388
+ if cultural_knowledge.content is not None:
389
+ content_dict["content"] = cultural_knowledge.content
390
+ if cultural_knowledge.categories is not None:
391
+ content_dict["categories"] = cultural_knowledge.categories
392
+ if cultural_knowledge.notes is not None:
393
+ content_dict["notes"] = cultural_knowledge.notes
394
+
395
+ return json.dumps(content_dict) if content_dict else None # type: ignore
396
+
397
+
398
+ def deserialize_cultural_knowledge_from_db(db_row: Dict[str, Any]) -> CulturalKnowledge:
399
+ """Deserialize a database row to a CulturalKnowledge object.
400
+
401
+ The database stores content as a JSON dict containing content, categories, and notes.
402
+ This method extracts those fields and converts them back to the model format.
403
+
404
+ Args:
405
+ db_row (Dict[str, Any]): The database row as a dictionary.
406
+
407
+ Returns:
408
+ CulturalKnowledge: The cultural knowledge object.
409
+ """
410
+ # Extract content, categories, and notes from the JSON content field
411
+ content_json = db_row.get("content", {}) or {}
412
+
413
+ if isinstance(content_json, str):
414
+ content_json = json.loads(content_json) if content_json else {}
415
+
416
+ return CulturalKnowledge.from_dict(
417
+ {
418
+ "id": db_row.get("id"),
419
+ "name": db_row.get("name"),
420
+ "summary": db_row.get("summary"),
421
+ "content": content_json.get("content"),
422
+ "categories": content_json.get("categories"),
423
+ "notes": content_json.get("notes"),
424
+ "metadata": db_row.get("metadata"),
425
+ "input": db_row.get("input"),
426
+ "created_at": db_row.get("created_at"),
427
+ "updated_at": db_row.get("updated_at"),
428
+ "agent_id": db_row.get("agent_id"),
429
+ "team_id": db_row.get("team_id"),
430
+ }
431
+ )
@@ -0,0 +1,3 @@
1
+ from agno.db.surrealdb.surrealdb import SurrealDb
2
+
3
+ __all__ = ["SurrealDb"]
@@ -0,0 +1,292 @@
1
+ from datetime import date, datetime, timedelta, timezone
2
+ from textwrap import dedent
3
+ from typing import Any, Callable, Dict, List, Optional, Union
4
+
5
+ from surrealdb import BlockingHttpSurrealConnection, BlockingWsSurrealConnection, RecordID
6
+
7
+ from agno.db.base import SessionType
8
+ from agno.db.surrealdb import utils
9
+ from agno.db.surrealdb.models import desurrealize_session, surrealize_dates
10
+ from agno.db.surrealdb.queries import WhereClause
11
+ from agno.utils.log import log_error
12
+
13
+
14
+ def get_all_sessions_for_metrics_calculation(
15
+ client: Union[BlockingWsSurrealConnection, BlockingHttpSurrealConnection],
16
+ table: str,
17
+ start_timestamp: Optional[datetime] = None,
18
+ end_timestamp: Optional[datetime] = None,
19
+ ) -> List[Dict[str, Any]]:
20
+ """
21
+ Get all sessions of all types (agent, team, workflow) as raw dictionaries.
22
+
23
+ Args:
24
+ start_timestamp (Optional[int]): The start timestamp to filter by. Defaults to None.
25
+ end_timestamp (Optional[int]): The end timestamp to filter by. Defaults to None.
26
+
27
+ Returns:
28
+ List[Dict[str, Any]]: List of session dictionaries with session_type field.
29
+
30
+ Raises:
31
+ Exception: If an error occurs during retrieval.
32
+ """
33
+ where = WhereClause()
34
+
35
+ # starting_date
36
+ if start_timestamp is not None:
37
+ where = where.and_("created_at", start_timestamp, ">=")
38
+
39
+ # ending_date
40
+ if end_timestamp is not None:
41
+ where = where.and_("created_at", end_timestamp, "<=")
42
+
43
+ where_clause, where_vars = where.build()
44
+
45
+ # Query
46
+ query = dedent(f"""
47
+ SELECT *
48
+ FROM {table}
49
+ {where_clause}
50
+ """)
51
+
52
+ results = utils.query(client, query, where_vars, dict)
53
+ return [desurrealize_session(x) for x in results]
54
+
55
+
56
+ def get_metrics_calculation_starting_date(
57
+ client: Union[BlockingWsSurrealConnection, BlockingHttpSurrealConnection], table: str, get_sessions: Callable
58
+ ) -> Optional[date]:
59
+ """Get the first date for which metrics calculation is needed:
60
+
61
+ 1. If there are metrics records, return the date of the first day without a complete metrics record.
62
+ 2. If there are no metrics records, return the date of the first recorded session.
63
+ 3. If there are no metrics records and no sessions records, return None.
64
+
65
+ Args:
66
+ table (Table): The table to get the starting date for.
67
+
68
+ Returns:
69
+ Optional[date]: The starting date for which metrics calculation is needed.
70
+ """
71
+ query = dedent(f"""
72
+ SELECT * FROM ONLY {table}
73
+ ORDER BY date DESC
74
+ LIMIT 1
75
+ """)
76
+ result = utils.query_one(client, query, {}, dict)
77
+ if result:
78
+ # 1. Return the date of the first day without a complete metrics record
79
+ result_date = result["date"]
80
+ assert isinstance(result_date, datetime)
81
+ result_date = result_date.date()
82
+
83
+ if result.get("completed"):
84
+ return result_date + timedelta(days=1)
85
+ else:
86
+ return result_date
87
+
88
+ # 2. No metrics records. Return the date of the first recorded session
89
+ first_session, _ = get_sessions(
90
+ session_type=SessionType.AGENT, # this is ignored because of component_id=None and deserialize=False
91
+ sort_by="created_at",
92
+ sort_order="asc",
93
+ limit=1,
94
+ component_id=None,
95
+ deserialize=False,
96
+ )
97
+ assert isinstance(first_session, list)
98
+
99
+ first_session_date = first_session[0]["created_at"] if first_session else None
100
+
101
+ # 3. No metrics records and no sessions records. Return None
102
+ if first_session_date is None:
103
+ return None
104
+
105
+ # Handle different types for created_at
106
+ if isinstance(first_session_date, datetime):
107
+ return first_session_date.date()
108
+ elif isinstance(first_session_date, int):
109
+ # Assume it's a Unix timestamp
110
+ return datetime.fromtimestamp(first_session_date, tz=timezone.utc).date()
111
+ elif isinstance(first_session_date, str):
112
+ # Try parsing as ISO format
113
+ return datetime.fromisoformat(first_session_date.replace("Z", "+00:00")).date()
114
+ else:
115
+ # If it's already a date object
116
+ if isinstance(first_session_date, date):
117
+ return first_session_date
118
+ raise ValueError(f"Unexpected type for created_at: {type(first_session_date)}")
119
+
120
+
121
+ def bulk_upsert_metrics(
122
+ client: Union[BlockingWsSurrealConnection, BlockingHttpSurrealConnection],
123
+ table: str,
124
+ metrics_records: List[Dict[str, Any]],
125
+ ) -> List[Dict[str, Any]]:
126
+ """Bulk upsert metrics into the database.
127
+
128
+ Args:
129
+ table (Table): The table to upsert into.
130
+ metrics_records (List[Dict[str, Any]]): The list of metrics records to upsert.
131
+
132
+ Returns:
133
+ list[dict]: The upserted metrics records.
134
+ """
135
+ if not metrics_records:
136
+ return []
137
+
138
+ metrics_records = [surrealize_dates(x) for x in metrics_records]
139
+
140
+ try:
141
+ results = []
142
+ from agno.utils.log import log_debug
143
+
144
+ for metric in metrics_records:
145
+ log_debug(f"Upserting metric: {metric}") # Add this
146
+ result = utils.query_one(
147
+ client,
148
+ "UPSERT $record CONTENT $content",
149
+ {"record": RecordID(table, metric["id"]), "content": metric},
150
+ dict,
151
+ )
152
+ if result:
153
+ results.append(result)
154
+ return results
155
+
156
+ except Exception as e:
157
+ import traceback
158
+
159
+ log_error(traceback.format_exc())
160
+ log_error(f"Error upserting metrics: {e}")
161
+
162
+ return []
163
+
164
+
165
+ def fetch_all_sessions_data(
166
+ sessions: List[Dict[str, Any]], dates_to_process: list[date], start_timestamp: int
167
+ ) -> Optional[dict]:
168
+ """Return all session data for the given dates, for all session types.
169
+
170
+ Args:
171
+ sessions (List[Dict[str, Any]]): The sessions to process.
172
+ dates_to_process (list[date]): The dates to fetch session data for.
173
+ start_timestamp (int): The start timestamp (fallback if created_at is missing).
174
+
175
+ Returns:
176
+ dict: A dictionary with dates as keys and session data as values, for all session types.
177
+
178
+ Example:
179
+ {
180
+ "2000-01-01": {
181
+ "agent": [<session1>, <session2>, ...],
182
+ "team": [...],
183
+ "workflow": [...],
184
+ }
185
+ }
186
+ """
187
+ if not dates_to_process:
188
+ return None
189
+
190
+ all_sessions_data: Dict[str, Dict[str, List[Dict[str, Any]]]] = {
191
+ date_to_process.isoformat(): {"agent": [], "team": [], "workflow": []} for date_to_process in dates_to_process
192
+ }
193
+
194
+ for session in sessions:
195
+ created_at = session.get("created_at", start_timestamp)
196
+
197
+ # Handle different types for created_at
198
+ if isinstance(created_at, datetime):
199
+ session_date = created_at.date().isoformat()
200
+ elif isinstance(created_at, int):
201
+ session_date = datetime.fromtimestamp(created_at, tz=timezone.utc).date().isoformat()
202
+ elif isinstance(created_at, date):
203
+ session_date = created_at.isoformat()
204
+ else:
205
+ # Fallback to start_timestamp if type is unexpected
206
+ session_date = datetime.fromtimestamp(start_timestamp, tz=timezone.utc).date().isoformat()
207
+
208
+ if session_date in all_sessions_data:
209
+ session_type = session.get("session_type", "agent") # Default to agent if missing
210
+ all_sessions_data[session_date][session_type].append(session)
211
+
212
+ return all_sessions_data
213
+
214
+
215
+ def calculate_date_metrics(date_to_process: date, sessions_data: dict) -> dict:
216
+ """Calculate metrics for the given single date.
217
+
218
+ Args:
219
+ date_to_process (date): The date to calculate metrics for.
220
+ sessions_data (dict): The sessions data to calculate metrics for.
221
+
222
+ Returns:
223
+ dict: The calculated metrics.
224
+ """
225
+ metrics = {
226
+ "users_count": 0,
227
+ "agent_sessions_count": 0,
228
+ "team_sessions_count": 0,
229
+ "workflow_sessions_count": 0,
230
+ "agent_runs_count": 0,
231
+ "team_runs_count": 0,
232
+ "workflow_runs_count": 0,
233
+ }
234
+ token_metrics = {
235
+ "input_tokens": 0,
236
+ "output_tokens": 0,
237
+ "total_tokens": 0,
238
+ "audio_total_tokens": 0,
239
+ "audio_input_tokens": 0,
240
+ "audio_output_tokens": 0,
241
+ "cache_read_tokens": 0,
242
+ "cache_write_tokens": 0,
243
+ "reasoning_tokens": 0,
244
+ }
245
+ model_counts: Dict[str, int] = {}
246
+
247
+ session_types = [
248
+ ("agent", "agent_sessions_count", "agent_runs_count"),
249
+ ("team", "team_sessions_count", "team_runs_count"),
250
+ ("workflow", "workflow_sessions_count", "workflow_runs_count"),
251
+ ]
252
+ all_user_ids = set()
253
+
254
+ for session_type, sessions_count_key, runs_count_key in session_types:
255
+ sessions = sessions_data.get(session_type, [])
256
+ metrics[sessions_count_key] = len(sessions)
257
+
258
+ for session in sessions:
259
+ if session.get("user_id"):
260
+ all_user_ids.add(session["user_id"])
261
+ metrics[runs_count_key] += len(session.get("runs", []))
262
+ if runs := session.get("runs", []):
263
+ for run in runs:
264
+ if model_id := run.get("model"):
265
+ model_provider = run.get("model_provider", "")
266
+ model_counts[f"{model_id}:{model_provider}"] = (
267
+ model_counts.get(f"{model_id}:{model_provider}", 0) + 1
268
+ )
269
+
270
+ session_metrics = session.get("session_data", {}).get("session_metrics", {})
271
+ for field in token_metrics:
272
+ token_metrics[field] += session_metrics.get(field, 0)
273
+
274
+ model_metrics = []
275
+ for model, count in model_counts.items():
276
+ model_id, model_provider = model.split(":")
277
+ model_metrics.append({"model_id": model_id, "model_provider": model_provider, "count": count})
278
+
279
+ metrics["users_count"] = len(all_user_ids)
280
+ current_time = datetime.now(timezone.utc)
281
+
282
+ return {
283
+ "id": date_to_process.isoformat(), # Changed: Use date as ID (e.g., "2025-10-16")
284
+ "date": current_time.replace(hour=0, minute=0, second=0, microsecond=0), # Date at midnight UTC
285
+ "completed": date_to_process < datetime.now(timezone.utc).date(),
286
+ "token_metrics": token_metrics,
287
+ "model_metrics": model_metrics,
288
+ "created_at": current_time,
289
+ "updated_at": current_time,
290
+ "aggregation_period": "daily",
291
+ **metrics,
292
+ }