edsl 0.1.36.dev2__py3-none-any.whl → 0.1.36.dev6__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 (257) hide show
  1. edsl/Base.py +303 -298
  2. edsl/BaseDiff.py +260 -260
  3. edsl/TemplateLoader.py +24 -24
  4. edsl/__init__.py +47 -47
  5. edsl/__version__.py +1 -1
  6. edsl/agents/Agent.py +804 -800
  7. edsl/agents/AgentList.py +337 -337
  8. edsl/agents/Invigilator.py +222 -222
  9. edsl/agents/InvigilatorBase.py +294 -294
  10. edsl/agents/PromptConstructor.py +312 -311
  11. edsl/agents/__init__.py +3 -3
  12. edsl/agents/descriptors.py +86 -86
  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 +289 -289
  26. edsl/config.py +149 -149
  27. edsl/conjure/AgentConstructionMixin.py +152 -152
  28. edsl/conjure/Conjure.py +62 -62
  29. edsl/conjure/InputData.py +659 -659
  30. edsl/conjure/InputDataCSV.py +48 -48
  31. edsl/conjure/InputDataMixinQuestionStats.py +182 -182
  32. edsl/conjure/InputDataPyRead.py +91 -91
  33. edsl/conjure/InputDataSPSS.py +8 -8
  34. edsl/conjure/InputDataStata.py +8 -8
  35. edsl/conjure/QuestionOptionMixin.py +76 -76
  36. edsl/conjure/QuestionTypeMixin.py +23 -23
  37. edsl/conjure/RawQuestion.py +65 -65
  38. edsl/conjure/SurveyResponses.py +7 -7
  39. edsl/conjure/__init__.py +9 -9
  40. edsl/conjure/naming_utilities.py +263 -263
  41. edsl/conjure/utilities.py +201 -201
  42. edsl/conversation/Conversation.py +238 -238
  43. edsl/conversation/car_buying.py +58 -58
  44. edsl/conversation/mug_negotiation.py +81 -81
  45. edsl/conversation/next_speaker_utilities.py +93 -93
  46. edsl/coop/PriceFetcher.py +54 -58
  47. edsl/coop/__init__.py +2 -2
  48. edsl/coop/coop.py +849 -815
  49. edsl/coop/utils.py +131 -131
  50. edsl/data/Cache.py +527 -527
  51. edsl/data/CacheEntry.py +228 -228
  52. edsl/data/CacheHandler.py +149 -149
  53. edsl/data/RemoteCacheSync.py +84 -0
  54. edsl/data/SQLiteDict.py +292 -292
  55. edsl/data/__init__.py +4 -4
  56. edsl/data/orm.py +10 -10
  57. edsl/data_transfer_models.py +73 -73
  58. edsl/enums.py +173 -173
  59. edsl/exceptions/__init__.py +50 -50
  60. edsl/exceptions/agents.py +40 -40
  61. edsl/exceptions/configuration.py +16 -16
  62. edsl/exceptions/coop.py +10 -2
  63. edsl/exceptions/data.py +14 -14
  64. edsl/exceptions/general.py +34 -34
  65. edsl/exceptions/jobs.py +33 -33
  66. edsl/exceptions/language_models.py +63 -63
  67. edsl/exceptions/prompts.py +15 -15
  68. edsl/exceptions/questions.py +91 -91
  69. edsl/exceptions/results.py +26 -26
  70. edsl/exceptions/surveys.py +34 -34
  71. edsl/inference_services/AnthropicService.py +87 -87
  72. edsl/inference_services/AwsBedrock.py +115 -115
  73. edsl/inference_services/AzureAI.py +217 -217
  74. edsl/inference_services/DeepInfraService.py +18 -18
  75. edsl/inference_services/GoogleService.py +156 -156
  76. edsl/inference_services/GroqService.py +20 -20
  77. edsl/inference_services/InferenceServiceABC.py +147 -119
  78. edsl/inference_services/InferenceServicesCollection.py +72 -68
  79. edsl/inference_services/MistralAIService.py +123 -123
  80. edsl/inference_services/OllamaService.py +18 -18
  81. edsl/inference_services/OpenAIService.py +224 -224
  82. edsl/inference_services/TestService.py +89 -89
  83. edsl/inference_services/TogetherAIService.py +170 -170
  84. edsl/inference_services/models_available_cache.py +118 -94
  85. edsl/inference_services/rate_limits_cache.py +25 -25
  86. edsl/inference_services/registry.py +39 -39
  87. edsl/inference_services/write_available.py +10 -10
  88. edsl/jobs/Answers.py +56 -56
  89. edsl/jobs/Jobs.py +1112 -1089
  90. edsl/jobs/__init__.py +1 -1
  91. edsl/jobs/buckets/BucketCollection.py +63 -63
  92. edsl/jobs/buckets/ModelBuckets.py +65 -65
  93. edsl/jobs/buckets/TokenBucket.py +248 -248
  94. edsl/jobs/interviews/Interview.py +651 -633
  95. edsl/jobs/interviews/InterviewExceptionCollection.py +99 -90
  96. edsl/jobs/interviews/InterviewExceptionEntry.py +182 -164
  97. edsl/jobs/interviews/InterviewStatistic.py +63 -63
  98. edsl/jobs/interviews/InterviewStatisticsCollection.py +25 -25
  99. edsl/jobs/interviews/InterviewStatusDictionary.py +78 -78
  100. edsl/jobs/interviews/InterviewStatusLog.py +92 -92
  101. edsl/jobs/interviews/ReportErrors.py +66 -66
  102. edsl/jobs/interviews/interview_status_enum.py +9 -9
  103. edsl/jobs/runners/JobsRunnerAsyncio.py +337 -343
  104. edsl/jobs/runners/JobsRunnerStatus.py +332 -332
  105. edsl/jobs/tasks/QuestionTaskCreator.py +242 -242
  106. edsl/jobs/tasks/TaskCreators.py +64 -64
  107. edsl/jobs/tasks/TaskHistory.py +441 -425
  108. edsl/jobs/tasks/TaskStatusLog.py +23 -23
  109. edsl/jobs/tasks/task_status_enum.py +163 -163
  110. edsl/jobs/tokens/InterviewTokenUsage.py +27 -27
  111. edsl/jobs/tokens/TokenUsage.py +34 -34
  112. edsl/language_models/LanguageModel.py +718 -718
  113. edsl/language_models/ModelList.py +102 -102
  114. edsl/language_models/RegisterLanguageModelsMeta.py +184 -184
  115. edsl/language_models/__init__.py +2 -2
  116. edsl/language_models/fake_openai_call.py +15 -15
  117. edsl/language_models/fake_openai_service.py +61 -61
  118. edsl/language_models/registry.py +137 -137
  119. edsl/language_models/repair.py +156 -156
  120. edsl/language_models/unused/ReplicateBase.py +83 -83
  121. edsl/language_models/utilities.py +64 -64
  122. edsl/notebooks/Notebook.py +259 -259
  123. edsl/notebooks/__init__.py +1 -1
  124. edsl/prompts/Prompt.py +358 -358
  125. edsl/prompts/__init__.py +2 -2
  126. edsl/questions/AnswerValidatorMixin.py +289 -289
  127. edsl/questions/QuestionBase.py +616 -616
  128. edsl/questions/QuestionBaseGenMixin.py +161 -161
  129. edsl/questions/QuestionBasePromptsMixin.py +266 -266
  130. edsl/questions/QuestionBudget.py +227 -227
  131. edsl/questions/QuestionCheckBox.py +359 -359
  132. edsl/questions/QuestionExtract.py +183 -183
  133. edsl/questions/QuestionFreeText.py +113 -113
  134. edsl/questions/QuestionFunctional.py +159 -155
  135. edsl/questions/QuestionList.py +231 -231
  136. edsl/questions/QuestionMultipleChoice.py +286 -286
  137. edsl/questions/QuestionNumerical.py +153 -153
  138. edsl/questions/QuestionRank.py +324 -324
  139. edsl/questions/Quick.py +41 -41
  140. edsl/questions/RegisterQuestionsMeta.py +71 -71
  141. edsl/questions/ResponseValidatorABC.py +174 -174
  142. edsl/questions/SimpleAskMixin.py +73 -73
  143. edsl/questions/__init__.py +26 -26
  144. edsl/questions/compose_questions.py +98 -98
  145. edsl/questions/decorators.py +21 -21
  146. edsl/questions/derived/QuestionLikertFive.py +76 -76
  147. edsl/questions/derived/QuestionLinearScale.py +87 -87
  148. edsl/questions/derived/QuestionTopK.py +91 -91
  149. edsl/questions/derived/QuestionYesNo.py +82 -82
  150. edsl/questions/descriptors.py +418 -418
  151. edsl/questions/prompt_templates/question_budget.jinja +13 -13
  152. edsl/questions/prompt_templates/question_checkbox.jinja +32 -32
  153. edsl/questions/prompt_templates/question_extract.jinja +11 -11
  154. edsl/questions/prompt_templates/question_free_text.jinja +3 -3
  155. edsl/questions/prompt_templates/question_linear_scale.jinja +11 -11
  156. edsl/questions/prompt_templates/question_list.jinja +17 -17
  157. edsl/questions/prompt_templates/question_multiple_choice.jinja +33 -33
  158. edsl/questions/prompt_templates/question_numerical.jinja +36 -36
  159. edsl/questions/question_registry.py +147 -147
  160. edsl/questions/settings.py +12 -12
  161. edsl/questions/templates/budget/answering_instructions.jinja +7 -7
  162. edsl/questions/templates/budget/question_presentation.jinja +7 -7
  163. edsl/questions/templates/checkbox/answering_instructions.jinja +10 -10
  164. edsl/questions/templates/checkbox/question_presentation.jinja +22 -22
  165. edsl/questions/templates/extract/answering_instructions.jinja +7 -7
  166. edsl/questions/templates/likert_five/answering_instructions.jinja +10 -10
  167. edsl/questions/templates/likert_five/question_presentation.jinja +11 -11
  168. edsl/questions/templates/linear_scale/answering_instructions.jinja +5 -5
  169. edsl/questions/templates/linear_scale/question_presentation.jinja +5 -5
  170. edsl/questions/templates/list/answering_instructions.jinja +3 -3
  171. edsl/questions/templates/list/question_presentation.jinja +5 -5
  172. edsl/questions/templates/multiple_choice/answering_instructions.jinja +9 -9
  173. edsl/questions/templates/multiple_choice/question_presentation.jinja +11 -11
  174. edsl/questions/templates/numerical/answering_instructions.jinja +6 -6
  175. edsl/questions/templates/numerical/question_presentation.jinja +6 -6
  176. edsl/questions/templates/rank/answering_instructions.jinja +11 -11
  177. edsl/questions/templates/rank/question_presentation.jinja +15 -15
  178. edsl/questions/templates/top_k/answering_instructions.jinja +8 -8
  179. edsl/questions/templates/top_k/question_presentation.jinja +22 -22
  180. edsl/questions/templates/yes_no/answering_instructions.jinja +6 -6
  181. edsl/questions/templates/yes_no/question_presentation.jinja +11 -11
  182. edsl/results/Dataset.py +293 -281
  183. edsl/results/DatasetExportMixin.py +693 -693
  184. edsl/results/DatasetTree.py +145 -145
  185. edsl/results/Result.py +433 -431
  186. edsl/results/Results.py +1158 -1146
  187. edsl/results/ResultsDBMixin.py +238 -238
  188. edsl/results/ResultsExportMixin.py +43 -43
  189. edsl/results/ResultsFetchMixin.py +33 -33
  190. edsl/results/ResultsGGMixin.py +121 -121
  191. edsl/results/ResultsToolsMixin.py +98 -98
  192. edsl/results/Selector.py +118 -118
  193. edsl/results/__init__.py +2 -2
  194. edsl/results/tree_explore.py +115 -115
  195. edsl/scenarios/FileStore.py +443 -443
  196. edsl/scenarios/Scenario.py +507 -496
  197. edsl/scenarios/ScenarioHtmlMixin.py +59 -59
  198. edsl/scenarios/ScenarioList.py +1101 -1101
  199. edsl/scenarios/ScenarioListExportMixin.py +52 -52
  200. edsl/scenarios/ScenarioListPdfMixin.py +261 -261
  201. edsl/scenarios/__init__.py +2 -2
  202. edsl/shared.py +1 -1
  203. edsl/study/ObjectEntry.py +173 -173
  204. edsl/study/ProofOfWork.py +113 -113
  205. edsl/study/SnapShot.py +80 -80
  206. edsl/study/Study.py +528 -528
  207. edsl/study/__init__.py +4 -4
  208. edsl/surveys/DAG.py +148 -148
  209. edsl/surveys/Memory.py +31 -31
  210. edsl/surveys/MemoryPlan.py +244 -244
  211. edsl/surveys/Rule.py +324 -324
  212. edsl/surveys/RuleCollection.py +387 -387
  213. edsl/surveys/Survey.py +1772 -1769
  214. edsl/surveys/SurveyCSS.py +261 -261
  215. edsl/surveys/SurveyExportMixin.py +259 -259
  216. edsl/surveys/SurveyFlowVisualizationMixin.py +121 -121
  217. edsl/surveys/SurveyQualtricsImport.py +284 -284
  218. edsl/surveys/__init__.py +3 -3
  219. edsl/surveys/base.py +53 -53
  220. edsl/surveys/descriptors.py +56 -56
  221. edsl/surveys/instructions/ChangeInstruction.py +47 -47
  222. edsl/surveys/instructions/Instruction.py +51 -34
  223. edsl/surveys/instructions/InstructionCollection.py +77 -77
  224. edsl/templates/error_reporting/base.html +23 -23
  225. edsl/templates/error_reporting/exceptions_by_model.html +34 -34
  226. edsl/templates/error_reporting/exceptions_by_question_name.html +16 -16
  227. edsl/templates/error_reporting/exceptions_by_type.html +16 -16
  228. edsl/templates/error_reporting/interview_details.html +115 -115
  229. edsl/templates/error_reporting/interviews.html +9 -9
  230. edsl/templates/error_reporting/overview.html +4 -4
  231. edsl/templates/error_reporting/performance_plot.html +1 -1
  232. edsl/templates/error_reporting/report.css +73 -73
  233. edsl/templates/error_reporting/report.html +117 -117
  234. edsl/templates/error_reporting/report.js +25 -25
  235. edsl/tools/__init__.py +1 -1
  236. edsl/tools/clusters.py +192 -192
  237. edsl/tools/embeddings.py +27 -27
  238. edsl/tools/embeddings_plotting.py +118 -118
  239. edsl/tools/plotting.py +112 -112
  240. edsl/tools/summarize.py +18 -18
  241. edsl/utilities/SystemInfo.py +28 -28
  242. edsl/utilities/__init__.py +22 -22
  243. edsl/utilities/ast_utilities.py +25 -25
  244. edsl/utilities/data/Registry.py +6 -6
  245. edsl/utilities/data/__init__.py +1 -1
  246. edsl/utilities/data/scooter_results.json +1 -1
  247. edsl/utilities/decorators.py +77 -77
  248. edsl/utilities/gcp_bucket/cloud_storage.py +96 -96
  249. edsl/utilities/interface.py +627 -627
  250. edsl/utilities/repair_functions.py +28 -28
  251. edsl/utilities/restricted_python.py +70 -70
  252. edsl/utilities/utilities.py +391 -391
  253. {edsl-0.1.36.dev2.dist-info → edsl-0.1.36.dev6.dist-info}/LICENSE +21 -21
  254. {edsl-0.1.36.dev2.dist-info → edsl-0.1.36.dev6.dist-info}/METADATA +1 -1
  255. edsl-0.1.36.dev6.dist-info/RECORD +279 -0
  256. edsl-0.1.36.dev2.dist-info/RECORD +0 -278
  257. {edsl-0.1.36.dev2.dist-info → edsl-0.1.36.dev6.dist-info}/WHEEL +0 -0
