create-ai-tooling-app 0.1.0__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (26) hide show
  1. create_ai_tooling_app-0.1.0/.claude/settings.local.json +7 -0
  2. create_ai_tooling_app-0.1.0/.gitignore +7 -0
  3. create_ai_tooling_app-0.1.0/PKG-INFO +9 -0
  4. create_ai_tooling_app-0.1.0/README.md +61 -0
  5. create_ai_tooling_app-0.1.0/pyproject.toml +21 -0
  6. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/__init__.py +0 -0
  7. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/cli.py +90 -0
  8. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/scaffold.py +57 -0
  9. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/.env.example.jinja +6 -0
  10. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/.gitignore.jinja +7 -0
  11. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/README.md.jinja +48 -0
  12. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/pyproject.toml.jinja +27 -0
  13. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/__init__.py.jinja +0 -0
  14. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/llm/__init__.py.jinja +0 -0
  15. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/llm/client.py.jinja +37 -0
  16. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/main.py.jinja +7 -0
  17. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/models/__init__.py.jinja +0 -0
  18. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/models/request.py.jinja +6 -0
  19. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/models/response.py.jinja +6 -0
  20. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/prompts/__init__.py.jinja +0 -0
  21. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/prompts/base.py.jinja +4 -0
  22. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/routes/__init__.py.jinja +0 -0
  23. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/src/routes/tool.py.jinja +16 -0
  24. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/tests/__init__.py.jinja +0 -0
  25. create_ai_tooling_app-0.1.0/src/create_ai_tooling_app/templates/tests/test_tool.py.jinja +11 -0
  26. create_ai_tooling_app-0.1.0/uv.lock +238 -0
