edsl 0.1.36.dev3__tar.gz → 0.1.36.dev4__tar.gz

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 (294) hide show
  1. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/PKG-INFO +1 -1
  2. edsl-0.1.36.dev4/edsl/__version__.py +1 -0
  3. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/coop/coop.py +3 -2
  4. edsl-0.1.36.dev4/edsl/data/RemoteCacheSync.py +84 -0
  5. edsl-0.1.36.dev4/edsl/exceptions/coop.py +10 -0
  6. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/Jobs.py +140 -158
  7. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/Interview.py +21 -3
  8. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/InterviewExceptionCollection.py +9 -0
  9. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/InterviewExceptionEntry.py +24 -6
  10. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/runners/JobsRunnerAsyncio.py +5 -14
  11. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/tasks/TaskHistory.py +22 -7
  12. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/Results.py +13 -1
  13. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/pyproject.toml +1 -1
  14. edsl-0.1.36.dev3/edsl/__version__.py +0 -1
  15. edsl-0.1.36.dev3/edsl/exceptions/coop.py +0 -2
  16. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/LICENSE +0 -0
  17. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/README.md +0 -0
  18. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/Base.py +0 -0
  19. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/BaseDiff.py +0 -0
  20. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/TemplateLoader.py +0 -0
  21. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/__init__.py +0 -0
  22. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/agents/Agent.py +0 -0
  23. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/agents/AgentList.py +0 -0
  24. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/agents/Invigilator.py +0 -0
  25. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/agents/InvigilatorBase.py +0 -0
  26. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/agents/PromptConstructor.py +0 -0
  27. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/agents/__init__.py +0 -0
  28. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/agents/descriptors.py +0 -0
  29. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/agents/prompt_helpers.py +0 -0
  30. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/AutoStudy.py +0 -0
  31. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/StageBase.py +0 -0
  32. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/StageGenerateSurvey.py +0 -0
  33. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/StageLabelQuestions.py +0 -0
  34. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/StagePersona.py +0 -0
  35. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/StagePersonaDimensionValueRanges.py +0 -0
  36. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/StagePersonaDimensionValues.py +0 -0
  37. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/StagePersonaDimensions.py +0 -0
  38. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/StageQuestions.py +0 -0
  39. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/SurveyCreatorPipeline.py +0 -0
  40. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/auto/utilities.py +0 -0
  41. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/base/Base.py +0 -0
  42. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/config.py +0 -0
  43. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/AgentConstructionMixin.py +0 -0
  44. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/Conjure.py +0 -0
  45. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/InputData.py +0 -0
  46. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/InputDataCSV.py +0 -0
  47. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/InputDataMixinQuestionStats.py +0 -0
  48. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/InputDataPyRead.py +0 -0
  49. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/InputDataSPSS.py +0 -0
  50. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/InputDataStata.py +0 -0
  51. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/QuestionOptionMixin.py +0 -0
  52. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/QuestionTypeMixin.py +0 -0
  53. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/RawQuestion.py +0 -0
  54. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/SurveyResponses.py +0 -0
  55. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/__init__.py +0 -0
  56. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/examples/placeholder.txt +0 -0
  57. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/naming_utilities.py +0 -0
  58. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conjure/utilities.py +0 -0
  59. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conversation/Conversation.py +0 -0
  60. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conversation/car_buying.py +0 -0
  61. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conversation/mug_negotiation.py +0 -0
  62. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/conversation/next_speaker_utilities.py +0 -0
  63. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/coop/PriceFetcher.py +0 -0
  64. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/coop/__init__.py +0 -0
  65. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/coop/utils.py +0 -0
  66. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/data/Cache.py +0 -0
  67. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/data/CacheEntry.py +0 -0
  68. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/data/CacheHandler.py +0 -0
  69. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/data/SQLiteDict.py +0 -0
  70. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/data/__init__.py +0 -0
  71. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/data/orm.py +0 -0
  72. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/data_transfer_models.py +0 -0
  73. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/enums.py +0 -0
  74. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/__init__.py +0 -0
  75. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/agents.py +0 -0
  76. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/configuration.py +0 -0
  77. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/data.py +0 -0
  78. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/general.py +0 -0
  79. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/jobs.py +0 -0
  80. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/language_models.py +0 -0
  81. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/prompts.py +0 -0
  82. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/questions.py +0 -0
  83. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/results.py +0 -0
  84. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/exceptions/surveys.py +0 -0
  85. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/AnthropicService.py +0 -0
  86. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/AwsBedrock.py +0 -0
  87. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/AzureAI.py +0 -0
  88. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/DeepInfraService.py +0 -0
  89. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/GoogleService.py +0 -0
  90. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/GroqService.py +0 -0
  91. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/InferenceServiceABC.py +0 -0
  92. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/InferenceServicesCollection.py +0 -0
  93. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/MistralAIService.py +0 -0
  94. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/OllamaService.py +0 -0
  95. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/OpenAIService.py +0 -0
  96. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/TestService.py +0 -0
  97. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/TogetherAIService.py +0 -0
  98. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/__init__.py +0 -0
  99. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/models_available_cache.py +0 -0
  100. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/rate_limits_cache.py +0 -0
  101. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/registry.py +0 -0
  102. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/inference_services/write_available.py +0 -0
  103. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/Answers.py +0 -0
  104. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/__init__.py +0 -0
  105. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/buckets/BucketCollection.py +0 -0
  106. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/buckets/ModelBuckets.py +0 -0
  107. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/buckets/TokenBucket.py +0 -0
  108. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/InterviewStatistic.py +0 -0
  109. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/InterviewStatisticsCollection.py +0 -0
  110. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/InterviewStatusDictionary.py +0 -0
  111. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/InterviewStatusLog.py +0 -0
  112. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/ReportErrors.py +0 -0
  113. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/interviews/interview_status_enum.py +0 -0
  114. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/runners/JobsRunnerStatus.py +0 -0
  115. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/runners/JobsRunnerStatusData.py +0 -0
  116. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/tasks/QuestionTaskCreator.py +0 -0
  117. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/tasks/TaskCreators.py +0 -0
  118. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/tasks/TaskStatusLog.py +0 -0
  119. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/tasks/task_status_enum.py +0 -0
  120. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/tokens/InterviewTokenUsage.py +0 -0
  121. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/jobs/tokens/TokenUsage.py +0 -0
  122. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/LanguageModel.py +0 -0
  123. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/ModelList.py +0 -0
  124. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/RegisterLanguageModelsMeta.py +0 -0
  125. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/__init__.py +0 -0
  126. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/fake_openai_call.py +0 -0
  127. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/fake_openai_service.py +0 -0
  128. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/registry.py +0 -0
  129. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/repair.py +0 -0
  130. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/unused/ReplicateBase.py +0 -0
  131. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/language_models/utilities.py +0 -0
  132. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/notebooks/Notebook.py +0 -0
  133. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/notebooks/__init__.py +0 -0
  134. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/prompts/Prompt.py +0 -0
  135. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/prompts/__init__.py +0 -0
  136. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/AnswerValidatorMixin.py +0 -0
  137. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionBase.py +0 -0
  138. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionBaseGenMixin.py +0 -0
  139. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionBasePromptsMixin.py +0 -0
  140. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionBudget.py +0 -0
  141. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionCheckBox.py +0 -0
  142. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionExtract.py +0 -0
  143. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionFreeText.py +0 -0
  144. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionFunctional.py +0 -0
  145. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionList.py +0 -0
  146. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionMultipleChoice.py +0 -0
  147. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionNumerical.py +0 -0
  148. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/QuestionRank.py +0 -0
  149. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/Quick.py +0 -0
  150. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/RegisterQuestionsMeta.py +0 -0
  151. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/ResponseValidatorABC.py +0 -0
  152. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/SimpleAskMixin.py +0 -0
  153. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/__init__.py +0 -0
  154. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/compose_questions.py +0 -0
  155. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/decorators.py +0 -0
  156. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/derived/QuestionLikertFive.py +0 -0
  157. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/derived/QuestionLinearScale.py +0 -0
  158. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/derived/QuestionTopK.py +0 -0
  159. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/derived/QuestionYesNo.py +0 -0
  160. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/derived/__init__.py +0 -0
  161. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/descriptors.py +0 -0
  162. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/prompt_templates/question_budget.jinja +0 -0
  163. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/prompt_templates/question_checkbox.jinja +0 -0
  164. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/prompt_templates/question_extract.jinja +0 -0
  165. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/prompt_templates/question_free_text.jinja +0 -0
  166. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/prompt_templates/question_linear_scale.jinja +0 -0
  167. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/prompt_templates/question_list.jinja +0 -0
  168. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/prompt_templates/question_multiple_choice.jinja +0 -0
  169. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/prompt_templates/question_numerical.jinja +0 -0
  170. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/question_registry.py +0 -0
  171. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/settings.py +0 -0
  172. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/__init__.py +0 -0
  173. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/__pycache__/__init__.cpython-311.pyc +0 -0
  174. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/budget/__init__.py +0 -0
  175. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/budget/__pycache__/__init__.cpython-311.pyc +0 -0
  176. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/budget/answering_instructions.jinja +0 -0
  177. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/budget/question_presentation.jinja +0 -0
  178. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/checkbox/__init__.py +0 -0
  179. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/checkbox/__pycache__/__init__.cpython-311.pyc +0 -0
  180. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/checkbox/answering_instructions.jinja +0 -0
  181. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/checkbox/question_presentation.jinja +0 -0
  182. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/extract/__init__.py +0 -0
  183. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/extract/__pycache__/__init__.cpython-311.pyc +0 -0
  184. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/extract/answering_instructions.jinja +0 -0
  185. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/extract/question_presentation.jinja +0 -0
  186. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/free_text/__init__.py +0 -0
  187. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/free_text/__pycache__/__init__.cpython-311.pyc +0 -0
  188. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/free_text/answering_instructions.jinja +0 -0
  189. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/free_text/question_presentation.jinja +0 -0
  190. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/likert_five/__init__.py +0 -0
  191. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/likert_five/__pycache__/__init__.cpython-311.pyc +0 -0
  192. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/likert_five/answering_instructions.jinja +0 -0
  193. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/likert_five/question_presentation.jinja +0 -0
  194. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/linear_scale/__init__.py +0 -0
  195. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/linear_scale/__pycache__/__init__.cpython-311.pyc +0 -0
  196. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/linear_scale/answering_instructions.jinja +0 -0
  197. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/linear_scale/question_presentation.jinja +0 -0
  198. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/list/__init__.py +0 -0
  199. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/list/__pycache__/__init__.cpython-311.pyc +0 -0
  200. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/list/answering_instructions.jinja +0 -0
  201. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/list/question_presentation.jinja +0 -0
  202. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/multiple_choice/__init__.py +0 -0
  203. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/multiple_choice/__pycache__/__init__.cpython-311.pyc +0 -0
  204. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/multiple_choice/answering_instructions.jinja +0 -0
  205. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/multiple_choice/html.jinja +0 -0
  206. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/multiple_choice/question_presentation.jinja +0 -0
  207. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/numerical/__init__.py +0 -0
  208. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/numerical/__pycache__/__init__.cpython-311.pyc +0 -0
  209. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/numerical/answering_instructions.jinja +0 -0
  210. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/numerical/question_presentation.jinja +0 -0
  211. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/rank/__init__.py +0 -0
  212. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/rank/__pycache__/__init__.cpython-311.pyc +0 -0
  213. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/rank/answering_instructions.jinja +0 -0
  214. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/rank/question_presentation.jinja +0 -0
  215. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/top_k/__init__.py +0 -0
  216. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/top_k/__pycache__/__init__.cpython-311.pyc +0 -0
  217. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/top_k/answering_instructions.jinja +0 -0
  218. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/top_k/question_presentation.jinja +0 -0
  219. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/yes_no/__init__.py +0 -0
  220. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/yes_no/__pycache__/__init__.cpython-311.pyc +0 -0
  221. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/yes_no/answering_instructions.jinja +0 -0
  222. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/questions/templates/yes_no/question_presentation.jinja +0 -0
  223. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/Dataset.py +0 -0
  224. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/DatasetExportMixin.py +0 -0
  225. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/DatasetTree.py +0 -0
  226. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/Result.py +0 -0
  227. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/ResultsDBMixin.py +0 -0
  228. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/ResultsExportMixin.py +0 -0
  229. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/ResultsFetchMixin.py +0 -0
  230. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/ResultsGGMixin.py +0 -0
  231. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/ResultsToolsMixin.py +0 -0
  232. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/Selector.py +0 -0
  233. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/__init__.py +0 -0
  234. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/results/tree_explore.py +0 -0
  235. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/scenarios/FileStore.py +0 -0
  236. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/scenarios/Scenario.py +0 -0
  237. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/scenarios/ScenarioHtmlMixin.py +0 -0
  238. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/scenarios/ScenarioList.py +0 -0
  239. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/scenarios/ScenarioListExportMixin.py +0 -0
  240. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/scenarios/ScenarioListPdfMixin.py +0 -0
  241. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/scenarios/__init__.py +0 -0
  242. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/shared.py +0 -0
  243. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/study/ObjectEntry.py +0 -0
  244. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/study/ProofOfWork.py +0 -0
  245. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/study/SnapShot.py +0 -0
  246. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/study/Study.py +0 -0
  247. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/study/__init__.py +0 -0
  248. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/DAG.py +0 -0
  249. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/Memory.py +0 -0
  250. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/MemoryPlan.py +0 -0
  251. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/Rule.py +0 -0
  252. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/RuleCollection.py +0 -0
  253. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/Survey.py +0 -0
  254. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/SurveyCSS.py +0 -0
  255. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/SurveyExportMixin.py +0 -0
  256. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/SurveyFlowVisualizationMixin.py +0 -0
  257. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/SurveyQualtricsImport.py +0 -0
  258. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/__init__.py +0 -0
  259. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/base.py +0 -0
  260. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/descriptors.py +0 -0
  261. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/instructions/ChangeInstruction.py +0 -0
  262. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/instructions/Instruction.py +0 -0
  263. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/instructions/InstructionCollection.py +0 -0
  264. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/surveys/instructions/__init__.py +0 -0
  265. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/base.html +0 -0
  266. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/exceptions_by_model.html +0 -0
  267. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/exceptions_by_question_name.html +0 -0
  268. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/exceptions_by_type.html +0 -0
  269. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/interview_details.html +0 -0
  270. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/interviews.html +0 -0
  271. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/overview.html +0 -0
  272. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/performance_plot.html +0 -0
  273. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/report.css +0 -0
  274. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/report.html +0 -0
  275. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/templates/error_reporting/report.js +0 -0
  276. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/tools/__init__.py +0 -0
  277. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/tools/clusters.py +0 -0
  278. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/tools/embeddings.py +0 -0
  279. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/tools/embeddings_plotting.py +0 -0
  280. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/tools/plotting.py +0 -0
  281. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/tools/summarize.py +0 -0
  282. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/SystemInfo.py +0 -0
  283. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/__init__.py +0 -0
  284. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/ast_utilities.py +0 -0
  285. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/data/Registry.py +0 -0
  286. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/data/__init__.py +0 -0
  287. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/data/scooter_results.json +0 -0
  288. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/decorators.py +0 -0
  289. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/gcp_bucket/__init__.py +0 -0
  290. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/gcp_bucket/cloud_storage.py +0 -0
  291. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/interface.py +0 -0
  292. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/repair_functions.py +0 -0
  293. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/restricted_python.py +0 -0
  294. {edsl-0.1.36.dev3 → edsl-0.1.36.dev4}/edsl/utilities/utilities.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: edsl
