veadk-python 0.2.10__py3-none-any.whl → 0.2.11__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.

Potentially problematic release.


This version of veadk-python might be problematic. Click here for more details.

Files changed (29) hide show
  1. veadk/agent.py +7 -3
  2. veadk/auth/veauth/ark_veauth.py +43 -51
  3. veadk/auth/veauth/utils.py +57 -0
  4. veadk/configs/model_configs.py +3 -3
  5. veadk/consts.py +9 -0
  6. veadk/knowledgebase/knowledgebase.py +19 -32
  7. veadk/memory/long_term_memory.py +39 -92
  8. veadk/memory/long_term_memory_backends/base_backend.py +4 -2
  9. veadk/memory/long_term_memory_backends/in_memory_backend.py +8 -6
  10. veadk/memory/long_term_memory_backends/mem0_backend.py +8 -8
  11. veadk/memory/long_term_memory_backends/opensearch_backend.py +40 -36
  12. veadk/memory/long_term_memory_backends/redis_backend.py +59 -46
  13. veadk/memory/long_term_memory_backends/vikingdb_memory_backend.py +53 -28
  14. veadk/memory/short_term_memory.py +9 -11
  15. veadk/runner.py +19 -11
  16. veadk/tools/builtin_tools/generate_image.py +11 -6
  17. veadk/tools/builtin_tools/image_edit.py +9 -4
  18. veadk/tools/builtin_tools/image_generate.py +9 -4
  19. veadk/tools/builtin_tools/load_knowledgebase.py +97 -0
  20. veadk/tools/builtin_tools/video_generate.py +6 -4
  21. veadk/utils/misc.py +6 -10
  22. veadk/utils/volcengine_sign.py +2 -0
  23. veadk/version.py +1 -1
  24. {veadk_python-0.2.10.dist-info → veadk_python-0.2.11.dist-info}/METADATA +2 -1
  25. {veadk_python-0.2.10.dist-info → veadk_python-0.2.11.dist-info}/RECORD +29 -27
  26. {veadk_python-0.2.10.dist-info → veadk_python-0.2.11.dist-info}/WHEEL +0 -0
  27. {veadk_python-0.2.10.dist-info → veadk_python-0.2.11.dist-info}/entry_points.txt +0 -0
  28. {veadk_python-0.2.10.dist-info → veadk_python-0.2.11.dist-info}/licenses/LICENSE +0 -0
  29. {veadk_python-0.2.10.dist-info → veadk_python-0.2.11.dist-info}/top_level.txt +0 -0
veadk/agent.py CHANGED
@@ -133,10 +133,14 @@ class Agent(LlmAgent):
133
133
  )
134
134
 
135
135
  if self.knowledgebase:
136
- from veadk.tools import load_knowledgebase_tool
136
+ from veadk.tools.builtin_tools.load_knowledgebase import (
137
+ LoadKnowledgebaseTool,
138
+ )
137
139
 
138
- load_knowledgebase_tool.knowledgebase = self.knowledgebase
139
- self.tools.append(load_knowledgebase_tool.load_knowledgebase_tool)
140
+ load_knowledgebase_tool = LoadKnowledgebaseTool(
141
+ knowledgebase=self.knowledgebase
142
+ )
143
+ self.tools.append(load_knowledgebase_tool)
140
144
 
141
145
  if self.long_term_memory is not None:
142
146
  from google.adk.tools import load_memory
@@ -14,64 +14,56 @@
14
14
 
15
15
  import os
16
16
 
17
- from typing_extensions import override
18
-
19
- from veadk.auth.veauth.base_veauth import BaseVeAuth
17
+ from veadk.auth.veauth.utils import get_credential_from_vefaas_iam
20
18
  from veadk.utils.logger import get_logger
21
19
  from veadk.utils.volcengine_sign import ve_request
22
20
 
23
21
  logger = get_logger(__name__)
24
22
 
25
23
 
26
- class ARKVeAuth(BaseVeAuth):
27
- def __init__(
28
- self,
29
- access_key: str = os.getenv("VOLCENGINE_ACCESS_KEY", ""),
30
- secret_key: str = os.getenv("VOLCENGINE_SECRET_KEY", ""),
31
- ) -> None:
32
- super().__init__(access_key, secret_key)
24
+ def get_ark_token(region: str = "cn-beijing") -> str:
25
+ logger.info("Fetching ARK token...")
33
26
 
