edsl 0.1.39.dev2__py3-none-any.whl → 0.1.39.dev3__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 (334) hide show
  1. edsl/Base.py +332 -385
  2. edsl/BaseDiff.py +260 -260
  3. edsl/TemplateLoader.py +24 -24
  4. edsl/__init__.py +49 -57
  5. edsl/__version__.py +1 -1
  6. edsl/agents/Agent.py +867 -1079
  7. edsl/agents/AgentList.py +413 -551
  8. edsl/agents/Invigilator.py +233 -285
  9. edsl/agents/InvigilatorBase.py +270 -254
  10. edsl/agents/PromptConstructor.py +354 -252
  11. edsl/agents/__init__.py +3 -2
  12. edsl/agents/descriptors.py +99 -99
  13. edsl/agents/prompt_helpers.py +129 -129
  14. edsl/auto/AutoStudy.py +117 -117
  15. edsl/auto/StageBase.py +230 -230
  16. edsl/auto/StageGenerateSurvey.py +178 -178
  17. edsl/auto/StageLabelQuestions.py +125 -125
  18. edsl/auto/StagePersona.py +61 -61
  19. edsl/auto/StagePersonaDimensionValueRanges.py +88 -88
  20. edsl/auto/StagePersonaDimensionValues.py +74 -74
  21. edsl/auto/StagePersonaDimensions.py +69 -69
  22. edsl/auto/StageQuestions.py +73 -73
  23. edsl/auto/SurveyCreatorPipeline.py +21 -21
  24. edsl/auto/utilities.py +224 -224
  25. edsl/base/Base.py +279 -279
  26. edsl/config.py +157 -177
  27. edsl/conversation/Conversation.py +290 -290
  28. edsl/conversation/car_buying.py +58 -59
  29. edsl/conversation/chips.py +95 -95
  30. edsl/conversation/mug_negotiation.py +81 -81
  31. edsl/conversation/next_speaker_utilities.py +93 -93
  32. edsl/coop/PriceFetcher.py +54 -54
  33. edsl/coop/__init__.py +2 -2
  34. edsl/coop/coop.py +1028 -1090
  35. edsl/coop/utils.py +131 -131
  36. edsl/data/Cache.py +555 -562
  37. edsl/data/CacheEntry.py +233 -230
  38. edsl/data/CacheHandler.py +149 -170
  39. edsl/data/RemoteCacheSync.py +78 -78
  40. edsl/data/SQLiteDict.py +292 -292
  41. edsl/data/__init__.py +4 -5
  42. edsl/data/orm.py +10 -10
  43. edsl/data_transfer_models.py +73 -74
  44. edsl/enums.py +175 -195
  45. edsl/exceptions/BaseException.py +21 -21
  46. edsl/exceptions/__init__.py +54 -54
  47. edsl/exceptions/agents.py +42 -54
  48. edsl/exceptions/cache.py +5 -5
  49. edsl/exceptions/configuration.py +16 -16
  50. edsl/exceptions/coop.py +10 -10
  51. edsl/exceptions/data.py +14 -14
  52. edsl/exceptions/general.py +34 -34
  53. edsl/exceptions/jobs.py +33 -33
  54. edsl/exceptions/language_models.py +63 -63
  55. edsl/exceptions/prompts.py +15 -15
  56. edsl/exceptions/questions.py +91 -109
  57. edsl/exceptions/results.py +29 -29
  58. edsl/exceptions/scenarios.py +22 -29
  59. edsl/exceptions/surveys.py +37 -37
  60. edsl/inference_services/AnthropicService.py +87 -84
  61. edsl/inference_services/AwsBedrock.py +120 -118
  62. edsl/inference_services/AzureAI.py +217 -215
  63. edsl/inference_services/DeepInfraService.py +18 -18
  64. edsl/inference_services/GoogleService.py +148 -139
  65. edsl/inference_services/GroqService.py +20 -20
  66. edsl/inference_services/InferenceServiceABC.py +147 -80
  67. edsl/inference_services/InferenceServicesCollection.py +97 -122
  68. edsl/inference_services/MistralAIService.py +123 -120
  69. edsl/inference_services/OllamaService.py +18 -18
  70. edsl/inference_services/OpenAIService.py +224 -221
  71. edsl/inference_services/PerplexityService.py +163 -160
  72. edsl/inference_services/TestService.py +89 -92
  73. edsl/inference_services/TogetherAIService.py +170 -170
  74. edsl/inference_services/models_available_cache.py +118 -118
  75. edsl/inference_services/rate_limits_cache.py +25 -25
  76. edsl/inference_services/registry.py +41 -41
  77. edsl/inference_services/write_available.py +10 -10
  78. edsl/jobs/Answers.py +56 -43
  79. edsl/jobs/Jobs.py +898 -757
  80. edsl/jobs/JobsChecks.py +147 -172
  81. edsl/jobs/JobsPrompts.py +268 -270
  82. edsl/jobs/JobsRemoteInferenceHandler.py +239 -287
  83. edsl/jobs/__init__.py +1 -1
  84. edsl/jobs/buckets/BucketCollection.py +63 -104
  85. edsl/jobs/buckets/ModelBuckets.py +65 -65
  86. edsl/jobs/buckets/TokenBucket.py +251 -283
  87. edsl/jobs/interviews/Interview.py +661 -358
  88. edsl/jobs/interviews/InterviewExceptionCollection.py +99 -99
  89. edsl/jobs/interviews/InterviewExceptionEntry.py +186 -186
  90. edsl/jobs/interviews/InterviewStatistic.py +63 -63
  91. edsl/jobs/interviews/InterviewStatisticsCollection.py +25 -25
  92. edsl/jobs/interviews/InterviewStatusDictionary.py +78 -78
  93. edsl/jobs/interviews/InterviewStatusLog.py +92 -92
  94. edsl/jobs/interviews/ReportErrors.py +66 -66
  95. edsl/jobs/interviews/interview_status_enum.py +9 -9
  96. edsl/jobs/runners/JobsRunnerAsyncio.py +466 -421
  97. edsl/jobs/runners/JobsRunnerStatus.py +330 -330
  98. edsl/jobs/tasks/QuestionTaskCreator.py +242 -244
  99. edsl/jobs/tasks/TaskCreators.py +64 -64
  100. edsl/jobs/tasks/TaskHistory.py +450 -449
  101. edsl/jobs/tasks/TaskStatusLog.py +23 -23
  102. edsl/jobs/tasks/task_status_enum.py +163 -161
  103. edsl/jobs/tokens/InterviewTokenUsage.py +27 -27
  104. edsl/jobs/tokens/TokenUsage.py +34 -34
  105. edsl/language_models/KeyLookup.py +30 -0
  106. edsl/language_models/LanguageModel.py +668 -571
  107. edsl/language_models/ModelList.py +155 -153
  108. edsl/language_models/RegisterLanguageModelsMeta.py +184 -184
  109. edsl/language_models/__init__.py +3 -2
  110. edsl/language_models/fake_openai_call.py +15 -15
  111. edsl/language_models/fake_openai_service.py +61 -61
  112. edsl/language_models/registry.py +190 -180
  113. edsl/language_models/repair.py +156 -156
  114. edsl/language_models/unused/ReplicateBase.py +83 -0
  115. edsl/language_models/utilities.py +64 -65
  116. edsl/notebooks/Notebook.py +258 -263
  117. edsl/notebooks/__init__.py +1 -1
  118. edsl/prompts/Prompt.py +362 -352
  119. edsl/prompts/__init__.py +2 -2
  120. edsl/questions/AnswerValidatorMixin.py +289 -334
  121. edsl/questions/QuestionBase.py +664 -509
  122. edsl/questions/QuestionBaseGenMixin.py +161 -165
  123. edsl/questions/QuestionBasePromptsMixin.py +217 -221
  124. edsl/questions/QuestionBudget.py +227 -227
  125. edsl/questions/QuestionCheckBox.py +359 -359
  126. edsl/questions/QuestionExtract.py +182 -182
  127. edsl/questions/QuestionFreeText.py +114 -113
  128. edsl/questions/QuestionFunctional.py +166 -166
  129. edsl/questions/QuestionList.py +231 -229
  130. edsl/questions/QuestionMultipleChoice.py +286 -330
  131. edsl/questions/QuestionNumerical.py +153 -151
  132. edsl/questions/QuestionRank.py +324 -314
  133. edsl/questions/Quick.py +41 -41
  134. edsl/questions/RegisterQuestionsMeta.py +71 -71
  135. edsl/questions/ResponseValidatorABC.py +174 -200
  136. edsl/questions/SimpleAskMixin.py +73 -74
  137. edsl/questions/__init__.py +26 -27
  138. edsl/questions/compose_questions.py +98 -98
  139. edsl/questions/decorators.py +21 -21
  140. edsl/questions/derived/QuestionLikertFive.py +76 -76
  141. edsl/questions/derived/QuestionLinearScale.py +87 -90
  142. edsl/questions/derived/QuestionTopK.py +93 -93
  143. edsl/questions/derived/QuestionYesNo.py +82 -82
  144. edsl/questions/descriptors.py +413 -427
  145. edsl/questions/prompt_templates/question_budget.jinja +13 -13
  146. edsl/questions/prompt_templates/question_checkbox.jinja +32 -32
  147. edsl/questions/prompt_templates/question_extract.jinja +11 -11
  148. edsl/questions/prompt_templates/question_free_text.jinja +3 -3
  149. edsl/questions/prompt_templates/question_linear_scale.jinja +11 -11
  150. edsl/questions/prompt_templates/question_list.jinja +17 -17
  151. edsl/questions/prompt_templates/question_multiple_choice.jinja +33 -33
  152. edsl/questions/prompt_templates/question_numerical.jinja +36 -36
  153. edsl/questions/question_registry.py +177 -177
  154. edsl/questions/settings.py +12 -12
  155. edsl/questions/templates/budget/answering_instructions.jinja +7 -7
  156. edsl/questions/templates/budget/question_presentation.jinja +7 -7
  157. edsl/questions/templates/checkbox/answering_instructions.jinja +10 -10
  158. edsl/questions/templates/checkbox/question_presentation.jinja +22 -22
  159. edsl/questions/templates/extract/answering_instructions.jinja +7 -7
  160. edsl/questions/templates/likert_five/answering_instructions.jinja +10 -10
  161. edsl/questions/templates/likert_five/question_presentation.jinja +11 -11
  162. edsl/questions/templates/linear_scale/answering_instructions.jinja +5 -5
  163. edsl/questions/templates/linear_scale/question_presentation.jinja +5 -5
  164. edsl/questions/templates/list/answering_instructions.jinja +3 -3
  165. edsl/questions/templates/list/question_presentation.jinja +5 -5
  166. edsl/questions/templates/multiple_choice/answering_instructions.jinja +9 -9
  167. edsl/questions/templates/multiple_choice/question_presentation.jinja +11 -11
  168. edsl/questions/templates/numerical/answering_instructions.jinja +6 -6
  169. edsl/questions/templates/numerical/question_presentation.jinja +6 -6
  170. edsl/questions/templates/rank/answering_instructions.jinja +11 -11
  171. edsl/questions/templates/rank/question_presentation.jinja +15 -15
  172. edsl/questions/templates/top_k/answering_instructions.jinja +8 -8
  173. edsl/questions/templates/top_k/question_presentation.jinja +22 -22
  174. edsl/questions/templates/yes_no/answering_instructions.jinja +6 -6
  175. edsl/questions/templates/yes_no/question_presentation.jinja +11 -11
  176. edsl/results/CSSParameterizer.py +108 -108
  177. edsl/results/Dataset.py +424 -587
  178. edsl/results/DatasetExportMixin.py +731 -653
  179. edsl/results/DatasetTree.py +275 -295
  180. edsl/results/Result.py +465 -451
  181. edsl/results/Results.py +1165 -1172
  182. edsl/results/ResultsDBMixin.py +238 -0
  183. edsl/results/ResultsExportMixin.py +43 -45
  184. edsl/results/ResultsFetchMixin.py +33 -33
  185. edsl/results/ResultsGGMixin.py +121 -121
  186. edsl/results/ResultsToolsMixin.py +98 -98
  187. edsl/results/Selector.py +135 -145
  188. edsl/results/TableDisplay.py +198 -125
  189. edsl/results/__init__.py +2 -2
  190. edsl/results/table_display.css +77 -77
  191. edsl/results/tree_explore.py +115 -115
  192. edsl/scenarios/FileStore.py +632 -511
  193. edsl/scenarios/Scenario.py +601 -498
  194. edsl/scenarios/ScenarioHtmlMixin.py +64 -65
  195. edsl/scenarios/ScenarioJoin.py +127 -131
  196. edsl/scenarios/ScenarioList.py +1287 -1430
  197. edsl/scenarios/ScenarioListExportMixin.py +52 -45
  198. edsl/scenarios/ScenarioListPdfMixin.py +261 -239
  199. edsl/scenarios/__init__.py +4 -3
  200. edsl/shared.py +1 -1
  201. edsl/study/ObjectEntry.py +173 -173
  202. edsl/study/ProofOfWork.py +113 -113
  203. edsl/study/SnapShot.py +80 -80
  204. edsl/study/Study.py +528 -521
  205. edsl/study/__init__.py +4 -4
  206. edsl/surveys/DAG.py +148 -148
  207. edsl/surveys/Memory.py +31 -31
  208. edsl/surveys/MemoryPlan.py +244 -244
  209. edsl/surveys/Rule.py +326 -327
  210. edsl/surveys/RuleCollection.py +387 -385
  211. edsl/surveys/Survey.py +1801 -1229
  212. edsl/surveys/SurveyCSS.py +261 -273
  213. edsl/surveys/SurveyExportMixin.py +259 -259
  214. edsl/surveys/{SurveyFlowVisualization.py → SurveyFlowVisualizationMixin.py} +179 -181
  215. edsl/surveys/SurveyQualtricsImport.py +284 -284
  216. edsl/surveys/__init__.py +3 -5
  217. edsl/surveys/base.py +53 -53
  218. edsl/surveys/descriptors.py +56 -60
  219. edsl/surveys/instructions/ChangeInstruction.py +49 -48
  220. edsl/surveys/instructions/Instruction.py +65 -56
  221. edsl/surveys/instructions/InstructionCollection.py +77 -82
  222. edsl/templates/error_reporting/base.html +23 -23
  223. edsl/templates/error_reporting/exceptions_by_model.html +34 -34
  224. edsl/templates/error_reporting/exceptions_by_question_name.html +16 -16
  225. edsl/templates/error_reporting/exceptions_by_type.html +16 -16
  226. edsl/templates/error_reporting/interview_details.html +115 -115
  227. edsl/templates/error_reporting/interviews.html +19 -19
  228. edsl/templates/error_reporting/overview.html +4 -4
  229. edsl/templates/error_reporting/performance_plot.html +1 -1
  230. edsl/templates/error_reporting/report.css +73 -73
  231. edsl/templates/error_reporting/report.html +117 -117
  232. edsl/templates/error_reporting/report.js +25 -25
  233. edsl/tools/__init__.py +1 -1
  234. edsl/tools/clusters.py +192 -192
  235. edsl/tools/embeddings.py +27 -27
  236. edsl/tools/embeddings_plotting.py +118 -118
  237. edsl/tools/plotting.py +112 -112
  238. edsl/tools/summarize.py +18 -18
  239. edsl/utilities/SystemInfo.py +28 -28
  240. edsl/utilities/__init__.py +22 -22
  241. edsl/utilities/ast_utilities.py +25 -25
  242. edsl/utilities/data/Registry.py +6 -6
  243. edsl/utilities/data/__init__.py +1 -1
  244. edsl/utilities/data/scooter_results.json +1 -1
  245. edsl/utilities/decorators.py +77 -77
  246. edsl/utilities/gcp_bucket/cloud_storage.py +96 -96
  247. edsl/utilities/interface.py +627 -627
  248. edsl/utilities/naming_utilities.py +263 -263
  249. edsl/utilities/repair_functions.py +28 -28
  250. edsl/utilities/restricted_python.py +70 -70
  251. edsl/utilities/utilities.py +424 -436
  252. {edsl-0.1.39.dev2.dist-info → edsl-0.1.39.dev3.dist-info}/LICENSE +21 -21
  253. {edsl-0.1.39.dev2.dist-info → edsl-0.1.39.dev3.dist-info}/METADATA +10 -12
  254. edsl-0.1.39.dev3.dist-info/RECORD +277 -0
  255. edsl/agents/QuestionInstructionPromptBuilder.py +0 -128
  256. edsl/agents/QuestionOptionProcessor.py +0 -172
  257. edsl/agents/QuestionTemplateReplacementsBuilder.py +0 -137
  258. edsl/coop/CoopFunctionsMixin.py +0 -15
  259. edsl/coop/ExpectedParrotKeyHandler.py +0 -125
  260. edsl/exceptions/inference_services.py +0 -5
  261. edsl/inference_services/AvailableModelCacheHandler.py +0 -184
  262. edsl/inference_services/AvailableModelFetcher.py +0 -209
  263. edsl/inference_services/ServiceAvailability.py +0 -135
  264. edsl/inference_services/data_structures.py +0 -62
  265. edsl/jobs/AnswerQuestionFunctionConstructor.py +0 -188
  266. edsl/jobs/FetchInvigilator.py +0 -40
  267. edsl/jobs/InterviewTaskManager.py +0 -98
  268. edsl/jobs/InterviewsConstructor.py +0 -48
  269. edsl/jobs/JobsComponentConstructor.py +0 -189
  270. edsl/jobs/JobsRemoteInferenceLogger.py +0 -239
  271. edsl/jobs/RequestTokenEstimator.py +0 -30
  272. edsl/jobs/buckets/TokenBucketAPI.py +0 -211
  273. edsl/jobs/buckets/TokenBucketClient.py +0 -191
  274. edsl/jobs/decorators.py +0 -35
  275. edsl/jobs/jobs_status_enums.py +0 -9
  276. edsl/jobs/loggers/HTMLTableJobLogger.py +0 -304
  277. edsl/language_models/ComputeCost.py +0 -63
  278. edsl/language_models/PriceManager.py +0 -127
  279. edsl/language_models/RawResponseHandler.py +0 -106
  280. edsl/language_models/ServiceDataSources.py +0 -0
  281. edsl/language_models/key_management/KeyLookup.py +0 -63
  282. edsl/language_models/key_management/KeyLookupBuilder.py +0 -273
  283. edsl/language_models/key_management/KeyLookupCollection.py +0 -38
  284. edsl/language_models/key_management/__init__.py +0 -0
  285. edsl/language_models/key_management/models.py +0 -131
  286. edsl/notebooks/NotebookToLaTeX.py +0 -142
  287. edsl/questions/ExceptionExplainer.py +0 -77
  288. edsl/questions/HTMLQuestion.py +0 -103
  289. edsl/questions/LoopProcessor.py +0 -149
  290. edsl/questions/QuestionMatrix.py +0 -265
  291. edsl/questions/ResponseValidatorFactory.py +0 -28
  292. edsl/questions/templates/matrix/__init__.py +0 -1
  293. edsl/questions/templates/matrix/answering_instructions.jinja +0 -5
  294. edsl/questions/templates/matrix/question_presentation.jinja +0 -20
  295. edsl/results/MarkdownToDocx.py +0 -122
  296. edsl/results/MarkdownToPDF.py +0 -111
  297. edsl/results/TextEditor.py +0 -50
  298. edsl/results/smart_objects.py +0 -96
  299. edsl/results/table_data_class.py +0 -12
  300. edsl/results/table_renderers.py +0 -118
  301. edsl/scenarios/ConstructDownloadLink.py +0 -109
  302. edsl/scenarios/DirectoryScanner.py +0 -96
  303. edsl/scenarios/DocumentChunker.py +0 -102
  304. edsl/scenarios/DocxScenario.py +0 -16
  305. edsl/scenarios/PdfExtractor.py +0 -40
  306. edsl/scenarios/ScenarioSelector.py +0 -156
  307. edsl/scenarios/file_methods.py +0 -85
  308. edsl/scenarios/handlers/__init__.py +0 -13
  309. edsl/scenarios/handlers/csv.py +0 -38
  310. edsl/scenarios/handlers/docx.py +0 -76
  311. edsl/scenarios/handlers/html.py +0 -37
  312. edsl/scenarios/handlers/json.py +0 -111
  313. edsl/scenarios/handlers/latex.py +0 -5
  314. edsl/scenarios/handlers/md.py +0 -51
  315. edsl/scenarios/handlers/pdf.py +0 -68
  316. edsl/scenarios/handlers/png.py +0 -39
  317. edsl/scenarios/handlers/pptx.py +0 -105
  318. edsl/scenarios/handlers/py.py +0 -294
  319. edsl/scenarios/handlers/sql.py +0 -313
  320. edsl/scenarios/handlers/sqlite.py +0 -149
  321. edsl/scenarios/handlers/txt.py +0 -33
  322. edsl/surveys/ConstructDAG.py +0 -92
  323. edsl/surveys/EditSurvey.py +0 -221
  324. edsl/surveys/InstructionHandler.py +0 -100
  325. edsl/surveys/MemoryManagement.py +0 -72
  326. edsl/surveys/RuleManager.py +0 -172
  327. edsl/surveys/Simulator.py +0 -75
  328. edsl/surveys/SurveyToApp.py +0 -141
  329. edsl/utilities/PrettyList.py +0 -56
  330. edsl/utilities/is_notebook.py +0 -18
  331. edsl/utilities/is_valid_variable_name.py +0 -11
  332. edsl/utilities/remove_edsl_version.py +0 -24
  333. edsl-0.1.39.dev2.dist-info/RECORD +0 -352
  334. {edsl-0.1.39.dev2.dist-info → edsl-0.1.39.dev3.dist-info}/WHEEL +0 -0
