edsl 0.1.39.dev3__py3-none-any.whl → 0.1.39.dev5__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- edsl/Base.py +413 -332
- edsl/BaseDiff.py +260 -260
- edsl/TemplateLoader.py +24 -24
- edsl/__init__.py +57 -49
- edsl/__version__.py +1 -1
- edsl/agents/Agent.py +1071 -867
- edsl/agents/AgentList.py +551 -413
- edsl/agents/Invigilator.py +284 -233
- edsl/agents/InvigilatorBase.py +257 -270
- edsl/agents/PromptConstructor.py +272 -354
- edsl/agents/QuestionInstructionPromptBuilder.py +128 -0
- edsl/agents/QuestionTemplateReplacementsBuilder.py +137 -0
- edsl/agents/__init__.py +2 -3
- edsl/agents/descriptors.py +99 -99
- edsl/agents/prompt_helpers.py +129 -129
- edsl/agents/question_option_processor.py +172 -0
- edsl/auto/AutoStudy.py +130 -117
- edsl/auto/StageBase.py +243 -230
- edsl/auto/StageGenerateSurvey.py +178 -178
- edsl/auto/StageLabelQuestions.py +125 -125
- edsl/auto/StagePersona.py +61 -61
- edsl/auto/StagePersonaDimensionValueRanges.py +88 -88
- edsl/auto/StagePersonaDimensionValues.py +74 -74
- edsl/auto/StagePersonaDimensions.py +69 -69
- edsl/auto/StageQuestions.py +74 -73
- edsl/auto/SurveyCreatorPipeline.py +21 -21
- edsl/auto/utilities.py +218 -224
- edsl/base/Base.py +279 -279
- edsl/config.py +177 -157
- edsl/conversation/Conversation.py +290 -290
- edsl/conversation/car_buying.py +59 -58
- edsl/conversation/chips.py +95 -95
- edsl/conversation/mug_negotiation.py +81 -81
- edsl/conversation/next_speaker_utilities.py +93 -93
- edsl/coop/CoopFunctionsMixin.py +15 -0
- edsl/coop/ExpectedParrotKeyHandler.py +125 -0
- edsl/coop/PriceFetcher.py +54 -54
- edsl/coop/__init__.py +2 -2
- edsl/coop/coop.py +1106 -1028
- edsl/coop/utils.py +131 -131
- edsl/data/Cache.py +573 -555
- edsl/data/CacheEntry.py +230 -233
- edsl/data/CacheHandler.py +168 -149
- edsl/data/RemoteCacheSync.py +186 -78
- edsl/data/SQLiteDict.py +292 -292
- edsl/data/__init__.py +5 -4
- edsl/data/orm.py +10 -10
- edsl/data_transfer_models.py +74 -73
- edsl/enums.py +202 -175
- edsl/exceptions/BaseException.py +21 -21
- edsl/exceptions/__init__.py +54 -54
- edsl/exceptions/agents.py +54 -42
- edsl/exceptions/cache.py +5 -5
- edsl/exceptions/configuration.py +16 -16
- edsl/exceptions/coop.py +10 -10
- edsl/exceptions/data.py +14 -14
- edsl/exceptions/general.py +34 -34
- edsl/exceptions/inference_services.py +5 -0
- edsl/exceptions/jobs.py +33 -33
- edsl/exceptions/language_models.py +63 -63
- edsl/exceptions/prompts.py +15 -15
- edsl/exceptions/questions.py +109 -91
- edsl/exceptions/results.py +29 -29
- edsl/exceptions/scenarios.py +29 -22
- edsl/exceptions/surveys.py +37 -37
- edsl/inference_services/AnthropicService.py +106 -87
- edsl/inference_services/AvailableModelCacheHandler.py +184 -0
- edsl/inference_services/AvailableModelFetcher.py +215 -0
- edsl/inference_services/AwsBedrock.py +118 -120
- edsl/inference_services/AzureAI.py +215 -217
- edsl/inference_services/DeepInfraService.py +18 -18
- edsl/inference_services/GoogleService.py +143 -148
- edsl/inference_services/GroqService.py +20 -20
- edsl/inference_services/InferenceServiceABC.py +80 -147
- edsl/inference_services/InferenceServicesCollection.py +138 -97
- edsl/inference_services/MistralAIService.py +120 -123
- edsl/inference_services/OllamaService.py +18 -18
- edsl/inference_services/OpenAIService.py +236 -224
- edsl/inference_services/PerplexityService.py +160 -163
- edsl/inference_services/ServiceAvailability.py +135 -0
- edsl/inference_services/TestService.py +90 -89
- edsl/inference_services/TogetherAIService.py +172 -170
- edsl/inference_services/data_structures.py +134 -0
- edsl/inference_services/models_available_cache.py +118 -118
- edsl/inference_services/rate_limits_cache.py +25 -25
- edsl/inference_services/registry.py +41 -41
- edsl/inference_services/write_available.py +10 -10
- edsl/jobs/AnswerQuestionFunctionConstructor.py +223 -0
- edsl/jobs/Answers.py +43 -56
- edsl/jobs/FetchInvigilator.py +47 -0
- edsl/jobs/InterviewTaskManager.py +98 -0
- edsl/jobs/InterviewsConstructor.py +50 -0
- edsl/jobs/Jobs.py +823 -898
- edsl/jobs/JobsChecks.py +172 -147
- edsl/jobs/JobsComponentConstructor.py +189 -0
- edsl/jobs/JobsPrompts.py +270 -268
- edsl/jobs/JobsRemoteInferenceHandler.py +311 -239
- edsl/jobs/JobsRemoteInferenceLogger.py +239 -0
- edsl/jobs/RequestTokenEstimator.py +30 -0
- edsl/jobs/__init__.py +1 -1
- edsl/jobs/async_interview_runner.py +138 -0
- edsl/jobs/buckets/BucketCollection.py +104 -63
- edsl/jobs/buckets/ModelBuckets.py +65 -65
- edsl/jobs/buckets/TokenBucket.py +283 -251
- 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 +396 -661
- edsl/jobs/interviews/InterviewExceptionCollection.py +99 -99
- edsl/jobs/interviews/InterviewExceptionEntry.py +186 -186
- edsl/jobs/interviews/InterviewStatistic.py +63 -63
- edsl/jobs/interviews/InterviewStatisticsCollection.py +25 -25
- edsl/jobs/interviews/InterviewStatusDictionary.py +78 -78
- edsl/jobs/interviews/InterviewStatusLog.py +92 -92
- edsl/jobs/interviews/ReportErrors.py +66 -66
- edsl/jobs/interviews/interview_status_enum.py +9 -9
- 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 -466
- edsl/jobs/runners/JobsRunnerStatus.py +297 -330
- edsl/jobs/tasks/QuestionTaskCreator.py +244 -242
- edsl/jobs/tasks/TaskCreators.py +64 -64
- edsl/jobs/tasks/TaskHistory.py +470 -450
- edsl/jobs/tasks/TaskStatusLog.py +23 -23
- edsl/jobs/tasks/task_status_enum.py +161 -163
- edsl/jobs/tokens/InterviewTokenUsage.py +27 -27
- edsl/jobs/tokens/TokenUsage.py +34 -34
- edsl/language_models/ComputeCost.py +63 -0
- edsl/language_models/LanguageModel.py +626 -668
- edsl/language_models/ModelList.py +164 -155
- edsl/language_models/PriceManager.py +127 -0
- edsl/language_models/RawResponseHandler.py +106 -0
- edsl/language_models/RegisterLanguageModelsMeta.py +184 -184
- edsl/language_models/ServiceDataSources.py +0 -0
- edsl/language_models/__init__.py +2 -3
- edsl/language_models/fake_openai_call.py +15 -15
- edsl/language_models/fake_openai_service.py +61 -61
- 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 +156 -156
- edsl/language_models/utilities.py +65 -64
- edsl/notebooks/Notebook.py +263 -258
- edsl/notebooks/NotebookToLaTeX.py +142 -0
- edsl/notebooks/__init__.py +1 -1
- edsl/prompts/Prompt.py +352 -362
- edsl/prompts/__init__.py +2 -2
- edsl/questions/ExceptionExplainer.py +77 -0
- edsl/questions/HTMLQuestion.py +103 -0
- edsl/questions/QuestionBase.py +518 -664
- edsl/questions/QuestionBasePromptsMixin.py +221 -217
- edsl/questions/QuestionBudget.py +227 -227
- edsl/questions/QuestionCheckBox.py +359 -359
- edsl/questions/QuestionExtract.py +180 -182
- edsl/questions/QuestionFreeText.py +113 -114
- edsl/questions/QuestionFunctional.py +166 -166
- edsl/questions/QuestionList.py +223 -231
- edsl/questions/QuestionMatrix.py +265 -0
- edsl/questions/QuestionMultipleChoice.py +330 -286
- edsl/questions/QuestionNumerical.py +151 -153
- edsl/questions/QuestionRank.py +314 -324
- edsl/questions/Quick.py +41 -41
- edsl/questions/SimpleAskMixin.py +74 -73
- edsl/questions/__init__.py +27 -26
- edsl/questions/{AnswerValidatorMixin.py → answer_validator_mixin.py} +334 -289
- edsl/questions/compose_questions.py +98 -98
- edsl/questions/data_structures.py +20 -0
- edsl/questions/decorators.py +21 -21
- edsl/questions/derived/QuestionLikertFive.py +76 -76
- edsl/questions/derived/QuestionLinearScale.py +90 -87
- edsl/questions/derived/QuestionTopK.py +93 -93
- edsl/questions/derived/QuestionYesNo.py +82 -82
- edsl/questions/descriptors.py +427 -413
- edsl/questions/loop_processor.py +149 -0
- edsl/questions/prompt_templates/question_budget.jinja +13 -13
- edsl/questions/prompt_templates/question_checkbox.jinja +32 -32
- edsl/questions/prompt_templates/question_extract.jinja +11 -11
- edsl/questions/prompt_templates/question_free_text.jinja +3 -3
- edsl/questions/prompt_templates/question_linear_scale.jinja +11 -11
- edsl/questions/prompt_templates/question_list.jinja +17 -17
- edsl/questions/prompt_templates/question_multiple_choice.jinja +33 -33
- edsl/questions/prompt_templates/question_numerical.jinja +36 -36
- edsl/questions/{QuestionBaseGenMixin.py → question_base_gen_mixin.py} +168 -161
- edsl/questions/question_registry.py +177 -177
- edsl/questions/{RegisterQuestionsMeta.py → register_questions_meta.py} +71 -71
- edsl/questions/{ResponseValidatorABC.py → response_validator_abc.py} +188 -174
- edsl/questions/response_validator_factory.py +34 -0
- edsl/questions/settings.py +12 -12
- edsl/questions/templates/budget/answering_instructions.jinja +7 -7
- edsl/questions/templates/budget/question_presentation.jinja +7 -7
- edsl/questions/templates/checkbox/answering_instructions.jinja +10 -10
- edsl/questions/templates/checkbox/question_presentation.jinja +22 -22
- edsl/questions/templates/extract/answering_instructions.jinja +7 -7
- edsl/questions/templates/likert_five/answering_instructions.jinja +10 -10
- edsl/questions/templates/likert_five/question_presentation.jinja +11 -11
- edsl/questions/templates/linear_scale/answering_instructions.jinja +5 -5
- edsl/questions/templates/linear_scale/question_presentation.jinja +5 -5
- edsl/questions/templates/list/answering_instructions.jinja +3 -3
- edsl/questions/templates/list/question_presentation.jinja +5 -5
- 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/answering_instructions.jinja +9 -9
- edsl/questions/templates/multiple_choice/question_presentation.jinja +11 -11
- edsl/questions/templates/numerical/answering_instructions.jinja +6 -6
- edsl/questions/templates/numerical/question_presentation.jinja +6 -6
- edsl/questions/templates/rank/answering_instructions.jinja +11 -11
- edsl/questions/templates/rank/question_presentation.jinja +15 -15
- edsl/questions/templates/top_k/answering_instructions.jinja +8 -8
- edsl/questions/templates/top_k/question_presentation.jinja +22 -22
- edsl/questions/templates/yes_no/answering_instructions.jinja +6 -6
- edsl/questions/templates/yes_no/question_presentation.jinja +11 -11
- edsl/results/CSSParameterizer.py +108 -108
- edsl/results/Dataset.py +587 -424
- edsl/results/DatasetExportMixin.py +594 -731
- edsl/results/DatasetTree.py +295 -275
- edsl/results/MarkdownToDocx.py +122 -0
- edsl/results/MarkdownToPDF.py +111 -0
- edsl/results/Result.py +557 -465
- edsl/results/Results.py +1183 -1165
- edsl/results/ResultsExportMixin.py +45 -43
- edsl/results/ResultsGGMixin.py +121 -121
- edsl/results/TableDisplay.py +125 -198
- edsl/results/TextEditor.py +50 -0
- edsl/results/__init__.py +2 -2
- edsl/results/file_exports.py +252 -0
- edsl/results/{ResultsFetchMixin.py → results_fetch_mixin.py} +33 -33
- edsl/results/{Selector.py → results_selector.py} +145 -135
- edsl/results/{ResultsToolsMixin.py → results_tools_mixin.py} +98 -98
- edsl/results/smart_objects.py +96 -0
- edsl/results/table_data_class.py +12 -0
- edsl/results/table_display.css +77 -77
- edsl/results/table_renderers.py +118 -0
- edsl/results/tree_explore.py +115 -115
- edsl/scenarios/ConstructDownloadLink.py +109 -0
- edsl/scenarios/DocumentChunker.py +102 -0
- edsl/scenarios/DocxScenario.py +16 -0
- edsl/scenarios/FileStore.py +511 -632
- edsl/scenarios/PdfExtractor.py +40 -0
- edsl/scenarios/Scenario.py +498 -601
- edsl/scenarios/ScenarioHtmlMixin.py +65 -64
- edsl/scenarios/ScenarioList.py +1458 -1287
- edsl/scenarios/ScenarioListExportMixin.py +45 -52
- edsl/scenarios/ScenarioListPdfMixin.py +239 -261
- edsl/scenarios/__init__.py +3 -4
- 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 +38 -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/{ScenarioJoin.py → scenario_join.py} +131 -127
- edsl/scenarios/scenario_selector.py +156 -0
- edsl/shared.py +1 -1
- edsl/study/ObjectEntry.py +173 -173
- edsl/study/ProofOfWork.py +113 -113
- edsl/study/SnapShot.py +80 -80
- edsl/study/Study.py +521 -528
- edsl/study/__init__.py +4 -4
- edsl/surveys/ConstructDAG.py +92 -0
- edsl/surveys/DAG.py +148 -148
- edsl/surveys/EditSurvey.py +221 -0
- edsl/surveys/InstructionHandler.py +100 -0
- edsl/surveys/Memory.py +31 -31
- edsl/surveys/MemoryManagement.py +72 -0
- edsl/surveys/MemoryPlan.py +244 -244
- edsl/surveys/Rule.py +327 -326
- edsl/surveys/RuleCollection.py +385 -387
- edsl/surveys/RuleManager.py +172 -0
- edsl/surveys/Simulator.py +75 -0
- edsl/surveys/Survey.py +1280 -1801
- edsl/surveys/SurveyCSS.py +273 -261
- edsl/surveys/SurveyExportMixin.py +259 -259
- edsl/surveys/{SurveyFlowVisualizationMixin.py → SurveyFlowVisualization.py} +181 -179
- edsl/surveys/SurveyQualtricsImport.py +284 -284
- edsl/surveys/SurveyToApp.py +141 -0
- edsl/surveys/__init__.py +5 -3
- edsl/surveys/base.py +53 -53
- edsl/surveys/descriptors.py +60 -56
- edsl/surveys/instructions/ChangeInstruction.py +48 -49
- edsl/surveys/instructions/Instruction.py +56 -65
- edsl/surveys/instructions/InstructionCollection.py +82 -77
- edsl/templates/error_reporting/base.html +23 -23
- edsl/templates/error_reporting/exceptions_by_model.html +34 -34
- edsl/templates/error_reporting/exceptions_by_question_name.html +16 -16
- edsl/templates/error_reporting/exceptions_by_type.html +16 -16
- edsl/templates/error_reporting/interview_details.html +115 -115
- edsl/templates/error_reporting/interviews.html +19 -19
- edsl/templates/error_reporting/overview.html +4 -4
- edsl/templates/error_reporting/performance_plot.html +1 -1
- edsl/templates/error_reporting/report.css +73 -73
- edsl/templates/error_reporting/report.html +117 -117
- edsl/templates/error_reporting/report.js +25 -25
- edsl/tools/__init__.py +1 -1
- edsl/tools/clusters.py +192 -192
- edsl/tools/embeddings.py +27 -27
- edsl/tools/embeddings_plotting.py +118 -118
- edsl/tools/plotting.py +112 -112
- edsl/tools/summarize.py +18 -18
- edsl/utilities/PrettyList.py +56 -0
- edsl/utilities/SystemInfo.py +28 -28
- edsl/utilities/__init__.py +22 -22
- edsl/utilities/ast_utilities.py +25 -25
- edsl/utilities/data/Registry.py +6 -6
- edsl/utilities/data/__init__.py +1 -1
- edsl/utilities/data/scooter_results.json +1 -1
- edsl/utilities/decorators.py +77 -77
- edsl/utilities/gcp_bucket/cloud_storage.py +96 -96
- edsl/utilities/interface.py +627 -627
- edsl/utilities/is_notebook.py +18 -0
- edsl/utilities/is_valid_variable_name.py +11 -0
- edsl/utilities/naming_utilities.py +263 -263
- edsl/utilities/remove_edsl_version.py +24 -0
- edsl/utilities/repair_functions.py +28 -28
- edsl/utilities/restricted_python.py +70 -70
- edsl/utilities/utilities.py +436 -424
- {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/LICENSE +21 -21
- {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/METADATA +13 -11
- edsl-0.1.39.dev5.dist-info/RECORD +358 -0
- {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/WHEEL +1 -1
- edsl/language_models/KeyLookup.py +0 -30
- edsl/language_models/registry.py +0 -190
- edsl/language_models/unused/ReplicateBase.py +0 -83
- edsl/results/ResultsDBMixin.py +0 -238
- edsl-0.1.39.dev3.dist-info/RECORD +0 -277
@@ -1,184 +1,184 @@
|
|
1
|
-
from abc import ABC, ABCMeta
|
2
|
-
from typing import Any, List, Callable
|
3
|
-
import inspect
|
4
|
-
from typing import get_type_hints
|
5
|
-
from edsl.exceptions.language_models import LanguageModelAttributeTypeError
|
6
|
-
from edsl.enums import InferenceServiceType
|
7
|
-
|
8
|
-
|
9
|
-
class RegisterLanguageModelsMeta(ABCMeta):
|
10
|
-
"""Metaclass to register output elements in a registry i.e., those that have a parent."""
|
11
|
-
|
12
|
-
_registry = {} # Initialize the registry as a dictionary
|
13
|
-
REQUIRED_CLASS_ATTRIBUTES = ["_model_", "_parameters_", "_inference_service_"]
|
14
|
-
|
15
|
-
def __init__(cls, name, bases, dct):
|
16
|
-
"""Register the class in the registry if it has a _model_ attribute."""
|
17
|
-
super(RegisterLanguageModelsMeta, cls).__init__(name, bases, dct)
|
18
|
-
# if name != "LanguageModel":
|
19
|
-
if (model_name := getattr(cls, "_model_", None)) is not None:
|
20
|
-
RegisterLanguageModelsMeta.check_required_class_variables(
|
21
|
-
cls, RegisterLanguageModelsMeta.REQUIRED_CLASS_ATTRIBUTES
|
22
|
-
)
|
23
|
-
|
24
|
-
## Check that model name is valid
|
25
|
-
# if not LanguageModelType.is_value_valid(model_name):
|
26
|
-
# acceptable_values = [item.value for item in LanguageModelType]
|
27
|
-
# raise LanguageModelAttributeTypeError(
|
28
|
-
# f"""A LanguageModel's model must be one of {LanguageModelType} values, which are
|
29
|
-
# {acceptable_values}. You passed {model_name}."""
|
30
|
-
# )
|
31
|
-
|
32
|
-
if not InferenceServiceType.is_value_valid(
|
33
|
-
inference_service := getattr(cls, "_inference_service_", None)
|
34
|
-
):
|
35
|
-
acceptable_values = [item.value for item in InferenceServiceType]
|
36
|
-
raise LanguageModelAttributeTypeError(
|
37
|
-
f"""A LanguageModel's model must have an _inference_service_ value from
|
38
|
-
{acceptable_values}. You passed {inference_service}."""
|
39
|
-
)
|
40
|
-
|
41
|
-
# LanguageModel children have to implement the async_execute_model_call method
|
42
|
-
RegisterLanguageModelsMeta.verify_method(
|
43
|
-
candidate_class=cls,
|
44
|
-
method_name="async_execute_model_call",
|
45
|
-
expected_return_type=dict[str, Any],
|
46
|
-
required_parameters=[("user_prompt", str), ("system_prompt", str)],
|
47
|
-
must_be_async=True,
|
48
|
-
)
|
49
|
-
# LanguageModel children have to implement the parse_response method
|
50
|
-
RegisterLanguageModelsMeta._registry[model_name] = cls
|
51
|
-
|
52
|
-
@classmethod
|
53
|
-
def get_registered_classes(cls):
|
54
|
-
"""Return the registry."""
|
55
|
-
return cls._registry
|
56
|
-
|
57
|
-
@staticmethod
|
58
|
-
def check_required_class_variables(
|
59
|
-
candidate_class: "LanguageModel", required_attributes: List[str] = None
|
60
|
-
):
|
61
|
-
"""Check if a class has the required attributes.
|
62
|
-
|
63
|
-
>>> class M:
|
64
|
-
... _model_ = "m"
|
65
|
-
... _parameters_ = {}
|
66
|
-
>>> RegisterLanguageModelsMeta.check_required_class_variables(M, ["_model_", "_parameters_"])
|
67
|
-
>>> class M2:
|
68
|
-
... _model_ = "m"
|
69
|
-
>>> RegisterLanguageModelsMeta.check_required_class_variables(M2, ["_model_", "_parameters_"])
|
70
|
-
Traceback (most recent call last):
|
71
|
-
...
|
72
|
-
Exception: Class M2 does not have required attribute _parameters_
|
73
|
-
"""
|
74
|
-
required_attributes = required_attributes or []
|
75
|
-
for attribute in required_attributes:
|
76
|
-
if not hasattr(candidate_class, attribute):
|
77
|
-
raise Exception(
|
78
|
-
f"Class {candidate_class.__name__} does not have required attribute {attribute}"
|
79
|
-
)
|
80
|
-
|
81
|
-
@staticmethod
|
82
|
-
def verify_method(
|
83
|
-
candidate_class: "LanguageModel",
|
84
|
-
method_name: str,
|
85
|
-
expected_return_type: Any,
|
86
|
-
required_parameters: List[tuple[str, Any]] = None,
|
87
|
-
must_be_async: bool = False,
|
88
|
-
):
|
89
|
-
"""Verify that a method is defined in a class, has the correct return type, and has the correct parameters."""
|
90
|
-
RegisterLanguageModelsMeta._check_method_defined(candidate_class, method_name)
|
91
|
-
|
92
|
-
required_parameters = required_parameters or []
|
93
|
-
method = getattr(candidate_class, method_name)
|
94
|
-
# signature = inspect.signature(method)
|
95
|
-
|
96
|
-
RegisterLanguageModelsMeta._check_return_type(method, expected_return_type)
|
97
|
-
|
98
|
-
if must_be_async:
|
99
|
-
RegisterLanguageModelsMeta._check_is_coroutine(method)
|
100
|
-
|
101
|
-
# Check the parameters
|
102
|
-
# params = signature.parameters
|
103
|
-
# for param_name, param_type in required_parameters:
|
104
|
-
# RegisterLanguageModelsMeta._verify_parameter(
|
105
|
-
# params, param_name, param_type, method_name
|
106
|
-
# )
|
107
|
-
|
108
|
-
@staticmethod
|
109
|
-
def _check_method_defined(cls, method_name):
|
110
|
-
"""Check if a method is defined in a class.
|
111
|
-
|
112
|
-
Example:
|
113
|
-
>>> class M:
|
114
|
-
... def f(self): pass
|
115
|
-
>>> RegisterLanguageModelsMeta._check_method_defined(M, "f")
|
116
|
-
>>> RegisterLanguageModelsMeta._check_method_defined(M, "g")
|
117
|
-
Traceback (most recent call last):
|
118
|
-
...
|
119
|
-
NotImplementedError: g method must be implemented.
|
120
|
-
"""
|
121
|
-
if not hasattr(cls, method_name):
|
122
|
-
raise NotImplementedError(f"{method_name} method must be implemented.")
|
123
|
-
|
124
|
-
@staticmethod
|
125
|
-
def _check_is_coroutine(func: Callable):
|
126
|
-
"""Check to make sure it's a coroutine function.
|
127
|
-
|
128
|
-
Example:
|
129
|
-
|
130
|
-
>>> def f(): pass
|
131
|
-
>>> RegisterLanguageModelsMeta._check_is_coroutine(f)
|
132
|
-
Traceback (most recent call last):
|
133
|
-
...
|
134
|
-
TypeError: A LangugeModel class with method f must be an asynchronous method.
|
135
|
-
"""
|
136
|
-
if not inspect.iscoroutinefunction(func):
|
137
|
-
raise TypeError(
|
138
|
-
f"A LangugeModel class with method {func.__name__} must be an asynchronous method."
|
139
|
-
)
|
140
|
-
|
141
|
-
@staticmethod
|
142
|
-
def _verify_parameter(params, param_name, param_type, method_name):
|
143
|
-
"""Verify that a parameter is defined in a method and has the correct type."""
|
144
|
-
pass
|
145
|
-
# if param_name not in params:
|
146
|
-
# raise TypeError(
|
147
|
-
# f"""Parameter "{param_name}" of method "{method_name}" must be defined.
|
148
|
-
# """
|
149
|
-
# )
|
150
|
-
# if params[param_name].annotation != param_type:
|
151
|
-
# raise TypeError(
|
152
|
-
# f"""Parameter "{param_name}" of method "{method_name}" must be of type {param_type.__name__}.
|
153
|
-
# Got {params[param_name].annotation} instead.
|
154
|
-
# """
|
155
|
-
# )
|
156
|
-
|
157
|
-
@staticmethod
|
158
|
-
def _check_return_type(method, expected_return_type):
|
159
|
-
"""
|
160
|
-
Check if the return type of a method is as expected.
|
161
|
-
|
162
|
-
Example:
|
163
|
-
"""
|
164
|
-
pass
|
165
|
-
# if inspect.isroutine(method):
|
166
|
-
# # return_type = inspect.signature(method).return_annotation
|
167
|
-
# return_type = get_type_hints(method)["return"]
|
168
|
-
# if return_type != expected_return_type:
|
169
|
-
# raise TypeError(
|
170
|
-
# f"Return type of {method.__name__} must be {expected_return_type}. Got {return_type}."
|
171
|
-
# )
|
172
|
-
|
173
|
-
@classmethod
|
174
|
-
def model_names_to_classes(cls):
|
175
|
-
"""Return a dictionary of model names to classes."""
|
176
|
-
d = {}
|
177
|
-
for classname, cls in cls._registry.items():
|
178
|
-
if hasattr(cls, "_model_"):
|
179
|
-
d[cls._model_] = cls
|
180
|
-
else:
|
181
|
-
raise Exception(
|
182
|
-
f"Class {classname} does not have a _model_ class attribute."
|
183
|
-
)
|
184
|
-
return d
|
1
|
+
from abc import ABC, ABCMeta
|
2
|
+
from typing import Any, List, Callable
|
3
|
+
import inspect
|
4
|
+
from typing import get_type_hints
|
5
|
+
from edsl.exceptions.language_models import LanguageModelAttributeTypeError
|
6
|
+
from edsl.enums import InferenceServiceType
|
7
|
+
|
8
|
+
|
9
|
+
class RegisterLanguageModelsMeta(ABCMeta):
|
10
|
+
"""Metaclass to register output elements in a registry i.e., those that have a parent."""
|
11
|
+
|
12
|
+
_registry = {} # Initialize the registry as a dictionary
|
13
|
+
REQUIRED_CLASS_ATTRIBUTES = ["_model_", "_parameters_", "_inference_service_"]
|
14
|
+
|
15
|
+
def __init__(cls, name, bases, dct):
|
16
|
+
"""Register the class in the registry if it has a _model_ attribute."""
|
17
|
+
super(RegisterLanguageModelsMeta, cls).__init__(name, bases, dct)
|
18
|
+
# if name != "LanguageModel":
|
19
|
+
if (model_name := getattr(cls, "_model_", None)) is not None:
|
20
|
+
RegisterLanguageModelsMeta.check_required_class_variables(
|
21
|
+
cls, RegisterLanguageModelsMeta.REQUIRED_CLASS_ATTRIBUTES
|
22
|
+
)
|
23
|
+
|
24
|
+
## Check that model name is valid
|
25
|
+
# if not LanguageModelType.is_value_valid(model_name):
|
26
|
+
# acceptable_values = [item.value for item in LanguageModelType]
|
27
|
+
# raise LanguageModelAttributeTypeError(
|
28
|
+
# f"""A LanguageModel's model must be one of {LanguageModelType} values, which are
|
29
|
+
# {acceptable_values}. You passed {model_name}."""
|
30
|
+
# )
|
31
|
+
|
32
|
+
if not InferenceServiceType.is_value_valid(
|
33
|
+
inference_service := getattr(cls, "_inference_service_", None)
|
34
|
+
):
|
35
|
+
acceptable_values = [item.value for item in InferenceServiceType]
|
36
|
+
raise LanguageModelAttributeTypeError(
|
37
|
+
f"""A LanguageModel's model must have an _inference_service_ value from
|
38
|
+
{acceptable_values}. You passed {inference_service}."""
|
39
|
+
)
|
40
|
+
|
41
|
+
# LanguageModel children have to implement the async_execute_model_call method
|
42
|
+
RegisterLanguageModelsMeta.verify_method(
|
43
|
+
candidate_class=cls,
|
44
|
+
method_name="async_execute_model_call",
|
45
|
+
expected_return_type=dict[str, Any],
|
46
|
+
required_parameters=[("user_prompt", str), ("system_prompt", str)],
|
47
|
+
must_be_async=True,
|
48
|
+
)
|
49
|
+
# LanguageModel children have to implement the parse_response method
|
50
|
+
RegisterLanguageModelsMeta._registry[model_name] = cls
|
51
|
+
|
52
|
+
@classmethod
|
53
|
+
def get_registered_classes(cls):
|
54
|
+
"""Return the registry."""
|
55
|
+
return cls._registry
|
56
|
+
|
57
|
+
@staticmethod
|
58
|
+
def check_required_class_variables(
|
59
|
+
candidate_class: "LanguageModel", required_attributes: List[str] = None
|
60
|
+
):
|
61
|
+
"""Check if a class has the required attributes.
|
62
|
+
|
63
|
+
>>> class M:
|
64
|
+
... _model_ = "m"
|
65
|
+
... _parameters_ = {}
|
66
|
+
>>> RegisterLanguageModelsMeta.check_required_class_variables(M, ["_model_", "_parameters_"])
|
67
|
+
>>> class M2:
|
68
|
+
... _model_ = "m"
|
69
|
+
>>> RegisterLanguageModelsMeta.check_required_class_variables(M2, ["_model_", "_parameters_"])
|
70
|
+
Traceback (most recent call last):
|
71
|
+
...
|
72
|
+
Exception: Class M2 does not have required attribute _parameters_
|
73
|
+
"""
|
74
|
+
required_attributes = required_attributes or []
|
75
|
+
for attribute in required_attributes:
|
76
|
+
if not hasattr(candidate_class, attribute):
|
77
|
+
raise Exception(
|
78
|
+
f"Class {candidate_class.__name__} does not have required attribute {attribute}"
|
79
|
+
)
|
80
|
+
|
81
|
+
@staticmethod
|
82
|
+
def verify_method(
|
83
|
+
candidate_class: "LanguageModel",
|
84
|
+
method_name: str,
|
85
|
+
expected_return_type: Any,
|
86
|
+
required_parameters: List[tuple[str, Any]] = None,
|
87
|
+
must_be_async: bool = False,
|
88
|
+
):
|
89
|
+
"""Verify that a method is defined in a class, has the correct return type, and has the correct parameters."""
|
90
|
+
RegisterLanguageModelsMeta._check_method_defined(candidate_class, method_name)
|
91
|
+
|
92
|
+
required_parameters = required_parameters or []
|
93
|
+
method = getattr(candidate_class, method_name)
|
94
|
+
# signature = inspect.signature(method)
|
95
|
+
|
96
|
+
RegisterLanguageModelsMeta._check_return_type(method, expected_return_type)
|
97
|
+
|
98
|
+
if must_be_async:
|
99
|
+
RegisterLanguageModelsMeta._check_is_coroutine(method)
|
100
|
+
|
101
|
+
# Check the parameters
|
102
|
+
# params = signature.parameters
|
103
|
+
# for param_name, param_type in required_parameters:
|
104
|
+
# RegisterLanguageModelsMeta._verify_parameter(
|
105
|
+
# params, param_name, param_type, method_name
|
106
|
+
# )
|
107
|
+
|
108
|
+
@staticmethod
|
109
|
+
def _check_method_defined(cls, method_name):
|
110
|
+
"""Check if a method is defined in a class.
|
111
|
+
|
112
|
+
Example:
|
113
|
+
>>> class M:
|
114
|
+
... def f(self): pass
|
115
|
+
>>> RegisterLanguageModelsMeta._check_method_defined(M, "f")
|
116
|
+
>>> RegisterLanguageModelsMeta._check_method_defined(M, "g")
|
117
|
+
Traceback (most recent call last):
|
118
|
+
...
|
119
|
+
NotImplementedError: g method must be implemented.
|
120
|
+
"""
|
121
|
+
if not hasattr(cls, method_name):
|
122
|
+
raise NotImplementedError(f"{method_name} method must be implemented.")
|
123
|
+
|
124
|
+
@staticmethod
|
125
|
+
def _check_is_coroutine(func: Callable):
|
126
|
+
"""Check to make sure it's a coroutine function.
|
127
|
+
|
128
|
+
Example:
|
129
|
+
|
130
|
+
>>> def f(): pass
|
131
|
+
>>> RegisterLanguageModelsMeta._check_is_coroutine(f)
|
132
|
+
Traceback (most recent call last):
|
133
|
+
...
|
134
|
+
TypeError: A LangugeModel class with method f must be an asynchronous method.
|
135
|
+
"""
|
136
|
+
if not inspect.iscoroutinefunction(func):
|
137
|
+
raise TypeError(
|
138
|
+
f"A LangugeModel class with method {func.__name__} must be an asynchronous method."
|
139
|
+
)
|
140
|
+
|
141
|
+
@staticmethod
|
142
|
+
def _verify_parameter(params, param_name, param_type, method_name):
|
143
|
+
"""Verify that a parameter is defined in a method and has the correct type."""
|
144
|
+
pass
|
145
|
+
# if param_name not in params:
|
146
|
+
# raise TypeError(
|
147
|
+
# f"""Parameter "{param_name}" of method "{method_name}" must be defined.
|
148
|
+
# """
|
149
|
+
# )
|
150
|
+
# if params[param_name].annotation != param_type:
|
151
|
+
# raise TypeError(
|
152
|
+
# f"""Parameter "{param_name}" of method "{method_name}" must be of type {param_type.__name__}.
|
153
|
+
# Got {params[param_name].annotation} instead.
|
154
|
+
# """
|
155
|
+
# )
|
156
|
+
|
157
|
+
@staticmethod
|
158
|
+
def _check_return_type(method, expected_return_type):
|
159
|
+
"""
|
160
|
+
Check if the return type of a method is as expected.
|
161
|
+
|
162
|
+
Example:
|
163
|
+
"""
|
164
|
+
pass
|
165
|
+
# if inspect.isroutine(method):
|
166
|
+
# # return_type = inspect.signature(method).return_annotation
|
167
|
+
# return_type = get_type_hints(method)["return"]
|
168
|
+
# if return_type != expected_return_type:
|
169
|
+
# raise TypeError(
|
170
|
+
# f"Return type of {method.__name__} must be {expected_return_type}. Got {return_type}."
|
171
|
+
# )
|
172
|
+
|
173
|
+
@classmethod
|
174
|
+
def model_names_to_classes(cls):
|
175
|
+
"""Return a dictionary of model names to classes."""
|
176
|
+
d = {}
|
177
|
+
for classname, cls in cls._registry.items():
|
178
|
+
if hasattr(cls, "_model_"):
|
179
|
+
d[cls._model_] = cls
|
180
|
+
else:
|
181
|
+
raise Exception(
|
182
|
+
f"Class {classname} does not have a _model_ class attribute."
|
183
|
+
)
|
184
|
+
return d
|
File without changes
|
edsl/language_models/__init__.py
CHANGED
@@ -1,3 +1,2 @@
|
|
1
|
-
from edsl.language_models.LanguageModel import LanguageModel
|
2
|
-
from edsl.language_models.
|
3
|
-
from edsl.language_models.KeyLookup import KeyLookup
|
1
|
+
from edsl.language_models.LanguageModel import LanguageModel
|
2
|
+
from edsl.language_models.model import Model
|
@@ -1,15 +1,15 @@
|
|
1
|
-
from openai import AsyncOpenAI
|
2
|
-
import asyncio
|
3
|
-
|
4
|
-
client = AsyncOpenAI(base_url="http://127.0.0.1:8000/v1", api_key="fake_key")
|
5
|
-
|
6
|
-
|
7
|
-
async def main():
|
8
|
-
response = await client.chat.completions.create(
|
9
|
-
model="gpt-3.5-turbo", messages=[{"role": "user", "content": "Question XX42"}]
|
10
|
-
)
|
11
|
-
print(response)
|
12
|
-
|
13
|
-
|
14
|
-
if __name__ == "__main__":
|
15
|
-
asyncio.run(main())
|
1
|
+
from openai import AsyncOpenAI
|
2
|
+
import asyncio
|
3
|
+
|
4
|
+
client = AsyncOpenAI(base_url="http://127.0.0.1:8000/v1", api_key="fake_key")
|
5
|
+
|
6
|
+
|
7
|
+
async def main():
|
8
|
+
response = await client.chat.completions.create(
|
9
|
+
model="gpt-3.5-turbo", messages=[{"role": "user", "content": "Question XX42"}]
|
10
|
+
)
|
11
|
+
print(response)
|
12
|
+
|
13
|
+
|
14
|
+
if __name__ == "__main__":
|
15
|
+
asyncio.run(main())
|
@@ -1,61 +1,61 @@
|
|
1
|
-
import threading
|
2
|
-
import asyncio
|
3
|
-
from fastapi import FastAPI, Request
|
4
|
-
from fastapi.responses import JSONResponse
|
5
|
-
import uvicorn
|
6
|
-
import json
|
7
|
-
from typing import Any
|
8
|
-
|
9
|
-
app = FastAPI()
|
10
|
-
|
11
|
-
|
12
|
-
async def generate_response(question_number: int) -> dict:
|
13
|
-
# Simulate some asynchronous work
|
14
|
-
await asyncio.sleep(1)
|
15
|
-
return {
|
16
|
-
"id": "chatcmpl-123",
|
17
|
-
"object": "chat.completion",
|
18
|
-
"created": 1677652288,
|
19
|
-
"model": "gpt-3.5-turbo-0613",
|
20
|
-
"choices": [
|
21
|
-
{
|
22
|
-
"index": 0,
|
23
|
-
"message": {
|
24
|
-
"role": "assistant",
|
25
|
-
"content": json.dumps(
|
26
|
-
{"answer": f"SPAM for question {question_number}!"}
|
27
|
-
),
|
28
|
-
},
|
29
|
-
"finish_reason": "stop",
|
30
|
-
}
|
31
|
-
],
|
32
|
-
"usage": {"prompt_tokens": 9, "completion_tokens": 12, "total_tokens": 21},
|
33
|
-
}
|
34
|
-
|
35
|
-
|
36
|
-
@app.post("/v1/chat/completions")
|
37
|
-
async def chat_completions(request: Request):
|
38
|
-
body = await request.json()
|
39
|
-
user_prompt = body["messages"][-1]["content"]
|
40
|
-
question_number = int(user_prompt.split("XX")[1])
|
41
|
-
|
42
|
-
response = await generate_response(question_number)
|
43
|
-
return JSONResponse(content=response)
|
44
|
-
|
45
|
-
|
46
|
-
def run_server():
|
47
|
-
uvicorn.run(app, host="127.0.0.1", port=8000)
|
48
|
-
|
49
|
-
|
50
|
-
if __name__ == "__main__":
|
51
|
-
# Start the server in a separate thread
|
52
|
-
server_thread = threading.Thread(target=run_server)
|
53
|
-
server_thread.start()
|
54
|
-
|
55
|
-
# Your main code here
|
56
|
-
# ...
|
57
|
-
|
58
|
-
# To use this with the OpenAI SDK:
|
59
|
-
# from openai import AsyncOpenAI
|
60
|
-
# client = AsyncOpenAI(base_url="http://127.0.0.1:8000/v1", api_key="fake_key")
|
61
|
-
# response = await client.chat.completions.create(model="gpt-3.5-turbo", messages=[...])
|
1
|
+
import threading
|
2
|
+
import asyncio
|
3
|
+
from fastapi import FastAPI, Request
|
4
|
+
from fastapi.responses import JSONResponse
|
5
|
+
import uvicorn
|
6
|
+
import json
|
7
|
+
from typing import Any
|
8
|
+
|
9
|
+
app = FastAPI()
|
10
|
+
|
11
|
+
|
12
|
+
async def generate_response(question_number: int) -> dict:
|
13
|
+
# Simulate some asynchronous work
|
14
|
+
await asyncio.sleep(1)
|
15
|
+
return {
|
16
|
+
"id": "chatcmpl-123",
|
17
|
+
"object": "chat.completion",
|
18
|
+
"created": 1677652288,
|
19
|
+
"model": "gpt-3.5-turbo-0613",
|
20
|
+
"choices": [
|
21
|
+
{
|
22
|
+
"index": 0,
|
23
|
+
"message": {
|
24
|
+
"role": "assistant",
|
25
|
+
"content": json.dumps(
|
26
|
+
{"answer": f"SPAM for question {question_number}!"}
|
27
|
+
),
|
28
|
+
},
|
29
|
+
"finish_reason": "stop",
|
30
|
+
}
|
31
|
+
],
|
32
|
+
"usage": {"prompt_tokens": 9, "completion_tokens": 12, "total_tokens": 21},
|
33
|
+
}
|
34
|
+
|
35
|
+
|
36
|
+
@app.post("/v1/chat/completions")
|
37
|
+
async def chat_completions(request: Request):
|
38
|
+
body = await request.json()
|
39
|
+
user_prompt = body["messages"][-1]["content"]
|
40
|
+
question_number = int(user_prompt.split("XX")[1])
|
41
|
+
|
42
|
+
response = await generate_response(question_number)
|
43
|
+
return JSONResponse(content=response)
|
44
|
+
|
45
|
+
|
46
|
+
def run_server():
|
47
|
+
uvicorn.run(app, host="127.0.0.1", port=8000)
|
48
|
+
|
49
|
+
|
50
|
+
if __name__ == "__main__":
|
51
|
+
# Start the server in a separate thread
|
52
|
+
server_thread = threading.Thread(target=run_server)
|
53
|
+
server_thread.start()
|
54
|
+
|
55
|
+
# Your main code here
|
56
|
+
# ...
|
57
|
+
|
58
|
+
# To use this with the OpenAI SDK:
|
59
|
+
# from openai import AsyncOpenAI
|
60
|
+
# client = AsyncOpenAI(base_url="http://127.0.0.1:8000/v1", api_key="fake_key")
|
61
|
+
# response = await client.chat.completions.create(model="gpt-3.5-turbo", messages=[...])
|
@@ -0,0 +1,63 @@
|
|
1
|
+
from collections import UserDict
|
2
|
+
from dataclasses import asdict
|
3
|
+
from edsl.enums import service_to_api_keyname
|
4
|
+
|
5
|
+
from edsl.language_models.key_management.models import LanguageModelInput
|
6
|
+
|
7
|
+
|
8
|
+
class KeyLookup(UserDict):
|
9
|
+
"""A class for looking up API keys and related configuration.
|
10
|
+
|
11
|
+
>>> from edsl.language_models.key_management.models import LanguageModelInput
|
12
|
+
>>> lookup = KeyLookup()
|
13
|
+
>>> lm_input = LanguageModelInput.example()
|
14
|
+
>>> lookup['test'] = lm_input
|
15
|
+
>>> lookup.to_dict()['test']['api_token']
|
16
|
+
'sk-abcd123'
|
17
|
+
>>> restored = KeyLookup.from_dict(lookup.to_dict())
|
18
|
+
>>> restored['test'].api_token
|
19
|
+
'sk-abcd123'
|
20
|
+
"""
|
21
|
+
|
22
|
+
def to_dict(self):
|
23
|
+
"""
|
24
|
+
>>> kl = KeyLookup.example()
|
25
|
+
>>> kl2 = KeyLookup.from_dict(kl.to_dict())
|
26
|
+
>>> kl2 == kl
|
27
|
+
True
|
28
|
+
>>> kl2 is kl
|
29
|
+
False
|
30
|
+
"""
|
31
|
+
return {k: asdict(v) for k, v in self.data.items()}
|
32
|
+
|
33
|
+
@classmethod
|
34
|
+
def from_dict(cls, d):
|
35
|
+
return cls({k: LanguageModelInput(**v) for k, v in d.items()})
|
36
|
+
|
37
|
+
@classmethod
|
38
|
+
def example(cls):
|
39
|
+
return cls(
|
40
|
+
{
|
41
|
+
"test": LanguageModelInput.example(),
|
42
|
+
"openai": LanguageModelInput.example(),
|
43
|
+
}
|
44
|
+
)
|
45
|
+
|
46
|
+
def to_dot_env(self):
|
47
|
+
"""Return a string representation of the key lookup collection for a .env file."""
|
48
|
+
lines = []
|
49
|
+
for service, lm_input in self.items():
|
50
|
+
if service != "test":
|
51
|
+
lines.append(f"EDSL_SERVICE_RPM_{service.upper()}={lm_input.rpm}")
|
52
|
+
lines.append(f"EDSL_SERVICE_TPM_{service.upper()}={lm_input.tpm}")
|
53
|
+
key_name = service_to_api_keyname.get(service, service)
|
54
|
+
lines.append(f"{key_name.upper()}={lm_input.api_token}")
|
55
|
+
if lm_input.api_id is not None:
|
56
|
+
lines.append(f"{service.upper()}_API_ID={lm_input.api_id}")
|
57
|
+
return "\n".join([f"{line}" for line in lines])
|
58
|
+
|
59
|
+
|
60
|
+
if __name__ == "__main__":
|
61
|
+
import doctest
|
62
|
+
|
63
|
+
doctest.testmod(optionflags=doctest.ELLIPSIS)
|