34
- self._token: str = ""
27
+ access_key = os.getenv("VOLCENGINE_ACCESS_KEY")
28
+ secret_key = os.getenv("VOLCENGINE_SECRET_KEY")
29
+ session_token = ""
35
30
 
36
- @override
37
- def _fetch_token(self) -> None:
38
- logger.info("Fetching ARK token...")
39
- # list api keys
40
- first_api_key_id = ""
41
- res = ve_request(
42
- request_body={"ProjectName": "default", "Filter": {}},
43
- action="ListApiKeys",
44
- ak=self.access_key,
45
- sk=self.secret_key,
46
- service="ark",
47
- version="2024-01-01",
48
- region="cn-beijing",
49
- host="open.volcengineapi.com",
50
- )
51
- try:
52
- first_api_key_id = res["Result"]["Items"][0]["Id"]
53
- except KeyError:
54
- raise ValueError(f"Failed to get ARK api key list: {res}")
31
+ if not (access_key and secret_key):
32
+ # try to get from vefaas iam
33
+ cred = get_credential_from_vefaas_iam()
34
+ access_key = cred.access_key_id
35
+ secret_key = cred.secret_access_key
36
+ session_token = cred.session_token
55
37
 
56
- # get raw api key
57
- res = ve_request(
58
- request_body={"Id": first_api_key_id},
59
- action="GetRawApiKey",
60
- ak=self.access_key,
61
- sk=self.secret_key,
62
- service="ark",
63
- version="2024-01-01",
64
- region="cn-beijing",
65
- host="open.volcengineapi.com",
66
- )
67
- try:
68
- self._token = res["Result"]["ApiKey"]
69
- except KeyError:
70
- raise ValueError(f"Failed to get ARK api key: {res}")
38
+ res = ve_request(
39
+ request_body={"ProjectName": "default", "Filter": {}},
40
+ header={"X-Security-Token": session_token},
41
+ action="ListApiKeys",
42
+ ak=access_key,
43
+ sk=secret_key,
44
+ service="ark",
45
+ version="2024-01-01",
46
+ region=region,
47
+ host="open.volcengineapi.com",
48
+ )
49
+ try:
50
+ first_api_key_id = res["Result"]["Items"][0]["Id"]
51
+ except KeyError:
52
+ raise ValueError(f"Failed to get ARK api key list: {res}")
71
53
 
72
- @property
73
- def token(self) -> str:
74
- if self._token:
75
- return self._token
76
- self._fetch_token()
77
- return self._token
54
+ # get raw api key
55
+ res = ve_request(
56
+ request_body={"Id": first_api_key_id},
57
+ header={"X-Security-Token": session_token},
58
+ action="GetRawApiKey",
59
+ ak=access_key,
60
+ sk=secret_key,
61
+ service="ark",
62
+ version="2024-01-01",
63
+ region=region,
64
+ host="open.volcengineapi.com",
65
+ )
66
+ try:
67
+ return res["Result"]["ApiKey"]
68
+ except KeyError:
69
+ raise ValueError(f"Failed to get ARK api key: {res}")
@@ -0,0 +1,57 @@
1
+ # Copyright (c) 2025 Beijing Volcano Engine Technology Co., Ltd. and/or its affiliates.
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+
15
+ import json
16
+ from pathlib import Path
17
+
18
+ from pydantic import BaseModel
19
+
20
+ from veadk.consts import VEFAAS_IAM_CRIDENTIAL_PATH
21
+ from veadk.utils.logger import get_logger
22
+
23
+ logger = get_logger(__name__)
24
+
25
+
26
+ class VeIAMCredential(BaseModel):
27
+ access_key_id: str
28
+ secret_access_key: str
29
+ session_token: str
30
+
31
+
32
+ def get_credential_from_vefaas_iam() -> VeIAMCredential:
33
+ """Get credential from VeFaaS IAM file"""
34
+ logger.info(
35
+ f"Get Volcegnine access key or secret key from environment variables failed, try to get from VeFaaS IAM file (path={VEFAAS_IAM_CRIDENTIAL_PATH})."
36
+ )
37
+
38
+ path = Path(VEFAAS_IAM_CRIDENTIAL_PATH)
39
+
40
+ if not path.exists():
41
+ logger.error(
42
+ f"Get Volcegnine access key or secret key from environment variables failed, and VeFaaS IAM file (path={VEFAAS_IAM_CRIDENTIAL_PATH}) not exists. Please check your configuration."
43
+ )
44
+ raise FileNotFoundError(
45
+ f"Get Volcegnine access key or secret key from environment variables failed, and VeFaaS IAM file (path={VEFAAS_IAM_CRIDENTIAL_PATH}) not exists. Please check your configuration."
46
+ )
47
+
48
+ with open(VEFAAS_IAM_CRIDENTIAL_PATH, "r") as f:
49
+ cred_dict = json.load(f)
50
+ access_key = cred_dict["access_key_id"]
51
+ secret_key = cred_dict["secret_access_key"]
52
+ session_token = cred_dict["session_token"]
53
+ return VeIAMCredential(
54
+ access_key_id=access_key,
55
+ secret_access_key=secret_key,
56
+ session_token=session_token,
57
+ )
@@ -17,7 +17,7 @@ from functools import cached_property
17
17
 
