Prototype implementation

This commit is contained in:
2026-05-08 14:26:48 +02:00
parent 315143a6fc
commit 14b0bc6d01
160 changed files with 5731 additions and 42 deletions

6
.env.example Normal file
View File

@@ -0,0 +1,6 @@
DATABASE_URL=postgres://vergabe:vergabe@localhost:5432/vergabe_db
SECRET_KEY=change-me-in-production
DEBUG=True
ALLOWED_HOSTS=localhost,127.0.0.1
MEDIA_ROOT=media/
MAX_UPLOAD_SIZE=52428800

7
.gitignore vendored
View File

@@ -171,6 +171,13 @@ cython_debug/
# Ruff stuff:
.ruff_cache/
# Build output
static/dist/
node_modules/
# Media uploads
media/
# PyPI configuration file
.pypirc

1
.python-version Normal file
View File

@@ -0,0 +1 @@
3.12

View File

@@ -6,11 +6,48 @@ This file provides guidance to Claude Code (claude.ai/code) when working with co
**Vergabe Teilnahme** is a web-based tender/bid management system (internal collaboration tool) that supports a company through the full lifecycle of public and private procurement bids — from initial research through post-award retrospective. The language of the application and all domain documentation is **German**.
This repo is currently **pre-implementation**. The authoritative requirements are in `wiki/ProductRequirementsDocument.md`.
The authoritative requirements are in `wiki/ProductRequirementsDocument.md`. Technical architecture in `wiki/ArchitectureBlueprint.md`. Use cases in `wiki/UseCaseCatalog.md`.
## Planned Tech Stack
## Tech Stack
The .gitignore targets **Python** (Django, Flask, uv, Ruff, pytest). No framework or tooling has been selected yet — check for a `pyproject.toml`, `Pipfile`, or `requirements.txt` before assuming.
**Django 6.x** · **uv** (package manager) · **Tailwind CSS v4** (via Vite) · **HTMX 2.x** · **Alpine.js 3.x** · **PostgreSQL 16+** (psycopg3)
## Entwicklungs-Commands
```bash
make db # PostgreSQL via Docker starten (oder infra-postgres-1 verwenden)
make dev # Django-Dev-Server (Port 8000)
make css # Tailwind CSS im Watch-Modus
make migrate # Migrations generieren und ausführen
make test # pytest ausführen
make lint # ruff + mypy
uv run manage.py test vergabe_teilnahme.apps.<app> # Einzelne App testen
uv run pytest vergabe_teilnahme/apps/<app>/tests/ # Einzelne Testdatei
```
## Projektstruktur
```
vergabe_teilnahme/
├── apps/ # Alle Django-Apps
│ ├── core/ # FlexibleModel, CustomAttribute, EntityFieldConfig, Freigabe
│ ├── accounts/ # Mitarbeiter (AbstractUser)
│ └── ... # je eine App pro Fachdomäne
├── settings/ # base.py, dev.py, prod.py
└── urls.py
static/
├── src/main.css # Tailwind-Quelldatei (mit @layer components und @theme brand tokens)
├── vendor/ # HTMX, Alpine.js (lokal, kein CDN)
└── dist/ # Build-Output (gitignored)
workplans/ # Ralph-Loop-Workplans (WP-0001 bis WP-0012)
wiki/ # PRD, Blueprint, Use-Case-Katalog
```
## Shared Infrastructure Note
Port 5432 is used by `infra-postgres-1` (the Custodian shared PostgreSQL container). The `vergabe_db` database and `vergabe` user are created there. `docker-compose.dev.yml` documents the intended standalone setup but is not started when infra container is active.
## Domain Model — Key Concepts

30
Makefile Normal file
View File

@@ -0,0 +1,30 @@
.PHONY: dev db migrate shell test lint css createsuperuser collectstatic
db:
docker compose -f docker-compose.dev.yml up -d 2>/dev/null || echo "DB already running"
dev:
uv run manage.py runserver 0.0.0.0:8000
css:
npm run dev
migrate:
uv run manage.py makemigrations
uv run manage.py migrate
shell:
uv run manage.py shell_plus 2>/dev/null || uv run manage.py shell
test:
uv run pytest
lint:
uv run ruff check .
uv run mypy vergabe_teilnahme/
createsuperuser:
uv run manage.py createsuperuser
collectstatic:
uv run manage.py collectstatic --noinput

10
conftest.py Normal file
View File

@@ -0,0 +1,10 @@
import pytest
@pytest.fixture
def mitarbeiter(db):
from vergabe_teilnahme.apps.accounts.models import Mitarbeiter
return Mitarbeiter.objects.create_user(
username='testuser', password='testpass', first_name='Test', last_name='User'
)

14
docker-compose.dev.yml Normal file
View File

@@ -0,0 +1,14 @@
services:
db:
image: postgres:16-alpine
environment:
POSTGRES_DB: vergabe_db
POSTGRES_USER: vergabe
POSTGRES_PASSWORD: vergabe
ports:
- "5432:5432"
volumes:
- postgres_data:/var/lib/postgresql/data
volumes:
postgres_data:

14
docker-compose.test.yml Normal file
View File

@@ -0,0 +1,14 @@
services:
db:
image: postgres:16-alpine
environment:
POSTGRES_DB: vergabe_test
POSTGRES_USER: vergabe
POSTGRES_PASSWORD: vergabe
ports:
- "5433:5432"
volumes:
- postgres_test_data:/var/lib/postgresql/data
volumes:
postgres_test_data:

22
manage.py Executable file
View File

@@ -0,0 +1,22 @@
#!/usr/bin/env python
"""Django's command-line utility for administrative tasks."""
import os
import sys
def main():
"""Run administrative tasks."""
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'vergabe_teilnahme.settings.dev')
try:
from django.core.management import execute_from_command_line
except ImportError as exc:
raise ImportError(
"Couldn't import Django. Are you sure it's installed and "
"available on your PYTHONPATH environment variable? Did you "
"forget to activate a virtual environment?"
) from exc
execute_from_command_line(sys.argv)
if __name__ == '__main__':
main()

1643
package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

12
package.json Normal file
View File

@@ -0,0 +1,12 @@
{
"type": "module",
"scripts": {
"dev": "vite build --watch",
"build": "vite build"
},
"devDependencies": {
"vite": "^5.0",
"@tailwindcss/vite": "^4.0",
"tailwindcss": "^4.0"
}
}

44
pyproject.toml Normal file
View File

@@ -0,0 +1,44 @@
[project]
name = "vergabe-teilnahme"
version = "0.1.0"
description = "Ausschreibungs- und Teilnahme-Management-System"
requires-python = ">=3.12"
dependencies = [
"django>=5.2",
"psycopg[binary]>=3.2",
"django-storages>=1.14",
"whitenoise>=6.7",
"python-decouple>=3.8",
"dj-database-url>=2.1",
]
[dependency-groups]
dev = [
"pytest-django>=4.8",
"pytest-cov>=5.0",
"factory-boy>=3.3",
"ruff>=0.4",
"mypy>=1.10",
"django-stubs>=5.0",
]
[tool.pytest.ini_options]
DJANGO_SETTINGS_MODULE = "vergabe_teilnahme.settings.dev"
python_files = ["test_*.py"]
python_classes = ["Test*"]
python_functions = ["test_*"]
addopts = "--tb=short -q"
[tool.ruff]
line-length = 100
target-version = "py312"
[tool.ruff.lint]
select = ["E", "F", "I", "N", "UP"]
[tool.mypy]
python_version = "3.12"
plugins = ["mypy_django_plugin.main"]
[tool.django-stubs]
django_settings_module = "vergabe_teilnahme.settings.dev"

43
static/src/main.css Normal file
View File

@@ -0,0 +1,43 @@
@import "tailwindcss";
@theme {
--color-brand-50: #f0f4ff;
--color-brand-100: #dce7ff;
--color-brand-500: #3b5bdb;
--color-brand-600: #2f4ac7;
--color-brand-700: #2541b2;
--color-brand-900: #152d99;
}
@layer base {
/* German-app base resets */
html {
font-family: ui-sans-serif, system-ui, sans-serif;
}
}
@layer components {
.card { @apply bg-white rounded-xl border border-slate-200 shadow-sm p-6; }
.btn-primary { @apply bg-brand-500 text-white px-4 py-2 rounded-lg hover:bg-brand-600 transition-colors; }
.btn-secondary { @apply bg-white text-slate-700 border border-slate-300 px-4 py-2 rounded-lg hover:bg-slate-50; }
.btn-danger { @apply bg-red-600 text-white px-4 py-2 rounded-lg hover:bg-red-700; }
.btn-ghost { @apply text-slate-600 px-3 py-2 rounded-lg hover:bg-slate-100; }
.field-row { @apply grid grid-cols-3 gap-4 py-3 border-b border-slate-100 last:border-0; }
.field-label { @apply text-sm font-medium text-slate-500 col-span-1; }
.field-value { @apply text-sm text-slate-900 col-span-2; }
.phase-badge { @apply inline-flex items-center justify-center w-7 h-7 rounded-full text-sm font-bold; }
.phase-todo { @apply inline-flex items-center justify-center w-7 h-7 rounded-full text-sm font-bold bg-slate-200 text-slate-500; }
.phase-active { @apply inline-flex items-center justify-center w-7 h-7 rounded-full text-sm font-bold bg-brand-500 text-white; }
.phase-done { @apply inline-flex items-center justify-center w-7 h-7 rounded-full text-sm font-bold bg-green-500 text-white; }
.phase-warn { @apply inline-flex items-center justify-center w-7 h-7 rounded-full text-sm font-bold bg-amber-400 text-amber-900; }
.section-title { @apply text-base font-semibold text-slate-900 mb-4; }
.page-title { @apply text-2xl font-bold text-slate-900; }
.form-input { @apply w-full rounded-lg border border-slate-300 px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-brand-500 focus:border-transparent; }
.form-label { @apply block text-sm font-medium text-slate-700 mb-1; }
.table-base { @apply w-full text-sm text-left; }
.table-header { @apply bg-slate-50 text-slate-500 font-medium text-xs uppercase tracking-wide; }
.table-row { @apply border-t border-slate-100 hover:bg-slate-50 transition-colors; }
.sidebar-link { @apply flex items-center px-3 py-2 rounded-lg text-sm text-slate-700 hover:bg-slate-100 transition-colors; }
.sidebar-link-active { @apply bg-brand-50 text-brand-700 font-medium; }
.sidebar-section-btn { @apply w-full flex items-center justify-between px-3 py-2 text-xs font-semibold text-slate-500 uppercase tracking-wide hover:text-slate-700; }
}

5
static/vendor/alpinejs/alpine.min.js vendored Normal file

File diff suppressed because one or more lines are too long

1
static/vendor/htmx/htmx.min.js vendored Normal file

File diff suppressed because one or more lines are too long

616
uv.lock generated Normal file
View File

