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,330 +1,286 @@
1
- from __future__ import annotations
2
- from typing import Union, Literal, Optional, List, Any
3
-
4
- from jinja2 import Template
5
- from pydantic import BaseModel, Field
6
-
7
- from edsl.scenarios.Scenario import Scenario
8
- from edsl.questions.QuestionBase import QuestionBase
9
- from edsl.questions.descriptors import QuestionOptionsDescriptor
10
- from edsl.questions.decorators import inject_exception
11
- from edsl.questions.ResponseValidatorABC import ResponseValidatorABC
12
-
13
-
14
- def create_response_model(choices: List[str], permissive: bool = False):
15
- """
16
- Create a ChoiceResponse model class with a predefined list of choices.
17
-
18
- :param choices: A list of allowed values for the answer field.
19
- :param permissive: If True, any value will be accepted as an answer.
20
- :return: A new Pydantic model class.
21
- """
22
- choice_tuple = tuple(choices)
23
-
24
- if not permissive:
25
-
26
- class ChoiceResponse(BaseModel):
27
- answer: Literal[choice_tuple] = Field(description="Selected choice")
28
- comment: Optional[str] = Field(None, description="Optional comment field")
29
- generated_tokens: Optional[Any] = Field(
30
- None, description="Generated tokens"
31
- )
32
-
33
- class Config:
34
- @staticmethod
35
- def json_schema_extra(schema: dict, model: BaseModel) -> None:
36
- for prop in schema.get("properties", {}).values():
37
- if prop.get("title") == "answer":
38
- prop["enum"] = choices
39
-
40
- else:
41
-
42
- class ChoiceResponse(BaseModel):
43
- answer: Any = Field(description="Selected choice (can be any value)")
44
- comment: Optional[str] = Field(None, description="Optional comment field")
45
- generated_tokens: Optional[Any] = Field(
46
- None, description="Generated tokens"
47
- )
48
-
49
- class Config:
50
- @staticmethod
51
- def json_schema_extra(schema: dict, model: BaseModel) -> None:
52
- for prop in schema.get("properties", {}).values():
53
- if prop.get("title") == "answer":
54
- prop["description"] += f". Suggested choices are: {choices}"
55
- schema["title"] += " (Permissive)"
56
-
57
- return ChoiceResponse
58
-
59
-
60
- class MultipleChoiceResponseValidator(ResponseValidatorABC):
61
- required_params = ["question_options", "use_code"]
62
-
63
- def fix(self, response, verbose=False):
64
- response_text = str(response.get("answer"))
65
- if response_text is None:
66
- response_text = response.get("generated_tokens", "")
67
-
68
- if verbose:
69
- print(f"Invalid generated tokens was: {response_text}")
70
-
71
- matches = []
72
- for idx, option in enumerate(self.question_options):
73
- if verbose:
74
- print("The options are: ", self.question_options)
75
- if str(option) in response_text:
76
- if verbose:
77
- print("Match found with option ", option)
78
- if option not in matches:
79
- matches.append(option)
80
-
81
- if verbose:
82
- print("The matches are: ", matches)
83
- if len(matches) == 1:
84
- proposed_data = {
85
- "answer": matches[0],
86
- "generated_tokens": response.get("generated_tokens", None),
87
- }
88
- try:
89
- self.response_model(**proposed_data)
90
- return proposed_data
91
- except Exception as e:
92
- if verbose:
93
- print(f"Proposed solution {proposed_data} is invalid. Error: {e}")
94
- return response
95
-
96
- valid_examples = [
97
- ({"answer": 1}, {"question_options": ["Good", "Great", "OK", "Bad"]})
98
- ]
99
-
100
- invalid_examples = [
101
- (
102
- {"answer": -1},
103
- {"question_options": ["Good", "Great", "OK", "Bad"]},
104
- "Answer code must be a non-negative integer",
105
- ),
106
- (
107
- {"answer": None},
108
- {"question_options": ["Good", "Great", "OK", "Bad"]},
109
- "Answer code must not be missing.",
110
- ),
111
- ]
112
-
113
-
114
- class QuestionMultipleChoice(QuestionBase):
115
- """This question prompts the agent to select one option from a list of options.
116
-
117
- https://docs.expectedparrot.com/en/latest/questions.html#questionmultiplechoice-class
118
-
119
- """
120
-
121
- question_type = "multiple_choice"
122
- purpose = "When options are known and limited"
123
- question_options: Union[list[str], list[list], list[float], list[int]] = (
124
- QuestionOptionsDescriptor()
125
- )
126
- _response_model = None
127
- response_validator_class = MultipleChoiceResponseValidator
128
-
129
- def __init__(
130
- self,
131
- question_name: str,
132
- question_text: str,
133
- question_options: Union[list[str], list[list], list[float], list[int]],
134
- include_comment: bool = True,
135
- use_code: bool = False,
136
- answering_instructions: Optional[str] = None,
137
- question_presentation: Optional[str] = None,
138
- permissive: bool = False,
139
- ):
140
- """Instantiate a new QuestionMultipleChoice.
141
-
142
- :param question_name: The name of the question.
143
- :param question_text: The text of the question.
144
- :param question_options: The options the agent should select from.
145
- :param include_comment: Whether to include a comment field.
146
- :param use_code: Whether to use code for the options.
147
- :param answering_instructions: Instructions for the question.
148
- :param question_presentation: The presentation of the question.
149
- :param permissive: Whether to force the answer to be one of the options.
150
-
151
- """
152
- self.question_name = question_name
153
- self.question_text = question_text
154
- self.question_options = question_options
155
-
156
- self._include_comment = include_comment
157
- self.use_code = use_code
158
- self.answering_instructions = answering_instructions
159
- self.question_presentation = question_presentation
160
- self.permissive = permissive
161
-
162
- ################
163
- # Answer methods
164
- ################
165
-
166
- def create_response_model(self, replacement_dict: dict = None):
167
- if replacement_dict is None:
168
- replacement_dict = {}
169
- # The replacement dict that could be from scenario, current answers, etc. to populate the response model
170
-
171
- if self.use_code:
172
- return create_response_model(
173
- list(range(len(self.question_options))), self.permissive
174
- )
175
- else:
176
- return create_response_model(self.question_options, self.permissive)
177
-
178
- @staticmethod
179
- def _translate_question_options(
180
- question_options, substitution_dict: dict
181
- ) -> list[str]:
182
-
183
- if isinstance(question_options, str):
184
- # If dynamic options are provided like {{ options }}, render them with the scenario
185
- # We can check if it's in the Scenario.
186
- from jinja2 import Environment, meta
187
-
188
- env = Environment()
189
- parsed_content = env.parse(question_options)
190
- template_variables = list(meta.find_undeclared_variables(parsed_content))
191
- # print("The template variables are: ", template_variables)
192
- question_option_key = template_variables[0]
193
- # We need to deal with possibility it's actually an answer to a question.
194
- potential_replacement = substitution_dict.get(question_option_key, None)
195
-
196
- if isinstance(potential_replacement, list):
197
- # translated_options = potential_replacement
198
- return potential_replacement
199
-
200
- if isinstance(potential_replacement, QuestionBase):
201
- if hasattr(potential_replacement, "answer") and isinstance(
202
- potential_replacement.answer, list
203
- ):
204
- return potential_replacement.answer
205
- # translated_options = potential_replacement.answer
206
-
207
- # if not isinstance(potential_replacement, list):
208
- # translated_options = potential_replacement
209
-
210
- if potential_replacement is None:
211
- # Nope - maybe it's in the substition dict?
212
- raise ValueError(
213
- f"Could not find the key '{question_option_key}' in the scenario."
214
- f"The substition dict was: '{substitution_dict}.'"
215
- f"The question options were: '{question_options}'."
216
- )
217
- else:
218
- translated_options = [
219
- Template(str(option)).render(substitution_dict)
220
- for option in question_options
221
- ]
222
- return translated_options
223
-
224
- def _translate_answer_code_to_answer(
225
- self, answer_code: int, replacements_dict: Optional[dict] = None
226
- ):
227
- """Translate the answer code to the actual answer.
228
-
229
- It is used to translate the answer code to the actual answer.
230
- The question options might be templates, so they need to be rendered with the scenario.
231
-
232
- >>> q = QuestionMultipleChoice.example()
233
- >>> q._translate_answer_code_to_answer('Good', {})
234
- 'Good'
235
-
236
- >>> q = QuestionMultipleChoice(question_name="how_feeling", question_text="How are you?", question_options=["{{emotion[0]}}", "emotion[1]"])
237
- >>> q._translate_answer_code_to_answer('Happy', {"emotion": ["Happy", "Sad"]})
238
- 'Happy'
239
-
240
- """
241
- if replacements_dict is None:
242
- replacements_dict = {}
243
- translated_options = self._translate_question_options(
244
- self.question_options, replacements_dict
245
- )
246
-
247
- if self._use_code:
248
- try:
249
- return translated_options[int(answer_code)]
250
- except IndexError:
251
- raise ValueError(
252
- f"Answer code is out of range. The answer code index was: {int(answer_code)}. The options were: {translated_options}."
253
- )
254
- except TypeError:
255
- raise ValueError(
256
- f"The answer code was: '{answer_code}.'",
257
- f"The options were: '{translated_options}'.",
258
- )
259
- else:
260
- # return translated_options[answer_code]
261
- return answer_code
262
-
263
- @property
264
- def question_html_content(self) -> str:
265
- """Return the HTML version of the question."""
266
- if hasattr(self, "option_labels"):
267
- option_labels = self.option_labels
268
- else:
269
- option_labels = {}
270
- question_html_content = Template(
271
- """
272
- {% for option in question_options %}
273
- <div>
274
- <input type="radio" id="{{ option }}" name="{{ question_name }}" value="{{ option }}">
275
- <label for="{{ option }}">
276
- {{ option }}
277
- {% if option in option_labels %}
278
- : {{ option_labels[option] }}
279
- {% endif %}
280
- </label>
281
- </div>
282
- {% endfor %}
283
- """
284
- ).render(
285
- question_name=self.question_name,
286
- question_options=self.question_options,
287
- option_labels=option_labels,
288
- )
289
- return question_html_content
290
-
291
- ################
292
- # Example
293
- ################
294
- @classmethod
295
- @inject_exception
296
- def example(cls, include_comment=False, use_code=False) -> QuestionMultipleChoice:
297
- """Return an example instance."""
298
- return cls(
299
- question_text="How are you?",
300
- question_options=["Good", "Great", "OK", "Bad"],
301
- question_name="how_feeling",
302
- include_comment=include_comment,
303
- use_code=use_code,
304
- )
305
-
306
-
307
- # def main():
308
- # """Create an example QuestionMultipleChoice and test its methods."""
309
- # from edsl.questions.QuestionMultipleChoice import QuestionMultipleChoice
310
-
311
- # q = QuestionMultipleChoice.example()
312
- # q.question_text
313
- # q.question_options
314
- # q.question_name
315
- # # validate an answer
316
- # q._validate_answer({"answer": 0, "comment": "I like custard"})
317
- # # translate answer code
318
- # q._translate_answer_code_to_answer(0, {})
319
- # # simulate answer
320
- # q._simulate_answer()
321
- # q._simulate_answer(human_readable=False)
322
- # # serialization (inherits from Question)
323
- # q.to_dict()
324
- # assert q.from_dict(q.to_dict()) == q
325
-
326
-
327
- if __name__ == "__main__":
328
- import doctest
329
-
330
- doctest.testmod(optionflags=doctest.ELLIPSIS)
1
+ from __future__ import annotations
2
+ from typing import Union, Literal, Optional, List, Any
3
+
4
+ from jinja2 import Template
5
+ from pydantic import BaseModel, Field
6
+
7
+ from edsl.scenarios.Scenario import Scenario
8
+ from edsl.questions.QuestionBase import QuestionBase
9
+ from edsl.questions.descriptors import QuestionOptionsDescriptor
10
+ from edsl.questions.decorators import inject_exception
11
+ from edsl.questions.ResponseValidatorABC import ResponseValidatorABC
12
+
13
+
14
+ def create_response_model(choices: List[str], permissive: bool = False):
15
+ """
16
+ Create a ChoiceResponse model class with a predefined list of choices.
17
+
18
+ :param choices: A list of allowed values for the answer field.
19
+ :param permissive: If True, any value will be accepted as an answer.
20
+ :return: A new Pydantic model class.
21
+ """
22
+ choice_tuple = tuple(choices)
23
+
24
+ if not permissive:
25
+
26
+ class ChoiceResponse(BaseModel):
27
+ answer: Literal[choice_tuple] = Field(description="Selected choice")
28
+ comment: Optional[str] = Field(None, description="Optional comment field")
29
+ generated_tokens: Optional[Any] = Field(
30
+ None, description="Generated tokens"
31
+ )
32
+
33
+ class Config:
34
+ @staticmethod
35
+ def json_schema_extra(schema: dict, model: BaseModel) -> None:
36
+ for prop in schema.get("properties", {}).values():
37
+ if prop.get("title") == "answer":
38
+ prop["enum"] = choices
39
+
40
+ else:
41
+
42
+ class ChoiceResponse(BaseModel):
43
+ answer: Any = Field(description="Selected choice (can be any value)")
44
+ comment: Optional[str] = Field(None, description="Optional comment field")
45
+ generated_tokens: Optional[Any] = Field(
46
+ None, description="Generated tokens"
47
+ )
48
+
49
+ class Config:
50
+ @staticmethod
51
+ def json_schema_extra(schema: dict, model: BaseModel) -> None:
52
+ for prop in schema.get("properties", {}).values():
53
+ if prop.get("title") == "answer":
54
+ prop["description"] += f". Suggested choices are: {choices}"
55
+ schema["title"] += " (Permissive)"
56
+
57
+ return ChoiceResponse
58
+
59
+
60
+ class MultipleChoiceResponseValidator(ResponseValidatorABC):
61
+ required_params = ["question_options", "use_code"]
62
+
63
+ def fix(self, response, verbose=False):
64
+ response_text = str(response.get("answer"))
65
+ if response_text is None:
66
+ response_text = response.get("generated_tokens", "")
67
+
68
+ if verbose:
69
+ print(f"Invalid generated tokens was: {response_text}")
70
+
71
+ matches = []
72
+ for idx, option in enumerate(self.question_options):
73
+ if verbose:
74
+ print("The options are: ", self.question_options)
75
+ if str(option) in response_text:
76
+ if verbose:
77
+ print("Match found with option ", option)
78
+ if option not in matches:
79
+ matches.append(option)
80
+
81
+ if verbose:
82
+ print("The matches are: ", matches)
83
+ if len(matches) == 1:
84
+ proposed_data = {
85
+ "answer": matches[0],
86
+ "generated_tokens": response.get("generated_tokens", None),
87
+ }
88
+ try:
89
+ self.response_model(**proposed_data)
90
+ return proposed_data
91
+ except Exception as e:
92
+ if verbose:
93
+ print(f"Proposed solution {proposed_data} is invalid. Error: {e}")
94
+ return response
95
+
96
+ valid_examples = [
97
+ ({"answer": 1}, {"question_options": ["Good", "Great", "OK", "Bad"]})
98
+ ]
99
+
100
+ invalid_examples = [
101
+ (
102
+ {"answer": -1},
103
+ {"question_options": ["Good", "Great", "OK", "Bad"]},
104
+ "Answer code must be a non-negative integer",
105
+ ),
106
+ (
107
+ {"answer": None},
108
+ {"question_options": ["Good", "Great", "OK", "Bad"]},
109
+ "Answer code must not be missing.",
110
+ ),
111
+ ]
112
+
113
+
114
+ class QuestionMultipleChoice(QuestionBase):
115
+ """This question prompts the agent to select one option from a list of options.
116
+
117
+ https://docs.expectedparrot.com/en/latest/questions.html#questionmultiplechoice-class
118
+
119
+ """
120
+
121
+ question_type = "multiple_choice"
122
+ purpose = "When options are known and limited"
123
+ question_options: Union[
124
+ list[str], list[list], list[float], list[int]
125
+ ] = QuestionOptionsDescriptor()
126
+ _response_model = None
127
+ response_validator_class = MultipleChoiceResponseValidator
128
+
129
+ def __init__(
130
+ self,
131
+ question_name: str,
132
+ question_text: str,
133
+ question_options: Union[list[str], list[list], list[float], list[int]],
134
+ include_comment: bool = True,
135
+ use_code: bool = False,
136
+ answering_instructions: Optional[str] = None,
137
+ question_presentation: Optional[str] = None,
138
+ permissive: bool = False,
139
+ ):
140
+ """Instantiate a new QuestionMultipleChoice.
141
+
142
+ :param question_name: The name of the question.
143
+ :param question_text: The text of the question.
144
+ :param question_options: The options the agent should select from.
145
+ :param include_comment: Whether to include a comment field.
146
+ :param use_code: Whether to use code for the options.
147
+ :param answering_instructions: Instructions for the question.
148
+ :param question_presentation: The presentation of the question.
149
+ :param permissive: Whether to force the answer to be one of the options.
150
+
151
+ """
152
+ self.question_name = question_name
153
+ self.question_text = question_text
154
+ self.question_options = question_options
155
+
156
+ self._include_comment = include_comment
157
+ self.use_code = use_code
158
+ self.answering_instructions = answering_instructions
159
+ self.question_presentation = question_presentation
160
+ self.permissive = permissive
161
+
162
+ ################
163
+ # Answer methods
164
+ ################
165
+
166
+ def create_response_model(self, replacement_dict: dict = None):
167
+ if replacement_dict is None:
168
+ replacement_dict = {}
169
+ # The replacement dict that could be from scenario, current answers, etc. to populate the response model
170
+
171
+ if self.use_code:
172
+ return create_response_model(
173
+ list(range(len(self.question_options))), self.permissive
174
+ )
175
+ else:
176
+ return create_response_model(self.question_options, self.permissive)
177
+
178
+ def _translate_answer_code_to_answer(
179
+ self, answer_code: int, scenario: Optional["Scenario"] = None
180
+ ):
181
+ """Translate the answer code to the actual answer.
182
+
183
+ It is used to translate the answer code to the actual answer.
184
+ The question options might be templates, so they need to be rendered with the scenario.
185
+
186
+ >>> q = QuestionMultipleChoice.example()
187
+ >>> q._translate_answer_code_to_answer('Good', {})
188
+ 'Good'
189
+
190
+ >>> q = QuestionMultipleChoice(question_name="how_feeling", question_text="How are you?", question_options=["{{emotion[0]}}", "emotion[1]"])
191
+ >>> q._translate_answer_code_to_answer('Happy', {"emotion": ["Happy", "Sad"]})
192
+ 'Happy'
193
+
194
+ """
195
+
196
+ scenario = scenario or Scenario()
197
+
198
+ if isinstance(self.question_options, str):
199
+ # If dynamic options are provided like {{ options }}, render them with the scenario
200
+ from jinja2 import Environment, meta
201
+
202
+ env = Environment()
203
+ parsed_content = env.parse(self.question_options)
204
+ question_option_key = list(meta.find_undeclared_variables(parsed_content))[
205
+ 0
206
+ ]
207
+ translated_options = scenario.get(question_option_key)
208
+ else:
209
+ translated_options = [
210
+ Template(str(option)).render(scenario)
211
+ for option in self.question_options
212
+ ]
213
+ if self._use_code:
214
+ return translated_options[int(answer_code)]
215
+ else:
216
+ # return translated_options[answer_code]
217
+ return answer_code
218
+
219
+ @property
220
+ def question_html_content(self) -> str:
221
+ """Return the HTML version of the question."""
222
+ if hasattr(self, "option_labels"):
223
+ option_labels = self.option_labels
224
+ else:
225
+ option_labels = {}
226
+ question_html_content = Template(
227
+ """
228
+ {% for option in question_options %}
229
+ <div>
230
+ <input type="radio" id="{{ option }}" name="{{ question_name }}" value="{{ option }}">
231
+ <label for="{{ option }}">
232
+ {{ option }}
233
+ {% if option in option_labels %}
234
+ : {{ option_labels[option] }}
235
+ {% endif %}
236
+ </label>
237
+ </div>
238
+ {% endfor %}
239
+ """
240
+ ).render(
241
+ question_name=self.question_name,
242
+ question_options=self.question_options,
243
+ option_labels=option_labels,
244
+ )
245
+ return question_html_content
246
+
247
+ ################
248
+ # Example
249
+ ################
250
+ @classmethod
251
+ @inject_exception
252
+ def example(cls, include_comment=False, use_code=False) -> QuestionMultipleChoice:
253
+ """Return an example instance."""
254
+ return cls(
255
+ question_text="How are you?",
256
+ question_options=["Good", "Great", "OK", "Bad"],
257
+ question_name="how_feeling",
258
+ include_comment=include_comment,
259
+ use_code=use_code,
260
+ )
261
+
262
+
263
+ # def main():
264
+ # """Create an example QuestionMultipleChoice and test its methods."""
265
+ # from edsl.questions.QuestionMultipleChoice import QuestionMultipleChoice
266
+
267
+ # q = QuestionMultipleChoice.example()
268
+ # q.question_text
269
+ # q.question_options
270
+ # q.question_name
271
+ # # validate an answer
272
+ # q._validate_answer({"answer": 0, "comment": "I like custard"})
273
+ # # translate answer code
274
+ # q._translate_answer_code_to_answer(0, {})
275
+ # # simulate answer
276
+ # q._simulate_answer()
277
+ # q._simulate_answer(human_readable=False)
278
+ # # serialization (inherits from Question)
279
+ # q.to_dict()
280
+ # assert q.from_dict(q.to_dict()) == q
281
+
282
+
283
+ if __name__ == "__main__":
284
+ import doctest
285
+
286
+ doctest.testmod(optionflags=doctest.ELLIPSIS)