@@ -1,221 +1,217 @@
1
- from importlib import resources
2
- from typing import Optional
3
- from edsl.exceptions.questions import QuestionAnswerValidationError
4
- from functools import lru_cache
5
-
6
-
7
- class TemplateManager:
8
- _instance = None
9
-
10
- def __new__(cls):
11
- if cls._instance is None:
12
- cls._instance = super().__new__(cls)
13
- cls._instance._template_cache = {}
14
- return cls._instance
15
-
16
- @lru_cache(maxsize=None)
17
- def get_template(self, question_type, template_name):
18
- if (question_type, template_name) not in self._template_cache:
19
- with resources.open_text(
20
- f"edsl.questions.templates.{question_type}", template_name
21
- ) as file:
22
- self._template_cache[(question_type, template_name)] = file.read()
23
- return self._template_cache[(question_type, template_name)]
24
-
25
-
26
- # Global instance
27
- template_manager = TemplateManager()
28
-
29
-
30
- class QuestionBasePromptsMixin:
31
- @property
32
- def model_instructions(self) -> dict:
33
- """Get the model-specific instructions for the question."""
34
- if not hasattr(self, "_model_instructions"):
35
- self._model_instructions = {}
36
- return self._model_instructions
37
-
38
- def _all_text(self) -> str:
39
- """Return the question text.
40
-
41
- >>> from edsl import QuestionMultipleChoice as Q
42
- >>> Q.example()._all_text()
43
- "how_feelingHow are you?['Good', 'Great', 'OK', 'Bad']"
44
- """
45
- txt = ""
46
- for key, value in self.data.items():
47
- if isinstance(value, str):
48
- txt += value
49
- elif isinstance(value, list):
50
- txt += "".join(str(value))
51
- return txt
52
-
53
- @model_instructions.setter
54
- def model_instructions(self, data: dict):
55
- """Set the model-specific instructions for the question."""
56
- self._model_instructions = data
57
-
58
- def add_model_instructions(
59
- self, *, instructions: str, model: Optional[str] = None
60
- ) -> None:
61
- """Add model-specific instructions for the question that override the default instructions.
62
-
63
- :param instructions: The instructions to add. This is typically a jinja2 template.
64
- :param model: The language model for this instruction.
65
-
66
- >>> from edsl.questions import QuestionFreeText
67
- >>> q = QuestionFreeText(question_name = "color", question_text = "What is your favorite color?")
68
- >>> q.add_model_instructions(instructions = "{{question_text}}. Answer in valid JSON like so {'answer': 'comment: <>}", model = "gpt3")
69
- >>> q.get_instructions(model = "gpt3")
70
- Prompt(text=\"""{{question_text}}. Answer in valid JSON like so {'answer': 'comment: <>}\""")
71
- """
72
- from edsl.language_models.registry import Model
73
-
74
- if not hasattr(self, "_model_instructions"):
75
- self._model_instructions = {}
76
- if model is None:
77
- # if not model is passed, all the models are mapped to this instruction, including 'None'
78
- self._model_instructions = {
79
- model_name: instructions
80
- for model_name in Model.available(name_only=True)
81
- }
82
- self._model_instructions.update({model: instructions})
83
- else:
84
- self._model_instructions.update({model: instructions})
85
-
86
- @classmethod
87
- def path_to_folder(cls) -> str:
88
- return resources.files(f"edsl.questions.templates", cls.question_type)
89
-
90
- @property
91
- def response_model(self) -> type["BaseModel"]:
92
- if self._response_model is not None:
93
- return self._response_model
94
- else:
95
- return self.create_response_model()
96
-
97
- @property
98
- def use_code(self) -> bool:
99
- if hasattr(self, "_use_code"):
100
- return self._use_code
101
- return True
102
-
103
- @use_code.setter
104
- def use_code(self, value: bool) -> None:
105
- self._use_code = value
106
-
107
- @property
108
- def include_comment(self) -> bool:
109
- if hasattr(self, "_include_comment"):
110
- return self._include_comment
111
- return True
112
-
113
- @include_comment.setter
114
- def include_comment(self, value: bool) -> None:
115
- self._include_comment = value
116
-
117
- @classmethod
118
- def default_answering_instructions(cls) -> str:
119
- # template_text = cls._read_template("answering_instructions.jinja")
120
- template_text = template_manager.get_template(
121
- cls.question_type, "answering_instructions.jinja"
122
- )
123
- from edsl.prompts import Prompt
124
-
125
- return Prompt(text=template_text)
126
-
127
- @classmethod
128
- def default_question_presentation(cls):
129
- template_text = template_manager.get_template(
130
- cls.question_type, "question_presentation.jinja"
131
- )
132
- from edsl.prompts import Prompt
133
-
134
- return Prompt(text=template_text)
135
-
136
- @property
137
- def answering_instructions(self) -> str:
138
- if self._answering_instructions is None:
139
- return self.default_answering_instructions()
140
- return self._answering_instructions
141
-
142
- @answering_instructions.setter
143
- def answering_instructions(self, value) -> None:
144
- self._answering_instructions = value
145
-
146
- @property
147
- def question_presentation(self):
148
- if self._question_presentation is None:
149
- return self.default_question_presentation()
150
- return self._question_presentation
151
-
152
- @question_presentation.setter
153
- def question_presentation(self, value):
154
- self._question_presentation = value
155
-
156
- def prompt_preview(self, scenario=None, agent=None):
157
- return self.new_default_instructions.render(
158
- self.data
159
- | {
160
- "include_comment": getattr(self, "_include_comment", True),
161
- "use_code": getattr(self, "_use_code", True),
162
- }
163
- | ({"scenario": scenario} or {})
164
- | ({"agent": agent} or {})
165
- )
166
-
167
- @classmethod
168
- def self_check(cls):
169
- q = cls.example()
170
- for answer, params in q.response_validator.valid_examples:
171
- for key, value in params.items():
172
- setattr(q, key, value)
173
- q._validate_answer(answer)
174
- for answer, params, reason in q.response_validator.invalid_examples:
175
- for key, value in params.items():
176
- setattr(q, key, value)
177
- try:
178
- q._validate_answer(answer)
179
- except QuestionAnswerValidationError:
180
- pass
181
- else:
182
- raise ValueError(f"Example {answer} should have failed for {reason}.")
183
-
184
- @property
185
- def new_default_instructions(self) -> "Prompt":
186
- "This is set up as a property because there are mutable question values that determine how it is rendered."
187
- from edsl.prompts import Prompt
188
-
189
- return Prompt(self.question_presentation) + Prompt(self.answering_instructions)
190
-
191
- @property
192
- def parameters(self) -> set[str]:
193
- """Return the parameters of the question."""
194
- from jinja2 import Environment, meta
195
-
196
- env = Environment()
197
- # Parse the template
198
- txt = self._all_text()
199
- # txt = self.question_text
200
- # if hasattr(self, "question_options"):
201
- # txt += " ".join(self.question_options)
202
- parsed_content = env.parse(txt)
203
- # Extract undeclared variables
204
- variables = meta.find_undeclared_variables(parsed_content)
205
- # Return as a list
206
- return set(variables)
207
-
208
- def get_instructions(self, model: Optional[str] = None) -> type["PromptBase"]:
209
- """Get the mathcing question-answering instructions for the question.
210
-
211
- :param model: The language model to use.
212
- """
213
- from edsl.prompts.Prompt import Prompt
214
-
215
- if model in self.model_instructions:
216
- return Prompt(text=self.model_instructions[model])
217
- else:
218
- if hasattr(self, "new_default_instructions"):
219
- return self.new_default_instructions
220
- else:
221
- return self.applicable_prompts(model)[0]()
1
+ from importlib import resources
2
+ from typing import Optional
3
+ from edsl.prompts import Prompt
4
+ from edsl.exceptions.questions import QuestionAnswerValidationError
5
+
6
+ from functools import lru_cache
7
+
8
+
9
+ class TemplateManager:
10
+ _instance = None
11
+
12
+ def __new__(cls):
13
+ if cls._instance is None:
14
+ cls._instance = super().__new__(cls)
15
+ cls._instance._template_cache = {}
16
+ return cls._instance
17
+
18
+ @lru_cache(maxsize=None)
19
+ def get_template(self, question_type, template_name):
20
+ if (question_type, template_name) not in self._template_cache:
21
+ with resources.open_text(
22
+ f"edsl.questions.templates.{question_type}", template_name
23
+ ) as file:
24
+ self._template_cache[(question_type, template_name)] = file.read()
25
+ return self._template_cache[(question_type, template_name)]
26
+
27
+
28
+ # Global instance
29
+ template_manager = TemplateManager()
30
+
31
+
32
+ class QuestionBasePromptsMixin:
33
+ @property
34
+ def model_instructions(self) -> dict:
35
+ """Get the model-specific instructions for the question."""
36
+ if not hasattr(self, "_model_instructions"):
37
+ self._model_instructions = {}
38
+ return self._model_instructions
39
+
40
+ def _all_text(self) -> str:
41
+ """Return the question text.
42
+
43
+ >>> from edsl import QuestionMultipleChoice as Q
44
+ >>> Q.example()._all_text()
45
+ "how_feelingHow are you?['Good', 'Great', 'OK', 'Bad']"
46
+ """
47
+ txt = ""
48
+ for key, value in self.data.items():
49
+ if isinstance(value, str):
50
+ txt += value
51
+ elif isinstance(value, list):
52
+ txt += "".join(str(value))
53
+ return txt
54
+
55
+ @model_instructions.setter
56
+ def model_instructions(self, data: dict):
57
+ """Set the model-specific instructions for the question."""
58
+ self._model_instructions = data
59
+
60
+ def add_model_instructions(
61
+ self, *, instructions: str, model: Optional[str] = None
62
+ ) -> None:
63
+ """Add model-specific instructions for the question that override the default instructions.
64
+
65
+ :param instructions: The instructions to add. This is typically a jinja2 template.
66
+ :param model: The language model for this instruction.
67
+
68
+ >>> from edsl.questions import QuestionFreeText
69
+ >>> q = QuestionFreeText(question_name = "color", question_text = "What is your favorite color?")
70
+ >>> q.add_model_instructions(instructions = "{{question_text}}. Answer in valid JSON like so {'answer': 'comment: <>}", model = "gpt3")
71
+ >>> q.get_instructions(model = "gpt3")
72
+ Prompt(text=\"""{{question_text}}. Answer in valid JSON like so {'answer': 'comment: <>}\""")
73
+ """
74
+ from edsl import Model
75
+
76
+ if not hasattr(self, "_model_instructions"):
77
+ self._model_instructions = {}
78
+ if model is None:
79
+ # if not model is passed, all the models are mapped to this instruction, including 'None'
80
+ self._model_instructions = {
81
+ model_name: instructions
82
+ for model_name in Model.available(name_only=True)
83
+ }
84
+ self._model_instructions.update({model: instructions})
85
+ else:
86
+ self._model_instructions.update({model: instructions})
87
+
88
+ @classmethod
89
+ def path_to_folder(cls) -> str:
90
+ return resources.files(f"edsl.questions.templates", cls.question_type)
91
+
92
+ @property
93
+ def response_model(self) -> type["BaseModel"]:
94
+ if self._response_model is not None:
95
+ return self._response_model
96
+ else:
97
+ return self.create_response_model()
98
+
99
+ @property
100
+ def use_code(self) -> bool:
101
+ if hasattr(self, "_use_code"):
102
+ return self._use_code
103
+ return True
104
+
105
+ @use_code.setter
106
+ def use_code(self, value: bool) -> None:
107
+ self._use_code = value
108
+
109
+ @property
110
+ def include_comment(self) -> bool:
111
+ if hasattr(self, "_include_comment"):
112
+ return self._include_comment
113
+ return True
114
+
115
+ @include_comment.setter
116
+ def include_comment(self, value: bool) -> None:
117
+ self._include_comment = value
118
+
119
+ @classmethod
120
+ def default_answering_instructions(cls) -> str:
121
+ # template_text = cls._read_template("answering_instructions.jinja")
122
+ template_text = template_manager.get_template(
123
+ cls.question_type, "answering_instructions.jinja"
124
+ )
125
+ return Prompt(text=template_text)
126
+
127
+ @classmethod
128
+ def default_question_presentation(cls):
129
+ template_text = template_manager.get_template(
130
+ cls.question_type, "question_presentation.jinja"
131
+ )
132
+ return Prompt(text=template_text)
133
+
134
+ @property
135
+ def answering_instructions(self) -> str:
136
+ if self._answering_instructions is None:
137
+ return self.default_answering_instructions()
138
+ return self._answering_instructions
139
+
140
+ @answering_instructions.setter
141
+ def answering_instructions(self, value) -> None:
142
+ self._answering_instructions = value
143
+
144
+ @property
145
+ def question_presentation(self):
146
+ if self._question_presentation is None:
147
+ return self.default_question_presentation()
148
+ return self._question_presentation
149
+
150
+ @question_presentation.setter
151
+ def question_presentation(self, value):
152
+ self._question_presentation = value
153
+
154
+ def prompt_preview(self, scenario=None, agent=None):
155
+ return self.new_default_instructions.render(
156
+ self.data
157
+ | {
158
+ "include_comment": getattr(self, "_include_comment", True),
159
+ "use_code": getattr(self, "_use_code", True),
160
+ }
161
+ | ({"scenario": scenario} or {})
162
+ | ({"agent": agent} or {})
163
+ )
164
+
165
+ @classmethod
166
+ def self_check(cls):
167
+ q = cls.example()
168
+ for answer, params in q.response_validator.valid_examples:
169
+ for key, value in params.items():
170
+ setattr(q, key, value)
171
+ q._validate_answer(answer)
172
+ for answer, params, reason in q.response_validator.invalid_examples:
173
+ for key, value in params.items():
174
+ setattr(q, key, value)
175
+ try:
176
+ q._validate_answer(answer)
177
+ except QuestionAnswerValidationError:
178
+ pass
179
+ else:
180
+ raise ValueError(f"Example {answer} should have failed for {reason}.")
181
+
182
+ @property
183
+ def new_default_instructions(self) -> "Prompt":
184
+ "This is set up as a property because there are mutable question values that determine how it is rendered."
185
+ return Prompt(self.question_presentation) + Prompt(self.answering_instructions)
186
+
187
+ @property
188
+ def parameters(self) -> set[str]:
189
+ """Return the parameters of the question."""
190
+ from jinja2 import Environment, meta
191
+
192
+ env = Environment()
193
+ # Parse the template
194
+ txt = self._all_text()
195
+ # txt = self.question_text
196
+ # if hasattr(self, "question_options"):
197
+ # txt += " ".join(self.question_options)
198
+ parsed_content = env.parse(txt)
199
+ # Extract undeclared variables
200
+ variables = meta.find_undeclared_variables(parsed_content)
201
+ # Return as a list
202
+ return set(variables)
203
+
204
+ def get_instructions(self, model: Optional[str] = None) -> type["PromptBase"]:
205
+ """Get the mathcing question-answering instructions for the question.
206
+
207
+ :param model: The language model to use.
208
+ """
209
+ from edsl.prompts.Prompt import Prompt
210
+
211
+ if model in self.model_instructions:
212
+ return Prompt(text=self.model_instructions[model])
213
+ else:
214
+ if hasattr(self, "new_default_instructions"):
215
+ return self.new_default_instructions
216
+ else:
217
+ return self.applicable_prompts(model)[0]()