3
- Version: 0.1.36.dev3
3
+ Version: 0.1.36.dev4
4
4
  Summary: Create and analyze LLM-based surveys
5
5
  Home-page: https://www.expectedparrot.com/
6
6
  License: MIT
@@ -0,0 +1 @@
1
+ __version__ = "0.1.36.dev4"
@@ -6,6 +6,7 @@ from typing import Any, Optional, Union, Literal
6
6
  from uuid import UUID
7
7
  import edsl
8
8
  from edsl import CONFIG, CacheEntry, Jobs, Survey
9
+ from edsl.exceptions.coop import CoopNoUUIDError, CoopServerResponseError
9
10
  from edsl.coop.utils import (
10
11
  EDSLObject,
11
12
  ObjectRegistry,
@@ -99,7 +100,7 @@ class Coop:
99
100
  if "Authorization" in message:
100
101
  print(message)
101
102
  message = "Please provide an Expected Parrot API key."
102
- raise Exception(message)
103
+ raise CoopServerResponseError(message)
103
104
 
104
105
  def _json_handle_none(self, value: Any) -> Any:
105
106
  """
@@ -116,7 +117,7 @@ class Coop:
116
117
  Resolve the uuid from a uuid or a url.
117
118
  """
118
119
  if not url and not uuid:
119
- raise Exception("No uuid or url provided for the object.")
120
+ raise CoopNoUUIDError("No uuid or url provided for the object.")
120
121
  if not uuid and url:
121
122
  uuid = url.split("/")[-1]
122
123
  return uuid
@@ -0,0 +1,84 @@
1
+ class RemoteCacheSync:
2
+ def __init__(self, coop, cache, output_func, remote_cache=True, remote_cache_description=""):
3
+ self.coop = coop
4
+ self.cache = cache
5
+ self._output = output_func
6
+ self.remote_cache = remote_cache
7
+ self.old_entry_keys = []
8
+ self.new_cache_entries = []
9
+ self.remote_cache_description = remote_cache_description
10
+
11
+ def __enter__(self):
12
+ if self.remote_cache:
13
+ self._sync_from_remote()
14
+ self.old_entry_keys = list(self.cache.keys())
15
+ return self
16
+
17
+ def __exit__(self, exc_type, exc_value, traceback):
18
+ if self.remote_cache:
19
+ self._sync_to_remote()
20
+ return False # Propagate exceptions
21
+
22
+ def _sync_from_remote(self):
23
+ cache_difference = self.coop.remote_cache_get_diff(self.cache.keys())
24
+ client_missing_cacheentries = cache_difference.get("client_missing_cacheentries", [])
25
+ missing_entry_count = len(client_missing_cacheentries)
26
+
27
+ if missing_entry_count > 0:
28
+ self._output(
29
+ f"Updating local cache with {missing_entry_count:,} new "
30
+ f"{'entry' if missing_entry_count == 1 else 'entries'} from remote..."
31
+ )
32
+ self.cache.add_from_dict({entry.key: entry for entry in client_missing_cacheentries})
33
+ self._output("Local cache updated!")
34
+ else:
35
+ self._output("No new entries to add to local cache.")
36
+
37
+ def _sync_to_remote(self):
38
+ cache_difference = self.coop.remote_cache_get_diff(self.cache.keys())
39
+ server_missing_cacheentry_keys = cache_difference.get("server_missing_cacheentry_keys", [])
40
+ server_missing_cacheentries = [
41
+ entry
42
+ for key in server_missing_cacheentry_keys
43
+ if (entry := self.cache.data.get(key)) is not None
44
+ ]
45
+
46
+ new_cache_entries = [
47
+ entry for entry in self.cache.values() if entry.key not in self.old_entry_keys
48
+ ]
49
+ server_missing_cacheentries.extend(new_cache_entries)
50
+ new_entry_count = len(server_missing_cacheentries)
51
+
52
+ if new_entry_count > 0:
53
+ self._output(
54
+ f"Updating remote cache with {new_entry_count:,} new "
55
+ f"{'entry' if new_entry_count == 1 else 'entries'}..."
56
+ )
57
+ self.coop.remote_cache_create_many(
58
+ server_missing_cacheentries,
59
+ visibility="private",
60
+ description=self.remote_cache_description,
61
+ )
62
+ self._output("Remote cache updated!")
63
+ else:
64
+ self._output("No new entries to add to remote cache.")
65
+
66
+ self._output(f"There are {len(self.cache.keys()):,} entries in the local cache.")
67
+
68
+ # # Usage example
69
+ # def run_job(self, n, progress_bar, cache, stop_on_exception, sidecar_model, print_exceptions, raise_validation_errors, use_remote_cache=True):
70
+ # with RemoteCacheSync(self.coop, cache, self._output, remote_cache=use_remote_cache):
71
+ # self._output("Running job...")
72
+ # results = self._run_local(
73
+ # n=n,
74
+ # progress_bar=progress_bar,
75
+ # cache=cache,
76
+ # stop_on_exception=stop_on_exception,
77
+ # sidecar_model=sidecar_model,
78
+ # print_exceptions=print_exceptions,
79
+ # raise_validation_errors=raise_validation_errors,
80
+ # )
81
+ # self._output("Job completed!")
82
+
83
+ # results.cache = cache.new_entries_cache()
84
+ # return results
@@ -0,0 +1,10 @@
1
+ class CoopErrors(Exception):
2
+ pass
3
+
4
+
5
+ class CoopNoUUIDError(CoopErrors):
6
+ pass
7
+
8
+
9
+ class CoopServerResponseError(CoopErrors):
10
+ pass
@@ -1,8 +1,10 @@
1
1
  # """The Jobs class is a collection of agents, scenarios and models and one survey."""
2
2
  from __future__ import annotations
3
3
  import warnings
4
+ import requests
4
5
  from itertools import product
5
6
  from typing import Optional, Union, Sequence, Generator
7
+
6
8
  from edsl.Base import Base
7
9
  from edsl.exceptions import MissingAPIKeyError
8
10
  from edsl.jobs.buckets.BucketCollection import BucketCollection
@@ -10,6 +12,9 @@ from edsl.jobs.interviews.Interview import Interview
10
12
  from edsl.jobs.runners.JobsRunnerAsyncio import JobsRunnerAsyncio
11
13
  from edsl.utilities.decorators import add_edsl_version, remove_edsl_version
12
14
 
15
+ from edsl.data.RemoteCacheSync import RemoteCacheSync
16
+ from edsl.exceptions.coop import CoopServerResponseError
17
+
13
18
 
14
19
  class Jobs(Base):
15
20
  """
@@ -203,10 +208,6 @@ class Jobs(Base):
203
208
  ]
204
209
  )
205
210
  return d
206
- # if table:
207
- # d.to_scenario_list().print(format="rich")
208
- # else:
209
- # return d
210
211
 
211
212
  def show_prompts(self) -> None:
212
213
  """Print the prompts."""
@@ -615,7 +616,7 @@ class Jobs(Base):
615
616
 
616
617
  def _output(self, message) -> None:
617
618
  """Check if a Job is verbose. If so, print the message."""
618
- if self.verbose:
619
+ if hasattr(self, "verbose") and self.verbose:
619
620
  print(message)
620
621
 
621
622
  def _check_parameters(self, strict=False, warn=False) -> None:
@@ -692,6 +693,122 @@ class Jobs(Base):
692
693
  return False
693
694
  return self._raise_validation_errors
694
695
 
696
+ def create_remote_inference_job(
697
+ self, iterations: int = 1, remote_inference_description: Optional[str] = None
698
+ ):
699
+ """ """
700
+ from edsl.coop.coop import Coop
701
+
702
+ coop = Coop()
703
+ self._output("Remote inference activated. Sending job to server...")
704
+ remote_job_creation_data = coop.remote_inference_create(
705
+ self,
706
+ description=remote_inference_description,
707
+ status="queued",
708
+ iterations=iterations,
709
+ )
710
+ return remote_job_creation_data
711
+
712
+ @staticmethod
713
+ def check_status(job_uuid):
714
+ from edsl.coop.coop import Coop
715
+
716
+ coop = Coop()
717
+ return coop.remote_inference_get(job_uuid)
718
+
719
+ def poll_remote_inference_job(
720
+ self, remote_job_creation_data: dict
721
+ ) -> Union[Results, None]:
722
+ from edsl.coop.coop import Coop
723
+ import time
724
+ from datetime import datetime
725
+ from edsl.config import CONFIG
726
+
727
+ expected_parrot_url = CONFIG.get("EXPECTED_PARROT_URL")
728
+
729
+ job_uuid = remote_job_creation_data.get("uuid")
730
+
731
+ coop = Coop()
732
+ job_in_queue = True
733
+ while job_in_queue:
734
+ remote_job_data = coop.remote_inference_get(job_uuid)
735
+ status = remote_job_data.get("status")
736
+ if status == "cancelled":
737
+ print("\r" + " " * 80 + "\r", end="")
738
+ print("Job cancelled by the user.")
739
+ print(
740
+ f"See {expected_parrot_url}/home/remote-inference for more details."
741
+ )
742
+ return None
743
+ elif status == "failed":
744
+ print("\r" + " " * 80 + "\r", end="")
745
+ print("Job failed.")
746
+ print(
747
+ f"See {expected_parrot_url}/home/remote-inference for more details."
748
+ )
749
+ return None
750
+ elif status == "completed":
751
+ results_uuid = remote_job_data.get("results_uuid")
752
+ results = coop.get(results_uuid, expected_object_type="results")
753
+ print("\r" + " " * 80 + "\r", end="")
754
+ print(
755
+ f"Job completed and Results stored on Coop (Results uuid={results_uuid})."
756
+ )
757
+ return results
758
+ else:
759
+ duration = 5
760
+ time_checked = datetime.now().strftime("%Y-%m-%d %I:%M:%S %p")
761
+ frames = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"]
762
+ start_time = time.time()
763
+ i = 0
764
+ while time.time() - start_time < duration:
765
+ print(
766
+ f"\r{frames[i % len(frames)]} Job status: {status} - last update: {time_checked}",
767
+ end="",
768
+ flush=True,
769
+ )
770
+ time.sleep(0.1)
771
+ i += 1
772
+
773
+ def use_remote_inference(self, disable_remote_inference: bool):
774
+ if disable_remote_inference:
775
+ return False
776
+ if not disable_remote_inference:
777
+ try:
778
+ from edsl import Coop
779
+
780
+ user_edsl_settings = Coop().edsl_settings
781
+ return user_edsl_settings.get("remote_inference", False)
782
+ except requests.ConnectionError:
783
+ pass
784
+ except CoopServerResponseError as e:
785
+ pass
786
+
787
+ return False
788
+
789
+ def use_remote_cache(self):
790
+ try:
791
+ from edsl import Coop
792
+
793
+ user_edsl_settings = Coop().edsl_settings
794
+ return user_edsl_settings.get("remote_caching", False)
795
+ except requests.ConnectionError:
796
+ pass
797
+ except CoopServerResponseError as e:
798
+ pass
799
+
800
+ return False
801
+
802
+ def check_api_keys(self):
803
+ from edsl import Model
804
+
805
+ for model in self.models + [Model()]:
806
+ if not model.has_valid_api_key():
807
+ raise MissingAPIKeyError(
808
+ model_name=str(model.model),
809
+ inference_service=model._inference_service_,
810
+ )
811
+
695
812
  def run(
696
813
  self,
697
814
  n: int = 1,
@@ -729,91 +846,17 @@ class Jobs(Base):
729
846
 
730
847
  self.verbose = verbose
731
848
 
732
- remote_cache = False
733
- remote_inference = False
734
-
735
- if not disable_remote_inference:
736
- try:
737
- coop = Coop()
738
- user_edsl_settings = Coop().edsl_settings
739
- remote_cache = user_edsl_settings.get("remote_caching", False)
740
- remote_inference = user_edsl_settings.get("remote_inference", False)
741
- except Exception:
742
- pass
743
-
744
- if remote_inference:
745
- import time
746
- from datetime import datetime
747
- from edsl.config import CONFIG
748
-
749
- expected_parrot_url = CONFIG.get("EXPECTED_PARROT_URL")
750
-
751
- self._output("Remote inference activated. Sending job to server...")
752
- if remote_cache:
753
- self._output(
754
- "Remote caching activated. The remote cache will be used for this job."
755
- )
756
-
757
- remote_job_creation_data = coop.remote_inference_create(
758
- self,
759
- description=remote_inference_description,
760
- status="queued",
761
- iterations=n,
849
+ if remote_inference := self.use_remote_inference(disable_remote_inference):
850
+ remote_job_creation_data = self.create_remote_inference_job(
851
+ iterations=n, remote_inference_description=remote_inference_description
762
852
  )
763
- time_queued = datetime.now().strftime("%m/%d/%Y %I:%M:%S %p")
764
- job_uuid = remote_job_creation_data.get("uuid")
765
- print(f"Remote inference started (Job uuid={job_uuid}).")
766
- # print(f"Job queued at {time_queued}.")
767
- job_in_queue = True
768
- while job_in_queue:
769
- remote_job_data = coop.remote_inference_get(job_uuid)
770
- status = remote_job_data.get("status")
771
- if status == "cancelled":
772
- print("\r" + " " * 80 + "\r", end="")
773
- print("Job cancelled by the user.")
774
- print(
775
- f"See {expected_parrot_url}/home/remote-inference for more details."
776
- )
777
- return None
778
- elif status == "failed":
779
- print("\r" + " " * 80 + "\r", end="")
780
- print("Job failed.")
781
- print(
782
- f"See {expected_parrot_url}/home/remote-inference for more details."
783
- )
784
- return None
785
- elif status == "completed":
786
- results_uuid = remote_job_data.get("results_uuid")
787
- results = coop.get(results_uuid, expected_object_type="results")
788
- print("\r" + " " * 80 + "\r", end="")
789
- print(
790
- f"Job completed and Results stored on Coop (Results uuid={results_uuid})."
791
- )
792
- return results
793
- else:
794
- duration = 5
795
- time_checked = datetime.now().strftime("%Y-%m-%d %I:%M:%S %p")
796
- frames = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"]
797
- start_time = time.time()
798
- i = 0
799
- while time.time() - start_time < duration:
800
- print(
801
- f"\r{frames[i % len(frames)]} Job status: {status} - last update: {time_checked}",
802
- end="",
803
- flush=True,
804
- )
805
- time.sleep(0.1)
806
- i += 1
807
- else:
808
- if check_api_keys:
809
- from edsl import Model
853
+ results = self.poll_remote_inference_job(remote_job_creation_data)
854
+ if results is None:
855
+ self._output("Job failed.")
856
+ return results
810
857
 
811
- for model in self.models + [Model()]:
812
- if not model.has_valid_api_key():
813
- raise MissingAPIKeyError(
814
- model_name=str(model.model),
815
- inference_service=model._inference_service_,
816
- )
858
+ if check_api_keys:
859
+ self.check_api_keys()
817
860
 
818
861
  # handle cache
819
862
  if cache is None or cache is True:
@@ -825,7 +868,14 @@ class Jobs(Base):
825
868
 
826
869
  cache = Cache()
827
870
 
828
- if not remote_cache:
871
+ remote_cache = self.use_remote_cache()
872
+ with RemoteCacheSync(
873
+ coop=Coop(),
874
+ cache=cache,
875
+ output_func=self._output,
876
+ remote_cache=remote_cache,
877
+ remote_cache_description=remote_cache_description,
878
+ ) as r:
829
879
  results = self._run_local(
830
880
  n=n,
831
881
  progress_bar=progress_bar,
@@ -836,75 +886,7 @@ class Jobs(Base):
836
886
  raise_validation_errors=raise_validation_errors,
837
887
  )
838
888
 
839
- results.cache = cache.new_entries_cache()
840
-
841
- self._output(f"There are {len(cache.keys()):,} entries in the local cache.")
842
- else:
843
- cache_difference = coop.remote_cache_get_diff(cache.keys())
844
-
845
- client_missing_cacheentries = cache_difference.get(
846
- "client_missing_cacheentries", []
847
- )
848
-
849
- missing_entry_count = len(client_missing_cacheentries)
850
- if missing_entry_count > 0:
851
- self._output(
852
- f"Updating local cache with {missing_entry_count:,} new "
853
- f"{'entry' if missing_entry_count == 1 else 'entries'} from remote..."
854
- )
855
- cache.add_from_dict(
856
- {entry.key: entry for entry in client_missing_cacheentries}
857
- )
858
- self._output("Local cache updated!")
859
- else:
860
- self._output("No new entries to add to local cache.")
861
-
862
- server_missing_cacheentry_keys = cache_difference.get(
863
- "server_missing_cacheentry_keys", []
864
- )
865
- server_missing_cacheentries = [
866
- entry
867
- for key in server_missing_cacheentry_keys
868
- if (entry := cache.data.get(key)) is not None
869
- ]
870
- old_entry_keys = [key for key in cache.keys()]
871
-
872
- self._output("Running job...")
873
- results = self._run_local(
874
- n=n,
875
- progress_bar=progress_bar,
876
- cache=cache,
877
- stop_on_exception=stop_on_exception,
878
- sidecar_model=sidecar_model,
879
- print_exceptions=print_exceptions,
880
- raise_validation_errors=raise_validation_errors,
881
- )
882
- self._output("Job completed!")
883
-
884
- new_cache_entries = list(
885
- [entry for entry in cache.values() if entry.key not in old_entry_keys]
886
- )
887
- server_missing_cacheentries.extend(new_cache_entries)
888
-
889
- new_entry_count = len(server_missing_cacheentries)
890
- if new_entry_count > 0:
891
- self._output(
892
- f"Updating remote cache with {new_entry_count:,} new "
893
- f"{'entry' if new_entry_count == 1 else 'entries'}..."
894
- )
895
- coop.remote_cache_create_many(
896
- server_missing_cacheentries,
897
- visibility="private",
898
- description=remote_cache_description,
899
- )
900
- self._output("Remote cache updated!")
901
- else:
902
- self._output("No new entries to add to remote cache.")
903
-
904
- results.cache = cache.new_entries_cache()
905
-
906
- self._output(f"There are {len(cache.keys()):,} entries in the local cache.")
907
-
889
+ results.cache = cache.new_entries_cache()
908
890
  return results
909
891
 
910
892
  def _run_local(self, *args, **kwargs):
@@ -159,13 +159,13 @@ class Interview:
159
159
  return self.task_creators.interview_status
160
160
 
161
161
  # region: Serialization
162
- def _to_dict(self, include_exceptions=False) -> dict[str, Any]:
162
+ def _to_dict(self, include_exceptions=True) -> dict[str, Any]:
163
163
  """Return a dictionary representation of the Interview instance.
164
164
  This is just for hashing purposes.
165
165
 
166
166
  >>> i = Interview.example()
167
167
  >>> hash(i)
168
- 1646262796627658719
168
+ 1217840301076717434
169
169
  """
170
170
  d = {
171
171
  "agent": self.agent._to_dict(),
@@ -177,11 +177,29 @@ class Interview:
177
177
  }
178
178
  if include_exceptions:
179
179
  d["exceptions"] = self.exceptions.to_dict()
180
+ return d
181
+
182
+ @classmethod
183
+ def from_dict(cls, d: dict[str, Any]) -> "Interview":
184
+ """Return an Interview instance from a dictionary."""
185
+ agent = Agent.from_dict(d["agent"])
186
+ survey = Survey.from_dict(d["survey"])
187
+ scenario = Scenario.from_dict(d["scenario"])
188
+ model = LanguageModel.from_dict(d["model"])
189
+ iteration = d["iteration"]
190
+ return cls(agent=agent, survey=survey, scenario=scenario, model=model, iteration=iteration)
180
191
 
181
192
  def __hash__(self) -> int:
182
193
  from edsl.utilities.utilities import dict_hash
183
194
 
184
- return dict_hash(self._to_dict())
195
+ return dict_hash(self._to_dict(include_exceptions=False))
196
+
197
+ def __eq__(self, other: "Interview") -> bool:
198
+ """
199
+ >>> from edsl.jobs.interviews.Interview import Interview; i = Interview.example(); d = i._to_dict(); i2 = Interview.from_dict(d); i == i2
200
+ True
201
+ """
202
+ return hash(self) == hash(other)
185
203
 
186
204
  # endregion
187
205
 
@@ -33,6 +33,15 @@ class InterviewExceptionCollection(UserDict):
33
33
  """Return the collection of exceptions as a dictionary."""
34
34
  newdata = {k: [e.to_dict() for e in v] for k, v in self.data.items()}
35
35
  return newdata
36
+
37
+ @classmethod
38
+ def from_dict(cls, data: dict) -> "InterviewExceptionCollection":
39
+ """Create an InterviewExceptionCollection from a dictionary."""
40
+ collection = cls()
41
+ for question_name, entries in data.items():
42
+ for entry in entries:
43
+ collection.add(question_name, InterviewExceptionEntry.from_dict(entry))
44
+ return collection
36
45
 
37
46
  def _repr_html_(self) -> str:
38
47
  from edsl.utilities.utilities import data_to_html
@@ -9,7 +9,6 @@ class InterviewExceptionEntry:
9
9
  self,
10
10
  *,
11
11
  exception: Exception,
12
- # failed_question: FailedQuestion,
13
12
  invigilator: "Invigilator",
14
13
  traceback_format="text",
15
14
  answers=None,
@@ -133,22 +132,41 @@ class InterviewExceptionEntry:
133
132
  )
134
133
  console.print(tb)
135
134
  return html_output.getvalue()
135
+
136
+ @staticmethod
137
+ def serialize_exception(exception: Exception) -> dict:
138
+ return {
139
+ "type": type(exception).__name__,
140
+ "message": str(exception),
141
+ "traceback": "".join(traceback.format_exception(type(exception), exception, exception.__traceback__)),
142
+ }
143
+
144
+ @staticmethod
145
+ def deserialize_exception(data: dict) -> Exception:
146
+ exception_class = globals()[data["type"]]
147
+ return exception_class(data["message"])
136
148
 
137
149
  def to_dict(self) -> dict:
138
150
  """Return the exception as a dictionary.
139
151
 
140
152
  >>> entry = InterviewExceptionEntry.example()
141
- >>> entry.to_dict()['exception']
142
- ValueError()
143
-
153
+ >>> _ = entry.to_dict()
144
154
  """
145
155
  return {
146
- "exception": self.exception,
156
+ "exception": self.serialize_exception(self.exception),
147
157
  "time": self.time,
148
158
  "traceback": self.traceback,
149
- # "failed_question": self.failed_question.to_dict(),
150
159
  "invigilator": self.invigilator.to_dict(),
151
160
  }
161
+
162
+ @classmethod
163
+ def from_dict(cls, data: dict) -> "InterviewExceptionEntry":
164
+ """Create an InterviewExceptionEntry from a dictionary."""
165
+ from edsl.agents.Invigilator import InvigilatorAI
166
+
167
+ exception = cls.deserialize_exception(data["exception"])
168
+ invigilator = InvigilatorAI.from_dict(data["invigilator"])
169
+ return cls(exception=exception, invigilator=invigilator)
152
170
 
153
171
  def push(self):
154
172
  from edsl import Coop
@@ -1,18 +1,12 @@
1
1
  from __future__ import annotations
2
2
  import time
3
- import math
4
3
  import asyncio
5
- import functools
6
4
  import threading
7
5
  from typing import Coroutine, List, AsyncGenerator, Optional, Union, Generator
8
6
  from contextlib import contextmanager
9
7
  from collections import UserList
10
8
 
11
- from rich.live import Live
12
- from rich.console import Console
13
-
14
9
  from edsl.results.Results import Results
15
- from edsl import shared_globals
16
10
  from edsl.jobs.interviews.Interview import Interview
17
11
  from edsl.jobs.runners.JobsRunnerStatus import JobsRunnerStatus
18
12
 
@@ -25,7 +19,6 @@ from edsl.results.Results import Results
25
19
  from edsl.language_models.LanguageModel import LanguageModel
26
20
  from edsl.data.Cache import Cache
27
21
 
28
-
29
22
  class StatusTracker(UserList):
30
23
  def __init__(self, total_tasks: int):
31
24
  self.total_tasks = total_tasks
@@ -48,8 +41,6 @@ class JobsRunnerAsyncio:
48
41
  self.bucket_collection: "BucketCollection" = jobs.bucket_collection
49
42
  self.total_interviews: List["Interview"] = []
50
43
 
51
- # self.jobs_runner_status = JobsRunnerStatus(self, n=1)
52
-
53
44
  async def run_async_generator(
54
45
  self,
55
46
  cache: Cache,
@@ -228,17 +219,16 @@ class JobsRunnerAsyncio:
228
219
  }
229
220
  interview_hashes = list(interview_lookup.keys())
230
221
 
222
+ task_history = TaskHistory(self.total_interviews, include_traceback=False)
223
+
231
224
  results = Results(
232
225
  survey=self.jobs.survey,
233
226
  data=sorted(
234
227
  raw_results, key=lambda x: interview_hashes.index(x.interview_hash)
235
228
  ),
229
+ task_history=task_history,
230
+ cache=cache,
236
231
  )
237
- results.cache = cache
238
- results.task_history = TaskHistory(
239
- self.total_interviews, include_traceback=False
240
- )
241
- results.has_unfixed_exceptions = results.task_history.has_unfixed_exceptions
242
232
  results.bucket_collection = self.bucket_collection
243
233
 
244
234
  if results.has_unfixed_exceptions and print_exceptions:
@@ -266,6 +256,7 @@ class JobsRunnerAsyncio:
266
256
  except Exception as e:
267
257
  print(e)
268
258
  remote_logging = False
259
+
269
260
  if remote_logging:
270
261
  filestore = HTMLFileStore(filepath)
271
262
  coop_details = filestore.push(description="Error report")