@@ -1,248 +1,248 @@
1
- from typing import Union, List, Any, Optional
2
- import asyncio
3
- import time
4
-
5
-
6
- class TokenBucket:
7
- """This is a token bucket used to respect rate limits to services."""
8
-
9
- def __init__(
10
- self,
11
- *,
12
- bucket_name: str,
13
- bucket_type: str,
14
- capacity: Union[int, float],
15
- refill_rate: Union[int, float],
16
- ):
17
- self.bucket_name = bucket_name
18
- self.bucket_type = bucket_type
19
- self.capacity = capacity # Maximum number of tokens
20
- self.added_tokens = 0
21
-
22
- self.target_rate = (
23
- capacity * 60
24
- ) # set this here because it can change with turbo mode
25
-
26
- self._old_capacity = capacity
27
- self.tokens = capacity # Current number of available tokens
28
- self.refill_rate = refill_rate # Rate at which tokens are refilled
29
- self._old_refill_rate = refill_rate
30
- self.last_refill = time.monotonic() # Last refill time
31
- self.log: List[Any] = []
32
- self.turbo_mode = False
33
-
34
- self.creation_time = time.monotonic()
35
-
36
- self.num_requests = 0
37
- self.num_released = 0
38
- self.tokens_returned = 0
39
-
40
- def turbo_mode_on(self):
41
- """Set the refill rate to infinity."""
42
- if self.turbo_mode:
43
- pass
44
- else:
45
- # pass
46
- self.turbo_mode = True
47
- self.capacity = float("inf")
48
- self.refill_rate = float("inf")
49
-
50
- def turbo_mode_off(self):
51
- """Restore the refill rate to its original value."""
52
- self.turbo_mode = False
53
- self.capacity = self._old_capacity
54
- self.refill_rate = self._old_refill_rate
55
-
56
- def __add__(self, other) -> "TokenBucket":
57
- """Combine two token buckets.
58
-
59
- The resulting bucket has the minimum capacity and refill rate of the two buckets.
60
- This is useful, for example, if we have two calls to the same model on the same service but have different temperatures.
61
- """
62
- return TokenBucket(
63
- bucket_name=self.bucket_name,
64
- bucket_type=self.bucket_type,
65
- capacity=min(self.capacity, other.capacity),
66
- refill_rate=min(self.refill_rate, other.refill_rate),
67
- )
68
-
69
- def __repr__(self):
70
- return f"TokenBucket(bucket_name={self.bucket_name}, bucket_type='{self.bucket_type}', capacity={self.capacity}, refill_rate={self.refill_rate})"
71
-
72
- def add_tokens(self, tokens: Union[int, float]) -> None:
73
- """Add tokens to the bucket, up to the maximum capacity.
74
-
75
- :param tokens: The number of tokens to add to the bucket.
76
-
77
- >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
78
- >>> bucket.tokens
79
- 10
80
- >>> bucket.add_tokens(5)
81
- >>> bucket.tokens
82
- 10
83
- """
84
- self.tokens_returned += tokens
85
- self.tokens = min(self.capacity, self.tokens + tokens)
86
- self.log.append((time.monotonic(), self.tokens))
87
-
88
- def refill(self) -> None:
89
- """Refill the bucket with new tokens based on elapsed time.
90
-
91
-
92
-
93
- >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
94
- >>> bucket.tokens = 0
95
- >>> bucket.refill()
96
- >>> bucket.tokens > 0
97
- True
98
- """
99
- """Refill the bucket with new tokens based on elapsed time."""
100
- now = time.monotonic()
101
- # print(f"Time is now: {now}; Last refill time: {self.last_refill}")
102
- elapsed = now - self.last_refill
103
- # print("Elapsed time: ", elapsed)
104
- refill_amount = elapsed * self.refill_rate
105
- self.tokens = min(self.capacity, self.tokens + refill_amount)
106
- self.last_refill = now
107
-
108
- if self.tokens < self.capacity:
109
- pass
110
- # print(f"Refilled. Current tokens: {self.tokens:.4f}")
111
- # print(f"Elapsed time: {elapsed:.4f} seconds")
112
- # print(f"Refill amount: {refill_amount:.4f}")
113
-
114
- self.log.append((now, self.tokens))
115
-
116
- def wait_time(self, requested_tokens: Union[float, int]) -> float:
117
- """Calculate the time to wait for the requested number of tokens."""
118
- # self.refill() # Update the current token count
119
- if self.tokens >= requested_tokens:
120
- return 0
121
- return (requested_tokens - self.tokens) / self.refill_rate
122
-
123
- async def get_tokens(
124
- self, amount: Union[int, float] = 1, cheat_bucket_capacity=True
125
- ) -> None:
126
- """Wait for the specified number of tokens to become available.
127
-
128
-
129
- :param amount: The number of tokens
130
- :param warn: If True, warn if the requested amount exceeds the bucket capacity.
131
-
132
- >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
133
- >>> asyncio.run(bucket.get_tokens(5))
134
- >>> bucket.tokens
135
- 5
136
- >>> asyncio.run(bucket.get_tokens(9))
137
- >>> bucket.tokens < 1
138
- True
139
-
140
- >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
141
- >>> asyncio.run(bucket.get_tokens(11, cheat_bucket_capacity=False))
142
- Traceback (most recent call last):
143
- ...
144
- ValueError: Requested amount exceeds bucket capacity. Bucket capacity: 10, requested amount: 11. As the bucket never overflows, the requested amount will never be available.
145
- >>> asyncio.run(bucket.get_tokens(11, cheat_bucket_capacity=True))
146
- >>> bucket.capacity
147
- 12.100000000000001
148
- """
149
- self.num_requests += amount
150
- if amount >= self.capacity:
151
- if not cheat_bucket_capacity:
152
- msg = f"Requested amount exceeds bucket capacity. Bucket capacity: {self.capacity}, requested amount: {amount}. As the bucket never overflows, the requested amount will never be available."
153
- raise ValueError(msg)
154
- else:
155
- self.capacity = amount * 1.10
156
- self._old_capacity = self.capacity
157
-
158
- start_time = time.monotonic()
159
- while True:
160
- self.refill() # Refill based on elapsed time
161
- if self.tokens >= amount:
162
- self.tokens -= amount
163
- break
164
-
165
- wait_time = self.wait_time(amount)
166
- if wait_time > 0:
167
- await asyncio.sleep(wait_time)
168
-
169
- self.num_released += amount
170
- now = time.monotonic()
171
- self.log.append((now, self.tokens))
172
- return None
173
-
174
- def get_log(self) -> list[tuple]:
175
- return self.log
176
-
177
- def visualize(self):
178
- """Visualize the token bucket over time."""
179
- times, tokens = zip(*self.get_log())
180
- start_time = times[0]
181
- times = [t - start_time for t in times] # Normalize time to start from 0
182
- from matplotlib import pyplot as plt
183
-
184
- plt.figure(figsize=(10, 6))
185
- plt.plot(times, tokens, label="Tokens Available")
186
- plt.xlabel("Time (seconds)", fontsize=12)
187
- plt.ylabel("Number of Tokens", fontsize=12)
188
- details = f"{self.bucket_name} ({self.bucket_type}) Bucket Usage Over Time\nCapacity: {self.capacity:.1f}, Refill Rate: {self.refill_rate:.1f}/second"
189
- plt.title(details, fontsize=14)
190
-
191
- plt.legend()
192
- plt.grid(True)
193
- plt.tight_layout()
194
- plt.show()
195
-
196
- def get_throughput(self, time_window: Optional[float] = None) -> float:
197
- """
198
- Calculate the empirical bucket throughput in tokens per minute for the specified time window.
199
-
200
- :param time_window: The time window in seconds to calculate the throughput for.
201
- :return: The throughput in tokens per minute.
202
-
203
- >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=100, refill_rate=10)
204
- >>> asyncio.run(bucket.get_tokens(50))
205
- >>> time.sleep(1) # Wait for 1 second
206
- >>> asyncio.run(bucket.get_tokens(30))
207
- >>> throughput = bucket.get_throughput(1)
208
- >>> 4750 < throughput < 4850
209
- True
210
- """
211
- now = time.monotonic()
212
-
213
- if time_window is None:
214
- start_time = self.creation_time
215
- else:
216
- start_time = now - time_window
217
-
218
- if start_time < self.creation_time:
219
- start_time = self.creation_time
220
-
221
- elapsed_time = now - start_time
222
-
223
- return (self.num_released / elapsed_time) * 60
224
-
225
- # # Filter log entries within the time window
226
- # relevant_log = [(t, tokens) for t, tokens in self.log if t >= start_time]
227
-
228
- # if len(relevant_log) < 2:
229
- # return 0 # Not enough data points to calculate throughput
230
-
231
- # # Calculate total tokens used
232
- # initial_tokens = relevant_log[0][1]
233
- # final_tokens = relevant_log[-1][1]
234
- # tokens_used = self.num_released - (final_tokens - initial_tokens)
235
-
236
- # # Calculate actual time elapsed
237
- # actual_time_elapsed = relevant_log[-1][0] - relevant_log[0][0]
238
-
239
- # # Calculate throughput in tokens per minute
240
- # throughput = (tokens_used / actual_time_elapsed) * 60
241
-
242
- # return throughput
243
-
244
-
245
- if __name__ == "__main__":
246
- import doctest
247
-
248
- doctest.testmod(optionflags=doctest.ELLIPSIS)
1
+ from typing import Union, List, Any, Optional
2
+ import asyncio
3
+ import time
4
+
5
+
6
+ class TokenBucket:
7
+ """This is a token bucket used to respect rate limits to services."""
8
+
9
+ def __init__(
10
+ self,
11
+ *,
12
+ bucket_name: str,
13
+ bucket_type: str,
14
+ capacity: Union[int, float],
15
+ refill_rate: Union[int, float],
16
+ ):
17
+ self.bucket_name = bucket_name
18
+ self.bucket_type = bucket_type
19
+ self.capacity = capacity # Maximum number of tokens
20
+ self.added_tokens = 0
21
+
22
+ self.target_rate = (
23
+ capacity * 60
24
+ ) # set this here because it can change with turbo mode
25
+
26
+ self._old_capacity = capacity
27
+ self.tokens = capacity # Current number of available tokens
28
+ self.refill_rate = refill_rate # Rate at which tokens are refilled
29
+ self._old_refill_rate = refill_rate
30
+ self.last_refill = time.monotonic() # Last refill time
31
+ self.log: List[Any] = []
32
+ self.turbo_mode = False
33
+
34
+ self.creation_time = time.monotonic()
35
+
36
+ self.num_requests = 0
37
+ self.num_released = 0
38
+ self.tokens_returned = 0
39
+
40
+ def turbo_mode_on(self):
41
+ """Set the refill rate to infinity."""
42
+ if self.turbo_mode:
43
+ pass
44
+ else:
45
+ # pass
46
+ self.turbo_mode = True
47
+ self.capacity = float("inf")
48
+ self.refill_rate = float("inf")
49
+
50
+ def turbo_mode_off(self):
51
+ """Restore the refill rate to its original value."""
52
+ self.turbo_mode = False
53
+ self.capacity = self._old_capacity
54
+ self.refill_rate = self._old_refill_rate
55
+
56
+ def __add__(self, other) -> "TokenBucket":
57
+ """Combine two token buckets.
58
+
59
+ The resulting bucket has the minimum capacity and refill rate of the two buckets.
60
+ This is useful, for example, if we have two calls to the same model on the same service but have different temperatures.
61
+ """
62
+ return TokenBucket(
63
+ bucket_name=self.bucket_name,
64
+ bucket_type=self.bucket_type,
65
+ capacity=min(self.capacity, other.capacity),
66
+ refill_rate=min(self.refill_rate, other.refill_rate),
67
+ )
68
+
69
+ def __repr__(self):
70
+ return f"TokenBucket(bucket_name={self.bucket_name}, bucket_type='{self.bucket_type}', capacity={self.capacity}, refill_rate={self.refill_rate})"
71
+
72
+ def add_tokens(self, tokens: Union[int, float]) -> None:
73
+ """Add tokens to the bucket, up to the maximum capacity.
74
+
75
+ :param tokens: The number of tokens to add to the bucket.
76
+
77
+ >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
78
+ >>> bucket.tokens
79
+ 10
80
+ >>> bucket.add_tokens(5)
81
+ >>> bucket.tokens
82
+ 10
83
+ """
84
+ self.tokens_returned += tokens
85
+ self.tokens = min(self.capacity, self.tokens + tokens)
86
+ self.log.append((time.monotonic(), self.tokens))
87
+
88
+ def refill(self) -> None:
89
+ """Refill the bucket with new tokens based on elapsed time.
90
+
91
+
92
+
93
+ >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
94
+ >>> bucket.tokens = 0
95
+ >>> bucket.refill()
96
+ >>> bucket.tokens > 0
97
+ True
98
+ """
99
+ """Refill the bucket with new tokens based on elapsed time."""
100
+ now = time.monotonic()
101
+ # print(f"Time is now: {now}; Last refill time: {self.last_refill}")
102
+ elapsed = now - self.last_refill
103
+ # print("Elapsed time: ", elapsed)
104
+ refill_amount = elapsed * self.refill_rate
105
+ self.tokens = min(self.capacity, self.tokens + refill_amount)
106
+ self.last_refill = now
107
+
108
+ if self.tokens < self.capacity:
109
+ pass
110
+ # print(f"Refilled. Current tokens: {self.tokens:.4f}")
111
+ # print(f"Elapsed time: {elapsed:.4f} seconds")
112
+ # print(f"Refill amount: {refill_amount:.4f}")
113
+
114
+ self.log.append((now, self.tokens))
115
+
116
+ def wait_time(self, requested_tokens: Union[float, int]) -> float:
117
+ """Calculate the time to wait for the requested number of tokens."""
118
+ # self.refill() # Update the current token count
119
+ if self.tokens >= requested_tokens:
120
+ return 0
121
+ return (requested_tokens - self.tokens) / self.refill_rate
122
+
123
+ async def get_tokens(
124
+ self, amount: Union[int, float] = 1, cheat_bucket_capacity=True
125
+ ) -> None:
126
+ """Wait for the specified number of tokens to become available.
127
+
128
+
129
+ :param amount: The number of tokens
130
+ :param warn: If True, warn if the requested amount exceeds the bucket capacity.
131
+
132
+ >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
133
+ >>> asyncio.run(bucket.get_tokens(5))
134
+ >>> bucket.tokens
135
+ 5
136
+ >>> asyncio.run(bucket.get_tokens(9))
137
+ >>> bucket.tokens < 1
138
+ True
139
+
140
+ >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
141
+ >>> asyncio.run(bucket.get_tokens(11, cheat_bucket_capacity=False))
142
+ Traceback (most recent call last):
143
+ ...
144
+ ValueError: Requested amount exceeds bucket capacity. Bucket capacity: 10, requested amount: 11. As the bucket never overflows, the requested amount will never be available.
145
+ >>> asyncio.run(bucket.get_tokens(11, cheat_bucket_capacity=True))
146
+ >>> bucket.capacity
147
+ 12.100000000000001
148
+ """
149
+ self.num_requests += amount
150
+ if amount >= self.capacity:
151
+ if not cheat_bucket_capacity:
152
+ msg = f"Requested amount exceeds bucket capacity. Bucket capacity: {self.capacity}, requested amount: {amount}. As the bucket never overflows, the requested amount will never be available."
153
+ raise ValueError(msg)
154
+ else:
155
+ self.capacity = amount * 1.10
156
+ self._old_capacity = self.capacity
157
+
158
+ start_time = time.monotonic()
159
+ while True:
160
+ self.refill() # Refill based on elapsed time
161
+ if self.tokens >= amount:
162
+ self.tokens -= amount
163
+ break
164
+
165
+ wait_time = self.wait_time(amount)
166
+ if wait_time > 0:
167
+ await asyncio.sleep(wait_time)
168
+
169
+ self.num_released += amount
170
+ now = time.monotonic()
171
+ self.log.append((now, self.tokens))
172
+ return None
173
+
174
+ def get_log(self) -> list[tuple]:
175
+ return self.log
176
+
177
+ def visualize(self):
178
+ """Visualize the token bucket over time."""
179
+ times, tokens = zip(*self.get_log())
180
+ start_time = times[0]
181
+ times = [t - start_time for t in times] # Normalize time to start from 0
182
+ from matplotlib import pyplot as plt
183
+
184
+ plt.figure(figsize=(10, 6))
185
+ plt.plot(times, tokens, label="Tokens Available")
186
+ plt.xlabel("Time (seconds)", fontsize=12)
187
+ plt.ylabel("Number of Tokens", fontsize=12)
188
+ details = f"{self.bucket_name} ({self.bucket_type}) Bucket Usage Over Time\nCapacity: {self.capacity:.1f}, Refill Rate: {self.refill_rate:.1f}/second"
189
+ plt.title(details, fontsize=14)
190
+
191
+ plt.legend()
192
+ plt.grid(True)
193
+ plt.tight_layout()
194
+ plt.show()
195
+
196
+ def get_throughput(self, time_window: Optional[float] = None) -> float:
197
+ """
198
+ Calculate the empirical bucket throughput in tokens per minute for the specified time window.
199
+
200
+ :param time_window: The time window in seconds to calculate the throughput for.
201
+ :return: The throughput in tokens per minute.
202
+
203
+ >>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=100, refill_rate=10)
204
+ >>> asyncio.run(bucket.get_tokens(50))
205
+ >>> time.sleep(1) # Wait for 1 second
206
+ >>> asyncio.run(bucket.get_tokens(30))
207
+ >>> throughput = bucket.get_throughput(1)
208
+ >>> 4750 < throughput < 4850
209
+ True
210
+ """
211
+ now = time.monotonic()
212
+
213
+ if time_window is None:
214
+ start_time = self.creation_time
215
+ else:
216
+ start_time = now - time_window
217
+
218
+ if start_time < self.creation_time:
219
+ start_time = self.creation_time
220
+
221
+ elapsed_time = now - start_time
222
+
223
+ return (self.num_released / elapsed_time) * 60
224
+
225
+ # # Filter log entries within the time window
226
+ # relevant_log = [(t, tokens) for t, tokens in self.log if t >= start_time]
227
+
228
+ # if len(relevant_log) < 2:
229
+ # return 0 # Not enough data points to calculate throughput
230
+
231
+ # # Calculate total tokens used
232
+ # initial_tokens = relevant_log[0][1]
233
+ # final_tokens = relevant_log[-1][1]
234
+ # tokens_used = self.num_released - (final_tokens - initial_tokens)
235
+
236
+ # # Calculate actual time elapsed
237
+ # actual_time_elapsed = relevant_log[-1][0] - relevant_log[0][0]
238
+
239
+ # # Calculate throughput in tokens per minute
240
+ # throughput = (tokens_used / actual_time_elapsed) * 60
241
+
242
+ # return throughput
243
+
244
+
245
+ if __name__ == "__main__":
246
+ import doctest
247
+
248
+ doctest.testmod(optionflags=doctest.ELLIPSIS)