@@ -0,0 +1,616 @@
version = 1
requires-python = ">=3.12"
resolution-markers = [
"python_full_version < '3.15'",
"python_full_version >= '3.15'",
]
[[package]]
name = "asgiref"
version = "3.11.1"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/63/40/f03da1264ae8f7cfdbf9146542e5e7e8100a4c66ab48e791df9a03d3f6c0/asgiref-3.11.1.tar.gz", hash = "sha256:5f184dc43b7e763efe848065441eac62229c9f7b0475f41f80e207a114eda4ce", size = 38550 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/5c/0a/a72d10ed65068e115044937873362e6e32fab1b7dce0046aeb224682c989/asgiref-3.11.1-py3-none-any.whl", hash = "sha256:e8667a091e69529631969fd45dc268fa79b99c92c5fcdda727757e52146ec133", size = 24345 },
]
[[package]]
name = "ast-serialize"
version = "0.3.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/a9/9d/912fefab0e30aee6a3af8a62bbea4a81b29afa4ba2c973d31170620a26de/ast_serialize-0.3.0.tar.gz", hash = "sha256:1bc3ca09a63a021376527c4e938deedd11d11d675ce850e6f9c7487f5889992b", size = 60689 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/6f/57/a54d4de491d6cdd7a4e4b0952cc3ca9f60dcefa7b5fb48d6d492debe1649/ast_serialize-0.3.0-cp314-cp314t-macosx_10_12_x86_64.whl", hash = "sha256:3a867927df59f76a18dc1d874a0b2c079b42c58972dca637905576deb0912e14", size = 1182966 },
{ url = "https://files.pythonhosted.org/packages/ee/9e/a5db014bb0f91b209236b57c429389e31290c0093532b8436d577699b2fa/ast_serialize-0.3.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:a6fb063bf040abf8321e7b8113a0554eda445ffc508aa51287f8808886a5ae22", size = 1171316 },
{ url = "https://files.pythonhosted.org/packages/15/59/fd55133e478c4326f60a11df02573bf7ccb2ac685810b50f1803d0f68053/ast_serialize-0.3.0-cp314-cp314t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5075cd8482573d743586779e5f9b652a015e37d4e95132d7e5a9bc5c8f483d8f", size = 1232234 },
{ url = "https://files.pythonhosted.org/packages/cc/79/0ca1d26357ecb4a697d74d00b73ef3137f24c140424125393a0de820eb09/ast_serialize-0.3.0-cp314-cp314t-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:41560b27794f4553b0f77811e9fb325b77db4a2b39018d437e09932275306e66", size = 1233437 },
{ url = "https://files.pythonhosted.org/packages/53/3e/7078ec94dd6e124b8e028ac77016a4f13c83fa1c145790f2e68f3816998b/ast_serialize-0.3.0-cp314-cp314t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b967c01ca74909c5d90e0fe4393401e2cc5da5ebd9a6262a19e45ffd3757dec8", size = 1440188 },
{ url = "https://files.pythonhosted.org/packages/21/16/cca7195ef55a012f8013c3442afa91d287a0a36dcf88b480b262475135b3/ast_serialize-0.3.0-cp314-cp314t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:424ebb8f46cd993f7cec4009d119312d8433dd90e6b0df0499cd2c91bdcc5af9", size = 1254211 },
{ url = "https://files.pythonhosted.org/packages/a0/0f/f3d4dfae67dee6580534361a6343367d34217e7d25cff858bd1d8f03b8ed/ast_serialize-0.3.0-cp314-cp314t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d14b1d566b56e2ee70b11fec1de7e0b94ec7cd83717ec7d189967841a361190e", size = 1255973 },
{ url = "https://files.pythonhosted.org/packages/14/41/55fbfe02c42f40fbe3e74eda167d977d555ff720ce1abfa08515236efd88/ast_serialize-0.3.0-cp314-cp314t-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:7ba30b18735f047ec11103d1ab92f4789cf1fea1e0dc89b04a2f5a0632fd79de", size = 1298629 },
{ url = "https://files.pythonhosted.org/packages/28/36/7d2501cacc7989fb8504aa9da2a2022a174200a59d4e6639de4367a57fdd/ast_serialize-0.3.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:e6ea0754cb7b0f682ebb005ffb0d18f8d17993490d9c289863cd69cacc4ab8df", size = 1408435 },
{ url = "https://files.pythonhosted.org/packages/03/e7/54e3b469c3fa0bf9cd532fa643d1d33b73303f8d70beac3e366b68dd64b7/ast_serialize-0.3.0-cp314-cp314t-musllinux_1_2_armv7l.whl", hash = "sha256:a0c5aa1073a5ba7b2abaa4b54abe8b8d75c4d1e2d54a2ff70b0ca6222fea5728", size = 1508174 },
{ url = "https://files.pythonhosted.org/packages/b5/2a/9b9621865b02c60539e26d9b114a312b4fa46aa703e33e79317174bfea21/ast_serialize-0.3.0-cp314-cp314t-musllinux_1_2_i686.whl", hash = "sha256:4e52650d834c1ea7791969a361de2c54c13b2fb4c519ec79445fa8b9021a147d", size = 1502354 },
{ url = "https://files.pythonhosted.org/packages/34/dd/f138bc5c43b0c414fdd12eefe15677839323078b6e75301ad7f96cd26d45/ast_serialize-0.3.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:15bd6af3f136c61dae27805eb6b8f3269e85a545c4c27ffe9e530ead78d2b36d", size = 1450504 },
{ url = "https://files.pythonhosted.org/packages/68/cf/97ef9e1c315601db74365955c8edd3292e3055500d6317602815dbdf08ae/ast_serialize-0.3.0-cp314-cp314t-win32.whl", hash = "sha256:d188bfe37b674b49708497683051d4b571366a668799c9b8e8a94513694969d9", size = 1058662 },
{ url = "https://files.pythonhosted.org/packages/f8/d6/e2c3483c31580fdb623f92ad38d2f856cde4b9205a3e6bd84760f3de7d82/ast_serialize-0.3.0-cp314-cp314t-win_amd64.whl", hash = "sha256:5832c2fdf8f8a6cf682b4cfcf677f5eaf39b4ddbc490f5480cfccdd1e7ce8fa1", size = 1100349 },
{ url = "https://files.pythonhosted.org/packages/ab/89/29abcb1fe18a429cda60c6e0bbd1d6e90499339842a2f548d7567542357e/ast_serialize-0.3.0-cp314-cp314t-win_arm64.whl", hash = "sha256:670f177188d128fb7f9f15b5ad0e1b553d22c34e3f584dcb83eb8077600437f0", size = 1072895 },
{ url = "https://files.pythonhosted.org/packages/bc/93/72abad83966ed6235647c9f956417dc1e17e997696388521910e3d1fa3f4/ast_serialize-0.3.0-cp39-abi3-macosx_10_12_x86_64.whl", hash = "sha256:2ec2fafa5e4313cc8feed96e436ebe19ac7bc6fa41fbc2827e826c48b9e4c3a9", size = 1190024 },
{ url = "https://files.pythonhosted.org/packages/85/4f/eb88584b2f0234e581762011208ca203252bf6c98e59b4769daa571f3576/ast_serialize-0.3.0-cp39-abi3-macosx_11_0_arm64.whl", hash = "sha256:ef6d3c08b7b4cd29b48410338e134764a00e76d25841eb02c1084e868c888ecc", size = 1178633 },
{ url = "https://files.pythonhosted.org/packages/56/51/cf1ec1ff3e616373d0dcbd5fad502e0029dc541f13ab642259762a7d127f/ast_serialize-0.3.0-cp39-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3d841424f41b886e98044abc80769c14a956e6e5ccd5fb5b0d9f5ead72be18a4", size = 1241351 },
{ url = "https://files.pythonhosted.org/packages/0d/44/68fcf50478cf1093f2d423f034ae06453122c8b415d8e21a44668eca485d/ast_serialize-0.3.0-cp39-abi3-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:d21453734ad39367ede5d37efe4f59f830ce1c09f432fc72a90e368f77a4a3e7", size = 1239582 },
{ url = "https://files.pythonhosted.org/packages/9d/c1/a6c9fa284eceb5fc6f21347e968445a051d7ca2c4d34e6a04314646dbcee/ast_serialize-0.3.0-cp39-abi3-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f5e110cdce2a347e1dd987529c88ef54d26f67848dce3eba1b3b2cc2cf085c94", size = 1448853 },
{ url = "https://files.pythonhosted.org/packages/23/5f/8ad3829a09e4e8c5328a53ce7d4711d660944e3e164c5f6abcc2c8f27167/ast_serialize-0.3.0-cp39-abi3-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3b6e23a98e57560a055f5c4b68700a0fd5ce483d2814c23140b3638c7f5d1e61", size = 1262204 },
{ url = "https://files.pythonhosted.org/packages/25/13/44aa28d97f10e25247e8576b5f6b2795d4fa1a80acc88acc942c508d06f7/ast_serialize-0.3.0-cp39-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c1c9e763d70293d65ce1e1ea8c943140c68d0953f0268c7ee0998f2e07f77dd0", size = 1266458 },
{ url = "https://files.pythonhosted.org/packages/d8/58/b3a8be3777cd3744324fd5cec0d80d37cd96fc7cbb0fb010e03dff1e870f/ast_serialize-0.3.0-cp39-abi3-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:4388a1796c228f1ce5c391426f7d21a0003ad3b47f677dbeded9bd1a85c7209f", size = 1308700 },
{ url = "https://files.pythonhosted.org/packages/13/03/f8312d6b57f5471a9dc7946f22b8798a1fc296d38c25766223aacadec42c/ast_serialize-0.3.0-cp39-abi3-musllinux_1_2_aarch64.whl", hash = "sha256:5283cdcc0c64c3d8b9b688dc6aaa012d9c0cf1380a7f774a6bae6a1c01b3205a", size = 1416724 },
{ url = "https://files.pythonhosted.org/packages/50/5d/13fc3789a7abac00559da2e2e9f386db4612aa1f84fc53d09bf714c37545/ast_serialize-0.3.0-cp39-abi3-musllinux_1_2_armv7l.whl", hash = "sha256:f5ef88cc5842a5d7a6ac09dc0d5fc2c98f5d276c1f076f866d55047ce886785b", size = 1515441 },
{ url = "https://files.pythonhosted.org/packages/eb/b9/7ab43fc7a23b1f970281093228f5f79bed6edeed7a3e672bde6d7a832a58/ast_serialize-0.3.0-cp39-abi3-musllinux_1_2_i686.whl", hash = "sha256:cc14bf402bdc0978594ecce783793de2c7470cd4f5cd7eb286ca97ed8ff7cba9", size = 1510522 },
{ url = "https://files.pythonhosted.org/packages/56/ec/d75fc2b788d319f1fad77c14156896f31afdfc68af85b505e5bdebcb9592/ast_serialize-0.3.0-cp39-abi3-musllinux_1_2_x86_64.whl", hash = "sha256:11eae0cf1b7b3e0678133cc2daa974ea972caf02eb4b3aa062af6fa9acd52c57", size = 1460917 },
{ url = "https://files.pythonhosted.org/packages/95/74/f99c81193a2725911e1911ae567ed27c2f2419332c7f3537366f9d238cac/ast_serialize-0.3.0-cp39-abi3-win32.whl", hash = "sha256:2db3dd99de5e6a5a11d7dda73de8750eb6e5baaf25245adf7bdcfe64b6108ae2", size = 1067804 },
{ url = "https://files.pythonhosted.org/packages/16/81/76af00c47daa151e89f98ae21fbbcb2840aaa9f5766579c4da76a3c57188/ast_serialize-0.3.0-cp39-abi3-win_amd64.whl", hash = "sha256:a2cd125adccf7969470621905d302750cd25951f22ea430d9a25b7be031e5549", size = 1105561 },
{ url = "https://files.pythonhosted.org/packages/bd/46/d3ec57ad500f598d1554bd14ce4df615960549ab2844961bc4e1f5fbd174/ast_serialize-0.3.0-cp39-abi3-win_arm64.whl", hash = "sha256:0dd00da29985f15f50dc35728b7e1e7c84507bccfea1d9914738530f1c72238a", size = 1077165 },
]
[[package]]
name = "colorama"
version = "0.4.6"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335 },
]
[[package]]
name = "coverage"
version = "7.13.5"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/9d/e0/70553e3000e345daff267cec284ce4cbf3fc141b6da229ac52775b5428f1/coverage-7.13.5.tar.gz", hash = "sha256:c81f6515c4c40141f83f502b07bbfa5c240ba25bbe73da7b33f1e5b6120ff179", size = 915967 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/a0/c3/a396306ba7db865bf96fc1fb3b7fd29bcbf3d829df642e77b13555163cd6/coverage-7.13.5-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:460cf0114c5016fa841214ff5564aa4864f11948da9440bc97e21ad1f4ba1e01", size = 219554 },
{ url = "https://files.pythonhosted.org/packages/a6/16/a68a19e5384e93f811dccc51034b1fd0b865841c390e3c931dcc4699e035/coverage-7.13.5-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:0e223ce4b4ed47f065bfb123687686512e37629be25cc63728557ae7db261422", size = 219908 },
{ url = "https://files.pythonhosted.org/packages/29/72/20b917c6793af3a5ceb7fb9c50033f3ec7865f2911a1416b34a7cfa0813b/coverage-7.13.5-cp312-cp312-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:6e3370441f4513c6252bf042b9c36d22491142385049243253c7e48398a15a9f", size = 251419 },
{ url = "https://files.pythonhosted.org/packages/8c/49/cd14b789536ac6a4778c453c6a2338bc0a2fb60c5a5a41b4008328b9acc1/coverage-7.13.5-cp312-cp312-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:03ccc709a17a1de074fb1d11f217342fb0d2b1582ed544f554fc9fc3f07e95f5", size = 254159 },
{ url = "https://files.pythonhosted.org/packages/9d/00/7b0edcfe64e2ed4c0340dac14a52ad0f4c9bd0b8b5e531af7d55b703db7c/coverage-7.13.5-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:3f4818d065964db3c1c66dc0fbdac5ac692ecbc875555e13374fdbe7eedb4376", size = 255270 },
{ url = "https://files.pythonhosted.org/packages/93/89/7ffc4ba0f5d0a55c1e84ea7cee39c9fc06af7b170513d83fbf3bbefce280/coverage-7.13.5-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:012d5319e66e9d5a218834642d6c35d265515a62f01157a45bcc036ecf947256", size = 257538 },
{ url = "https://files.pythonhosted.org/packages/81/bd/73ddf85f93f7e6fa83e77ccecb6162d9415c79007b4bc124008a4995e4a7/coverage-7.13.5-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:8dd02af98971bdb956363e4827d34425cb3df19ee550ef92855b0acb9c7ce51c", size = 251821 },
{ url = "https://files.pythonhosted.org/packages/a0/81/278aff4e8dec4926a0bcb9486320752811f543a3ce5b602cc7a29978d073/coverage-7.13.5-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:f08fd75c50a760c7eb068ae823777268daaf16a80b918fa58eea888f8e3919f5", size = 253191 },
{ url = "https://files.pythonhosted.org/packages/70/ee/fe1621488e2e0a58d7e94c4800f0d96f79671553488d401a612bebae324b/coverage-7.13.5-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:843ea8643cf967d1ac7e8ecd4bb00c99135adf4816c0c0593fdcc47b597fcf09", size = 251337 },
{ url = "https://files.pythonhosted.org/packages/37/a6/f79fb37aa104b562207cc23cb5711ab6793608e246cae1e93f26b2236ed9/coverage-7.13.5-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:9d44d7aa963820b1b971dbecd90bfe5fe8f81cff79787eb6cca15750bd2f79b9", size = 255404 },
{ url = "https://files.pythonhosted.org/packages/75/f0/ed15262a58ec81ce457ceb717b7f78752a1713556b19081b76e90896e8d4/coverage-7.13.5-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:7132bed4bd7b836200c591410ae7d97bf7ae8be6fc87d160b2bd881df929e7bf", size = 250903 },
{ url = "https://files.pythonhosted.org/packages/0f/e9/9129958f20e7e9d4d56d51d42ccf708d15cac355ff4ac6e736e97a9393d2/coverage-7.13.5-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:a698e363641b98843c517817db75373c83254781426e94ada3197cabbc2c919c", size = 252780 },
{ url = "https://files.pythonhosted.org/packages/a4/d7/0ad9b15812d81272db94379fe4c6df8fd17781cc7671fdfa30c76ba5ff7b/coverage-7.13.5-cp312-cp312-win32.whl", hash = "sha256:bdba0a6b8812e8c7df002d908a9a2ea3c36e92611b5708633c50869e6d922fdf", size = 222093 },
{ url = "https://files.pythonhosted.org/packages/29/3d/821a9a5799fac2556bcf0bd37a70d1d11fa9e49784b6d22e92e8b2f85f18/coverage-7.13.5-cp312-cp312-win_amd64.whl", hash = "sha256:d2c87e0c473a10bffe991502eac389220533024c8082ec1ce849f4218dded810", size = 222900 },
{ url = "https://files.pythonhosted.org/packages/d4/fa/2238c2ad08e35cf4f020ea721f717e09ec3152aea75d191a7faf3ef009a8/coverage-7.13.5-cp312-cp312-win_arm64.whl", hash = "sha256:bf69236a9a81bdca3bff53796237aab096cdbf8d78a66ad61e992d9dac7eb2de", size = 221515 },
{ url = "https://files.pythonhosted.org/packages/74/8c/74fedc9663dcf168b0a059d4ea756ecae4da77a489048f94b5f512a8d0b3/coverage-7.13.5-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:5ec4af212df513e399cf11610cc27063f1586419e814755ab362e50a85ea69c1", size = 219576 },
{ url = "https://files.pythonhosted.org/packages/0c/c9/44fb661c55062f0818a6ffd2685c67aa30816200d5f2817543717d4b92eb/coverage-7.13.5-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:941617e518602e2d64942c88ec8499f7fbd49d3f6c4327d3a71d43a1973032f3", size = 219942 },
{ url = "https://files.pythonhosted.org/packages/5f/13/93419671cee82b780bab7ea96b67c8ef448f5f295f36bf5031154ec9a790/coverage-7.13.5-cp313-cp313-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:da305e9937617ee95c2e39d8ff9f040e0487cbf1ac174f777ed5eddd7a7c1f26", size = 250935 },
{ url = "https://files.pythonhosted.org/packages/ac/68/1666e3a4462f8202d836920114fa7a5ee9275d1fa45366d336c551a162dd/coverage-7.13.5-cp313-cp313-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:78e696e1cc714e57e8b25760b33a8b1026b7048d270140d25dafe1b0a1ee05a3", size = 253541 },
{ url = "https://files.pythonhosted.org/packages/4e/5e/3ee3b835647be646dcf3c65a7c6c18f87c27326a858f72ab22c12730773d/coverage-7.13.5-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:02ca0eed225b2ff301c474aeeeae27d26e2537942aa0f87491d3e147e784a82b", size = 254780 },
{ url = "https://files.pythonhosted.org/packages/44/b3/cb5bd1a04cfcc49ede6cd8409d80bee17661167686741e041abc7ee1b9a9/coverage-7.13.5-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:04690832cbea4e4663d9149e05dba142546ca05cb1848816760e7f58285c970a", size = 256912 },
{ url = "https://files.pythonhosted.org/packages/1b/66/c1dceb7b9714473800b075f5c8a84f4588f887a90eb8645282031676e242/coverage-7.13.5-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:0590e44dd2745c696a778f7bab6aa95256de2cbc8b8cff4f7db8ff09813d6969", size = 251165 },
{ url = "https://files.pythonhosted.org/packages/b7/62/5502b73b97aa2e53ea22a39cf8649ff44827bef76d90bf638777daa27a9d/coverage-7.13.5-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:d7cfad2d6d81dd298ab6b89fe72c3b7b05ec7544bdda3b707ddaecff8d25c161", size = 252908 },
{ url = "https://files.pythonhosted.org/packages/7d/37/7792c2d69854397ca77a55c4646e5897c467928b0e27f2d235d83b5d08c6/coverage-7.13.5-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:e092b9499de38ae0fbfbc603a74660eb6ff3e869e507b50d85a13b6db9863e15", size = 250873 },
{ url = "https://files.pythonhosted.org/packages/a3/23/bc866fb6163be52a8a9e5d708ba0d3b1283c12158cefca0a8bbb6e247a43/coverage-7.13.5-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:48c39bc4a04d983a54a705a6389512883d4a3b9862991b3617d547940e9f52b1", size = 255030 },
{ url = "https://files.pythonhosted.org/packages/7d/8b/ef67e1c222ef49860701d346b8bbb70881bef283bd5f6cbba68a39a086c7/coverage-7.13.5-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:2d3807015f138ffea1ed9afeeb8624fd781703f2858b62a8dd8da5a0994c57b6", size = 250694 },
{ url = "https://files.pythonhosted.org/packages/46/0d/866d1f74f0acddbb906db212e096dee77a8e2158ca5e6bb44729f9d93298/coverage-7.13.5-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:ee2aa19e03161671ec964004fb74b2257805d9710bf14a5c704558b9d8dbaf17", size = 252469 },
{ url = "https://files.pythonhosted.org/packages/7a/f5/be742fec31118f02ce42b21c6af187ad6a344fed546b56ca60caacc6a9a0/coverage-7.13.5-cp313-cp313-win32.whl", hash = "sha256:ce1998c0483007608c8382f4ff50164bfc5bd07a2246dd272aa4043b75e61e85", size = 222112 },
{ url = "https://files.pythonhosted.org/packages/66/40/7732d648ab9d069a46e686043241f01206348e2bbf128daea85be4d6414b/coverage-7.13.5-cp313-cp313-win_amd64.whl", hash = "sha256:631efb83f01569670a5e866ceb80fe483e7c159fac6f167e6571522636104a0b", size = 222923 },
{ url = "https://files.pythonhosted.org/packages/48/af/fea819c12a095781f6ccd504890aaddaf88b8fab263c4940e82c7b770124/coverage-7.13.5-cp313-cp313-win_arm64.whl", hash = "sha256:f4cd16206ad171cbc2470dbea9103cf9a7607d5fe8c242fdf1edf36174020664", size = 221540 },
{ url = "https://files.pythonhosted.org/packages/23/d2/17879af479df7fbbd44bd528a31692a48f6b25055d16482fdf5cdb633805/coverage-7.13.5-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:0428cbef5783ad91fe240f673cc1f76b25e74bbfe1a13115e4aa30d3f538162d", size = 220262 },
{ url = "https://files.pythonhosted.org/packages/5b/4c/d20e554f988c8f91d6a02c5118f9abbbf73a8768a3048cb4962230d5743f/coverage-7.13.5-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:e0b216a19534b2427cc201a26c25da4a48633f29a487c61258643e89d28200c0", size = 220617 },
{ url = "https://files.pythonhosted.org/packages/29/9c/f9f5277b95184f764b24e7231e166dfdb5780a46d408a2ac665969416d61/coverage-7.13.5-cp313-cp313t-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:972a9cd27894afe4bc2b1480107054e062df08e671df7c2f18c205e805ccd806", size = 261912 },
{ url = "https://files.pythonhosted.org/packages/d5/f6/7f1ab39393eeb50cfe4747ae8ef0e4fc564b989225aa1152e13a180d74f8/coverage-7.13.5-cp313-cp313t-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:4b59148601efcd2bac8c4dbf1f0ad6391693ccf7a74b8205781751637076aee3", size = 263987 },
{ url = "https://files.pythonhosted.org/packages/a0/d7/62c084fb489ed9c6fbdf57e006752e7c516ea46fd690e5ed8b8617c7d52e/coverage-7.13.5-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:505d7083c8b0c87a8fa8c07370c285847c1f77739b22e299ad75a6af6c32c5c9", size = 266416 },
{ url = "https://files.pythonhosted.org/packages/a9/f6/df63d8660e1a0bff6125947afda112a0502736f470d62ca68b288ea762d8/coverage-7.13.5-cp313-cp313t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:60365289c3741e4db327e7baff2a4aaacf22f788e80fa4683393891b70a89fbd", size = 267558 },
{ url = "https://files.pythonhosted.org/packages/5b/02/353ca81d36779bd108f6d384425f7139ac3c58c750dcfaafe5d0bee6436b/coverage-7.13.5-cp313-cp313t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:1b88c69c8ef5d4b6fe7dea66d6636056a0f6a7527c440e890cf9259011f5e606", size = 261163 },
{ url = "https://files.pythonhosted.org/packages/2c/16/2e79106d5749bcaf3aee6d309123548e3276517cd7851faa8da213bc61bf/coverage-7.13.5-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:5b13955d31d1633cf9376908089b7cebe7d15ddad7aeaabcbe969a595a97e95e", size = 263981 },
{ url = "https://files.pythonhosted.org/packages/29/c7/c29e0c59ffa6942030ae6f50b88ae49988e7e8da06de7ecdbf49c6d4feae/coverage-7.13.5-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:f70c9ab2595c56f81a89620e22899eea8b212a4041bd728ac6f4a28bf5d3ddd0", size = 261604 },
{ url = "https://files.pythonhosted.org/packages/40/48/097cdc3db342f34006a308ab41c3a7c11c3f0d84750d340f45d88a782e00/coverage-7.13.5-cp313-cp313t-musllinux_1_2_ppc64le.whl", hash = "sha256:084b84a8c63e8d6fc7e3931b316a9bcafca1458d753c539db82d31ed20091a87", size = 265321 },
{ url = "https://files.pythonhosted.org/packages/bb/1f/4994af354689e14fd03a75f8ec85a9a68d94e0188bbdab3fc1516b55e512/coverage-7.13.5-cp313-cp313t-musllinux_1_2_riscv64.whl", hash = "sha256:ad14385487393e386e2ea988b09d62dd42c397662ac2dabc3832d71253eee479", size = 260502 },
{ url = "https://files.pythonhosted.org/packages/22/c6/9bb9ef55903e628033560885f5c31aa227e46878118b63ab15dc7ba87797/coverage-7.13.5-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:7f2c47b36fe7709a6e83bfadf4eefb90bd25fbe4014d715224c4316f808e59a2", size = 262688 },
{ url = "https://files.pythonhosted.org/packages/14/4f/f5df9007e50b15e53e01edea486814783a7f019893733d9e4d6caad75557/coverage-7.13.5-cp313-cp313t-win32.whl", hash = "sha256:67e9bc5449801fad0e5dff329499fb090ba4c5800b86805c80617b4e29809b2a", size = 222788 },
{ url = "https://files.pythonhosted.org/packages/e1/98/aa7fccaa97d0f3192bec013c4e6fd6d294a6ed44b640e6bb61f479e00ed5/coverage-7.13.5-cp313-cp313t-win_amd64.whl", hash = "sha256:da86cdcf10d2519e10cabb8ac2de03da1bcb6e4853790b7fbd48523332e3a819", size = 223851 },
{ url = "https://files.pythonhosted.org/packages/3d/8b/e5c469f7352651e5f013198e9e21f97510b23de957dd06a84071683b4b60/coverage-7.13.5-cp313-cp313t-win_arm64.whl", hash = "sha256:0ecf12ecb326fe2c339d93fc131816f3a7367d223db37817208905c89bded911", size = 222104 },
{ url = "https://files.pythonhosted.org/packages/8e/77/39703f0d1d4b478bfd30191d3c14f53caf596fac00efb3f8f6ee23646439/coverage-7.13.5-cp314-cp314-macosx_10_15_x86_64.whl", hash = "sha256:fbabfaceaeb587e16f7008f7795cd80d20ec548dc7f94fbb0d4ec2e038ce563f", size = 219621 },
{ url = "https://files.pythonhosted.org/packages/e2/3e/51dff36d99ae14639a133d9b164d63e628532e2974d8b1edb99dd1ebc733/coverage-7.13.5-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:9bb2a28101a443669a423b665939381084412b81c3f8c0fcfbac57f4e30b5b8e", size = 219953 },
{ url = "https://files.pythonhosted.org/packages/6a/6c/1f1917b01eb647c2f2adc9962bd66c79eb978951cab61bdc1acab3290c07/coverage-7.13.5-cp314-cp314-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:bd3a2fbc1c6cccb3c5106140d87cc6a8715110373ef42b63cf5aea29df8c217a", size = 250992 },
{ url = "https://files.pythonhosted.org/packages/22/e5/06b1f88f42a5a99df42ce61208bdec3bddb3d261412874280a19796fc09c/coverage-7.13.5-cp314-cp314-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:6c36ddb64ed9d7e496028d1d00dfec3e428e0aabf4006583bb1839958d280510", size = 253503 },
{ url = "https://files.pythonhosted.org/packages/80/28/2a148a51e5907e504fa7b85490277734e6771d8844ebcc48764a15e28155/coverage-7.13.5-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:380e8e9084d8eb38db3a9176a1a4f3c0082c3806fa0dc882d1d87abc3c789247", size = 254852 },
{ url = "https://files.pythonhosted.org/packages/61/77/50e8d3d85cc0b7ebe09f30f151d670e302c7ff4a1bf6243f71dd8b0981fa/coverage-7.13.5-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:e808af52a0513762df4d945ea164a24b37f2f518cbe97e03deaa0ee66139b4d6", size = 257161 },
{ url = "https://files.pythonhosted.org/packages/3b/c4/b5fd1d4b7bf8d0e75d997afd3925c59ba629fc8616f1b3aae7605132e256/coverage-7.13.5-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:e301d30dd7e95ae068671d746ba8c34e945a82682e62918e41b2679acd2051a0", size = 251021 },
{ url = "https://files.pythonhosted.org/packages/f8/66/6ea21f910e92d69ef0b1c3346ea5922a51bad4446c9126db2ae96ee24c4c/coverage-7.13.5-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:800bc829053c80d240a687ceeb927a94fd108bbdc68dfbe505d0d75ab578a882", size = 252858 },
{ url = "https://files.pythonhosted.org/packages/9e/ea/879c83cb5d61aa2a35fb80e72715e92672daef8191b84911a643f533840c/coverage-7.13.5-cp314-cp314-musllinux_1_2_i686.whl", hash = "sha256:0b67af5492adb31940ee418a5a655c28e48165da5afab8c7fa6fd72a142f8740", size = 250823 },
{ url = "https://files.pythonhosted.org/packages/8a/fb/616d95d3adb88b9803b275580bdeee8bd1b69a886d057652521f83d7322f/coverage-7.13.5-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:c9136ff29c3a91e25b1d1552b5308e53a1e0653a23e53b6366d7c2dcbbaf8a16", size = 255099 },
{ url = "https://files.pythonhosted.org/packages/1c/93/25e6917c90ec1c9a56b0b26f6cad6408e5f13bb6b35d484a0d75c9cf000d/coverage-7.13.5-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:cff784eef7f0b8f6cb28804fbddcfa99f89efe4cc35fb5627e3ac58f91ed3ac0", size = 250638 },
{ url = "https://files.pythonhosted.org/packages/fc/7b/dc1776b0464145a929deed214aef9fb1493f159b59ff3c7eeeedf91eddd0/coverage-7.13.5-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:68a4953be99b17ac3c23b6efbc8a38330d99680c9458927491d18700ef23ded0", size = 252295 },
{ url = "https://files.pythonhosted.org/packages/ea/fb/99cbbc56a26e07762a2740713f3c8f9f3f3106e3a3dd8cc4474954bccd34/coverage-7.13.5-cp314-cp314-win32.whl", hash = "sha256:35a31f2b1578185fbe6aa2e74cea1b1d0bbf4c552774247d9160d29b80ed56cc", size = 222360 },
{ url = "https://files.pythonhosted.org/packages/8d/b7/4758d4f73fb536347cc5e4ad63662f9d60ba9118cb6785e9616b2ce5d7fa/coverage-7.13.5-cp314-cp314-win_amd64.whl", hash = "sha256:2aa055ae1857258f9e0045be26a6d62bdb47a72448b62d7b55f4820f361a2633", size = 223174 },
{ url = "https://files.pythonhosted.org/packages/2c/f2/24d84e1dfe70f8ac9fdf30d338239860d0d1d5da0bda528959d0ebc9da28/coverage-7.13.5-cp314-cp314-win_arm64.whl", hash = "sha256:1b11eef33edeae9d142f9b4358edb76273b3bfd30bc3df9a4f95d0e49caf94e8", size = 221739 },
{ url = "https://files.pythonhosted.org/packages/60/5b/4a168591057b3668c2428bff25dd3ebc21b629d666d90bcdfa0217940e84/coverage-7.13.5-cp314-cp314t-macosx_10_15_x86_64.whl", hash = "sha256:10a0c37f0b646eaff7cce1874c31d1f1ccb297688d4c747291f4f4c70741cc8b", size = 220351 },
{ url = "https://files.pythonhosted.org/packages/f5/21/1fd5c4dbfe4a58b6b99649125635df46decdfd4a784c3cd6d410d303e370/coverage-7.13.5-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:b5db73ba3c41c7008037fa731ad5459fc3944cb7452fc0aa9f822ad3533c583c", size = 220612 },
{ url = "https://files.pythonhosted.org/packages/d6/fe/2a924b3055a5e7e4512655a9d4609781b0d62334fa0140c3e742926834e2/coverage-7.13.5-cp314-cp314t-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:750db93a81e3e5a9831b534be7b1229df848b2e125a604fe6651e48aa070e5f9", size = 261985 },
{ url = "https://files.pythonhosted.org/packages/d7/0d/c8928f2bd518c45990fe1a2ab8db42e914ef9b726c975facc4282578c3eb/coverage-7.13.5-cp314-cp314t-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:9ddb4f4a5479f2539644be484da179b653273bca1a323947d48ab107b3ed1f29", size = 264107 },
{ url = "https://files.pythonhosted.org/packages/ef/ae/4ae35bbd9a0af9d820362751f0766582833c211224b38665c0f8de3d487f/coverage-7.13.5-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:d8a7a2049c14f413163e2bdabd37e41179b1d1ccb10ffc6ccc4b7a718429c607", size = 266513 },
{ url = "https://files.pythonhosted.org/packages/9c/20/d326174c55af36f74eac6ae781612d9492f060ce8244b570bb9d50d9d609/coverage-7.13.5-cp314-cp314t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:e1c85e0b6c05c592ea6d8768a66a254bfb3874b53774b12d4c89c481eb78cb90", size = 267650 },
{ url = "https://files.pythonhosted.org/packages/7a/5e/31484d62cbd0eabd3412e30d74386ece4a0837d4f6c3040a653878bfc019/coverage-7.13.5-cp314-cp314t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:777c4d1eff1b67876139d24288aaf1817f6c03d6bae9c5cc8d27b83bcfe38fe3", size = 261089 },
{ url = "https://files.pythonhosted.org/packages/e9/d8/49a72d6de146eebb0b7e48cc0f4bc2c0dd858e3d4790ab2b39a2872b62bd/coverage-7.13.5-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:6697e29b93707167687543480a40f0db8f356e86d9f67ddf2e37e2dfd91a9dab", size = 263982 },
{ url = "https://files.pythonhosted.org/packages/06/3b/0351f1bd566e6e4dd39e978efe7958bde1d32f879e85589de147654f57bb/coverage-7.13.5-cp314-cp314t-musllinux_1_2_i686.whl", hash = "sha256:8fdf453a942c3e4d99bd80088141c4c6960bb232c409d9c3558e2dbaa3998562", size = 261579 },
{ url = "https://files.pythonhosted.org/packages/5d/ce/796a2a2f4017f554d7810f5c573449b35b1e46788424a548d4d19201b222/coverage-7.13.5-cp314-cp314t-musllinux_1_2_ppc64le.whl", hash = "sha256:32ca0c0114c9834a43f045a87dcebd69d108d8ffb666957ea65aa132f50332e2", size = 265316 },
{ url = "https://files.pythonhosted.org/packages/3d/16/d5ae91455541d1a78bc90abf495be600588aff8f6db5c8b0dae739fa39c9/coverage-7.13.5-cp314-cp314t-musllinux_1_2_riscv64.whl", hash = "sha256:8769751c10f339021e2638cd354e13adeac54004d1941119b2c96fe5276d45ea", size = 260427 },
{ url = "https://files.pythonhosted.org/packages/48/11/07f413dba62db21fb3fad5d0de013a50e073cc4e2dc4306e770360f6dfc8/coverage-7.13.5-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:cec2d83125531bd153175354055cdb7a09987af08a9430bd173c937c6d0fba2a", size = 262745 },
{ url = "https://files.pythonhosted.org/packages/91/15/d792371332eb4663115becf4bad47e047d16234b1aff687b1b18c58d60ae/coverage-7.13.5-cp314-cp314t-win32.whl", hash = "sha256:0cd9ed7a8b181775459296e402ca4fb27db1279740a24e93b3b41942ebe4b215", size = 223146 },
{ url = "https://files.pythonhosted.org/packages/db/51/37221f59a111dca5e85be7dbf09696323b5b9f13ff65e0641d535ed06ea8/coverage-7.13.5-cp314-cp314t-win_amd64.whl", hash = "sha256:301e3b7dfefecaca37c9f1aa6f0049b7d4ab8dd933742b607765d757aca77d43", size = 224254 },
{ url = "https://files.pythonhosted.org/packages/54/83/6acacc889de8987441aa7d5adfbdbf33d288dad28704a67e574f1df9bcbb/coverage-7.13.5-cp314-cp314t-win_arm64.whl", hash = "sha256:9dacc2ad679b292709e0f5fc1ac74a6d4d5562e424058962c7bb0c658ad25e45", size = 222276 },
{ url = "https://files.pythonhosted.org/packages/9e/ee/a4cf96b8ce1e566ed238f0659ac2d3f007ed1d14b181bcb684e19561a69a/coverage-7.13.5-py3-none-any.whl", hash = "sha256:34b02417cf070e173989b3db962f7ed56d2f644307b2cf9d5a0f258e13084a61", size = 211346 },
]
[[package]]
name = "dj-database-url"
version = "3.1.2"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "django" },
]
sdist = { url = "https://files.pythonhosted.org/packages/03/f6/00b625e9d371b980aa261011d0dc906a16444cb688f94215e0dc86996eb5/dj_database_url-3.1.2.tar.gz", hash = "sha256:63c20e4bbaa51690dfd4c8d189521f6bf6bc9da9fcdb23d95d2ee8ee87f9ec62", size = 11490 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/cf/a9/57c66006373381f1d3e5bd94216f1d371228a89f443d3030e010f73dd198/dj_database_url-3.1.2-py3-none-any.whl", hash = "sha256:544e015fee3efa5127a1eb1cca465f4ace578265b3671fe61d0ed7dbafb5ec8a", size = 8953 },
]
[[package]]
name = "django"
version = "6.0.5"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "asgiref" },
{ name = "sqlparse" },
{ name = "tzdata", marker = "sys_platform == 'win32'" },
]
sdist = { url = "https://files.pythonhosted.org/packages/5e/f1/bf85f0d29ef76abf901f193fe8fef4769d3da7794197832bc30151c071d8/django-6.0.5.tar.gz", hash = "sha256:bc6d6872e98a2864c836e42edd644b362db311147dd5aa8d5b82ba7a032f5269", size = 10924131 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/94/5b/1328f8b84fce040c404f76822bf8c57d254e368e8cbd8bd67ec2b26d75f5/django-6.0.5-py3-none-any.whl", hash = "sha256:9d58a7cb49244e74c8e161d5e403a46d6209f1009ba40f5a66d6aa0d0786a8f0", size = 8368680 },
]
[[package]]
name = "django-storages"
version = "1.14.6"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "django" },
]
sdist = { url = "https://files.pythonhosted.org/packages/ff/d6/2e50e378fff0408d558f36c4acffc090f9a641fd6e084af9e54d45307efa/django_storages-1.14.6.tar.gz", hash = "sha256:7a25ce8f4214f69ac9c7ce87e2603887f7ae99326c316bc8d2d75375e09341c9", size = 87587 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/1f/21/3cedee63417bc5553eed0c204be478071c9ab208e5e259e97287590194f1/django_storages-1.14.6-py3-none-any.whl", hash = "sha256:11b7b6200e1cb5ffcd9962bd3673a39c7d6a6109e8096f0e03d46fab3d3aabd9", size = 33095 },
]
[[package]]
name = "django-stubs"
version = "6.0.3"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "django" },
{ name = "django-stubs-ext" },
{ name = "types-pyyaml" },
{ name = "typing-extensions" },
]
sdist = { url = "https://files.pythonhosted.org/packages/86/0c/8d0d875af79bf774c1c3997c84aa118dba3a77be12086b9c14e130e8ec72/django_stubs-6.0.3.tar.gz", hash = "sha256:ee895f403c373608eeb50822f0733f9d9ec5ab12731d4ab58956053bb95fdd9e", size = 278214 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/80/a3/6751b7684d20fc4f228bdd3dd8341d382ab3faaf65d3d050c0d59ab0a1b0/django_stubs-6.0.3-py3-none-any.whl", hash = "sha256:5fee22bcbbad59a78c727a820b6f4e68ff442ca76a922b7002e57c25dd7cb390", size = 541570 },
]
[[package]]
name = "django-stubs-ext"
version = "6.0.3"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "django" },
{ name = "typing-extensions" },
]
sdist = { url = "https://files.pythonhosted.org/packages/fb/e6/5dcdaa785ec3eed5fc196c7e68fb7ad9d9fe6d5acccea4690e65f2546417/django_stubs_ext-6.0.3.tar.gz", hash = "sha256:3307d42132bc295d5744de6276bc5fdf6896efc70f891e21c0ae8bdf529d2762", size = 6663 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/10/fa/0a3a05c29d6295dbd52fa3cb4047a95de11ba4f2696072d6f3f2c1e6f370/django_stubs_ext-6.0.3-py3-none-any.whl", hash = "sha256:9e4105955419ae310d7da9cfd808e039d4dae3092c628f021057bb4f2c237f8f", size = 10354 },
]
[[package]]
name = "factory-boy"
version = "3.3.3"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "faker" },
]
sdist = { url = "https://files.pythonhosted.org/packages/ba/98/75cacae9945f67cfe323829fc2ac451f64517a8a330b572a06a323997065/factory_boy-3.3.3.tar.gz", hash = "sha256:866862d226128dfac7f2b4160287e899daf54f2612778327dd03d0e2cb1e3d03", size = 164146 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/27/8d/2bc5f5546ff2ccb3f7de06742853483ab75bf74f36a92254702f8baecc79/factory_boy-3.3.3-py2.py3-none-any.whl", hash = "sha256:1c39e3289f7e667c4285433f305f8d506efc2fe9c73aaea4151ebd5cdea394fc", size = 37036 },
]
[[package]]
name = "faker"
version = "40.15.0"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "tzdata", marker = "platform_system == 'Windows'" },
]
sdist = { url = "https://files.pythonhosted.org/packages/7f/13/6741787bd91c4109c7bed047d68273965cd52ce8a5f773c471b949334b6d/faker-40.15.0.tar.gz", hash = "sha256:20f3a6ec8c266b74d4c554e34118b21c3c2056c0b4a519d15c8decb3a4e6e795", size = 1967447 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/a7/a7/a600f8f30d4505e89166de51dd121bd540ab8e560e8cf0901de00a81de8c/faker-40.15.0-py3-none-any.whl", hash = "sha256:71ab3c3370da9d2205ab74ffb0fd51273063ad562b3a3bb69d0026a20923e318", size = 2004447 },
]
[[package]]
name = "iniconfig"
version = "2.3.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/72/34/14ca021ce8e5dfedc35312d08ba8bf51fdd999c576889fc2c24cb97f4f10/iniconfig-2.3.0.tar.gz", hash = "sha256:c76315c77db068650d49c5b56314774a7804df16fee4402c1f19d6d15d8c4730", size = 20503 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/cb/b1/3846dd7f199d53cb17f49cba7e651e9ce294d8497c8c150530ed11865bb8/iniconfig-2.3.0-py3-none-any.whl", hash = "sha256:f631c04d2c48c52b84d0d0549c99ff3859c98df65b3101406327ecc7d53fbf12", size = 7484 },
]
[[package]]
name = "librt"
version = "0.10.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/39/cb/c1945e506893b5b8577fb45a60c80e3ffe4a82092a04a6f29b0b951d9a24/librt-0.10.0.tar.gz", hash = "sha256:1aba1e8aa4e3307a7be68a74149545fde7451964dc0235a8bec5704a17bdda42", size = 191799 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/12/8e/cbb5b6f6e45e65c10a42449a69eaccc44d73e6a081ea752fbc5221c6dc1c/librt-0.10.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:b4b58a44b407e91f633dafee008de9ddea6aa2a555ed94929c099260910bd0ba", size = 77327 },
{ url = "https://files.pythonhosted.org/packages/e9/3d/8233cbee8e99e6a8992f02bfc2dec8d787509566a511d1fde2574ee7473f/librt-0.10.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:950b79b11762531bdf45a9df909d2f9a2a8445c70c88665c01d14c8511a27dc5", size = 79971 },
{ url = "https://files.pythonhosted.org/packages/87/6f/5264b298cef2b72fc97d2dde56c66181eda35204bf5dcd1ed0c3d0a0a782/librt-0.10.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4538453f51be197633b425912c150e25b0667252d3741c53e8368176d98d9d37", size = 246559 },
{ url = "https://files.pythonhosted.org/packages/07/7b/19b1b859cc60d5f99276cc2b3144d91556c6d1b1e4ebb50359696bebf7a8/librt-0.10.0-cp312-cp312-manylinux2014_i686.manylinux_2_17_i686.manylinux_2_28_i686.whl", hash = "sha256:70b955f091beac93e994a0b7ec616934f63b3ea5c3d6d7af847562f935aceca7", size = 235216 },
{ url = "https://files.pythonhosted.org/packages/6e/56/a2f40717142a8af46289f57874ef914353d8faccd5e4f8e594ab1e16e8c7/librt-0.10.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:483e685e06b6163728ba6c85d74315176be7190f432ec2a41226e5e14355d5f0", size = 263108 },
{ url = "https://files.pythonhosted.org/packages/67/ca/15c625c3bdc0167c01e04ef8878317e9713f3bfa788438342f7a94c7b22c/librt-0.10.0-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:7ac53d946a009d1a38c44a60812708c9458fb2a239a5f630d8e625571386650f", size = 255280 },
{ url = "https://files.pythonhosted.org/packages/ed/c5/ba301d571d9e05844e2435b73aba30bee77bb75ce155c9affcfd2173dd03/librt-0.10.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:bc8771c9fcf0ea894ca41fdc2abd83572c2fbda221f232d86e718614e57ff513", size = 268829 },
{ url = "https://files.pythonhosted.org/packages/8b/60/af70e135bc1f1fe15dd3894b1e4bbefc7ecdf911749a925a39eb86ceb2a1/librt-0.10.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:70805dbc5257892ac572f86290a61e3c8d90224ecce1a8b2d1f7ed51965417f4", size = 262051 },
{ url = "https://files.pythonhosted.org/packages/83/c2/c8236eb8b421bac5a172ba208f965abaa89805da2a3fa112bdf1764caf8f/librt-0.10.0-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:d3b4f300f7bcba6e2ff73fb8bef1898479e9772bfa2682998c636391633ec826", size = 264347 },
{ url = "https://files.pythonhosted.org/packages/d6/f5/15b6d32bc25dacd4a60886a683d8128d6219910c122202b995a40dd4f8d2/librt-0.10.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:943bc943f92f4fb3408fae62485c6a3ad68ce4f2ee205643a39641525c19a276", size = 286482 },
{ url = "https://files.pythonhosted.org/packages/fb/8e/b1b959bacd323eb4360579db992513e1406d1c6ef7edb57b5511fd0666fd/librt-0.10.0-cp312-cp312-win32.whl", hash = "sha256:6065c1a758fba1010b41401013903d3d5d2750eab425ddedd584abac31d0630e", size = 62955 },
{ url = "https://files.pythonhosted.org/packages/9e/4c/d4cd6e4b9fc24098e63cc85537d1b6689682aee96809c38f08072067cc2b/librt-0.10.0-cp312-cp312-win_amd64.whl", hash = "sha256:d788ecbe208ab352dab0e105cc06057bf9a2fc7e58cabb0d751ad9e30062b9e2", size = 71191 },
{ url = "https://files.pythonhosted.org/packages/2b/19/8641da1f63d24b92354a492f893c022d6b3a0df44e70c8eff49364613983/librt-0.10.0-cp312-cp312-win_arm64.whl", hash = "sha256:6003d1f295bdba02656dc81308208fc060d0a51d8c0d0a6db70f7f3c57b9ba0a", size = 61432 },
{ url = "https://files.pythonhosted.org/packages/e5/29/681a75c82f4cc90d29e4b257a3299b79fe13fe927a04c57b8109d70b6957/librt-0.10.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:f0ede79d682e73f91c1b599a76d78b7464b9b5d213754cedb13372d9df36e596", size = 77299 },
{ url = "https://files.pythonhosted.org/packages/62/24/0c7ca445a55d04be79cac19819437fd094782347fa116f6681844fa6143e/librt-0.10.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:e0ba0b131fdb336c8b9c948e397f4a7e649d0f783b529f07b647bf4961df392e", size = 79930 },
{ url = "https://files.pythonhosted.org/packages/fe/1f/1e2b8f6443ef9e9a81e89486ca70e22f3684f93db003ce6eaefc3d0839b9/librt-0.10.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:2728117da2afb96fb957768725ee43dc9a2d73b031e02da424b818a3cdd3a275", size = 246195 },
{ url = "https://files.pythonhosted.org/packages/74/61/9dc9e03de0439ad84c1c240aac8b747f12c90cb797ea6042f7bdb8d3410f/librt-0.10.0-cp313-cp313-manylinux2014_i686.manylinux_2_17_i686.manylinux_2_28_i686.whl", hash = "sha256:723ba80594c49cdf0584196fc430752262605dc9449902fc9bd3d9b79976cb77", size = 234951 },
{ url = "https://files.pythonhosted.org/packages/55/f4/635223117d7590875bca441275065a3bf491203ad4208bd1cc3ffd90c5a1/librt-0.10.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:7292edaaca294a61a978c53a3c7d6130d099b0dfbc8f0a65916cdc6b891b9852", size = 262768 },
{ url = "https://files.pythonhosted.org/packages/e5/66/b04152d0cd8b6ca2b428a8bd3230343230c35ed304a932f35b5375f2f828/librt-0.10.0-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:89fe9d539f2c10a1666633eeeac507ce95dd06d9ecc58de3c6390dba156a3d3a", size = 255075 },
{ url = "https://files.pythonhosted.org/packages/35/1e/25bac4c7f2ca36f0e612cade186970683cf79153d96beccc3a11a9e19b97/librt-0.10.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:4efa7b9587503fa5b67f40593302b9c8836d211d222ff9f7cafe67be5f8f0b10", size = 268559 },
{ url = "https://files.pythonhosted.org/packages/18/54/4601faab35b6632a13200faa146ca62bfd111ffbe2568be430d65c89493a/librt-0.10.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:22dc982ef59df0136df36092ccbdbb570ced8aafb33e49585739b2f1de1c13b6", size = 261753 },
{ url = "https://files.pythonhosted.org/packages/1b/cf/39f4023509e94fade8b074666fa3292db9cb6b34ea5dcbe7af53df9fca1d/librt-0.10.0-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:6f2e5f3606253a84cea719c94a3bb1c54487b5d617d0254d46e0920d8a06be3f", size = 264055 },
{ url = "https://files.pythonhosted.org/packages/8e/00/40247209fc46a8e308a91412d5206aedf8efb667ee89eb625820106a5c2f/librt-0.10.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:40884bfaa1e29f6b6a9be255007d8f359bfc9e61d68bdef8ed3158bfcbc95df9", size = 286190 },
{ url = "https://files.pythonhosted.org/packages/d8/6e/5566beb94431a985abe1787af5ef86e087750172ff9d0bbf20f93e88132d/librt-0.10.0-cp313-cp313-win32.whl", hash = "sha256:3cd34cd8254eba756660bff6c2da91278248184301054fe3e4feb073bdd49b14", size = 62949 },
{ url = "https://files.pythonhosted.org/packages/d0/c2/3ea3301d6c8dff51d39dbe8ed75db3dc92896947d4afb5eeadf821c1e67f/librt-0.10.0-cp313-cp313-win_amd64.whl", hash = "sha256:7baac5313e2d8dce1386f97777a8d03ab28f5fe1e780b3b9ac2ee7544551fedc", size = 71152 },
{ url = "https://files.pythonhosted.org/packages/3c/de/5d49cb92cadcbc77d3abc27b93fd6030ed8437487dde2eae38cab5e6704d/librt-0.10.0-cp313-cp313-win_arm64.whl", hash = "sha256:afc5b4406c8e2515698d922a5c7823a009312835ea58196671fff40e35cb8166", size = 61336 },
{ url = "https://files.pythonhosted.org/packages/6a/64/7165e08108cc185a13a9c069f0685e6ef92e70e07fddf7edf5e7348c6316/librt-0.10.0-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:f09588a30e6a22ec624090d72a3ab1a6d4d5485c3ed739603e76aa3c16efa688", size = 76794 },
{ url = "https://files.pythonhosted.org/packages/ae/ef/bf8613febf651b90c5222ee79dea5ae58d4cc2b544df69d3033424448934/librt-0.10.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:131ade118d12bd7a0adc4e655474a553f1b76cf78385868885944d21d51e45e0", size = 79662 },
{ url = "https://files.pythonhosted.org/packages/b6/67/9eddd165c1d8397bdf99b38bf12b5a55b3def5035b49eedb49f2775d1430/librt-0.10.0-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b8b9ab28e40d011c373a189eae900c916e66d6fbecf7983e9e4883089ee085ef", size = 242390 },
{ url = "https://files.pythonhosted.org/packages/10/d1/d95da80334501866cd37004ab5d7483220d05862fab4b5405394f0264f0d/librt-0.10.0-cp314-cp314-manylinux2014_i686.manylinux_2_17_i686.manylinux_2_28_i686.whl", hash = "sha256:67c39bb30da73bae1f293d1ed8bc2f8f6642649dd0928d3600aeff3041ac23d6", size = 232603 },
{ url = "https://files.pythonhosted.org/packages/0c/fa/e6d64d28718bc1be4e1736fcb037ca1c4dfca927e7167df75a7d5215665e/librt-0.10.0-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:8c3273c6b774614f093c8927c2bf1b077d0fefde988fe98f46a333734e5597ab", size = 259187 },
{ url = "https://files.pythonhosted.org/packages/72/3f/3fdb77e7f937dad59cfd76b720be7e7643400ec76b2da35befab8d66ba30/librt-0.10.0-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:9dd7c1b86a4baa583ab5db977484b93a2c474e69e96ef3e9538387ea54229cb9", size = 251846 },
{ url = "https://files.pythonhosted.org/packages/18/ca/f4d49133dd86a6f55d79eca30bf412fa722f511a9abe67f62f57aa64e66a/librt-0.10.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:a77385c5a202e831149f7ad03be9e67cf80e957e52c614e83dcb822c95222eb8", size = 264936 },
{ url = "https://files.pythonhosted.org/packages/de/66/a8df2fbadc1f6c1827a096d11c40175bd526133480bd3bc88ec64a03d257/librt-0.10.0-cp314-cp314-musllinux_1_2_i686.whl", hash = "sha256:c6a5eafa74b5655bad59886138ed68426f098a6beb8cb95a71f2cc3cd8bb33fe", size = 258699 },
{ url = "https://files.pythonhosted.org/packages/bb/73/1e3c83613fe05451bb969e27b68a573d177f08d5f63533cc29fec0989658/librt-0.10.0-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:1fc93d0439204c50ab4d1512611ce2c206f1b369b419f69c7c27c761561e3291", size = 259825 },
{ url = "https://files.pythonhosted.org/packages/09/24/5e2f926ee9d3ef348d9339526d7062abb5c44d8419e3179528c01d78c102/librt-0.10.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:79e713c178bc7a744adfbee6b4619a288eecc0c914da2a9313a20255abe2f0cf", size = 282548 },
{ url = "https://files.pythonhosted.org/packages/fc/7d/3e89ed6ad0162561fa8bef9df3195e24263104c955713cd0237d3711fad2/librt-0.10.0-cp314-cp314-win32.whl", hash = "sha256:2eba9d955a68c41d9f326be3da42f163ec3518b7ab20f1c826224e7bed71e0bf", size = 58970 },
{ url = "https://files.pythonhosted.org/packages/76/25/579e731c94a7086a268bfa3e7a4945cd47836bebd3cbf3faeafd2e7eaef9/librt-0.10.0-cp314-cp314-win_amd64.whl", hash = "sha256:cbfaf7f5145e9917f5d18bffa298eff6a19d74e7b8b11dabdca95785befe8dbf", size = 67260 },
{ url = "https://files.pythonhosted.org/packages/6e/f8/235822b7ae0b2334f12ee18bcf2476d07924077a5efeea57dbe927704be2/librt-0.10.0-cp314-cp314-win_arm64.whl", hash = "sha256:8d6d385d1969849a6b1397114df22714b6ded917bada98668e3e974dc663477e", size = 57156 },
{ url = "https://files.pythonhosted.org/packages/9f/e3/9b919cbf1e8eb770bf91bb7df28125e0f1daf4587169afefd95402636e9a/librt-0.10.0-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:6c3a82d3bd32631ef5c79922dfc028520c9ad840255979ab4d908271818039ee", size = 79150 },
{ url = "https://files.pythonhosted.org/packages/6a/f5/72a944aa3bc3498169a168087eff58ca48b58bf1b704e59d091fd30739f3/librt-0.10.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:d64cc66005dc324c9bb1fa3fc2841f529002f6eb15966d55e46d430f56955a6a", size = 82304 },
{ url = "https://files.pythonhosted.org/packages/9c/e3/fcc290a33e295019759472dfa794d204e43504b276ac65eab7fd9da20ea3/librt-0.10.0-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9bb562cd28c88cd2c6a9a6c78f99dc39348d6b16c94adc25de0e574acf1176e9", size = 272556 },
{ url = "https://files.pythonhosted.org/packages/fd/54/546975e4c997573885e7f040a05012f8838e06fb12b0c3c1fbb76254e9d7/librt-0.10.0-cp314-cp314t-manylinux2014_i686.manylinux_2_17_i686.manylinux_2_28_i686.whl", hash = "sha256:b809aa2854d019c28773b03605df22adc675ee4f3f4402d673581313e8906119", size = 256941 },
{ url = "https://files.pythonhosted.org/packages/70/8c/f1d03401571b331653acddbd4e8cd955c06d945241dd08b25192fac0d04b/librt-0.10.0-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:cc15acabdd519bd4176fdadc2119e5e3093485d86f89138daf47e5b4cedb983a", size = 285855 },
{ url = "https://files.pythonhosted.org/packages/0c/08/62cf80ff046c339faf56718b3a940244d4beb70f1c6407289b5830ec11e9/librt-0.10.0-cp314-cp314t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:b1b2d835307d08ddadd94568e2369648ec9173bd3eea6d7f52a1abe717c81f98", size = 275321 },
{ url = "https://files.pythonhosted.org/packages/d9/ea/da5918d4070362e9a4d2ee9cd34f9dc84902daad8fd4275f8504a727ff4e/librt-0.10.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:d261c6a2f93335a5167887fb0223e8b98ffce20ee3fde242e8e58a37ece6d0e5", size = 293993 },
{ url = "https://files.pythonhosted.org/packages/c9/8d/68b6086bed1fcdc314c640ea04e31e52d18052e08059fa595409d66a51a9/librt-0.10.0-cp314-cp314t-musllinux_1_2_i686.whl", hash = "sha256:e2ffd44963f8e7f68995504d90f9881d64e94dc1d8e310039b9526108fc0c0f7", size = 284254 },
{ url = "https://files.pythonhosted.org/packages/06/c8/b810f1d84ec34a5a7ed93d7b510ab04164d75fbdf23088d5c3fbe6b08357/librt-0.10.0-cp314-cp314t-musllinux_1_2_riscv64.whl", hash = "sha256:5f285f6455ed495791c4d8630e5af732960adea93cac4c893d15619f2eae53e8", size = 284925 },
{ url = "https://files.pythonhosted.org/packages/5a/00/3c82d4158c5a2c62528b8fccce65a8c9ad700e480e86f9389387435089a5/librt-0.10.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:f6034ff52e663d34c7b82ef2aa2f94ad7c1d939e2368e63b06844bc4d127d2e1", size = 307830 },
{ url = "https://files.pythonhosted.org/packages/99/3a/9c635ac3e8a00383ff689161d3eac8a30b3b2ddc711b40471e6b8983ea29/librt-0.10.0-cp314-cp314t-win32.whl", hash = "sha256:657860fd877fba6a241ea088ef99f63ca819945d3c715265da670bad56c37ebe", size = 60147 },
{ url = "https://files.pythonhosted.org/packages/dc/e8/6f65f3e565d4ac212cddddd552eacc8035ffdf941ca0ad6fe945a211d41f/librt-0.10.0-cp314-cp314t-win_amd64.whl", hash = "sha256:56ded2d66010203a0cb5af063b609e3f079531a0e5e576d618dece859fd2e1af", size = 68649 },
{ url = "https://files.pythonhosted.org/packages/51/78/a0705a67cacd81e5fa01a5035b3adbdfbb43a7b8d4bd27e2b282ae61baf2/librt-0.10.0-cp314-cp314t-win_arm64.whl", hash = "sha256:1ee63f30abf18ed4830fdbaf87b2b6f4bba1e198d46085c314edde4045e56715", size = 58247 },
]
[[package]]
name = "mypy"
version = "2.0.0"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "ast-serialize" },
{ name = "librt", marker = "platform_python_implementation != 'PyPy'" },
{ name = "mypy-extensions" },
{ name = "pathspec" },
{ name = "typing-extensions" },
]
sdist = { url = "https://files.pythonhosted.org/packages/cf/dc/7e6d49f04fca40b9dd5c752a51a432ffe67fb45200702bc9eee0cb4bbb26/mypy-2.0.0.tar.gz", hash = "sha256:1a9e3900ac5c40f1fe813506c7739da6e6f0eab2729067ebd94bfb0bbba53532", size = 3869036 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/f6/4b/f6cd12ef1eb63be1c342da3e8ca811d2280276177f6de4ef20cb2366d79b/mypy-2.0.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:660790551c988e69d8bf7a35c8b4149edeb22f4a339165702be843532e9dcdb5", size = 14756610 },
{ url = "https://files.pythonhosted.org/packages/32/73/67d09ca28bee21feaca264b2a680cf2d300bcc2071136ad064928324c843/mypy-2.0.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:7a15bf92cd8781f8e72f69ffa7e30d1f434402d065ee1ecd5223ef2ef100f914", size = 13554270 },
{ url = "https://files.pythonhosted.org/packages/61/b3/44718b5c6b1b5a27440ff2effe6a1be0fa2a190c0f4e2e21a83728416f95/mypy-2.0.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4ff370b43d7def05bbcd2f5267f0bcda72dd6a552ef2ea9375b02d6fe06da270", size = 13924663 },
{ url = "https://files.pythonhosted.org/packages/6a/2b/bbb9cc5773f946846a7c340097e59bcf84095437dda0d56bb4f6cf1f6541/mypy-2.0.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:37bd246590a018e5a11703b7b09c39d47ede3df5ba3fa863c5b8590b465beb01", size = 14946862 },
{ url = "https://files.pythonhosted.org/packages/43/25/e9318566f443a5130b4ff0ad3367ee6c4c4c49ff083fe5214a7318c18282/mypy-2.0.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:cce87e92214fac8bf8feb8a680d0c1b6fb748d50e9b57fbb13e4b1d83a3ed19b", size = 15175090 },
{ url = "https://files.pythonhosted.org/packages/67/65/2ec28c834f21e164c33bc296a7db538ad50c74f83e517c7a0be95ff6de86/mypy-2.0.0-cp312-cp312-win_amd64.whl", hash = "sha256:e19e9cb69b66a4141009d24898259914fa2b71d026de0b46edf9fafdbf4fd46e", size = 11052899 },
{ url = "https://files.pythonhosted.org/packages/9e/72/d1ec625cfc9bd101c07a6834ef1f94e820296f8fdbad2eb03f50e0983f8c/mypy-2.0.0-cp312-cp312-win_arm64.whl", hash = "sha256:b021614cb08d44785b025982163ec3c39c94bff766ead071fa9e82b4ef6f62cd", size = 9972935 },
{ url = "https://files.pythonhosted.org/packages/e5/c6/996a1e535e5d0d597c3b1460fc962733091f885f312e749350eb2ac10965/mypy-2.0.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:9ef5f581b61240d1cc629b12f8df6565ed6ffac0d82ed745eef7833222ab50b9", size = 14737259 },
{ url = "https://files.pythonhosted.org/packages/94/c5/0f9460e26b77f434bd53f47d1ce32a3cd4580c92a5331fa5dfc059f9421a/mypy-2.0.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:20e3470a165dbc249bdfbe8d1c5172727ef22688cffc279f8c3aa264ab9d4d9a", size = 13538377 },
{ url = "https://files.pythonhosted.org/packages/b2/3e/8ea2f8dd1e5c9c279fb3c28193bdb850adf4d3d8172880abad829eced609/mypy-2.0.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:224ba142eee8b4d65d4db657cb1fc22abec30b135ded6ab297302ba1f62e505d", size = 13914264 },
{ url = "https://files.pythonhosted.org/packages/be/ce/78bd3b8520f676acee9dab48ea71473e68f6d5cf14b59fbd800bea50a92b/mypy-2.0.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2e879ad8a03908ff74d15e8a9b42bf049918e6798d52c011011f1873d0b5877e", size = 14926761 },
{ url = "https://files.pythonhosted.org/packages/61/ef/b52fa340522da3d22e669117c3b83155c2660f7cdc035856958fbfffb224/mypy-2.0.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:65c5c15bcbd18d6fe927cc55c459597a3517d69cc3123f067be3b020010e115e", size = 15157014 },
{ url = "https://files.pythonhosted.org/packages/7a/0c/dde7614250c6d017936c7aa3bb63b9b52c7cfd298d3f1be9be45f307870b/mypy-2.0.0-cp313-cp313-win_amd64.whl", hash = "sha256:d1a068acd7c9fb77e9f8923f1556f2f49d6d7895821121b8d97fa5642b9c52f5", size = 11067049 },
{ url = "https://files.pythonhosted.org/packages/27/ec/1d6af4830a94a285442db19caa02f160cc1a255e4f324eec5458e6c2bafb/mypy-2.0.0-cp313-cp313-win_arm64.whl", hash = "sha256:ef9d96da1ddffbc21f27d3939319b6846d12393baa17c4d2f3e81e040e73ce2c", size = 9967903 },
{ url = "https://files.pythonhosted.org/packages/ce/2c/6fefe954207860aed6eeb91776795e64a257d3ce0360862288984ce121f5/mypy-2.0.0-cp314-cp314-macosx_10_15_x86_64.whl", hash = "sha256:c918c64e8ce36557851b0347f84eb12f1965d3a06813c36df253eb0c0afd1d82", size = 14729633 },
{ url = "https://files.pythonhosted.org/packages/23/d6/d336f5b820af189eb0390cce21de62d264c0a4e64713dfbe81bfc4fc7739/mypy-2.0.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:301f1a8ccc7d79b542ee218b28bb49443a83e194eb3d10da63ff1649e5aa5d34", size = 13559524 },
{ url = "https://files.pythonhosted.org/packages/af/a6/d7bb54fde1770f0484e5fbdbdce37a41e95ed0a1cd493ec60ead111e356c/mypy-2.0.0-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:fdf4ef489d44ce350bac3fd699907834e551d4c934e9cc862ef201215ab1558d", size = 13936018 },
{ url = "https://files.pythonhosted.org/packages/7d/ba/5be51316b91e6a6bf6e3a8adb3de500e7e1fb5bf9491743b8cbc81a34a2c/mypy-2.0.0-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9cde2d0989f912fc850890f727d0d76495e7a6c5bdd9912a1efdb64952b4398d", size = 14910712 },
{ url = "https://files.pythonhosted.org/packages/b7/37/e2c8c3b373e20ebfb66e6c83a99027fd67df4ec43b08879f74e822d2dc4c/mypy-2.0.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:cdf05693c231a14fe37dbfce192a3a1372c26a833af4a80f550547742952e719", size = 15141499 },
{ url = "https://files.pythonhosted.org/packages/12/36/07756f933e00416d912e35878cfcf89a593a3350a885691c0bb85ae0226a/mypy-2.0.0-cp314-cp314-win_amd64.whl", hash = "sha256:73aee2da33a2237e66cbe84a94780e53599847e86bb3aa7b93e405e8cd9905f2", size = 11240511 },
{ url = "https://files.pythonhosted.org/packages/70/05/79ac1f20f2397353f3845f7b8bb5d8006cda7c8ef9092f04f9de3c6135f2/mypy-2.0.0-cp314-cp314-win_arm64.whl", hash = "sha256:1f6dcd8f39971f41edab2728c877c4ac8b50ad3c387ff2770423b79a05d23910", size = 10149336 },
{ url = "https://files.pythonhosted.org/packages/53/e0/0db84e0ebbad6e99e566c68e4b465784f2a2294f7719e8db9d509ef23087/mypy-2.0.0-cp314-cp314t-macosx_10_15_x86_64.whl", hash = "sha256:a04e980b9275c76159da66c6e1723c7798306f9802b31bdaf9358d0c84030ce8", size = 15797362 },
{ url = "https://files.pythonhosted.org/packages/0a/a4/14cc0768164dd53bec48aa41a20270b18df9bf72aa5054278bf133608315/mypy-2.0.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:33f9cf4825469b2bc73c53ba55f6d9a9b4cdb60f9e6e228745581520f29b8771", size = 14635914 },
{ url = "https://files.pythonhosted.org/packages/08/48/d866a3e23b4dc5974c77d9cf65a435bf22de01a84dd4620917950e233960/mypy-2.0.0-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:191675c3c7dc2a5c7722a035a6909c277f14046c5e4e02aa5fbf65f8524f08ad", size = 15270866 },
{ url = "https://files.pythonhosted.org/packages/71/eb/de9ef94958eb2078a6b908ceb247757dc384d3a238d3bd6ed7d81de5eaf8/mypy-2.0.0-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c3d26c4321a3b06fc9f04c741e0733af693f82d823f8e64e47b2e63b7f19fa84", size = 16093131 },
{ url = "https://files.pythonhosted.org/packages/ad/07/0ab2c1a9d26e90942612724cbd5788f16b7810c5dd39bfcf79286c6c4524/mypy-2.0.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:bbcbc4d5917ca6ce12de70e051de7f533e3bf92d548b41a38a2232a6fe356525", size = 16330685 },
{ url = "https://files.pythonhosted.org/packages/a6/8f/46f85d1371a5be642dad263828118ae1efd536d91d8bd2000c68acff3920/mypy-2.0.0-cp314-cp314t-win_amd64.whl", hash = "sha256:dbc6ba6d40572ae49268531565793a8f07eac7fc65ad76d482c9b4c8765b6043", size = 12752017 },
{ url = "https://files.pythonhosted.org/packages/7a/e6/94ca48800cac19eb28a58188a768aaec0d16cac0f373915f073058ab0855/mypy-2.0.0-cp314-cp314t-win_arm64.whl", hash = "sha256:77926029dfcb7e1a3ecb0acb2ddbb24ca36be03f7d623e1759ad5376be8f6c01", size = 10527097 },
{ url = "https://files.pythonhosted.org/packages/5c/14/fd0694aa594d6e9f9fd16ce821be2eff295197a273262ef56ddcc1388d68/mypy-2.0.0-py3-none-any.whl", hash = "sha256:8a92b2be3146b4fa1f062af7eb05574cbf3e6eb8e1f14704af1075423144e4e5", size = 2673434 },
]
[[package]]
name = "mypy-extensions"
version = "1.1.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/a2/6e/371856a3fb9d31ca8dac321cda606860fa4548858c0cc45d9d1d4ca2628b/mypy_extensions-1.1.0.tar.gz", hash = "sha256:52e68efc3284861e772bbcd66823fde5ae21fd2fdb51c62a211403730b916558", size = 6343 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/79/7b/2c79738432f5c924bef5071f933bcc9efd0473bac3b4aa584a6f7c1c8df8/mypy_extensions-1.1.0-py3-none-any.whl", hash = "sha256:1be4cccdb0f2482337c4743e60421de3a356cd97508abadd57d47403e94f5505", size = 4963 },
]
[[package]]
name = "packaging"
version = "26.2"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/d7/f1/e7a6dd94a8d4a5626c03e4e99c87f241ba9e350cd9e6d75123f992427270/packaging-26.2.tar.gz", hash = "sha256:ff452ff5a3e828ce110190feff1178bb1f2ea2281fa2075aadb987c2fb221661", size = 228134 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/df/b2/87e62e8c3e2f4b32e5fe99e0b86d576da1312593b39f47d8ceef365e95ed/packaging-26.2-py3-none-any.whl", hash = "sha256:5fc45236b9446107ff2415ce77c807cee2862cb6fac22b8a73826d0693b0980e", size = 100195 },
]
[[package]]
name = "pathspec"
version = "1.1.1"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/5a/82/42f767fc1c1143d6fd36efb827202a2d997a375e160a71eb2888a925aac1/pathspec-1.1.1.tar.gz", hash = "sha256:17db5ecd524104a120e173814c90367a96a98d07c45b2e10c2f3919fff91bf5a", size = 135180 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/f1/d9/7fb5aa316bc299258e68c73ba3bddbc499654a07f151cba08f6153988714/pathspec-1.1.1-py3-none-any.whl", hash = "sha256:a00ce642f577bf7f473932318056212bc4f8bfdf53128c78bbd5af0b9b20b189", size = 57328 },
]
[[package]]
name = "pluggy"
version = "1.6.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/f9/e2/3e91f31a7d2b083fe6ef3fa267035b518369d9511ffab804f839851d2779/pluggy-1.6.0.tar.gz", hash = "sha256:7dcc130b76258d33b90f61b658791dede3486c3e6bfb003ee5c9bfb396dd22f3", size = 69412 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538 },
]
[[package]]
name = "psycopg"
version = "3.3.4"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "typing-extensions", marker = "python_full_version < '3.13'" },
{ name = "tzdata", marker = "sys_platform == 'win32'" },
]
sdist = { url = "https://files.pythonhosted.org/packages/db/2f/cb91e5502ec9de1de6f1b76cfbf69531932725361168bb06963620c77e2e/psycopg-3.3.4.tar.gz", hash = "sha256:e21207764952cff81b6b8bdacad9a3939f2793367fdac2987b3aac36a651b5bc", size = 165799 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/5c/e0/7b3dee031daae7743609ce3c746565d4a3ed7c2c186479eb48e34e838c64/psycopg-3.3.4-py3-none-any.whl", hash = "sha256:b6bbc25ccf05c8fad3b061d9db2ef0909a555171b84b07f29458a447253d679a", size = 213001 },
]
[package.optional-dependencies]
binary = [
{ name = "psycopg-binary", marker = "implementation_name != 'pypy'" },
]
[[package]]
name = "psycopg-binary"
version = "3.3.4"
source = { registry = "https://pypi.org/simple" }
wheels = [
{ url = "https://files.pythonhosted.org/packages/95/7d/03818e13ba7f36de93573c93ee3482006d3dfa8b0f8d28df511bad0a1a92/psycopg_binary-3.3.4-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:5ab28a2a7649df3b72e6b674b4c190e448e8e77cf496a65bd846472048de2089", size = 4591122 },
{ url = "https://files.pythonhosted.org/packages/a5/b9/11b341edf8d54e2694726b273fe9652b254d989f4f63e3ac6816ad6b55f4/psycopg_binary-3.3.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:6402a9d8146cf4b3974ded3fd28a971e83dc6a0333eb7822524a3aa20b546578", size = 4669943 },
{ url = "https://files.pythonhosted.org/packages/8b/18/4665bacd65e7865b4372fcd8abb8b9186ada4b0025f8c2ca691b364a556c/psycopg_binary-3.3.4-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:580ae30a5f95ccd90008ec697d3ed6a4a2047a516407ad904283fa42086936e9", size = 5469697 },
{ url = "https://files.pythonhosted.org/packages/7c/b1/b83136c6e510593d9b0c759ba5384337bc4ad82d19fda675adc4b2703c84/psycopg_binary-3.3.4-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:e7510c37550f91a187e3660a8cc50d4b760f8c3b8b2f89ebc5698cd2c7f2c85d", size = 5152995 },
{ url = "https://files.pythonhosted.org/packages/67/8d/a9821e2a648afe6091989929982a3b0f00b2631a859cb81379728f08fb75/psycopg_binary-3.3.4-cp312-cp312-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:77df19583501ea288eaf15ac0fe7ad01e6d8091a91d5c41df5c718f307d8e31b", size = 6738180 },
{ url = "https://files.pythonhosted.org/packages/7e/58/2e349e8d23905dc2317b80ac65f48fb6f821a4777a4e994a60da91c4850f/psycopg_binary-3.3.4-cp312-cp312-manylinux_2_38_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:018fbed325936da502feb546642c982dcc4b9ffdea32dfef78dbf3b7f7ad4070", size = 4978828 },
{ url = "https://files.pythonhosted.org/packages/45/48/57b00d03b4721878326122a1f1e6b0a90b85bcaec56b5b2f8ea6cfa45235/psycopg_binary-3.3.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:17a21953a9e5ff3a16dab692625a3676e2f101db5e40072f39dbee2250194d68", size = 4509757 },
{ url = "https://files.pythonhosted.org/packages/25/37/33b47d8c007df69aec500df5889767c4d313748e8e9e27a2fef8a6dabcee/psycopg_binary-3.3.4-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:eb05ee1c2b817d27c537333224c9e83c7afb86fe7296ba970990068baf819b16", size = 4190546 },
{ url = "https://files.pythonhosted.org/packages/ca/c6/32b0835dbc2122617902b649d76a91c1e75406e76bf3d595b0c3bb5ffad6/psycopg_binary-3.3.4-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:773d573e11f437ce0bdb95b7c18dc58390494f96d43f8b45b9760436114f7652", size = 3926197 },
{ url = "https://files.pythonhosted.org/packages/cd/68/d190ef0c0c5b16ded07831dabc8ddd412f4cdab07ec6e30ed38d9bda0e1f/psycopg_binary-3.3.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:71e55ccbdfae79a2ed9c6369c3008a3025817ff9d7e27b32a2d84e2a4267e66e", size = 4236627 },
{ url = "https://files.pythonhosted.org/packages/25/8f/81dcbc2e8454b74d14881275ea45f00791052dac531a9fa8be1730d1685b/psycopg_binary-3.3.4-cp312-cp312-win_amd64.whl", hash = "sha256:494ca54901be8cf9eb7e02c25b731f2317c378efa44f43e8f9bd0e1184ae7be4", size = 3560782 },
{ url = "https://files.pythonhosted.org/packages/09/43/13e9c406fbbf354580476e248a16b64802a376873ebe6339e30bb655572d/psycopg_binary-3.3.4-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:fbd1d4ed566895ad2d3bf4ddfd8bae90026930ddf29df3b9d91d32c8c47866a7", size = 4590377 },
{ url = "https://files.pythonhosted.org/packages/22/be/2923cd7c3683e7afdecf4f10796a18de02f5c5ddc0969aa2ad0a8cdd3bbd/psycopg_binary-3.3.4-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:75a9067e236f9b9ae3535b66fe99bddb33d39c0de10112e49b9ab11eee53dc31", size = 4669023 },
{ url = "https://files.pythonhosted.org/packages/96/a0/2c913d6fe13d6a8bd13597d36739bf47af063ad9399e402cfecab16f3c1e/psycopg_binary-3.3.4-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:b56b603ebcea8aa10b46228b8410ba7f13e7c2ee54389d4d9be0927fd8ce2a70", size = 5467423 },
{ url = "https://files.pythonhosted.org/packages/e7/38/205d10bc1ad0df4a21c5c51659126bd3ea0ef98fcad1e852f78c249bb9c3/psycopg_binary-3.3.4-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:c677c4ad433cb7150c8cd304a0769ae3bcfbe5ea0676eb53faa7b1443b16d0d3", size = 5151137 },
{ url = "https://files.pythonhosted.org/packages/36/fc/f0381ddcd45eff3bb70dbca6823a996048d7f507b2ec3fc92c6fabc0fe87/psycopg_binary-3.3.4-cp313-cp313-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:26df2717e59c0473e4465a97dfb1b7afebaa479277870fd5784d1436470db47c", size = 6736671 },
{ url = "https://files.pythonhosted.org/packages/95/40/fa545ae152c24327651e5624e4902121e808270be36c10b12e9939be09bc/psycopg_binary-3.3.4-cp313-cp313-manylinux_2_38_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:1dc1f79fd16bb1f3f4421417a514607539f17804d95c7ed617265369d1981cae", size = 4979601 },
{ url = "https://files.pythonhosted.org/packages/86/e4/2f8a47ee97f90cd2b933d0463081d35631ff419de2b8c984a5f369857de0/psycopg_binary-3.3.4-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:136f199a407b5348b9b857c504aff60c77622a28482e7195839ce1b51238c4cc", size = 4510513 },
{ url = "https://files.pythonhosted.org/packages/0e/0e/94e842ff4a7f98ed162580ca2e8b8864b28c1e0350f2443f8ee47f821167/psycopg_binary-3.3.4-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:b6f5a29e9c775b9f12a1a717aa7a2c80f9e1db6f27ba44a5b59c80ac61d2ffcf", size = 4187243 },
{ url = "https://files.pythonhosted.org/packages/d0/83/fc6c174b672e29b7de996ea77b6cbddf46c891751c3355f6974292baa6b4/psycopg_binary-3.3.4-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:ee17a2cf4943cde261adfad1bbc5bf38d6b3776d7afff74c7cabcbeaeb08c260", size = 3927347 },
{ url = "https://files.pythonhosted.org/packages/e9/65/768364d4a97a15b1a7f47ba52688c1686f22941d8332a8398cefc468e25f/psycopg_binary-3.3.4-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:5c4ab71be17bdca30cb34c34c4e1496e2f5d6f20c199c12bad226070b22ef9bf", size = 4236393 },
{ url = "https://files.pythonhosted.org/packages/bd/3b/218efbc9e645becd80cdf651acda05f85cfe546b7a9c0458c7cbc8fe1f74/psycopg_binary-3.3.4-cp313-cp313-win_amd64.whl", hash = "sha256:dbfdb9b6cc79f31104a7b162a2b921b765fcc62af6c00540a167a8de47e4ed38", size = 3564592 },
{ url = "https://files.pythonhosted.org/packages/48/a6/828c9185701dab71b234c2a76c38a08b098ebfec5020716b4e93807492b5/psycopg_binary-3.3.4-cp314-cp314-macosx_10_15_x86_64.whl", hash = "sha256:28b7398fdd19db3232c884fb24550bdfe951221f510e195e233299e4c9b78f97", size = 4607292 },
{ url = "https://files.pythonhosted.org/packages/92/58/5b40dbc9d839045c9dae956960e4fb6d20bcabe6c59a2aa34fc3a371913f/psycopg_binary-3.3.4-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:1fbaa292a3c8bb61b45df1ad3da1908ccee7cb889db9425e3557d9e34e2a4829", size = 4687023 },
{ url = "https://files.pythonhosted.org/packages/85/a9/793f0ac107a9003b48441d0d1f9f616d96e0f37458dd8dc12528ceff55fb/psycopg_binary-3.3.4-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:94596f9e7633ee3f6440711d43bb70aa31cc0a46a900ab8b4201a366ace5c9e7", size = 5486985 },
{ url = "https://files.pythonhosted.org/packages/8f/26/42e8533497e2592334f68ec529cf5f840f7fa4e99575a4bb61aa184dbfbf/psycopg_binary-3.3.4-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:8c0056529e68dbe9184cd4019a1f3d8f3a4ead2f6fc7a5afcf27d3314edd1277", size = 5168745 },
{ url = "https://files.pythonhosted.org/packages/15/af/b7151776cc08d5935d45c833ec818a9beb417cf7c08239af1aafbdae78ee/psycopg_binary-3.3.4-cp314-cp314-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:2c09aad7051326e7603c14e50636db9c01f78272dc54b3accff03d46370461e6", size = 6761486 },
{ url = "https://files.pythonhosted.org/packages/d0/ed/c92533b9124712d592cbf1cd6c76da933a2e0acea81dfe1fbe7e735f0cff/psycopg_binary-3.3.4-cp314-cp314-manylinux_2_38_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:514404ed543efd620c85602b747df2a23cf1241b4067199e1a66f2d2757aaa41", size = 4997427 },
{ url = "https://files.pythonhosted.org/packages/a2/23/ccadfd0de416aa188356daa199453af24087b042e296088706d190ae0295/psycopg_binary-3.3.4-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:46893c26858be12cc49ca4226ed6a60b4bfccadd946b3bebb783a60b38788228", size = 4533549 },
{ url = "https://files.pythonhosted.org/packages/fd/a0/c8f43cee36386f7bc891ab41a9d31ea07cf9826038e732da79f26b1e5f34/psycopg_binary-3.3.4-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:df1d567fc430f6df15c9fcf67d87685fc49bdb325adc0db5af1adfb2f44eb5c9", size = 4210256 },
{ url = "https://files.pythonhosted.org/packages/4e/2c/c1547871be3790676e8868b38655496422f94f0978dfb66b74bdba2f1676/psycopg_binary-3.3.4-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:6b9016b1714da4dd5ecaaa75b82098aa5a0b87854ce9b092e21c27c4ae23e014", size = 3946204 },
{ url = "https://files.pythonhosted.org/packages/c4/b1/f6670f00fa7ea601584623f6c11602ab92117d83eaff885e0210f6de7418/psycopg_binary-3.3.4-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:47c656a8a7ba6eb0cff1801a4caaa9c8bdc12d03080e273aff1c8ac39971a77e", size = 4255811 },
{ url = "https://files.pythonhosted.org/packages/eb/e6/5fff07a70d1f945ed90ae131c3bd76cab32beff7c58c6db15ad5820b6d1f/psycopg_binary-3.3.4-cp314-cp314-win_amd64.whl", hash = "sha256:c37e024c07308cd06cf3ec51bfd0e7f6157585a4d84d1bce4a7f5f7913719bf8", size = 3666849 },
]
[[package]]
name = "pygments"
version = "2.20.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/c3/b2/bc9c9196916376152d655522fdcebac55e66de6603a76a02bca1b6414f6c/pygments-2.20.0.tar.gz", hash = "sha256:6757cd03768053ff99f3039c1a36d6c0aa0b263438fcab17520b30a303a82b5f", size = 4955991 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/f4/7e/a72dd26f3b0f4f2bf1dd8923c85f7ceb43172af56d63c7383eb62b332364/pygments-2.20.0-py3-none-any.whl", hash = "sha256:81a9e26dd42fd28a23a2d169d86d7ac03b46e2f8b59ed4698fb4785f946d0176", size = 1231151 },
]
[[package]]
name = "pytest"
version = "9.0.3"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "colorama", marker = "sys_platform == 'win32'" },
{ name = "iniconfig" },
{ name = "packaging" },
{ name = "pluggy" },
{ name = "pygments" },
]
sdist = { url = "https://files.pythonhosted.org/packages/7d/0d/549bd94f1a0a402dc8cf64563a117c0f3765662e2e668477624baeec44d5/pytest-9.0.3.tar.gz", hash = "sha256:b86ada508af81d19edeb213c681b1d48246c1a91d304c6c81a427674c17eb91c", size = 1572165 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/d4/24/a372aaf5c9b7208e7112038812994107bc65a84cd00e0354a88c2c77a617/pytest-9.0.3-py3-none-any.whl", hash = "sha256:2c5efc453d45394fdd706ade797c0a81091eccd1d6e4bccfcd476e2b8e0ab5d9", size = 375249 },
]
[[package]]
name = "pytest-cov"
version = "7.1.0"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "coverage" },
{ name = "pluggy" },
{ name = "pytest" },
]
sdist = { url = "https://files.pythonhosted.org/packages/b1/51/a849f96e117386044471c8ec2bd6cfebacda285da9525c9106aeb28da671/pytest_cov-7.1.0.tar.gz", hash = "sha256:30674f2b5f6351aa09702a9c8c364f6a01c27aae0c1366ae8016160d1efc56b2", size = 55592 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/9d/7a/d968e294073affff457b041c2be9868a40c1c71f4a35fcc1e45e5493067b/pytest_cov-7.1.0-py3-none-any.whl", hash = "sha256:a0461110b7865f9a271aa1b51e516c9a95de9d696734a2f71e3e78f46e1d4678", size = 22876 },
]
[[package]]
name = "pytest-django"
version = "4.12.0"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "pytest" },
]
sdist = { url = "https://files.pythonhosted.org/packages/13/2b/db9a193df89e5660137f5428063bcc2ced7ad790003b26974adf5c5ceb3b/pytest_django-4.12.0.tar.gz", hash = "sha256:df94ec819a83c8979c8f6de13d9cdfbe76e8c21d39473cfe2b40c9fc9be3c758", size = 91156 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/83/a5/41d091f697c09609e7ef1d5d61925494e0454ebf51de7de05f0f0a728f1d/pytest_django-4.12.0-py3-none-any.whl", hash = "sha256:3ff300c49f8350ba2953b90297d23bf5f589db69545f56f1ec5f8cff5da83e85", size = 26123 },
]
[[package]]
name = "python-decouple"
version = "3.8"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/e1/97/373dcd5844ec0ea5893e13c39a2c67e7537987ad8de3842fe078db4582fa/python-decouple-3.8.tar.gz", hash = "sha256:ba6e2657d4f376ecc46f77a3a615e058d93ba5e465c01bbe57289bfb7cce680f", size = 9612 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/a2/d4/9193206c4563ec771faf2ccf54815ca7918529fe81f6adb22ee6d0e06622/python_decouple-3.8-py3-none-any.whl", hash = "sha256:d0d45340815b25f4de59c974b855bb38d03151d81b037d9e3f463b0c9f8cbd66", size = 9947 },
]
[[package]]
name = "ruff"
version = "0.15.12"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/99/43/3291f1cc9106f4c63bdce7a8d0df5047fe8422a75b091c16b5e9355e0b11/ruff-0.15.12.tar.gz", hash = "sha256:ecea26adb26b4232c0c2ca19ccbc0083a68344180bba2a600605538ce51a40a6", size = 4643852 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/c3/6e/e78ffb61d4686f3d96ba3df2c801161843746dcbcbb17a1e927d4829312b/ruff-0.15.12-py3-none-linux_armv6l.whl", hash = "sha256:f86f176e188e94d6bdbc09f09bfd9dc729059ad93d0e7390b5a73efe19f8861c", size = 10640713 },
{ url = "https://files.pythonhosted.org/packages/ae/08/a317bc231fb9e7b93e4ef3089501e51922ff88d6936ce5cf870c4fe55419/ruff-0.15.12-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:e3bcd123364c3770b8e1b7baaf343cc99a35f197c5c6e8af79015c666c423a6c", size = 11069267 },
{ url = "https://files.pythonhosted.org/packages/aa/a4/f828e9718d3dce1f5f11c39c4f65afd32783c8b2aebb2e3d259e492c47bd/ruff-0.15.12-py3-none-macosx_11_0_arm64.whl", hash = "sha256:fe87510d000220aa1ed530d4448a7c696a0cae1213e5ec30e5874287b66557b5", size = 10397182 },
{ url = "https://files.pythonhosted.org/packages/71/e0/3310fc6d1b5e1fdea22bf3b1b807c7e187b581021b0d7d4514cccdb5fb71/ruff-0.15.12-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:84a1630093121375a3e2a95b4a6dc7b59e2b4ee76216e32d81aae550a832d002", size = 10758012 },
{ url = "https://files.pythonhosted.org/packages/11/c1/a606911aee04c324ddaa883ae418f3569792fd3c4a10c50e0dd0a2311e1e/ruff-0.15.12-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:fb129f40f114f089ebe0ca56c0d251cf2061b17651d464bb6478dc01e69f11f5", size = 10447479 },
{ url = "https://files.pythonhosted.org/packages/9d/68/4201e8444f0894f21ab4aeeaee68aa4f10b51613514a20d80bd628d57e88/ruff-0.15.12-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b0c862b172d695db7598426b8af465e7e9ac00a3ea2a3630ee67eb82e366aaa6", size = 11234040 },
{ url = "https://files.pythonhosted.org/packages/34/ff/8a6d6cf4ccc23fd67060874e832c18919d1557a0611ebef03fdb01fff11e/ruff-0.15.12-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2849ea9f3484c3aca43a82f484210370319e7170df4dfe4843395ddf6c57bc33", size = 12087377 },
{ url = "https://files.pythonhosted.org/packages/85/f6/c669cf73f5152f623d34e69866a46d5e6185816b19fcd5b6dd8a2d299922/ruff-0.15.12-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9e77c7e51c07fe396826d5969a5b846d9cd4c402535835fb6e21ce8b28fef847", size = 11367784 },
{ url = "https://files.pythonhosted.org/packages/e8/39/c61d193b8a1daaa8977f7dea9e8d8ba866e02ea7b65d32f6861693aa4c12/ruff-0.15.12-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:83b2f4f2f3b1026b5fb449b467d9264bf22067b600f7b6f41fc5958909f449d0", size = 11344088 },
{ url = "https://files.pythonhosted.org/packages/c2/8d/49afab3645e31e12c590acb6d3b5b69d7aab5b81926dbaf7461f9441f37a/ruff-0.15.12-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:9ba3b8f1afd7e2e43d8943e55f249e13f9682fde09711644a6e7290eb4f3e339", size = 11271770 },
{ url = "https://files.pythonhosted.org/packages/46/06/33f41fe94403e2b755481cdfb9b7ef3e4e0ed031c4581124658d935d52b4/ruff-0.15.12-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:e852ba9fdc890655e1d78f2df1499efbe0e54126bd405362154a75e2bde159c5", size = 10719355 },
{ url = "https://files.pythonhosted.org/packages/0d/59/18aa4e014debbf559670e4048e39260a85c7fcee84acfd761ac01e7b8d35/ruff-0.15.12-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:dd8aed930da53780d22fc70bdf84452c843cf64f8cb4eb38984319c24c5cd5fd", size = 10462758 },
{ url = "https://files.pythonhosted.org/packages/25/e7/cc9f16fd0f3b5fddcbd7ec3d6ae30c8f3fde1047f32a4093a98d633c6570/ruff-0.15.12-py3-none-musllinux_1_2_i686.whl", hash = "sha256:01da3988d225628b709493d7dc67c3b9b12c0210016b08690ef9bd27970b262b", size = 10953498 },
{ url = "https://files.pythonhosted.org/packages/72/7a/a9ba7f98c7a575978698f4230c5e8cc54bbc761af34f560818f933dafa0c/ruff-0.15.12-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:9cae0f92bd5700d1213188b31cd3bdd2b315361296d10b96b8e2337d3d11f53e", size = 11447765 },
{ url = "https://files.pythonhosted.org/packages/ea/f9/0ae446942c846b8266059ad8a30702a35afae55f5cdc54c5adf8d7afdc27/ruff-0.15.12-py3-none-win32.whl", hash = "sha256:d0185894e038d7043ba8fd6aee7499ece6462dc0ea9f1e260c7451807c714c20", size = 10657277 },
{ url = "https://files.pythonhosted.org/packages/33/f1/9614e03e1cdcbf9437570b5400ced8a720b5db22b28d8e0f1bda429f660d/ruff-0.15.12-py3-none-win_amd64.whl", hash = "sha256:c87a162d61ab3adca47c03f7f717c68672edec7d1b5499e652331780fe74950d", size = 11837758 },
{ url = "https://files.pythonhosted.org/packages/c0/98/6beb4b351e472e5f4c4613f7c35a5290b8be2497e183825310c4c3a3984b/ruff-0.15.12-py3-none-win_arm64.whl", hash = "sha256:a538f7a82d061cee7be55542aca1d86d1393d55d81d4fcc314370f4340930d4f", size = 11120821 },
]
[[package]]
name = "sqlparse"
version = "0.5.5"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/90/76/437d71068094df0726366574cf3432a4ed754217b436eb7429415cf2d480/sqlparse-0.5.5.tar.gz", hash = "sha256:e20d4a9b0b8585fdf63b10d30066c7c94c5d7a7ec47c889a2d83a3caa93ff28e", size = 120815 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/49/4b/359f28a903c13438ef59ebeee215fb25da53066db67b305c125f1c6d2a25/sqlparse-0.5.5-py3-none-any.whl", hash = "sha256:12a08b3bf3eec877c519589833aed092e2444e68240a3577e8e26148acc7b1ba", size = 46138 },
]
[[package]]
name = "types-pyyaml"
version = "6.0.12.20260508"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/b9/6b/f9d82598121b2f0532238381aec27734c24a0ff548ac352b358c2857a176/types_pyyaml-6.0.12.20260508.tar.gz", hash = "sha256:5ae42149c3ebf7aaaf6c65ee49af590c80f0ba52e9e3f75a75c5564b33556fa6", size = 17776 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/b0/80/96690460fe7f4e4b4ab0ecd79a8609aec2ae63927ad97625f3c747ca6a1d/types_pyyaml-6.0.12.20260508-py3-none-any.whl", hash = "sha256:edc094ed3a918b0c6232f71a5b67fdf38e76e17517b7d87bfbb9fc27d442fb51", size = 20309 },
]
[[package]]
name = "typing-extensions"
version = "4.15.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/72/94/1a15dd82efb362ac84269196e94cf00f187f7ed21c242792a923cdb1c61f/typing_extensions-4.15.0.tar.gz", hash = "sha256:0cea48d173cc12fa28ecabc3b837ea3cf6f38c6d1136f85cbaaf598984861466", size = 109391 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/18/67/36e9267722cc04a6b9f15c7f3441c2363321a3ea07da7ae0c0707beb2a9c/typing_extensions-4.15.0-py3-none-any.whl", hash = "sha256:f0fa19c6845758ab08074a0cfa8b7aecb71c999ca73d62883bc25cc018c4e548", size = 44614 },
]
[[package]]
name = "tzdata"
version = "2026.2"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/ba/19/1b9b0e29f30c6d35cb345486df41110984ea67ae69dddbc0e8a100999493/tzdata-2026.2.tar.gz", hash = "sha256:9173fde7d80d9018e02a662e168e5a2d04f87c41ea174b139fbef642eda62d10", size = 198254 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/ce/e4/dccd7f47c4b64213ac01ef921a1337ee6e30e8c6466046018326977efd95/tzdata-2026.2-py2.py3-none-any.whl", hash = "sha256:bbe9af844f658da81a5f95019480da3a89415801f6cc966806612cc7169bffe7", size = 349321 },
]
[[package]]
name = "vergabe-teilnahme"
version = "0.1.0"
source = { virtual = "." }
dependencies = [
{ name = "dj-database-url" },
{ name = "django" },
{ name = "django-storages" },
{ name = "psycopg", extra = ["binary"] },
{ name = "python-decouple" },
{ name = "whitenoise" },
]
[package.dev-dependencies]
dev = [
{ name = "django-stubs" },
{ name = "factory-boy" },
{ name = "mypy" },
{ name = "pytest-cov" },
{ name = "pytest-django" },
{ name = "ruff" },
]
[package.metadata]
requires-dist = [
{ name = "dj-database-url", specifier = ">=2.1" },
{ name = "django", specifier = ">=5.2" },
{ name = "django-storages", specifier = ">=1.14" },
{ name = "psycopg", extras = ["binary"], specifier = ">=3.2" },
{ name = "python-decouple", specifier = ">=3.8" },
{ name = "whitenoise", specifier = ">=6.7" },
]
[package.metadata.requires-dev]
dev = [
{ name = "django-stubs", specifier = ">=5.0" },
{ name = "factory-boy", specifier = ">=3.3" },
{ name = "mypy", specifier = ">=1.10" },
{ name = "pytest-cov", specifier = ">=5.0" },
{ name = "pytest-django", specifier = ">=4.8" },
{ name = "ruff", specifier = ">=0.4" },
]
[[package]]
name = "whitenoise"
version = "6.12.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/cb/2a/55b3f3a4ec326cd077c1c3defeee656b9298372a69229134d930151acd01/whitenoise-6.12.0.tar.gz", hash = "sha256:f723ebb76a112e98816ff80fcea0a6c9b8ecde835f8ddda25df7a30a3c2db6ad", size = 26841 }
wheels = [
{ url = "https://files.pythonhosted.org/packages/db/eb/d5583a11486211f3ebd4b385545ae787f32363d453c19fffd81106c9c138/whitenoise-6.12.0-py3-none-any.whl", hash = "sha256:fc5e8c572e33ebf24795b47b6a7da8da3c00cff2349f5b04c02f28d0cc5a3cc2", size = 20302 },
]

