edsl 0.1.38.dev3__py3-none-any.whl → 0.1.39__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 -303
  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 -858
  7. edsl/agents/AgentList.py +551 -362
  8. edsl/agents/Invigilator.py +284 -222
  9. edsl/agents/InvigilatorBase.py +257 -284
  10. edsl/agents/PromptConstructor.py +272 -353
  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 -149
  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 -961
  40. edsl/coop/utils.py +131 -131
  41. edsl/data/Cache.py +573 -530
  42. edsl/data/CacheEntry.py +230 -228
  43. edsl/data/CacheHandler.py +168 -149
  44. edsl/data/RemoteCacheSync.py +186 -97
  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 -173
  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 -156
  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 -0
  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 -39
  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 -1358
  94. edsl/jobs/JobsChecks.py +172 -0
  95. edsl/jobs/JobsComponentConstructor.py +189 -0
  96. edsl/jobs/JobsPrompts.py +270 -0
  97. edsl/jobs/JobsRemoteInferenceHandler.py +311 -0
  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 -361
  123. edsl/jobs/runners/JobsRunnerStatus.py +298 -332
  124. edsl/jobs/tasks/QuestionTaskCreator.py +244 -242
  125. edsl/jobs/tasks/TaskCreators.py +64 -64
  126. edsl/jobs/tasks/TaskHistory.py +470 -451
  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 -708
  133. edsl/language_models/ModelList.py +164 -109
  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 -357
  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 -660
  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 -183
  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 -147
  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 -0
  220. edsl/results/Dataset.py +587 -293
  221. edsl/results/DatasetExportMixin.py +594 -717
  222. edsl/results/DatasetTree.py +295 -145
  223. edsl/results/MarkdownToDocx.py +122 -0
  224. edsl/results/MarkdownToPDF.py +111 -0
  225. edsl/results/Result.py +557 -456
  226. edsl/results/Results.py +1183 -1071
  227. edsl/results/ResultsExportMixin.py +45 -43
  228. edsl/results/ResultsGGMixin.py +121 -121
  229. edsl/results/TableDisplay.py +125 -0
  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 +78 -0
  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 +543 -458
  245. edsl/scenarios/PdfExtractor.py +40 -0
  246. edsl/scenarios/Scenario.py +498 -544
  247. edsl/scenarios/ScenarioHtmlMixin.py +65 -64
  248. edsl/scenarios/ScenarioList.py +1458 -1112
  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 +49 -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/scenario_join.py +131 -0
  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 -1787
  288. edsl/surveys/SurveyCSS.py +273 -261
  289. edsl/surveys/SurveyExportMixin.py +259 -259
  290. edsl/surveys/{SurveyFlowVisualizationMixin.py → SurveyFlowVisualization.py} +181 -121
  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 -53
  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 -10
  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 -409
  333. {edsl-0.1.38.dev3.dist-info → edsl-0.1.39.dist-info}/LICENSE +21 -21
  334. {edsl-0.1.38.dev3.dist-info → edsl-0.1.39.dist-info}/METADATA +13 -10
  335. edsl-0.1.39.dist-info/RECORD +358 -0
  336. {edsl-0.1.38.dev3.dist-info → edsl-0.1.39.dist-info}/WHEEL +1 -1
  337. edsl/language_models/KeyLookup.py +0 -30
  338. edsl/language_models/registry.py +0 -137
  339. edsl/language_models/unused/ReplicateBase.py +0 -83
  340. edsl/results/ResultsDBMixin.py +0 -238
  341. edsl-0.1.38.dev3.dist-info/RECORD +0 -269
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,97 +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
- )
79
-
80
-
81
- # # Usage example
82
- # def run_job(self, n, progress_bar, cache, stop_on_exception, sidecar_model, print_exceptions, raise_validation_errors, use_remote_cache=True):
83
- # with RemoteCacheSync(self.coop, cache, self._output, remote_cache=use_remote_cache):
84
- # self._output("Running job...")
85
- # results = self._run_local(
86
- # n=n,
87
- # progress_bar=progress_bar,
88
- # cache=cache,
89
- # stop_on_exception=stop_on_exception,
90
- # sidecar_model=sidecar_model,
91
- # print_exceptions=print_exceptions,
92
- # raise_validation_errors=raise_validation_errors,
93
- # )
94
- # self._output("Job completed!")
95
-
96
- # results.cache = cache.new_entries_cache()
97
- # return results
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()