18
18
  from pydantic_settings import BaseSettings, SettingsConfigDict
19
19
 
20
- from veadk.auth.veauth.ark_veauth import ARKVeAuth
20
+ from veadk.auth.veauth.ark_veauth import get_ark_token
21
21
  from veadk.consts import (
22
22
  DEFAULT_MODEL_AGENT_API_BASE,
23
23
  DEFAULT_MODEL_AGENT_NAME,
@@ -39,7 +39,7 @@ class ModelConfig(BaseSettings):
39
39
 
40
40
  @cached_property
41
41
  def api_key(self) -> str:
42
- return os.getenv("MODEL_AGENT_API_KEY") or ARKVeAuth().token
42
+ return os.getenv("MODEL_AGENT_API_KEY") or get_ark_token()
43
43
 
44
44
 
45
45
  class EmbeddingModelConfig(BaseSettings):
@@ -56,7 +56,7 @@ class EmbeddingModelConfig(BaseSettings):
56
56
 
57
57
  @cached_property
58
58
  def api_key(self) -> str:
59
- return os.getenv("MODEL_EMBEDDING_API_KEY") or ARKVeAuth().token
59
+ return os.getenv("MODEL_EMBEDDING_API_KEY") or get_ark_token()
60
60
 
61
61
 
62
62
  class NormalEmbeddingModelConfig(BaseSettings):
veadk/consts.py CHANGED
@@ -63,6 +63,15 @@ DEFAULT_TOS_BUCKET_NAME = "ark-tutorial"
63
63
  DEFAULT_COZELOOP_SPACE_NAME = "VeADK Space"
64
64
 
65
65
  DEFAULT_TEXT_TO_IMAGE_MODEL_NAME = "doubao-seedream-3-0-t2i-250415"
66
+ DEFAULT_TEXT_TO_IMAGE_MODEL_API_BASE = "https://ark.cn-beijing.volces.com/api/v3/"
67
+
66
68
  DEFAULT_IMAGE_EDIT_MODEL_NAME = "doubao-seededit-3-0-i2i-250628"
69
+ DEFAULT_IMAGE_EDIT_MODEL_API_BASE = "https://ark.cn-beijing.volces.com/api/v3/"
70
+
67
71
  DEFAULT_VIDEO_MODEL_NAME = "doubao-seedance-1-0-pro-250528"
72
+ DEFAULT_VIDEO_MODEL_API_BASE = "https://ark.cn-beijing.volces.com/api/v3/"
73
+
68
74
  DEFAULT_IMAGE_GENERATE_MODEL_NAME = "doubao-seedream-4-0-250828"
75
+ DEFAULT_IMAGE_GENERATE_MODEL_API_BASE = "https://ark.cn-beijing.volces.com/api/v3/"
76
+
77
+ VEFAAS_IAM_CRIDENTIAL_PATH = "/var/run/secrets/iam/credential"
@@ -12,10 +12,11 @@
12
12
  # See the License for the specific language governing permissions and
13
13
  # limitations under the License.
14
14
 
15
- from typing import Any, Callable, Literal
15
+ from __future__ import annotations
16
+
17
+ from typing import Any, Callable, Literal, Union
16
18
 
17
19
  from pydantic import BaseModel, Field
18
- from typing_extensions import Union
19
20
 
20
21
  from veadk.knowledgebase.backends.base_backend import BaseKnowledgebaseBackend
21
22
  from veadk.knowledgebase.entry import KnowledgebaseEntry
@@ -54,11 +55,11 @@ def _get_backend_cls(backend: str) -> type[BaseKnowledgebaseBackend]:
54
55
  raise ValueError(f"Unsupported knowledgebase backend: {backend}")
55
56
 
56
57
 
57
- def build_knowledgebase_index(app_name: str):
58
- return f"veadk_kb_{app_name}"
58
+ class KnowledgeBase(BaseModel):
59
+ name: str = "user_knowledgebase"
59
60
 
61
+ description: str = "This knowledgebase stores some user-related information."
60
62
 
61
- class KnowledgeBase(BaseModel):
62
63
  backend: Union[
63
64
  Literal["local", "opensearch", "viking", "redis"], BaseKnowledgebaseBackend
64
65
  ] = "local"
@@ -73,9 +74,7 @@ class KnowledgeBase(BaseModel):
73
74
  """Configuration for the backend"""
74
75
 
75
76
  top_k: int = 10
76
- """Number of top similar documents to retrieve during search.
77
-
78
- Default is 10."""
77
+ """Number of top similar documents to retrieve during search"""
79
78
 
80
79
  app_name: str = ""
81
80
 
@@ -85,38 +84,27 @@ class KnowledgeBase(BaseModel):
85
84
  def model_post_init(self, __context: Any) -> None:
86
85
  if isinstance(self.backend, BaseKnowledgebaseBackend):
87
86
  self._backend = self.backend
87
+ self.index = self._backend.index
88
88
  logger.info(
89
89
  f"Initialized knowledgebase with provided backend instance {self._backend.__class__.__name__}"
90
90
  )
91
91
  return
92
92
 
93
- # must provide at least one of them
94
- if not self.app_name and not self.index:
95
- raise ValueError(
96
- "Either `app_name` or `index` must be provided one of them."
97
- )
98
-
99
- # priority use index
100
- if self.app_name and self.index:
101
- logger.warning(
102
- "`app_name` and `index` are both provided, using `index` as the knowledgebase index name."
103
- )
93
+ # Once user define backend config, use it directly
94
+ if self.backend_config:
95
+ self._backend = _get_backend_cls(self.backend)(**self.backend_config)
96
+ return
104
97
 
105
- # generate index name if `index` not provided but `app_name` is provided
106
- if self.app_name and not self.index:
107
- self.index = build_knowledgebase_index(self.app_name)
108
- logger.info(
109
- f"Knowledgebase index is set to {self.index} (generated by the app_name: {self.app_name})."
110
- )
98
+ self.index = self.index or self.app_name
99
+ if not self.index:
100
+ raise ValueError("Either `index` or `app_name` must be provided.")
111
101
 
112
102
  logger.info(
113
- f"Initializing knowledgebase: backend={self.backend} top_k={self.top_k}"
114
- )
115
- self._backend = _get_backend_cls(self.backend)(
116
- index=self.index, **self.backend_config if self.backend_config else {}
103
+ f"Initializing knowledgebase: backend={self.backend} index={self.index} top_k={self.top_k}"
117
104
  )
105
+ self._backend = _get_backend_cls(self.backend)(index=self.index)
118
106
  logger.info(
119
- f"Initialized knowledgebase with backend {self._backend.__class__.__name__}"
107
+ f"Initialized knowledgebase with backend {self.backend.__class__.__name__}"
120
108
  )
121
109
 
122
110
  def add_from_directory(self, directory: str, **kwargs) -> bool:
@@ -133,8 +121,7 @@ class KnowledgeBase(BaseModel):
133
121
 
134
122
  def search(self, query: str, top_k: int = 0, **kwargs) -> list[KnowledgebaseEntry]:
135
123
  """Search knowledge from knowledgebase"""
136
- if top_k == 0:
137
- top_k = self.top_k
124
+ top_k = top_k if top_k != 0 else self.top_k
138
125
 
139
126
  _entries = self._backend.search(query=query, top_k=top_k, **kwargs)
140
127
 
@@ -72,10 +72,6 @@ def _get_backend_cls(backend: str) -> type[BaseLongTermMemoryBackend]:
72
72
  raise ValueError(f"Unsupported long term memory backend: {backend}")
73
73
 
74
74
 
75
- def build_long_term_memory_index(app_name: str, user_id: str):
76
- return f"{app_name}_{user_id}"
77
-
78
-
79
75
  class LongTermMemory(BaseMemoryService, BaseModel):
80
76
  backend: Union[
81
77
  Literal["local", "opensearch", "redis", "viking", "viking_mem", "mem0"],
@@ -89,54 +85,48 @@ class LongTermMemory(BaseMemoryService, BaseModel):
89
85
  top_k: int = 5
90
86
  """Number of top similar documents to retrieve during search."""
91
87
 
88
+ index: str = ""
89
+
92
90
  app_name: str = ""
93
91
 
94
92
  user_id: str = ""
93
+ """Deprecated attribute"""
95
94
 
96
95
  def model_post_init(self, __context: Any) -> None:
97
- if self.backend == "viking_mem":
98
- logger.warning(
99
- "The `viking_mem` backend is deprecated, please use `viking` instead."
100
- )
101
- self.backend = "viking"
102
-
103
- self._backend = None
104
-
105
96
  # Once user define a backend instance, use it directly
106
97
  if isinstance(self.backend, BaseLongTermMemoryBackend):
107
98
  self._backend = self.backend
99
+ self.index = self._backend.index
108
100
  logger.info(
109
- f"Initialized long term memory with provided backend instance {self._backend.__class__.__name__}"
101
+ f"Initialized long term memory with provided backend instance {self._backend.__class__.__name__}, index={self.index}"
110
102
  )
111
103
  return
112
104
 
105
+ # Once user define backend config, use it directly
113
106
  if self.backend_config:
114
- logger.warning(
115
- f"Initialized long term memory backend {self.backend} with config. We will ignore `app_name` and `user_id` if provided."
116
- )
117
107
  self._backend = _get_backend_cls(self.backend)(**self.backend_config)
118
- _index = self.backend_config.get("index", None)
119
- if _index:
120
- self._index = _index
121
- logger.info(f"Long term memory index set to {self._index}.")
122
- else:
123
- logger.warning(
124
- "Cannot find index via backend_config, please set `index` parameter."
125
- )
126
108
  return
127
109
 
128
- if self.app_name and self.user_id:
129
- self._index = build_long_term_memory_index(
130
- app_name=self.app_name, user_id=self.user_id
131
- )
132
- logger.info(f"Long term memory index set to {self._index}.")
133
- self._backend = _get_backend_cls(self.backend)(
134
- index=self._index, **self.backend_config if self.backend_config else {}
110
+ # Check index
111
+ self.index = self.index or self.app_name
112
+ if not self.index:
113
+ logger.warning(
114
+ "Attribute `index` or `app_name` not provided, use `default_app` instead."
135
115
  )
136
- else:
116
+ self.index = "default_app"
117
+
118
+ # Forward compliance
119
+ if self.backend == "viking_mem":
137
120
  logger.warning(
138
- "Neither `backend_instance`, `backend_config`, nor (`app_name`/`user_id`) is provided, the long term memory storage will initialize when adding a session."
121
+ "The `viking_mem` backend is deprecated, change to `viking` instead."
139
122
  )
123
+ self.backend = "viking"
124
+
125
+ self._backend = _get_backend_cls(self.backend)(index=self.index)
126
+
127
+ logger.info(
128
+ f"Initialized long term memory with provided backend instance {self._backend.__class__.__name__}, index={self.index}"
129
+ )
140
130
 
141
131
  def _filter_and_convert_events(self, events: list[Event]) -> list[str]:
142
132
  final_events = []
@@ -164,75 +154,32 @@ class LongTermMemory(BaseMemoryService, BaseModel):
164
154
  self,
165
155
  session: Session,
166
156
  ):
167
- app_name = session.app_name
168
157
  user_id = session.user_id
169
-
170
- if not self._backend and isinstance(self.backend, str):
171
- self._index = build_long_term_memory_index(app_name, user_id)
172
- self._backend = _get_backend_cls(self.backend)(
173
- index=self._index, **self.backend_config if self.backend_config else {}
174
- )
175
- logger.info(
176
- f"Initialize long term memory backend now, index is {self._index}"
177
- )
178
-
179
- if not self._index and self._index != build_long_term_memory_index(
180
- app_name, user_id
181
- ):
182
- logger.warning(
183
- f"The `app_name` or `user_id` is different from the initialized one, skip add session to memory. Initialized index: {self._index}, current built index: {build_long_term_memory_index(app_name, user_id)}"
184
- )
185
- return
186
158
  event_strings = self._filter_and_convert_events(session.events)
187
159
 
188
160
  logger.info(
189
- f"Adding {len(event_strings)} events to long term memory: index={self._index}"
161
+ f"Adding {len(event_strings)} events to long term memory: index={self.index}"
162
+ )
163
+ self._backend.save_memory(user_id=user_id, event_strings=event_strings)
164
+ logger.info(
165
+ f"Added {len(event_strings)} events to long term memory: index={self.index}, user_id={user_id}"
190
166
  )
191
-
192
- if self._backend:
193
- self._backend.save_memory(event_strings=event_strings, user_id=user_id)
194
-
195
- logger.info(
196
- f"Added {len(event_strings)} events to long term memory: index={self._index}"
197
- )
198
- else:
199
- logger.error(
200
- "Long term memory backend initialize failed, cannot add session to memory."
201
- )
202
167
 
203
168
  @override
204
- async def search_memory(self, *, app_name: str, user_id: str, query: str):
205
- # prevent model invoke `load_memory` before add session to this memory
206
- if not self._backend and isinstance(self.backend, str):
207
- self._index = build_long_term_memory_index(app_name, user_id)
208
- self._backend = _get_backend_cls(self.backend)(
209
- index=self._index, **self.backend_config if self.backend_config else {}
210
- )
211
- logger.info(
212
- f"Initialize long term memory backend now, index is {self._index}"
213
- )
169
+ async def search_memory(
170
+ self, *, app_name: str, user_id: str, query: str
171
+ ) -> SearchMemoryResponse:
172
+ logger.info(f"Search memory with query={query}")
214
173
 
215
- if not self._index and self._index != build_long_term_memory_index(
216
- app_name, user_id
217
- ):
218
- logger.warning(
219
- f"The `app_name` or `user_id` is different from the initialized one. Initialized index: {self._index}, current built index: {build_long_term_memory_index(app_name, user_id)}. Search memory return empty list."
174
+ memory_chunks = []
175
+ try:
176
+ memory_chunks = self._backend.search_memory(
177
+ query=query, top_k=self.top_k, user_id=user_id
220
178
  )
221
- return SearchMemoryResponse(memories=[])
222
-
223
- if not self._backend:
179
+ except Exception as e:
224
180
  logger.error(
225
- "Long term memory backend is not initialized, cannot search memory."
181
+ f"Exception orrcus during memory search: {e}. Return empty memory chunks"
226
182
  )
227
- return SearchMemoryResponse(memories=[])
228
-
229
- logger.info(
230
- f"Searching long term memory: query={query} index={self._index} top_k={self.top_k}"
231
- )
232
-
233
- memory_chunks = self._backend.search_memory(
234
- query=query, top_k=self.top_k, user_id=user_id
235
- )
236
183
 
237
184
  memory_events = []
238
185
  for memory in memory_chunks:
@@ -260,6 +207,6 @@ class LongTermMemory(BaseMemoryService, BaseModel):
260
207
  )
261
208
 
262
209
  logger.info(
263
- f"Return {len(memory_events)} memory events for query: {query} index={self._index}"
210
+ f"Return {len(memory_events)} memory events for query: {query} index={self.index} user_id={user_id}"
264
211
  )
265
212
  return SearchMemoryResponse(memories=memory_events)
@@ -25,9 +25,11 @@ class BaseLongTermMemoryBackend(ABC, BaseModel):
25
25
  """Check the index name is valid or not"""
26
26
 
27
27
  @abstractmethod
28
- def save_memory(self, event_strings: list[str], **kwargs) -> bool:
28
+ def save_memory(self, user_id: str, event_strings: list[str], **kwargs) -> bool:
29
29
  """Save memory to long term memory backend"""
30
30
 
31
31
  @abstractmethod
32
- def search_memory(self, query: str, top_k: int, **kwargs) -> list[str]:
32
+ def search_memory(
33
+ self, user_id: str, query: str, top_k: int, **kwargs
34
+ ) -> list[str]:
33
35
  """Retrieve memory from long term memory backend"""
@@ -29,10 +29,6 @@ class InMemoryLTMBackend(BaseLongTermMemoryBackend):
29
29
  embedding_config: EmbeddingModelConfig = Field(default_factory=EmbeddingModelConfig)
30
30
  """Embedding model configs"""
31
31
 
32
- def precheck_index_naming(self):
33
- # no checking
34
- pass
35
-
36
32
  def model_post_init(self, __context: Any) -> None:
37
33
  self._embed_model = OpenAILikeEmbedding(
38
34
  model_name=self.embedding_config.name,
@@ -41,8 +37,12 @@ class InMemoryLTMBackend(BaseLongTermMemoryBackend):
41
37
  )
42
38
  self._vector_index = VectorStoreIndex([], embed_model=self._embed_model)
43
39
 
40
+ def precheck_index_naming(self):
41
+ # no checking
42
+ pass
43
+
44
44
  @override
45
- def save_memory(self, event_strings: list[str], **kwargs) -> bool:
45
+ def save_memory(self, user_id: str, event_strings: list[str], **kwargs) -> bool:
46
46
  for event_string in event_strings:
47
47
  document = Document(text=event_string)
48
48
  nodes = self._split_documents([document])
@@ -50,7 +50,9 @@ class InMemoryLTMBackend(BaseLongTermMemoryBackend):
50
50
  return True
51
51
 
52
52
  @override
53
- def search_memory(self, query: str, top_k: int, **kwargs) -> list[str]:
53
+ def search_memory(
54
+ self, user_id: str, query: str, top_k: int, **kwargs
55
+ ) -> list[str]:
54
56
  _retriever = self._vector_index.as_retriever(similarity_top_k=top_k)
55
57
  retrieved_nodes = _retriever.retrieve(query)
56
58
  return [node.text for node in retrieved_nodes]
@@ -13,12 +13,11 @@
13
13
  # limitations under the License.
14
14
 
15
15
  from typing import Any
16
- from typing_extensions import override
16
+
17
17
  from pydantic import Field
18
+ from typing_extensions import override
18
19
 
19
20
  from veadk.configs.database_configs import Mem0Config
20
-
21
-
22
21
  from veadk.memory.long_term_memory_backends.base_backend import (
23
22
  BaseLongTermMemoryBackend,
24
23
  )
@@ -66,7 +65,9 @@ class Mem0LTMBackend(BaseLongTermMemoryBackend):
66
65
  pass
67
66
 
68
67
  @override
69
- def save_memory(self, event_strings: list[str], **kwargs) -> bool:
68
+ def save_memory(
69
+ self, event_strings: list[str], user_id: str = "default_user", **kwargs
70
+ ) -> bool:
70
71
  """Save memory to Mem0
71
72
 
72
73
  Args:
@@ -76,8 +77,6 @@ class Mem0LTMBackend(BaseLongTermMemoryBackend):
76
77
  Returns:
77
78
  bool: True if saved successfully, False otherwise
78
79
  """
79
- user_id = kwargs.get("user_id", "default_user")
80
-
81
80
  try:
82
81
  logger.info(
83
82
  f"Saving {len(event_strings)} events to Mem0 for user: {user_id}"
@@ -100,7 +99,9 @@ class Mem0LTMBackend(BaseLongTermMemoryBackend):
100
99
  return False
101
100
 
102
101
  @override
103
- def search_memory(self, query: str, top_k: int, **kwargs) -> list[str]:
102
+ def search_memory(
103
+ self, query: str, top_k: int, user_id: str = "default_user", **kwargs
104
+ ) -> list[str]:
104
105
  """Search memory from Mem0
105
106
 
106
107
  Args:
@@ -111,7 +112,6 @@ class Mem0LTMBackend(BaseLongTermMemoryBackend):
111
112
  Returns:
112
113
  list[str]: List of memory strings
113
114
  """
114
- user_id = kwargs.get("user_id", "default_user")
115
115
 
116
116
  try:
117
117
  logger.info(