basic-memory 0.7.0__py3-none-any.whl → 0.17.4__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 basic-memory might be problematic. Click here for more details.

Files changed (195) hide show
  1. basic_memory/__init__.py +5 -1
  2. basic_memory/alembic/alembic.ini +119 -0
  3. basic_memory/alembic/env.py +130 -20
  4. basic_memory/alembic/migrations.py +4 -9
  5. basic_memory/alembic/versions/314f1ea54dc4_add_postgres_full_text_search_support_.py +131 -0
  6. basic_memory/alembic/versions/502b60eaa905_remove_required_from_entity_permalink.py +51 -0
  7. basic_memory/alembic/versions/5fe1ab1ccebe_add_projects_table.py +120 -0
  8. basic_memory/alembic/versions/647e7a75e2cd_project_constraint_fix.py +112 -0
  9. basic_memory/alembic/versions/6830751f5fb6_merge_multiple_heads.py +24 -0
  10. basic_memory/alembic/versions/9d9c1cb7d8f5_add_mtime_and_size_columns_to_entity_.py +49 -0
  11. basic_memory/alembic/versions/a1b2c3d4e5f6_fix_project_foreign_keys.py +49 -0
  12. basic_memory/alembic/versions/a2b3c4d5e6f7_add_search_index_entity_cascade.py +56 -0
  13. basic_memory/alembic/versions/b3c3938bacdb_relation_to_name_unique_index.py +44 -0
  14. basic_memory/alembic/versions/cc7172b46608_update_search_index_schema.py +113 -0
  15. basic_memory/alembic/versions/e7e1f4367280_add_scan_watermark_tracking_to_project.py +37 -0
  16. basic_memory/alembic/versions/f8a9b2c3d4e5_add_pg_trgm_for_fuzzy_link_resolution.py +239 -0
  17. basic_memory/alembic/versions/g9a0b3c4d5e6_add_external_id_to_project_and_entity.py +173 -0
  18. basic_memory/api/app.py +87 -20
  19. basic_memory/api/container.py +133 -0
  20. basic_memory/api/routers/__init__.py +4 -1
  21. basic_memory/api/routers/directory_router.py +84 -0
  22. basic_memory/api/routers/importer_router.py +152 -0
  23. basic_memory/api/routers/knowledge_router.py +180 -23
  24. basic_memory/api/routers/management_router.py +80 -0
  25. basic_memory/api/routers/memory_router.py +9 -64
  26. basic_memory/api/routers/project_router.py +460 -0
  27. basic_memory/api/routers/prompt_router.py +260 -0
  28. basic_memory/api/routers/resource_router.py +136 -11
  29. basic_memory/api/routers/search_router.py +5 -5
  30. basic_memory/api/routers/utils.py +169 -0
  31. basic_memory/api/template_loader.py +292 -0
  32. basic_memory/api/v2/__init__.py +35 -0
  33. basic_memory/api/v2/routers/__init__.py +21 -0
  34. basic_memory/api/v2/routers/directory_router.py +93 -0
  35. basic_memory/api/v2/routers/importer_router.py +181 -0
  36. basic_memory/api/v2/routers/knowledge_router.py +427 -0
  37. basic_memory/api/v2/routers/memory_router.py +130 -0
  38. basic_memory/api/v2/routers/project_router.py +359 -0
  39. basic_memory/api/v2/routers/prompt_router.py +269 -0
  40. basic_memory/api/v2/routers/resource_router.py +286 -0
  41. basic_memory/api/v2/routers/search_router.py +73 -0
  42. basic_memory/cli/app.py +80 -10
  43. basic_memory/cli/auth.py +300 -0
  44. basic_memory/cli/commands/__init__.py +15 -2
  45. basic_memory/cli/commands/cloud/__init__.py +6 -0
  46. basic_memory/cli/commands/cloud/api_client.py +127 -0
  47. basic_memory/cli/commands/cloud/bisync_commands.py +110 -0
  48. basic_memory/cli/commands/cloud/cloud_utils.py +108 -0
  49. basic_memory/cli/commands/cloud/core_commands.py +195 -0
  50. basic_memory/cli/commands/cloud/rclone_commands.py +397 -0
  51. basic_memory/cli/commands/cloud/rclone_config.py +110 -0
  52. basic_memory/cli/commands/cloud/rclone_installer.py +263 -0
  53. basic_memory/cli/commands/cloud/upload.py +240 -0
  54. basic_memory/cli/commands/cloud/upload_command.py +124 -0
  55. basic_memory/cli/commands/command_utils.py +99 -0
  56. basic_memory/cli/commands/db.py +87 -12
  57. basic_memory/cli/commands/format.py +198 -0
  58. basic_memory/cli/commands/import_chatgpt.py +47 -223
  59. basic_memory/cli/commands/import_claude_conversations.py +48 -171
  60. basic_memory/cli/commands/import_claude_projects.py +53 -160
  61. basic_memory/cli/commands/import_memory_json.py +55 -111
  62. basic_memory/cli/commands/mcp.py +67 -11
  63. basic_memory/cli/commands/project.py +889 -0
  64. basic_memory/cli/commands/status.py +52 -34
  65. basic_memory/cli/commands/telemetry.py +81 -0
  66. basic_memory/cli/commands/tool.py +341 -0
  67. basic_memory/cli/container.py +84 -0
  68. basic_memory/cli/main.py +14 -6
  69. basic_memory/config.py +580 -26
  70. basic_memory/db.py +285 -28
  71. basic_memory/deps/__init__.py +293 -0
  72. basic_memory/deps/config.py +26 -0
  73. basic_memory/deps/db.py +56 -0
  74. basic_memory/deps/importers.py +200 -0
  75. basic_memory/deps/projects.py +238 -0
  76. basic_memory/deps/repositories.py +179 -0
  77. basic_memory/deps/services.py +480 -0
  78. basic_memory/deps.py +16 -185
  79. basic_memory/file_utils.py +318 -54
  80. basic_memory/ignore_utils.py +297 -0
  81. basic_memory/importers/__init__.py +27 -0
  82. basic_memory/importers/base.py +100 -0
  83. basic_memory/importers/chatgpt_importer.py +245 -0
  84. basic_memory/importers/claude_conversations_importer.py +192 -0
  85. basic_memory/importers/claude_projects_importer.py +184 -0
  86. basic_memory/importers/memory_json_importer.py +128 -0
  87. basic_memory/importers/utils.py +61 -0
  88. basic_memory/markdown/entity_parser.py +182 -23
  89. basic_memory/markdown/markdown_processor.py +70 -7
  90. basic_memory/markdown/plugins.py +43 -23
  91. basic_memory/markdown/schemas.py +1 -1
  92. basic_memory/markdown/utils.py +38 -14
  93. basic_memory/mcp/async_client.py +135 -4
  94. basic_memory/mcp/clients/__init__.py +28 -0
  95. basic_memory/mcp/clients/directory.py +70 -0
  96. basic_memory/mcp/clients/knowledge.py +176 -0
  97. basic_memory/mcp/clients/memory.py +120 -0
  98. basic_memory/mcp/clients/project.py +89 -0
  99. basic_memory/mcp/clients/resource.py +71 -0
  100. basic_memory/mcp/clients/search.py +65 -0
  101. basic_memory/mcp/container.py +110 -0
  102. basic_memory/mcp/project_context.py +155 -0
  103. basic_memory/mcp/prompts/__init__.py +19 -0
  104. basic_memory/mcp/prompts/ai_assistant_guide.py +70 -0
  105. basic_memory/mcp/prompts/continue_conversation.py +62 -0
  106. basic_memory/mcp/prompts/recent_activity.py +188 -0
  107. basic_memory/mcp/prompts/search.py +57 -0
  108. basic_memory/mcp/prompts/utils.py +162 -0
  109. basic_memory/mcp/resources/ai_assistant_guide.md +283 -0
  110. basic_memory/mcp/resources/project_info.py +71 -0
  111. basic_memory/mcp/server.py +61 -9
  112. basic_memory/mcp/tools/__init__.py +33 -21
  113. basic_memory/mcp/tools/build_context.py +120 -0
  114. basic_memory/mcp/tools/canvas.py +152 -0
  115. basic_memory/mcp/tools/chatgpt_tools.py +190 -0
  116. basic_memory/mcp/tools/delete_note.py +249 -0
  117. basic_memory/mcp/tools/edit_note.py +325 -0
  118. basic_memory/mcp/tools/list_directory.py +157 -0
  119. basic_memory/mcp/tools/move_note.py +549 -0
  120. basic_memory/mcp/tools/project_management.py +204 -0
  121. basic_memory/mcp/tools/read_content.py +281 -0
  122. basic_memory/mcp/tools/read_note.py +265 -0
  123. basic_memory/mcp/tools/recent_activity.py +528 -0
  124. basic_memory/mcp/tools/search.py +377 -24
  125. basic_memory/mcp/tools/utils.py +402 -16
  126. basic_memory/mcp/tools/view_note.py +78 -0
  127. basic_memory/mcp/tools/write_note.py +230 -0
  128. basic_memory/models/__init__.py +3 -2
  129. basic_memory/models/knowledge.py +82 -17
  130. basic_memory/models/project.py +93 -0
  131. basic_memory/models/search.py +68 -8
  132. basic_memory/project_resolver.py +222 -0
  133. basic_memory/repository/__init__.py +2 -0
  134. basic_memory/repository/entity_repository.py +437 -8
  135. basic_memory/repository/observation_repository.py +36 -3
  136. basic_memory/repository/postgres_search_repository.py +451 -0
  137. basic_memory/repository/project_info_repository.py +10 -0
  138. basic_memory/repository/project_repository.py +140 -0
  139. basic_memory/repository/relation_repository.py +79 -4
  140. basic_memory/repository/repository.py +148 -29
  141. basic_memory/repository/search_index_row.py +95 -0
  142. basic_memory/repository/search_repository.py +79 -268
  143. basic_memory/repository/search_repository_base.py +241 -0
  144. basic_memory/repository/sqlite_search_repository.py +437 -0
  145. basic_memory/runtime.py +61 -0
  146. basic_memory/schemas/__init__.py +22 -9
  147. basic_memory/schemas/base.py +131 -12
  148. basic_memory/schemas/cloud.py +50 -0
  149. basic_memory/schemas/directory.py +31 -0
  150. basic_memory/schemas/importer.py +35 -0
  151. basic_memory/schemas/memory.py +194 -25
  152. basic_memory/schemas/project_info.py +213 -0
  153. basic_memory/schemas/prompt.py +90 -0
  154. basic_memory/schemas/request.py +56 -2
  155. basic_memory/schemas/response.py +85 -28
  156. basic_memory/schemas/search.py +36 -35
  157. basic_memory/schemas/sync_report.py +72 -0
  158. basic_memory/schemas/v2/__init__.py +27 -0
  159. basic_memory/schemas/v2/entity.py +133 -0
  160. basic_memory/schemas/v2/resource.py +47 -0
  161. basic_memory/services/__init__.py +2 -1
  162. basic_memory/services/context_service.py +451 -138
  163. basic_memory/services/directory_service.py +310 -0
  164. basic_memory/services/entity_service.py +636 -71
  165. basic_memory/services/exceptions.py +21 -0
  166. basic_memory/services/file_service.py +402 -33
  167. basic_memory/services/initialization.py +216 -0
  168. basic_memory/services/link_resolver.py +50 -56
  169. basic_memory/services/project_service.py +888 -0
  170. basic_memory/services/search_service.py +232 -37
  171. basic_memory/sync/__init__.py +4 -2
  172. basic_memory/sync/background_sync.py +26 -0
  173. basic_memory/sync/coordinator.py +160 -0
  174. basic_memory/sync/sync_service.py +1200 -109
  175. basic_memory/sync/watch_service.py +432 -135
  176. basic_memory/telemetry.py +249 -0
  177. basic_memory/templates/prompts/continue_conversation.hbs +110 -0
  178. basic_memory/templates/prompts/search.hbs +101 -0
  179. basic_memory/utils.py +407 -54
  180. basic_memory-0.17.4.dist-info/METADATA +617 -0
  181. basic_memory-0.17.4.dist-info/RECORD +193 -0
  182. {basic_memory-0.7.0.dist-info → basic_memory-0.17.4.dist-info}/WHEEL +1 -1
  183. {basic_memory-0.7.0.dist-info → basic_memory-0.17.4.dist-info}/entry_points.txt +1 -0
  184. basic_memory/alembic/README +0 -1
  185. basic_memory/cli/commands/sync.py +0 -206
  186. basic_memory/cli/commands/tools.py +0 -157
  187. basic_memory/mcp/tools/knowledge.py +0 -68
  188. basic_memory/mcp/tools/memory.py +0 -170
  189. basic_memory/mcp/tools/notes.py +0 -202
  190. basic_memory/schemas/discovery.py +0 -28
  191. basic_memory/sync/file_change_scanner.py +0 -158
  192. basic_memory/sync/utils.py +0 -31
  193. basic_memory-0.7.0.dist-info/METADATA +0 -378
  194. basic_memory-0.7.0.dist-info/RECORD +0 -82
  195. {basic_memory-0.7.0.dist-info → basic_memory-0.17.4.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,480 @@
1
+ """Service dependency injection for basic-memory.
2
+
3
+ This module provides service-layer dependencies:
4
+ - EntityParser, MarkdownProcessor
5
+ - FileService, EntityService
6
+ - SearchService, LinkResolver, ContextService
7
+ - SyncService, ProjectService, DirectoryService
8
+ """
9
+
10
+ from typing import Annotated
11
+
12
+ from fastapi import Depends
13
+ from loguru import logger
14
+
15
+ from basic_memory.deps.config import AppConfigDep
16
+ from basic_memory.deps.projects import (
17
+ ProjectConfigDep,
18
+ ProjectConfigV2Dep,
19
+ ProjectConfigV2ExternalDep,
20
+ ProjectRepositoryDep,
21
+ )
22
+ from basic_memory.deps.repositories import (
23
+ EntityRepositoryDep,
24
+ EntityRepositoryV2Dep,
25
+ EntityRepositoryV2ExternalDep,
26
+ ObservationRepositoryDep,
27
+ ObservationRepositoryV2Dep,
28
+ ObservationRepositoryV2ExternalDep,
29
+ RelationRepositoryDep,
30
+ RelationRepositoryV2Dep,
31
+ RelationRepositoryV2ExternalDep,
32
+ SearchRepositoryDep,
33
+ SearchRepositoryV2Dep,
34
+ SearchRepositoryV2ExternalDep,
35
+ )
36
+ from basic_memory.markdown import EntityParser
37
+ from basic_memory.markdown.markdown_processor import MarkdownProcessor
38
+ from basic_memory.services import EntityService, ProjectService
39
+ from basic_memory.services.context_service import ContextService
40
+ from basic_memory.services.directory_service import DirectoryService
41
+ from basic_memory.services.file_service import FileService
42
+ from basic_memory.services.link_resolver import LinkResolver
43
+ from basic_memory.services.search_service import SearchService
44
+ from basic_memory.sync import SyncService
45
+
46
+
47
+ # --- Entity Parser ---
48
+
49
+
50
+ async def get_entity_parser(project_config: ProjectConfigDep) -> EntityParser:
51
+ return EntityParser(project_config.home)
52
+
53
+
54
+ EntityParserDep = Annotated["EntityParser", Depends(get_entity_parser)]
55
+
56
+
57
+ async def get_entity_parser_v2(project_config: ProjectConfigV2Dep) -> EntityParser: # pragma: no cover
58
+ return EntityParser(project_config.home)
59
+
60
+
61
+ EntityParserV2Dep = Annotated["EntityParser", Depends(get_entity_parser_v2)]
62
+
63
+
64
+ async def get_entity_parser_v2_external(project_config: ProjectConfigV2ExternalDep) -> EntityParser:
65
+ return EntityParser(project_config.home)
66
+
67
+
68
+ EntityParserV2ExternalDep = Annotated["EntityParser", Depends(get_entity_parser_v2_external)]
69
+
70
+
71
+ # --- Markdown Processor ---
72
+
73
+
74
+ async def get_markdown_processor(
75
+ entity_parser: EntityParserDep, app_config: AppConfigDep
76
+ ) -> MarkdownProcessor:
77
+ return MarkdownProcessor(entity_parser, app_config=app_config)
78
+
79
+
80
+ MarkdownProcessorDep = Annotated[MarkdownProcessor, Depends(get_markdown_processor)]
81
+
82
+
83
+ async def get_markdown_processor_v2( # pragma: no cover
84
+ entity_parser: EntityParserV2Dep, app_config: AppConfigDep
85
+ ) -> MarkdownProcessor:
86
+ return MarkdownProcessor(entity_parser, app_config=app_config)
87
+
88
+
89
+ MarkdownProcessorV2Dep = Annotated[MarkdownProcessor, Depends(get_markdown_processor_v2)]
90
+
91
+
92
+ async def get_markdown_processor_v2_external(
93
+ entity_parser: EntityParserV2ExternalDep, app_config: AppConfigDep
94
+ ) -> MarkdownProcessor:
95
+ return MarkdownProcessor(entity_parser, app_config=app_config)
96
+
97
+
98
+ MarkdownProcessorV2ExternalDep = Annotated[
99
+ MarkdownProcessor, Depends(get_markdown_processor_v2_external)
100
+ ]
101
+
102
+
103
+ # --- File Service ---
104
+
105
+
106
+ async def get_file_service(
107
+ project_config: ProjectConfigDep,
108
+ markdown_processor: MarkdownProcessorDep,
109
+ app_config: AppConfigDep,
110
+ ) -> FileService:
111
+ file_service = FileService(project_config.home, markdown_processor, app_config=app_config)
112
+ logger.debug(
113
+ f"Created FileService for project: {project_config.name}, base_path: {project_config.home} "
114
+ )
115
+ return file_service
116
+
117
+
118
+ FileServiceDep = Annotated[FileService, Depends(get_file_service)]
119
+
120
+
121
+ async def get_file_service_v2( # pragma: no cover
122
+ project_config: ProjectConfigV2Dep,
123
+ markdown_processor: MarkdownProcessorV2Dep,
124
+ app_config: AppConfigDep,
125
+ ) -> FileService:
126
+ file_service = FileService(project_config.home, markdown_processor, app_config=app_config)
127
+ logger.debug(
128
+ f"Created FileService for project: {project_config.name}, base_path: {project_config.home}"
129
+ )
130
+ return file_service
131
+
132
+
133
+ FileServiceV2Dep = Annotated[FileService, Depends(get_file_service_v2)]
134
+
135
+
136
+ async def get_file_service_v2_external(
137
+ project_config: ProjectConfigV2ExternalDep,
138
+ markdown_processor: MarkdownProcessorV2ExternalDep,
139
+ app_config: AppConfigDep,
140
+ ) -> FileService:
141
+ file_service = FileService(project_config.home, markdown_processor, app_config=app_config)
142
+ logger.debug(
143
+ f"Created FileService for project: {project_config.name}, base_path: {project_config.home}"
144
+ )
145
+ return file_service
146
+
147
+
148
+ FileServiceV2ExternalDep = Annotated[FileService, Depends(get_file_service_v2_external)]
149
+
150
+
151
+ # --- Search Service ---
152
+
153
+
154
+ async def get_search_service(
155
+ search_repository: SearchRepositoryDep,
156
+ entity_repository: EntityRepositoryDep,
157
+ file_service: FileServiceDep,
158
+ ) -> SearchService:
159
+ """Create SearchService with dependencies."""
160
+ return SearchService(search_repository, entity_repository, file_service)
161
+
162
+
163
+ SearchServiceDep = Annotated[SearchService, Depends(get_search_service)]
164
+
165
+
166
+ async def get_search_service_v2( # pragma: no cover
167
+ search_repository: SearchRepositoryV2Dep,
168
+ entity_repository: EntityRepositoryV2Dep,
169
+ file_service: FileServiceV2Dep,
170
+ ) -> SearchService:
171
+ """Create SearchService for v2 API."""
172
+ return SearchService(search_repository, entity_repository, file_service)
173
+
174
+
175
+ SearchServiceV2Dep = Annotated[SearchService, Depends(get_search_service_v2)]
176
+
177
+
178
+ async def get_search_service_v2_external(
179
+ search_repository: SearchRepositoryV2ExternalDep,
180
+ entity_repository: EntityRepositoryV2ExternalDep,
181
+ file_service: FileServiceV2ExternalDep,
182
+ ) -> SearchService:
183
+ """Create SearchService for v2 API (uses external_id)."""
184
+ return SearchService(search_repository, entity_repository, file_service)
185
+
186
+
187
+ SearchServiceV2ExternalDep = Annotated[SearchService, Depends(get_search_service_v2_external)]
188
+
189
+
190
+ # --- Link Resolver ---
191
+
192
+
193
+ async def get_link_resolver(
194
+ entity_repository: EntityRepositoryDep, search_service: SearchServiceDep
195
+ ) -> LinkResolver:
196
+ return LinkResolver(entity_repository=entity_repository, search_service=search_service)
197
+
198
+
199
+ LinkResolverDep = Annotated[LinkResolver, Depends(get_link_resolver)]
200
+
201
+
202
+ async def get_link_resolver_v2( # pragma: no cover
203
+ entity_repository: EntityRepositoryV2Dep, search_service: SearchServiceV2Dep
204
+ ) -> LinkResolver:
205
+ return LinkResolver(entity_repository=entity_repository, search_service=search_service)
206
+
207
+
208
+ LinkResolverV2Dep = Annotated[LinkResolver, Depends(get_link_resolver_v2)]
209
+
210
+
211
+ async def get_link_resolver_v2_external(
212
+ entity_repository: EntityRepositoryV2ExternalDep, search_service: SearchServiceV2ExternalDep
213
+ ) -> LinkResolver:
214
+ return LinkResolver(entity_repository=entity_repository, search_service=search_service)
215
+
216
+
217
+ LinkResolverV2ExternalDep = Annotated[LinkResolver, Depends(get_link_resolver_v2_external)]
218
+
219
+
220
+ # --- Entity Service ---
221
+
222
+
223
+ async def get_entity_service(
224
+ entity_repository: EntityRepositoryDep,
225
+ observation_repository: ObservationRepositoryDep,
226
+ relation_repository: RelationRepositoryDep,
227
+ entity_parser: EntityParserDep,
228
+ file_service: FileServiceDep,
229
+ link_resolver: LinkResolverDep,
230
+ search_service: SearchServiceDep,
231
+ app_config: AppConfigDep,
232
+ ) -> EntityService:
233
+ """Create EntityService with repository."""
234
+ return EntityService(
235
+ entity_repository=entity_repository,
236
+ observation_repository=observation_repository,
237
+ relation_repository=relation_repository,
238
+ entity_parser=entity_parser,
239
+ file_service=file_service,
240
+ link_resolver=link_resolver,
241
+ search_service=search_service,
242
+ app_config=app_config,
243
+ )
244
+
245
+
246
+ EntityServiceDep = Annotated[EntityService, Depends(get_entity_service)]
247
+
248
+
249
+ async def get_entity_service_v2( # pragma: no cover
250
+ entity_repository: EntityRepositoryV2Dep,
251
+ observation_repository: ObservationRepositoryV2Dep,
252
+ relation_repository: RelationRepositoryV2Dep,
253
+ entity_parser: EntityParserV2Dep,
254
+ file_service: FileServiceV2Dep,
255
+ link_resolver: LinkResolverV2Dep,
256
+ search_service: SearchServiceV2Dep,
257
+ app_config: AppConfigDep,
258
+ ) -> EntityService:
259
+ """Create EntityService for v2 API."""
260
+ return EntityService(
261
+ entity_repository=entity_repository,
262
+ observation_repository=observation_repository,
263
+ relation_repository=relation_repository,
264
+ entity_parser=entity_parser,
265
+ file_service=file_service,
266
+ link_resolver=link_resolver,
267
+ search_service=search_service,
268
+ app_config=app_config,
269
+ )
270
+
271
+
272
+ EntityServiceV2Dep = Annotated[EntityService, Depends(get_entity_service_v2)]
273
+
274
+
275
+ async def get_entity_service_v2_external(
276
+ entity_repository: EntityRepositoryV2ExternalDep,
277
+ observation_repository: ObservationRepositoryV2ExternalDep,
278
+ relation_repository: RelationRepositoryV2ExternalDep,
279
+ entity_parser: EntityParserV2ExternalDep,
280
+ file_service: FileServiceV2ExternalDep,
281
+ link_resolver: LinkResolverV2ExternalDep,
282
+ search_service: SearchServiceV2ExternalDep,
283
+ app_config: AppConfigDep,
284
+ ) -> EntityService:
285
+ """Create EntityService for v2 API (uses external_id)."""
286
+ return EntityService(
287
+ entity_repository=entity_repository,
288
+ observation_repository=observation_repository,
289
+ relation_repository=relation_repository,
290
+ entity_parser=entity_parser,
291
+ file_service=file_service,
292
+ link_resolver=link_resolver,
293
+ search_service=search_service,
294
+ app_config=app_config,
295
+ )
296
+
297
+
298
+ EntityServiceV2ExternalDep = Annotated[EntityService, Depends(get_entity_service_v2_external)]
299
+
300
+
301
+ # --- Context Service ---
302
+
303
+
304
+ async def get_context_service(
305
+ search_repository: SearchRepositoryDep,
306
+ entity_repository: EntityRepositoryDep,
307
+ observation_repository: ObservationRepositoryDep,
308
+ ) -> ContextService:
309
+ return ContextService(
310
+ search_repository=search_repository,
311
+ entity_repository=entity_repository,
312
+ observation_repository=observation_repository,
313
+ )
314
+
315
+
316
+ ContextServiceDep = Annotated[ContextService, Depends(get_context_service)]
317
+
318
+
319
+ async def get_context_service_v2( # pragma: no cover
320
+ search_repository: SearchRepositoryV2Dep,
321
+ entity_repository: EntityRepositoryV2Dep,
322
+ observation_repository: ObservationRepositoryV2Dep,
323
+ ) -> ContextService:
324
+ """Create ContextService for v2 API."""
325
+ return ContextService(
326
+ search_repository=search_repository,
327
+ entity_repository=entity_repository,
328
+ observation_repository=observation_repository,
329
+ )
330
+
331
+
332
+ ContextServiceV2Dep = Annotated[ContextService, Depends(get_context_service_v2)]
333
+
334
+
335
+ async def get_context_service_v2_external(
336
+ search_repository: SearchRepositoryV2ExternalDep,
337
+ entity_repository: EntityRepositoryV2ExternalDep,
338
+ observation_repository: ObservationRepositoryV2ExternalDep,
339
+ ) -> ContextService:
340
+ """Create ContextService for v2 API (uses external_id)."""
341
+ return ContextService(
342
+ search_repository=search_repository,
343
+ entity_repository=entity_repository,
344
+ observation_repository=observation_repository,
345
+ )
346
+
347
+
348
+ ContextServiceV2ExternalDep = Annotated[ContextService, Depends(get_context_service_v2_external)]
349
+
350
+
351
+ # --- Sync Service ---
352
+
353
+
354
+ async def get_sync_service(
355
+ app_config: AppConfigDep,
356
+ entity_service: EntityServiceDep,
357
+ entity_parser: EntityParserDep,
358
+ entity_repository: EntityRepositoryDep,
359
+ relation_repository: RelationRepositoryDep,
360
+ project_repository: ProjectRepositoryDep,
361
+ search_service: SearchServiceDep,
362
+ file_service: FileServiceDep,
363
+ ) -> SyncService: # pragma: no cover
364
+ return SyncService(
365
+ app_config=app_config,
366
+ entity_service=entity_service,
367
+ entity_parser=entity_parser,
368
+ entity_repository=entity_repository,
369
+ relation_repository=relation_repository,
370
+ project_repository=project_repository,
371
+ search_service=search_service,
372
+ file_service=file_service,
373
+ )
374
+
375
+
376
+ SyncServiceDep = Annotated[SyncService, Depends(get_sync_service)]
377
+
378
+
379
+ async def get_sync_service_v2(
380
+ app_config: AppConfigDep,
381
+ entity_service: EntityServiceV2Dep,
382
+ entity_parser: EntityParserV2Dep,
383
+ entity_repository: EntityRepositoryV2Dep,
384
+ relation_repository: RelationRepositoryV2Dep,
385
+ project_repository: ProjectRepositoryDep,
386
+ search_service: SearchServiceV2Dep,
387
+ file_service: FileServiceV2Dep,
388
+ ) -> SyncService: # pragma: no cover
389
+ """Create SyncService for v2 API."""
390
+ return SyncService(
391
+ app_config=app_config,
392
+ entity_service=entity_service,
393
+ entity_parser=entity_parser,
394
+ entity_repository=entity_repository,
395
+ relation_repository=relation_repository,
396
+ project_repository=project_repository,
397
+ search_service=search_service,
398
+ file_service=file_service,
399
+ )
400
+
401
+
402
+ SyncServiceV2Dep = Annotated[SyncService, Depends(get_sync_service_v2)]
403
+
404
+
405
+ async def get_sync_service_v2_external(
406
+ app_config: AppConfigDep,
407
+ entity_service: EntityServiceV2ExternalDep,
408
+ entity_parser: EntityParserV2ExternalDep,
409
+ entity_repository: EntityRepositoryV2ExternalDep,
410
+ relation_repository: RelationRepositoryV2ExternalDep,
411
+ project_repository: ProjectRepositoryDep,
412
+ search_service: SearchServiceV2ExternalDep,
413
+ file_service: FileServiceV2ExternalDep,
414
+ ) -> SyncService: # pragma: no cover
415
+ """Create SyncService for v2 API (uses external_id)."""
416
+ return SyncService(
417
+ app_config=app_config,
418
+ entity_service=entity_service,
419
+ entity_parser=entity_parser,
420
+ entity_repository=entity_repository,
421
+ relation_repository=relation_repository,
422
+ project_repository=project_repository,
423
+ search_service=search_service,
424
+ file_service=file_service,
425
+ )
426
+
427
+
428
+ SyncServiceV2ExternalDep = Annotated[SyncService, Depends(get_sync_service_v2_external)]
429
+
430
+
431
+ # --- Project Service ---
432
+
433
+
434
+ async def get_project_service(
435
+ project_repository: ProjectRepositoryDep,
436
+ ) -> ProjectService:
437
+ """Create ProjectService with repository."""
438
+ return ProjectService(repository=project_repository)
439
+
440
+
441
+ ProjectServiceDep = Annotated[ProjectService, Depends(get_project_service)]
442
+
443
+
444
+ # --- Directory Service ---
445
+
446
+
447
+ async def get_directory_service(
448
+ entity_repository: EntityRepositoryDep,
449
+ ) -> DirectoryService:
450
+ """Create DirectoryService with dependencies."""
451
+ return DirectoryService(
452
+ entity_repository=entity_repository,
453
+ )
454
+
455
+
456
+ DirectoryServiceDep = Annotated[DirectoryService, Depends(get_directory_service)]
457
+
458
+
459
+ async def get_directory_service_v2( # pragma: no cover
460
+ entity_repository: EntityRepositoryV2Dep,
461
+ ) -> DirectoryService:
462
+ """Create DirectoryService for v2 API (uses integer project_id from path)."""
463
+ return DirectoryService(
464
+ entity_repository=entity_repository,
465
+ )
466
+
467
+
468
+ DirectoryServiceV2Dep = Annotated[DirectoryService, Depends(get_directory_service_v2)]
469
+
470
+
471
+ async def get_directory_service_v2_external(
472
+ entity_repository: EntityRepositoryV2ExternalDep,
473
+ ) -> DirectoryService:
474
+ """Create DirectoryService for v2 API (uses external_id from path)."""
475
+ return DirectoryService(
476
+ entity_repository=entity_repository,
477
+ )
478
+
479
+
480
+ DirectoryServiceV2ExternalDep = Annotated[DirectoryService, Depends(get_directory_service_v2_external)]
basic_memory/deps.py CHANGED
@@ -1,185 +1,16 @@
1
- """Dependency injection functions for basic-memory services."""
2
-
3
- from typing import Annotated
4
-
5
- import logfire
6
- from fastapi import Depends
7
- from sqlalchemy.ext.asyncio import (
8
- AsyncSession,
9
- AsyncEngine,
10
- async_sessionmaker,
11
- )
12
-
13
- from basic_memory import db
14
- from basic_memory.config import ProjectConfig, config
15
- from basic_memory.markdown import EntityParser
16
- from basic_memory.markdown.markdown_processor import MarkdownProcessor
17
- from basic_memory.repository.entity_repository import EntityRepository
18
- from basic_memory.repository.observation_repository import ObservationRepository
19
- from basic_memory.repository.relation_repository import RelationRepository
20
- from basic_memory.repository.search_repository import SearchRepository
21
- from basic_memory.services import (
22
- EntityService,
23
- )
24
- from basic_memory.services.context_service import ContextService
25
- from basic_memory.services.file_service import FileService
26
- from basic_memory.services.link_resolver import LinkResolver
27
- from basic_memory.services.search_service import SearchService
28
-
29
-
30
- ## project
31
-
32
-
33
- def get_project_config() -> ProjectConfig: # pragma: no cover
34
- return config
35
-
36
-
37
- ProjectConfigDep = Annotated[ProjectConfig, Depends(get_project_config)] # pragma: no cover
38
-
39
-
40
- ## sqlalchemy
41
-
42
-
43
- async def get_engine_factory(
44
- project_config: ProjectConfigDep,
45
- ) -> tuple[AsyncEngine, async_sessionmaker[AsyncSession]]: # pragma: no cover
46
- """Get engine and session maker."""
47
- engine, session_maker = await db.get_or_create_db(project_config.database_path)
48
- if project_config.env != "test":
49
- logfire.instrument_sqlalchemy(engine=engine)
50
- return engine, session_maker
51
-
52
-
53
- EngineFactoryDep = Annotated[
54
- tuple[AsyncEngine, async_sessionmaker[AsyncSession]], Depends(get_engine_factory)
55
- ]
56
-
57
-
58
- async def get_session_maker(engine_factory: EngineFactoryDep) -> async_sessionmaker[AsyncSession]:
59
- """Get session maker."""
60
- _, session_maker = engine_factory
61
- return session_maker
62
-
63
-
64
- SessionMakerDep = Annotated[async_sessionmaker, Depends(get_session_maker)]
65
-
66
-
67
- ## repositories
68
-
69
-
70
- async def get_entity_repository(
71
- session_maker: SessionMakerDep,
72
- ) -> EntityRepository:
73
- """Create an EntityRepository instance."""
74
- return EntityRepository(session_maker)
75
-
76
-
77
- EntityRepositoryDep = Annotated[EntityRepository, Depends(get_entity_repository)]
78
-
79
-
80
- async def get_observation_repository(
81
- session_maker: SessionMakerDep,
82
- ) -> ObservationRepository:
83
- """Create an ObservationRepository instance."""
84
- return ObservationRepository(session_maker)
85
-
86
-
87
- ObservationRepositoryDep = Annotated[ObservationRepository, Depends(get_observation_repository)]
88
-
89
-
90
- async def get_relation_repository(
91
- session_maker: SessionMakerDep,
92
- ) -> RelationRepository:
93
- """Create a RelationRepository instance."""
94
- return RelationRepository(session_maker)
95
-
96
-
97
- RelationRepositoryDep = Annotated[RelationRepository, Depends(get_relation_repository)]
98
-
99
-
100
- async def get_search_repository(
101
- session_maker: SessionMakerDep,
102
- ) -> SearchRepository:
103
- """Create a SearchRepository instance."""
104
- return SearchRepository(session_maker)
105
-
106
-
107
- SearchRepositoryDep = Annotated[SearchRepository, Depends(get_search_repository)]
108
-
109
-
110
- ## services
111
-
112
-
113
- async def get_entity_parser(project_config: ProjectConfigDep) -> EntityParser:
114
- return EntityParser(project_config.home)
115
-
116
-
117
- EntityParserDep = Annotated["EntityParser", Depends(get_entity_parser)]
118
-
119
-
120
- async def get_markdown_processor(entity_parser: EntityParserDep) -> MarkdownProcessor:
121
- return MarkdownProcessor(entity_parser)
122
-
123
-
124
- MarkdownProcessorDep = Annotated[MarkdownProcessor, Depends(get_markdown_processor)]
125
-
126
-
127
- async def get_file_service(
128
- project_config: ProjectConfigDep, markdown_processor: MarkdownProcessorDep
129
- ) -> FileService:
130
- return FileService(project_config.home, markdown_processor)
131
-
132
-
133
- FileServiceDep = Annotated[FileService, Depends(get_file_service)]
134
-
135
-
136
- async def get_entity_service(
137
- entity_repository: EntityRepositoryDep,
138
- observation_repository: ObservationRepositoryDep,
139
- relation_repository: RelationRepositoryDep,
140
- entity_parser: EntityParserDep,
141
- file_service: FileServiceDep,
142
- link_resolver: "LinkResolverDep",
143
- ) -> EntityService:
144
- """Create EntityService with repository."""
145
- return EntityService(
146
- entity_repository=entity_repository,
147
- observation_repository=observation_repository,
148
- relation_repository=relation_repository,
149
- entity_parser=entity_parser,
150
- file_service=file_service,
151
- link_resolver=link_resolver,
152
- )
153
-
154
-
155
- EntityServiceDep = Annotated[EntityService, Depends(get_entity_service)]
156
-
157
-
158
- async def get_search_service(
159
- search_repository: SearchRepositoryDep,
160
- entity_repository: EntityRepositoryDep,
161
- file_service: FileServiceDep,
162
- ) -> SearchService:
163
- """Create SearchService with dependencies."""
164
- return SearchService(search_repository, entity_repository, file_service)
165
-
166
-
167
- SearchServiceDep = Annotated[SearchService, Depends(get_search_service)]
168
-
169
-
170
- async def get_link_resolver(
171
- entity_repository: EntityRepositoryDep, search_service: SearchServiceDep
172
- ) -> LinkResolver:
173
- return LinkResolver(entity_repository=entity_repository, search_service=search_service)
174
-
175
-
176
- LinkResolverDep = Annotated[LinkResolver, Depends(get_link_resolver)]
177
-
178
-
179
- async def get_context_service(
180
- search_repository: SearchRepositoryDep, entity_repository: EntityRepositoryDep
181
- ) -> ContextService:
182
- return ContextService(search_repository, entity_repository)
183
-
184
-
185
- ContextServiceDep = Annotated[ContextService, Depends(get_context_service)]
1
+ """Dependency injection functions for basic-memory services.
2
+
3
+ DEPRECATED: This module is a backwards-compatibility shim.
4
+ Import from basic_memory.deps package submodules instead:
5
+ - basic_memory.deps.config for configuration
6
+ - basic_memory.deps.db for database/session
7
+ - basic_memory.deps.projects for project resolution
8
+ - basic_memory.deps.repositories for data access
9
+ - basic_memory.deps.services for business logic
10
+ - basic_memory.deps.importers for import functionality
11
+
12
+ This file will be removed once all callers are migrated.
13
+ """
14
+
15
+ # Re-export everything from the deps package for backwards compatibility
16
+ from basic_memory.deps import * # noqa: F401, F403 # pragma: no cover