4b0d56ff0e
Этап 1 миграции TestingWebApp на целевой стек (Python/Flask/Jinja),
БД остаётся clinic_tests.
E1.0 — База Flask-приложения: SQLAlchemy/psycopg2 пул, Flask sessions,
фабрика create_app, blueprint main с / и /health, base.html в стиле
кабинета HR (Tailwind CDN + Manrope + Material Symbols), 404/500.
E1.1 — Auth + /api/me: Flask sessions (signed cookie) вместо JWT,
bcrypt + Werkzeug, опц. HR_AUTH=1 с UPSERT в clinic_tests.users по
staff_id. UI /login, JSON /api/auth/{login,logout,me}, декораторы
@login_required / @require_role.
E1.2 — Тесты: список + редактор. 10 эндпоинтов, сервисы test_draft,
test_access, test_chain, ai_editor, llm_client, draft_validator,
editor_content. UI /tests (каталог + создание) и /tests/<id>/edit
(редактор с AI). Полный мобильный UX (аккордеоны/drag-n-drop) — в E1.7.
E1.3 — Импорт документов: pypdf + python-docx, эндпоинт
POST /api/tests/import/document, кнопка «Импорт документа» в
AI-панели редактора, лимит 16 МБ.
E1.8 — AI v2: страница /settings (статус ENV-ключа + ping),
ai/generate-by-title (без сетки), ai/check (рецензия), ai/improve
(массовое было→стало с чекбоксами). Унифицированный ответ AI-ошибок:
{ error, code, settingsUrl }.
Docker:
- docker-compose.dev.yml: добавлены DATABASE_URL, HR_AUTH/HR_DATABASE_URL,
DEEPSEEK_API_KEY/OPENAI_API_KEY/LLM_BASE_URL/LLM_MODEL и сеть postgres
для testing-flask.
Документация:
- docs/migration-final.md — двух-этапный план (Этап 1: унификация
стека внутри TestingWebApp; Этап 2: слияние с tgFlaskForm).
- docs/migration-final-inventory.md — карта 22 эндпоинтов Express.
Made-with: Cursor
59 lines
2.5 KiB
HTML
59 lines
2.5 KiB
HTML
{% extends "base.html" %}
|
|
{% block title %}Вход — Тестирование{% endblock %}
|
|
|
|
{% block content %}
|
|
<section class="mx-auto max-w-md mt-8">
|
|
<div class="rounded-2xl bg-white shadow-sm border border-ink-300/60 p-6">
|
|
<div class="flex items-center gap-2">
|
|
<span class="material-symbols-outlined text-brand-600">login</span>
|
|
<h1 class="text-xl font-semibold">Вход в систему</h1>
|
|
</div>
|
|
<p class="mt-1 text-sm text-ink-500">
|
|
Используйте логин и пароль.
|
|
{% if hr_auth_enabled %}
|
|
Учётка кадровой системы (HR).
|
|
{% endif %}
|
|
</p>
|
|
|
|
{% with messages = get_flashed_messages(with_categories=true) %}
|
|
{% if messages %}
|
|
<div class="mt-4 space-y-2">
|
|
{% for category, msg in messages %}
|
|
<div class="px-3 py-2 rounded-lg text-sm
|
|
{% if category == 'error' %}bg-red-50 text-red-700 border border-red-200
|
|
{% else %}bg-brand-50 text-brand-700 border border-brand-100{% endif %}">
|
|
{{ msg }}
|
|
</div>
|
|
{% endfor %}
|
|
</div>
|
|
{% endif %}
|
|
{% endwith %}
|
|
|
|
<form method="post" action="{{ url_for('auth.login_submit') }}" class="mt-5 space-y-4" novalidate>
|
|
<input type="hidden" name="next" value="{{ next or '/' }}">
|
|
|
|
<label class="block">
|
|
<span class="text-sm font-medium text-ink-700">Логин</span>
|
|
<input type="text" name="login" value="{{ login or '' }}" required autofocus autocomplete="username"
|
|
class="mt-1 w-full rounded-lg border border-ink-300 bg-white px-3 py-2 text-ink-900
|
|
focus:border-brand-500 focus:ring-2 focus:ring-brand-500/20" />
|
|
</label>
|
|
|
|
<label class="block">
|
|
<span class="text-sm font-medium text-ink-700">Пароль</span>
|
|
<input type="password" name="password" required autocomplete="current-password"
|
|
class="mt-1 w-full rounded-lg border border-ink-300 bg-white px-3 py-2 text-ink-900
|
|
focus:border-brand-500 focus:ring-2 focus:ring-brand-500/20" />
|
|
</label>
|
|
|
|
<button type="submit"
|
|
class="w-full inline-flex items-center justify-center gap-2 rounded-lg
|
|
bg-brand-600 hover:bg-brand-700 text-white font-medium px-4 py-2 transition">
|
|
<span class="material-symbols-outlined text-base">login</span>
|
|
Войти
|
|
</button>
|
|
</form>
|
|
</div>
|
|
</section>
|
|
{% endblock %}
|