edsl 0.1.14__py3-none-any.whl → 0.1.40__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.
- edsl/Base.py +348 -38
- edsl/BaseDiff.py +260 -0
- edsl/TemplateLoader.py +24 -0
- edsl/__init__.py +46 -10
- edsl/__version__.py +1 -0
- edsl/agents/Agent.py +842 -144
- edsl/agents/AgentList.py +521 -25
- edsl/agents/Invigilator.py +250 -374
- edsl/agents/InvigilatorBase.py +257 -0
- edsl/agents/PromptConstructor.py +272 -0
- edsl/agents/QuestionInstructionPromptBuilder.py +128 -0
- edsl/agents/QuestionTemplateReplacementsBuilder.py +137 -0
- edsl/agents/descriptors.py +43 -13
- edsl/agents/prompt_helpers.py +129 -0
- edsl/agents/question_option_processor.py +172 -0
- edsl/auto/AutoStudy.py +130 -0
- edsl/auto/StageBase.py +243 -0
- edsl/auto/StageGenerateSurvey.py +178 -0
- edsl/auto/StageLabelQuestions.py +125 -0
- edsl/auto/StagePersona.py +61 -0
- edsl/auto/StagePersonaDimensionValueRanges.py +88 -0
- edsl/auto/StagePersonaDimensionValues.py +74 -0
- edsl/auto/StagePersonaDimensions.py +69 -0
- edsl/auto/StageQuestions.py +74 -0
- edsl/auto/SurveyCreatorPipeline.py +21 -0
- edsl/auto/utilities.py +218 -0
- edsl/base/Base.py +279 -0
- edsl/config.py +121 -104
- edsl/conversation/Conversation.py +290 -0
- edsl/conversation/car_buying.py +59 -0
- edsl/conversation/chips.py +95 -0
- edsl/conversation/mug_negotiation.py +81 -0
- edsl/conversation/next_speaker_utilities.py +93 -0
- edsl/coop/CoopFunctionsMixin.py +15 -0
- edsl/coop/ExpectedParrotKeyHandler.py +125 -0
- edsl/coop/PriceFetcher.py +54 -0
- edsl/coop/__init__.py +1 -0
- edsl/coop/coop.py +1029 -134
- edsl/coop/utils.py +131 -0
- edsl/data/Cache.py +560 -89
- edsl/data/CacheEntry.py +230 -0
- edsl/data/CacheHandler.py +168 -0
- edsl/data/RemoteCacheSync.py +186 -0
- edsl/data/SQLiteDict.py +292 -0
- edsl/data/__init__.py +5 -3
- edsl/data/orm.py +6 -33
- edsl/data_transfer_models.py +74 -27
- edsl/enums.py +165 -8
- edsl/exceptions/BaseException.py +21 -0
- edsl/exceptions/__init__.py +52 -46
- edsl/exceptions/agents.py +33 -15
- edsl/exceptions/cache.py +5 -0
- edsl/exceptions/coop.py +8 -0
- edsl/exceptions/general.py +34 -0
- edsl/exceptions/inference_services.py +5 -0
- edsl/exceptions/jobs.py +15 -0
- edsl/exceptions/language_models.py +46 -1
- edsl/exceptions/questions.py +80 -5
- edsl/exceptions/results.py +16 -5
- edsl/exceptions/scenarios.py +29 -0
- edsl/exceptions/surveys.py +13 -10
- edsl/inference_services/AnthropicService.py +106 -0
- edsl/inference_services/AvailableModelCacheHandler.py +184 -0
- edsl/inference_services/AvailableModelFetcher.py +215 -0
- edsl/inference_services/AwsBedrock.py +118 -0
- edsl/inference_services/AzureAI.py +215 -0
- edsl/inference_services/DeepInfraService.py +18 -0
- edsl/inference_services/GoogleService.py +143 -0
- edsl/inference_services/GroqService.py +20 -0
- edsl/inference_services/InferenceServiceABC.py +80 -0
- edsl/inference_services/InferenceServicesCollection.py +138 -0
- edsl/inference_services/MistralAIService.py +120 -0
- edsl/inference_services/OllamaService.py +18 -0
- edsl/inference_services/OpenAIService.py +236 -0
- edsl/inference_services/PerplexityService.py +160 -0
- edsl/inference_services/ServiceAvailability.py +135 -0
- edsl/inference_services/TestService.py +90 -0
- edsl/inference_services/TogetherAIService.py +172 -0
- edsl/inference_services/data_structures.py +134 -0
- edsl/inference_services/models_available_cache.py +118 -0
- edsl/inference_services/rate_limits_cache.py +25 -0
- edsl/inference_services/registry.py +41 -0
- edsl/inference_services/write_available.py +10 -0
- edsl/jobs/AnswerQuestionFunctionConstructor.py +223 -0
- edsl/jobs/Answers.py +21 -20
- edsl/jobs/FetchInvigilator.py +47 -0
- edsl/jobs/InterviewTaskManager.py +98 -0
- edsl/jobs/InterviewsConstructor.py +50 -0
- edsl/jobs/Jobs.py +684 -204
- edsl/jobs/JobsChecks.py +172 -0
- edsl/jobs/JobsComponentConstructor.py +189 -0
- edsl/jobs/JobsPrompts.py +270 -0
- edsl/jobs/JobsRemoteInferenceHandler.py +311 -0
- edsl/jobs/JobsRemoteInferenceLogger.py +239 -0
- edsl/jobs/RequestTokenEstimator.py +30 -0
- edsl/jobs/async_interview_runner.py +138 -0
- edsl/jobs/buckets/BucketCollection.py +104 -0
- edsl/jobs/buckets/ModelBuckets.py +65 -0
- edsl/jobs/buckets/TokenBucket.py +283 -0
- edsl/jobs/buckets/TokenBucketAPI.py +211 -0
- edsl/jobs/buckets/TokenBucketClient.py +191 -0
- edsl/jobs/check_survey_scenario_compatibility.py +85 -0
- edsl/jobs/data_structures.py +120 -0
- edsl/jobs/decorators.py +35 -0
- edsl/jobs/interviews/Interview.py +392 -0
- edsl/jobs/interviews/InterviewExceptionCollection.py +99 -0
- edsl/jobs/interviews/InterviewExceptionEntry.py +186 -0
- edsl/jobs/interviews/InterviewStatistic.py +63 -0
- edsl/jobs/interviews/InterviewStatisticsCollection.py +25 -0
- edsl/jobs/interviews/InterviewStatusDictionary.py +78 -0
- edsl/jobs/interviews/InterviewStatusLog.py +92 -0
- edsl/jobs/interviews/ReportErrors.py +66 -0
- edsl/jobs/interviews/interview_status_enum.py +9 -0
- edsl/jobs/jobs_status_enums.py +9 -0
- edsl/jobs/loggers/HTMLTableJobLogger.py +304 -0
- edsl/jobs/results_exceptions_handler.py +98 -0
- edsl/jobs/runners/JobsRunnerAsyncio.py +151 -110
- edsl/jobs/runners/JobsRunnerStatus.py +298 -0
- edsl/jobs/tasks/QuestionTaskCreator.py +244 -0
- edsl/jobs/tasks/TaskCreators.py +64 -0
- edsl/jobs/tasks/TaskHistory.py +470 -0
- edsl/jobs/tasks/TaskStatusLog.py +23 -0
- edsl/jobs/tasks/task_status_enum.py +161 -0
- edsl/jobs/tokens/InterviewTokenUsage.py +27 -0
- edsl/jobs/tokens/TokenUsage.py +34 -0
- edsl/language_models/ComputeCost.py +63 -0
- edsl/language_models/LanguageModel.py +507 -386
- edsl/language_models/ModelList.py +164 -0
- edsl/language_models/PriceManager.py +127 -0
- edsl/language_models/RawResponseHandler.py +106 -0
- edsl/language_models/RegisterLanguageModelsMeta.py +184 -0
- edsl/language_models/__init__.py +1 -8
- edsl/language_models/fake_openai_call.py +15 -0
- edsl/language_models/fake_openai_service.py +61 -0
- edsl/language_models/key_management/KeyLookup.py +63 -0
- edsl/language_models/key_management/KeyLookupBuilder.py +273 -0
- edsl/language_models/key_management/KeyLookupCollection.py +38 -0
- edsl/language_models/key_management/__init__.py +0 -0
- edsl/language_models/key_management/models.py +131 -0
- edsl/language_models/model.py +256 -0
- edsl/language_models/repair.py +109 -41
- edsl/language_models/utilities.py +65 -0
- edsl/notebooks/Notebook.py +263 -0
- edsl/notebooks/NotebookToLaTeX.py +142 -0
- edsl/notebooks/__init__.py +1 -0
- edsl/prompts/Prompt.py +222 -93
- edsl/prompts/__init__.py +1 -1
- edsl/questions/ExceptionExplainer.py +77 -0
- edsl/questions/HTMLQuestion.py +103 -0
- edsl/questions/QuestionBase.py +518 -0
- edsl/questions/QuestionBasePromptsMixin.py +221 -0
- edsl/questions/QuestionBudget.py +164 -67
- edsl/questions/QuestionCheckBox.py +281 -62
- edsl/questions/QuestionDict.py +343 -0
- edsl/questions/QuestionExtract.py +136 -50
- edsl/questions/QuestionFreeText.py +79 -55
- edsl/questions/QuestionFunctional.py +138 -41
- edsl/questions/QuestionList.py +184 -57
- edsl/questions/QuestionMatrix.py +265 -0
- edsl/questions/QuestionMultipleChoice.py +293 -69
- edsl/questions/QuestionNumerical.py +109 -56
- edsl/questions/QuestionRank.py +244 -49
- edsl/questions/Quick.py +41 -0
- edsl/questions/SimpleAskMixin.py +74 -0
- edsl/questions/__init__.py +9 -6
- edsl/questions/{AnswerValidatorMixin.py → answer_validator_mixin.py} +153 -38
- edsl/questions/compose_questions.py +13 -7
- edsl/questions/data_structures.py +20 -0
- edsl/questions/decorators.py +21 -0
- edsl/questions/derived/QuestionLikertFive.py +28 -26
- edsl/questions/derived/QuestionLinearScale.py +41 -28
- edsl/questions/derived/QuestionTopK.py +34 -26
- edsl/questions/derived/QuestionYesNo.py +40 -27
- edsl/questions/descriptors.py +228 -74
- edsl/questions/loop_processor.py +149 -0
- edsl/questions/prompt_templates/question_budget.jinja +13 -0
- edsl/questions/prompt_templates/question_checkbox.jinja +32 -0
- edsl/questions/prompt_templates/question_extract.jinja +11 -0
- edsl/questions/prompt_templates/question_free_text.jinja +3 -0
- edsl/questions/prompt_templates/question_linear_scale.jinja +11 -0
- edsl/questions/prompt_templates/question_list.jinja +17 -0
- edsl/questions/prompt_templates/question_multiple_choice.jinja +33 -0
- edsl/questions/prompt_templates/question_numerical.jinja +37 -0
- edsl/questions/question_base_gen_mixin.py +168 -0
- edsl/questions/question_registry.py +130 -46
- edsl/questions/register_questions_meta.py +71 -0
- edsl/questions/response_validator_abc.py +188 -0
- edsl/questions/response_validator_factory.py +34 -0
- edsl/questions/settings.py +5 -2
- edsl/questions/templates/__init__.py +0 -0
- edsl/questions/templates/budget/__init__.py +0 -0
- edsl/questions/templates/budget/answering_instructions.jinja +7 -0
- edsl/questions/templates/budget/question_presentation.jinja +7 -0
- edsl/questions/templates/checkbox/__init__.py +0 -0
- edsl/questions/templates/checkbox/answering_instructions.jinja +10 -0
- edsl/questions/templates/checkbox/question_presentation.jinja +22 -0
- edsl/questions/templates/dict/__init__.py +0 -0
- edsl/questions/templates/dict/answering_instructions.jinja +21 -0
- edsl/questions/templates/dict/question_presentation.jinja +1 -0
- edsl/questions/templates/extract/__init__.py +0 -0
- edsl/questions/templates/extract/answering_instructions.jinja +7 -0
- edsl/questions/templates/extract/question_presentation.jinja +1 -0
- edsl/questions/templates/free_text/__init__.py +0 -0
- edsl/questions/templates/free_text/answering_instructions.jinja +0 -0
- edsl/questions/templates/free_text/question_presentation.jinja +1 -0
- edsl/questions/templates/likert_five/__init__.py +0 -0
- edsl/questions/templates/likert_five/answering_instructions.jinja +10 -0
- edsl/questions/templates/likert_five/question_presentation.jinja +12 -0
- edsl/questions/templates/linear_scale/__init__.py +0 -0
- edsl/questions/templates/linear_scale/answering_instructions.jinja +5 -0
- edsl/questions/templates/linear_scale/question_presentation.jinja +5 -0
- edsl/questions/templates/list/__init__.py +0 -0
- edsl/questions/templates/list/answering_instructions.jinja +4 -0
- edsl/questions/templates/list/question_presentation.jinja +5 -0
- edsl/questions/templates/matrix/__init__.py +1 -0
- edsl/questions/templates/matrix/answering_instructions.jinja +5 -0
- edsl/questions/templates/matrix/question_presentation.jinja +20 -0
- edsl/questions/templates/multiple_choice/__init__.py +0 -0
- edsl/questions/templates/multiple_choice/answering_instructions.jinja +9 -0
- edsl/questions/templates/multiple_choice/html.jinja +0 -0
- edsl/questions/templates/multiple_choice/question_presentation.jinja +12 -0
- edsl/questions/templates/numerical/__init__.py +0 -0
- edsl/questions/templates/numerical/answering_instructions.jinja +7 -0
- edsl/questions/templates/numerical/question_presentation.jinja +7 -0
- edsl/questions/templates/rank/__init__.py +0 -0
- edsl/questions/templates/rank/answering_instructions.jinja +11 -0
- edsl/questions/templates/rank/question_presentation.jinja +15 -0
- edsl/questions/templates/top_k/__init__.py +0 -0
- edsl/questions/templates/top_k/answering_instructions.jinja +8 -0
- edsl/questions/templates/top_k/question_presentation.jinja +22 -0
- edsl/questions/templates/yes_no/__init__.py +0 -0
- edsl/questions/templates/yes_no/answering_instructions.jinja +6 -0
- edsl/questions/templates/yes_no/question_presentation.jinja +12 -0
- edsl/results/CSSParameterizer.py +108 -0
- edsl/results/Dataset.py +550 -19
- edsl/results/DatasetExportMixin.py +594 -0
- edsl/results/DatasetTree.py +295 -0
- edsl/results/MarkdownToDocx.py +122 -0
- edsl/results/MarkdownToPDF.py +111 -0
- edsl/results/Result.py +477 -173
- edsl/results/Results.py +987 -269
- edsl/results/ResultsExportMixin.py +28 -125
- edsl/results/ResultsGGMixin.py +83 -15
- edsl/results/TableDisplay.py +125 -0
- edsl/results/TextEditor.py +50 -0
- edsl/results/__init__.py +1 -1
- edsl/results/file_exports.py +252 -0
- edsl/results/results_fetch_mixin.py +33 -0
- edsl/results/results_selector.py +145 -0
- edsl/results/results_tools_mixin.py +98 -0
- edsl/results/smart_objects.py +96 -0
- edsl/results/table_data_class.py +12 -0
- edsl/results/table_display.css +78 -0
- edsl/results/table_renderers.py +118 -0
- edsl/results/tree_explore.py +115 -0
- edsl/scenarios/ConstructDownloadLink.py +109 -0
- edsl/scenarios/DocumentChunker.py +102 -0
- edsl/scenarios/DocxScenario.py +16 -0
- edsl/scenarios/FileStore.py +543 -0
- edsl/scenarios/PdfExtractor.py +40 -0
- edsl/scenarios/Scenario.py +431 -62
- edsl/scenarios/ScenarioHtmlMixin.py +65 -0
- edsl/scenarios/ScenarioList.py +1415 -45
- edsl/scenarios/ScenarioListExportMixin.py +45 -0
- edsl/scenarios/ScenarioListPdfMixin.py +239 -0
- edsl/scenarios/__init__.py +2 -0
- edsl/scenarios/directory_scanner.py +96 -0
- edsl/scenarios/file_methods.py +85 -0
- edsl/scenarios/handlers/__init__.py +13 -0
- edsl/scenarios/handlers/csv.py +49 -0
- edsl/scenarios/handlers/docx.py +76 -0
- edsl/scenarios/handlers/html.py +37 -0
- edsl/scenarios/handlers/json.py +111 -0
- edsl/scenarios/handlers/latex.py +5 -0
- edsl/scenarios/handlers/md.py +51 -0
- edsl/scenarios/handlers/pdf.py +68 -0
- edsl/scenarios/handlers/png.py +39 -0
- edsl/scenarios/handlers/pptx.py +105 -0
- edsl/scenarios/handlers/py.py +294 -0
- edsl/scenarios/handlers/sql.py +313 -0
- edsl/scenarios/handlers/sqlite.py +149 -0
- edsl/scenarios/handlers/txt.py +33 -0
- edsl/scenarios/scenario_join.py +131 -0
- edsl/scenarios/scenario_selector.py +156 -0
- edsl/shared.py +1 -0
- edsl/study/ObjectEntry.py +173 -0
- edsl/study/ProofOfWork.py +113 -0
- edsl/study/SnapShot.py +80 -0
- edsl/study/Study.py +521 -0
- edsl/study/__init__.py +4 -0
- edsl/surveys/ConstructDAG.py +92 -0
- edsl/surveys/DAG.py +92 -11
- edsl/surveys/EditSurvey.py +221 -0
- edsl/surveys/InstructionHandler.py +100 -0
- edsl/surveys/Memory.py +9 -4
- edsl/surveys/MemoryManagement.py +72 -0
- edsl/surveys/MemoryPlan.py +156 -35
- edsl/surveys/Rule.py +221 -74
- edsl/surveys/RuleCollection.py +241 -61
- edsl/surveys/RuleManager.py +172 -0
- edsl/surveys/Simulator.py +75 -0
- edsl/surveys/Survey.py +1079 -339
- edsl/surveys/SurveyCSS.py +273 -0
- edsl/surveys/SurveyExportMixin.py +235 -40
- edsl/surveys/SurveyFlowVisualization.py +181 -0
- edsl/surveys/SurveyQualtricsImport.py +284 -0
- edsl/surveys/SurveyToApp.py +141 -0
- edsl/surveys/__init__.py +4 -2
- edsl/surveys/base.py +19 -3
- edsl/surveys/descriptors.py +17 -6
- edsl/surveys/instructions/ChangeInstruction.py +48 -0
- edsl/surveys/instructions/Instruction.py +56 -0
- edsl/surveys/instructions/InstructionCollection.py +82 -0
- edsl/surveys/instructions/__init__.py +0 -0
- edsl/templates/error_reporting/base.html +24 -0
- edsl/templates/error_reporting/exceptions_by_model.html +35 -0
- edsl/templates/error_reporting/exceptions_by_question_name.html +17 -0
- edsl/templates/error_reporting/exceptions_by_type.html +17 -0
- edsl/templates/error_reporting/interview_details.html +116 -0
- edsl/templates/error_reporting/interviews.html +19 -0
- edsl/templates/error_reporting/overview.html +5 -0
- edsl/templates/error_reporting/performance_plot.html +2 -0
- edsl/templates/error_reporting/report.css +74 -0
- edsl/templates/error_reporting/report.html +118 -0
- edsl/templates/error_reporting/report.js +25 -0
- edsl/tools/__init__.py +1 -0
- edsl/tools/clusters.py +192 -0
- edsl/tools/embeddings.py +27 -0
- edsl/tools/embeddings_plotting.py +118 -0
- edsl/tools/plotting.py +112 -0
- edsl/tools/summarize.py +18 -0
- edsl/utilities/PrettyList.py +56 -0
- edsl/utilities/SystemInfo.py +5 -0
- edsl/utilities/__init__.py +21 -20
- edsl/utilities/ast_utilities.py +3 -0
- edsl/utilities/data/Registry.py +2 -0
- edsl/utilities/decorators.py +41 -0
- edsl/utilities/gcp_bucket/__init__.py +0 -0
- edsl/utilities/gcp_bucket/cloud_storage.py +96 -0
- edsl/utilities/interface.py +310 -60
- edsl/utilities/is_notebook.py +18 -0
- edsl/utilities/is_valid_variable_name.py +11 -0
- edsl/utilities/naming_utilities.py +263 -0
- edsl/utilities/remove_edsl_version.py +24 -0
- edsl/utilities/repair_functions.py +28 -0
- edsl/utilities/restricted_python.py +70 -0
- edsl/utilities/utilities.py +203 -13
- edsl-0.1.40.dist-info/METADATA +111 -0
- edsl-0.1.40.dist-info/RECORD +362 -0
- {edsl-0.1.14.dist-info → edsl-0.1.40.dist-info}/WHEEL +1 -1
- edsl/agents/AgentListExportMixin.py +0 -24
- edsl/coop/old.py +0 -31
- edsl/data/Database.py +0 -141
- edsl/data/crud.py +0 -121
- edsl/jobs/Interview.py +0 -417
- edsl/jobs/JobsRunner.py +0 -63
- edsl/jobs/JobsRunnerStatusMixin.py +0 -115
- edsl/jobs/base.py +0 -47
- edsl/jobs/buckets.py +0 -166
- edsl/jobs/runners/JobsRunnerDryRun.py +0 -19
- edsl/jobs/runners/JobsRunnerStreaming.py +0 -54
- edsl/jobs/task_management.py +0 -218
- edsl/jobs/token_tracking.py +0 -78
- edsl/language_models/DeepInfra.py +0 -69
- edsl/language_models/OpenAI.py +0 -98
- edsl/language_models/model_interfaces/GeminiPro.py +0 -66
- edsl/language_models/model_interfaces/LanguageModelOpenAIFour.py +0 -8
- edsl/language_models/model_interfaces/LanguageModelOpenAIThreeFiveTurbo.py +0 -8
- edsl/language_models/model_interfaces/LlamaTwo13B.py +0 -21
- edsl/language_models/model_interfaces/LlamaTwo70B.py +0 -21
- edsl/language_models/model_interfaces/Mixtral8x7B.py +0 -24
- edsl/language_models/registry.py +0 -81
- edsl/language_models/schemas.py +0 -15
- edsl/language_models/unused/ReplicateBase.py +0 -83
- edsl/prompts/QuestionInstructionsBase.py +0 -6
- edsl/prompts/library/agent_instructions.py +0 -29
- edsl/prompts/library/agent_persona.py +0 -17
- edsl/prompts/library/question_budget.py +0 -26
- edsl/prompts/library/question_checkbox.py +0 -32
- edsl/prompts/library/question_extract.py +0 -19
- edsl/prompts/library/question_freetext.py +0 -14
- edsl/prompts/library/question_linear_scale.py +0 -20
- edsl/prompts/library/question_list.py +0 -22
- edsl/prompts/library/question_multiple_choice.py +0 -44
- edsl/prompts/library/question_numerical.py +0 -31
- edsl/prompts/library/question_rank.py +0 -21
- edsl/prompts/prompt_config.py +0 -33
- edsl/prompts/registry.py +0 -185
- edsl/questions/Question.py +0 -240
- edsl/report/InputOutputDataTypes.py +0 -134
- edsl/report/RegressionMixin.py +0 -28
- edsl/report/ReportOutputs.py +0 -1228
- edsl/report/ResultsFetchMixin.py +0 -106
- edsl/report/ResultsOutputMixin.py +0 -14
- edsl/report/demo.ipynb +0 -645
- edsl/results/ResultsDBMixin.py +0 -184
- edsl/surveys/SurveyFlowVisualizationMixin.py +0 -92
- edsl/trackers/Tracker.py +0 -91
- edsl/trackers/TrackerAPI.py +0 -196
- edsl/trackers/TrackerTasks.py +0 -70
- edsl/utilities/pastebin.py +0 -141
- edsl-0.1.14.dist-info/METADATA +0 -69
- edsl-0.1.14.dist-info/RECORD +0 -141
- /edsl/{language_models/model_interfaces → inference_services}/__init__.py +0 -0
- /edsl/{report/__init__.py → jobs/runners/JobsRunnerStatusData.py} +0 -0
- /edsl/{trackers/__init__.py → language_models/ServiceDataSources.py} +0 -0
- {edsl-0.1.14.dist-info → edsl-0.1.40.dist-info}/LICENSE +0 -0
@@ -0,0 +1,188 @@
|
|
1
|
+
from abc import ABC, abstractmethod
|
2
|
+
from typing import Optional, Any, List, TypedDict
|
3
|
+
|
4
|
+
from pydantic import BaseModel, Field, field_validator, ValidationError
|
5
|
+
|
6
|
+
from edsl.exceptions.questions import QuestionAnswerValidationError
|
7
|
+
from edsl.questions.ExceptionExplainer import ExceptionExplainer
|
8
|
+
|
9
|
+
from edsl.questions.data_structures import (
|
10
|
+
RawEdslAnswerDict,
|
11
|
+
EdslAnswerDict,
|
12
|
+
)
|
13
|
+
|
14
|
+
|
15
|
+
class ResponseValidatorABC(ABC):
|
16
|
+
required_params: List[str] = []
|
17
|
+
|
18
|
+
def __init_subclass__(cls, **kwargs):
|
19
|
+
"""This is a metaclass that ensures that all subclasses of ResponseValidatorABC have the required class variables."""
|
20
|
+
super().__init_subclass__(**kwargs)
|
21
|
+
required_class_vars = ["required_params", "valid_examples", "invalid_examples"]
|
22
|
+
for var in required_class_vars:
|
23
|
+
if not hasattr(cls, var):
|
24
|
+
raise ValueError(f"Class {cls.__name__} must have a '{var}' attribute.")
|
25
|
+
|
26
|
+
def __init__(
|
27
|
+
self,
|
28
|
+
response_model: type[BaseModel],
|
29
|
+
exception_to_throw: Optional[Exception] = None,
|
30
|
+
override_answer: Optional[dict] = None,
|
31
|
+
**kwargs,
|
32
|
+
):
|
33
|
+
self.response_model = response_model
|
34
|
+
self.exception_to_throw = exception_to_throw # for testing
|
35
|
+
self.override_answer = override_answer # for testing
|
36
|
+
self.original_exception = None
|
37
|
+
|
38
|
+
# Validate required parameters
|
39
|
+
missing_params = [
|
40
|
+
param for param in self.required_params if param not in kwargs
|
41
|
+
]
|
42
|
+
if missing_params:
|
43
|
+
raise ValueError(
|
44
|
+
f"Missing required parameters: {', '.join(missing_params)}"
|
45
|
+
)
|
46
|
+
|
47
|
+
# Set attributes
|
48
|
+
for key, value in kwargs.items():
|
49
|
+
setattr(self, key, value)
|
50
|
+
|
51
|
+
if not hasattr(self, "permissive"):
|
52
|
+
self.permissive = False
|
53
|
+
|
54
|
+
self.fixes_tried = 0 # how many times we've tried to fix the answer
|
55
|
+
|
56
|
+
def _preprocess(self, data: RawEdslAnswerDict) -> RawEdslAnswerDict:
|
57
|
+
"""This is for testing purposes. A question can be given an exception to throw or an answer to always return.
|
58
|
+
|
59
|
+
>>> rv = ResponseValidatorABC.example()
|
60
|
+
>>> rv.override_answer = {"answer": 42}
|
61
|
+
>>> rv.validate({"answer": 23})
|
62
|
+
{'answer': 42, 'comment': None, 'generated_tokens': None}
|
63
|
+
"""
|
64
|
+
if self.exception_to_throw:
|
65
|
+
raise self.exception_to_throw
|
66
|
+
return self.override_answer if self.override_answer else data
|
67
|
+
|
68
|
+
def _base_validate(self, data: RawEdslAnswerDict) -> BaseModel:
|
69
|
+
"""This is the main validation function. It takes the response_model and checks the data against it,
|
70
|
+
returning the instantiated model.
|
71
|
+
|
72
|
+
>>> rv = ResponseValidatorABC.example("numerical")
|
73
|
+
>>> rv._base_validate({"answer": 42})
|
74
|
+
ConstrainedNumericResponse(answer=42, comment=None, generated_tokens=None)
|
75
|
+
"""
|
76
|
+
try:
|
77
|
+
return self.response_model(**data)
|
78
|
+
except ValidationError as e:
|
79
|
+
raise QuestionAnswerValidationError(
|
80
|
+
message=str(e), pydantic_error=e, data=data, model=self.response_model
|
81
|
+
)
|
82
|
+
|
83
|
+
def post_validation_answer_convert(self, data):
|
84
|
+
return data
|
85
|
+
|
86
|
+
def validate(
|
87
|
+
self,
|
88
|
+
raw_edsl_answer_dict: RawEdslAnswerDict,
|
89
|
+
fix=False,
|
90
|
+
verbose=False,
|
91
|
+
replacement_dict: dict = None,
|
92
|
+
) -> EdslAnswerDict:
|
93
|
+
"""This is the main validation function.
|
94
|
+
|
95
|
+
>>> rv = ResponseValidatorABC.example("numerical")
|
96
|
+
>>> rv.validate({"answer": 42})
|
97
|
+
{'answer': 42, 'comment': None, 'generated_tokens': None}
|
98
|
+
>>> rv.max_value
|
99
|
+
86.7
|
100
|
+
>>> rv.validate({"answer": "120"})
|
101
|
+
Traceback (most recent call last):
|
102
|
+
...
|
103
|
+
edsl.exceptions.questions.QuestionAnswerValidationError:...
|
104
|
+
>>> from edsl import QuestionNumerical
|
105
|
+
>>> q = QuestionNumerical.example()
|
106
|
+
>>> q.permissive = True
|
107
|
+
>>> rv = q.response_validator
|
108
|
+
>>> rv.validate({"answer": "120"})
|
109
|
+
{'answer': 120, 'comment': None, 'generated_tokens': None}
|
110
|
+
>>> rv.validate({"answer": "poo"})
|
111
|
+
Traceback (most recent call last):
|
112
|
+
...
|
113
|
+
edsl.exceptions.questions.QuestionAnswerValidationError:...
|
114
|
+
"""
|
115
|
+
proposed_edsl_answer_dict = self._preprocess(raw_edsl_answer_dict)
|
116
|
+
try:
|
117
|
+
pydantic_edsl_answer: BaseModel = self._base_validate(
|
118
|
+
proposed_edsl_answer_dict
|
119
|
+
)
|
120
|
+
edsl_answer_dict = self._extract_answer(pydantic_edsl_answer)
|
121
|
+
return self._post_process(edsl_answer_dict)
|
122
|
+
except QuestionAnswerValidationError as e:
|
123
|
+
return self._handle_exception(e, raw_edsl_answer_dict)
|
124
|
+
|
125
|
+
def human_explanation(self, e: QuestionAnswerValidationError):
|
126
|
+
explanation = ExceptionExplainer(e, model_response=e.data).explain()
|
127
|
+
return explanation
|
128
|
+
|
129
|
+
def _handle_exception(self, e: Exception, raw_edsl_answer_dict) -> EdslAnswerDict:
|
130
|
+
if self.fixes_tried == 0:
|
131
|
+
self.original_exception = e
|
132
|
+
|
133
|
+
if self.fixes_tried == 0 and hasattr(self, "fix"):
|
134
|
+
self.fixes_tried += 1
|
135
|
+
fixed_data = self.fix(raw_edsl_answer_dict)
|
136
|
+
try:
|
137
|
+
return self.validate(fixed_data, fix=True) # early return if validates
|
138
|
+
except Exception as e:
|
139
|
+
pass # we don't log failed fixes
|
140
|
+
|
141
|
+
# If the exception is already a QuestionAnswerValidationError, raise it
|
142
|
+
if isinstance(self.original_exception, QuestionAnswerValidationError):
|
143
|
+
raise self.original_exception
|
144
|
+
|
145
|
+
# If nothing worked, raise the original exception
|
146
|
+
raise QuestionAnswerValidationError(
|
147
|
+
message=self.original_exception,
|
148
|
+
pydantic_error=self.original_exception,
|
149
|
+
data=raw_edsl_answer_dict,
|
150
|
+
model=self.response_model,
|
151
|
+
)
|
152
|
+
|
153
|
+
def _check_constraints(self, pydantic_edsl_answer: BaseModel) -> dict:
|
154
|
+
pass
|
155
|
+
|
156
|
+
def _extract_answer(self, response: BaseModel) -> EdslAnswerDict:
|
157
|
+
return response.model_dump()
|
158
|
+
|
159
|
+
def _post_process(self, edsl_answer_dict: EdslAnswerDict) -> EdslAnswerDict:
|
160
|
+
return edsl_answer_dict
|
161
|
+
|
162
|
+
@classmethod
|
163
|
+
def example(cls, question_type="numerical"):
|
164
|
+
from edsl import Question
|
165
|
+
|
166
|
+
q = Question.example(question_type)
|
167
|
+
return q.response_validator
|
168
|
+
|
169
|
+
|
170
|
+
def main():
|
171
|
+
rv = ResponseValidatorABC.example()
|
172
|
+
print(rv.validate({"answer": 42}))
|
173
|
+
|
174
|
+
|
175
|
+
# Example usage
|
176
|
+
if __name__ == "__main__":
|
177
|
+
import doctest
|
178
|
+
|
179
|
+
doctest.testmod(optionflags=doctest.ELLIPSIS)
|
180
|
+
|
181
|
+
rv = ResponseValidatorABC.example()
|
182
|
+
# print(rv.validate({"answer": 42}))
|
183
|
+
|
184
|
+
rv = ResponseValidatorABC.example()
|
185
|
+
try:
|
186
|
+
rv.validate({"answer": "120"})
|
187
|
+
except QuestionAnswerValidationError as e:
|
188
|
+
print(rv.human_explanation(e))
|
@@ -0,0 +1,34 @@
|
|
1
|
+
from edsl.questions.data_structures import BaseModel
|
2
|
+
from edsl.questions.response_validator_abc import ResponseValidatorABC
|
3
|
+
|
4
|
+
|
5
|
+
class ResponseValidatorFactory:
|
6
|
+
"""Factory class to create a response validator for a question."""
|
7
|
+
|
8
|
+
def __init__(self, question):
|
9
|
+
self.question = question
|
10
|
+
|
11
|
+
@property
|
12
|
+
def response_model(self) -> type["BaseModel"]:
|
13
|
+
if self.question._response_model is not None:
|
14
|
+
return self.question._response_model
|
15
|
+
else:
|
16
|
+
return self.question.create_response_model()
|
17
|
+
|
18
|
+
@property
|
19
|
+
def response_validator(self) -> "ResponseValidatorABC":
|
20
|
+
"""Return the response validator."""
|
21
|
+
params = (
|
22
|
+
{
|
23
|
+
"response_model": self.question.response_model,
|
24
|
+
}
|
25
|
+
| {k: getattr(self.question, k) for k in self.validator_parameters}
|
26
|
+
| {"exception_to_throw": getattr(self.question, "exception_to_throw", None)}
|
27
|
+
| {"override_answer": getattr(self.question, "override_answer", None)}
|
28
|
+
)
|
29
|
+
return self.question.response_validator_class(**params)
|
30
|
+
|
31
|
+
@property
|
32
|
+
def validator_parameters(self) -> list[str]:
|
33
|
+
"""Return the parameters required for the response validator."""
|
34
|
+
return self.question.response_validator_class.required_params
|
edsl/questions/settings.py
CHANGED
@@ -1,9 +1,12 @@
|
|
1
|
+
"""Settings for the questions module."""
|
2
|
+
|
3
|
+
|
1
4
|
class Settings:
|
2
5
|
"""Settings for the questions module."""
|
3
6
|
|
4
7
|
MAX_ANSWER_LENGTH = 2000
|
5
8
|
MAX_EXPRESSION_CONSTRAINT_LENGTH = 1000
|
6
|
-
MAX_NUM_OPTIONS =
|
9
|
+
MAX_NUM_OPTIONS = 200
|
7
10
|
MIN_NUM_OPTIONS = 2
|
8
|
-
MAX_OPTION_LENGTH =
|
11
|
+
MAX_OPTION_LENGTH = 10000
|
9
12
|
MAX_QUESTION_LENGTH = 100000
|
File without changes
|
File without changes
|
@@ -0,0 +1,7 @@
|
|
1
|
+
Return only a comma-separated list the values in the same order as the options, with 0s included, on one line, in square braces.
|
2
|
+
|
3
|
+
Example: if there are 4 options, the response should be "[25,25,25,25]" to allocate 25 to each option.
|
4
|
+
|
5
|
+
{% if include_comment %}
|
6
|
+
After the answer, you can put a comment explaining your choice on the next line.
|
7
|
+
{% endif %}
|
File without changes
|
@@ -0,0 +1,10 @@
|
|
1
|
+
{# Answering Instructions #}
|
2
|
+
{% if use_code %}
|
3
|
+
Please respond only with a comma-separated list of the code of the options that apply, with square brackets. E.g., [0, 1, 3]
|
4
|
+
{% else %}
|
5
|
+
Please respond only with a comma-separated list of the options that apply, with square brackets. E.g., ['Good', 'Bad', 'Ugly']
|
6
|
+
{% endif %}
|
7
|
+
{% if include_comment %}
|
8
|
+
After the answer, you can put a comment explaining your choice on the next line.
|
9
|
+
{% endif %}
|
10
|
+
|
@@ -0,0 +1,22 @@
|
|
1
|
+
{{question_text}}
|
2
|
+
{% if use_code %}
|
3
|
+
{% for option in question_options %}
|
4
|
+
{{ loop.index0 }}: {{option}}
|
5
|
+
{% endfor %}
|
6
|
+
{% else %}
|
7
|
+
{% for option in question_options %}
|
8
|
+
{{ option }}
|
9
|
+
{% endfor %}
|
10
|
+
{% endif %}
|
11
|
+
|
12
|
+
{# Restrictions #}
|
13
|
+
{% if min_selections != None and max_selections != None and min_selections == max_selections %}
|
14
|
+
You must select exactly {{min_selections}} options.
|
15
|
+
{% elif min_selections != None and max_selections != None %}
|
16
|
+
Minimum number of options that must be selected: {{min_selections}}.
|
17
|
+
Maximum number of options that must be selected: {{max_selections}}.
|
18
|
+
{% elif min_selections != None %}
|
19
|
+
Minimum number of options that must be selected: {{min_selections}}.
|
20
|
+
{% elif max_selections != None %}
|
21
|
+
Maximum number of options that must be selected: {{max_selections}}.
|
22
|
+
{% endif %}
|
File without changes
|
@@ -0,0 +1,21 @@
|
|
1
|
+
Please respond with a dictionary using the following keys: {{ answer_keys | join(', ') }}.
|
2
|
+
|
3
|
+
{% if value_descriptions %}
|
4
|
+
Here are descriptions of the values to provide:
|
5
|
+
{% for idx in range(answer_keys | length) %}
|
6
|
+
- "{{ answer_keys[idx] }}": "{{ value_descriptions[idx] }}"
|
7
|
+
{% endfor %}
|
8
|
+
{% endif %}
|
9
|
+
|
10
|
+
{% if value_types %}
|
11
|
+
The values should be formatted in the following types:
|
12
|
+
{% for idx in range(answer_keys | length) %}
|
13
|
+
- "{{ answer_keys[idx] }}": "{{ value_types[idx] }}"
|
14
|
+
{% endfor %}
|
15
|
+
{% endif %}
|
16
|
+
|
17
|
+
If you do not have a value for a given key, use "null".
|
18
|
+
|
19
|
+
{% if include_comment %}
|
20
|
+
After the answer, you can put a comment explaining your response on the next line.
|
21
|
+
{% endif %}
|
@@ -0,0 +1 @@
|
|
1
|
+
{{question_text}}
|
File without changes
|
@@ -0,0 +1 @@
|
|
1
|
+
{{question_text}}
|
File without changes
|
File without changes
|
@@ -0,0 +1 @@
|
|
1
|
+
{{question_text}}
|
File without changes
|
@@ -0,0 +1,10 @@
|
|
1
|
+
{# Answering Instructions #}
|
2
|
+
{% if use_code %}
|
3
|
+
Respond only with the code corresponding to one of the options.
|
4
|
+
{% else %}
|
5
|
+
Respond only with a string corresponding to one of the options.
|
6
|
+
{% endif %}
|
7
|
+
{% if include_comment %}
|
8
|
+
After the answer, you can put a comment explaining why you chose that option on the next line.
|
9
|
+
{% endif %}
|
10
|
+
|
@@ -0,0 +1,12 @@
|
|
1
|
+
{# Question Presention #}
|
2
|
+
{{question_text}}
|
3
|
+
{% if use_code %}
|
4
|
+
{%- for option in question_options %}
|
5
|
+
{{ loop.index0 }}: {{option}}
|
6
|
+
{% endfor %}
|
7
|
+
{% else %}
|
8
|
+
{% for option in question_options %}
|
9
|
+
{{option}}
|
10
|
+
{% endfor %}
|
11
|
+
{% endif %}
|
12
|
+
Only 1 option may be selected.
|
File without changes
|
File without changes
|
@@ -0,0 +1 @@
|
|
1
|
+
|
@@ -0,0 +1,20 @@
|
|
1
|
+
{{question_text}}
|
2
|
+
|
3
|
+
Rows:
|
4
|
+
{% for item in question_items %}
|
5
|
+
{{ loop.index0 }}: {{item}}
|
6
|
+
{% endfor %}
|
7
|
+
|
8
|
+
Columns:
|
9
|
+
{% for option in question_options %}
|
10
|
+
{{ loop.index0 }}: {{option}}
|
11
|
+
{%- if option in option_labels %}
|
12
|
+
({{option_labels[option]}})
|
13
|
+
{%- endif %}
|
14
|
+
{% endfor %}
|
15
|
+
|
16
|
+
|
17
|
+
Select one column option for each row.
|
18
|
+
{% if required %}
|
19
|
+
All rows require a response.
|
20
|
+
{% endif %}
|
File without changes
|
@@ -0,0 +1,9 @@
|
|
1
|
+
{# Answering Instructions #}
|
2
|
+
{% if use_code %}
|
3
|
+
Respond only with the code corresponding to one of the options.
|
4
|
+
{% else %}
|
5
|
+
Respond only with a string corresponding to one of the options.
|
6
|
+
{% endif %}
|
7
|
+
{% if include_comment %}
|
8
|
+
After the answer, you can put a comment explaining why you chose that option on the next line.
|
9
|
+
{% endif %}
|
File without changes
|
@@ -0,0 +1,12 @@
|
|
1
|
+
{# Question Presention #}
|
2
|
+
{{question_text}}
|
3
|
+
{% if use_code %}
|
4
|
+
{%- for option in question_options %}
|
5
|
+
{{ loop.index0 }}: {{option}}
|
6
|
+
{% endfor %}
|
7
|
+
{% else %}
|
8
|
+
{% for option in question_options %}
|
9
|
+
{{option}}
|
10
|
+
{% endfor %}
|
11
|
+
{% endif %}
|
12
|
+
Only 1 option may be selected.
|
File without changes
|
@@ -0,0 +1,7 @@
|
|
1
|
+
This question requires a numerical response in the form of an integer or decimal (e.g., -12, 0, 1, 2, 3.45, ...).
|
2
|
+
Respond with just your number on a single line.
|
3
|
+
If your response is equivalent to zero, report '0'
|
4
|
+
|
5
|
+
{% if include_comment %}
|
6
|
+
After the answer, put a comment explaining your choice on the next line.
|
7
|
+
{% endif %}
|
File without changes
|
@@ -0,0 +1,11 @@
|
|
1
|
+
{# Answering Instructions #}
|
2
|
+
{% if use_code %}
|
3
|
+
Please respond only with a comma-separated list of the code of the raked options, with square brackets. E.g., [0, 1, 3]
|
4
|
+
{% else %}
|
5
|
+
Please respond only with a comma-separated list of the ranked options, with square brackets. E.g., ['Good', 'Bad', 'Ugly']
|
6
|
+
{% endif %}
|
7
|
+
{% if include_comment %}
|
8
|
+
After the answer, you can put a comment explaining your choice on the next line.
|
9
|
+
{% endif %}
|
10
|
+
|
11
|
+
|
@@ -0,0 +1,15 @@
|
|
1
|
+
{{question_text}}
|
2
|
+
{% if use_code %}
|
3
|
+
The options are
|
4
|
+
{% for option in question_options %}
|
5
|
+
{{ loop.index0 }}: {{option}}
|
6
|
+
{% endfor %}
|
7
|
+
{% else %}
|
8
|
+
The options are:
|
9
|
+
{% for option in question_options %}
|
10
|
+
{{option}}
|
11
|
+
{% endfor %}
|
12
|
+
{% endif %}
|
13
|
+
{% if num_selections %}
|
14
|
+
You can inlcude up to {{num_selections}} options in your answer.
|
15
|
+
{% endif %}
|
File without changes
|
@@ -0,0 +1,8 @@
|
|
1
|
+
{# Answering Instructions #}
|
2
|
+
Please respond with valid JSON, formatted like so:
|
3
|
+
{% if include_comment %}
|
4
|
+
{"answer": [<put comma-separated list here>], "comment": "<put explanation here>"}
|
5
|
+
{% else %}
|
6
|
+
{"answer": [<put comma-separated list here>]}
|
7
|
+
{% endif %}
|
8
|
+
|
@@ -0,0 +1,22 @@
|
|
1
|
+
{{question_text}}
|
2
|
+
{% if use_code %}
|
3
|
+
{% for option in question_options %}
|
4
|
+
{{ loop.index0 }}: {{option}}
|
5
|
+
{% endfor %}
|
6
|
+
{% else %}
|
7
|
+
{% for option in question_options %}
|
8
|
+
{{ option }}
|
9
|
+
{% endfor %}
|
10
|
+
{% endif %}
|
11
|
+
|
12
|
+
{# Restrictions #}
|
13
|
+
{% if min_selections != None and max_selections != None and min_selections == max_selections %}
|
14
|
+
You must select exactly {{min_selections}} options.
|
15
|
+
{% elif min_selections != None and max_selections != None %}
|
16
|
+
Minimum number of options that must be selected: {{min_selections}}.
|
17
|
+
Maximum number of options that must be selected: {{max_selections}}.
|
18
|
+
{% elif min_selections != None %}
|
19
|
+
Minimum number of options that must be selected: {{min_selections}}.
|
20
|
+
{% elif max_selections != None %}
|
21
|
+
Maximum number of options that must be selected: {{max_selections}}.
|
22
|
+
{% endif %}
|
File without changes
|
@@ -0,0 +1,12 @@
|
|
1
|
+
{# Question Presention #}
|
2
|
+
{{question_text}}
|
3
|
+
{% if use_code %}
|
4
|
+
{%- for option in question_options %}
|
5
|
+
{{ loop.index0 }}: {{option}}
|
6
|
+
{% endfor %}
|
7
|
+
{% else %}
|
8
|
+
{% for option in question_options %}
|
9
|
+
{{option}}
|
10
|
+
{% endfor %}
|
11
|
+
{% endif %}
|
12
|
+
Only 1 option may be selected.
|