View File

View File

View File

@@ -0,0 +1,12 @@
from django.contrib import admin
from django.contrib.auth.admin import UserAdmin
from .models import Mitarbeiter
@admin.register(Mitarbeiter)
class MitarbeiterAdmin(UserAdmin):
fieldsets = UserAdmin.fieldsets + (
('Firmen-Details', {'fields': ('rolle', 'mobilnummer', 'organisationseinheit')}),
)
list_display = ['username', 'get_full_name', 'rolle', 'organisationseinheit', 'is_active']

View File

@@ -0,0 +1,5 @@
from django.apps import AppConfig
class AccountsConfig(AppConfig):
name = 'vergabe_teilnahme.apps.accounts'

View File

@@ -0,0 +1,46 @@
# Generated by Django 6.0.5 on 2026-05-08 08:32
import django.contrib.auth.models
import django.contrib.auth.validators
import django.utils.timezone
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
('auth', '0012_alter_user_first_name_max_length'),
]
operations = [
migrations.CreateModel(
name='Mitarbeiter',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('password', models.CharField(max_length=128, verbose_name='password')),
('last_login', models.DateTimeField(blank=True, null=True, verbose_name='last login')),
('is_superuser', models.BooleanField(default=False, help_text='Designates that this user has all permissions without explicitly assigning them.', verbose_name='superuser status')),
('username', models.CharField(error_messages={'unique': 'A user with that username already exists.'}, help_text='Required. 150 characters or fewer. Letters, digits and @/./+/-/_ only.', max_length=150, unique=True, validators=[django.contrib.auth.validators.UnicodeUsernameValidator()], verbose_name='username')),
('first_name', models.CharField(blank=True, max_length=150, verbose_name='first name')),
('last_name', models.CharField(blank=True, max_length=150, verbose_name='last name')),
('email', models.EmailField(blank=True, max_length=254, verbose_name='email address')),
('is_staff', models.BooleanField(default=False, help_text='Designates whether the user can log into this admin site.', verbose_name='staff status')),
('is_active', models.BooleanField(default=True, help_text='Designates whether this user should be treated as active. Unselect this instead of deleting accounts.', verbose_name='active')),
('date_joined', models.DateTimeField(default=django.utils.timezone.now, verbose_name='date joined')),
('rolle', models.CharField(choices=[('ersteller', 'Ersteller'), ('kalkulator', 'Kalkulator'), ('jurist', 'Jurist'), ('freigeber', 'Freigeber'), ('beobachter', 'Beobachter'), ('admin', 'Admin'), ('geschaeftsfuehrung', 'Geschäftsführung'), ('projektleiter', 'Projektleiter')], default='ersteller', max_length=30)),
('mobilnummer', models.CharField(blank=True, max_length=30)),
('organisationseinheit', models.CharField(blank=True, max_length=100)),
('groups', models.ManyToManyField(blank=True, help_text='The groups this user belongs to. A user will get all permissions granted to each of their groups.', related_name='user_set', related_query_name='user', to='auth.group', verbose_name='groups')),
('user_permissions', models.ManyToManyField(blank=True, help_text='Specific permissions for this user.', related_name='user_set', related_query_name='user', to='auth.permission', verbose_name='user permissions')),
],
options={
'verbose_name': 'Mitarbeiter',
'verbose_name_plural': 'Mitarbeiter',
},
managers=[
('objects', django.contrib.auth.models.UserManager()),
],
),
]

