mirror of
https://github.com/wasrusgen/zov-tech.git
synced 2026-06-03 17:24:49 +00:00
Users.role теперь хранит CSV-список ролей: 'manager,measurer'.
Парсим, добавляем, отзываем — все через sheets.parse_roles / grant_role /
revoke_role / list_users_with_role. Старые однострочные значения работают
как раньше (legacy compat).
Backend:
- /api/me возвращает roles[] (массив), role (главная для legacy-UI),
plus capabilities {measurer, assembler} для staff
- /api/grant_role (admin-only) — добавить/отозвать роль
- /api/staff_list (manager-only) — список сотрудников по роли
(будет использоваться в dropdown «выбрать замерщика»)
- При role=staff отдаём отдельный кабинет; если у юзера нет measurer/
assembler — возвращаем error=no_staff_role
Bot:
- /start — 3-я reply-кнопка [🔧 Я сотрудник]. При тапе MiniApp получает
?role=staff и решает кабинет по capabilities.
- /whoami — сотрудник присылает свой Telegram ID, пересылает куратору
чтобы тот выдал роль через /api/grant_role.
MiniApp:
- renderStaff() — заглушка кабинета сотрудника с шапкой (имя, аватар,
список ролей) и пустым inbox («Пока пусто»). Если есть measurer —
быстрая кнопка «Сделать новый замер».
- При error=no_staff_role — экран с инструкцией как получить роль.
- CSS .staff-head / .staff-no-role.
Cache bust v=20260513f.
99 lines
3.7 KiB
Python
99 lines
3.7 KiB
Python
import time
|
||
|
||
from aiogram import F, Router
|
||
from aiogram.filters import Command, CommandStart
|
||
from aiogram.types import (
|
||
KeyboardButton,
|
||
Message,
|
||
ReplyKeyboardMarkup,
|
||
ReplyKeyboardRemove,
|
||
WebAppInfo,
|
||
)
|
||
|
||
from config import Config
|
||
|
||
router = Router(name="start")
|
||
|
||
|
||
# ============================================================
|
||
# URL helpers
|
||
# ============================================================
|
||
|
||
def _bust_cache(url: str) -> str:
|
||
"""Append unique timestamp so Telegram WebView не кеширует между сессиями."""
|
||
sep = "&" if "?" in url else "?"
|
||
return f"{url}{sep}t={int(time.time())}"
|
||
|
||
|
||
def _with_query(url: str, **params: str) -> str:
|
||
sep = "&" if "?" in url else "?"
|
||
pairs = "&".join(f"{k}={v}" for k, v in params.items() if v)
|
||
return f"{url}{sep}{pairs}" if pairs else url
|
||
|
||
|
||
def _wapp(miniapp_url: str, role: str) -> WebAppInfo:
|
||
return WebAppInfo(url=_bust_cache(_with_query(miniapp_url, role=role)))
|
||
|
||
|
||
# ============================================================
|
||
# Reply keyboard — выбор роли. Три кнопки, все WebApp.
|
||
# ============================================================
|
||
|
||
def role_choice_kb(miniapp_url: str) -> ReplyKeyboardMarkup:
|
||
return ReplyKeyboardMarkup(
|
||
keyboard=[
|
||
[
|
||
KeyboardButton(text="👤 Я менеджер", web_app=_wapp(miniapp_url, "manager")),
|
||
KeyboardButton(text="🏠 Я клиент", web_app=_wapp(miniapp_url, "client")),
|
||
],
|
||
[
|
||
KeyboardButton(text="🔧 Я сотрудник", web_app=_wapp(miniapp_url, "staff")),
|
||
],
|
||
],
|
||
resize_keyboard=True,
|
||
is_persistent=True,
|
||
input_field_placeholder="Выберите кто вы…",
|
||
)
|
||
|
||
|
||
# ============================================================
|
||
# Commands
|
||
# ============================================================
|
||
|
||
@router.message(CommandStart())
|
||
async def cmd_start(message: Message, config: Config) -> None:
|
||
await message.answer(
|
||
"👋 Здравствуйте, я бот-помощник от Руслана ВАСИЛЬЕВА.\n\n"
|
||
"Выберите, кто вы — кабинет откроется одним тапом.\n\n"
|
||
"<i>«Сотрудник» — для замерщиков и сборщиков ЗОВ. Если вы менеджер или клиент — выбирайте свою роль.</i>",
|
||
reply_markup=role_choice_kb(config.miniapp_url),
|
||
)
|
||
|
||
|
||
@router.message(Command("menu"))
|
||
async def cmd_menu(message: Message, config: Config) -> None:
|
||
await message.answer("Выберите роль:", reply_markup=role_choice_kb(config.miniapp_url))
|
||
|
||
|
||
@router.message(Command("hide"))
|
||
async def cmd_hide(message: Message) -> None:
|
||
await message.answer("Клавиатура скрыта. Вернуть — /menu", reply_markup=ReplyKeyboardRemove())
|
||
|
||
|
||
# ============================================================
|
||
# /whoami — сотрудник присылает свой ID куратору, чтобы тот выдал роль
|
||
# ============================================================
|
||
|
||
@router.message(Command("whoami"))
|
||
async def cmd_whoami(message: Message) -> None:
|
||
user = message.from_user
|
||
if not user:
|
||
return
|
||
await message.answer(
|
||
f"<b>Ваш Telegram ID:</b> <code>{user.id}</code>\n"
|
||
f"Username: @{user.username or '—'}\n"
|
||
f"Имя: {user.first_name or ''} {user.last_name or ''}".strip()
|
||
+ "\n\n"
|
||
"<i>Перешлите это сообщение куратору @wasrusgen чтобы вам выдали роль замерщика/сборщика.</i>"
|
||
)
|