- Receive Evotor webhooks: POST /user/create, /user/verify, /user/token
- Create users in pending status; match to existing users by email/phone
- Send invite link via Celery notification task; user sets password at /invite
- Abstract EmailProvider/SMSProvider with ConsoleEmailProvider default
- Role-based access control: role enum on users + roles/permissions tables
- Admin panel: /admin/users (list, filter, search, paginate), user detail card
with activate/suspend/reset-password/send-invite/edit/delete actions
- Admin roles management: /admin/roles with per-role permission assignment
- Extend user profile card: role, status, Evotor ID, email confirmation badge
- Auth routes: register, login, logout, confirm-email, forgot/reset password
- Alembic migrations 0002 (full schema + new fields) and 0003 (RBAC + seeds)
- Port Pico CSS + Bootstrap Icons UI from Node.js commit (854c912)
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
44 lines
2.0 KiB
HTML
44 lines
2.0 KiB
HTML
{% extends "base.html" %}
|
|
{% block title %}Редактировать профиль — ЭВОСИНК{% endblock %}
|
|
|
|
{% block content %}
|
|
<div class="row justify-center">
|
|
<div class="col-sm-10 col-md-7 col-lg-6">
|
|
<article class="card mt-4">
|
|
<header>
|
|
<h1><i class="bi bi-pencil me-2"></i>Редактировать профиль</h1>
|
|
</header>
|
|
<div class="card-body">
|
|
<form method="post" action="/profile/edit">
|
|
<div class="row gap-2 mb-2">
|
|
<div class="col">
|
|
<label for="first_name">Имя
|
|
<input type="text" id="first_name" name="first_name"
|
|
value="{{ form.first_name if form else user.first_name }}" required>
|
|
</label>
|
|
</div>
|
|
<div class="col">
|
|
<label for="last_name">Фамилия
|
|
<input type="text" id="last_name" name="last_name"
|
|
value="{{ form.last_name if form else user.last_name }}" required>
|
|
</label>
|
|
</div>
|
|
</div>
|
|
<label>Email
|
|
<input type="email" value="{{ user.email }}" disabled>
|
|
</label>
|
|
<label for="phone">Телефон
|
|
<input type="tel" id="phone" name="phone"
|
|
value="{{ form.phone if form else user.phone }}" required>
|
|
</label>
|
|
<div class="d-flex gap-2">
|
|
<button type="submit">Сохранить</button>
|
|
<a href="/profile" role="button" class="outline secondary">Отмена</a>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
</article>
|
|
</div>
|
|
</div>
|
|
{% endblock %}
|