Compare commits
119 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 7cecb722d8 | |||
| 11b145a967 | |||
| 105b3c59d7 | |||
| 3bea327043 | |||
| c6aea78ff3 | |||
| 01c49f2df0 | |||
| 9deccd3a9d | |||
| 81d4ec2c97 | |||
| c0a5482e1a | |||
| 5a18cacb2e | |||
| 121272fdfe | |||
| ccf11457ca | |||
| e492d4fc2d | |||
| 11f6b1bcc9 | |||
| 06d40fdbc8 | |||
| 043ed9ce45 | |||
| 320aba2877 | |||
| e3fdac15b5 | |||
| 105a051c2f | |||
| de9f56c97d | |||
| 007806a5d8 | |||
| c9627e51a2 | |||
| 2a3285996e | |||
| 025c7c2f9a | |||
| e6e6d17b72 | |||
| 563e118f23 | |||
| e2303490e9 | |||
| 9c1c6c2483 | |||
| c0c8f852d2 | |||
| ac6e2455a1 | |||
| 9374ff16ed | |||
| 17b92b25f4 | |||
| d2edbf16cc | |||
| b16627c2b6 | |||
| 4f7afb3bc9 | |||
| 5baf63e9ad | |||
| a0d9d1bc44 | |||
| f46f2bb5d3 | |||
| 46527fe761 | |||
| d0a25895ab | |||
| 05faa9e32d | |||
| 0dbd4064ac | |||
| 0f03da0a60 | |||
| 6d90ba8274 | |||
| 35894bf89e | |||
| 6394b1fe8c | |||
| d170c83b9e | |||
| 4a2d1d2d38 | |||
| 706f20e403 | |||
| 4d3362d93f | |||
| b03929174a | |||
| 7e2747ec73 | |||
| ae6be912e3 | |||
| 116bed16a8 | |||
| 063de7ee3e | |||
| 5c4ec562d0 | |||
| dbd481566c | |||
| 3f4571d3a7 | |||
| 8c1e7991cd | |||
| c1fdba510b | |||
| 435399dcf2 | |||
| ddaa0f4279 | |||
| b205967f1a | |||
| 7457315d6f | |||
| 59f9904d66 | |||
| 3b91a009ea | |||
| a6ae5aac31 | |||
| dc26b4d7e4 | |||
| bc6136d91e | |||
| 2e95841ca8 | |||
| a7c8127f90 | |||
| cad4e5c30e | |||
| 77647e4bb8 | |||
| 17c631aef2 | |||
| 89b5196676 | |||
| ab1d2f1683 | |||
| 1bcd88db32 | |||
| 63e613c061 | |||
| dbf59c544a | |||
| 14b9bf15f2 | |||
| 5dee2d87f5 | |||
| b71488097e | |||
| 6e92419cff | |||
| fdb3445bec | |||
| c1f5d96e25 | |||
| c874f7b797 | |||
| aefed5abd4 | |||
| 25c22b2ff5 | |||
| cb40c2438d | |||
| 2a76ec0fb8 | |||
| 57c8714889 | |||
| 8220f2060f | |||
| 41f2ea6e90 | |||
| 5082dd4fcf | |||
| cfbda4ca05 | |||
| 0218890a7a | |||
| a1ec688ec8 | |||
| 2529df4157 | |||
| a8f2b10956 | |||
| 3228e77c7f | |||
| 621ef553e7 | |||
| 5f3516e703 | |||
| 2eb7d86e48 | |||
| 3e291b0ed5 | |||
| a5ba4111cf | |||
| f45985041b | |||
| 9c91057798 | |||
| 675ac1226e | |||
| b80002aa36 | |||
| bb8cbb7a40 | |||
| 93e7c1ac66 | |||
| 4d6651827b | |||
| 9e7a202f42 | |||
| 1c4cfb71c0 | |||
| ecc2236937 | |||
| 3002db6534 | |||
| 176f1105ab | |||
| b6af5f047c | |||
| 66e7f5eea7 |
@@ -6,5 +6,19 @@ TELEGRAM_BOT_TOKEN=YOUR_BOT_TOKEN_HERE
|
|||||||
# Найти его можно в информации о боте у @BotFather.
|
# Найти его можно в информации о боте у @BotFather.
|
||||||
TELEGRAM_BOT_USERNAME=YOUR_BOT_USERNAME_HERE
|
TELEGRAM_BOT_USERNAME=YOUR_BOT_USERNAME_HERE
|
||||||
|
|
||||||
|
# HTTPS URL Mini App dashboard, например: https://your-domain.example/miniapp
|
||||||
|
# Используется ботом для кнопки меню Telegram и кнопки /start.
|
||||||
|
TELEGRAM_MINI_APP_URL=
|
||||||
|
|
||||||
# Пароль для базы данных PostgreSQL
|
# Пароль для базы данных PostgreSQL
|
||||||
POSTGRES_PASSWORD=StrongPasswordForDatabase
|
POSTGRES_PASSWORD=StrongPasswordForDatabase
|
||||||
|
|
||||||
|
# Локальный порт веб-интерфейса GM-Relay
|
||||||
|
GMRELAY_WEB_PORT=8080
|
||||||
|
|
||||||
|
# === Backup ===
|
||||||
|
# Сколько дней хранить дампы PostgreSQL (default: 7)
|
||||||
|
BACKUP_RETENTION_DAYS=7
|
||||||
|
|
||||||
|
# Имя Docker volume для резервных копий БД
|
||||||
|
BACKUP_VOLUME_NAME=game_pgbackups
|
||||||
|
|||||||
+54
-24
@@ -6,7 +6,7 @@ on:
|
|||||||
- main
|
- main
|
||||||
|
|
||||||
env:
|
env:
|
||||||
VERSION: 1.0.0
|
VERSION: 2.0.0
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
# ЧАСТЬ 1: Собираем образы и кладем в Gitea (чтобы делиться с ребятами)
|
# ЧАСТЬ 1: Собираем образы и кладем в Gitea (чтобы делиться с ребятами)
|
||||||
@@ -23,33 +23,62 @@ jobs:
|
|||||||
username: toutsu
|
username: toutsu
|
||||||
password: ${{ secrets.GIT_TOKEN }}
|
password: ${{ secrets.GIT_TOKEN }}
|
||||||
|
|
||||||
- name: Build and push Bot
|
- name: Build Bot image
|
||||||
uses: docker/build-push-action@v5
|
run: |
|
||||||
with:
|
docker build \
|
||||||
context: .
|
--label "org.opencontainers.image.source=https://git.codeanddice.ru/${{ gitea.repository }}" \
|
||||||
file: src/GmRelay.Bot/Dockerfile
|
-f src/GmRelay.Bot/Dockerfile \
|
||||||
push: true
|
-t git.codeanddice.ru/toutsu/gmrelay-bot:latest \
|
||||||
tags: |
|
-t git.codeanddice.ru/toutsu/gmrelay-bot:${{ env.VERSION }} \
|
||||||
git.codeanddice.ru/toutsu/gmrelay-bot:latest
|
.
|
||||||
git.codeanddice.ru/toutsu/gmrelay-bot:${{ env.VERSION }}
|
|
||||||
labels: |
|
|
||||||
org.opencontainers.image.source=https://git.codeanddice.ru/${{ gitea.repository }}
|
|
||||||
|
|
||||||
- name: Build and push Web
|
- name: Push Bot image
|
||||||
uses: docker/build-push-action@v5
|
run: |
|
||||||
with:
|
docker push git.codeanddice.ru/toutsu/gmrelay-bot:latest
|
||||||
context: .
|
docker push git.codeanddice.ru/toutsu/gmrelay-bot:${{ env.VERSION }}
|
||||||
file: src/GmRelay.Web/Dockerfile
|
|
||||||
push: true
|
- name: Build Web image
|
||||||
tags: |
|
run: |
|
||||||
git.codeanddice.ru/toutsu/gmrelay-web:latest
|
docker build \
|
||||||
|
--label "org.opencontainers.image.source=https://git.codeanddice.ru/${{ gitea.repository }}" \
|
||||||
|
-f src/GmRelay.Web/Dockerfile \
|
||||||
|
-t git.codeanddice.ru/toutsu/gmrelay-web:latest \
|
||||||
|
-t git.codeanddice.ru/toutsu/gmrelay-web:${{ env.VERSION }} \
|
||||||
|
.
|
||||||
|
|
||||||
|
- name: Push Web image
|
||||||
|
run: |
|
||||||
|
docker push git.codeanddice.ru/toutsu/gmrelay-web:latest
|
||||||
|
docker push git.codeanddice.ru/toutsu/gmrelay-web:${{ env.VERSION }}
|
||||||
|
|
||||||
|
# ЧАСТЬ 1.5: Сканируем собранные образы на уязвимости
|
||||||
|
scan-images:
|
||||||
|
needs: build-and-push
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Install Trivy
|
||||||
|
run: |
|
||||||
|
curl -sfL https://raw.githubusercontent.com/aquasecurity/trivy/main/contrib/install.sh | sh -s -- -b /usr/local/bin
|
||||||
|
|
||||||
|
- name: Scan Bot image
|
||||||
|
run: |
|
||||||
|
trivy image \
|
||||||
|
--severity HIGH,CRITICAL \
|
||||||
|
--exit-code 1 \
|
||||||
|
--format table \
|
||||||
|
git.codeanddice.ru/toutsu/gmrelay-bot:${{ env.VERSION }}
|
||||||
|
|
||||||
|
- name: Scan Web image
|
||||||
|
run: |
|
||||||
|
trivy image \
|
||||||
|
--severity HIGH,CRITICAL \
|
||||||
|
--exit-code 1 \
|
||||||
|
--format table \
|
||||||
git.codeanddice.ru/toutsu/gmrelay-web:${{ env.VERSION }}
|
git.codeanddice.ru/toutsu/gmrelay-web:${{ env.VERSION }}
|
||||||
labels: |
|
|
||||||
org.opencontainers.image.source=https://git.codeanddice.ru/${{ gitea.repository }}
|
|
||||||
|
|
||||||
# ЧАСТЬ 2: Запускаем эти образы на самом сервере
|
# ЧАСТЬ 2: Запускаем эти образы на самом сервере
|
||||||
deploy:
|
deploy:
|
||||||
needs: build-and-push
|
needs: scan-images
|
||||||
runs-on: ubuntu-latest # Тот же локальный раннер
|
runs-on: ubuntu-latest # Тот же локальный раннер
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
@@ -60,6 +89,7 @@ jobs:
|
|||||||
echo "TELEGRAM_BOT_TOKEN=${{ secrets.TELEGRAM_BOT_TOKEN }}" > .env
|
echo "TELEGRAM_BOT_TOKEN=${{ secrets.TELEGRAM_BOT_TOKEN }}" > .env
|
||||||
echo "POSTGRES_PASSWORD=${{ secrets.POSTGRES_PASSWORD }}" >> .env
|
echo "POSTGRES_PASSWORD=${{ secrets.POSTGRES_PASSWORD }}" >> .env
|
||||||
echo "TELEGRAM_BOT_USERNAME=${{ secrets.TELEGRAM_BOT_USERNAME }}" >> .env
|
echo "TELEGRAM_BOT_USERNAME=${{ secrets.TELEGRAM_BOT_USERNAME }}" >> .env
|
||||||
|
echo "TELEGRAM_MINI_APP_URL=${{ secrets.TELEGRAM_MINI_APP_URL }}" >> .env
|
||||||
|
|
||||||
- name: Deploy Containers
|
- name: Deploy Containers
|
||||||
run: |
|
run: |
|
||||||
@@ -70,4 +100,4 @@ jobs:
|
|||||||
docker compose pull bot web
|
docker compose pull bot web
|
||||||
|
|
||||||
# Запускаем! Флаг -d оставит их работать в фоне.
|
# Запускаем! Флаг -d оставит их работать в фоне.
|
||||||
docker compose up -d
|
docker compose up -d
|
||||||
|
|||||||
@@ -0,0 +1,78 @@
|
|||||||
|
name: PR Checks
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test-and-build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup .NET
|
||||||
|
uses: actions/setup-dotnet@v4
|
||||||
|
with:
|
||||||
|
dotnet-version: '10.0.x'
|
||||||
|
|
||||||
|
- name: Restore dependencies
|
||||||
|
run: dotnet restore
|
||||||
|
|
||||||
|
- name: Verify Trivy dependency scan inputs
|
||||||
|
run: |
|
||||||
|
lock_count="$(find . -name packages.lock.json -not -path "*/bin/*" -not -path "*/obj/*" | tee trivy-targets.txt | wc -l)"
|
||||||
|
echo "Trivy NuGet lock files: ${lock_count}"
|
||||||
|
if [ "${lock_count}" -eq 0 ]; then
|
||||||
|
echo "::error::No packages.lock.json files found. Trivy would scan 0 NuGet dependency files."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# ── Linting ──
|
||||||
|
|
||||||
|
- name: Lint C# code style
|
||||||
|
run: dotnet format --verify-no-changes --verbosity diagnostic
|
||||||
|
|
||||||
|
# ── Security ──
|
||||||
|
|
||||||
|
- name: Check NuGet packages for vulnerabilities
|
||||||
|
run: |
|
||||||
|
dotnet list package --vulnerable --include-transitive 2>&1 | tee nuget-audit.txt
|
||||||
|
if grep -qi "has the following vulnerable packages" nuget-audit.txt; then
|
||||||
|
echo "::error::Vulnerable NuGet packages found!"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "No vulnerable packages detected."
|
||||||
|
|
||||||
|
- name: Install Trivy
|
||||||
|
run: |
|
||||||
|
curl -sfL https://raw.githubusercontent.com/aquasecurity/trivy/main/contrib/install.sh | sh -s -- -b /usr/local/bin
|
||||||
|
trivy --version
|
||||||
|
|
||||||
|
- name: Trivy filesystem security scan
|
||||||
|
run: |
|
||||||
|
set +e
|
||||||
|
trivy fs --scanners vuln,misconfig,secret --exit-code 1 --severity HIGH,CRITICAL . 2>&1 | tee trivy-scan.log
|
||||||
|
trivy_exit="${PIPESTATUS[0]}"
|
||||||
|
if ! grep -Eq "Number of language-specific files[[:space:]]+num=[1-9][0-9]*" trivy-scan.log; then
|
||||||
|
echo "::error::Trivy did not detect any language-specific dependency files."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
exit "${trivy_exit}"
|
||||||
|
|
||||||
|
# ── Build (includes SAST via SecurityCodeScan Roslyn analyzer) ──
|
||||||
|
|
||||||
|
- name: Build Shared
|
||||||
|
run: dotnet build src/GmRelay.Shared/GmRelay.Shared.csproj --no-restore
|
||||||
|
|
||||||
|
- name: Build Bot (compile check, includes SAST)
|
||||||
|
run: dotnet build src/GmRelay.Bot/GmRelay.Bot.csproj --no-restore
|
||||||
|
|
||||||
|
- name: Build Web (compile check, includes SAST)
|
||||||
|
run: dotnet build src/GmRelay.Web/GmRelay.Web.csproj --no-restore
|
||||||
|
|
||||||
|
# ── Tests ──
|
||||||
|
|
||||||
|
- name: Run tests
|
||||||
|
run: dotnet test tests/GmRelay.Bot.Tests/GmRelay.Bot.Tests.csproj --verbosity normal
|
||||||
BIN
Binary file not shown.
@@ -1,10 +1,15 @@
|
|||||||
<Project>
|
<Project>
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<Version>1.0.0</Version>
|
<Version>2.0.0</Version>
|
||||||
<TargetFramework>net10.0</TargetFramework>
|
<TargetFramework>net10.0</TargetFramework>
|
||||||
<LangVersion>preview</LangVersion>
|
<LangVersion>preview</LangVersion>
|
||||||
<Nullable>enable</Nullable>
|
<Nullable>enable</Nullable>
|
||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
|
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
|
||||||
|
<RestorePackagesWithLockFile>true</RestorePackagesWithLockFile>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="SecurityCodeScan.VS2019" Version="5.6.7" PrivateAssets="all" />
|
||||||
|
</ItemGroup>
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2026 Toutsu
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||||
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
||||||
|
IN THE SOFTWARE.
|
||||||
@@ -4,133 +4,170 @@
|
|||||||
|
|
||||||
Проект разработан с упором на производительность, архитектуру Vertical Slice, Native AOT (для бота) и удобство развертывания с использованием .NET Aspire.
|
Проект разработан с упором на производительность, архитектуру Vertical Slice, Native AOT (для бота) и удобство развертывания с использованием .NET Aspire.
|
||||||
|
|
||||||
|
**Текущая версия:** `v1.15.0`.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## ✨ Ключевые возможности
|
## ✨ Key Features
|
||||||
|
|
||||||
### 🤖 Telegram Бот
|
### 🤖 Telegram Bot
|
||||||
- **📅 Создание расписаний (Batch Sessions)**: Создавайте сразу несколько игр одним сообщением (на неделю или месяц вперед).
|
- **📅 Создание расписаний (Batch Sessions)**: Создавайте сразу несколько игр одним сообщением изменения (на недельный месяц в перед).
|
||||||
- **✋ Интерактивная запись**: Игроки записываются на конкретные даты нажатием одной кнопки.
|
- **🖼 Обложки расписаний**: И batch-посту можно прикрепить фото к `/newsession` или указать строку `Картинка: https://...`; бот отправит обложку перед сообщением записи.
|
||||||
- **📁 Поддержка Форумов (Telegram Topics)**: Бот автоматически создает тему во вложенных чатах Telegram под каждую новую пачку игр.
|
- **⚡ Быстрые повторы расписания**: Для регулярной кампании можно указать одну дату, количество игр и интервал, а бот сам развернёт повторяющийся batch.
|
||||||
- **❌ Управление сессиями**: Мастер может отменять отдельные игры прямо в общем сообщении расписания.
|
- **✋ Интерактивная запись и выход**: Игроки записываются на конкретные даты и самостоятельно снимают запись нажатием одной кнопки.
|
||||||
- **🗓 Экспорт в Календарь**: Генерация файла `.ics` для добавления всех игр в Google, Apple или Яндекс Календарь одной командой.
|
- **👥 Лимит мест и лист ожидания**: ГМ задаёт максимальный состав, бот не переполняет сессию, автоматически ведёт очередь ожидания и освобождённое место отдаёт первому ожидающему.
|
||||||
- **🚀 Native AOT**: Скомпилирован в нативный бинарный файл. Мгновенный запуск и минимальное потребление памяти. Идеально для **Raspberry Pi**.
|
- **📁 Поддержка Форумов (Telegram Topics)**: Если `/newsession` запущен в теме форума Telegram, расписание и групповые уведомления остаются в этой теме; при запуске из корня форума бот создает отдельную тему и сообщает о необходимости прав admin/Manage Topics, если их не хватает.
|
||||||
|
- **❌ Управление сессиями**: Owner и назначенные co-GM могут создавать, отменять, удалять и переносить игры из Telegram через `/listsessions`; публичный пост записи показывает только кнопки игроков.
|
||||||
|
- **🔄 Голосование за перенос**: Быстрый поиск свободного места с через свободное недель и кнопками новых времени и дедлайном.
|
||||||
|
- **🔔 Уведомления**: Игрок получают за 24 часа, напоминание за 1 час, ссылку перед игрой, отмены и переносы; групповые уведомления при этом остаются.
|
||||||
|
- **🕐 Режим уведомлений batch**: Для каждой пачки можно выбрать `В группе и в личку` или `Только в группе`.
|
||||||
|
- **⬆️ Управление очередью**: Веб-интерфейс показывает заполненность, лист ожидания и позволяет ГМу поднять первого игрока из очереди.
|
||||||
|
- **🔄 Автоматическая синхронизация**: Любые изменения в веб-интерфейсе мгновенно обновляют сообщения с расписанием в Telegram-чатах игроков.
|
||||||
|
|
||||||
### 🌐 Web Dashboard (Blazor Server)
|
### 🌐 Web Dashboard (Blazor Server)
|
||||||
- **🔐 Авторизация через Telegram**: Безопасный вход с использованием Telegram Login Widget (HMAC-SHA256 валидация).
|
- **🔐 Авторизация через Telegram**: Telegram Login Widget с HMAC-SHA256 валидацией.
|
||||||
- **📝 Удобное редактирование**: Веб-интерфейс для детального редактирования сессий, изменения дат, названий и статусов.
|
- **📱 Telegram Mini App Dashboard**: Мобильная панель открывается из Telegram, проверяет `initData` на сервере, учитывает safe-area телефона и верхнюю панель Telegram.
|
||||||
- **🔄 Автоматическая синхронизация**: Любые изменения в веб-интерфейсе мгновенно обновляют сообщения с расписанием в Telegram-чатах игроков.
|
- **✏️ Редактирование**: Детальное изменение дат, названий и статусов сессий.
|
||||||
- **🕒 Управление временем**: UI адаптирован под московское время (UTC+3), в то время как база данных работает в UTC.
|
- **🤝 Co-GM и делегирование**: Owner назначает помощников по Telegram ID; co-GM управляет расписанием, но **не может назначать других co-GM**.
|
||||||
|
- **📋 Шаблоны кампаний**: Вкладка `Шаблоны` отдельно от страницы группы: сохранение типовых параметров и запуск нового batch из шаблона.
|
||||||
|
- **📦 Bulk-операции для Batch Sessions**:
|
||||||
|
- обновить общий `title`/`link` у всей пачки;
|
||||||
|
- перенести пачку на фиксированный шаг в днях;
|
||||||
|
- клонировать batch на следующую неделю или месяц.
|
||||||
|
- **⬆️ Управление очередью**: Заполненность, лист ожидания и ручное повышение игрока из очереди.
|
||||||
|
- **📜 История изменений сессий**: Страница `/session/{id}/history` показывает аудит-лог всех значимых изменений (время, ссылка, название, участники, статус) с указанием акторов и дат.
|
||||||
|
- **📊 Статистика посещаемости**: Страница `/group/{id}/stats` показывает долю присутствия, количество пропусков и среднюю явку по каждому игроку группы.
|
||||||
|
- **🔄 Автосинхронизация**: Изменения в вебе мгновенно перерисовывают Telegram-сообщения расписания.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 🛠 Технологический стек
|
## 🛠 Технологический стек
|
||||||
|
|
||||||
- **Язык**: C# 14 (.NET 10)
|
| Компонент | Технология |
|
||||||
- **Архитектура**: Vertical Slice Architecture, общая библиотека (`GmRelay.Shared`) для доменной логики.
|
|---|---|
|
||||||
- **Бот**: Telegram.Bot, Native AOT.
|
| Язык | C# 14 (.NET 10) |
|
||||||
- **Веб-интерфейс**: Blazor Server.
|
| Архитектура | Vertical Slice + общая библиотека `GmRelay.Shared` |
|
||||||
- **Оркестрация**: .NET Aspire (`GmRelay.AppHost`).
|
| Бот | Telegram.Bot, **Native AOT** |
|
||||||
- **База данных**: PostgreSQL
|
| Веб | Blazor Server |
|
||||||
- **ORM**: Dapper (с использованием Dapper.AOT для source generators).
|
| Оркестрация | .NET Aspire (`GmRelay.AppHost`) |
|
||||||
- **Миграции**: DbUp.
|
| БД | PostgreSQL |
|
||||||
- **Развертывание**: Docker Compose + Multi-arch (AMD64/ARM64).
|
| ORM | Dapper + **Dapper.AOT** (source generators) |
|
||||||
|
| Миграции | DbUp |
|
||||||
|
| Развёртывание | Docker Compose, Multi-arch (**AMD64/ARM64**) |
|
||||||
|
|
||||||
|
> [!NOTE]
|
||||||
|
> При использовании Dapper в режиме Native AOT все SQL-запросы используют строго типизированные DTO; динамические типы (`dynamic`) не поддерживаются.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 🚀 Быстрый старт (Docker Compose)
|
## 🚀 Быстрый старт (Docker Compose)
|
||||||
|
|
||||||
Проект использует Docker Compose для одновременного запуска базы данных, бота и веб-интерфейса.
|
**Требования:** Docker и Docker Compose.
|
||||||
|
|
||||||
### 1. Подготовка
|
|
||||||
Убедитесь, что у вас установлены **Docker** и **Docker Compose**.
|
|
||||||
|
|
||||||
### 2. Настройка окружения
|
|
||||||
Скопируйте файл-шаблон и заполните его значениями:
|
|
||||||
|
|
||||||
|
### 1. Настройка окружения
|
||||||
```bash
|
```bash
|
||||||
cp .env.example .env
|
cp .env.example .env
|
||||||
```
|
```
|
||||||
|
|
||||||
Отредактируйте `.env`:
|
**Ключевые переменные `.env`:**
|
||||||
|
|
||||||
```env
|
```env
|
||||||
# Токен вашего бота от @BotFather (используется и для бота, и как секретный ключ для веб-авторизации)
|
# Токен от @BotFather (используется ботом и как секретный ключ веб-авторизации)
|
||||||
TELEGRAM_BOT_TOKEN=ваш_токен_здесь
|
TELEGRAM_BOT_TOKEN=ваш_токен_здесь
|
||||||
|
|
||||||
# Имя вашего бота в Telegram (без @), например: GmRelayBot.
|
# Имя бота без @ (для Telegram Login Widget)
|
||||||
# Найти его можно в информации о боте у @BotFather.
|
|
||||||
# Используется для работы виджета авторизации (Telegram Login Widget).
|
|
||||||
TELEGRAM_BOT_USERNAME=ваше_имя_бота_здесь
|
TELEGRAM_BOT_USERNAME=ваше_имя_бота_здесь
|
||||||
|
|
||||||
# Пароль для базы данных PostgreSQL
|
# HTTPS URL Mini App, например https://your-domain.example/miniapp
|
||||||
|
TELEGRAM_MINI_APP_URL=https://your-domain.example/miniapp
|
||||||
|
|
||||||
POSTGRES_PASSWORD=ваш_надежный_пароль
|
POSTGRES_PASSWORD=ваш_надежный_пароль
|
||||||
|
GMRELAY_WEB_PORT=8080
|
||||||
```
|
```
|
||||||
|
|
||||||
*(Опционально)* Настройте домен Telegram бота в @BotFather командой `/setdomain` для работы виджета авторизации на вашем сайте.
|
**Настройка в @BotFather:**
|
||||||
|
- Команда `/setdomain` для работы виджета авторизации на вашем домене.
|
||||||
|
- Для Mini App настройте домен Web Dashboard и menu button на URL из `TELEGRAM_MINI_APP_URL`.
|
||||||
|
- Начиная с **v1.9.3** дополнительных действий для фикса входа не требуется: fallback выполняется внутри активного Telegram WebView по тому же HTTPS-адресу `/miniapp`.
|
||||||
|
|
||||||
### 3. Запуск
|
### 2. Запуск
|
||||||
Выполните команду:
|
|
||||||
```bash
|
```bash
|
||||||
docker compose up -d -build
|
docker compose up -d
|
||||||
```
|
|
||||||
Инфраструктура автоматически:
|
|
||||||
- Поднимет PostgreSQL.
|
|
||||||
- Запустит бота (применив миграции БД).
|
|
||||||
- Запустит веб-интерфейс (доступен по умолчанию на порту **8080** внутри контейнера).
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## ⚙️ Настройка бота в Telegram
|
|
||||||
|
|
||||||
Чтобы бот работал корректно:
|
|
||||||
1. **Добавьте бота в группу** (или Супергруппу/Форум).
|
|
||||||
2. **Назначьте бота Администратором**.
|
|
||||||
3. **Необходимые права**:
|
|
||||||
* `Выбор тем` (Managed Topics) — **обязательно** для Форумов.
|
|
||||||
* `Отправка сообщений`.
|
|
||||||
* `Закрепление сообщений` — рекомендуется.
|
|
||||||
|
|
||||||
> [!TIP]
|
|
||||||
> Колонку "Мастер" (GM) бот определяет по первому человеку, который создал сессию в этой группе. Только этот пользователь сможет отменять игры через кнопки бота и редактировать их в веб-интерфейсе.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 📝 Инструкция для Мастера
|
|
||||||
|
|
||||||
### Создание расписания игр
|
|
||||||
Используйте команду `/newsession` с описанием в следующем формате:
|
|
||||||
|
|
||||||
```text
|
|
||||||
/newsession
|
|
||||||
Название: Легенды Берега Мечей (D&D 5e)
|
|
||||||
Время: 15.05.2024 19:30
|
|
||||||
Время: 22.05.2024 19:00
|
|
||||||
Ссылка: https://discord.gg/invite-link
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Другие команды
|
**Автоматически выполняется:**
|
||||||
- `/listsessions` — Показать список всех актуальных игр в этой группе.
|
- создание Docker-сети и volume PostgreSQL;
|
||||||
- `/reschedulesession` — Перенести сессию на другое время с голосованием игроков.
|
- подъём PostgreSQL (`db:5432`);
|
||||||
- `/deletesession` — Удалить сессию.
|
- запуск бота с плавной миграцией (DbUp);
|
||||||
- `/exportcalendar` — Получить `.ics` файл с играми.
|
- запуск веб-приложения с подключением к БД и Telegram API.
|
||||||
- `/help` — Справка по формату.
|
|
||||||
|
### 3. Первоначальная настройка
|
||||||
|
1. Напишите боту `/start`.
|
||||||
|
2. Создайте группу через `/newgroup`.
|
||||||
|
3. Откройте Mini App или Web Dashboard для расширенного управления.
|
||||||
|
|
||||||
|
## 💾 Backup и восстановление
|
||||||
|
|
||||||
|
Проект включает автоматический ежедневный backup PostgreSQL через сервис `db-backup` в Docker Compose.
|
||||||
|
|
||||||
|
### Как это работает
|
||||||
|
- **Каждый день в 03:00** выполняется `pg_dump` базы `gmrelay_db`.
|
||||||
|
- Дампы сжимаются (`gzip`) и сохраняются в volume `pgbackups` (`/backups`).
|
||||||
|
- Формат имени: `gmrelay_db_YYYYMMDD_HHMMSS.sql.gz`.
|
||||||
|
- Ротация: по умолчанию хранятся последние **7 дней** (настраивается через `BACKUP_RETENTION_DAYS`).
|
||||||
|
|
||||||
|
### Проверка бэкапов
|
||||||
|
```bash
|
||||||
|
docker compose exec db-backup ls -la /backups
|
||||||
|
```
|
||||||
|
|
||||||
|
### Ручное создание дампа
|
||||||
|
```bash
|
||||||
|
docker compose exec db-backup sh -c "pg_dump -h db -U gmrelay -d gmrelay_db | gzip > /backups/gmrelay_db_manual.sql.gz"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Восстановление из бэкапа
|
||||||
|
```bash
|
||||||
|
# Использовать последний автоматический бэкап
|
||||||
|
./scripts/restore.sh
|
||||||
|
|
||||||
|
# Или указать конкретный файл
|
||||||
|
./scripts/restore.sh backups/gmrelay_db_20260512_030000.sql.gz
|
||||||
|
```
|
||||||
|
|
||||||
|
> [!WARNING]
|
||||||
|
> Восстановление **перезаписывает текущую базу данных**. Убедитесь, что вы понимаете последствия, прежде чем запускать `restore.sh`.
|
||||||
|
|
||||||
|
### Переменные окружения (опциональные)
|
||||||
|
```env
|
||||||
|
BACKUP_RETENTION_DAYS=7
|
||||||
|
BACKUP_VOLUME_NAME=game_pgbackups
|
||||||
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 🏗 Разработка и запуск локально (.NET Aspire)
|
## 🗂 Структура репозитория
|
||||||
|
|
||||||
Для локальной разработки проще всего использовать .NET Aspire:
|
```
|
||||||
|
├── src/
|
||||||
1. Установите [.NET 10 SDK](https://dotnet.microsoft.com/download/dotnet/10.0) и workload Aspire.
|
│ ├── GmRelay.AppHost/ # .NET Aspire orchestrator
|
||||||
2. Откройте решение `GM-Relay.slnx`.
|
│ ├── GmRelay.Bot/ # Telegram-бот (Native AOT)
|
||||||
3. Установите переменные окружения (или user secrets) для `GmRelay.AppHost`.
|
│ ├── GmRelay.Migrator/ # DbUp-миграции
|
||||||
4. Запустите проект `GmRelay.AppHost`. Aspire Dashboard запустится автоматически, предоставляя удобный мониторинг БД, бота и веб-интерфейса.
|
│ ├── GmRelay.ServiceDefaults/ # Aspire service defaults
|
||||||
|
│ ├── GmRelay.Shared/ # Общие доменные модели
|
||||||
> [!NOTE]
|
│ ├── GmRelay.Web/ # Blazor Server dashboard
|
||||||
> При использовании **Dapper** в режиме Native AOT, все SQL-запросы используют строго типизированные DTO. Динамические типы (`dynamic`) не поддерживаются.
|
│ └── GmRelay.Worker/ # Background workers
|
||||||
|
├── tests/
|
||||||
|
│ └── GmRelay.Bot.Tests/ # xUnit + NSubstitute
|
||||||
|
├── compose.yaml # Docker Compose (AMD64 + ARM64)
|
||||||
|
└── .env.example # Шаблон переменных окружения
|
||||||
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 📜 Лицензия
|
## 📜 Лицензия
|
||||||
Проект распространяется под лицензией MIT. Использование в некоммерческих целях приветствуется.
|
|
||||||
|
MIT License. См. [LICENSE](./LICENSE).
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
*Построено с ❤️ для TTRPG-сообщества.*
|
||||||
|
|||||||
+70
-20
@@ -1,52 +1,102 @@
|
|||||||
services:
|
services:
|
||||||
db:
|
db:
|
||||||
image: postgres:17-alpine
|
image: postgres:17-alpine
|
||||||
container_name: gmrelay_db
|
|
||||||
restart: always
|
restart: always
|
||||||
environment:
|
environment:
|
||||||
POSTGRES_USER: gmrelay
|
POSTGRES_USER: gmrelay
|
||||||
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
|
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:?Set POSTGRES_PASSWORD in .env}
|
||||||
POSTGRES_DB: gmrelay_db
|
POSTGRES_DB: gmrelay_db
|
||||||
volumes:
|
volumes:
|
||||||
- pgdata:/var/lib/postgresql/data
|
- pgdata:/var/lib/postgresql/data
|
||||||
ports:
|
networks:
|
||||||
- "5432:5432"
|
- gmrelay
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test: [ "CMD-SHELL", "pg_isready -U gmrelay -d gmrelay_db" ]
|
test: [ "CMD-SHELL", "pg_isready -U gmrelay -d gmrelay_db" ]
|
||||||
interval: 3s
|
interval: 3s
|
||||||
timeout: 3s
|
timeout: 3s
|
||||||
retries: 10
|
retries: 10
|
||||||
|
|
||||||
bot:
|
db-backup:
|
||||||
image: git.codeanddice.ru/toutsu/gmrelay-bot:1.0.0
|
image: postgres:17-alpine
|
||||||
container_name: gmrelay_bot
|
restart: unless-stopped
|
||||||
restart: always
|
|
||||||
network_mode: host
|
|
||||||
depends_on:
|
depends_on:
|
||||||
db:
|
db:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
environment:
|
environment:
|
||||||
- "ConnectionStrings__gmrelaydb=Host=127.0.0.1;Port=5432;Database=gmrelay_db;Username=gmrelay;Password=${POSTGRES_PASSWORD}"
|
POSTGRES_USER: gmrelay
|
||||||
- "Telegram__BotToken=${TELEGRAM_BOT_TOKEN}"
|
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:?Set POSTGRES_PASSWORD in .env}
|
||||||
|
POSTGRES_DB: gmrelay_db
|
||||||
|
PGPASSWORD: ${POSTGRES_PASSWORD:?Set POSTGRES_PASSWORD in .env}
|
||||||
|
BACKUP_RETENTION_DAYS: ${BACKUP_RETENTION_DAYS:-7}
|
||||||
|
volumes:
|
||||||
|
- pgbackups:/backups
|
||||||
|
networks:
|
||||||
|
- gmrelay
|
||||||
|
entrypoint: ["sh", "-c"]
|
||||||
|
command:
|
||||||
|
- |
|
||||||
|
cat > /usr/local/bin/backup.sh << 'EOF'
|
||||||
|
#!/bin/sh
|
||||||
|
set -e
|
||||||
|
TMPFILE="/tmp/backup_$$.sql"
|
||||||
|
pg_dump -h db -U gmrelay -d gmrelay_db > "$TMPFILE"
|
||||||
|
gzip "$TMPFILE"
|
||||||
|
mv "$TMPFILE.gz" "/backups/gmrelay_db_$(date +%Y%m%d_%H%M%S).sql.gz"
|
||||||
|
find /backups -name 'gmrelay_db_*.sql.gz' -type f -mtime +${BACKUP_RETENTION_DAYS} -delete
|
||||||
|
EOF
|
||||||
|
chmod +x /usr/local/bin/backup.sh
|
||||||
|
echo "0 3 * * * /usr/local/bin/backup.sh" | crontab -
|
||||||
|
crond -f
|
||||||
|
|
||||||
|
bot:
|
||||||
|
image: git.codeanddice.ru/toutsu/gmrelay-bot:2.0.0
|
||||||
|
restart: always
|
||||||
|
depends_on:
|
||||||
|
db:
|
||||||
|
condition: service_healthy
|
||||||
|
environment:
|
||||||
|
- "ConnectionStrings__gmrelaydb=Host=db;Port=5432;Database=gmrelay_db;Username=gmrelay;Password=${POSTGRES_PASSWORD:?Set POSTGRES_PASSWORD in .env}"
|
||||||
|
- "Telegram__BotToken=${TELEGRAM_BOT_TOKEN:?Set TELEGRAM_BOT_TOKEN in .env}"
|
||||||
|
- "Telegram__MiniAppUrl=${TELEGRAM_MINI_APP_URL:-}"
|
||||||
|
networks:
|
||||||
|
- gmrelay
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "wget -qO- http://localhost:8081/health || exit 1"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 3
|
||||||
|
|
||||||
web:
|
web:
|
||||||
image: git.codeanddice.ru/toutsu/gmrelay-web:1.0.0
|
image: git.codeanddice.ru/toutsu/gmrelay-web:2.0.0
|
||||||
container_name: gmrelay_web
|
|
||||||
restart: always
|
restart: always
|
||||||
network_mode: host
|
|
||||||
depends_on:
|
depends_on:
|
||||||
db:
|
db:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
environment:
|
environment:
|
||||||
- "ConnectionStrings__gmrelaydb=Host=127.0.0.1;Port=5432;Database=gmrelay_db;Username=gmrelay;Password=${POSTGRES_PASSWORD}"
|
- "ConnectionStrings__gmrelaydb=Host=db;Port=5432;Database=gmrelay_db;Username=gmrelay;Password=${POSTGRES_PASSWORD:?Set POSTGRES_PASSWORD in .env}"
|
||||||
- "Telegram__BotToken=${TELEGRAM_BOT_TOKEN}"
|
- "Telegram__BotToken=${TELEGRAM_BOT_TOKEN:?Set TELEGRAM_BOT_TOKEN in .env}"
|
||||||
- "Telegram__BotUsername=${TELEGRAM_BOT_USERNAME}"
|
- "Telegram__BotUsername=${TELEGRAM_BOT_USERNAME:?Set TELEGRAM_BOT_USERNAME in .env}"
|
||||||
|
- "Telegram__MiniAppUrl=${TELEGRAM_MINI_APP_URL:-}"
|
||||||
|
ports:
|
||||||
|
- "${GMRELAY_WEB_PORT:-8080}:8080"
|
||||||
volumes:
|
volumes:
|
||||||
- web_keys:/app/dataprotection-keys
|
- web_keys:/app/dataprotection-keys
|
||||||
|
networks:
|
||||||
|
- gmrelay
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "wget -qO- http://localhost:8080/health || exit 1"]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 3
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
pgdata:
|
pgdata:
|
||||||
external: true
|
name: ${POSTGRES_VOLUME_NAME:-game_pgdata}
|
||||||
name: game_pgdata
|
|
||||||
web_keys:
|
web_keys:
|
||||||
name: gmrelay_web_keys
|
name: ${WEB_KEYS_VOLUME_NAME:-gmrelay_web_keys}
|
||||||
|
pgbackups:
|
||||||
|
name: ${BACKUP_VOLUME_NAME:-game_pgbackups}
|
||||||
|
|
||||||
|
networks:
|
||||||
|
gmrelay:
|
||||||
|
driver: bridge
|
||||||
|
|||||||
@@ -0,0 +1,65 @@
|
|||||||
|
# ADR 002: Platform-Neutral Batch Rendering
|
||||||
|
|
||||||
|
## Status
|
||||||
|
|
||||||
|
**Accepted** — implemented in v1.10.0 (PR #42).
|
||||||
|
|
||||||
|
## Context
|
||||||
|
|
||||||
|
`SessionBatchRenderer` жил в `GmRelay.Shared` и напрямую зависел от `Telegram.Bot` (`InlineKeyboardMarkup`, `ParseMode.Html`). Это создавало проблемы:
|
||||||
|
|
||||||
|
1. **Shared не был platform-neutral.** Любой платформенный проект (Discord, Slack, WebSocket) тащил Telegram-зависимость.
|
||||||
|
2. **Дублирование логики.** `GmRelay.Web` использовал тот же рендерер через прямую зависимость от `Shared`, но Web — это не Telegram-клиент.
|
||||||
|
3. **Невозможно написать unit-тесты без Telegram-объектов.** Smoke-тесты создавали InlineKeyboardMarkup даже для проверки чисто доменной логики.
|
||||||
|
|
||||||
|
## Decision
|
||||||
|
|
||||||
|
Разделить рендеринг на две стадии:
|
||||||
|
|
||||||
|
1. **View Builder (platform-neutral)** — собирает view model из доменных DTO.
|
||||||
|
2. **Platform Renderer (platform-specific)** — превращает view model в платформенное представление.
|
||||||
|
|
||||||
|
```
|
||||||
|
Domain DTOs
|
||||||
|
│
|
||||||
|
▼
|
||||||
|
SessionBatchViewBuilder (Shared)
|
||||||
|
│
|
||||||
|
▼
|
||||||
|
SessionBatchViewModel (platform-neutral)
|
||||||
|
│
|
||||||
|
├──► TelegramSessionBatchRenderer ──► HTML + InlineKeyboardMarkup
|
||||||
|
│
|
||||||
|
└──► DiscordSessionBatchRenderer ──► (issue #26)
|
||||||
|
```
|
||||||
|
|
||||||
|
### Изменённые компоненты
|
||||||
|
|
||||||
|
| Компонент | Было | Стало |
|
||||||
|
|---|---|---|
|
||||||
|
| `SessionBatchRenderer` | `GmRelay.Shared.Rendering` | Удалён |
|
||||||
|
| `SessionBatchViewBuilder` | — | `GmRelay.Shared.Rendering` |
|
||||||
|
| `SessionBatchViewModel` | — | `GmRelay.Shared.Rendering` |
|
||||||
|
| `TelegramSessionBatchRenderer` | — | `GmRelay.Bot` + `GmRelay.Web` |
|
||||||
|
| `DiscordSessionBatchRenderer` | — | `GmRelay.Shared.Rendering` (stub) |
|
||||||
|
| `BatchMessageEditor` | `GmRelay.Shared.Rendering` | `GmRelay.Bot` + `GmRelay.Web` |
|
||||||
|
|
||||||
|
## Consequences
|
||||||
|
|
||||||
|
### Positive
|
||||||
|
|
||||||
|
- `GmRelay.Shared` больше не зависит от `Telegram.Bot`. Чистый platform-agnostic проект.
|
||||||
|
- Можно добавить `DiscordSessionBatchRenderer` без изменений в `Shared`.
|
||||||
|
- Unit-тесты ViewBuilder не создают `InlineKeyboardMarkup`.
|
||||||
|
- Логика подсчёта игроков, сортировки сессий и генерации действий — в одном месте (ViewBuilder).
|
||||||
|
|
||||||
|
### Negative
|
||||||
|
|
||||||
|
- **Временное дублирование.** `TelegramSessionBatchRenderer` и `BatchMessageEditor` скопированы в `Bot` и `Web`. Планируется вынести в `GmRelay.Shared.Telegram` при появлении третьего Telegram-потребителя.
|
||||||
|
- **Дополнительная стадия.** Теперь два вызова вместо одного: `Build` + `Render`. Этоtrade-off за чистоту абстракции.
|
||||||
|
|
||||||
|
## Related
|
||||||
|
|
||||||
|
- Issue #22 — этот рефакторинг.
|
||||||
|
- Issue #26 — Discord Bot MVP (потребитель новой архитектуры).
|
||||||
|
- ADR 001 — vertical slice, native AOT, Aspire (`docs/adr/0001-use-vertical-slice-native-aot-and-aspire.md`).
|
||||||
@@ -0,0 +1,438 @@
|
|||||||
|
# Player List + Kick + Waitlist Promotion Implementation Plan
|
||||||
|
|
||||||
|
> **For Hermes:** Use subagent-driven-development skill to implement this plan task-by-task.
|
||||||
|
|
||||||
|
**Goal:** Add a player list (with names) to the Web UI session views, allow GM to kick a specific player, and auto-promote the next waitlisted player.
|
||||||
|
|
||||||
|
**Architecture:** Extend `ISessionStore` with participant queries and a remove method. Update `GroupDetails.razor` to show expandable participant lists. Reuse existing `PromoteWaitlistedPlayerAsync` logic after removal.
|
||||||
|
|
||||||
|
**Tech Stack:** C# 14, Blazor SSR, Dapper, PostgreSQL
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 1: Add domain model for WebParticipant
|
||||||
|
|
||||||
|
**Objective:** Create a DTO to represent a session participant in the web layer.
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: `src/GmRelay.Web/Services/SessionService.cs`
|
||||||
|
|
||||||
|
**Step 1: Add record**
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
public sealed record WebParticipant(
|
||||||
|
Guid Id,
|
||||||
|
long TelegramId,
|
||||||
|
string DisplayName,
|
||||||
|
string? TelegramUsername,
|
||||||
|
string RsvpStatus,
|
||||||
|
string RegistrationStatus,
|
||||||
|
bool IsGm,
|
||||||
|
DateTime? RespondedAt);
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 2: Commit**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git add src/GmRelay.Web/Services/SessionService.cs
|
||||||
|
git commit -m "feat: add WebParticipant record"
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 2: Add GetSessionParticipantsAsync to ISessionStore
|
||||||
|
|
||||||
|
**Objective:** Retrieve all participants for a session with full player info.
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: `src/GmRelay.Web/Services/ISessionStore.cs`
|
||||||
|
- Modify: `src/GmRelay.Web/Services/SessionService.cs`
|
||||||
|
- Modify: `src/GmRelay.Web/Services/AuthorizedSessionService.cs`
|
||||||
|
|
||||||
|
**Step 1: Add to interface**
|
||||||
|
|
||||||
|
In `ISessionStore.cs`, add:
|
||||||
|
```csharp
|
||||||
|
Task<List<WebParticipant>> GetSessionParticipantsAsync(Guid sessionId);
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 2: Implement in SessionService**
|
||||||
|
|
||||||
|
In `SessionService.cs`, add:
|
||||||
|
```csharp
|
||||||
|
public async Task<List<WebParticipant>> GetSessionParticipantsAsync(Guid sessionId)
|
||||||
|
{
|
||||||
|
await using var conn = await dataSource.OpenConnectionAsync();
|
||||||
|
return (await conn.QueryAsync<WebParticipant>(
|
||||||
|
"""
|
||||||
|
SELECT sp.id AS Id,
|
||||||
|
p.telegram_id AS TelegramId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
sp.rsvp_status AS RsvpStatus,
|
||||||
|
sp.registration_status AS RegistrationStatus,
|
||||||
|
sp.is_gm AS IsGm,
|
||||||
|
sp.responded_at AS RespondedAt
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON p.id = sp.player_id
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
ORDER BY sp.is_gm DESC,
|
||||||
|
CASE sp.registration_status WHEN 'Active' THEN 0 ELSE 1 END,
|
||||||
|
sp.created_at
|
||||||
|
""",
|
||||||
|
new { SessionId = sessionId })).ToList();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 3: Add authorized wrapper**
|
||||||
|
|
||||||
|
In `AuthorizedSessionService.cs`, add:
|
||||||
|
```csharp
|
||||||
|
public async Task<List<WebParticipant>?> GetSessionParticipantsForGmAsync(Guid sessionId, long gmId)
|
||||||
|
{
|
||||||
|
var session = await GetSessionForGmAsync(sessionId, gmId);
|
||||||
|
if (session is null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return await sessionStore.GetSessionParticipantsAsync(sessionId);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 4: Commit**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git add src/GmRelay.Web/Services/ISessionStore.cs
|
||||||
|
|
||||||
|
git add src/GmRelay.Web/Services/SessionService.cs
|
||||||
|
|
||||||
|
git add src/GmRelay.Web/Services/AuthorizedSessionService.cs
|
||||||
|
|
||||||
|
git commit -m "feat: add GetSessionParticipantsAsync"
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 3: Add RemovePlayerFromSessionAsync with waitlist promotion
|
||||||
|
|
||||||
|
**Objective:** Allow GM to remove a specific player; auto-promote next waitlisted player if conditions met.
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: `src/GmRelay.Web/Services/ISessionStore.cs`
|
||||||
|
- Modify: `src/GmRelay.Web/Services/SessionService.cs`
|
||||||
|
- Modify: `src/GmRelay.Web/Services/AuthorizedSessionService.cs`
|
||||||
|
|
||||||
|
**Step 1: Add to interface**
|
||||||
|
|
||||||
|
In `ISessionStore.cs`, add:
|
||||||
|
```csharp
|
||||||
|
Task RemovePlayerFromSessionAsync(Guid sessionId, Guid groupId, Guid participantId);
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 2: Implement in SessionService**
|
||||||
|
|
||||||
|
In `SessionService.cs`, add:
|
||||||
|
```csharp
|
||||||
|
public async Task RemovePlayerFromSessionAsync(Guid sessionId, Guid groupId, Guid participantId)
|
||||||
|
{
|
||||||
|
await using var conn = await dataSource.OpenConnectionAsync();
|
||||||
|
await using var transaction = await conn.BeginTransactionAsync();
|
||||||
|
|
||||||
|
var session = await conn.QuerySingleOrDefaultAsync<WebSession>(
|
||||||
|
@"SELECT s.id, s.group_id AS GroupId, s.title, s.scheduled_at AS ScheduledAt, s.status, s.join_link AS JoinLink,
|
||||||
|
s.batch_id AS BatchId, s.batch_message_id AS BatchMessageId,
|
||||||
|
g.telegram_chat_id AS TelegramChatId,
|
||||||
|
s.max_players AS MaxPlayers,
|
||||||
|
0 AS ActivePlayerCount,
|
||||||
|
0 AS WaitlistedPlayerCount,
|
||||||
|
s.notification_mode AS NotificationMode
|
||||||
|
FROM sessions s
|
||||||
|
JOIN game_groups g ON g.id = s.group_id
|
||||||
|
WHERE s.id = @SessionId AND s.group_id = @GroupId
|
||||||
|
FOR UPDATE",
|
||||||
|
new { SessionId = sessionId, GroupId = groupId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (session is null)
|
||||||
|
{
|
||||||
|
throw new SessionAccessDeniedException(sessionId, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify participant exists in this session
|
||||||
|
var participant = await conn.QuerySingleOrDefaultAsync<WebParticipant>(
|
||||||
|
"""
|
||||||
|
SELECT sp.id AS Id,
|
||||||
|
p.telegram_id AS TelegramId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
sp.rsvp_status AS RsvpStatus,
|
||||||
|
sp.registration_status AS RegistrationStatus,
|
||||||
|
sp.is_gm AS IsGm,
|
||||||
|
sp.responded_at AS RespondedAt
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON p.id = sp.player_id
|
||||||
|
WHERE sp.id = @ParticipantId AND sp.session_id = @SessionId
|
||||||
|
""",
|
||||||
|
new { ParticipantId = participantId, SessionId = sessionId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (participant is null)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException("Участник не найден в этой сессии.");
|
||||||
|
}
|
||||||
|
|
||||||
|
bool wasActive = participant.RegistrationStatus == ParticipantRegistrationStatus.Active;
|
||||||
|
|
||||||
|
await conn.ExecuteAsync(
|
||||||
|
"DELETE FROM session_participants WHERE id = @ParticipantId",
|
||||||
|
new { ParticipantId = participantId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
WebPromotedParticipantDto? promoted = null;
|
||||||
|
|
||||||
|
if (wasActive)
|
||||||
|
{
|
||||||
|
promoted = await conn.QuerySingleOrDefaultAsync<WebPromotedParticipantDto>(
|
||||||
|
"""
|
||||||
|
SELECT sp.id AS ParticipantRowId,
|
||||||
|
p.display_name AS DisplayName
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON p.id = sp.player_id
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Waitlisted
|
||||||
|
ORDER BY sp.created_at ASC, sp.id ASC
|
||||||
|
LIMIT 1
|
||||||
|
FOR UPDATE OF sp
|
||||||
|
""",
|
||||||
|
new { SessionId = sessionId, Waitlisted = ParticipantRegistrationStatus.Waitlisted },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (promoted is not null)
|
||||||
|
{
|
||||||
|
await conn.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
UPDATE session_participants
|
||||||
|
SET registration_status = @Active,
|
||||||
|
rsvp_status = @Pending,
|
||||||
|
responded_at = NULL
|
||||||
|
WHERE id = @ParticipantRowId
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
promoted.ParticipantRowId,
|
||||||
|
Active = ParticipantRegistrationStatus.Active,
|
||||||
|
Pending = RsvpStatus.Pending
|
||||||
|
},
|
||||||
|
transaction);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await transaction.CommitAsync();
|
||||||
|
|
||||||
|
// Notifications
|
||||||
|
await bot.SendMessage(
|
||||||
|
session.TelegramChatId,
|
||||||
|
$"🚪 <b>{System.Net.WebUtility.HtmlEncode(participant.DisplayName)}</b> удален(а) из сессии «{System.Net.WebUtility.HtmlEncode(session.Title)}».",
|
||||||
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html);
|
||||||
|
|
||||||
|
if (promoted is not null)
|
||||||
|
{
|
||||||
|
await bot.SendMessage(
|
||||||
|
session.TelegramChatId,
|
||||||
|
$"⬆️ <b>{System.Net.WebUtility.HtmlEncode(promoted.DisplayName)}</b> переведен(а) из листа ожидания в основной состав «{System.Net.WebUtility.HtmlEncode(session.Title)}».",
|
||||||
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (session.BatchMessageId.HasValue)
|
||||||
|
{
|
||||||
|
await TryUpdateBatchMessageAsync(session.BatchId, session.TelegramChatId, session.BatchMessageId.Value, session.Title);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 3: Add authorized wrapper**
|
||||||
|
|
||||||
|
In `AuthorizedSessionService.cs`, add:
|
||||||
|
```csharp
|
||||||
|
public async Task RemovePlayerFromSessionForGmAsync(Guid sessionId, long gmId, Guid participantId)
|
||||||
|
{
|
||||||
|
var session = await GetSessionForGmAsync(sessionId, gmId);
|
||||||
|
if (session is null)
|
||||||
|
{
|
||||||
|
throw new SessionAccessDeniedException(sessionId, gmId);
|
||||||
|
}
|
||||||
|
|
||||||
|
await sessionStore.RemovePlayerFromSessionAsync(sessionId, session.GroupId, participantId);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 4: Commit**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git add src/GmRelay.Web/Services/ISessionStore.cs
|
||||||
|
|
||||||
|
git add src/GmRelay.Web/Services/SessionService.cs
|
||||||
|
|
||||||
|
git add src/GmRelay.Web/Services/AuthorizedSessionService.cs
|
||||||
|
|
||||||
|
git commit -m "feat: add RemovePlayerFromSessionAsync with waitlist promotion"
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 4: Modify GroupDetails.razor to show participant list
|
||||||
|
|
||||||
|
**Objective:** Add expandable player lists to each session row with kick buttons.
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: `src/GmRelay.Web/Components/Pages/GroupDetails.razor`
|
||||||
|
|
||||||
|
**Step 1:** Add `participants` dictionary and `kickingParticipantId` state variables.
|
||||||
|
|
||||||
|
**Step 2:** Add `LoadParticipants(Guid sessionId)` and `KickParticipant(Guid sessionId, Guid participantId)` methods.
|
||||||
|
|
||||||
|
**Step 3:** In desktop table, add a new column or expand row with participant list.
|
||||||
|
|
||||||
|
**Step 4:** In mobile cards, add expandable participant section.
|
||||||
|
|
||||||
|
**Step 5:** Add styles to `app.css` if needed (badge styles are already present).
|
||||||
|
|
||||||
|
**Step 6:** Commit
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git add src/GmRelay.Web/Components/Pages/GroupDetails.razor
|
||||||
|
|
||||||
|
git add src/GmRelay.Web/wwwroot/app.css
|
||||||
|
|
||||||
|
git commit -m "feat: show player list and kick button in GroupDetails"
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 5: Modify EditSession.razor to show participant list
|
||||||
|
|
||||||
|
**Objective:** Show participant list on the edit page with kick capability.
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: `src/GmRelay.Web/Components/Pages/EditSession.razor`
|
||||||
|
|
||||||
|
**Step 1:** Load participants in `OnInitializedAsync`.
|
||||||
|
|
||||||
|
**Step 2:** Render participant list below the edit form.
|
||||||
|
|
||||||
|
**Step 3:** Add kick button for each non-GM participant.
|
||||||
|
|
||||||
|
**Step 4:** Commit
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git add src/GmRelay.Web/Components/Pages/EditSession.razor
|
||||||
|
|
||||||
|
git commit -m "feat: show player list and kick button in EditSession"
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 6: Add backend tests
|
||||||
|
|
||||||
|
**Objective:** Cover new GetSessionParticipants and RemovePlayerFromSession logic.
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Create: `tests/GmRelay.Bot.Tests/Web/SessionParticipantTests.cs`
|
||||||
|
|
||||||
|
**Step 1:** Write tests for `GetSessionParticipantsForGmAsync`.
|
||||||
|
|
||||||
|
**Step 2:** Write tests for `RemovePlayerFromSessionForGmAsync` including waitlist promotion.
|
||||||
|
|
||||||
|
**Step 3:** Run tests
|
||||||
|
|
||||||
|
```bash
|
||||||
|
dotnet test tests/GmRelay.Bot.Tests/GmRelay.Bot.Tests.csproj -v n
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 4:** Commit
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git add tests/GmRelay.Bot.Tests/Web/SessionParticipantTests.cs
|
||||||
|
|
||||||
|
git commit -m "test: add SessionParticipant service tests"
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 7: Update README
|
||||||
|
|
||||||
|
**Objective:** Bump version and document new features.
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: `README.md`
|
||||||
|
|
||||||
|
**Step 1:** Change version from `v1.9.6` to `v1.9.7`.
|
||||||
|
|
||||||
|
**Step 2:** Add bullet under Web Dashboard: player list with kick and auto-promote.
|
||||||
|
|
||||||
|
**Step 3:** Commit
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git add README.md
|
||||||
|
|
||||||
|
git commit -m "docs: bump README to v1.9.7, document player list kick"
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 8: Update Wiki
|
||||||
|
|
||||||
|
**Objective:** Update `Руководство ГМа` page with player management instructions.
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: Wiki page `Руководство ГМа`
|
||||||
|
|
||||||
|
**Step 1:** Read current wiki content via MCP.
|
||||||
|
|
||||||
|
**Step 2:** Add section about viewing player list and removing players.
|
||||||
|
|
||||||
|
**Step 3:** Update via MCP.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 9: Push branch and run CI
|
||||||
|
|
||||||
|
**Objective:** Push branch, monitor workflow, fix issues.
|
||||||
|
|
||||||
|
**Step 1:** Push
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git push -u origin feat/player-list-kick-waitlist
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 2:** Check workflow run via MCP gitea actions.
|
||||||
|
|
||||||
|
**Step 3:** Fix any issues.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Task 10: Merge and create release
|
||||||
|
|
||||||
|
**Objective:** Merge PR (or fast-forward), tag, create release.
|
||||||
|
|
||||||
|
**Step 1:** Merge to main
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git checkout main
|
||||||
|
|
||||||
|
git merge --no-ff feat/player-list-kick-waitlist -m "feat: player list, kick, and waitlist promotion (#X)"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 2:** Tag v1.9.7
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git tag v1.9.7
|
||||||
|
|
||||||
|
git push origin main --tags
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 3:** Create release via MCP gitea_create_release.
|
||||||
|
|
||||||
|
---
|
||||||
@@ -0,0 +1,69 @@
|
|||||||
|
# Telegram Mini App Dashboard Implementation Plan
|
||||||
|
|
||||||
|
> **For agentic workers:** REQUIRED SUB-SKILL: Use superpowers:subagent-driven-development (recommended) or superpowers:executing-plans to implement this plan task-by-task. Steps use checkbox (`- [ ]`) syntax for tracking.
|
||||||
|
|
||||||
|
**Goal:** Add a Telegram Mini App mobile dashboard that reuses the existing Web Dashboard and validates Telegram WebApp `initData` on the server.
|
||||||
|
|
||||||
|
**Architecture:** Extend `TelegramAuthService` for WebApp init data, add a `/miniapp` Blazor entry page plus `/auth/telegram-webapp` endpoint, and add bot entry points through an inline WebApp button and optional menu button setup. Existing application/domain services remain the only write path.
|
||||||
|
|
||||||
|
**Tech Stack:** .NET 10, Blazor Server, Telegram.Bot, xUnit, Dapper/Npgsql-backed existing services.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Task 1: Telegram WebApp Authentication
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: `src/GmRelay.Web/Services/TelegramAuthService.cs`
|
||||||
|
- Modify: `src/GmRelay.Web/Program.cs`
|
||||||
|
- Test: `tests/GmRelay.Bot.Tests/Web/TelegramAuthServiceTests.cs`
|
||||||
|
|
||||||
|
- [ ] Write failing tests for valid WebApp `initData`, tampered hash, and expired auth date.
|
||||||
|
- [ ] Run `dotnet test tests/GmRelay.Bot.Tests/GmRelay.Bot.Tests.csproj --filter TelegramAuthServiceTests`.
|
||||||
|
- [ ] Implement WebApp HMAC verification using the Telegram `WebAppData` secret derivation.
|
||||||
|
- [ ] Add `/auth/telegram-webapp` endpoint that signs in using the same claims as `/auth/telegram`.
|
||||||
|
- [ ] Re-run the filtered tests.
|
||||||
|
|
||||||
|
### Task 2: Mini App Entry Page
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Create: `src/GmRelay.Web/Components/Pages/MiniApp.razor`
|
||||||
|
- Modify: `src/GmRelay.Web/Components/App.razor`
|
||||||
|
- Modify: `src/GmRelay.Web/wwwroot/app.css`
|
||||||
|
- Test: `tests/GmRelay.Bot.Tests/Web/MiniAppDashboardTests.cs`
|
||||||
|
|
||||||
|
- [ ] Write failing tests that assert `/miniapp`, `telegram-web-app.js`, `authenticateTelegramMiniApp`, and Mini App CSS hooks exist.
|
||||||
|
- [ ] Run `dotnet test tests/GmRelay.Bot.Tests/GmRelay.Bot.Tests.csproj --filter MiniAppDashboardTests`.
|
||||||
|
- [ ] Implement `/miniapp` to post `Telegram.WebApp.initData` to `/auth/telegram-webapp`, expand/ready the Mini App, and show fallback login when opened outside Telegram.
|
||||||
|
- [ ] Add CSS for a mobile-first Mini App shell and compact dashboard spacing.
|
||||||
|
- [ ] Re-run the filtered tests.
|
||||||
|
|
||||||
|
### Task 3: Bot Entry Points
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Create: `src/GmRelay.Bot/Infrastructure/Telegram/TelegramMiniAppMenuButtonService.cs`
|
||||||
|
- Modify: `src/GmRelay.Bot/Infrastructure/Telegram/UpdateRouter.cs`
|
||||||
|
- Modify: `src/GmRelay.Bot/Program.cs`
|
||||||
|
- Test: `tests/GmRelay.Bot.Tests/Infrastructure/Telegram/TelegramMiniAppEntryPointTests.cs`
|
||||||
|
|
||||||
|
- [ ] Write failing tests that assert `/start` exposes a WebApp button and startup registers the menu button service.
|
||||||
|
- [ ] Run `dotnet test tests/GmRelay.Bot.Tests/GmRelay.Bot.Tests.csproj --filter TelegramMiniAppEntryPointTests`.
|
||||||
|
- [ ] Add a configurable `Telegram:MiniAppUrl` entry point; when missing, keep existing command behavior.
|
||||||
|
- [ ] Add hosted service that calls `SetChatMenuButton` with `MenuButtonWebApp` only when the URL is configured.
|
||||||
|
- [ ] Re-run the filtered tests.
|
||||||
|
|
||||||
|
### Task 4: Docs, Versions, and Release Prep
|
||||||
|
|
||||||
|
**Files:**
|
||||||
|
- Modify: `Directory.Build.props`
|
||||||
|
- Modify: `compose.yaml`
|
||||||
|
- Modify: `.gitea/workflows/deploy.yml`
|
||||||
|
- Modify: `src/GmRelay.Web/wwwroot/app.css`
|
||||||
|
- Modify: `src/GmRelay.Web/Components/Layout/NavMenu.razor`
|
||||||
|
- Modify: `README.md`
|
||||||
|
- Wiki: `Home`, `Быстрый старт`, `Руководство ГМа`, `Развёртывание`, `Архитектура`, `Разработка`
|
||||||
|
|
||||||
|
- [ ] Update project/container/workflow/UI versions to `1.9.0`.
|
||||||
|
- [ ] Document `TELEGRAM_MINI_APP_URL`, BotFather `/setmenubutton`, `/miniapp`, and WebApp auth.
|
||||||
|
- [ ] Run `dotnet test tests/GmRelay.Bot.Tests/GmRelay.Bot.Tests.csproj --collect:"XPlat Code Coverage"`.
|
||||||
|
- [ ] Run `dotnet build GM-Relay.slnx -c Release`.
|
||||||
|
- [ ] Commit, push, close issue #17, update wiki, create tag/release `v1.9.0`.
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
# Telegram Mini App Dashboard Design
|
||||||
|
|
||||||
|
## Goal
|
||||||
|
|
||||||
|
Issue #17 adds a Telegram Mini App dashboard as the mobile entry point for the existing Web Dashboard. Owner and co-GM users must be able to open the dashboard from Telegram, pass server-side Telegram WebApp `initData` validation, and manage only their own groups.
|
||||||
|
|
||||||
|
## Scope
|
||||||
|
|
||||||
|
- Add Mini App authentication using Telegram WebApp `initData`.
|
||||||
|
- Add a `/miniapp` entry page that signs the user into the existing cookie auth flow, then opens the regular dashboard UI in mobile-first mode.
|
||||||
|
- Reuse `AuthorizedSessionService`, `SessionService`, and existing Blazor pages for groups, sessions, templates, waitlist promotion, edit forms, and bulk batch operations.
|
||||||
|
- Add bot entry points: a Mini App button in `/start` and a configurable default menu button when `Telegram:MiniAppUrl` is set.
|
||||||
|
- Update README, wiki, deployment config, and visible version strings to `1.9.0`.
|
||||||
|
|
||||||
|
## Architecture
|
||||||
|
|
||||||
|
The Mini App is not a second dashboard implementation. It is a Telegram-authenticated entrance into the existing Blazor dashboard. This keeps authorization, domain operations, Telegram message synchronization, and Web Dashboard behavior in one place.
|
||||||
|
|
||||||
|
`TelegramAuthService` gains a second verification method for WebApp `initData`. The server accepts the raw URL-encoded init payload at `/auth/telegram-webapp`, verifies the Telegram HMAC with the bot token, extracts the user id/name from the embedded `user` JSON, and issues the same auth cookie as the login widget endpoint.
|
||||||
|
|
||||||
|
`/miniapp` loads `telegram-web-app.js`, posts `window.Telegram.WebApp.initData` to the server endpoint, expands the WebApp viewport, and redirects to `/`. If a user opens `/miniapp` outside Telegram, the page shows the regular login fallback.
|
||||||
|
|
||||||
|
## Data Flow
|
||||||
|
|
||||||
|
1. User opens the Mini App from the bot menu button or `/start` inline button.
|
||||||
|
2. Telegram injects `initData` into the WebApp JavaScript API.
|
||||||
|
3. `/miniapp` posts `{ initData }` to `/auth/telegram-webapp`.
|
||||||
|
4. The server verifies the WebApp signature and expiry.
|
||||||
|
5. The server creates the same claims used by Telegram Login Widget.
|
||||||
|
6. Existing Blazor pages load groups through `AuthorizedSessionService`.
|
||||||
|
7. Any edit, waitlist, template, or batch action still goes through existing services and keeps Telegram messages synchronized.
|
||||||
|
|
||||||
|
## Error Handling
|
||||||
|
|
||||||
|
- Missing or invalid init data returns `401` and leaves the user on the Mini App page.
|
||||||
|
- Expired auth data is rejected with the same 24-hour window used by the Login Widget.
|
||||||
|
- A verified Telegram user with no owner/co-GM groups sees the existing empty dashboard state.
|
||||||
|
- Direct navigation to a foreign group/session still redirects to `/access-denied` through existing authorization checks.
|
||||||
|
|
||||||
|
## Testing
|
||||||
|
|
||||||
|
- Unit tests cover valid and invalid WebApp `initData`.
|
||||||
|
- File-level regression tests ensure `/miniapp`, `/auth/telegram-webapp`, Telegram WebApp script loading, bot Mini App button, menu button setup, and mobile Mini App CSS hooks remain present.
|
||||||
|
- Existing `AuthorizedSessionServiceTests` continue covering owner/co-GM access behavior.
|
||||||
@@ -0,0 +1,77 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
# GM-Relay PostgreSQL Backup Restore Script
|
||||||
|
# Usage: ./scripts/restore.sh [backup_file]
|
||||||
|
# If no file is provided, uses the most recent backup.
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
PROJECT_ROOT="$(cd "${SCRIPT_DIR}/.." && pwd)"
|
||||||
|
|
||||||
|
# Check required env
|
||||||
|
if [ -z "${POSTGRES_PASSWORD:-}" ]; then
|
||||||
|
if [ -f "${PROJECT_ROOT}/.env" ]; then
|
||||||
|
# shellcheck source=/dev/null
|
||||||
|
set -a
|
||||||
|
. "${PROJECT_ROOT}/.env"
|
||||||
|
set +a
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -z "${POSTGRES_PASSWORD:-}" ]; then
|
||||||
|
echo "ERROR: POSTGRES_PASSWORD is not set. Please set it in your environment or .env file."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
BACKUP_DIR="${PROJECT_ROOT}/backups"
|
||||||
|
|
||||||
|
# Determine backup file
|
||||||
|
if [ $# -ge 1 ]; then
|
||||||
|
BACKUP_FILE="$1"
|
||||||
|
else
|
||||||
|
BACKUP_FILE=$(find "${BACKUP_DIR}" -name 'gmrelay_db_*.sql.gz' -type f -printf '%T+ %p\n' 2>/dev/null | sort -r | head -n1 | cut -d' ' -f2-)
|
||||||
|
if [ -z "${BACKUP_FILE}" ]; then
|
||||||
|
echo "ERROR: No backup files found in ${BACKUP_DIR}."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -f "${BACKUP_FILE}" ]; then
|
||||||
|
echo "ERROR: Backup file not found: ${BACKUP_FILE}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "=================================================="
|
||||||
|
echo " GM-Relay PostgreSQL Restore"
|
||||||
|
echo "=================================================="
|
||||||
|
echo ""
|
||||||
|
echo "Backup file: ${BACKUP_FILE}"
|
||||||
|
echo "Database: gmrelay_db"
|
||||||
|
echo "User: gmrelay"
|
||||||
|
echo ""
|
||||||
|
read -p "This will OVERWRITE the current database. Are you sure? [y/N] " CONFIRM
|
||||||
|
|
||||||
|
if [[ ! "${CONFIRM}" =~ ^[Yy]$ ]]; then
|
||||||
|
echo "Restore cancelled."
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "Restoring database from ${BACKUP_FILE}..."
|
||||||
|
|
||||||
|
# Restore using docker compose exec to leverage the running postgres container
|
||||||
|
COMPOSE_ARGS="-f ${PROJECT_ROOT}/compose.yaml"
|
||||||
|
|
||||||
|
docker compose ${COMPOSE_ARGS} exec -T -e PGPASSWORD="${POSTGRES_PASSWORD}" db psql \
|
||||||
|
-U gmrelay \
|
||||||
|
-d gmrelay_db \
|
||||||
|
-c "DROP SCHEMA public CASCADE; CREATE SCHEMA public;" 2>/dev/null || true
|
||||||
|
|
||||||
|
gunzip -c "${BACKUP_FILE}" | docker compose ${COMPOSE_ARGS} exec -T -e PGPASSWORD="${POSTGRES_PASSWORD}" db psql \
|
||||||
|
-U gmrelay \
|
||||||
|
-d gmrelay_db
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "=================================================="
|
||||||
|
echo " Restore completed successfully!"
|
||||||
|
echo "=================================================="
|
||||||
@@ -0,0 +1,687 @@
|
|||||||
|
{
|
||||||
|
"version": 1,
|
||||||
|
"dependencies": {
|
||||||
|
"net10.0": {
|
||||||
|
"Aspire.Dashboard.Sdk.win-x64": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[13.2.1, )",
|
||||||
|
"resolved": "13.2.1",
|
||||||
|
"contentHash": "KLB9rXwY8kg2taWwxsJFoK0cAuupSZurcv1zTyYMqLyNuwvYYjs65Yz3g/cgh22QlUfOT3tOh+Jzk5MdJhy5+w=="
|
||||||
|
},
|
||||||
|
"Aspire.Hosting.AppHost": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[13.2.1, )",
|
||||||
|
"resolved": "13.2.1",
|
||||||
|
"contentHash": "4B/eoZPwOobxpMpvYnqe/EcXabjPhZJhfxlHXv5gdKd16duoWbHnvvAZJsVI3WUpakCwmsCiTrT4sNGfW8H+IQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"AspNetCore.HealthChecks.Uris": "9.0.0",
|
||||||
|
"Aspire.Hosting": "13.2.1",
|
||||||
|
"Google.Protobuf": "3.33.5",
|
||||||
|
"Grpc.AspNetCore": "2.76.0",
|
||||||
|
"Grpc.Net.ClientFactory": "2.76.0",
|
||||||
|
"Grpc.Tools": "2.78.0",
|
||||||
|
"Humanizer.Core": "2.14.1",
|
||||||
|
"JsonPatch.Net": "3.3.0",
|
||||||
|
"KubernetesClient": "18.0.13",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileSystemGlobbing": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Http": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5",
|
||||||
|
"ModelContextProtocol": "1.0.0",
|
||||||
|
"Newtonsoft.Json": "13.0.4",
|
||||||
|
"Polly.Core": "8.6.5",
|
||||||
|
"Semver": "3.0.0",
|
||||||
|
"StreamJsonRpc": "2.22.23",
|
||||||
|
"System.IO.Hashing": "10.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Aspire.Hosting.Orchestration.win-x64": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[13.2.1, )",
|
||||||
|
"resolved": "13.2.1",
|
||||||
|
"contentHash": "39lRUH4WuCsBaYB7fZH1/r81SSJIXrA8WphBlAdP1QT95+1sKQHzXJuXU4nzKpBLv4oZmjcWzvA+FDMGZbWmkw=="
|
||||||
|
},
|
||||||
|
"Aspire.Hosting.PostgreSQL": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[13.2.1, )",
|
||||||
|
"resolved": "13.2.1",
|
||||||
|
"contentHash": "7F/nmeplR9cYE/B/E1haRjnkoBRQ/voMXpnK/SNJoXSFs4Vb/g00CDDvI/xfH3SAV7Xq8ekWa9ZbX56JuQ+YiA==",
|
||||||
|
"dependencies": {
|
||||||
|
"AspNetCore.HealthChecks.NpgSql": "9.0.0",
|
||||||
|
"AspNetCore.HealthChecks.Uris": "9.0.0",
|
||||||
|
"Aspire.Hosting": "13.2.1",
|
||||||
|
"Google.Protobuf": "3.33.5",
|
||||||
|
"Grpc.AspNetCore": "2.76.0",
|
||||||
|
"Grpc.Net.ClientFactory": "2.76.0",
|
||||||
|
"Grpc.Tools": "2.78.0",
|
||||||
|
"Humanizer.Core": "2.14.1",
|
||||||
|
"JsonPatch.Net": "3.3.0",
|
||||||
|
"KubernetesClient": "18.0.13",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": "8.0.25",
|
||||||
|
"Microsoft.Extensions.FileSystemGlobbing": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Http": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5",
|
||||||
|
"ModelContextProtocol": "1.0.0",
|
||||||
|
"Newtonsoft.Json": "13.0.4",
|
||||||
|
"Polly.Core": "8.6.5",
|
||||||
|
"Semver": "3.0.0",
|
||||||
|
"StreamJsonRpc": "2.22.23",
|
||||||
|
"System.IO.Hashing": "10.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"SecurityCodeScan.VS2019": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[5.6.7, )",
|
||||||
|
"resolved": "5.6.7",
|
||||||
|
"contentHash": "WIE9RJswdSc2j+rLz2gW6U+gMUjMHzY2j7C/CL8/R2olXNM/+twarfMnWqm+rZodDBvaYDApJyxM8mVYf9FGrQ=="
|
||||||
|
},
|
||||||
|
"Aspire.Hosting": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "13.2.1",
|
||||||
|
"contentHash": "GY/T5iK2F4K3Sk60VUeVnTX1MhCjSaX48+qPUjA/rI1x1ONHevHzFj+Gc3fNlGEaZGY8L87hSxwGrV+Bjd5EJw==",
|
||||||
|
"dependencies": {
|
||||||
|
"AspNetCore.HealthChecks.Uris": "9.0.0",
|
||||||
|
"Google.Protobuf": "3.33.5",
|
||||||
|
"Grpc.AspNetCore": "2.76.0",
|
||||||
|
"Grpc.Net.ClientFactory": "2.76.0",
|
||||||
|
"Grpc.Tools": "2.78.0",
|
||||||
|
"Humanizer.Core": "2.14.1",
|
||||||
|
"JsonPatch.Net": "3.3.0",
|
||||||
|
"KubernetesClient": "18.0.13",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": "8.0.25",
|
||||||
|
"Microsoft.Extensions.FileSystemGlobbing": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Http": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5",
|
||||||
|
"ModelContextProtocol": "1.0.0",
|
||||||
|
"Newtonsoft.Json": "13.0.4",
|
||||||
|
"Polly.Core": "8.6.5",
|
||||||
|
"Semver": "3.0.0",
|
||||||
|
"StreamJsonRpc": "2.22.23",
|
||||||
|
"System.IO.Hashing": "10.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"AspNetCore.HealthChecks.NpgSql": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "9.0.0",
|
||||||
|
"contentHash": "npc58/AD5zuVxERdhCl2Kb7WnL37mwX42SJcXIwvmEig0/dugOLg3SIwtfvvh3TnvTwR/sk5LYNkkPaBdks61A==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": "8.0.11",
|
||||||
|
"Npgsql": "8.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"AspNetCore.HealthChecks.Uris": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "9.0.0",
|
||||||
|
"contentHash": "XYdNlA437KeF8p9qOpZFyNqAN+c0FXt/JjTvzH/Qans0q0O3pPE8KPnn39ucQQjR/Roum1vLTP3kXiUs8VHyuA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": "8.0.11",
|
||||||
|
"Microsoft.Extensions.Http": "8.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Fractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "7.3.0",
|
||||||
|
"contentHash": "2bETFWLBc8b7Ut2SVi+bxhGVwiSpknHYGBh2PADyGWONLkTxT7bKyDRhF8ao+XUv90tq8Fl7GTPxSI5bacIRJw=="
|
||||||
|
},
|
||||||
|
"Google.Protobuf": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "3.33.5",
|
||||||
|
"contentHash": "XEzLpCTosZb5I6eGSPn7rAES0VfkJkn3Cqydh0W39POdZwkdhPhOmAROTFJF9g0ardst4ulNXRm/q/iXwNu+Qw=="
|
||||||
|
},
|
||||||
|
"Grpc.AspNetCore": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.76.0",
|
||||||
|
"contentHash": "LyXMmpN2Ba0TE35SOLSKbGqIYtJuhc1UgiaGfoW1X8KJERV70QI5KGW+ckEY7MrXoFWN/uWo4B70siVhbDmCgQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Google.Protobuf": "3.31.1",
|
||||||
|
"Grpc.AspNetCore.Server.ClientFactory": "2.76.0",
|
||||||
|
"Grpc.Tools": "2.76.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Grpc.AspNetCore.Server": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.76.0",
|
||||||
|
"contentHash": "diSC/ZeNdSdxHdYSOpYwuSBBDYpuNVtJQFJfiBB0WrYOQ4lVMmdxuUZJcViahQyo8pCvS3Mueo5lqFxwwMF/iw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Grpc.Net.Common": "2.76.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Grpc.AspNetCore.Server.ClientFactory": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.76.0",
|
||||||
|
"contentHash": "y5KGO1GO0N2L/hCCMR05mmoK8j+v8rKvZ+9nothAxKx2Tf2CwV8f4TM5K0GkKfDsp4vrc4lm90MU6E+DeN7YIw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Grpc.AspNetCore.Server": "2.76.0",
|
||||||
|
"Grpc.Net.ClientFactory": "2.76.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Grpc.Core.Api": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.76.0",
|
||||||
|
"contentHash": "cSxC2tdnFdXXuBgIn1pjc4YBx7LXTCp4M0qn+SMBS35VWZY+cEQYLWTBDDhdBH1HzU7BV+ncVZlniGQHMpRJKQ=="
|
||||||
|
},
|
||||||
|
"Grpc.Net.Client": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.76.0",
|
||||||
|
"contentHash": "K1oldmqw2+Gn69nGRzZLhqSiUZwelX1GrBu/cUl9wNf1C0uB61vFS6JcxUUv9P8VoUJhFsmV44JA6lI2EUt4xw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Grpc.Net.Common": "2.76.0",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "8.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Grpc.Net.ClientFactory": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.76.0",
|
||||||
|
"contentHash": "XI+kO69L9AV8B9N0UQOmH911r6MOEp9huHiavEsY56DJYuzJ9KAxNGy37dpV6CLbgCaN2uKmpOsZ9Pao6bmpVQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Grpc.Net.Client": "2.76.0",
|
||||||
|
"Microsoft.Extensions.Http": "8.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Grpc.Net.Common": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.76.0",
|
||||||
|
"contentHash": "bZpiMVYgvpB44/wBh1RotrkqC7bg2FOasLri2GhR3hMKyzsiTxCoDE49YjPrJeFc4RW0wS8u+EInI09sjxVFRA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Grpc.Core.Api": "2.76.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Grpc.Tools": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.78.0",
|
||||||
|
"contentHash": "6jPG2gHon+w2PczW8jjrCRnW/g9eEfCdd7aK6mDooptWtuPsV3ZxAwKKEx7LGEDVoT4c2SViRl8Yu3L1XiWIIg=="
|
||||||
|
},
|
||||||
|
"Humanizer.Core": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.14.1",
|
||||||
|
"contentHash": "lQKvtaTDOXnoVJ20ibTuSIOf2i0uO0MPbDhd1jm238I+U/2ZnRENj0cktKZhtchBMtCUSRQ5v4xBCUbKNmyVMw=="
|
||||||
|
},
|
||||||
|
"Json.More.Net": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.1.0",
|
||||||
|
"contentHash": "qtwsyAsL55y2vB2/sK4Pjg3ZyVzD5KKSpV3lOAMHlnjFfsjQ/86eHJfQT9aV1YysVXzF4+xyHOZbh7Iu3YQ7Lg=="
|
||||||
|
},
|
||||||
|
"JsonPatch.Net": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "3.3.0",
|
||||||
|
"contentHash": "GIcMMDtzfzVfIpQgey8w7dhzcw6jG5nD4DDAdQCTmHfblkCvN7mI8K03to8YyUhKMl4PTR6D6nLSvWmyOGFNTg==",
|
||||||
|
"dependencies": {
|
||||||
|
"JsonPointer.Net": "5.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"JsonPointer.Net": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "5.2.0",
|
||||||
|
"contentHash": "qe1F7Tr/p4mgwLPU9P60MbYkp+xnL2uCPnWXGgzfR/AZCunAZIC0RZ32dLGJJEhSuLEfm0YF/1R3u5C7mEVq+w==",
|
||||||
|
"dependencies": {
|
||||||
|
"Humanizer.Core": "2.14.1",
|
||||||
|
"Json.More.Net": "2.1.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"KubernetesClient": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "18.0.13",
|
||||||
|
"contentHash": "X5IuxmydftB148XeULtc7rD5/RvqLuW5SzkIjFovPgJpvV4RAoRqNPruVB7GEFu1Xg+zHVIk88WqdV8JjbgHbA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Fractions": "7.3.0",
|
||||||
|
"YamlDotNet": "16.3.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"MessagePack": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.5.192",
|
||||||
|
"contentHash": "Jtle5MaFeIFkdXtxQeL9Tu2Y3HsAQGoSntOzrn6Br/jrl6c8QmG22GEioT5HBtZJR0zw0s46OnKU8ei2M3QifA==",
|
||||||
|
"dependencies": {
|
||||||
|
"MessagePack.Annotations": "2.5.192",
|
||||||
|
"Microsoft.NET.StringTools": "17.6.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"MessagePack.Annotations": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.5.192",
|
||||||
|
"contentHash": "jaJuwcgovWIZ8Zysdyf3b7b34/BrADw4v82GaEZymUhDd3ScMPrYd/cttekeDteJJPXseJxp04yTIcxiVUjTWg=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.AI.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.3.0",
|
||||||
|
"contentHash": "hDjDvUERvUH3HBMs2MDusOcGJBjAHOG5pJIU2x/HZEa4e1UthNKt89cwMi3B+ogJo6skki1XFjfgGN3ksnVqvQ=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Caching.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.3",
|
||||||
|
"contentHash": "5dtXBvI8t3z8pF4tB38JYgi/enCL/DwSXxpqShgFz3SHJ7IzqFIMs6Gu5ik8sNZzcO9qQs3xIDpB3vDamkYG+Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "8Rx5sqg04FttxrumyG6bmoRuFRgYzK6IVwF1i0/o0cXfKBdDeVpJejKHtJCMjyg9E/DNMVqpqOGe/tCT5gYvVA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "P09QpTHjqHmCLQOTC+WyLkoRNxek4NIvfWt+TnU0etoDUSRxcltyd6+j/ouRbMdLR0j44GqGO+lhI2M4fAHG4g==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "99Z4rjyXopb1MIazDSPcvwYCUdYNO01Cf1GUs2WUjIFAbkGmwzj2vPa2k+3pheJRV+YgNd2QqRKHAri0oBAU4Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.CommandLine": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "or9fOLopMUTJOQVJ3bou4aD6PwvsiKf4kZC4EE5sRRKSkmh+wfk/LekJXRjAX88X+1JA9zHjDo+5fiQ7z3MY/A==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.EnvironmentVariables": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "tchMGQ+zVTO40np/Zzg2Li/TIR8bksQgg4UVXZa0OzeFCKWnIYtxE2FVs+eSmjPGCjMS2voZbwN/mUcYfpSTuA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.FileExtensions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "OhTr0O79dP49734lLTqVveivVX9sDXxbI/8vjELAZTHXqoN90mdpgTAgwicJED42iaHMCcZcK6Bj+8wNyBikaw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Physical": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.Json": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "brBM/WP0YAUYh2+QqSYVdK8eQHYQTtTEUJXJ+84Zkdo2buGLja9VSrMIhgoeBUU7JBmcskAib8Lb/N83bvxgYQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.FileExtensions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.UserSecrets": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "fhdG6UV9lIp70QhNkVyaHciUVq25IPFkczheVJL9bIFvmnJ+Zghaie6dWkDbbVmxZlHl9gj3zTDxMxJs5zNhIA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Json": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Physical": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.DependencyInjection": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "v1SVsowG6YE1YnHVGmLWz57YTRCQRx9pH5ebIESXfm5isI9gA3QaMyg/oMTzPpXYZwSAVDzYItGJKfmV+pqXkQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "iVMtq9eRvzyhx8949EGT0OCYJfXi737SbRVzWXE5GrOgGj5AaZ9eUuxA/BSUfmOMALKn/g8KfFaNQw0eiB3lyA=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "vAJHd4yOpmKoK+jBuYV7a3y+Ab9U4ARCc29b6qvMy276RgJFw9LFs0DdsPqOL3ahwzyrX7tM+i4cCxU/RX0qAg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "/nYGrpa9/0BZofrVpBbbj+Ns8ZesiPE0V/KxsuHgDgHQopIzN54nRaQGSuvPw16/kI9sW1Zox5yyAPqvf0Jz6A==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "REdt95QXHscGdtw/UUgyCW2lF9DJcAOJxmebKW2IkgUjuCAdMODIi2HNOWg5utW98nm8ekgV0Gjqs/sljwwqMw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "NrIMTy7dpqxAvA6kHAYH8cXID/YgeNOy0OqFKpLtkPu5X4WS/basX91UszANzVrMNRAICJ2GOnGiRxJtsRyEQw=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "nCBmCx0Xemlu65ZiWMcXbvfvtznKxf4/YYKF9R28QkqdI9lTikedGqzJ28/xmdGGsxUnsP5/3TQGpiPwVjK0dA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.FileProviders.Physical": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "dMu5kUPSfol1Rqhmr6nWPSmbFjDe9w6bkoKithG17bWTZA0UyKirTatM5mqYUN3mGpNA0MorlusIoVTh6J7o5g==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileSystemGlobbing": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.FileSystemGlobbing": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "mOE3ARusNQR0a5x8YOcnUbfyyXGqoAWQtEc7qFOfNJgruDWQLo39Re+3/Lzj5pLPFuFYj8hN4dgKzaSQDKiOCw=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Hosting": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "8i7e5IBdiKLNqt/+ciWrS8U95Rv5DClaaj7ulkZbimnCi4uREWd+lXzkp3joofFuIPOlAzV4AckxLTIELv2jdg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.CommandLine": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.EnvironmentVariables": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.FileExtensions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Json": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.UserSecrets": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Physical": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Console": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Debug": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.EventLog": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.EventSource": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "+Wb7KAMVZTomwJkQrjuPTe5KBzGod7N8XeG+ScxRlkPOB4sZLG4ccVwjV4Phk5BCJt7uIMnGHVoN6ZMVploX+g==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Http": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "AiFvHYM8nP0wPC7bGPI3NHQlSYSLqjjT7DMJUuuxhd+7pz3O89iu2gdQfgACy5DxsXENiok5i1bMacJL7KR8jA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "+XTMKQyDWg4ODoNHU/BN3BaI1jhGO7VCS+BnzT/4IauiG6y2iPAte7MyD7rHKS+hNP0TkFkjrae8DFjDUxtcxg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "9HOdqlDtPptVcmKAjsQ/Nr5Rxfq6FMYLdhvZh1lVmeKR738qeYecQD7+ldooXf+u2KzzR1kafSphWngIM3C6ug==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.Configuration": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "cSgxsDgfP0+gmVRPVoNHI/KIDavIZxh+CxE6tSLPlYTogqccDnjBFI9CgEsiNuMP6+fiuXUwhhlTz36uUEpwbQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.Console": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "PMs2gha2v24hvH5o5KQem5aNK4mN0BhhCWlMqsg9tzifWKzjeQi2tyPOP/RaWMVvalOhVLcrmoMYPqbnia/epg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.Debug": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "/VacEkBQ02A8PBXSa6YpbIXCuisYy6JJr62/+ANJDZE+RMBfZMcXJXLfr/LpyLE6pgdp17Wxlt7e7R9zvkwZ3Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.EventLog": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "0ezhWYJS4/6KrqQel9JL+Tr4n+4EX2TF5EYiaysBWNNEM2c3Gtj1moD39esfgk8OHblSX+UFjtZ3z0c4i9tRvw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"System.Diagnostics.EventLog": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.EventSource": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "vN+aq1hBFXyYvY5Ow9WyeR66drKQxRZmas4lAjh6QWfryPkjTn1uLtX5AFIxyDaZj78v5TG2sELUyvrXpAPQQw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Options": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "MDaQMdUplw0AIRhWWmbLA7yQEXaLIHb+9CTroTiNS8OlI0LMXS4LCxtopqauiqGCWlRgJ+xyraVD8t6veRAFbw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "BB9uUW3+6Rxu1R97OB1H/13lUF8P2+H1+eDhpZlK30kDh/6E4EKHBUqTp+ilXQmZLzsRErxON8aBSR6WpUKJdg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Primitives": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "/HUHJ0tw/LQvD0DZrz50eQy/3z7PfX7WWEaXnjKTV9/TNdcgFlNTZGo49QhS7PTmhDqMyHRMqAXSBxLh0vso4g=="
|
||||||
|
},
|
||||||
|
"Microsoft.NET.StringTools": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "17.6.3",
|
||||||
|
"contentHash": "N0ZIanl1QCgvUumEL1laasU0a7sOE5ZwLZVTn0pAePnfhq8P7SvTjF8Axq+CnavuQkmdQpGNXQ1efZtu5kDFbA=="
|
||||||
|
},
|
||||||
|
"Microsoft.VisualStudio.Threading.Only": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "17.13.61",
|
||||||
|
"contentHash": "vl5a2URJYCO5m+aZZtNlAXAMz28e2pUotRuoHD7RnCWOCeoyd8hWp5ZBaLNYq4iEj2oeJx5ZxiSboAjVmB20Qg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.VisualStudio.Validation": "17.8.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.VisualStudio.Validation": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "17.8.8",
|
||||||
|
"contentHash": "rWXThIpyQd4YIXghNkiv2+VLvzS+MCMKVRDR0GAMlflsdo+YcAN2g2r5U1Ah98OFjQMRexTFtXQQ2LkajxZi3g=="
|
||||||
|
},
|
||||||
|
"ModelContextProtocol": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.0.0",
|
||||||
|
"contentHash": "W7UX8AQ1qMjXyCDcpP25u/L1W2vIIgfhLX/B2ZtTU1VUyILXdmVbdRjkQesKVPT/wPMpYXIHUcZJTPdsGfKSfQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Caching.Abstractions": "10.0.3",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.3",
|
||||||
|
"ModelContextProtocol.Core": "1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"ModelContextProtocol.Core": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.0.0",
|
||||||
|
"contentHash": "QKboiQEq2MJMGeQ029Gy6xqge88abm0Px9lnG7hueOyf+EDCxi5SUATV+Df7GwT+NwWzkEsYG271bUQD+LGhEg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.AI.Abstractions": "10.3.0",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Nerdbank.Streams": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.12.87",
|
||||||
|
"contentHash": "oDKOeKZ865I5X8qmU3IXMyrAnssYEiYWTobPGdrqubN3RtTzEHIv+D6fwhdcfrdhPJzHjCkK/ORztR/IsnmA6g==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.VisualStudio.Threading.Only": "17.13.61",
|
||||||
|
"Microsoft.VisualStudio.Validation": "17.8.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Newtonsoft.Json": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "13.0.4",
|
||||||
|
"contentHash": "pdgNNMai3zv51W5aq268sujXUyx7SNdE2bj1wZcWjAQrKMFZV260lbqYop1d2GM67JI1huLRwxo9ZqnfF/lC6A=="
|
||||||
|
},
|
||||||
|
"Npgsql": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.0.3",
|
||||||
|
"contentHash": "6WEmzsQJCZAlUG1pThKg/RmeF6V+I0DmBBBE/8YzpRtEzhyZzKcK7ulMANDm5CkxrALBEC8H+5plxHWtIL7xnA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "8.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Polly.Core": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.6.5",
|
||||||
|
"contentHash": "t+sUVrIwvo7UmsgHGgOG9F0GDZSRIm47u2ylH17Gvcv1q5hNEwgD5GoBlFyc0kh/pebmPyrAgvGsR/65ZBaXlg=="
|
||||||
|
},
|
||||||
|
"Semver": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "3.0.0",
|
||||||
|
"contentHash": "9jZCicsVgTebqkAujRWtC9J1A5EQVlu0TVKHcgoCuv345ve5DYf4D1MjhKEnQjdRZo6x/vdv6QQrYFs7ilGzLA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Primitives": "5.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"StreamJsonRpc": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "2.22.23",
|
||||||
|
"contentHash": "Ahq6uUFPnU9alny5h4agyX74th3PRq3NQCRNaDOqWcx20WT06mH/wENSk5IbHDc8BmfreQVEIBx5IXLBbsLFIA==",
|
||||||
|
"dependencies": {
|
||||||
|
"MessagePack": "2.5.192",
|
||||||
|
"Microsoft.VisualStudio.Threading.Only": "17.13.61",
|
||||||
|
"Microsoft.VisualStudio.Validation": "17.8.8",
|
||||||
|
"Nerdbank.Streams": "2.12.87",
|
||||||
|
"Newtonsoft.Json": "13.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"System.Diagnostics.EventLog": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "wugvy+pBVzjQEnRs9wMTWwoaeNFX3hsaHeVHFDIvJSWXp7wfmNWu3mxAwBIE6pyW+g6+rHa1Of5fTzb0QVqUTA=="
|
||||||
|
},
|
||||||
|
"System.IO.Hashing": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.3",
|
||||||
|
"contentHash": "La6ICwsdTKhVX+LKN+pvFjQRR3LhLwq3uKdi2knjLzRyPYBSydF4cjXidYxIiTcDD6XVYdsBWQEI8ZxiZ/OdIg=="
|
||||||
|
},
|
||||||
|
"YamlDotNet": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "16.3.0",
|
||||||
|
"contentHash": "SgMOdxbz8X65z8hraIs6hOEdnkH6hESTAIUa7viEngHOYaH+6q5XJmwr1+yb9vJpNQ19hCQY69xbFsLtXpobQA=="
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -30,8 +30,16 @@ RUN dotnet publish "GmRelay.Bot.csproj" -c Release -a $TARGETARCH -o /app/publis
|
|||||||
FROM mcr.microsoft.com/dotnet/runtime-deps:10.0-noble AS final
|
FROM mcr.microsoft.com/dotnet/runtime-deps:10.0-noble AS final
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
|
# Устанавливаем wget для healthcheck
|
||||||
|
RUN apt-get update && apt-get install -y --no-install-recommends wget \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
# Копируем только AOT-результаты из билда
|
# Копируем только AOT-результаты из билда
|
||||||
COPY --from=build /app/publish .
|
COPY --from=build /app/publish .
|
||||||
|
|
||||||
|
EXPOSE 8081
|
||||||
|
|
||||||
|
USER $APP_UID
|
||||||
|
|
||||||
# Запуск скомпилированного AOT бинарного файла напрямую
|
# Запуск скомпилированного AOT бинарного файла напрямую
|
||||||
ENTRYPOINT ["./GmRelay.Bot"]
|
ENTRYPOINT ["./GmRelay.Bot"]
|
||||||
|
|||||||
@@ -1,14 +1,11 @@
|
|||||||
using Dapper;
|
using Dapper;
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
using GmRelay.Bot.Features.Confirmation.SendConfirmation;
|
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types.ReplyMarkups;
|
using Telegram.Bot.Types.ReplyMarkups;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Confirmation.HandleRsvp;
|
namespace GmRelay.Bot.Features.Confirmation.HandleRsvp;
|
||||||
|
|
||||||
// ── Command ──────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
public sealed record HandleRsvpCommand(
|
public sealed record HandleRsvpCommand(
|
||||||
Guid SessionId,
|
Guid SessionId,
|
||||||
long TelegramUserId,
|
long TelegramUserId,
|
||||||
@@ -17,15 +14,15 @@ public sealed record HandleRsvpCommand(
|
|||||||
long ChatId,
|
long ChatId,
|
||||||
int MessageId);
|
int MessageId);
|
||||||
|
|
||||||
// ── DTOs ─────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
internal sealed record RsvpCounts(int Total, int Confirmed, int Declined);
|
internal sealed record RsvpCounts(int Total, int Confirmed, int Declined);
|
||||||
|
|
||||||
internal sealed record SessionContext(
|
internal sealed record SessionContext(
|
||||||
string Title,
|
string Title,
|
||||||
DateTime ScheduledAt,
|
DateTime ScheduledAt,
|
||||||
|
string Status,
|
||||||
long GmTelegramId,
|
long GmTelegramId,
|
||||||
long TelegramChatId);
|
long TelegramChatId,
|
||||||
|
int? ThreadId);
|
||||||
|
|
||||||
internal sealed record ParticipantRsvp(
|
internal sealed record ParticipantRsvp(
|
||||||
long TelegramId,
|
long TelegramId,
|
||||||
@@ -33,21 +30,6 @@ internal sealed record ParticipantRsvp(
|
|||||||
string? TelegramUsername,
|
string? TelegramUsername,
|
||||||
string RsvpStatus);
|
string RsvpStatus);
|
||||||
|
|
||||||
// ── Handler ──────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Handles the "Буду" / "Не смогу" callback query.
|
|
||||||
///
|
|
||||||
/// Flow:
|
|
||||||
/// 1. Validate that the user is a participant in this session
|
|
||||||
/// 2. Record or update their RSVP (idempotent)
|
|
||||||
/// 3. If declined → alert GM privately, revert session if was Confirmed
|
|
||||||
/// 4. If all non-GM players confirmed → mark session Confirmed, notify group + GM
|
|
||||||
/// 5. Update the inline keyboard to show current RSVP status
|
|
||||||
///
|
|
||||||
/// Concurrency: two simultaneous clicks on different rows don't conflict (MVCC).
|
|
||||||
/// The last EditMessage wins, which is fine — both reflect up-to-date state.
|
|
||||||
/// </summary>
|
|
||||||
public sealed class HandleRsvpHandler(
|
public sealed class HandleRsvpHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
ITelegramBotClient bot,
|
ITelegramBotClient bot,
|
||||||
@@ -58,19 +40,19 @@ public sealed class HandleRsvpHandler(
|
|||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
await using var transaction = await connection.BeginTransactionAsync(ct);
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
|
||||||
// ── 1. Validate participant ──────────────────────────────────
|
|
||||||
|
|
||||||
var participantExists = await connection.ExecuteScalarAsync<bool>(
|
var participantExists = await connection.ExecuteScalarAsync<bool>(
|
||||||
"""
|
"""
|
||||||
SELECT EXISTS (
|
SELECT EXISTS (
|
||||||
SELECT 1 FROM session_participants sp
|
SELECT 1
|
||||||
|
FROM session_participants sp
|
||||||
JOIN players p ON p.id = sp.player_id
|
JOIN players p ON p.id = sp.player_id
|
||||||
WHERE sp.session_id = @SessionId
|
WHERE sp.session_id = @SessionId
|
||||||
AND p.telegram_id = @TelegramUserId
|
AND p.telegram_id = @TelegramUserId
|
||||||
AND sp.is_gm = false
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
)
|
)
|
||||||
""",
|
""",
|
||||||
new { command.SessionId, command.TelegramUserId },
|
new { command.SessionId, command.TelegramUserId, Active = ParticipantRegistrationStatus.Active },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
if (!participantExists)
|
if (!participantExists)
|
||||||
@@ -82,8 +64,6 @@ public sealed class HandleRsvpHandler(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── 2. Record RSVP (idempotent) ─────────────────────────────
|
|
||||||
|
|
||||||
var updated = await connection.ExecuteAsync(
|
var updated = await connection.ExecuteAsync(
|
||||||
"""
|
"""
|
||||||
UPDATE session_participants
|
UPDATE session_participants
|
||||||
@@ -91,14 +71,14 @@ public sealed class HandleRsvpHandler(
|
|||||||
responded_at = now()
|
responded_at = now()
|
||||||
WHERE session_id = @SessionId
|
WHERE session_id = @SessionId
|
||||||
AND player_id = (SELECT id FROM players WHERE telegram_id = @TelegramUserId)
|
AND player_id = (SELECT id FROM players WHERE telegram_id = @TelegramUserId)
|
||||||
|
AND registration_status = @Active
|
||||||
AND rsvp_status != @Status
|
AND rsvp_status != @Status
|
||||||
""",
|
""",
|
||||||
new { command.SessionId, command.TelegramUserId, command.Status },
|
new { command.SessionId, command.TelegramUserId, command.Status, Active = ParticipantRegistrationStatus.Active },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
if (updated == 0)
|
if (updated == 0)
|
||||||
{
|
{
|
||||||
// Already in this state — just dismiss the loading spinner
|
|
||||||
var alreadyText = command.Status == RsvpStatus.Confirmed
|
var alreadyText = command.Status == RsvpStatus.Confirmed
|
||||||
? "Вы уже подтвердили участие."
|
? "Вы уже подтвердили участие."
|
||||||
: "Вы уже отказались от участия.";
|
: "Вы уже отказались от участия.";
|
||||||
@@ -110,13 +90,14 @@ public sealed class HandleRsvpHandler(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── 3. Load session context ─────────────────────────────────
|
|
||||||
|
|
||||||
var session = await connection.QuerySingleAsync<SessionContext>(
|
var session = await connection.QuerySingleAsync<SessionContext>(
|
||||||
"""
|
"""
|
||||||
SELECT s.title, s.scheduled_at AS ScheduledAt,
|
SELECT s.title,
|
||||||
|
s.scheduled_at AS ScheduledAt,
|
||||||
|
s.status AS Status,
|
||||||
g.gm_telegram_id AS GmTelegramId,
|
g.gm_telegram_id AS GmTelegramId,
|
||||||
g.telegram_chat_id AS TelegramChatId
|
g.telegram_chat_id AS TelegramChatId,
|
||||||
|
s.thread_id AS ThreadId
|
||||||
FROM sessions s
|
FROM sessions s
|
||||||
JOIN game_groups g ON g.id = s.group_id
|
JOIN game_groups g ON g.id = s.group_id
|
||||||
WHERE s.id = @SessionId
|
WHERE s.id = @SessionId
|
||||||
@@ -124,26 +105,27 @@ public sealed class HandleRsvpHandler(
|
|||||||
new { command.SessionId },
|
new { command.SessionId },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
// ── 4. Handle decline ───────────────────────────────────────
|
|
||||||
|
|
||||||
if (command.Status == RsvpStatus.Declined)
|
if (command.Status == RsvpStatus.Declined)
|
||||||
{
|
{
|
||||||
// Revert session to ConfirmationSent if it was Confirmed
|
var decision = RsvpFlowRules.Evaluate(command.Status, session.Status, totalParticipants: 0, confirmedParticipants: 0);
|
||||||
await connection.ExecuteAsync(
|
|
||||||
"""
|
if (decision.ShouldRevertSessionToConfirmationSent)
|
||||||
UPDATE sessions
|
{
|
||||||
SET status = @ConfirmationSent, updated_at = now()
|
await connection.ExecuteAsync(
|
||||||
WHERE id = @SessionId AND status = @Confirmed
|
"""
|
||||||
""",
|
UPDATE sessions
|
||||||
new
|
SET status = @ConfirmationSent, updated_at = now()
|
||||||
{
|
WHERE id = @SessionId AND status = @Confirmed
|
||||||
command.SessionId,
|
""",
|
||||||
ConfirmationSent = SessionStatus.ConfirmationSent,
|
new
|
||||||
Confirmed = SessionStatus.Confirmed
|
{
|
||||||
},
|
command.SessionId,
|
||||||
transaction);
|
ConfirmationSent = SessionStatus.ConfirmationSent,
|
||||||
|
Confirmed = SessionStatus.Confirmed
|
||||||
|
},
|
||||||
|
transaction);
|
||||||
|
}
|
||||||
|
|
||||||
// Alert GM immediately via private message
|
|
||||||
var declinedPlayer = await connection.QuerySingleAsync<string>(
|
var declinedPlayer = await connection.QuerySingleAsync<string>(
|
||||||
"SELECT display_name FROM players WHERE telegram_id = @TelegramUserId",
|
"SELECT display_name FROM players WHERE telegram_id = @TelegramUserId",
|
||||||
new { command.TelegramUserId },
|
new { command.TelegramUserId },
|
||||||
@@ -151,7 +133,6 @@ public sealed class HandleRsvpHandler(
|
|||||||
|
|
||||||
await transaction.CommitAsync(ct);
|
await transaction.CommitAsync(ct);
|
||||||
|
|
||||||
// Send alert outside transaction (network call)
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await bot.SendMessage(
|
await bot.SendMessage(
|
||||||
@@ -161,38 +142,38 @@ public sealed class HandleRsvpHandler(
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
logger.LogWarning(ex, "Failed to send decline alert to GM for session {SessionId}",
|
logger.LogWarning(ex, "Failed to send decline alert to GM for session {SessionId}", command.SessionId);
|
||||||
command.SessionId);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await bot.AnswerCallbackQuery(
|
await bot.AnswerCallbackQuery(
|
||||||
callbackQueryId: command.CallbackQueryId,
|
callbackQueryId: command.CallbackQueryId,
|
||||||
text: "Вы отказались от участия.",
|
text: decision.CallbackText,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
}
|
}
|
||||||
// ── 5. Handle confirm — check if ALL confirmed ──────────────
|
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
var counts = await connection.QuerySingleAsync<RsvpCounts>(
|
var counts = await connection.QuerySingleAsync<RsvpCounts>(
|
||||||
"""
|
"""
|
||||||
SELECT
|
SELECT
|
||||||
count(*) AS Total,
|
count(*) AS Total,
|
||||||
count(*) FILTER (WHERE rsvp_status = @Confirmed) AS Confirmed,
|
count(*) FILTER (WHERE rsvp_status = @Confirmed) AS Confirmed,
|
||||||
count(*) FILTER (WHERE rsvp_status = @Declined) AS Declined
|
count(*) FILTER (WHERE rsvp_status = @Declined) AS Declined
|
||||||
FROM session_participants
|
FROM session_participants
|
||||||
WHERE session_id = @SessionId AND is_gm = false
|
WHERE session_id = @SessionId AND is_gm = false
|
||||||
|
AND registration_status = @Active
|
||||||
""",
|
""",
|
||||||
new
|
new
|
||||||
{
|
{
|
||||||
command.SessionId,
|
command.SessionId,
|
||||||
Confirmed = RsvpStatus.Confirmed,
|
Confirmed = RsvpStatus.Confirmed,
|
||||||
Declined = RsvpStatus.Declined
|
Declined = RsvpStatus.Declined,
|
||||||
|
Active = ParticipantRegistrationStatus.Active
|
||||||
},
|
},
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
var allConfirmed = counts.Confirmed == counts.Total;
|
var decision = RsvpFlowRules.Evaluate(command.Status, session.Status, counts.Total, counts.Confirmed);
|
||||||
|
|
||||||
if (allConfirmed)
|
if (decision.ShouldMarkSessionConfirmed)
|
||||||
{
|
{
|
||||||
await connection.ExecuteAsync(
|
await connection.ExecuteAsync(
|
||||||
"""
|
"""
|
||||||
@@ -206,23 +187,24 @@ public sealed class HandleRsvpHandler(
|
|||||||
|
|
||||||
await transaction.CommitAsync(ct);
|
await transaction.CommitAsync(ct);
|
||||||
|
|
||||||
if (allConfirmed)
|
if (decision.ShouldNotifyGroup)
|
||||||
{
|
{
|
||||||
// Notify group
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await bot.SendMessage(
|
await bot.SendMessage(
|
||||||
chatId: session.TelegramChatId,
|
chatId: session.TelegramChatId,
|
||||||
|
messageThreadId: session.ThreadId,
|
||||||
text: $"🎉 Игра «{session.Title}» подтверждена! Все участники на месте.",
|
text: $"🎉 Игра «{session.Title}» подтверждена! Все участники на месте.",
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
logger.LogWarning(ex, "Failed to send group confirmation for session {SessionId}",
|
logger.LogWarning(ex, "Failed to send group confirmation for session {SessionId}", command.SessionId);
|
||||||
command.SessionId);
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Notify GM privately
|
if (decision.ShouldNotifyGm)
|
||||||
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await bot.SendMessage(
|
await bot.SendMessage(
|
||||||
@@ -232,27 +214,20 @@ public sealed class HandleRsvpHandler(
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
logger.LogWarning(ex, "Failed to send GM confirmation for session {SessionId}",
|
logger.LogWarning(ex, "Failed to send GM confirmation for session {SessionId}", command.SessionId);
|
||||||
command.SessionId);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await bot.AnswerCallbackQuery(
|
await bot.AnswerCallbackQuery(
|
||||||
callbackQueryId: command.CallbackQueryId,
|
callbackQueryId: command.CallbackQueryId,
|
||||||
text: "Вы подтвердили участие!",
|
text: decision.CallbackText,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── 6. Update inline keyboard message ───────────────────────
|
|
||||||
|
|
||||||
await UpdateConfirmationMessage(command, session, ct);
|
await UpdateConfirmationMessage(command, session, ct);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
private async Task UpdateConfirmationMessage(HandleRsvpCommand command, SessionContext session, CancellationToken ct)
|
||||||
/// Re-renders the confirmation message with current RSVP statuses.
|
|
||||||
/// </summary>
|
|
||||||
private async Task UpdateConfirmationMessage(
|
|
||||||
HandleRsvpCommand command, SessionContext session, CancellationToken ct)
|
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -260,16 +235,18 @@ public sealed class HandleRsvpHandler(
|
|||||||
|
|
||||||
var participants = (await connection.QueryAsync<ParticipantRsvp>(
|
var participants = (await connection.QueryAsync<ParticipantRsvp>(
|
||||||
"""
|
"""
|
||||||
SELECT p.telegram_id AS TelegramId,
|
SELECT p.telegram_id AS TelegramId,
|
||||||
p.display_name AS DisplayName,
|
p.display_name AS DisplayName,
|
||||||
p.telegram_username AS TelegramUsername,
|
p.telegram_username AS TelegramUsername,
|
||||||
sp.rsvp_status AS RsvpStatus
|
sp.rsvp_status AS RsvpStatus
|
||||||
FROM session_participants sp
|
FROM session_participants sp
|
||||||
JOIN players p ON p.id = sp.player_id
|
JOIN players p ON p.id = sp.player_id
|
||||||
WHERE sp.session_id = @SessionId AND sp.is_gm = false
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
ORDER BY sp.responded_at NULLS LAST
|
ORDER BY sp.responded_at NULLS LAST
|
||||||
""",
|
""",
|
||||||
new { command.SessionId })).ToList();
|
new { command.SessionId, Active = ParticipantRegistrationStatus.Active })).ToList();
|
||||||
|
|
||||||
var confirmed = participants.Where(p => p.RsvpStatus == RsvpStatus.Confirmed).ToList();
|
var confirmed = participants.Where(p => p.RsvpStatus == RsvpStatus.Confirmed).ToList();
|
||||||
var declined = participants.Where(p => p.RsvpStatus == RsvpStatus.Declined).ToList();
|
var declined = participants.Where(p => p.RsvpStatus == RsvpStatus.Declined).ToList();
|
||||||
@@ -279,34 +256,47 @@ public sealed class HandleRsvpHandler(
|
|||||||
{
|
{
|
||||||
$"🎲 Подтвердите участие в «{session.Title}»",
|
$"🎲 Подтвердите участие в «{session.Title}»",
|
||||||
$"📅 {session.ScheduledAt.FormatMoscow()} (МСК)",
|
$"📅 {session.ScheduledAt.FormatMoscow()} (МСК)",
|
||||||
""
|
string.Empty
|
||||||
};
|
};
|
||||||
|
|
||||||
foreach (var p in confirmed)
|
foreach (var participant in confirmed)
|
||||||
lines.Add($" ✅ {FormatName(p)}");
|
{
|
||||||
foreach (var p in declined)
|
lines.Add($" ✅ {FormatName(participant)}");
|
||||||
lines.Add($" ❌ ~~{FormatName(p)}~~");
|
}
|
||||||
foreach (var p in pending)
|
|
||||||
lines.Add($" ⏳ {FormatName(p)}");
|
|
||||||
|
|
||||||
lines.Add("");
|
foreach (var participant in declined)
|
||||||
|
{
|
||||||
|
lines.Add($" ❌ ~~{FormatName(participant)}~~");
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var participant in pending)
|
||||||
|
{
|
||||||
|
lines.Add($" ⏳ {FormatName(participant)}");
|
||||||
|
}
|
||||||
|
|
||||||
|
lines.Add(string.Empty);
|
||||||
|
|
||||||
if (confirmed.Count == participants.Count)
|
if (confirmed.Count == participants.Count)
|
||||||
|
{
|
||||||
lines.Add($"Статус: ✅ все подтвердили ({confirmed.Count}/{participants.Count})");
|
lines.Add($"Статус: ✅ все подтвердили ({confirmed.Count}/{participants.Count})");
|
||||||
|
}
|
||||||
else if (declined.Count > 0)
|
else if (declined.Count > 0)
|
||||||
|
{
|
||||||
lines.Add($"Статус: ⚠️ есть отказы ({confirmed.Count}/{participants.Count} подтвердили)");
|
lines.Add($"Статус: ⚠️ есть отказы ({confirmed.Count}/{participants.Count} подтвердили)");
|
||||||
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
lines.Add($"Статус: ожидаем подтверждения ({confirmed.Count}/{participants.Count})");
|
lines.Add($"Статус: ожидаем подтверждения ({confirmed.Count}/{participants.Count})");
|
||||||
|
}
|
||||||
|
|
||||||
var text = string.Join("\n", lines);
|
var text = string.Join("\n", lines);
|
||||||
|
|
||||||
// Keep buttons unless everyone confirmed
|
|
||||||
var replyMarkup = confirmed.Count == participants.Count
|
var replyMarkup = confirmed.Count == participants.Count
|
||||||
? null
|
? null
|
||||||
: new InlineKeyboardMarkup([
|
: new InlineKeyboardMarkup([
|
||||||
[
|
[
|
||||||
InlineKeyboardButton.WithCallbackData("\u2705 Буду", $"rsvp:confirm:{command.SessionId}"),
|
InlineKeyboardButton.WithCallbackData("✅ Буду", $"rsvp:confirm:{command.SessionId}"),
|
||||||
InlineKeyboardButton.WithCallbackData("\u274c Не смогу", $"rsvp:decline:{command.SessionId}")
|
InlineKeyboardButton.WithCallbackData("❌ Не смогу", $"rsvp:decline:{command.SessionId}")
|
||||||
]
|
]
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@@ -319,12 +309,10 @@ public sealed class HandleRsvpHandler(
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
// EditMessage can fail if message is too old or unchanged — non-critical
|
logger.LogWarning(ex, "Failed to update confirmation message for session {SessionId}", command.SessionId);
|
||||||
logger.LogWarning(ex, "Failed to update confirmation message for session {SessionId}",
|
|
||||||
command.SessionId);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static string FormatName(ParticipantRsvp p) =>
|
private static string FormatName(ParticipantRsvp participant) =>
|
||||||
p.TelegramUsername is not null ? $"@{p.TelegramUsername}" : p.DisplayName;
|
participant.TelegramUsername is not null ? $"@{participant.TelegramUsername}" : participant.DisplayName;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,42 @@
|
|||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Features.Confirmation.HandleRsvp;
|
||||||
|
|
||||||
|
internal sealed record RsvpFlowDecision(
|
||||||
|
string CallbackText,
|
||||||
|
bool ShouldAlertGm,
|
||||||
|
bool ShouldRevertSessionToConfirmationSent,
|
||||||
|
bool ShouldMarkSessionConfirmed,
|
||||||
|
bool ShouldNotifyGroup,
|
||||||
|
bool ShouldNotifyGm);
|
||||||
|
|
||||||
|
internal static class RsvpFlowRules
|
||||||
|
{
|
||||||
|
public static RsvpFlowDecision Evaluate(
|
||||||
|
string requestedStatus,
|
||||||
|
string currentSessionStatus,
|
||||||
|
int totalParticipants,
|
||||||
|
int confirmedParticipants)
|
||||||
|
{
|
||||||
|
if (requestedStatus == RsvpStatus.Declined)
|
||||||
|
{
|
||||||
|
return new RsvpFlowDecision(
|
||||||
|
CallbackText: "\u0412\u044b \u043e\u0442\u043a\u0430\u0437\u0430\u043b\u0438\u0441\u044c \u043e\u0442 \u0443\u0447\u0430\u0441\u0442\u0438\u044f.",
|
||||||
|
ShouldAlertGm: true,
|
||||||
|
ShouldRevertSessionToConfirmationSent: currentSessionStatus == SessionStatus.Confirmed,
|
||||||
|
ShouldMarkSessionConfirmed: false,
|
||||||
|
ShouldNotifyGroup: false,
|
||||||
|
ShouldNotifyGm: false);
|
||||||
|
}
|
||||||
|
|
||||||
|
var everyoneConfirmed = confirmedParticipants == totalParticipants;
|
||||||
|
|
||||||
|
return new RsvpFlowDecision(
|
||||||
|
CallbackText: "\u0412\u044b \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0434\u0438\u043b\u0438 \u0443\u0447\u0430\u0441\u0442\u0438\u0435!",
|
||||||
|
ShouldAlertGm: false,
|
||||||
|
ShouldRevertSessionToConfirmationSent: false,
|
||||||
|
ShouldMarkSessionConfirmed: everyoneConfirmed,
|
||||||
|
ShouldNotifyGroup: everyoneConfirmed,
|
||||||
|
ShouldNotifyGm: everyoneConfirmed);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
namespace GmRelay.Bot.Features.Confirmation.SendConfirmation;
|
||||||
|
|
||||||
|
public interface ISendConfirmationHandler
|
||||||
|
{
|
||||||
|
Task HandleAsync(Guid sessionId, CancellationToken ct);
|
||||||
|
}
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
using Dapper;
|
using Dapper;
|
||||||
|
using GmRelay.Bot.Features.Notifications;
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
@@ -13,7 +14,9 @@ internal sealed record SessionInfo(
|
|||||||
string Title,
|
string Title,
|
||||||
DateTime ScheduledAt,
|
DateTime ScheduledAt,
|
||||||
Guid GroupId,
|
Guid GroupId,
|
||||||
long TelegramChatId);
|
long TelegramChatId,
|
||||||
|
int? ThreadId,
|
||||||
|
string NotificationMode);
|
||||||
|
|
||||||
internal sealed record ParticipantInfo(
|
internal sealed record ParticipantInfo(
|
||||||
long TelegramId,
|
long TelegramId,
|
||||||
@@ -29,7 +32,8 @@ internal sealed record ParticipantInfo(
|
|||||||
public sealed class SendConfirmationHandler(
|
public sealed class SendConfirmationHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
ITelegramBotClient bot,
|
ITelegramBotClient bot,
|
||||||
ILogger<SendConfirmationHandler> logger)
|
DirectSessionNotificationSender directSender,
|
||||||
|
ILogger<SendConfirmationHandler> logger) : ISendConfirmationHandler
|
||||||
{
|
{
|
||||||
public async Task HandleAsync(Guid sessionId, CancellationToken ct)
|
public async Task HandleAsync(Guid sessionId, CancellationToken ct)
|
||||||
{
|
{
|
||||||
@@ -39,7 +43,9 @@ public sealed class SendConfirmationHandler(
|
|||||||
var session = await connection.QuerySingleOrDefaultAsync<SessionInfo>(
|
var session = await connection.QuerySingleOrDefaultAsync<SessionInfo>(
|
||||||
"""
|
"""
|
||||||
SELECT s.id, s.title, s.scheduled_at AS ScheduledAt, s.group_id AS GroupId,
|
SELECT s.id, s.title, s.scheduled_at AS ScheduledAt, s.group_id AS GroupId,
|
||||||
g.telegram_chat_id AS TelegramChatId
|
g.telegram_chat_id AS TelegramChatId,
|
||||||
|
s.thread_id AS ThreadId,
|
||||||
|
s.notification_mode AS NotificationMode
|
||||||
FROM sessions s
|
FROM sessions s
|
||||||
JOIN game_groups g ON g.id = s.group_id
|
JOIN game_groups g ON g.id = s.group_id
|
||||||
WHERE s.id = @SessionId AND s.status = @Planned
|
WHERE s.id = @SessionId AND s.status = @Planned
|
||||||
@@ -60,9 +66,11 @@ public sealed class SendConfirmationHandler(
|
|||||||
p.telegram_username AS TelegramUsername
|
p.telegram_username AS TelegramUsername
|
||||||
FROM session_participants sp
|
FROM session_participants sp
|
||||||
JOIN players p ON p.id = sp.player_id
|
JOIN players p ON p.id = sp.player_id
|
||||||
WHERE sp.session_id = @SessionId AND sp.is_gm = false
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
""",
|
""",
|
||||||
new { SessionId = sessionId })).ToList();
|
new { SessionId = sessionId, Active = ParticipantRegistrationStatus.Active })).ToList();
|
||||||
|
|
||||||
if (participants.Count == 0)
|
if (participants.Count == 0)
|
||||||
{
|
{
|
||||||
@@ -93,18 +101,21 @@ public sealed class SendConfirmationHandler(
|
|||||||
// 4. Send to group
|
// 4. Send to group
|
||||||
var message = await bot.SendMessage(
|
var message = await bot.SendMessage(
|
||||||
chatId: session.TelegramChatId,
|
chatId: session.TelegramChatId,
|
||||||
|
messageThreadId: session.ThreadId,
|
||||||
text: text,
|
text: text,
|
||||||
replyMarkup: keyboard,
|
replyMarkup: keyboard,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
|
|
||||||
// 5. Update session status and store message ID
|
// 5. Update session status, store message ID, and mark confirmation sent
|
||||||
await connection.ExecuteAsync(
|
await connection.ExecuteAsync(
|
||||||
"""
|
"""
|
||||||
UPDATE sessions
|
UPDATE sessions
|
||||||
SET status = @Status,
|
SET status = @Status,
|
||||||
confirmation_message_id = @MessageId,
|
confirmation_message_id = @MessageId,
|
||||||
|
confirmation_sent_at = now(),
|
||||||
updated_at = now()
|
updated_at = now()
|
||||||
WHERE id = @SessionId
|
WHERE id = @SessionId
|
||||||
|
AND confirmation_sent_at IS NULL
|
||||||
""",
|
""",
|
||||||
new
|
new
|
||||||
{
|
{
|
||||||
@@ -113,6 +124,26 @@ public sealed class SendConfirmationHandler(
|
|||||||
MessageId = message.MessageId
|
MessageId = message.MessageId
|
||||||
});
|
});
|
||||||
|
|
||||||
|
var mode = SessionNotificationModeExtensions.FromDatabaseValue(session.NotificationMode);
|
||||||
|
if (mode.ShouldSendDirectMessages())
|
||||||
|
{
|
||||||
|
var directText = $"""
|
||||||
|
🎲 <b>Подтвердите участие в игре</b>
|
||||||
|
|
||||||
|
📌 <b>{System.Net.WebUtility.HtmlEncode(session.Title)}</b>
|
||||||
|
📅 {session.ScheduledAt.FormatMoscow()} (МСК)
|
||||||
|
|
||||||
|
Ответьте кнопкой в групповом сообщении расписания.
|
||||||
|
""";
|
||||||
|
|
||||||
|
await directSender.SendAsync(
|
||||||
|
participants.Select(p => new DirectNotificationRecipient(p.TelegramId, p.DisplayName)),
|
||||||
|
directText,
|
||||||
|
"confirmation",
|
||||||
|
sessionId,
|
||||||
|
ct);
|
||||||
|
}
|
||||||
|
|
||||||
logger.LogInformation(
|
logger.LogInformation(
|
||||||
"Confirmation sent for session {SessionId} ({Title}), message_id={MessageId}",
|
"Confirmation sent for session {SessionId} ({Title}), message_id={MessageId}",
|
||||||
sessionId, session.Title, message.MessageId);
|
sessionId, session.Title, message.MessageId);
|
||||||
|
|||||||
@@ -0,0 +1,41 @@
|
|||||||
|
using Telegram.Bot;
|
||||||
|
using Telegram.Bot.Types.Enums;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Features.Notifications;
|
||||||
|
|
||||||
|
public sealed record DirectNotificationRecipient(long TelegramId, string DisplayName);
|
||||||
|
|
||||||
|
public sealed class DirectSessionNotificationSender(
|
||||||
|
ITelegramBotClient bot,
|
||||||
|
ILogger<DirectSessionNotificationSender> logger)
|
||||||
|
{
|
||||||
|
public async Task SendAsync(
|
||||||
|
IEnumerable<DirectNotificationRecipient> recipients,
|
||||||
|
string htmlText,
|
||||||
|
string notificationKind,
|
||||||
|
Guid sessionId,
|
||||||
|
CancellationToken ct)
|
||||||
|
{
|
||||||
|
foreach (var recipient in recipients)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await bot.SendMessage(
|
||||||
|
chatId: recipient.TelegramId,
|
||||||
|
text: htmlText,
|
||||||
|
parseMode: ParseMode.Html,
|
||||||
|
cancellationToken: ct);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogWarning(
|
||||||
|
ex,
|
||||||
|
"Failed to send {NotificationKind} DM for session {SessionId} to player {TelegramId} ({DisplayName})",
|
||||||
|
notificationKind,
|
||||||
|
sessionId,
|
||||||
|
recipient.TelegramId,
|
||||||
|
recipient.DisplayName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
namespace GmRelay.Bot.Features.Reminders.SendJoinLink;
|
||||||
|
|
||||||
|
public interface ISendJoinLinkHandler
|
||||||
|
{
|
||||||
|
Task HandleAsync(Guid sessionId, CancellationToken ct);
|
||||||
|
}
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
using Dapper;
|
using Dapper;
|
||||||
|
using GmRelay.Bot.Features.Notifications;
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
@@ -12,7 +13,9 @@ internal sealed record JoinLinkSession(
|
|||||||
string Title,
|
string Title,
|
||||||
string JoinLink,
|
string JoinLink,
|
||||||
DateTime ScheduledAt,
|
DateTime ScheduledAt,
|
||||||
long TelegramChatId);
|
long TelegramChatId,
|
||||||
|
int? ThreadId,
|
||||||
|
string NotificationMode);
|
||||||
|
|
||||||
internal sealed record ConfirmedPlayer(
|
internal sealed record ConfirmedPlayer(
|
||||||
long TelegramId,
|
long TelegramId,
|
||||||
@@ -28,7 +31,8 @@ internal sealed record ConfirmedPlayer(
|
|||||||
public sealed class SendJoinLinkHandler(
|
public sealed class SendJoinLinkHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
ITelegramBotClient bot,
|
ITelegramBotClient bot,
|
||||||
ILogger<SendJoinLinkHandler> logger)
|
DirectSessionNotificationSender directSender,
|
||||||
|
ILogger<SendJoinLinkHandler> logger) : ISendJoinLinkHandler
|
||||||
{
|
{
|
||||||
public async Task HandleAsync(Guid sessionId, CancellationToken ct)
|
public async Task HandleAsync(Guid sessionId, CancellationToken ct)
|
||||||
{
|
{
|
||||||
@@ -38,7 +42,9 @@ public sealed class SendJoinLinkHandler(
|
|||||||
var session = await connection.QuerySingleOrDefaultAsync<JoinLinkSession>(
|
var session = await connection.QuerySingleOrDefaultAsync<JoinLinkSession>(
|
||||||
"""
|
"""
|
||||||
SELECT s.id, s.title, s.join_link AS JoinLink, s.scheduled_at AS ScheduledAt,
|
SELECT s.id, s.title, s.join_link AS JoinLink, s.scheduled_at AS ScheduledAt,
|
||||||
g.telegram_chat_id AS TelegramChatId
|
g.telegram_chat_id AS TelegramChatId,
|
||||||
|
s.thread_id AS ThreadId,
|
||||||
|
s.notification_mode AS NotificationMode
|
||||||
FROM sessions s
|
FROM sessions s
|
||||||
JOIN game_groups g ON g.id = s.group_id
|
JOIN game_groups g ON g.id = s.group_id
|
||||||
WHERE s.id = @SessionId
|
WHERE s.id = @SessionId
|
||||||
@@ -63,8 +69,14 @@ public sealed class SendJoinLinkHandler(
|
|||||||
JOIN players p ON p.id = sp.player_id
|
JOIN players p ON p.id = sp.player_id
|
||||||
WHERE sp.session_id = @SessionId
|
WHERE sp.session_id = @SessionId
|
||||||
AND sp.rsvp_status = @Confirmed
|
AND sp.rsvp_status = @Confirmed
|
||||||
|
AND sp.registration_status = @Active
|
||||||
""",
|
""",
|
||||||
new { SessionId = sessionId, Confirmed = RsvpStatus.Confirmed })).ToList();
|
new
|
||||||
|
{
|
||||||
|
SessionId = sessionId,
|
||||||
|
Confirmed = RsvpStatus.Confirmed,
|
||||||
|
Active = ParticipantRegistrationStatus.Active
|
||||||
|
})).ToList();
|
||||||
|
|
||||||
// 3. Build message with player mentions
|
// 3. Build message with player mentions
|
||||||
var mentions = string.Join(", ", players.Select(p =>
|
var mentions = string.Join(", ", players.Select(p =>
|
||||||
@@ -84,6 +96,7 @@ public sealed class SendJoinLinkHandler(
|
|||||||
// 4. Send
|
// 4. Send
|
||||||
var message = await bot.SendMessage(
|
var message = await bot.SendMessage(
|
||||||
chatId: session.TelegramChatId,
|
chatId: session.TelegramChatId,
|
||||||
|
messageThreadId: session.ThreadId,
|
||||||
text: text,
|
text: text,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
|
|
||||||
@@ -96,6 +109,24 @@ public sealed class SendJoinLinkHandler(
|
|||||||
""",
|
""",
|
||||||
new { SessionId = sessionId, MessageId = message.MessageId });
|
new { SessionId = sessionId, MessageId = message.MessageId });
|
||||||
|
|
||||||
|
var mode = SessionNotificationModeExtensions.FromDatabaseValue(session.NotificationMode);
|
||||||
|
if (mode.ShouldSendDirectMessages())
|
||||||
|
{
|
||||||
|
var directText = $"""
|
||||||
|
🎮 <b>Игра начинается через 5 минут</b>
|
||||||
|
|
||||||
|
📌 <b>{System.Net.WebUtility.HtmlEncode(session.Title)}</b>
|
||||||
|
🔗 {System.Net.WebUtility.HtmlEncode(session.JoinLink)}
|
||||||
|
""";
|
||||||
|
|
||||||
|
await directSender.SendAsync(
|
||||||
|
players.Select(p => new DirectNotificationRecipient(p.TelegramId, p.DisplayName)),
|
||||||
|
directText,
|
||||||
|
"join-link",
|
||||||
|
sessionId,
|
||||||
|
ct);
|
||||||
|
}
|
||||||
|
|
||||||
logger.LogInformation(
|
logger.LogInformation(
|
||||||
"Join link sent for session {SessionId} ({Title}), message_id={MessageId}",
|
"Join link sent for session {SessionId} ({Title}), message_id={MessageId}",
|
||||||
sessionId, session.Title, message.MessageId);
|
sessionId, session.Title, message.MessageId);
|
||||||
|
|||||||
@@ -0,0 +1,6 @@
|
|||||||
|
namespace GmRelay.Bot.Features.Reminders.SendOneHourReminder;
|
||||||
|
|
||||||
|
public interface ISendOneHourReminderHandler
|
||||||
|
{
|
||||||
|
Task HandleAsync(Guid sessionId, CancellationToken ct);
|
||||||
|
}
|
||||||
@@ -0,0 +1,97 @@
|
|||||||
|
using Dapper;
|
||||||
|
using GmRelay.Bot.Features.Notifications;
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
using Npgsql;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Features.Reminders.SendOneHourReminder;
|
||||||
|
|
||||||
|
internal sealed record OneHourReminderSession(
|
||||||
|
Guid Id,
|
||||||
|
string Title,
|
||||||
|
string JoinLink,
|
||||||
|
DateTime ScheduledAt,
|
||||||
|
string NotificationMode);
|
||||||
|
|
||||||
|
public sealed class SendOneHourReminderHandler(
|
||||||
|
NpgsqlDataSource dataSource,
|
||||||
|
DirectSessionNotificationSender directSender,
|
||||||
|
ILogger<SendOneHourReminderHandler> logger) : ISendOneHourReminderHandler
|
||||||
|
{
|
||||||
|
public async Task HandleAsync(Guid sessionId, CancellationToken ct)
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
|
||||||
|
var session = await connection.QuerySingleOrDefaultAsync<OneHourReminderSession>(
|
||||||
|
"""
|
||||||
|
SELECT id,
|
||||||
|
title,
|
||||||
|
join_link AS JoinLink,
|
||||||
|
scheduled_at AS ScheduledAt,
|
||||||
|
notification_mode AS NotificationMode
|
||||||
|
FROM sessions
|
||||||
|
WHERE id = @SessionId
|
||||||
|
AND status IN (@Confirmed, @ConfirmationSent)
|
||||||
|
AND one_hour_reminder_processed_at IS NULL
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
SessionId = sessionId,
|
||||||
|
Confirmed = SessionStatus.Confirmed,
|
||||||
|
ConfirmationSent = SessionStatus.ConfirmationSent
|
||||||
|
});
|
||||||
|
|
||||||
|
if (session is null)
|
||||||
|
{
|
||||||
|
logger.LogWarning("Session {SessionId} not eligible for one-hour reminder", sessionId);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var recipients = (await connection.QueryAsync<DirectNotificationRecipient>(
|
||||||
|
"""
|
||||||
|
SELECT p.telegram_id AS TelegramId,
|
||||||
|
p.display_name AS DisplayName
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON p.id = sp.player_id
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
|
AND sp.rsvp_status != @Declined
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
SessionId = sessionId,
|
||||||
|
Active = ParticipantRegistrationStatus.Active,
|
||||||
|
Declined = RsvpStatus.Declined
|
||||||
|
})).ToList();
|
||||||
|
|
||||||
|
var mode = SessionNotificationModeExtensions.FromDatabaseValue(session.NotificationMode);
|
||||||
|
if (mode.ShouldSendDirectMessages() && recipients.Count > 0)
|
||||||
|
{
|
||||||
|
var text = $"""
|
||||||
|
⏰ <b>Игра начнётся примерно через 1 час</b>
|
||||||
|
|
||||||
|
📌 <b>{System.Net.WebUtility.HtmlEncode(session.Title)}</b>
|
||||||
|
📅 {session.ScheduledAt.FormatMoscow()} (МСК)
|
||||||
|
🔗 {System.Net.WebUtility.HtmlEncode(session.JoinLink)}
|
||||||
|
""";
|
||||||
|
|
||||||
|
await directSender.SendAsync(recipients, text, "one-hour-reminder", session.Id, ct);
|
||||||
|
}
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
UPDATE sessions
|
||||||
|
SET one_hour_reminder_processed_at = now(),
|
||||||
|
updated_at = now()
|
||||||
|
WHERE id = @SessionId
|
||||||
|
AND one_hour_reminder_processed_at IS NULL
|
||||||
|
""",
|
||||||
|
new { SessionId = sessionId });
|
||||||
|
|
||||||
|
logger.LogInformation(
|
||||||
|
"One-hour reminder processed for session {SessionId} ({Title}) with mode {NotificationMode}",
|
||||||
|
sessionId,
|
||||||
|
session.Title,
|
||||||
|
session.NotificationMode);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,9 +1,11 @@
|
|||||||
using Dapper;
|
using Dapper;
|
||||||
|
using GmRelay.Bot.Features.Notifications;
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
using GmRelay.Shared.Rendering;
|
using GmRelay.Shared.Rendering;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types;
|
using Telegram.Bot.Types;
|
||||||
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
||||||
|
|
||||||
@@ -12,28 +14,41 @@ public sealed record CancelSessionCommand(
|
|||||||
long TelegramUserId,
|
long TelegramUserId,
|
||||||
string CallbackQueryId,
|
string CallbackQueryId,
|
||||||
long ChatId,
|
long ChatId,
|
||||||
|
int? MessageThreadId,
|
||||||
int MessageId);
|
int MessageId);
|
||||||
|
|
||||||
// DTOs for AOT compilation
|
// DTOs for AOT compilation
|
||||||
internal sealed record CancelSessionInfoDto(string Title, Guid BatchId, long GmId);
|
internal sealed record CancelSessionInfoDto(string Title, Guid BatchId, int? BatchMessageId, bool CanManage, string NotificationMode);
|
||||||
|
|
||||||
public sealed class CancelSessionHandler(
|
public sealed class CancelSessionHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
ITelegramBotClient bot,
|
ITelegramBotClient bot,
|
||||||
|
DirectSessionNotificationSender directSender,
|
||||||
ILogger<CancelSessionHandler> logger)
|
ILogger<CancelSessionHandler> logger)
|
||||||
{
|
{
|
||||||
public async Task HandleAsync(CancelSessionCommand command, CancellationToken ct)
|
public async Task HandleAsync(CancelSessionCommand command, CancellationToken ct)
|
||||||
{
|
{
|
||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
await using var transaction = await connection.BeginTransactionAsync(ct);
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
|
||||||
// 1. Проверяем, что запрос делает ГМ данной сессии
|
// 1. Проверяем, что запрос делает управляющий данной группы.
|
||||||
var session = await connection.QuerySingleOrDefaultAsync<CancelSessionInfoDto>(
|
var session = await connection.QuerySingleOrDefaultAsync<CancelSessionInfoDto>(
|
||||||
@"SELECT s.title as Title, s.batch_id as BatchId, g.gm_telegram_id as GmId
|
"""
|
||||||
FROM sessions s
|
SELECT s.title AS Title,
|
||||||
JOIN game_groups g ON s.group_id = g.id
|
s.batch_id AS BatchId,
|
||||||
WHERE s.id = @SessionId",
|
s.batch_message_id AS BatchMessageId,
|
||||||
new { command.SessionId }, transaction);
|
s.notification_mode AS NotificationMode,
|
||||||
|
EXISTS (
|
||||||
|
SELECT 1
|
||||||
|
FROM group_managers gm
|
||||||
|
JOIN players p ON p.id = gm.player_id
|
||||||
|
WHERE gm.group_id = s.group_id
|
||||||
|
AND p.telegram_id = @TelegramUserId
|
||||||
|
) AS CanManage
|
||||||
|
FROM sessions s
|
||||||
|
WHERE s.id = @SessionId
|
||||||
|
""",
|
||||||
|
new { command.SessionId, command.TelegramUserId }, transaction);
|
||||||
|
|
||||||
if (session == null)
|
if (session == null)
|
||||||
{
|
{
|
||||||
@@ -41,48 +56,87 @@ public sealed class CancelSessionHandler(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (session.GmId != command.TelegramUserId)
|
if (!session.CanManage)
|
||||||
{
|
{
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Только Мастер Игры (GM) может отменять сессию.", showAlert: true, cancellationToken: ct);
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Только owner или co-GM может отменять сессию.", showAlert: true, cancellationToken: ct);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. Отменяем сессию
|
// 2. Отменяем сессию
|
||||||
await connection.ExecuteAsync("UPDATE sessions SET status = 'Cancelled' WHERE id = @Id", new { Id = command.SessionId }, transaction);
|
await connection.ExecuteAsync(
|
||||||
|
"UPDATE sessions SET status = @Status WHERE id = @Id",
|
||||||
|
new { Id = command.SessionId, Status = SessionStatus.Cancelled },
|
||||||
|
transaction);
|
||||||
|
|
||||||
// 3. Загружаем весь батч для перерисовки
|
// 3. Загружаем весь батч для перерисовки
|
||||||
var batchSessions = await connection.QueryAsync<SessionBatchDto>(
|
var batchSessions = await connection.QueryAsync<SessionBatchDto>(
|
||||||
@"SELECT id as SessionId, scheduled_at as ScheduledAt, status as Status FROM sessions WHERE batch_id = @BatchId ORDER BY scheduled_at",
|
@"SELECT id as SessionId, scheduled_at as ScheduledAt, status as Status, max_players as MaxPlayers, join_link as JoinLink
|
||||||
|
FROM sessions
|
||||||
|
WHERE batch_id = @BatchId
|
||||||
|
ORDER BY scheduled_at",
|
||||||
new { BatchId = session.BatchId }, transaction);
|
new { BatchId = session.BatchId }, transaction);
|
||||||
|
|
||||||
var batchParticipants = await connection.QueryAsync<ParticipantBatchDto>(
|
var batchParticipants = await connection.QueryAsync<ParticipantBatchDto>(
|
||||||
@"SELECT sp.session_id as SessionId, p.display_name as DisplayName, p.telegram_username as TelegramUsername
|
@"SELECT sp.session_id as SessionId,
|
||||||
|
p.display_name as DisplayName,
|
||||||
|
p.telegram_username as TelegramUsername,
|
||||||
|
sp.registration_status as RegistrationStatus
|
||||||
FROM session_participants sp
|
FROM session_participants sp
|
||||||
JOIN players p ON sp.player_id = p.id
|
JOIN players p ON sp.player_id = p.id
|
||||||
JOIN sessions s ON sp.session_id = s.id
|
JOIN sessions s ON sp.session_id = s.id
|
||||||
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
||||||
ORDER BY sp.responded_at ASC, p.created_at ASC",
|
ORDER BY sp.registration_status ASC, sp.created_at ASC, sp.responded_at ASC, p.created_at ASC",
|
||||||
new { BatchId = session.BatchId }, transaction);
|
new { BatchId = session.BatchId }, transaction);
|
||||||
|
|
||||||
|
var directRecipients = (await connection.QueryAsync<DirectNotificationRecipient>(
|
||||||
|
"""
|
||||||
|
SELECT p.telegram_id AS TelegramId,
|
||||||
|
p.display_name AS DisplayName
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON sp.player_id = p.id
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
|
""",
|
||||||
|
new { command.SessionId, Active = ParticipantRegistrationStatus.Active },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
await transaction.CommitAsync(ct);
|
await transaction.CommitAsync(ct);
|
||||||
|
|
||||||
// 4. Перерисовываем сообщение
|
// 4. Перерисовываем сообщение
|
||||||
var renderResult = SessionBatchRenderer.Render(session.Title, batchSessions.ToList(), batchParticipants.ToList());
|
var view = SessionBatchViewBuilder.Build(session.Title, batchSessions.ToList(), batchParticipants.ToList());
|
||||||
|
var renderResult = TelegramSessionBatchRenderer.Render(view);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await bot.EditMessageText(
|
await BatchMessageEditor.EditBatchMessageAsync(
|
||||||
|
bot,
|
||||||
chatId: command.ChatId,
|
chatId: command.ChatId,
|
||||||
messageId: command.MessageId,
|
messageId: session.BatchMessageId ?? command.MessageId,
|
||||||
text: renderResult.Text,
|
text: renderResult.Text,
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
replyMarkup: renderResult.Markup,
|
||||||
replyMarkup: renderResult.Markup,
|
ct);
|
||||||
cancellationToken: ct);
|
|
||||||
|
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Сессия отменена!", cancellationToken: ct);
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Сессия отменена!", cancellationToken: ct);
|
||||||
|
|
||||||
// Опционально: написать отдельное сообщение в чат
|
// Опционально: написать отдельное сообщение в чат
|
||||||
await bot.SendMessage(command.ChatId, $"❌ <b>Внимание!</b> Сессия \"{System.Net.WebUtility.HtmlEncode(session.Title)}\" отменена.", parseMode: Telegram.Bot.Types.Enums.ParseMode.Html, cancellationToken: ct);
|
await bot.SendMessage(
|
||||||
|
chatId: command.ChatId,
|
||||||
|
messageThreadId: command.MessageThreadId,
|
||||||
|
text: $"❌ <b>Внимание!</b> Сессия \"{System.Net.WebUtility.HtmlEncode(session.Title)}\" отменена.",
|
||||||
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
|
cancellationToken: ct);
|
||||||
|
|
||||||
|
var mode = SessionNotificationModeExtensions.FromDatabaseValue(session.NotificationMode);
|
||||||
|
if (mode.ShouldSendDirectMessages())
|
||||||
|
{
|
||||||
|
await directSender.SendAsync(
|
||||||
|
directRecipients,
|
||||||
|
$"❌ <b>Сессия отменена</b>\n\n📌 <b>{System.Net.WebUtility.HtmlEncode(session.Title)}</b>",
|
||||||
|
"session-cancelled",
|
||||||
|
command.SessionId,
|
||||||
|
ct);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,14 +1,15 @@
|
|||||||
using System.Text.RegularExpressions;
|
|
||||||
using Dapper;
|
using Dapper;
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
using GmRelay.Shared.Rendering;
|
using GmRelay.Shared.Rendering;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types;
|
using Telegram.Bot.Types;
|
||||||
using Telegram.Bot.Types.ReplyMarkups;
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
||||||
|
|
||||||
|
internal sealed record SessionCreationGroupAccessDto(Guid GroupId, bool CanManage);
|
||||||
|
|
||||||
public sealed class CreateSessionHandler(
|
public sealed class CreateSessionHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
ITelegramBotClient botClient,
|
ITelegramBotClient botClient,
|
||||||
@@ -16,49 +17,56 @@ public sealed class CreateSessionHandler(
|
|||||||
{
|
{
|
||||||
public async Task HandleAsync(Message message, CancellationToken cancellationToken)
|
public async Task HandleAsync(Message message, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var text = message.Text ?? "";
|
var parseResult = NewSessionCommandParser.Parse(message.Text ?? message.Caption, DateTimeOffset.UtcNow);
|
||||||
|
|
||||||
string? title = null;
|
|
||||||
string? link = null;
|
|
||||||
var scheduledTimes = new List<DateTimeOffset>();
|
|
||||||
|
|
||||||
foreach (var line in text.Split('\n'))
|
foreach (var timeInput in parseResult.PastTimeInputs)
|
||||||
{
|
{
|
||||||
var trimmed = line.Trim();
|
await botClient.SendMessage(
|
||||||
if (trimmed.StartsWith("Название:", StringComparison.OrdinalIgnoreCase))
|
message.Chat.Id,
|
||||||
title = trimmed["Название:".Length..].Trim();
|
$"⚠️ Предупреждение: дата {timeInput} находится в прошлом и будет пропущена.",
|
||||||
else if (trimmed.StartsWith("Ссылка:", StringComparison.OrdinalIgnoreCase))
|
cancellationToken: cancellationToken);
|
||||||
link = trimmed["Ссылка:".Length..].Trim();
|
|
||||||
else if (trimmed.StartsWith("Время:", StringComparison.OrdinalIgnoreCase))
|
|
||||||
{
|
|
||||||
var timeStr = trimmed["Время:".Length..].Trim();
|
|
||||||
if (MoscowTime.TryParseMoscow(timeStr, out var scheduledAt))
|
|
||||||
{
|
|
||||||
if (scheduledAt > DateTimeOffset.UtcNow)
|
|
||||||
scheduledTimes.Add(scheduledAt);
|
|
||||||
else
|
|
||||||
await botClient.SendMessage(message.Chat.Id, $"⚠️ Предупреждение: Дата {timeStr} находится в прошлом и будет пропущена.", cancellationToken: cancellationToken);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
await botClient.SendMessage(message.Chat.Id, $"⚠️ Предупреждение: Некорректный формат времени '{timeStr}'. Пропущено.", cancellationToken: cancellationToken);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(title) || string.IsNullOrEmpty(link) || scheduledTimes.Count == 0)
|
foreach (var timeInput in parseResult.InvalidTimeInputs)
|
||||||
|
{
|
||||||
|
await botClient.SendMessage(
|
||||||
|
message.Chat.Id,
|
||||||
|
$"⚠️ Предупреждение: некорректный формат времени '{timeInput}'. Пропущено.",
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var seatLimitInput in parseResult.InvalidSeatLimitInputs)
|
||||||
|
{
|
||||||
|
await botClient.SendMessage(
|
||||||
|
message.Chat.Id,
|
||||||
|
$"⚠️ Предупреждение: некорректный лимит мест '{seatLimitInput}'. Укажите целое число больше 0.",
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var recurringInput in parseResult.InvalidRecurringInputs)
|
||||||
|
{
|
||||||
|
await botClient.SendMessage(
|
||||||
|
message.Chat.Id,
|
||||||
|
$"⚠️ Предупреждение: некорректный повтор расписания '{recurringInput}'. Укажите число игр 1-52 и шаг 1-365 дней.",
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!parseResult.IsValid)
|
||||||
{
|
{
|
||||||
await botClient.SendMessage(
|
await botClient.SendMessage(
|
||||||
chatId: message.Chat.Id,
|
chatId: message.Chat.Id,
|
||||||
text: "❌ Не удалось распознать формат. Пожалуйста, используйте шаблон:\n\n/newsession\nНазвание: My Game\nВремя: 15.05.2026 19:30\nВремя: 22.05.2026 19:30\nСсылка: https://link",
|
text: "❌ Не удалось распознать формат. Пожалуйста, используйте шаблон:\n\n/newsession\nНазвание: My Game\nВремя: 15.05.2026 19:30\nВремя: 22.05.2026 19:30\nМест: 4\nСсылка: https://link\nКартинка: https://cover\n\nДля повтора можно указать одну дату и строки:\nИгр: 4\nИнтервал: 7",
|
||||||
cancellationToken: cancellationToken);
|
cancellationToken: cancellationToken);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var title = parseResult.Title!;
|
||||||
|
var link = parseResult.Link!;
|
||||||
|
var imageReference = GetBatchImageReference(message, parseResult.ImageUrl);
|
||||||
var gmId = message.From!.Id;
|
var gmId = message.From!.Id;
|
||||||
var gmName = message.From.FirstName + (string.IsNullOrEmpty(message.From.LastName) ? "" : $" {message.From.LastName}");
|
var gmName = message.From.FirstName + (string.IsNullOrEmpty(message.From.LastName) ? string.Empty : $" {message.From.LastName}");
|
||||||
var gmUsername = message.From.Username;
|
var gmUsername = message.From.Username;
|
||||||
|
|
||||||
var chatId = message.Chat.Id;
|
var chatId = message.Chat.Id;
|
||||||
var chatTitle = message.Chat.Title ?? "Private Chat";
|
var chatTitle = message.Chat.Title ?? "Private Chat";
|
||||||
|
|
||||||
@@ -67,70 +75,201 @@ public sealed class CreateSessionHandler(
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
// 1. Убеждаемся, что GM зарегистрирован
|
|
||||||
await connection.ExecuteAsync(
|
await connection.ExecuteAsync(
|
||||||
@"INSERT INTO players (telegram_id, display_name, telegram_username)
|
"""
|
||||||
VALUES (@TgId, @Name, @Username)
|
INSERT INTO players (telegram_id, display_name, telegram_username, platform, external_user_id, external_username)
|
||||||
ON CONFLICT (telegram_id) DO UPDATE SET display_name = EXCLUDED.display_name, telegram_username = EXCLUDED.telegram_username;",
|
VALUES (@TgId, @Name, @Username, 'Telegram', @TgId::TEXT, @Username)
|
||||||
|
ON CONFLICT (telegram_id) DO UPDATE
|
||||||
|
SET display_name = EXCLUDED.display_name,
|
||||||
|
telegram_username = EXCLUDED.telegram_username,
|
||||||
|
platform = COALESCE(players.platform, 'Telegram'),
|
||||||
|
external_user_id = COALESCE(players.external_user_id, EXCLUDED.telegram_id::TEXT),
|
||||||
|
external_username = COALESCE(players.external_username, EXCLUDED.telegram_username);
|
||||||
|
""",
|
||||||
new { TgId = gmId, Name = gmName, Username = gmUsername },
|
new { TgId = gmId, Name = gmName, Username = gmUsername },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
// 2. Убеждаемся, что Группа зарегистрирована
|
var existingGroup = await connection.QuerySingleOrDefaultAsync<SessionCreationGroupAccessDto>(
|
||||||
var groupId = await connection.ExecuteScalarAsync<Guid>(
|
"""
|
||||||
@"INSERT INTO game_groups (telegram_chat_id, name, gm_telegram_id)
|
SELECT g.id AS GroupId,
|
||||||
VALUES (@ChatId, @ChatName, @GmId)
|
EXISTS (
|
||||||
ON CONFLICT (telegram_chat_id) DO UPDATE SET name = EXCLUDED.name
|
SELECT 1
|
||||||
RETURNING id;",
|
FROM group_managers gm
|
||||||
new { ChatId = chatId, ChatName = chatTitle, GmId = gmId },
|
JOIN players p ON p.id = gm.player_id
|
||||||
|
WHERE gm.group_id = g.id
|
||||||
|
AND COALESCE(p.external_user_id, p.telegram_id::TEXT) = @GmId::TEXT
|
||||||
|
) AS CanManage
|
||||||
|
FROM game_groups g
|
||||||
|
WHERE COALESCE(g.external_group_id, g.telegram_chat_id::TEXT) = @ChatId::TEXT
|
||||||
|
""",
|
||||||
|
new { ChatId = chatId, GmId = gmId },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
int? messageThreadId = null;
|
Guid groupId;
|
||||||
if (message.Chat.IsForum)
|
if (existingGroup is null)
|
||||||
{
|
{
|
||||||
var topic = await botClient.CreateForumTopic(
|
groupId = await connection.ExecuteScalarAsync<Guid>(
|
||||||
chatId: chatId,
|
"""
|
||||||
name: $"🎲 Игры: {title}",
|
INSERT INTO game_groups (telegram_chat_id, name, gm_telegram_id, platform, external_group_id)
|
||||||
cancellationToken: cancellationToken);
|
VALUES (@ChatId, @ChatName, @GmId, 'Telegram', @ChatId::TEXT)
|
||||||
messageThreadId = topic.MessageThreadId;
|
RETURNING id;
|
||||||
|
""",
|
||||||
|
new { ChatId = chatId, ChatName = chatTitle, GmId = gmId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
INSERT INTO group_managers (group_id, player_id, role)
|
||||||
|
SELECT @GroupId, p.id, @OwnerRole
|
||||||
|
FROM players p
|
||||||
|
WHERE COALESCE(p.external_user_id, p.telegram_id::TEXT) = @GmId::TEXT
|
||||||
|
ON CONFLICT (group_id, player_id) DO NOTHING
|
||||||
|
""",
|
||||||
|
new { GroupId = groupId, GmId = gmId, OwnerRole = GroupManagerRoleExtensions.OwnerValue },
|
||||||
|
transaction);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (!existingGroup.CanManage)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(cancellationToken);
|
||||||
|
await botClient.SendMessage(
|
||||||
|
chatId,
|
||||||
|
"⛔ Только owner или co-GM этой группы может создавать игровые сессии.",
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
groupId = existingGroup.GroupId;
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"UPDATE game_groups SET name = @ChatName WHERE id = @GroupId",
|
||||||
|
new { ChatName = chatTitle, GroupId = groupId },
|
||||||
|
transaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
var topicDestination = TelegramTopicRouting.ResolveNewScheduleDestination(
|
||||||
|
message.Chat.IsForum,
|
||||||
|
message.MessageThreadId);
|
||||||
|
var messageThreadId = topicDestination.MessageThreadId;
|
||||||
|
var topicCreatedByBot = topicDestination.TopicCreatedByBot;
|
||||||
|
if (topicDestination.ShouldCreateForumTopic)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var topic = await botClient.CreateForumTopic(
|
||||||
|
chatId: chatId,
|
||||||
|
name: $"🎲 Игры: {title}",
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
messageThreadId = topic.MessageThreadId;
|
||||||
|
}
|
||||||
|
catch (Telegram.Bot.Exceptions.ApiRequestException ex)
|
||||||
|
when (TelegramTopicRouting.IsMissingForumTopicRightsError(ex.Message))
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(cancellationToken);
|
||||||
|
await botClient.SendMessage(
|
||||||
|
chatId,
|
||||||
|
TelegramTopicRouting.MissingForumTopicRightsMessage,
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. Создаем сессии в цикле с общим batch_id
|
|
||||||
var batchId = Guid.NewGuid();
|
var batchId = Guid.NewGuid();
|
||||||
var sessions = new List<SessionBatchDto>();
|
var sessions = new List<SessionBatchDto>();
|
||||||
|
|
||||||
foreach (var dt in scheduledTimes.OrderBy(d => d))
|
foreach (var scheduledAt in parseResult.ScheduledTimes.OrderBy(value => value))
|
||||||
{
|
{
|
||||||
var sessionId = await connection.ExecuteScalarAsync<Guid>(
|
var sessionId = await connection.ExecuteScalarAsync<Guid>(
|
||||||
@"INSERT INTO sessions (batch_id, group_id, title, join_link, scheduled_at, status, thread_id)
|
"""
|
||||||
VALUES (@BatchId, @GroupId, @Title, @Link, @ScheduledAt, 'Planned', @ThreadId)
|
INSERT INTO sessions (batch_id, group_id, title, join_link, scheduled_at, status, thread_id, topic_created_by_bot, max_players)
|
||||||
RETURNING id;",
|
VALUES (@BatchId, @GroupId, @Title, @Link, @ScheduledAt, @Status, @ThreadId, @TopicCreatedByBot, @MaxPlayers)
|
||||||
new { BatchId = batchId, GroupId = groupId, Title = title, Link = link, ScheduledAt = dt, ThreadId = messageThreadId },
|
RETURNING id;
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
BatchId = batchId,
|
||||||
|
GroupId = groupId,
|
||||||
|
Title = title,
|
||||||
|
Link = link,
|
||||||
|
ScheduledAt = scheduledAt,
|
||||||
|
ThreadId = messageThreadId,
|
||||||
|
TopicCreatedByBot = topicCreatedByBot,
|
||||||
|
MaxPlayers = parseResult.MaxPlayers,
|
||||||
|
Status = SessionStatus.Planned
|
||||||
|
},
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
sessions.Add(new SessionBatchDto(sessionId, dt.UtcDateTime, "Planned"));
|
sessions.Add(new SessionBatchDto(sessionId, scheduledAt.UtcDateTime, SessionStatus.Planned, parseResult.MaxPlayers, link));
|
||||||
}
|
}
|
||||||
|
|
||||||
await transaction.CommitAsync(cancellationToken);
|
await transaction.CommitAsync(cancellationToken);
|
||||||
logger.LogInformation("Создан батч {BatchId} с {Count} сессиями в группе {GroupId}", batchId, sessions.Count, groupId);
|
logger.LogInformation("Создан батч {BatchId} с {Count} сессиями в группе {GroupId}", batchId, sessions.Count, groupId);
|
||||||
|
|
||||||
// 4. Отправляем сообщение в чат
|
var view = SessionBatchViewBuilder.Build(title, sessions, Array.Empty<ParticipantBatchDto>());
|
||||||
var renderResult = SessionBatchRenderer.Render(title, sessions, Array.Empty<ParticipantBatchDto>());
|
var renderResult = TelegramSessionBatchRenderer.Render(view);
|
||||||
|
|
||||||
|
Message batchMessage;
|
||||||
|
|
||||||
var batchMessage = await botClient.SendMessage(
|
if (imageReference is not null && renderResult.Text.Length <= 1024)
|
||||||
chatId: chatId,
|
{
|
||||||
messageThreadId: messageThreadId,
|
// Картинка + расписание умещаются в одном Telegram-фото с подписью
|
||||||
text: renderResult.Text,
|
try
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
{
|
||||||
replyMarkup: renderResult.Markup,
|
batchMessage = await botClient.SendPhoto(
|
||||||
cancellationToken: cancellationToken);
|
chatId: chatId,
|
||||||
|
messageThreadId: messageThreadId,
|
||||||
|
photo: InputFile.FromString(imageReference),
|
||||||
|
caption: renderResult.Text,
|
||||||
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
|
replyMarkup: renderResult.Markup,
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogWarning(ex, "Не удалось отправить картинку для батча {BatchId}, отправляем текстом", batchId);
|
||||||
|
batchMessage = await botClient.SendMessage(
|
||||||
|
chatId: chatId,
|
||||||
|
messageThreadId: messageThreadId,
|
||||||
|
text: renderResult.Text,
|
||||||
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
|
replyMarkup: renderResult.Markup,
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Текст слишком длинный для caption — fallback на два сообщения
|
||||||
|
if (imageReference is not null)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await botClient.SendPhoto(
|
||||||
|
chatId: chatId,
|
||||||
|
messageThreadId: messageThreadId,
|
||||||
|
photo: InputFile.FromString(imageReference),
|
||||||
|
caption: $"🎲 {System.Net.WebUtility.HtmlEncode(title)}",
|
||||||
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogWarning(ex, "Не удалось отправить картинку для батча {BatchId}", batchId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
batchMessage = await botClient.SendMessage(
|
||||||
|
chatId: chatId,
|
||||||
|
messageThreadId: messageThreadId,
|
||||||
|
text: renderResult.Text,
|
||||||
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
|
replyMarkup: renderResult.Markup,
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
// 4b. Сохраняем message_id батч-сообщения для дальнейшего редактирования
|
|
||||||
await connection.ExecuteAsync(
|
await connection.ExecuteAsync(
|
||||||
"UPDATE sessions SET batch_message_id = @MsgId WHERE batch_id = @BatchId",
|
"UPDATE sessions SET batch_message_id = @MsgId WHERE batch_id = @BatchId",
|
||||||
new { MsgId = batchMessage.MessageId, BatchId = batchId });
|
new { MsgId = batchMessage.MessageId, BatchId = batchId });
|
||||||
|
|
||||||
// 5. Удаляем исходное сообщение с командой /newsession, чтобы не спамить
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await botClient.DeleteMessage(
|
await botClient.DeleteMessage(
|
||||||
@@ -150,4 +289,20 @@ public sealed class CreateSessionHandler(
|
|||||||
await botClient.SendMessage(chatId, "💥 Произошла ошибка базы данных при создании сессии.", cancellationToken: cancellationToken);
|
await botClient.SendMessage(chatId, "💥 Произошла ошибка базы данных при создании сессии.", cancellationToken: cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal static string? GetBatchImageReference(Message message, string? parsedImageUrl)
|
||||||
|
{
|
||||||
|
var attachedPhotoFileId = message.Photo?
|
||||||
|
.OrderByDescending(photo => photo.FileSize ?? 0)
|
||||||
|
.ThenByDescending(photo => photo.Width * photo.Height)
|
||||||
|
.FirstOrDefault()
|
||||||
|
?.FileId;
|
||||||
|
|
||||||
|
if (!string.IsNullOrWhiteSpace(attachedPhotoFileId))
|
||||||
|
{
|
||||||
|
return attachedPhotoFileId;
|
||||||
|
}
|
||||||
|
|
||||||
|
return string.IsNullOrWhiteSpace(parsedImageUrl) ? null : parsedImageUrl.Trim();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,9 +2,9 @@ using Dapper;
|
|||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types;
|
using Telegram.Bot.Types;
|
||||||
using Telegram.Bot.Types.ReplyMarkups;
|
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
using GmRelay.Shared.Rendering;
|
using GmRelay.Shared.Rendering;
|
||||||
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
||||||
|
|
||||||
@@ -18,7 +18,7 @@ public sealed record JoinSessionCommand(
|
|||||||
int MessageId);
|
int MessageId);
|
||||||
|
|
||||||
// DTOs for AOT compilation
|
// DTOs for AOT compilation
|
||||||
internal sealed record JoinSessionBatchDto(Guid BatchId, string Title);
|
internal sealed record JoinSessionBatchDto(Guid BatchId, string Title, int? MaxPlayers);
|
||||||
|
|
||||||
public sealed class JoinSessionHandler(
|
public sealed class JoinSessionHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
@@ -29,24 +29,86 @@ public sealed class JoinSessionHandler(
|
|||||||
{
|
{
|
||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
await using var transaction = await connection.BeginTransactionAsync(ct);
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
var transactionCommitted = false;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
// 1. Убеждаемся, что игрок есть в базе
|
// 1. Убеждаемся, что игрок есть в базе
|
||||||
var playerId = await connection.ExecuteScalarAsync<Guid>(
|
var playerId = await connection.ExecuteScalarAsync<Guid>(
|
||||||
@"INSERT INTO players (telegram_id, display_name, telegram_username)
|
@"INSERT INTO players (telegram_id, display_name, telegram_username, platform, external_user_id, external_username)
|
||||||
VALUES (@TgId, @Name, @Username)
|
VALUES (@TgId, @Name, @Username, 'Telegram', @TgId::TEXT, @Username)
|
||||||
ON CONFLICT (telegram_id) DO UPDATE SET display_name = EXCLUDED.display_name, telegram_username = EXCLUDED.telegram_username
|
ON CONFLICT (telegram_id) DO UPDATE
|
||||||
|
SET display_name = EXCLUDED.display_name,
|
||||||
|
telegram_username = EXCLUDED.telegram_username,
|
||||||
|
platform = COALESCE(players.platform, 'Telegram'),
|
||||||
|
external_user_id = COALESCE(players.external_user_id, EXCLUDED.telegram_id::TEXT),
|
||||||
|
external_username = COALESCE(players.external_username, EXCLUDED.telegram_username)
|
||||||
RETURNING id;",
|
RETURNING id;",
|
||||||
new { TgId = command.TelegramUserId, Name = command.DisplayName, Username = command.TelegramUsername },
|
new { TgId = command.TelegramUserId, Name = command.DisplayName, Username = command.TelegramUsername },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
// 2. Добавляем в участники сессии (статус Pending, так как за 24 часа нужно будет финальное подтверждение)
|
// 2. Блокируем сессию на время расчета мест, чтобы параллельные нажатия не переполнили состав.
|
||||||
|
var batchInfo = await connection.QuerySingleOrDefaultAsync<JoinSessionBatchDto>(
|
||||||
|
@"SELECT batch_id as BatchId, title as Title, max_players as MaxPlayers
|
||||||
|
FROM sessions
|
||||||
|
WHERE id = @SessionId
|
||||||
|
FOR UPDATE",
|
||||||
|
new { command.SessionId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (batchInfo is null)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Сессия не найдена.", cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var existingRegistrationStatus = await connection.ExecuteScalarAsync<string?>(
|
||||||
|
"""
|
||||||
|
SELECT sp.registration_status
|
||||||
|
FROM session_participants sp
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.player_id = @PlayerId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
""",
|
||||||
|
new { command.SessionId, PlayerId = playerId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (existingRegistrationStatus is not null)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
var alreadyText = existingRegistrationStatus == ParticipantRegistrationStatus.Waitlisted
|
||||||
|
? "Вы уже в листе ожидания!"
|
||||||
|
: "Вы уже записаны!";
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, alreadyText, cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var activeParticipants = await connection.ExecuteScalarAsync<int>(
|
||||||
|
"""
|
||||||
|
SELECT COUNT(*)
|
||||||
|
FROM session_participants
|
||||||
|
WHERE session_id = @SessionId
|
||||||
|
AND is_gm = false
|
||||||
|
AND registration_status = @Active
|
||||||
|
""",
|
||||||
|
new { command.SessionId, Active = ParticipantRegistrationStatus.Active },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
var registrationStatus = SessionCapacityRules.DecideJoinStatus(batchInfo.MaxPlayers, activeParticipants);
|
||||||
|
|
||||||
|
// 3. Добавляем в основной состав или лист ожидания. RSVP остается Pending до финального подтверждения.
|
||||||
var inserted = await connection.ExecuteAsync(
|
var inserted = await connection.ExecuteAsync(
|
||||||
@"INSERT INTO session_participants (session_id, player_id, is_gm, rsvp_status)
|
@"INSERT INTO session_participants (session_id, player_id, is_gm, rsvp_status, registration_status)
|
||||||
VALUES (@SessionId, @PlayerId, false, 'Pending')
|
VALUES (@SessionId, @PlayerId, false, @Pending, @RegistrationStatus)
|
||||||
ON CONFLICT (session_id, player_id) DO NOTHING;",
|
ON CONFLICT (session_id, player_id) DO NOTHING;",
|
||||||
new { SessionId = command.SessionId, PlayerId = playerId },
|
new
|
||||||
|
{
|
||||||
|
command.SessionId,
|
||||||
|
PlayerId = playerId,
|
||||||
|
Pending = RsvpStatus.Pending,
|
||||||
|
RegistrationStatus = registrationStatus
|
||||||
|
},
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
if (inserted == 0)
|
if (inserted == 0)
|
||||||
@@ -56,45 +118,58 @@ public sealed class JoinSessionHandler(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. Получаем batch_id по session_id
|
|
||||||
var batchInfo = await connection.QuerySingleAsync<JoinSessionBatchDto>(
|
|
||||||
@"SELECT batch_id as BatchId, title as Title FROM sessions WHERE id = @SessionId",
|
|
||||||
new { command.SessionId }, transaction);
|
|
||||||
|
|
||||||
// Загружаем весь батч для перерисовки
|
// Загружаем весь батч для перерисовки
|
||||||
var batchSessions = await connection.QueryAsync<SessionBatchDto>(
|
var batchSessions = await connection.QueryAsync<SessionBatchDto>(
|
||||||
@"SELECT id as SessionId, scheduled_at as ScheduledAt, status as Status FROM sessions WHERE batch_id = @BatchId ORDER BY scheduled_at",
|
@"SELECT id as SessionId, scheduled_at as ScheduledAt, status as Status, max_players as MaxPlayers, join_link as JoinLink
|
||||||
|
FROM sessions
|
||||||
|
WHERE batch_id = @BatchId
|
||||||
|
ORDER BY scheduled_at",
|
||||||
new { BatchId = batchInfo.BatchId }, transaction);
|
new { BatchId = batchInfo.BatchId }, transaction);
|
||||||
|
|
||||||
var batchParticipants = await connection.QueryAsync<ParticipantBatchDto>(
|
var batchParticipants = await connection.QueryAsync<ParticipantBatchDto>(
|
||||||
@"SELECT sp.session_id as SessionId, p.display_name as DisplayName, p.telegram_username as TelegramUsername
|
@"SELECT sp.session_id as SessionId,
|
||||||
|
p.display_name as DisplayName,
|
||||||
|
p.telegram_username as TelegramUsername,
|
||||||
|
sp.registration_status as RegistrationStatus
|
||||||
FROM session_participants sp
|
FROM session_participants sp
|
||||||
JOIN players p ON sp.player_id = p.id
|
JOIN players p ON sp.player_id = p.id
|
||||||
JOIN sessions s ON sp.session_id = s.id
|
JOIN sessions s ON sp.session_id = s.id
|
||||||
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
||||||
ORDER BY sp.responded_at ASC, p.created_at ASC",
|
ORDER BY sp.registration_status ASC, sp.created_at ASC, sp.responded_at ASC, p.created_at ASC",
|
||||||
new { BatchId = batchInfo.BatchId }, transaction);
|
new { BatchId = batchInfo.BatchId }, transaction);
|
||||||
|
|
||||||
await transaction.CommitAsync(ct);
|
await transaction.CommitAsync(ct);
|
||||||
|
transactionCommitted = true;
|
||||||
|
|
||||||
// 4. Перерисовываем сообщение
|
// 4. Перерисовываем сообщение
|
||||||
var renderResult = SessionBatchRenderer.Render(batchInfo.Title, batchSessions.ToList(), batchParticipants.ToList());
|
var view = SessionBatchViewBuilder.Build(batchInfo.Title, batchSessions.ToList(), batchParticipants.ToList());
|
||||||
|
var renderResult = TelegramSessionBatchRenderer.Render(view);
|
||||||
|
|
||||||
await bot.EditMessageText(
|
await BatchMessageEditor.EditBatchMessageAsync(
|
||||||
|
bot,
|
||||||
chatId: command.ChatId,
|
chatId: command.ChatId,
|
||||||
messageId: command.MessageId,
|
messageId: command.MessageId,
|
||||||
text: renderResult.Text,
|
text: renderResult.Text,
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
|
||||||
replyMarkup: renderResult.Markup,
|
replyMarkup: renderResult.Markup,
|
||||||
cancellationToken: ct);
|
ct);
|
||||||
|
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Вы успешно записаны!", cancellationToken: ct);
|
var callbackText = registrationStatus == ParticipantRegistrationStatus.Waitlisted
|
||||||
|
? "Основной состав заполнен. Вы добавлены в лист ожидания."
|
||||||
|
: "Вы успешно записаны!";
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, callbackText, cancellationToken: ct);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
logger.LogError(ex, "Ошибка при добавлении игрока к сессии");
|
logger.LogError(ex, "Ошибка при добавлении игрока к сессии");
|
||||||
await transaction.RollbackAsync(ct);
|
if (!transactionCommitted)
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Произошла ошибка при регистрации.", cancellationToken: ct);
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
}
|
||||||
|
|
||||||
|
var errorText = transactionCommitted
|
||||||
|
? "Регистрация сохранена, но не удалось обновить сообщение расписания."
|
||||||
|
: "Произошла ошибка при регистрации.";
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, errorText, cancellationToken: ct);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,220 @@
|
|||||||
|
using Dapper;
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
using GmRelay.Shared.Rendering;
|
||||||
|
using Npgsql;
|
||||||
|
using Telegram.Bot;
|
||||||
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
||||||
|
|
||||||
|
public sealed record LeaveSessionCommand(
|
||||||
|
Guid SessionId,
|
||||||
|
long TelegramUserId,
|
||||||
|
string CallbackQueryId,
|
||||||
|
long ChatId,
|
||||||
|
int MessageId);
|
||||||
|
|
||||||
|
internal sealed record LeaveSessionInfoDto(string Title, Guid BatchId, string Status, int? MaxPlayers);
|
||||||
|
internal sealed record LeaveSessionParticipantDto(Guid ParticipantRowId, string DisplayName, string RegistrationStatus);
|
||||||
|
internal sealed record LeaveSessionPromotionDto(Guid ParticipantRowId, string DisplayName);
|
||||||
|
|
||||||
|
public sealed class LeaveSessionHandler(
|
||||||
|
NpgsqlDataSource dataSource,
|
||||||
|
ITelegramBotClient bot,
|
||||||
|
ILogger<LeaveSessionHandler> logger)
|
||||||
|
{
|
||||||
|
public async Task HandleAsync(LeaveSessionCommand command, CancellationToken ct)
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
var transactionCommitted = false;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var session = await connection.QuerySingleOrDefaultAsync<LeaveSessionInfoDto>(
|
||||||
|
"""
|
||||||
|
SELECT title AS Title,
|
||||||
|
batch_id AS BatchId,
|
||||||
|
status AS Status,
|
||||||
|
max_players AS MaxPlayers
|
||||||
|
FROM sessions
|
||||||
|
WHERE id = @SessionId
|
||||||
|
FOR UPDATE
|
||||||
|
""",
|
||||||
|
new { command.SessionId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (session is null)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Сессия не найдена.", cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (SessionStatus.IsCancelled(session.Status))
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Сессия уже отменена.", cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var participant = await connection.QuerySingleOrDefaultAsync<LeaveSessionParticipantDto>(
|
||||||
|
"""
|
||||||
|
SELECT sp.id AS ParticipantRowId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
sp.registration_status AS RegistrationStatus
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON p.id = sp.player_id
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND p.telegram_id = @TelegramUserId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
FOR UPDATE OF sp
|
||||||
|
""",
|
||||||
|
new { command.SessionId, command.TelegramUserId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (participant is null)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Вы не записаны на эту сессию.", cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
DELETE FROM session_participants
|
||||||
|
WHERE id = @ParticipantRowId
|
||||||
|
""",
|
||||||
|
new { participant.ParticipantRowId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
var activeParticipantsAfterLeave = await connection.ExecuteScalarAsync<int>(
|
||||||
|
"""
|
||||||
|
SELECT COUNT(*)
|
||||||
|
FROM session_participants
|
||||||
|
WHERE session_id = @SessionId
|
||||||
|
AND is_gm = false
|
||||||
|
AND registration_status = @Active
|
||||||
|
""",
|
||||||
|
new { command.SessionId, Active = ParticipantRegistrationStatus.Active },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
var waitlistedParticipants = await connection.ExecuteScalarAsync<int>(
|
||||||
|
"""
|
||||||
|
SELECT COUNT(*)
|
||||||
|
FROM session_participants
|
||||||
|
WHERE session_id = @SessionId
|
||||||
|
AND is_gm = false
|
||||||
|
AND registration_status = @Waitlisted
|
||||||
|
""",
|
||||||
|
new { command.SessionId, Waitlisted = ParticipantRegistrationStatus.Waitlisted },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
string? promotedDisplayName = null;
|
||||||
|
if (SessionCapacityRules.ShouldPromoteAfterParticipantLeaves(
|
||||||
|
participant.RegistrationStatus,
|
||||||
|
session.MaxPlayers,
|
||||||
|
activeParticipantsAfterLeave,
|
||||||
|
waitlistedParticipants))
|
||||||
|
{
|
||||||
|
var promoted = await connection.QuerySingleAsync<LeaveSessionPromotionDto>(
|
||||||
|
"""
|
||||||
|
SELECT sp.id AS ParticipantRowId,
|
||||||
|
p.display_name AS DisplayName
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON p.id = sp.player_id
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Waitlisted
|
||||||
|
ORDER BY sp.created_at ASC, sp.id ASC
|
||||||
|
LIMIT 1
|
||||||
|
FOR UPDATE OF sp
|
||||||
|
""",
|
||||||
|
new { command.SessionId, Waitlisted = ParticipantRegistrationStatus.Waitlisted },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
UPDATE session_participants
|
||||||
|
SET registration_status = @Active,
|
||||||
|
rsvp_status = @Pending,
|
||||||
|
responded_at = NULL
|
||||||
|
WHERE id = @ParticipantRowId
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
promoted.ParticipantRowId,
|
||||||
|
Active = ParticipantRegistrationStatus.Active,
|
||||||
|
Pending = RsvpStatus.Pending
|
||||||
|
},
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
promotedDisplayName = promoted.DisplayName;
|
||||||
|
}
|
||||||
|
|
||||||
|
var batchSessions = (await connection.QueryAsync<SessionBatchDto>(
|
||||||
|
"""
|
||||||
|
SELECT id AS SessionId,
|
||||||
|
scheduled_at AS ScheduledAt,
|
||||||
|
status AS Status,
|
||||||
|
max_players AS MaxPlayers,
|
||||||
|
join_link AS JoinLink
|
||||||
|
FROM sessions
|
||||||
|
WHERE batch_id = @BatchId
|
||||||
|
ORDER BY scheduled_at
|
||||||
|
""",
|
||||||
|
new { session.BatchId },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
|
var batchParticipants = (await connection.QueryAsync<ParticipantBatchDto>(
|
||||||
|
"""
|
||||||
|
SELECT sp.session_id AS SessionId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
sp.registration_status AS RegistrationStatus
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON sp.player_id = p.id
|
||||||
|
JOIN sessions s ON sp.session_id = s.id
|
||||||
|
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
||||||
|
ORDER BY sp.registration_status ASC, sp.created_at ASC, sp.responded_at ASC, p.created_at ASC
|
||||||
|
""",
|
||||||
|
new { session.BatchId },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
|
await transaction.CommitAsync(ct);
|
||||||
|
transactionCommitted = true;
|
||||||
|
|
||||||
|
var view = SessionBatchViewBuilder.Build(session.Title, batchSessions, batchParticipants);
|
||||||
|
var renderResult = TelegramSessionBatchRenderer.Render(view);
|
||||||
|
|
||||||
|
await BatchMessageEditor.EditBatchMessageAsync(
|
||||||
|
bot,
|
||||||
|
chatId: command.ChatId,
|
||||||
|
messageId: command.MessageId,
|
||||||
|
text: renderResult.Text,
|
||||||
|
replyMarkup: renderResult.Markup,
|
||||||
|
ct);
|
||||||
|
|
||||||
|
var callbackText = participant.RegistrationStatus == ParticipantRegistrationStatus.Waitlisted
|
||||||
|
? "Вы удалены из листа ожидания."
|
||||||
|
: promotedDisplayName is null
|
||||||
|
? "Вы отписались от сессии."
|
||||||
|
: $"Вы отписались от сессии. Место получил(а) {promotedDisplayName}.";
|
||||||
|
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, callbackText, cancellationToken: ct);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogError(ex, "Ошибка при самостоятельной отмене записи на сессию {SessionId}", command.SessionId);
|
||||||
|
if (!transactionCommitted)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
}
|
||||||
|
|
||||||
|
var errorText = transactionCommitted
|
||||||
|
? "Запись снята, но не удалось обновить сообщение расписания."
|
||||||
|
: "Произошла ошибка при отмене записи.";
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, errorText, cancellationToken: ct);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,184 @@
|
|||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
||||||
|
|
||||||
|
internal sealed record NewSessionParseResult(
|
||||||
|
string? Title,
|
||||||
|
string? Link,
|
||||||
|
string? ImageUrl,
|
||||||
|
int? MaxPlayers,
|
||||||
|
IReadOnlyList<DateTimeOffset> ScheduledTimes,
|
||||||
|
IReadOnlyList<string> PastTimeInputs,
|
||||||
|
IReadOnlyList<string> InvalidTimeInputs,
|
||||||
|
IReadOnlyList<string> InvalidSeatLimitInputs,
|
||||||
|
IReadOnlyList<string> InvalidRecurringInputs)
|
||||||
|
{
|
||||||
|
public bool IsValid =>
|
||||||
|
!string.IsNullOrWhiteSpace(Title) &&
|
||||||
|
!string.IsNullOrWhiteSpace(Link) &&
|
||||||
|
ScheduledTimes.Count > 0 &&
|
||||||
|
InvalidSeatLimitInputs.Count == 0 &&
|
||||||
|
InvalidRecurringInputs.Count == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
internal static class NewSessionCommandParser
|
||||||
|
{
|
||||||
|
private const int MaxRecurringSessionCount = 52;
|
||||||
|
private const int MaxRecurringIntervalDays = 365;
|
||||||
|
private const string TitlePrefix = "\u041d\u0430\u0437\u0432\u0430\u043d\u0438\u0435:";
|
||||||
|
private const string TimePrefix = "\u0412\u0440\u0435\u043c\u044f:";
|
||||||
|
private const string LinkPrefix = "\u0421\u0441\u044b\u043b\u043a\u0430:";
|
||||||
|
private static readonly string[] ImagePrefixes =
|
||||||
|
[
|
||||||
|
"\u041a\u0430\u0440\u0442\u0438\u043d\u043a\u0430:",
|
||||||
|
"\u0418\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u0435:",
|
||||||
|
"\u041e\u0431\u043b\u043e\u0436\u043a\u0430:"
|
||||||
|
];
|
||||||
|
private static readonly string[] SeatLimitPrefixes =
|
||||||
|
[
|
||||||
|
"\u041c\u0435\u0441\u0442:",
|
||||||
|
"\u041b\u0438\u043c\u0438\u0442:",
|
||||||
|
"\u041c\u0430\u043a\u0441\u0438\u043c\u0443\u043c:"
|
||||||
|
];
|
||||||
|
private static readonly string[] RecurringCountPrefixes =
|
||||||
|
[
|
||||||
|
"\u0418\u0433\u0440:",
|
||||||
|
"\u0421\u0435\u0441\u0441\u0438\u0439:",
|
||||||
|
"\u041f\u043e\u0432\u0442\u043e\u0440\u043e\u0432:"
|
||||||
|
];
|
||||||
|
private static readonly string[] RecurringIntervalPrefixes =
|
||||||
|
[
|
||||||
|
"\u0428\u0430\u0433:",
|
||||||
|
"\u0418\u043d\u0442\u0435\u0440\u0432\u0430\u043b:"
|
||||||
|
];
|
||||||
|
|
||||||
|
public static NewSessionParseResult Parse(string? text, DateTimeOffset nowUtc)
|
||||||
|
{
|
||||||
|
string? title = null;
|
||||||
|
string? link = null;
|
||||||
|
string? imageUrl = null;
|
||||||
|
int? maxPlayers = null;
|
||||||
|
int? recurringCount = null;
|
||||||
|
var recurringIntervalDays = 7;
|
||||||
|
var scheduledTimes = new List<DateTimeOffset>();
|
||||||
|
var pastTimeInputs = new List<string>();
|
||||||
|
var invalidTimeInputs = new List<string>();
|
||||||
|
var invalidSeatLimitInputs = new List<string>();
|
||||||
|
var invalidRecurringInputs = new List<string>();
|
||||||
|
|
||||||
|
foreach (var line in (text ?? string.Empty).Split('\n', StringSplitOptions.TrimEntries))
|
||||||
|
{
|
||||||
|
if (line.StartsWith(TitlePrefix, StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
title = line[TitlePrefix.Length..].Trim();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (line.StartsWith(LinkPrefix, StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
link = line[LinkPrefix.Length..].Trim();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var imagePrefix = ImagePrefixes.FirstOrDefault(prefix =>
|
||||||
|
line.StartsWith(prefix, StringComparison.OrdinalIgnoreCase));
|
||||||
|
if (imagePrefix is not null)
|
||||||
|
{
|
||||||
|
imageUrl = line[imagePrefix.Length..].Trim();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var seatLimitPrefix = SeatLimitPrefixes.FirstOrDefault(prefix =>
|
||||||
|
line.StartsWith(prefix, StringComparison.OrdinalIgnoreCase));
|
||||||
|
if (seatLimitPrefix is not null)
|
||||||
|
{
|
||||||
|
var seatLimitInput = line[seatLimitPrefix.Length..].Trim();
|
||||||
|
if (int.TryParse(seatLimitInput, out var parsedMaxPlayers) && parsedMaxPlayers > 0)
|
||||||
|
{
|
||||||
|
maxPlayers = parsedMaxPlayers;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
invalidSeatLimitInputs.Add(seatLimitInput);
|
||||||
|
}
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var recurringCountPrefix = RecurringCountPrefixes.FirstOrDefault(prefix =>
|
||||||
|
line.StartsWith(prefix, StringComparison.OrdinalIgnoreCase));
|
||||||
|
if (recurringCountPrefix is not null)
|
||||||
|
{
|
||||||
|
var recurringInput = line[recurringCountPrefix.Length..].Trim();
|
||||||
|
if (int.TryParse(recurringInput, out var parsedCount) &&
|
||||||
|
parsedCount is >= 1 and <= MaxRecurringSessionCount)
|
||||||
|
{
|
||||||
|
recurringCount = parsedCount;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
invalidRecurringInputs.Add(recurringInput);
|
||||||
|
}
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var recurringIntervalPrefix = RecurringIntervalPrefixes.FirstOrDefault(prefix =>
|
||||||
|
line.StartsWith(prefix, StringComparison.OrdinalIgnoreCase));
|
||||||
|
if (recurringIntervalPrefix is not null)
|
||||||
|
{
|
||||||
|
var recurringInput = line[recurringIntervalPrefix.Length..].Trim();
|
||||||
|
if (int.TryParse(recurringInput, out var parsedInterval) &&
|
||||||
|
parsedInterval is >= 1 and <= MaxRecurringIntervalDays)
|
||||||
|
{
|
||||||
|
recurringIntervalDays = parsedInterval;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
invalidRecurringInputs.Add(recurringInput);
|
||||||
|
}
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!line.StartsWith(TimePrefix, StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var timeInput = line[TimePrefix.Length..].Trim();
|
||||||
|
if (!MoscowTime.TryParseMoscow(timeInput, out var scheduledAt))
|
||||||
|
{
|
||||||
|
invalidTimeInputs.Add(timeInput);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (scheduledAt <= nowUtc)
|
||||||
|
{
|
||||||
|
pastTimeInputs.Add(timeInput);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
scheduledTimes.Add(scheduledAt);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recurringCount.HasValue && scheduledTimes.Count == 1)
|
||||||
|
{
|
||||||
|
var firstScheduledTime = scheduledTimes[0];
|
||||||
|
scheduledTimes = Enumerable.Range(0, recurringCount.Value)
|
||||||
|
.Select(index => firstScheduledTime.AddDays(recurringIntervalDays * index))
|
||||||
|
.ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
return new NewSessionParseResult(
|
||||||
|
title,
|
||||||
|
link,
|
||||||
|
imageUrl,
|
||||||
|
maxPlayers,
|
||||||
|
scheduledTimes,
|
||||||
|
pastTimeInputs,
|
||||||
|
invalidTimeInputs,
|
||||||
|
invalidSeatLimitInputs,
|
||||||
|
invalidRecurringInputs);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,194 @@
|
|||||||
|
using Dapper;
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
using GmRelay.Shared.Rendering;
|
||||||
|
using Npgsql;
|
||||||
|
using Telegram.Bot;
|
||||||
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Features.Sessions.CreateSession;
|
||||||
|
|
||||||
|
public sealed record PromoteWaitlistedPlayerCommand(
|
||||||
|
Guid SessionId,
|
||||||
|
long TelegramUserId,
|
||||||
|
string CallbackQueryId,
|
||||||
|
long ChatId,
|
||||||
|
int MessageId);
|
||||||
|
|
||||||
|
internal sealed record PromoteWaitlistSessionDto(string Title, Guid BatchId, int? BatchMessageId, bool CanManage, int? MaxPlayers);
|
||||||
|
internal sealed record WaitlistedParticipantDto(Guid ParticipantRowId, string DisplayName);
|
||||||
|
|
||||||
|
public sealed class PromoteWaitlistedPlayerHandler(
|
||||||
|
NpgsqlDataSource dataSource,
|
||||||
|
ITelegramBotClient bot,
|
||||||
|
ILogger<PromoteWaitlistedPlayerHandler> logger)
|
||||||
|
{
|
||||||
|
public async Task HandleAsync(PromoteWaitlistedPlayerCommand command, CancellationToken ct)
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
var transactionCommitted = false;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var session = await connection.QuerySingleOrDefaultAsync<PromoteWaitlistSessionDto>(
|
||||||
|
"""
|
||||||
|
SELECT s.title AS Title,
|
||||||
|
s.batch_id AS BatchId,
|
||||||
|
s.batch_message_id AS BatchMessageId,
|
||||||
|
s.max_players AS MaxPlayers,
|
||||||
|
EXISTS (
|
||||||
|
SELECT 1
|
||||||
|
FROM group_managers gm
|
||||||
|
JOIN players p ON p.id = gm.player_id
|
||||||
|
WHERE gm.group_id = s.group_id
|
||||||
|
AND p.telegram_id = @TelegramUserId
|
||||||
|
) AS CanManage
|
||||||
|
FROM sessions s
|
||||||
|
WHERE s.id = @SessionId
|
||||||
|
FOR UPDATE
|
||||||
|
""",
|
||||||
|
new { command.SessionId, command.TelegramUserId },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (session is null)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Сессия не найдена.", cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!session.CanManage)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Только owner или co-GM может поднимать игроков из листа ожидания.", showAlert: true, cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var activeParticipants = await connection.ExecuteScalarAsync<int>(
|
||||||
|
"""
|
||||||
|
SELECT COUNT(*)
|
||||||
|
FROM session_participants
|
||||||
|
WHERE session_id = @SessionId
|
||||||
|
AND is_gm = false
|
||||||
|
AND registration_status = @Active
|
||||||
|
""",
|
||||||
|
new { command.SessionId, Active = ParticipantRegistrationStatus.Active },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
var waitlistedParticipants = await connection.ExecuteScalarAsync<int>(
|
||||||
|
"""
|
||||||
|
SELECT COUNT(*)
|
||||||
|
FROM session_participants
|
||||||
|
WHERE session_id = @SessionId
|
||||||
|
AND is_gm = false
|
||||||
|
AND registration_status = @Waitlisted
|
||||||
|
""",
|
||||||
|
new { command.SessionId, Waitlisted = ParticipantRegistrationStatus.Waitlisted },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (waitlistedParticipants == 0)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Лист ожидания пуст.", cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!SessionCapacityRules.CanPromoteWaitlistedPlayer(session.MaxPlayers, activeParticipants, waitlistedParticipants))
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Нет свободных мест. Увеличьте лимит перед повышением игрока.", showAlert: true, cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var promoted = await connection.QuerySingleAsync<WaitlistedParticipantDto>(
|
||||||
|
"""
|
||||||
|
SELECT sp.id AS ParticipantRowId,
|
||||||
|
p.display_name AS DisplayName
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON p.id = sp.player_id
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Waitlisted
|
||||||
|
ORDER BY sp.created_at ASC, sp.id ASC
|
||||||
|
LIMIT 1
|
||||||
|
FOR UPDATE OF sp
|
||||||
|
""",
|
||||||
|
new { command.SessionId, Waitlisted = ParticipantRegistrationStatus.Waitlisted },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
UPDATE session_participants
|
||||||
|
SET registration_status = @Active,
|
||||||
|
rsvp_status = @Pending,
|
||||||
|
responded_at = NULL
|
||||||
|
WHERE id = @ParticipantRowId
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
promoted.ParticipantRowId,
|
||||||
|
Active = ParticipantRegistrationStatus.Active,
|
||||||
|
Pending = RsvpStatus.Pending
|
||||||
|
},
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
var batchSessions = (await connection.QueryAsync<SessionBatchDto>(
|
||||||
|
"""
|
||||||
|
SELECT id AS SessionId,
|
||||||
|
scheduled_at AS ScheduledAt,
|
||||||
|
status AS Status,
|
||||||
|
max_players AS MaxPlayers,
|
||||||
|
join_link AS JoinLink
|
||||||
|
FROM sessions
|
||||||
|
WHERE batch_id = @BatchId
|
||||||
|
ORDER BY scheduled_at
|
||||||
|
""",
|
||||||
|
new { session.BatchId },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
|
var batchParticipants = (await connection.QueryAsync<ParticipantBatchDto>(
|
||||||
|
"""
|
||||||
|
SELECT sp.session_id AS SessionId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
sp.registration_status AS RegistrationStatus
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON sp.player_id = p.id
|
||||||
|
JOIN sessions s ON sp.session_id = s.id
|
||||||
|
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
||||||
|
ORDER BY sp.registration_status ASC, sp.created_at ASC, sp.responded_at ASC, p.created_at ASC
|
||||||
|
""",
|
||||||
|
new { session.BatchId },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
|
await transaction.CommitAsync(ct);
|
||||||
|
transactionCommitted = true;
|
||||||
|
|
||||||
|
var view = SessionBatchViewBuilder.Build(session.Title, batchSessions, batchParticipants);
|
||||||
|
var renderResult = TelegramSessionBatchRenderer.Render(view);
|
||||||
|
|
||||||
|
await BatchMessageEditor.EditBatchMessageAsync(
|
||||||
|
bot,
|
||||||
|
chatId: command.ChatId,
|
||||||
|
messageId: session.BatchMessageId ?? command.MessageId,
|
||||||
|
text: renderResult.Text,
|
||||||
|
replyMarkup: renderResult.Markup,
|
||||||
|
ct);
|
||||||
|
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, $"{promoted.DisplayName} переведен(а) в основной состав.", cancellationToken: ct);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogError(ex, "Ошибка при повышении игрока из листа ожидания для сессии {SessionId}", command.SessionId);
|
||||||
|
if (!transactionCommitted)
|
||||||
|
{
|
||||||
|
await transaction.RollbackAsync(ct);
|
||||||
|
}
|
||||||
|
|
||||||
|
var errorText = transactionCommitted
|
||||||
|
? "Игрок повышен, но не удалось обновить сообщение расписания."
|
||||||
|
: "Ошибка при обновлении листа ожидания.";
|
||||||
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, errorText, cancellationToken: ct);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,8 +1,11 @@
|
|||||||
using System.Text;
|
using System.Text;
|
||||||
using Dapper;
|
using Dapper;
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
using Microsoft.Extensions.Configuration;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types;
|
using Telegram.Bot.Types;
|
||||||
|
using Telegram.Bot.Types.ReplyMarkups;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Sessions.ExportCalendar;
|
namespace GmRelay.Bot.Features.Sessions.ExportCalendar;
|
||||||
|
|
||||||
@@ -10,21 +13,22 @@ internal sealed record CalendarSessionDto(Guid Id, string Title, DateTime Schedu
|
|||||||
|
|
||||||
public sealed class ExportCalendarHandler(
|
public sealed class ExportCalendarHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
ITelegramBotClient botClient)
|
ITelegramBotClient botClient,
|
||||||
|
IConfiguration configuration)
|
||||||
{
|
{
|
||||||
public async Task HandleAsync(Message message, CancellationToken cancellationToken)
|
public async Task HandleAsync(Message message, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
await using var connection = await dataSource.OpenConnectionAsync(cancellationToken);
|
await using var connection = await dataSource.OpenConnectionAsync(cancellationToken);
|
||||||
|
|
||||||
var sessions = await connection.QueryAsync<CalendarSessionDto>(
|
var sessions = await connection.QueryAsync<CalendarSessionDto>(
|
||||||
@"SELECT s.id as Id, s.title as Title, s.scheduled_at as ScheduledAt
|
@"SELECT s.id as Id, s.title as Title, s.scheduled_at as ScheduledAt"
|
||||||
FROM sessions s
|
+ " FROM sessions s"
|
||||||
JOIN game_groups g ON s.group_id = g.id
|
+ " JOIN game_groups g ON s.group_id = g.id"
|
||||||
WHERE g.telegram_chat_id = @ChatId
|
+ " WHERE g.telegram_chat_id = @ChatId"
|
||||||
AND s.status = 'Planned'
|
+ " AND s.status = @Planned"
|
||||||
AND s.scheduled_at > NOW()
|
+ " AND s.scheduled_at > NOW()"
|
||||||
ORDER BY s.scheduled_at ASC",
|
+ " ORDER BY s.scheduled_at ASC",
|
||||||
new { ChatId = message.Chat.Id });
|
new { ChatId = message.Chat.Id, Planned = SessionStatus.Planned });
|
||||||
|
|
||||||
var sessionsList = sessions.ToList();
|
var sessionsList = sessions.ToList();
|
||||||
|
|
||||||
@@ -53,8 +57,6 @@ public sealed class ExportCalendarHandler(
|
|||||||
sb.AppendLine($"DTSTART:{dtStart}");
|
sb.AppendLine($"DTSTART:{dtStart}");
|
||||||
sb.AppendLine($"DTEND:{dtEnd}");
|
sb.AppendLine($"DTEND:{dtEnd}");
|
||||||
sb.AppendLine($"SUMMARY:{s.Title}");
|
sb.AppendLine($"SUMMARY:{s.Title}");
|
||||||
// Escape special chars according to iCal standards (RFC 5545) -- simple escaping for summary
|
|
||||||
// In a fuller implementation we'd escape \r\n, commas, etc. But titles are mostly plain text.
|
|
||||||
sb.AppendLine("END:VEVENT");
|
sb.AppendLine("END:VEVENT");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -65,11 +67,45 @@ public sealed class ExportCalendarHandler(
|
|||||||
|
|
||||||
var inputFile = InputFile.FromStream(stream, "schedule.ics");
|
var inputFile = InputFile.FromStream(stream, "schedule.ics");
|
||||||
|
|
||||||
|
// Create calendar subscription
|
||||||
|
string? subscriptionUrl = null;
|
||||||
|
var baseUrl = configuration["Web:BaseUrl"];
|
||||||
|
var senderId = message.From?.Id;
|
||||||
|
if (!string.IsNullOrWhiteSpace(baseUrl) && senderId.HasValue)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var token = Guid.NewGuid().ToString("N");
|
||||||
|
var groupId = await connection.QueryFirstOrDefaultAsync<Guid?>(
|
||||||
|
@"SELECT id FROM game_groups WHERE telegram_chat_id = @ChatId",
|
||||||
|
new { ChatId = message.Chat.Id });
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
@"INSERT INTO calendar_subscriptions (id, token, user_telegram_id, group_id, filter_type, created_at, expires_at)
|
||||||
|
VALUES (gen_random_uuid(), @token, @userTelegramId, @groupId, @filterType, now(), NULL)",
|
||||||
|
new { token, userTelegramId = senderId.Value, groupId, filterType = (int)CalendarSubscriptionFilter.SpecificGroup });
|
||||||
|
|
||||||
|
subscriptionUrl = $"{baseUrl.TrimEnd('/')}/calendar/{token}.ics";
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
// Non-critical: if subscription creation fails, still send the file
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var replyMarkup = subscriptionUrl is not null
|
||||||
|
? new InlineKeyboardMarkup(new[]
|
||||||
|
{
|
||||||
|
new[] { InlineKeyboardButton.WithUrl("🔗 Подписаться на календарь", subscriptionUrl) }
|
||||||
|
})
|
||||||
|
: null;
|
||||||
|
|
||||||
await botClient.SendDocument(
|
await botClient.SendDocument(
|
||||||
chatId: message.Chat.Id,
|
chatId: message.Chat.Id,
|
||||||
document: inputFile,
|
document: inputFile,
|
||||||
caption: "📅 <b>Ваш календарь игр!</b>\nОткройте файл на устройстве, чтобы добавить события в свой календарь.",
|
caption: "📅 <b>Ваш календарь игр!</b>\nОткройте файл на устройстве, чтобы добавить события в свой календарь.",
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
|
replyMarkup: replyMarkup,
|
||||||
messageThreadId: message.MessageThreadId,
|
messageThreadId: message.MessageThreadId,
|
||||||
cancellationToken: cancellationToken);
|
cancellationToken: cancellationToken);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
using Dapper;
|
using Dapper;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Sessions.ListSessions;
|
namespace GmRelay.Bot.Features.Sessions.ListSessions;
|
||||||
@@ -12,7 +13,13 @@ public sealed record DeleteSessionCommand(
|
|||||||
long ChatId,
|
long ChatId,
|
||||||
int MessageId);
|
int MessageId);
|
||||||
|
|
||||||
internal sealed record DeleteSessionInfoDto(string Title, Guid BatchId, long GmId, int? ThreadId);
|
internal sealed record DeleteSessionInfoDto(
|
||||||
|
string Title,
|
||||||
|
Guid BatchId,
|
||||||
|
Guid GroupId,
|
||||||
|
bool CanManage,
|
||||||
|
int? ThreadId,
|
||||||
|
bool TopicCreatedByBot);
|
||||||
|
|
||||||
public sealed class DeleteSessionHandler(
|
public sealed class DeleteSessionHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
@@ -24,13 +31,25 @@ public sealed class DeleteSessionHandler(
|
|||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
await using var transaction = await connection.BeginTransactionAsync(ct);
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
|
||||||
// 1. Fetch session and verify GM
|
// 1. Fetch session and verify group manager.
|
||||||
var session = await connection.QuerySingleOrDefaultAsync<DeleteSessionInfoDto>(
|
var session = await connection.QuerySingleOrDefaultAsync<DeleteSessionInfoDto>(
|
||||||
@"SELECT s.title as Title, s.batch_id as BatchId, s.thread_id as ThreadId, g.gm_telegram_id as GmId
|
"""
|
||||||
FROM sessions s
|
SELECT s.title AS Title,
|
||||||
JOIN game_groups g ON s.group_id = g.id
|
s.batch_id AS BatchId,
|
||||||
WHERE s.id = @SessionId",
|
s.group_id AS GroupId,
|
||||||
new { command.SessionId }, transaction);
|
s.thread_id AS ThreadId,
|
||||||
|
s.topic_created_by_bot AS TopicCreatedByBot,
|
||||||
|
EXISTS (
|
||||||
|
SELECT 1
|
||||||
|
FROM group_managers gm
|
||||||
|
JOIN players p ON p.id = gm.player_id
|
||||||
|
WHERE gm.group_id = s.group_id
|
||||||
|
AND p.telegram_id = @TelegramUserId
|
||||||
|
) AS CanManage
|
||||||
|
FROM sessions s
|
||||||
|
WHERE s.id = @SessionId
|
||||||
|
""",
|
||||||
|
new { command.SessionId, command.TelegramUserId }, transaction);
|
||||||
|
|
||||||
if (session == null)
|
if (session == null)
|
||||||
{
|
{
|
||||||
@@ -38,24 +57,32 @@ public sealed class DeleteSessionHandler(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (session.GmId != command.TelegramUserId)
|
if (!session.CanManage)
|
||||||
{
|
{
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Только Мастер Игры (GM) может удалять сессию.", showAlert: true, cancellationToken: ct);
|
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Только owner или co-GM может удалять сессию.", showAlert: true, cancellationToken: ct);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. Delete session
|
// 2. Delete session
|
||||||
await connection.ExecuteAsync("DELETE FROM sessions WHERE id = @Id", new { Id = command.SessionId }, transaction);
|
await connection.ExecuteAsync("DELETE FROM sessions WHERE id = @Id", new { Id = command.SessionId }, transaction);
|
||||||
|
|
||||||
// 3. Check if any sessions are left in the batch
|
var remainingInTopic = session.ThreadId.HasValue
|
||||||
var remainingInBatch = await connection.ExecuteScalarAsync<int>(
|
? await connection.ExecuteScalarAsync<int>(
|
||||||
"SELECT COUNT(*) FROM sessions WHERE batch_id = @BatchId",
|
"""
|
||||||
new { BatchId = session.BatchId }, transaction);
|
SELECT COUNT(*)
|
||||||
|
FROM sessions
|
||||||
|
WHERE group_id = @GroupId
|
||||||
|
AND thread_id = @ThreadId
|
||||||
|
""",
|
||||||
|
new { session.GroupId, ThreadId = session.ThreadId.Value },
|
||||||
|
transaction)
|
||||||
|
: 0;
|
||||||
|
|
||||||
await transaction.CommitAsync(ct);
|
await transaction.CommitAsync(ct);
|
||||||
|
|
||||||
// 4. If no sessions left and we have a forum topic, delete the topic
|
// 4. If no sessions are left in a bot-owned forum topic, delete the topic.
|
||||||
if (remainingInBatch == 0 && session.ThreadId.HasValue)
|
if (session.ThreadId.HasValue &&
|
||||||
|
TelegramTopicRouting.ShouldDeleteForumTopic(session.TopicCreatedByBot, remainingInTopic))
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -74,45 +101,49 @@ public sealed class DeleteSessionHandler(
|
|||||||
// A simple way is to re-render the list:
|
// A simple way is to re-render the list:
|
||||||
await using var readConnection = await dataSource.OpenConnectionAsync(ct);
|
await using var readConnection = await dataSource.OpenConnectionAsync(ct);
|
||||||
var sessions = await readConnection.QueryAsync<SessionListItemDto>(
|
var sessions = await readConnection.QueryAsync<SessionListItemDto>(
|
||||||
@"SELECT s.id as Id, s.title as Title, s.scheduled_at as ScheduledAt, s.status as Status,
|
@"SELECT s.id as Id, s.title as Title, s.scheduled_at as ScheduledAt, s.status as Status, s.max_players as MaxPlayers,
|
||||||
COUNT(sp.id) FILTER (WHERE sp.is_gm = false) as PlayerCount,
|
COUNT(sp.id) FILTER (WHERE sp.is_gm = false AND sp.registration_status = @Active) as PlayerCount,
|
||||||
g.gm_telegram_id as GmId
|
COUNT(sp.id) FILTER (WHERE sp.is_gm = false AND sp.registration_status = @Waitlisted) as WaitlistCount,
|
||||||
|
EXISTS (
|
||||||
|
SELECT 1
|
||||||
|
FROM group_managers gm
|
||||||
|
JOIN players manager_player ON manager_player.id = gm.player_id
|
||||||
|
WHERE gm.group_id = s.group_id
|
||||||
|
AND manager_player.telegram_id = @TelegramUserId
|
||||||
|
) AS CanManage
|
||||||
FROM sessions s
|
FROM sessions s
|
||||||
JOIN game_groups g ON s.group_id = g.id
|
JOIN game_groups g ON s.group_id = g.id
|
||||||
LEFT JOIN session_participants sp ON s.id = sp.session_id
|
LEFT JOIN session_participants sp ON s.id = sp.session_id
|
||||||
WHERE g.telegram_chat_id = @ChatId AND s.status != 'Cancelled' AND s.scheduled_at > NOW()
|
WHERE g.telegram_chat_id = @ChatId AND s.status != @Cancelled AND s.scheduled_at > NOW()
|
||||||
GROUP BY s.id, s.title, s.scheduled_at, s.status, g.gm_telegram_id
|
GROUP BY s.id, s.title, s.scheduled_at, s.status, s.max_players, s.group_id
|
||||||
ORDER BY s.scheduled_at ASC",
|
ORDER BY s.scheduled_at ASC",
|
||||||
new { ChatId = command.ChatId });
|
new
|
||||||
|
{
|
||||||
|
ChatId = command.ChatId,
|
||||||
|
command.TelegramUserId,
|
||||||
|
Cancelled = SessionStatus.Cancelled,
|
||||||
|
Active = ParticipantRegistrationStatus.Active,
|
||||||
|
Waitlisted = ParticipantRegistrationStatus.Waitlisted
|
||||||
|
});
|
||||||
|
|
||||||
var sessionsList = sessions.ToList();
|
var sessionsList = sessions.ToList();
|
||||||
|
|
||||||
if (sessionsList.Count == 0)
|
if (sessionsList.Count == 0)
|
||||||
{
|
{
|
||||||
try { await bot.EditMessageText(command.ChatId, command.MessageId, "📭 В этой группе нет предстоящих игр.", cancellationToken: ct); } catch {}
|
try { await bot.EditMessageText(command.ChatId, command.MessageId, "📭 В этой группе нет предстоящих игр.", cancellationToken: ct); } catch { }
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var text = "📅 <b>Ближайшие игры:</b>\n\n";
|
var renderResult = SessionListMessageRenderer.Render(sessionsList);
|
||||||
foreach (var s in sessionsList)
|
|
||||||
{
|
|
||||||
text += $"🔹 <b>{s.ScheduledAt.FormatMoscow()}</b> — {System.Net.WebUtility.HtmlEncode(s.Title)} (Участников: {s.PlayerCount})\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
var isGm = command.TelegramUserId == sessionsList.First().GmId;
|
|
||||||
var keyboard = isGm
|
|
||||||
? new Telegram.Bot.Types.ReplyMarkups.InlineKeyboardMarkup(
|
|
||||||
sessionsList.Select(s => new[] { Telegram.Bot.Types.ReplyMarkups.InlineKeyboardButton.WithCallbackData($"🗑 Удалить {s.ScheduledAt.FormatMoscowShort()}", $"delete_session:{s.Id}") }))
|
|
||||||
: null;
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await bot.EditMessageText(
|
await bot.EditMessageText(
|
||||||
command.ChatId,
|
command.ChatId,
|
||||||
command.MessageId,
|
command.MessageId,
|
||||||
text,
|
renderResult.Text,
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
replyMarkup: keyboard,
|
replyMarkup: renderResult.Markup,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
|
|||||||
@@ -3,10 +3,59 @@ using GmRelay.Shared.Domain;
|
|||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types;
|
using Telegram.Bot.Types;
|
||||||
|
using Telegram.Bot.Types.ReplyMarkups;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Sessions.ListSessions;
|
namespace GmRelay.Bot.Features.Sessions.ListSessions;
|
||||||
|
|
||||||
internal sealed record SessionListItemDto(Guid Id, string Title, DateTime ScheduledAt, string Status, int PlayerCount, long GmId);
|
internal sealed record SessionListItemDto(Guid Id, string Title, DateTime ScheduledAt, string Status, int? MaxPlayers, int PlayerCount, int WaitlistCount, bool CanManage);
|
||||||
|
|
||||||
|
internal static class SessionListMessageRenderer
|
||||||
|
{
|
||||||
|
public static (string Text, InlineKeyboardMarkup? Markup) Render(IReadOnlyList<SessionListItemDto> sessions)
|
||||||
|
{
|
||||||
|
var text = "📅 <b>Ближайшие игры:</b>\n\n";
|
||||||
|
foreach (var session in sessions)
|
||||||
|
{
|
||||||
|
var seats = session.MaxPlayers.HasValue
|
||||||
|
? $"{session.PlayerCount}/{session.MaxPlayers.Value}"
|
||||||
|
: session.PlayerCount.ToString(System.Globalization.CultureInfo.InvariantCulture);
|
||||||
|
var waitlist = session.WaitlistCount > 0 ? $", ожидание: {session.WaitlistCount}" : string.Empty;
|
||||||
|
text += $"🔹 <b>{session.ScheduledAt.FormatMoscow()}</b> — {System.Net.WebUtility.HtmlEncode(session.Title)} (Места: {seats}{waitlist})\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
var canManage = sessions.Count > 0 && sessions.First().CanManage;
|
||||||
|
if (!canManage)
|
||||||
|
{
|
||||||
|
return (text, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
var buttons = new List<InlineKeyboardButton[]>();
|
||||||
|
foreach (var session in sessions)
|
||||||
|
{
|
||||||
|
var dateTitle = session.ScheduledAt.FormatMoscowShort();
|
||||||
|
buttons.Add(
|
||||||
|
[
|
||||||
|
InlineKeyboardButton.WithCallbackData($"❌ {dateTitle}", $"cancel_session:{session.Id}"),
|
||||||
|
InlineKeyboardButton.WithCallbackData($"⏰ {dateTitle}", $"reschedule_session:{session.Id}")
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (SessionCapacityRules.CanPromoteWaitlistedPlayer(session.MaxPlayers, session.PlayerCount, session.WaitlistCount))
|
||||||
|
{
|
||||||
|
buttons.Add(
|
||||||
|
[
|
||||||
|
InlineKeyboardButton.WithCallbackData($"⬆️ Из ожидания {dateTitle}", $"promote_waitlist:{session.Id}")
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
buttons.Add(
|
||||||
|
[
|
||||||
|
InlineKeyboardButton.WithCallbackData($"🗑 Удалить {dateTitle}", $"delete_session:{session.Id}")
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (text, new InlineKeyboardMarkup(buttons));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public sealed class ListSessionsHandler(
|
public sealed class ListSessionsHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
@@ -17,16 +66,30 @@ public sealed class ListSessionsHandler(
|
|||||||
await using var connection = await dataSource.OpenConnectionAsync(cancellationToken);
|
await using var connection = await dataSource.OpenConnectionAsync(cancellationToken);
|
||||||
|
|
||||||
var sessions = await connection.QueryAsync<SessionListItemDto>(
|
var sessions = await connection.QueryAsync<SessionListItemDto>(
|
||||||
@"SELECT s.id as Id, s.title as Title, s.scheduled_at as ScheduledAt, s.status as Status,
|
@"SELECT s.id as Id, s.title as Title, s.scheduled_at as ScheduledAt, s.status as Status, s.max_players as MaxPlayers,
|
||||||
COUNT(sp.id) FILTER (WHERE sp.is_gm = false) as PlayerCount,
|
COUNT(sp.id) FILTER (WHERE sp.is_gm = false AND sp.registration_status = @Active) as PlayerCount,
|
||||||
g.gm_telegram_id as GmId
|
COUNT(sp.id) FILTER (WHERE sp.is_gm = false AND sp.registration_status = @Waitlisted) as WaitlistCount,
|
||||||
|
EXISTS (
|
||||||
|
SELECT 1
|
||||||
|
FROM group_managers gm
|
||||||
|
JOIN players manager_player ON manager_player.id = gm.player_id
|
||||||
|
WHERE gm.group_id = s.group_id
|
||||||
|
AND manager_player.telegram_id = @TelegramUserId
|
||||||
|
) AS CanManage
|
||||||
FROM sessions s
|
FROM sessions s
|
||||||
JOIN game_groups g ON s.group_id = g.id
|
JOIN game_groups g ON s.group_id = g.id
|
||||||
LEFT JOIN session_participants sp ON s.id = sp.session_id
|
LEFT JOIN session_participants sp ON s.id = sp.session_id
|
||||||
WHERE g.telegram_chat_id = @ChatId AND s.status != 'Cancelled' AND s.scheduled_at > NOW()
|
WHERE g.telegram_chat_id = @ChatId AND s.status != @Cancelled AND s.scheduled_at > NOW()
|
||||||
GROUP BY s.id, s.title, s.scheduled_at, s.status, g.gm_telegram_id
|
GROUP BY s.id, s.title, s.scheduled_at, s.status, s.max_players, s.group_id
|
||||||
ORDER BY s.scheduled_at ASC",
|
ORDER BY s.scheduled_at ASC",
|
||||||
new { ChatId = message.Chat.Id });
|
new
|
||||||
|
{
|
||||||
|
ChatId = message.Chat.Id,
|
||||||
|
TelegramUserId = message.From?.Id,
|
||||||
|
Cancelled = SessionStatus.Cancelled,
|
||||||
|
Active = ParticipantRegistrationStatus.Active,
|
||||||
|
Waitlisted = ParticipantRegistrationStatus.Waitlisted
|
||||||
|
});
|
||||||
|
|
||||||
var sessionsList = sessions.ToList();
|
var sessionsList = sessions.ToList();
|
||||||
|
|
||||||
@@ -39,23 +102,13 @@ public sealed class ListSessionsHandler(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var text = "📅 <b>Ближайшие игры:</b>\n\n";
|
var renderResult = SessionListMessageRenderer.Render(sessionsList);
|
||||||
foreach (var s in sessionsList)
|
|
||||||
{
|
|
||||||
text += $"🔹 <b>{s.ScheduledAt.FormatMoscow()}</b> — {System.Net.WebUtility.HtmlEncode(s.Title)} (Участников: {s.PlayerCount})\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
var isGm = message.From?.Id == sessionsList.First().GmId;
|
|
||||||
var keyboard = isGm
|
|
||||||
? new Telegram.Bot.Types.ReplyMarkups.InlineKeyboardMarkup(
|
|
||||||
sessionsList.Select(s => new[] { Telegram.Bot.Types.ReplyMarkups.InlineKeyboardButton.WithCallbackData($"🗑 Удалить {s.ScheduledAt.FormatMoscowShort()}", $"delete_session:{s.Id}") }))
|
|
||||||
: null;
|
|
||||||
|
|
||||||
await botClient.SendMessage(
|
await botClient.SendMessage(
|
||||||
chatId: message.Chat.Id,
|
chatId: message.Chat.Id,
|
||||||
text: text,
|
text: renderResult.Text,
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
replyMarkup: keyboard,
|
replyMarkup: renderResult.Markup,
|
||||||
cancellationToken: cancellationToken);
|
cancellationToken: cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+204
-52
@@ -1,10 +1,12 @@
|
|||||||
using Dapper;
|
using Dapper;
|
||||||
|
using GmRelay.Bot.Features.Notifications;
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
using GmRelay.Shared.Rendering;
|
using GmRelay.Shared.Rendering;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types;
|
using Telegram.Bot.Types;
|
||||||
using Telegram.Bot.Types.ReplyMarkups;
|
using Telegram.Bot.Types.ReplyMarkups;
|
||||||
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Sessions.RescheduleSession;
|
namespace GmRelay.Bot.Features.Sessions.RescheduleSession;
|
||||||
|
|
||||||
@@ -12,20 +14,26 @@ namespace GmRelay.Bot.Features.Sessions.RescheduleSession;
|
|||||||
|
|
||||||
internal sealed record AwaitingProposalDto(
|
internal sealed record AwaitingProposalDto(
|
||||||
Guid Id, Guid SessionId, string Title, DateTime CurrentScheduledAt,
|
Guid Id, Guid SessionId, string Title, DateTime CurrentScheduledAt,
|
||||||
Guid BatchId, int? BatchMessageId, long TelegramChatId);
|
Guid BatchId, int? BatchMessageId, long TelegramChatId, int? ThreadId, string NotificationMode);
|
||||||
|
|
||||||
internal sealed record VoteParticipantDto(Guid PlayerId, string DisplayName, string? TelegramUsername);
|
internal sealed record VoteParticipantDto(
|
||||||
|
Guid PlayerId,
|
||||||
|
string DisplayName,
|
||||||
|
string? TelegramUsername,
|
||||||
|
long TelegramId = 0);
|
||||||
|
|
||||||
// ── Handler ──────────────────────────────────────────────────────────
|
// ── Handler ──────────────────────────────────────────────────────────
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Handles text input from the GM who has an AwaitingTime proposal.
|
/// Handles text input from the GM who has an AwaitingTime proposal.
|
||||||
/// Parses the new time, creates a voting message, and tags all participants.
|
/// Parses reschedule options with a voting deadline, creates a voting message,
|
||||||
|
/// and tags all participants.
|
||||||
/// If no participants are registered, reschedules immediately.
|
/// If no participants are registered, reschedules immediately.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public sealed class HandleRescheduleTimeInputHandler(
|
public sealed class HandleRescheduleTimeInputHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
ITelegramBotClient bot,
|
ITelegramBotClient bot,
|
||||||
|
DirectSessionNotificationSender directSender,
|
||||||
ILogger<HandleRescheduleTimeInputHandler> logger)
|
ILogger<HandleRescheduleTimeInputHandler> logger)
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -48,13 +56,22 @@ public sealed class HandleRescheduleTimeInputHandler(
|
|||||||
"""
|
"""
|
||||||
SELECT rp.id AS Id, rp.session_id AS SessionId, s.title AS Title, s.scheduled_at AS CurrentScheduledAt,
|
SELECT rp.id AS Id, rp.session_id AS SessionId, s.title AS Title, s.scheduled_at AS CurrentScheduledAt,
|
||||||
s.batch_id AS BatchId, s.batch_message_id AS BatchMessageId,
|
s.batch_id AS BatchId, s.batch_message_id AS BatchMessageId,
|
||||||
g.telegram_chat_id AS TelegramChatId
|
g.telegram_chat_id AS TelegramChatId,
|
||||||
|
s.thread_id AS ThreadId,
|
||||||
|
s.notification_mode AS NotificationMode
|
||||||
FROM reschedule_proposals rp
|
FROM reschedule_proposals rp
|
||||||
JOIN sessions s ON s.id = rp.session_id
|
JOIN sessions s ON s.id = rp.session_id
|
||||||
JOIN game_groups g ON g.id = s.group_id
|
JOIN game_groups g ON g.id = s.group_id
|
||||||
WHERE rp.proposed_by = @GmId
|
WHERE rp.proposed_by = @GmId
|
||||||
AND rp.status = 'AwaitingTime'
|
AND rp.status = 'AwaitingTime'
|
||||||
AND g.telegram_chat_id = @ChatId
|
AND g.telegram_chat_id = @ChatId
|
||||||
|
AND EXISTS (
|
||||||
|
SELECT 1
|
||||||
|
FROM group_managers gm
|
||||||
|
JOIN players manager_player ON manager_player.id = gm.player_id
|
||||||
|
WHERE gm.group_id = s.group_id
|
||||||
|
AND manager_player.telegram_id = @GmId
|
||||||
|
)
|
||||||
ORDER BY rp.created_at DESC
|
ORDER BY rp.created_at DESC
|
||||||
LIMIT 1
|
LIMIT 1
|
||||||
""",
|
""",
|
||||||
@@ -63,82 +80,135 @@ public sealed class HandleRescheduleTimeInputHandler(
|
|||||||
if (proposal is null)
|
if (proposal is null)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// 2. Parse the new time
|
// 2. Parse voting input
|
||||||
if (!MoscowTime.TryParseMoscow(text, out var newTime))
|
if (!RescheduleVotingInput.TryParse(text, DateTimeOffset.UtcNow, out var votingInput, out var parseError))
|
||||||
{
|
{
|
||||||
await bot.SendMessage(
|
await bot.SendMessage(
|
||||||
chatId: chatId,
|
chatId: chatId,
|
||||||
text: "⚠️ Не удалось распознать время. Используйте формат: <code>ДД.ММ.ГГГГ ЧЧ:ММ</code>\nНапример: <code>25.04.2026 19:30</code>",
|
messageThreadId: proposal.ThreadId,
|
||||||
|
text: $"⚠️ {parseError}\n\nИспользуйте формат:\n<code>25.04.2026 19:30\n26.04.2026 18:00\nДедлайн: 25.04.2026 12:00</code>",
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newTime <= DateTimeOffset.UtcNow)
|
|
||||||
{
|
|
||||||
await bot.SendMessage(
|
|
||||||
chatId: chatId,
|
|
||||||
text: "⚠️ Новое время должно быть в будущем. Попробуйте снова.",
|
|
||||||
cancellationToken: ct);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 3. Load participants (non-GM) signed up for this session
|
// 3. Load participants (non-GM) signed up for this session
|
||||||
var participants = (await connection.QueryAsync<VoteParticipantDto>(
|
var participants = (await connection.QueryAsync<VoteParticipantDto>(
|
||||||
"""
|
"""
|
||||||
SELECT p.id AS PlayerId, p.display_name AS DisplayName, p.telegram_username AS TelegramUsername
|
SELECT p.id AS PlayerId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
p.telegram_id AS TelegramId
|
||||||
FROM session_participants sp
|
FROM session_participants sp
|
||||||
JOIN players p ON p.id = sp.player_id
|
JOIN players p ON p.id = sp.player_id
|
||||||
WHERE sp.session_id = @SessionId AND sp.is_gm = false
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
""",
|
""",
|
||||||
new { proposal.SessionId })).ToList();
|
new { proposal.SessionId, Active = ParticipantRegistrationStatus.Active })).ToList();
|
||||||
|
|
||||||
// 4. If no participants — reschedule immediately
|
// 4. If no participants — reschedule immediately
|
||||||
if (participants.Count == 0)
|
if (participants.Count == 0)
|
||||||
{
|
{
|
||||||
await RescheduleImmediately(connection, proposal, newTime, chatId, ct);
|
await RescheduleImmediately(connection, proposal, votingInput.Options[0], chatId, ct);
|
||||||
await TryDeleteMessage(chatId, message.MessageId, ct);
|
await TryDeleteMessage(chatId, message.MessageId, ct);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. Create voting message
|
// 5. Create voting message
|
||||||
await using var transaction = await connection.BeginTransactionAsync(ct);
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
var options = votingInput.Options
|
||||||
|
.Select((proposedAt, index) => new RescheduleOptionDto(
|
||||||
|
Guid.NewGuid(),
|
||||||
|
index + 1,
|
||||||
|
proposedAt))
|
||||||
|
.ToList();
|
||||||
|
|
||||||
// Update proposal with proposed time and Voting status
|
|
||||||
await connection.ExecuteAsync(
|
await connection.ExecuteAsync(
|
||||||
"""
|
"""
|
||||||
UPDATE reschedule_proposals
|
UPDATE reschedule_proposals
|
||||||
SET proposed_at = @ProposedAt, status = 'Voting', vote_chat_id = @ChatId
|
SET voting_deadline_at = @Deadline, status = 'Voting', vote_chat_id = @ChatId
|
||||||
WHERE id = @Id
|
WHERE id = @Id
|
||||||
""",
|
""",
|
||||||
new { ProposedAt = newTime, ChatId = chatId, Id = proposal.Id },
|
new { votingInput.Deadline, ChatId = chatId, Id = proposal.Id },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
|
foreach (var option in options)
|
||||||
|
{
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
INSERT INTO reschedule_options (id, proposal_id, proposed_at, display_order)
|
||||||
|
VALUES (@OptionId, @ProposalId, @ProposedAt, @DisplayOrder)
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
option.OptionId,
|
||||||
|
ProposalId = proposal.Id,
|
||||||
|
option.ProposedAt,
|
||||||
|
option.DisplayOrder
|
||||||
|
},
|
||||||
|
transaction);
|
||||||
|
}
|
||||||
|
|
||||||
await transaction.CommitAsync(ct);
|
await transaction.CommitAsync(ct);
|
||||||
|
|
||||||
// Build voting message text
|
var voteText = BuildVotingMessage(
|
||||||
var voteText = BuildVotingMessage(proposal.Title, proposal.CurrentScheduledAt, newTime, participants, []);
|
proposal.Title,
|
||||||
|
proposal.CurrentScheduledAt,
|
||||||
var keyboard = new InlineKeyboardMarkup([
|
votingInput.Deadline,
|
||||||
[
|
options,
|
||||||
InlineKeyboardButton.WithCallbackData("✅ Согласен", $"reschedule_vote:yes:{proposal.Id}"),
|
participants,
|
||||||
InlineKeyboardButton.WithCallbackData("❌ Против", $"reschedule_vote:no:{proposal.Id}")
|
[]);
|
||||||
]
|
var keyboard = BuildVotingKeyboard(options);
|
||||||
]);
|
|
||||||
|
|
||||||
var voteMsg = await bot.SendMessage(
|
var voteMsg = await bot.SendMessage(
|
||||||
chatId: chatId,
|
chatId: chatId,
|
||||||
|
messageThreadId: proposal.ThreadId,
|
||||||
text: voteText,
|
text: voteText,
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
replyMarkup: keyboard,
|
replyMarkup: keyboard,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
|
|
||||||
|
var mode = SessionNotificationModeExtensions.FromDatabaseValue(proposal.NotificationMode);
|
||||||
|
if (mode.ShouldSendDirectMessages())
|
||||||
|
{
|
||||||
|
var optionsText = string.Join(
|
||||||
|
"\n",
|
||||||
|
options.Select(option => $"{option.DisplayOrder}. <b>{option.ProposedAt.FormatMoscow()}</b> (МСК)"));
|
||||||
|
var directText = $"""
|
||||||
|
🔄 <b>Голосование за перенос сессии</b>
|
||||||
|
|
||||||
|
📌 <b>{System.Net.WebUtility.HtmlEncode(proposal.Title)}</b>
|
||||||
|
📅 Текущее время: <b>{proposal.CurrentScheduledAt.FormatMoscow()}</b> (МСК)
|
||||||
|
🗳 Варианты:
|
||||||
|
{optionsText}
|
||||||
|
|
||||||
|
⏳ Дедлайн: <b>{votingInput.Deadline.FormatMoscow()}</b> (МСК)
|
||||||
|
|
||||||
|
Проголосуйте кнопкой в групповом сообщении.
|
||||||
|
""";
|
||||||
|
|
||||||
|
await directSender.SendAsync(
|
||||||
|
participants.Select(p => new DirectNotificationRecipient(
|
||||||
|
p.TelegramId,
|
||||||
|
p.DisplayName)),
|
||||||
|
directText,
|
||||||
|
"reschedule-vote",
|
||||||
|
proposal.SessionId,
|
||||||
|
ct);
|
||||||
|
}
|
||||||
|
|
||||||
// Store vote message ID
|
// Store vote message ID
|
||||||
await connection.ExecuteAsync(
|
await connection.ExecuteAsync(
|
||||||
"UPDATE reschedule_proposals SET vote_message_id = @MsgId WHERE id = @Id",
|
"UPDATE reschedule_proposals SET vote_message_id = @MsgId WHERE id = @Id",
|
||||||
new { MsgId = voteMsg.MessageId, Id = proposal.Id });
|
new { MsgId = voteMsg.MessageId, Id = proposal.Id });
|
||||||
|
|
||||||
logger.LogInformation("Reschedule voting started for session {SessionId}, proposal {ProposalId}", proposal.SessionId, proposal.Id);
|
logger.LogInformation(
|
||||||
|
"Reschedule voting started for session {SessionId}, proposal {ProposalId}, options {OptionCount}, deadline {Deadline}",
|
||||||
|
proposal.SessionId,
|
||||||
|
proposal.Id,
|
||||||
|
options.Count,
|
||||||
|
votingInput.Deadline);
|
||||||
|
|
||||||
// Delete GM's time input message
|
// Delete GM's time input message
|
||||||
await TryDeleteMessage(chatId, message.MessageId, ct);
|
await TryDeleteMessage(chatId, message.MessageId, ct);
|
||||||
@@ -154,10 +224,16 @@ public sealed class HandleRescheduleTimeInputHandler(
|
|||||||
|
|
||||||
await connection.ExecuteAsync(
|
await connection.ExecuteAsync(
|
||||||
"""
|
"""
|
||||||
UPDATE sessions SET scheduled_at = @NewTime, status = 'Planned', updated_at = now()
|
UPDATE sessions
|
||||||
|
SET scheduled_at = @NewTime,
|
||||||
|
status = @Status,
|
||||||
|
confirmation_message_id = NULL,
|
||||||
|
confirmation_sent_at = NULL,
|
||||||
|
one_hour_reminder_processed_at = NULL,
|
||||||
|
updated_at = now()
|
||||||
WHERE id = @SessionId
|
WHERE id = @SessionId
|
||||||
""",
|
""",
|
||||||
new { NewTime = newTime, proposal.SessionId },
|
new { NewTime = newTime, proposal.SessionId, Status = SessionStatus.Planned },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
await connection.ExecuteAsync(
|
await connection.ExecuteAsync(
|
||||||
@@ -169,6 +245,7 @@ public sealed class HandleRescheduleTimeInputHandler(
|
|||||||
|
|
||||||
await bot.SendMessage(
|
await bot.SendMessage(
|
||||||
chatId: chatId,
|
chatId: chatId,
|
||||||
|
messageThreadId: proposal.ThreadId,
|
||||||
text: $"✅ Сессия «{proposal.Title}» перенесена!\n\n📅 Новое время: <b>{newTime.ToOffset(TimeSpan.FromHours(3)).ToString("d MMMM yyyy, HH:mm", System.Globalization.CultureInfo.GetCultureInfo("ru-RU"))}</b> (МСК)\n\n<i>Участников нет — голосование не требуется.</i>",
|
text: $"✅ Сессия «{proposal.Title}» перенесена!\n\n📅 Новое время: <b>{newTime.ToOffset(TimeSpan.FromHours(3)).ToString("d MMMM yyyy, HH:mm", System.Globalization.CultureInfo.GetCultureInfo("ru-RU"))}</b> (МСК)\n\n<i>Участников нет — голосование не требуется.</i>",
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
@@ -180,33 +257,105 @@ public sealed class HandleRescheduleTimeInputHandler(
|
|||||||
}
|
}
|
||||||
|
|
||||||
internal static string BuildVotingMessage(
|
internal static string BuildVotingMessage(
|
||||||
string title, DateTime currentTime, DateTimeOffset newTime,
|
string title,
|
||||||
|
DateTime currentTime,
|
||||||
|
DateTimeOffset deadline,
|
||||||
|
IReadOnlyList<RescheduleOptionDto> options,
|
||||||
IReadOnlyList<VoteParticipantDto> participants,
|
IReadOnlyList<VoteParticipantDto> participants,
|
||||||
IReadOnlyCollection<Guid> approvedPlayerIds)
|
IReadOnlyList<RescheduleOptionVoteDto> votes)
|
||||||
{
|
{
|
||||||
|
var votesByOption = votes
|
||||||
|
.GroupBy(v => v.OptionId)
|
||||||
|
.ToDictionary(g => g.Key, g => g.ToList());
|
||||||
|
var votedPlayerIds = votes.Select(v => v.PlayerId).ToHashSet();
|
||||||
|
var pendingParticipants = participants
|
||||||
|
.Where(p => !votedPlayerIds.Contains(p.PlayerId))
|
||||||
|
.Select(FormatParticipantName)
|
||||||
|
.ToList();
|
||||||
|
|
||||||
var lines = new List<string>
|
var lines = new List<string>
|
||||||
{
|
{
|
||||||
$"🔄 <b>Перенос сессии «{System.Net.WebUtility.HtmlEncode(title)}»</b>",
|
$"🔄 <b>Перенос сессии «{System.Net.WebUtility.HtmlEncode(title)}»</b>",
|
||||||
"",
|
"",
|
||||||
$"📅 Текущее время: <b>{currentTime.FormatMoscow()}</b> (МСК)",
|
$"📅 Текущее время: <b>{currentTime.FormatMoscow()}</b> (МСК)",
|
||||||
$"📅 Новое время: <b>{newTime.ToOffset(TimeSpan.FromHours(3)).ToString("d MMMM yyyy, HH:mm", System.Globalization.CultureInfo.GetCultureInfo("ru-RU"))}</b> (МСК)",
|
$"⏳ Дедлайн: <b>{deadline.FormatMoscow()}</b> (МСК)",
|
||||||
"",
|
"",
|
||||||
"Для переноса нужно согласие всех участников:"
|
"Выберите один из вариантов:"
|
||||||
};
|
};
|
||||||
|
|
||||||
foreach (var p in participants)
|
foreach (var option in options.OrderBy(x => x.DisplayOrder))
|
||||||
{
|
{
|
||||||
var name = p.TelegramUsername is not null ? $"@{p.TelegramUsername}" : p.DisplayName;
|
var optionVotes = votesByOption.GetValueOrDefault(option.OptionId, []);
|
||||||
var icon = approvedPlayerIds.Contains(p.PlayerId) ? "✅" : "⏳";
|
lines.Add(
|
||||||
lines.Add($" {icon} {name}");
|
$"{option.DisplayOrder}. <b>{option.ProposedAt.FormatMoscow()}</b> (МСК) — {FormatVoteCount(optionVotes.Count)}");
|
||||||
|
|
||||||
|
if (optionVotes.Count > 0)
|
||||||
|
{
|
||||||
|
lines.Add($" {string.Join(", ", optionVotes.Select(FormatParticipantName))}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pendingParticipants.Count > 0)
|
||||||
|
{
|
||||||
|
lines.Add("");
|
||||||
|
lines.Add($"Не проголосовали: {string.Join(", ", pendingParticipants)}");
|
||||||
}
|
}
|
||||||
|
|
||||||
lines.Add("");
|
lines.Add("");
|
||||||
lines.Add($"Голоса: {approvedPlayerIds.Count}/{participants.Count} ✅");
|
lines.Add($"Голосов: {votedPlayerIds.Count}/{participants.Count}");
|
||||||
|
lines.Add("Правило: побеждает вариант с большинством голосов к дедлайну; при ничьей перенос не применяется.");
|
||||||
|
|
||||||
return string.Join("\n", lines);
|
return string.Join("\n", lines);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal static InlineKeyboardMarkup BuildVotingKeyboard(IReadOnlyList<RescheduleOptionDto> options)
|
||||||
|
{
|
||||||
|
return new InlineKeyboardMarkup(
|
||||||
|
options
|
||||||
|
.OrderBy(option => option.DisplayOrder)
|
||||||
|
.Select(option => new[]
|
||||||
|
{
|
||||||
|
InlineKeyboardButton.WithCallbackData(
|
||||||
|
$"{option.DisplayOrder}. {FormatButtonTime(option.ProposedAt)}",
|
||||||
|
$"reschedule_vote:{option.OptionId}")
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
internal static string FormatParticipantName(VoteParticipantDto participant)
|
||||||
|
{
|
||||||
|
return participant.TelegramUsername is { Length: > 0 } username
|
||||||
|
? $"@{System.Net.WebUtility.HtmlEncode(username)}"
|
||||||
|
: System.Net.WebUtility.HtmlEncode(participant.DisplayName);
|
||||||
|
}
|
||||||
|
|
||||||
|
internal static string FormatParticipantName(RescheduleOptionVoteDto vote)
|
||||||
|
{
|
||||||
|
return vote.TelegramUsername is { Length: > 0 } username
|
||||||
|
? $"@{System.Net.WebUtility.HtmlEncode(username)}"
|
||||||
|
: System.Net.WebUtility.HtmlEncode(vote.DisplayName);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string FormatVoteCount(int count)
|
||||||
|
{
|
||||||
|
var modulo100 = count % 100;
|
||||||
|
var modulo10 = count % 10;
|
||||||
|
var word = modulo100 is >= 11 and <= 14
|
||||||
|
? "голосов"
|
||||||
|
: modulo10 switch
|
||||||
|
{
|
||||||
|
1 => "голос",
|
||||||
|
>= 2 and <= 4 => "голоса",
|
||||||
|
_ => "голосов"
|
||||||
|
};
|
||||||
|
|
||||||
|
return $"{count} {word}";
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string FormatButtonTime(DateTimeOffset utc)
|
||||||
|
=> utc.ToOffset(TimeSpan.FromHours(3)).ToString(
|
||||||
|
"dd.MM HH:mm",
|
||||||
|
System.Globalization.CultureInfo.InvariantCulture);
|
||||||
|
|
||||||
private async Task TryUpdateBatchMessage(AwaitingProposalDto proposal, CancellationToken ct)
|
private async Task TryUpdateBatchMessage(AwaitingProposalDto proposal, CancellationToken ct)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@@ -214,32 +363,35 @@ public sealed class HandleRescheduleTimeInputHandler(
|
|||||||
await using var conn = await dataSource.OpenConnectionAsync(ct);
|
await using var conn = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
|
||||||
var batchSessions = (await conn.QueryAsync<SessionBatchDto>(
|
var batchSessions = (await conn.QueryAsync<SessionBatchDto>(
|
||||||
"SELECT id AS SessionId, scheduled_at AS ScheduledAt, status AS Status FROM sessions WHERE batch_id = @BatchId ORDER BY scheduled_at",
|
"SELECT id AS SessionId, scheduled_at AS ScheduledAt, status AS Status, max_players AS MaxPlayers, join_link AS JoinLink FROM sessions WHERE batch_id = @BatchId ORDER BY scheduled_at",
|
||||||
new { proposal.BatchId })).ToList();
|
new { proposal.BatchId })).ToList();
|
||||||
|
|
||||||
var batchParticipants = (await conn.QueryAsync<ParticipantBatchDto>(
|
var batchParticipants = (await conn.QueryAsync<ParticipantBatchDto>(
|
||||||
"""
|
"""
|
||||||
SELECT sp.session_id AS SessionId, p.display_name AS DisplayName, p.telegram_username AS TelegramUsername
|
SELECT sp.session_id AS SessionId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
sp.registration_status AS RegistrationStatus
|
||||||
FROM session_participants sp
|
FROM session_participants sp
|
||||||
JOIN players p ON sp.player_id = p.id
|
JOIN players p ON sp.player_id = p.id
|
||||||
JOIN sessions s ON sp.session_id = s.id
|
JOIN sessions s ON sp.session_id = s.id
|
||||||
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
||||||
ORDER BY sp.responded_at ASC, p.created_at ASC
|
ORDER BY sp.registration_status ASC, sp.created_at ASC, sp.responded_at ASC, p.created_at ASC
|
||||||
""",
|
""",
|
||||||
new { proposal.BatchId })).ToList();
|
new { proposal.BatchId })).ToList();
|
||||||
|
|
||||||
if (proposal.BatchMessageId.HasValue)
|
if (proposal.BatchMessageId.HasValue)
|
||||||
{
|
{
|
||||||
var renderResult = SessionBatchRenderer.Render(
|
var view = SessionBatchViewBuilder.Build(proposal.Title, batchSessions, batchParticipants);
|
||||||
proposal.Title, batchSessions, batchParticipants);
|
var renderResult = TelegramSessionBatchRenderer.Render(view);
|
||||||
|
|
||||||
await bot.EditMessageText(
|
await BatchMessageEditor.EditBatchMessageAsync(
|
||||||
|
bot,
|
||||||
chatId: proposal.TelegramChatId,
|
chatId: proposal.TelegramChatId,
|
||||||
messageId: proposal.BatchMessageId.Value,
|
messageId: proposal.BatchMessageId.Value,
|
||||||
text: renderResult.Text,
|
text: renderResult.Text,
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
|
||||||
replyMarkup: renderResult.Markup,
|
replyMarkup: renderResult.Markup,
|
||||||
cancellationToken: ct);
|
ct);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|||||||
+97
-234
@@ -1,48 +1,24 @@
|
|||||||
using Dapper;
|
using Dapper;
|
||||||
using GmRelay.Shared.Domain;
|
using GmRelay.Shared.Domain;
|
||||||
using GmRelay.Shared.Rendering;
|
|
||||||
using GmRelay.Bot.Features.Sessions.CreateSession;
|
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types.ReplyMarkups;
|
|
||||||
|
|
||||||
namespace GmRelay.Bot.Features.Sessions.RescheduleSession;
|
namespace GmRelay.Bot.Features.Sessions.RescheduleSession;
|
||||||
|
|
||||||
// ── Command ──────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
public sealed record HandleRescheduleVoteCommand(
|
public sealed record HandleRescheduleVoteCommand(
|
||||||
Guid ProposalId,
|
Guid OptionId,
|
||||||
string Vote, // "yes" or "no"
|
|
||||||
long TelegramUserId,
|
long TelegramUserId,
|
||||||
string CallbackQueryId,
|
string CallbackQueryId,
|
||||||
long ChatId,
|
long ChatId,
|
||||||
int MessageId);
|
int MessageId);
|
||||||
|
|
||||||
// ── DTOs ─────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
internal sealed record VoteProposalDto(
|
internal sealed record VoteProposalDto(
|
||||||
Guid Id,
|
Guid Id,
|
||||||
Guid SessionId,
|
Guid SessionId,
|
||||||
DateTime ProposedAt,
|
DateTimeOffset VotingDeadlineAt,
|
||||||
string Title,
|
string Title,
|
||||||
DateTime CurrentScheduledAt,
|
DateTime CurrentScheduledAt);
|
||||||
Guid BatchId,
|
|
||||||
string SessionStatus,
|
|
||||||
long TelegramChatId,
|
|
||||||
int? ConfirmationMessageId,
|
|
||||||
int? BatchMessageId);
|
|
||||||
|
|
||||||
internal sealed record VoteCountDto(int Total, int Approved);
|
|
||||||
|
|
||||||
// ── Handler ──────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Handles "✅ Согласен" / "❌ Против" votes on a reschedule proposal.
|
|
||||||
///
|
|
||||||
/// If anyone votes no → proposal rejected, old time stays.
|
|
||||||
/// If all vote yes → session time updated, batch message re-rendered,
|
|
||||||
/// session status reset to Planned so confirmation triggers work correctly.
|
|
||||||
/// </summary>
|
|
||||||
public sealed class HandleRescheduleVoteHandler(
|
public sealed class HandleRescheduleVoteHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
ITelegramBotClient bot,
|
ITelegramBotClient bot,
|
||||||
@@ -53,31 +29,40 @@ public sealed class HandleRescheduleVoteHandler(
|
|||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
await using var transaction = await connection.BeginTransactionAsync(ct);
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
|
||||||
// 1. Load proposal + session info
|
|
||||||
var proposal = await connection.QuerySingleOrDefaultAsync<VoteProposalDto>(
|
var proposal = await connection.QuerySingleOrDefaultAsync<VoteProposalDto>(
|
||||||
"""
|
"""
|
||||||
SELECT rp.id AS Id, rp.session_id AS SessionId, rp.proposed_at AS ProposedAt,
|
SELECT rp.id AS Id,
|
||||||
s.title AS Title, s.scheduled_at AS CurrentScheduledAt,
|
rp.session_id AS SessionId,
|
||||||
s.batch_id AS BatchId, s.status AS SessionStatus,
|
rp.voting_deadline_at AS VotingDeadlineAt,
|
||||||
s.confirmation_message_id AS ConfirmationMessageId,
|
s.title AS Title,
|
||||||
s.batch_message_id AS BatchMessageId,
|
s.scheduled_at AS CurrentScheduledAt
|
||||||
g.telegram_chat_id AS TelegramChatId
|
FROM reschedule_options ro
|
||||||
FROM reschedule_proposals rp
|
JOIN reschedule_proposals rp ON rp.id = ro.proposal_id
|
||||||
JOIN sessions s ON s.id = rp.session_id
|
JOIN sessions s ON s.id = rp.session_id
|
||||||
JOIN game_groups g ON g.id = s.group_id
|
WHERE ro.id = @OptionId AND rp.status = 'Voting'
|
||||||
WHERE rp.id = @ProposalId AND rp.status = 'Voting'
|
|
||||||
""",
|
""",
|
||||||
new { command.ProposalId },
|
new { command.OptionId },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
if (proposal is null)
|
if (proposal is null)
|
||||||
{
|
{
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId,
|
await bot.AnswerCallbackQuery(
|
||||||
"Голосование уже завершено или не найдено.", cancellationToken: ct);
|
command.CallbackQueryId,
|
||||||
|
"Голосование уже завершено или не найдено.",
|
||||||
|
cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (proposal.VotingDeadlineAt <= DateTimeOffset.UtcNow)
|
||||||
|
{
|
||||||
|
await bot.AnswerCallbackQuery(
|
||||||
|
command.CallbackQueryId,
|
||||||
|
"Дедлайн уже прошёл. Результаты скоро будут применены.",
|
||||||
|
showAlert: true,
|
||||||
|
cancellationToken: ct);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. Verify voter is a participant of this session
|
|
||||||
var playerId = await connection.ExecuteScalarAsync<Guid?>(
|
var playerId = await connection.ExecuteScalarAsync<Guid?>(
|
||||||
"""
|
"""
|
||||||
SELECT p.id
|
SELECT p.id
|
||||||
@@ -86,229 +71,107 @@ public sealed class HandleRescheduleVoteHandler(
|
|||||||
WHERE sp.session_id = @SessionId
|
WHERE sp.session_id = @SessionId
|
||||||
AND p.telegram_id = @TelegramUserId
|
AND p.telegram_id = @TelegramUserId
|
||||||
AND sp.is_gm = false
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
""",
|
""",
|
||||||
new { proposal.SessionId, command.TelegramUserId },
|
new { proposal.SessionId, command.TelegramUserId, Active = ParticipantRegistrationStatus.Active },
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
if (playerId is null)
|
if (playerId is null)
|
||||||
{
|
{
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId,
|
await bot.AnswerCallbackQuery(
|
||||||
"Вы не являетесь участником этой сессии.", cancellationToken: ct);
|
command.CallbackQueryId,
|
||||||
|
"Вы не являетесь участником этой сессии.",
|
||||||
|
cancellationToken: ct);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. Record vote (upsert)
|
await connection.ExecuteAsync(
|
||||||
var inserted = await connection.ExecuteAsync(
|
|
||||||
"""
|
"""
|
||||||
INSERT INTO reschedule_votes (proposal_id, player_id, vote)
|
INSERT INTO reschedule_option_votes (proposal_id, player_id, option_id)
|
||||||
VALUES (@ProposalId, @PlayerId, @Vote)
|
VALUES (@ProposalId, @PlayerId, @OptionId)
|
||||||
ON CONFLICT (proposal_id, player_id) DO UPDATE SET vote = EXCLUDED.vote, voted_at = now()
|
ON CONFLICT (proposal_id, player_id) DO UPDATE
|
||||||
|
SET option_id = EXCLUDED.option_id,
|
||||||
|
voted_at = now()
|
||||||
""",
|
""",
|
||||||
new { command.ProposalId, PlayerId = playerId.Value, command.Vote },
|
new
|
||||||
|
{
|
||||||
|
ProposalId = proposal.Id,
|
||||||
|
PlayerId = playerId.Value,
|
||||||
|
command.OptionId
|
||||||
|
},
|
||||||
transaction);
|
transaction);
|
||||||
|
|
||||||
// 4. Handle "no" vote — immediately reject
|
|
||||||
if (command.Vote == "no")
|
|
||||||
{
|
|
||||||
await connection.ExecuteAsync(
|
|
||||||
"UPDATE reschedule_proposals SET status = 'Rejected' WHERE id = @Id",
|
|
||||||
new { Id = command.ProposalId },
|
|
||||||
transaction);
|
|
||||||
|
|
||||||
await transaction.CommitAsync(ct);
|
|
||||||
|
|
||||||
// Get voter's name
|
|
||||||
var voterName = await connection.QuerySingleOrDefaultAsync<string>(
|
|
||||||
"SELECT display_name FROM players WHERE telegram_id = @TgId",
|
|
||||||
new { TgId = command.TelegramUserId });
|
|
||||||
|
|
||||||
// Update voting message — show rejection
|
|
||||||
try
|
|
||||||
{
|
|
||||||
await bot.EditMessageText(
|
|
||||||
chatId: command.ChatId,
|
|
||||||
messageId: command.MessageId,
|
|
||||||
text: $"❌ <b>Перенос сессии «{System.Net.WebUtility.HtmlEncode(proposal.Title)}» отклонён!</b>\n\n{voterName ?? "Участник"} проголосовал(а) против. Время сессии остаётся прежним:\n📅 <b>{proposal.CurrentScheduledAt.FormatMoscow()}</b> (МСК)",
|
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
|
||||||
cancellationToken: ct);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
logger.LogWarning(ex, "Failed to update vote message after rejection");
|
|
||||||
}
|
|
||||||
|
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId, "Вы проголосовали против переноса.", cancellationToken: ct);
|
|
||||||
logger.LogInformation("Reschedule proposal {ProposalId} rejected by player {PlayerId}", command.ProposalId, playerId);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 5. Handle "yes" vote — check if all approved
|
|
||||||
var participants = (await connection.QueryAsync<VoteParticipantDto>(
|
var participants = (await connection.QueryAsync<VoteParticipantDto>(
|
||||||
"""
|
"""
|
||||||
SELECT p.id AS PlayerId, p.display_name AS DisplayName, p.telegram_username AS TelegramUsername
|
SELECT p.id AS PlayerId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
p.telegram_id AS TelegramId
|
||||||
FROM session_participants sp
|
FROM session_participants sp
|
||||||
JOIN players p ON p.id = sp.player_id
|
JOIN players p ON p.id = sp.player_id
|
||||||
WHERE sp.session_id = @SessionId AND sp.is_gm = false
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
|
ORDER BY p.display_name
|
||||||
""",
|
""",
|
||||||
new { proposal.SessionId },
|
new { proposal.SessionId, Active = ParticipantRegistrationStatus.Active },
|
||||||
transaction)).ToList();
|
transaction)).ToList();
|
||||||
|
|
||||||
var approvedPlayerIds = (await connection.QueryAsync<Guid>(
|
var options = (await connection.QueryAsync<RescheduleOptionDto>(
|
||||||
"""
|
"""
|
||||||
SELECT player_id FROM reschedule_votes
|
SELECT id AS OptionId,
|
||||||
WHERE proposal_id = @ProposalId AND vote = 'yes'
|
display_order AS DisplayOrder,
|
||||||
|
proposed_at AS ProposedAt
|
||||||
|
FROM reschedule_options
|
||||||
|
WHERE proposal_id = @ProposalId
|
||||||
|
ORDER BY display_order
|
||||||
""",
|
""",
|
||||||
new { command.ProposalId },
|
new { ProposalId = proposal.Id },
|
||||||
transaction)).ToHashSet();
|
transaction)).ToList();
|
||||||
|
|
||||||
var allApproved = approvedPlayerIds.Count == participants.Count;
|
var votes = (await connection.QueryAsync<RescheduleOptionVoteDto>(
|
||||||
|
"""
|
||||||
|
SELECT rov.option_id AS OptionId,
|
||||||
|
p.id AS PlayerId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername
|
||||||
|
FROM reschedule_option_votes rov
|
||||||
|
JOIN players p ON p.id = rov.player_id
|
||||||
|
WHERE rov.proposal_id = @ProposalId
|
||||||
|
ORDER BY rov.voted_at, p.display_name
|
||||||
|
""",
|
||||||
|
new { ProposalId = proposal.Id },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
if (allApproved)
|
await transaction.CommitAsync(ct);
|
||||||
{
|
|
||||||
// 6. All approved — reschedule!
|
|
||||||
var newTime = new DateTimeOffset(proposal.ProposedAt, TimeSpan.Zero); // ProposedAt is stored in UTC
|
|
||||||
|
|
||||||
// Update session time and reset status to Planned for fresh notification cycle
|
var voteText = HandleRescheduleTimeInputHandler.BuildVotingMessage(
|
||||||
await connection.ExecuteAsync(
|
proposal.Title,
|
||||||
"""
|
proposal.CurrentScheduledAt,
|
||||||
UPDATE sessions
|
proposal.VotingDeadlineAt,
|
||||||
SET scheduled_at = @NewTime,
|
options,
|
||||||
status = 'Planned',
|
participants,
|
||||||
confirmation_message_id = NULL,
|
votes);
|
||||||
link_message_id = NULL,
|
var keyboard = HandleRescheduleTimeInputHandler.BuildVotingKeyboard(options);
|
||||||
updated_at = now()
|
|
||||||
WHERE id = @SessionId
|
|
||||||
""",
|
|
||||||
new { NewTime = newTime, proposal.SessionId },
|
|
||||||
transaction);
|
|
||||||
|
|
||||||
await connection.ExecuteAsync(
|
|
||||||
"UPDATE reschedule_proposals SET status = 'Approved' WHERE id = @Id",
|
|
||||||
new { Id = command.ProposalId },
|
|
||||||
transaction);
|
|
||||||
|
|
||||||
// Reset all participant RSVP to Pending for the new confirmation cycle
|
|
||||||
await connection.ExecuteAsync(
|
|
||||||
"""
|
|
||||||
UPDATE session_participants
|
|
||||||
SET rsvp_status = 'Pending', responded_at = NULL
|
|
||||||
WHERE session_id = @SessionId AND is_gm = false
|
|
||||||
""",
|
|
||||||
new { proposal.SessionId },
|
|
||||||
transaction);
|
|
||||||
|
|
||||||
await transaction.CommitAsync(ct);
|
|
||||||
|
|
||||||
// Update voting message — show approval
|
|
||||||
try
|
|
||||||
{
|
|
||||||
await bot.EditMessageText(
|
|
||||||
chatId: command.ChatId,
|
|
||||||
messageId: command.MessageId,
|
|
||||||
text: $"✅ <b>Перенос сессии «{System.Net.WebUtility.HtmlEncode(proposal.Title)}» одобрен!</b>\n\nВсе участники согласились.\n📅 Новое время: <b>{proposal.ProposedAt.FormatMoscow()}</b> (МСК)\n\n<i>Уведомления будут приходить согласно новому расписанию.</i>",
|
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
|
||||||
cancellationToken: ct);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
logger.LogWarning(ex, "Failed to update vote message after approval");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Re-render batch message
|
|
||||||
await TryUpdateBatchMessage(proposal, ct);
|
|
||||||
|
|
||||||
logger.LogInformation("Session {SessionId} rescheduled to {NewTime} (proposal {ProposalId})",
|
|
||||||
proposal.SessionId, newTime, command.ProposalId);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Not all voted yet — update the voting message to show progress
|
|
||||||
await transaction.CommitAsync(ct);
|
|
||||||
|
|
||||||
var voteText = HandleRescheduleTimeInputHandler.BuildVotingMessage(
|
|
||||||
proposal.Title, proposal.CurrentScheduledAt,
|
|
||||||
new DateTimeOffset(proposal.ProposedAt, TimeSpan.Zero),
|
|
||||||
participants, approvedPlayerIds);
|
|
||||||
|
|
||||||
var keyboard = new InlineKeyboardMarkup([
|
|
||||||
[
|
|
||||||
InlineKeyboardButton.WithCallbackData("✅ Согласен", $"reschedule_vote:yes:{command.ProposalId}"),
|
|
||||||
InlineKeyboardButton.WithCallbackData("❌ Против", $"reschedule_vote:no:{command.ProposalId}")
|
|
||||||
]
|
|
||||||
]);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
await bot.EditMessageText(
|
|
||||||
chatId: command.ChatId,
|
|
||||||
messageId: command.MessageId,
|
|
||||||
text: voteText,
|
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
|
||||||
replyMarkup: keyboard,
|
|
||||||
cancellationToken: ct);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
logger.LogWarning(ex, "Failed to update vote message with progress");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId,
|
|
||||||
allApproved ? "Вы подтвердили перенос! Все согласны — время обновлено." : "Вы подтвердили перенос!",
|
|
||||||
cancellationToken: ct);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Re-renders the batch schedule message to reflect the updated session time.
|
|
||||||
/// If batch_message_id is stored, edits the original message. Otherwise sends a notification.
|
|
||||||
/// </summary>
|
|
||||||
private async Task TryUpdateBatchMessage(VoteProposalDto proposal, CancellationToken ct)
|
|
||||||
{
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
await bot.EditMessageText(
|
||||||
|
chatId: command.ChatId,
|
||||||
var batchSessions = (await connection.QueryAsync<SessionBatchDto>(
|
messageId: command.MessageId,
|
||||||
"SELECT id AS SessionId, scheduled_at AS ScheduledAt, status AS Status FROM sessions WHERE batch_id = @BatchId ORDER BY scheduled_at",
|
text: voteText,
|
||||||
new { proposal.BatchId })).ToList();
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
|
replyMarkup: keyboard,
|
||||||
var batchParticipants = (await connection.QueryAsync<ParticipantBatchDto>(
|
cancellationToken: ct);
|
||||||
"""
|
|
||||||
SELECT sp.session_id AS SessionId, p.display_name AS DisplayName, p.telegram_username AS TelegramUsername
|
|
||||||
FROM session_participants sp
|
|
||||||
JOIN players p ON sp.player_id = p.id
|
|
||||||
JOIN sessions s ON sp.session_id = s.id
|
|
||||||
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
|
||||||
ORDER BY sp.responded_at ASC, p.created_at ASC
|
|
||||||
""",
|
|
||||||
new { proposal.BatchId })).ToList();
|
|
||||||
|
|
||||||
if (proposal.BatchMessageId.HasValue)
|
|
||||||
{
|
|
||||||
// Edit the original batch schedule message in-place
|
|
||||||
var renderResult = SessionBatchRenderer.Render(proposal.Title, batchSessions, batchParticipants);
|
|
||||||
|
|
||||||
await bot.EditMessageText(
|
|
||||||
chatId: proposal.TelegramChatId,
|
|
||||||
messageId: proposal.BatchMessageId.Value,
|
|
||||||
text: renderResult.Text,
|
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
|
||||||
replyMarkup: renderResult.Markup,
|
|
||||||
cancellationToken: ct);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Fallback for sessions created before V005 migration (no batch_message_id)
|
|
||||||
await bot.SendMessage(
|
|
||||||
chatId: proposal.TelegramChatId,
|
|
||||||
text: $"📢 Расписание обновлено! Сессия «{proposal.Title}» перенесена на <b>{proposal.ProposedAt.FormatMoscow()}</b> (МСК).",
|
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
|
||||||
cancellationToken: ct);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
logger.LogWarning(ex, "Failed to update batch message for proposal {ProposalId}", proposal.Id);
|
logger.LogWarning(ex, "Failed to update reschedule vote message for proposal {ProposalId}", proposal.Id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await bot.AnswerCallbackQuery(
|
||||||
|
command.CallbackQueryId,
|
||||||
|
"Ваш голос учтён. До дедлайна его можно изменить.",
|
||||||
|
cancellationToken: ct);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using Dapper;
|
using Dapper;
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
|
|
||||||
@@ -11,18 +12,19 @@ public sealed record InitiateRescheduleCommand(
|
|||||||
long TelegramUserId,
|
long TelegramUserId,
|
||||||
string CallbackQueryId,
|
string CallbackQueryId,
|
||||||
long ChatId,
|
long ChatId,
|
||||||
|
int? MessageThreadId,
|
||||||
int MessageId);
|
int MessageId);
|
||||||
|
|
||||||
// ── DTOs ─────────────────────────────────────────────────────────────
|
// ── DTOs ─────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
internal sealed record RescheduleSessionInfoDto(string Title, long GmId);
|
internal sealed record RescheduleSessionInfoDto(string Title, bool CanManage);
|
||||||
|
|
||||||
// ── Handler ──────────────────────────────────────────────────────────
|
// ── Handler ──────────────────────────────────────────────────────────
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Handles the "⏰ Перенести" button press from the batch message.
|
/// Handles the "⏰ Перенести" button press from the batch message.
|
||||||
/// Creates a reschedule proposal in AwaitingTime status and prompts
|
/// Creates a reschedule proposal in AwaitingTime status and prompts
|
||||||
/// the GM to enter the new time via a regular text message.
|
/// the GM to enter 2-3 new time options and a voting deadline.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public sealed class InitiateRescheduleHandler(
|
public sealed class InitiateRescheduleHandler(
|
||||||
NpgsqlDataSource dataSource,
|
NpgsqlDataSource dataSource,
|
||||||
@@ -33,15 +35,21 @@ public sealed class InitiateRescheduleHandler(
|
|||||||
{
|
{
|
||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
|
||||||
// 1. Verify GM ownership
|
// 1. Verify group management access.
|
||||||
var session = await connection.QuerySingleOrDefaultAsync<RescheduleSessionInfoDto>(
|
var session = await connection.QuerySingleOrDefaultAsync<RescheduleSessionInfoDto>(
|
||||||
"""
|
"""
|
||||||
SELECT s.title AS Title, g.gm_telegram_id AS GmId
|
SELECT s.title AS Title,
|
||||||
|
EXISTS (
|
||||||
|
SELECT 1
|
||||||
|
FROM group_managers gm
|
||||||
|
JOIN players p ON p.id = gm.player_id
|
||||||
|
WHERE gm.group_id = s.group_id
|
||||||
|
AND p.telegram_id = @TelegramUserId
|
||||||
|
) AS CanManage
|
||||||
FROM sessions s
|
FROM sessions s
|
||||||
JOIN game_groups g ON s.group_id = g.id
|
WHERE s.id = @SessionId AND s.status != @Cancelled
|
||||||
WHERE s.id = @SessionId AND s.status != 'Cancelled'
|
|
||||||
""",
|
""",
|
||||||
new { command.SessionId });
|
new { command.SessionId, command.TelegramUserId, Cancelled = SessionStatus.Cancelled });
|
||||||
|
|
||||||
if (session is null)
|
if (session is null)
|
||||||
{
|
{
|
||||||
@@ -49,10 +57,10 @@ public sealed class InitiateRescheduleHandler(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (session.GmId != command.TelegramUserId)
|
if (!session.CanManage)
|
||||||
{
|
{
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId,
|
await bot.AnswerCallbackQuery(command.CallbackQueryId,
|
||||||
"Только Мастер Игры (GM) может переносить сессию.", showAlert: true, cancellationToken: ct);
|
"Только owner или co-GM может переносить сессию.", showAlert: true, cancellationToken: ct);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -85,11 +93,21 @@ public sealed class InitiateRescheduleHandler(
|
|||||||
|
|
||||||
// 4. Prompt GM in chat
|
// 4. Prompt GM in chat
|
||||||
await bot.AnswerCallbackQuery(command.CallbackQueryId,
|
await bot.AnswerCallbackQuery(command.CallbackQueryId,
|
||||||
"Введите новое время в чат (формат: ДД.ММ.ГГГГ ЧЧ:ММ)", cancellationToken: ct);
|
"Введите 2-3 варианта времени и дедлайн голосования.", cancellationToken: ct);
|
||||||
|
|
||||||
await bot.SendMessage(
|
await bot.SendMessage(
|
||||||
chatId: command.ChatId,
|
chatId: command.ChatId,
|
||||||
text: $"⏰ Укажите новое время для сессии «{session.Title}» в формате:\n<code>ДД.ММ.ГГГГ ЧЧ:ММ</code>\n\nНапример: <code>25.04.2026 19:30</code>",
|
messageThreadId: command.MessageThreadId,
|
||||||
|
text: $"""
|
||||||
|
⏰ Укажите 2-3 варианта времени для сессии «{session.Title}» и дедлайн голосования.
|
||||||
|
|
||||||
|
Формат:
|
||||||
|
<code>25.04.2026 19:30
|
||||||
|
26.04.2026 18:00
|
||||||
|
Дедлайн: 25.04.2026 12:00</code>
|
||||||
|
|
||||||
|
Дедлайн должен быть в будущем и раньше первого предложенного времени.
|
||||||
|
""",
|
||||||
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
parseMode: Telegram.Bot.Types.Enums.ParseMode.Html,
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,69 @@
|
|||||||
|
namespace GmRelay.Bot.Features.Sessions.RescheduleSession;
|
||||||
|
|
||||||
|
internal enum RescheduleVoteOutcome
|
||||||
|
{
|
||||||
|
Pending,
|
||||||
|
Rejected,
|
||||||
|
Approved
|
||||||
|
}
|
||||||
|
|
||||||
|
internal sealed record RescheduleVoteDecision(
|
||||||
|
RescheduleVoteOutcome Outcome,
|
||||||
|
string Reason,
|
||||||
|
Guid? SelectedOptionId = null,
|
||||||
|
string CallbackText = "",
|
||||||
|
bool ShouldRescheduleSession = false,
|
||||||
|
bool ShouldResetParticipantRsvps = false);
|
||||||
|
|
||||||
|
internal static class RescheduleVoteRules
|
||||||
|
{
|
||||||
|
public static RescheduleVoteDecision SelectWinner(IReadOnlyList<RescheduleOptionVoteCount> voteCounts)
|
||||||
|
{
|
||||||
|
var maxVotes = voteCounts.Count == 0 ? 0 : voteCounts.Max(x => x.VoteCount);
|
||||||
|
if (maxVotes == 0)
|
||||||
|
{
|
||||||
|
return new RescheduleVoteDecision(
|
||||||
|
RescheduleVoteOutcome.Rejected,
|
||||||
|
"Никто не проголосовал до дедлайна, перенос не применяется.");
|
||||||
|
}
|
||||||
|
|
||||||
|
var winners = voteCounts.Where(x => x.VoteCount == maxVotes).ToList();
|
||||||
|
if (winners.Count > 1)
|
||||||
|
{
|
||||||
|
return new RescheduleVoteDecision(
|
||||||
|
RescheduleVoteOutcome.Rejected,
|
||||||
|
"Голоса разделились поровну, перенос не применяется.");
|
||||||
|
}
|
||||||
|
|
||||||
|
return new RescheduleVoteDecision(
|
||||||
|
RescheduleVoteOutcome.Approved,
|
||||||
|
"Победил вариант с большинством голосов.",
|
||||||
|
winners[0].OptionId,
|
||||||
|
ShouldRescheduleSession: true,
|
||||||
|
ShouldResetParticipantRsvps: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static RescheduleVoteDecision Evaluate(string vote, int totalParticipants, int approvedParticipants)
|
||||||
|
{
|
||||||
|
if (string.Equals(vote, "no", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
return new RescheduleVoteDecision(
|
||||||
|
Outcome: RescheduleVoteOutcome.Rejected,
|
||||||
|
Reason: "\u041e\u0434\u0438\u043d \u0438\u0437 \u0443\u0447\u0430\u0441\u0442\u043d\u0438\u043a\u043e\u0432 \u043e\u0442\u043a\u043b\u043e\u043d\u0438\u043b \u043f\u0435\u0440\u0435\u043d\u043e\u0441.",
|
||||||
|
CallbackText: "\u0412\u044b \u043f\u0440\u043e\u0433\u043e\u043b\u043e\u0441\u043e\u0432\u0430\u043b\u0438 \u043f\u0440\u043e\u0442\u0438\u0432 \u043f\u0435\u0440\u0435\u043d\u043e\u0441\u0430.");
|
||||||
|
}
|
||||||
|
|
||||||
|
var everyoneApproved = approvedParticipants == totalParticipants;
|
||||||
|
|
||||||
|
return new RescheduleVoteDecision(
|
||||||
|
Outcome: everyoneApproved ? RescheduleVoteOutcome.Approved : RescheduleVoteOutcome.Pending,
|
||||||
|
Reason: everyoneApproved
|
||||||
|
? "\u0412\u0441\u0435 \u0443\u0447\u0430\u0441\u0442\u043d\u0438\u043a\u0438 \u0441\u043e\u0433\u043b\u0430\u0441\u043d\u044b."
|
||||||
|
: "\u0413\u043e\u043b\u043e\u0441\u043e\u0432\u0430\u043d\u0438\u0435 \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0430\u0435\u0442\u0441\u044f.",
|
||||||
|
CallbackText: everyoneApproved
|
||||||
|
? "\u0412\u044b \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0434\u0438\u043b\u0438 \u043f\u0435\u0440\u0435\u043d\u043e\u0441! \u0412\u0441\u0435 \u0441\u043e\u0433\u043b\u0430\u0441\u043d\u044b \u2014 \u0432\u0440\u0435\u043c\u044f \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u043e."
|
||||||
|
: "\u0412\u044b \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0434\u0438\u043b\u0438 \u043f\u0435\u0440\u0435\u043d\u043e\u0441!",
|
||||||
|
ShouldRescheduleSession: everyoneApproved,
|
||||||
|
ShouldResetParticipantRsvps: everyoneApproved);
|
||||||
|
}
|
||||||
|
}
|
||||||
+370
@@ -0,0 +1,370 @@
|
|||||||
|
using Dapper;
|
||||||
|
using GmRelay.Bot.Features.Notifications;
|
||||||
|
using GmRelay.Bot.Infrastructure.Scheduling;
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
using GmRelay.Shared.Rendering;
|
||||||
|
using Npgsql;
|
||||||
|
using Telegram.Bot;
|
||||||
|
using Telegram.Bot.Types.Enums;
|
||||||
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Features.Sessions.RescheduleSession;
|
||||||
|
|
||||||
|
internal sealed record DueRescheduleProposalDto(
|
||||||
|
Guid Id,
|
||||||
|
Guid SessionId,
|
||||||
|
DateTimeOffset VotingDeadlineAt,
|
||||||
|
string Title,
|
||||||
|
DateTime CurrentScheduledAt,
|
||||||
|
Guid BatchId,
|
||||||
|
int? BatchMessageId,
|
||||||
|
int? VoteMessageId,
|
||||||
|
long TelegramChatId,
|
||||||
|
int? ThreadId,
|
||||||
|
string NotificationMode);
|
||||||
|
|
||||||
|
public sealed class RescheduleVotingDeadlineService(
|
||||||
|
NpgsqlDataSource dataSource,
|
||||||
|
ITelegramBotClient bot,
|
||||||
|
DirectSessionNotificationSender directSender,
|
||||||
|
ISystemClock clock,
|
||||||
|
ILogger<RescheduleVotingDeadlineService> logger) : BackgroundService
|
||||||
|
{
|
||||||
|
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await ProcessDueProposals(stoppingToken);
|
||||||
|
|
||||||
|
using var timer = new PeriodicTimer(TimeSpan.FromMinutes(1));
|
||||||
|
while (await timer.WaitForNextTickAsync(stoppingToken))
|
||||||
|
{
|
||||||
|
await ProcessDueProposals(stoppingToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (OperationCanceledException) when (stoppingToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task ProcessDueProposals(CancellationToken ct)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
var proposalIds = (await connection.QueryAsync<Guid>(
|
||||||
|
"""
|
||||||
|
SELECT id
|
||||||
|
FROM reschedule_proposals
|
||||||
|
WHERE status = 'Voting'
|
||||||
|
AND voting_deadline_at IS NOT NULL
|
||||||
|
AND voting_deadline_at <= @Now
|
||||||
|
ORDER BY voting_deadline_at
|
||||||
|
LIMIT 25
|
||||||
|
""",
|
||||||
|
new { Now = clock.UtcNow.UtcDateTime })).ToList();
|
||||||
|
|
||||||
|
foreach (var proposalId in proposalIds)
|
||||||
|
{
|
||||||
|
await FinalizeProposal(proposalId, ct);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (OperationCanceledException) when (ct.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogError(ex, "Failed to process due reschedule voting proposals");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task FinalizeProposal(Guid proposalId, CancellationToken ct)
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
await using var transaction = await connection.BeginTransactionAsync(ct);
|
||||||
|
|
||||||
|
var proposal = await connection.QuerySingleOrDefaultAsync<DueRescheduleProposalDto>(
|
||||||
|
"""
|
||||||
|
SELECT rp.id AS Id,
|
||||||
|
rp.session_id AS SessionId,
|
||||||
|
rp.voting_deadline_at AS VotingDeadlineAt,
|
||||||
|
rp.vote_message_id AS VoteMessageId,
|
||||||
|
s.title AS Title,
|
||||||
|
s.scheduled_at AS CurrentScheduledAt,
|
||||||
|
s.batch_id AS BatchId,
|
||||||
|
s.batch_message_id AS BatchMessageId,
|
||||||
|
s.notification_mode AS NotificationMode,
|
||||||
|
s.thread_id AS ThreadId,
|
||||||
|
g.telegram_chat_id AS TelegramChatId
|
||||||
|
FROM reschedule_proposals rp
|
||||||
|
JOIN sessions s ON s.id = rp.session_id
|
||||||
|
JOIN game_groups g ON g.id = s.group_id
|
||||||
|
WHERE rp.id = @ProposalId
|
||||||
|
AND rp.status = 'Voting'
|
||||||
|
AND rp.voting_deadline_at IS NOT NULL
|
||||||
|
AND rp.voting_deadline_at <= @Now
|
||||||
|
FOR UPDATE
|
||||||
|
""",
|
||||||
|
new { ProposalId = proposalId, Now = clock.UtcNow.UtcDateTime },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
if (proposal is null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var participants = (await connection.QueryAsync<VoteParticipantDto>(
|
||||||
|
"""
|
||||||
|
SELECT p.id AS PlayerId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
p.telegram_id AS TelegramId
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON p.id = sp.player_id
|
||||||
|
WHERE sp.session_id = @SessionId
|
||||||
|
AND sp.is_gm = false
|
||||||
|
AND sp.registration_status = @Active
|
||||||
|
ORDER BY p.display_name
|
||||||
|
""",
|
||||||
|
new { proposal.SessionId, Active = ParticipantRegistrationStatus.Active },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
|
var options = (await connection.QueryAsync<RescheduleOptionDto>(
|
||||||
|
"""
|
||||||
|
SELECT id AS OptionId,
|
||||||
|
display_order AS DisplayOrder,
|
||||||
|
proposed_at AS ProposedAt
|
||||||
|
FROM reschedule_options
|
||||||
|
WHERE proposal_id = @ProposalId
|
||||||
|
ORDER BY display_order
|
||||||
|
""",
|
||||||
|
new { ProposalId = proposal.Id },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
|
var votes = (await connection.QueryAsync<RescheduleOptionVoteDto>(
|
||||||
|
"""
|
||||||
|
SELECT rov.option_id AS OptionId,
|
||||||
|
p.id AS PlayerId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername
|
||||||
|
FROM reschedule_option_votes rov
|
||||||
|
JOIN players p ON p.id = rov.player_id
|
||||||
|
WHERE rov.proposal_id = @ProposalId
|
||||||
|
ORDER BY rov.voted_at, p.display_name
|
||||||
|
""",
|
||||||
|
new { ProposalId = proposal.Id },
|
||||||
|
transaction)).ToList();
|
||||||
|
|
||||||
|
var voteCounts = options
|
||||||
|
.Select(option => new RescheduleOptionVoteCount(
|
||||||
|
option.OptionId,
|
||||||
|
votes.Count(vote => vote.OptionId == option.OptionId)))
|
||||||
|
.ToList();
|
||||||
|
var decision = RescheduleVoteRules.SelectWinner(voteCounts);
|
||||||
|
var selectedOption = decision.SelectedOptionId is { } selectedOptionId
|
||||||
|
? options.Single(x => x.OptionId == selectedOptionId)
|
||||||
|
: null;
|
||||||
|
|
||||||
|
if (selectedOption is not null)
|
||||||
|
{
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
UPDATE sessions
|
||||||
|
SET scheduled_at = @NewTime,
|
||||||
|
status = @Status,
|
||||||
|
confirmation_message_id = NULL,
|
||||||
|
confirmation_sent_at = NULL,
|
||||||
|
link_message_id = NULL,
|
||||||
|
one_hour_reminder_processed_at = NULL,
|
||||||
|
updated_at = now()
|
||||||
|
WHERE id = @SessionId
|
||||||
|
""",
|
||||||
|
new { NewTime = selectedOption.ProposedAt, proposal.SessionId, Status = SessionStatus.Planned },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
UPDATE session_participants
|
||||||
|
SET rsvp_status = 'Pending',
|
||||||
|
responded_at = NULL
|
||||||
|
WHERE session_id = @SessionId
|
||||||
|
AND is_gm = false
|
||||||
|
AND registration_status = @Active
|
||||||
|
""",
|
||||||
|
new { proposal.SessionId, Active = ParticipantRegistrationStatus.Active },
|
||||||
|
transaction);
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"""
|
||||||
|
UPDATE reschedule_proposals
|
||||||
|
SET status = 'Approved',
|
||||||
|
selected_option_id = @SelectedOptionId,
|
||||||
|
proposed_at = @ProposedAt
|
||||||
|
WHERE id = @ProposalId
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
ProposalId = proposal.Id,
|
||||||
|
SelectedOptionId = selectedOption.OptionId,
|
||||||
|
ProposedAt = selectedOption.ProposedAt
|
||||||
|
},
|
||||||
|
transaction);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
"UPDATE reschedule_proposals SET status = 'Rejected' WHERE id = @ProposalId",
|
||||||
|
new { ProposalId = proposal.Id },
|
||||||
|
transaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
var directRecipients = participants
|
||||||
|
.Select(p => new DirectNotificationRecipient(p.TelegramId, p.DisplayName))
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
await transaction.CommitAsync(ct);
|
||||||
|
|
||||||
|
await TryUpdateVoteMessage(proposal, options, participants, votes, decision, selectedOption, ct);
|
||||||
|
|
||||||
|
if (selectedOption is not null)
|
||||||
|
{
|
||||||
|
await TryUpdateBatchMessage(proposal, ct);
|
||||||
|
}
|
||||||
|
|
||||||
|
var mode = SessionNotificationModeExtensions.FromDatabaseValue(proposal.NotificationMode);
|
||||||
|
if (mode.ShouldSendDirectMessages())
|
||||||
|
{
|
||||||
|
await SendDirectResult(proposal, directRecipients, decision, selectedOption, ct);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.LogInformation(
|
||||||
|
"Finalized reschedule proposal {ProposalId} for session {SessionId} with outcome {Outcome}",
|
||||||
|
proposal.Id,
|
||||||
|
proposal.SessionId,
|
||||||
|
decision.Outcome);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task TryUpdateVoteMessage(
|
||||||
|
DueRescheduleProposalDto proposal,
|
||||||
|
IReadOnlyList<RescheduleOptionDto> options,
|
||||||
|
IReadOnlyList<VoteParticipantDto> participants,
|
||||||
|
IReadOnlyList<RescheduleOptionVoteDto> votes,
|
||||||
|
RescheduleVoteDecision decision,
|
||||||
|
RescheduleOptionDto? selectedOption,
|
||||||
|
CancellationToken ct)
|
||||||
|
{
|
||||||
|
if (proposal.VoteMessageId is null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var resultText = selectedOption is not null
|
||||||
|
? $"✅ <b>Голосование завершено.</b>\nПобедил вариант {selectedOption.DisplayOrder}: <b>{selectedOption.ProposedAt.FormatMoscow()}</b> (МСК)."
|
||||||
|
: $"❌ <b>Голосование завершено.</b>\n{System.Net.WebUtility.HtmlEncode(decision.Reason)}";
|
||||||
|
|
||||||
|
var text = $"""
|
||||||
|
{HandleRescheduleTimeInputHandler.BuildVotingMessage(
|
||||||
|
proposal.Title,
|
||||||
|
proposal.CurrentScheduledAt,
|
||||||
|
proposal.VotingDeadlineAt,
|
||||||
|
options,
|
||||||
|
participants,
|
||||||
|
votes)}
|
||||||
|
|
||||||
|
{resultText}
|
||||||
|
""";
|
||||||
|
|
||||||
|
await bot.EditMessageText(
|
||||||
|
chatId: proposal.TelegramChatId,
|
||||||
|
messageId: proposal.VoteMessageId.Value,
|
||||||
|
text: text,
|
||||||
|
parseMode: ParseMode.Html,
|
||||||
|
cancellationToken: ct);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogWarning(ex, "Failed to update finalized reschedule vote message for proposal {ProposalId}", proposal.Id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task TryUpdateBatchMessage(DueRescheduleProposalDto proposal, CancellationToken ct)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
|
||||||
|
var batchSessions = (await connection.QueryAsync<SessionBatchDto>(
|
||||||
|
"SELECT id AS SessionId, scheduled_at AS ScheduledAt, status AS Status, max_players AS MaxPlayers, join_link AS JoinLink FROM sessions WHERE batch_id = @BatchId ORDER BY scheduled_at",
|
||||||
|
new { proposal.BatchId })).ToList();
|
||||||
|
|
||||||
|
var batchParticipants = (await connection.QueryAsync<ParticipantBatchDto>(
|
||||||
|
"""
|
||||||
|
SELECT sp.session_id AS SessionId,
|
||||||
|
p.display_name AS DisplayName,
|
||||||
|
p.telegram_username AS TelegramUsername,
|
||||||
|
sp.registration_status AS RegistrationStatus
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN players p ON sp.player_id = p.id
|
||||||
|
JOIN sessions s ON sp.session_id = s.id
|
||||||
|
WHERE s.batch_id = @BatchId AND sp.is_gm = false
|
||||||
|
ORDER BY sp.registration_status ASC, sp.created_at ASC, sp.responded_at ASC, p.created_at ASC
|
||||||
|
""",
|
||||||
|
new { proposal.BatchId })).ToList();
|
||||||
|
|
||||||
|
if (proposal.BatchMessageId.HasValue)
|
||||||
|
{
|
||||||
|
var view = SessionBatchViewBuilder.Build(proposal.Title, batchSessions, batchParticipants);
|
||||||
|
var renderResult = TelegramSessionBatchRenderer.Render(view);
|
||||||
|
|
||||||
|
await BatchMessageEditor.EditBatchMessageAsync(
|
||||||
|
bot,
|
||||||
|
chatId: proposal.TelegramChatId,
|
||||||
|
messageId: proposal.BatchMessageId.Value,
|
||||||
|
text: renderResult.Text,
|
||||||
|
replyMarkup: renderResult.Markup,
|
||||||
|
ct);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await bot.SendMessage(
|
||||||
|
chatId: proposal.TelegramChatId,
|
||||||
|
messageThreadId: proposal.ThreadId,
|
||||||
|
text: $"📣 Расписание обновлено после голосования за перенос сессии «{System.Net.WebUtility.HtmlEncode(proposal.Title)}».",
|
||||||
|
parseMode: ParseMode.Html,
|
||||||
|
cancellationToken: ct);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogWarning(ex, "Failed to update batch message for finalized proposal {ProposalId}", proposal.Id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task SendDirectResult(
|
||||||
|
DueRescheduleProposalDto proposal,
|
||||||
|
IReadOnlyList<DirectNotificationRecipient> recipients,
|
||||||
|
RescheduleVoteDecision decision,
|
||||||
|
RescheduleOptionDto? selectedOption,
|
||||||
|
CancellationToken ct)
|
||||||
|
{
|
||||||
|
var htmlText = selectedOption is not null
|
||||||
|
? $"""
|
||||||
|
✅ <b>Сессия перенесена по итогам голосования</b>
|
||||||
|
|
||||||
|
📌 <b>{System.Net.WebUtility.HtmlEncode(proposal.Title)}</b>
|
||||||
|
📅 Новое время: <b>{selectedOption.ProposedAt.FormatMoscow()}</b> (МСК)
|
||||||
|
"""
|
||||||
|
: $"""
|
||||||
|
❌ <b>Перенос сессии отклонён по итогам голосования</b>
|
||||||
|
|
||||||
|
📌 <b>{System.Net.WebUtility.HtmlEncode(proposal.Title)}</b>
|
||||||
|
📅 Время остаётся прежним: <b>{proposal.CurrentScheduledAt.FormatMoscow()}</b> (МСК)
|
||||||
|
Причина: {System.Net.WebUtility.HtmlEncode(decision.Reason)}
|
||||||
|
""";
|
||||||
|
|
||||||
|
await directSender.SendAsync(
|
||||||
|
recipients,
|
||||||
|
htmlText,
|
||||||
|
selectedOption is not null ? "reschedule-vote-approved" : "reschedule-vote-rejected",
|
||||||
|
proposal.SessionId,
|
||||||
|
ct);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,110 @@
|
|||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Features.Sessions.RescheduleSession;
|
||||||
|
|
||||||
|
internal sealed record RescheduleVotingInput(
|
||||||
|
IReadOnlyList<DateTimeOffset> Options,
|
||||||
|
DateTimeOffset Deadline)
|
||||||
|
{
|
||||||
|
private static readonly Regex DateTimePattern = new(
|
||||||
|
@"(?<date>\d{1,2}\.\d{2}\.\d{4})\s+(?<time>\d{1,2}:\d{2})",
|
||||||
|
RegexOptions.CultureInvariant);
|
||||||
|
|
||||||
|
public static bool TryParse(
|
||||||
|
string text,
|
||||||
|
DateTimeOffset nowUtc,
|
||||||
|
out RescheduleVotingInput input,
|
||||||
|
out string error)
|
||||||
|
{
|
||||||
|
input = new RescheduleVotingInput([], default);
|
||||||
|
error = string.Empty;
|
||||||
|
|
||||||
|
var options = new List<DateTimeOffset>();
|
||||||
|
DateTimeOffset? deadline = null;
|
||||||
|
|
||||||
|
foreach (var rawLine in text.Split('\n', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries))
|
||||||
|
{
|
||||||
|
var line = rawLine.Trim();
|
||||||
|
var match = DateTimePattern.Match(line);
|
||||||
|
if (!match.Success)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var value = $"{match.Groups["date"].Value} {match.Groups["time"].Value}";
|
||||||
|
if (!MoscowTime.TryParseMoscow(value, out var parsed))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (IsDeadlineLine(line))
|
||||||
|
{
|
||||||
|
deadline = parsed;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
options.Add(parsed);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (options.Count is < 2 or > 3)
|
||||||
|
{
|
||||||
|
error = "Укажите от 2 до 3 вариантов времени.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (options.Distinct().Count() != options.Count)
|
||||||
|
{
|
||||||
|
error = "Варианты времени не должны повторяться.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deadline is null)
|
||||||
|
{
|
||||||
|
error = "Укажите дедлайн голосования строкой «Дедлайн: ДД.ММ.ГГГГ ЧЧ:ММ».";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (options.Any(option => option <= nowUtc))
|
||||||
|
{
|
||||||
|
error = "Все варианты времени должны быть в будущем.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deadline.Value <= nowUtc)
|
||||||
|
{
|
||||||
|
error = "Дедлайн голосования должен быть в будущем.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deadline.Value >= options.Min())
|
||||||
|
{
|
||||||
|
error = "Дедлайн голосования должен быть раньше первого предложенного времени.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
input = new RescheduleVotingInput(options, deadline.Value);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static bool IsDeadlineLine(string line)
|
||||||
|
{
|
||||||
|
var normalized = line.TrimStart('-', '*', ' ', '\t').ToLowerInvariant();
|
||||||
|
|
||||||
|
return normalized.StartsWith("дедлайн", StringComparison.Ordinal)
|
||||||
|
|| normalized.StartsWith("deadline", StringComparison.Ordinal)
|
||||||
|
|| normalized.StartsWith("до:", StringComparison.Ordinal);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal sealed record RescheduleOptionDto(
|
||||||
|
Guid OptionId,
|
||||||
|
int DisplayOrder,
|
||||||
|
DateTimeOffset ProposedAt);
|
||||||
|
|
||||||
|
internal sealed record RescheduleOptionVoteDto(
|
||||||
|
Guid OptionId,
|
||||||
|
Guid PlayerId,
|
||||||
|
string DisplayName,
|
||||||
|
string? TelegramUsername);
|
||||||
|
|
||||||
|
internal sealed record RescheduleOptionVoteCount(
|
||||||
|
Guid OptionId,
|
||||||
|
int VoteCount);
|
||||||
@@ -22,7 +22,7 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Aspire.Npgsql" Version="13.2.1" />
|
<PackageReference Include="Aspire.Npgsql" Version="13.2.2" />
|
||||||
<PackageReference Include="Dapper" Version="2.1.72" />
|
<PackageReference Include="Dapper" Version="2.1.72" />
|
||||||
<PackageReference Include="Dapper.AOT" Version="1.0.48" />
|
<PackageReference Include="Dapper.AOT" Version="1.0.48" />
|
||||||
<PackageReference Include="dbup-postgresql" Version="7.0.1" />
|
<PackageReference Include="dbup-postgresql" Version="7.0.1" />
|
||||||
|
|||||||
@@ -0,0 +1,101 @@
|
|||||||
|
using System.Net;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Health;
|
||||||
|
|
||||||
|
public sealed class BotHealthCheckHostedService : IHostedService
|
||||||
|
{
|
||||||
|
private readonly ILogger<BotHealthCheckHostedService> _logger;
|
||||||
|
private readonly string _prefix;
|
||||||
|
private HttpListener? _listener;
|
||||||
|
private CancellationTokenSource? _cts;
|
||||||
|
private Task? _listenerTask;
|
||||||
|
|
||||||
|
public BotHealthCheckHostedService(
|
||||||
|
ILogger<BotHealthCheckHostedService> logger,
|
||||||
|
IConfiguration configuration)
|
||||||
|
{
|
||||||
|
_logger = logger;
|
||||||
|
_prefix = configuration.GetValue("HealthCheck:Prefix", "http://+:8081/")!;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task StartAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
_cts = new CancellationTokenSource();
|
||||||
|
_listener = new HttpListener();
|
||||||
|
_listener.Prefixes.Add(_prefix);
|
||||||
|
_listener.Start();
|
||||||
|
|
||||||
|
_logger.LogInformation("Health check server started on {Prefix}", _prefix);
|
||||||
|
|
||||||
|
_listenerTask = Task.Run(async () => await ListenAsync(_cts.Token), cancellationToken);
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task StopAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
_cts?.Cancel();
|
||||||
|
_listener?.Stop();
|
||||||
|
|
||||||
|
if (_listenerTask != null)
|
||||||
|
{
|
||||||
|
await Task.WhenAny(_listenerTask, Task.Delay(TimeSpan.FromSeconds(5), cancellationToken));
|
||||||
|
}
|
||||||
|
|
||||||
|
_listener?.Close();
|
||||||
|
_logger.LogInformation("Health check server stopped");
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task ListenAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
while (_listener?.IsListening == true && !cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var context = await _listener.GetContextAsync();
|
||||||
|
_ = Task.Run(() => HandleRequestAsync(context), cancellationToken);
|
||||||
|
}
|
||||||
|
catch (HttpListenerException) when (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
catch (ObjectDisposedException)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error in health check listener");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleRequestAsync(HttpListenerContext context)
|
||||||
|
{
|
||||||
|
var response = context.Response;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var request = context.Request;
|
||||||
|
|
||||||
|
if (request.Url?.AbsolutePath == "/health")
|
||||||
|
{
|
||||||
|
response.StatusCode = (int)HttpStatusCode.OK;
|
||||||
|
response.ContentType = "application/json";
|
||||||
|
var body = "{\"status\":\"healthy\"}"u8.ToArray();
|
||||||
|
await response.OutputStream.WriteAsync(body);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
response.StatusCode = (int)HttpStatusCode.NotFound;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error handling health check request");
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
response.Close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using Npgsql;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Logging;
|
||||||
|
|
||||||
|
public static partial class SecretRedactor
|
||||||
|
{
|
||||||
|
public static string RedactConnectionString(string? connectionString)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(connectionString))
|
||||||
|
{
|
||||||
|
return string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var builder = new NpgsqlConnectionStringBuilder(connectionString);
|
||||||
|
if (!string.IsNullOrWhiteSpace(builder.Password))
|
||||||
|
{
|
||||||
|
builder.Password = "***";
|
||||||
|
}
|
||||||
|
|
||||||
|
return builder.ToString();
|
||||||
|
}
|
||||||
|
catch (ArgumentException)
|
||||||
|
{
|
||||||
|
return RedactText(connectionString);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string RedactText(string? text)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(text))
|
||||||
|
{
|
||||||
|
return string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SecretKeyValueRegex().Replace(
|
||||||
|
text,
|
||||||
|
static match => $"{match.Groups["key"].Value}={GetRedactedValue()}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string GetRedactedValue() => "***";
|
||||||
|
|
||||||
|
[GeneratedRegex(@"(?<key>password|pwd|passwd|token|secret|api[-_]?key)\s*=\s*(?<value>[^;\s,]+)", RegexOptions.IgnoreCase | RegexOptions.CultureInvariant)]
|
||||||
|
private static partial Regex SecretKeyValueRegex();
|
||||||
|
}
|
||||||
@@ -0,0 +1,86 @@
|
|||||||
|
using Dapper;
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
using Npgsql;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Scheduling;
|
||||||
|
|
||||||
|
public interface ISessionTriggerStore
|
||||||
|
{
|
||||||
|
Task<IReadOnlyList<Guid>> GetSessionsNeedingConfirmationAsync(DateTimeOffset now, CancellationToken ct);
|
||||||
|
Task<IReadOnlyList<Guid>> GetSessionsNeedingOneHourReminderAsync(DateTimeOffset now, CancellationToken ct);
|
||||||
|
Task<IReadOnlyList<Guid>> GetSessionsNeedingJoinLinkAsync(DateTimeOffset now, CancellationToken ct);
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class DbSessionTriggerStore(NpgsqlDataSource dataSource) : ISessionTriggerStore
|
||||||
|
{
|
||||||
|
private static readonly TimeSpan ConfirmationLeadTime = TimeSpan.FromHours(24);
|
||||||
|
private static readonly TimeSpan OneHourReminderLeadTime = TimeSpan.FromHours(1);
|
||||||
|
private static readonly TimeSpan JoinLinkLeadTime = TimeSpan.FromMinutes(5);
|
||||||
|
|
||||||
|
public async Task<IReadOnlyList<Guid>> GetSessionsNeedingConfirmationAsync(DateTimeOffset now, CancellationToken ct)
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
|
||||||
|
var results = await connection.QueryAsync<Guid>(
|
||||||
|
"""
|
||||||
|
SELECT id
|
||||||
|
FROM sessions
|
||||||
|
WHERE status = @Planned
|
||||||
|
AND scheduled_at - @LeadTime <= @Now
|
||||||
|
AND confirmation_sent_at IS NULL
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
Planned = SessionStatus.Planned,
|
||||||
|
LeadTime = ConfirmationLeadTime,
|
||||||
|
Now = now.UtcDateTime
|
||||||
|
});
|
||||||
|
|
||||||
|
return results.ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<IReadOnlyList<Guid>> GetSessionsNeedingOneHourReminderAsync(DateTimeOffset now, CancellationToken ct)
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
|
||||||
|
var results = await connection.QueryAsync<Guid>(
|
||||||
|
"""
|
||||||
|
SELECT id
|
||||||
|
FROM sessions
|
||||||
|
WHERE status IN (@Confirmed, @ConfirmationSent)
|
||||||
|
AND scheduled_at - @LeadTime <= @Now
|
||||||
|
AND one_hour_reminder_processed_at IS NULL
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
Confirmed = SessionStatus.Confirmed,
|
||||||
|
ConfirmationSent = SessionStatus.ConfirmationSent,
|
||||||
|
LeadTime = OneHourReminderLeadTime,
|
||||||
|
Now = now.UtcDateTime
|
||||||
|
});
|
||||||
|
|
||||||
|
return results.ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<IReadOnlyList<Guid>> GetSessionsNeedingJoinLinkAsync(DateTimeOffset now, CancellationToken ct)
|
||||||
|
{
|
||||||
|
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
||||||
|
|
||||||
|
var results = await connection.QueryAsync<Guid>(
|
||||||
|
"""
|
||||||
|
SELECT id
|
||||||
|
FROM sessions
|
||||||
|
WHERE status = @Confirmed
|
||||||
|
AND scheduled_at - @LeadTime <= @Now
|
||||||
|
AND link_message_id IS NULL
|
||||||
|
""",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
Confirmed = SessionStatus.Confirmed,
|
||||||
|
LeadTime = JoinLinkLeadTime,
|
||||||
|
Now = now.UtcDateTime
|
||||||
|
});
|
||||||
|
|
||||||
|
return results.ToList();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
namespace GmRelay.Bot.Infrastructure.Scheduling;
|
||||||
|
|
||||||
|
public interface ISystemClock
|
||||||
|
{
|
||||||
|
DateTimeOffset UtcNow { get; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class SystemClock : ISystemClock
|
||||||
|
{
|
||||||
|
public DateTimeOffset UtcNow => DateTimeOffset.UtcNow;
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class FakeSystemClock : ISystemClock
|
||||||
|
{
|
||||||
|
public DateTimeOffset UtcNow { get; set; } = DateTimeOffset.UtcNow;
|
||||||
|
}
|
||||||
@@ -1,28 +1,27 @@
|
|||||||
using Dapper;
|
|
||||||
using GmRelay.Shared.Domain;
|
|
||||||
using GmRelay.Bot.Features.Confirmation.SendConfirmation;
|
using GmRelay.Bot.Features.Confirmation.SendConfirmation;
|
||||||
using GmRelay.Bot.Features.Reminders.SendJoinLink;
|
using GmRelay.Bot.Features.Reminders.SendJoinLink;
|
||||||
using Npgsql;
|
using GmRelay.Bot.Features.Reminders.SendOneHourReminder;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Infrastructure.Scheduling;
|
namespace GmRelay.Bot.Infrastructure.Scheduling;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Stateless scheduler: wakes every 60 seconds, queries PostgreSQL for actionable sessions.
|
/// Stateless scheduler: wakes every 60 seconds, queries PostgreSQL for actionable sessions.
|
||||||
/// Two triggers:
|
/// Three triggers:
|
||||||
/// T-24h: send confirmation request with inline keyboard
|
/// T-24h: send confirmation request with inline keyboard
|
||||||
|
/// T-1h: send one-hour direct reminder
|
||||||
/// T-5min: send join link to all confirmed players
|
/// T-5min: send join link to all confirmed players
|
||||||
///
|
///
|
||||||
/// If the Raspberry Pi reboots, nothing is lost — all state is in the DB.
|
/// If the Raspberry Pi reboots, nothing is lost — all state is in the DB.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public sealed class SessionSchedulerService(
|
public sealed class SessionSchedulerService(
|
||||||
NpgsqlDataSource dataSource,
|
ISessionTriggerStore triggerStore,
|
||||||
SendConfirmationHandler confirmationHandler,
|
ISendConfirmationHandler confirmationHandler,
|
||||||
SendJoinLinkHandler joinLinkHandler,
|
ISendOneHourReminderHandler oneHourReminderHandler,
|
||||||
|
ISendJoinLinkHandler joinLinkHandler,
|
||||||
|
ISystemClock clock,
|
||||||
ILogger<SessionSchedulerService> logger) : BackgroundService
|
ILogger<SessionSchedulerService> logger) : BackgroundService
|
||||||
{
|
{
|
||||||
private static readonly TimeSpan TickInterval = TimeSpan.FromMinutes(1);
|
private static readonly TimeSpan TickInterval = TimeSpan.FromMinutes(1);
|
||||||
private static readonly TimeSpan ConfirmationLeadTime = TimeSpan.FromHours(24);
|
|
||||||
private static readonly TimeSpan JoinLinkLeadTime = TimeSpan.FromMinutes(5);
|
|
||||||
|
|
||||||
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
||||||
{
|
{
|
||||||
@@ -30,13 +29,11 @@ public sealed class SessionSchedulerService(
|
|||||||
|
|
||||||
using var timer = new PeriodicTimer(TickInterval);
|
using var timer = new PeriodicTimer(TickInterval);
|
||||||
|
|
||||||
// Run immediately on startup, then on each tick
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await ProcessConfirmationTriggers(stoppingToken);
|
await TickAsync(stoppingToken);
|
||||||
await ProcessJoinLinkTriggers(stoppingToken);
|
|
||||||
}
|
}
|
||||||
catch (OperationCanceledException) when (stoppingToken.IsCancellationRequested)
|
catch (OperationCanceledException) when (stoppingToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
@@ -53,21 +50,30 @@ public sealed class SessionSchedulerService(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// T-24h trigger: find sessions that need confirmation requests sent.
|
/// Runs a single scheduler tick using the current clock time.
|
||||||
/// Condition: status='Planned' AND scheduled_at minus 24h is in the past.
|
/// Public so it can be called from integration tests with a fake clock.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private async Task ProcessConfirmationTriggers(CancellationToken ct)
|
public async Task TickAsync(CancellationToken ct)
|
||||||
{
|
{
|
||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
var now = clock.UtcNow;
|
||||||
|
|
||||||
var sessionIds = await connection.QueryAsync<Guid>(
|
await ProcessConfirmationTriggers(now, ct);
|
||||||
"""
|
await ProcessOneHourReminderTriggers(now, ct);
|
||||||
SELECT id
|
await ProcessJoinLinkTriggers(now, ct);
|
||||||
FROM sessions
|
}
|
||||||
WHERE status = @Planned
|
|
||||||
AND scheduled_at - @LeadTime <= now()
|
private async Task ProcessConfirmationTriggers(DateTimeOffset now, CancellationToken ct)
|
||||||
""",
|
{
|
||||||
new { Planned = SessionStatus.Planned, LeadTime = ConfirmationLeadTime });
|
IReadOnlyList<Guid> sessionIds;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
sessionIds = await triggerStore.GetSessionsNeedingConfirmationAsync(now, ct);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogError(ex, "Failed to query confirmation triggers");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
foreach (var sessionId in sessionIds)
|
foreach (var sessionId in sessionIds)
|
||||||
{
|
{
|
||||||
@@ -83,23 +89,45 @@ public sealed class SessionSchedulerService(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
private async Task ProcessOneHourReminderTriggers(DateTimeOffset now, CancellationToken ct)
|
||||||
/// T-5min trigger: find confirmed sessions that need join links sent.
|
|
||||||
/// Condition: status='Confirmed' AND scheduled_at minus 5min is in the past AND link not yet sent.
|
|
||||||
/// </summary>
|
|
||||||
private async Task ProcessJoinLinkTriggers(CancellationToken ct)
|
|
||||||
{
|
{
|
||||||
await using var connection = await dataSource.OpenConnectionAsync(ct);
|
IReadOnlyList<Guid> sessionIds;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
sessionIds = await triggerStore.GetSessionsNeedingOneHourReminderAsync(now, ct);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogError(ex, "Failed to query one-hour reminder triggers");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var sessionIds = await connection.QueryAsync<Guid>(
|
foreach (var sessionId in sessionIds)
|
||||||
"""
|
{
|
||||||
SELECT id
|
try
|
||||||
FROM sessions
|
{
|
||||||
WHERE status = @Confirmed
|
await oneHourReminderHandler.HandleAsync(sessionId, ct);
|
||||||
AND scheduled_at - @LeadTime <= now()
|
logger.LogInformation("One-hour reminder processed for session {SessionId}", sessionId);
|
||||||
AND link_message_id IS NULL
|
}
|
||||||
""",
|
catch (Exception ex)
|
||||||
new { Confirmed = SessionStatus.Confirmed, LeadTime = JoinLinkLeadTime });
|
{
|
||||||
|
logger.LogError(ex, "Failed to process one-hour reminder for session {SessionId}", sessionId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task ProcessJoinLinkTriggers(DateTimeOffset now, CancellationToken ct)
|
||||||
|
{
|
||||||
|
IReadOnlyList<Guid> sessionIds;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
sessionIds = await triggerStore.GetSessionsNeedingJoinLinkAsync(now, ct);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogError(ex, "Failed to query join-link triggers");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
foreach (var sessionId in sessionIds)
|
foreach (var sessionId in sessionIds)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -0,0 +1,51 @@
|
|||||||
|
using Telegram.Bot;
|
||||||
|
using Telegram.Bot.Types.Enums;
|
||||||
|
using Telegram.Bot.Types.ReplyMarkups;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Handles editing batch messages that may be either text or photo messages.
|
||||||
|
/// When the batch was created with SendPhoto (image + caption), we need
|
||||||
|
/// EditMessageCaption instead of EditMessageText.
|
||||||
|
/// </summary>
|
||||||
|
public static class BatchMessageEditor
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Edits a batch message, automatically detecting whether it is a text or photo message.
|
||||||
|
/// Tries EditMessageText first; on failure falls back to EditMessageCaption.
|
||||||
|
/// </summary>
|
||||||
|
public static async Task EditBatchMessageAsync(
|
||||||
|
ITelegramBotClient bot,
|
||||||
|
long chatId,
|
||||||
|
int messageId,
|
||||||
|
string text,
|
||||||
|
InlineKeyboardMarkup? replyMarkup,
|
||||||
|
CancellationToken ct = default)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await bot.EditMessageText(
|
||||||
|
chatId: chatId,
|
||||||
|
messageId: messageId,
|
||||||
|
text: text,
|
||||||
|
parseMode: ParseMode.Html,
|
||||||
|
replyMarkup: replyMarkup,
|
||||||
|
cancellationToken: ct);
|
||||||
|
}
|
||||||
|
catch (global::Telegram.Bot.Exceptions.ApiRequestException ex)
|
||||||
|
when (ex.Message.Contains("there is no text in the message", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
// The batch message is a photo — use EditMessageCaption instead.
|
||||||
|
// Caption is limited to 1024 chars; if text exceeds that, truncate gracefully.
|
||||||
|
var caption = text.Length <= 1024 ? text : text[..1021] + "...";
|
||||||
|
await bot.EditMessageCaption(
|
||||||
|
chatId: chatId,
|
||||||
|
messageId: messageId,
|
||||||
|
caption: caption,
|
||||||
|
parseMode: ParseMode.Html,
|
||||||
|
replyMarkup: replyMarkup,
|
||||||
|
cancellationToken: ct);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
using Telegram.Bot.Types;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
public interface ITelegramUpdateHandler
|
||||||
|
{
|
||||||
|
Task RouteAsync(Update update, CancellationToken ct);
|
||||||
|
}
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
using Telegram.Bot.Types;
|
||||||
|
using Telegram.Bot.Types.Enums;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
public interface ITelegramUpdateSource
|
||||||
|
{
|
||||||
|
Task<Update[]> GetUpdatesAsync(
|
||||||
|
int offset,
|
||||||
|
int? limit = null,
|
||||||
|
int? timeout = null,
|
||||||
|
IEnumerable<UpdateType>? allowedUpdates = null,
|
||||||
|
CancellationToken cancellationToken = default);
|
||||||
|
}
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
using Telegram.Bot;
|
|
||||||
using Telegram.Bot.Types;
|
using Telegram.Bot.Types;
|
||||||
using Telegram.Bot.Types.Enums;
|
using Telegram.Bot.Types.Enums;
|
||||||
|
|
||||||
@@ -9,35 +8,21 @@ namespace GmRelay.Bot.Infrastructure.Telegram;
|
|||||||
/// Stateless — all state is in PostgreSQL. Safe to restart at any time.
|
/// Stateless — all state is in PostgreSQL. Safe to restart at any time.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public sealed class TelegramBotService(
|
public sealed class TelegramBotService(
|
||||||
ITelegramBotClient bot,
|
ITelegramUpdateSource updateSource,
|
||||||
UpdateRouter router,
|
ITelegramUpdateHandler updateHandler,
|
||||||
ILogger<TelegramBotService> logger) : BackgroundService
|
ILogger<TelegramBotService> logger) : BackgroundService
|
||||||
{
|
{
|
||||||
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
||||||
{
|
{
|
||||||
logger.LogInformation("Telegram bot polling started");
|
logger.LogInformation("Telegram bot polling started");
|
||||||
|
|
||||||
// Skip any pending updates from before this startup
|
var offset = await GetStartupOffsetAsync(stoppingToken);
|
||||||
try
|
|
||||||
{
|
|
||||||
var pending = await bot.GetUpdates(offset: -1, limit: 1, cancellationToken: stoppingToken);
|
|
||||||
if (pending.Length > 0)
|
|
||||||
{
|
|
||||||
logger.LogInformation("Skipped {Count} pending update(s)", pending[^1].Id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
logger.LogWarning(ex, "Failed to clear pending updates, continuing anyway");
|
|
||||||
}
|
|
||||||
|
|
||||||
var offset = 0;
|
|
||||||
|
|
||||||
while (!stoppingToken.IsCancellationRequested)
|
while (!stoppingToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var updates = await bot.GetUpdates(
|
var updates = await updateSource.GetUpdatesAsync(
|
||||||
offset: offset,
|
offset: offset,
|
||||||
timeout: 30,
|
timeout: 30,
|
||||||
allowedUpdates: [UpdateType.Message, UpdateType.CallbackQuery],
|
allowedUpdates: [UpdateType.Message, UpdateType.CallbackQuery],
|
||||||
@@ -47,7 +32,7 @@ public sealed class TelegramBotService(
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await router.RouteAsync(update, stoppingToken);
|
await updateHandler.RouteAsync(update, stoppingToken);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -70,4 +55,33 @@ public sealed class TelegramBotService(
|
|||||||
|
|
||||||
logger.LogInformation("Telegram bot polling stopped");
|
logger.LogInformation("Telegram bot polling stopped");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task<int> GetStartupOffsetAsync(CancellationToken stoppingToken)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var pending = await updateSource.GetUpdatesAsync(
|
||||||
|
offset: -1,
|
||||||
|
limit: 1,
|
||||||
|
cancellationToken: stoppingToken);
|
||||||
|
|
||||||
|
if (pending.Length == 0)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
var startupOffset = pending[^1].Id + 1;
|
||||||
|
logger.LogInformation(
|
||||||
|
"Skipping pending updates through {LastPendingUpdateId}; starting polling from offset {StartupOffset}",
|
||||||
|
pending[^1].Id,
|
||||||
|
startupOffset);
|
||||||
|
|
||||||
|
return startupOffset;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogWarning(ex, "Failed to determine startup offset, continuing from offset 0");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,46 @@
|
|||||||
|
using Telegram.Bot;
|
||||||
|
using Telegram.Bot.Types;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
public sealed class TelegramMiniAppMenuButtonService(
|
||||||
|
ITelegramBotClient bot,
|
||||||
|
IConfiguration configuration,
|
||||||
|
ILogger<TelegramMiniAppMenuButtonService> logger) : IHostedService
|
||||||
|
{
|
||||||
|
public async Task StartAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var miniAppUrl = configuration["Telegram:MiniAppUrl"];
|
||||||
|
if (string.IsNullOrWhiteSpace(miniAppUrl))
|
||||||
|
{
|
||||||
|
logger.LogInformation("Telegram Mini App URL is not configured; menu button setup skipped.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Uri.TryCreate(miniAppUrl, UriKind.Absolute, out var uri) ||
|
||||||
|
(uri.Scheme != Uri.UriSchemeHttps && !uri.IsLoopback))
|
||||||
|
{
|
||||||
|
logger.LogWarning("Telegram Mini App URL {MiniAppUrl} is not a valid HTTPS URL.", miniAppUrl);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await bot.SetChatMenuButton(
|
||||||
|
menuButton: new MenuButtonWebApp
|
||||||
|
{
|
||||||
|
Text = "Dashboard",
|
||||||
|
WebApp = new WebAppInfo(miniAppUrl)
|
||||||
|
},
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
|
||||||
|
logger.LogInformation("Telegram Mini App menu button configured for {MiniAppUrl}.", miniAppUrl);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
logger.LogWarning(ex, "Failed to configure Telegram Mini App menu button.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask;
|
||||||
|
}
|
||||||
@@ -0,0 +1,63 @@
|
|||||||
|
// NOTE: duplicated in GmRelay.Web/Services/TelegramSessionBatchRenderer.cs
|
||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
using GmRelay.Shared.Rendering;
|
||||||
|
using Telegram.Bot.Types.ReplyMarkups;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
public static class TelegramSessionBatchRenderer
|
||||||
|
{
|
||||||
|
public static (string Text, InlineKeyboardMarkup Markup) Render(SessionBatchViewModel view)
|
||||||
|
{
|
||||||
|
var messageText = $"🎲 <b>Новые игры:</b> {System.Net.WebUtility.HtmlEncode(view.Title)}\n\n" +
|
||||||
|
$"<b>Расписание:</b>\n\n";
|
||||||
|
|
||||||
|
var buttons = new List<InlineKeyboardButton[]>();
|
||||||
|
|
||||||
|
foreach (var session in view.Sessions)
|
||||||
|
{
|
||||||
|
messageText += $"📅 <b>{session.ScheduledAt.FormatMoscow()}</b>\n";
|
||||||
|
messageText += session.MaxPlayers.HasValue
|
||||||
|
? $"👥 Места: {session.ActivePlayerCount}/{session.MaxPlayers.Value}\n"
|
||||||
|
: $"👥 Игроки ({session.ActivePlayerCount}):\n";
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(session.JoinLink))
|
||||||
|
{
|
||||||
|
messageText += $"🔗 <a href=\"{System.Net.WebUtility.HtmlEncode(session.JoinLink)}\">Ссылка на игру</a>\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (session.ActivePlayers.Count > 0)
|
||||||
|
{
|
||||||
|
messageText += string.Join("\n", session.ActivePlayers.Select(p =>
|
||||||
|
$" 👤 {(p.TelegramUsername != null ? "@" + p.TelegramUsername : p.DisplayName)}")) + "\n";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
messageText += " <i>Пока никто не записался</i>\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (session.WaitlistedPlayers.Count > 0)
|
||||||
|
{
|
||||||
|
messageText += $"⏳ Лист ожидания ({session.WaitlistedPlayers.Count}):\n";
|
||||||
|
messageText += string.Join("\n", session.WaitlistedPlayers.Select(p =>
|
||||||
|
$" ⏱ {(p.TelegramUsername != null ? "@" + p.TelegramUsername : p.DisplayName)}")) + "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (GmRelay.Shared.Domain.SessionStatus.IsCancelled(session.Status))
|
||||||
|
{
|
||||||
|
messageText += "❌ <i>Сессия отменена</i>\n\n";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
messageText += "\n";
|
||||||
|
var actionRow = session.AvailableActions
|
||||||
|
.Select(a => InlineKeyboardButton.WithCallbackData(a.Label, $"{a.ActionKey}:{a.SessionId}"))
|
||||||
|
.ToArray();
|
||||||
|
if (actionRow.Length > 0)
|
||||||
|
buttons.Add(actionRow);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (messageText, new InlineKeyboardMarkup(buttons));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
namespace GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
public sealed record TelegramTopicDestination(
|
||||||
|
int? MessageThreadId,
|
||||||
|
bool ShouldCreateForumTopic,
|
||||||
|
bool TopicCreatedByBot);
|
||||||
|
|
||||||
|
public static class TelegramTopicRouting
|
||||||
|
{
|
||||||
|
public const string MissingForumTopicRightsMessage =
|
||||||
|
"Не удалось создать Telegram topic. Сделайте бота admin и включите право Manage Topics, затем повторите команду.";
|
||||||
|
|
||||||
|
public static TelegramTopicDestination ResolveNewScheduleDestination(
|
||||||
|
bool chatIsForum,
|
||||||
|
int? incomingMessageThreadId)
|
||||||
|
{
|
||||||
|
if (!chatIsForum)
|
||||||
|
{
|
||||||
|
return new TelegramTopicDestination(null, ShouldCreateForumTopic: false, TopicCreatedByBot: false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (incomingMessageThreadId.HasValue)
|
||||||
|
{
|
||||||
|
return new TelegramTopicDestination(
|
||||||
|
incomingMessageThreadId,
|
||||||
|
ShouldCreateForumTopic: false,
|
||||||
|
TopicCreatedByBot: false);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new TelegramTopicDestination(null, ShouldCreateForumTopic: true, TopicCreatedByBot: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool ShouldDeleteForumTopic(bool topicCreatedByBot, int remainingSessionsInTopic) =>
|
||||||
|
topicCreatedByBot && remainingSessionsInTopic == 0;
|
||||||
|
|
||||||
|
public static bool IsMissingForumTopicRightsError(string apiError) =>
|
||||||
|
apiError.Contains("not enough rights", StringComparison.OrdinalIgnoreCase) ||
|
||||||
|
apiError.Contains("CHAT_ADMIN_REQUIRED", StringComparison.OrdinalIgnoreCase) ||
|
||||||
|
apiError.Contains("not an administrator", StringComparison.OrdinalIgnoreCase);
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
using Telegram.Bot;
|
||||||
|
using Telegram.Bot.Types;
|
||||||
|
using Telegram.Bot.Types.Enums;
|
||||||
|
|
||||||
|
namespace GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
|
public sealed class TelegramUpdateSource(ITelegramBotClient bot) : ITelegramUpdateSource
|
||||||
|
{
|
||||||
|
public Task<Update[]> GetUpdatesAsync(
|
||||||
|
int offset,
|
||||||
|
int? limit = null,
|
||||||
|
int? timeout = null,
|
||||||
|
IEnumerable<UpdateType>? allowedUpdates = null,
|
||||||
|
CancellationToken cancellationToken = default) =>
|
||||||
|
bot.GetUpdates(
|
||||||
|
offset: offset,
|
||||||
|
limit: limit,
|
||||||
|
timeout: timeout,
|
||||||
|
allowedUpdates: allowedUpdates,
|
||||||
|
cancellationToken: cancellationToken);
|
||||||
|
}
|
||||||
@@ -9,6 +9,7 @@ using GmRelay.Bot.Features.Sessions.RescheduleSession;
|
|||||||
using Telegram.Bot;
|
using Telegram.Bot;
|
||||||
using Telegram.Bot.Types;
|
using Telegram.Bot.Types;
|
||||||
using Telegram.Bot.Types.Enums;
|
using Telegram.Bot.Types.Enums;
|
||||||
|
using Telegram.Bot.Types.ReplyMarkups;
|
||||||
|
|
||||||
namespace GmRelay.Bot.Infrastructure.Telegram;
|
namespace GmRelay.Bot.Infrastructure.Telegram;
|
||||||
|
|
||||||
@@ -20,6 +21,8 @@ public sealed class UpdateRouter(
|
|||||||
HandleRsvpHandler rsvpHandler,
|
HandleRsvpHandler rsvpHandler,
|
||||||
CreateSessionHandler createSessionHandler,
|
CreateSessionHandler createSessionHandler,
|
||||||
JoinSessionHandler joinSessionHandler,
|
JoinSessionHandler joinSessionHandler,
|
||||||
|
LeaveSessionHandler leaveSessionHandler,
|
||||||
|
PromoteWaitlistedPlayerHandler promoteWaitlistedPlayerHandler,
|
||||||
CancelSessionHandler cancelSessionHandler,
|
CancelSessionHandler cancelSessionHandler,
|
||||||
DeleteSessionHandler deleteSessionHandler,
|
DeleteSessionHandler deleteSessionHandler,
|
||||||
ListSessionsHandler listSessionsHandler,
|
ListSessionsHandler listSessionsHandler,
|
||||||
@@ -28,7 +31,8 @@ public sealed class UpdateRouter(
|
|||||||
HandleRescheduleTimeInputHandler rescheduleTimeInputHandler,
|
HandleRescheduleTimeInputHandler rescheduleTimeInputHandler,
|
||||||
HandleRescheduleVoteHandler rescheduleVoteHandler,
|
HandleRescheduleVoteHandler rescheduleVoteHandler,
|
||||||
ITelegramBotClient bot,
|
ITelegramBotClient bot,
|
||||||
ILogger<UpdateRouter> logger)
|
IConfiguration configuration,
|
||||||
|
ILogger<UpdateRouter> logger) : ITelegramUpdateHandler
|
||||||
{
|
{
|
||||||
public async Task RouteAsync(Update update, CancellationToken ct)
|
public async Task RouteAsync(Update update, CancellationToken ct)
|
||||||
{
|
{
|
||||||
@@ -38,17 +42,26 @@ public sealed class UpdateRouter(
|
|||||||
await HandleCallbackQueryAsync(query, ct);
|
await HandleCallbackQueryAsync(query, ct);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case { Message: { Text: { } text } message } when text.StartsWith('/'):
|
case { Message: { } message }:
|
||||||
await HandleCommandAsync(message, text, ct);
|
var commandText = GetCommandText(message);
|
||||||
break;
|
if (commandText.StartsWith("/", StringComparison.Ordinal))
|
||||||
|
{
|
||||||
|
await HandleCommandAsync(message, commandText, ct);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (message.Text is not null)
|
||||||
|
{
|
||||||
|
await rescheduleTimeInputHandler.TryHandleAsync(message, ct);
|
||||||
|
}
|
||||||
|
|
||||||
// Non-command text messages — check for reschedule time input
|
|
||||||
case { Message: { Text: { } } message } when !message.Text!.StartsWith('/'):
|
|
||||||
await rescheduleTimeInputHandler.TryHandleAsync(message, ct);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal static string GetCommandText(Message message)
|
||||||
|
=> (message.Text ?? message.Caption ?? string.Empty).TrimStart();
|
||||||
|
|
||||||
private async Task HandleCallbackQueryAsync(CallbackQuery query, CancellationToken ct)
|
private async Task HandleCallbackQueryAsync(CallbackQuery query, CancellationToken ct)
|
||||||
{
|
{
|
||||||
if (query.Data is not { } data || query.Message is not { } message)
|
if (query.Data is not { } data || query.Message is not { } message)
|
||||||
@@ -67,11 +80,24 @@ public sealed class UpdateRouter(
|
|||||||
CallbackQueryId: query.Id,
|
CallbackQueryId: query.Id,
|
||||||
ChatId: message.Chat.Id,
|
ChatId: message.Chat.Id,
|
||||||
MessageId: message.MessageId);
|
MessageId: message.MessageId);
|
||||||
|
|
||||||
await joinSessionHandler.HandleAsync(command, ct);
|
await joinSessionHandler.HandleAsync(command, ct);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (action == "leave_session" && parts.Length >= 2 && Guid.TryParse(parts[1], out var leaveSessionId))
|
||||||
|
{
|
||||||
|
var command = new LeaveSessionCommand(
|
||||||
|
SessionId: leaveSessionId,
|
||||||
|
TelegramUserId: query.From.Id,
|
||||||
|
CallbackQueryId: query.Id,
|
||||||
|
ChatId: message.Chat.Id,
|
||||||
|
MessageId: message.MessageId);
|
||||||
|
|
||||||
|
await leaveSessionHandler.HandleAsync(command, ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (action == "cancel_session" && parts.Length >= 2 && Guid.TryParse(parts[1], out var cancelSessionId))
|
if (action == "cancel_session" && parts.Length >= 2 && Guid.TryParse(parts[1], out var cancelSessionId))
|
||||||
{
|
{
|
||||||
var command = new CancelSessionCommand(
|
var command = new CancelSessionCommand(
|
||||||
@@ -79,12 +105,26 @@ public sealed class UpdateRouter(
|
|||||||
TelegramUserId: query.From.Id,
|
TelegramUserId: query.From.Id,
|
||||||
CallbackQueryId: query.Id,
|
CallbackQueryId: query.Id,
|
||||||
ChatId: message.Chat.Id,
|
ChatId: message.Chat.Id,
|
||||||
|
MessageThreadId: message.MessageThreadId,
|
||||||
MessageId: message.MessageId);
|
MessageId: message.MessageId);
|
||||||
|
|
||||||
await cancelSessionHandler.HandleAsync(command, ct);
|
await cancelSessionHandler.HandleAsync(command, ct);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (action == "promote_waitlist" && parts.Length >= 2 && Guid.TryParse(parts[1], out var promoteSessionId))
|
||||||
|
{
|
||||||
|
var command = new PromoteWaitlistedPlayerCommand(
|
||||||
|
SessionId: promoteSessionId,
|
||||||
|
TelegramUserId: query.From.Id,
|
||||||
|
CallbackQueryId: query.Id,
|
||||||
|
ChatId: message.Chat.Id,
|
||||||
|
MessageId: message.MessageId);
|
||||||
|
|
||||||
|
await promoteWaitlistedPlayerHandler.HandleAsync(command, ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (action == "delete_session" && parts.Length >= 2 && Guid.TryParse(parts[1], out var deleteSessionId))
|
if (action == "delete_session" && parts.Length >= 2 && Guid.TryParse(parts[1], out var deleteSessionId))
|
||||||
{
|
{
|
||||||
var command = new DeleteSessionCommand(
|
var command = new DeleteSessionCommand(
|
||||||
@@ -105,21 +145,17 @@ public sealed class UpdateRouter(
|
|||||||
TelegramUserId: query.From.Id,
|
TelegramUserId: query.From.Id,
|
||||||
CallbackQueryId: query.Id,
|
CallbackQueryId: query.Id,
|
||||||
ChatId: message.Chat.Id,
|
ChatId: message.Chat.Id,
|
||||||
|
MessageThreadId: message.MessageThreadId,
|
||||||
MessageId: message.MessageId);
|
MessageId: message.MessageId);
|
||||||
|
|
||||||
await initiateRescheduleHandler.HandleAsync(command, ct);
|
await initiateRescheduleHandler.HandleAsync(command, ct);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (action == "reschedule_vote" && parts.Length >= 3 && Guid.TryParse(parts[2], out var proposalId))
|
if (action == "reschedule_vote" && parts.Length >= 2 && Guid.TryParse(parts[1], out var optionId))
|
||||||
{
|
{
|
||||||
var vote = parts[1]; // "yes" or "no"
|
|
||||||
if (vote is not ("yes" or "no"))
|
|
||||||
return;
|
|
||||||
|
|
||||||
var command = new HandleRescheduleVoteCommand(
|
var command = new HandleRescheduleVoteCommand(
|
||||||
ProposalId: proposalId,
|
OptionId: optionId,
|
||||||
Vote: vote,
|
|
||||||
TelegramUserId: query.From.Id,
|
TelegramUserId: query.From.Id,
|
||||||
CallbackQueryId: query.Id,
|
CallbackQueryId: query.Id,
|
||||||
ChatId: message.Chat.Id,
|
ChatId: message.Chat.Id,
|
||||||
@@ -165,10 +201,7 @@ public sealed class UpdateRouter(
|
|||||||
switch (command)
|
switch (command)
|
||||||
{
|
{
|
||||||
case "/start":
|
case "/start":
|
||||||
await bot.SendMessage(
|
await SendStartMessageAsync(message, ct);
|
||||||
chatId: message.Chat.Id,
|
|
||||||
text: "GM-Relay Bot ready. Use /help for commands.",
|
|
||||||
cancellationToken: ct);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case "/newsession":
|
case "/newsession":
|
||||||
@@ -192,9 +225,17 @@ public sealed class UpdateRouter(
|
|||||||
/newsession
|
/newsession
|
||||||
Название: My Game
|
Название: My Game
|
||||||
Время: 15.05.2026 19:30
|
Время: 15.05.2026 19:30
|
||||||
|
Мест: 4
|
||||||
Ссылка: https://link
|
Ссылка: https://link
|
||||||
|
Картинка: https://cover
|
||||||
|
|
||||||
|
Для регулярного расписания можно указать одну дату:
|
||||||
|
Игр: 4
|
||||||
|
Интервал: 7
|
||||||
|
|
||||||
/listsessions — список предстоящих сессий
|
/listsessions — список предстоящих сессий
|
||||||
|
Для owner/co-GM /listsessions показывает кнопки отмены, переноса, удаления и повышения из листа ожидания.
|
||||||
|
Игроки могут записаться кнопкой «На дату» и сняться кнопкой «Выйти».
|
||||||
/help — эта справка
|
/help — эта справка
|
||||||
""",
|
""",
|
||||||
cancellationToken: ct);
|
cancellationToken: ct);
|
||||||
@@ -206,4 +247,24 @@ public sealed class UpdateRouter(
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task SendStartMessageAsync(Message message, CancellationToken ct)
|
||||||
|
{
|
||||||
|
var miniAppUrl = configuration["Telegram:MiniAppUrl"];
|
||||||
|
if (string.IsNullOrWhiteSpace(miniAppUrl))
|
||||||
|
{
|
||||||
|
await bot.SendMessage(
|
||||||
|
chatId: message.Chat.Id,
|
||||||
|
text: "GM-Relay Bot ready. Use /help for commands.",
|
||||||
|
cancellationToken: ct);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await bot.SendMessage(
|
||||||
|
chatId: message.Chat.Id,
|
||||||
|
text: "GM-Relay Bot ready. Откройте dashboard внутри Telegram или используйте /help для команд.",
|
||||||
|
replyMarkup: new InlineKeyboardMarkup(
|
||||||
|
InlineKeyboardButton.WithWebApp("Открыть dashboard", new WebAppInfo(miniAppUrl))),
|
||||||
|
cancellationToken: ct);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,16 @@
|
|||||||
|
-- Add per-session seat limits and participant waitlist support.
|
||||||
|
ALTER TABLE sessions
|
||||||
|
ADD COLUMN max_players INTEGER,
|
||||||
|
ADD CONSTRAINT ck_sessions_max_players CHECK (max_players IS NULL OR max_players > 0);
|
||||||
|
|
||||||
|
ALTER TABLE session_participants
|
||||||
|
ADD COLUMN registration_status VARCHAR(50) NOT NULL DEFAULT 'Active'
|
||||||
|
CHECK (registration_status IN ('Active', 'Waitlisted')),
|
||||||
|
ADD COLUMN created_at TIMESTAMPTZ NOT NULL DEFAULT now();
|
||||||
|
|
||||||
|
CREATE INDEX ix_session_participants_session_registration_status
|
||||||
|
ON session_participants (session_id, registration_status);
|
||||||
|
|
||||||
|
CREATE INDEX ix_session_participants_waitlist_order
|
||||||
|
ON session_participants (session_id, created_at, id)
|
||||||
|
WHERE registration_status = 'Waitlisted';
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
ALTER TABLE sessions
|
||||||
|
ADD COLUMN notification_mode VARCHAR(50) NOT NULL DEFAULT 'GroupAndDirect'
|
||||||
|
CHECK (notification_mode IN ('GroupAndDirect', 'GroupOnly')),
|
||||||
|
ADD COLUMN one_hour_reminder_processed_at TIMESTAMPTZ;
|
||||||
|
|
||||||
|
CREATE INDEX ix_sessions_one_hour_reminders ON sessions (scheduled_at)
|
||||||
|
WHERE status IN ('Confirmed', 'ConfirmationSent')
|
||||||
|
AND one_hour_reminder_processed_at IS NULL;
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
-- Add explicit owner/co-GM management roles for each Telegram group.
|
||||||
|
|
||||||
|
INSERT INTO players (telegram_id, display_name)
|
||||||
|
SELECT DISTINCT gg.gm_telegram_id,
|
||||||
|
'GM ' || gg.gm_telegram_id::text
|
||||||
|
FROM game_groups gg
|
||||||
|
ON CONFLICT (telegram_id) DO NOTHING;
|
||||||
|
|
||||||
|
CREATE TABLE group_managers (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
group_id UUID NOT NULL REFERENCES game_groups(id) ON DELETE CASCADE,
|
||||||
|
player_id UUID NOT NULL REFERENCES players(id) ON DELETE CASCADE,
|
||||||
|
role VARCHAR(50) NOT NULL CHECK (role IN ('Owner', 'CoGm')),
|
||||||
|
added_by_player_id UUID REFERENCES players(id) ON DELETE SET NULL,
|
||||||
|
created_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
||||||
|
UNIQUE (group_id, player_id)
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO group_managers (group_id, player_id, role)
|
||||||
|
SELECT gg.id, p.id, 'Owner'
|
||||||
|
FROM game_groups gg
|
||||||
|
JOIN players p ON p.telegram_id = gg.gm_telegram_id
|
||||||
|
ON CONFLICT (group_id, player_id) DO NOTHING;
|
||||||
|
|
||||||
|
CREATE INDEX ix_group_managers_group_role ON group_managers (group_id, role);
|
||||||
|
CREATE INDEX ix_group_managers_player ON group_managers (player_id);
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
-- Multi-option reschedule voting with a deadline.
|
||||||
|
|
||||||
|
ALTER TABLE reschedule_proposals
|
||||||
|
ADD COLUMN voting_deadline_at TIMESTAMPTZ,
|
||||||
|
ADD COLUMN selected_option_id UUID;
|
||||||
|
|
||||||
|
CREATE TABLE reschedule_options (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
proposal_id UUID NOT NULL REFERENCES reschedule_proposals(id) ON DELETE CASCADE,
|
||||||
|
proposed_at TIMESTAMPTZ NOT NULL,
|
||||||
|
display_order INTEGER NOT NULL CHECK (display_order BETWEEN 1 AND 3),
|
||||||
|
created_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
||||||
|
UNIQUE (proposal_id, id),
|
||||||
|
UNIQUE (proposal_id, display_order),
|
||||||
|
UNIQUE (proposal_id, proposed_at)
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE reschedule_option_votes (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
proposal_id UUID NOT NULL REFERENCES reschedule_proposals(id) ON DELETE CASCADE,
|
||||||
|
player_id UUID NOT NULL REFERENCES players(id) ON DELETE CASCADE,
|
||||||
|
option_id UUID NOT NULL,
|
||||||
|
voted_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
||||||
|
UNIQUE (proposal_id, player_id),
|
||||||
|
FOREIGN KEY (proposal_id, option_id)
|
||||||
|
REFERENCES reschedule_options(proposal_id, id)
|
||||||
|
ON DELETE CASCADE
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX ix_reschedule_voting_deadline
|
||||||
|
ON reschedule_proposals (voting_deadline_at)
|
||||||
|
WHERE status = 'Voting';
|
||||||
|
|
||||||
|
CREATE INDEX ix_reschedule_option_votes_option
|
||||||
|
ON reschedule_option_votes (option_id);
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
CREATE TABLE campaign_templates (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
group_id UUID NOT NULL REFERENCES game_groups(id) ON DELETE CASCADE,
|
||||||
|
name VARCHAR(200) NOT NULL,
|
||||||
|
title VARCHAR(500) NOT NULL,
|
||||||
|
join_link TEXT NOT NULL,
|
||||||
|
session_count INTEGER NOT NULL CHECK (session_count BETWEEN 1 AND 52),
|
||||||
|
interval_days INTEGER NOT NULL CHECK (interval_days BETWEEN 1 AND 365),
|
||||||
|
max_players INTEGER CHECK (max_players IS NULL OR max_players > 0),
|
||||||
|
notification_mode VARCHAR(32) NOT NULL DEFAULT 'GroupAndDirect'
|
||||||
|
CHECK (notification_mode IN ('GroupAndDirect', 'GroupOnly')),
|
||||||
|
created_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
||||||
|
updated_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
||||||
|
UNIQUE (group_id, name)
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX ix_campaign_templates_group ON campaign_templates (group_id, created_at DESC);
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
CREATE TABLE calendar_subscriptions (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
token TEXT UNIQUE NOT NULL,
|
||||||
|
user_telegram_id BIGINT NOT NULL,
|
||||||
|
group_id UUID REFERENCES game_groups(id) ON DELETE CASCADE,
|
||||||
|
filter_type SMALLINT NOT NULL DEFAULT 0,
|
||||||
|
created_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
||||||
|
expires_at TIMESTAMPTZ
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX ix_calendar_subscriptions_user_telegram_id ON calendar_subscriptions (user_telegram_id);
|
||||||
@@ -0,0 +1,66 @@
|
|||||||
|
-- =============================================================
|
||||||
|
-- Attendance statistics view for GM analytics
|
||||||
|
-- Returns per-player aggregated metrics for a given game group.
|
||||||
|
-- NOTE: waitlist count reflects CURRENT registration_status only.
|
||||||
|
-- Full historical waitlist tracking will come with #15.
|
||||||
|
-- =============================================================
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION get_group_attendance_stats(p_group_id UUID)
|
||||||
|
RETURNS TABLE (
|
||||||
|
player_id UUID,
|
||||||
|
display_name VARCHAR,
|
||||||
|
telegram_username VARCHAR,
|
||||||
|
total_sessions BIGINT,
|
||||||
|
confirmed_count BIGINT,
|
||||||
|
declined_count BIGINT,
|
||||||
|
no_response_count BIGINT,
|
||||||
|
waitlisted_count BIGINT,
|
||||||
|
cancellation_affected_count BIGINT,
|
||||||
|
attendance_rate NUMERIC
|
||||||
|
) AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN QUERY
|
||||||
|
WITH player_sessions AS (
|
||||||
|
SELECT
|
||||||
|
sp.player_id,
|
||||||
|
s.id AS session_id,
|
||||||
|
sp.rsvp_status,
|
||||||
|
sp.registration_status,
|
||||||
|
s.status AS session_status,
|
||||||
|
s.scheduled_at
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN sessions s ON s.id = sp.session_id
|
||||||
|
WHERE s.group_id = p_group_id
|
||||||
|
),
|
||||||
|
player_totals AS (
|
||||||
|
SELECT
|
||||||
|
ps.player_id,
|
||||||
|
COUNT(*) FILTER (WHERE ps.session_status <> 'Cancelled') AS total_sessions,
|
||||||
|
COUNT(*) FILTER (WHERE ps.rsvp_status = 'Confirmed' AND ps.session_status <> 'Cancelled') AS confirmed_count,
|
||||||
|
COUNT(*) FILTER (WHERE ps.rsvp_status = 'Declined' AND ps.session_status <> 'Cancelled') AS declined_count,
|
||||||
|
COUNT(*) FILTER (WHERE ps.rsvp_status = 'Pending' AND ps.scheduled_at < NOW() AND ps.session_status <> 'Cancelled') AS no_response_count,
|
||||||
|
COUNT(*) FILTER (WHERE ps.registration_status = 'Waitlisted' AND ps.session_status <> 'Cancelled') AS waitlisted_count,
|
||||||
|
COUNT(*) FILTER (WHERE ps.session_status = 'Cancelled') AS cancellation_affected_count
|
||||||
|
FROM player_sessions ps
|
||||||
|
GROUP BY ps.player_id
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
pt.player_id,
|
||||||
|
p.display_name,
|
||||||
|
COALESCE(p.external_username, p.telegram_username) AS telegram_username,
|
||||||
|
pt.total_sessions,
|
||||||
|
pt.confirmed_count,
|
||||||
|
pt.declined_count,
|
||||||
|
pt.no_response_count,
|
||||||
|
pt.waitlisted_count,
|
||||||
|
pt.cancellation_affected_count,
|
||||||
|
ROUND(
|
||||||
|
100.0 * pt.confirmed_count
|
||||||
|
/ NULLIF(pt.total_sessions, 0),
|
||||||
|
1
|
||||||
|
) AS attendance_rate
|
||||||
|
FROM player_totals pt
|
||||||
|
JOIN players p ON p.id = pt.player_id
|
||||||
|
ORDER BY pt.confirmed_count DESC, pt.total_sessions DESC;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql STABLE;
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
CREATE EXTENSION IF NOT EXISTS "pgcrypto";
|
||||||
|
|
||||||
|
CREATE TABLE session_audit_log (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
session_id UUID NOT NULL REFERENCES sessions(id) ON DELETE CASCADE,
|
||||||
|
actor_telegram_id BIGINT NOT NULL,
|
||||||
|
actor_name VARCHAR(255) NOT NULL,
|
||||||
|
change_type VARCHAR(50) NOT NULL
|
||||||
|
CHECK (change_type IN ('Title','Time','Link','MaxPlayers','Status','WaitlistPromote','PlayerRemoved','BatchRescheduled','Cancelled')),
|
||||||
|
old_value TEXT,
|
||||||
|
new_value TEXT,
|
||||||
|
changed_at TIMESTAMPTZ NOT NULL DEFAULT now()
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX ix_session_audit_log_session_id ON session_audit_log(session_id);
|
||||||
|
CREATE INDEX ix_session_audit_log_changed_at ON session_audit_log(changed_at);
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
ALTER TABLE sessions
|
||||||
|
ADD COLUMN confirmation_sent_at TIMESTAMPTZ;
|
||||||
|
|
||||||
|
-- Update existing ConfirmationSent sessions to have a sentinel value
|
||||||
|
-- so they don't get re-processed after migration
|
||||||
|
UPDATE sessions
|
||||||
|
SET confirmation_sent_at = now()
|
||||||
|
WHERE status = 'ConfirmationSent';
|
||||||
|
|
||||||
|
-- Partial index for efficient T-24h query
|
||||||
|
CREATE INDEX ix_sessions_confirmation_reminders ON sessions (scheduled_at)
|
||||||
|
WHERE status = 'Planned'
|
||||||
|
AND confirmation_sent_at IS NULL;
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
ALTER TABLE sessions
|
||||||
|
ADD COLUMN topic_created_by_bot BOOLEAN NOT NULL DEFAULT FALSE;
|
||||||
|
|
||||||
|
UPDATE sessions
|
||||||
|
SET topic_created_by_bot = TRUE
|
||||||
|
WHERE thread_id IS NOT NULL;
|
||||||
@@ -0,0 +1,119 @@
|
|||||||
|
-- =============================================================
|
||||||
|
-- V016: Add platform identity columns and platform_messages table
|
||||||
|
-- =============================================================
|
||||||
|
-- Scope: Prepare schema for multi-platform support (Discord, etc).
|
||||||
|
-- Legacy telegram_* columns are retained for backward compatibility.
|
||||||
|
-- =============================================================
|
||||||
|
|
||||||
|
-- -- Players: platform-agnostic identity
|
||||||
|
ALTER TABLE players
|
||||||
|
ADD COLUMN platform VARCHAR(50),
|
||||||
|
ADD COLUMN external_user_id VARCHAR(255),
|
||||||
|
ADD COLUMN external_username VARCHAR(255);
|
||||||
|
|
||||||
|
CREATE UNIQUE INDEX ix_players_platform_external_user_id
|
||||||
|
ON players (platform, external_user_id)
|
||||||
|
WHERE platform IS NOT NULL AND external_user_id IS NOT NULL;
|
||||||
|
|
||||||
|
-- -- Game groups: platform-agnostic identity
|
||||||
|
ALTER TABLE game_groups
|
||||||
|
ADD COLUMN platform VARCHAR(50),
|
||||||
|
ADD COLUMN external_group_id VARCHAR(255),
|
||||||
|
ADD COLUMN external_channel_id VARCHAR(255);
|
||||||
|
|
||||||
|
CREATE UNIQUE INDEX ix_game_groups_platform_external_group_id
|
||||||
|
ON game_groups (platform, external_group_id)
|
||||||
|
WHERE platform IS NOT NULL AND external_group_id IS NOT NULL;
|
||||||
|
|
||||||
|
-- -- Backfill existing Telegram data
|
||||||
|
UPDATE players
|
||||||
|
SET platform = 'Telegram',
|
||||||
|
external_user_id = telegram_id::TEXT,
|
||||||
|
external_username = telegram_username
|
||||||
|
WHERE platform IS NULL;
|
||||||
|
|
||||||
|
UPDATE game_groups
|
||||||
|
SET platform = 'Telegram',
|
||||||
|
external_group_id = telegram_chat_id::TEXT
|
||||||
|
WHERE platform IS NULL;
|
||||||
|
|
||||||
|
-- -- Platform messages: store per-platform message references
|
||||||
|
CREATE TABLE platform_messages (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
platform VARCHAR(50) NOT NULL,
|
||||||
|
group_id UUID REFERENCES game_groups(id) ON DELETE CASCADE,
|
||||||
|
batch_id UUID,
|
||||||
|
session_id UUID REFERENCES sessions(id) ON DELETE CASCADE,
|
||||||
|
external_channel_id VARCHAR(255),
|
||||||
|
external_thread_id VARCHAR(255),
|
||||||
|
external_message_id VARCHAR(255) NOT NULL,
|
||||||
|
purpose VARCHAR(50) NOT NULL,
|
||||||
|
created_at TIMESTAMPTZ NOT NULL DEFAULT now()
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX ix_platform_messages_group_id ON platform_messages(group_id);
|
||||||
|
CREATE INDEX ix_platform_messages_batch_id ON platform_messages(batch_id);
|
||||||
|
CREATE INDEX ix_platform_messages_session_id ON platform_messages(session_id);
|
||||||
|
CREATE INDEX ix_platform_messages_platform_message
|
||||||
|
ON platform_messages (platform, external_message_id);
|
||||||
|
|
||||||
|
-- -- Recreate attendance stats function for new columns (prod back-compat)
|
||||||
|
CREATE OR REPLACE FUNCTION get_group_attendance_stats(p_group_id UUID)
|
||||||
|
RETURNS TABLE (
|
||||||
|
player_id UUID,
|
||||||
|
display_name VARCHAR,
|
||||||
|
telegram_username VARCHAR,
|
||||||
|
total_sessions BIGINT,
|
||||||
|
confirmed_count BIGINT,
|
||||||
|
declined_count BIGINT,
|
||||||
|
no_response_count BIGINT,
|
||||||
|
waitlisted_count BIGINT,
|
||||||
|
cancellation_affected_count BIGINT,
|
||||||
|
attendance_rate NUMERIC
|
||||||
|
) AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN QUERY
|
||||||
|
WITH player_sessions AS (
|
||||||
|
SELECT
|
||||||
|
sp.player_id,
|
||||||
|
s.id AS session_id,
|
||||||
|
sp.rsvp_status,
|
||||||
|
sp.registration_status,
|
||||||
|
s.status AS session_status,
|
||||||
|
s.scheduled_at
|
||||||
|
FROM session_participants sp
|
||||||
|
JOIN sessions s ON s.id = sp.session_id
|
||||||
|
WHERE s.group_id = p_group_id
|
||||||
|
),
|
||||||
|
player_totals AS (
|
||||||
|
SELECT
|
||||||
|
ps.player_id,
|
||||||
|
COUNT(*) FILTER (WHERE ps.session_status <> 'Cancelled') AS total_sessions,
|
||||||
|
COUNT(*) FILTER (WHERE ps.rsvp_status = 'Confirmed' AND ps.session_status <> 'Cancelled') AS confirmed_count,
|
||||||
|
COUNT(*) FILTER (WHERE ps.rsvp_status = 'Declined' AND ps.session_status <> 'Cancelled') AS declined_count,
|
||||||
|
COUNT(*) FILTER (WHERE ps.rsvp_status = 'Pending' AND ps.scheduled_at < NOW() AND ps.session_status <> 'Cancelled') AS no_response_count,
|
||||||
|
COUNT(*) FILTER (WHERE ps.registration_status = 'Waitlisted' AND ps.session_status <> 'Cancelled') AS waitlisted_count,
|
||||||
|
COUNT(*) FILTER (WHERE ps.session_status = 'Cancelled') AS cancellation_affected_count
|
||||||
|
FROM player_sessions ps
|
||||||
|
GROUP BY ps.player_id
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
pt.player_id,
|
||||||
|
p.display_name,
|
||||||
|
COALESCE(p.external_username, p.telegram_username) AS telegram_username,
|
||||||
|
pt.total_sessions,
|
||||||
|
pt.confirmed_count,
|
||||||
|
pt.declined_count,
|
||||||
|
pt.no_response_count,
|
||||||
|
pt.waitlisted_count,
|
||||||
|
pt.cancellation_affected_count,
|
||||||
|
ROUND(
|
||||||
|
100.0 * pt.confirmed_count
|
||||||
|
/ NULLIF(pt.total_sessions, 0),
|
||||||
|
1
|
||||||
|
) AS attendance_rate
|
||||||
|
FROM player_totals pt
|
||||||
|
JOIN players p ON p.id = pt.player_id
|
||||||
|
ORDER BY pt.confirmed_count DESC, pt.total_sessions DESC;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql STABLE;
|
||||||
@@ -1,9 +1,13 @@
|
|||||||
using GmRelay.Bot.Features.Confirmation.HandleRsvp;
|
using GmRelay.Bot.Features.Confirmation.HandleRsvp;
|
||||||
using GmRelay.Bot.Features.Confirmation.SendConfirmation;
|
using GmRelay.Bot.Features.Confirmation.SendConfirmation;
|
||||||
|
using GmRelay.Bot.Features.Notifications;
|
||||||
using GmRelay.Bot.Features.Reminders.SendJoinLink;
|
using GmRelay.Bot.Features.Reminders.SendJoinLink;
|
||||||
|
using GmRelay.Bot.Features.Reminders.SendOneHourReminder;
|
||||||
using GmRelay.Bot.Features.Sessions.CreateSession;
|
using GmRelay.Bot.Features.Sessions.CreateSession;
|
||||||
using GmRelay.Bot.Features.Sessions.RescheduleSession;
|
using GmRelay.Bot.Features.Sessions.RescheduleSession;
|
||||||
using GmRelay.Bot.Infrastructure.Database;
|
using GmRelay.Bot.Infrastructure.Database;
|
||||||
|
using GmRelay.Bot.Infrastructure.Health;
|
||||||
|
using GmRelay.Bot.Infrastructure.Logging;
|
||||||
using GmRelay.Bot.Infrastructure.Scheduling;
|
using GmRelay.Bot.Infrastructure.Scheduling;
|
||||||
using GmRelay.Bot.Infrastructure.Telegram;
|
using GmRelay.Bot.Infrastructure.Telegram;
|
||||||
using Npgsql;
|
using Npgsql;
|
||||||
@@ -20,11 +24,16 @@ builder.AddServiceDefaults();
|
|||||||
builder.Services.AddSingleton<NpgsqlDataSource>(sp =>
|
builder.Services.AddSingleton<NpgsqlDataSource>(sp =>
|
||||||
{
|
{
|
||||||
var config = sp.GetRequiredService<IConfiguration>();
|
var config = sp.GetRequiredService<IConfiguration>();
|
||||||
|
var loggerFactory = sp.GetRequiredService<ILoggerFactory>();
|
||||||
var connectionString = config.GetConnectionString("gmrelaydb")
|
var connectionString = config.GetConnectionString("gmrelaydb")
|
||||||
?? throw new InvalidOperationException(
|
?? throw new InvalidOperationException(
|
||||||
"ConnectionStrings:gmrelaydb is required. Set via environment variable ConnectionStrings__gmrelaydb.");
|
"ConnectionStrings:gmrelaydb is required. Set via environment variable ConnectionStrings__gmrelaydb.");
|
||||||
|
|
||||||
Console.WriteLine($"[DBG] Master ConnectionString => {connectionString}");
|
var logger = loggerFactory.CreateLogger("GmRelay.Bot.Startup");
|
||||||
|
logger.LogInformation(
|
||||||
|
"Configured PostgreSQL data source with connection string {ConnectionString}",
|
||||||
|
SecretRedactor.RedactConnectionString(connectionString));
|
||||||
|
|
||||||
return NpgsqlDataSource.Create(connectionString);
|
return NpgsqlDataSource.Create(connectionString);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -40,13 +49,21 @@ builder.Services.AddSingleton<ITelegramBotClient>(sp =>
|
|||||||
"Telegram:BotToken is required. Set via environment variable Telegram__BotToken or appsettings.json.");
|
"Telegram:BotToken is required. Set via environment variable Telegram__BotToken or appsettings.json.");
|
||||||
return new TelegramBotClient(token);
|
return new TelegramBotClient(token);
|
||||||
});
|
});
|
||||||
|
builder.Services.AddSingleton<ITelegramUpdateSource, TelegramUpdateSource>();
|
||||||
|
|
||||||
// ── Feature handlers (explicit registration — AOT safe) ──────────────
|
// ── Feature handlers (explicit registration — AOT safe) ──────────────
|
||||||
builder.Services.AddSingleton<SendConfirmationHandler>();
|
builder.Services.AddSingleton<SendConfirmationHandler>();
|
||||||
|
builder.Services.AddSingleton<ISendConfirmationHandler>(sp => sp.GetRequiredService<SendConfirmationHandler>());
|
||||||
|
builder.Services.AddSingleton<DirectSessionNotificationSender>();
|
||||||
builder.Services.AddSingleton<HandleRsvpHandler>();
|
builder.Services.AddSingleton<HandleRsvpHandler>();
|
||||||
builder.Services.AddSingleton<SendJoinLinkHandler>();
|
builder.Services.AddSingleton<SendJoinLinkHandler>();
|
||||||
|
builder.Services.AddSingleton<ISendJoinLinkHandler>(sp => sp.GetRequiredService<SendJoinLinkHandler>());
|
||||||
|
builder.Services.AddSingleton<SendOneHourReminderHandler>();
|
||||||
|
builder.Services.AddSingleton<ISendOneHourReminderHandler>(sp => sp.GetRequiredService<SendOneHourReminderHandler>());
|
||||||
builder.Services.AddSingleton<CreateSessionHandler>();
|
builder.Services.AddSingleton<CreateSessionHandler>();
|
||||||
builder.Services.AddSingleton<JoinSessionHandler>();
|
builder.Services.AddSingleton<JoinSessionHandler>();
|
||||||
|
builder.Services.AddSingleton<LeaveSessionHandler>();
|
||||||
|
builder.Services.AddSingleton<PromoteWaitlistedPlayerHandler>();
|
||||||
builder.Services.AddSingleton<CancelSessionHandler>();
|
builder.Services.AddSingleton<CancelSessionHandler>();
|
||||||
builder.Services.AddSingleton<GmRelay.Bot.Features.Sessions.ListSessions.DeleteSessionHandler>();
|
builder.Services.AddSingleton<GmRelay.Bot.Features.Sessions.ListSessions.DeleteSessionHandler>();
|
||||||
builder.Services.AddSingleton<GmRelay.Bot.Features.Sessions.ListSessions.ListSessionsHandler>();
|
builder.Services.AddSingleton<GmRelay.Bot.Features.Sessions.ListSessions.ListSessionsHandler>();
|
||||||
@@ -57,10 +74,20 @@ builder.Services.AddSingleton<HandleRescheduleVoteHandler>();
|
|||||||
|
|
||||||
// ── Telegram infrastructure ──────────────────────────────────────────
|
// ── Telegram infrastructure ──────────────────────────────────────────
|
||||||
builder.Services.AddSingleton<UpdateRouter>();
|
builder.Services.AddSingleton<UpdateRouter>();
|
||||||
|
builder.Services.AddSingleton<ITelegramUpdateHandler>(sp => sp.GetRequiredService<UpdateRouter>());
|
||||||
|
builder.Services.AddHostedService<TelegramMiniAppMenuButtonService>();
|
||||||
builder.Services.AddHostedService<TelegramBotService>();
|
builder.Services.AddHostedService<TelegramBotService>();
|
||||||
|
|
||||||
|
// ── Clock and scheduling ──────────────────────────────────────────────
|
||||||
|
builder.Services.AddSingleton<ISystemClock, SystemClock>();
|
||||||
|
builder.Services.AddSingleton<ISessionTriggerStore, DbSessionTriggerStore>();
|
||||||
|
|
||||||
// ── Session scheduler ────────────────────────────────────────────────
|
// ── Session scheduler ────────────────────────────────────────────────
|
||||||
builder.Services.AddHostedService<SessionSchedulerService>();
|
builder.Services.AddHostedService<SessionSchedulerService>();
|
||||||
|
builder.Services.AddHostedService<RescheduleVotingDeadlineService>();
|
||||||
|
|
||||||
|
// ── Health check server ──────────────────────────────────────────────
|
||||||
|
builder.Services.AddHostedService<BotHealthCheckHostedService>();
|
||||||
|
|
||||||
var host = builder.Build();
|
var host = builder.Build();
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,3 @@
|
|||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
|
||||||
|
[assembly: InternalsVisibleTo("GmRelay.Bot.Tests")]
|
||||||
@@ -7,6 +7,10 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"Telegram": {
|
"Telegram": {
|
||||||
"BotToken": ""
|
"BotToken": "",
|
||||||
|
"MiniAppUrl": ""
|
||||||
|
},
|
||||||
|
"Web": {
|
||||||
|
"BaseUrl": ""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,689 @@
|
|||||||
|
{
|
||||||
|
"version": 1,
|
||||||
|
"dependencies": {
|
||||||
|
"net10.0": {
|
||||||
|
"Aspire.Npgsql": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[13.2.2, )",
|
||||||
|
"resolved": "13.2.2",
|
||||||
|
"contentHash": "nEYgziWN7hksgEQEWy24JypcMCU8gKYcIIyPL05JfdXxUWuPRLotH/KOeuHevAjSEOYkL3dtGakBkJAuPobGmA==",
|
||||||
|
"dependencies": {
|
||||||
|
"AspNetCore.HealthChecks.NpgSql": "9.0.0",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5",
|
||||||
|
"Npgsql.DependencyInjection": "10.0.1",
|
||||||
|
"Npgsql.OpenTelemetry": "10.0.1",
|
||||||
|
"OpenTelemetry.Extensions.Hosting": "1.15.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Dapper": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[2.1.72, )",
|
||||||
|
"resolved": "2.1.72",
|
||||||
|
"contentHash": "ns4mGqQd9a/MhP8m6w556vVlZIa0/MfUu03zrxjZC/jlr1uVCsUac8bkdB+Fs98Llbd56rRSo1eZH5VVmeGZyw=="
|
||||||
|
},
|
||||||
|
"Dapper.AOT": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[1.0.48, )",
|
||||||
|
"resolved": "1.0.48",
|
||||||
|
"contentHash": "rsLM3yKr4g+YKKox9lhc8D+kz67P7Q9+xdyn1LmCsoYr1kYpJSm+Nt6slo5UrfUrcTiGJ57zUlyO8XUdV7G7iA=="
|
||||||
|
},
|
||||||
|
"dbup-postgresql": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[7.0.1, )",
|
||||||
|
"resolved": "7.0.1",
|
||||||
|
"contentHash": "mRnmENWWPuuMZ538gOd1mZnzucx6FQk0anmw3EABjGfcbp24FDb9QdGepYrDiaM8K9s5/gd49+5cmBOlniH/lg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Npgsql": "10.0.1",
|
||||||
|
"dbup-core": "6.1.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.DotNet.ILCompiler": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[10.0.5, )",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "yadTZIkStCVsG8nGwvfroSfBApPsgjQbodQyaIfp53dgayE0qhZpywixiCB6lx57JYQ+KVg1m1AFLrj54pxpZg=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Hosting": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[10.0.5, )",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "8i7e5IBdiKLNqt/+ciWrS8U95Rv5DClaaj7ulkZbimnCi4uREWd+lXzkp3joofFuIPOlAzV4AckxLTIELv2jdg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.CommandLine": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.EnvironmentVariables": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.FileExtensions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Json": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.UserSecrets": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Physical": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Console": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Debug": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.EventLog": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.EventSource": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.NET.ILLink.Tasks": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[10.0.5, )",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "A+5ZuQ0f449tM+MQrhf6R9ZX7lYpjk/ODEwLYKrnF6111rtARx8fVsm4YznUnQiKnnXfaXNBqgxmil6RW3L3SA=="
|
||||||
|
},
|
||||||
|
"Npgsql": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[10.0.2, )",
|
||||||
|
"resolved": "10.0.2",
|
||||||
|
"contentHash": "q5RfBI+wywJSFUNDE1L4ZbHEHCFTblo8Uf6A6oe4feOUFYiUQXyAf9GBh5qEZpvJaHiEbpBPkQumjEhXCJxdrg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"SecurityCodeScan.VS2019": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[5.6.7, )",
|
||||||
|
"resolved": "5.6.7",
|
||||||
|
"contentHash": "WIE9RJswdSc2j+rLz2gW6U+gMUjMHzY2j7C/CL8/R2olXNM/+twarfMnWqm+rZodDBvaYDApJyxM8mVYf9FGrQ=="
|
||||||
|
},
|
||||||
|
"Telegram.Bot": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[22.9.5.3, )",
|
||||||
|
"resolved": "22.9.5.3",
|
||||||
|
"contentHash": "7u8rZU9Vx9XEyIm6pB+dAlITsi1v63I+hKo7IEXGiQZnVjzvZgPs9yDCP17/Cwm7lgjCNEqknlbv/yoBnsUYFw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": "8.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"AspNetCore.HealthChecks.NpgSql": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "9.0.0",
|
||||||
|
"contentHash": "npc58/AD5zuVxERdhCl2Kb7WnL37mwX42SJcXIwvmEig0/dugOLg3SIwtfvvh3TnvTwR/sk5LYNkkPaBdks61A==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": "8.0.11",
|
||||||
|
"Npgsql": "8.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"dbup-core": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "6.1.1",
|
||||||
|
"contentHash": "kgpuyJVEFJHoIj/slnc994Go88aoeZqNDfGHDBr4sh7CsEWwJhOTCt/FJqO4ziUImL5L0NEY0kxxOiNgPKI2Fw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "8.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.AmbientMetadata.Application": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "CNrEjaOCZ8d1HtB0mvpiX4EWxLkee2xy+CsYXxmsEYJSFgw3OmF9pIhP/tCTeYBHhpsKJj5wM63G8IBFGxAcsw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": "10.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Compliance.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "1a4xDAT6fRyP8t419q3WvWMmMslDTvI7OAZLWBhn5rysFG0bl5xFenTswd1xAbT/3u3mx4Xyb5bPx+V+18tJeQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.ObjectPool": "10.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "8Rx5sqg04FttxrumyG6bmoRuFRgYzK6IVwF1i0/o0cXfKBdDeVpJejKHtJCMjyg9E/DNMVqpqOGe/tCT5gYvVA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "P09QpTHjqHmCLQOTC+WyLkoRNxek4NIvfWt+TnU0etoDUSRxcltyd6+j/ouRbMdLR0j44GqGO+lhI2M4fAHG4g==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "99Z4rjyXopb1MIazDSPcvwYCUdYNO01Cf1GUs2WUjIFAbkGmwzj2vPa2k+3pheJRV+YgNd2QqRKHAri0oBAU4Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.CommandLine": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "or9fOLopMUTJOQVJ3bou4aD6PwvsiKf4kZC4EE5sRRKSkmh+wfk/LekJXRjAX88X+1JA9zHjDo+5fiQ7z3MY/A==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.EnvironmentVariables": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "tchMGQ+zVTO40np/Zzg2Li/TIR8bksQgg4UVXZa0OzeFCKWnIYtxE2FVs+eSmjPGCjMS2voZbwN/mUcYfpSTuA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.FileExtensions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "OhTr0O79dP49734lLTqVveivVX9sDXxbI/8vjELAZTHXqoN90mdpgTAgwicJED42iaHMCcZcK6Bj+8wNyBikaw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Physical": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.Json": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "brBM/WP0YAUYh2+QqSYVdK8eQHYQTtTEUJXJ+84Zkdo2buGLja9VSrMIhgoeBUU7JBmcskAib8Lb/N83bvxgYQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.FileExtensions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Configuration.UserSecrets": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "fhdG6UV9lIp70QhNkVyaHciUVq25IPFkczheVJL9bIFvmnJ+Zghaie6dWkDbbVmxZlHl9gj3zTDxMxJs5zNhIA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Json": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Physical": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.DependencyInjection": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "v1SVsowG6YE1YnHVGmLWz57YTRCQRx9pH5ebIESXfm5isI9gA3QaMyg/oMTzPpXYZwSAVDzYItGJKfmV+pqXkQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "iVMtq9eRvzyhx8949EGT0OCYJfXi737SbRVzWXE5GrOgGj5AaZ9eUuxA/BSUfmOMALKn/g8KfFaNQw0eiB3lyA=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.DependencyInjection.AutoActivation": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "Z/OI261l7LnxyODKPx0trQyIHFyicCR/akfn64lGOjPcf4FpAZ7ePAGl2HPvQBUBSNfPTF0gWeCfuFmyftMgYA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "vAJHd4yOpmKoK+jBuYV7a3y+Ab9U4ARCc29b6qvMy276RgJFw9LFs0DdsPqOL3ahwzyrX7tM+i4cCxU/RX0qAg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "/nYGrpa9/0BZofrVpBbbj+Ns8ZesiPE0V/KxsuHgDgHQopIzN54nRaQGSuvPw16/kI9sW1Zox5yyAPqvf0Jz6A==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics.ExceptionSummarization": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "3qMK1D40D10kb5TdBtFJpzz6/WH0NinWs68ZZS8jCFgHMXDiOjGiPOneMmIocCP/wnUUW4Hzf8lMsIE1xIGxDA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "REdt95QXHscGdtw/UUgyCW2lF9DJcAOJxmebKW2IkgUjuCAdMODIi2HNOWg5utW98nm8ekgV0Gjqs/sljwwqMw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics.HealthChecks.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "NrIMTy7dpqxAvA6kHAYH8cXID/YgeNOy0OqFKpLtkPu5X4WS/basX91UszANzVrMNRAICJ2GOnGiRxJtsRyEQw=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Features": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.2",
|
||||||
|
"contentHash": "X7tm2aV2w3lN9roSSGhl19lz4w76HvdiuKNhIv2XOiorYII9XCm66o/z9IJ0+QwkgvEv5gMZDM6rV6uwABHEQQ=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "nCBmCx0Xemlu65ZiWMcXbvfvtznKxf4/YYKF9R28QkqdI9lTikedGqzJ28/xmdGGsxUnsP5/3TQGpiPwVjK0dA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.FileProviders.Physical": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "dMu5kUPSfol1Rqhmr6nWPSmbFjDe9w6bkoKithG17bWTZA0UyKirTatM5mqYUN3mGpNA0MorlusIoVTh6J7o5g==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileSystemGlobbing": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.FileSystemGlobbing": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "mOE3ARusNQR0a5x8YOcnUbfyyXGqoAWQtEc7qFOfNJgruDWQLo39Re+3/Lzj5pLPFuFYj8hN4dgKzaSQDKiOCw=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "+Wb7KAMVZTomwJkQrjuPTe5KBzGod7N8XeG+ScxRlkPOB4sZLG4ccVwjV4Phk5BCJt7uIMnGHVoN6ZMVploX+g==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Diagnostics.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.FileProviders.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Http": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.2",
|
||||||
|
"contentHash": "egUPC0xydb1ugCMcRyJ6zaOGOzx7N4coOVlGeLcIsXhUf1xHHwZeX+ob7JuG0dXExFduHYE/t+4/4y8BLlBKmw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Diagnostics": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Http.Diagnostics": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "I0FBgF6yZRwYH9E3KQ2vHm80YZ7YBj+52GDsmOWXPBv/p15b/wUoNupV9kw3LnSNVsWMqlGbiuZgBnHpMwPh+Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Http": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Telemetry": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Http.Resilience": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "Lg+OjBW+ODDbM4Ax4LoERvQ1dqSZ8I2gQc2+B0/WOWl2+PunLJ3xb3x8MtHGfcb/Mp98RoMpwRKm6Aj9mzXwrA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Http.Diagnostics": "10.2.0",
|
||||||
|
"Microsoft.Extensions.ObjectPool": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Resilience": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "+XTMKQyDWg4ODoNHU/BN3BaI1jhGO7VCS+BnzT/4IauiG6y2iPAte7MyD7rHKS+hNP0TkFkjrae8DFjDUxtcxg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "9HOdqlDtPptVcmKAjsQ/Nr5Rxfq6FMYLdhvZh1lVmeKR738qeYecQD7+ldooXf+u2KzzR1kafSphWngIM3C6ug==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.Configuration": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "cSgxsDgfP0+gmVRPVoNHI/KIDavIZxh+CxE6tSLPlYTogqccDnjBFI9CgEsiNuMP6+fiuXUwhhlTz36uUEpwbQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.Console": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "PMs2gha2v24hvH5o5KQem5aNK4mN0BhhCWlMqsg9tzifWKzjeQi2tyPOP/RaWMVvalOhVLcrmoMYPqbnia/epg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Configuration": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.Debug": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "/VacEkBQ02A8PBXSa6YpbIXCuisYy6JJr62/+ANJDZE+RMBfZMcXJXLfr/LpyLE6pgdp17Wxlt7e7R9zvkwZ3Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.EventLog": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "0ezhWYJS4/6KrqQel9JL+Tr4n+4EX2TF5EYiaysBWNNEM2c3Gtj1moD39esfgk8OHblSX+UFjtZ3z0c4i9tRvw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"System.Diagnostics.EventLog": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Logging.EventSource": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "vN+aq1hBFXyYvY5Ow9WyeR66drKQxRZmas4lAjh6QWfryPkjTn1uLtX5AFIxyDaZj78v5TG2sELUyvrXpAPQQw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.ObjectPool": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.2",
|
||||||
|
"contentHash": "kpCp4m7nwJVBcRKWXYHdVK/W0dkKyyFOjCmKVdO+zKThWvUxP1V+jVEP9FGpqRu4GPl9041SEXu2f+U/l825nQ=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Options": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "MDaQMdUplw0AIRhWWmbLA7yQEXaLIHb+9CTroTiNS8OlI0LMXS4LCxtopqauiqGCWlRgJ+xyraVD8t6veRAFbw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "BB9uUW3+6Rxu1R97OB1H/13lUF8P2+H1+eDhpZlK30kDh/6E4EKHBUqTp+ilXQmZLzsRErxON8aBSR6WpUKJdg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.5",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.5",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Primitives": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "/HUHJ0tw/LQvD0DZrz50eQy/3z7PfX7WWEaXnjKTV9/TNdcgFlNTZGo49QhS7PTmhDqMyHRMqAXSBxLh0vso4g=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Resilience": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "v4WOdAOFxB3AcsUkZWNcHL3mYzs4KAPtHO8rkoQlFKOBoD3KyjjAL+h3tRwSK5i4UpF/yhxsQRY0JxKj4osxxw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Diagnostics": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Diagnostics.ExceptionSummarization": "10.2.0",
|
||||||
|
"Microsoft.Extensions.Options.ConfigurationExtensions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Telemetry.Abstractions": "10.2.0",
|
||||||
|
"Polly.Extensions": "8.4.2",
|
||||||
|
"Polly.RateLimiting": "8.4.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.ServiceDiscovery": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "AHTPfiKodj66xA8RwRkFD4q11V2AvzcuDsujv6ViPkOPtvBEYcPVplHakK56pPzWlX08MDS+TAQXfFXAeP7J5w==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Http": "10.0.2",
|
||||||
|
"Microsoft.Extensions.ServiceDiscovery.Abstractions": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.ServiceDiscovery.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "sANlOvfqfw/yfych4CLlHSKSWzIie6mQG7w83gVur1foNOafyHxcgpoQMvBf+KiB4Tpls6P1/Z77IIQSK8hxFg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Configuration.Binder": "10.0.2",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Features": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Primitives": "10.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Telemetry": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "ssW5gosYlewNH/ISTyaLD/XfJT4GSjwShOUKv61fpXrqVmHkhuIA/5bBAGStM1XbzJjt9IG2vzfdHTu4zlX9Ew==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.AmbientMetadata.Application": "10.2.0",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.AutoActivation": "10.2.0",
|
||||||
|
"Microsoft.Extensions.Logging.Configuration": "10.0.2",
|
||||||
|
"Microsoft.Extensions.ObjectPool": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Telemetry.Abstractions": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Telemetry.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "6V4V6NX6RLUYWwV89DeW/4zK5xOycYHWhsfMXSpKVGgMHfXcczmbk6hBeqTnRPzhpATYcOWlmA6hk1jgdxUugA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Compliance.Abstractions": "10.2.0",
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "10.0.2",
|
||||||
|
"Microsoft.Extensions.ObjectPool": "10.0.2",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Npgsql.DependencyInjection": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.1",
|
||||||
|
"contentHash": "YHFa4vD27sNIfv6s5q8Zi1fLvKfmK1xcpMv0PUvXOxDFbRmuMRSHwpZTbPvsAlj97q1/o7DfyynLqfqrCm1VnA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "8.0.0",
|
||||||
|
"Npgsql": "10.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Npgsql.OpenTelemetry": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.1",
|
||||||
|
"contentHash": "G9fEIBaHggZXWfDSDnKLc0XwKcbuU6i2eXp7zDqpgYxbhCmIN9fRgaSOGyyMNHSo/yY1IB4G4CjW5VO/SKRR0g==",
|
||||||
|
"dependencies": {
|
||||||
|
"Npgsql": "10.0.1",
|
||||||
|
"OpenTelemetry.API": "1.14.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "N0i6WjPoHPbZyms1ugbDIFAJFuGlpeExJMU/+XSL0lQRUkg/D0utFkDoLXf8Z1km5B+xVZ2GyMXXiX8qdeNmPg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Diagnostics.Abstractions": "10.0.0",
|
||||||
|
"Microsoft.Extensions.Logging.Configuration": "10.0.0",
|
||||||
|
"OpenTelemetry.Api.ProviderBuilderExtensions": "1.15.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Api": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "fX+fkCysfPut+qCcT3bKqyX4QN9Saf4CgX8HLOHywEVD+Xr7sULtfuypITpoDysjx8R59dn/3mWhgimMH8cm/g=="
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Api.ProviderBuilderExtensions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "SYn0lqYDwLMWhv/zlNGsQcl2yX++yTumanX46bmOZE/ZDOd1WjPBO2kZaZgKLEZTZk48pavIFGJ6vOvxXgWVFQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.DependencyInjection.Abstractions": "10.0.0",
|
||||||
|
"OpenTelemetry.Api": "1.15.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Exporter.OpenTelemetryProtocol": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "FEXJepcseTGbATiCkUfP7ipoFEYYfl/0UmmUwi0KxCPg9PaUA8ab2P1LGopK+/HExasJ1ZutFhZrN6WvUIR23g==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry": "1.15.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Extensions.Hosting": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "u8n/W8yIlqv0BXZmvId1iVaeWXG42tGKdTkuLYg5g57Y/r9CeUNzqtrSHNdG5IoO8iPX79w3v+WsbAHgUQbfeg==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Hosting.Abstractions": "10.0.0",
|
||||||
|
"OpenTelemetry": "1.15.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Instrumentation.AspNetCore": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.2",
|
||||||
|
"contentHash": "2nPd7r0ug/gd6/CNFL6Rlu+RSQ9WYGSGHAYQ1ssbSqyzKJpqTunfx2I/1O0WB5k+L0cyXbG4XVZpoSoUc3M7wg==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry.Api.ProviderBuilderExtensions": "[1.15.3, 2.0.0)"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Instrumentation.Http": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.1",
|
||||||
|
"contentHash": "vFO4Fj/dXkoVNGo/nhoGpO2zYQmZwr4jTID7oRGo+XlQ8LqksyZjUXQ4p39RfUvTID7IzzL8Qe71tW7CcAFymA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Configuration": "10.0.0",
|
||||||
|
"Microsoft.Extensions.Options": "10.0.0",
|
||||||
|
"OpenTelemetry.Api.ProviderBuilderExtensions": "[1.15.3, 2.0.0)"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Instrumentation.Runtime": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.1",
|
||||||
|
"contentHash": "cpPwlUT5HXcLGPaIgsbSy0W9eFYAPGVbTP1p8/uyQ4Osvf5BJuPpEXE7crL09SmEd44r0DGNKDtsqxaAz0HxQw==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry.Api": "[1.15.3, 2.0.0)"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Polly.Core": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.4.2",
|
||||||
|
"contentHash": "BpE2I6HBYYA5tF0Vn4eoQOGYTYIK1BlF5EXVgkWGn3mqUUjbXAr13J6fZVbp7Q3epRR8yshacBMlsHMhpOiV3g=="
|
||||||
|
},
|
||||||
|
"Polly.Extensions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.4.2",
|
||||||
|
"contentHash": "GZ9vRVmR0jV2JtZavt+pGUsQ1O1cuRKG7R7VOZI6ZDy9y6RNPvRvXK1tuS4ffUrv8L0FTea59oEuQzgS0R7zSA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Logging.Abstractions": "8.0.0",
|
||||||
|
"Microsoft.Extensions.Options": "8.0.0",
|
||||||
|
"Polly.Core": "8.4.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Polly.RateLimiting": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.4.2",
|
||||||
|
"contentHash": "ehTImQ/eUyO07VYW2WvwSmU9rRH200SKJ/3jku9rOkyWE0A2JxNFmAVms8dSn49QLSjmjFRRSgfNyOgr/2PSmA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Polly.Core": "8.4.2",
|
||||||
|
"System.Threading.RateLimiting": "8.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"System.Diagnostics.EventLog": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "wugvy+pBVzjQEnRs9wMTWwoaeNFX3hsaHeVHFDIvJSWXp7wfmNWu3mxAwBIE6pyW+g6+rHa1Of5fTzb0QVqUTA=="
|
||||||
|
},
|
||||||
|
"System.Threading.RateLimiting": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.0.0",
|
||||||
|
"contentHash": "7mu9v0QDv66ar3DpGSZHg9NuNcxDaaAcnMULuZlaTpP9+hwXhrxNGsF5GmLkSHxFdb5bBc1TzeujsRgTrPWi+Q=="
|
||||||
|
},
|
||||||
|
"gmrelay.servicedefaults": {
|
||||||
|
"type": "Project",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Http.Resilience": "[10.2.0, )",
|
||||||
|
"Microsoft.Extensions.ServiceDiscovery": "[10.2.0, )",
|
||||||
|
"OpenTelemetry.Exporter.OpenTelemetryProtocol": "[1.15.3, )",
|
||||||
|
"OpenTelemetry.Extensions.Hosting": "[1.15.3, )",
|
||||||
|
"OpenTelemetry.Instrumentation.AspNetCore": "[1.15.2, )",
|
||||||
|
"OpenTelemetry.Instrumentation.Http": "[1.15.1, )",
|
||||||
|
"OpenTelemetry.Instrumentation.Runtime": "[1.15.1, )"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"gmrelay.shared": {
|
||||||
|
"type": "Project"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"net10.0/win-x64": {
|
||||||
|
"Microsoft.DotNet.ILCompiler": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[10.0.5, )",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "yadTZIkStCVsG8nGwvfroSfBApPsgjQbodQyaIfp53dgayE0qhZpywixiCB6lx57JYQ+KVg1m1AFLrj54pxpZg==",
|
||||||
|
"dependencies": {
|
||||||
|
"runtime.win-x64.Microsoft.DotNet.ILCompiler": "10.0.5"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"runtime.win-x64.Microsoft.DotNet.ILCompiler": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "vblLkpVhSDYOmrEW0jypX7YVtLg7idU1QzUyx45ZdZ2sFUSSf3mYFCr0FW3+KZgXWpN1ve9ZPrxNywvHISF4bA=="
|
||||||
|
},
|
||||||
|
"System.Diagnostics.EventLog": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.0.5",
|
||||||
|
"contentHash": "wugvy+pBVzjQEnRs9wMTWwoaeNFX3hsaHeVHFDIvJSWXp7wfmNWu3mxAwBIE6pyW+g6+rHa1Of5fTzb0QVqUTA=="
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -12,11 +12,11 @@
|
|||||||
|
|
||||||
<PackageReference Include="Microsoft.Extensions.Http.Resilience" Version="10.2.0" />
|
<PackageReference Include="Microsoft.Extensions.Http.Resilience" Version="10.2.0" />
|
||||||
<PackageReference Include="Microsoft.Extensions.ServiceDiscovery" Version="10.2.0" />
|
<PackageReference Include="Microsoft.Extensions.ServiceDiscovery" Version="10.2.0" />
|
||||||
<PackageReference Include="OpenTelemetry.Exporter.OpenTelemetryProtocol" Version="1.15.0" />
|
<PackageReference Include="OpenTelemetry.Exporter.OpenTelemetryProtocol" Version="1.15.3" />
|
||||||
<PackageReference Include="OpenTelemetry.Extensions.Hosting" Version="1.15.0" />
|
<PackageReference Include="OpenTelemetry.Extensions.Hosting" Version="1.15.3" />
|
||||||
<PackageReference Include="OpenTelemetry.Instrumentation.AspNetCore" Version="1.15.0" />
|
<PackageReference Include="OpenTelemetry.Instrumentation.AspNetCore" Version="1.15.2" />
|
||||||
<PackageReference Include="OpenTelemetry.Instrumentation.Http" Version="1.15.0" />
|
<PackageReference Include="OpenTelemetry.Instrumentation.Http" Version="1.15.1" />
|
||||||
<PackageReference Include="OpenTelemetry.Instrumentation.Runtime" Version="1.15.0" />
|
<PackageReference Include="OpenTelemetry.Instrumentation.Runtime" Version="1.15.1" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@@ -0,0 +1,181 @@
|
|||||||
|
{
|
||||||
|
"version": 1,
|
||||||
|
"dependencies": {
|
||||||
|
"net10.0": {
|
||||||
|
"Microsoft.Extensions.Http.Resilience": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[10.2.0, )",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "Lg+OjBW+ODDbM4Ax4LoERvQ1dqSZ8I2gQc2+B0/WOWl2+PunLJ3xb3x8MtHGfcb/Mp98RoMpwRKm6Aj9mzXwrA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Http.Diagnostics": "10.2.0",
|
||||||
|
"Microsoft.Extensions.Resilience": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.ServiceDiscovery": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[10.2.0, )",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "AHTPfiKodj66xA8RwRkFD4q11V2AvzcuDsujv6ViPkOPtvBEYcPVplHakK56pPzWlX08MDS+TAQXfFXAeP7J5w==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.ServiceDiscovery.Abstractions": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Exporter.OpenTelemetryProtocol": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[1.15.3, )",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "FEXJepcseTGbATiCkUfP7ipoFEYYfl/0UmmUwi0KxCPg9PaUA8ab2P1LGopK+/HExasJ1ZutFhZrN6WvUIR23g==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry": "1.15.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Extensions.Hosting": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[1.15.3, )",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "u8n/W8yIlqv0BXZmvId1iVaeWXG42tGKdTkuLYg5g57Y/r9CeUNzqtrSHNdG5IoO8iPX79w3v+WsbAHgUQbfeg==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry": "1.15.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Instrumentation.AspNetCore": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[1.15.2, )",
|
||||||
|
"resolved": "1.15.2",
|
||||||
|
"contentHash": "2nPd7r0ug/gd6/CNFL6Rlu+RSQ9WYGSGHAYQ1ssbSqyzKJpqTunfx2I/1O0WB5k+L0cyXbG4XVZpoSoUc3M7wg==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry.Api.ProviderBuilderExtensions": "[1.15.3, 2.0.0)"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Instrumentation.Http": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[1.15.1, )",
|
||||||
|
"resolved": "1.15.1",
|
||||||
|
"contentHash": "vFO4Fj/dXkoVNGo/nhoGpO2zYQmZwr4jTID7oRGo+XlQ8LqksyZjUXQ4p39RfUvTID7IzzL8Qe71tW7CcAFymA==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry.Api.ProviderBuilderExtensions": "[1.15.3, 2.0.0)"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Instrumentation.Runtime": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[1.15.1, )",
|
||||||
|
"resolved": "1.15.1",
|
||||||
|
"contentHash": "cpPwlUT5HXcLGPaIgsbSy0W9eFYAPGVbTP1p8/uyQ4Osvf5BJuPpEXE7crL09SmEd44r0DGNKDtsqxaAz0HxQw==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry.Api": "[1.15.3, 2.0.0)"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"SecurityCodeScan.VS2019": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[5.6.7, )",
|
||||||
|
"resolved": "5.6.7",
|
||||||
|
"contentHash": "WIE9RJswdSc2j+rLz2gW6U+gMUjMHzY2j7C/CL8/R2olXNM/+twarfMnWqm+rZodDBvaYDApJyxM8mVYf9FGrQ=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.AmbientMetadata.Application": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "CNrEjaOCZ8d1HtB0mvpiX4EWxLkee2xy+CsYXxmsEYJSFgw3OmF9pIhP/tCTeYBHhpsKJj5wM63G8IBFGxAcsw=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Compliance.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "1a4xDAT6fRyP8t419q3WvWMmMslDTvI7OAZLWBhn5rysFG0bl5xFenTswd1xAbT/3u3mx4Xyb5bPx+V+18tJeQ=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.DependencyInjection.AutoActivation": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "Z/OI261l7LnxyODKPx0trQyIHFyicCR/akfn64lGOjPcf4FpAZ7ePAGl2HPvQBUBSNfPTF0gWeCfuFmyftMgYA=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Diagnostics.ExceptionSummarization": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "3qMK1D40D10kb5TdBtFJpzz6/WH0NinWs68ZZS8jCFgHMXDiOjGiPOneMmIocCP/wnUUW4Hzf8lMsIE1xIGxDA=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Http.Diagnostics": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "I0FBgF6yZRwYH9E3KQ2vHm80YZ7YBj+52GDsmOWXPBv/p15b/wUoNupV9kw3LnSNVsWMqlGbiuZgBnHpMwPh+Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Telemetry": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Resilience": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "v4WOdAOFxB3AcsUkZWNcHL3mYzs4KAPtHO8rkoQlFKOBoD3KyjjAL+h3tRwSK5i4UpF/yhxsQRY0JxKj4osxxw==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Diagnostics.ExceptionSummarization": "10.2.0",
|
||||||
|
"Microsoft.Extensions.Telemetry.Abstractions": "10.2.0",
|
||||||
|
"Polly.Extensions": "8.4.2",
|
||||||
|
"Polly.RateLimiting": "8.4.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.ServiceDiscovery.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "sANlOvfqfw/yfych4CLlHSKSWzIie6mQG7w83gVur1foNOafyHxcgpoQMvBf+KiB4Tpls6P1/Z77IIQSK8hxFg=="
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Telemetry": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "ssW5gosYlewNH/ISTyaLD/XfJT4GSjwShOUKv61fpXrqVmHkhuIA/5bBAGStM1XbzJjt9IG2vzfdHTu4zlX9Ew==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.AmbientMetadata.Application": "10.2.0",
|
||||||
|
"Microsoft.Extensions.DependencyInjection.AutoActivation": "10.2.0",
|
||||||
|
"Microsoft.Extensions.Telemetry.Abstractions": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Microsoft.Extensions.Telemetry.Abstractions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "10.2.0",
|
||||||
|
"contentHash": "6V4V6NX6RLUYWwV89DeW/4zK5xOycYHWhsfMXSpKVGgMHfXcczmbk6hBeqTnRPzhpATYcOWlmA6hk1jgdxUugA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Microsoft.Extensions.Compliance.Abstractions": "10.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "N0i6WjPoHPbZyms1ugbDIFAJFuGlpeExJMU/+XSL0lQRUkg/D0utFkDoLXf8Z1km5B+xVZ2GyMXXiX8qdeNmPg==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry.Api.ProviderBuilderExtensions": "1.15.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Api": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "fX+fkCysfPut+qCcT3bKqyX4QN9Saf4CgX8HLOHywEVD+Xr7sULtfuypITpoDysjx8R59dn/3mWhgimMH8cm/g=="
|
||||||
|
},
|
||||||
|
"OpenTelemetry.Api.ProviderBuilderExtensions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "1.15.3",
|
||||||
|
"contentHash": "SYn0lqYDwLMWhv/zlNGsQcl2yX++yTumanX46bmOZE/ZDOd1WjPBO2kZaZgKLEZTZk48pavIFGJ6vOvxXgWVFQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"OpenTelemetry.Api": "1.15.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Polly.Core": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.4.2",
|
||||||
|
"contentHash": "BpE2I6HBYYA5tF0Vn4eoQOGYTYIK1BlF5EXVgkWGn3mqUUjbXAr13J6fZVbp7Q3epRR8yshacBMlsHMhpOiV3g=="
|
||||||
|
},
|
||||||
|
"Polly.Extensions": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.4.2",
|
||||||
|
"contentHash": "GZ9vRVmR0jV2JtZavt+pGUsQ1O1cuRKG7R7VOZI6ZDy9y6RNPvRvXK1tuS4ffUrv8L0FTea59oEuQzgS0R7zSA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Polly.Core": "8.4.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Polly.RateLimiting": {
|
||||||
|
"type": "Transitive",
|
||||||
|
"resolved": "8.4.2",
|
||||||
|
"contentHash": "ehTImQ/eUyO07VYW2WvwSmU9rRH200SKJ/3jku9rOkyWE0A2JxNFmAVms8dSn49QLSjmjFRRSgfNyOgr/2PSmA==",
|
||||||
|
"dependencies": {
|
||||||
|
"Polly.Core": "8.4.2"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
namespace GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
public enum CalendarSubscriptionFilter
|
||||||
|
{
|
||||||
|
AllMyGroups = 0,
|
||||||
|
SpecificGroup = 1
|
||||||
|
}
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
namespace GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
public enum GroupManagerRole
|
||||||
|
{
|
||||||
|
Owner,
|
||||||
|
CoGm
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class GroupManagerRoleExtensions
|
||||||
|
{
|
||||||
|
public const string OwnerValue = "Owner";
|
||||||
|
public const string CoGmValue = "CoGm";
|
||||||
|
|
||||||
|
public static string ToDatabaseValue(this GroupManagerRole role) => role switch
|
||||||
|
{
|
||||||
|
GroupManagerRole.Owner => OwnerValue,
|
||||||
|
GroupManagerRole.CoGm => CoGmValue,
|
||||||
|
_ => throw new ArgumentOutOfRangeException(nameof(role), role, "Unknown group manager role.")
|
||||||
|
};
|
||||||
|
|
||||||
|
public static GroupManagerRole FromDatabaseValue(string value) => value switch
|
||||||
|
{
|
||||||
|
OwnerValue => GroupManagerRole.Owner,
|
||||||
|
CoGmValue => GroupManagerRole.CoGm,
|
||||||
|
_ => throw new ArgumentOutOfRangeException(nameof(value), value, "Unknown group manager role.")
|
||||||
|
};
|
||||||
|
|
||||||
|
public static string ToDisplayName(this GroupManagerRole role) => role switch
|
||||||
|
{
|
||||||
|
GroupManagerRole.Owner => "Owner",
|
||||||
|
GroupManagerRole.CoGm => "Co-GM",
|
||||||
|
_ => throw new ArgumentOutOfRangeException(nameof(role), role, "Unknown group manager role.")
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -11,7 +11,7 @@ public static class MoscowTime
|
|||||||
public static string FormatMoscow(this DateTimeOffset utc)
|
public static string FormatMoscow(this DateTimeOffset utc)
|
||||||
=> utc.ToOffset(MoscowOffset).ToString("d MMMM yyyy, HH:mm", System.Globalization.CultureInfo.GetCultureInfo("ru-RU"));
|
=> utc.ToOffset(MoscowOffset).ToString("d MMMM yyyy, HH:mm", System.Globalization.CultureInfo.GetCultureInfo("ru-RU"));
|
||||||
|
|
||||||
public static DateTime ToMoscow(this DateTime utcDt) => utcDt.Add(MoscowOffset);
|
public static DateTime ToMoscow(this DateTime utcDt) => DateTime.SpecifyKind(utcDt.Add(MoscowOffset), DateTimeKind.Unspecified);
|
||||||
|
|
||||||
public static string FormatMoscow(this DateTime utcDt)
|
public static string FormatMoscow(this DateTime utcDt)
|
||||||
=> utcDt.Add(MoscowOffset).ToString("d MMMM yyyy, HH:mm", System.Globalization.CultureInfo.GetCultureInfo("ru-RU"));
|
=> utcDt.Add(MoscowOffset).ToString("d MMMM yyyy, HH:mm", System.Globalization.CultureInfo.GetCultureInfo("ru-RU"));
|
||||||
@@ -21,7 +21,7 @@ public static class MoscowTime
|
|||||||
|
|
||||||
public static bool TryParseMoscow(string text, out DateTimeOffset utcTime)
|
public static bool TryParseMoscow(string text, out DateTimeOffset utcTime)
|
||||||
{
|
{
|
||||||
if (DateTime.TryParseExact(text, new[] { "dd.MM.yyyy HH:mm", "dd.MM.yyyy H:mm", "d.MM.yyyy HH:mm" },
|
if (DateTime.TryParseExact(text, new[] { "dd.MM.yyyy HH:mm", "dd.MM.yyyy H:mm", "d.MM.yyyy HH:mm" },
|
||||||
System.Globalization.CultureInfo.InvariantCulture, System.Globalization.DateTimeStyles.None, out var localDt))
|
System.Globalization.CultureInfo.InvariantCulture, System.Globalization.DateTimeStyles.None, out var localDt))
|
||||||
{
|
{
|
||||||
utcTime = new DateTimeOffset(localDt, MoscowOffset).ToUniversalTime();
|
utcTime = new DateTimeOffset(localDt, MoscowOffset).ToUniversalTime();
|
||||||
|
|||||||
@@ -0,0 +1,9 @@
|
|||||||
|
namespace GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
public static class ParticipantRegistrationStatus
|
||||||
|
{
|
||||||
|
public const string Active = "Active";
|
||||||
|
public const string Waitlisted = "Waitlisted";
|
||||||
|
|
||||||
|
public static readonly string[] All = [Active, Waitlisted];
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
namespace GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
public static class SessionCapacityRules
|
||||||
|
{
|
||||||
|
public static string DecideJoinStatus(int? maxPlayers, int activeParticipants)
|
||||||
|
{
|
||||||
|
if (!maxPlayers.HasValue)
|
||||||
|
{
|
||||||
|
return ParticipantRegistrationStatus.Active;
|
||||||
|
}
|
||||||
|
|
||||||
|
return activeParticipants < maxPlayers.Value
|
||||||
|
? ParticipantRegistrationStatus.Active
|
||||||
|
: ParticipantRegistrationStatus.Waitlisted;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool CanPromoteWaitlistedPlayer(int? maxPlayers, int activeParticipants, int waitlistedParticipants)
|
||||||
|
{
|
||||||
|
if (waitlistedParticipants <= 0)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return !maxPlayers.HasValue || activeParticipants < maxPlayers.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool ShouldPromoteAfterParticipantLeaves(
|
||||||
|
string removedRegistrationStatus,
|
||||||
|
int? maxPlayers,
|
||||||
|
int activeParticipantsAfterLeave,
|
||||||
|
int waitlistedParticipants)
|
||||||
|
{
|
||||||
|
return removedRegistrationStatus == ParticipantRegistrationStatus.Active
|
||||||
|
&& CanPromoteWaitlistedPlayer(maxPlayers, activeParticipantsAfterLeave, waitlistedParticipants);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
namespace GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
public enum SessionNotificationMode
|
||||||
|
{
|
||||||
|
GroupAndDirect,
|
||||||
|
GroupOnly
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class SessionNotificationModeExtensions
|
||||||
|
{
|
||||||
|
public const string GroupAndDirectValue = nameof(SessionNotificationMode.GroupAndDirect);
|
||||||
|
public const string GroupOnlyValue = nameof(SessionNotificationMode.GroupOnly);
|
||||||
|
|
||||||
|
public static bool ShouldSendDirectMessages(this SessionNotificationMode mode) =>
|
||||||
|
mode == SessionNotificationMode.GroupAndDirect;
|
||||||
|
|
||||||
|
public static string ToDatabaseValue(this SessionNotificationMode mode) =>
|
||||||
|
mode switch
|
||||||
|
{
|
||||||
|
SessionNotificationMode.GroupAndDirect => GroupAndDirectValue,
|
||||||
|
SessionNotificationMode.GroupOnly => GroupOnlyValue,
|
||||||
|
_ => throw new ArgumentOutOfRangeException(nameof(mode), mode, "Unknown notification mode.")
|
||||||
|
};
|
||||||
|
|
||||||
|
public static SessionNotificationMode FromDatabaseValue(string? value) =>
|
||||||
|
value switch
|
||||||
|
{
|
||||||
|
null or "" => SessionNotificationMode.GroupAndDirect,
|
||||||
|
GroupAndDirectValue => SessionNotificationMode.GroupAndDirect,
|
||||||
|
GroupOnlyValue => SessionNotificationMode.GroupOnly,
|
||||||
|
_ => throw new ArgumentOutOfRangeException(nameof(value), value, "Unknown notification mode.")
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
using System.Collections.Frozen;
|
||||||
|
|
||||||
namespace GmRelay.Shared.Domain;
|
namespace GmRelay.Shared.Domain;
|
||||||
|
|
||||||
public static class SessionStatus
|
public static class SessionStatus
|
||||||
@@ -6,4 +8,13 @@ public static class SessionStatus
|
|||||||
public const string ConfirmationSent = "ConfirmationSent";
|
public const string ConfirmationSent = "ConfirmationSent";
|
||||||
public const string Confirmed = "Confirmed";
|
public const string Confirmed = "Confirmed";
|
||||||
public const string Cancelled = "Cancelled";
|
public const string Cancelled = "Cancelled";
|
||||||
|
|
||||||
|
public static IReadOnlySet<string> All { get; } =
|
||||||
|
new[] { Planned, ConfirmationSent, Confirmed, Cancelled }
|
||||||
|
.ToFrozenSet(StringComparer.Ordinal);
|
||||||
|
|
||||||
|
public static bool IsKnown(string status) => All.Contains(status);
|
||||||
|
|
||||||
|
public static bool IsCancelled(string status) =>
|
||||||
|
string.Equals(status, Cancelled, StringComparison.Ordinal);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,8 +7,4 @@
|
|||||||
<LangVersion>preview</LangVersion>
|
<LangVersion>preview</LangVersion>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="Telegram.Bot" Version="22.9.5.3" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@@ -0,0 +1,13 @@
|
|||||||
|
namespace GmRelay.Shared.Rendering;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Заглушка для Discord-рендерера.
|
||||||
|
/// Реальная реализация будет добавлена в проект GmRelay.DiscordBot (issue #26).
|
||||||
|
/// </summary>
|
||||||
|
public static class DiscordSessionBatchRenderer
|
||||||
|
{
|
||||||
|
public static object Render(SessionBatchViewModel view)
|
||||||
|
{
|
||||||
|
throw new NotImplementedException("Discord renderer will be implemented in issue #26.");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
namespace GmRelay.Shared.Rendering;
|
||||||
|
|
||||||
|
public sealed record SessionBatchDto(Guid SessionId, DateTime ScheduledAt, string Status, int? MaxPlayers, string JoinLink);
|
||||||
|
public sealed record ParticipantBatchDto(Guid SessionId, string DisplayName, string? TelegramUsername, string RegistrationStatus);
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
using GmRelay.Shared.Domain;
|
|
||||||
using Telegram.Bot.Types.ReplyMarkups;
|
|
||||||
|
|
||||||
namespace GmRelay.Shared.Rendering;
|
|
||||||
|
|
||||||
public sealed record SessionBatchDto(Guid SessionId, DateTime ScheduledAt, string Status);
|
|
||||||
public sealed record ParticipantBatchDto(Guid SessionId, string DisplayName, string? TelegramUsername);
|
|
||||||
|
|
||||||
public static class SessionBatchRenderer
|
|
||||||
{
|
|
||||||
public static (string Text, InlineKeyboardMarkup Markup) Render(
|
|
||||||
string title,
|
|
||||||
IReadOnlyList<SessionBatchDto> sessions,
|
|
||||||
IReadOnlyList<ParticipantBatchDto> participants)
|
|
||||||
{
|
|
||||||
var activeSessions = sessions.OrderBy(s => s.ScheduledAt).ToList();
|
|
||||||
|
|
||||||
var messageText = $"🎲 <b>Новые игры:</b> {System.Net.WebUtility.HtmlEncode(title)}\n\n" +
|
|
||||||
$"<b>Расписание:</b>\n\n";
|
|
||||||
|
|
||||||
var buttons = new List<InlineKeyboardButton[]>();
|
|
||||||
|
|
||||||
foreach (var session in activeSessions)
|
|
||||||
{
|
|
||||||
var sessionPlayers = participants.Where(p => p.SessionId == session.SessionId).ToList();
|
|
||||||
|
|
||||||
messageText += $"📅 <b>{session.ScheduledAt.FormatMoscow()}</b>\n";
|
|
||||||
messageText += $"👥 Игроки ({sessionPlayers.Count}):\n";
|
|
||||||
|
|
||||||
if (sessionPlayers.Count > 0)
|
|
||||||
{
|
|
||||||
messageText += string.Join("\n", sessionPlayers.Select(p => $" 👤 {(p.TelegramUsername != null ? "@" + p.TelegramUsername : p.DisplayName)}")) + "\n";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
messageText += " <i>Пока никто не записался</i>\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (session.Status == "Cancelled")
|
|
||||||
{
|
|
||||||
messageText += "❌ <i>Сессия отменена</i>\n\n";
|
|
||||||
}
|
|
||||||
else if (session.Status == "RecruitmentClosed")
|
|
||||||
{
|
|
||||||
messageText += "🔒 <i>Набор завершен</i>\n\n";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
messageText += "\n";
|
|
||||||
var dateTitle = session.ScheduledAt.FormatMoscowShort();
|
|
||||||
buttons.Add(new[]
|
|
||||||
{
|
|
||||||
InlineKeyboardButton.WithCallbackData($"✋ На {dateTitle}", $"join_session:{session.SessionId}"),
|
|
||||||
InlineKeyboardButton.WithCallbackData($"❌ Отменить {dateTitle} (ГМ)", $"cancel_session:{session.SessionId}"),
|
|
||||||
InlineKeyboardButton.WithCallbackData($"⏰ (ГМ)", $"reschedule_session:{session.SessionId}")
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return (messageText, new InlineKeyboardMarkup(buttons));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
namespace GmRelay.Shared.Rendering;
|
||||||
|
|
||||||
|
public static class SessionBatchViewBuilder
|
||||||
|
{
|
||||||
|
public static SessionBatchViewModel Build(
|
||||||
|
string title,
|
||||||
|
IReadOnlyList<SessionBatchDto> sessions,
|
||||||
|
IReadOnlyList<ParticipantBatchDto> participants)
|
||||||
|
{
|
||||||
|
var orderedSessions = sessions.OrderBy(s => s.ScheduledAt).ToList();
|
||||||
|
var sessionItems = new List<SessionViewItem>();
|
||||||
|
|
||||||
|
foreach (var session in orderedSessions)
|
||||||
|
{
|
||||||
|
var activePlayers = participants
|
||||||
|
.Where(p => p.SessionId == session.SessionId && p.RegistrationStatus == ParticipantRegistrationStatus.Active)
|
||||||
|
.Select(p => new PlayerViewItem(p.DisplayName, p.TelegramUsername, p.RegistrationStatus))
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
var waitlistedPlayers = participants
|
||||||
|
.Where(p => p.SessionId == session.SessionId && p.RegistrationStatus == ParticipantRegistrationStatus.Waitlisted)
|
||||||
|
.Select(p => new PlayerViewItem(p.DisplayName, p.TelegramUsername, p.RegistrationStatus))
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
var actions = new List<AvailableAction>();
|
||||||
|
if (!SessionStatus.IsCancelled(session.Status))
|
||||||
|
{
|
||||||
|
var dateTitle = session.ScheduledAt.FormatMoscowShort();
|
||||||
|
var joinLabel = GetJoinButtonText(session, activePlayers.Count, dateTitle);
|
||||||
|
actions.Add(new AvailableAction("join_session", joinLabel, session.SessionId));
|
||||||
|
actions.Add(new AvailableAction("leave_session", $"🚪 Выйти {dateTitle}", session.SessionId));
|
||||||
|
}
|
||||||
|
|
||||||
|
sessionItems.Add(new SessionViewItem(
|
||||||
|
session.SessionId,
|
||||||
|
session.ScheduledAt,
|
||||||
|
session.Status,
|
||||||
|
session.MaxPlayers,
|
||||||
|
session.JoinLink,
|
||||||
|
activePlayers.Count,
|
||||||
|
activePlayers,
|
||||||
|
waitlistedPlayers,
|
||||||
|
actions));
|
||||||
|
}
|
||||||
|
|
||||||
|
return new SessionBatchViewModel(title, sessionItems);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string GetJoinButtonText(SessionBatchDto session, int activePlayers, string dateTitle)
|
||||||
|
{
|
||||||
|
if (session.MaxPlayers.HasValue && activePlayers >= session.MaxPlayers.Value)
|
||||||
|
{
|
||||||
|
return $"⏳ В лист ожидания {dateTitle}";
|
||||||
|
}
|
||||||
|
|
||||||
|
return $"✋ На {dateTitle}";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// trigger pr
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
using GmRelay.Shared.Domain;
|
||||||
|
|
||||||
|
namespace GmRelay.Shared.Rendering;
|
||||||
|
|
||||||
|
public sealed record SessionBatchViewModel(
|
||||||
|
string Title,
|
||||||
|
IReadOnlyList<SessionViewItem> Sessions);
|
||||||
|
|
||||||
|
public sealed record SessionViewItem(
|
||||||
|
Guid SessionId,
|
||||||
|
DateTime ScheduledAt,
|
||||||
|
string Status,
|
||||||
|
int? MaxPlayers,
|
||||||
|
string JoinLink,
|
||||||
|
int ActivePlayerCount,
|
||||||
|
IReadOnlyList<PlayerViewItem> ActivePlayers,
|
||||||
|
IReadOnlyList<PlayerViewItem> WaitlistedPlayers,
|
||||||
|
IReadOnlyList<AvailableAction> AvailableActions);
|
||||||
|
|
||||||
|
public sealed record PlayerViewItem(
|
||||||
|
string DisplayName,
|
||||||
|
string? TelegramUsername,
|
||||||
|
string RegistrationStatus);
|
||||||
|
|
||||||
|
public sealed record AvailableAction(
|
||||||
|
string ActionKey,
|
||||||
|
string Label,
|
||||||
|
Guid SessionId);
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
{
|
||||||
|
"version": 1,
|
||||||
|
"dependencies": {
|
||||||
|
"net10.0": {
|
||||||
|
"SecurityCodeScan.VS2019": {
|
||||||
|
"type": "Direct",
|
||||||
|
"requested": "[5.6.7, )",
|
||||||
|
"resolved": "5.6.7",
|
||||||
|
"contentHash": "WIE9RJswdSc2j+rLz2gW6U+gMUjMHzY2j7C/CL8/R2olXNM/+twarfMnWqm+rZodDBvaYDApJyxM8mVYf9FGrQ=="
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,14 +1,19 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="ru">
|
||||||
|
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8" />
|
<meta charset="utf-8" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
|
<meta name="description" content="GM-Relay — панель управления для Мастеров Игры. Управляйте сессиями настольных ролевых игр через Telegram." />
|
||||||
|
<meta name="theme-color" content="#0a0e1a" />
|
||||||
<base href="/" />
|
<base href="/" />
|
||||||
<ResourcePreloader />
|
<ResourcePreloader />
|
||||||
<link rel="stylesheet" href="@Assets["lib/bootstrap/dist/css/bootstrap.min.css"]" />
|
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
||||||
|
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&display=swap" rel="stylesheet" />
|
||||||
<link rel="stylesheet" href="@Assets["app.css"]" />
|
<link rel="stylesheet" href="@Assets["app.css"]" />
|
||||||
<link rel="stylesheet" href="@Assets["GmRelay.Web.styles.css"]" />
|
<link rel="stylesheet" href="@Assets["GmRelay.Web.styles.css"]" />
|
||||||
|
<script src="https://telegram.org/js/telegram-web-app.js"></script>
|
||||||
<ImportMap />
|
<ImportMap />
|
||||||
<link rel="icon" type="image/png" href="favicon.png" />
|
<link rel="icon" type="image/png" href="favicon.png" />
|
||||||
<HeadOutlet @rendermode="InteractiveServer" />
|
<HeadOutlet @rendermode="InteractiveServer" />
|
||||||
@@ -18,6 +23,277 @@
|
|||||||
<Routes @rendermode="InteractiveServer" />
|
<Routes @rendermode="InteractiveServer" />
|
||||||
<ReconnectModal />
|
<ReconnectModal />
|
||||||
<script src="@Assets["_framework/blazor.web.js"]"></script>
|
<script src="@Assets["_framework/blazor.web.js"]"></script>
|
||||||
|
<script>
|
||||||
|
window.waitForTelegramMiniApp = async function (timeoutMs) {
|
||||||
|
var deadline = Date.now() + (timeoutMs || 3000);
|
||||||
|
|
||||||
|
while (Date.now() <= deadline) {
|
||||||
|
if (window.Telegram && window.Telegram.WebApp) {
|
||||||
|
return window.Telegram.WebApp;
|
||||||
|
}
|
||||||
|
|
||||||
|
await new Promise(function (resolve) {
|
||||||
|
setTimeout(resolve, 100);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
window.waitForTelegramMiniAppInitData = async function (timeoutMs) {
|
||||||
|
var deadline = Date.now() + (timeoutMs || 3000);
|
||||||
|
|
||||||
|
while (Date.now() <= deadline) {
|
||||||
|
if (window.Telegram && window.Telegram.WebApp && window.Telegram.WebApp.initData) {
|
||||||
|
return window.Telegram.WebApp;
|
||||||
|
}
|
||||||
|
|
||||||
|
await new Promise(function (resolve) {
|
||||||
|
setTimeout(resolve, 100);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
window.syncTelegramMiniAppViewport = function (webApp) {
|
||||||
|
if (!webApp) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var root = document.documentElement;
|
||||||
|
var safeArea = webApp.safeAreaInset || {};
|
||||||
|
var contentSafeArea = webApp.contentSafeAreaInset || {};
|
||||||
|
var setPx = function (name, value) {
|
||||||
|
root.style.setProperty(name, Math.max(0, Number(value) || 0) + 'px');
|
||||||
|
};
|
||||||
|
|
||||||
|
setPx('--gm-tg-safe-top', safeArea.top);
|
||||||
|
setPx('--gm-tg-safe-right', safeArea.right);
|
||||||
|
setPx('--gm-tg-safe-bottom', safeArea.bottom);
|
||||||
|
setPx('--gm-tg-safe-left', safeArea.left);
|
||||||
|
setPx('--gm-tg-content-safe-top', contentSafeArea.top);
|
||||||
|
setPx('--gm-tg-content-safe-right', contentSafeArea.right);
|
||||||
|
setPx('--gm-tg-content-safe-bottom', contentSafeArea.bottom);
|
||||||
|
setPx('--gm-tg-content-safe-left', contentSafeArea.left);
|
||||||
|
|
||||||
|
if (webApp.viewportHeight) {
|
||||||
|
root.style.setProperty('--gm-tg-viewport-height', webApp.viewportHeight + 'px');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
window.prepareTelegramMiniApp = function (webApp) {
|
||||||
|
if (!webApp) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
document.body.classList.add('telegram-mini-app');
|
||||||
|
window.syncTelegramMiniAppViewport(webApp);
|
||||||
|
|
||||||
|
try {
|
||||||
|
webApp.ready();
|
||||||
|
} catch (error) {
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
webApp.expand();
|
||||||
|
} catch (error) {
|
||||||
|
}
|
||||||
|
|
||||||
|
if (webApp.onEvent && !window.gmRelayTelegramMiniAppViewportEventsRegistered) {
|
||||||
|
window.gmRelayTelegramMiniAppViewportEventsRegistered = true;
|
||||||
|
webApp.onEvent('safeAreaChanged', function () {
|
||||||
|
window.syncTelegramMiniAppViewport(webApp);
|
||||||
|
});
|
||||||
|
webApp.onEvent('contentSafeAreaChanged', function () {
|
||||||
|
window.syncTelegramMiniAppViewport(webApp);
|
||||||
|
});
|
||||||
|
webApp.onEvent('viewportChanged', function () {
|
||||||
|
window.syncTelegramMiniAppViewport(webApp);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
(async function () {
|
||||||
|
var webApp = await window.waitForTelegramMiniApp(1000);
|
||||||
|
window.prepareTelegramMiniApp(webApp);
|
||||||
|
})();
|
||||||
|
|
||||||
|
window.loadTelegramWidget = function (botUsername, authUrl) {
|
||||||
|
var container = document.getElementById('telegram-login-container');
|
||||||
|
if (!container) return;
|
||||||
|
container.innerHTML = '';
|
||||||
|
window.gmRelayTelegramLoginAuthUrl = authUrl || '/auth/telegram-login';
|
||||||
|
var script = document.createElement('script');
|
||||||
|
script.async = true;
|
||||||
|
script.src = 'https://telegram.org/js/telegram-widget.js?22';
|
||||||
|
script.setAttribute('data-telegram-login', botUsername);
|
||||||
|
script.setAttribute('data-size', 'large');
|
||||||
|
script.setAttribute('data-onauth', 'window.handleTelegramLogin(user)');
|
||||||
|
script.setAttribute('data-request-access', 'write');
|
||||||
|
container.appendChild(script);
|
||||||
|
};
|
||||||
|
|
||||||
|
window.handleTelegramLogin = async function (user) {
|
||||||
|
if (!user) {
|
||||||
|
window.location.href = '/login?error=auth_failed';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
var response = await fetch(window.gmRelayTelegramLoginAuthUrl || '/auth/telegram-login', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
credentials: 'same-origin',
|
||||||
|
cache: 'no-store',
|
||||||
|
body: JSON.stringify(user)
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
window.location.href = '/login?error=auth_failed';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var payload = await response.json();
|
||||||
|
window.location.href = payload.redirectUrl || '/';
|
||||||
|
} catch (error) {
|
||||||
|
window.location.href = '/login?error=auth_failed';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
window.watchTelegramMiniAppLogin = function (statusUrl, redirectUrl, reloadOnReturn) {
|
||||||
|
window.gmRelayMiniAppLoginReloadOnReturn =
|
||||||
|
window.gmRelayMiniAppLoginReloadOnReturn || reloadOnReturn === true;
|
||||||
|
|
||||||
|
if (window.gmRelayMiniAppLoginWatcher) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
window.gmRelayMiniAppLoginLeftPage = false;
|
||||||
|
|
||||||
|
var stopWatching = function () {
|
||||||
|
if (window.gmRelayMiniAppLoginWatcher) {
|
||||||
|
window.clearInterval(window.gmRelayMiniAppLoginWatcher);
|
||||||
|
window.gmRelayMiniAppLoginWatcher = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var reloadAfterExternalLogin = function () {
|
||||||
|
if (!window.gmRelayMiniAppLoginReloadOnReturn || !window.gmRelayMiniAppLoginLeftPage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
window.gmRelayMiniAppLoginLeftPage = false;
|
||||||
|
|
||||||
|
try {
|
||||||
|
var refreshKey = 'gmrelay-miniapp-login-refresh:' + window.location.pathname;
|
||||||
|
if (window.sessionStorage && window.sessionStorage.getItem(refreshKey) === '1') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (window.sessionStorage) {
|
||||||
|
window.sessionStorage.setItem(refreshKey, '1');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
}
|
||||||
|
|
||||||
|
window.location.reload();
|
||||||
|
};
|
||||||
|
|
||||||
|
var allowNextExternalLoginReload = function () {
|
||||||
|
window.gmRelayMiniAppLoginLeftPage = true;
|
||||||
|
|
||||||
|
try {
|
||||||
|
var refreshKey = 'gmrelay-miniapp-login-refresh:' + window.location.pathname;
|
||||||
|
if (window.sessionStorage) {
|
||||||
|
window.sessionStorage.removeItem(refreshKey);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var checkLogin = async function (reloadWhenUnauthenticated) {
|
||||||
|
try {
|
||||||
|
var response = await fetch(statusUrl, {
|
||||||
|
credentials: 'same-origin',
|
||||||
|
cache: 'no-store'
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var payload = await response.json();
|
||||||
|
if (payload.authenticated) {
|
||||||
|
stopWatching();
|
||||||
|
window.location.href = redirectUrl || '/';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (reloadWhenUnauthenticated) {
|
||||||
|
reloadAfterExternalLogin();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
window.gmRelayMiniAppLoginWatcher = window.setInterval(checkLogin, 1000);
|
||||||
|
window.addEventListener('blur', function () {
|
||||||
|
allowNextExternalLoginReload();
|
||||||
|
});
|
||||||
|
window.addEventListener('focus', function () {
|
||||||
|
void checkLogin(true);
|
||||||
|
});
|
||||||
|
document.addEventListener('visibilitychange', function () {
|
||||||
|
if (document.hidden) {
|
||||||
|
allowNextExternalLoginReload();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
void checkLogin(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
void checkLogin(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
window.authenticateTelegramMiniApp = async function (authUrl, redirectUrl) {
|
||||||
|
var webApp = await window.waitForTelegramMiniApp(3000);
|
||||||
|
if (!webApp) {
|
||||||
|
return { authenticated: false, reason: 'telegram-webapp-missing' };
|
||||||
|
}
|
||||||
|
|
||||||
|
window.prepareTelegramMiniApp(webApp);
|
||||||
|
|
||||||
|
if (!webApp.initData) {
|
||||||
|
return { authenticated: false, reason: 'telegram-init-data-empty' };
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
var response = await fetch(authUrl, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
credentials: 'same-origin',
|
||||||
|
cache: 'no-store',
|
||||||
|
body: JSON.stringify({ initData: webApp.initData })
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
return {
|
||||||
|
authenticated: false,
|
||||||
|
reason: 'telegram-auth-failed',
|
||||||
|
status: response.status
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
var payload = await response.json();
|
||||||
|
window.location.href = payload.redirectUrl || redirectUrl || '/';
|
||||||
|
return { authenticated: true, redirectUrl: payload.redirectUrl || redirectUrl || '/' };
|
||||||
|
} catch (error) {
|
||||||
|
return { authenticated: false, reason: 'telegram-auth-failed' };
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
</body>
|
</body>
|
||||||
|
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
@@ -1,19 +1,15 @@
|
|||||||
@inherits LayoutComponentBase
|
@inherits LayoutComponentBase
|
||||||
|
|
||||||
<div class="page">
|
<div class="page">
|
||||||
<div class="sidebar">
|
<aside class="sidebar">
|
||||||
<NavMenu />
|
<NavMenu />
|
||||||
</div>
|
</aside>
|
||||||
|
|
||||||
<main>
|
<div class="main-area">
|
||||||
<div class="top-row px-4">
|
<article class="content">
|
||||||
<a href="https://github.com/Toutsu/GmRelayBot" target="_blank">О проекте</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<article class="content px-4">
|
|
||||||
@Body
|
@Body
|
||||||
</article>
|
</article>
|
||||||
</main>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="blazor-error-ui" data-nosnippet>
|
<div id="blazor-error-ui" data-nosnippet>
|
||||||
|
|||||||
@@ -1,86 +1,39 @@
|
|||||||
.page {
|
.page {
|
||||||
position: relative;
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
min-height: 100vh;
|
||||||
}
|
|
||||||
|
|
||||||
main {
|
|
||||||
flex: 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar {
|
.sidebar {
|
||||||
background-image: linear-gradient(180deg, rgb(5, 39, 103) 0%, #3a0647 70%);
|
width: var(--sidebar-width);
|
||||||
}
|
background: linear-gradient(180deg, #0f1629 0%, #1a0a2e 100%);
|
||||||
|
border-right: 1px solid var(--border-color);
|
||||||
.top-row {
|
position: fixed;
|
||||||
background-color: #f7f7f7;
|
top: 0;
|
||||||
border-bottom: 1px solid #d6d5d5;
|
left: 0;
|
||||||
justify-content: flex-end;
|
height: 100vh;
|
||||||
height: 3.5rem;
|
z-index: 100;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
flex-direction: column;
|
||||||
|
transition: transform var(--transition-smooth);
|
||||||
}
|
}
|
||||||
|
|
||||||
.top-row ::deep a, .top-row ::deep .btn-link {
|
.main-area {
|
||||||
white-space: nowrap;
|
flex: 1;
|
||||||
margin-left: 1.5rem;
|
margin-left: var(--sidebar-width);
|
||||||
text-decoration: none;
|
min-height: 100vh;
|
||||||
}
|
|
||||||
|
|
||||||
.top-row ::deep a:hover, .top-row ::deep .btn-link:hover {
|
|
||||||
text-decoration: underline;
|
|
||||||
}
|
|
||||||
|
|
||||||
.top-row ::deep a:first-child {
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 640.98px) {
|
|
||||||
.top-row {
|
|
||||||
justify-content: space-between;
|
|
||||||
}
|
|
||||||
|
|
||||||
.top-row ::deep a, .top-row ::deep .btn-link {
|
|
||||||
margin-left: 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 641px) {
|
.content {
|
||||||
.page {
|
padding: 1.5rem 2rem;
|
||||||
flex-direction: row;
|
max-width: 100%;
|
||||||
}
|
|
||||||
|
|
||||||
.sidebar {
|
|
||||||
width: 250px;
|
|
||||||
height: 100vh;
|
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.top-row {
|
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
z-index: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.top-row.auth ::deep a:first-child {
|
|
||||||
flex: 1;
|
|
||||||
text-align: right;
|
|
||||||
width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.top-row, article {
|
|
||||||
padding-left: 2rem !important;
|
|
||||||
padding-right: 1.5rem !important;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* === Error UI === */
|
||||||
#blazor-error-ui {
|
#blazor-error-ui {
|
||||||
color-scheme: light only;
|
background: var(--bg-secondary);
|
||||||
background: lightyellow;
|
border-top: 1px solid var(--border-color);
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2);
|
box-shadow: 0 -4px 16px rgba(0, 0, 0, 0.3);
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
display: none;
|
display: none;
|
||||||
left: 0;
|
left: 0;
|
||||||
@@ -88,11 +41,49 @@ main {
|
|||||||
position: fixed;
|
position: fixed;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
z-index: 1000;
|
z-index: 1000;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
font-size: 0.875rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#blazor-error-ui .reload {
|
||||||
|
color: var(--accent-secondary);
|
||||||
|
margin-left: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
#blazor-error-ui .dismiss {
|
#blazor-error-ui .dismiss {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
right: 0.75rem;
|
right: 0.75rem;
|
||||||
top: 0.5rem;
|
top: 0.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* === Mobile Responsive === */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.sidebar {
|
||||||
|
transform: none;
|
||||||
|
width: 100%;
|
||||||
|
height: auto;
|
||||||
|
min-height: 0;
|
||||||
|
position: sticky;
|
||||||
|
border-right: none;
|
||||||
|
border-bottom: 1px solid var(--border-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.page {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.main-area {
|
||||||
|
margin-left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
padding: 1rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 769px) and (max-width: 1024px) {
|
||||||
|
.content {
|
||||||
|
padding: 1.25rem 1.5rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,41 +1,82 @@
|
|||||||
<div class="top-row ps-3 navbar navbar-dark">
|
@inject NavigationManager Navigation
|
||||||
<div class="container-fluid">
|
|
||||||
<a class="navbar-brand" href="">GM-Relay Web</a>
|
<div class="nav-header">
|
||||||
</div>
|
<a class="nav-brand" href="">
|
||||||
|
<img src="logo.png" alt="GM-Relay" class="nav-brand-icon" />
|
||||||
|
<span class="nav-brand-text">GM-Relay</span>
|
||||||
|
</a>
|
||||||
|
<button class="nav-toggle" @onclick="ToggleMenu" aria-label="Переключить меню">
|
||||||
|
<svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
|
<line x1="3" y1="6" x2="21" y2="6"/>
|
||||||
|
<line x1="3" y1="12" x2="21" y2="12"/>
|
||||||
|
<line x1="3" y1="18" x2="21" y2="18"/>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<input type="checkbox" title="Навигационное меню" class="navbar-toggler" />
|
<nav class="nav-body @(isOpen ? "open" : "")">
|
||||||
|
<AuthorizeView>
|
||||||
|
<Authorized>
|
||||||
|
<div class="nav-section">
|
||||||
|
<NavLink class="nav-item" href="" Match="NavLinkMatch.All" @onclick="CloseMenu">
|
||||||
|
<svg class="nav-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<path d="M3 9l9-7 9 7v11a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z"/>
|
||||||
|
<polyline points="9 22 9 12 15 12 15 22"/>
|
||||||
|
</svg>
|
||||||
|
Главная страница
|
||||||
|
</NavLink>
|
||||||
|
<NavLink class="nav-item" href="templates" @onclick="CloseMenu">
|
||||||
|
<svg class="nav-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<rect x="3" y="4" width="18" height="16" rx="2"/>
|
||||||
|
<path d="M7 8h10"/>
|
||||||
|
<path d="M7 12h6"/>
|
||||||
|
<path d="M7 16h8"/>
|
||||||
|
</svg>
|
||||||
|
Шаблоны
|
||||||
|
</NavLink>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="nav-scrollable" onclick="document.querySelector('.navbar-toggler').click()">
|
<div class="nav-footer">
|
||||||
<nav class="nav flex-column">
|
<div class="nav-user">
|
||||||
<AuthorizeView>
|
<div class="nav-user-avatar">
|
||||||
<Authorized>
|
@(context.User.Identity?.Name?.Substring(0, 1).ToUpper() ?? "?")
|
||||||
<div class="nav-item px-3">
|
|
||||||
<NavLink class="nav-link" href="" Match="NavLinkMatch.All">
|
|
||||||
<span class="bi bi-house-door-fill-nav-menu" aria-hidden="true"></span> Панель управления
|
|
||||||
</NavLink>
|
|
||||||
</div>
|
|
||||||
<div class="nav-item px-3 mt-auto">
|
|
||||||
<div class="nav-link text-light">
|
|
||||||
<span class="bi bi-person-fill" aria-hidden="true"></span> @context.User.Identity?.Name
|
|
||||||
</div>
|
</div>
|
||||||
|
<span class="nav-user-name">@context.User.Identity?.Name</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="nav-item px-3">
|
|
||||||
<form action="/auth/logout" method="post">
|
<form action="/auth/logout" method="post">
|
||||||
<AntiforgeryToken />
|
<AntiforgeryToken />
|
||||||
<button type="submit" class="nav-link btn btn-link text-light text-start w-100 p-0 shadow-none border-0">
|
<button type="submit" class="nav-logout-btn">
|
||||||
<span class="bi bi-box-arrow-right" aria-hidden="true"></span> Выйти
|
<svg class="nav-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
</button>
|
<path d="M9 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1-2 2h4"/>
|
||||||
</form>
|
<polyline points="16 17 21 12 16 7"/>
|
||||||
</div>
|
<line x1="21" y1="12" x2="9" y2="12"/>
|
||||||
</Authorized>
|
</svg>
|
||||||
<NotAuthorized>
|
Выход
|
||||||
<div class="nav-item px-3">
|
</button>
|
||||||
<NavLink class="nav-link" href="login">
|
</form>
|
||||||
<span class="bi bi-person-badge-nav-menu" aria-hidden="true"></span> Войти
|
|
||||||
</NavLink>
|
<div class="nav-version">v2.0.0</div>
|
||||||
</div>
|
</div>
|
||||||
</NotAuthorized>
|
</Authorized>
|
||||||
</AuthorizeView>
|
<NotAuthorized>
|
||||||
</nav>
|
<div class="nav-section">
|
||||||
</div>
|
<NavLink class="nav-item" href="login" @onclick="CloseMenu">
|
||||||
|
<svg class="nav-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<path d="M15 3h4a2 2 0 0 1 2 2v14a2 2 0 0 1-2 2h-4"/>
|
||||||
|
<polyline points="10 17 15 12 10 7"/>
|
||||||
|
<line x1="15" y1="12" x2="3" y2="12"/>
|
||||||
|
</svg>
|
||||||
|
Вход
|
||||||
|
</NavLink>
|
||||||
|
</div>
|
||||||
|
</NotAuthorized>
|
||||||
|
</AuthorizeView>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
private bool isOpen;
|
||||||
|
|
||||||
|
private void ToggleMenu() => isOpen = !isOpen;
|
||||||
|
private void CloseMenu() => isOpen = false;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,105 +1,210 @@
|
|||||||
.navbar-toggler {
|
/* === Nav Header === */
|
||||||
appearance: none;
|
.nav-header {
|
||||||
cursor: pointer;
|
display: flex;
|
||||||
width: 3.5rem;
|
align-items: center;
|
||||||
height: 2.5rem;
|
justify-content: space-between;
|
||||||
color: white;
|
padding: 1.25rem 1rem;
|
||||||
position: absolute;
|
border-bottom: 1px solid var(--border-color);
|
||||||
top: 0.5rem;
|
|
||||||
right: 1rem;
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
||||||
background: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e") no-repeat center/1.75rem rgba(255, 255, 255, 0.1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar-toggler:checked {
|
.nav-brand {
|
||||||
background-color: rgba(255, 255, 255, 0.5);
|
display: flex;
|
||||||
|
align-items: baseline;
|
||||||
|
gap: 0.75rem;
|
||||||
|
text-decoration: none;
|
||||||
|
color: var(--text-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
.top-row {
|
.nav-brand-icon {
|
||||||
min-height: 3.5rem;
|
width: 1.5rem;
|
||||||
background-color: rgba(0,0,0,0.4);
|
height: 1.5rem;
|
||||||
}
|
object-fit: contain;
|
||||||
|
|
||||||
.navbar-brand {
|
|
||||||
font-size: 1.1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.bi {
|
|
||||||
display: inline-block;
|
|
||||||
position: relative;
|
|
||||||
width: 1.25rem;
|
|
||||||
height: 1.25rem;
|
|
||||||
margin-right: 0.75rem;
|
|
||||||
top: -1px;
|
|
||||||
background-size: cover;
|
|
||||||
}
|
|
||||||
|
|
||||||
.bi-house-door-fill-nav-menu {
|
|
||||||
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-house-door-fill' viewBox='0 0 16 16'%3E%3Cpath d='M6.5 14.5v-3.505c0-.245.25-.495.5-.495h2c.25 0 .5.25.5.5v3.5a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5v-7a.5.5 0 0 0-.146-.354L13 5.793V2.5a.5.5 0 0 0-.5-.5h-1a.5.5 0 0 0-.5.5v1.293L8.354 1.146a.5.5 0 0 0-.708 0l-6 6A.5.5 0 0 0 1.5 7.5v7a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5Z'/%3E%3C/svg%3E");
|
|
||||||
}
|
|
||||||
|
|
||||||
.bi-plus-square-fill-nav-menu {
|
|
||||||
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-plus-square-fill' viewBox='0 0 16 16'%3E%3Cpath d='M2 0a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V2a2 2 0 0 0-2-2H2zm6.5 4.5v3h3a.5.5 0 0 1 0 1h-3v3a.5.5 0 0 1-1 0v-3h-3a.5.5 0 0 1 0-1h3v-3a.5.5 0 0 1 1 0z'/%3E%3C/svg%3E");
|
|
||||||
}
|
|
||||||
|
|
||||||
.bi-list-nested-nav-menu {
|
|
||||||
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-list-nested' viewBox='0 0 16 16'%3E%3Cpath fill-rule='evenodd' d='M4.5 11.5A.5.5 0 0 1 5 11h10a.5.5 0 0 1 0 1H5a.5.5 0 0 1-.5-.5zm-2-4A.5.5 0 0 1 3 7h10a.5.5 0 0 1 0 1H3a.5.5 0 0 1-.5-.5zm-2-4A.5.5 0 0 1 1 3h10a.5.5 0 0 1 0 1H1a.5.5 0 0 1-.5-.5z'/%3E%3C/svg%3E");
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item {
|
|
||||||
font-size: 0.9rem;
|
|
||||||
padding-bottom: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item:first-of-type {
|
|
||||||
padding-top: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item:last-of-type {
|
|
||||||
padding-bottom: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep .nav-link {
|
|
||||||
color: #d7d7d7;
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
border-radius: 4px;
|
|
||||||
height: 3rem;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
line-height: 3rem;
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep a.active {
|
|
||||||
background-color: rgba(255,255,255,0.37);
|
|
||||||
color: white;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep .nav-link:hover {
|
|
||||||
background-color: rgba(255,255,255,0.1);
|
|
||||||
color: white;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-scrollable {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.navbar-toggler:checked ~ .nav-scrollable {
|
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 641px) {
|
.nav-brand-text {
|
||||||
.navbar-toggler {
|
font-size: 1.125rem;
|
||||||
|
font-weight: 700;
|
||||||
|
background: var(--accent-gradient);
|
||||||
|
-webkit-background-clip: text;
|
||||||
|
-webkit-text-fill-color: transparent;
|
||||||
|
background-clip: text;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-toggle {
|
||||||
|
display: none;
|
||||||
|
background: none;
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
color: var(--text-secondary);
|
||||||
|
padding: 0.375rem;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all var(--transition-fast);
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-toggle:hover {
|
||||||
|
background: var(--bg-surface);
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* === Nav Body === */
|
||||||
|
.nav-body {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
padding: 0.75rem 0;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-section {
|
||||||
|
padding: 0 0.75rem;
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* === Nav Items === */
|
||||||
|
.nav-section ::deep .nav-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.75rem;
|
||||||
|
width: 100%;
|
||||||
|
padding: 0.625rem 0.875rem;
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
color: var(--text-secondary);
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: 0.875rem;
|
||||||
|
font-weight: 500;
|
||||||
|
transition: all var(--transition-normal);
|
||||||
|
white-space: nowrap;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-section ::deep .nav-item:hover {
|
||||||
|
background: rgba(255, 255, 255, 0.06);
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-section ::deep .nav-item.active {
|
||||||
|
background: rgba(124, 58, 237, 0.15);
|
||||||
|
color: var(--accent-primary);
|
||||||
|
border: 1px solid rgba(124, 58, 237, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-icon {
|
||||||
|
width: 1.125rem;
|
||||||
|
height: 1.125rem;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* === Nav Footer === */
|
||||||
|
.nav-footer {
|
||||||
|
padding: 0.75rem;
|
||||||
|
border-top: 1px solid var(--border-color);
|
||||||
|
margin-top: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-user {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.625rem;
|
||||||
|
padding: 0.5rem 0.5rem 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-user-avatar {
|
||||||
|
width: 2rem;
|
||||||
|
height: 2rem;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: var(--accent-gradient);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 0.8125rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: white;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-user-name {
|
||||||
|
font-size: 0.8125rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: var(--text-primary);
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-logout-btn {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.75rem;
|
||||||
|
width: 100%;
|
||||||
|
padding: 0.5rem 0.875rem;
|
||||||
|
background: none;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-family: 'Inter', sans-serif;
|
||||||
|
font-size: 0.8125rem;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all var(--transition-normal);
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-logout-btn:hover {
|
||||||
|
background: var(--status-danger-bg);
|
||||||
|
color: var(--status-danger);
|
||||||
|
border-color: rgba(239, 68, 68, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-version {
|
||||||
|
text-align: center;
|
||||||
|
font-size: 0.6875rem;
|
||||||
|
color: var(--text-muted);
|
||||||
|
padding-top: 0.5rem;
|
||||||
|
opacity: 0.6;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* === Mobile === */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.nav-toggle {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
position: fixed;
|
||||||
|
top: 0.75rem;
|
||||||
|
left: 0.75rem;
|
||||||
|
z-index: 200;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
width: 2.5rem;
|
||||||
|
height: 2.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-body {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-scrollable {
|
.nav-body.open {
|
||||||
/* Never collapse the sidebar for wide screens */
|
display: flex;
|
||||||
display: block;
|
position: fixed;
|
||||||
|
inset: 0;
|
||||||
|
z-index: 200;
|
||||||
|
background: linear-gradient(180deg, #0f1629 0%, #1a0a2e 100%);
|
||||||
|
padding-top: 4.5rem;
|
||||||
|
padding-left: 1rem;
|
||||||
|
padding-right: 1rem;
|
||||||
|
padding-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
/* Allow sidebar to scroll for tall menus */
|
.nav-header {
|
||||||
height: calc(100vh - 3.5rem);
|
padding-left: 3.75rem;
|
||||||
overflow-y: auto;
|
padding-right: 0.75rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 769px) {
|
||||||
|
.nav-body {
|
||||||
|
height: calc(100vh - 4.5rem);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,27 @@
|
|||||||
|
@page "/access-denied"
|
||||||
|
|
||||||
|
<PageTitle>Доступ запрещен — GM-Relay</PageTitle>
|
||||||
|
|
||||||
|
<div class="page-container">
|
||||||
|
<div class="glass-card" style="max-width: 640px;">
|
||||||
|
<div class="empty-state">
|
||||||
|
<div class="empty-state-icon">⛔</div>
|
||||||
|
<div class="empty-state-title">Доступ запрещен</div>
|
||||||
|
<p class="empty-state-text">Эта группа или сессия недоступна для вашей учётной записи.</p>
|
||||||
|
<a href="/" class="btn-gm btn-gm-primary">← На главную</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
[CascadingParameter]
|
||||||
|
private HttpContext? HttpContext { get; set; }
|
||||||
|
|
||||||
|
protected override void OnInitialized()
|
||||||
|
{
|
||||||
|
if (HttpContext is not null && !HttpContext.Response.HasStarted)
|
||||||
|
{
|
||||||
|
HttpContext.Response.StatusCode = StatusCodes.Status403Forbidden;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,402 @@
|
|||||||
|
@page "/templates"
|
||||||
|
@using GmRelay.Web.Services
|
||||||
|
@using GmRelay.Shared.Domain
|
||||||
|
@using Microsoft.AspNetCore.Authorization
|
||||||
|
@using Microsoft.AspNetCore.Components.Authorization
|
||||||
|
@attribute [Authorize]
|
||||||
|
@inject AuthorizedSessionService SessionService
|
||||||
|
@inject AuthenticationStateProvider AuthStateProvider
|
||||||
|
@inject NavigationManager Navigation
|
||||||
|
|
||||||
|
<PageTitle>Шаблоны кампаний — GM-Relay</PageTitle>
|
||||||
|
|
||||||
|
<div class="page-container">
|
||||||
|
<ul class="gm-breadcrumb animate-fade-in">
|
||||||
|
<li><a href="/">Главная</a></li>
|
||||||
|
<li class="active">Шаблоны кампаний</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<div class="page-header animate-fade-in">
|
||||||
|
<h2>📋 Шаблоны кампаний</h2>
|
||||||
|
<p>Сохраняйте типовые кампании один раз и применяйте их на странице нужной группы.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (!string.IsNullOrEmpty(errorMessage))
|
||||||
|
{
|
||||||
|
<div class="gm-alert gm-alert-danger" style="margin-bottom: 1rem;">
|
||||||
|
⚠️ @errorMessage
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (!string.IsNullOrEmpty(successMessage))
|
||||||
|
{
|
||||||
|
<div class="gm-alert gm-alert-success" style="margin-bottom: 1rem;">
|
||||||
|
✅ @successMessage
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (groups is null)
|
||||||
|
{
|
||||||
|
<div class="glass-card" style="padding: 2rem;">
|
||||||
|
<div class="skeleton skeleton-text" style="width: 80%; margin-bottom: 1rem;"></div>
|
||||||
|
<div class="skeleton skeleton-text" style="width: 60%; margin-bottom: 0.75rem;"></div>
|
||||||
|
<div class="skeleton skeleton-text" style="width: 70%;"></div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else if (groups.Count == 0)
|
||||||
|
{
|
||||||
|
<div class="glass-card">
|
||||||
|
<div class="empty-state">
|
||||||
|
<div class="empty-state-icon">🤖</div>
|
||||||
|
<div class="empty-state-title">Нет доступных групп</div>
|
||||||
|
<p class="empty-state-text">Добавьте бота GM-Relay в группу Telegram, чтобы создать первый шаблон кампании.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="glass-card campaign-template-panel animate-slide-up">
|
||||||
|
<div class="batch-bulk-header">
|
||||||
|
<div>
|
||||||
|
<h3>Группа для шаблонов</h3>
|
||||||
|
<p>@(SelectedGroup?.Name ?? "Выберите группу")</p>
|
||||||
|
</div>
|
||||||
|
@if (SelectedGroup is not null)
|
||||||
|
{
|
||||||
|
<span class="status-badge @(SelectedGroup.ManagerRole == GroupManagerRoleExtensions.OwnerValue ? "status-success" : "status-info")">
|
||||||
|
@FormatRole(SelectedGroup.ManagerRole)
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="template-group-selector">
|
||||||
|
<select value="@selectedGroupId" @onchange="OnSelectedGroupChanged" class="gm-form-control">
|
||||||
|
@foreach (var group in groups)
|
||||||
|
{
|
||||||
|
<option value="@group.Id">@group.Name</option>
|
||||||
|
}
|
||||||
|
</select>
|
||||||
|
@if (SelectedGroup is not null)
|
||||||
|
{
|
||||||
|
<a href="/group/@SelectedGroup.Id" class="btn-gm btn-gm-outline">Открыть группу →</a>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="glass-card campaign-template-panel animate-slide-up">
|
||||||
|
<div class="batch-bulk-header">
|
||||||
|
<div>
|
||||||
|
<h3>Новый шаблон</h3>
|
||||||
|
<p>Эти параметры будут использоваться при запуске batch из группы.</p>
|
||||||
|
</div>
|
||||||
|
<span class="status-badge status-info">Template</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<EditForm Model="@templateModel" OnValidSubmit="CreateCampaignTemplate">
|
||||||
|
<div class="campaign-template-fields">
|
||||||
|
<div class="gm-form-group">
|
||||||
|
<label class="gm-form-label">Название шаблона</label>
|
||||||
|
<InputText @bind-Value="templateModel.Name" class="gm-form-control" />
|
||||||
|
</div>
|
||||||
|
<div class="gm-form-group">
|
||||||
|
<label class="gm-form-label">Название кампании</label>
|
||||||
|
<InputText @bind-Value="templateModel.Title" class="gm-form-control" />
|
||||||
|
</div>
|
||||||
|
<div class="gm-form-group">
|
||||||
|
<label class="gm-form-label">Ссылка</label>
|
||||||
|
<InputText @bind-Value="templateModel.JoinLink" class="gm-form-control" />
|
||||||
|
</div>
|
||||||
|
<div class="gm-form-group">
|
||||||
|
<label class="gm-form-label">Игр</label>
|
||||||
|
<InputNumber @bind-Value="templateModel.SessionCount" class="gm-form-control" min="1" max="52" />
|
||||||
|
</div>
|
||||||
|
<div class="gm-form-group">
|
||||||
|
<label class="gm-form-label">Интервал, дней</label>
|
||||||
|
<InputNumber @bind-Value="templateModel.IntervalDays" class="gm-form-control" min="1" max="365" />
|
||||||
|
</div>
|
||||||
|
<div class="gm-form-group">
|
||||||
|
<label class="gm-form-label">Мест</label>
|
||||||
|
<InputNumber @bind-Value="templateModel.MaxPlayers" class="gm-form-control" min="1" />
|
||||||
|
</div>
|
||||||
|
<div class="gm-form-group">
|
||||||
|
<label class="gm-form-label">Уведомления</label>
|
||||||
|
<select @bind="templateModel.NotificationMode" class="gm-form-control">
|
||||||
|
<option value="@SessionNotificationModeExtensions.GroupAndDirectValue">В группе и в личку</option>
|
||||||
|
<option value="@SessionNotificationModeExtensions.GroupOnlyValue">Только в группе</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button type="submit" class="btn-gm btn-gm-primary" disabled="@isCreatingTemplate">
|
||||||
|
@(isCreatingTemplate ? "⏳ Сохраняем..." : "💾 Сохранить шаблон")
|
||||||
|
</button>
|
||||||
|
</EditForm>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="glass-card campaign-template-panel animate-slide-up">
|
||||||
|
<div class="batch-bulk-header">
|
||||||
|
<div>
|
||||||
|
<h3>Сохранённые шаблоны</h3>
|
||||||
|
<p>@campaignTemplateModels.Count для выбранной группы</p>
|
||||||
|
</div>
|
||||||
|
<span class="status-badge status-info">@campaignTemplateModels.Count</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (campaignTemplates is null)
|
||||||
|
{
|
||||||
|
<div class="skeleton skeleton-text" style="width: 70%; margin-bottom: 0.75rem;"></div>
|
||||||
|
<div class="skeleton skeleton-text" style="width: 55%;"></div>
|
||||||
|
}
|
||||||
|
else if (campaignTemplateModels.Count == 0)
|
||||||
|
{
|
||||||
|
<div class="empty-state empty-state-compact">
|
||||||
|
<div class="empty-state-title">Шаблонов пока нет</div>
|
||||||
|
<p class="empty-state-text">Создайте первый шаблон для выбранной группы.</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="campaign-template-list">
|
||||||
|
@foreach (var template in campaignTemplateModels)
|
||||||
|
{
|
||||||
|
<div class="campaign-template-row template-management-row">
|
||||||
|
<div class="campaign-template-info">
|
||||||
|
<h3>@template.Name</h3>
|
||||||
|
<p>@FormatTemplateSummary(template)</p>
|
||||||
|
</div>
|
||||||
|
<div class="template-management-actions">
|
||||||
|
<span class="status-badge status-neutral">@FormatLocalMoscow(template.UpdatedAt.ToMoscow())</span>
|
||||||
|
<button type="button" class="btn-gm btn-gm-danger" disabled="@(deletingTemplateId == template.Id)" @onclick="() => DeleteCampaignTemplate(template)">
|
||||||
|
@(deletingTemplateId == template.Id ? "⏳ Удаляем..." : "Удалить")
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
private List<WebGameGroup>? groups;
|
||||||
|
private List<WebCampaignTemplate>? campaignTemplates;
|
||||||
|
private List<CampaignTemplateManagementModel> campaignTemplateModels = [];
|
||||||
|
private Guid selectedGroupId;
|
||||||
|
private Guid? deletingTemplateId;
|
||||||
|
private bool isCreatingTemplate;
|
||||||
|
private long telegramId;
|
||||||
|
private string? errorMessage;
|
||||||
|
private string? successMessage;
|
||||||
|
private CampaignTemplateEditModel templateModel = new();
|
||||||
|
|
||||||
|
private WebGameGroup? SelectedGroup => groups?.FirstOrDefault(group => group.Id == selectedGroupId);
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
var authState = await AuthStateProvider.GetAuthenticationStateAsync();
|
||||||
|
if (!authState.User.TryGetTelegramId(out telegramId))
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
groups = await SessionService.GetGroupsForGmAsync(telegramId);
|
||||||
|
selectedGroupId = groups.FirstOrDefault()?.Id ?? Guid.Empty;
|
||||||
|
|
||||||
|
if (selectedGroupId != Guid.Empty)
|
||||||
|
{
|
||||||
|
await LoadTemplates();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task OnSelectedGroupChanged(ChangeEventArgs args)
|
||||||
|
{
|
||||||
|
if (!Guid.TryParse(args.Value?.ToString(), out var groupId))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
selectedGroupId = groupId;
|
||||||
|
errorMessage = null;
|
||||||
|
successMessage = null;
|
||||||
|
await LoadTemplates();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task LoadTemplates()
|
||||||
|
{
|
||||||
|
campaignTemplates = null;
|
||||||
|
campaignTemplateModels = [];
|
||||||
|
|
||||||
|
var templates = await SessionService.GetCampaignTemplatesForGmAsync(selectedGroupId, telegramId);
|
||||||
|
if (templates is null)
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
campaignTemplates = templates;
|
||||||
|
RebuildCampaignTemplateModels();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task CreateCampaignTemplate()
|
||||||
|
{
|
||||||
|
errorMessage = null;
|
||||||
|
successMessage = null;
|
||||||
|
|
||||||
|
if (selectedGroupId == Guid.Empty)
|
||||||
|
{
|
||||||
|
errorMessage = "Выберите группу для шаблона.";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ValidateCampaignTemplate(templateModel))
|
||||||
|
{
|
||||||
|
errorMessage = "Шаблон должен иметь название, ссылку, 1-52 игр, шаг 1-365 дней и положительный лимит мест, если он указан.";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
isCreatingTemplate = true;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await SessionService.CreateCampaignTemplateForGmAsync(
|
||||||
|
selectedGroupId,
|
||||||
|
telegramId,
|
||||||
|
new CreateCampaignTemplateRequest(
|
||||||
|
templateModel.Name,
|
||||||
|
templateModel.Title,
|
||||||
|
templateModel.JoinLink,
|
||||||
|
templateModel.SessionCount,
|
||||||
|
templateModel.IntervalDays,
|
||||||
|
templateModel.MaxPlayers,
|
||||||
|
SessionNotificationModeExtensions.FromDatabaseValue(templateModel.NotificationMode)));
|
||||||
|
|
||||||
|
templateModel = new();
|
||||||
|
successMessage = "Шаблон кампании сохранён.";
|
||||||
|
await LoadTemplates();
|
||||||
|
}
|
||||||
|
catch (SessionAccessDeniedException)
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
errorMessage = "Не удалось сохранить шаблон: " + ex.Message;
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
isCreatingTemplate = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task DeleteCampaignTemplate(CampaignTemplateManagementModel template)
|
||||||
|
{
|
||||||
|
errorMessage = null;
|
||||||
|
successMessage = null;
|
||||||
|
deletingTemplateId = template.Id;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await SessionService.DeleteCampaignTemplateForGmAsync(template.Id, telegramId);
|
||||||
|
successMessage = "Шаблон кампании удалён.";
|
||||||
|
await LoadTemplates();
|
||||||
|
}
|
||||||
|
catch (SessionAccessDeniedException)
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
errorMessage = "Не удалось удалить шаблон: " + ex.Message;
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
deletingTemplateId = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void RebuildCampaignTemplateModels()
|
||||||
|
{
|
||||||
|
campaignTemplateModels = campaignTemplates?
|
||||||
|
.OrderByDescending(template => template.UpdatedAt)
|
||||||
|
.ThenBy(template => template.Name)
|
||||||
|
.Select(template => new CampaignTemplateManagementModel
|
||||||
|
{
|
||||||
|
Id = template.Id,
|
||||||
|
Name = template.Name,
|
||||||
|
Title = template.Title,
|
||||||
|
JoinLink = template.JoinLink,
|
||||||
|
SessionCount = template.SessionCount,
|
||||||
|
IntervalDays = template.IntervalDays,
|
||||||
|
MaxPlayers = template.MaxPlayers,
|
||||||
|
NotificationMode = template.NotificationMode,
|
||||||
|
UpdatedAt = template.UpdatedAt
|
||||||
|
})
|
||||||
|
.ToList() ?? [];
|
||||||
|
}
|
||||||
|
|
||||||
|
private static bool ValidateCampaignTemplate(CampaignTemplateEditModel template)
|
||||||
|
{
|
||||||
|
template.Name = template.Name.Trim();
|
||||||
|
template.Title = template.Title.Trim();
|
||||||
|
template.JoinLink = template.JoinLink.Trim();
|
||||||
|
|
||||||
|
if (template.MaxPlayers.HasValue && template.MaxPlayers.Value <= 0)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return template.Name.Length > 0 &&
|
||||||
|
template.Title.Length > 0 &&
|
||||||
|
template.JoinLink.Length > 0 &&
|
||||||
|
template.SessionCount is >= 1 and <= 52 &&
|
||||||
|
template.IntervalDays is >= 1 and <= 365;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string FormatTemplateSummary(CampaignTemplateManagementModel template)
|
||||||
|
{
|
||||||
|
var seats = template.MaxPlayers.HasValue
|
||||||
|
? $"{template.MaxPlayers.Value.ToString(System.Globalization.CultureInfo.InvariantCulture)} мест"
|
||||||
|
: "без лимита";
|
||||||
|
|
||||||
|
return $"{template.Title} · {template.SessionCount} игр · каждые {template.IntervalDays} дн. · {seats} · {FormatNotificationMode(template.NotificationMode)}";
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string FormatNotificationMode(string notificationMode) =>
|
||||||
|
SessionNotificationModeExtensions.FromDatabaseValue(notificationMode) switch
|
||||||
|
{
|
||||||
|
SessionNotificationMode.GroupOnly => "только группа",
|
||||||
|
_ => "группа и личка"
|
||||||
|
};
|
||||||
|
|
||||||
|
private static string FormatRole(string role) =>
|
||||||
|
GroupManagerRoleExtensions.FromDatabaseValue(role).ToDisplayName();
|
||||||
|
|
||||||
|
private static string FormatLocalMoscow(DateTime localMoscow) =>
|
||||||
|
localMoscow.ToString("d MMMM yyyy, HH:mm", System.Globalization.CultureInfo.GetCultureInfo("ru-RU"));
|
||||||
|
|
||||||
|
private sealed class CampaignTemplateEditModel
|
||||||
|
{
|
||||||
|
public string Name { get; set; } = "";
|
||||||
|
public string Title { get; set; } = "";
|
||||||
|
public string JoinLink { get; set; } = "";
|
||||||
|
public int SessionCount { get; set; } = 6;
|
||||||
|
public int IntervalDays { get; set; } = 7;
|
||||||
|
public int? MaxPlayers { get; set; }
|
||||||
|
public string NotificationMode { get; set; } = SessionNotificationModeExtensions.GroupAndDirectValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
private sealed class CampaignTemplateManagementModel
|
||||||
|
{
|
||||||
|
public Guid Id { get; init; }
|
||||||
|
public string Name { get; init; } = "";
|
||||||
|
public string Title { get; init; } = "";
|
||||||
|
public string JoinLink { get; init; } = "";
|
||||||
|
public int SessionCount { get; init; }
|
||||||
|
public int IntervalDays { get; init; }
|
||||||
|
public int? MaxPlayers { get; init; }
|
||||||
|
public string NotificationMode { get; init; } = SessionNotificationModeExtensions.GroupAndDirectValue;
|
||||||
|
public DateTime UpdatedAt { get; init; }
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,61 +2,77 @@
|
|||||||
@using GmRelay.Web.Services
|
@using GmRelay.Web.Services
|
||||||
@using GmRelay.Shared.Domain
|
@using GmRelay.Shared.Domain
|
||||||
@using Microsoft.AspNetCore.Authorization
|
@using Microsoft.AspNetCore.Authorization
|
||||||
|
@using Microsoft.AspNetCore.Components.Authorization
|
||||||
@attribute [Authorize]
|
@attribute [Authorize]
|
||||||
@inject SessionService SessionService
|
@inject AuthorizedSessionService SessionService
|
||||||
|
@inject AuthenticationStateProvider AuthStateProvider
|
||||||
@inject NavigationManager Navigation
|
@inject NavigationManager Navigation
|
||||||
|
|
||||||
<PageTitle>Редактирование сессии - GM-Relay</PageTitle>
|
<PageTitle>Редактирование сессии — GM-Relay</PageTitle>
|
||||||
|
|
||||||
<div class="container mt-4">
|
<div class="page-container">
|
||||||
<nav aria-label="breadcrumb">
|
<ul class="gm-breadcrumb animate-fade-in">
|
||||||
<ol class="breadcrumb">
|
<li><a href="/">Главная</a></li>
|
||||||
<li class="breadcrumb-item"><a href="/">Главная</a></li>
|
<li class="active">Редактирование сессии</li>
|
||||||
<li class="breadcrumb-item active">Редактирование сессии</li>
|
</ul>
|
||||||
</ol>
|
|
||||||
</nav>
|
|
||||||
|
|
||||||
<h2>Редактирование сессии</h2>
|
<div class="page-header animate-fade-in">
|
||||||
|
<h2>✏️ Редактирование сессии</h2>
|
||||||
|
</div>
|
||||||
|
|
||||||
@if (session == null)
|
@if (session == null)
|
||||||
{
|
{
|
||||||
<p>Загрузка деталей сессии...</p>
|
<div class="glass-card" style="padding: 2rem;">
|
||||||
|
<div class="skeleton skeleton-text" style="width: 50%; margin-bottom: 1.5rem;"></div>
|
||||||
|
<div class="skeleton skeleton-text" style="width: 100%; height: 2.5rem; margin-bottom: 1.5rem;"></div>
|
||||||
|
<div class="skeleton skeleton-text" style="width: 50%; margin-bottom: 1.5rem;"></div>
|
||||||
|
<div class="skeleton skeleton-text" style="width: 100%; height: 2.5rem; margin-bottom: 1.5rem;"></div>
|
||||||
|
<div class="skeleton skeleton-text" style="width: 30%; height: 2.5rem;"></div>
|
||||||
|
</div>
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
<div class="card shadow-sm mt-4">
|
<div class="glass-card animate-slide-up" style="max-width: 640px;">
|
||||||
<div class="card-body">
|
<EditForm Model="@model" OnValidSubmit="HandleSubmit">
|
||||||
<EditForm Model="@model" OnValidSubmit="HandleSubmit">
|
<div class="gm-form-group">
|
||||||
<div class="mb-3">
|
<label class="gm-form-label">Название игры</label>
|
||||||
<label class="form-label font-weight-bold">Название игры</label>
|
<InputText @bind-Value="model.Title" class="gm-form-control" placeholder="например, D&D 5e: Dragon's Hoard" />
|
||||||
<InputText @bind-Value="model.Title" class="form-control" placeholder="например, D&D 5e: Dragon's Hoard" />
|
<div class="gm-form-hint">Изменение этого поля обновит все сессии в одной группе.</div>
|
||||||
<div class="form-text">Изменение этого поля обновит все сессии в одной группе.</div>
|
</div>
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="mb-3">
|
<div class="gm-form-group">
|
||||||
<label class="form-label font-weight-bold">Запланированное время (МСК UTC+3)</label>
|
<label class="gm-form-label">Запланированное время (МСК, UTC+3)</label>
|
||||||
<input type="datetime-local" @bind="model.ScheduledAtLocal" class="form-control" />
|
<input type="datetime-local" @bind="model.ScheduledAtLocal" class="gm-form-control" />
|
||||||
<div class="form-text">Текущее: @session.ScheduledAt.FormatMoscow()</div>
|
<div class="gm-form-hint">Текущее: @session.ScheduledAt.FormatMoscow()</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="mb-3">
|
<div class="gm-form-group">
|
||||||
<label class="form-label font-weight-bold">Ссылка для подключения</label>
|
<label class="gm-form-label">Ссылка для подключения</label>
|
||||||
<InputText @bind-Value="model.JoinLink" class="form-control" placeholder="Ссылка на Discord или VTT" />
|
<InputText @bind-Value="model.JoinLink" class="gm-form-control" placeholder="Ссылка на Discord или VTT" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="mt-4">
|
<div class="gm-form-group">
|
||||||
<button type="submit" class="btn btn-success" disabled="@isSubmitting">
|
<label class="gm-form-label">Лимит мест</label>
|
||||||
@(isSubmitting ? "Сохранение..." : "Сохранить изменения")
|
<InputNumber @bind-Value="model.MaxPlayers" class="gm-form-control" min="1" placeholder="Без лимита" />
|
||||||
</button>
|
<div class="gm-form-hint">Пустое значение означает запись без лимита. Если лимит заполнен, новые игроки попадут в лист ожидания.</div>
|
||||||
<button type="button" class="btn btn-outline-secondary ms-2" @onclick="GoBack">Отмена</button>
|
</div>
|
||||||
</div>
|
|
||||||
</EditForm>
|
<div style="display: flex; gap: 0.75rem; margin-top: 1.5rem;">
|
||||||
</div>
|
<button type="submit" class="btn-gm btn-gm-success" disabled="@isSubmitting">
|
||||||
|
@(isSubmitting ? "⏳ Сохранение..." : "✅ Сохранить изменения")
|
||||||
|
</button>
|
||||||
|
<button type="button" class="btn-gm btn-gm-outline" @onclick="GoBack">
|
||||||
|
Отмена
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</EditForm>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@if (!string.IsNullOrEmpty(errorMessage))
|
@if (!string.IsNullOrEmpty(errorMessage))
|
||||||
{
|
{
|
||||||
<div class="alert alert-danger mt-3">@errorMessage</div>
|
<div class="gm-alert gm-alert-danger" style="margin-top: 1rem; max-width: 640px;">
|
||||||
|
⚠️ @errorMessage
|
||||||
|
</div>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
@@ -65,19 +81,29 @@
|
|||||||
[Parameter] public Guid SessionId { get; set; }
|
[Parameter] public Guid SessionId { get; set; }
|
||||||
private WebSession? session;
|
private WebSession? session;
|
||||||
private SessionEditModel model = new();
|
private SessionEditModel model = new();
|
||||||
private bool isSubmitting = false;
|
private bool isSubmitting;
|
||||||
private string? errorMessage;
|
private string? errorMessage;
|
||||||
|
|
||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
{
|
{
|
||||||
session = await SessionService.GetSessionAsync(SessionId);
|
var authState = await AuthStateProvider.GetAuthenticationStateAsync();
|
||||||
if (session != null)
|
if (!authState.User.TryGetTelegramId(out var telegramId))
|
||||||
{
|
{
|
||||||
model.Title = session.Title;
|
Navigation.NavigateTo("/access-denied");
|
||||||
// Convert UTC to Moscow for the picker
|
return;
|
||||||
model.ScheduledAtLocal = session.ScheduledAt.ToMoscow();
|
|
||||||
model.JoinLink = session.JoinLink;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
session = await SessionService.GetSessionForGmAsync(SessionId, telegramId);
|
||||||
|
if (session is null)
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
model.Title = session.Title;
|
||||||
|
model.ScheduledAtLocal = session.ScheduledAt.ToMoscow();
|
||||||
|
model.JoinLink = session.JoinLink;
|
||||||
|
model.MaxPlayers = session.MaxPlayers;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task HandleSubmit()
|
private async Task HandleSubmit()
|
||||||
@@ -87,13 +113,22 @@
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
// The value from <input type="datetime-local"> is considered as "unspecified" or local to browser.
|
var authState = await AuthStateProvider.GetAuthenticationStateAsync();
|
||||||
// We treat it as Moscow time (UTC+3) and convert to UTC.
|
if (!authState.User.TryGetTelegramId(out var telegramId))
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var utcTime = new DateTimeOffset(model.ScheduledAtLocal, TimeSpan.FromHours(3)).ToUniversalTime().UtcDateTime;
|
var utcTime = new DateTimeOffset(model.ScheduledAtLocal, TimeSpan.FromHours(3)).ToUniversalTime().UtcDateTime;
|
||||||
|
|
||||||
await SessionService.UpdateSessionAsync(SessionId, model.Title, utcTime, model.JoinLink);
|
await SessionService.UpdateSessionForGmAsync(SessionId, telegramId, model.Title, utcTime, model.JoinLink, model.MaxPlayers);
|
||||||
Navigation.NavigateTo($"/group/{session!.GroupId}");
|
Navigation.NavigateTo($"/group/{session!.GroupId}");
|
||||||
}
|
}
|
||||||
|
catch (SessionAccessDeniedException)
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
errorMessage = "Не удалось сохранить изменения: " + ex.Message;
|
errorMessage = "Не удалось сохранить изменения: " + ex.Message;
|
||||||
@@ -111,5 +146,6 @@
|
|||||||
public string Title { get; set; } = "";
|
public string Title { get; set; } = "";
|
||||||
public DateTime ScheduledAtLocal { get; set; } = DateTime.Now;
|
public DateTime ScheduledAtLocal { get; set; } = DateTime.Now;
|
||||||
public string JoinLink { get; set; } = "";
|
public string JoinLink { get; set; } = "";
|
||||||
|
public int? MaxPlayers { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,28 +1,26 @@
|
|||||||
@page "/Error"
|
@page "/Error"
|
||||||
@using System.Diagnostics
|
@using System.Diagnostics
|
||||||
|
|
||||||
<PageTitle>Ошибка</PageTitle>
|
<PageTitle>Ошибка — GM-Relay</PageTitle>
|
||||||
|
|
||||||
<h1 class="text-danger">Ошибка.</h1>
|
<div class="page-container">
|
||||||
<h2 class="text-danger">Произошла ошибка при обработке вашего запроса.</h2>
|
<div class="error-page">
|
||||||
|
<div class="error-page-icon">⚠️</div>
|
||||||
|
<h1 class="error-page-title">Произошла ошибка</h1>
|
||||||
|
<p class="error-page-text">При обработке вашего запроса что-то пошло не так. Пожалуйста, попробуйте снова.</p>
|
||||||
|
|
||||||
@if (ShowRequestId)
|
@if (ShowRequestId)
|
||||||
{
|
{
|
||||||
<p>
|
<p style="font-size: 0.75rem; color: var(--text-muted); font-family: monospace;">
|
||||||
<strong>ID запроса:</strong> <code>@RequestId</code>
|
ID запроса: @RequestId
|
||||||
</p>
|
</p>
|
||||||
}
|
}
|
||||||
|
|
||||||
<h3>Режим разработки</h3>
|
<a href="/" class="btn-gm btn-gm-primary" style="margin-top: 0.5rem;">
|
||||||
<p>
|
← На главную
|
||||||
Переключение на среду <strong>Development</strong> отобразит более подробную информацию о произошедшей ошибке.
|
</a>
|
||||||
</p>
|
</div>
|
||||||
<p>
|
</div>
|
||||||
<strong>Среда Development не должна быть включена для развернутых приложений.</strong>
|
|
||||||
Это может привести к отображению конфиденциальной информации из исключений конечным пользователям.
|
|
||||||
Для локальной отладки включите среду <strong>Development</strong>, установив переменную среды <strong>ASPNETCORE_ENVIRONMENT</strong> в значение <strong>Development</strong>
|
|
||||||
и перезапустите приложение.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
@code{
|
@code{
|
||||||
[CascadingParameter]
|
[CascadingParameter]
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,235 @@
|
|||||||
|
@page "/group/{GroupId:guid}/stats"
|
||||||
|
@using GmRelay.Web.Services
|
||||||
|
@using GmRelay.Shared.Domain
|
||||||
|
@using Microsoft.AspNetCore.Authorization
|
||||||
|
@using Microsoft.AspNetCore.Components.Authorization
|
||||||
|
@using System.Security.Claims
|
||||||
|
@attribute [Authorize]
|
||||||
|
@inject ISessionStore SessionStore
|
||||||
|
@inject AuthenticationStateProvider AuthStateProvider
|
||||||
|
@inject NavigationManager Navigation
|
||||||
|
|
||||||
|
<PageTitle>Статистика — GM-Relay</PageTitle>
|
||||||
|
|
||||||
|
<div class="page-container">
|
||||||
|
<ul class="gm-breadcrumb animate-fade-in">
|
||||||
|
<li><a href="/">Главная</a></li>
|
||||||
|
<li><a href="/group/@GroupId">Сессии группы</a></li>
|
||||||
|
<li class="active">Статистика</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<div class="page-header animate-fade-in">
|
||||||
|
<h2>📊 Статистика посещаемости</h2>
|
||||||
|
<p class="page-subtitle">Надёжность состава и качество расписания</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@if (!string.IsNullOrEmpty(errorMessage))
|
||||||
|
{
|
||||||
|
<div class="gm-alert gm-alert-danger" style="margin-bottom: 1rem;">
|
||||||
|
⚠️ @errorMessage
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (stats is null)
|
||||||
|
{
|
||||||
|
<div class="loading-spinner">⏳ Загружаем статистику…</div>
|
||||||
|
}
|
||||||
|
else if (stats.Count == 0)
|
||||||
|
{
|
||||||
|
<div class="empty-state">
|
||||||
|
<div class="empty-icon">📈</div>
|
||||||
|
<h3>Пока нет данных</h3>
|
||||||
|
<p>После первых сессий здесь появится аналитика.</p>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<div class="glass-card animate-slide-up" style="margin-bottom: 1rem;">
|
||||||
|
<div class="stats-summary" style="display: grid; grid-template-columns: repeat(auto-fit, minmax(120px, 1fr)); gap: 1rem; margin-bottom: 1.5rem;">
|
||||||
|
<div class="stat-card">
|
||||||
|
<div class="stat-value">@stats.Count</div>
|
||||||
|
<div class="stat-label">Игроков</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-card">
|
||||||
|
<div class="stat-value">@TotalSessions</div>
|
||||||
|
<div class="stat-label">Сессий</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-card">
|
||||||
|
<div class="stat-value">@AvgAttendanceRate%</div>
|
||||||
|
<div class="stat-label">Средняя посещаемость</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-card">
|
||||||
|
<div class="stat-value">@topPlayer?.DisplayName</div>
|
||||||
|
<div class="stat-label">Самый стабильный</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="table-responsive">
|
||||||
|
<table class="gm-table" style="width: 100%;">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th @onclick="@(() => SortBy("player"))" style="cursor:pointer;" class="sortable">Игрок @(sortColumn == "player" ? (sortDesc ? "▼" : "▲") : "")</th>
|
||||||
|
<th @onclick="@(() => SortBy("total"))" style="cursor:pointer; text-align:center;" class="sortable">Всего @(sortColumn == "total" ? (sortDesc ? "▼" : "▲") : "")</th>
|
||||||
|
<th @onclick="@(() => SortBy("confirmed"))" style="cursor:pointer; text-align:center;" class="sortable">✅ @(sortColumn == "confirmed" ? (sortDesc ? "▼" : "▲") : "")</th>
|
||||||
|
<th @onclick="@(() => SortBy("declined"))" style="cursor:pointer; text-align:center;" class="sortable">❌ @(sortColumn == "declined" ? (sortDesc ? "▼" : "▲") : "")</th>
|
||||||
|
<th @onclick="@(() => SortBy("noresponse"))" style="cursor:pointer; text-align:center;" class="sortable">💤 @(sortColumn == "noresponse" ? (sortDesc ? "▼" : "▲") : "")</th>
|
||||||
|
<th @onclick="@(() => SortBy("waitlist"))" style="cursor:pointer; text-align:center;" class="sortable">⏳ @(sortColumn == "waitlist" ? (sortDesc ? "▼" : "▲") : "")</th>
|
||||||
|
<th @onclick="@(() => SortBy("rate"))" style="cursor:pointer; text-align:center;" class="sortable">% @(sortColumn == "rate" ? (sortDesc ? "▼" : "▲") : "")</th>
|
||||||
|
<th @onclick="@(() => SortBy("cancelled"))" style="cursor:pointer; text-align:center;" class="sortable">🚫 @(sortColumn == "cancelled" ? (sortDesc ? "▼" : "▲") : "")</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
@foreach (var s in sortedStats)
|
||||||
|
{
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
<div class="player-info">
|
||||||
|
<span class="player-name">@s.DisplayName</span>
|
||||||
|
@if (!string.IsNullOrEmpty(s.TelegramUsername))
|
||||||
|
{
|
||||||
|
<span class="player-username">@@@s.TelegramUsername</span>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
<td style="text-align:center;">@s.TotalSessions</td>
|
||||||
|
<td style="text-align:center;">@s.ConfirmedCount</td>
|
||||||
|
<td style="text-align:center;">@s.DeclinedCount</td>
|
||||||
|
<td style="text-align:center;">@s.NoResponseCount</td>
|
||||||
|
<td style="text-align:center;">@s.WaitlistedCount</td>
|
||||||
|
<td style="text-align:center;">
|
||||||
|
<span class="rate-badge @AttendanceBadgeClass(s.AttendanceRate)">
|
||||||
|
@s.AttendanceRate%
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td style="text-align:center;">@s.CancellationAffectedCount</td>
|
||||||
|
</tr>
|
||||||
|
}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.stat-card {
|
||||||
|
background: var(--card-bg-secondary, rgba(255,255,255,0.05));
|
||||||
|
border-radius: 0.75rem;
|
||||||
|
padding: 1rem;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.stat-value {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--accent-color, #7cb97a);
|
||||||
|
}
|
||||||
|
.stat-label {
|
||||||
|
font-size: 0.8rem;
|
||||||
|
color: var(--text-muted, #94a3b8);
|
||||||
|
margin-top: 0.25rem;
|
||||||
|
}
|
||||||
|
.player-info {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
.player-name {
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
.player-username {
|
||||||
|
font-size: 0.8rem;
|
||||||
|
color: var(--text-muted, #94a3b8);
|
||||||
|
}
|
||||||
|
.rate-badge {
|
||||||
|
display: inline-block;
|
||||||
|
padding: 0.25rem 0.5rem;
|
||||||
|
border-radius: 0.5rem;
|
||||||
|
font-weight: 600;
|
||||||
|
font-size: 0.875rem;
|
||||||
|
}
|
||||||
|
.rate-excellent { background: rgba(34,197,94,0.15); color: #22c55e; }
|
||||||
|
.rate-good { background: rgba(234,179,8,0.15); color: #eab308; }
|
||||||
|
.rate-poor { background: rgba(239,68,68,0.15); color: #ef4444; }
|
||||||
|
</style>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
[Parameter] public Guid GroupId { get; set; }
|
||||||
|
private List<PlayerAttendanceStats>? stats;
|
||||||
|
private List<PlayerAttendanceStats> sortedStats = new();
|
||||||
|
private string? errorMessage;
|
||||||
|
private string sortColumn = "confirmed";
|
||||||
|
private bool sortDesc = true;
|
||||||
|
private int TotalSessions => stats?.Count > 0 ? (int)(stats.Max(s => s.TotalSessions)) : 0;
|
||||||
|
private int AvgAttendanceRate => stats?.Count > 0 ? (int)(stats.Average(s => s.AttendanceRate)) : 0;
|
||||||
|
private PlayerAttendanceStats? topPlayer => stats?.OrderByDescending(s => s.AttendanceRate).ThenByDescending(s => s.ConfirmedCount).FirstOrDefault();
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
var authState = await AuthStateProvider.GetAuthenticationStateAsync();
|
||||||
|
var user = authState.User;
|
||||||
|
if (!user.Identity?.IsAuthenticated ?? true)
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/login");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var telegramIdClaim = user.FindFirst("telegram_id")?.Value
|
||||||
|
?? user.FindFirst(ClaimTypes.NameIdentifier)?.Value;
|
||||||
|
if (!long.TryParse(telegramIdClaim, out var telegramId))
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/login");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (!await SessionStore.IsGroupManagerAsync(GroupId, telegramId))
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
stats = await SessionStore.GetGroupAttendanceStatsAsync(GroupId) ?? new();
|
||||||
|
UpdateSortedStats();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
errorMessage = $"Ошибка загрузки статистики: {ex.Message}";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void SortBy(string column)
|
||||||
|
{
|
||||||
|
if (sortColumn == column)
|
||||||
|
sortDesc = !sortDesc;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sortColumn = column;
|
||||||
|
sortDesc = true;
|
||||||
|
}
|
||||||
|
UpdateSortedStats();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void UpdateSortedStats()
|
||||||
|
{
|
||||||
|
if (stats is null) { sortedStats = new(); return; }
|
||||||
|
IOrderedEnumerable<PlayerAttendanceStats> ordered = sortColumn switch
|
||||||
|
{
|
||||||
|
"player" => sortDesc ? stats.OrderByDescending(s => s.DisplayName) : stats.OrderBy(s => s.DisplayName),
|
||||||
|
"total" => sortDesc ? stats.OrderByDescending(s => s.TotalSessions) : stats.OrderBy(s => s.TotalSessions),
|
||||||
|
"confirmed" => sortDesc ? stats.OrderByDescending(s => s.ConfirmedCount) : stats.OrderBy(s => s.ConfirmedCount),
|
||||||
|
"declined" => sortDesc ? stats.OrderByDescending(s => s.DeclinedCount) : stats.OrderBy(s => s.DeclinedCount),
|
||||||
|
"noresponse" => sortDesc ? stats.OrderByDescending(s => s.NoResponseCount) : stats.OrderBy(s => s.NoResponseCount),
|
||||||
|
"waitlist" => sortDesc ? stats.OrderByDescending(s => s.WaitlistedCount) : stats.OrderBy(s => s.WaitlistedCount),
|
||||||
|
"rate" => sortDesc ? stats.OrderByDescending(s => s.AttendanceRate) : stats.OrderBy(s => s.AttendanceRate),
|
||||||
|
"cancelled" => sortDesc ? stats.OrderByDescending(s => s.CancellationAffectedCount) : stats.OrderBy(s => s.CancellationAffectedCount),
|
||||||
|
_ => stats.OrderByDescending(s => s.ConfirmedCount)
|
||||||
|
};
|
||||||
|
sortedStats = ordered.ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
private string SortIndicator(string column) => sortColumn == column ? (sortDesc ? "▼" : "▲") : "";
|
||||||
|
|
||||||
|
private string AttendanceBadgeClass(decimal rate) => rate switch
|
||||||
|
{
|
||||||
|
>= 75m => "rate-excellent",
|
||||||
|
>= 50m => "rate-good",
|
||||||
|
_ => "rate-poor"
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -1,50 +1,88 @@
|
|||||||
@page "/"
|
@page "/"
|
||||||
@using Microsoft.AspNetCore.Authorization
|
@using Microsoft.AspNetCore.Authorization
|
||||||
@using Microsoft.AspNetCore.Components.Authorization
|
@using Microsoft.AspNetCore.Components.Authorization
|
||||||
|
@using GmRelay.Shared.Domain
|
||||||
@using GmRelay.Web.Services
|
@using GmRelay.Web.Services
|
||||||
@attribute [Authorize]
|
@attribute [Authorize]
|
||||||
@inject SessionService SessionService
|
@inject AuthorizedSessionService SessionService
|
||||||
@inject AuthenticationStateProvider AuthStateProvider
|
@inject AuthenticationStateProvider AuthStateProvider
|
||||||
|
@inject NavigationManager Navigation
|
||||||
|
|
||||||
<PageTitle>Панель управления - GM-Relay</PageTitle>
|
<PageTitle>Панель управления — GM-Relay</PageTitle>
|
||||||
|
|
||||||
<div class="container mt-4">
|
<div class="page-container">
|
||||||
<h2>Добро пожаловать, @userName!</h2>
|
<div class="page-header animate-fade-in">
|
||||||
<p class="text-muted">Выберите группу для управления играми.</p>
|
<h2>Добро пожаловать, @userName! 👋</h2>
|
||||||
|
<p>Выберите группу для управления игровыми сессиями.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="row mt-4">
|
@if (groups == null)
|
||||||
@if (groups == null)
|
{
|
||||||
{
|
<div class="card-grid">
|
||||||
<p>Загрузка групп...</p>
|
@for (int i = 0; i < 3; i++)
|
||||||
}
|
{
|
||||||
else if (groups.Count == 0)
|
<div class="skeleton skeleton-card"></div>
|
||||||
{
|
}
|
||||||
<div class="col-12">
|
</div>
|
||||||
<div class="card bg-light">
|
}
|
||||||
<div class="card-body text-center">
|
else if (groups.Count == 0)
|
||||||
<p class="mb-0">У вас еще нет зарегистрированных групп. Сначала добавьте бота в группу Telegram!</p>
|
{
|
||||||
</div>
|
<div class="glass-card">
|
||||||
</div>
|
<div class="empty-state">
|
||||||
|
<div class="empty-state-icon">🤖</div>
|
||||||
|
<div class="empty-state-title">Нет зарегистрированных групп</div>
|
||||||
|
<p class="empty-state-text">Добавьте бота GM-Relay в свою группу Telegram, чтобы начать управлять игровыми сессиями.</p>
|
||||||
</div>
|
</div>
|
||||||
}
|
</div>
|
||||||
else
|
}
|
||||||
{
|
else
|
||||||
|
{
|
||||||
|
<div class="card-grid stagger-children">
|
||||||
@foreach (var group in groups)
|
@foreach (var group in groups)
|
||||||
{
|
{
|
||||||
<div class="col-md-4 mb-3">
|
<div class="glass-card group-card">
|
||||||
<div class="card h-100 shadow-sm">
|
<div class="group-card-icon">🎮</div>
|
||||||
<div class="card-body">
|
<h3 class="group-card-title">@group.Name</h3>
|
||||||
<h5 class="card-title">@group.Name</h5>
|
<p class="group-card-id">ID: @group.TelegramChatId</p>
|
||||||
<p class="card-text text-muted">ID: @group.TelegramChatId</p>
|
<span class="status-badge @(group.ManagerRole == GroupManagerRoleExtensions.OwnerValue ? "status-success" : "status-info")" style="align-self: flex-start; margin-bottom: 1rem;">
|
||||||
<a href="/group/@group.Id" class="btn btn-primary">Посмотреть игры</a>
|
@FormatRole(group.ManagerRole)
|
||||||
</div>
|
</span>
|
||||||
</div>
|
<a href="/group/@group.Id" class="btn-gm btn-gm-primary" style="width: 100%; justify-content: center; margin-top: auto;">
|
||||||
|
Посмотреть игры →
|
||||||
|
</a>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
}
|
</div>
|
||||||
</div>
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.group-card {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
min-height: 180px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-card-icon {
|
||||||
|
font-size: 2rem;
|
||||||
|
margin-bottom: 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-card-title {
|
||||||
|
font-size: 1.125rem;
|
||||||
|
font-weight: 600;
|
||||||
|
margin-bottom: 0.25rem;
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-card-id {
|
||||||
|
font-size: 0.75rem;
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-family: 'Courier New', monospace;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
@code {
|
@code {
|
||||||
private List<WebGameGroup>? groups;
|
private List<WebGameGroup>? groups;
|
||||||
private string userName = "";
|
private string userName = "";
|
||||||
@@ -55,10 +93,15 @@
|
|||||||
var user = authState.User;
|
var user = authState.User;
|
||||||
userName = user.Identity?.Name ?? "Мастер Игры";
|
userName = user.Identity?.Name ?? "Мастер Игры";
|
||||||
|
|
||||||
var telegramIdClaim = user.FindFirst("TelegramId")?.Value;
|
if (!user.TryGetTelegramId(out var telegramId))
|
||||||
if (long.TryParse(telegramIdClaim, out var telegramId))
|
|
||||||
{
|
{
|
||||||
groups = await SessionService.GetGroupsForGmAsync(telegramId);
|
Navigation.NavigateTo("/access-denied");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
groups = await SessionService.GetGroupsForGmAsync(telegramId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static string FormatRole(string role) =>
|
||||||
|
GroupManagerRoleExtensions.FromDatabaseValue(role).ToDisplayName();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,37 +2,29 @@
|
|||||||
@using Microsoft.AspNetCore.Components.Authorization
|
@using Microsoft.AspNetCore.Components.Authorization
|
||||||
@inject NavigationManager Navigation
|
@inject NavigationManager Navigation
|
||||||
@inject IConfiguration Configuration
|
@inject IConfiguration Configuration
|
||||||
|
@inject IJSRuntime JS
|
||||||
|
|
||||||
<PageTitle>Вход - GM-Relay</PageTitle>
|
<PageTitle>Вход — GM-Relay</PageTitle>
|
||||||
|
|
||||||
<div class="container">
|
<div class="login-page">
|
||||||
<div class="row justify-content-center mt-5">
|
<div class="login-card">
|
||||||
<div class="col-md-6 text-center">
|
<img src="logo.png" alt="GM-Relay" class="login-logo" />
|
||||||
<h3>Панель управления GM-Relay</h3>
|
<h1 class="login-title">GM-Relay</h1>
|
||||||
<p class="text-muted">Пожалуйста, войдите как Мастер Игры для управления сессиями.</p>
|
<p class="login-subtitle">Войдите через Telegram для управления игровыми сессиями</p>
|
||||||
|
|
||||||
<div class="mt-4">
|
@if (Navigation.Uri.Contains("error=auth_failed"))
|
||||||
@if (Navigation.Uri.Contains("error=auth_failed"))
|
{
|
||||||
{
|
<div class="gm-alert gm-alert-danger" style="margin-bottom: 1.5rem; justify-content: center;">
|
||||||
<div class="alert alert-danger">Ошибка аутентификации. Пожалуйста, попробуйте снова.</div>
|
⚠️ Ошибка аутентификации. Пожалуйста, попробуйте снова.
|
||||||
}
|
|
||||||
|
|
||||||
@* Telegram Login Widget *@
|
|
||||||
<div id="telegram-login-container">
|
|
||||||
<script async src="https://telegram.org/js/telegram-widget.js?22"
|
|
||||||
data-telegram-login="@BotUsername"
|
|
||||||
data-size="large"
|
|
||||||
data-auth-url="@AuthUrl"
|
|
||||||
data-request-access="write"></script>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
}
|
||||||
|
|
||||||
|
<div id="telegram-login-container"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@code {
|
@code {
|
||||||
private string BotUsername => Configuration["Telegram:BotUsername"] ?? "GmRelayBot";
|
private string BotUsername => Configuration["Telegram:BotUsername"] ?? "GmRelayBot";
|
||||||
private string AuthUrl => Navigation.ToAbsoluteUri("/auth/telegram").ToString();
|
|
||||||
|
|
||||||
[CascadingParameter]
|
[CascadingParameter]
|
||||||
private Task<AuthenticationState>? AuthStateTask { get; set; }
|
private Task<AuthenticationState>? AuthStateTask { get; set; }
|
||||||
@@ -48,4 +40,13 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected override async Task OnAfterRenderAsync(bool firstRender)
|
||||||
|
{
|
||||||
|
if (firstRender)
|
||||||
|
{
|
||||||
|
await JS.InvokeVoidAsync("loadTelegramWidget", BotUsername, "/auth/telegram-login");
|
||||||
|
await JS.InvokeVoidAsync("watchTelegramMiniAppLogin", "/auth/status", "/", false);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,103 @@
|
|||||||
|
@page "/miniapp"
|
||||||
|
@using Microsoft.AspNetCore.Components.Authorization
|
||||||
|
@using System.Text.Json.Serialization
|
||||||
|
@inject IJSRuntime JS
|
||||||
|
@inject NavigationManager Navigation
|
||||||
|
|
||||||
|
<PageTitle>Mini App Dashboard — GM-Relay</PageTitle>
|
||||||
|
|
||||||
|
<div class="mini-app-page">
|
||||||
|
<div class="mini-app-auth-card" data-auth-status="@miniAppAuthStatus">
|
||||||
|
<img src="logo.png" alt="GM-Relay" class="mini-app-logo" />
|
||||||
|
<h1>GM-Relay</h1>
|
||||||
|
<p>@statusMessage</p>
|
||||||
|
|
||||||
|
@if (showFallback)
|
||||||
|
{
|
||||||
|
<a href="/login" class="btn-gm btn-gm-primary">Войти через Telegram</a>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
private string statusMessage = "Открываем dashboard внутри Telegram...";
|
||||||
|
private string miniAppAuthStatus = "starting";
|
||||||
|
private bool showFallback;
|
||||||
|
|
||||||
|
[CascadingParameter]
|
||||||
|
private Task<AuthenticationState>? AuthStateTask { get; set; }
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
if (AuthStateTask is null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var user = (await AuthStateTask).User;
|
||||||
|
if (user.Identity?.IsAuthenticated == true)
|
||||||
|
{
|
||||||
|
Navigation.NavigateTo("/");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async Task OnAfterRenderAsync(bool firstRender)
|
||||||
|
{
|
||||||
|
if (!firstRender)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var result = await JS.InvokeAsync<MiniAppAuthResult>(
|
||||||
|
"authenticateTelegramMiniApp",
|
||||||
|
"/auth/telegram-webapp",
|
||||||
|
"/");
|
||||||
|
|
||||||
|
if (!result.Authenticated)
|
||||||
|
{
|
||||||
|
miniAppAuthStatus = string.IsNullOrWhiteSpace(result.Reason)
|
||||||
|
? "telegram-auth-failed"
|
||||||
|
: result.Reason;
|
||||||
|
statusMessage = GetStatusMessage(miniAppAuthStatus);
|
||||||
|
showFallback = true;
|
||||||
|
StateHasChanged();
|
||||||
|
await TryWatchLoginAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (JSException)
|
||||||
|
{
|
||||||
|
miniAppAuthStatus = "telegram-auth-failed";
|
||||||
|
statusMessage = "Не удалось получить данные Telegram Mini App. Попробуйте открыть dashboard из бота.";
|
||||||
|
showFallback = true;
|
||||||
|
StateHasChanged();
|
||||||
|
await TryWatchLoginAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task TryWatchLoginAsync()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await JS.InvokeVoidAsync("watchTelegramMiniAppLogin", "/auth/status", "/");
|
||||||
|
}
|
||||||
|
catch (JSException)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string GetStatusMessage(string reason) => reason switch
|
||||||
|
{
|
||||||
|
"telegram-webapp-missing" => "Mini App API не найден. Если страница открыта в браузере, войдите через Telegram.",
|
||||||
|
"telegram-init-data-empty" => "Telegram открыл страницу без Mini App initData. Попробуйте войти через Telegram на этом экране.",
|
||||||
|
"telegram-auth-failed" => "Не удалось проверить Telegram Mini App. Попробуйте войти через Telegram.",
|
||||||
|
_ => "Mini App доступен из Telegram. Для браузера используйте обычный вход."
|
||||||
|
};
|
||||||
|
|
||||||
|
private sealed record MiniAppAuthResult(
|
||||||
|
[property: JsonPropertyName("authenticated")] bool Authenticated,
|
||||||
|
[property: JsonPropertyName("reason")] string? Reason,
|
||||||
|
[property: JsonPropertyName("status")] int? Status,
|
||||||
|
[property: JsonPropertyName("redirectUrl")] string? RedirectUrl);
|
||||||
|
}
|
||||||
@@ -1,5 +1,13 @@
|
|||||||
@page "/not-found"
|
@page "/not-found"
|
||||||
@layout MainLayout
|
@layout MainLayout
|
||||||
|
|
||||||
<h3>Не найдено</h3>
|
<PageTitle>404 — GM-Relay</PageTitle>
|
||||||
<p>Извините, страница, которую вы ищете, не существует.</p>
|
|
||||||
|
<div class="error-page">
|
||||||
|
<div class="error-page-icon">🔍</div>
|
||||||
|
<h1 class="error-page-title">Страница не найдена</h1>
|
||||||
|
<p class="error-page-text">Извините, страница, которую вы ищете, не существует или была перемещена.</p>
|
||||||
|
<a href="/" class="btn-gm btn-gm-primary">
|
||||||
|
← Вернуться на главную
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user