View File

@@ -0,0 +1,28 @@
# Generated by Django 6.0.5 on 2026-05-08 10:32
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('accounts', '0001_initial'),
]
operations = [
migrations.AlterField(
model_name='mitarbeiter',
name='mobilnummer',
field=models.CharField(blank=True, max_length=50),
),
migrations.AlterField(
model_name='mitarbeiter',
name='organisationseinheit',
field=models.CharField(blank=True, max_length=200),
),
migrations.AlterField(
model_name='mitarbeiter',
name='rolle',
field=models.CharField(blank=True, choices=[('bid_manager', 'Bid Manager'), ('fachexperte', 'Fachverantwortlicher'), ('vertrieb', 'Vertrieb / Account Management'), ('pricing', 'Pricing / Controlling'), ('recht', 'Recht / Compliance'), ('geschaeftsfuehrung', 'Geschäftsführung'), ('projektleitung', 'Projektleitung Umsetzung'), ('admin', 'Administrator')], max_length=30),
),
]

View File

@@ -0,0 +1,25 @@
from django.contrib.auth.models import AbstractUser
from django.db import models
class Mitarbeiter(AbstractUser):
ROLLE_CHOICES = [
('bid_manager', 'Bid Manager'),
('fachexperte', 'Fachverantwortlicher'),
('vertrieb', 'Vertrieb / Account Management'),
('pricing', 'Pricing / Controlling'),
('recht', 'Recht / Compliance'),
('geschaeftsfuehrung', 'Geschäftsführung'),
('projektleitung', 'Projektleitung Umsetzung'),
('admin', 'Administrator'),
]
rolle = models.CharField(max_length=30, choices=ROLLE_CHOICES, blank=True)
mobilnummer = models.CharField(max_length=50, blank=True)
organisationseinheit = models.CharField(max_length=200, blank=True)
def __str__(self):
return self.get_full_name() or self.username
class Meta:
verbose_name = 'Mitarbeiter'
verbose_name_plural = 'Mitarbeiter'

