edsl 0.1.38.dev2__py3-none-any.whl → 0.1.38.dev3__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (248) hide show
  1. edsl/Base.py +303 -303
  2. edsl/BaseDiff.py +260 -260
  3. edsl/TemplateLoader.py +24 -24
  4. edsl/__init__.py +49 -49
  5. edsl/__version__.py +1 -1
  6. edsl/agents/Agent.py +858 -858
  7. edsl/agents/AgentList.py +362 -362
  8. edsl/agents/Invigilator.py +222 -222
  9. edsl/agents/InvigilatorBase.py +284 -284
  10. edsl/agents/PromptConstructor.py +353 -353
  11. edsl/agents/__init__.py +3 -3
  12. edsl/agents/descriptors.py +99 -99
  13. edsl/agents/prompt_helpers.py +129 -129
  14. edsl/auto/AutoStudy.py +117 -117
  15. edsl/auto/StageBase.py +230 -230
  16. edsl/auto/StageGenerateSurvey.py +178 -178
  17. edsl/auto/StageLabelQuestions.py +125 -125
  18. edsl/auto/StagePersona.py +61 -61
  19. edsl/auto/StagePersonaDimensionValueRanges.py +88 -88
  20. edsl/auto/StagePersonaDimensionValues.py +74 -74
  21. edsl/auto/StagePersonaDimensions.py +69 -69
  22. edsl/auto/StageQuestions.py +73 -73
  23. edsl/auto/SurveyCreatorPipeline.py +21 -21
  24. edsl/auto/utilities.py +224 -224
  25. edsl/base/Base.py +279 -279
  26. edsl/config.py +149 -149
  27. edsl/conversation/Conversation.py +290 -290
  28. edsl/conversation/car_buying.py +58 -58
  29. edsl/conversation/chips.py +95 -95
  30. edsl/conversation/mug_negotiation.py +81 -81
  31. edsl/conversation/next_speaker_utilities.py +93 -93
  32. edsl/coop/PriceFetcher.py +54 -54
  33. edsl/coop/__init__.py +2 -2
  34. edsl/coop/coop.py +961 -961
  35. edsl/coop/utils.py +131 -131
  36. edsl/data/Cache.py +530 -530
  37. edsl/data/CacheEntry.py +228 -228
  38. edsl/data/CacheHandler.py +149 -149
  39. edsl/data/RemoteCacheSync.py +97 -97
  40. edsl/data/SQLiteDict.py +292 -292
  41. edsl/data/__init__.py +4 -4
  42. edsl/data/orm.py +10 -10
  43. edsl/data_transfer_models.py +73 -73
  44. edsl/enums.py +173 -173
  45. edsl/exceptions/BaseException.py +21 -21
  46. edsl/exceptions/__init__.py +54 -54
  47. edsl/exceptions/agents.py +42 -42
  48. edsl/exceptions/cache.py +5 -5
  49. edsl/exceptions/configuration.py +16 -16
  50. edsl/exceptions/coop.py +10 -10
  51. edsl/exceptions/data.py +14 -14
  52. edsl/exceptions/general.py +34 -34
  53. edsl/exceptions/jobs.py +33 -33
  54. edsl/exceptions/language_models.py +63 -63
  55. edsl/exceptions/prompts.py +15 -15
  56. edsl/exceptions/questions.py +91 -91
  57. edsl/exceptions/results.py +29 -29
  58. edsl/exceptions/scenarios.py +22 -22
  59. edsl/exceptions/surveys.py +37 -37
  60. edsl/inference_services/AnthropicService.py +87 -87
  61. edsl/inference_services/AwsBedrock.py +120 -120
  62. edsl/inference_services/AzureAI.py +217 -217
  63. edsl/inference_services/DeepInfraService.py +18 -18
  64. edsl/inference_services/GoogleService.py +156 -156
  65. edsl/inference_services/GroqService.py +20 -20
  66. edsl/inference_services/InferenceServiceABC.py +147 -147
  67. edsl/inference_services/InferenceServicesCollection.py +97 -97
  68. edsl/inference_services/MistralAIService.py +123 -123
  69. edsl/inference_services/OllamaService.py +18 -18
  70. edsl/inference_services/OpenAIService.py +224 -224
  71. edsl/inference_services/TestService.py +89 -89
  72. edsl/inference_services/TogetherAIService.py +170 -170
  73. edsl/inference_services/models_available_cache.py +118 -118
  74. edsl/inference_services/rate_limits_cache.py +25 -25
  75. edsl/inference_services/registry.py +39 -39
  76. edsl/inference_services/write_available.py +10 -10
  77. edsl/jobs/Answers.py +56 -56
  78. edsl/jobs/Jobs.py +1358 -1358
  79. edsl/jobs/__init__.py +1 -1
  80. edsl/jobs/buckets/BucketCollection.py +63 -63
  81. edsl/jobs/buckets/ModelBuckets.py +65 -65
  82. edsl/jobs/buckets/TokenBucket.py +251 -251
  83. edsl/jobs/interviews/Interview.py +661 -661
  84. edsl/jobs/interviews/InterviewExceptionCollection.py +99 -99
  85. edsl/jobs/interviews/InterviewExceptionEntry.py +186 -186
  86. edsl/jobs/interviews/InterviewStatistic.py +63 -63
  87. edsl/jobs/interviews/InterviewStatisticsCollection.py +25 -25
  88. edsl/jobs/interviews/InterviewStatusDictionary.py +78 -78
  89. edsl/jobs/interviews/InterviewStatusLog.py +92 -92
  90. edsl/jobs/interviews/ReportErrors.py +66 -66
  91. edsl/jobs/interviews/interview_status_enum.py +9 -9
  92. edsl/jobs/runners/JobsRunnerAsyncio.py +361 -361
  93. edsl/jobs/runners/JobsRunnerStatus.py +332 -332
  94. edsl/jobs/tasks/QuestionTaskCreator.py +242 -242
  95. edsl/jobs/tasks/TaskCreators.py +64 -64
  96. edsl/jobs/tasks/TaskHistory.py +451 -451
  97. edsl/jobs/tasks/TaskStatusLog.py +23 -23
  98. edsl/jobs/tasks/task_status_enum.py +163 -163
  99. edsl/jobs/tokens/InterviewTokenUsage.py +27 -27
  100. edsl/jobs/tokens/TokenUsage.py +34 -34
  101. edsl/language_models/KeyLookup.py +30 -30
  102. edsl/language_models/LanguageModel.py +708 -708
  103. edsl/language_models/ModelList.py +109 -109
  104. edsl/language_models/RegisterLanguageModelsMeta.py +184 -184
  105. edsl/language_models/__init__.py +3 -3
  106. edsl/language_models/fake_openai_call.py +15 -15
  107. edsl/language_models/fake_openai_service.py +61 -61
  108. edsl/language_models/registry.py +137 -137
  109. edsl/language_models/repair.py +156 -156
  110. edsl/language_models/unused/ReplicateBase.py +83 -83
  111. edsl/language_models/utilities.py +64 -64
  112. edsl/notebooks/Notebook.py +258 -258
  113. edsl/notebooks/__init__.py +1 -1
  114. edsl/prompts/Prompt.py +357 -357
  115. edsl/prompts/__init__.py +2 -2
  116. edsl/questions/AnswerValidatorMixin.py +289 -289
  117. edsl/questions/QuestionBase.py +660 -660
  118. edsl/questions/QuestionBaseGenMixin.py +161 -161
  119. edsl/questions/QuestionBasePromptsMixin.py +217 -217
  120. edsl/questions/QuestionBudget.py +227 -227
  121. edsl/questions/QuestionCheckBox.py +359 -359
  122. edsl/questions/QuestionExtract.py +183 -183
  123. edsl/questions/QuestionFreeText.py +114 -114
  124. edsl/questions/QuestionFunctional.py +166 -166
  125. edsl/questions/QuestionList.py +231 -231
  126. edsl/questions/QuestionMultipleChoice.py +286 -286
  127. edsl/questions/QuestionNumerical.py +153 -153
  128. edsl/questions/QuestionRank.py +324 -324
  129. edsl/questions/Quick.py +41 -41
  130. edsl/questions/RegisterQuestionsMeta.py +71 -71
  131. edsl/questions/ResponseValidatorABC.py +174 -174
  132. edsl/questions/SimpleAskMixin.py +73 -73
  133. edsl/questions/__init__.py +26 -26
  134. edsl/questions/compose_questions.py +98 -98
  135. edsl/questions/decorators.py +21 -21
  136. edsl/questions/derived/QuestionLikertFive.py +76 -76
  137. edsl/questions/derived/QuestionLinearScale.py +87 -87
  138. edsl/questions/derived/QuestionTopK.py +93 -93
  139. edsl/questions/derived/QuestionYesNo.py +82 -82
  140. edsl/questions/descriptors.py +413 -413
  141. edsl/questions/prompt_templates/question_budget.jinja +13 -13
  142. edsl/questions/prompt_templates/question_checkbox.jinja +32 -32
  143. edsl/questions/prompt_templates/question_extract.jinja +11 -11
  144. edsl/questions/prompt_templates/question_free_text.jinja +3 -3
  145. edsl/questions/prompt_templates/question_linear_scale.jinja +11 -11
  146. edsl/questions/prompt_templates/question_list.jinja +17 -17
  147. edsl/questions/prompt_templates/question_multiple_choice.jinja +33 -33
  148. edsl/questions/prompt_templates/question_numerical.jinja +36 -36
  149. edsl/questions/question_registry.py +147 -147
  150. edsl/questions/settings.py +12 -12
  151. edsl/questions/templates/budget/answering_instructions.jinja +7 -7
  152. edsl/questions/templates/budget/question_presentation.jinja +7 -7
  153. edsl/questions/templates/checkbox/answering_instructions.jinja +10 -10
  154. edsl/questions/templates/checkbox/question_presentation.jinja +22 -22
  155. edsl/questions/templates/extract/answering_instructions.jinja +7 -7
  156. edsl/questions/templates/likert_five/answering_instructions.jinja +10 -10
  157. edsl/questions/templates/likert_five/question_presentation.jinja +11 -11
  158. edsl/questions/templates/linear_scale/answering_instructions.jinja +5 -5
  159. edsl/questions/templates/linear_scale/question_presentation.jinja +5 -5
  160. edsl/questions/templates/list/answering_instructions.jinja +3 -3
  161. edsl/questions/templates/list/question_presentation.jinja +5 -5
  162. edsl/questions/templates/multiple_choice/answering_instructions.jinja +9 -9
  163. edsl/questions/templates/multiple_choice/question_presentation.jinja +11 -11
  164. edsl/questions/templates/numerical/answering_instructions.jinja +6 -6
  165. edsl/questions/templates/numerical/question_presentation.jinja +6 -6
  166. edsl/questions/templates/rank/answering_instructions.jinja +11 -11
  167. edsl/questions/templates/rank/question_presentation.jinja +15 -15
  168. edsl/questions/templates/top_k/answering_instructions.jinja +8 -8
  169. edsl/questions/templates/top_k/question_presentation.jinja +22 -22
  170. edsl/questions/templates/yes_no/answering_instructions.jinja +6 -6
  171. edsl/questions/templates/yes_no/question_presentation.jinja +11 -11
  172. edsl/results/Dataset.py +293 -293
  173. edsl/results/DatasetExportMixin.py +717 -717
  174. edsl/results/DatasetTree.py +145 -145
  175. edsl/results/Result.py +456 -456
  176. edsl/results/Results.py +1071 -1071
  177. edsl/results/ResultsDBMixin.py +238 -238
  178. edsl/results/ResultsExportMixin.py +43 -43
  179. edsl/results/ResultsFetchMixin.py +33 -33
  180. edsl/results/ResultsGGMixin.py +121 -121
  181. edsl/results/ResultsToolsMixin.py +98 -98
  182. edsl/results/Selector.py +135 -135
  183. edsl/results/__init__.py +2 -2
  184. edsl/results/tree_explore.py +115 -115
  185. edsl/scenarios/FileStore.py +458 -458
  186. edsl/scenarios/Scenario.py +544 -544
  187. edsl/scenarios/ScenarioHtmlMixin.py +64 -64
  188. edsl/scenarios/ScenarioList.py +1112 -1112
  189. edsl/scenarios/ScenarioListExportMixin.py +52 -52
  190. edsl/scenarios/ScenarioListPdfMixin.py +261 -261
  191. edsl/scenarios/__init__.py +4 -4
  192. edsl/shared.py +1 -1
  193. edsl/study/ObjectEntry.py +173 -173
  194. edsl/study/ProofOfWork.py +113 -113
  195. edsl/study/SnapShot.py +80 -80
  196. edsl/study/Study.py +528 -528
  197. edsl/study/__init__.py +4 -4
  198. edsl/surveys/DAG.py +148 -148
  199. edsl/surveys/Memory.py +31 -31
  200. edsl/surveys/MemoryPlan.py +244 -244
  201. edsl/surveys/Rule.py +326 -326
  202. edsl/surveys/RuleCollection.py +387 -387
  203. edsl/surveys/Survey.py +1787 -1787
  204. edsl/surveys/SurveyCSS.py +261 -261
  205. edsl/surveys/SurveyExportMixin.py +259 -259
  206. edsl/surveys/SurveyFlowVisualizationMixin.py +121 -121
  207. edsl/surveys/SurveyQualtricsImport.py +284 -284
  208. edsl/surveys/__init__.py +3 -3
  209. edsl/surveys/base.py +53 -53
  210. edsl/surveys/descriptors.py +56 -56
  211. edsl/surveys/instructions/ChangeInstruction.py +49 -49
  212. edsl/surveys/instructions/Instruction.py +53 -53
  213. edsl/surveys/instructions/InstructionCollection.py +77 -77
  214. edsl/templates/error_reporting/base.html +23 -23
  215. edsl/templates/error_reporting/exceptions_by_model.html +34 -34
  216. edsl/templates/error_reporting/exceptions_by_question_name.html +16 -16
  217. edsl/templates/error_reporting/exceptions_by_type.html +16 -16
  218. edsl/templates/error_reporting/interview_details.html +115 -115
  219. edsl/templates/error_reporting/interviews.html +9 -9
  220. edsl/templates/error_reporting/overview.html +4 -4
  221. edsl/templates/error_reporting/performance_plot.html +1 -1
  222. edsl/templates/error_reporting/report.css +73 -73
  223. edsl/templates/error_reporting/report.html +117 -117
  224. edsl/templates/error_reporting/report.js +25 -25
  225. edsl/tools/__init__.py +1 -1
  226. edsl/tools/clusters.py +192 -192
  227. edsl/tools/embeddings.py +27 -27
  228. edsl/tools/embeddings_plotting.py +118 -118
  229. edsl/tools/plotting.py +112 -112
  230. edsl/tools/summarize.py +18 -18
  231. edsl/utilities/SystemInfo.py +28 -28
  232. edsl/utilities/__init__.py +22 -22
  233. edsl/utilities/ast_utilities.py +25 -25
  234. edsl/utilities/data/Registry.py +6 -6
  235. edsl/utilities/data/__init__.py +1 -1
  236. edsl/utilities/data/scooter_results.json +1 -1
  237. edsl/utilities/decorators.py +77 -77
  238. edsl/utilities/gcp_bucket/cloud_storage.py +96 -96
  239. edsl/utilities/interface.py +627 -627
  240. edsl/utilities/naming_utilities.py +263 -263
  241. edsl/utilities/repair_functions.py +28 -28
  242. edsl/utilities/restricted_python.py +70 -70
  243. edsl/utilities/utilities.py +409 -409
  244. {edsl-0.1.38.dev2.dist-info → edsl-0.1.38.dev3.dist-info}/LICENSE +21 -21
  245. {edsl-0.1.38.dev2.dist-info → edsl-0.1.38.dev3.dist-info}/METADATA +1 -1
  246. edsl-0.1.38.dev3.dist-info/RECORD +269 -0
  247. edsl-0.1.38.dev2.dist-info/RECORD +0 -269
  248. {edsl-0.1.38.dev2.dist-info → edsl-0.1.38.dev3.dist-info}/WHEEL +0 -0
