models-dev 1.0.22__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- models_dev-1.0.22/.gitignore +10 -0
- models_dev-1.0.22/LICENSE +21 -0
- models_dev-1.0.22/PKG-INFO +27 -0
- models_dev-1.0.22/README.md +5 -0
- models_dev-1.0.22/pyproject.toml +48 -0
- models_dev-1.0.22/scripts/update.py +147 -0
- models_dev-1.0.22/src/models_dev/__init__.py +73 -0
- models_dev-1.0.22/src/models_dev/_loader.py +110 -0
- models_dev-1.0.22/src/models_dev/_types.py +102 -0
- models_dev-1.0.22/src/models_dev/data.json.gz +0 -0
- models_dev-1.0.22/src/models_dev/py.typed +0 -0
- models_dev-1.0.22/tests/test_data.py +92 -0
- models_dev-1.0.22/uv.lock +213 -0
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 Vladimir Klimontovich
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
Metadata-Version: 2.4
|
|
2
|
+
Name: models-dev
|
|
3
|
+
Version: 1.0.22
|
|
4
|
+
Summary: Typed Python interface to models.dev API data
|
|
5
|
+
Project-URL: Homepage, https://github.com/vklimontovich/models-dev
|
|
6
|
+
Project-URL: Documentation, https://models.dev
|
|
7
|
+
Author: vklmn
|
|
8
|
+
License-Expression: MIT
|
|
9
|
+
License-File: LICENSE
|
|
10
|
+
Keywords: ai,anthropic,llm,models,openai
|
|
11
|
+
Classifier: Development Status :: 4 - Beta
|
|
12
|
+
Classifier: Intended Audience :: Developers
|
|
13
|
+
Classifier: License :: OSI Approved :: MIT License
|
|
14
|
+
Classifier: Programming Language :: Python :: 3
|
|
15
|
+
Classifier: Programming Language :: Python :: 3.10
|
|
16
|
+
Classifier: Programming Language :: Python :: 3.11
|
|
17
|
+
Classifier: Programming Language :: Python :: 3.12
|
|
18
|
+
Classifier: Programming Language :: Python :: 3.13
|
|
19
|
+
Classifier: Typing :: Typed
|
|
20
|
+
Requires-Python: >=3.10
|
|
21
|
+
Description-Content-Type: text/markdown
|
|
22
|
+
|
|
23
|
+
# models-dev
|
|
24
|
+
|
|
25
|
+
Typed Python package for [models.dev](https://models.dev) data. Access 2000+ LLM models from 75+ providers with full typing support. No HTTP calls - data is bundled and auto-updated hourly.
|
|
26
|
+
|
|
27
|
+
See [GitHub](https://github.com/vklimontovich/models-dev) for documentation and examples.
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
# models-dev
|
|
2
|
+
|
|
3
|
+
Typed Python package for [models.dev](https://models.dev) data. Access 2000+ LLM models from 75+ providers with full typing support. No HTTP calls - data is bundled and auto-updated hourly.
|
|
4
|
+
|
|
5
|
+
See [GitHub](https://github.com/vklimontovich/models-dev) for documentation and examples.
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
[project]
|
|
2
|
+
name = "models-dev"
|
|
3
|
+
version = "1.0.22"
|
|
4
|
+
description = "Typed Python interface to models.dev API data"
|
|
5
|
+
readme = "README.md"
|
|
6
|
+
requires-python = ">=3.10"
|
|
7
|
+
license = "MIT"
|
|
8
|
+
authors = [{ name = "vklmn" }]
|
|
9
|
+
keywords = ["llm", "models", "ai", "openai", "anthropic"]
|
|
10
|
+
classifiers = [
|
|
11
|
+
"Development Status :: 4 - Beta",
|
|
12
|
+
"Intended Audience :: Developers",
|
|
13
|
+
"License :: OSI Approved :: MIT License",
|
|
14
|
+
"Programming Language :: Python :: 3",
|
|
15
|
+
"Programming Language :: Python :: 3.10",
|
|
16
|
+
"Programming Language :: Python :: 3.11",
|
|
17
|
+
"Programming Language :: Python :: 3.12",
|
|
18
|
+
"Programming Language :: Python :: 3.13",
|
|
19
|
+
"Typing :: Typed",
|
|
20
|
+
]
|
|
21
|
+
|
|
22
|
+
[project.urls]
|
|
23
|
+
Homepage = "https://github.com/vklimontovich/models-dev"
|
|
24
|
+
Documentation = "https://models.dev"
|
|
25
|
+
|
|
26
|
+
[dependency-groups]
|
|
27
|
+
dev = ["ruff", "ty", "pytest"]
|
|
28
|
+
|
|
29
|
+
[build-system]
|
|
30
|
+
requires = ["hatchling"]
|
|
31
|
+
build-backend = "hatchling.build"
|
|
32
|
+
|
|
33
|
+
[tool.hatch.build.targets.wheel]
|
|
34
|
+
packages = ["src/models_dev"]
|
|
35
|
+
artifacts = ["src/models_dev/data.json.gz"]
|
|
36
|
+
|
|
37
|
+
[tool.ruff]
|
|
38
|
+
target-version = "py310"
|
|
39
|
+
line-length = 100
|
|
40
|
+
|
|
41
|
+
[tool.ruff.lint]
|
|
42
|
+
select = ["E", "F", "W", "I", "UP", "B", "C4", "SIM", "RUF"]
|
|
43
|
+
|
|
44
|
+
[tool.ty.environment]
|
|
45
|
+
python-version = "3.10"
|
|
46
|
+
|
|
47
|
+
[tool.pytest.ini_options]
|
|
48
|
+
testpaths = ["tests"]
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
# /// script
|
|
2
|
+
# dependencies = ["anthropic", "httpx", "gitpython"]
|
|
3
|
+
# ///
|
|
4
|
+
"""Update data.json.gz from models.dev.
|
|
5
|
+
|
|
6
|
+
Usage:
|
|
7
|
+
uv run scripts/update.py [--dry-run]
|
|
8
|
+
"""
|
|
9
|
+
|
|
10
|
+
import argparse
|
|
11
|
+
import difflib
|
|
12
|
+
import gzip
|
|
13
|
+
import json
|
|
14
|
+
import os
|
|
15
|
+
import re
|
|
16
|
+
import sys
|
|
17
|
+
from pathlib import Path
|
|
18
|
+
|
|
19
|
+
import git
|
|
20
|
+
import httpx
|
|
21
|
+
|
|
22
|
+
DATA_URL = "https://models.dev/api.json"
|
|
23
|
+
ROOT = Path(__file__).parent.parent.parent
|
|
24
|
+
DATA_PATH = ROOT / "python" / "src" / "models_dev" / "data.json.gz"
|
|
25
|
+
PYPROJECT_PATH = ROOT / "python" / "pyproject.toml"
|
|
26
|
+
|
|
27
|
+
|
|
28
|
+
def fetch_latest() -> str:
|
|
29
|
+
"""Fetch latest data as formatted JSON string."""
|
|
30
|
+
resp = httpx.get(DATA_URL, timeout=30)
|
|
31
|
+
resp.raise_for_status()
|
|
32
|
+
return json.dumps(resp.json(), indent=2, sort_keys=True)
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
def load_current() -> str:
|
|
36
|
+
"""Load current data as formatted JSON string."""
|
|
37
|
+
if not DATA_PATH.exists():
|
|
38
|
+
return ""
|
|
39
|
+
with gzip.open(DATA_PATH, "rt", encoding="utf-8") as f:
|
|
40
|
+
return json.dumps(json.load(f), indent=2, sort_keys=True)
|
|
41
|
+
|
|
42
|
+
|
|
43
|
+
def compute_diff(old: str, new: str) -> str:
|
|
44
|
+
"""Compute unified diff."""
|
|
45
|
+
old_lines = old.splitlines(keepends=True)
|
|
46
|
+
new_lines = new.splitlines(keepends=True)
|
|
47
|
+
diff = difflib.unified_diff(old_lines, new_lines, fromfile="old", tofile="new")
|
|
48
|
+
return "".join(diff)
|
|
49
|
+
|
|
50
|
+
|
|
51
|
+
def generate_commit_message(diff: str) -> str:
|
|
52
|
+
"""Generate commit message using Anthropic API."""
|
|
53
|
+
api_key = os.environ.get("ANTHROPIC_API_KEY")
|
|
54
|
+
if not api_key:
|
|
55
|
+
return "chore: update models data"
|
|
56
|
+
|
|
57
|
+
import anthropic
|
|
58
|
+
|
|
59
|
+
client = anthropic.Anthropic(api_key=api_key)
|
|
60
|
+
prompt = (
|
|
61
|
+
"Generate a commit message for this diff. Use conventional commits. Be concise. "
|
|
62
|
+
"NEVER wrap code in ``` etc. Output just commit message. "
|
|
63
|
+
"ABSOLUTELY no formatting, plain ASCII"
|
|
64
|
+
)
|
|
65
|
+
prompt = f"{prompt}\n\n{diff[:10000]}"
|
|
66
|
+
resp = client.messages.create(
|
|
67
|
+
model="claude-haiku-4-5",
|
|
68
|
+
max_tokens=1000,
|
|
69
|
+
messages=[{"role": "user", "content": prompt}],
|
|
70
|
+
)
|
|
71
|
+
return resp.content[0].text.strip()
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+
def get_version() -> str:
|
|
75
|
+
content = PYPROJECT_PATH.read_text()
|
|
76
|
+
match = re.search(r'^version = "(.+)"', content, re.MULTILINE)
|
|
77
|
+
return match.group(1) if match else "0.0.0"
|
|
78
|
+
|
|
79
|
+
|
|
80
|
+
def bump_version() -> str:
|
|
81
|
+
"""Bump patch version and return new version."""
|
|
82
|
+
content = PYPROJECT_PATH.read_text()
|
|
83
|
+
old = get_version()
|
|
84
|
+
parts = old.split(".")
|
|
85
|
+
parts[-1] = str(int(parts[-1]) + 1)
|
|
86
|
+
new = ".".join(parts)
|
|
87
|
+
content = re.sub(r'^version = ".+"', f'version = "{new}"', content, flags=re.MULTILINE)
|
|
88
|
+
PYPROJECT_PATH.write_text(content)
|
|
89
|
+
return new
|
|
90
|
+
|
|
91
|
+
|
|
92
|
+
def save_data(data_str: str) -> None:
|
|
93
|
+
"""Save data to gzipped JSON."""
|
|
94
|
+
with gzip.open(DATA_PATH, "wt", encoding="utf-8") as f:
|
|
95
|
+
f.write(json.dumps(json.loads(data_str), separators=(",", ":")))
|
|
96
|
+
|
|
97
|
+
|
|
98
|
+
def commit(message: str) -> None:
|
|
99
|
+
"""Commit changes."""
|
|
100
|
+
repo = git.Repo(ROOT)
|
|
101
|
+
repo.index.add(["python/pyproject.toml", "python/src/models_dev/data.json.gz"])
|
|
102
|
+
repo.index.commit(message)
|
|
103
|
+
|
|
104
|
+
|
|
105
|
+
def main() -> int:
|
|
106
|
+
parser = argparse.ArgumentParser()
|
|
107
|
+
parser.add_argument("--dry-run", action="store_true")
|
|
108
|
+
args = parser.parse_args()
|
|
109
|
+
|
|
110
|
+
print("Fetching latest data...")
|
|
111
|
+
new_data = fetch_latest()
|
|
112
|
+
old_data = load_current()
|
|
113
|
+
|
|
114
|
+
print("Computing diff...")
|
|
115
|
+
diff = compute_diff(old_data, new_data)
|
|
116
|
+
|
|
117
|
+
if not diff:
|
|
118
|
+
print("No changes")
|
|
119
|
+
return 0
|
|
120
|
+
|
|
121
|
+
print(f"Changes detected ({len(diff)} chars)")
|
|
122
|
+
print(diff[:500])
|
|
123
|
+
|
|
124
|
+
print("Generating commit message...")
|
|
125
|
+
message = generate_commit_message(diff)
|
|
126
|
+
print(f"Message: {message}")
|
|
127
|
+
|
|
128
|
+
if args.dry_run:
|
|
129
|
+
print("Dry run, stopping")
|
|
130
|
+
return 0
|
|
131
|
+
|
|
132
|
+
print("Saving data...")
|
|
133
|
+
save_data(new_data)
|
|
134
|
+
|
|
135
|
+
print("Bumping version...")
|
|
136
|
+
version = bump_version()
|
|
137
|
+
print(f"New version: {version}")
|
|
138
|
+
|
|
139
|
+
print("Committing...")
|
|
140
|
+
commit(message)
|
|
141
|
+
|
|
142
|
+
print("Done")
|
|
143
|
+
return 0
|
|
144
|
+
|
|
145
|
+
|
|
146
|
+
if __name__ == "__main__":
|
|
147
|
+
sys.exit(main())
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
"""Typed Python interface to models.dev API data.
|
|
2
|
+
|
|
3
|
+
Example usage:
|
|
4
|
+
from models_dev import providers, get_provider, get_model_by_id
|
|
5
|
+
|
|
6
|
+
# Iterate all providers
|
|
7
|
+
for provider in providers():
|
|
8
|
+
print(provider.name, len(provider.models))
|
|
9
|
+
|
|
10
|
+
# Get specific provider
|
|
11
|
+
openai = get_provider("openai")
|
|
12
|
+
print(openai.name) # "OpenAI"
|
|
13
|
+
|
|
14
|
+
# Get model by ID
|
|
15
|
+
gpt4o = get_model_by_id("openai", "gpt-4o")
|
|
16
|
+
print(gpt4o.cost.input) # price per 1M tokens
|
|
17
|
+
|
|
18
|
+
# Get model by name
|
|
19
|
+
gpt4o = openai.get_model_by_name("GPT-4o")
|
|
20
|
+
"""
|
|
21
|
+
|
|
22
|
+
from collections.abc import Iterator
|
|
23
|
+
|
|
24
|
+
from ._loader import _get_providers
|
|
25
|
+
from ._types import (
|
|
26
|
+
Cost,
|
|
27
|
+
InputModality,
|
|
28
|
+
Interleaved,
|
|
29
|
+
Limit,
|
|
30
|
+
Modalities,
|
|
31
|
+
Model,
|
|
32
|
+
ModelProvider,
|
|
33
|
+
OutputModality,
|
|
34
|
+
Provider,
|
|
35
|
+
Status,
|
|
36
|
+
)
|
|
37
|
+
|
|
38
|
+
__all__ = [
|
|
39
|
+
"Cost",
|
|
40
|
+
"InputModality",
|
|
41
|
+
"Interleaved",
|
|
42
|
+
"Limit",
|
|
43
|
+
"Modalities",
|
|
44
|
+
"Model",
|
|
45
|
+
"ModelProvider",
|
|
46
|
+
"OutputModality",
|
|
47
|
+
"Provider",
|
|
48
|
+
"Status",
|
|
49
|
+
"get_model_by_id",
|
|
50
|
+
"get_model_by_name",
|
|
51
|
+
"get_provider",
|
|
52
|
+
"providers",
|
|
53
|
+
]
|
|
54
|
+
|
|
55
|
+
|
|
56
|
+
def providers() -> Iterator[Provider]:
|
|
57
|
+
"""Iterate over all providers."""
|
|
58
|
+
yield from _get_providers().values()
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
def get_provider(provider_id: str) -> Provider:
|
|
62
|
+
"""Get a provider by ID. Raises KeyError if not found."""
|
|
63
|
+
return _get_providers()[provider_id]
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
def get_model_by_id(provider_id: str, model_id: str) -> Model:
|
|
67
|
+
"""Get a model by provider ID and model ID. Raises KeyError if not found."""
|
|
68
|
+
return get_provider(provider_id).models[model_id]
|
|
69
|
+
|
|
70
|
+
|
|
71
|
+
def get_model_by_name(provider_id: str, name: str) -> Model:
|
|
72
|
+
"""Get a model by provider ID and model name. Raises KeyError if not found."""
|
|
73
|
+
return get_provider(provider_id).get_model_by_name(name)
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
"""Load and deserialize models.dev data into typed objects."""
|
|
2
|
+
|
|
3
|
+
import gzip
|
|
4
|
+
import json
|
|
5
|
+
from functools import cache
|
|
6
|
+
from pathlib import Path
|
|
7
|
+
from typing import Any
|
|
8
|
+
|
|
9
|
+
from ._types import (
|
|
10
|
+
Cost,
|
|
11
|
+
Interleaved,
|
|
12
|
+
Limit,
|
|
13
|
+
Modalities,
|
|
14
|
+
Model,
|
|
15
|
+
ModelProvider,
|
|
16
|
+
Provider,
|
|
17
|
+
)
|
|
18
|
+
|
|
19
|
+
_DATA_PATH = Path(__file__).parent / "data.json.gz"
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
def _load_raw_data() -> dict[str, Any]:
|
|
23
|
+
with gzip.open(_DATA_PATH, "rt", encoding="utf-8") as f:
|
|
24
|
+
return json.load(f)
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
def _load_cost(data: dict[str, Any] | None) -> Cost | None:
|
|
28
|
+
if data is None:
|
|
29
|
+
return None
|
|
30
|
+
return Cost(
|
|
31
|
+
input=data.get("input"),
|
|
32
|
+
output=data.get("output"),
|
|
33
|
+
cache_read=data.get("cache_read"),
|
|
34
|
+
cache_write=data.get("cache_write"),
|
|
35
|
+
reasoning=data.get("reasoning"),
|
|
36
|
+
input_audio=data.get("input_audio"),
|
|
37
|
+
output_audio=data.get("output_audio"),
|
|
38
|
+
context_over_200k=data.get("context_over_200k"),
|
|
39
|
+
)
|
|
40
|
+
|
|
41
|
+
|
|
42
|
+
def _load_limit(data: dict[str, Any]) -> Limit:
|
|
43
|
+
return Limit(
|
|
44
|
+
context=data.get("context"),
|
|
45
|
+
input=data.get("input"),
|
|
46
|
+
output=data.get("output"),
|
|
47
|
+
)
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def _load_modalities(data: dict[str, Any]) -> Modalities:
|
|
51
|
+
return Modalities(
|
|
52
|
+
input=tuple(data.get("input", [])),
|
|
53
|
+
output=tuple(data.get("output", [])),
|
|
54
|
+
)
|
|
55
|
+
|
|
56
|
+
|
|
57
|
+
def _load_interleaved(data: dict[str, Any] | bool | None) -> Interleaved | None:
|
|
58
|
+
if data is None or data is False:
|
|
59
|
+
return None
|
|
60
|
+
if data is True:
|
|
61
|
+
return Interleaved()
|
|
62
|
+
return Interleaved(field=data.get("field"))
|
|
63
|
+
|
|
64
|
+
|
|
65
|
+
def _load_model_provider(data: dict[str, Any] | None) -> ModelProvider | None:
|
|
66
|
+
if data is None:
|
|
67
|
+
return None
|
|
68
|
+
return ModelProvider(npm=data["npm"])
|
|
69
|
+
|
|
70
|
+
|
|
71
|
+
def _load_model(data: dict[str, Any]) -> Model:
|
|
72
|
+
return Model(
|
|
73
|
+
id=data["id"],
|
|
74
|
+
name=data["name"],
|
|
75
|
+
attachment=data["attachment"],
|
|
76
|
+
reasoning=data["reasoning"],
|
|
77
|
+
tool_call=data["tool_call"],
|
|
78
|
+
open_weights=data["open_weights"],
|
|
79
|
+
release_date=data["release_date"],
|
|
80
|
+
last_updated=data["last_updated"],
|
|
81
|
+
modalities=_load_modalities(data["modalities"]),
|
|
82
|
+
limit=_load_limit(data["limit"]),
|
|
83
|
+
family=data.get("family"),
|
|
84
|
+
structured_output=data.get("structured_output"),
|
|
85
|
+
temperature=data.get("temperature"),
|
|
86
|
+
knowledge=data.get("knowledge"),
|
|
87
|
+
cost=_load_cost(data.get("cost")),
|
|
88
|
+
status=data.get("status"),
|
|
89
|
+
interleaved=_load_interleaved(data.get("interleaved")),
|
|
90
|
+
provider=_load_model_provider(data.get("provider")),
|
|
91
|
+
)
|
|
92
|
+
|
|
93
|
+
|
|
94
|
+
def _load_provider(data: dict[str, Any]) -> Provider:
|
|
95
|
+
models = {mid: _load_model(mdata) for mid, mdata in data["models"].items()}
|
|
96
|
+
return Provider(
|
|
97
|
+
id=data["id"],
|
|
98
|
+
name=data["name"],
|
|
99
|
+
env=tuple(data["env"]),
|
|
100
|
+
npm=data["npm"],
|
|
101
|
+
doc=data["doc"],
|
|
102
|
+
models=models,
|
|
103
|
+
api=data.get("api"),
|
|
104
|
+
)
|
|
105
|
+
|
|
106
|
+
|
|
107
|
+
@cache
|
|
108
|
+
def _get_providers() -> dict[str, Provider]:
|
|
109
|
+
raw = _load_raw_data()
|
|
110
|
+
return {pid: _load_provider(pdata) for pid, pdata in raw.items()}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
"""Type definitions for models.dev data structures."""
|
|
2
|
+
|
|
3
|
+
from collections.abc import Mapping
|
|
4
|
+
from dataclasses import dataclass
|
|
5
|
+
from typing import Literal
|
|
6
|
+
|
|
7
|
+
InputModality = Literal["text", "image", "audio", "video", "pdf"]
|
|
8
|
+
OutputModality = Literal["text", "image", "audio", "video"]
|
|
9
|
+
Status = Literal["alpha", "deprecated"]
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
@dataclass(frozen=True, slots=True)
|
|
13
|
+
class Cost:
|
|
14
|
+
"""Model pricing in USD per 1M tokens."""
|
|
15
|
+
|
|
16
|
+
input: float | None = None
|
|
17
|
+
output: float | None = None
|
|
18
|
+
cache_read: float | None = None
|
|
19
|
+
cache_write: float | None = None
|
|
20
|
+
reasoning: float | None = None
|
|
21
|
+
input_audio: float | None = None
|
|
22
|
+
output_audio: float | None = None
|
|
23
|
+
context_over_200k: float | None = None
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
@dataclass(frozen=True, slots=True)
|
|
27
|
+
class Limit:
|
|
28
|
+
"""Model token limits."""
|
|
29
|
+
|
|
30
|
+
context: int | None = None
|
|
31
|
+
input: int | None = None
|
|
32
|
+
output: int | None = None
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
@dataclass(frozen=True, slots=True)
|
|
36
|
+
class Modalities:
|
|
37
|
+
"""Supported input/output modalities."""
|
|
38
|
+
|
|
39
|
+
input: tuple[InputModality, ...]
|
|
40
|
+
output: tuple[OutputModality, ...]
|
|
41
|
+
|
|
42
|
+
|
|
43
|
+
@dataclass(frozen=True, slots=True)
|
|
44
|
+
class Interleaved:
|
|
45
|
+
"""Streaming reasoning configuration."""
|
|
46
|
+
|
|
47
|
+
field: str | None = None
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
@dataclass(frozen=True, slots=True)
|
|
51
|
+
class ModelProvider:
|
|
52
|
+
"""SDK override for cross-provider models."""
|
|
53
|
+
|
|
54
|
+
npm: str
|
|
55
|
+
|
|
56
|
+
|
|
57
|
+
@dataclass(frozen=True, slots=True)
|
|
58
|
+
class Model:
|
|
59
|
+
"""AI model metadata."""
|
|
60
|
+
|
|
61
|
+
id: str
|
|
62
|
+
name: str
|
|
63
|
+
attachment: bool
|
|
64
|
+
reasoning: bool
|
|
65
|
+
tool_call: bool
|
|
66
|
+
open_weights: bool
|
|
67
|
+
release_date: str
|
|
68
|
+
last_updated: str
|
|
69
|
+
modalities: Modalities
|
|
70
|
+
limit: Limit
|
|
71
|
+
family: str | None = None
|
|
72
|
+
structured_output: bool | None = None
|
|
73
|
+
temperature: bool | None = None
|
|
74
|
+
knowledge: str | None = None
|
|
75
|
+
cost: Cost | None = None
|
|
76
|
+
status: Status | None = None
|
|
77
|
+
interleaved: Interleaved | None = None
|
|
78
|
+
provider: ModelProvider | None = None
|
|
79
|
+
|
|
80
|
+
|
|
81
|
+
@dataclass(frozen=True, slots=True)
|
|
82
|
+
class Provider:
|
|
83
|
+
"""AI provider metadata."""
|
|
84
|
+
|
|
85
|
+
id: str
|
|
86
|
+
name: str
|
|
87
|
+
env: tuple[str, ...]
|
|
88
|
+
npm: str
|
|
89
|
+
doc: str
|
|
90
|
+
models: Mapping[str, Model]
|
|
91
|
+
api: str | None = None
|
|
92
|
+
|
|
93
|
+
def get_model_by_id(self, model_id: str) -> Model:
|
|
94
|
+
"""Get model by ID. Raises KeyError if not found."""
|
|
95
|
+
return self.models[model_id]
|
|
96
|
+
|
|
97
|
+
def get_model_by_name(self, name: str) -> Model:
|
|
98
|
+
"""Get model by name. Raises KeyError if not found."""
|
|
99
|
+
for model in self.models.values():
|
|
100
|
+
if model.name == name:
|
|
101
|
+
return model
|
|
102
|
+
raise KeyError(f"Model with name '{name}' not found")
|
|
Binary file
|
|
File without changes
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
"""Tests for models_dev data integrity."""
|
|
2
|
+
|
|
3
|
+
import gzip
|
|
4
|
+
import json
|
|
5
|
+
from pathlib import Path
|
|
6
|
+
|
|
7
|
+
import pytest
|
|
8
|
+
|
|
9
|
+
from models_dev import Model, Provider, get_provider, providers
|
|
10
|
+
|
|
11
|
+
DATA_PATH = Path(__file__).parent.parent / "src" / "models_dev" / "data.json.gz"
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class TestDataIntegrity:
|
|
15
|
+
"""Test that data.json.gz is valid and loads correctly."""
|
|
16
|
+
|
|
17
|
+
def test_data_file_exists(self) -> None:
|
|
18
|
+
assert DATA_PATH.exists(), "data.json.gz must exist"
|
|
19
|
+
|
|
20
|
+
def test_data_is_valid_gzip(self) -> None:
|
|
21
|
+
with gzip.open(DATA_PATH, "rt") as f:
|
|
22
|
+
data = json.load(f)
|
|
23
|
+
assert isinstance(data, dict)
|
|
24
|
+
assert len(data) > 0
|
|
25
|
+
|
|
26
|
+
def test_all_providers_load(self) -> None:
|
|
27
|
+
provider_list = list(providers())
|
|
28
|
+
assert len(provider_list) > 0
|
|
29
|
+
for p in provider_list:
|
|
30
|
+
assert isinstance(p, Provider)
|
|
31
|
+
assert p.id
|
|
32
|
+
assert p.name
|
|
33
|
+
|
|
34
|
+
def test_all_models_have_required_fields(self) -> None:
|
|
35
|
+
for provider in providers():
|
|
36
|
+
for model in provider.models.values():
|
|
37
|
+
assert isinstance(model, Model)
|
|
38
|
+
assert model.id
|
|
39
|
+
assert model.name
|
|
40
|
+
assert model.modalities is not None
|
|
41
|
+
assert model.limit is not None
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
class TestKnownProviders:
|
|
45
|
+
"""Test that known providers exist and have expected data."""
|
|
46
|
+
|
|
47
|
+
@pytest.mark.parametrize("provider_id", ["openai", "anthropic", "google"])
|
|
48
|
+
def test_major_providers_exist(self, provider_id: str) -> None:
|
|
49
|
+
provider = get_provider(provider_id)
|
|
50
|
+
assert provider.id == provider_id
|
|
51
|
+
assert len(provider.models) > 0
|
|
52
|
+
|
|
53
|
+
def test_openai_has_gpt4(self) -> None:
|
|
54
|
+
openai = get_provider("openai")
|
|
55
|
+
model_ids = list(openai.models.keys())
|
|
56
|
+
assert any("gpt-4" in m for m in model_ids)
|
|
57
|
+
|
|
58
|
+
def test_anthropic_has_claude(self) -> None:
|
|
59
|
+
anthropic = get_provider("anthropic")
|
|
60
|
+
model_ids = list(anthropic.models.keys())
|
|
61
|
+
assert any("claude" in m for m in model_ids)
|
|
62
|
+
|
|
63
|
+
|
|
64
|
+
class TestModelFields:
|
|
65
|
+
"""Test model field types and values."""
|
|
66
|
+
|
|
67
|
+
def test_cost_fields_are_numeric(self) -> None:
|
|
68
|
+
for provider in providers():
|
|
69
|
+
for model in provider.models.values():
|
|
70
|
+
if model.cost:
|
|
71
|
+
if model.cost.input is not None:
|
|
72
|
+
assert isinstance(model.cost.input, (int, float))
|
|
73
|
+
if model.cost.output is not None:
|
|
74
|
+
assert isinstance(model.cost.output, (int, float))
|
|
75
|
+
|
|
76
|
+
def test_limit_fields_are_integers(self) -> None:
|
|
77
|
+
for provider in providers():
|
|
78
|
+
for model in provider.models.values():
|
|
79
|
+
if model.limit.context is not None:
|
|
80
|
+
assert isinstance(model.limit.context, int)
|
|
81
|
+
if model.limit.output is not None:
|
|
82
|
+
assert isinstance(model.limit.output, int)
|
|
83
|
+
|
|
84
|
+
def test_modalities_are_valid(self) -> None:
|
|
85
|
+
valid_input = {"text", "image", "audio", "video", "pdf"}
|
|
86
|
+
valid_output = {"text", "image", "audio", "video"}
|
|
87
|
+
for provider in providers():
|
|
88
|
+
for model in provider.models.values():
|
|
89
|
+
for m in model.modalities.input:
|
|
90
|
+
assert m in valid_input, f"Invalid input modality: {m}"
|
|
91
|
+
for m in model.modalities.output:
|
|
92
|
+
assert m in valid_output, f"Invalid output modality: {m}"
|
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
version = 1
|
|
2
|
+
revision = 3
|
|
3
|
+
requires-python = ">=3.10"
|
|
4
|
+
|
|
5
|
+
[[package]]
|
|
6
|
+
name = "colorama"
|
|
7
|
+
version = "0.4.6"
|
|
8
|
+
source = { registry = "https://pypi.org/simple" }
|
|
9
|
+
sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" }
|
|
10
|
+
wheels = [
|
|
11
|
+
{ url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" },
|
|
12
|
+
]
|
|
13
|
+
|
|
14
|
+
[[package]]
|
|
15
|
+
name = "exceptiongroup"
|
|
16
|
+
version = "1.3.1"
|
|
17
|
+
source = { registry = "https://pypi.org/simple" }
|
|
18
|
+
dependencies = [
|
|
19
|
+
{ name = "typing-extensions", marker = "python_full_version < '3.13'" },
|
|
20
|
+
]
|
|
21
|
+
sdist = { url = "https://files.pythonhosted.org/packages/50/79/66800aadf48771f6b62f7eb014e352e5d06856655206165d775e675a02c9/exceptiongroup-1.3.1.tar.gz", hash = "sha256:8b412432c6055b0b7d14c310000ae93352ed6754f70fa8f7c34141f91c4e3219", size = 30371, upload-time = "2025-11-21T23:01:54.787Z" }
|
|
22
|
+
wheels = [
|
|
23
|
+
{ url = "https://files.pythonhosted.org/packages/8a/0e/97c33bf5009bdbac74fd2beace167cab3f978feb69cc36f1ef79360d6c4e/exceptiongroup-1.3.1-py3-none-any.whl", hash = "sha256:a7a39a3bd276781e98394987d3a5701d0c4edffb633bb7a5144577f82c773598", size = 16740, upload-time = "2025-11-21T23:01:53.443Z" },
|
|
24
|
+
]
|
|
25
|
+
|
|
26
|
+
[[package]]
|
|
27
|
+
name = "iniconfig"
|
|
28
|
+
version = "2.3.0"
|
|
29
|
+
source = { registry = "https://pypi.org/simple" }
|
|
30
|
+
sdist = { url = "https://files.pythonhosted.org/packages/72/34/14ca021ce8e5dfedc35312d08ba8bf51fdd999c576889fc2c24cb97f4f10/iniconfig-2.3.0.tar.gz", hash = "sha256:c76315c77db068650d49c5b56314774a7804df16fee4402c1f19d6d15d8c4730", size = 20503, upload-time = "2025-10-18T21:55:43.219Z" }
|
|
31
|
+
wheels = [
|
|
32
|
+
{ url = "https://files.pythonhosted.org/packages/cb/b1/3846dd7f199d53cb17f49cba7e651e9ce294d8497c8c150530ed11865bb8/iniconfig-2.3.0-py3-none-any.whl", hash = "sha256:f631c04d2c48c52b84d0d0549c99ff3859c98df65b3101406327ecc7d53fbf12", size = 7484, upload-time = "2025-10-18T21:55:41.639Z" },
|
|
33
|
+
]
|
|
34
|
+
|
|
35
|
+
[[package]]
|
|
36
|
+
name = "models-dev"
|
|
37
|
+
version = "1.0.22"
|
|
38
|
+
source = { editable = "." }
|
|
39
|
+
|
|
40
|
+
[package.dev-dependencies]
|
|
41
|
+
dev = [
|
|
42
|
+
{ name = "pytest" },
|
|
43
|
+
{ name = "ruff" },
|
|
44
|
+
{ name = "ty" },
|
|
45
|
+
]
|
|
46
|
+
|
|
47
|
+
[package.metadata]
|
|
48
|
+
|
|
49
|
+
[package.metadata.requires-dev]
|
|
50
|
+
dev = [
|
|
51
|
+
{ name = "pytest" },
|
|
52
|
+
{ name = "ruff" },
|
|
53
|
+
{ name = "ty" },
|
|
54
|
+
]
|
|
55
|
+
|
|
56
|
+
[[package]]
|
|
57
|
+
name = "packaging"
|
|
58
|
+
version = "25.0"
|
|
59
|
+
source = { registry = "https://pypi.org/simple" }
|
|
60
|
+
sdist = { url = "https://files.pythonhosted.org/packages/a1/d4/1fc4078c65507b51b96ca8f8c3ba19e6a61c8253c72794544580a7b6c24d/packaging-25.0.tar.gz", hash = "sha256:d443872c98d677bf60f6a1f2f8c1cb748e8fe762d2bf9d3148b5599295b0fc4f", size = 165727, upload-time = "2025-04-19T11:48:59.673Z" }
|
|
61
|
+
wheels = [
|
|
62
|
+
{ url = "https://files.pythonhosted.org/packages/20/12/38679034af332785aac8774540895e234f4d07f7545804097de4b666afd8/packaging-25.0-py3-none-any.whl", hash = "sha256:29572ef2b1f17581046b3a2227d5c611fb25ec70ca1ba8554b24b0e69331a484", size = 66469, upload-time = "2025-04-19T11:48:57.875Z" },
|
|
63
|
+
]
|
|
64
|
+
|
|
65
|
+
[[package]]
|
|
66
|
+
name = "pluggy"
|
|
67
|
+
version = "1.6.0"
|
|
68
|
+
source = { registry = "https://pypi.org/simple" }
|
|
69
|
+
sdist = { url = "https://files.pythonhosted.org/packages/f9/e2/3e91f31a7d2b083fe6ef3fa267035b518369d9511ffab804f839851d2779/pluggy-1.6.0.tar.gz", hash = "sha256:7dcc130b76258d33b90f61b658791dede3486c3e6bfb003ee5c9bfb396dd22f3", size = 69412, upload-time = "2025-05-15T12:30:07.975Z" }
|
|
70
|
+
wheels = [
|
|
71
|
+
{ url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538, upload-time = "2025-05-15T12:30:06.134Z" },
|
|
72
|
+
]
|
|
73
|
+
|
|
74
|
+
[[package]]
|
|
75
|
+
name = "pygments"
|
|
76
|
+
version = "2.19.2"
|
|
77
|
+
source = { registry = "https://pypi.org/simple" }
|
|
78
|
+
sdist = { url = "https://files.pythonhosted.org/packages/b0/77/a5b8c569bf593b0140bde72ea885a803b82086995367bf2037de0159d924/pygments-2.19.2.tar.gz", hash = "sha256:636cb2477cec7f8952536970bc533bc43743542f70392ae026374600add5b887", size = 4968631, upload-time = "2025-06-21T13:39:12.283Z" }
|
|
79
|
+
wheels = [
|
|
80
|
+
{ url = "https://files.pythonhosted.org/packages/c7/21/705964c7812476f378728bdf590ca4b771ec72385c533964653c68e86bdc/pygments-2.19.2-py3-none-any.whl", hash = "sha256:86540386c03d588bb81d44bc3928634ff26449851e99741617ecb9037ee5ec0b", size = 1225217, upload-time = "2025-06-21T13:39:07.939Z" },
|
|
81
|
+
]
|
|
82
|
+
|
|
83
|
+
[[package]]
|
|
84
|
+
name = "pytest"
|
|
85
|
+
version = "9.0.2"
|
|
86
|
+
source = { registry = "https://pypi.org/simple" }
|
|
87
|
+
dependencies = [
|
|
88
|
+
{ name = "colorama", marker = "sys_platform == 'win32'" },
|
|
89
|
+
{ name = "exceptiongroup", marker = "python_full_version < '3.11'" },
|
|
90
|
+
{ name = "iniconfig" },
|
|
91
|
+
{ name = "packaging" },
|
|
92
|
+
{ name = "pluggy" },
|
|
93
|
+
{ name = "pygments" },
|
|
94
|
+
{ name = "tomli", marker = "python_full_version < '3.11'" },
|
|
95
|
+
]
|
|
96
|
+
sdist = { url = "https://files.pythonhosted.org/packages/d1/db/7ef3487e0fb0049ddb5ce41d3a49c235bf9ad299b6a25d5780a89f19230f/pytest-9.0.2.tar.gz", hash = "sha256:75186651a92bd89611d1d9fc20f0b4345fd827c41ccd5c299a868a05d70edf11", size = 1568901, upload-time = "2025-12-06T21:30:51.014Z" }
|
|
97
|
+
wheels = [
|
|
98
|
+
{ url = "https://files.pythonhosted.org/packages/3b/ab/b3226f0bd7cdcf710fbede2b3548584366da3b19b5021e74f5bde2a8fa3f/pytest-9.0.2-py3-none-any.whl", hash = "sha256:711ffd45bf766d5264d487b917733b453d917afd2b0ad65223959f59089f875b", size = 374801, upload-time = "2025-12-06T21:30:49.154Z" },
|
|
99
|
+
]
|
|
100
|
+
|
|
101
|
+
[[package]]
|
|
102
|
+
name = "ruff"
|
|
103
|
+
version = "0.14.11"
|
|
104
|
+
source = { registry = "https://pypi.org/simple" }
|
|
105
|
+
sdist = { url = "https://files.pythonhosted.org/packages/d4/77/9a7fe084d268f8855d493e5031ea03fa0af8cc05887f638bf1c4e3363eb8/ruff-0.14.11.tar.gz", hash = "sha256:f6dc463bfa5c07a59b1ff2c3b9767373e541346ea105503b4c0369c520a66958", size = 5993417, upload-time = "2026-01-08T19:11:58.322Z" }
|
|
106
|
+
wheels = [
|
|
107
|
+
{ url = "https://files.pythonhosted.org/packages/f0/a6/a4c40a5aaa7e331f245d2dc1ac8ece306681f52b636b40ef87c88b9f7afd/ruff-0.14.11-py3-none-linux_armv6l.whl", hash = "sha256:f6ff2d95cbd335841a7217bdfd9c1d2e44eac2c584197ab1385579d55ff8830e", size = 12951208, upload-time = "2026-01-08T19:12:09.218Z" },
|
|
108
|
+
{ url = "https://files.pythonhosted.org/packages/5c/5c/360a35cb7204b328b685d3129c08aca24765ff92b5a7efedbdd6c150d555/ruff-0.14.11-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:6f6eb5c1c8033680f4172ea9c8d3706c156223010b8b97b05e82c59bdc774ee6", size = 13330075, upload-time = "2026-01-08T19:12:02.549Z" },
|
|
109
|
+
{ url = "https://files.pythonhosted.org/packages/1b/9e/0cc2f1be7a7d33cae541824cf3f95b4ff40d03557b575912b5b70273c9ec/ruff-0.14.11-py3-none-macosx_11_0_arm64.whl", hash = "sha256:f2fc34cc896f90080fca01259f96c566f74069a04b25b6205d55379d12a6855e", size = 12257809, upload-time = "2026-01-08T19:12:00.366Z" },
|
|
110
|
+
{ url = "https://files.pythonhosted.org/packages/a7/e5/5faab97c15bb75228d9f74637e775d26ac703cc2b4898564c01ab3637c02/ruff-0.14.11-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:53386375001773ae812b43205d6064dae49ff0968774e6befe16a994fc233caa", size = 12678447, upload-time = "2026-01-08T19:12:13.899Z" },
|
|
111
|
+
{ url = "https://files.pythonhosted.org/packages/1b/33/e9767f60a2bef779fb5855cab0af76c488e0ce90f7bb7b8a45c8a2ba4178/ruff-0.14.11-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:a697737dce1ca97a0a55b5ff0434ee7205943d4874d638fe3ae66166ff46edbe", size = 12758560, upload-time = "2026-01-08T19:11:42.55Z" },
|
|
112
|
+
{ url = "https://files.pythonhosted.org/packages/eb/84/4c6cf627a21462bb5102f7be2a320b084228ff26e105510cd2255ea868e5/ruff-0.14.11-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6845ca1da8ab81ab1dce755a32ad13f1db72e7fba27c486d5d90d65e04d17b8f", size = 13599296, upload-time = "2026-01-08T19:11:30.371Z" },
|
|
113
|
+
{ url = "https://files.pythonhosted.org/packages/88/e1/92b5ed7ea66d849f6157e695dc23d5d6d982bd6aa8d077895652c38a7cae/ruff-0.14.11-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:e36ce2fd31b54065ec6f76cb08d60159e1b32bdf08507862e32f47e6dde8bcbf", size = 15048981, upload-time = "2026-01-08T19:12:04.742Z" },
|
|
114
|
+
{ url = "https://files.pythonhosted.org/packages/61/df/c1bd30992615ac17c2fb64b8a7376ca22c04a70555b5d05b8f717163cf9f/ruff-0.14.11-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:590bcc0e2097ecf74e62a5c10a6b71f008ad82eb97b0a0079e85defe19fe74d9", size = 14633183, upload-time = "2026-01-08T19:11:40.069Z" },
|
|
115
|
+
{ url = "https://files.pythonhosted.org/packages/04/e9/fe552902f25013dd28a5428a42347d9ad20c4b534834a325a28305747d64/ruff-0.14.11-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:53fe71125fc158210d57fe4da26e622c9c294022988d08d9347ec1cf782adafe", size = 14050453, upload-time = "2026-01-08T19:11:37.555Z" },
|
|
116
|
+
{ url = "https://files.pythonhosted.org/packages/ae/93/f36d89fa021543187f98991609ce6e47e24f35f008dfe1af01379d248a41/ruff-0.14.11-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a35c9da08562f1598ded8470fcfef2afb5cf881996e6c0a502ceb61f4bc9c8a3", size = 13757889, upload-time = "2026-01-08T19:12:07.094Z" },
|
|
117
|
+
{ url = "https://files.pythonhosted.org/packages/b7/9f/c7fb6ecf554f28709a6a1f2a7f74750d400979e8cd47ed29feeaa1bd4db8/ruff-0.14.11-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:0f3727189a52179393ecf92ec7057c2210203e6af2676f08d92140d3e1ee72c1", size = 13955832, upload-time = "2026-01-08T19:11:55.064Z" },
|
|
118
|
+
{ url = "https://files.pythonhosted.org/packages/db/a0/153315310f250f76900a98278cf878c64dfb6d044e184491dd3289796734/ruff-0.14.11-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:eb09f849bd37147a789b85995ff734a6c4a095bed5fd1608c4f56afc3634cde2", size = 12586522, upload-time = "2026-01-08T19:11:35.356Z" },
|
|
119
|
+
{ url = "https://files.pythonhosted.org/packages/2f/2b/a73a2b6e6d2df1d74bf2b78098be1572191e54bec0e59e29382d13c3adc5/ruff-0.14.11-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:c61782543c1231bf71041461c1f28c64b961d457d0f238ac388e2ab173d7ecb7", size = 12724637, upload-time = "2026-01-08T19:11:47.796Z" },
|
|
120
|
+
{ url = "https://files.pythonhosted.org/packages/f0/41/09100590320394401cd3c48fc718a8ba71c7ddb1ffd07e0ad6576b3a3df2/ruff-0.14.11-py3-none-musllinux_1_2_i686.whl", hash = "sha256:82ff352ea68fb6766140381748e1f67f83c39860b6446966cff48a315c3e2491", size = 13145837, upload-time = "2026-01-08T19:11:32.87Z" },
|
|
121
|
+
{ url = "https://files.pythonhosted.org/packages/3b/d8/e035db859d1d3edf909381eb8ff3e89a672d6572e9454093538fe6f164b0/ruff-0.14.11-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:728e56879df4ca5b62a9dde2dd0eb0edda2a55160c0ea28c4025f18c03f86984", size = 13850469, upload-time = "2026-01-08T19:12:11.694Z" },
|
|
122
|
+
{ url = "https://files.pythonhosted.org/packages/4e/02/bb3ff8b6e6d02ce9e3740f4c17dfbbfb55f34c789c139e9cd91985f356c7/ruff-0.14.11-py3-none-win32.whl", hash = "sha256:337c5dd11f16ee52ae217757d9b82a26400be7efac883e9e852646f1557ed841", size = 12851094, upload-time = "2026-01-08T19:11:45.163Z" },
|
|
123
|
+
{ url = "https://files.pythonhosted.org/packages/58/f1/90ddc533918d3a2ad628bc3044cdfc094949e6d4b929220c3f0eb8a1c998/ruff-0.14.11-py3-none-win_amd64.whl", hash = "sha256:f981cea63d08456b2c070e64b79cb62f951aa1305282974d4d5216e6e0178ae6", size = 14001379, upload-time = "2026-01-08T19:11:52.591Z" },
|
|
124
|
+
{ url = "https://files.pythonhosted.org/packages/c4/1c/1dbe51782c0e1e9cfce1d1004752672d2d4629ea46945d19d731ad772b3b/ruff-0.14.11-py3-none-win_arm64.whl", hash = "sha256:649fb6c9edd7f751db276ef42df1f3df41c38d67d199570ae2a7bd6cbc3590f0", size = 12938644, upload-time = "2026-01-08T19:11:50.027Z" },
|
|
125
|
+
]
|
|
126
|
+
|
|
127
|
+
[[package]]
|
|
128
|
+
name = "tomli"
|
|
129
|
+
version = "2.4.0"
|
|
130
|
+
source = { registry = "https://pypi.org/simple" }
|
|
131
|
+
sdist = { url = "https://files.pythonhosted.org/packages/82/30/31573e9457673ab10aa432461bee537ce6cef177667deca369efb79df071/tomli-2.4.0.tar.gz", hash = "sha256:aa89c3f6c277dd275d8e243ad24f3b5e701491a860d5121f2cdd399fbb31fc9c", size = 17477, upload-time = "2026-01-11T11:22:38.165Z" }
|
|
132
|
+
wheels = [
|
|
133
|
+
{ url = "https://files.pythonhosted.org/packages/3c/d9/3dc2289e1f3b32eb19b9785b6a006b28ee99acb37d1d47f78d4c10e28bf8/tomli-2.4.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b5ef256a3fd497d4973c11bf142e9ed78b150d36f5773f1ca6088c230ffc5867", size = 153663, upload-time = "2026-01-11T11:21:45.27Z" },
|
|
134
|
+
{ url = "https://files.pythonhosted.org/packages/51/32/ef9f6845e6b9ca392cd3f64f9ec185cc6f09f0a2df3db08cbe8809d1d435/tomli-2.4.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:5572e41282d5268eb09a697c89a7bee84fae66511f87533a6f88bd2f7b652da9", size = 148469, upload-time = "2026-01-11T11:21:46.873Z" },
|
|
135
|
+
{ url = "https://files.pythonhosted.org/packages/d6/c2/506e44cce89a8b1b1e047d64bd495c22c9f71f21e05f380f1a950dd9c217/tomli-2.4.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:551e321c6ba03b55676970b47cb1b73f14a0a4dce6a3e1a9458fd6d921d72e95", size = 236039, upload-time = "2026-01-11T11:21:48.503Z" },
|
|
136
|
+
{ url = "https://files.pythonhosted.org/packages/b3/40/e1b65986dbc861b7e986e8ec394598187fa8aee85b1650b01dd925ca0be8/tomli-2.4.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:5e3f639a7a8f10069d0e15408c0b96a2a828cfdec6fca05296ebcdcc28ca7c76", size = 243007, upload-time = "2026-01-11T11:21:49.456Z" },
|
|
137
|
+
{ url = "https://files.pythonhosted.org/packages/9c/6f/6e39ce66b58a5b7ae572a0f4352ff40c71e8573633deda43f6a379d56b3e/tomli-2.4.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:1b168f2731796b045128c45982d3a4874057626da0e2ef1fdd722848b741361d", size = 240875, upload-time = "2026-01-11T11:21:50.755Z" },
|
|
138
|
+
{ url = "https://files.pythonhosted.org/packages/aa/ad/cb089cb190487caa80204d503c7fd0f4d443f90b95cf4ef5cf5aa0f439b0/tomli-2.4.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:133e93646ec4300d651839d382d63edff11d8978be23da4cc106f5a18b7d0576", size = 246271, upload-time = "2026-01-11T11:21:51.81Z" },
|
|
139
|
+
{ url = "https://files.pythonhosted.org/packages/0b/63/69125220e47fd7a3a27fd0de0c6398c89432fec41bc739823bcc66506af6/tomli-2.4.0-cp311-cp311-win32.whl", hash = "sha256:b6c78bdf37764092d369722d9946cb65b8767bfa4110f902a1b2542d8d173c8a", size = 96770, upload-time = "2026-01-11T11:21:52.647Z" },
|
|
140
|
+
{ url = "https://files.pythonhosted.org/packages/1e/0d/a22bb6c83f83386b0008425a6cd1fa1c14b5f3dd4bad05e98cf3dbbf4a64/tomli-2.4.0-cp311-cp311-win_amd64.whl", hash = "sha256:d3d1654e11d724760cdb37a3d7691f0be9db5fbdaef59c9f532aabf87006dbaa", size = 107626, upload-time = "2026-01-11T11:21:53.459Z" },
|
|
141
|
+
{ url = "https://files.pythonhosted.org/packages/2f/6d/77be674a3485e75cacbf2ddba2b146911477bd887dda9d8c9dfb2f15e871/tomli-2.4.0-cp311-cp311-win_arm64.whl", hash = "sha256:cae9c19ed12d4e8f3ebf46d1a75090e4c0dc16271c5bce1c833ac168f08fb614", size = 94842, upload-time = "2026-01-11T11:21:54.831Z" },
|
|
142
|
+
{ url = "https://files.pythonhosted.org/packages/3c/43/7389a1869f2f26dba52404e1ef13b4784b6b37dac93bac53457e3ff24ca3/tomli-2.4.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:920b1de295e72887bafa3ad9f7a792f811847d57ea6b1215154030cf131f16b1", size = 154894, upload-time = "2026-01-11T11:21:56.07Z" },
|
|
143
|
+
{ url = "https://files.pythonhosted.org/packages/e9/05/2f9bf110b5294132b2edf13fe6ca6ae456204f3d749f623307cbb7a946f2/tomli-2.4.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:7d6d9a4aee98fac3eab4952ad1d73aee87359452d1c086b5ceb43ed02ddb16b8", size = 149053, upload-time = "2026-01-11T11:21:57.467Z" },
|
|
144
|
+
{ url = "https://files.pythonhosted.org/packages/e8/41/1eda3ca1abc6f6154a8db4d714a4d35c4ad90adc0bcf700657291593fbf3/tomli-2.4.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:36b9d05b51e65b254ea6c2585b59d2c4cb91c8a3d91d0ed0f17591a29aaea54a", size = 243481, upload-time = "2026-01-11T11:21:58.661Z" },
|
|
145
|
+
{ url = "https://files.pythonhosted.org/packages/d2/6d/02ff5ab6c8868b41e7d4b987ce2b5f6a51d3335a70aa144edd999e055a01/tomli-2.4.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:1c8a885b370751837c029ef9bc014f27d80840e48bac415f3412e6593bbc18c1", size = 251720, upload-time = "2026-01-11T11:22:00.178Z" },
|
|
146
|
+
{ url = "https://files.pythonhosted.org/packages/7b/57/0405c59a909c45d5b6f146107c6d997825aa87568b042042f7a9c0afed34/tomli-2.4.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:8768715ffc41f0008abe25d808c20c3d990f42b6e2e58305d5da280ae7d1fa3b", size = 247014, upload-time = "2026-01-11T11:22:01.238Z" },
|
|
147
|
+
{ url = "https://files.pythonhosted.org/packages/2c/0e/2e37568edd944b4165735687cbaf2fe3648129e440c26d02223672ee0630/tomli-2.4.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:7b438885858efd5be02a9a133caf5812b8776ee0c969fea02c45e8e3f296ba51", size = 251820, upload-time = "2026-01-11T11:22:02.727Z" },
|
|
148
|
+
{ url = "https://files.pythonhosted.org/packages/5a/1c/ee3b707fdac82aeeb92d1a113f803cf6d0f37bdca0849cb489553e1f417a/tomli-2.4.0-cp312-cp312-win32.whl", hash = "sha256:0408e3de5ec77cc7f81960c362543cbbd91ef883e3138e81b729fc3eea5b9729", size = 97712, upload-time = "2026-01-11T11:22:03.777Z" },
|
|
149
|
+
{ url = "https://files.pythonhosted.org/packages/69/13/c07a9177d0b3bab7913299b9278845fc6eaaca14a02667c6be0b0a2270c8/tomli-2.4.0-cp312-cp312-win_amd64.whl", hash = "sha256:685306e2cc7da35be4ee914fd34ab801a6acacb061b6a7abca922aaf9ad368da", size = 108296, upload-time = "2026-01-11T11:22:04.86Z" },
|
|
150
|
+
{ url = "https://files.pythonhosted.org/packages/18/27/e267a60bbeeee343bcc279bb9e8fbed0cbe224bc7b2a3dc2975f22809a09/tomli-2.4.0-cp312-cp312-win_arm64.whl", hash = "sha256:5aa48d7c2356055feef06a43611fc401a07337d5b006be13a30f6c58f869e3c3", size = 94553, upload-time = "2026-01-11T11:22:05.854Z" },
|
|
151
|
+
{ url = "https://files.pythonhosted.org/packages/34/91/7f65f9809f2936e1f4ce6268ae1903074563603b2a2bd969ebbda802744f/tomli-2.4.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:84d081fbc252d1b6a982e1870660e7330fb8f90f676f6e78b052ad4e64714bf0", size = 154915, upload-time = "2026-01-11T11:22:06.703Z" },
|
|
152
|
+
{ url = "https://files.pythonhosted.org/packages/20/aa/64dd73a5a849c2e8f216b755599c511badde80e91e9bc2271baa7b2cdbb1/tomli-2.4.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:9a08144fa4cba33db5255f9b74f0b89888622109bd2776148f2597447f92a94e", size = 149038, upload-time = "2026-01-11T11:22:07.56Z" },
|
|
153
|
+
{ url = "https://files.pythonhosted.org/packages/9e/8a/6d38870bd3d52c8d1505ce054469a73f73a0fe62c0eaf5dddf61447e32fa/tomli-2.4.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:c73add4bb52a206fd0c0723432db123c0c75c280cbd67174dd9d2db228ebb1b4", size = 242245, upload-time = "2026-01-11T11:22:08.344Z" },
|
|
154
|
+
{ url = "https://files.pythonhosted.org/packages/59/bb/8002fadefb64ab2669e5b977df3f5e444febea60e717e755b38bb7c41029/tomli-2.4.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:1fb2945cbe303b1419e2706e711b7113da57b7db31ee378d08712d678a34e51e", size = 250335, upload-time = "2026-01-11T11:22:09.951Z" },
|
|
155
|
+
{ url = "https://files.pythonhosted.org/packages/a5/3d/4cdb6f791682b2ea916af2de96121b3cb1284d7c203d97d92d6003e91c8d/tomli-2.4.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:bbb1b10aa643d973366dc2cb1ad94f99c1726a02343d43cbc011edbfac579e7c", size = 245962, upload-time = "2026-01-11T11:22:11.27Z" },
|
|
156
|
+
{ url = "https://files.pythonhosted.org/packages/f2/4a/5f25789f9a460bd858ba9756ff52d0830d825b458e13f754952dd15fb7bb/tomli-2.4.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:4cbcb367d44a1f0c2be408758b43e1ffb5308abe0ea222897d6bfc8e8281ef2f", size = 250396, upload-time = "2026-01-11T11:22:12.325Z" },
|
|
157
|
+
{ url = "https://files.pythonhosted.org/packages/aa/2f/b73a36fea58dfa08e8b3a268750e6853a6aac2a349241a905ebd86f3047a/tomli-2.4.0-cp313-cp313-win32.whl", hash = "sha256:7d49c66a7d5e56ac959cb6fc583aff0651094ec071ba9ad43df785abc2320d86", size = 97530, upload-time = "2026-01-11T11:22:13.865Z" },
|
|
158
|
+
{ url = "https://files.pythonhosted.org/packages/3b/af/ca18c134b5d75de7e8dc551c5234eaba2e8e951f6b30139599b53de9c187/tomli-2.4.0-cp313-cp313-win_amd64.whl", hash = "sha256:3cf226acb51d8f1c394c1b310e0e0e61fecdd7adcb78d01e294ac297dd2e7f87", size = 108227, upload-time = "2026-01-11T11:22:15.224Z" },
|
|
159
|
+
{ url = "https://files.pythonhosted.org/packages/22/c3/b386b832f209fee8073c8138ec50f27b4460db2fdae9ffe022df89a57f9b/tomli-2.4.0-cp313-cp313-win_arm64.whl", hash = "sha256:d20b797a5c1ad80c516e41bc1fb0443ddb5006e9aaa7bda2d71978346aeb9132", size = 94748, upload-time = "2026-01-11T11:22:16.009Z" },
|
|
160
|
+
{ url = "https://files.pythonhosted.org/packages/f3/c4/84047a97eb1004418bc10bdbcfebda209fca6338002eba2dc27cc6d13563/tomli-2.4.0-cp314-cp314-macosx_10_15_x86_64.whl", hash = "sha256:26ab906a1eb794cd4e103691daa23d95c6919cc2fa9160000ac02370cc9dd3f6", size = 154725, upload-time = "2026-01-11T11:22:17.269Z" },
|
|
161
|
+
{ url = "https://files.pythonhosted.org/packages/a8/5d/d39038e646060b9d76274078cddf146ced86dc2b9e8bbf737ad5983609a0/tomli-2.4.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:20cedb4ee43278bc4f2fee6cb50daec836959aadaf948db5172e776dd3d993fc", size = 148901, upload-time = "2026-01-11T11:22:18.287Z" },
|
|
162
|
+
{ url = "https://files.pythonhosted.org/packages/73/e5/383be1724cb30f4ce44983d249645684a48c435e1cd4f8b5cded8a816d3c/tomli-2.4.0-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:39b0b5d1b6dd03684b3fb276407ebed7090bbec989fa55838c98560c01113b66", size = 243375, upload-time = "2026-01-11T11:22:19.154Z" },
|
|
163
|
+
{ url = "https://files.pythonhosted.org/packages/31/f0/bea80c17971c8d16d3cc109dc3585b0f2ce1036b5f4a8a183789023574f2/tomli-2.4.0-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:a26d7ff68dfdb9f87a016ecfd1e1c2bacbe3108f4e0f8bcd2228ef9a766c787d", size = 250639, upload-time = "2026-01-11T11:22:20.168Z" },
|
|
164
|
+
{ url = "https://files.pythonhosted.org/packages/2c/8f/2853c36abbb7608e3f945d8a74e32ed3a74ee3a1f468f1ffc7d1cb3abba6/tomli-2.4.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:20ffd184fb1df76a66e34bd1b36b4a4641bd2b82954befa32fe8163e79f1a702", size = 246897, upload-time = "2026-01-11T11:22:21.544Z" },
|
|
165
|
+
{ url = "https://files.pythonhosted.org/packages/49/f0/6c05e3196ed5337b9fe7ea003e95fd3819a840b7a0f2bf5a408ef1dad8ed/tomli-2.4.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:75c2f8bbddf170e8effc98f5e9084a8751f8174ea6ccf4fca5398436e0320bc8", size = 254697, upload-time = "2026-01-11T11:22:23.058Z" },
|
|
166
|
+
{ url = "https://files.pythonhosted.org/packages/f3/f5/2922ef29c9f2951883525def7429967fc4d8208494e5ab524234f06b688b/tomli-2.4.0-cp314-cp314-win32.whl", hash = "sha256:31d556d079d72db7c584c0627ff3a24c5d3fb4f730221d3444f3efb1b2514776", size = 98567, upload-time = "2026-01-11T11:22:24.033Z" },
|
|
167
|
+
{ url = "https://files.pythonhosted.org/packages/7b/31/22b52e2e06dd2a5fdbc3ee73226d763b184ff21fc24e20316a44ccc4d96b/tomli-2.4.0-cp314-cp314-win_amd64.whl", hash = "sha256:43e685b9b2341681907759cf3a04e14d7104b3580f808cfde1dfdb60ada85475", size = 108556, upload-time = "2026-01-11T11:22:25.378Z" },
|
|
168
|
+
{ url = "https://files.pythonhosted.org/packages/48/3d/5058dff3255a3d01b705413f64f4306a141a8fd7a251e5a495e3f192a998/tomli-2.4.0-cp314-cp314-win_arm64.whl", hash = "sha256:3d895d56bd3f82ddd6faaff993c275efc2ff38e52322ea264122d72729dca2b2", size = 96014, upload-time = "2026-01-11T11:22:26.138Z" },
|
|
169
|
+
{ url = "https://files.pythonhosted.org/packages/b8/4e/75dab8586e268424202d3a1997ef6014919c941b50642a1682df43204c22/tomli-2.4.0-cp314-cp314t-macosx_10_15_x86_64.whl", hash = "sha256:5b5807f3999fb66776dbce568cc9a828544244a8eb84b84b9bafc080c99597b9", size = 163339, upload-time = "2026-01-11T11:22:27.143Z" },
|
|
170
|
+
{ url = "https://files.pythonhosted.org/packages/06/e3/b904d9ab1016829a776d97f163f183a48be6a4deb87304d1e0116a349519/tomli-2.4.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:c084ad935abe686bd9c898e62a02a19abfc9760b5a79bc29644463eaf2840cb0", size = 159490, upload-time = "2026-01-11T11:22:28.399Z" },
|
|
171
|
+
{ url = "https://files.pythonhosted.org/packages/e3/5a/fc3622c8b1ad823e8ea98a35e3c632ee316d48f66f80f9708ceb4f2a0322/tomli-2.4.0-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:0f2e3955efea4d1cfbcb87bc321e00dc08d2bcb737fd1d5e398af111d86db5df", size = 269398, upload-time = "2026-01-11T11:22:29.345Z" },
|
|
172
|
+
{ url = "https://files.pythonhosted.org/packages/fd/33/62bd6152c8bdd4c305ad9faca48f51d3acb2df1f8791b1477d46ff86e7f8/tomli-2.4.0-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0e0fe8a0b8312acf3a88077a0802565cb09ee34107813bba1c7cd591fa6cfc8d", size = 276515, upload-time = "2026-01-11T11:22:30.327Z" },
|
|
173
|
+
{ url = "https://files.pythonhosted.org/packages/4b/ff/ae53619499f5235ee4211e62a8d7982ba9e439a0fb4f2f351a93d67c1dd2/tomli-2.4.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:413540dce94673591859c4c6f794dfeaa845e98bf35d72ed59636f869ef9f86f", size = 273806, upload-time = "2026-01-11T11:22:32.56Z" },
|
|
174
|
+
{ url = "https://files.pythonhosted.org/packages/47/71/cbca7787fa68d4d0a9f7072821980b39fbb1b6faeb5f5cf02f4a5559fa28/tomli-2.4.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:0dc56fef0e2c1c470aeac5b6ca8cc7b640bb93e92d9803ddaf9ea03e198f5b0b", size = 281340, upload-time = "2026-01-11T11:22:33.505Z" },
|
|
175
|
+
{ url = "https://files.pythonhosted.org/packages/f5/00/d595c120963ad42474cf6ee7771ad0d0e8a49d0f01e29576ee9195d9ecdf/tomli-2.4.0-cp314-cp314t-win32.whl", hash = "sha256:d878f2a6707cc9d53a1be1414bbb419e629c3d6e67f69230217bb663e76b5087", size = 108106, upload-time = "2026-01-11T11:22:34.451Z" },
|
|
176
|
+
{ url = "https://files.pythonhosted.org/packages/de/69/9aa0c6a505c2f80e519b43764f8b4ba93b5a0bbd2d9a9de6e2b24271b9a5/tomli-2.4.0-cp314-cp314t-win_amd64.whl", hash = "sha256:2add28aacc7425117ff6364fe9e06a183bb0251b03f986df0e78e974047571fd", size = 120504, upload-time = "2026-01-11T11:22:35.764Z" },
|
|
177
|
+
{ url = "https://files.pythonhosted.org/packages/b3/9f/f1668c281c58cfae01482f7114a4b88d345e4c140386241a1a24dcc9e7bc/tomli-2.4.0-cp314-cp314t-win_arm64.whl", hash = "sha256:2b1e3b80e1d5e52e40e9b924ec43d81570f0e7d09d11081b797bc4692765a3d4", size = 99561, upload-time = "2026-01-11T11:22:36.624Z" },
|
|
178
|
+
{ url = "https://files.pythonhosted.org/packages/23/d1/136eb2cb77520a31e1f64cbae9d33ec6df0d78bdf4160398e86eec8a8754/tomli-2.4.0-py3-none-any.whl", hash = "sha256:1f776e7d669ebceb01dee46484485f43a4048746235e683bcdffacdf1fb4785a", size = 14477, upload-time = "2026-01-11T11:22:37.446Z" },
|
|
179
|
+
]
|
|
180
|
+
|
|
181
|
+
[[package]]
|
|
182
|
+
name = "ty"
|
|
183
|
+
version = "0.0.11"
|
|
184
|
+
source = { registry = "https://pypi.org/simple" }
|
|
185
|
+
sdist = { url = "https://files.pythonhosted.org/packages/bc/45/5ae578480168d4b3c08cf8e5eac3caf8eb7acdb1a06a9bed7519564bd9b4/ty-0.0.11.tar.gz", hash = "sha256:ebcbc7d646847cb6610de1da4ffc849d8b800e29fd1e9ebb81ba8f3fbac88c25", size = 4920340, upload-time = "2026-01-09T21:06:01.592Z" }
|
|
186
|
+
wheels = [
|
|
187
|
+
{ url = "https://files.pythonhosted.org/packages/0f/34/b1d05cdcd01589a8d2e63011e0a1e24dcefdc2a09d024fee3e27755963f6/ty-0.0.11-py3-none-linux_armv6l.whl", hash = "sha256:68f0b8d07b0a2ea7ec63a08ba2624f853e4f9fa1a06fce47fb453fa279dead5a", size = 9521748, upload-time = "2026-01-09T21:06:13.221Z" },
|
|
188
|
+
{ url = "https://files.pythonhosted.org/packages/43/21/f52d93f4b3784b91bfbcabd01b84dc82128f3a9de178536bcf82968f3367/ty-0.0.11-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:cbf82d7ef0618e9ae3cc3c37c33abcfa302c9b3e3b8ff11d71076f98481cb1a8", size = 9454903, upload-time = "2026-01-09T21:06:42.363Z" },
|
|
189
|
+
{ url = "https://files.pythonhosted.org/packages/ad/01/3a563dba8b1255e474c35e1c3810b7589e81ae8c41df401b6a37c8e2cde9/ty-0.0.11-py3-none-macosx_11_0_arm64.whl", hash = "sha256:121987c906e02264c3b511b95cb9f8a3cdd66f3283b8bbab678ca3525652e304", size = 8823417, upload-time = "2026-01-09T21:06:26.315Z" },
|
|
190
|
+
{ url = "https://files.pythonhosted.org/packages/6f/b1/99b87222c05d3a28fb7bbfb85df4efdde8cb6764a24c1b138f3a615283dd/ty-0.0.11-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:999390b6cc045fe5e1b3da1c2c9ae8e8c0def23b69455e7c9191ba9ffd747023", size = 9290785, upload-time = "2026-01-09T21:05:59.028Z" },
|
|
191
|
+
{ url = "https://files.pythonhosted.org/packages/3d/9f/598809a8fff2194f907ba6de07ac3d7b7788342592d8f8b98b1b50c2fb49/ty-0.0.11-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:ed504d78eb613c49be3c848f236b345b6c13dc6bcfc4b202790a60a97e1d8f35", size = 9359392, upload-time = "2026-01-09T21:06:37.459Z" },
|
|
192
|
+
{ url = "https://files.pythonhosted.org/packages/71/3e/aeea2a97b38f3dcd9f8224bf83609848efa4bc2f484085508165567daa7b/ty-0.0.11-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7fedc8b43cc8a9991e0034dd205f957a8380dd29bfce36f2a35b5d321636dfd9", size = 9852973, upload-time = "2026-01-09T21:06:21.245Z" },
|
|
193
|
+
{ url = "https://files.pythonhosted.org/packages/72/40/86173116995e38f954811a86339ac4c00a2d8058cc245d3e4903bc4a132c/ty-0.0.11-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:0808bdfb7efe09881bf70249b85b0498fb8b75fbb036ce251c496c20adb10075", size = 10796113, upload-time = "2026-01-09T21:06:16.034Z" },
|
|
194
|
+
{ url = "https://files.pythonhosted.org/packages/69/71/97c92c401dacae9baa3696163ebe8371635ebf34ba9fda781110d0124857/ty-0.0.11-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:07185b3e38b18c562056dfbc35fb51d866f872977ea1ebcd64ca24a001b5b4f1", size = 10432137, upload-time = "2026-01-09T21:06:07.498Z" },
|
|
195
|
+
{ url = "https://files.pythonhosted.org/packages/18/10/9ab43f3cfc5f7792f6bc97620f54d0a0a81ef700be84ea7f6be330936a99/ty-0.0.11-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b5c72f1ada8eb5be984502a600f71d1a3099e12fb6f3c0607aaba2f86f0e9d80", size = 10240520, upload-time = "2026-01-09T21:06:34.823Z" },
|
|
196
|
+
{ url = "https://files.pythonhosted.org/packages/74/18/8dd4fe6df1fd66f3e83b4798eddb1d8482d9d9b105f25099b76703402ebb/ty-0.0.11-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:25f88e8789072830348cb59b761d5ced70642ed5600673b4bf6a849af71eca8b", size = 9973340, upload-time = "2026-01-09T21:06:39.657Z" },
|
|
197
|
+
{ url = "https://files.pythonhosted.org/packages/e4/0b/fb2301450cf8f2d7164944d6e1e659cac9ec7021556cc173d54947cf8ef4/ty-0.0.11-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:f370e1047a62dcedcd06e2b27e1f0b16c7f8ea2361d9070fcbf0d0d69baaa192", size = 9262101, upload-time = "2026-01-09T21:06:28.989Z" },
|
|
198
|
+
{ url = "https://files.pythonhosted.org/packages/f7/8c/d6374af023541072dee1c8bcfe8242669363a670b7619e6fffcc7415a995/ty-0.0.11-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:52be34047ed6177bfcef9247459a767ec03d775714855e262bca1fb015895e8a", size = 9382756, upload-time = "2026-01-09T21:06:24.097Z" },
|
|
199
|
+
{ url = "https://files.pythonhosted.org/packages/0d/44/edd1e63ffa8d49d720c475c2c1c779084e5efe50493afdc261938705d10a/ty-0.0.11-py3-none-musllinux_1_2_i686.whl", hash = "sha256:b9e5762ccb3778779378020b8d78f936b3f52ea83f18785319cceba3ae85d8e6", size = 9553944, upload-time = "2026-01-09T21:06:18.426Z" },
|
|
200
|
+
{ url = "https://files.pythonhosted.org/packages/35/cd/4afdb0d182d23d07ff287740c4954cc6dde5c3aed150ec3f2a1d72b00f71/ty-0.0.11-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:e9334646ee3095e778e3dbc45fdb2bddfc16acc7804283830ad84991ece16dd7", size = 10060365, upload-time = "2026-01-09T21:06:45.083Z" },
|
|
201
|
+
{ url = "https://files.pythonhosted.org/packages/d1/94/a009ad9d8b359933cfea8721c689c0331189be28650d74dcc6add4d5bb09/ty-0.0.11-py3-none-win32.whl", hash = "sha256:44cfb7bb2d6784bd7ffe7b5d9ea90851d9c4723729c50b5f0732d4b9a2013cfc", size = 9040448, upload-time = "2026-01-09T21:06:32.241Z" },
|
|
202
|
+
{ url = "https://files.pythonhosted.org/packages/df/04/5a5dfd0aec0ea99ead1e824ee6e347fb623c464da7886aa1e3660fb0f36c/ty-0.0.11-py3-none-win_amd64.whl", hash = "sha256:1bb205db92715d4a13343bfd5b0c59ce8c0ca0daa34fb220ec9120fc66ccbda7", size = 9780112, upload-time = "2026-01-09T21:06:04.69Z" },
|
|
203
|
+
{ url = "https://files.pythonhosted.org/packages/ad/07/47d4fccd7bcf5eea1c634d518d6cb233f535a85d0b63fcd66815759e2fa0/ty-0.0.11-py3-none-win_arm64.whl", hash = "sha256:4688bd87b2dc5c85da277bda78daba14af2e66f3dda4d98f3604e3de75519eba", size = 9194038, upload-time = "2026-01-09T21:06:10.152Z" },
|
|
204
|
+
]
|
|
205
|
+
|
|
206
|
+
[[package]]
|
|
207
|
+
name = "typing-extensions"
|
|
208
|
+
version = "4.15.0"
|
|
209
|
+
source = { registry = "https://pypi.org/simple" }
|
|
210
|
+
sdist = { url = "https://files.pythonhosted.org/packages/72/94/1a15dd82efb362ac84269196e94cf00f187f7ed21c242792a923cdb1c61f/typing_extensions-4.15.0.tar.gz", hash = "sha256:0cea48d173cc12fa28ecabc3b837ea3cf6f38c6d1136f85cbaaf598984861466", size = 109391, upload-time = "2025-08-25T13:49:26.313Z" }
|
|
211
|
+
wheels = [
|
|
212
|
+
{ url = "https://files.pythonhosted.org/packages/18/67/36e9267722cc04a6b9f15c7f3441c2363321a3ea07da7ae0c0707beb2a9c/typing_extensions-4.15.0-py3-none-any.whl", hash = "sha256:f0fa19c6845758ab08074a0cfa8b7aecb71c999ca73d62883bc25cc018c4e548", size = 44614, upload-time = "2025-08-25T13:49:24.86Z" },
|
|
213
|
+
]
|