View File

@@ -0,0 +1,3 @@
from django.test import TestCase
# Create your tests here.

View File

@@ -0,0 +1,3 @@
from django.shortcuts import render
# Create your views here.

View File

@@ -0,0 +1,15 @@
from django.contrib import admin
from .models import Aufgabe, Bieterfrage
@admin.register(Aufgabe)
class AufgabeAdmin(admin.ModelAdmin):
list_display = ['titel', 'typ', 'status', 'prioritaet', 'frist', 'verantwortlicher']
list_filter = ['typ', 'status', 'prioritaet']
@admin.register(Bieterfrage)
class BieterfragAdmin(admin.ModelAdmin):
list_display = ['fragentext', 'status', 'prioritaet', 'einreichungsdatum', 'eingearbeitet']
list_filter = ['status', 'prioritaet', 'eingearbeitet']

View File

@@ -0,0 +1,5 @@
from django.apps import AppConfig
class AufgabenConfig(AppConfig):
name = 'vergabe_teilnahme.apps.aufgaben'

View File

@@ -0,0 +1,51 @@
# Generated by Django 6.0.5 on 2026-05-08 10:33
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
]
operations = [
migrations.CreateModel(
name='Aufgabe',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('titel', models.CharField(max_length=400)),
('beschreibung', models.TextField(blank=True)),
('typ', models.CharField(choices=[('fachlich', 'Fachlich'), ('rechtlich', 'Rechtlich'), ('kaufmaennisch', 'Kaufmännisch'), ('technisch', 'Technisch'), ('subunternehmer', 'Subunternehmer'), ('dokument', 'Dokument'), ('preis', 'Preis')], default='fachlich', max_length=20)),
('status', models.CharField(choices=[('offen', 'Offen'), ('in_bearbeitung', 'In Bearbeitung'), ('wartend_intern', 'Wartend (intern)'), ('wartend_sub', 'Wartend (Subunternehmer)'), ('wartend_ausschreiber', 'Wartend (Ausschreiber)'), ('erledigt', 'Erledigt'), ('verworfen', 'Verworfen'), ('ueberfaellig', 'Überfällig')], default='offen', max_length=25)),
('prioritaet', models.PositiveSmallIntegerField(choices=[(1, 'Hoch'), (2, 'Mittel'), (3, 'Niedrig')], default=2)),
('frist', models.DateField(blank=True, null=True)),
('ergebnis', models.TextField(blank=True)),
],
options={
'verbose_name': 'Aufgabe',
'verbose_name_plural': 'Aufgaben',
'ordering': ['prioritaet', 'frist'],
},
),
migrations.CreateModel(
name='Bieterfrage',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('fragentext', models.TextField()),
('begruendung', models.TextField(blank=True)),
('status', models.CharField(choices=[('entwurf', 'Entwurf'), ('abgestimmt', 'Abgestimmt'), ('eingereicht', 'Eingereicht'), ('beantwortet', 'Beantwortet'), ('eingearbeitet', 'Eingearbeitet')], default='entwurf', max_length=20)),
('prioritaet', models.PositiveSmallIntegerField(choices=[(1, 'Hoch'), (2, 'Mittel'), (3, 'Niedrig')], default=2)),
('einreichungsdatum', models.DateField(blank=True, null=True)),
('antwort', models.TextField(blank=True)),
('auswirkung_angebot', models.TextField(blank=True)),
('eingearbeitet', models.BooleanField(default=False)),
],
options={
'verbose_name': 'Bieterfrage',
'verbose_name_plural': 'Bieterfragen',
'ordering': ['prioritaet', 'status'],
},
),
]