@@ -1,52 +1,52 @@
1
- """Mixin class for exporting results."""
2
-
3
- from functools import wraps
4
- from edsl.results.DatasetExportMixin import DatasetExportMixin
5
-
6
-
7
- def to_dataset(func):
8
- """Convert the object to a Dataset object before calling the function."""
9
-
10
- @wraps(func)
11
- def wrapper(self, *args, **kwargs):
12
- """Return the function with the Results object converted to a Dataset object."""
13
- if self.__class__.__name__ == "ScenarioList":
14
- return func(self.to_dataset(), *args, **kwargs)
15
- else:
16
- raise Exception(
17
- f"Class {self.__class__.__name__} not recognized as a Results or Dataset object."
18
- )
19
-
20
- return wrapper
21
-
22
-
23
- def decorate_methods_from_mixin(cls, mixin_cls):
24
- for attr_name, attr_value in mixin_cls.__dict__.items():
25
- if callable(attr_value) and not attr_name.startswith("__"):
26
- setattr(cls, attr_name, to_dataset(attr_value))
27
- return cls
28
-
29
-
30
- # def decorate_all_methods(cls):
31
- # for attr_name, attr_value in cls.__dict__.items():
32
- # if callable(attr_value):
33
- # setattr(cls, attr_name, to_dataset(attr_value))
34
- # return cls
35
-
36
-
37
- # @decorate_all_methods
38
- class ScenarioListExportMixin(DatasetExportMixin):
39
- """Mixin class for exporting Results objects."""
40
-
41
- def __init_subclass__(cls, **kwargs):
42
- super().__init_subclass__(**kwargs)
43
- decorate_methods_from_mixin(cls, DatasetExportMixin)
44
-
45
- def to_docx(self, filename: str):
46
- """Export the ScenarioList to a .docx file."""
47
- dataset = self.to_dataset()
48
- from edsl.results.DatasetTree import Tree
49
-
50
- tree = Tree(dataset)
51
- tree.construct_tree()
52
- tree.to_docx(filename)
1
+ """Mixin class for exporting results."""
2
+
3
+ from functools import wraps
4
+ from edsl.results.DatasetExportMixin import DatasetExportMixin
5
+
6
+
7
+ def to_dataset(func):
8
+ """Convert the object to a Dataset object before calling the function."""
9
+
10
+ @wraps(func)
11
+ def wrapper(self, *args, **kwargs):
12
+ """Return the function with the Results object converted to a Dataset object."""
13
+ if self.__class__.__name__ == "ScenarioList":
14
+ return func(self.to_dataset(), *args, **kwargs)
15
+ else:
16
+ raise Exception(
17
+ f"Class {self.__class__.__name__} not recognized as a Results or Dataset object."
18
+ )
19
+
20
+ return wrapper
21
+
22
+
23
+ def decorate_methods_from_mixin(cls, mixin_cls):
24
+ for attr_name, attr_value in mixin_cls.__dict__.items():
25
+ if callable(attr_value) and not attr_name.startswith("__"):
26
+ setattr(cls, attr_name, to_dataset(attr_value))
27
+ return cls
28
+
29
+
30
+ # def decorate_all_methods(cls):
31
+ # for attr_name, attr_value in cls.__dict__.items():
32
+ # if callable(attr_value):
33
+ # setattr(cls, attr_name, to_dataset(attr_value))
34
+ # return cls
35
+
36
+
37
+ # @decorate_all_methods
38
+ class ScenarioListExportMixin(DatasetExportMixin):
39
+ """Mixin class for exporting Results objects."""
40
+
41
+ def __init_subclass__(cls, **kwargs):
42
+ super().__init_subclass__(**kwargs)
43
+ decorate_methods_from_mixin(cls, DatasetExportMixin)
44
+
45
+ def to_docx(self, filename: str):
46
+ """Export the ScenarioList to a .docx file."""
47
+ dataset = self.to_dataset()
48
+ from edsl.results.DatasetTree import Tree
49
+
50
+ tree = Tree(dataset)
51
+ tree.construct_tree()
52
+ tree.to_docx(filename)
@@ -1,261 +1,261 @@
1
- import fitz # PyMuPDF
2
- import os
3
- import copy
4
- import subprocess
5
- import requests
6
- import tempfile
7
- import os
8
-
9
- # import urllib.parse as urlparse
10
- from urllib.parse import urlparse
11
-
12
- # from edsl import Scenario
13
-
14
- import requests
15
- import re
16
- import tempfile
17
- import os
18
- import atexit
19
- from urllib.parse import urlparse, parse_qs
20
-
21
-
22
- class GoogleDriveDownloader:
23
- _temp_dir = None
24
- _temp_file_path = None
25
-
26
- @classmethod
27
- def fetch_from_drive(cls, url, filename=None):
28
- # Extract file ID from the URL
29
- file_id = cls._extract_file_id(url)
30
- if not file_id:
31
- raise ValueError("Invalid Google Drive URL")
32
-
33
- # Construct the download URL
34
- download_url = f"https://drive.google.com/uc?export=download&id={file_id}"
35
-
36
- # Send a GET request to the URL
37
- session = requests.Session()
38
- response = session.get(download_url, stream=True)
39
- response.raise_for_status()
40
-
41
- # Check for large file download prompt
42
- for key, value in response.cookies.items():
43
- if key.startswith("download_warning"):
44
- params = {"id": file_id, "confirm": value}
45
- response = session.get(download_url, params=params, stream=True)
46
- break
47
-
48
- # Create a temporary file to save the download
49
- if not filename:
50
- filename = "downloaded_file"
51
-
52
- if cls._temp_dir is None:
53
- cls._temp_dir = tempfile.TemporaryDirectory()
54
- atexit.register(cls._cleanup)
55
-
56
- cls._temp_file_path = os.path.join(cls._temp_dir.name, filename)
57
-
58
- # Write the content to the temporary file
59
- with open(cls._temp_file_path, "wb") as f:
60
- for chunk in response.iter_content(32768):
61
- if chunk:
62
- f.write(chunk)
63
-
64
- print(f"File saved to: {cls._temp_file_path}")
65
-
66
- return cls._temp_file_path
67
-
68
- @staticmethod
69
- def _extract_file_id(url):
70
- # Try to extract file ID from '/file/d/' format
71
- file_id_match = re.search(r"/d/([a-zA-Z0-9-_]+)", url)
72
- if file_id_match:
73
- return file_id_match.group(1)
74
-
75
- # If not found, try to extract from 'open?id=' format
76
- parsed_url = urlparse(url)
77
- query_params = parse_qs(parsed_url.query)
78
- if "id" in query_params:
79
- return query_params["id"][0]
80
-
81
- return None
82
-
83
- @classmethod
84
- def _cleanup(cls):
85
- if cls._temp_dir:
86
- cls._temp_dir.cleanup()
87
-
88
- @classmethod
89
- def get_temp_file_path(cls):
90
- return cls._temp_file_path
91
-
92
-
93
- def fetch_and_save_pdf(url, filename):
94
- # Send a GET request to the URL
95
- response = requests.get(url)
96
-
97
- # Check if the request was successful
98
- response.raise_for_status()
99
-
100
- # Create a temporary directory
101
- with tempfile.TemporaryDirectory() as temp_dir:
102
- # Construct the full path for the file
103
- temp_file_path = os.path.join(temp_dir, filename)
104
-
105
- # Write the content to the temporary file
106
- with open(temp_file_path, "wb") as file:
107
- file.write(response.content)
108
-
109
- print(f"PDF saved to: {temp_file_path}")
110
-
111
- # Here you can perform operations with the file
112
- # The file will be automatically deleted when you exit this block
113
-
114
- return temp_file_path
115
-
116
-
117
- # Example usage:
118
- # url = "https://example.com/sample.pdf"
119
- # fetch_and_save_pdf(url, "sample.pdf")
120
-
121
-
122
- class ScenarioListPdfMixin:
123
- @classmethod
124
- def from_pdf(cls, filename_or_url, collapse_pages=False):
125
- # Check if the input is a URL
126
- if cls.is_url(filename_or_url):
127
- # Check if it's a Google Drive URL
128
- if "drive.google.com" in filename_or_url:
129
- temp_filename = GoogleDriveDownloader.fetch_from_drive(
130
- filename_or_url, "temp_pdf.pdf"
131
- )
132
- else:
133
- # For other URLs, use the previous fetch_and_save_pdf function
134
- temp_filename = fetch_and_save_pdf(filename_or_url, "temp_pdf.pdf")
135
-
136
- scenarios = list(cls.extract_text_from_pdf(temp_filename))
137
- else:
138
- # If it's not a URL, assume it's a local file path
139
- scenarios = list(cls.extract_text_from_pdf(filename_or_url))
140
- if not collapse_pages:
141
- return cls(scenarios)
142
- else:
143
- txt = ""
144
- for scenario in scenarios:
145
- txt += scenario["text"]
146
- from edsl.scenarios import Scenario
147
-
148
- base_scenario = copy.copy(scenarios[0])
149
- base_scenario["text"] = txt
150
- return base_scenario
151
-
152
- @staticmethod
153
- def is_url(string):
154
- try:
155
- result = urlparse(string)
156
- return all([result.scheme, result.netloc])
157
- except ValueError:
158
- return False
159
-
160
- @classmethod
161
- def _from_pdf_to_image(cls, pdf_path, image_format="jpeg"):
162
- """
163
- Convert each page of a PDF into an image and create Scenario instances.
164
-
165
- :param pdf_path: Path to the PDF file.
166
- :param image_format: Format of the output images (default is 'jpeg').
167
- :return: ScenarioList instance containing the Scenario instances.
168
- """
169
- import tempfile
170
- from pdf2image import convert_from_path
171
- from edsl.scenarios import Scenario
172
-
173
- with tempfile.TemporaryDirectory() as output_folder:
174
- # Convert PDF to images
175
- images = convert_from_path(pdf_path)
176
-
177
- scenarios = []
178
-
179
- # Save each page as an image and create Scenario instances
180
- for i, image in enumerate(images):
181
- image_path = os.path.join(output_folder, f"page_{i+1}.{image_format}")
182
- image.save(image_path, image_format.upper())
183
-
184
- scenario = Scenario._from_filepath_image(image_path)
185
- scenarios.append(scenario)
186
-
187
- # print(f"Saved {len(images)} pages as images in {output_folder}")
188
- return cls(scenarios)
189
-
190
- @staticmethod
191
- def extract_text_from_pdf(pdf_path):
192
- from edsl import Scenario
193
-
194
- # TODO: Add test case
195
- # Ensure the file exists
196
- if not os.path.exists(pdf_path):
197
- raise FileNotFoundError(f"The file {pdf_path} does not exist.")
198
-
199
- # Open the PDF file
200
- document = fitz.open(pdf_path)
201
-
202
- # Get the filename from the path
203
- filename = os.path.basename(pdf_path)
204
-
205
- # Iterate through each page and extract text
206
- for page_num in range(len(document)):
207
- page = document.load_page(page_num)
208
- text = page.get_text()
209
-
210
- # Create a dictionary for the current page
211
- page_info = {"filename": filename, "page": page_num + 1, "text": text}
212
- yield Scenario(page_info)
213
-
214
- def create_hello_world_pdf(pdf_path):
215
- # LaTeX content
216
- latex_content = r"""
217
- \documentclass{article}
218
- \title{Hello World}
219
- \author{John}
220
- \date{\today}
221
- \begin{document}
222
- \maketitle
223
- \section{Hello, World!}
224
- This is a simple hello world example created with LaTeX and Python.
225
- \end{document}
226
- """
227
-
228
- # Create a .tex file
229
- tex_filename = pdf_path + ".tex"
230
- with open(tex_filename, "w") as tex_file:
231
- tex_file.write(latex_content)
232
-
233
- # Compile the .tex file to PDF
234
- subprocess.run(["pdflatex", tex_filename], check=True)
235
-
236
- # Optionally, clean up auxiliary files generated by pdflatex
237
- aux_files = [pdf_path + ext for ext in [".aux", ".log"]]
238
- for aux_file in aux_files:
239
- try:
240
- os.remove(aux_file)
241
- except FileNotFoundError:
242
- pass
243
-
244
-
245
- if __name__ == "__main__":
246
- pass
247
-
248
- # from edsl import ScenarioList
249
-
250
- # class ScenarioListNew(ScenarioList, ScenaroListPdfMixin):
251
- # pass
252
-
253
- # #ScenarioListNew.create_hello_world_pdf('hello_world')
254
- # #scenarios = ScenarioListNew.from_pdf('hello_world.pdf')
255
- # #print(scenarios)
256
-
257
- # from edsl import ScenarioList, QuestionFreeText
258
- # homo_silicus = ScenarioList.from_pdf('w31122.pdf')
259
- # q = QuestionFreeText(question_text = "What is the key point of the text in {{ text }}?", question_name = "key_point")
260
- # results = q.by(homo_silicus).run(progress_bar = True)
261
- # results.select('scenario.page', 'answer.key_point').order_by('page').print()
1
+ import fitz # PyMuPDF
2
+ import os
3
+ import copy
4
+ import subprocess
5
+ import requests
6
+ import tempfile
7
+ import os
8
+
9
+ # import urllib.parse as urlparse
10
+ from urllib.parse import urlparse
11
+
12
+ # from edsl import Scenario
13
+
14
+ import requests
15
+ import re
16
+ import tempfile
17
+ import os
18
+ import atexit
19
+ from urllib.parse import urlparse, parse_qs
20
+
21
+
22
+ class GoogleDriveDownloader:
23
+ _temp_dir = None
24
+ _temp_file_path = None
25
+
26
+ @classmethod
27
+ def fetch_from_drive(cls, url, filename=None):
28
+ # Extract file ID from the URL
29
+ file_id = cls._extract_file_id(url)
30
+ if not file_id:
31
+ raise ValueError("Invalid Google Drive URL")
32
+
33
+ # Construct the download URL
34
+ download_url = f"https://drive.google.com/uc?export=download&id={file_id}"
35
+
36
+ # Send a GET request to the URL
37
+ session = requests.Session()
38
+ response = session.get(download_url, stream=True)
39
+ response.raise_for_status()
40
+
41
+ # Check for large file download prompt
42
+ for key, value in response.cookies.items():
43
+ if key.startswith("download_warning"):
44
+ params = {"id": file_id, "confirm": value}
45
+ response = session.get(download_url, params=params, stream=True)
46
+ break
47
+
48
+ # Create a temporary file to save the download
49
+ if not filename:
50
+ filename = "downloaded_file"
51
+
52
+ if cls._temp_dir is None:
53
+ cls._temp_dir = tempfile.TemporaryDirectory()
54
+ atexit.register(cls._cleanup)
55
+
56
+ cls._temp_file_path = os.path.join(cls._temp_dir.name, filename)
57
+
58
+ # Write the content to the temporary file
59
+ with open(cls._temp_file_path, "wb") as f:
60
+ for chunk in response.iter_content(32768):
61
+ if chunk:
62
+ f.write(chunk)
63
+
64
+ print(f"File saved to: {cls._temp_file_path}")
65
+
66
+ return cls._temp_file_path
67
+
68
+ @staticmethod
69
+ def _extract_file_id(url):
70
+ # Try to extract file ID from '/file/d/' format
71
+ file_id_match = re.search(r"/d/([a-zA-Z0-9-_]+)", url)
72
+ if file_id_match:
73
+ return file_id_match.group(1)
74
+
75
+ # If not found, try to extract from 'open?id=' format
76
+ parsed_url = urlparse(url)
77
+ query_params = parse_qs(parsed_url.query)
78
+ if "id" in query_params:
79
+ return query_params["id"][0]
80
+
81
+ return None
82
+
83
+ @classmethod
84
+ def _cleanup(cls):
85
+ if cls._temp_dir:
86
+ cls._temp_dir.cleanup()
87
+
88
+ @classmethod
89
+ def get_temp_file_path(cls):
90
+ return cls._temp_file_path
91
+
92
+
93
+ def fetch_and_save_pdf(url, filename):
94
+ # Send a GET request to the URL
95
+ response = requests.get(url)
96
+
97
+ # Check if the request was successful
98
+ response.raise_for_status()
99
+
100
+ # Create a temporary directory
101
+ with tempfile.TemporaryDirectory() as temp_dir:
102
+ # Construct the full path for the file
103
+ temp_file_path = os.path.join(temp_dir, filename)
104
+
105
+ # Write the content to the temporary file
106
+ with open(temp_file_path, "wb") as file:
107
+ file.write(response.content)
108
+
109
+ print(f"PDF saved to: {temp_file_path}")
110
+
111
+ # Here you can perform operations with the file
112
+ # The file will be automatically deleted when you exit this block
113
+
114
+ return temp_file_path
115
+
116
+
117
+ # Example usage:
118
+ # url = "https://example.com/sample.pdf"
119
+ # fetch_and_save_pdf(url, "sample.pdf")
120
+
121
+
122
+ class ScenarioListPdfMixin:
123
+ @classmethod
124
+ def from_pdf(cls, filename_or_url, collapse_pages=False):
125
+ # Check if the input is a URL
126
+ if cls.is_url(filename_or_url):
127
+ # Check if it's a Google Drive URL
128
+ if "drive.google.com" in filename_or_url:
129
+ temp_filename = GoogleDriveDownloader.fetch_from_drive(
130
+ filename_or_url, "temp_pdf.pdf"
131
+ )
132
+ else:
133
+ # For other URLs, use the previous fetch_and_save_pdf function
134
+ temp_filename = fetch_and_save_pdf(filename_or_url, "temp_pdf.pdf")
135
+
136
+ scenarios = list(cls.extract_text_from_pdf(temp_filename))
137
+ else:
138
+ # If it's not a URL, assume it's a local file path
139
+ scenarios = list(cls.extract_text_from_pdf(filename_or_url))
140
+ if not collapse_pages:
141
+ return cls(scenarios)
142
+ else:
143
+ txt = ""
144
+ for scenario in scenarios:
145
+ txt += scenario["text"]
146
+ from edsl.scenarios import Scenario
147
+
148
+ base_scenario = copy.copy(scenarios[0])
149
+ base_scenario["text"] = txt
150
+ return base_scenario
151
+
152
+ @staticmethod
153
+ def is_url(string):
154
+ try:
155
+ result = urlparse(string)
156
+ return all([result.scheme, result.netloc])
157
+ except ValueError:
158
+ return False
159
+
160
+ @classmethod
161
+ def _from_pdf_to_image(cls, pdf_path, image_format="jpeg"):
162
+ """
163
+ Convert each page of a PDF into an image and create Scenario instances.
164
+
165
+ :param pdf_path: Path to the PDF file.
166
+ :param image_format: Format of the output images (default is 'jpeg').
167
+ :return: ScenarioList instance containing the Scenario instances.
168
+ """
169
+ import tempfile
170
+ from pdf2image import convert_from_path
171
+ from edsl.scenarios import Scenario
172
+
173
+ with tempfile.TemporaryDirectory() as output_folder:
174
+ # Convert PDF to images
175
+ images = convert_from_path(pdf_path)
176
+
177
+ scenarios = []
178
+
179
+ # Save each page as an image and create Scenario instances
180
+ for i, image in enumerate(images):
181
+ image_path = os.path.join(output_folder, f"page_{i+1}.{image_format}")
182
+ image.save(image_path, image_format.upper())
183
+
184
+ scenario = Scenario._from_filepath_image(image_path)
185
+ scenarios.append(scenario)
186
+
187
+ # print(f"Saved {len(images)} pages as images in {output_folder}")
188
+ return cls(scenarios)
189
+
190
+ @staticmethod
191
+ def extract_text_from_pdf(pdf_path):
192
+ from edsl import Scenario
193
+
194
+ # TODO: Add test case
195
+ # Ensure the file exists
196
+ if not os.path.exists(pdf_path):
197
+ raise FileNotFoundError(f"The file {pdf_path} does not exist.")
198
+
199
+ # Open the PDF file
200
+ document = fitz.open(pdf_path)
201
+
202
+ # Get the filename from the path
203
+ filename = os.path.basename(pdf_path)
204
+
205
+ # Iterate through each page and extract text
206
+ for page_num in range(len(document)):
207
+ page = document.load_page(page_num)
208
+ text = page.get_text()
209
+
210
+ # Create a dictionary for the current page
211
+ page_info = {"filename": filename, "page": page_num + 1, "text": text}
212
+ yield Scenario(page_info)
213
+
214
+ def create_hello_world_pdf(pdf_path):
215
+ # LaTeX content
216
+ latex_content = r"""
217
+ \documentclass{article}
218
+ \title{Hello World}
219
+ \author{John}
220
+ \date{\today}
221
+ \begin{document}
222
+ \maketitle
223
+ \section{Hello, World!}
224
+ This is a simple hello world example created with LaTeX and Python.
225
+ \end{document}
226
+ """
227
+
228
+ # Create a .tex file
229
+ tex_filename = pdf_path + ".tex"
230
+ with open(tex_filename, "w") as tex_file:
231
+ tex_file.write(latex_content)
232
+
233
+ # Compile the .tex file to PDF
234
+ subprocess.run(["pdflatex", tex_filename], check=True)
235
+
236
+ # Optionally, clean up auxiliary files generated by pdflatex
237
+ aux_files = [pdf_path + ext for ext in [".aux", ".log"]]
238
+ for aux_file in aux_files:
239
+ try:
240
+ os.remove(aux_file)
241
+ except FileNotFoundError:
242
+ pass
243
+
244
+
245
+ if __name__ == "__main__":
246
+ pass
247
+
248
+ # from edsl import ScenarioList
249
+
250
+ # class ScenarioListNew(ScenarioList, ScenaroListPdfMixin):
251
+ # pass
252
+
253
+ # #ScenarioListNew.create_hello_world_pdf('hello_world')
254
+ # #scenarios = ScenarioListNew.from_pdf('hello_world.pdf')
255
+ # #print(scenarios)
256
+
257
+ # from edsl import ScenarioList, QuestionFreeText
258
+ # homo_silicus = ScenarioList.from_pdf('w31122.pdf')
259
+ # q = QuestionFreeText(question_text = "What is the key point of the text in {{ text }}?", question_name = "key_point")
260
+ # results = q.by(homo_silicus).run(progress_bar = True)
261
+ # results.select('scenario.page', 'answer.key_point').order_by('page').print()
@@ -1,4 +1,4 @@
1
- from edsl.scenarios.Scenario import Scenario
2
- from edsl.scenarios.ScenarioList import ScenarioList
3
-
4
- # from edsl.scenarios.FileStore import FileStore
1
+ from edsl.scenarios.Scenario import Scenario
2
+ from edsl.scenarios.ScenarioList import ScenarioList
3
+
4
+ # from edsl.scenarios.FileStore import FileStore
edsl/shared.py CHANGED
@@ -1 +1 @@
1
- shared_globals = {}
1
+ shared_globals = {}