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.
Files changed (341) hide show
  1. edsl/Base.py +413 -332
  2. edsl/BaseDiff.py +260 -260
  3. edsl/TemplateLoader.py +24 -24
  4. edsl/__init__.py +57 -49
  5. edsl/__version__.py +1 -1
  6. edsl/agents/Agent.py +1071 -867
  7. edsl/agents/AgentList.py +551 -413
  8. edsl/agents/Invigilator.py +284 -233
  9. edsl/agents/InvigilatorBase.py +257 -270
  10. edsl/agents/PromptConstructor.py +272 -354
  11. edsl/agents/QuestionInstructionPromptBuilder.py +128 -0
  12. edsl/agents/QuestionTemplateReplacementsBuilder.py +137 -0
  13. edsl/agents/__init__.py +2 -3
  14. edsl/agents/descriptors.py +99 -99
  15. edsl/agents/prompt_helpers.py +129 -129
  16. edsl/agents/question_option_processor.py +172 -0
  17. edsl/auto/AutoStudy.py +130 -117
  18. edsl/auto/StageBase.py +243 -230
  19. edsl/auto/StageGenerateSurvey.py +178 -178
  20. edsl/auto/StageLabelQuestions.py +125 -125
  21. edsl/auto/StagePersona.py +61 -61
  22. edsl/auto/StagePersonaDimensionValueRanges.py +88 -88
  23. edsl/auto/StagePersonaDimensionValues.py +74 -74
  24. edsl/auto/StagePersonaDimensions.py +69 -69
  25. edsl/auto/StageQuestions.py +74 -73
  26. edsl/auto/SurveyCreatorPipeline.py +21 -21
  27. edsl/auto/utilities.py +218 -224
  28. edsl/base/Base.py +279 -279
  29. edsl/config.py +177 -157
  30. edsl/conversation/Conversation.py +290 -290
  31. edsl/conversation/car_buying.py +59 -58
  32. edsl/conversation/chips.py +95 -95
  33. edsl/conversation/mug_negotiation.py +81 -81
  34. edsl/conversation/next_speaker_utilities.py +93 -93
  35. edsl/coop/CoopFunctionsMixin.py +15 -0
  36. edsl/coop/ExpectedParrotKeyHandler.py +125 -0
  37. edsl/coop/PriceFetcher.py +54 -54
  38. edsl/coop/__init__.py +2 -2
  39. edsl/coop/coop.py +1106 -1028
  40. edsl/coop/utils.py +131 -131
  41. edsl/data/Cache.py +573 -555
  42. edsl/data/CacheEntry.py +230 -233
  43. edsl/data/CacheHandler.py +168 -149
  44. edsl/data/RemoteCacheSync.py +186 -78
  45. edsl/data/SQLiteDict.py +292 -292
  46. edsl/data/__init__.py +5 -4
  47. edsl/data/orm.py +10 -10
  48. edsl/data_transfer_models.py +74 -73
  49. edsl/enums.py +202 -175
  50. edsl/exceptions/BaseException.py +21 -21
  51. edsl/exceptions/__init__.py +54 -54
  52. edsl/exceptions/agents.py +54 -42
  53. edsl/exceptions/cache.py +5 -5
  54. edsl/exceptions/configuration.py +16 -16
  55. edsl/exceptions/coop.py +10 -10
  56. edsl/exceptions/data.py +14 -14
  57. edsl/exceptions/general.py +34 -34
  58. edsl/exceptions/inference_services.py +5 -0
  59. edsl/exceptions/jobs.py +33 -33
  60. edsl/exceptions/language_models.py +63 -63
  61. edsl/exceptions/prompts.py +15 -15
  62. edsl/exceptions/questions.py +109 -91
  63. edsl/exceptions/results.py +29 -29
  64. edsl/exceptions/scenarios.py +29 -22
  65. edsl/exceptions/surveys.py +37 -37
  66. edsl/inference_services/AnthropicService.py +106 -87
  67. edsl/inference_services/AvailableModelCacheHandler.py +184 -0
  68. edsl/inference_services/AvailableModelFetcher.py +215 -0
  69. edsl/inference_services/AwsBedrock.py +118 -120
  70. edsl/inference_services/AzureAI.py +215 -217
  71. edsl/inference_services/DeepInfraService.py +18 -18
  72. edsl/inference_services/GoogleService.py +143 -148
  73. edsl/inference_services/GroqService.py +20 -20
  74. edsl/inference_services/InferenceServiceABC.py +80 -147
  75. edsl/inference_services/InferenceServicesCollection.py +138 -97
  76. edsl/inference_services/MistralAIService.py +120 -123
  77. edsl/inference_services/OllamaService.py +18 -18
  78. edsl/inference_services/OpenAIService.py +236 -224
  79. edsl/inference_services/PerplexityService.py +160 -163
  80. edsl/inference_services/ServiceAvailability.py +135 -0
  81. edsl/inference_services/TestService.py +90 -89
  82. edsl/inference_services/TogetherAIService.py +172 -170
  83. edsl/inference_services/data_structures.py +134 -0
  84. edsl/inference_services/models_available_cache.py +118 -118
  85. edsl/inference_services/rate_limits_cache.py +25 -25
  86. edsl/inference_services/registry.py +41 -41
  87. edsl/inference_services/write_available.py +10 -10
  88. edsl/jobs/AnswerQuestionFunctionConstructor.py +223 -0
  89. edsl/jobs/Answers.py +43 -56
  90. edsl/jobs/FetchInvigilator.py +47 -0
  91. edsl/jobs/InterviewTaskManager.py +98 -0
  92. edsl/jobs/InterviewsConstructor.py +50 -0
  93. edsl/jobs/Jobs.py +823 -898
  94. edsl/jobs/JobsChecks.py +172 -147
  95. edsl/jobs/JobsComponentConstructor.py +189 -0
  96. edsl/jobs/JobsPrompts.py +270 -268
  97. edsl/jobs/JobsRemoteInferenceHandler.py +311 -239
  98. edsl/jobs/JobsRemoteInferenceLogger.py +239 -0
  99. edsl/jobs/RequestTokenEstimator.py +30 -0
  100. edsl/jobs/__init__.py +1 -1
  101. edsl/jobs/async_interview_runner.py +138 -0
  102. edsl/jobs/buckets/BucketCollection.py +104 -63
  103. edsl/jobs/buckets/ModelBuckets.py +65 -65
  104. edsl/jobs/buckets/TokenBucket.py +283 -251
  105. edsl/jobs/buckets/TokenBucketAPI.py +211 -0
  106. edsl/jobs/buckets/TokenBucketClient.py +191 -0
  107. edsl/jobs/check_survey_scenario_compatibility.py +85 -0
  108. edsl/jobs/data_structures.py +120 -0
  109. edsl/jobs/decorators.py +35 -0
  110. edsl/jobs/interviews/Interview.py +396 -661
  111. edsl/jobs/interviews/InterviewExceptionCollection.py +99 -99
  112. edsl/jobs/interviews/InterviewExceptionEntry.py +186 -186
  113. edsl/jobs/interviews/InterviewStatistic.py +63 -63
  114. edsl/jobs/interviews/InterviewStatisticsCollection.py +25 -25
  115. edsl/jobs/interviews/InterviewStatusDictionary.py +78 -78
  116. edsl/jobs/interviews/InterviewStatusLog.py +92 -92
  117. edsl/jobs/interviews/ReportErrors.py +66 -66
  118. edsl/jobs/interviews/interview_status_enum.py +9 -9
  119. edsl/jobs/jobs_status_enums.py +9 -0
  120. edsl/jobs/loggers/HTMLTableJobLogger.py +304 -0
  121. edsl/jobs/results_exceptions_handler.py +98 -0
  122. edsl/jobs/runners/JobsRunnerAsyncio.py +151 -466
  123. edsl/jobs/runners/JobsRunnerStatus.py +297 -330
  124. edsl/jobs/tasks/QuestionTaskCreator.py +244 -242
  125. edsl/jobs/tasks/TaskCreators.py +64 -64
  126. edsl/jobs/tasks/TaskHistory.py +470 -450
  127. edsl/jobs/tasks/TaskStatusLog.py +23 -23
  128. edsl/jobs/tasks/task_status_enum.py +161 -163
  129. edsl/jobs/tokens/InterviewTokenUsage.py +27 -27
  130. edsl/jobs/tokens/TokenUsage.py +34 -34
  131. edsl/language_models/ComputeCost.py +63 -0
  132. edsl/language_models/LanguageModel.py +626 -668
  133. edsl/language_models/ModelList.py +164 -155
  134. edsl/language_models/PriceManager.py +127 -0
  135. edsl/language_models/RawResponseHandler.py +106 -0
  136. edsl/language_models/RegisterLanguageModelsMeta.py +184 -184
  137. edsl/language_models/ServiceDataSources.py +0 -0
  138. edsl/language_models/__init__.py +2 -3
  139. edsl/language_models/fake_openai_call.py +15 -15
  140. edsl/language_models/fake_openai_service.py +61 -61
  141. edsl/language_models/key_management/KeyLookup.py +63 -0
  142. edsl/language_models/key_management/KeyLookupBuilder.py +273 -0
  143. edsl/language_models/key_management/KeyLookupCollection.py +38 -0
  144. edsl/language_models/key_management/__init__.py +0 -0
  145. edsl/language_models/key_management/models.py +131 -0
  146. edsl/language_models/model.py +256 -0
  147. edsl/language_models/repair.py +156 -156
  148. edsl/language_models/utilities.py +65 -64
  149. edsl/notebooks/Notebook.py +263 -258
  150. edsl/notebooks/NotebookToLaTeX.py +142 -0
  151. edsl/notebooks/__init__.py +1 -1
  152. edsl/prompts/Prompt.py +352 -362
  153. edsl/prompts/__init__.py +2 -2
  154. edsl/questions/ExceptionExplainer.py +77 -0
  155. edsl/questions/HTMLQuestion.py +103 -0
  156. edsl/questions/QuestionBase.py +518 -664
  157. edsl/questions/QuestionBasePromptsMixin.py +221 -217
  158. edsl/questions/QuestionBudget.py +227 -227
  159. edsl/questions/QuestionCheckBox.py +359 -359
  160. edsl/questions/QuestionExtract.py +180 -182
  161. edsl/questions/QuestionFreeText.py +113 -114
  162. edsl/questions/QuestionFunctional.py +166 -166
  163. edsl/questions/QuestionList.py +223 -231
  164. edsl/questions/QuestionMatrix.py +265 -0
  165. edsl/questions/QuestionMultipleChoice.py +330 -286
  166. edsl/questions/QuestionNumerical.py +151 -153
  167. edsl/questions/QuestionRank.py +314 -324
  168. edsl/questions/Quick.py +41 -41
  169. edsl/questions/SimpleAskMixin.py +74 -73
  170. edsl/questions/__init__.py +27 -26
  171. edsl/questions/{AnswerValidatorMixin.py → answer_validator_mixin.py} +334 -289
  172. edsl/questions/compose_questions.py +98 -98
  173. edsl/questions/data_structures.py +20 -0
  174. edsl/questions/decorators.py +21 -21
  175. edsl/questions/derived/QuestionLikertFive.py +76 -76
  176. edsl/questions/derived/QuestionLinearScale.py +90 -87
  177. edsl/questions/derived/QuestionTopK.py +93 -93
  178. edsl/questions/derived/QuestionYesNo.py +82 -82
  179. edsl/questions/descriptors.py +427 -413
  180. edsl/questions/loop_processor.py +149 -0
  181. edsl/questions/prompt_templates/question_budget.jinja +13 -13
  182. edsl/questions/prompt_templates/question_checkbox.jinja +32 -32
  183. edsl/questions/prompt_templates/question_extract.jinja +11 -11
  184. edsl/questions/prompt_templates/question_free_text.jinja +3 -3
  185. edsl/questions/prompt_templates/question_linear_scale.jinja +11 -11
  186. edsl/questions/prompt_templates/question_list.jinja +17 -17
  187. edsl/questions/prompt_templates/question_multiple_choice.jinja +33 -33
  188. edsl/questions/prompt_templates/question_numerical.jinja +36 -36
  189. edsl/questions/{QuestionBaseGenMixin.py → question_base_gen_mixin.py} +168 -161
  190. edsl/questions/question_registry.py +177 -177
  191. edsl/questions/{RegisterQuestionsMeta.py → register_questions_meta.py} +71 -71
  192. edsl/questions/{ResponseValidatorABC.py → response_validator_abc.py} +188 -174
  193. edsl/questions/response_validator_factory.py +34 -0
  194. edsl/questions/settings.py +12 -12
  195. edsl/questions/templates/budget/answering_instructions.jinja +7 -7
  196. edsl/questions/templates/budget/question_presentation.jinja +7 -7
  197. edsl/questions/templates/checkbox/answering_instructions.jinja +10 -10
  198. edsl/questions/templates/checkbox/question_presentation.jinja +22 -22
  199. edsl/questions/templates/extract/answering_instructions.jinja +7 -7
  200. edsl/questions/templates/likert_five/answering_instructions.jinja +10 -10
  201. edsl/questions/templates/likert_five/question_presentation.jinja +11 -11
  202. edsl/questions/templates/linear_scale/answering_instructions.jinja +5 -5
  203. edsl/questions/templates/linear_scale/question_presentation.jinja +5 -5
  204. edsl/questions/templates/list/answering_instructions.jinja +3 -3
  205. edsl/questions/templates/list/question_presentation.jinja +5 -5
  206. edsl/questions/templates/matrix/__init__.py +1 -0
  207. edsl/questions/templates/matrix/answering_instructions.jinja +5 -0
  208. edsl/questions/templates/matrix/question_presentation.jinja +20 -0
  209. edsl/questions/templates/multiple_choice/answering_instructions.jinja +9 -9
  210. edsl/questions/templates/multiple_choice/question_presentation.jinja +11 -11
  211. edsl/questions/templates/numerical/answering_instructions.jinja +6 -6
  212. edsl/questions/templates/numerical/question_presentation.jinja +6 -6
  213. edsl/questions/templates/rank/answering_instructions.jinja +11 -11
  214. edsl/questions/templates/rank/question_presentation.jinja +15 -15
  215. edsl/questions/templates/top_k/answering_instructions.jinja +8 -8
  216. edsl/questions/templates/top_k/question_presentation.jinja +22 -22
  217. edsl/questions/templates/yes_no/answering_instructions.jinja +6 -6
  218. edsl/questions/templates/yes_no/question_presentation.jinja +11 -11
  219. edsl/results/CSSParameterizer.py +108 -108
  220. edsl/results/Dataset.py +587 -424
  221. edsl/results/DatasetExportMixin.py +594 -731
  222. edsl/results/DatasetTree.py +295 -275
  223. edsl/results/MarkdownToDocx.py +122 -0
  224. edsl/results/MarkdownToPDF.py +111 -0
  225. edsl/results/Result.py +557 -465
  226. edsl/results/Results.py +1183 -1165
  227. edsl/results/ResultsExportMixin.py +45 -43
  228. edsl/results/ResultsGGMixin.py +121 -121
  229. edsl/results/TableDisplay.py +125 -198
  230. edsl/results/TextEditor.py +50 -0
  231. edsl/results/__init__.py +2 -2
  232. edsl/results/file_exports.py +252 -0
  233. edsl/results/{ResultsFetchMixin.py → results_fetch_mixin.py} +33 -33
  234. edsl/results/{Selector.py → results_selector.py} +145 -135
  235. edsl/results/{ResultsToolsMixin.py → results_tools_mixin.py} +98 -98
  236. edsl/results/smart_objects.py +96 -0
  237. edsl/results/table_data_class.py +12 -0
  238. edsl/results/table_display.css +77 -77
  239. edsl/results/table_renderers.py +118 -0
  240. edsl/results/tree_explore.py +115 -115
  241. edsl/scenarios/ConstructDownloadLink.py +109 -0
  242. edsl/scenarios/DocumentChunker.py +102 -0
  243. edsl/scenarios/DocxScenario.py +16 -0
  244. edsl/scenarios/FileStore.py +511 -632
  245. edsl/scenarios/PdfExtractor.py +40 -0
  246. edsl/scenarios/Scenario.py +498 -601
  247. edsl/scenarios/ScenarioHtmlMixin.py +65 -64
  248. edsl/scenarios/ScenarioList.py +1458 -1287
  249. edsl/scenarios/ScenarioListExportMixin.py +45 -52
  250. edsl/scenarios/ScenarioListPdfMixin.py +239 -261
  251. edsl/scenarios/__init__.py +3 -4
  252. edsl/scenarios/directory_scanner.py +96 -0
  253. edsl/scenarios/file_methods.py +85 -0
  254. edsl/scenarios/handlers/__init__.py +13 -0
  255. edsl/scenarios/handlers/csv.py +38 -0
  256. edsl/scenarios/handlers/docx.py +76 -0
  257. edsl/scenarios/handlers/html.py +37 -0
  258. edsl/scenarios/handlers/json.py +111 -0
  259. edsl/scenarios/handlers/latex.py +5 -0
  260. edsl/scenarios/handlers/md.py +51 -0
  261. edsl/scenarios/handlers/pdf.py +68 -0
  262. edsl/scenarios/handlers/png.py +39 -0
  263. edsl/scenarios/handlers/pptx.py +105 -0
  264. edsl/scenarios/handlers/py.py +294 -0
  265. edsl/scenarios/handlers/sql.py +313 -0
  266. edsl/scenarios/handlers/sqlite.py +149 -0
  267. edsl/scenarios/handlers/txt.py +33 -0
  268. edsl/scenarios/{ScenarioJoin.py → scenario_join.py} +131 -127
  269. edsl/scenarios/scenario_selector.py +156 -0
  270. edsl/shared.py +1 -1
  271. edsl/study/ObjectEntry.py +173 -173
  272. edsl/study/ProofOfWork.py +113 -113
  273. edsl/study/SnapShot.py +80 -80
  274. edsl/study/Study.py +521 -528
  275. edsl/study/__init__.py +4 -4
  276. edsl/surveys/ConstructDAG.py +92 -0
  277. edsl/surveys/DAG.py +148 -148
  278. edsl/surveys/EditSurvey.py +221 -0
  279. edsl/surveys/InstructionHandler.py +100 -0
  280. edsl/surveys/Memory.py +31 -31
  281. edsl/surveys/MemoryManagement.py +72 -0
  282. edsl/surveys/MemoryPlan.py +244 -244
  283. edsl/surveys/Rule.py +327 -326
  284. edsl/surveys/RuleCollection.py +385 -387
  285. edsl/surveys/RuleManager.py +172 -0
  286. edsl/surveys/Simulator.py +75 -0
  287. edsl/surveys/Survey.py +1280 -1801
  288. edsl/surveys/SurveyCSS.py +273 -261
  289. edsl/surveys/SurveyExportMixin.py +259 -259
  290. edsl/surveys/{SurveyFlowVisualizationMixin.py → SurveyFlowVisualization.py} +181 -179
  291. edsl/surveys/SurveyQualtricsImport.py +284 -284
  292. edsl/surveys/SurveyToApp.py +141 -0
  293. edsl/surveys/__init__.py +5 -3
  294. edsl/surveys/base.py +53 -53
  295. edsl/surveys/descriptors.py +60 -56
  296. edsl/surveys/instructions/ChangeInstruction.py +48 -49
  297. edsl/surveys/instructions/Instruction.py +56 -65
  298. edsl/surveys/instructions/InstructionCollection.py +82 -77
  299. edsl/templates/error_reporting/base.html +23 -23
  300. edsl/templates/error_reporting/exceptions_by_model.html +34 -34
  301. edsl/templates/error_reporting/exceptions_by_question_name.html +16 -16
  302. edsl/templates/error_reporting/exceptions_by_type.html +16 -16
  303. edsl/templates/error_reporting/interview_details.html +115 -115
  304. edsl/templates/error_reporting/interviews.html +19 -19
  305. edsl/templates/error_reporting/overview.html +4 -4
  306. edsl/templates/error_reporting/performance_plot.html +1 -1
  307. edsl/templates/error_reporting/report.css +73 -73
  308. edsl/templates/error_reporting/report.html +117 -117
  309. edsl/templates/error_reporting/report.js +25 -25
  310. edsl/tools/__init__.py +1 -1
  311. edsl/tools/clusters.py +192 -192
  312. edsl/tools/embeddings.py +27 -27
  313. edsl/tools/embeddings_plotting.py +118 -118
  314. edsl/tools/plotting.py +112 -112
  315. edsl/tools/summarize.py +18 -18
  316. edsl/utilities/PrettyList.py +56 -0
  317. edsl/utilities/SystemInfo.py +28 -28
  318. edsl/utilities/__init__.py +22 -22
  319. edsl/utilities/ast_utilities.py +25 -25
  320. edsl/utilities/data/Registry.py +6 -6
  321. edsl/utilities/data/__init__.py +1 -1
  322. edsl/utilities/data/scooter_results.json +1 -1
  323. edsl/utilities/decorators.py +77 -77
  324. edsl/utilities/gcp_bucket/cloud_storage.py +96 -96
  325. edsl/utilities/interface.py +627 -627
  326. edsl/utilities/is_notebook.py +18 -0
  327. edsl/utilities/is_valid_variable_name.py +11 -0
  328. edsl/utilities/naming_utilities.py +263 -263
  329. edsl/utilities/remove_edsl_version.py +24 -0
  330. edsl/utilities/repair_functions.py +28 -28
  331. edsl/utilities/restricted_python.py +70 -70
  332. edsl/utilities/utilities.py +436 -424
  333. {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/LICENSE +21 -21
  334. {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/METADATA +13 -11
  335. edsl-0.1.39.dev5.dist-info/RECORD +358 -0
  336. {edsl-0.1.39.dev3.dist-info → edsl-0.1.39.dev5.dist-info}/WHEEL +1 -1
  337. edsl/language_models/KeyLookup.py +0 -30
  338. edsl/language_models/registry.py +0 -190
  339. edsl/language_models/unused/ReplicateBase.py +0 -83
  340. edsl/results/ResultsDBMixin.py +0 -238
  341. edsl-0.1.39.dev3.dist-info/RECORD +0 -277
@@ -1,23 +1,23 @@
1
- from collections import UserList
2
-
3
-
4
- class TaskStatusLog(UserList):
5
- """A list of TaskStatusEntry objects."""
6
-
7
- @property
8
- def min_time(self):
9
- return self[0]["log_time"]
10
-
11
- @property
12
- def max_time(self):
13
- return self[-1]["log_time"]
14
-
15
- def status_at_time(self, t):
16
- """Return the status at time t.
17
-
18
- TODO: Could re-factor with bisect to make this faster.
19
- """
20
- for entry in self:
21
- if entry["log_time"] > t:
22
- return entry["value"]
23
- return self[-1]["value"]
1
+ from collections import UserList
2
+
3
+
4
+ class TaskStatusLog(UserList):
5
+ """A list of TaskStatusEntry objects."""
6
+
7
+ @property
8
+ def min_time(self):
9
+ return self[0]["log_time"]
10
+
11
+ @property
12
+ def max_time(self):
13
+ return self[-1]["log_time"]
14
+
15
+ def status_at_time(self, t):
16
+ """Return the status at time t.
17
+
18
+ TODO: Could re-factor with bisect to make this faster.
19
+ """
20
+ for entry in self:
21
+ if entry["log_time"] > t:
22
+ return entry["value"]
23
+ return self[-1]["value"]
@@ -1,163 +1,161 @@
1
- from __future__ import annotations
2
- from collections import UserDict
3
- import enum
4
- import time
5
-
6
- # from edsl.jobs.tasks.TaskStatusLogEntry import TaskStatusLogEntry
7
-
8
-
9
- class TaskStatus(enum.Enum):
10
- "These are the possible states a task can be in."
11
- NOT_STARTED = enum.auto()
12
- WAITING_FOR_DEPENDENCIES = enum.auto()
13
- CANCELLED = enum.auto()
14
- PARENT_FAILED = enum.auto()
15
- WAITING_FOR_REQUEST_CAPACITY = enum.auto()
16
- WAITING_FOR_TOKEN_CAPACITY = enum.auto()
17
- API_CALL_IN_PROGRESS = enum.auto()
18
- SUCCESS = enum.auto()
19
- FAILED = enum.auto()
20
-
21
-
22
- class TaskStatusLogEntry(UserDict):
23
- def __init__(self, log_time, value):
24
- self.data = {"log_time": log_time, "value": value}
25
- super().__init__(self.data)
26
-
27
-
28
- class TaskStatusDescriptor:
29
- "The descriptor ensures that the task status is always an instance of the TaskStatus enum."
30
-
31
- def __init__(self):
32
- self._task_status = None
33
-
34
- def __get__(self, instance, owner):
35
- return self._task_status
36
-
37
- def __set__(self, instance, value):
38
- """Ensure that the value is an instance of TaskStatus."""
39
- if not isinstance(value, TaskStatus):
40
- raise ValueError("Value must be an instance of TaskStatus enum")
41
- t = time.monotonic()
42
- if hasattr(instance, "status_log"):
43
- instance.status_log.append(TaskStatusLogEntry(t, value))
44
- self._task_status = value
45
-
46
- def __delete__(self, instance):
47
- self._task_status = None
48
-
49
-
50
- status_colors = {
51
- TaskStatus.NOT_STARTED: "grey",
52
- TaskStatus.WAITING_FOR_DEPENDENCIES: "orange",
53
- TaskStatus.WAITING_FOR_REQUEST_CAPACITY: "yellow",
54
- TaskStatus.WAITING_FOR_TOKEN_CAPACITY: "gold",
55
- TaskStatus.CANCELLED: "white",
56
- TaskStatus.PARENT_FAILED: "darkred",
57
- TaskStatus.FAILED: "red",
58
- TaskStatus.API_CALL_IN_PROGRESS: "blue",
59
- TaskStatus.SUCCESS: "green",
60
- }
61
-
62
-
63
- def get_enum_from_string(str_key):
64
- """Parse the string to extract the enum member name."""
65
- try:
66
- _, member_name = str_key.split(".")
67
- enum_member = getattr(TaskStatus, member_name)
68
- return enum_member
69
- except ValueError:
70
- return str_key
71
-
72
-
73
- class InterviewTaskLogDict(UserDict):
74
- """A dictionary of TaskStatusLog objects.
75
-
76
- The key is the name of the task.
77
- """
78
-
79
- @property
80
- def min_time(self):
81
- return min([log.min_time for log in self.values()])
82
-
83
- @property
84
- def max_time(self):
85
- return max([log.max_time for log in self.values()])
86
-
87
- def status_matrix(self, num_periods):
88
- """Return a matrix of status values."""
89
- start_time = self.min_time
90
- end_time = self.max_time
91
- time_increment = (end_time - start_time) / num_periods
92
- status_matrix = {}
93
- time_periods = [start_time + i * time_increment for i in range(num_periods)]
94
- for task_name, log in self.items():
95
- status_matrix[task_name] = [log.status_at_time(t) for t in time_periods]
96
- return status_matrix
97
-
98
- def numerical_matrix(self, num_periods):
99
- """Return a numerical matrix of status values."""
100
- status_dicts = self.status_matrix(num_periods)
101
-
102
- num_cols = num_periods
103
- num_rows = len(status_dicts)
104
- matrix = [[0 for _ in range(num_cols)] for _ in range(num_rows)]
105
-
106
- for row_index, (task_name, status_list) in enumerate(status_dicts.items()):
107
- matrix[row_index] = [
108
- list(status_colors.keys()).index(status) for status in status_list
109
- ]
110
-
111
- index_to_names = {i: name for i, name in enumerate(status_dicts.keys())}
112
- return matrix, index_to_names
113
-
114
- def visualize(self, num_periods=10):
115
- """Visualize the status matrix with outlined squares."""
116
- import matplotlib.pyplot as plt
117
- from matplotlib.colors import ListedColormap
118
- import numpy as np
119
- from matplotlib.patches import Rectangle
120
-
121
- # Define your custom colormap
122
- custom_cmap = ListedColormap(list(status_colors.values()))
123
-
124
- # Generate the matrix
125
- matrix, index_to_names = self.numerical_matrix(num_periods)
126
-
127
- # Create the figure and axes
128
- plt.figure(figsize=(10, 5))
129
- ax = plt.gca()
130
-
131
- # Display the matrix and keep a reference to the imshow object
132
- im = ax.imshow(matrix, aspect="auto", cmap=custom_cmap)
133
-
134
- # Adding color bar, now correctly associating it with 'im'
135
- cbar = plt.colorbar(im, ticks=range(len(status_colors)), label="Task Status")
136
-
137
- cbar_labels = [status.name for status in status_colors.keys()]
138
- # breakpoint()
139
- cbar.set_ticklabels(cbar_labels) # Setting the custom labels for the colorbar
140
-
141
- im.set_clim(
142
- -0.5, len(status_colors) - 0.5
143
- ) # Setting color limits directly on the imshow object
144
-
145
- # Outline each cell by drawing rectangles
146
- for (j, i), val in np.ndenumerate(matrix):
147
- ax.add_patch(
148
- Rectangle(
149
- (i - 0.5, j - 0.5), 1, 1, fill=False, edgecolor="black", lw=0.5
150
- )
151
- )
152
-
153
- # Set custom y-axis ticks and labels
154
- yticks = list(index_to_names.keys())
155
- yticklabels = list(index_to_names.values())
156
- plt.yticks(ticks=yticks, labels=yticklabels)
157
-
158
- # Show the plot
159
- plt.show()
160
-
161
-
162
- if __name__ == "__main__":
163
- pass
1
+ from __future__ import annotations
2
+ from collections import UserDict
3
+ import enum
4
+ import time
5
+
6
+
7
+ class TaskStatus(enum.Enum):
8
+ "These are the possible states a task can be in."
9
+ NOT_STARTED = enum.auto()
10
+ WAITING_FOR_DEPENDENCIES = enum.auto()
11
+ CANCELLED = enum.auto()
12
+ PARENT_FAILED = enum.auto()
13
+ WAITING_FOR_REQUEST_CAPACITY = enum.auto()
14
+ WAITING_FOR_TOKEN_CAPACITY = enum.auto()
15
+ API_CALL_IN_PROGRESS = enum.auto()
16
+ SUCCESS = enum.auto()
17
+ FAILED = enum.auto()
18
+
19
+
20
+ class TaskStatusLogEntry(UserDict):
21
+ def __init__(self, log_time, value):
22
+ self.data = {"log_time": log_time, "value": value}
23
+ super().__init__(self.data)
24
+
25
+
26
+ class TaskStatusDescriptor:
27
+ "The descriptor ensures that the task status is always an instance of the TaskStatus enum."
28
+
29
+ def __init__(self):
30
+ self._task_status = None
31
+
32
+ def __get__(self, instance, owner):
33
+ return self._task_status
34
+
35
+ def __set__(self, instance, value):
36
+ """Ensure that the value is an instance of TaskStatus."""
37
+ if not isinstance(value, TaskStatus):
38
+ raise ValueError("Value must be an instance of TaskStatus enum")
39
+ t = time.monotonic()
40
+ if hasattr(instance, "status_log"):
41
+ instance.status_log.append(TaskStatusLogEntry(t, value))
42
+ self._task_status = value
43
+
44
+ def __delete__(self, instance):
45
+ self._task_status = None
46
+
47
+
48
+ status_colors = {
49
+ TaskStatus.NOT_STARTED: "grey",
50
+ TaskStatus.WAITING_FOR_DEPENDENCIES: "orange",
51
+ TaskStatus.WAITING_FOR_REQUEST_CAPACITY: "yellow",
52
+ TaskStatus.WAITING_FOR_TOKEN_CAPACITY: "gold",
53
+ TaskStatus.CANCELLED: "white",
54
+ TaskStatus.PARENT_FAILED: "darkred",
55
+ TaskStatus.FAILED: "red",
56
+ TaskStatus.API_CALL_IN_PROGRESS: "blue",
57
+ TaskStatus.SUCCESS: "green",
58
+ }
59
+
60
+
61
+ def get_enum_from_string(str_key):
62
+ """Parse the string to extract the enum member name."""
63
+ try:
64
+ _, member_name = str_key.split(".")
65
+ enum_member = getattr(TaskStatus, member_name)
66
+ return enum_member
67
+ except ValueError:
68
+ return str_key
69
+
70
+
71
+ class InterviewTaskLogDict(UserDict):
72
+ """A dictionary of TaskStatusLog objects.
73
+
74
+ The key is the name of the task.
75
+ """
76
+
77
+ @property
78
+ def min_time(self):
79
+ return min([log.min_time for log in self.values()])
80
+
81
+ @property
82
+ def max_time(self):
83
+ return max([log.max_time for log in self.values()])
84
+
85
+ def status_matrix(self, num_periods):
86
+ """Return a matrix of status values."""
87
+ start_time = self.min_time
88
+ end_time = self.max_time
89
+ time_increment = (end_time - start_time) / num_periods
90
+ status_matrix = {}
91
+ time_periods = [start_time + i * time_increment for i in range(num_periods)]
92
+ for task_name, log in self.items():
93
+ status_matrix[task_name] = [log.status_at_time(t) for t in time_periods]
94
+ return status_matrix
95
+
96
+ def numerical_matrix(self, num_periods):
97
+ """Return a numerical matrix of status values."""
98
+ status_dicts = self.status_matrix(num_periods)
99
+
100
+ num_cols = num_periods
101
+ num_rows = len(status_dicts)
102
+ matrix = [[0 for _ in range(num_cols)] for _ in range(num_rows)]
103
+
104
+ for row_index, (task_name, status_list) in enumerate(status_dicts.items()):
105
+ matrix[row_index] = [
106
+ list(status_colors.keys()).index(status) for status in status_list
107
+ ]
108
+
109
+ index_to_names = {i: name for i, name in enumerate(status_dicts.keys())}
110
+ return matrix, index_to_names
111
+
112
+ def visualize(self, num_periods=10):
113
+ """Visualize the status matrix with outlined squares."""
114
+ import matplotlib.pyplot as plt
115
+ from matplotlib.colors import ListedColormap
116
+ import numpy as np
117
+ from matplotlib.patches import Rectangle
118
+
119
+ # Define your custom colormap
120
+ custom_cmap = ListedColormap(list(status_colors.values()))
121
+
122
+ # Generate the matrix
123
+ matrix, index_to_names = self.numerical_matrix(num_periods)
124
+
125
+ # Create the figure and axes
126
+ plt.figure(figsize=(10, 5))
127
+ ax = plt.gca()
128
+
129
+ # Display the matrix and keep a reference to the imshow object
130
+ im = ax.imshow(matrix, aspect="auto", cmap=custom_cmap)
131
+
132
+ # Adding color bar, now correctly associating it with 'im'
133
+ cbar = plt.colorbar(im, ticks=range(len(status_colors)), label="Task Status")
134
+
135
+ cbar_labels = [status.name for status in status_colors.keys()]
136
+ # breakpoint()
137
+ cbar.set_ticklabels(cbar_labels) # Setting the custom labels for the colorbar
138
+
139
+ im.set_clim(
140
+ -0.5, len(status_colors) - 0.5
141
+ ) # Setting color limits directly on the imshow object
142
+
143
+ # Outline each cell by drawing rectangles
144
+ for (j, i), val in np.ndenumerate(matrix):
145
+ ax.add_patch(
146
+ Rectangle(
147
+ (i - 0.5, j - 0.5), 1, 1, fill=False, edgecolor="black", lw=0.5
148
+ )
149
+ )
150
+
151
+ # Set custom y-axis ticks and labels
152
+ yticks = list(index_to_names.keys())
153
+ yticklabels = list(index_to_names.values())
154
+ plt.yticks(ticks=yticks, labels=yticklabels)
155
+
156
+ # Show the plot
157
+ plt.show()
158
+
159
+
160
+ if __name__ == "__main__":
161
+ pass
@@ -1,27 +1,27 @@
1
- from edsl.jobs.tokens.TokenUsage import TokenUsage
2
- from edsl.enums import TokenPricing
3
-
4
-
5
- class InterviewTokenUsage:
6
- def __init__(
7
- self, new_token_usage: TokenUsage = None, cached_token_usage: TokenUsage = None
8
- ):
9
- self.new_token_usage = new_token_usage or TokenUsage(from_cache=False)
10
- self.cached_token_usage = cached_token_usage or TokenUsage(from_cache=True)
11
-
12
- def __add__(self, other):
13
- if not isinstance(other, InterviewTokenUsage):
14
- raise ValueError(f"Can't add {type(other)} to InterviewTokenSummary")
15
- return InterviewTokenUsage(
16
- new_token_usage=self.new_token_usage + other.new_token_usage,
17
- cached_token_usage=self.cached_token_usage + other.cached_token_usage,
18
- )
19
-
20
- def __repr__(self):
21
- return f"InterviewTokenUsage(new_token_usage={self.new_token_usage}, cached_token_usage={self.cached_token_usage})"
22
-
23
- def cost(self, prices: TokenPricing):
24
- return self.new_token_usage.cost(prices)
25
-
26
- def saved(self, prices: TokenPricing):
27
- return self.cached_token_usage.cost(prices)
1
+ from edsl.jobs.tokens.TokenUsage import TokenUsage
2
+ from edsl.enums import TokenPricing
3
+
4
+
5
+ class InterviewTokenUsage:
6
+ def __init__(
7
+ self, new_token_usage: TokenUsage = None, cached_token_usage: TokenUsage = None
8
+ ):
9
+ self.new_token_usage = new_token_usage or TokenUsage(from_cache=False)
10
+ self.cached_token_usage = cached_token_usage or TokenUsage(from_cache=True)
11
+
12
+ def __add__(self, other):
13
+ if not isinstance(other, InterviewTokenUsage):
14
+ raise ValueError(f"Can't add {type(other)} to InterviewTokenSummary")
15
+ return InterviewTokenUsage(
16
+ new_token_usage=self.new_token_usage + other.new_token_usage,
17
+ cached_token_usage=self.cached_token_usage + other.cached_token_usage,
18
+ )
19
+
20
+ def __repr__(self):
21
+ return f"InterviewTokenUsage(new_token_usage={self.new_token_usage}, cached_token_usage={self.cached_token_usage})"
22
+
23
+ def cost(self, prices: TokenPricing):
24
+ return self.new_token_usage.cost(prices)
25
+
26
+ def saved(self, prices: TokenPricing):
27
+ return self.cached_token_usage.cost(prices)
@@ -1,34 +1,34 @@
1
- from edsl.enums import TokenPricing
2
-
3
-
4
- class TokenUsage:
5
- def __init__(
6
- self, from_cache: bool, prompt_tokens: int = 0, completion_tokens: int = 0
7
- ):
8
- self.from_cache = from_cache
9
- self.prompt_tokens = prompt_tokens
10
- self.completion_tokens = completion_tokens
11
-
12
- def add_tokens(self, prompt_tokens, completion_tokens):
13
- self.prompt_tokens += prompt_tokens
14
- self.completion_tokens += completion_tokens
15
-
16
- def __add__(self, other):
17
- if not isinstance(other, TokenUsage):
18
- raise ValueError(f"Can't add {type(other)} to InterviewTokenUsage")
19
- if self.from_cache != other.from_cache:
20
- raise ValueError(f"Can't add token usages from different sources")
21
- return TokenUsage(
22
- from_cache=self.from_cache,
23
- prompt_tokens=self.prompt_tokens + other.prompt_tokens,
24
- completion_tokens=self.completion_tokens + other.completion_tokens,
25
- )
26
-
27
- def __repr__(self):
28
- return f"TokenUsage(from_cache={self.from_cache}, prompt_tokens={self.prompt_tokens}, completion_tokens={self.completion_tokens})"
29
-
30
- def cost(self, prices: TokenPricing):
31
- return (
32
- self.prompt_tokens * prices.prompt_token_price
33
- + self.completion_tokens * prices.completion_token_price
34
- )
1
+ from edsl.enums import TokenPricing
2
+
3
+
4
+ class TokenUsage:
5
+ def __init__(
6
+ self, from_cache: bool, prompt_tokens: int = 0, completion_tokens: int = 0
7
+ ):
8
+ self.from_cache = from_cache
9
+ self.prompt_tokens = prompt_tokens
10
+ self.completion_tokens = completion_tokens
11
+
12
+ def add_tokens(self, prompt_tokens, completion_tokens):
13
+ self.prompt_tokens += prompt_tokens
14
+ self.completion_tokens += completion_tokens
15
+
16
+ def __add__(self, other):
17
+ if not isinstance(other, TokenUsage):
18
+ raise ValueError(f"Can't add {type(other)} to InterviewTokenUsage")
19
+ if self.from_cache != other.from_cache:
20
+ raise ValueError(f"Can't add token usages from different sources")
21
+ return TokenUsage(
22
+ from_cache=self.from_cache,
23
+ prompt_tokens=self.prompt_tokens + other.prompt_tokens,
24
+ completion_tokens=self.completion_tokens + other.completion_tokens,
25
+ )
26
+
27
+ def __repr__(self):
28
+ return f"TokenUsage(from_cache={self.from_cache}, prompt_tokens={self.prompt_tokens}, completion_tokens={self.completion_tokens})"
29
+
30
+ def cost(self, prices: TokenPricing):
31
+ return (
32
+ self.prompt_tokens * prices.prompt_token_price
33
+ + self.completion_tokens * prices.completion_token_price
34
+ )
@@ -0,0 +1,63 @@
1
+ from typing import Any, Union
2
+
3
+
4
+ class ComputeCost:
5
+ def __init__(self, language_model: "LanguageModel"):
6
+ self.language_model = language_model
7
+ self._price_lookup = None
8
+
9
+ @property
10
+ def price_lookup(self):
11
+ if self._price_lookup is None:
12
+ from edsl.coop import Coop
13
+
14
+ c = Coop()
15
+ self._price_lookup = c.fetch_prices()
16
+ return self._price_lookup
17
+
18
+ def cost(self, raw_response: dict[str, Any]) -> Union[float, str]:
19
+ """Return the dollar cost of a raw response."""
20
+
21
+ usage = self.get_usage_dict(raw_response)
22
+ from edsl.coop import Coop
23
+
24
+ c = Coop()
25
+ price_lookup = c.fetch_prices()
26
+ key = (self._inference_service_, self.model)
27
+ if key not in price_lookup:
28
+ return f"Could not find price for model {self.model} in the price lookup."
29
+
30
+ relevant_prices = price_lookup[key]
31
+ try:
32
+ input_tokens = int(usage[self.input_token_name])
33
+ output_tokens = int(usage[self.output_token_name])
34
+ except Exception as e:
35
+ return f"Could not fetch tokens from model response: {e}"
36
+
37
+ try:
38
+ inverse_output_price = relevant_prices["output"]["one_usd_buys"]
39
+ inverse_input_price = relevant_prices["input"]["one_usd_buys"]
40
+ except Exception as e:
41
+ if "output" not in relevant_prices:
42
+ return f"Could not fetch prices from {relevant_prices} - {e}; Missing 'output' key."
43
+ if "input" not in relevant_prices:
44
+ return f"Could not fetch prices from {relevant_prices} - {e}; Missing 'input' key."
45
+ return f"Could not fetch prices from {relevant_prices} - {e}"
46
+
47
+ if inverse_input_price == "infinity":
48
+ input_cost = 0
49
+ else:
50
+ try:
51
+ input_cost = input_tokens / float(inverse_input_price)
52
+ except Exception as e:
53
+ return f"Could not compute input price - {e}."
54
+
55
+ if inverse_output_price == "infinity":
56
+ output_cost = 0
57
+ else:
58
+ try:
59
+ output_cost = output_tokens / float(inverse_output_price)
60
+ except Exception as e:
61
+ return f"Could not compute output price - {e}"
62
+
63
+ return input_cost + output_cost