View File

@@ -0,0 +1,71 @@
# Generated by Django 6.0.5 on 2026-05-08 10:33
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
('aufgaben', '0001_initial'),
('ausschreibungen', '0001_initial'),
('dokumente', '0002_initial'),
('lose', '0001_initial'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.AddField(
model_name='aufgabe',
name='anforderung',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='aufgaben', to='lose.anforderung'),
),
migrations.AddField(
model_name='aufgabe',
name='ausschreibung',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='aufgaben', to='ausschreibungen.ausschreibung'),
),
migrations.AddField(
model_name='aufgabe',
name='dokument',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='aufgaben', to='dokumente.dokument'),
),
migrations.AddField(
model_name='aufgabe',
name='los',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='aufgaben', to='lose.los'),
),
migrations.AddField(
model_name='aufgabe',
name='verantwortlicher',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL),
),
migrations.AddField(
model_name='bieterfrage',
name='anforderung',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='bieterfragen', to='lose.anforderung'),
),
migrations.AddField(
model_name='bieterfrage',
name='ausschreibung',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='bieterfragen', to='ausschreibungen.ausschreibung'),
),
migrations.AddField(
model_name='bieterfrage',
name='dokument',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='bieterfragen', to='dokumente.dokument'),
),
migrations.AddField(
model_name='bieterfrage',
name='verfasser',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL),
),
migrations.AddField(
model_name='aufgabe',
name='bieterfrage',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='aufgaben', to='aufgaben.bieterfrage'),
),
]

View File

@@ -0,0 +1,113 @@
from datetime import date
from django.db import models
from vergabe_teilnahme.apps.core.models import FlexibleModel
class Aufgabe(FlexibleModel):
TYP_CHOICES = [
('fachlich', 'Fachlich'),
('rechtlich', 'Rechtlich'),
('kaufmaennisch', 'Kaufmännisch'),
('technisch', 'Technisch'),
('subunternehmer', 'Subunternehmer'),
('dokument', 'Dokument'),
('preis', 'Preis'),
]
STATUS_CHOICES = [
('offen', 'Offen'),
('in_bearbeitung', 'In Bearbeitung'),
('wartend_intern', 'Wartend (intern)'),
('wartend_sub', 'Wartend (Subunternehmer)'),
('wartend_ausschreiber', 'Wartend (Ausschreiber)'),
('erledigt', 'Erledigt'),
('verworfen', 'Verworfen'),
('ueberfaellig', 'Überfällig'),
]
PRIORITAET_CHOICES = [(1, 'Hoch'), (2, 'Mittel'), (3, 'Niedrig')]
ausschreibung = models.ForeignKey(
'ausschreibungen.Ausschreibung', on_delete=models.CASCADE, related_name='aufgaben'
)
los = models.ForeignKey(
'lose.Los', on_delete=models.SET_NULL, null=True, blank=True, related_name='aufgaben'
)
anforderung = models.ForeignKey(
'lose.Anforderung', on_delete=models.SET_NULL,
null=True, blank=True, related_name='aufgaben'
)
bieterfrage = models.ForeignKey(
'Bieterfrage', on_delete=models.SET_NULL,
null=True, blank=True, related_name='aufgaben'
)
dokument = models.ForeignKey(
'dokumente.Dokument', on_delete=models.SET_NULL,
null=True, blank=True, related_name='aufgaben'
)
titel = models.CharField(max_length=400)
beschreibung = models.TextField(blank=True)
typ = models.CharField(max_length=20, choices=TYP_CHOICES, default='fachlich')
status = models.CharField(max_length=25, choices=STATUS_CHOICES, default='offen')
prioritaet = models.PositiveSmallIntegerField(choices=PRIORITAET_CHOICES, default=2)
frist = models.DateField(null=True, blank=True)
verantwortlicher = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL, null=True, blank=True
)
ergebnis = models.TextField(blank=True)
class Meta:
ordering = ['prioritaet', 'frist']
verbose_name = 'Aufgabe'
verbose_name_plural = 'Aufgaben'
def __str__(self):
return self.titel
@property
def ist_ueberfaellig(self):
if not self.frist:
return False
return self.frist < date.today() and self.status not in ['erledigt', 'verworfen']
class Bieterfrage(FlexibleModel):
STATUS_CHOICES = [
('entwurf', 'Entwurf'),
('abgestimmt', 'Abgestimmt'),
('eingereicht', 'Eingereicht'),
('beantwortet', 'Beantwortet'),
('eingearbeitet', 'Eingearbeitet'),
]
PRIORITAET_CHOICES = [(1, 'Hoch'), (2, 'Mittel'), (3, 'Niedrig')]
ausschreibung = models.ForeignKey(
'ausschreibungen.Ausschreibung', on_delete=models.CASCADE, related_name='bieterfragen'
)
anforderung = models.ForeignKey(
'lose.Anforderung', on_delete=models.SET_NULL,
null=True, blank=True, related_name='bieterfragen'
)
dokument = models.ForeignKey(
'dokumente.Dokument', on_delete=models.SET_NULL,
null=True, blank=True, related_name='bieterfragen'
)
fragentext = models.TextField()
begruendung = models.TextField(blank=True)
status = models.CharField(max_length=20, choices=STATUS_CHOICES, default='entwurf')
prioritaet = models.PositiveSmallIntegerField(choices=PRIORITAET_CHOICES, default=2)
einreichungsdatum = models.DateField(null=True, blank=True)
antwort = models.TextField(blank=True)
auswirkung_angebot = models.TextField(blank=True)
eingearbeitet = models.BooleanField(default=False)
verfasser = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL, null=True, blank=True
)
class Meta:
ordering = ['prioritaet', 'status']
verbose_name = 'Bieterfrage'
verbose_name_plural = 'Bieterfragen'
def __str__(self):
return self.fragentext[:80]

View File

@@ -0,0 +1,3 @@
from django.test import TestCase
# Create your tests here.

View File

@@ -0,0 +1,2 @@
from django.urls import path
urlpatterns = []

View File

@@ -0,0 +1,3 @@
from django.shortcuts import render
# Create your views here.

View File

@@ -0,0 +1,11 @@
from django.contrib import admin
from .models import Ausschreibung
@admin.register(Ausschreibung)
class AusschreibungAdmin(admin.ModelAdmin):
list_display = ['titel', 'ausschreiber', 'status', 'abgabe_bis', 'teilnahmeentscheidung']
list_filter = ['status', 'teilnahmeentscheidung', 'vergabeart']
search_fields = ['titel', 'ausschreiber', 'vergabenummer']
date_hierarchy = 'abgabe_bis'

View File

@@ -0,0 +1,5 @@
from django.apps import AppConfig
class AusschreibungenConfig(AppConfig):
name = 'vergabe_teilnahme.apps.ausschreibungen'

View File

@@ -0,0 +1,53 @@
# Generated by Django 6.0.5 on 2026-05-08 10:32
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.CreateModel(
name='Ausschreibung',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('titel', models.CharField(max_length=400)),
('ausschreiber', models.CharField(max_length=300)),
('vergabeplattform', models.CharField(blank=True, max_length=200)),
('vergabenummer', models.CharField(blank=True, max_length=100)),
('vergabeart', models.CharField(blank=True, choices=[('oeffentlich', 'Öffentliche Ausschreibung'), ('beschraenkt', 'Beschränkte Ausschreibung'), ('freihanding', 'Freihändige Vergabe'), ('wettbewerb', 'Wettbewerb'), ('rahmenvertrag', 'Rahmenvertrag'), ('sonstige', 'Sonstige')], max_length=30)),
('status', models.PositiveSmallIntegerField(choices=[(1, 'Recherche & Unterlagen'), (2, 'Teilnahmeentscheidung'), (3, 'Detaillierte Durchsicht'), (4, 'Klärungsphase'), (5, 'Preismodell'), (6, 'Unterlagen finalisieren'), (7, 'Abgabe'), (8, 'Zuschlag / Nachbetrachtung'), (9, 'Abgegeben'), (10, 'Gewonnen'), (11, 'Verloren'), (12, 'Aufgehoben'), (13, 'Zurückgezogen')], default=1)),
('teilnahmeentscheidung', models.CharField(choices=[('offen', 'Offen'), ('teilnahme', 'Teilnahme'), ('ablehnung', 'Ablehnung')], default='offen', max_length=20)),
('entscheidungsbegruendung', models.TextField(blank=True)),
('veroeffentlichungsdatum', models.DateField(blank=True, null=True)),
('bieterfragen_bis', models.DateField(blank=True, null=True)),
('abgabe_bis', models.DateTimeField(blank=True, null=True)),
('bindefrist', models.DateField(blank=True, null=True)),
('leistungsbeschreibung', models.TextField(blank=True)),
('branche', models.CharField(blank=True, max_length=200)),
('schlagwoerter', models.CharField(blank=True, max_length=500)),
('geschaetztes_volumen', models.DecimalField(blank=True, decimal_places=2, max_digits=14, null=True)),
('laufzeit', models.CharField(blank=True, max_length=100)),
('optionen', models.TextField(blank=True)),
('fundstelle_url', models.URLField(blank=True, max_length=1000)),
('unterlagen_erhalten', models.BooleanField(default=False)),
('unterlagen_erhalten_am', models.DateField(blank=True, null=True)),
('erstellt_am', models.DateTimeField(auto_now_add=True)),
('geaendert_am', models.DateTimeField(auto_now=True)),
('bid_manager', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='verwaltete_ausschreibungen', to=settings.AUTH_USER_MODEL)),
('team', models.ManyToManyField(blank=True, related_name='team_ausschreibungen', to=settings.AUTH_USER_MODEL)),
],
options={
'verbose_name': 'Ausschreibung',
'verbose_name_plural': 'Ausschreibungen',
'ordering': ['-erstellt_am'],
},
),
]

View File

@@ -0,0 +1,107 @@
from datetime import date
from django.db import models
from vergabe_teilnahme.apps.core.models import FlexibleModel
class Ausschreibung(FlexibleModel):
STATUS_CHOICES = [
(1, 'Recherche & Unterlagen'),
(2, 'Teilnahmeentscheidung'),
(3, 'Detaillierte Durchsicht'),
(4, 'Klärungsphase'),
(5, 'Preismodell'),
(6, 'Unterlagen finalisieren'),
(7, 'Abgabe'),
(8, 'Zuschlag / Nachbetrachtung'),
(9, 'Abgegeben'),
(10, 'Gewonnen'),
(11, 'Verloren'),
(12, 'Aufgehoben'),
(13, 'Zurückgezogen'),
]
TEILNAHME_CHOICES = [
('offen', 'Offen'),
('teilnahme', 'Teilnahme'),
('ablehnung', 'Ablehnung'),
]
VERGABEART_CHOICES = [
('oeffentlich', 'Öffentliche Ausschreibung'),
('beschraenkt', 'Beschränkte Ausschreibung'),
('freihanding', 'Freihändige Vergabe'),
('wettbewerb', 'Wettbewerb'),
('rahmenvertrag', 'Rahmenvertrag'),
('sonstige', 'Sonstige'),
]
titel = models.CharField(max_length=400)
ausschreiber = models.CharField(max_length=300)
vergabeplattform = models.CharField(max_length=200, blank=True)
vergabenummer = models.CharField(max_length=100, blank=True)
vergabeart = models.CharField(max_length=30, choices=VERGABEART_CHOICES, blank=True)
status = models.PositiveSmallIntegerField(choices=STATUS_CHOICES, default=1)
teilnahmeentscheidung = models.CharField(
max_length=20, choices=TEILNAHME_CHOICES, default='offen'
)
entscheidungsbegruendung = models.TextField(blank=True)
# Fristen
veroeffentlichungsdatum = models.DateField(null=True, blank=True)
bieterfragen_bis = models.DateField(null=True, blank=True)
abgabe_bis = models.DateTimeField(null=True, blank=True)
bindefrist = models.DateField(null=True, blank=True)
# Verantwortlichkeiten
bid_manager = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL,
null=True, blank=True, related_name='verwaltete_ausschreibungen'
)
team = models.ManyToManyField(
'accounts.Mitarbeiter', blank=True, related_name='team_ausschreibungen'
)
# Beschreibung & Klassifizierung
leistungsbeschreibung = models.TextField(blank=True)
branche = models.CharField(max_length=200, blank=True)
schlagwoerter = models.CharField(max_length=500, blank=True)
geschaetztes_volumen = models.DecimalField(max_digits=14, decimal_places=2, null=True, blank=True)
laufzeit = models.CharField(max_length=100, blank=True)
optionen = models.TextField(blank=True)
# Herkunft & Dokumente
fundstelle_url = models.URLField(max_length=1000, blank=True)
unterlagen_erhalten = models.BooleanField(default=False)
unterlagen_erhalten_am = models.DateField(null=True, blank=True)
# Timestamps
erstellt_am = models.DateTimeField(auto_now_add=True)
geaendert_am = models.DateTimeField(auto_now=True)
class Meta:
ordering = ['-erstellt_am']
verbose_name = 'Ausschreibung'
verbose_name_plural = 'Ausschreibungen'
def __str__(self):
return self.titel
@property
def ist_aktiv(self):
return 1 <= self.status <= 9
@property
def naechste_frist(self):
heute = date.today()
kandidaten = []
if self.bieterfragen_bis and self.bieterfragen_bis >= heute:
kandidaten.append(self.bieterfragen_bis)
if self.abgabe_bis:
abgabe = self.abgabe_bis.date() if hasattr(self.abgabe_bis, 'date') else self.abgabe_bis
if abgabe >= heute:
kandidaten.append(abgabe)
return min(kandidaten) if kandidaten else None
@property
def phase_nummer(self):
return min(8, max(1, self.status))

View File

@@ -0,0 +1,3 @@
from django.test import TestCase
# Create your tests here.

View File

@@ -0,0 +1,9 @@
from django.urls import path
from . import views
app_name = 'ausschreibungen'
urlpatterns = [
path('', views.dashboard, name='dashboard'),
]

View File

@@ -0,0 +1,7 @@
from django.shortcuts import render
def dashboard(request):
return render(request, 'ausschreibungen/dashboard.html', {
'breadcrumbs': [{'label': 'Übersicht', 'url': None}],
})

View File

@@ -0,0 +1,27 @@
from django.contrib import admin
from .models import Entscheidungsregel, Leistungsblatt, Nachweis, Referenz
@admin.register(Nachweis)
class NachweisAdmin(admin.ModelAdmin):
list_display = ['titel', 'kategorie', 'gueltig_bis', 'freigabestatus', 'ist_abgelaufen']
list_filter = ['kategorie', 'freigabestatus', 'fuer_oeffentliche']
@admin.register(Referenz)
class ReferenzAdmin(admin.ModelAdmin):
list_display = ['referenztitel', 'kunde', 'branche', 'freigabestatus_verwendung']
list_filter = ['freigabestatus_verwendung', 'verwendbar_fuer_ausschreibungen']
search_fields = ['referenztitel', 'kunde']
@admin.register(Leistungsblatt)
class LeistungsblattAdmin(admin.ModelAdmin):
list_display = ['produktfunktion', 'version', 'eigentuemer']
@admin.register(Entscheidungsregel)
class EntscheidungsregelAdmin(admin.ModelAdmin):
list_display = ['regelname', 'kategorie', 'empfehlung', 'aktiv']
list_filter = ['kategorie', 'empfehlung', 'aktiv']

View File

@@ -0,0 +1,5 @@
from django.apps import AppConfig
class BibliothekConfig(AppConfig):
name = 'vergabe_teilnahme.apps.bibliothek'

View File

@@ -0,0 +1,115 @@
# Generated by Django 6.0.5 on 2026-05-08 10:33
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.CreateModel(
name='Entscheidungsregel',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('regelname', models.CharField(max_length=300)),
('beschreibung', models.TextField(blank=True)),
('kategorie', models.CharField(choices=[('ausschlusskriterium', 'Ausschlusskriterium'), ('frist', 'Fristlage'), ('referenz', 'Referenzanforderung'), ('wirtschaftlichkeit', 'Wirtschaftlichkeit'), ('ressourcen', 'Ressourcenverfügbarkeit'), ('sonstiges', 'Sonstiges')], max_length=30)),
('gewichtung', models.DecimalField(decimal_places=2, default=1.0, max_digits=5)),
('bewertungslogik', models.TextField(blank=True)),
('schwellenwert', models.DecimalField(blank=True, decimal_places=2, max_digits=10, null=True)),
('empfehlung', models.CharField(choices=[('teilnehmen', 'Teilnehmen'), ('nicht_teilnehmen', 'Nicht teilnehmen'), ('pruefen', 'Prüfen')], max_length=20)),
('begruendung', models.TextField(blank=True)),
('gueltig_von', models.DateField(blank=True, null=True)),
('gueltig_bis', models.DateField(blank=True, null=True)),
('aktiv', models.BooleanField(default=True)),
('verantwortlicher', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)),
],
options={
'verbose_name': 'Entscheidungsregel',
'verbose_name_plural': 'Entscheidungsregeln',
'ordering': ['regelname'],
},
),
migrations.CreateModel(
name='Leistungsblatt',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('produktfunktion', models.CharField(max_length=300)),
('beschreibung', models.TextField(blank=True)),
('leistungsumfang', models.TextField(blank=True)),
('grenzen_ausschluesse', models.TextField(blank=True)),
('technische_voraussetzungen', models.TextField(blank=True)),
('typische_nachweise', models.TextField(blank=True)),
('version', models.CharField(default='1.0', max_length=20)),
('eigentuemer', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)),
],
options={
'verbose_name': 'Leistungsblatt',
'verbose_name_plural': 'Leistungsblätter',
'ordering': ['produktfunktion'],
},
),
migrations.CreateModel(
name='Nachweis',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('titel', models.CharField(max_length=300)),
('kurzbeschreibung', models.TextField(blank=True)),
('dokumenttyp', models.CharField(blank=True, max_length=100)),
('kategorie', models.CharField(blank=True, max_length=100)),
('datei', models.FileField(blank=True, null=True, upload_to='nachweise/%Y/')),
('version', models.CharField(default='1.0', max_length=20)),
('status', models.CharField(default='aktiv', max_length=20)),
('gueltig_ab', models.DateField(blank=True, null=True)),
('gueltig_bis', models.DateField(blank=True, null=True)),
('freigabestatus', models.CharField(choices=[('intern_freigegeben', 'Intern freigegeben'), ('extern_freigegeben', 'Extern freigegeben'), ('eingeschraenkt', 'Eingeschränkt'), ('nicht_freigegeben', 'Nicht freigegeben')], default='intern_freigegeben', max_length=30)),
('letzte_pruefung', models.DateField(blank=True, null=True)),
('zugehoerige_standards', models.TextField(blank=True)),
('vertraulichkeit', models.CharField(choices=[('intern', 'Intern'), ('streng_vertraulich', 'Streng vertraulich'), ('oeffentlich', 'Öffentlich')], default='intern', max_length=20)),
('sprache', models.CharField(default='de', max_length=5)),
('fuer_oeffentliche', models.BooleanField(default=True)),
('fuer_privatwirtschaftliche', models.BooleanField(default=True)),
('eigentuemer', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='eigene_nachweise', to=settings.AUTH_USER_MODEL)),
],
options={
'verbose_name': 'Nachweis',
'verbose_name_plural': 'Nachweise',
'ordering': ['titel'],
},
),
migrations.CreateModel(
name='Referenz',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('referenztitel', models.CharField(max_length=400)),
('kunde', models.CharField(max_length=300)),
('branche', models.CharField(blank=True, max_length=200)),
('oeffentlich_oder_privat', models.CharField(blank=True, max_length=20)),
('leistungsbeschreibung', models.TextField(blank=True)),
('eingesetzte_produkte', models.TextField(blank=True)),
('projektzeitraum', models.CharField(blank=True, max_length=100)),
('vertragsvolumen', models.DecimalField(blank=True, decimal_places=2, max_digits=14, null=True)),
('ansprechpartner_referenzkunde', models.CharField(blank=True, max_length=200)),
('freigabestatus_verwendung', models.CharField(choices=[('freigegeben', 'Freigegeben'), ('eingeschraenkt', 'Eingeschränkt'), ('nicht_freigegeben', 'Nicht freigegeben')], default='freigegeben', max_length=30)),
('vertraulichkeit', models.CharField(default='intern', max_length=20)),
('whitepaper', models.FileField(blank=True, null=True, upload_to='referenzen/')),
('kurzfassung', models.TextField(blank=True)),
('langfassung', models.TextField(blank=True)),
('verwendbar_fuer_ausschreibungen', models.BooleanField(default=True)),
('einschraenkungen_verwendung', models.TextField(blank=True)),
('leistungsblaetter', models.ManyToManyField(blank=True, to='bibliothek.leistungsblatt')),
],
options={
'verbose_name': 'Referenz',
'verbose_name_plural': 'Referenzen',
'ordering': ['referenztitel'],
},
),
]

View File

@@ -0,0 +1,154 @@
from datetime import date
from django.db import models
from vergabe_teilnahme.apps.core.models import FlexibleModel
class Leistungsblatt(FlexibleModel):
produktfunktion = models.CharField(max_length=300)
beschreibung = models.TextField(blank=True)
leistungsumfang = models.TextField(blank=True)
grenzen_ausschluesse = models.TextField(blank=True)
technische_voraussetzungen = models.TextField(blank=True)
typische_nachweise = models.TextField(blank=True)
version = models.CharField(max_length=20, default='1.0')
eigentuemer = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL, null=True, blank=True
)
class Meta:
verbose_name = 'Leistungsblatt'
verbose_name_plural = 'Leistungsblätter'
ordering = ['produktfunktion']
def __str__(self):
return self.produktfunktion
class Nachweis(FlexibleModel):
FREIGABE_CHOICES = [
('intern_freigegeben', 'Intern freigegeben'),
('extern_freigegeben', 'Extern freigegeben'),
('eingeschraenkt', 'Eingeschränkt'),
('nicht_freigegeben', 'Nicht freigegeben'),
]
VERTRAULICHKEIT_CHOICES = [
('intern', 'Intern'),
('streng_vertraulich', 'Streng vertraulich'),
('oeffentlich', 'Öffentlich'),
]
titel = models.CharField(max_length=300)
kurzbeschreibung = models.TextField(blank=True)
dokumenttyp = models.CharField(max_length=100, blank=True)
kategorie = models.CharField(max_length=100, blank=True)
datei = models.FileField(upload_to='nachweise/%Y/', null=True, blank=True)
version = models.CharField(max_length=20, default='1.0')
status = models.CharField(max_length=20, default='aktiv')
gueltig_ab = models.DateField(null=True, blank=True)
gueltig_bis = models.DateField(null=True, blank=True)
eigentuemer = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL,
null=True, blank=True, related_name='eigene_nachweise'
)
freigabestatus = models.CharField(
max_length=30, choices=FREIGABE_CHOICES, default='intern_freigegeben'
)
letzte_pruefung = models.DateField(null=True, blank=True)
zugehoerige_standards = models.TextField(blank=True)
vertraulichkeit = models.CharField(
max_length=20, choices=VERTRAULICHKEIT_CHOICES, default='intern'
)
sprache = models.CharField(max_length=5, default='de')
fuer_oeffentliche = models.BooleanField(default=True)
fuer_privatwirtschaftliche = models.BooleanField(default=True)
class Meta:
verbose_name = 'Nachweis'
verbose_name_plural = 'Nachweise'
ordering = ['titel']
def __str__(self):
return self.titel
@property
def ist_abgelaufen(self):
if not self.gueltig_bis:
return False
return self.gueltig_bis < date.today()
class Referenz(FlexibleModel):
FREIGABE_CHOICES = [
('freigegeben', 'Freigegeben'),
('eingeschraenkt', 'Eingeschränkt'),
('nicht_freigegeben', 'Nicht freigegeben'),
]
referenztitel = models.CharField(max_length=400)
kunde = models.CharField(max_length=300)
branche = models.CharField(max_length=200, blank=True)
oeffentlich_oder_privat = models.CharField(max_length=20, blank=True)
leistungsbeschreibung = models.TextField(blank=True)
eingesetzte_produkte = models.TextField(blank=True)
projektzeitraum = models.CharField(max_length=100, blank=True)
vertragsvolumen = models.DecimalField(max_digits=14, decimal_places=2, null=True, blank=True)
ansprechpartner_referenzkunde = models.CharField(max_length=200, blank=True)
freigabestatus_verwendung = models.CharField(
max_length=30, choices=FREIGABE_CHOICES, default='freigegeben'
)
vertraulichkeit = models.CharField(max_length=20, default='intern')
whitepaper = models.FileField(upload_to='referenzen/', null=True, blank=True)
kurzfassung = models.TextField(blank=True)
langfassung = models.TextField(blank=True)
verwendbar_fuer_ausschreibungen = models.BooleanField(default=True)
einschraenkungen_verwendung = models.TextField(blank=True)
leistungsblaetter = models.ManyToManyField(Leistungsblatt, blank=True)
class Meta:
verbose_name = 'Referenz'
verbose_name_plural = 'Referenzen'
ordering = ['referenztitel']
def __str__(self):
return self.referenztitel
class Entscheidungsregel(FlexibleModel):
KATEGORIE_CHOICES = [
('ausschlusskriterium', 'Ausschlusskriterium'),
('frist', 'Fristlage'),
('referenz', 'Referenzanforderung'),
('wirtschaftlichkeit', 'Wirtschaftlichkeit'),
('ressourcen', 'Ressourcenverfügbarkeit'),
('sonstiges', 'Sonstiges'),
]
EMPFEHLUNG_CHOICES = [
('teilnehmen', 'Teilnehmen'),
('nicht_teilnehmen', 'Nicht teilnehmen'),
('pruefen', 'Prüfen'),
]
regelname = models.CharField(max_length=300)
beschreibung = models.TextField(blank=True)
kategorie = models.CharField(max_length=30, choices=KATEGORIE_CHOICES)
gewichtung = models.DecimalField(max_digits=5, decimal_places=2, default=1.0)
bewertungslogik = models.TextField(blank=True)
schwellenwert = models.DecimalField(max_digits=10, decimal_places=2, null=True, blank=True)
empfehlung = models.CharField(max_length=20, choices=EMPFEHLUNG_CHOICES)
begruendung = models.TextField(blank=True)
gueltig_von = models.DateField(null=True, blank=True)
gueltig_bis = models.DateField(null=True, blank=True)
aktiv = models.BooleanField(default=True)
verantwortlicher = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL, null=True, blank=True
)
class Meta:
verbose_name = 'Entscheidungsregel'
verbose_name_plural = 'Entscheidungsregeln'
ordering = ['regelname']
def __str__(self):
return self.regelname