@@ -0,0 +1,7 @@
1
+ {
2
+ "permissions": {
3
+ "allow": [
4
+ "Bash(uv sync:*)"
5
+ ]
6
+ }
7
+ }
@@ -0,0 +1,7 @@
1
+ .venv/
2
+ __pycache__/
3
+ *.pyc
4
+ *.pyo
5
+ .pytest_cache/
6
+ dist/
7
+ *.egg-info/
@@ -0,0 +1,9 @@
1
+ Metadata-Version: 2.4
2
+ Name: create-ai-tooling-app
3
+ Version: 0.1.0
4
+ Summary: CLI to scaffold Python AI tooling API projects
5
+ Requires-Python: >=3.11
6
+ Requires-Dist: jinja2>=3.1.0
7
+ Requires-Dist: questionary>=2.0.1
8
+ Requires-Dist: rich>=13.0.0
9
+ Requires-Dist: typer>=0.12.0
@@ -0,0 +1,61 @@
1
+ # create-ai-tooling-app
2
+
3
+ A CLI that scaffolds Python AI tooling API projects — batteries included, minimal by design.
4
+
5
+ ## Usage
6
+
7
+ ```bash
8
+ uvx create-ai-tooling-app
9
+ ```
10
+
11
+ Or, if working locally:
12
+
13
+ ```bash
14
+ uv run create-ai-tooling-app [output-dir]
15
+ ```
16
+
17
+ You'll be prompted for:
18
+ - **Project name** — used for the directory and package name
19
+ - **LLM provider** — Anthropic (Claude) or OpenAI
20
+
21
+ ## What gets generated
22
+
23
+ ```
24
+ my-project/
25
+ ├── pyproject.toml # uv-managed, includes fastapi, pydantic, and your chosen LLM SDK
26
+ ├── .env.example # API key placeholder
27
+ ├── .gitignore
28
+ ├── README.md
29
+ └── src/
30
+ │ └── my_project/
31
+ │ ├── main.py # FastAPI app entrypoint
32
+ │ ├── models/
33
+ │ │ ├── request.py # Pydantic input model
34
+ │ │ └── response.py # Pydantic output model
35
+ │ ├── prompts/
36
+ │ │ └── base.py # Prompt builder
37
+ │ ├── llm/
38
+ │ │ └── client.py # LLM client wrapper
39
+ │ └── routes/
40
+ │ └── tool.py # POST /run endpoint
41
+ └── tests/
42
+ └── test_tool.py
43
+ ```
44
+
45
+ The request flow:
46
+
47
+ ```
48
+ POST /run
49
+ → validate input (Pydantic ToolRequest)
50
+ → build prompt (prompts/base.py)
51
+ → call LLM (llm/client.py)
52
+ → validate output (Pydantic ToolResponse)
53
+ → return response
54
+ ```
55
+
56
+ ## Development
57
+
58
+ ```bash
59
+ uv sync
60
+ uv run create-ai-tooling-app
61
+ ```
@@ -0,0 +1,21 @@
1
+ [project]
2
+ name = "create-ai-tooling-app"
3
+ version = "0.1.0"
4
+ description = "CLI to scaffold Python AI tooling API projects"
5
+ requires-python = ">=3.11"
6
+ dependencies = [
7
+ "typer>=0.12.0",
8
+ "questionary>=2.0.1",
9
+ "jinja2>=3.1.0",
10
+ "rich>=13.0.0",
11
+ ]
12
+
13
+ [project.scripts]
14
+ create-ai-tooling-app = "create_ai_tooling_app.cli:app"
15
+
16
+ [build-system]
17
+ requires = ["hatchling"]
18
+ build-backend = "hatchling.build"
19
+
20
+ [tool.hatch.build.targets.wheel]
21
+ packages = ["src/create_ai_tooling_app"]
@@ -0,0 +1,90 @@
1
+ import re
2
+ import sys
3
+ from pathlib import Path
4
+
5
+ import questionary
6
+ import typer
7
+ from rich.console import Console
8
+ from rich.panel import Panel
9
+
10
+ from .scaffold import scaffold_project
11
+
12
+ app = typer.Typer(add_completion=False)
13
+ console = Console()
14
+
15
+
16
+ def to_slug(name: str) -> str:
17
+ slug = name.lower().strip()
18
+ slug = re.sub(r"[^\w\s-]", "", slug)
19
+ slug = re.sub(r"[\s-]+", "_", slug)
20
+ return slug
21
+
22
+
23
+ @app.command()
24
+ def main(
25
+ output_dir: Path = typer.Argument(
26
+ default=None,
27
+ help="Directory to create the project in. Defaults to current directory.",
28
+ ),
29
+ ) -> None:
30
+ console.print(
31
+ Panel(
32
+ "[bold cyan]create-ai-tooling-app[/bold cyan]\n"
33
+ "[dim]Scaffold a Python AI tooling API project[/dim]",
34
+ expand=False,
35
+ )
36
+ )
37
+
38
+ project_name: str = questionary.text(
39
+ "Project name:",
40
+ validate=lambda v: bool(v.strip()) or "Project name cannot be empty.",
41
+ ).ask()
42
+
43
+ if project_name is None:
44
+ sys.exit(0)
45
+
46
+ project_slug = to_slug(project_name)
47
+
48
+ provider: str = questionary.select(
49
+ "LLM provider:",
50
+ choices=[
51
+ questionary.Choice("Anthropic (Claude)", value="anthropic"),
52
+ questionary.Choice("OpenAI", value="openai"),
53
+ ],
54
+ ).ask()
55
+
56
+ if provider is None:
57
+ sys.exit(0)
58
+
59
+ target = (output_dir or Path.cwd()) / project_slug
60
+
61
+ if target.exists():
62
+ overwrite = questionary.confirm(
63
+ f"Directory '{target}' already exists. Overwrite?", default=False
64
+ ).ask()
65
+ if not overwrite:
66
+ console.print("[yellow]Aborted.[/yellow]")
67
+ sys.exit(0)
68
+
69
+ console.print(f"\n[dim]Scaffolding [bold]{project_name}[/bold] in {target}...[/dim]")
70
+
71
+ scaffold_project(
72
+ project_name=project_name,
73
+ project_slug=project_slug,
74
+ provider=provider,
75
+ target=target,
76
+ )
77
+
78
+ api_key_var = "ANTHROPIC_API_KEY" if provider == "anthropic" else "OPENAI_API_KEY"
79
+
80
+ console.print(
81
+ Panel(
82
+ f"[bold green]Done![/bold green] Your project is ready.\n\n"
83
+ f" [bold]cd {project_slug}[/bold]\n"
84
+ f" [bold]cp .env.example .env[/bold] # add your {api_key_var}\n"
85
+ f" [bold]uv sync[/bold]\n"
86
+ f" [bold]uv run fastapi dev src/{project_slug}/main.py[/bold]",
87
+ title="Next steps",
88
+ expand=False,
89
+ )
90
+ )
@@ -0,0 +1,57 @@
1
+ import shutil
2
+ from pathlib import Path
3
+
4
+ from jinja2 import Environment, PackageLoader
5
+
6
+ TEMPLATES_DIR = Path(__file__).parent / "templates"
7
+
8
+
9
+ def scaffold_project(
10
+ project_name: str,
11
+ project_slug: str,
12
+ provider: str,
13
+ target: Path,
14
+ ) -> None:
15
+ ctx = {
16
+ "project_name": project_name,
17
+ "project_slug": project_slug,
18
+ "provider": provider,
19
+ }
20
+
21
+ env = Environment(
22
+ loader=PackageLoader("create_ai_tooling_app", "templates"),
23
+ keep_trailing_newline=True,
24
+ )
25
+
26
+ if target.exists():
27
+ shutil.rmtree(target)
28
+ target.mkdir(parents=True)
29
+
30
+ # Files rendered from Jinja2 templates
31
+ template_files = [
32
+ ("README.md.jinja", "README.md"),
33
+ ("pyproject.toml.jinja", "pyproject.toml"),
34
+ (".env.example.jinja", ".env.example"),
35
+ (".gitignore.jinja", ".gitignore"),
36
+ ("src/__init__.py.jinja", f"src/{project_slug}/__init__.py"),
37
+ ("src/main.py.jinja", f"src/{project_slug}/main.py"),
38
+ ("src/models/__init__.py.jinja", f"src/{project_slug}/models/__init__.py"),
39
+ ("src/models/request.py.jinja", f"src/{project_slug}/models/request.py"),
40
+ ("src/models/response.py.jinja", f"src/{project_slug}/models/response.py"),
41
+ ("src/prompts/__init__.py.jinja", f"src/{project_slug}/prompts/__init__.py"),
42
+ ("src/prompts/base.py.jinja", f"src/{project_slug}/prompts/base.py"),
43
+ ("src/llm/__init__.py.jinja", f"src/{project_slug}/llm/__init__.py"),
44
+ ("src/llm/client.py.jinja", f"src/{project_slug}/llm/client.py"),
45
+ ("src/routes/__init__.py.jinja", f"src/{project_slug}/routes/__init__.py"),
46
+ ("src/routes/tool.py.jinja", f"src/{project_slug}/routes/tool.py"),
47
+ ("tests/__init__.py.jinja", "tests/__init__.py"),
48
+ ("tests/test_tool.py.jinja", "tests/test_tool.py"),
49
+ ]
50
+
51
+ for template_path, output_path in template_files:
52
+ template = env.get_template(template_path)
53
+ rendered = template.render(**ctx)
54
+ out = target / output_path
55
+ out.parent.mkdir(parents=True, exist_ok=True)
56
+ out.write_text(rendered)
57
+
@@ -0,0 +1,6 @@
1
+ {%- if provider == "anthropic" %}
2
+ ANTHROPIC_API_KEY=your-api-key-here
3
+ {%- elif provider == "openai" %}
4
+ OPENAI_API_KEY=your-api-key-here
5
+ {%- endif %}
6
+
@@ -0,0 +1,7 @@
1
+ .env
2
+ .venv/
3
+ __pycache__/
4
+ *.pyc
5
+ *.pyo
6
+ .pytest_cache/
7
+ dist/
@@ -0,0 +1,48 @@
1
+ # {{ project_name }}
2
+
3
+ A Python AI tooling API built with FastAPI, Pydantic, and {{ "Anthropic (Claude)" if provider == "anthropic" else "OpenAI" }}.
4
+
5
+ ## Setup
6
+
7
+ ```bash
8
+ cp .env.example .env # add your {{ "ANTHROPIC_API_KEY" if provider == "anthropic" else "OPENAI_API_KEY" }}
9
+ uv sync
10
+ ```
11
+
12
+ ## Run
13
+
14
+ ```bash
15
+ uv run fastapi dev src/{{ project_slug }}/main.py
16
+ ```
17
+
18
+ The API will be available at `http://localhost:8000`. Interactive docs at `http://localhost:8000/docs`.
19
+
20
+ ## Usage
21
+
22
+ ```bash
23
+ curl -X POST http://localhost:8000/run \
24
+ -H "Content-Type: application/json" \
25
+ -d '{"input": "your input here"}'
26
+ ```
27
+
28
+ ## Project structure
29
+
30
+ ```
31
+ src/{{ project_slug }}/
32
+ ├── main.py # FastAPI app entrypoint
33
+ ├── models/
34
+ │ ├── request.py # Pydantic input model — add your request fields here
35
+ │ └── response.py # Pydantic output model — add your response fields here
36
+ ├── prompts/
37
+ │ └── base.py # Prompt builder — edit your system prompt here
38
+ ├── llm/
39
+ │ └── client.py # LLM client wrapper
40
+ └── routes/
41
+ └── tool.py # POST /run endpoint
42
+ ```
43
+
44
+ ## Tests
45
+
46
+ ```bash
47
+ uv run pytest
48
+ ```
@@ -0,0 +1,27 @@
1
+ [project]
2
+ name = "{{ project_slug }}"
3
+ version = "0.1.0"
4
+ requires-python = ">=3.11"
5
+ dependencies = [
6
+ "fastapi[standard]>=0.111.0",
7
+ "pydantic>=2.0.0",
8
+ "python-dotenv>=1.0.0",
9
+ {%- if provider == "anthropic" %}
10
+ "anthropic>=0.25.0",
11
+ {%- elif provider == "openai" %}
12
+ "openai>=1.0.0",
13
+ {%- endif %}
14
+ ]
15
+
16
+ [project.optional-dependencies]
17
+ dev = [
18
+ "pytest>=8.0.0",
19
+ "httpx>=0.27.0",
20
+ ]
21
+
22
+ [build-system]
23
+ requires = ["hatchling"]
24
+ build-backend = "hatchling.build"
25
+
26
+ [tool.hatch.build.targets.wheel]
27
+ packages = ["src/{{ project_slug }}"]
@@ -0,0 +1,37 @@
1
+ import os
2
+
3
+ from dotenv import load_dotenv
4
+ {%- if provider == "anthropic" %}
5
+ import anthropic
6
+ {%- elif provider == "openai" %}
7
+ import openai
8
+ {%- endif %}
9
+
10
+ load_dotenv()
11
+
12
+
13
+ class LLMClient:
14
+ {%- if provider == "anthropic" %}
15
+ def __init__(self) -> None:
16
+ self.client = anthropic.Anthropic(api_key=os.environ["ANTHROPIC_API_KEY"])
17
+ self.model = "claude-opus-4-6"
18
+
19
+ def run(self, prompt: str) -> str:
20
+ message = self.client.messages.create(
21
+ model=self.model,
22
+ max_tokens=1024,
23
+ messages=[{"role": "user", "content": prompt}],
24
+ )
25
+ return message.content[0].text
26
+ {%- elif provider == "openai" %}
27
+ def __init__(self) -> None:
28
+ self.client = openai.OpenAI(api_key=os.environ["OPENAI_API_KEY"])
29
+ self.model = "gpt-4o"
30
+
31
+ def run(self, prompt: str) -> str:
32
+ response = self.client.chat.completions.create(
33
+ model=self.model,
34
+ messages=[{"role": "user", "content": prompt}],
35
+ )
36
+ return response.choices[0].message.content
37
+ {%- endif %}
@@ -0,0 +1,7 @@
1
+ from fastapi import FastAPI
2
+
3
+ from .routes.tool import router
4
+
5
+ app = FastAPI(title="{{ project_name }}")
6
+
7
+ app.include_router(router)
@@ -0,0 +1,6 @@
1
+ from pydantic import BaseModel
2
+
3
+
4
+ class ToolRequest(BaseModel):
5
+ input: str
6
+ # Add your request fields here
@@ -0,0 +1,6 @@
1
+ from pydantic import BaseModel
2
+
3
+
4
+ class ToolResponse(BaseModel):
5
+ output: str
6
+ # Add your response fields here
@@ -0,0 +1,4 @@
1
+ def build_prompt(input: str) -> str:
2
+ return f"""You are a helpful assistant.
3
+
4
+ {input}"""
@@ -0,0 +1,16 @@
1
+ from fastapi import APIRouter
2
+
3
+ from ..llm.client import LLMClient
4
+ from ..models.request import ToolRequest
5
+ from ..models.response import ToolResponse
6
+ from ..prompts.base import build_prompt
7
+
8
+ router = APIRouter()
9
+ client = LLMClient()
10
+
11
+
12
+ @router.post("/run", response_model=ToolResponse)
13
+ def run(request: ToolRequest) -> ToolResponse:
14
+ prompt = build_prompt(request.input)
15
+ output = client.run(prompt)
16
+ return ToolResponse(output=output)
@@ -0,0 +1,11 @@
1
+ from fastapi.testclient import TestClient
2
+
3
+ from src.{{ project_slug }}.main import app
4
+
5
+ client = TestClient(app)
6
+
7
+
8
+ def test_run():
9
+ response = client.post("/run", json={"input": "Hello"})
10
+ assert response.status_code == 200
11
+ assert "output" in response.json()
@@ -0,0 +1,238 @@
1
+ version = 1
2
+ revision = 3
3
+ requires-python = ">=3.11"
4
+
5
+ [[package]]
6
+ name = "annotated-doc"
7
+ version = "0.0.4"
8
+ source = { registry = "https://pypi.org/simple" }
9
+ sdist = { url = "https://files.pythonhosted.org/packages/57/ba/046ceea27344560984e26a590f90bc7f4a75b06701f653222458922b558c/annotated_doc-0.0.4.tar.gz", hash = "sha256:fbcda96e87e9c92ad167c2e53839e57503ecfda18804ea28102353485033faa4", size = 7288, upload-time = "2025-11-10T22:07:42.062Z" }
10
+ wheels = [
11
+ { url = "https://files.pythonhosted.org/packages/1e/d3/26bf1008eb3d2daa8ef4cacc7f3bfdc11818d111f7e2d0201bc6e3b49d45/annotated_doc-0.0.4-py3-none-any.whl", hash = "sha256:571ac1dc6991c450b25a9c2d84a3705e2ae7a53467b5d111c24fa8baabbed320", size = 5303, upload-time = "2025-11-10T22:07:40.673Z" },
12
+ ]
13
+
14
+ [[package]]
15
+ name = "click"
16
+ version = "8.3.1"
17
+ source = { registry = "https://pypi.org/simple" }
18
+ dependencies = [
19
+ { name = "colorama", marker = "sys_platform == 'win32'" },
20
+ ]
21
+ sdist = { url = "https://files.pythonhosted.org/packages/3d/fa/656b739db8587d7b5dfa22e22ed02566950fbfbcdc20311993483657a5c0/click-8.3.1.tar.gz", hash = "sha256:12ff4785d337a1bb490bb7e9c2b1ee5da3112e94a8622f26a6c77f5d2fc6842a", size = 295065, upload-time = "2025-11-15T20:45:42.706Z" }
22
+ wheels = [
23
+ { url = "https://files.pythonhosted.org/packages/98/78/01c019cdb5d6498122777c1a43056ebb3ebfeef2076d9d026bfe15583b2b/click-8.3.1-py3-none-any.whl", hash = "sha256:981153a64e25f12d547d3426c367a4857371575ee7ad18df2a6183ab0545b2a6", size = 108274, upload-time = "2025-11-15T20:45:41.139Z" },
24
+ ]
25
+
26
+ [[package]]
27
+ name = "colorama"
28
+ version = "0.4.6"
29
+ source = { registry = "https://pypi.org/simple" }
30
+ 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" }
31
+ wheels = [
32
+ { 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" },
33
+ ]
34
+
35
+ [[package]]
36
+ name = "create-ai-tooling-app"
37
+ version = "0.1.0"
38
+ source = { editable = "." }
39
+ dependencies = [
40
+ { name = "jinja2" },
41
+ { name = "questionary" },
42
+ { name = "rich" },
43
+ { name = "typer" },
44
+ ]
45
+
46
+ [package.metadata]
47
+ requires-dist = [
48
+ { name = "jinja2", specifier = ">=3.1.0" },
49
+ { name = "questionary", specifier = ">=2.0.1" },
50
+ { name = "rich", specifier = ">=13.0.0" },
51
+ { name = "typer", specifier = ">=0.12.0" },
52
+ ]
53
+
54
+ [[package]]
55
+ name = "jinja2"
56
+ version = "3.1.6"
57
+ source = { registry = "https://pypi.org/simple" }
58
+ dependencies = [
59
+ { name = "markupsafe" },
60
+ ]
61
+ sdist = { url = "https://files.pythonhosted.org/packages/df/bf/f7da0350254c0ed7c72f3e33cef02e048281fec7ecec5f032d4aac52226b/jinja2-3.1.6.tar.gz", hash = "sha256:0137fb05990d35f1275a587e9aee6d56da821fc83491a0fb838183be43f66d6d", size = 245115, upload-time = "2025-03-05T20:05:02.478Z" }
62
+ wheels = [
63
+ { url = "https://files.pythonhosted.org/packages/62/a1/3d680cbfd5f4b8f15abc1d571870c5fc3e594bb582bc3b64ea099db13e56/jinja2-3.1.6-py3-none-any.whl", hash = "sha256:85ece4451f492d0c13c5dd7c13a64681a86afae63a5f347908daf103ce6d2f67", size = 134899, upload-time = "2025-03-05T20:05:00.369Z" },
64
+ ]
65
+
66
+ [[package]]
67
+ name = "markdown-it-py"
68
+ version = "4.0.0"
69
+ source = { registry = "https://pypi.org/simple" }
70
+ dependencies = [
71
+ { name = "mdurl" },
72
+ ]
73
+ sdist = { url = "https://files.pythonhosted.org/packages/5b/f5/4ec618ed16cc4f8fb3b701563655a69816155e79e24a17b651541804721d/markdown_it_py-4.0.0.tar.gz", hash = "sha256:cb0a2b4aa34f932c007117b194e945bd74e0ec24133ceb5bac59009cda1cb9f3", size = 73070, upload-time = "2025-08-11T12:57:52.854Z" }
74
+ wheels = [
75
+ { url = "https://files.pythonhosted.org/packages/94/54/e7d793b573f298e1c9013b8c4dade17d481164aa517d1d7148619c2cedbf/markdown_it_py-4.0.0-py3-none-any.whl", hash = "sha256:87327c59b172c5011896038353a81343b6754500a08cd7a4973bb48c6d578147", size = 87321, upload-time = "2025-08-11T12:57:51.923Z" },
76
+ ]
77
+
78
+ [[package]]
79
+ name = "markupsafe"
80
+ version = "3.0.3"
81
+ source = { registry = "https://pypi.org/simple" }
82
+ sdist = { url = "https://files.pythonhosted.org/packages/7e/99/7690b6d4034fffd95959cbe0c02de8deb3098cc577c67bb6a24fe5d7caa7/markupsafe-3.0.3.tar.gz", hash = "sha256:722695808f4b6457b320fdc131280796bdceb04ab50fe1795cd540799ebe1698", size = 80313, upload-time = "2025-09-27T18:37:40.426Z" }
83
+ wheels = [
84
+ { url = "https://files.pythonhosted.org/packages/08/db/fefacb2136439fc8dd20e797950e749aa1f4997ed584c62cfb8ef7c2be0e/markupsafe-3.0.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:1cc7ea17a6824959616c525620e387f6dd30fec8cb44f649e31712db02123dad", size = 11631, upload-time = "2025-09-27T18:36:18.185Z" },
85
+ { url = "https://files.pythonhosted.org/packages/e1/2e/5898933336b61975ce9dc04decbc0a7f2fee78c30353c5efba7f2d6ff27a/markupsafe-3.0.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:4bd4cd07944443f5a265608cc6aab442e4f74dff8088b0dfc8238647b8f6ae9a", size = 12058, upload-time = "2025-09-27T18:36:19.444Z" },
86
+ { url = "https://files.pythonhosted.org/packages/1d/09/adf2df3699d87d1d8184038df46a9c80d78c0148492323f4693df54e17bb/markupsafe-3.0.3-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6b5420a1d9450023228968e7e6a9ce57f65d148ab56d2313fcd589eee96a7a50", size = 24287, upload-time = "2025-09-27T18:36:20.768Z" },
87
+ { url = "https://files.pythonhosted.org/packages/30/ac/0273f6fcb5f42e314c6d8cd99effae6a5354604d461b8d392b5ec9530a54/markupsafe-3.0.3-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0bf2a864d67e76e5c9a34dc26ec616a66b9888e25e7b9460e1c76d3293bd9dbf", size = 22940, upload-time = "2025-09-27T18:36:22.249Z" },
88
+ { url = "https://files.pythonhosted.org/packages/19/ae/31c1be199ef767124c042c6c3e904da327a2f7f0cd63a0337e1eca2967a8/markupsafe-3.0.3-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:bc51efed119bc9cfdf792cdeaa4d67e8f6fcccab66ed4bfdd6bde3e59bfcbb2f", size = 21887, upload-time = "2025-09-27T18:36:23.535Z" },
89
+ { url = "https://files.pythonhosted.org/packages/b2/76/7edcab99d5349a4532a459e1fe64f0b0467a3365056ae550d3bcf3f79e1e/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:068f375c472b3e7acbe2d5318dea141359e6900156b5b2ba06a30b169086b91a", size = 23692, upload-time = "2025-09-27T18:36:24.823Z" },
90
+ { url = "https://files.pythonhosted.org/packages/a4/28/6e74cdd26d7514849143d69f0bf2399f929c37dc2b31e6829fd2045b2765/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:7be7b61bb172e1ed687f1754f8e7484f1c8019780f6f6b0786e76bb01c2ae115", size = 21471, upload-time = "2025-09-27T18:36:25.95Z" },
91
+ { url = "https://files.pythonhosted.org/packages/62/7e/a145f36a5c2945673e590850a6f8014318d5577ed7e5920a4b3448e0865d/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f9e130248f4462aaa8e2552d547f36ddadbeaa573879158d721bbd33dfe4743a", size = 22923, upload-time = "2025-09-27T18:36:27.109Z" },
92
+ { url = "https://files.pythonhosted.org/packages/0f/62/d9c46a7f5c9adbeeeda52f5b8d802e1094e9717705a645efc71b0913a0a8/markupsafe-3.0.3-cp311-cp311-win32.whl", hash = "sha256:0db14f5dafddbb6d9208827849fad01f1a2609380add406671a26386cdf15a19", size = 14572, upload-time = "2025-09-27T18:36:28.045Z" },
93
+ { url = "https://files.pythonhosted.org/packages/83/8a/4414c03d3f891739326e1783338e48fb49781cc915b2e0ee052aa490d586/markupsafe-3.0.3-cp311-cp311-win_amd64.whl", hash = "sha256:de8a88e63464af587c950061a5e6a67d3632e36df62b986892331d4620a35c01", size = 15077, upload-time = "2025-09-27T18:36:29.025Z" },
94
+ { url = "https://files.pythonhosted.org/packages/35/73/893072b42e6862f319b5207adc9ae06070f095b358655f077f69a35601f0/markupsafe-3.0.3-cp311-cp311-win_arm64.whl", hash = "sha256:3b562dd9e9ea93f13d53989d23a7e775fdfd1066c33494ff43f5418bc8c58a5c", size = 13876, upload-time = "2025-09-27T18:36:29.954Z" },
95
+ { url = "https://files.pythonhosted.org/packages/5a/72/147da192e38635ada20e0a2e1a51cf8823d2119ce8883f7053879c2199b5/markupsafe-3.0.3-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:d53197da72cc091b024dd97249dfc7794d6a56530370992a5e1a08983ad9230e", size = 11615, upload-time = "2025-09-27T18:36:30.854Z" },
96
+ { url = "https://files.pythonhosted.org/packages/9a/81/7e4e08678a1f98521201c3079f77db69fb552acd56067661f8c2f534a718/markupsafe-3.0.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:1872df69a4de6aead3491198eaf13810b565bdbeec3ae2dc8780f14458ec73ce", size = 12020, upload-time = "2025-09-27T18:36:31.971Z" },
97
+ { url = "https://files.pythonhosted.org/packages/1e/2c/799f4742efc39633a1b54a92eec4082e4f815314869865d876824c257c1e/markupsafe-3.0.3-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:3a7e8ae81ae39e62a41ec302f972ba6ae23a5c5396c8e60113e9066ef893da0d", size = 24332, upload-time = "2025-09-27T18:36:32.813Z" },
98
+ { url = "https://files.pythonhosted.org/packages/3c/2e/8d0c2ab90a8c1d9a24f0399058ab8519a3279d1bd4289511d74e909f060e/markupsafe-3.0.3-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:d6dd0be5b5b189d31db7cda48b91d7e0a9795f31430b7f271219ab30f1d3ac9d", size = 22947, upload-time = "2025-09-27T18:36:33.86Z" },
99
+ { url = "https://files.pythonhosted.org/packages/2c/54/887f3092a85238093a0b2154bd629c89444f395618842e8b0c41783898ea/markupsafe-3.0.3-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:94c6f0bb423f739146aec64595853541634bde58b2135f27f61c1ffd1cd4d16a", size = 21962, upload-time = "2025-09-27T18:36:35.099Z" },
100
+ { url = "https://files.pythonhosted.org/packages/c9/2f/336b8c7b6f4a4d95e91119dc8521402461b74a485558d8f238a68312f11c/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:be8813b57049a7dc738189df53d69395eba14fb99345e0a5994914a3864c8a4b", size = 23760, upload-time = "2025-09-27T18:36:36.001Z" },
101
+ { url = "https://files.pythonhosted.org/packages/32/43/67935f2b7e4982ffb50a4d169b724d74b62a3964bc1a9a527f5ac4f1ee2b/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:83891d0e9fb81a825d9a6d61e3f07550ca70a076484292a70fde82c4b807286f", size = 21529, upload-time = "2025-09-27T18:36:36.906Z" },
102
+ { url = "https://files.pythonhosted.org/packages/89/e0/4486f11e51bbba8b0c041098859e869e304d1c261e59244baa3d295d47b7/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:77f0643abe7495da77fb436f50f8dab76dbc6e5fd25d39589a0f1fe6548bfa2b", size = 23015, upload-time = "2025-09-27T18:36:37.868Z" },
103
+ { url = "https://files.pythonhosted.org/packages/2f/e1/78ee7a023dac597a5825441ebd17170785a9dab23de95d2c7508ade94e0e/markupsafe-3.0.3-cp312-cp312-win32.whl", hash = "sha256:d88b440e37a16e651bda4c7c2b930eb586fd15ca7406cb39e211fcff3bf3017d", size = 14540, upload-time = "2025-09-27T18:36:38.761Z" },
104
+ { url = "https://files.pythonhosted.org/packages/aa/5b/bec5aa9bbbb2c946ca2733ef9c4ca91c91b6a24580193e891b5f7dbe8e1e/markupsafe-3.0.3-cp312-cp312-win_amd64.whl", hash = "sha256:26a5784ded40c9e318cfc2bdb30fe164bdb8665ded9cd64d500a34fb42067b1c", size = 15105, upload-time = "2025-09-27T18:36:39.701Z" },
105
+ { url = "https://files.pythonhosted.org/packages/e5/f1/216fc1bbfd74011693a4fd837e7026152e89c4bcf3e77b6692fba9923123/markupsafe-3.0.3-cp312-cp312-win_arm64.whl", hash = "sha256:35add3b638a5d900e807944a078b51922212fb3dedb01633a8defc4b01a3c85f", size = 13906, upload-time = "2025-09-27T18:36:40.689Z" },
106
+ { url = "https://files.pythonhosted.org/packages/38/2f/907b9c7bbba283e68f20259574b13d005c121a0fa4c175f9bed27c4597ff/markupsafe-3.0.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:e1cf1972137e83c5d4c136c43ced9ac51d0e124706ee1c8aa8532c1287fa8795", size = 11622, upload-time = "2025-09-27T18:36:41.777Z" },
107
+ { url = "https://files.pythonhosted.org/packages/9c/d9/5f7756922cdd676869eca1c4e3c0cd0df60ed30199ffd775e319089cb3ed/markupsafe-3.0.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:116bb52f642a37c115f517494ea5feb03889e04df47eeff5b130b1808ce7c219", size = 12029, upload-time = "2025-09-27T18:36:43.257Z" },
108
+ { url = "https://files.pythonhosted.org/packages/00/07/575a68c754943058c78f30db02ee03a64b3c638586fba6a6dd56830b30a3/markupsafe-3.0.3-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:133a43e73a802c5562be9bbcd03d090aa5a1fe899db609c29e8c8d815c5f6de6", size = 24374, upload-time = "2025-09-27T18:36:44.508Z" },
109
+ { url = "https://files.pythonhosted.org/packages/a9/21/9b05698b46f218fc0e118e1f8168395c65c8a2c750ae2bab54fc4bd4e0e8/markupsafe-3.0.3-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ccfcd093f13f0f0b7fdd0f198b90053bf7b2f02a3927a30e63f3ccc9df56b676", size = 22980, upload-time = "2025-09-27T18:36:45.385Z" },
110
+ { url = "https://files.pythonhosted.org/packages/7f/71/544260864f893f18b6827315b988c146b559391e6e7e8f7252839b1b846a/markupsafe-3.0.3-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:509fa21c6deb7a7a273d629cf5ec029bc209d1a51178615ddf718f5918992ab9", size = 21990, upload-time = "2025-09-27T18:36:46.916Z" },
111
+ { url = "https://files.pythonhosted.org/packages/c2/28/b50fc2f74d1ad761af2f5dcce7492648b983d00a65b8c0e0cb457c82ebbe/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:a4afe79fb3de0b7097d81da19090f4df4f8d3a2b3adaa8764138aac2e44f3af1", size = 23784, upload-time = "2025-09-27T18:36:47.884Z" },
112
+ { url = "https://files.pythonhosted.org/packages/ed/76/104b2aa106a208da8b17a2fb72e033a5a9d7073c68f7e508b94916ed47a9/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:795e7751525cae078558e679d646ae45574b47ed6e7771863fcc079a6171a0fc", size = 21588, upload-time = "2025-09-27T18:36:48.82Z" },
113
+ { url = "https://files.pythonhosted.org/packages/b5/99/16a5eb2d140087ebd97180d95249b00a03aa87e29cc224056274f2e45fd6/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:8485f406a96febb5140bfeca44a73e3ce5116b2501ac54fe953e488fb1d03b12", size = 23041, upload-time = "2025-09-27T18:36:49.797Z" },
114
+ { url = "https://files.pythonhosted.org/packages/19/bc/e7140ed90c5d61d77cea142eed9f9c303f4c4806f60a1044c13e3f1471d0/markupsafe-3.0.3-cp313-cp313-win32.whl", hash = "sha256:bdd37121970bfd8be76c5fb069c7751683bdf373db1ed6c010162b2a130248ed", size = 14543, upload-time = "2025-09-27T18:36:51.584Z" },
115
+ { url = "https://files.pythonhosted.org/packages/05/73/c4abe620b841b6b791f2edc248f556900667a5a1cf023a6646967ae98335/markupsafe-3.0.3-cp313-cp313-win_amd64.whl", hash = "sha256:9a1abfdc021a164803f4d485104931fb8f8c1efd55bc6b748d2f5774e78b62c5", size = 15113, upload-time = "2025-09-27T18:36:52.537Z" },
116
+ { url = "https://files.pythonhosted.org/packages/f0/3a/fa34a0f7cfef23cf9500d68cb7c32dd64ffd58a12b09225fb03dd37d5b80/markupsafe-3.0.3-cp313-cp313-win_arm64.whl", hash = "sha256:7e68f88e5b8799aa49c85cd116c932a1ac15caaa3f5db09087854d218359e485", size = 13911, upload-time = "2025-09-27T18:36:53.513Z" },
117
+ { url = "https://files.pythonhosted.org/packages/e4/d7/e05cd7efe43a88a17a37b3ae96e79a19e846f3f456fe79c57ca61356ef01/markupsafe-3.0.3-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:218551f6df4868a8d527e3062d0fb968682fe92054e89978594c28e642c43a73", size = 11658, upload-time = "2025-09-27T18:36:54.819Z" },
118
+ { url = "https://files.pythonhosted.org/packages/99/9e/e412117548182ce2148bdeacdda3bb494260c0b0184360fe0d56389b523b/markupsafe-3.0.3-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:3524b778fe5cfb3452a09d31e7b5adefeea8c5be1d43c4f810ba09f2ceb29d37", size = 12066, upload-time = "2025-09-27T18:36:55.714Z" },
119
+ { url = "https://files.pythonhosted.org/packages/bc/e6/fa0ffcda717ef64a5108eaa7b4f5ed28d56122c9a6d70ab8b72f9f715c80/markupsafe-3.0.3-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4e885a3d1efa2eadc93c894a21770e4bc67899e3543680313b09f139e149ab19", size = 25639, upload-time = "2025-09-27T18:36:56.908Z" },
120
+ { url = "https://files.pythonhosted.org/packages/96/ec/2102e881fe9d25fc16cb4b25d5f5cde50970967ffa5dddafdb771237062d/markupsafe-3.0.3-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:8709b08f4a89aa7586de0aadc8da56180242ee0ada3999749b183aa23df95025", size = 23569, upload-time = "2025-09-27T18:36:57.913Z" },
121
+ { url = "https://files.pythonhosted.org/packages/4b/30/6f2fce1f1f205fc9323255b216ca8a235b15860c34b6798f810f05828e32/markupsafe-3.0.3-cp313-cp313t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:b8512a91625c9b3da6f127803b166b629725e68af71f8184ae7e7d54686a56d6", size = 23284, upload-time = "2025-09-27T18:36:58.833Z" },
122
+ { url = "https://files.pythonhosted.org/packages/58/47/4a0ccea4ab9f5dcb6f79c0236d954acb382202721e704223a8aafa38b5c8/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:9b79b7a16f7fedff2495d684f2b59b0457c3b493778c9eed31111be64d58279f", size = 24801, upload-time = "2025-09-27T18:36:59.739Z" },
123
+ { url = "https://files.pythonhosted.org/packages/6a/70/3780e9b72180b6fecb83a4814d84c3bf4b4ae4bf0b19c27196104149734c/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_riscv64.whl", hash = "sha256:12c63dfb4a98206f045aa9563db46507995f7ef6d83b2f68eda65c307c6829eb", size = 22769, upload-time = "2025-09-27T18:37:00.719Z" },
124
+ { url = "https://files.pythonhosted.org/packages/98/c5/c03c7f4125180fc215220c035beac6b9cb684bc7a067c84fc69414d315f5/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:8f71bc33915be5186016f675cd83a1e08523649b0e33efdb898db577ef5bb009", size = 23642, upload-time = "2025-09-27T18:37:01.673Z" },
125
+ { url = "https://files.pythonhosted.org/packages/80/d6/2d1b89f6ca4bff1036499b1e29a1d02d282259f3681540e16563f27ebc23/markupsafe-3.0.3-cp313-cp313t-win32.whl", hash = "sha256:69c0b73548bc525c8cb9a251cddf1931d1db4d2258e9599c28c07ef3580ef354", size = 14612, upload-time = "2025-09-27T18:37:02.639Z" },
126
+ { url = "https://files.pythonhosted.org/packages/2b/98/e48a4bfba0a0ffcf9925fe2d69240bfaa19c6f7507b8cd09c70684a53c1e/markupsafe-3.0.3-cp313-cp313t-win_amd64.whl", hash = "sha256:1b4b79e8ebf6b55351f0d91fe80f893b4743f104bff22e90697db1590e47a218", size = 15200, upload-time = "2025-09-27T18:37:03.582Z" },
127
+ { url = "https://files.pythonhosted.org/packages/0e/72/e3cc540f351f316e9ed0f092757459afbc595824ca724cbc5a5d4263713f/markupsafe-3.0.3-cp313-cp313t-win_arm64.whl", hash = "sha256:ad2cf8aa28b8c020ab2fc8287b0f823d0a7d8630784c31e9ee5edea20f406287", size = 13973, upload-time = "2025-09-27T18:37:04.929Z" },
128
+ { url = "https://files.pythonhosted.org/packages/33/8a/8e42d4838cd89b7dde187011e97fe6c3af66d8c044997d2183fbd6d31352/markupsafe-3.0.3-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:eaa9599de571d72e2daf60164784109f19978b327a3910d3e9de8c97b5b70cfe", size = 11619, upload-time = "2025-09-27T18:37:06.342Z" },
129
+ { url = "https://files.pythonhosted.org/packages/b5/64/7660f8a4a8e53c924d0fa05dc3a55c9cee10bbd82b11c5afb27d44b096ce/markupsafe-3.0.3-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:c47a551199eb8eb2121d4f0f15ae0f923d31350ab9280078d1e5f12b249e0026", size = 12029, upload-time = "2025-09-27T18:37:07.213Z" },
130
+ { url = "https://files.pythonhosted.org/packages/da/ef/e648bfd021127bef5fa12e1720ffed0c6cbb8310c8d9bea7266337ff06de/markupsafe-3.0.3-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:f34c41761022dd093b4b6896d4810782ffbabe30f2d443ff5f083e0cbbb8c737", size = 24408, upload-time = "2025-09-27T18:37:09.572Z" },
131
+ { url = "https://files.pythonhosted.org/packages/41/3c/a36c2450754618e62008bf7435ccb0f88053e07592e6028a34776213d877/markupsafe-3.0.3-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:457a69a9577064c05a97c41f4e65148652db078a3a509039e64d3467b9e7ef97", size = 23005, upload-time = "2025-09-27T18:37:10.58Z" },
132
+ { url = "https://files.pythonhosted.org/packages/bc/20/b7fdf89a8456b099837cd1dc21974632a02a999ec9bf7ca3e490aacd98e7/markupsafe-3.0.3-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:e8afc3f2ccfa24215f8cb28dcf43f0113ac3c37c2f0f0806d8c70e4228c5cf4d", size = 22048, upload-time = "2025-09-27T18:37:11.547Z" },
133
+ { url = "https://files.pythonhosted.org/packages/9a/a7/591f592afdc734f47db08a75793a55d7fbcc6902a723ae4cfbab61010cc5/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:ec15a59cf5af7be74194f7ab02d0f59a62bdcf1a537677ce67a2537c9b87fcda", size = 23821, upload-time = "2025-09-27T18:37:12.48Z" },
134
+ { url = "https://files.pythonhosted.org/packages/7d/33/45b24e4f44195b26521bc6f1a82197118f74df348556594bd2262bda1038/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:0eb9ff8191e8498cca014656ae6b8d61f39da5f95b488805da4bb029cccbfbaf", size = 21606, upload-time = "2025-09-27T18:37:13.485Z" },
135
+ { url = "https://files.pythonhosted.org/packages/ff/0e/53dfaca23a69fbfbbf17a4b64072090e70717344c52eaaaa9c5ddff1e5f0/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:2713baf880df847f2bece4230d4d094280f4e67b1e813eec43b4c0e144a34ffe", size = 23043, upload-time = "2025-09-27T18:37:14.408Z" },
136
+ { url = "https://files.pythonhosted.org/packages/46/11/f333a06fc16236d5238bfe74daccbca41459dcd8d1fa952e8fbd5dccfb70/markupsafe-3.0.3-cp314-cp314-win32.whl", hash = "sha256:729586769a26dbceff69f7a7dbbf59ab6572b99d94576a5592625d5b411576b9", size = 14747, upload-time = "2025-09-27T18:37:15.36Z" },
137
+ { url = "https://files.pythonhosted.org/packages/28/52/182836104b33b444e400b14f797212f720cbc9ed6ba34c800639d154e821/markupsafe-3.0.3-cp314-cp314-win_amd64.whl", hash = "sha256:bdc919ead48f234740ad807933cdf545180bfbe9342c2bb451556db2ed958581", size = 15341, upload-time = "2025-09-27T18:37:16.496Z" },
138
+ { url = "https://files.pythonhosted.org/packages/6f/18/acf23e91bd94fd7b3031558b1f013adfa21a8e407a3fdb32745538730382/markupsafe-3.0.3-cp314-cp314-win_arm64.whl", hash = "sha256:5a7d5dc5140555cf21a6fefbdbf8723f06fcd2f63ef108f2854de715e4422cb4", size = 14073, upload-time = "2025-09-27T18:37:17.476Z" },
139
+ { url = "https://files.pythonhosted.org/packages/3c/f0/57689aa4076e1b43b15fdfa646b04653969d50cf30c32a102762be2485da/markupsafe-3.0.3-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:1353ef0c1b138e1907ae78e2f6c63ff67501122006b0f9abad68fda5f4ffc6ab", size = 11661, upload-time = "2025-09-27T18:37:18.453Z" },
140
+ { url = "https://files.pythonhosted.org/packages/89/c3/2e67a7ca217c6912985ec766c6393b636fb0c2344443ff9d91404dc4c79f/markupsafe-3.0.3-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:1085e7fbddd3be5f89cc898938f42c0b3c711fdcb37d75221de2666af647c175", size = 12069, upload-time = "2025-09-27T18:37:19.332Z" },
141
+ { url = "https://files.pythonhosted.org/packages/f0/00/be561dce4e6ca66b15276e184ce4b8aec61fe83662cce2f7d72bd3249d28/markupsafe-3.0.3-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1b52b4fb9df4eb9ae465f8d0c228a00624de2334f216f178a995ccdcf82c4634", size = 25670, upload-time = "2025-09-27T18:37:20.245Z" },
142
+ { url = "https://files.pythonhosted.org/packages/50/09/c419f6f5a92e5fadde27efd190eca90f05e1261b10dbd8cbcb39cd8ea1dc/markupsafe-3.0.3-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:fed51ac40f757d41b7c48425901843666a6677e3e8eb0abcff09e4ba6e664f50", size = 23598, upload-time = "2025-09-27T18:37:21.177Z" },
143
+ { url = "https://files.pythonhosted.org/packages/22/44/a0681611106e0b2921b3033fc19bc53323e0b50bc70cffdd19f7d679bb66/markupsafe-3.0.3-cp314-cp314t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:f190daf01f13c72eac4efd5c430a8de82489d9cff23c364c3ea822545032993e", size = 23261, upload-time = "2025-09-27T18:37:22.167Z" },
144
+ { url = "https://files.pythonhosted.org/packages/5f/57/1b0b3f100259dc9fffe780cfb60d4be71375510e435efec3d116b6436d43/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:e56b7d45a839a697b5eb268c82a71bd8c7f6c94d6fd50c3d577fa39a9f1409f5", size = 24835, upload-time = "2025-09-27T18:37:23.296Z" },
145
+ { url = "https://files.pythonhosted.org/packages/26/6a/4bf6d0c97c4920f1597cc14dd720705eca0bf7c787aebc6bb4d1bead5388/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_riscv64.whl", hash = "sha256:f3e98bb3798ead92273dc0e5fd0f31ade220f59a266ffd8a4f6065e0a3ce0523", size = 22733, upload-time = "2025-09-27T18:37:24.237Z" },
146
+ { url = "https://files.pythonhosted.org/packages/14/c7/ca723101509b518797fedc2fdf79ba57f886b4aca8a7d31857ba3ee8281f/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:5678211cb9333a6468fb8d8be0305520aa073f50d17f089b5b4b477ea6e67fdc", size = 23672, upload-time = "2025-09-27T18:37:25.271Z" },
147
+ { url = "https://files.pythonhosted.org/packages/fb/df/5bd7a48c256faecd1d36edc13133e51397e41b73bb77e1a69deab746ebac/markupsafe-3.0.3-cp314-cp314t-win32.whl", hash = "sha256:915c04ba3851909ce68ccc2b8e2cd691618c4dc4c4232fb7982bca3f41fd8c3d", size = 14819, upload-time = "2025-09-27T18:37:26.285Z" },
148
+ { url = "https://files.pythonhosted.org/packages/1a/8a/0402ba61a2f16038b48b39bccca271134be00c5c9f0f623208399333c448/markupsafe-3.0.3-cp314-cp314t-win_amd64.whl", hash = "sha256:4faffd047e07c38848ce017e8725090413cd80cbc23d86e55c587bf979e579c9", size = 15426, upload-time = "2025-09-27T18:37:27.316Z" },
149
+ { url = "https://files.pythonhosted.org/packages/70/bc/6f1c2f612465f5fa89b95bead1f44dcb607670fd42891d8fdcd5d039f4f4/markupsafe-3.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:32001d6a8fc98c8cb5c947787c5d08b0a50663d139f1305bac5885d98d9b40fa", size = 14146, upload-time = "2025-09-27T18:37:28.327Z" },
150
+ ]
151
+
152
+ [[package]]
153
+ name = "mdurl"
154
+ version = "0.1.2"
155
+ source = { registry = "https://pypi.org/simple" }
156
+ sdist = { url = "https://files.pythonhosted.org/packages/d6/54/cfe61301667036ec958cb99bd3efefba235e65cdeb9c84d24a8293ba1d90/mdurl-0.1.2.tar.gz", hash = "sha256:bb413d29f5eea38f31dd4754dd7377d4465116fb207585f97bf925588687c1ba", size = 8729, upload-time = "2022-08-14T12:40:10.846Z" }
157
+ wheels = [
158
+ { url = "https://files.pythonhosted.org/packages/b3/38/89ba8ad64ae25be8de66a6d463314cf1eb366222074cfda9ee839c56a4b4/mdurl-0.1.2-py3-none-any.whl", hash = "sha256:84008a41e51615a49fc9966191ff91509e3c40b939176e643fd50a5c2196b8f8", size = 9979, upload-time = "2022-08-14T12:40:09.779Z" },
159
+ ]
160
+
161
+ [[package]]
162
+ name = "prompt-toolkit"
163
+ version = "3.0.52"
164
+ source = { registry = "https://pypi.org/simple" }
165
+ dependencies = [
166
+ { name = "wcwidth" },
167
+ ]
168
+ sdist = { url = "https://files.pythonhosted.org/packages/a1/96/06e01a7b38dce6fe1db213e061a4602dd6032a8a97ef6c1a862537732421/prompt_toolkit-3.0.52.tar.gz", hash = "sha256:28cde192929c8e7321de85de1ddbe736f1375148b02f2e17edd840042b1be855", size = 434198, upload-time = "2025-08-27T15:24:02.057Z" }
169
+ wheels = [
170
+ { url = "https://files.pythonhosted.org/packages/84/03/0d3ce49e2505ae70cf43bc5bb3033955d2fc9f932163e84dc0779cc47f48/prompt_toolkit-3.0.52-py3-none-any.whl", hash = "sha256:9aac639a3bbd33284347de5ad8d68ecc044b91a762dc39b7c21095fcd6a19955", size = 391431, upload-time = "2025-08-27T15:23:59.498Z" },
171
+ ]
172
+
173
+ [[package]]
174
+ name = "pygments"
175
+ version = "2.19.2"
176
+ source = { registry = "https://pypi.org/simple" }
177
+ 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" }
178
+ wheels = [
179
+ { 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" },
180
+ ]
181
+
182
+ [[package]]
183
+ name = "questionary"
184
+ version = "2.1.1"
185
+ source = { registry = "https://pypi.org/simple" }
186
+ dependencies = [
187
+ { name = "prompt-toolkit" },
188
+ ]
189
+ sdist = { url = "https://files.pythonhosted.org/packages/f6/45/eafb0bba0f9988f6a2520f9ca2df2c82ddfa8d67c95d6625452e97b204a5/questionary-2.1.1.tar.gz", hash = "sha256:3d7e980292bb0107abaa79c68dd3eee3c561b83a0f89ae482860b181c8bd412d", size = 25845, upload-time = "2025-08-28T19:00:20.851Z" }
190
+ wheels = [
191
+ { url = "https://files.pythonhosted.org/packages/3c/26/1062c7ec1b053db9e499b4d2d5bc231743201b74051c973dadeac80a8f43/questionary-2.1.1-py3-none-any.whl", hash = "sha256:a51af13f345f1cdea62347589fbb6df3b290306ab8930713bfae4d475a7d4a59", size = 36753, upload-time = "2025-08-28T19:00:19.56Z" },
192
+ ]
193
+
194
+ [[package]]
195
+ name = "rich"
196
+ version = "14.3.3"
197
+ source = { registry = "https://pypi.org/simple" }
198
+ dependencies = [
199
+ { name = "markdown-it-py" },
200
+ { name = "pygments" },
201
+ ]
202
+ sdist = { url = "https://files.pythonhosted.org/packages/b3/c6/f3b320c27991c46f43ee9d856302c70dc2d0fb2dba4842ff739d5f46b393/rich-14.3.3.tar.gz", hash = "sha256:b8daa0b9e4eef54dd8cf7c86c03713f53241884e814f4e2f5fb342fe520f639b", size = 230582, upload-time = "2026-02-19T17:23:12.474Z" }
203
+ wheels = [
204
+ { url = "https://files.pythonhosted.org/packages/14/25/b208c5683343959b670dc001595f2f3737e051da617f66c31f7c4fa93abc/rich-14.3.3-py3-none-any.whl", hash = "sha256:793431c1f8619afa7d3b52b2cdec859562b950ea0d4b6b505397612db8d5362d", size = 310458, upload-time = "2026-02-19T17:23:13.732Z" },
205
+ ]
206
+
207
+ [[package]]
208
+ name = "shellingham"
209
+ version = "1.5.4"
210
+ source = { registry = "https://pypi.org/simple" }
211
+ sdist = { url = "https://files.pythonhosted.org/packages/58/15/8b3609fd3830ef7b27b655beb4b4e9c62313a4e8da8c676e142cc210d58e/shellingham-1.5.4.tar.gz", hash = "sha256:8dbca0739d487e5bd35ab3ca4b36e11c4078f3a234bfce294b0a0291363404de", size = 10310, upload-time = "2023-10-24T04:13:40.426Z" }
212
+ wheels = [
213
+ { url = "https://files.pythonhosted.org/packages/e0/f9/0595336914c5619e5f28a1fb793285925a8cd4b432c9da0a987836c7f822/shellingham-1.5.4-py2.py3-none-any.whl", hash = "sha256:7ecfff8f2fd72616f7481040475a65b2bf8af90a56c89140852d1120324e8686", size = 9755, upload-time = "2023-10-24T04:13:38.866Z" },
214
+ ]
215
+
216
+ [[package]]
217
+ name = "typer"
218
+ version = "0.24.1"
219
+ source = { registry = "https://pypi.org/simple" }
220
+ dependencies = [
221
+ { name = "annotated-doc" },
222
+ { name = "click" },
223
+ { name = "rich" },
224
+ { name = "shellingham" },
225
+ ]
226
+ sdist = { url = "https://files.pythonhosted.org/packages/f5/24/cb09efec5cc954f7f9b930bf8279447d24618bb6758d4f6adf2574c41780/typer-0.24.1.tar.gz", hash = "sha256:e39b4732d65fbdcde189ae76cf7cd48aeae72919dea1fdfc16593be016256b45", size = 118613, upload-time = "2026-02-21T16:54:40.609Z" }
227
+ wheels = [
228
+ { url = "https://files.pythonhosted.org/packages/4a/91/48db081e7a63bb37284f9fbcefda7c44c277b18b0e13fbc36ea2335b71e6/typer-0.24.1-py3-none-any.whl", hash = "sha256:112c1f0ce578bfb4cab9ffdabc68f031416ebcc216536611ba21f04e9aa84c9e", size = 56085, upload-time = "2026-02-21T16:54:41.616Z" },
229
+ ]
230
+
231
+ [[package]]
232
+ name = "wcwidth"
233
+ version = "0.6.0"
234
+ source = { registry = "https://pypi.org/simple" }
235
+ sdist = { url = "https://files.pythonhosted.org/packages/35/a2/8e3becb46433538a38726c948d3399905a4c7cabd0df578ede5dc51f0ec2/wcwidth-0.6.0.tar.gz", hash = "sha256:cdc4e4262d6ef9a1a57e018384cbeb1208d8abbc64176027e2c2455c81313159", size = 159684, upload-time = "2026-02-06T19:19:40.919Z" }
236
+ wheels = [
237
+ { url = "https://files.pythonhosted.org/packages/68/5a/199c59e0a824a3db2b89c5d2dade7ab5f9624dbf6448dc291b46d5ec94d3/wcwidth-0.6.0-py3-none-any.whl", hash = "sha256:1a3a1e510b553315f8e146c54764f4fb6264ffad731b3d78088cdb1478ffbdad", size = 94189, upload-time = "2026-02-06T19:19:39.646Z" },
238
+ ]