edsl 0.1.39.dev3__py3-none-any.whl → 0.1.39.dev5__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 (341) hide show
  1. edsl/Base.py +413 -332
  2. edsl/BaseDiff.py +260 -260
  3. edsl/TemplateLoader.py +24 -24
  4. edsl/__init__.py +57 -49
  5. edsl/__version__.py +1 -1
  6. edsl/agents/Agent.py +1071 -867
  7. edsl/agents/AgentList.py +551 -413
  8. edsl/agents/Invigilator.py +284 -233
  9. edsl/agents/InvigilatorBase.py +257 -270
  10. edsl/agents/PromptConstructor.py +272 -354
  11. edsl/agents/QuestionInstructionPromptBuilder.py +128 -0
  12. edsl/agents/QuestionTemplateReplacementsBuilder.py +137 -0
  13. edsl/agents/__init__.py +2 -3
  14. edsl/agents/descriptors.py +99 -99
  15. edsl/agents/prompt_helpers.py +129 -129
  16. edsl/agents/question_option_processor.py +172 -0
  17. edsl/auto/AutoStudy.py +130 -117
  18. edsl/auto/StageBase.py +243 -230
  19. edsl/auto/StageGenerateSurvey.py +178 -178
  20. edsl/auto/StageLabelQuestions.py +125 -125
  21. edsl/auto/StagePersona.py +61 -61
  22. edsl/auto/StagePersonaDimensionValueRanges.py +88 -88
  23. edsl/auto/StagePersonaDimensionValues.py +74 -74
  24. edsl/auto/StagePersonaDimensions.py +69 -69
  25. edsl/auto/StageQuestions.py +74 -73
  26. edsl/auto/SurveyCreatorPipeline.py +21 -21
  27. edsl/auto/utilities.py +218 -224
  28. edsl/base/Base.py +279 -279
  29. edsl/config.py +177 -157
  30. edsl/conversation/Conversation.py +290 -290
  31. edsl/conversation/car_buying.py +59 -58
  32. edsl/conversation/chips.py +95 -95
  33. edsl/conversation/mug_negotiation.py +81 -81
  34. edsl/conversation/next_speaker_utilities.py +93 -93
  35. edsl/coop/CoopFunctionsMixin.py +15 -0
  36. edsl/coop/ExpectedParrotKeyHandler.py +125 -0
  37. edsl/coop/PriceFetcher.py +54 -54
  38. edsl/coop/__init__.py +2 -2
  39. edsl/coop/coop.py +1106 -1028
  40. edsl/coop/utils.py +131 -131
  41. edsl/data/Cache.py +573 -555
  42. edsl/data/CacheEntry.py +230 -233
  43. edsl/data/CacheHandler.py +168 -149
  44. edsl/data/RemoteCacheSync.py +186 -78
  45. edsl/data/SQLiteDict.py +292 -292
  46. edsl/data/__init__.py +5 -4
  47. edsl/data/orm.py +10 -10
  48. edsl/data_transfer_models.py +74 -73
  49. edsl/enums.py +202 -175
  50. edsl/exceptions/BaseException.py +21 -21
  51. edsl/exceptions/__init__.py +54 -54
  52. edsl/exceptions/agents.py +54 -42
  53. edsl/exceptions/cache.py +5 -5
  54. edsl/exceptions/configuration.py +16 -16
  55. edsl/exceptions/coop.py +10 -10
  56. edsl/exceptions/data.py +14 -14
  57. edsl/exceptions/general.py +34 -34
  58. edsl/exceptions/inference_services.py +5 -0
  59. edsl/exceptions/jobs.py +33 -33
  60. edsl/exceptions/language_models.py +63 -63
  61. edsl/exceptions/prompts.py +15 -15
  62. edsl/exceptions/questions.py +109 -91
  63. edsl/exceptions/results.py +29 -29
  64. edsl/exceptions/scenarios.py +29 -22
  65. edsl/exceptions/surveys.py +37 -37
  66. edsl/inference_services/AnthropicService.py +106 -87
  67. edsl/inference_services/AvailableModelCacheHandler.py +184 -0
  68. edsl/inference_services/AvailableModelFetcher.py +215 -0
  69. edsl/inference_services/AwsBedrock.py +118 -120
  70. edsl/inference_services/AzureAI.py +215 -217
  71. edsl/inference_services/DeepInfraService.py +18 -18
  72. edsl/inference_services/GoogleService.py +143 -148
  73. edsl/inference_services/GroqService.py +20 -20
  74. edsl/inference_services/InferenceServiceABC.py +80 -147
  75. edsl/inference_services/InferenceServicesCollection.py +138 -97
  76. edsl/inference_services/MistralAIService.py +120 -123
  77. edsl/inference_services/OllamaService.py +18 -18
  78. edsl/inference_services/OpenAIService.py +236 -224
  79. edsl/inference_services/PerplexityService.py +160 -163
  80. edsl/inference_services/ServiceAvailability.py +135 -0
  81. edsl/inference_services/TestService.py +90 -89
  82. edsl/inference_services/TogetherAIService.py +172 -170
  83. edsl/inference_services/data_structures.py +134 -0
  84. edsl/inference_services/models_available_cache.py +118 -118
  85. edsl/inference_services/rate_limits_cache.py +25 -25
  86. edsl/inference_services/registry.py +41 -41
  87. edsl/inference_services/write_available.py +10 -10
  88. edsl/jobs/AnswerQuestionFunctionConstructor.py +223 -0
  89. edsl/jobs/Answers.py +43 -56
  90. edsl/jobs/FetchInvigilator.py +47 -0
  91. edsl/jobs/InterviewTaskManager.py +98 -0
  92. edsl/jobs/InterviewsConstructor.py +50 -0
  93. edsl/jobs/Jobs.py +823 -898
  94. edsl/jobs/JobsChecks.py +172 -147
  95. edsl/jobs/JobsComponentConstructor.py +189 -0
  96. edsl/jobs/JobsPrompts.py +270 -268
  97. edsl/jobs/JobsRemoteInferenceHandler.py +311 -239
  98. edsl/jobs/JobsRemoteInferenceLogger.py +239 -0
  99. edsl/jobs/RequestTokenEstimator.py +30 -0
  100. edsl/jobs/__init__.py +1 -1
  101. edsl/jobs/async_interview_runner.py +138 -0
  102. edsl/jobs/buckets/BucketCollection.py +104 -63
  103. edsl/jobs/buckets/ModelBuckets.py +65 -65
  104. edsl/jobs/buckets/TokenBucket.py +283 -251
  105. edsl/jobs/buckets/TokenBucketAPI.py +211 -0
  106. edsl/jobs/buckets/TokenBucketClient.py +191 -0
  107. edsl/jobs/check_survey_scenario_compatibility.py +85 -0
  108. edsl/jobs/data_structures.py +120 -0
  109. edsl/jobs/decorators.py +35 -0
  110. edsl/jobs/interviews/Interview.py +396 -661
  111. edsl/jobs/interviews/InterviewExceptionCollection.py +99 -99
  112. edsl/jobs/interviews/InterviewExceptionEntry.py +186 -186
  113. edsl/jobs/interviews/InterviewStatistic.py +63 -63
  114. edsl/jobs/interviews/InterviewStatisticsCollection.py +25 -25
  115. edsl/jobs/interviews/InterviewStatusDictionary.py +78 -78
  116. edsl/jobs/interviews/InterviewStatusLog.py +92 -92
  117. edsl/jobs/interviews/ReportErrors.py +66 -66
  118. edsl/jobs/interviews/interview_status_enum.py +9 -9
  119. edsl/jobs/jobs_status_enums.py +9 -0
  120. edsl/jobs/loggers/HTMLTableJobLogger.py +304 -0
  121. edsl/jobs/results_exceptions_handler.py +98 -0
  122. edsl/jobs/runners/JobsRunnerAsyncio.py +151 -466
  123. edsl/jobs/runners/JobsRunnerStatus.py +297 -330
  124. edsl/jobs/tasks/QuestionTaskCreator.py +244 -242
  125. edsl/jobs/tasks/TaskCreators.py +64 -64
  126. edsl/jobs/tasks/TaskHistory.py +470 -450
  127. edsl/jobs/tasks/TaskStatusLog.py +23 -23
  128. edsl/jobs/tasks/task_status_enum.py +161 -163
  129. edsl/jobs/tokens/InterviewTokenUsage.py +27 -27
  130. edsl/jobs/tokens/TokenUsage.py +34 -34
  131. edsl/language_models/ComputeCost.py +63 -0
  132. edsl/language_models/LanguageModel.py +626 -668
  133. edsl/language_models/ModelList.py +164 -155
  134. edsl/language_models/PriceManager.py +127 -0
  135. edsl/language_models/RawResponseHandler.py +106 -0
  136. edsl/language_models/RegisterLanguageModelsMeta.py +184 -184
  137. edsl/language_models/ServiceDataSources.py +0 -0
  138. edsl/language_models/__init__.py +2 -3
  139. edsl/language_models/fake_openai_call.py +15 -15
  140. edsl/language_models/fake_openai_service.py +61 -61
  141. edsl/language_models/key_management/KeyLookup.py +63 -0
  142. edsl/language_models/key_management/KeyLookupBuilder.py +273 -0
  143. edsl/language_models/key_management/KeyLookupCollection.py +38 -0
  144. edsl/language_models/key_management/__init__.py +0 -0
  145. edsl/language_models/key_management/models.py +131 -0
  146. edsl/language_models/model.py +256 -0
  147. edsl/language_models/repair.py +156 -156
  148. edsl/language_models/utilities.py +65 -64
  149. edsl/notebooks/Notebook.py +263 -258
  150. edsl/notebooks/NotebookToLaTeX.py +142 -0
  151. edsl/notebooks/__init__.py +1 -1
  152. edsl/prompts/Prompt.py +352 -362
  153. edsl/prompts/__init__.py +2 -2
  154. edsl/questions/ExceptionExplainer.py +77 -0
  155. edsl/questions/HTMLQuestion.py +103 -0
  156. edsl/questions/QuestionBase.py +518 -664
  157. edsl/questions/QuestionBasePromptsMixin.py +221 -217
  158. edsl/questions/QuestionBudget.py +227 -227
  159. edsl/questions/QuestionCheckBox.py +359 -359
  160. edsl/questions/QuestionExtract.py +180 -182
  161. edsl/questions/QuestionFreeText.py +113 -114
  162. edsl/questions/QuestionFunctional.py +166 -166
  163. edsl/questions/QuestionList.py +223 -231
  164. edsl/questions/QuestionMatrix.py +265 -0
  165. edsl/questions/QuestionMultipleChoice.py +330 -286
  166. edsl/questions/QuestionNumerical.py +151 -153
  167. edsl/questions/QuestionRank.py +314 -324
  168. edsl/questions/Quick.py +41 -41
  169. edsl/questions/SimpleAskMixin.py +74 -73
  170. edsl/questions/__init__.py +27 -26
  171. edsl/questions/{AnswerValidatorMixin.py → answer_validator_mixin.py} +334 -289
  172. edsl/questions/compose_questions.py +98 -98
  173. edsl/questions/data_structures.py +20 -0
  174. edsl/questions/decorators.py +21 -21
  175. edsl/questions/derived/QuestionLikertFive.py +76 -76
  176. edsl/questions/derived/QuestionLinearScale.py +90 -87
  177. edsl/questions/derived/QuestionTopK.py +93 -93
  178. edsl/questions/derived/QuestionYesNo.py +82 -82
  179. edsl/questions/descriptors.py +427 -413
  180. edsl/questions/loop_processor.py +149 -0
  181. edsl/questions/prompt_templates/question_budget.jinja +13 -13
  182. edsl/questions/prompt_templates/question_checkbox.jinja +32 -32
  183. edsl/questions/prompt_templates/question_extract.jinja +11 -11
  184. edsl/questions/prompt_templates/question_free_text.jinja +3 -3
  185. edsl/questions/prompt_templates/question_linear_scale.jinja +11 -11
  186. edsl/questions/prompt_templates/question_list.jinja +17 -17
  187. edsl/questions/prompt_templates/question_multiple_choice.jinja +33 -33
  188. edsl/questions/prompt_templates/question_numerical.jinja +36 -36
  189. edsl/questions/{QuestionBaseGenMixin.py → question_base_gen_mixin.py} +168 -161
  190. edsl/questions/question_registry.py +177 -177
  191. edsl/questions/{RegisterQuestionsMeta.py → register_questions_meta.py} +71 -71
  192. edsl/questions/{ResponseValidatorABC.py → response_validator_abc.py} +188 -174
  193. edsl/questions/response_validator_factory.py +34 -0
  194. edsl/questions/settings.py +12 -12
  195. edsl/questions/templates/budget/answering_instructions.jinja +7 -7
  196. edsl/questions/templates/budget/question_presentation.jinja +7 -7
  197. edsl/questions/templates/checkbox/answering_instructions.jinja +10 -10
  198. edsl/questions/templates/checkbox/question_presentation.jinja +22 -22
  199. edsl/questions/templates/extract/answering_instructions.jinja +7 -7
  200. edsl/questions/templates/likert_five/answering_instructions.jinja +10 -10
  201. edsl/questions/templates/likert_five/question_presentation.jinja +11 -11
  202. edsl/questions/templates/linear_scale/answering_instructions.jinja +5 -5
  203. edsl/questions/templates/linear_scale/question_presentation.jinja +5 -5
  204. edsl/questions/templates/list/answering_instructions.jinja +3 -3
  205. edsl/questions/templates/list/question_presentation.jinja +5 -5
  206. edsl/questions/templates/matrix/__init__.py +1 -0
  207. edsl/questions/templates/matrix/answering_instructions.jinja +5 -0
  208. edsl/questions/templates/matrix/question_presentation.jinja +20 -0
  209. edsl/questions/templates/multiple_choice/answering_instructions.jinja +9 -9
  210. edsl/questions/templates/multiple_choice/question_presentation.jinja +11 -11
  211. edsl/questions/templates/numerical/answering_instructions.jinja +6 -6
  212. edsl/questions/templates/numerical/question_presentation.jinja +6 -6
  213. edsl/questions/templates/rank/answering_instructions.jinja +11 -11
  214. edsl/questions/templates/rank/question_presentation.jinja +15 -15
  215. edsl/questions/templates/top_k/answering_instructions.jinja +8 -8
  216. edsl/questions/templates/top_k/question_presentation.jinja +22 -22
  217. edsl/questions/templates/yes_no/answering_instructions.jinja +6 -6
  218. edsl/questions/templates/yes_no/question_presentation.jinja +11 -11
  219. edsl/results/CSSParameterizer.py +108 -108
  220. edsl/results/Dataset.py +587 -424
  221. edsl/results/DatasetExportMixin.py +594 -731
  222. edsl/results/DatasetTree.py +295 -275
  223. edsl/results/MarkdownToDocx.py +122 -0
  224. edsl/results/MarkdownToPDF.py +111 -0
  225. edsl/results/Result.py +557 -465
  226. edsl/results/Results.py +1183 -1165
  227. edsl/results/ResultsExportMixin.py +45 -43
  228. edsl/results/ResultsGGMixin.py +121 -121
  229. edsl/results/TableDisplay.py +125 -198
  230. edsl/results/TextEditor.py +50 -0
  231. edsl/results/__init__.py +2 -2
  232. edsl/results/file_exports.py +252 -0
  233. edsl/results/{ResultsFetchMixin.py → results_fetch_mixin.py} +33 -33
  234. edsl/results/{Selector.py → results_selector.py} +145 -135
  235. edsl/results/{ResultsToolsMixin.py → results_tools_mixin.py} +98 -98
  236. edsl/results/smart_objects.py +96 -0
  237. edsl/results/table_data_class.py +12 -0
  238. edsl/results/table_display.css +77 -77
  239. edsl/results/table_renderers.py +118 -0
  240. edsl/results/tree_explore.py +115 -115
  241. edsl/scenarios/ConstructDownloadLink.py +109 -0
  242. edsl/scenarios/DocumentChunker.py +102 -0
  243. edsl/scenarios/DocxScenario.py +16 -0
  244. edsl/scenarios/FileStore.py +511 -632
  245. edsl/scenarios/PdfExtractor.py +40 -0
  246. edsl/scenarios/Scenario.py +498 -601
  247. edsl/scenarios/ScenarioHtmlMixin.py +65 -64
  248. edsl/scenarios/ScenarioList.py +1458 -1287
  249. edsl/scenarios/ScenarioListExportMixin.py +45 -52
  250. edsl/scenarios/ScenarioListPdfMixin.py +239 -261
  251. edsl/scenarios/__init__.py +3 -4
  252. edsl/scenarios/directory_scanner.py +96 -0
  253. edsl/scenarios/file_methods.py +85 -0
  254. edsl/scenarios/handlers/__init__.py +13 -0
  255. edsl/scenarios/handlers/csv.py +38 -0
  256. edsl/scenarios/handlers/docx.py +76 -0
  257. edsl/scenarios/handlers/html.py +37 -0
  258. edsl/scenarios/handlers/json.py +111 -0
  259. edsl/scenarios/handlers/latex.py +5 -0
  260. edsl/scenarios/handlers/md.py +51 -0
  261. edsl/scenarios/handlers/pdf.py +68 -0
  262. edsl/scenarios/handlers/png.py +39 -0
  263. edsl/scenarios/handlers/pptx.py +105 -0
  264. edsl/scenarios/handlers/py.py +294 -0
  265. edsl/scenarios/handlers/sql.py +313 -0
  266. edsl/scenarios/handlers/sqlite.py +149 -0
  267. edsl/scenarios/handlers/txt.py +33 -0
  268. edsl/scenarios/{ScenarioJoin.py → scenario_join.py} +131 -127
  269. edsl/scenarios/scenario_selector.py +156 -0
  270. edsl/shared.py +1 -1
  271. edsl/study/ObjectEntry.py +173 -173
  272. edsl/study/ProofOfWork.py +113 -113
  273. edsl/study/SnapShot.py +80 -80
  274. edsl/study/Study.py +521 -528
  275. edsl/study/__init__.py +4 -4
  276. edsl/surveys/ConstructDAG.py +92 -0
  277. edsl/surveys/DAG.py +148 -148
  278. edsl/surveys/EditSurvey.py +221 -0
  279. edsl/surveys/InstructionHandler.py +100 -0
  280. edsl/surveys/Memory.py +31 -31
  281. edsl/surveys/MemoryManagement.py +72 -0
  282. edsl/surveys/MemoryPlan.py +244 -244
  283. edsl/surveys/Rule.py +327 -326
  284. edsl/surveys/RuleCollection.py +385 -387
  285. edsl/surveys/RuleManager.py +172 -0
  286. edsl/surveys/Simulator.py +75 -0
  287. edsl/surveys/Survey.py +1280 -1801
  288. edsl/surveys/SurveyCSS.py +273 -261
  289. edsl/surveys/SurveyExportMixin.py +259 -259
  290. edsl/surveys/{SurveyFlowVisualizationMixin.py → SurveyFlowVisualization.py} +181 -179
  291. edsl/surveys/SurveyQualtricsImport.py +284 -284
  292. edsl/surveys/SurveyToApp.py +141 -0
  293. edsl/surveys/__init__.py +5 -3
  294. edsl/surveys/base.py +53 -53
  295. edsl/surveys/descriptors.py +60 -56
  296. edsl/surveys/instructions/ChangeInstruction.py +48 -49
  297. edsl/surveys/instructions/Instruction.py +56 -65
  298. edsl/surveys/instructions/InstructionCollection.py +82 -77
  299. edsl/templates/error_reporting/base.html +23 -23
  300. edsl/templates/error_reporting/exceptions_by_model.html +34 -34
  301. edsl/templates/error_reporting/exceptions_by_question_name.html +16 -16
  302. edsl/templates/error_reporting/exceptions_by_type.html +16 -16
  303. edsl/templates/error_reporting/interview_details.html +115 -115
  304. edsl/templates/error_reporting/interviews.html +19 -19
  305. edsl/templates/error_reporting/overview.html +4 -4
  306. edsl/templates/error_reporting/performance_plot.html +1 -1
  307. edsl/templates/error_reporting/report.css +73 -73
  308. edsl/templates/error_reporting/report.html +117 -117
  309. edsl/templates/error_reporting/report.js +25 -25
  310. edsl/tools/__init__.py +1 -1
  311. edsl/tools/clusters.py +192 -192
  312. edsl/tools/embeddings.py +27 -27
  313. edsl/tools/embeddings_plotting.py +118 -118
  314. edsl/tools/plotting.py +112 -112
  315. edsl/tools/summarize.py +18 -18
  316. edsl/utilities/PrettyList.py +56 -0
  317. edsl/utilities/SystemInfo.py +28 -28
  318. edsl/utilities/__init__.py +22 -22
  319. edsl/utilities/ast_utilities.py +25 -25
  320. edsl/utilities/data/Registry.py +6 -6
  321. edsl/utilities/data/__init__.py +1 -1
  322. edsl/utilities/data/scooter_results.json +1 -1
  323. edsl/utilities/decorators.py +77 -77
  324. edsl/utilities/gcp_bucket/cloud_storage.py +96 -96
  325. edsl/utilities/interface.py +627 -627
  326. edsl/utilities/is_notebook.py +18 -0
  327. edsl/utilities/is_valid_variable_name.py +11 -0
  328. edsl/utilities/naming_utilities.py +263 -263
  329. edsl/utilities/remove_edsl_version.py +24 -0
  330. edsl/utilities/repair_functions.py +28 -28
  331. edsl/utilities/restricted_python.py +70 -70
  332. edsl/utilities/utilities.py +436 -424
  333. {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/LICENSE +21 -21
  334. {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/METADATA +13 -11
  335. edsl-0.1.39.dev5.dist-info/RECORD +358 -0
  336. {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/WHEEL +1 -1
  337. edsl/language_models/KeyLookup.py +0 -30
  338. edsl/language_models/registry.py +0 -190
  339. edsl/language_models/unused/ReplicateBase.py +0 -83
  340. edsl/results/ResultsDBMixin.py +0 -238
  341. edsl-0.1.39.dev3.dist-info/RECORD +0 -277
edsl/data/CacheHandler.py CHANGED
@@ -1,149 +1,168 @@
1
- from __future__ import annotations
2
- import ast
3
- import json
4
- import os
5
- import shutil
6
- import sqlite3
7
- from edsl.config import CONFIG
8
- from edsl.data.Cache import Cache
9
- from edsl.data.CacheEntry import CacheEntry
10
- from edsl.data.SQLiteDict import SQLiteDict
11
-
12
- from edsl.config import CONFIG
13
-
14
-
15
- def set_session_cache(cache: Cache) -> None:
16
- """
17
- Set the session cache.
18
- """
19
- CONFIG.EDSL_SESSION_CACHE = cache
20
-
21
-
22
- def unset_session_cache() -> None:
23
- """
24
- Unset the session cache.
25
- """
26
- if hasattr(CONFIG, "EDSL_SESSION_CACHE"):
27
- del CONFIG.EDSL_SESSION_CACHE
28
-
29
-
30
- class CacheHandler:
31
- """
32
- This CacheHandler figures out what caches are available and does migrations, as needed.
33
- """
34
-
35
- CACHE_PATH = CONFIG.get("EDSL_DATABASE_PATH")
36
-
37
- def __init__(self, test: bool = False):
38
- self.test = test
39
- self.create_cache_directory()
40
- self.cache = self.gen_cache()
41
- old_data = self.from_old_sqlite_cache()
42
- self.cache.add_from_dict(old_data)
43
-
44
- def create_cache_directory(self, notify=False) -> None:
45
- """
46
- Create the cache directory if one is required and it does not exist.
47
- """
48
- path = self.CACHE_PATH.replace("sqlite:///", "")
49
- dir_path = os.path.dirname(path)
50
- if dir_path and not os.path.exists(dir_path):
51
- os.makedirs(dir_path)
52
- if notify:
53
- print(f"Created cache directory: {dir_path}")
54
-
55
- def gen_cache(self) -> Cache:
56
- """
57
- Generate a Cache object.
58
- """
59
- if self.test:
60
- return Cache(data={})
61
-
62
- if hasattr(CONFIG, "EDSL_SESSION_CACHE"):
63
- return CONFIG.EDSL_SESSION_CACHE
64
-
65
- cache = Cache(data=SQLiteDict(self.CACHE_PATH))
66
- return cache
67
-
68
- def from_old_sqlite_cache(
69
- self, path: str = "edsl_cache.db"
70
- ) -> dict[str, CacheEntry]:
71
- """
72
- Convert an old-style cache to the new format.
73
- - NB: Not worth converting to sqlalchemy - this is a one-time operation.
74
- """
75
- old_data = {}
76
- if not os.path.exists(os.path.join(os.getcwd(), path)):
77
- return old_data
78
- try:
79
- conn = sqlite3.connect(path)
80
- with conn:
81
- cur = conn.cursor()
82
- table_name = "responses"
83
- cur.execute(f"PRAGMA table_info({table_name})")
84
- columns = cur.fetchall()
85
- schema = {column[1]: column[2] for column in columns}
86
- data = cur.execute(f"SELECT * FROM {table_name}").fetchall()
87
- for row in data:
88
- entry = self._parse_old_cache_entry(row, schema)
89
- old_data[entry.key] = entry
90
- print(
91
- f"Found old cache at {path} with {len(old_data)} entries.\n"
92
- f"We will convert this to the new cache format.\n"
93
- f"The old cache is backed up to {path}.bak"
94
- )
95
- shutil.copy(path, f"{path}.bak")
96
- os.remove(path)
97
- except sqlite3.OperationalError:
98
- print("Found an old Cache but could not convert it to new format.")
99
-
100
- return old_data
101
-
102
- def _parse_old_cache_entry(self, row: tuple, schema) -> CacheEntry:
103
- """
104
- Parse an old cache entry.
105
- """
106
- entry_dict = {k: row[i] for i, k in enumerate(schema.keys())}
107
- _ = entry_dict.pop("id")
108
- entry_dict["user_prompt"] = entry_dict.pop("prompt")
109
- parameters = entry_dict["parameters"]
110
- entry_dict["parameters"] = ast.literal_eval(parameters)
111
- entry = CacheEntry(**entry_dict)
112
- return entry
113
-
114
- def get_cache(self) -> Cache:
115
- return self.cache
116
-
117
- ###############
118
- # NOT IN USE
119
- ###############
120
- def from_sqlite(uri="new_edsl_cache.db") -> dict[str, CacheEntry]:
121
- """
122
- Read in a new-style sqlite cache and return a dictionary of dictionaries.
123
- """
124
- conn = sqlite3.connect(uri)
125
- with conn:
126
- cur = conn.cursor()
127
- data = cur.execute("SELECT key, value FROM data").fetchall()
128
- newdata = {}
129
- for _, value in data:
130
- entry = CacheEntry.from_dict(json.loads(value))
131
- newdata[entry.key] = entry
132
- return newdata
133
-
134
- def from_jsonl(filename="edsl_cache.jsonl") -> dict[str, CacheEntry]:
135
- """Read in a jsonl file and return a dictionary of CacheEntry objects."""
136
- with open(filename, "a+") as f:
137
- f.seek(0)
138
- lines = f.readlines()
139
- newdata = {}
140
- for line in lines:
141
- d = json.loads(line)
142
- key = list(d.keys())[0]
143
- value = list(d.values())[0]
144
- newdata[key] = CacheEntry.from_dict(value)
145
- return newdata
146
-
147
-
148
- if __name__ == "__main__":
149
- ch = CacheHandler()
1
+ from __future__ import annotations
2
+ import ast
3
+ import json
4
+ import os
5
+ import shutil
6
+ from typing import TYPE_CHECKING
7
+
8
+ if TYPE_CHECKING:
9
+ from edsl.data.Cache import Cache
10
+ from edsl.data.CacheEntry import CacheEntry
11
+
12
+
13
+ def set_session_cache(cache: "Cache") -> None:
14
+ """
15
+ Set the session cache.
16
+ """
17
+ from edsl.config import CONFIG
18
+
19
+ CONFIG.EDSL_SESSION_CACHE = cache
20
+
21
+
22
+ def unset_session_cache() -> None:
23
+ """
24
+ Unset the session cache.
25
+ """
26
+ from edsl.config import CONFIG
27
+
28
+ if hasattr(CONFIG, "EDSL_SESSION_CACHE"):
29
+ del CONFIG.EDSL_SESSION_CACHE
30
+
31
+
32
+ class CacheHandler:
33
+ """
34
+ This CacheHandler figures out what caches are available and does migrations, as needed.
35
+ """
36
+
37
+ @property
38
+ def CACHE_PATH(self):
39
+ from edsl.config import CONFIG
40
+
41
+ return CONFIG.get("EDSL_DATABASE_PATH")
42
+
43
+ def __init__(self, test: bool = False):
44
+ self.test = test
45
+ self.create_cache_directory()
46
+ self.cache = self.gen_cache()
47
+ old_data = self.from_old_sqlite_cache()
48
+ self.cache.add_from_dict(old_data)
49
+
50
+ def create_cache_directory(self, notify=False) -> None:
51
+ """
52
+ Create the cache directory if one is required and it does not exist.
53
+ """
54
+ path = self.CACHE_PATH.replace("sqlite:///", "")
55
+ dir_path = os.path.dirname(path)
56
+ if dir_path and not os.path.exists(dir_path):
57
+ os.makedirs(dir_path)
58
+ if notify:
59
+ print(f"Created cache directory: {dir_path}")
60
+
61
+ def gen_cache(self) -> "Cache":
62
+ """
63
+ Generate a Cache object.
64
+ """
65
+ from edsl.data.Cache import Cache
66
+
67
+ if self.test:
68
+ return Cache(data={})
69
+
70
+ from edsl.config import CONFIG
71
+
72
+ if hasattr(CONFIG, "EDSL_SESSION_CACHE"):
73
+ return CONFIG.EDSL_SESSION_CACHE
74
+
75
+ from edsl.data.SQLiteDict import SQLiteDict
76
+
77
+ cache = Cache(data=SQLiteDict(self.CACHE_PATH))
78
+ return cache
79
+
80
+ def from_old_sqlite_cache(
81
+ self, path: str = "edsl_cache.db"
82
+ ) -> dict[str, CacheEntry]:
83
+ """
84
+ Convert an old-style cache to the new format.
85
+ - NB: Not worth converting to sqlalchemy - this is a one-time operation.
86
+ """
87
+ old_data = {}
88
+ if not os.path.exists(os.path.join(os.getcwd(), path)):
89
+ return old_data
90
+ try:
91
+ import sqlite3
92
+
93
+ conn = sqlite3.connect(path)
94
+ with conn:
95
+ cur = conn.cursor()
96
+ table_name = "responses"
97
+ cur.execute(f"PRAGMA table_info({table_name})")
98
+ columns = cur.fetchall()
99
+ schema = {column[1]: column[2] for column in columns}
100
+ data = cur.execute(f"SELECT * FROM {table_name}").fetchall()
101
+ for row in data:
102
+ entry = self._parse_old_cache_entry(row, schema)
103
+ old_data[entry.key] = entry
104
+ print(
105
+ f"Found old cache at {path} with {len(old_data)} entries.\n"
106
+ f"We will convert this to the new cache format.\n"
107
+ f"The old cache is backed up to {path}.bak"
108
+ )
109
+ shutil.copy(path, f"{path}.bak")
110
+ os.remove(path)
111
+ except sqlite3.OperationalError:
112
+ print("Found an old Cache but could not convert it to new format.")
113
+
114
+ return old_data
115
+
116
+ def _parse_old_cache_entry(self, row: tuple, schema) -> CacheEntry:
117
+ """
118
+ Parse an old cache entry.
119
+ """
120
+ entry_dict = {k: row[i] for i, k in enumerate(schema.keys())}
121
+ _ = entry_dict.pop("id")
122
+ entry_dict["user_prompt"] = entry_dict.pop("prompt")
123
+ parameters = entry_dict["parameters"]
124
+ entry_dict["parameters"] = ast.literal_eval(parameters)
125
+ from edsl.data.CacheEntry import CacheEntry
126
+
127
+ entry = CacheEntry(**entry_dict)
128
+ return entry
129
+
130
+ def get_cache(self) -> Cache:
131
+ return self.cache
132
+
133
+ ###############
134
+ # NOT IN USE
135
+ ###############
136
+ def from_sqlite(uri="new_edsl_cache.db") -> dict[str, "CacheEntry"]:
137
+ """
138
+ Read in a new-style sqlite cache and return a dictionary of dictionaries.
139
+ """
140
+ conn = sqlite3.connect(uri)
141
+ with conn:
142
+ cur = conn.cursor()
143
+ data = cur.execute("SELECT key, value FROM data").fetchall()
144
+ newdata = {}
145
+ for _, value in data:
146
+ entry = CacheEntry.from_dict(json.loads(value))
147
+ newdata[entry.key] = entry
148
+ return newdata
149
+
150
+ def from_jsonl(filename="edsl_cache.jsonl") -> dict[str, "CacheEntry"]:
151
+ """Read in a jsonl file and return a dictionary of CacheEntry objects."""
152
+ with open(filename, "a+") as f:
153
+ f.seek(0)
154
+ lines = f.readlines()
155
+ newdata = {}
156
+ for line in lines:
157
+ d = json.loads(line)
158
+ key = list(d.keys())[0]
159
+ value = list(d.values())[0]
160
+ newdata[key] = CacheEntry.from_dict(value)
161
+ return newdata
162
+
163
+
164
+ if __name__ == "__main__":
165
+ # ch = CacheHandler()
166
+ import doctest
167
+
168
+ doctest.testmod()
@@ -1,78 +1,186 @@
1
- class RemoteCacheSync:
2
- def __init__(
3
- self, coop, cache, output_func, remote_cache=True, remote_cache_description=""
4
- ):
5
- self.coop = coop
6
- self.cache = cache
7
- self._output = output_func
8
- self.remote_cache = remote_cache
9
- self.old_entry_keys = []
10
- self.new_cache_entries = []
11
- self.remote_cache_description = remote_cache_description
12
-
13
- def __enter__(self):
14
- if self.remote_cache:
15
- self._sync_from_remote()
16
- self.old_entry_keys = list(self.cache.keys())
17
- return self
18
-
19
- def __exit__(self, exc_type, exc_value, traceback):
20
- if self.remote_cache:
21
- self._sync_to_remote()
22
- return False # Propagate exceptions
23
-
24
- def _sync_from_remote(self):
25
- cache_difference = self.coop.remote_cache_get_diff(self.cache.keys())
26
- client_missing_cacheentries = cache_difference.get(
27
- "client_missing_cacheentries", []
28
- )
29
- missing_entry_count = len(client_missing_cacheentries)
30
-
31
- if missing_entry_count > 0:
32
- self._output(
33
- f"Updating local cache with {missing_entry_count:,} new "
34
- f"{'entry' if missing_entry_count == 1 else 'entries'} from remote..."
35
- )
36
- self.cache.add_from_dict(
37
- {entry.key: entry for entry in client_missing_cacheentries}
38
- )
39
- self._output("Local cache updated!")
40
- else:
41
- self._output("No new entries to add to local cache.")
42
-
43
- def _sync_to_remote(self):
44
- cache_difference = self.coop.remote_cache_get_diff(self.cache.keys())
45
- server_missing_cacheentry_keys = cache_difference.get(
46
- "server_missing_cacheentry_keys", []
47
- )
48
- server_missing_cacheentries = [
49
- entry
50
- for key in server_missing_cacheentry_keys
51
- if (entry := self.cache.data.get(key)) is not None
52
- ]
53
-
54
- new_cache_entries = [
55
- entry
56
- for entry in self.cache.values()
57
- if entry.key not in self.old_entry_keys
58
- ]
59
- server_missing_cacheentries.extend(new_cache_entries)
60
- new_entry_count = len(server_missing_cacheentries)
61
-
62
- if new_entry_count > 0:
63
- self._output(
64
- f"Updating remote cache with {new_entry_count:,} new "
65
- f"{'entry' if new_entry_count == 1 else 'entries'}..."
66
- )
67
- self.coop.remote_cache_create_many(
68
- server_missing_cacheentries,
69
- visibility="private",
70
- description=self.remote_cache_description,
71
- )
72
- self._output("Remote cache updated!")
73
- else:
74
- self._output("No new entries to add to remote cache.")
75
-
76
- self._output(
77
- f"There are {len(self.cache.keys()):,} entries in the local cache."
78
- )
1
+ from typing import List, Dict, Any, Optional, TYPE_CHECKING, Callable
2
+ from dataclasses import dataclass
3
+ from contextlib import AbstractContextManager
4
+ from collections import UserList
5
+
6
+ if TYPE_CHECKING:
7
+ from .Cache import Cache
8
+ from edsl.coop.coop import Coop
9
+ from .CacheEntry import CacheEntry
10
+
11
+ from logging import Logger
12
+
13
+
14
+ class CacheKeyList(UserList):
15
+ def __init__(self, data: List[str]):
16
+ super().__init__(data)
17
+ self.data = data
18
+
19
+ def __repr__(self):
20
+ import reprlib
21
+
22
+ keys_repr = reprlib.repr(self.data)
23
+ return f"CacheKeyList({keys_repr})"
24
+
25
+
26
+ class CacheEntriesList(UserList):
27
+ def __init__(self, data: List["CacheEntry"]):
28
+ super().__init__(data)
29
+ self.data = data
30
+
31
+ def __repr__(self):
32
+ import reprlib
33
+
34
+ entries_repr = reprlib.repr(self.data)
35
+ return f"CacheEntries({entries_repr})"
36
+
37
+ def to_cache(self) -> "Cache":
38
+ from edsl.data.Cache import Cache
39
+
40
+ return Cache({entry.key: entry for entry in self.data})
41
+
42
+
43
+ @dataclass
44
+ class CacheDifference:
45
+ client_missing_entries: CacheEntriesList
46
+ server_missing_keys: List[str]
47
+
48
+ def __repr__(self):
49
+ """Returns a string representation of the CacheDifference object."""
50
+ import reprlib
51
+
52
+ missing_entries_repr = reprlib.repr(self.client_missing_entries)
53
+ missing_keys_repr = reprlib.repr(self.server_missing_keys)
54
+ return f"CacheDifference(client_missing_entries={missing_entries_repr}, server_missing_keys={missing_keys_repr})"
55
+
56
+
57
+ class RemoteCacheSync(AbstractContextManager):
58
+ """Synchronizes a local cache with a remote cache.
59
+
60
+ Handles bidirectional synchronization:
61
+ - Downloads missing entries from remote to local cache
62
+ - Uploads new local entries to remote cache
63
+ """
64
+
65
+ def __init__(
66
+ self,
67
+ coop: "Coop",
68
+ cache: "Cache",
69
+ output_func: Callable,
70
+ remote_cache: bool = True,
71
+ remote_cache_description: str = "",
72
+ ):
73
+ """
74
+ Initializes a RemoteCacheSync object.
75
+
76
+ :param coop: Coop object for interacting with the remote cache
77
+ :param cache: Cache object for local cache
78
+ :param output_func: Function for outputting messages
79
+ :param remote_cache: Whether to enable remote cache synchronization
80
+ :param remote_cache_description: Description for remote cache entries
81
+
82
+ """
83
+ self.coop = coop
84
+ self.cache = cache
85
+ self._output = output_func
86
+ self.remote_cache_enabled = remote_cache
87
+ self.remote_cache_description = remote_cache_description
88
+ self.initial_cache_keys = []
89
+
90
+ def __enter__(self) -> "RemoteCacheSync":
91
+ if self.remote_cache_enabled:
92
+ self._sync_from_remote()
93
+ self.initial_cache_keys = list(self.cache.keys())
94
+ return self
95
+
96
+ def __exit__(self, exc_type, exc_value, traceback):
97
+ if self.remote_cache_enabled:
98
+ self._sync_to_remote()
99
+ return False # Propagate exceptions
100
+
101
+ def _get_cache_difference(self) -> CacheDifference:
102
+ """Retrieves differences between local and remote caches."""
103
+ diff = self.coop.remote_cache_get_diff(self.cache.keys())
104
+ return CacheDifference(
105
+ client_missing_entries=diff.get("client_missing_cacheentries", []),
106
+ server_missing_keys=diff.get("server_missing_cacheentry_keys", []),
107
+ )
108
+
109
+ def _sync_from_remote(self) -> None:
110
+ """Downloads missing entries from remote cache to local cache."""
111
+ diff: CacheDifference = self._get_cache_difference()
112
+ missing_count = len(diff.client_missing_entries)
113
+
114
+ if missing_count == 0:
115
+ self._output("No new entries to add to local cache.")
116
+ return
117
+
118
+ self._output(
119
+ f"Updating local cache with {missing_count:,} new "
120
+ f"{'entry' if missing_count == 1 else 'entries'} from remote..."
121
+ )
122
+
123
+ self.cache.add_from_dict(
124
+ {entry.key: entry for entry in diff.client_missing_entries}
125
+ )
126
+ self._output("Local cache updated!")
127
+
128
+ def _get_entries_to_upload(self, diff: CacheDifference) -> CacheEntriesList:
129
+ """Determines which entries need to be uploaded to remote cache."""
130
+ # Get entries for keys missing from server
131
+ server_missing_entries = CacheEntriesList(
132
+ [
133
+ entry
134
+ for key in diff.server_missing_keys
135
+ if (entry := self.cache.data.get(key)) is not None
136
+ ]
137
+ )
138
+
139
+ # Get newly added entries since sync started
140
+ new_entries = CacheEntriesList(
141
+ [
142
+ entry
143
+ for entry in self.cache.values()
144
+ if entry.key not in self.initial_cache_keys
145
+ ]
146
+ )
147
+
148
+ return server_missing_entries + new_entries
149
+
150
+ def _sync_to_remote(self) -> None:
151
+ """Uploads new local entries to remote cache."""
152
+ diff: CacheDifference = self._get_cache_difference()
153
+ entries_to_upload: CacheEntriesList = self._get_entries_to_upload(diff)
154
+ upload_count = len(entries_to_upload)
155
+
156
+ if upload_count > 0:
157
+ self._output(
158
+ f"Updating remote cache with {upload_count:,} new "
159
+ f"{'entry' if upload_count == 1 else 'entries'}..."
160
+ )
161
+
162
+ self.coop.remote_cache_create_many(
163
+ entries_to_upload,
164
+ visibility="private",
165
+ description=self.remote_cache_description,
166
+ )
167
+ self._output("Remote cache updated!")
168
+ else:
169
+ self._output("No new entries to add to remote cache.")
170
+
171
+ self._output(
172
+ f"There are {len(self.cache.keys()):,} entries in the local cache."
173
+ )
174
+
175
+
176
+ if __name__ == "__main__":
177
+ import doctest
178
+
179
+ doctest.testmod()
180
+
181
+ from edsl.coop.coop import Coop
182
+ from edsl.data.Cache import Cache
183
+ from edsl.data.CacheEntry import CacheEntry
184
+
185
+ r = RemoteCacheSync(Coop(), Cache(), print)
186
+ diff = r._get_cache_difference()