View File

@@ -0,0 +1,3 @@
from django.test import TestCase
# Create your tests here.

View File

@@ -0,0 +1,2 @@
from django.urls import path
urlpatterns = []

View File

@@ -0,0 +1,3 @@
from django.shortcuts import render
# Create your views here.

View File

View File

@@ -0,0 +1,22 @@
from django.contrib import admin
from .models import CustomAttribute, EntityFieldConfig, Freigabe
@admin.register(EntityFieldConfig)
class EntityFieldConfigAdmin(admin.ModelAdmin):
list_display = ['entity_type', 'field_name', 'is_hidden', 'display_label', 'sort_order']
list_filter = ['entity_type', 'is_hidden']
search_fields = ['entity_type', 'field_name']
@admin.register(CustomAttribute)
class CustomAttributeAdmin(admin.ModelAdmin):
list_display = ['label', 'key', 'data_type', 'value', 'content_type', 'object_id']
list_filter = ['data_type', 'content_type']
@admin.register(Freigabe)
class FreigabeAdmin(admin.ModelAdmin):
list_display = ['freigabe_typ', 'status', 'freigebende_person', 'timestamp']
list_filter = ['freigabe_typ', 'status']

View File

@@ -0,0 +1,5 @@
from django.apps import AppConfig
class CoreConfig(AppConfig):
name = 'vergabe_teilnahme.apps.core'

View File

@@ -0,0 +1,13 @@
def vergabe_context(request):
context = {}
ausschreibung_id = None
if hasattr(request, 'resolver_match') and request.resolver_match:
kwargs = request.resolver_match.kwargs
ausschreibung_id = kwargs.get('ausschreibung_id') or kwargs.get('pk')
if ausschreibung_id:
try:
from vergabe_teilnahme.apps.ausschreibungen.models import Ausschreibung
context['current_ausschreibung'] = Ausschreibung.objects.get(pk=ausschreibung_id)
except (Ausschreibung.DoesNotExist, ValueError):
pass
return context

View File

@@ -0,0 +1,239 @@
from datetime import date, timedelta
from django.core.management.base import BaseCommand
class Command(BaseCommand):
help = 'Erstellt Entwicklungs-Seed-Daten (idempotent)'
def handle(self, *args, **options):
from vergabe_teilnahme.apps.accounts.models import Mitarbeiter
from vergabe_teilnahme.apps.ausschreibungen.models import Ausschreibung
from vergabe_teilnahme.apps.aufgaben.models import Aufgabe
from vergabe_teilnahme.apps.bibliothek.models import Entscheidungsregel, Nachweis
from vergabe_teilnahme.apps.lose.models import Anforderung, Los
from vergabe_teilnahme.apps.marktbegleiter.models import Marktbegleiter
from vergabe_teilnahme.apps.partner.models import Dienstleistertyp, Subunternehmer
from vergabe_teilnahme.apps.preise.models import Preispunkt
self.stdout.write('Erstelle Seed-Daten...')
# 1. Mitarbeiter
bm, _ = Mitarbeiter.objects.get_or_create(
username='max.muster',
defaults={
'first_name': 'Max', 'last_name': 'Muster',
'rolle': 'bid_manager', 'organisationseinheit': 'Vertrieb',
'email': 'max.muster@example.com',
}
)
bm.set_password('testpass123')
bm.save(update_fields=['password'])
fach, _ = Mitarbeiter.objects.get_or_create(
username='anna.fach',
defaults={
'first_name': 'Anna', 'last_name': 'Fach',
'rolle': 'fachexperte', 'organisationseinheit': 'Technik',
'email': 'anna.fach@example.com',
}
)
fach.set_password('testpass123')
fach.save(update_fields=['password'])
gf, _ = Mitarbeiter.objects.get_or_create(
username='georg.chef',
defaults={
'first_name': 'Georg', 'last_name': 'Chef',
'rolle': 'geschaeftsfuehrung', 'organisationseinheit': 'GF',
'email': 'georg.chef@example.com', 'is_staff': True,
}
)
gf.set_password('testpass123')
gf.save(update_fields=['password'])
self.stdout.write(f' ✓ 3 Mitarbeiter')
# 2. Dienstleistertypen
dt_it, _ = Dienstleistertyp.objects.get_or_create(
name='IT-Dienstleister',
defaults={'beschreibung': 'Software- und IT-Infrastruktur-Dienstleister'}
)
dt_recht, _ = Dienstleistertyp.objects.get_or_create(
name='Rechtsberatung',
defaults={'beschreibung': 'Anwaltskanzleien und Rechtsberatungsunternehmen'}
)
self.stdout.write(f' ✓ 2 Dienstleistertypen')
# 3. Subunternehmer
sub, _ = Subunternehmer.objects.get_or_create(
name='TechPartner GmbH',
defaults={
'dienstleistertyp': dt_it,
'praeferenz': 'bevorzugt',
'ort': 'München',
'email': 'kontakt@techpartner.example.com',
'leistungsprofil': 'Cloud-Infrastruktur, DevOps, Security',
}
)
self.stdout.write(f' ✓ 1 Subunternehmer')
# 4. Nachweise
nachweis_iso, _ = Nachweis.objects.get_or_create(
titel='ISO 27001 Zertifikat',
defaults={
'kategorie': 'Zertifizierung',
'version': '2.0',
'gueltig_ab': date.today() - timedelta(days=365),
'gueltig_bis': date.today() + timedelta(days=365),
'eigentuemer': bm,
'freigabestatus': 'intern_freigegeben',
}
)
nachweis_dsgvo, _ = Nachweis.objects.get_or_create(
titel='DSGVO-Datenschutzerklärung',
defaults={
'kategorie': 'Rechtlich',
'version': '1.2',
'gueltig_bis': date.today() + timedelta(days=730),
'eigentuemer': bm,
'freigabestatus': 'intern_freigegeben',
}
)
self.stdout.write(f' ✓ 2 Nachweise')
# 5. Marktbegleiter
mb, _ = Marktbegleiter.objects.get_or_create(
name='Konkurrent AG',
defaults={
'kurzbeschreibung': 'Größter Wettbewerber im IT-Bereich',
'relevante_branchen': 'IT, Behörden, Finanzwesen',
'bekannte_staerken': 'Große Kundenbasis, günstige Preise',
'bekannte_schwaechen': 'Langsamer Support, wenig Flexibilität',
}
)
self.stdout.write(f' ✓ 1 Marktbegleiter')
# 6. Entscheidungsregeln
Entscheidungsregel.objects.get_or_create(
regelname='Muss-Anforderung nicht erfüllbar',
defaults={
'kategorie': 'ausschlusskriterium',
'empfehlung': 'nicht_teilnehmen',
'begruendung': 'Falls eine Muss-Anforderung nicht erfüllt werden kann.',
'aktiv': True,
}
)
Entscheidungsregel.objects.get_or_create(
regelname='Abgabefrist zu kurz',
defaults={
'kategorie': 'frist',
'empfehlung': 'pruefen',
'begruendung': 'Weniger als 2 Wochen bis zur Abgabe.',
'aktiv': True,
}
)
Entscheidungsregel.objects.get_or_create(
regelname='Fehlende Referenz',
defaults={
'kategorie': 'referenz',
'empfehlung': 'nicht_teilnehmen',
'begruendung': 'Keine passende Referenz für die geforderte Branche vorhanden.',
'aktiv': True,
}
)
self.stdout.write(f' ✓ 3 Entscheidungsregeln')
# 7. Ausschreibung mit Losen, Anforderungen, Aufgaben, Preispunkten
ausschreibung, _ = Ausschreibung.objects.get_or_create(
vergabenummer='DEV-2026-001',
defaults={
'titel': 'Digitalisierung Verwaltungsportal Phase 2',
'ausschreiber': 'Beispiel-Behörde GmbH',
'vergabeplattform': 'DTVP',
'status': 4,
'bieterfragen_bis': date.today() + timedelta(days=5),
'abgabe_bis': None,
'bid_manager': bm,
'branche': 'Öffentliche Verwaltung',
'leistungsbeschreibung': 'Entwicklung und Betrieb eines digitalen Verwaltungsportals.',
}
)
los1, _ = Los.objects.get_or_create(
ausschreibung=ausschreibung, losnummer='1',
defaults={'lostitel': 'Softwareentwicklung', 'zustaendiger': fach}
)
los2, _ = Los.objects.get_or_create(
ausschreibung=ausschreibung, losnummer='2',
defaults={'lostitel': 'Hosting & Betrieb', 'zustaendiger': bm}
)
anf1, _ = Anforderung.objects.get_or_create(
ausschreibung=ausschreibung,
titel='ISO 27001 Zertifizierung nachweisen',
defaults={
'verbindlichkeit': 'muss', 'ausschlusskriterium': True,
'kategorie': 'zertifizierung', 'los': los1,
}
)
anf2, _ = Anforderung.objects.get_or_create(
ausschreibung=ausschreibung,
titel='Mindestens 3 Referenzprojekte in der öffentlichen Verwaltung',
defaults={
'verbindlichkeit': 'muss', 'kategorie': 'referenz', 'los': los1,
}
)
anf3, _ = Anforderung.objects.get_or_create(
ausschreibung=ausschreibung,
titel='SLA 99,5% Verfügbarkeit',
defaults={
'verbindlichkeit': 'soll', 'kategorie': 'technisch', 'los': los2,
}
)
Aufgabe.objects.get_or_create(
ausschreibung=ausschreibung,
titel='ISO-Zertifikat aktualisieren und einreichen',
defaults={
'typ': 'dokument', 'prioritaet': 1,
'verantwortlicher': bm, 'anforderung': anf1,
}
)
Aufgabe.objects.get_or_create(
ausschreibung=ausschreibung,
titel='Referenzliste zusammenstellen',
defaults={
'typ': 'fachlich', 'prioritaet': 2,
'verantwortlicher': fach, 'anforderung': anf2,
}
)
from decimal import Decimal
Preispunkt.objects.get_or_create(
ausschreibung=ausschreibung,
konkrete_leistung='Entwicklung Frontend',
defaults={
'leistungstyp': 'Entwicklung',
'menge': Decimal('1'),
'mengeneinheit': 'Pauschal',
'einzelpreis': Decimal('85000.00'),
'los': los1,
}
)
Preispunkt.objects.get_or_create(
ausschreibung=ausschreibung,
konkrete_leistung='Hosting p.a.',
defaults={
'leistungstyp': 'Betrieb',
'menge': Decimal('1'),
'mengeneinheit': 'Jahr',
'einzelpreis': Decimal('24000.00'),
'wiederkehrend': True,
'los': los2,
'vergleichsgewicht': Decimal('0.8'),
}
)
self.stdout.write(f' ✓ 1 Ausschreibung mit 2 Losen, 3 Anforderungen, 2 Aufgaben, 2 Preispunkten')
self.stdout.write(self.style.SUCCESS('\nSeed-Daten erfolgreich erstellt.'))

View File

@@ -0,0 +1,70 @@
# Generated by Django 6.0.5 on 2026-05-08 10:32
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
('contenttypes', '0002_remove_content_type_name'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.CreateModel(
name='EntityFieldConfig',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('entity_type', models.CharField(max_length=100)),
('field_name', models.CharField(max_length=100)),
('is_hidden', models.BooleanField(default=False)),
('display_label', models.CharField(blank=True, max_length=200)),
('sort_order', models.PositiveSmallIntegerField(default=0)),
],
options={
'verbose_name': 'Feldkonfiguration',
'verbose_name_plural': 'Feldkonfigurationen',
'unique_together': {('entity_type', 'field_name')},
},
),
migrations.CreateModel(
name='Freigabe',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('object_id', models.PositiveIntegerField()),
('freigabe_typ', models.CharField(choices=[('teilnahme', 'Teilnahme-Freigabe'), ('ablehnung', 'Ablehnungs-Freigabe'), ('recht', 'Rechtliche Freigabe'), ('preis', 'Preis-Freigabe'), ('abgabe', 'Abgabe-Freigabe'), ('standarddokument', 'Standarddokument-Freigabe'), ('referenz', 'Referenz-Freigabe')], max_length=30)),
('status', models.CharField(choices=[('erteilt', 'Erteilt'), ('abgelehnt', 'Abgelehnt'), ('ausstehend', 'Ausstehend')], default='erteilt', max_length=20)),
('kommentar', models.TextField(blank=True)),
('timestamp', models.DateTimeField(auto_now_add=True)),
('content_type', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='contenttypes.contenttype')),
('freigebende_person', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='erteilte_freigaben', to=settings.AUTH_USER_MODEL)),
],
options={
'verbose_name': 'Freigabe',
'verbose_name_plural': 'Freigaben',
'ordering': ['-timestamp'],
},
),
migrations.CreateModel(
name='CustomAttribute',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('object_id', models.PositiveIntegerField()),
('key', models.CharField(max_length=100)),
('label', models.CharField(max_length=200)),
('value', models.TextField(blank=True)),
('data_type', models.CharField(choices=[('text', 'Text'), ('number', 'Zahl'), ('date', 'Datum'), ('boolean', 'Ja/Nein'), ('url', 'URL'), ('email', 'E-Mail')], default='text', max_length=20)),
('sort_order', models.PositiveSmallIntegerField(default=0)),
('created_at', models.DateTimeField(auto_now_add=True)),
('content_type', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='contenttypes.contenttype')),
],
options={
'ordering': ['sort_order', 'created_at'],
'indexes': [models.Index(fields=['content_type', 'object_id'], name='core_custom_content_b275a1_idx')],
},
),
]

View File

@@ -0,0 +1,106 @@
from django.conf import settings
from django.contrib.contenttypes.fields import GenericForeignKey, GenericRelation
from django.contrib.contenttypes.models import ContentType
from django.db import models
class EntityFieldConfig(models.Model):
entity_type = models.CharField(max_length=100)
field_name = models.CharField(max_length=100)
is_hidden = models.BooleanField(default=False)
display_label = models.CharField(max_length=200, blank=True)
sort_order = models.PositiveSmallIntegerField(default=0)
class Meta:
unique_together = ('entity_type', 'field_name')
verbose_name = 'Feldkonfiguration'
verbose_name_plural = 'Feldkonfigurationen'
def __str__(self):
return f'{self.entity_type}.{self.field_name}'
class CustomAttribute(models.Model):
DATA_TYPE_CHOICES = [
('text', 'Text'),
('number', 'Zahl'),
('date', 'Datum'),
('boolean', 'Ja/Nein'),
('url', 'URL'),
('email', 'E-Mail'),
]
content_type = models.ForeignKey(ContentType, on_delete=models.CASCADE)
object_id = models.PositiveIntegerField()
content_object = GenericForeignKey('content_type', 'object_id')
key = models.CharField(max_length=100)
label = models.CharField(max_length=200)
value = models.TextField(blank=True)
data_type = models.CharField(max_length=20, choices=DATA_TYPE_CHOICES, default='text')
sort_order = models.PositiveSmallIntegerField(default=0)
created_at = models.DateTimeField(auto_now_add=True)
class Meta:
ordering = ['sort_order', 'created_at']
indexes = [models.Index(fields=['content_type', 'object_id'])]
def __str__(self):
return f'{self.label}: {self.value}'
class Freigabe(models.Model):
TYP_CHOICES = [
('teilnahme', 'Teilnahme-Freigabe'),
('ablehnung', 'Ablehnungs-Freigabe'),
('recht', 'Rechtliche Freigabe'),
('preis', 'Preis-Freigabe'),
('abgabe', 'Abgabe-Freigabe'),
('standarddokument', 'Standarddokument-Freigabe'),
('referenz', 'Referenz-Freigabe'),
]
STATUS_CHOICES = [
('erteilt', 'Erteilt'),
('abgelehnt', 'Abgelehnt'),
('ausstehend', 'Ausstehend'),
]
content_type = models.ForeignKey(ContentType, on_delete=models.CASCADE)
object_id = models.PositiveIntegerField()
content_object = GenericForeignKey('content_type', 'object_id')
freigabe_typ = models.CharField(max_length=30, choices=TYP_CHOICES)
freigebende_person = models.ForeignKey(
settings.AUTH_USER_MODEL, on_delete=models.PROTECT, related_name='erteilte_freigaben'
)
status = models.CharField(max_length=20, choices=STATUS_CHOICES, default='erteilt')
kommentar = models.TextField(blank=True)
timestamp = models.DateTimeField(auto_now_add=True)
class Meta:
ordering = ['-timestamp']
verbose_name = 'Freigabe'
verbose_name_plural = 'Freigaben'
def __str__(self):
return f'{self.get_freigabe_typ_display()}{self.status}'
class FlexibleModel(models.Model):
custom_attributes = GenericRelation(CustomAttribute)
freigaben = GenericRelation(Freigabe)
def get_hidden_fields(self):
model_name = self._meta.model_name
return set(
EntityFieldConfig.objects.filter(
entity_type=model_name, is_hidden=True
).values_list('field_name', flat=True)
)
def get_visible_field_names(self):
hidden = self.get_hidden_fields()
return [
f.name
for f in self._meta.get_fields()
if hasattr(f, 'column') and not f.name.startswith('_') and f.name not in hidden
]
class Meta:
abstract = True

View File

@@ -0,0 +1,106 @@
from datetime import date
from decimal import Decimal
PHASEN = [
(1, 'Recherche & Unterlagen'),
(2, 'Teilnahmeentscheidung'),
(3, 'Detaillierte Durchsicht'),
(4, 'Bieterfragen & Klärung'),
(5, 'Preismodell'),
(6, 'Unterlagen finalisieren'),
(7, 'Abgabe'),
(8, 'Zuschlag / Nachbetrachtung'),
]
# Maps Ausschreibung.status integer to phase number
STATUS_TO_PHASE = {
1: 1, 2: 1, # Recherche
3: 2, # Teilnahmeentscheidung
4: 3, 5: 3, # Durchsicht
6: 4, 7: 4, # Bieterfragen
8: 5, # Preise
9: 6, # Finalisierung
10: 7, 11: 7, # Abgabe
12: 8, 13: 8, # Nachbetrachtung
}
def build_phase_nav(ausschreibung, current_url=''):
aktuelle_phase = STATUS_TO_PHASE.get(ausschreibung.status, 1)
base = f'/ausschreibungen/{ausschreibung.pk}'
phase_urls = {
1: f'{base}/',
2: f'{base}/teilnahmeentscheidung/',
3: f'{base}/anforderungen/',
4: f'{base}/bieterfragen/',
5: f'{base}/preise/',
6: f'{base}/dokumente/',
7: f'{base}/abgabe/',
8: f'{base}/nachbetrachtung/',
}
return [
{
'nummer': num,
'name': name,
'url': phase_urls[num],
'aktiv': num == aktuelle_phase,
'erledigt': num < aktuelle_phase,
'warnung': False,
}
for num, name in PHASEN
]
def get_deadline_warnings(ausschreibung):
"""Gibt Liste von Warnungen für nahende Fristen zurück."""
warnings = []
heute = date.today()
if ausschreibung.bieterfragen_bis:
delta = (ausschreibung.bieterfragen_bis - heute).days
if delta <= 3:
warnings.append({
'typ': 'bieterfragen',
'tage': delta,
'farbe': 'red' if delta <= 1 else 'amber',
})
if ausschreibung.abgabe_bis:
abgabe_date = (
ausschreibung.abgabe_bis.date()
if hasattr(ausschreibung.abgabe_bis, 'date')
else ausschreibung.abgabe_bis
)
delta = (abgabe_date - heute).days
if delta <= 14:
warnings.append({
'typ': 'abgabe',
'tage': delta,
'farbe': 'red' if delta <= 3 else 'amber',
})
return warnings
def gewichteter_durchschnitt(preispunkte, feld='einzelpreis'):
"""Berechnet gewichteten Durchschnitt für Preispunkte.
Punkte mit Gewicht 0,0 werden ausgeschlossen.
Gibt None zurück wenn keine verwertbaren Punkte vorhanden.
"""
relevante = [
p for p in preispunkte
if getattr(p, feld) is not None and p.vergleichsgewicht > 0
]
if not relevante:
return None
summe_gewichte = sum(p.vergleichsgewicht for p in relevante)
if summe_gewichte == 0:
return None
summe = sum(getattr(p, feld) * p.vergleichsgewicht for p in relevante)
werte = [getattr(p, feld) for p in relevante]
return {
'wert': summe / summe_gewichte,
'summe_gewichte': summe_gewichte,
'anzahl': len(relevante),
'minimum': min(werte),
'maximum': max(werte),
'ungewichtet': sum(werte) / len(werte),
}

View File

