edsl 0.1.46__py3-none-any.whl → 0.1.48__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- edsl/__init__.py +44 -39
- edsl/__version__.py +1 -1
- edsl/agents/__init__.py +4 -2
- edsl/agents/{Agent.py → agent.py} +442 -152
- edsl/agents/{AgentList.py → agent_list.py} +220 -162
- edsl/agents/descriptors.py +46 -7
- edsl/{exceptions/agents.py → agents/exceptions.py} +3 -12
- edsl/base/__init__.py +75 -0
- edsl/base/base_class.py +1303 -0
- edsl/base/data_transfer_models.py +114 -0
- edsl/base/enums.py +215 -0
- edsl/base.py +8 -0
- edsl/buckets/__init__.py +25 -0
- edsl/buckets/bucket_collection.py +324 -0
- edsl/buckets/model_buckets.py +206 -0
- edsl/buckets/token_bucket.py +502 -0
- edsl/{jobs/buckets/TokenBucketAPI.py → buckets/token_bucket_api.py} +1 -1
- edsl/buckets/token_bucket_client.py +509 -0
- edsl/caching/__init__.py +20 -0
- edsl/caching/cache.py +814 -0
- edsl/caching/cache_entry.py +427 -0
- edsl/{data/CacheHandler.py → caching/cache_handler.py} +14 -15
- edsl/caching/exceptions.py +24 -0
- edsl/caching/orm.py +30 -0
- edsl/{data/RemoteCacheSync.py → caching/remote_cache_sync.py} +3 -3
- edsl/caching/sql_dict.py +441 -0
- edsl/config/__init__.py +8 -0
- edsl/config/config_class.py +177 -0
- edsl/config.py +4 -176
- edsl/conversation/Conversation.py +7 -7
- edsl/conversation/car_buying.py +4 -4
- edsl/conversation/chips.py +6 -6
- edsl/coop/__init__.py +25 -2
- edsl/coop/coop.py +430 -113
- edsl/coop/{ExpectedParrotKeyHandler.py → ep_key_handling.py} +86 -10
- edsl/coop/exceptions.py +62 -0
- edsl/coop/price_fetcher.py +126 -0
- edsl/coop/utils.py +89 -24
- edsl/data_transfer_models.py +5 -72
- edsl/dataset/__init__.py +10 -0
- edsl/{results/Dataset.py → dataset/dataset.py} +116 -36
- edsl/dataset/dataset_operations_mixin.py +1492 -0
- edsl/{results/DatasetTree.py → dataset/dataset_tree.py} +156 -75
- edsl/{results/TableDisplay.py → dataset/display/table_display.py} +18 -7
- edsl/{results → dataset/display}/table_renderers.py +58 -2
- edsl/{results → dataset}/file_exports.py +4 -5
- edsl/{results → dataset}/smart_objects.py +2 -2
- edsl/enums.py +5 -205
- edsl/inference_services/__init__.py +5 -0
- edsl/inference_services/{AvailableModelCacheHandler.py → available_model_cache_handler.py} +2 -3
- edsl/inference_services/{AvailableModelFetcher.py → available_model_fetcher.py} +8 -14
- edsl/inference_services/data_structures.py +3 -2
- edsl/{exceptions/inference_services.py → inference_services/exceptions.py} +1 -1
- edsl/inference_services/{InferenceServiceABC.py → inference_service_abc.py} +1 -1
- edsl/inference_services/{InferenceServicesCollection.py → inference_services_collection.py} +8 -7
- edsl/inference_services/registry.py +4 -41
- edsl/inference_services/{ServiceAvailability.py → service_availability.py} +5 -25
- edsl/inference_services/services/__init__.py +31 -0
- edsl/inference_services/{AnthropicService.py → services/anthropic_service.py} +3 -3
- edsl/inference_services/{AwsBedrock.py → services/aws_bedrock.py} +2 -2
- edsl/inference_services/{AzureAI.py → services/azure_ai.py} +2 -2
- edsl/inference_services/{DeepInfraService.py → services/deep_infra_service.py} +1 -3
- edsl/inference_services/{DeepSeekService.py → services/deep_seek_service.py} +2 -4
- edsl/inference_services/{GoogleService.py → services/google_service.py} +5 -4
- edsl/inference_services/{GroqService.py → services/groq_service.py} +1 -1
- edsl/inference_services/{MistralAIService.py → services/mistral_ai_service.py} +3 -3
- edsl/inference_services/{OllamaService.py → services/ollama_service.py} +1 -7
- edsl/inference_services/{OpenAIService.py → services/open_ai_service.py} +5 -6
- edsl/inference_services/{PerplexityService.py → services/perplexity_service.py} +12 -12
- edsl/inference_services/{TestService.py → services/test_service.py} +7 -6
- edsl/inference_services/{TogetherAIService.py → services/together_ai_service.py} +2 -6
- edsl/inference_services/{XAIService.py → services/xai_service.py} +1 -1
- edsl/inference_services/write_available.py +1 -2
- edsl/instructions/__init__.py +6 -0
- edsl/{surveys/instructions/Instruction.py → instructions/instruction.py} +11 -6
- edsl/{surveys/instructions/InstructionCollection.py → instructions/instruction_collection.py} +10 -5
- edsl/{surveys/InstructionHandler.py → instructions/instruction_handler.py} +3 -3
- edsl/{jobs/interviews → interviews}/ReportErrors.py +2 -2
- edsl/interviews/__init__.py +4 -0
- edsl/{jobs/AnswerQuestionFunctionConstructor.py → interviews/answering_function.py} +45 -18
- edsl/{jobs/interviews/InterviewExceptionEntry.py → interviews/exception_tracking.py} +107 -22
- edsl/interviews/interview.py +638 -0
- edsl/{jobs/interviews/InterviewStatusDictionary.py → interviews/interview_status_dictionary.py} +21 -12
- edsl/{jobs/interviews/InterviewStatusLog.py → interviews/interview_status_log.py} +16 -7
- edsl/{jobs/InterviewTaskManager.py → interviews/interview_task_manager.py} +12 -7
- edsl/{jobs/RequestTokenEstimator.py → interviews/request_token_estimator.py} +8 -3
- edsl/{jobs/interviews/InterviewStatistic.py → interviews/statistics.py} +36 -10
- edsl/invigilators/__init__.py +38 -0
- edsl/invigilators/invigilator_base.py +477 -0
- edsl/{agents/Invigilator.py → invigilators/invigilators.py} +263 -10
- edsl/invigilators/prompt_constructor.py +476 -0
- edsl/{agents → invigilators}/prompt_helpers.py +2 -1
- edsl/{agents/QuestionInstructionPromptBuilder.py → invigilators/question_instructions_prompt_builder.py} +18 -13
- edsl/{agents → invigilators}/question_option_processor.py +96 -21
- edsl/{agents/QuestionTemplateReplacementsBuilder.py → invigilators/question_template_replacements_builder.py} +64 -12
- edsl/jobs/__init__.py +7 -1
- edsl/jobs/async_interview_runner.py +99 -35
- edsl/jobs/check_survey_scenario_compatibility.py +7 -5
- edsl/jobs/data_structures.py +153 -22
- edsl/{exceptions/jobs.py → jobs/exceptions.py} +2 -1
- edsl/jobs/{FetchInvigilator.py → fetch_invigilator.py} +4 -4
- edsl/jobs/{loggers/HTMLTableJobLogger.py → html_table_job_logger.py} +6 -2
- edsl/jobs/{Jobs.py → jobs.py} +321 -155
- edsl/jobs/{JobsChecks.py → jobs_checks.py} +15 -7
- edsl/jobs/{JobsComponentConstructor.py → jobs_component_constructor.py} +20 -17
- edsl/jobs/{InterviewsConstructor.py → jobs_interview_constructor.py} +10 -5
- edsl/jobs/jobs_pricing_estimation.py +347 -0
- edsl/jobs/{JobsRemoteInferenceLogger.py → jobs_remote_inference_logger.py} +4 -3
- edsl/jobs/jobs_runner_asyncio.py +282 -0
- edsl/jobs/{JobsRemoteInferenceHandler.py → remote_inference.py} +19 -22
- edsl/jobs/results_exceptions_handler.py +2 -2
- edsl/key_management/__init__.py +28 -0
- edsl/key_management/key_lookup.py +161 -0
- edsl/{language_models/key_management/KeyLookupBuilder.py → key_management/key_lookup_builder.py} +118 -47
- edsl/key_management/key_lookup_collection.py +82 -0
- edsl/key_management/models.py +218 -0
- edsl/language_models/__init__.py +7 -2
- edsl/language_models/{ComputeCost.py → compute_cost.py} +18 -3
- edsl/{exceptions/language_models.py → language_models/exceptions.py} +2 -1
- edsl/language_models/language_model.py +1080 -0
- edsl/language_models/model.py +10 -25
- edsl/language_models/{ModelList.py → model_list.py} +9 -14
- edsl/language_models/{RawResponseHandler.py → raw_response_handler.py} +1 -1
- edsl/language_models/{RegisterLanguageModelsMeta.py → registry.py} +1 -1
- edsl/language_models/repair.py +4 -4
- edsl/language_models/utilities.py +4 -4
- edsl/notebooks/__init__.py +3 -1
- edsl/notebooks/{Notebook.py → notebook.py} +7 -8
- edsl/prompts/__init__.py +1 -1
- edsl/{exceptions/prompts.py → prompts/exceptions.py} +3 -1
- edsl/prompts/{Prompt.py → prompt.py} +101 -95
- edsl/questions/HTMLQuestion.py +1 -1
- edsl/questions/__init__.py +154 -25
- edsl/questions/answer_validator_mixin.py +1 -1
- edsl/questions/compose_questions.py +4 -3
- edsl/questions/derived/question_likert_five.py +166 -0
- edsl/questions/derived/{QuestionLinearScale.py → question_linear_scale.py} +4 -4
- edsl/questions/derived/{QuestionTopK.py → question_top_k.py} +4 -4
- edsl/questions/derived/{QuestionYesNo.py → question_yes_no.py} +4 -5
- edsl/questions/descriptors.py +24 -30
- edsl/questions/loop_processor.py +65 -19
- edsl/questions/question_base.py +881 -0
- edsl/questions/question_base_gen_mixin.py +15 -16
- edsl/questions/{QuestionBasePromptsMixin.py → question_base_prompts_mixin.py} +2 -2
- edsl/questions/{QuestionBudget.py → question_budget.py} +3 -4
- edsl/questions/{QuestionCheckBox.py → question_check_box.py} +16 -16
- edsl/questions/{QuestionDict.py → question_dict.py} +39 -5
- edsl/questions/{QuestionExtract.py → question_extract.py} +9 -9
- edsl/questions/question_free_text.py +282 -0
- edsl/questions/{QuestionFunctional.py → question_functional.py} +6 -5
- edsl/questions/{QuestionList.py → question_list.py} +6 -7
- edsl/questions/{QuestionMatrix.py → question_matrix.py} +6 -5
- edsl/questions/{QuestionMultipleChoice.py → question_multiple_choice.py} +126 -21
- edsl/questions/{QuestionNumerical.py → question_numerical.py} +5 -5
- edsl/questions/{QuestionRank.py → question_rank.py} +6 -6
- edsl/questions/question_registry.py +10 -16
- edsl/questions/register_questions_meta.py +8 -4
- edsl/questions/response_validator_abc.py +17 -16
- edsl/results/__init__.py +4 -1
- edsl/{exceptions/results.py → results/exceptions.py} +1 -1
- edsl/results/report.py +197 -0
- edsl/results/{Result.py → result.py} +131 -45
- edsl/results/{Results.py → results.py} +420 -216
- edsl/results/results_selector.py +344 -25
- edsl/scenarios/__init__.py +30 -3
- edsl/scenarios/{ConstructDownloadLink.py → construct_download_link.py} +7 -0
- edsl/scenarios/directory_scanner.py +156 -13
- edsl/scenarios/document_chunker.py +186 -0
- edsl/scenarios/exceptions.py +101 -0
- edsl/scenarios/file_methods.py +2 -3
- edsl/scenarios/file_store.py +755 -0
- edsl/scenarios/handlers/__init__.py +14 -14
- edsl/scenarios/handlers/{csv.py → csv_file_store.py} +1 -2
- edsl/scenarios/handlers/{docx.py → docx_file_store.py} +8 -7
- edsl/scenarios/handlers/{html.py → html_file_store.py} +1 -2
- edsl/scenarios/handlers/{jpeg.py → jpeg_file_store.py} +1 -1
- edsl/scenarios/handlers/{json.py → json_file_store.py} +1 -1
- edsl/scenarios/handlers/latex_file_store.py +5 -0
- edsl/scenarios/handlers/{md.py → md_file_store.py} +1 -1
- edsl/scenarios/handlers/{pdf.py → pdf_file_store.py} +2 -2
- edsl/scenarios/handlers/{png.py → png_file_store.py} +1 -1
- edsl/scenarios/handlers/{pptx.py → pptx_file_store.py} +8 -7
- edsl/scenarios/handlers/{py.py → py_file_store.py} +1 -3
- edsl/scenarios/handlers/{sql.py → sql_file_store.py} +2 -1
- edsl/scenarios/handlers/{sqlite.py → sqlite_file_store.py} +2 -3
- edsl/scenarios/handlers/{txt.py → txt_file_store.py} +1 -1
- edsl/scenarios/scenario.py +928 -0
- edsl/scenarios/scenario_join.py +18 -5
- edsl/scenarios/{ScenarioList.py → scenario_list.py} +424 -106
- edsl/scenarios/{ScenarioListPdfMixin.py → scenario_list_pdf_tools.py} +16 -15
- edsl/scenarios/scenario_selector.py +5 -1
- edsl/study/ObjectEntry.py +2 -2
- edsl/study/SnapShot.py +5 -5
- edsl/study/Study.py +20 -21
- edsl/study/__init__.py +6 -4
- edsl/surveys/__init__.py +7 -4
- edsl/surveys/dag/__init__.py +2 -0
- edsl/surveys/{ConstructDAG.py → dag/construct_dag.py} +3 -3
- edsl/surveys/{DAG.py → dag/dag.py} +13 -10
- edsl/surveys/descriptors.py +1 -1
- edsl/surveys/{EditSurvey.py → edit_survey.py} +9 -9
- edsl/{exceptions/surveys.py → surveys/exceptions.py} +1 -2
- edsl/surveys/memory/__init__.py +3 -0
- edsl/surveys/{MemoryPlan.py → memory/memory_plan.py} +10 -9
- edsl/surveys/rules/__init__.py +3 -0
- edsl/surveys/{Rule.py → rules/rule.py} +103 -43
- edsl/surveys/{RuleCollection.py → rules/rule_collection.py} +21 -30
- edsl/surveys/{RuleManager.py → rules/rule_manager.py} +19 -13
- edsl/surveys/survey.py +1743 -0
- edsl/surveys/{SurveyExportMixin.py → survey_export.py} +22 -27
- edsl/surveys/{SurveyFlowVisualization.py → survey_flow_visualization.py} +11 -2
- edsl/surveys/{Simulator.py → survey_simulator.py} +10 -3
- edsl/tasks/__init__.py +32 -0
- edsl/{jobs/tasks/QuestionTaskCreator.py → tasks/question_task_creator.py} +115 -57
- edsl/tasks/task_creators.py +135 -0
- edsl/{jobs/tasks/TaskHistory.py → tasks/task_history.py} +86 -47
- edsl/{jobs/tasks → tasks}/task_status_enum.py +91 -7
- edsl/tasks/task_status_log.py +85 -0
- edsl/tokens/__init__.py +2 -0
- edsl/tokens/interview_token_usage.py +53 -0
- edsl/utilities/PrettyList.py +1 -1
- edsl/utilities/SystemInfo.py +25 -22
- edsl/utilities/__init__.py +29 -21
- edsl/utilities/gcp_bucket/__init__.py +2 -0
- edsl/utilities/gcp_bucket/cloud_storage.py +99 -96
- edsl/utilities/interface.py +44 -536
- edsl/{results/MarkdownToPDF.py → utilities/markdown_to_pdf.py} +13 -5
- edsl/utilities/repair_functions.py +1 -1
- {edsl-0.1.46.dist-info → edsl-0.1.48.dist-info}/METADATA +3 -2
- edsl-0.1.48.dist-info/RECORD +347 -0
- edsl/Base.py +0 -426
- edsl/BaseDiff.py +0 -260
- edsl/agents/InvigilatorBase.py +0 -260
- edsl/agents/PromptConstructor.py +0 -318
- edsl/auto/AutoStudy.py +0 -130
- edsl/auto/StageBase.py +0 -243
- edsl/auto/StageGenerateSurvey.py +0 -178
- edsl/auto/StageLabelQuestions.py +0 -125
- edsl/auto/StagePersona.py +0 -61
- edsl/auto/StagePersonaDimensionValueRanges.py +0 -88
- edsl/auto/StagePersonaDimensionValues.py +0 -74
- edsl/auto/StagePersonaDimensions.py +0 -69
- edsl/auto/StageQuestions.py +0 -74
- edsl/auto/SurveyCreatorPipeline.py +0 -21
- edsl/auto/utilities.py +0 -218
- edsl/base/Base.py +0 -279
- edsl/coop/PriceFetcher.py +0 -54
- edsl/data/Cache.py +0 -580
- edsl/data/CacheEntry.py +0 -230
- edsl/data/SQLiteDict.py +0 -292
- edsl/data/__init__.py +0 -5
- edsl/data/orm.py +0 -10
- edsl/exceptions/cache.py +0 -5
- edsl/exceptions/coop.py +0 -14
- edsl/exceptions/data.py +0 -14
- edsl/exceptions/scenarios.py +0 -29
- edsl/jobs/Answers.py +0 -43
- edsl/jobs/JobsPrompts.py +0 -354
- edsl/jobs/buckets/BucketCollection.py +0 -134
- edsl/jobs/buckets/ModelBuckets.py +0 -65
- edsl/jobs/buckets/TokenBucket.py +0 -283
- edsl/jobs/buckets/TokenBucketClient.py +0 -191
- edsl/jobs/interviews/Interview.py +0 -395
- edsl/jobs/interviews/InterviewExceptionCollection.py +0 -99
- edsl/jobs/interviews/InterviewStatisticsCollection.py +0 -25
- edsl/jobs/runners/JobsRunnerAsyncio.py +0 -163
- edsl/jobs/runners/JobsRunnerStatusData.py +0 -0
- edsl/jobs/tasks/TaskCreators.py +0 -64
- edsl/jobs/tasks/TaskStatusLog.py +0 -23
- edsl/jobs/tokens/InterviewTokenUsage.py +0 -27
- edsl/language_models/LanguageModel.py +0 -635
- edsl/language_models/ServiceDataSources.py +0 -0
- edsl/language_models/key_management/KeyLookup.py +0 -63
- edsl/language_models/key_management/KeyLookupCollection.py +0 -38
- edsl/language_models/key_management/models.py +0 -137
- edsl/questions/QuestionBase.py +0 -539
- edsl/questions/QuestionFreeText.py +0 -130
- edsl/questions/derived/QuestionLikertFive.py +0 -76
- edsl/results/DatasetExportMixin.py +0 -911
- edsl/results/ResultsExportMixin.py +0 -45
- edsl/results/TextEditor.py +0 -50
- edsl/results/results_fetch_mixin.py +0 -33
- edsl/results/results_tools_mixin.py +0 -98
- edsl/scenarios/DocumentChunker.py +0 -104
- edsl/scenarios/FileStore.py +0 -564
- edsl/scenarios/Scenario.py +0 -548
- edsl/scenarios/ScenarioHtmlMixin.py +0 -65
- edsl/scenarios/ScenarioListExportMixin.py +0 -45
- edsl/scenarios/handlers/latex.py +0 -5
- edsl/shared.py +0 -1
- edsl/surveys/Survey.py +0 -1306
- edsl/surveys/SurveyQualtricsImport.py +0 -284
- edsl/surveys/SurveyToApp.py +0 -141
- edsl/surveys/instructions/__init__.py +0 -0
- edsl/tools/__init__.py +0 -1
- edsl/tools/clusters.py +0 -192
- edsl/tools/embeddings.py +0 -27
- edsl/tools/embeddings_plotting.py +0 -118
- edsl/tools/plotting.py +0 -112
- edsl/tools/summarize.py +0 -18
- edsl/utilities/data/Registry.py +0 -6
- edsl/utilities/data/__init__.py +0 -1
- edsl/utilities/data/scooter_results.json +0 -1
- edsl-0.1.46.dist-info/RECORD +0 -366
- /edsl/coop/{CoopFunctionsMixin.py → coop_functions.py} +0 -0
- /edsl/{results → dataset/display}/CSSParameterizer.py +0 -0
- /edsl/{language_models/key_management → dataset/display}/__init__.py +0 -0
- /edsl/{results → dataset/display}/table_data_class.py +0 -0
- /edsl/{results → dataset/display}/table_display.css +0 -0
- /edsl/{results/ResultsGGMixin.py → dataset/r/ggplot.py} +0 -0
- /edsl/{results → dataset}/tree_explore.py +0 -0
- /edsl/{surveys/instructions/ChangeInstruction.py → instructions/change_instruction.py} +0 -0
- /edsl/{jobs/interviews → interviews}/interview_status_enum.py +0 -0
- /edsl/jobs/{runners/JobsRunnerStatus.py → jobs_runner_status.py} +0 -0
- /edsl/language_models/{PriceManager.py → price_manager.py} +0 -0
- /edsl/language_models/{fake_openai_call.py → unused/fake_openai_call.py} +0 -0
- /edsl/language_models/{fake_openai_service.py → unused/fake_openai_service.py} +0 -0
- /edsl/notebooks/{NotebookToLaTeX.py → notebook_to_latex.py} +0 -0
- /edsl/{exceptions/questions.py → questions/exceptions.py} +0 -0
- /edsl/questions/{SimpleAskMixin.py → simple_ask_mixin.py} +0 -0
- /edsl/surveys/{Memory.py → memory/memory.py} +0 -0
- /edsl/surveys/{MemoryManagement.py → memory/memory_management.py} +0 -0
- /edsl/surveys/{SurveyCSS.py → survey_css.py} +0 -0
- /edsl/{jobs/tokens/TokenUsage.py → tokens/token_usage.py} +0 -0
- /edsl/{results/MarkdownToDocx.py → utilities/markdown_to_docx.py} +0 -0
- /edsl/{TemplateLoader.py → utilities/template_loader.py} +0 -0
- {edsl-0.1.46.dist-info → edsl-0.1.48.dist-info}/LICENSE +0 -0
- {edsl-0.1.46.dist-info → edsl-0.1.48.dist-info}/WHEEL +0 -0
edsl/jobs/buckets/TokenBucket.py
DELETED
@@ -1,283 +0,0 @@
|
|
1
|
-
from typing import Union, List, Any, Optional
|
2
|
-
import asyncio
|
3
|
-
import time
|
4
|
-
from threading import RLock
|
5
|
-
from edsl.jobs.decorators import synchronized_class
|
6
|
-
|
7
|
-
from typing import Union, List, Any, Optional
|
8
|
-
import asyncio
|
9
|
-
import time
|
10
|
-
from threading import RLock
|
11
|
-
from edsl.jobs.decorators import synchronized_class
|
12
|
-
|
13
|
-
|
14
|
-
@synchronized_class
|
15
|
-
class TokenBucket:
|
16
|
-
"""This is a token bucket used to respect rate limits to services.
|
17
|
-
It can operate either locally or remotely via a REST API based on initialization parameters.
|
18
|
-
"""
|
19
|
-
|
20
|
-
def __new__(
|
21
|
-
cls,
|
22
|
-
*,
|
23
|
-
bucket_name: str,
|
24
|
-
bucket_type: str,
|
25
|
-
capacity: Union[int, float],
|
26
|
-
refill_rate: Union[int, float],
|
27
|
-
remote_url: Optional[str] = None,
|
28
|
-
):
|
29
|
-
"""Factory method to create either a local or remote token bucket.
|
30
|
-
|
31
|
-
Args:
|
32
|
-
bucket_name: Name of the bucket
|
33
|
-
bucket_type: Type of the bucket
|
34
|
-
capacity: Maximum number of tokens
|
35
|
-
refill_rate: Rate at which tokens are refilled
|
36
|
-
remote_url: If provided, creates a remote token bucket client
|
37
|
-
"""
|
38
|
-
if remote_url is not None:
|
39
|
-
# Import here to avoid circular imports
|
40
|
-
from edsl.jobs.buckets.TokenBucketClient import TokenBucketClient
|
41
|
-
|
42
|
-
return TokenBucketClient(
|
43
|
-
bucket_name=bucket_name,
|
44
|
-
bucket_type=bucket_type,
|
45
|
-
capacity=capacity,
|
46
|
-
refill_rate=refill_rate,
|
47
|
-
api_base_url=remote_url,
|
48
|
-
)
|
49
|
-
|
50
|
-
# Create a local token bucket
|
51
|
-
instance = super(TokenBucket, cls).__new__(cls)
|
52
|
-
return instance
|
53
|
-
|
54
|
-
def __init__(
|
55
|
-
self,
|
56
|
-
*,
|
57
|
-
bucket_name: str,
|
58
|
-
bucket_type: str,
|
59
|
-
capacity: Union[int, float],
|
60
|
-
refill_rate: Union[int, float],
|
61
|
-
remote_url: Optional[str] = None,
|
62
|
-
):
|
63
|
-
# Skip initialization if this is a remote bucket
|
64
|
-
if remote_url is not None:
|
65
|
-
return
|
66
|
-
|
67
|
-
self.bucket_name = bucket_name
|
68
|
-
self.bucket_type = bucket_type
|
69
|
-
self.capacity = capacity
|
70
|
-
self.added_tokens = 0
|
71
|
-
self._lock = RLock()
|
72
|
-
|
73
|
-
self.target_rate = (
|
74
|
-
capacity * 60
|
75
|
-
) # set this here because it can change with turbo mode
|
76
|
-
|
77
|
-
self._old_capacity = capacity
|
78
|
-
self.tokens = capacity # Current number of available tokens
|
79
|
-
self.refill_rate = refill_rate # Rate at which tokens are refilled
|
80
|
-
self._old_refill_rate = refill_rate
|
81
|
-
self.last_refill = time.monotonic() # Last refill time
|
82
|
-
self.log: List[Any] = []
|
83
|
-
self.turbo_mode = False
|
84
|
-
|
85
|
-
self.creation_time = time.monotonic()
|
86
|
-
|
87
|
-
self.num_requests = 0
|
88
|
-
self.num_released = 0
|
89
|
-
self.tokens_returned = 0
|
90
|
-
|
91
|
-
def turbo_mode_on(self):
|
92
|
-
"""Set the refill rate to infinity."""
|
93
|
-
if self.turbo_mode:
|
94
|
-
pass
|
95
|
-
else:
|
96
|
-
# pass
|
97
|
-
self.turbo_mode = True
|
98
|
-
self.capacity = float("inf")
|
99
|
-
self.refill_rate = float("inf")
|
100
|
-
|
101
|
-
def turbo_mode_off(self):
|
102
|
-
"""Restore the refill rate to its original value."""
|
103
|
-
self.turbo_mode = False
|
104
|
-
self.capacity = self._old_capacity
|
105
|
-
self.refill_rate = self._old_refill_rate
|
106
|
-
|
107
|
-
def __add__(self, other) -> "TokenBucket":
|
108
|
-
"""Combine two token buckets.
|
109
|
-
|
110
|
-
The resulting bucket has the minimum capacity and refill rate of the two buckets.
|
111
|
-
This is useful, for example, if we have two calls to the same model on the same service but have different temperatures.
|
112
|
-
"""
|
113
|
-
return TokenBucket(
|
114
|
-
bucket_name=self.bucket_name,
|
115
|
-
bucket_type=self.bucket_type,
|
116
|
-
capacity=min(self.capacity, other.capacity),
|
117
|
-
refill_rate=min(self.refill_rate, other.refill_rate),
|
118
|
-
)
|
119
|
-
|
120
|
-
def __repr__(self):
|
121
|
-
return f"TokenBucket(bucket_name={self.bucket_name}, bucket_type='{self.bucket_type}', capacity={self.capacity}, refill_rate={self.refill_rate})"
|
122
|
-
|
123
|
-
def add_tokens(self, tokens: Union[int, float]) -> None:
|
124
|
-
"""Add tokens to the bucket, up to the maximum capacity.
|
125
|
-
|
126
|
-
:param tokens: The number of tokens to add to the bucket.
|
127
|
-
|
128
|
-
>>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
|
129
|
-
>>> bucket.tokens
|
130
|
-
10
|
131
|
-
>>> bucket.add_tokens(5)
|
132
|
-
>>> bucket.tokens
|
133
|
-
10
|
134
|
-
"""
|
135
|
-
self.tokens_returned += tokens
|
136
|
-
self.tokens = min(self.capacity, self.tokens + tokens)
|
137
|
-
self.log.append((time.monotonic(), self.tokens))
|
138
|
-
|
139
|
-
def refill(self) -> None:
|
140
|
-
"""Refill the bucket with new tokens based on elapsed time.
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
>>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
|
145
|
-
>>> bucket.tokens = 0
|
146
|
-
>>> bucket.refill()
|
147
|
-
>>> bucket.tokens > 0
|
148
|
-
True
|
149
|
-
"""
|
150
|
-
"""Refill the bucket with new tokens based on elapsed time."""
|
151
|
-
now = time.monotonic()
|
152
|
-
# print(f"Time is now: {now}; Last refill time: {self.last_refill}")
|
153
|
-
elapsed = now - self.last_refill
|
154
|
-
# print("Elapsed time: ", elapsed)
|
155
|
-
refill_amount = elapsed * self.refill_rate
|
156
|
-
self.tokens = min(self.capacity, self.tokens + refill_amount)
|
157
|
-
self.last_refill = now
|
158
|
-
|
159
|
-
if self.tokens < self.capacity:
|
160
|
-
pass
|
161
|
-
# print(f"Refilled. Current tokens: {self.tokens:.4f}")
|
162
|
-
# print(f"Elapsed time: {elapsed:.4f} seconds")
|
163
|
-
# print(f"Refill amount: {refill_amount:.4f}")
|
164
|
-
|
165
|
-
self.log.append((now, self.tokens))
|
166
|
-
|
167
|
-
def wait_time(self, requested_tokens: Union[float, int]) -> float:
|
168
|
-
"""Calculate the time to wait for the requested number of tokens."""
|
169
|
-
# self.refill() # Update the current token count
|
170
|
-
if self.tokens >= requested_tokens:
|
171
|
-
return 0
|
172
|
-
return (requested_tokens - self.tokens) / self.refill_rate
|
173
|
-
|
174
|
-
async def get_tokens(
|
175
|
-
self, amount: Union[int, float] = 1, cheat_bucket_capacity=True
|
176
|
-
) -> None:
|
177
|
-
"""Wait for the specified number of tokens to become available.
|
178
|
-
|
179
|
-
|
180
|
-
:param amount: The number of tokens
|
181
|
-
:param warn: If True, warn if the requested amount exceeds the bucket capacity.
|
182
|
-
|
183
|
-
>>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
|
184
|
-
>>> asyncio.run(bucket.get_tokens(5))
|
185
|
-
>>> bucket.tokens
|
186
|
-
5
|
187
|
-
>>> asyncio.run(bucket.get_tokens(9))
|
188
|
-
>>> bucket.tokens < 1
|
189
|
-
True
|
190
|
-
|
191
|
-
>>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=10, refill_rate=1)
|
192
|
-
>>> asyncio.run(bucket.get_tokens(11, cheat_bucket_capacity=False))
|
193
|
-
Traceback (most recent call last):
|
194
|
-
...
|
195
|
-
ValueError: Requested amount exceeds bucket capacity. Bucket capacity: 10, requested amount: 11. As the bucket never overflows, the requested amount will never be available.
|
196
|
-
>>> asyncio.run(bucket.get_tokens(11, cheat_bucket_capacity=True))
|
197
|
-
>>> bucket.capacity
|
198
|
-
12.100000000000001
|
199
|
-
"""
|
200
|
-
self.num_requests += amount
|
201
|
-
if amount >= self.capacity:
|
202
|
-
if not cheat_bucket_capacity:
|
203
|
-
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."
|
204
|
-
raise ValueError(msg)
|
205
|
-
else:
|
206
|
-
self.capacity = amount * 1.10
|
207
|
-
self._old_capacity = self.capacity
|
208
|
-
|
209
|
-
start_time = time.monotonic()
|
210
|
-
while True:
|
211
|
-
self.refill() # Refill based on elapsed time
|
212
|
-
if self.tokens >= amount:
|
213
|
-
self.tokens -= amount
|
214
|
-
break
|
215
|
-
|
216
|
-
wait_time = self.wait_time(amount)
|
217
|
-
if wait_time > 0:
|
218
|
-
await asyncio.sleep(wait_time)
|
219
|
-
|
220
|
-
self.num_released += amount
|
221
|
-
now = time.monotonic()
|
222
|
-
self.log.append((now, self.tokens))
|
223
|
-
return None
|
224
|
-
|
225
|
-
def get_log(self) -> list[tuple]:
|
226
|
-
return self.log
|
227
|
-
|
228
|
-
def visualize(self):
|
229
|
-
"""Visualize the token bucket over time."""
|
230
|
-
times, tokens = zip(*self.get_log())
|
231
|
-
start_time = times[0]
|
232
|
-
times = [t - start_time for t in times] # Normalize time to start from 0
|
233
|
-
from matplotlib import pyplot as plt
|
234
|
-
|
235
|
-
plt.figure(figsize=(10, 6))
|
236
|
-
plt.plot(times, tokens, label="Tokens Available")
|
237
|
-
plt.xlabel("Time (seconds)", fontsize=12)
|
238
|
-
plt.ylabel("Number of Tokens", fontsize=12)
|
239
|
-
details = f"{self.bucket_name} ({self.bucket_type}) Bucket Usage Over Time\nCapacity: {self.capacity:.1f}, Refill Rate: {self.refill_rate:.1f}/second"
|
240
|
-
plt.title(details, fontsize=14)
|
241
|
-
|
242
|
-
plt.legend()
|
243
|
-
plt.grid(True)
|
244
|
-
plt.tight_layout()
|
245
|
-
plt.show()
|
246
|
-
|
247
|
-
def get_throughput(self, time_window: Optional[float] = None) -> float:
|
248
|
-
"""
|
249
|
-
Calculate the empirical bucket throughput in tokens per minute for the specified time window.
|
250
|
-
|
251
|
-
:param time_window: The time window in seconds to calculate the throughput for.
|
252
|
-
:return: The throughput in tokens per minute.
|
253
|
-
|
254
|
-
>>> bucket = TokenBucket(bucket_name="test", bucket_type="test", capacity=100, refill_rate=10)
|
255
|
-
>>> asyncio.run(bucket.get_tokens(50))
|
256
|
-
>>> time.sleep(1) # Wait for 1 second
|
257
|
-
>>> asyncio.run(bucket.get_tokens(30))
|
258
|
-
>>> throughput = bucket.get_throughput(1)
|
259
|
-
>>> 4750 < throughput < 4850
|
260
|
-
True
|
261
|
-
"""
|
262
|
-
now = time.monotonic()
|
263
|
-
|
264
|
-
if time_window is None:
|
265
|
-
start_time = self.creation_time
|
266
|
-
else:
|
267
|
-
start_time = now - time_window
|
268
|
-
|
269
|
-
if start_time < self.creation_time:
|
270
|
-
start_time = self.creation_time
|
271
|
-
|
272
|
-
elapsed_time = now - start_time
|
273
|
-
|
274
|
-
if elapsed_time == 0:
|
275
|
-
return self.num_released / 0.001
|
276
|
-
|
277
|
-
return (self.num_released / elapsed_time) * 60
|
278
|
-
|
279
|
-
|
280
|
-
if __name__ == "__main__":
|
281
|
-
import doctest
|
282
|
-
|
283
|
-
doctest.testmod(optionflags=doctest.ELLIPSIS)
|
@@ -1,191 +0,0 @@
|
|
1
|
-
from typing import Union, Optional
|
2
|
-
import asyncio
|
3
|
-
import time
|
4
|
-
import aiohttp
|
5
|
-
|
6
|
-
|
7
|
-
class TokenBucketClient:
|
8
|
-
"""REST API client version of TokenBucket that maintains the same interface
|
9
|
-
by delegating to a server running the original TokenBucket implementation."""
|
10
|
-
|
11
|
-
def __init__(
|
12
|
-
self,
|
13
|
-
*,
|
14
|
-
bucket_name: str,
|
15
|
-
bucket_type: str,
|
16
|
-
capacity: Union[int, float],
|
17
|
-
refill_rate: Union[int, float],
|
18
|
-
api_base_url: str = "http://localhost:8000",
|
19
|
-
):
|
20
|
-
self.bucket_name = bucket_name
|
21
|
-
self.bucket_type = bucket_type
|
22
|
-
self.capacity = capacity
|
23
|
-
self.refill_rate = refill_rate
|
24
|
-
self.api_base_url = api_base_url
|
25
|
-
self.bucket_id = f"{bucket_name}_{bucket_type}"
|
26
|
-
|
27
|
-
# Initialize the bucket on the server
|
28
|
-
asyncio.run(self._create_bucket())
|
29
|
-
|
30
|
-
# Cache some values locally
|
31
|
-
self.creation_time = time.monotonic()
|
32
|
-
self.turbo_mode = False
|
33
|
-
|
34
|
-
async def _create_bucket(self):
|
35
|
-
async with aiohttp.ClientSession() as session:
|
36
|
-
payload = {
|
37
|
-
"bucket_name": self.bucket_name,
|
38
|
-
"bucket_type": self.bucket_type,
|
39
|
-
"capacity": self.capacity,
|
40
|
-
"refill_rate": self.refill_rate,
|
41
|
-
}
|
42
|
-
async with session.post(
|
43
|
-
f"{self.api_base_url}/bucket",
|
44
|
-
json=payload,
|
45
|
-
) as response:
|
46
|
-
if response.status != 200:
|
47
|
-
raise ValueError(f"Unexpected error: {await response.text()}")
|
48
|
-
|
49
|
-
result = await response.json()
|
50
|
-
if result["status"] == "existing":
|
51
|
-
# Update our local values to match the existing bucket
|
52
|
-
self.capacity = float(result["bucket"]["capacity"])
|
53
|
-
self.refill_rate = float(result["bucket"]["refill_rate"])
|
54
|
-
|
55
|
-
def turbo_mode_on(self):
|
56
|
-
"""Set the refill rate to infinity."""
|
57
|
-
asyncio.run(self._set_turbo_mode(True))
|
58
|
-
self.turbo_mode = True
|
59
|
-
|
60
|
-
def turbo_mode_off(self):
|
61
|
-
"""Restore the refill rate to its original value."""
|
62
|
-
asyncio.run(self._set_turbo_mode(False))
|
63
|
-
self.turbo_mode = False
|
64
|
-
|
65
|
-
async def add_tokens(self, amount: Union[int, float]):
|
66
|
-
"""Add tokens to the bucket."""
|
67
|
-
async with aiohttp.ClientSession() as session:
|
68
|
-
async with session.post(
|
69
|
-
f"{self.api_base_url}/bucket/{self.bucket_id}/add_tokens",
|
70
|
-
params={"amount": amount},
|
71
|
-
) as response:
|
72
|
-
if response.status != 200:
|
73
|
-
raise ValueError(f"Failed to add tokens: {await response.text()}")
|
74
|
-
|
75
|
-
async def _set_turbo_mode(self, state: bool):
|
76
|
-
async with aiohttp.ClientSession() as session:
|
77
|
-
async with session.post(
|
78
|
-
f"{self.api_base_url}/bucket/{self.bucket_id}/turbo_mode/{str(state).lower()}"
|
79
|
-
) as response:
|
80
|
-
if response.status != 200:
|
81
|
-
raise ValueError(
|
82
|
-
f"Failed to set turbo mode: {await response.text()}"
|
83
|
-
)
|
84
|
-
|
85
|
-
async def get_tokens(
|
86
|
-
self, amount: Union[int, float] = 1, cheat_bucket_capacity=True
|
87
|
-
) -> None:
|
88
|
-
async with aiohttp.ClientSession() as session:
|
89
|
-
async with session.post(
|
90
|
-
f"{self.api_base_url}/bucket/{self.bucket_id}/get_tokens",
|
91
|
-
params={
|
92
|
-
"amount": amount,
|
93
|
-
"cheat_bucket_capacity": int(cheat_bucket_capacity),
|
94
|
-
},
|
95
|
-
) as response:
|
96
|
-
if response.status != 200:
|
97
|
-
raise ValueError(f"Failed to get tokens: {await response.text()}")
|
98
|
-
|
99
|
-
def get_throughput(self, time_window: Optional[float] = None) -> float:
|
100
|
-
status = asyncio.run(self._get_status())
|
101
|
-
now = time.monotonic()
|
102
|
-
|
103
|
-
if time_window is None:
|
104
|
-
start_time = self.creation_time
|
105
|
-
else:
|
106
|
-
start_time = now - time_window
|
107
|
-
|
108
|
-
if start_time < self.creation_time:
|
109
|
-
start_time = self.creation_time
|
110
|
-
|
111
|
-
elapsed_time = now - start_time
|
112
|
-
|
113
|
-
if elapsed_time == 0:
|
114
|
-
return status["num_released"] / 0.001
|
115
|
-
|
116
|
-
return (status["num_released"] / elapsed_time) * 60
|
117
|
-
|
118
|
-
async def _get_status(self) -> dict:
|
119
|
-
async with aiohttp.ClientSession() as session:
|
120
|
-
async with session.get(
|
121
|
-
f"{self.api_base_url}/bucket/{self.bucket_id}/status"
|
122
|
-
) as response:
|
123
|
-
if response.status != 200:
|
124
|
-
raise ValueError(
|
125
|
-
f"Failed to get bucket status: {await response.text()}"
|
126
|
-
)
|
127
|
-
return await response.json()
|
128
|
-
|
129
|
-
def __add__(self, other) -> "TokenBucketClient":
|
130
|
-
"""Combine two token buckets."""
|
131
|
-
return TokenBucketClient(
|
132
|
-
bucket_name=self.bucket_name,
|
133
|
-
bucket_type=self.bucket_type,
|
134
|
-
capacity=min(self.capacity, other.capacity),
|
135
|
-
refill_rate=min(self.refill_rate, other.refill_rate),
|
136
|
-
api_base_url=self.api_base_url,
|
137
|
-
)
|
138
|
-
|
139
|
-
@property
|
140
|
-
def tokens(self) -> float:
|
141
|
-
"""Get the number of tokens remaining in the bucket."""
|
142
|
-
status = asyncio.run(self._get_status())
|
143
|
-
return float(status["tokens"])
|
144
|
-
|
145
|
-
def wait_time(self, requested_tokens: Union[float, int]) -> float:
|
146
|
-
"""Calculate the time to wait for the requested number of tokens."""
|
147
|
-
# self.refill() # Update the current token count
|
148
|
-
if self.tokens >= float(requested_tokens):
|
149
|
-
return 0.0
|
150
|
-
try:
|
151
|
-
return (requested_tokens - self.tokens) / self.refill_rate
|
152
|
-
except Exception as e:
|
153
|
-
raise ValueError(f"Error calculating wait time: {e}")
|
154
|
-
|
155
|
-
# def wait_time(self, num_tokens: Union[int, float]) -> float:
|
156
|
-
# return 0 # TODO - Need to implement this on the server side
|
157
|
-
|
158
|
-
def visualize(self):
|
159
|
-
"""Visualize the token bucket over time."""
|
160
|
-
status = asyncio.run(self._get_status())
|
161
|
-
times, tokens = zip(*status["log"])
|
162
|
-
start_time = times[0]
|
163
|
-
times = [t - start_time for t in times]
|
164
|
-
|
165
|
-
from matplotlib import pyplot as plt
|
166
|
-
|
167
|
-
plt.figure(figsize=(10, 6))
|
168
|
-
plt.plot(times, tokens, label="Tokens Available")
|
169
|
-
plt.xlabel("Time (seconds)", fontsize=12)
|
170
|
-
plt.ylabel("Number of Tokens", fontsize=12)
|
171
|
-
details = f"{self.bucket_name} ({self.bucket_type}) Bucket Usage Over Time\nCapacity: {self.capacity:.1f}, Refill Rate: {self.refill_rate:.1f}/second"
|
172
|
-
plt.title(details, fontsize=14)
|
173
|
-
plt.legend()
|
174
|
-
plt.grid(True)
|
175
|
-
plt.tight_layout()
|
176
|
-
plt.show()
|
177
|
-
|
178
|
-
|
179
|
-
if __name__ == "__main__":
|
180
|
-
import doctest
|
181
|
-
|
182
|
-
doctest.testmod()
|
183
|
-
# bucket = TokenBucketClient(
|
184
|
-
# bucket_name="test", bucket_type="test", capacity=100, refill_rate=10
|
185
|
-
# )
|
186
|
-
# asyncio.run(bucket.get_tokens(50))
|
187
|
-
# time.sleep(1) # Wait for 1 second
|
188
|
-
# asyncio.run(bucket.get_tokens(30))
|
189
|
-
# throughput = bucket.get_throughput(1)
|
190
|
-
# print(throughput)
|
191
|
-
# bucket.visualize()
|