@@ -0,0 +1,71 @@
from django import template
from django.utils.safestring import mark_safe
register = template.Library()
STATUS_COLORS = {
'offen': 'bg-slate-100 text-slate-700',
'in_bearbeitung': 'bg-blue-100 text-blue-700',
'erledigt': 'bg-green-100 text-green-700',
'freigegeben': 'bg-green-100 text-green-700',
'erteilt': 'bg-green-100 text-green-700',
'gewonnen': 'bg-green-100 text-green-700',
'intern_freigegeben': 'bg-green-100 text-green-700',
'ueberfaellig': 'bg-red-100 text-red-700',
'nicht_erfuellbar': 'bg-red-100 text-red-700',
'verloren': 'bg-red-100 text-red-700',
'abgelehnt': 'bg-red-100 text-red-700',
'ausstehend': 'bg-amber-100 text-amber-700',
'in_pruefung': 'bg-amber-100 text-amber-700',
'wartend_intern': 'bg-amber-100 text-amber-700',
'wartend_sub': 'bg-amber-100 text-amber-700',
'wartend_ausschreiber': 'bg-amber-100 text-amber-700',
'archiviert': 'bg-gray-100 text-gray-500',
'ersetzt': 'bg-gray-100 text-gray-500',
'verworfen': 'bg-gray-100 text-gray-500',
}
@register.inclusion_tag('partials/status_badge.html')
def status_badge(value, display_label=None):
css = STATUS_COLORS.get(str(value), 'bg-slate-100 text-slate-700')
label = display_label or str(value).replace('_', ' ').capitalize()
return {'css': css, 'label': label}
@register.simple_tag
def phase_badge(nummer, zustand='todo'):
css_map = {
'todo': 'phase-todo',
'active': 'phase-active',
'done': 'phase-done',
'warn': 'phase-warn',
}
css = css_map.get(zustand, 'phase-todo')
return mark_safe(f'<span class="{css}">{nummer}</span>')
def _is_field_hidden(entity_type, field_name):
from vergabe_teilnahme.apps.core.models import EntityFieldConfig
return EntityFieldConfig.objects.filter(
entity_type=entity_type, field_name=field_name, is_hidden=True
).exists()
def _get_field_label(entity_type, field_name, default_label):
from vergabe_teilnahme.apps.core.models import EntityFieldConfig
try:
cfg = EntityFieldConfig.objects.get(entity_type=entity_type, field_name=field_name)
return cfg.display_label or default_label
except EntityFieldConfig.DoesNotExist:
return default_label
@register.inclusion_tag('partials/field_row.html')
def render_field(obj, field_name, label=None, force_show=False):
entity_type = obj._meta.model_name
if not force_show and _is_field_hidden(entity_type, field_name):
return {'hidden': True}
value = getattr(obj, field_name, None)
display_label = _get_field_label(entity_type, field_name, label or field_name)
return {'hidden': False, 'label': display_label, 'value': value, 'field_name': field_name}

View File

@@ -0,0 +1,3 @@
from django.test import TestCase
# Create your tests here.

View File

@@ -0,0 +1,41 @@
from decimal import Decimal
import pytest
from vergabe_teilnahme.apps.core.services import gewichteter_durchschnitt
class FakePreispunkt:
def __init__(self, einzelpreis, vergleichsgewicht):
self.einzelpreis = Decimal(str(einzelpreis)) if einzelpreis is not None else None
self.vergleichsgewicht = Decimal(str(vergleichsgewicht))
class TestGewichteterDurchschnitt:
def test_blueprint_beispiel(self):
# 100×1,0 + 80×0,2 + 110×1,2 = 100+16+132=248 / (1,0+0,2+1,2)=2,4 = 103,333...
punkte = [
FakePreispunkt(100, '1.0'),
FakePreispunkt(80, '0.2'),
FakePreispunkt(110, '1.2'),
]
result = gewichteter_durchschnitt(punkte)
assert result is not None
assert abs(result['wert'] - Decimal('103.333')) < Decimal('0.001')
assert result['anzahl'] == 3
assert result['minimum'] == Decimal('80')
assert result['maximum'] == Decimal('110')
def test_leerer_input(self):
assert gewichteter_durchschnitt([]) is None
def test_alle_gewichte_null(self):
punkte = [FakePreispunkt(100, '0.0'), FakePreispunkt(200, '0.0')]
assert gewichteter_durchschnitt(punkte) is None
def test_einzelpreis_none_wird_ausgeschlossen(self):
punkte = [FakePreispunkt(None, '1.0'), FakePreispunkt(100, '1.0')]
result = gewichteter_durchschnitt(punkte)
assert result is not None
assert result['anzahl'] == 1
assert result['wert'] == Decimal('100')

View File

@@ -0,0 +1,14 @@
def make_breadcrumbs(*args):
"""Build a breadcrumbs list from alternating label/url pairs or (label, url) tuples.
Usage:
make_breadcrumbs(('Ausschreibungen', '/ausschreibungen/'), ('Mein Tender', None))
"""
crumbs = []
for item in args:
if isinstance(item, (list, tuple)) and len(item) == 2:
label, url = item
else:
label, url = item, None
crumbs.append({'label': label, 'url': url})
return crumbs

View File

@@ -0,0 +1,29 @@
from django.http import HttpResponse
from django.shortcuts import render
from django.views.decorators.http import require_GET
def custom_404(request, exception=None):
return render(request, 'errors/404.html', status=404)
def custom_500(request):
return render(request, 'errors/500.html', status=500)
@require_GET
def suche(request):
q = request.GET.get('q', '').strip()
if not q or len(q) < 2:
return HttpResponse('')
from vergabe_teilnahme.apps.ausschreibungen.models import Ausschreibung
ausschreibungen = Ausschreibung.objects.filter(titel__icontains=q)[:5]
if not ausschreibungen:
return HttpResponse('')
items = ''.join(
f'<a href="/ausschreibungen/{a.pk}/" '
f'class="block px-4 py-2 text-sm text-slate-700 hover:bg-slate-50">'
f'{a.titel}</a>'
for a in ausschreibungen
)
return HttpResponse(items)

View File

@@ -0,0 +1,10 @@
from django.contrib import admin
from .models import Dokument
@admin.register(Dokument)
class DokumentAdmin(admin.ModelAdmin):
list_display = ['dateiname', 'kategorie', 'status', 'version', 'upload_datum', 'finale_abgabeversion']
list_filter = ['kategorie', 'status', 'finale_abgabeversion']
search_fields = ['dateiname', 'beschreibung']

View File

@@ -0,0 +1,5 @@
from django.apps import AppConfig
class DokumenteConfig(AppConfig):
name = 'vergabe_teilnahme.apps.dokumente'

View File

@@ -0,0 +1,36 @@
# Generated by Django 6.0.5 on 2026-05-08 10:33
import django.db.models.deletion
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
('ausschreibungen', '0001_initial'),
]
operations = [
migrations.CreateModel(
name='Dokument',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('datei', models.FileField(upload_to='dokumente/%Y/%m/')),
('dateiname', models.CharField(blank=True, max_length=300)),
('kategorie', models.CharField(choices=[('angebotsunterlage', 'Angebotsunterlage'), ('leistungsbeschreibung', 'Leistungsbeschreibung'), ('preisblatt', 'Preisblatt'), ('referenz', 'Referenz'), ('nachweis', 'Nachweis'), ('abgabenachweis', 'Abgabenachweis'), ('kommunikation', 'Kommunikation'), ('intern', 'Intern'), ('sonstiges', 'Sonstiges')], default='sonstiges', max_length=30)),
('status', models.CharField(choices=[('entwurf', 'Entwurf'), ('in_pruefung', 'In Prüfung'), ('freigegeben', 'Freigegeben'), ('final_abgegeben', 'Final abgegeben'), ('ersetzt', 'Ersetzt'), ('archiviert', 'Archiviert')], default='entwurf', max_length=20)),
('version', models.CharField(default='1.0', max_length=20)),
('beschreibung', models.TextField(blank=True)),
('finale_abgabeversion', models.BooleanField(default=False)),
('upload_datum', models.DateTimeField(auto_now_add=True)),
('ausschreibung', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='dokumente', to='ausschreibungen.ausschreibung')),
],
options={
'verbose_name': 'Dokument',
'verbose_name_plural': 'Dokumente',
'ordering': ['-upload_datum'],
},
),
]

View File

@@ -0,0 +1,34 @@
# Generated by Django 6.0.5 on 2026-05-08 10:33
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
('dokumente', '0001_initial'),
('lose', '0001_initial'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.AddField(
model_name='dokument',
name='los',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='dokumente', to='lose.los'),
),
migrations.AddField(
model_name='dokument',
name='pruefer',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='zu_pruefende_dokumente', to=settings.AUTH_USER_MODEL),
),
migrations.AddField(
model_name='dokument',
name='verantwortlicher',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='verantwortete_dokumente', to=settings.AUTH_USER_MODEL),
),
]

View File

@@ -0,0 +1,97 @@
from django.conf import settings
from django.contrib.contenttypes.fields import GenericRelation
from django.core.exceptions import ValidationError
from django.db import models
from django.db.models.signals import pre_save
from django.dispatch import receiver
from vergabe_teilnahme.apps.core.models import FlexibleModel, Freigabe
ALLOWED_EXTENSIONS = {'.pdf', '.docx', '.xlsx', '.zip', '.png', '.jpg', '.jpeg'}
class Dokument(FlexibleModel):
STATUS_CHOICES = [
('entwurf', 'Entwurf'),
('in_pruefung', 'In Prüfung'),
('freigegeben', 'Freigegeben'),
('final_abgegeben', 'Final abgegeben'),
('ersetzt', 'Ersetzt'),
('archiviert', 'Archiviert'),
]
KATEGORIE_CHOICES = [
('angebotsunterlage', 'Angebotsunterlage'),
('leistungsbeschreibung', 'Leistungsbeschreibung'),
('preisblatt', 'Preisblatt'),
('referenz', 'Referenz'),
('nachweis', 'Nachweis'),
('abgabenachweis', 'Abgabenachweis'),
('kommunikation', 'Kommunikation'),
('intern', 'Intern'),
('sonstiges', 'Sonstiges'),
]
ausschreibung = models.ForeignKey(
'ausschreibungen.Ausschreibung', on_delete=models.SET_NULL,
null=True, blank=True, related_name='dokumente'
)
los = models.ForeignKey(
'lose.Los', on_delete=models.SET_NULL,
null=True, blank=True, related_name='dokumente'
)
datei = models.FileField(upload_to='dokumente/%Y/%m/')
dateiname = models.CharField(max_length=300, blank=True)
kategorie = models.CharField(max_length=30, choices=KATEGORIE_CHOICES, default='sonstiges')
status = models.CharField(max_length=20, choices=STATUS_CHOICES, default='entwurf')
version = models.CharField(max_length=20, default='1.0')
beschreibung = models.TextField(blank=True)
verantwortlicher = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL,
null=True, blank=True, related_name='verantwortete_dokumente'
)
pruefer = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL,
null=True, blank=True, related_name='zu_pruefende_dokumente'
)
finale_abgabeversion = models.BooleanField(default=False)
upload_datum = models.DateTimeField(auto_now_add=True)
freigaben_rel = GenericRelation(Freigabe)
class Meta:
ordering = ['-upload_datum']
verbose_name = 'Dokument'
verbose_name_plural = 'Dokumente'
def __str__(self):
return self.dateiname or self.datei.name
def clean(self):
if self.datei:
import os
ext = os.path.splitext(self.datei.name)[1].lower()
if ext not in ALLOWED_EXTENSIONS:
raise ValidationError(
f'Dateityp "{ext}" nicht erlaubt. Erlaubt: {", ".join(ALLOWED_EXTENSIONS)}'
)
max_size = getattr(settings, 'MAX_UPLOAD_SIZE', 52428800)
if hasattr(self.datei, 'size') and self.datei.size > max_size:
raise ValidationError(
f'Datei zu groß. Maximum: {max_size // 1024 // 1024} MB'
)
def save(self, *args, **kwargs):
if self.datei and not self.dateiname:
import os
self.dateiname = os.path.basename(self.datei.name)
super().save(*args, **kwargs)
@receiver(pre_save, sender=Dokument)
def set_final_status(sender, instance, **kwargs):
if instance.pk:
try:
old = Dokument.objects.get(pk=instance.pk)
if not old.finale_abgabeversion and instance.finale_abgabeversion:
instance.status = 'final_abgegeben'
except Dokument.DoesNotExist:
pass

View File

@@ -0,0 +1,3 @@
from django.test import TestCase
# Create your tests here.

View File

@@ -0,0 +1,2 @@
from django.urls import path
urlpatterns = []

View File

@@ -0,0 +1,3 @@
from django.shortcuts import render
# Create your views here.

View File

@@ -0,0 +1,10 @@
from django.contrib import admin
from .models import Feedbackeintrag
@admin.register(Feedbackeintrag)
class FeedbackeintragAdmin(admin.ModelAdmin):
list_display = ['titel', 'kategorie', 'dringlichkeit', 'status', 'datum', 'erfasst_von']
list_filter = ['kategorie', 'dringlichkeit', 'status']
search_fields = ['titel', 'beschreibung']

View File

@@ -0,0 +1,5 @@
from django.apps import AppConfig
class FeedbackConfig(AppConfig):
name = 'vergabe_teilnahme.apps.feedback'

View File

@@ -0,0 +1,42 @@
# Generated by Django 6.0.5 on 2026-05-08 10:33
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
('ausschreibungen', '0001_initial'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.CreateModel(
name='Feedbackeintrag',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('titel', models.CharField(default='Ohne Titel', max_length=300)),
('beschreibung', models.TextField()),
('seite_kontext', models.CharField(blank=True, max_length=500)),
('kategorie', models.CharField(choices=[('fehler', 'Fehler'), ('verbesserung', 'Verbesserungsvorschlag'), ('hinweis', 'Hinweis')], default='hinweis', max_length=20)),
('dringlichkeit', models.CharField(choices=[('niedrig', 'Niedrig'), ('mittel', 'Mittel'), ('hoch', 'Hoch'), ('kritisch', 'Kritisch')], default='mittel', max_length=10)),
('prioritaet', models.PositiveSmallIntegerField(default=2)),
('status', models.CharField(choices=[('neu', 'Neu'), ('in_bearbeitung', 'In Bearbeitung'), ('umgesetzt', 'Umgesetzt'), ('abgelehnt', 'Abgelehnt')], default='neu', max_length=20)),
('datum', models.DateTimeField(auto_now_add=True)),
('bewertung', models.TextField(blank=True)),
('entscheidung', models.TextField(blank=True)),
('umsetzungshinweis', models.TextField(blank=True)),
('ausschreibung', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='feedback', to='ausschreibungen.ausschreibung')),
('erfasst_von', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)),
],
options={
'verbose_name': 'Feedbackeintrag',
'verbose_name_plural': 'Feedbackeinträge',
'ordering': ['-datum'],
},
),
]

View File

@@ -0,0 +1,50 @@
from django.db import models
from vergabe_teilnahme.apps.core.models import FlexibleModel
class Feedbackeintrag(FlexibleModel):
KATEGORIE_CHOICES = [
('fehler', 'Fehler'),
('verbesserung', 'Verbesserungsvorschlag'),
('hinweis', 'Hinweis'),
]
DRINGLICHKEIT_CHOICES = [
('niedrig', 'Niedrig'),
('mittel', 'Mittel'),
('hoch', 'Hoch'),
('kritisch', 'Kritisch'),
]
STATUS_CHOICES = [
('neu', 'Neu'),
('in_bearbeitung', 'In Bearbeitung'),
('umgesetzt', 'Umgesetzt'),
('abgelehnt', 'Abgelehnt'),
]
titel = models.CharField(max_length=300, default='Ohne Titel')
beschreibung = models.TextField()
seite_kontext = models.CharField(max_length=500, blank=True)
ausschreibung = models.ForeignKey(
'ausschreibungen.Ausschreibung', on_delete=models.SET_NULL,
null=True, blank=True, related_name='feedback'
)
kategorie = models.CharField(max_length=20, choices=KATEGORIE_CHOICES, default='hinweis')
dringlichkeit = models.CharField(max_length=10, choices=DRINGLICHKEIT_CHOICES, default='mittel')
prioritaet = models.PositiveSmallIntegerField(default=2)
status = models.CharField(max_length=20, choices=STATUS_CHOICES, default='neu')
erfasst_von = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL, null=True, blank=True
)
datum = models.DateTimeField(auto_now_add=True)
bewertung = models.TextField(blank=True)
entscheidung = models.TextField(blank=True)
umsetzungshinweis = models.TextField(blank=True)
class Meta:
verbose_name = 'Feedbackeintrag'
verbose_name_plural = 'Feedbackeinträge'
ordering = ['-datum']
def __str__(self):
return self.titel

View File

@@ -0,0 +1,3 @@
from django.test import TestCase
# Create your tests here.

View File

@@ -0,0 +1,10 @@
from django.urls import path
from . import views
app_name = 'feedback'
urlpatterns = [
path('modal/', views.modal, name='modal'),
path('', views.submit, name='submit'),
]

View File

@@ -0,0 +1,46 @@
from django.http import HttpResponse
from django.shortcuts import render
from django.views.decorators.http import require_GET, require_POST
from .models import Feedbackeintrag
@require_GET
def modal(request):
return render(request, 'partials/feedback_modal.html')
@require_POST
def submit(request):
beschreibung = request.POST.get('beschreibung', '').strip()
if not beschreibung:
return render(request, 'partials/feedback_modal.html',
{'error': 'Beschreibung ist erforderlich.'})
entry = Feedbackeintrag(
titel=request.POST.get('titel', 'Ohne Titel') or 'Ohne Titel',
beschreibung=beschreibung,
seite_kontext=request.POST.get('seite_kontext', ''),
kategorie=request.POST.get('kategorie', 'hinweis'),
dringlichkeit=request.POST.get('dringlichkeit', 'mittel'),
)
ausschreibung_pk = request.POST.get('ausschreibung')
if ausschreibung_pk:
try:
from vergabe_teilnahme.apps.ausschreibungen.models import Ausschreibung
entry.ausschreibung = Ausschreibung.objects.get(pk=ausschreibung_pk)
except (Ausschreibung.DoesNotExist, ValueError):
pass
if request.user.is_authenticated:
entry.erfasst_von = request.user
entry.save()
return HttpResponse(
'<div x-data="{open:true}" x-show="open" x-cloak '
'class="fixed inset-0 bg-black/30 z-50 flex items-center justify-center">'
'<div class="bg-white rounded-xl shadow-xl p-8 text-center max-w-sm mx-4">'
'<p class="text-2xl mb-2">✅</p>'
'<p class="font-medium text-slate-900">Danke für dein Feedback!</p>'
'<button @click="open=false" class="btn-secondary mt-4">Schließen</button>'
'</div></div>'
)

View File

View File

@@ -0,0 +1,15 @@
from django.contrib import admin
from .models import Anforderung, Los
@admin.register(Los)
class LosAdmin(admin.ModelAdmin):
list_display = ['losnummer', 'lostitel', 'ausschreibung', 'teilnahme', 'zustaendiger']
list_filter = ['teilnahme']
@admin.register(Anforderung)
class AnforderungAdmin(admin.ModelAdmin):
list_display = ['titel', 'verbindlichkeit', 'kategorie', 'erfuellungsstatus', 'ausschlusskriterium']
list_filter = ['verbindlichkeit', 'kategorie', 'erfuellungsstatus', 'ausschlusskriterium']

View File

@@ -0,0 +1,5 @@
from django.apps import AppConfig
class LoseConfig(AppConfig):
name = 'vergabe_teilnahme.apps.lose'

View File

@@ -0,0 +1,63 @@
# Generated by Django 6.0.5 on 2026-05-08 10:33
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
initial = True
dependencies = [
('ausschreibungen', '0001_initial'),
('bibliothek', '0001_initial'),
('dokumente', '0001_initial'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.CreateModel(
name='Los',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('losnummer', models.CharField(max_length=50)),
('lostitel', models.CharField(max_length=300)),
('beschreibung', models.TextField(blank=True)),
('abgrenzung', models.TextField(blank=True)),
('teilnahme', models.BooleanField(null=True)),
('status', models.CharField(blank=True, max_length=50)),
('ausschreibung', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='lose', to='ausschreibungen.ausschreibung')),
('zustaendiger', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)),
],
options={
'verbose_name': 'Los',
'verbose_name_plural': 'Lose',
'ordering': ['losnummer'],
},
),
migrations.CreateModel(
name='Anforderung',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('titel', models.CharField(max_length=400)),
('beschreibung', models.TextField(blank=True)),
('quelle_im_dokument', models.CharField(blank=True, max_length=300)),
('kategorie', models.CharField(blank=True, choices=[('fachlich', 'Fachlich'), ('rechtlich', 'Rechtlich'), ('kaufmaennisch', 'Kaufmännisch'), ('technisch', 'Technisch'), ('zertifizierung', 'Zertifizierung'), ('referenz', 'Referenz'), ('sonstiges', 'Sonstiges')], max_length=30)),
('verbindlichkeit', models.CharField(choices=[('muss', 'Muss-Kriterium'), ('soll', 'Soll-Kriterium'), ('kann', 'Kann-Kriterium'), ('info', 'Information')], default='muss', max_length=10)),
('ausschlusskriterium', models.BooleanField(default=False)),
('bewertungskriterium', models.BooleanField(default=False)),
('erfuellungsstatus', models.CharField(choices=[('offen', 'Offen'), ('erfuellt', 'Erfüllt'), ('teilweise', 'Teilweise erfüllt'), ('nicht_erfuellt', 'Nicht erfüllt'), ('entfaellt', 'Entfällt')], default='offen', max_length=20)),
('nachweis_erforderlich', models.BooleanField(default=False)),
('ausschreibung', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='anforderungen', to='ausschreibungen.ausschreibung')),
('dokumente', models.ManyToManyField(blank=True, to='dokumente.dokument')),
('nachweise', models.ManyToManyField(blank=True, to='bibliothek.nachweis')),
('zustaendiger', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)),
('los', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='anforderungen', to='lose.los')),
],
options={
'verbose_name': 'Anforderung',
'verbose_name_plural': 'Anforderungen',
},
),
]

View File

@@ -0,0 +1,81 @@
from django.db import models
from vergabe_teilnahme.apps.core.models import FlexibleModel
class Los(FlexibleModel):
ausschreibung = models.ForeignKey(
'ausschreibungen.Ausschreibung', on_delete=models.CASCADE, related_name='lose'
)
losnummer = models.CharField(max_length=50)
lostitel = models.CharField(max_length=300)
beschreibung = models.TextField(blank=True)
abgrenzung = models.TextField(blank=True)
zustaendiger = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL, null=True, blank=True
)
teilnahme = models.BooleanField(null=True)
status = models.CharField(max_length=50, blank=True)
class Meta:
ordering = ['losnummer']
verbose_name = 'Los'
verbose_name_plural = 'Lose'
def __str__(self):
return f'Los {self.losnummer}: {self.lostitel}'
class Anforderung(FlexibleModel):
VERBINDLICHKEIT_CHOICES = [
('muss', 'Muss-Kriterium'),
('soll', 'Soll-Kriterium'),
('kann', 'Kann-Kriterium'),
('info', 'Information'),
]
KATEGORIE_CHOICES = [
('fachlich', 'Fachlich'),
('rechtlich', 'Rechtlich'),
('kaufmaennisch', 'Kaufmännisch'),
('technisch', 'Technisch'),
('zertifizierung', 'Zertifizierung'),
('referenz', 'Referenz'),
('sonstiges', 'Sonstiges'),
]
ERFUELLUNG_CHOICES = [
('offen', 'Offen'),
('erfuellt', 'Erfüllt'),
('teilweise', 'Teilweise erfüllt'),
('nicht_erfuellt', 'Nicht erfüllt'),
('entfaellt', 'Entfällt'),
]
ausschreibung = models.ForeignKey(
'ausschreibungen.Ausschreibung', on_delete=models.CASCADE, related_name='anforderungen'
)
los = models.ForeignKey(
Los, on_delete=models.SET_NULL, null=True, blank=True, related_name='anforderungen'
)
titel = models.CharField(max_length=400)
beschreibung = models.TextField(blank=True)
quelle_im_dokument = models.CharField(max_length=300, blank=True)
kategorie = models.CharField(max_length=30, choices=KATEGORIE_CHOICES, blank=True)
verbindlichkeit = models.CharField(max_length=10, choices=VERBINDLICHKEIT_CHOICES, default='muss')
ausschlusskriterium = models.BooleanField(default=False)
bewertungskriterium = models.BooleanField(default=False)
zustaendiger = models.ForeignKey(
'accounts.Mitarbeiter', on_delete=models.SET_NULL, null=True, blank=True
)
erfuellungsstatus = models.CharField(
max_length=20, choices=ERFUELLUNG_CHOICES, default='offen'
)
nachweis_erforderlich = models.BooleanField(default=False)
dokumente = models.ManyToManyField('dokumente.Dokument', blank=True)
nachweise = models.ManyToManyField('bibliothek.Nachweis', blank=True)
class Meta:
verbose_name = 'Anforderung'
verbose_name_plural = 'Anforderungen'
def __str__(self):
return self.titel

View File

@@ -0,0 +1,3 @@
from django.test import TestCase
# Create your tests here.

View File

@@ -0,0 +1,2 @@
from django.urls import path
urlpatterns = []

Some files were not shown because too many files have changed in this diff Show More