Compare commits
18 Commits
bdd7d0424f
...
localizati
| Author | SHA1 | Date | |
|---|---|---|---|
| 4c87f02679 | |||
| 9106af4f8e | |||
| 240864617f | |||
| 0bbeb0767b | |||
| 88d9ccd75d | |||
| 9281388959 | |||
| 0566901fa4 | |||
| e907dffe8c | |||
| fdd0580554 | |||
| 29d6255f22 | |||
| 77ae161b1a | |||
| d02a742278 | |||
| 35a977536b | |||
| 155e4d3b7b | |||
| 713eadc643 | |||
| 5ea3e8c1f3 | |||
| 85027a7747 | |||
| e275a9856b |
57
.env.example
57
.env.example
@@ -1,30 +1,65 @@
|
||||
# Telegram Bot Configuration
|
||||
# Telegram Tinder Bot Configuration
|
||||
# Rename this file to .env before starting the application
|
||||
|
||||
# === REQUIRED SETTINGS ===
|
||||
|
||||
# Telegram Bot Token (Get from @BotFather)
|
||||
TELEGRAM_BOT_TOKEN=your_telegram_bot_token_here
|
||||
|
||||
# Database Configuration
|
||||
# For local development (when running the bot directly)
|
||||
DB_HOST=localhost
|
||||
DB_PORT=5432
|
||||
DB_NAME=telegram_tinder_bot
|
||||
DB_USERNAME=postgres
|
||||
DB_PASSWORD=your_password_here
|
||||
|
||||
# Application Settings
|
||||
# === APPLICATION SETTINGS ===
|
||||
|
||||
# Environment (development, production)
|
||||
NODE_ENV=development
|
||||
|
||||
# Port for health checks
|
||||
PORT=3000
|
||||
|
||||
# Optional: Redis for caching (if using)
|
||||
REDIS_HOST=localhost
|
||||
REDIS_PORT=6379
|
||||
REDIS_PASSWORD=
|
||||
# === FILE UPLOAD SETTINGS ===
|
||||
|
||||
# Optional: File upload settings
|
||||
# Path for storing uploaded files
|
||||
UPLOAD_PATH=./uploads
|
||||
|
||||
# Maximum file size for uploads (in bytes, default: 5MB)
|
||||
MAX_FILE_SIZE=5242880
|
||||
|
||||
# Optional: External services
|
||||
GOOGLE_MAPS_API_KEY=your_google_maps_key
|
||||
CLOUDINARY_URL=your_cloudinary_url
|
||||
# === LOGGING ===
|
||||
|
||||
# Security
|
||||
# Log level (error, warn, info, debug)
|
||||
LOG_LEVEL=info
|
||||
|
||||
# Path for storing log files
|
||||
LOG_PATH=./logs
|
||||
|
||||
# === SECURITY ===
|
||||
|
||||
# Secret key for JWT tokens
|
||||
JWT_SECRET=your_jwt_secret_here
|
||||
|
||||
# Encryption key for sensitive data
|
||||
ENCRYPTION_KEY=your_encryption_key_here
|
||||
|
||||
# === KAKAO MAPS API ===
|
||||
|
||||
# Kakao REST API Key for location services (Get from https://developers.kakao.com/)
|
||||
# You can use either KAKAO_REST_API_KEY or KAKAO_MAP_REST_KEY
|
||||
KAKAO_REST_API_KEY=your_kakao_rest_api_key_here
|
||||
# KAKAO_MAP_REST_KEY=your_kakao_rest_api_key_here
|
||||
|
||||
# === ADVANCED SETTINGS ===
|
||||
|
||||
# Notification check interval in milliseconds (default: 60000 - 1 minute)
|
||||
NOTIFICATION_CHECK_INTERVAL=60000
|
||||
|
||||
# Number of matches to show per page
|
||||
MATCHES_PER_PAGE=10
|
||||
|
||||
# Number of profiles to load at once
|
||||
PROFILES_BATCH_SIZE=5
|
||||
|
||||
68
.env.production
Normal file
68
.env.production
Normal file
@@ -0,0 +1,68 @@
|
||||
# Конфигурация Telegram Tinder Bot для Production
|
||||
|
||||
# === НЕОБХОДИМЫЕ НАСТРОЙКИ ===
|
||||
|
||||
# Токен Telegram бота (получить у @BotFather)
|
||||
TELEGRAM_BOT_TOKEN=your_bot_token_here
|
||||
|
||||
# Настройки базы данных PostgreSQL
|
||||
DB_HOST=db
|
||||
DB_PORT=5432
|
||||
DB_NAME=telegram_tinder_bot
|
||||
DB_USERNAME=postgres
|
||||
DB_PASSWORD=your_secure_password_here
|
||||
|
||||
# === НАСТРОЙКИ ПРИЛОЖЕНИЯ ===
|
||||
|
||||
# Окружение
|
||||
NODE_ENV=production
|
||||
|
||||
# Порт для проверок работоспособности
|
||||
PORT=3000
|
||||
|
||||
# === НАСТРОЙКИ ЗАГРУЗКИ ФАЙЛОВ ===
|
||||
|
||||
# Путь для хранения загруженных файлов
|
||||
UPLOAD_PATH=./uploads
|
||||
|
||||
# Максимальный размер загружаемого файла (в байтах, по умолчанию: 5MB)
|
||||
MAX_FILE_SIZE=5242880
|
||||
|
||||
# Разрешенные типы файлов
|
||||
ALLOWED_FILE_TYPES=image/jpeg,image/png,image/gif
|
||||
|
||||
# === ЛОГИРОВАНИЕ ===
|
||||
|
||||
# Уровень логирования (error, warn, info, debug)
|
||||
LOG_LEVEL=info
|
||||
|
||||
# Путь для хранения лог-файлов
|
||||
LOG_PATH=./logs
|
||||
|
||||
# === БЕЗОПАСНОСТЬ ===
|
||||
|
||||
# Секретный ключ для JWT токенов
|
||||
JWT_SECRET=your_jwt_secret_here
|
||||
|
||||
# Ключ шифрования для чувствительных данных
|
||||
ENCRYPTION_KEY=your_encryption_key_here
|
||||
|
||||
# === РАСШИРЕННЫЕ НАСТРОЙКИ ===
|
||||
|
||||
# Интервал проверки уведомлений в миллисекундах (по умолчанию: 60000 - 1 минута)
|
||||
NOTIFICATION_CHECK_INTERVAL=60000
|
||||
|
||||
# Количество матчей для отображения на странице
|
||||
MATCHES_PER_PAGE=10
|
||||
|
||||
# Количество профилей для загрузки за один раз
|
||||
PROFILES_BATCH_SIZE=5
|
||||
|
||||
# === НАСТРОЙКИ DOCKER ===
|
||||
|
||||
# Имя хоста для доступа извне
|
||||
EXTERNAL_HOSTNAME=your_domain.com
|
||||
|
||||
# Настройки кеширования (Redis, если используется)
|
||||
CACHE_HOST=redis
|
||||
CACHE_PORT=6379
|
||||
29
Dockerfile
29
Dockerfile
@@ -8,14 +8,15 @@ WORKDIR /app
|
||||
COPY package*.json ./
|
||||
COPY tsconfig.json ./
|
||||
|
||||
# Install dependencies
|
||||
RUN npm ci --only=production && npm cache clean --force
|
||||
# Install all dependencies (including devDependencies for build)
|
||||
RUN npm ci && npm cache clean --force
|
||||
|
||||
# Copy source code
|
||||
COPY src/ ./src/
|
||||
COPY .env.example ./
|
||||
|
||||
# Build the application
|
||||
RUN npm run build
|
||||
# Build the application (using Linux-compatible build command)
|
||||
RUN npm run build:linux
|
||||
|
||||
# Production stage
|
||||
FROM node:18-alpine AS production
|
||||
@@ -31,11 +32,19 @@ RUN npm ci --only=production && npm cache clean --force
|
||||
|
||||
# Copy built application from builder stage
|
||||
COPY --from=builder /app/dist ./dist
|
||||
COPY --from=builder /app/.env.example ./.env.example
|
||||
|
||||
# Copy configuration files
|
||||
COPY config/ ./config/
|
||||
# Copy database migrations
|
||||
COPY src/database/migrations/ ./dist/database/migrations/
|
||||
|
||||
# Create uploads directory
|
||||
# Copy locales
|
||||
COPY src/locales/ ./dist/locales/
|
||||
|
||||
# Copy scripts
|
||||
COPY scripts/startup.sh ./startup.sh
|
||||
RUN chmod +x ./startup.sh
|
||||
|
||||
# Create directories
|
||||
RUN mkdir -p uploads logs
|
||||
|
||||
# Create non-root user for security
|
||||
@@ -53,7 +62,7 @@ EXPOSE 3000
|
||||
|
||||
# Health check
|
||||
HEALTHCHECK --interval=30s --timeout=3s --start-period=5s --retries=3 \
|
||||
CMD node -e "require('http').get('http://localhost:3000/health', (res) => { process.exit(res.statusCode === 200 ? 0 : 1) })" || exit 1
|
||||
CMD wget --no-verbose --tries=1 --spider http://localhost:3000/health || exit 1
|
||||
|
||||
# Start the application
|
||||
CMD ["node", "dist/bot.js"]
|
||||
# Start the application with migration script
|
||||
CMD ["./startup.sh"]
|
||||
|
||||
107
Makefile
Normal file
107
Makefile
Normal file
@@ -0,0 +1,107 @@
|
||||
# Makefile для Telegram Tinder Bot
|
||||
|
||||
.PHONY: help install update run migrate fix-docker clean clear-interactions
|
||||
|
||||
# Значения по умолчанию
|
||||
DB_HOST ?= db
|
||||
DB_PORT ?= 5432
|
||||
DB_NAME ?= telegram_tinder_bot
|
||||
DB_USERNAME ?= postgres
|
||||
DB_PASSWORD ?= postgres
|
||||
|
||||
# Основные команды
|
||||
help:
|
||||
@echo "========== Telegram Tinder Bot Makefile =========="
|
||||
@echo "make install - Установка зависимостей"
|
||||
@echo "make update - Обновление кода из репозитория"
|
||||
@echo "make run - Запуск бота в контейнере"
|
||||
@echo "make migrate - Применение миграций базы данных"
|
||||
@echo "make fix-docker - Исправление проблем с Docker"
|
||||
@echo "make clear-interactions - Очистка матчей, свайпов и сообщений"
|
||||
@echo "make clean - Очистка и остановка контейнеров"
|
||||
|
||||
install:
|
||||
@echo "Установка зависимостей..."
|
||||
@if ! command -v docker &> /dev/null; then \
|
||||
echo "Установка Docker..."; \
|
||||
echo "Удаление конфликтующих пакетов..."; \
|
||||
sudo apt remove -y docker.io containerd runc 2>/dev/null || true; \
|
||||
sudo apt update; \
|
||||
sudo apt install -y apt-transport-https ca-certificates curl software-properties-common; \
|
||||
curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo gpg --dearmor -o /usr/share/keyrings/docker-archive-keyring.gpg; \
|
||||
echo "deb [arch=$$(dpkg --print-architecture) signed-by=/usr/share/keyrings/docker-archive-keyring.gpg] https://download.docker.com/linux/ubuntu $$(lsb_release -cs) stable" | sudo tee /etc/apt/sources.list.d/docker.list > /dev/null; \
|
||||
sudo apt update; \
|
||||
sudo apt install -y docker-ce docker-ce-cli containerd.io docker-compose-plugin; \
|
||||
sudo systemctl start docker; \
|
||||
sudo systemctl enable docker; \
|
||||
sudo usermod -aG docker $$USER; \
|
||||
echo "Docker установлен. Перелогиньтесь для применения группы docker."; \
|
||||
fi
|
||||
@if [ ! -f .env ]; then \
|
||||
echo "Создание .env файла..."; \
|
||||
cp .env.example .env 2>/dev/null || cp .env.production .env 2>/dev/null || echo "NODE_ENV=production" > .env; \
|
||||
echo "Пожалуйста, отредактируйте файл .env!"; \
|
||||
fi
|
||||
@mkdir -p logs uploads && chmod -R 777 logs uploads
|
||||
|
||||
update:
|
||||
@echo "Обновление кода..."
|
||||
@git fetch --all
|
||||
@git pull origin main || git pull origin master || echo "Не удалось обновить код"
|
||||
@if [ -f package.json ]; then npm ci || npm install; fi
|
||||
@echo "Пересборка контейнеров..."
|
||||
@docker compose down || docker-compose down || true
|
||||
@docker compose build || docker-compose build
|
||||
@docker compose up -d || docker-compose up -d
|
||||
@echo "Применение миграций к базе данных..."
|
||||
@sleep 5
|
||||
@make migrate
|
||||
@echo "✅ Обновление завершено! Бот перезапущен с новой версией."
|
||||
|
||||
run:
|
||||
@echo "Запуск бота..."
|
||||
@docker-compose down || true
|
||||
@make fix-docker
|
||||
@docker-compose build
|
||||
@docker-compose up -d
|
||||
@echo "Бот запущен! Для просмотра логов: docker-compose logs -f"
|
||||
|
||||
migrate:
|
||||
@echo "Применение миграций к базе данных..."
|
||||
@if [ ! -f .env ]; then \
|
||||
echo "❌ Файл .env не найден! Создайте его перед применением миграций."; \
|
||||
exit 1; \
|
||||
fi
|
||||
@. ./.env && export $$(cat .env | grep -v '^#' | xargs) && \
|
||||
echo "Подключение к БД: $$DB_HOST:$$DB_PORT/$$DB_NAME ($$DB_USERNAME)" && \
|
||||
echo "Создание расширения uuid-ossp..." && \
|
||||
PGPASSWORD="$$DB_PASSWORD" psql -h $$DB_HOST -p $$DB_PORT -U $$DB_USERNAME -d $$DB_NAME \
|
||||
-c "CREATE EXTENSION IF NOT EXISTS \"uuid-ossp\";" 2>/dev/null || true && \
|
||||
echo "Применение consolidated.sql..." && \
|
||||
PGPASSWORD="$$DB_PASSWORD" psql -h $$DB_HOST -p $$DB_PORT -U $$DB_USERNAME -d $$DB_NAME \
|
||||
-f sql/consolidated.sql 2>&1 | grep -E "(ERROR|CREATE|ALTER)" || true && \
|
||||
echo "Применение дополнительных миграций..." && \
|
||||
for sql_file in sql/add_*.sql; do \
|
||||
[ -f "$$sql_file" ] && echo " - Применение $$(basename $$sql_file)..." && \
|
||||
PGPASSWORD="$$DB_PASSWORD" psql -h $$DB_HOST -p $$DB_PORT -U $$DB_USERNAME -d $$DB_NAME \
|
||||
-f "$$sql_file" 2>&1 | grep -v "NOTICE" || true; \
|
||||
done && \
|
||||
echo "✅ Миграции применены успешно!"
|
||||
|
||||
fix-docker:
|
||||
@echo "Исправление Docker конфигурации..."
|
||||
@if [ -f Dockerfile ] && grep -q "RUN npm run build" Dockerfile; then \
|
||||
sed -i 's/RUN npm run build/RUN npm run build:linux/g' Dockerfile; \
|
||||
fi
|
||||
@docker rm -f postgres-tinder adminer-tinder telegram-tinder-bot 2>/dev/null || true
|
||||
@docker system prune -f --volumes >/dev/null 2>&1 || true
|
||||
|
||||
clear-interactions:
|
||||
@echo "Очистка взаимодействий пользователей..."
|
||||
@bash bin/clear_interactions.sh
|
||||
|
||||
clean:
|
||||
@echo "Очистка..."
|
||||
@docker-compose down || true
|
||||
@rm -rf temp_migrations node_modules/.cache
|
||||
@echo "Очистка завершена"
|
||||
43
README.md
43
README.md
@@ -268,8 +268,32 @@ npm run dev
|
||||
- Node.js 16+
|
||||
- PostgreSQL 12+
|
||||
- Telegram Bot Token (получить у [@BotFather](https://t.me/BotFather))
|
||||
- Docker и Docker Compose (опционально)
|
||||
|
||||
### 2. Установка
|
||||
### 2. Установка и запуск
|
||||
|
||||
#### С использованием стартовых скриптов (рекомендуется)
|
||||
|
||||
```bash
|
||||
# Клонировать репозиторий
|
||||
git clone <repository-url>
|
||||
cd telegram-tinder-bot
|
||||
|
||||
# На Windows:
|
||||
.\start.bat
|
||||
|
||||
# На Linux/macOS:
|
||||
chmod +x start.sh
|
||||
./start.sh
|
||||
```
|
||||
|
||||
Скрипт автоматически:
|
||||
- Проверит наличие файла .env и создаст его из шаблона при необходимости
|
||||
- Предложит выбор между запуском с локальной БД или подключением к внешней
|
||||
- Настроит все необходимые параметры окружения
|
||||
- Запустит контейнеры Docker
|
||||
|
||||
#### Без Docker
|
||||
|
||||
```bash
|
||||
# Клонировать репозиторий
|
||||
@@ -279,24 +303,17 @@ cd telegram-tinder-bot
|
||||
# Установить зависимости
|
||||
npm install
|
||||
|
||||
# Скомпилировать TypeScript
|
||||
npm run build
|
||||
```
|
||||
# Скопировать файл конфигурации
|
||||
cp .env.example .env
|
||||
# Отредактируйте файл .env и укажите свой TELEGRAM_BOT_TOKEN
|
||||
|
||||
### 3. Настройка базы данных
|
||||
|
||||
```bash
|
||||
# Создать базу данных PostgreSQL
|
||||
createdb telegram_tinder_bot
|
||||
|
||||
# Запустить миграции
|
||||
psql -d telegram_tinder_bot -f src/database/migrations/init.sql
|
||||
```
|
||||
npm run migrate:up
|
||||
|
||||
### 4. Запуск бота
|
||||
|
||||
```bash
|
||||
# Компиляция TypeScript
|
||||
# Скомпилировать TypeScript
|
||||
npm run build
|
||||
|
||||
# Запуск бота
|
||||
|
||||
49
additional_translations.json
Normal file
49
additional_translations.json
Normal file
@@ -0,0 +1,49 @@
|
||||
{
|
||||
"noProfile": {
|
||||
"message": "❌ У вас пока нет профиля.\\nСоздайте его для начала использования бота!",
|
||||
"createButton": "🚀 Создать профиль"
|
||||
},
|
||||
"noMatches": {
|
||||
"message": "💔 У вас пока нет матчей.\\n\\n🔍 Попробуйте просмотреть больше анкет!\\nИспользуйте /browse для поиска.",
|
||||
"browsing": "🔍 Найти еще"
|
||||
},
|
||||
"matches": {
|
||||
"title": "💕 Ваши матчи:",
|
||||
"openChats": "💬 Открыть чаты",
|
||||
"nativeChats": "📱 Нативные чаты",
|
||||
"findMore": "🔍 Найти еще",
|
||||
"cityNotSpecified": "Не указан"
|
||||
},
|
||||
"profileCreation": {
|
||||
"start": "👋 Давайте создадим ваш профиль!\\n\\n📝 Сначала напишите ваше имя:",
|
||||
"enterName": "❌ Пожалуйста, отправьте текстовое сообщение с вашим именем",
|
||||
"enterAge": "📅 Отлично! Теперь укажите ваш возраст:",
|
||||
"ageNotNumber": "❌ Пожалуйста, отправьте число",
|
||||
"ageInvalid": "❌ Возраст должен быть числом от 18 до 100",
|
||||
"enterCity": "📍 Прекрасно! В каком городе вы живете?",
|
||||
"cityText": "❌ Пожалуйста, отправьте название города",
|
||||
"enterBio": "📝 Теперь расскажите немного о себе (био):\\n\\n💡 Например: хобби, интересы, что ищете в отношениях и т.д.",
|
||||
"bioText": "❌ Пожалуйста, отправьте текстовое описание",
|
||||
"enterPhoto": "📸 Отлично! Теперь отправьте ваше фото:\\n\\n💡 Лучше использовать качественное фото лица",
|
||||
"photoRequired": "❌ Пожалуйста, отправьте фотографию",
|
||||
"error": "❌ Произошла ошибка. Попробуйте еще раз.",
|
||||
"createError": "❌ Ошибка при создании профиля. Попробуйте еще раз позже."
|
||||
},
|
||||
"profileView": {
|
||||
"cityNotSpecified": "Не указан",
|
||||
"bioNotSpecified": "Описание не указано",
|
||||
"editProfile": "✏️ Редактировать",
|
||||
"managePhotos": "📸 Фото",
|
||||
"startBrowsing": "🔍 Начать поиск",
|
||||
"backToBrowsing": "👈 Назад"
|
||||
},
|
||||
"browsing": {
|
||||
"noMoreProfiles": "🎉 Вы просмотрели всех доступных кандидатов!\\n\\n⏰ Попробуйте позже - возможно появятся новые анкеты!",
|
||||
"needProfile": "❌ Сначала создайте профиль!\\nИспользуйте команду /start",
|
||||
"cityNotSpecified": "Не указан"
|
||||
},
|
||||
"general": {
|
||||
"greeting": "Привет! 👋\\n\\nИспользуйте команды для навигации:\\n/start - Главное меню\\n/help - Справка\\n/profile - Мой профиль\\n/browse - Поиск анкет",
|
||||
"photoManagement": "📸 Для управления фотографиями используйте:"
|
||||
}
|
||||
}
|
||||
85
bin/CLEAR_INTERACTIONS_README.md
Normal file
85
bin/CLEAR_INTERACTIONS_README.md
Normal file
@@ -0,0 +1,85 @@
|
||||
# Скрипт очистки взаимодействий пользователей
|
||||
|
||||
## Описание
|
||||
|
||||
Этот скрипт удаляет все взаимодействия между пользователями, оставляя только сами профили. Полезно для тестирования или сброса состояния приложения.
|
||||
|
||||
## Что удаляется
|
||||
|
||||
- ✅ **Messages** - все сообщения в чатах
|
||||
- ✅ **Matches** - все матчи между пользователями
|
||||
- ✅ **Profile Views** - все просмотры профилей
|
||||
- ✅ **Swipes** - все свайпы (лайки, дизлайки, суперлайки)
|
||||
- ✅ **Notifications** - все уведомления
|
||||
|
||||
## Что НЕ удаляется
|
||||
|
||||
- ❌ **Users** - пользователи остаются
|
||||
- ❌ **Profiles** - профили пользователей остаются
|
||||
|
||||
## Использование
|
||||
|
||||
### Способ 1: Через Makefile (рекомендуется)
|
||||
|
||||
```bash
|
||||
make clear-interactions
|
||||
```
|
||||
|
||||
### Способ 2: Прямой запуск скрипта
|
||||
|
||||
```bash
|
||||
./bin/clear_interactions.sh
|
||||
```
|
||||
|
||||
### Способ 3: Прямое выполнение SQL
|
||||
|
||||
```bash
|
||||
PGPASSWORD='your_password' psql -h host -U username -d database -f sql/clear_interactions.sql
|
||||
```
|
||||
|
||||
## Подтверждение
|
||||
|
||||
Скрипт запросит подтверждение перед выполнением:
|
||||
|
||||
```
|
||||
Вы уверены, что хотите продолжить? (yes/no):
|
||||
```
|
||||
|
||||
Введите `yes` для продолжения или `no` для отмены.
|
||||
|
||||
## Требования
|
||||
|
||||
- Файл `.env` должен существовать и содержать переменные:
|
||||
- `DB_HOST`
|
||||
- `DB_PORT`
|
||||
- `DB_NAME`
|
||||
- `DB_USERNAME`
|
||||
- `DB_PASSWORD`
|
||||
|
||||
## Вывод
|
||||
|
||||
После успешного выполнения скрипт покажет статистику:
|
||||
|
||||
```
|
||||
table_name | remaining_records
|
||||
-------------------+-------------------
|
||||
messages | 0
|
||||
matches | 0
|
||||
profile_views | 0
|
||||
swipes | 0
|
||||
notifications | 0
|
||||
users | 2
|
||||
profiles | 2
|
||||
```
|
||||
|
||||
## Безопасность
|
||||
|
||||
- Скрипт использует транзакцию (BEGIN/COMMIT) для безопасности
|
||||
- Все операции выполняются атомарно
|
||||
- В случае ошибки изменения откатываются
|
||||
|
||||
## Примечания
|
||||
|
||||
- ⚠️ **Необратимая операция!** Удаленные данные нельзя восстановить
|
||||
- 💡 Рекомендуется делать резервную копию БД перед запуском
|
||||
- 🔒 Убедитесь, что у вас есть права на удаление данных в БД
|
||||
54
bin/QUICK_FIX.md
Normal file
54
bin/QUICK_FIX.md
Normal file
@@ -0,0 +1,54 @@
|
||||
# Быстрое исправление проблем с миграциями
|
||||
|
||||
## Проблема
|
||||
При запуске миграций возникают ошибки с TypeScript-файлами и проблемы с модульными разрешениями.
|
||||
|
||||
## Быстрое решение
|
||||
|
||||
1. **Примените прямые SQL-миграции (рекомендуемый способ)**:
|
||||
```bash
|
||||
chmod +x bin/apply_direct_sql.sh
|
||||
./bin/apply_direct_sql.sh
|
||||
```
|
||||
Этот скрипт создаст и применит консолидированную SQL-миграцию, которая создаст все необходимые таблицы.
|
||||
|
||||
2. **Создайте консолидированную JS-миграцию**:
|
||||
```bash
|
||||
chmod +x bin/create_consolidated_migration.sh
|
||||
./bin/create_consolidated_migration.sh
|
||||
```
|
||||
Затем примените её:
|
||||
```bash
|
||||
DATABASE_URL="postgres://$DB_USERNAME:$DB_PASSWORD@$DB_HOST:$DB_PORT/$DB_NAME" npx node-pg-migrate up
|
||||
```
|
||||
|
||||
## Проверка результата
|
||||
|
||||
После выполнения миграций проверьте наличие таблиц в базе данных:
|
||||
|
||||
```bash
|
||||
export PGPASSWORD=$DB_PASSWORD
|
||||
psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -c "\dt"
|
||||
```
|
||||
|
||||
## Если проблемы сохраняются
|
||||
|
||||
1. **Проверьте доступность базы данных**:
|
||||
```bash
|
||||
export PGPASSWORD=$DB_PASSWORD
|
||||
psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -c "SELECT 1"
|
||||
```
|
||||
|
||||
2. **Проверьте правильность переменных окружения**:
|
||||
```bash
|
||||
echo "DB_HOST: $DB_HOST"
|
||||
echo "DB_PORT: $DB_PORT"
|
||||
echo "DB_NAME: $DB_NAME"
|
||||
echo "DB_USERNAME: $DB_USERNAME"
|
||||
```
|
||||
|
||||
3. **Установите PostgreSQL-клиент**, если он отсутствует:
|
||||
```bash
|
||||
apt-get update
|
||||
apt-get install -y postgresql-client
|
||||
```
|
||||
@@ -2,6 +2,24 @@
|
||||
|
||||
Этот документ описывает процесс автоматического обновления бота с помощью созданных скриптов.
|
||||
|
||||
## Доступные скрипты
|
||||
|
||||
### apply_all_patches.sh
|
||||
Применяет все SQL патчи к базе данных в правильном порядке:
|
||||
- Основная схема (consolidated.sql)
|
||||
- Исправление триггера looking_for
|
||||
- Добавление колонок job и state
|
||||
|
||||
```bash
|
||||
./bin/apply_all_patches.sh
|
||||
```
|
||||
|
||||
### apply_migrations.sh
|
||||
Применяет Node.js миграции через node-pg-migrate.
|
||||
|
||||
### apply_direct_sql.sh
|
||||
Применяет SQL файлы напрямую через psql.
|
||||
|
||||
## Скрипт обновления
|
||||
|
||||
Скрипт обновления выполняет следующие действия:
|
||||
|
||||
76
bin/apply_all_patches.sh
Executable file
76
bin/apply_all_patches.sh
Executable file
@@ -0,0 +1,76 @@
|
||||
#!/bin/bash
|
||||
# Применение всех патчей базы данных
|
||||
# Использование: ./bin/apply_all_patches.sh
|
||||
|
||||
set -e # Остановка при ошибке
|
||||
|
||||
# Загрузка переменных окружения
|
||||
if [ -f .env ]; then
|
||||
source .env
|
||||
else
|
||||
echo "❌ Файл .env не найден!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Проверка обязательных переменных
|
||||
if [ -z "$DB_HOST" ] || [ -z "$DB_PORT" ] || [ -z "$DB_NAME" ] || [ -z "$DB_USERNAME" ] || [ -z "$DB_PASSWORD" ]; then
|
||||
echo "❌ Не все переменные DB_* заданы в .env"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo "🔧 Применение патчей к базе данных..."
|
||||
echo "📍 Сервер: $DB_HOST:$DB_PORT"
|
||||
echo "📂 База данных: $DB_NAME"
|
||||
echo ""
|
||||
|
||||
# Функция применения патча
|
||||
apply_patch() {
|
||||
local patch_file=$1
|
||||
local description=$2
|
||||
|
||||
if [ ! -f "$patch_file" ]; then
|
||||
echo "⚠️ Патч $patch_file не найден, пропуск..."
|
||||
return
|
||||
fi
|
||||
|
||||
echo "📝 Применение: $description"
|
||||
if PGPASSWORD="$DB_PASSWORD" psql -h "$DB_HOST" -p "$DB_PORT" -U "$DB_USERNAME" -d "$DB_NAME" -f "$patch_file" > /dev/null 2>&1; then
|
||||
echo "✅ Патч применен: $patch_file"
|
||||
else
|
||||
echo "⚠️ Ошибка при применении: $patch_file (возможно уже применен)"
|
||||
fi
|
||||
echo ""
|
||||
}
|
||||
|
||||
# Применение патчей в правильном порядке
|
||||
apply_patch "sql/consolidated.sql" "Основная схема БД (16 таблиц)"
|
||||
apply_patch "sql/fix_looking_for_column.sql" "Исправление триггера и колонки looking_for"
|
||||
apply_patch "sql/add_job_and_state_columns.sql" "Добавление колонок job и state"
|
||||
|
||||
echo "🎉 Все патчи обработаны!"
|
||||
echo ""
|
||||
echo "🔍 Проверка применения патчей..."
|
||||
|
||||
# Проверка критичных колонок
|
||||
PGPASSWORD="$DB_PASSWORD" psql -h "$DB_HOST" -p "$DB_PORT" -U "$DB_USERNAME" -d "$DB_NAME" << 'EOF'
|
||||
SELECT
|
||||
CASE
|
||||
WHEN EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'profiles' AND column_name = 'job')
|
||||
THEN '✅ profiles.job существует'
|
||||
ELSE '❌ profiles.job НЕ НАЙДЕНА'
|
||||
END as status_job,
|
||||
CASE
|
||||
WHEN EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'users' AND column_name = 'state')
|
||||
THEN '✅ users.state существует'
|
||||
ELSE '❌ users.state НЕ НАЙДЕНА'
|
||||
END as status_state,
|
||||
CASE
|
||||
WHEN EXISTS (SELECT 1 FROM information_schema.columns WHERE table_name = 'profiles' AND column_name = 'interested_in')
|
||||
THEN '✅ profiles.interested_in существует'
|
||||
ELSE '❌ profiles.interested_in НЕ НАЙДЕНА'
|
||||
END as status_interested_in;
|
||||
EOF
|
||||
|
||||
echo ""
|
||||
echo "✅ Готово! Теперь можно перезапустить бота:"
|
||||
echo " docker compose restart bot"
|
||||
208
bin/apply_direct_sql.sh
Normal file
208
bin/apply_direct_sql.sh
Normal file
@@ -0,0 +1,208 @@
|
||||
#!/bin/bash
|
||||
# apply_direct_sql.sh - Прямое выполнение SQL-миграций с помощью psql
|
||||
|
||||
echo "🚀 Прямое выполнение SQL-миграций..."
|
||||
|
||||
# Загрузка переменных окружения из .env
|
||||
if [ -f .env ]; then
|
||||
echo "📝 Загрузка переменных окружения из .env..."
|
||||
set -o allexport
|
||||
source .env
|
||||
set +o allexport
|
||||
else
|
||||
echo "⚠️ Файл .env не найден, используем значения по умолчанию"
|
||||
export DB_HOST="localhost"
|
||||
export DB_PORT="5432"
|
||||
export DB_NAME="telegram_tinder_bot"
|
||||
export DB_USERNAME="postgres"
|
||||
export DB_PASSWORD="postgres"
|
||||
fi
|
||||
|
||||
# Создаем консолидированный SQL-файл
|
||||
echo "📝 Создание консолидированного SQL-файла..."
|
||||
consolidated_sql="consolidated_migration.sql"
|
||||
|
||||
cat > "$consolidated_sql" << EOL
|
||||
-- Консолидированная миграция для Telegram Tinder Bot
|
||||
-- Создана автоматически: $(date)
|
||||
|
||||
-- Создаем таблицу migrations, если её еще нет
|
||||
CREATE TABLE IF NOT EXISTS migrations (
|
||||
id SERIAL PRIMARY KEY,
|
||||
name VARCHAR(255) NOT NULL,
|
||||
executed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
-- Основная структура базы данных
|
||||
|
||||
-- Таблица пользователей
|
||||
CREATE TABLE IF NOT EXISTS users (
|
||||
id BIGINT PRIMARY KEY,
|
||||
username VARCHAR(255),
|
||||
first_name VARCHAR(255),
|
||||
last_name VARCHAR(255),
|
||||
language_code VARCHAR(10),
|
||||
is_bot BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
state VARCHAR(50) DEFAULT 'START',
|
||||
state_data JSONB,
|
||||
gender VARCHAR(10),
|
||||
looking_for VARCHAR(10),
|
||||
bio TEXT,
|
||||
age INTEGER,
|
||||
location VARCHAR(255),
|
||||
photos JSONB DEFAULT '[]'::jsonb,
|
||||
interests TEXT[],
|
||||
premium BOOLEAN DEFAULT FALSE,
|
||||
premium_expires_at TIMESTAMP
|
||||
);
|
||||
|
||||
-- Таблица профилей
|
||||
CREATE TABLE IF NOT EXISTS profiles (
|
||||
id SERIAL PRIMARY KEY,
|
||||
user_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
name VARCHAR(255),
|
||||
age INTEGER,
|
||||
gender VARCHAR(10),
|
||||
bio TEXT,
|
||||
photos JSONB DEFAULT '[]'::jsonb,
|
||||
interests TEXT[],
|
||||
location VARCHAR(255),
|
||||
religion VARCHAR(50),
|
||||
education VARCHAR(255),
|
||||
job VARCHAR(255),
|
||||
height INTEGER,
|
||||
smoking VARCHAR(50),
|
||||
drinking VARCHAR(50),
|
||||
looking_for VARCHAR(10),
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
-- Таблица лайков
|
||||
CREATE TABLE IF NOT EXISTS likes (
|
||||
id SERIAL PRIMARY KEY,
|
||||
from_user_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
to_user_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
is_like BOOLEAN NOT NULL,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
UNIQUE(from_user_id, to_user_id)
|
||||
);
|
||||
|
||||
-- Таблица матчей
|
||||
CREATE TABLE IF NOT EXISTS matches (
|
||||
id SERIAL PRIMARY KEY,
|
||||
user1_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
user2_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
active BOOLEAN DEFAULT TRUE,
|
||||
UNIQUE(user1_id, user2_id)
|
||||
);
|
||||
|
||||
-- Таблица сообщений
|
||||
CREATE TABLE IF NOT EXISTS messages (
|
||||
id SERIAL PRIMARY KEY,
|
||||
match_id INTEGER REFERENCES matches(id) ON DELETE CASCADE,
|
||||
sender_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
message_text TEXT,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
-- Таблица просмотров профилей
|
||||
CREATE TABLE IF NOT EXISTS profile_views (
|
||||
id SERIAL PRIMARY KEY,
|
||||
viewer_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
viewed_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
viewed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
UNIQUE(viewer_id, viewed_id)
|
||||
);
|
||||
|
||||
-- Таблица уведомлений
|
||||
CREATE TABLE IF NOT EXISTS notifications (
|
||||
id SERIAL PRIMARY KEY,
|
||||
user_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
data JSONB,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
scheduled_for TIMESTAMP,
|
||||
processed BOOLEAN DEFAULT FALSE
|
||||
);
|
||||
|
||||
-- Индексы для оптимизации запросов
|
||||
|
||||
-- Индексы для таблицы пользователей
|
||||
CREATE INDEX IF NOT EXISTS users_username_idx ON users(username);
|
||||
CREATE INDEX IF NOT EXISTS users_gender_idx ON users(gender);
|
||||
CREATE INDEX IF NOT EXISTS users_looking_for_idx ON users(looking_for);
|
||||
CREATE INDEX IF NOT EXISTS users_premium_idx ON users(premium);
|
||||
|
||||
-- Индексы для таблицы лайков
|
||||
CREATE INDEX IF NOT EXISTS likes_from_user_id_idx ON likes(from_user_id);
|
||||
CREATE INDEX IF NOT EXISTS likes_to_user_id_idx ON likes(to_user_id);
|
||||
CREATE INDEX IF NOT EXISTS likes_is_like_idx ON likes(is_like);
|
||||
|
||||
-- Индексы для таблицы матчей
|
||||
CREATE INDEX IF NOT EXISTS matches_user1_id_idx ON matches(user1_id);
|
||||
CREATE INDEX IF NOT EXISTS matches_user2_id_idx ON matches(user2_id);
|
||||
CREATE INDEX IF NOT EXISTS matches_active_idx ON matches(active);
|
||||
|
||||
-- Индексы для таблицы сообщений
|
||||
CREATE INDEX IF NOT EXISTS messages_match_id_idx ON messages(match_id);
|
||||
CREATE INDEX IF NOT EXISTS messages_sender_id_idx ON messages(sender_id);
|
||||
|
||||
-- Индексы для таблицы профилей
|
||||
CREATE INDEX IF NOT EXISTS profiles_user_id_idx ON profiles(user_id);
|
||||
CREATE INDEX IF NOT EXISTS profiles_gender_idx ON profiles(gender);
|
||||
CREATE INDEX IF NOT EXISTS profiles_looking_for_idx ON profiles(looking_for);
|
||||
|
||||
-- Индексы для таблицы просмотров профилей
|
||||
CREATE INDEX IF NOT EXISTS profile_views_viewer_id_idx ON profile_views(viewer_id);
|
||||
CREATE INDEX IF NOT EXISTS profile_views_viewed_id_idx ON profile_views(viewed_id);
|
||||
|
||||
-- Индексы для таблицы уведомлений
|
||||
CREATE INDEX IF NOT EXISTS notifications_user_id_idx ON notifications(user_id);
|
||||
CREATE INDEX IF NOT EXISTS notifications_scheduled_for_idx ON notifications(scheduled_for);
|
||||
CREATE INDEX IF NOT EXISTS notifications_processed_idx ON notifications(processed);
|
||||
|
||||
-- Запись о выполнении миграции
|
||||
INSERT INTO migrations (name) VALUES ('consolidated_migration.sql')
|
||||
ON CONFLICT DO NOTHING;
|
||||
EOL
|
||||
|
||||
echo "✅ Консолидированный SQL-файл создан: $consolidated_sql"
|
||||
|
||||
# Вывод информации о подключении
|
||||
echo "🔍 Используемые параметры подключения:"
|
||||
echo "DB_HOST: $DB_HOST"
|
||||
echo "DB_PORT: $DB_PORT"
|
||||
echo "DB_NAME: $DB_NAME"
|
||||
echo "DB_USERNAME: $DB_USERNAME"
|
||||
echo "DB_PASSWORD: ********"
|
||||
|
||||
# Проверка наличия psql
|
||||
if command -v psql >/dev/null; then
|
||||
echo "✅ Найдена команда psql, продолжаем..."
|
||||
else
|
||||
echo "⚠️ Команда psql не найдена, установите PostgreSQL клиент:"
|
||||
echo "apt-get update && apt-get install -y postgresql-client"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Применение миграции
|
||||
echo "🔄 Применение консолидированной миграции..."
|
||||
export PGPASSWORD=$DB_PASSWORD
|
||||
psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -f "$consolidated_sql"
|
||||
|
||||
# Проверка результата
|
||||
if [ $? -eq 0 ]; then
|
||||
echo "✅ Миграция успешно применена!"
|
||||
else
|
||||
echo "❌ Ошибка при применении миграции!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Удаление временного файла
|
||||
rm -f "$consolidated_sql"
|
||||
|
||||
echo "🚀 Миграция базы данных завершена!"
|
||||
75
bin/apply_migrations.sh
Normal file
75
bin/apply_migrations.sh
Normal file
@@ -0,0 +1,75 @@
|
||||
#!/bin/bash
|
||||
# apply_migrations.sh - Скрипт для ручного применения миграций
|
||||
|
||||
echo "🔄 Ручное применение миграций базы данных..."
|
||||
|
||||
# Загрузка переменных окружения из .env
|
||||
if [ -f .env ]; then
|
||||
echo "📝 Загрузка переменных окружения из .env..."
|
||||
export $(grep -v '^#' .env | xargs)
|
||||
else
|
||||
echo "⚠️ Файл .env не найден, используем значения по умолчанию"
|
||||
export DB_HOST="localhost"
|
||||
export DB_PORT="5432"
|
||||
export DB_NAME="telegram_tinder_bot"
|
||||
export DB_USERNAME="postgres"
|
||||
export DB_PASSWORD="postgres"
|
||||
fi
|
||||
|
||||
# Проверка на существование директории миграций
|
||||
if [ ! -d "migrations" ] && [ ! -d "src/database/migrations" ]; then
|
||||
echo "❌ Не найдены директории с миграциями!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Вывод информации о подключении
|
||||
echo "🔍 Используемые параметры подключения:"
|
||||
echo "DB_HOST: $DB_HOST"
|
||||
echo "DB_PORT: $DB_PORT"
|
||||
echo "DB_NAME: $DB_NAME"
|
||||
echo "DB_USERNAME: $DB_USERNAME"
|
||||
echo "DB_PASSWORD: ********"
|
||||
|
||||
# Проверка подключения к базе данных
|
||||
echo "🔍 Проверка подключения к базе данных..."
|
||||
if command -v pg_isready >/dev/null; then
|
||||
pg_isready -h $DB_HOST -p $DB_PORT -U $DB_USERNAME
|
||||
if [ $? -ne 0 ]; then
|
||||
echo "❌ Не удалось подключиться к базе данных!"
|
||||
exit 1
|
||||
fi
|
||||
else
|
||||
echo "⚠️ Утилита pg_isready не найдена, пропускаем проверку"
|
||||
fi
|
||||
|
||||
# Копирование миграций JS в отдельную директорию
|
||||
echo "📂 Копирование только JS-миграций во временную директорию..."
|
||||
mkdir -p temp_migrations
|
||||
find migrations -name "*.js" -exec cp {} temp_migrations/ \;
|
||||
|
||||
# Применение миграций
|
||||
echo "🔄 Применение миграций с помощью node-pg-migrate..."
|
||||
DATABASE_URL="postgres://$DB_USERNAME:$DB_PASSWORD@$DB_HOST:$DB_PORT/$DB_NAME" npx node-pg-migrate up --migrations-dir=temp_migrations
|
||||
|
||||
# Проверка результата
|
||||
if [ $? -eq 0 ]; then
|
||||
echo "✅ Миграции успешно применены!"
|
||||
else
|
||||
echo "❌ Ошибка при применении миграций!"
|
||||
echo "⚠️ Пытаемся применить миграции из других источников..."
|
||||
|
||||
# Попробуем применить SQL-миграции напрямую
|
||||
if [ -d "src/database/migrations" ]; then
|
||||
echo "📂 Найдены SQL-миграции. Пытаемся применить их напрямую..."
|
||||
for sql_file in src/database/migrations/*.sql; do
|
||||
if [ -f "$sql_file" ]; then
|
||||
echo "🔄 Применение миграции $sql_file..."
|
||||
PGPASSWORD=$DB_PASSWORD psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -f "$sql_file" || echo "⚠️ Ошибка при применении $sql_file"
|
||||
fi
|
||||
done
|
||||
fi
|
||||
fi
|
||||
|
||||
# Очистка временных файлов
|
||||
echo "🧹 Очистка временных файлов..."
|
||||
rm -rf temp_migrations
|
||||
54
bin/backup_db.sh
Normal file
54
bin/backup_db.sh
Normal file
@@ -0,0 +1,54 @@
|
||||
#!/bin/bash
|
||||
# backup_db.sh - Script for backing up the PostgreSQL database
|
||||
|
||||
echo "📦 Backing up PostgreSQL database..."
|
||||
|
||||
# Default backup directory
|
||||
BACKUP_DIR="${BACKUP_DIR:-/var/backups/tg_tinder_bot}"
|
||||
BACKUP_FILENAME="tg_tinder_bot_$(date +%Y%m%d_%H%M%S).sql"
|
||||
BACKUP_PATH="$BACKUP_DIR/$BACKUP_FILENAME"
|
||||
|
||||
# Create backup directory if it doesn't exist
|
||||
mkdir -p "$BACKUP_DIR"
|
||||
|
||||
# Check if running in docker-compose environment
|
||||
if [ -f /.dockerenv ] || [ -f /proc/self/cgroup ] && grep -q docker /proc/self/cgroup; then
|
||||
echo "🐳 Running in Docker environment, using docker-compose exec..."
|
||||
docker-compose exec -T db pg_dump -U postgres telegram_tinder_bot > "$BACKUP_PATH"
|
||||
else
|
||||
# Check if PGPASSWORD is set in environment
|
||||
if [ -z "$PGPASSWORD" ]; then
|
||||
# If .env file exists, try to get password from there
|
||||
if [ -f .env ]; then
|
||||
DB_PASSWORD=$(grep DB_PASSWORD .env | cut -d '=' -f2)
|
||||
export PGPASSWORD="$DB_PASSWORD"
|
||||
else
|
||||
echo "⚠️ No DB_PASSWORD found in environment or .env file."
|
||||
echo "Please enter PostgreSQL password:"
|
||||
read -s PGPASSWORD
|
||||
export PGPASSWORD
|
||||
fi
|
||||
fi
|
||||
|
||||
echo "💾 Backing up database to $BACKUP_PATH..."
|
||||
pg_dump -h localhost -U postgres -d telegram_tinder_bot > "$BACKUP_PATH"
|
||||
fi
|
||||
|
||||
# Check if backup was successful
|
||||
if [ $? -eq 0 ]; then
|
||||
echo "✅ Backup completed successfully: $BACKUP_PATH"
|
||||
echo "📊 Backup size: $(du -h $BACKUP_PATH | cut -f1)"
|
||||
|
||||
# Compress the backup
|
||||
gzip -f "$BACKUP_PATH"
|
||||
echo "🗜️ Compressed backup: $BACKUP_PATH.gz"
|
||||
|
||||
# Keep only the last 7 backups
|
||||
echo "🧹 Cleaning up old backups..."
|
||||
find "$BACKUP_DIR" -name "tg_tinder_bot_*.sql.gz" -type f -mtime +7 -delete
|
||||
|
||||
echo "🎉 Backup process completed!"
|
||||
else
|
||||
echo "❌ Backup failed!"
|
||||
exit 1
|
||||
fi
|
||||
71
bin/clear_interactions.sh
Executable file
71
bin/clear_interactions.sh
Executable file
@@ -0,0 +1,71 @@
|
||||
#!/bin/bash
|
||||
|
||||
# Скрипт для очистки всех взаимодействий между пользователями
|
||||
# Использование: ./clear_interactions.sh
|
||||
|
||||
set -e
|
||||
|
||||
# Цвета для вывода
|
||||
RED='\033[0;31m'
|
||||
GREEN='\033[0;32m'
|
||||
YELLOW='\033[1;33m'
|
||||
NC='\033[0m' # No Color
|
||||
|
||||
echo -e "${YELLOW}================================================${NC}"
|
||||
echo -e "${YELLOW} Скрипт очистки взаимодействий пользователей${NC}"
|
||||
echo -e "${YELLOW}================================================${NC}"
|
||||
echo ""
|
||||
echo -e "${RED}ВНИМАНИЕ!${NC} Будут удалены:"
|
||||
echo " - Все сообщения (messages)"
|
||||
echo " - Все матчи (matches)"
|
||||
echo " - Все просмотры профилей (profile_views)"
|
||||
echo " - Все свайпы (swipes)"
|
||||
echo " - Все уведомления (notifications)"
|
||||
echo ""
|
||||
echo -e "Профили пользователей ${GREEN}НЕ${NC} будут удалены."
|
||||
echo ""
|
||||
|
||||
# Запрос подтверждения
|
||||
read -p "Вы уверены, что хотите продолжить? (yes/no): " confirmation
|
||||
|
||||
if [ "$confirmation" != "yes" ]; then
|
||||
echo -e "${YELLOW}Операция отменена.${NC}"
|
||||
exit 0
|
||||
fi
|
||||
|
||||
echo ""
|
||||
echo -e "${YELLOW}Загрузка переменных окружения...${NC}"
|
||||
|
||||
# Загрузка переменных из .env файла
|
||||
if [ -f .env ]; then
|
||||
export $(cat .env | grep -v '^#' | xargs)
|
||||
else
|
||||
echo -e "${RED}Ошибка: файл .env не найден!${NC}"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Проверка наличия необходимых переменных
|
||||
if [ -z "$DB_HOST" ] || [ -z "$DB_PORT" ] || [ -z "$DB_NAME" ] || [ -z "$DB_USERNAME" ] || [ -z "$DB_PASSWORD" ]; then
|
||||
echo -e "${RED}Ошибка: не все переменные БД определены в .env${NC}"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo -e "${GREEN}Переменные загружены успешно.${NC}"
|
||||
echo ""
|
||||
echo -e "${YELLOW}Выполнение SQL скрипта...${NC}"
|
||||
|
||||
# Выполнение SQL скрипта
|
||||
PGPASSWORD="$DB_PASSWORD" psql -h "$DB_HOST" -p "$DB_PORT" -U "$DB_USERNAME" -d "$DB_NAME" -f sql/clear_interactions.sql
|
||||
|
||||
if [ $? -eq 0 ]; then
|
||||
echo ""
|
||||
echo -e "${GREEN}================================================${NC}"
|
||||
echo -e "${GREEN} ✅ Очистка выполнена успешно!${NC}"
|
||||
echo -e "${GREEN}================================================${NC}"
|
||||
else
|
||||
echo ""
|
||||
echo -e "${RED}================================================${NC}"
|
||||
echo -e "${RED} ❌ Ошибка при выполнении очистки!${NC}"
|
||||
echo -e "${RED}================================================${NC}"
|
||||
exit 1
|
||||
fi
|
||||
52
bin/compile_ts_migrations.sh
Normal file
52
bin/compile_ts_migrations.sh
Normal file
@@ -0,0 +1,52 @@
|
||||
#!/bin/bash
|
||||
# compile_ts_migrations.sh - Скрипт для компиляции TS миграций в JS
|
||||
|
||||
echo "🔄 Компиляция TypeScript миграций в JavaScript..."
|
||||
|
||||
# Проверка наличия TypeScript файлов
|
||||
if [ ! -f "migrations/*.ts" ] && [ ! -d "node_modules/typescript" ]; then
|
||||
echo "📦 Установка TypeScript..."
|
||||
npm install --no-save typescript
|
||||
fi
|
||||
|
||||
# Создание временного tsconfig для миграций
|
||||
echo "📝 Создание временного tsconfig.json для миграций..."
|
||||
cat > migrations/tsconfig.json << EOL
|
||||
{
|
||||
"compilerOptions": {
|
||||
"target": "es2020",
|
||||
"module": "commonjs",
|
||||
"esModuleInterop": true,
|
||||
"forceConsistentCasingInFileNames": true,
|
||||
"moduleResolution": "node",
|
||||
"resolveJsonModule": true,
|
||||
"skipLibCheck": true,
|
||||
"strict": false,
|
||||
"outDir": "../temp_migrations",
|
||||
"baseUrl": "..",
|
||||
"paths": {
|
||||
"*": ["node_modules/*"]
|
||||
}
|
||||
},
|
||||
"include": ["./*.ts"]
|
||||
}
|
||||
EOL
|
||||
|
||||
# Компиляция TS файлов
|
||||
echo "🔄 Компиляция TypeScript миграций..."
|
||||
npx tsc -p migrations/tsconfig.json
|
||||
|
||||
# Подтверждение
|
||||
if [ $? -eq 0 ]; then
|
||||
echo "✅ Миграции успешно скомпилированы в директорию temp_migrations/"
|
||||
|
||||
# Проверка, были ли созданы файлы
|
||||
file_count=$(find temp_migrations -name "*.js" | wc -l)
|
||||
echo "📊 Скомпилировано файлов: $file_count"
|
||||
else
|
||||
echo "❌ Ошибка при компиляции миграций!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Очистка временных файлов
|
||||
rm migrations/tsconfig.json
|
||||
188
bin/create_consolidated_migration.sh
Normal file
188
bin/create_consolidated_migration.sh
Normal file
@@ -0,0 +1,188 @@
|
||||
#!/bin/bash
|
||||
# create_consolidated_migration.sh - Создание консолидированной миграции из всех источников
|
||||
|
||||
echo "🚀 Создание консолидированной миграции..."
|
||||
|
||||
# Создаем каталог для миграций если его нет
|
||||
mkdir -p migrations
|
||||
|
||||
# Текущее время для имени файла
|
||||
timestamp=$(date +%s)
|
||||
|
||||
# Путь к консолидированной миграции
|
||||
consolidated_file="migrations/${timestamp}_consolidated_migration.js"
|
||||
|
||||
echo "📝 Создание файла $consolidated_file..."
|
||||
|
||||
# Создаем JS-миграцию
|
||||
cat > "$consolidated_file" << EOL
|
||||
/* eslint-disable camelcase */
|
||||
|
||||
exports.shorthands = undefined;
|
||||
|
||||
exports.up = pgm => {
|
||||
// Консолидированная миграция, созданная автоматически
|
||||
|
||||
// Создаем таблицу migrations, если её ещё нет
|
||||
pgm.sql(\`
|
||||
CREATE TABLE IF NOT EXISTS migrations (
|
||||
id SERIAL PRIMARY KEY,
|
||||
name VARCHAR(255) NOT NULL,
|
||||
executed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
\`);
|
||||
|
||||
// Создаем основную структуру базы данных
|
||||
pgm.sql(\`
|
||||
-- Таблица пользователей
|
||||
CREATE TABLE IF NOT EXISTS users (
|
||||
id BIGINT PRIMARY KEY,
|
||||
username VARCHAR(255),
|
||||
first_name VARCHAR(255),
|
||||
last_name VARCHAR(255),
|
||||
language_code VARCHAR(10),
|
||||
is_bot BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
state VARCHAR(50) DEFAULT 'START',
|
||||
state_data JSONB,
|
||||
gender VARCHAR(10),
|
||||
looking_for VARCHAR(10),
|
||||
bio TEXT,
|
||||
age INTEGER,
|
||||
location VARCHAR(255),
|
||||
photos JSONB DEFAULT '[]'::jsonb,
|
||||
interests TEXT[],
|
||||
premium BOOLEAN DEFAULT FALSE,
|
||||
premium_expires_at TIMESTAMP
|
||||
);
|
||||
|
||||
-- Таблица профилей
|
||||
CREATE TABLE IF NOT EXISTS profiles (
|
||||
id SERIAL PRIMARY KEY,
|
||||
user_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
name VARCHAR(255),
|
||||
age INTEGER,
|
||||
gender VARCHAR(10),
|
||||
bio TEXT,
|
||||
photos JSONB DEFAULT '[]'::jsonb,
|
||||
interests TEXT[],
|
||||
location VARCHAR(255),
|
||||
religion VARCHAR(50),
|
||||
education VARCHAR(255),
|
||||
job VARCHAR(255),
|
||||
height INTEGER,
|
||||
smoking VARCHAR(50),
|
||||
drinking VARCHAR(50),
|
||||
looking_for VARCHAR(10),
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
-- Таблица лайков
|
||||
CREATE TABLE IF NOT EXISTS likes (
|
||||
id SERIAL PRIMARY KEY,
|
||||
from_user_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
to_user_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
is_like BOOLEAN NOT NULL,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
UNIQUE(from_user_id, to_user_id)
|
||||
);
|
||||
|
||||
-- Таблица матчей
|
||||
CREATE TABLE IF NOT EXISTS matches (
|
||||
id SERIAL PRIMARY KEY,
|
||||
user1_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
user2_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
active BOOLEAN DEFAULT TRUE,
|
||||
UNIQUE(user1_id, user2_id)
|
||||
);
|
||||
|
||||
-- Таблица сообщений
|
||||
CREATE TABLE IF NOT EXISTS messages (
|
||||
id SERIAL PRIMARY KEY,
|
||||
match_id INTEGER REFERENCES matches(id) ON DELETE CASCADE,
|
||||
sender_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
message_text TEXT,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
-- Таблица просмотров профилей
|
||||
CREATE TABLE IF NOT EXISTS profile_views (
|
||||
id SERIAL PRIMARY KEY,
|
||||
viewer_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
viewed_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
viewed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
UNIQUE(viewer_id, viewed_id)
|
||||
);
|
||||
|
||||
-- Таблица уведомлений
|
||||
CREATE TABLE IF NOT EXISTS notifications (
|
||||
id SERIAL PRIMARY KEY,
|
||||
user_id BIGINT REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
data JSONB,
|
||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||
scheduled_for TIMESTAMP,
|
||||
processed BOOLEAN DEFAULT FALSE
|
||||
);
|
||||
\`);
|
||||
|
||||
// Индексы для оптимизации запросов
|
||||
pgm.sql(\`
|
||||
-- Индексы для таблицы пользователей
|
||||
CREATE INDEX IF NOT EXISTS users_username_idx ON users(username);
|
||||
CREATE INDEX IF NOT EXISTS users_gender_idx ON users(gender);
|
||||
CREATE INDEX IF NOT EXISTS users_looking_for_idx ON users(looking_for);
|
||||
CREATE INDEX IF NOT EXISTS users_premium_idx ON users(premium);
|
||||
|
||||
-- Индексы для таблицы лайков
|
||||
CREATE INDEX IF NOT EXISTS likes_from_user_id_idx ON likes(from_user_id);
|
||||
CREATE INDEX IF NOT EXISTS likes_to_user_id_idx ON likes(to_user_id);
|
||||
CREATE INDEX IF NOT EXISTS likes_is_like_idx ON likes(is_like);
|
||||
|
||||
-- Индексы для таблицы матчей
|
||||
CREATE INDEX IF NOT EXISTS matches_user1_id_idx ON matches(user1_id);
|
||||
CREATE INDEX IF NOT EXISTS matches_user2_id_idx ON matches(user2_id);
|
||||
CREATE INDEX IF NOT EXISTS matches_active_idx ON matches(active);
|
||||
|
||||
-- Индексы для таблицы сообщений
|
||||
CREATE INDEX IF NOT EXISTS messages_match_id_idx ON messages(match_id);
|
||||
CREATE INDEX IF NOT EXISTS messages_sender_id_idx ON messages(sender_id);
|
||||
|
||||
-- Индексы для таблицы профилей
|
||||
CREATE INDEX IF NOT EXISTS profiles_user_id_idx ON profiles(user_id);
|
||||
CREATE INDEX IF NOT EXISTS profiles_gender_idx ON profiles(gender);
|
||||
CREATE INDEX IF NOT EXISTS profiles_looking_for_idx ON profiles(looking_for);
|
||||
|
||||
-- Индексы для таблицы просмотров профилей
|
||||
CREATE INDEX IF NOT EXISTS profile_views_viewer_id_idx ON profile_views(viewer_id);
|
||||
CREATE INDEX IF NOT EXISTS profile_views_viewed_id_idx ON profile_views(viewed_id);
|
||||
|
||||
-- Индексы для таблицы уведомлений
|
||||
CREATE INDEX IF NOT EXISTS notifications_user_id_idx ON notifications(user_id);
|
||||
CREATE INDEX IF NOT EXISTS notifications_scheduled_for_idx ON notifications(scheduled_for);
|
||||
CREATE INDEX IF NOT EXISTS notifications_processed_idx ON notifications(processed);
|
||||
\`);
|
||||
};
|
||||
|
||||
exports.down = pgm => {
|
||||
// Эта функция не будет фактически использоваться,
|
||||
// но для полноты оставляем возможность отката
|
||||
pgm.sql(\`
|
||||
DROP TABLE IF EXISTS notifications;
|
||||
DROP TABLE IF EXISTS profile_views;
|
||||
DROP TABLE IF EXISTS messages;
|
||||
DROP TABLE IF EXISTS matches;
|
||||
DROP TABLE IF EXISTS likes;
|
||||
DROP TABLE IF EXISTS profiles;
|
||||
DROP TABLE IF EXISTS users;
|
||||
\`);
|
||||
};
|
||||
EOL
|
||||
|
||||
echo "✅ Консолидированная миграция создана: $consolidated_file"
|
||||
echo ""
|
||||
echo "🚀 Для применения миграции выполните:"
|
||||
echo "DATABASE_URL=postgres://\${DB_USERNAME}:\${DB_PASSWORD}@\${DB_HOST}:\${DB_PORT}/\${DB_NAME} npx node-pg-migrate up"
|
||||
72
bin/create_release.sh
Normal file
72
bin/create_release.sh
Normal file
@@ -0,0 +1,72 @@
|
||||
#!/bin/bash
|
||||
# Скрипт для создания релиза Telegram Tinder Bot
|
||||
|
||||
# Получение версии из package.json
|
||||
VERSION=$(grep -m1 "version" package.json | cut -d'"' -f4)
|
||||
RELEASE_NAME="tg-tinder-bot-v$VERSION"
|
||||
RELEASE_DIR="bin/releases/$RELEASE_NAME"
|
||||
|
||||
echo "🚀 Создание релиза $RELEASE_NAME"
|
||||
|
||||
# Создание директории релиза
|
||||
mkdir -p "$RELEASE_DIR"
|
||||
|
||||
# Очистка временных файлов
|
||||
echo "🧹 Очистка временных файлов..."
|
||||
rm -rf dist node_modules
|
||||
|
||||
# Установка зависимостей
|
||||
echo "📦 Установка зависимостей production..."
|
||||
npm ci --only=production
|
||||
|
||||
# Сборка проекта
|
||||
echo "🔧 Сборка проекта..."
|
||||
npm run build
|
||||
|
||||
# Копирование файлов релиза
|
||||
echo "📋 Копирование файлов..."
|
||||
cp -r dist "$RELEASE_DIR/"
|
||||
cp -r src/locales "$RELEASE_DIR/dist/"
|
||||
cp package.json package-lock.json .env.example "$RELEASE_DIR/"
|
||||
cp -r bin/start_bot.* bin/install_ubuntu.sh "$RELEASE_DIR/"
|
||||
cp README.md LICENSE "$RELEASE_DIR/" 2>/dev/null || echo "Файлы документации не найдены"
|
||||
cp sql/consolidated.sql "$RELEASE_DIR/"
|
||||
cp docker-compose.yml Dockerfile "$RELEASE_DIR/"
|
||||
cp deploy.sh "$RELEASE_DIR/" && chmod +x "$RELEASE_DIR/deploy.sh"
|
||||
|
||||
# Создание README для релиза
|
||||
cat > "$RELEASE_DIR/RELEASE.md" << EOL
|
||||
# Telegram Tinder Bot v$VERSION
|
||||
|
||||
Эта папка содержит релиз Telegram Tinder Bot версии $VERSION.
|
||||
|
||||
## Содержимое
|
||||
|
||||
- \`dist/\` - Скомпилированный код
|
||||
- \`package.json\` - Зависимости и скрипты
|
||||
- \`.env.example\` - Пример конфигурации
|
||||
- \`docker-compose.yml\` и \`Dockerfile\` - Для запуска через Docker
|
||||
- \`consolidated.sql\` - SQL-скрипт для инициализации базы данных
|
||||
- \`deploy.sh\` - Скрипт для простого деплоя
|
||||
|
||||
## Быстрый старт
|
||||
|
||||
1. Создайте файл \`.env\` на основе \`.env.example\`
|
||||
2. Запустите бота одним из способов:
|
||||
- Через Docker: \`./deploy.sh\`
|
||||
- Через Node.js: \`node dist/bot.js\`
|
||||
|
||||
## Дата релиза
|
||||
|
||||
$(date "+%d.%m.%Y %H:%M")
|
||||
EOL
|
||||
|
||||
# Архивирование релиза
|
||||
echo "📦 Создание архива..."
|
||||
cd bin/releases
|
||||
zip -r "$RELEASE_NAME.zip" "$RELEASE_NAME"
|
||||
cd ../..
|
||||
|
||||
echo "✅ Релиз создан успешно!"
|
||||
echo "📂 Релиз доступен в: bin/releases/$RELEASE_NAME"
|
||||
echo "📦 Архив релиза: bin/releases/$RELEASE_NAME.zip"
|
||||
60
bin/find_hardcoded_texts.sh
Executable file
60
bin/find_hardcoded_texts.sh
Executable file
@@ -0,0 +1,60 @@
|
||||
#!/bin/bash
|
||||
|
||||
# Скрипт для поиска всех хардкод-текстов на русском языке в TypeScript файлах
|
||||
|
||||
echo "========================================="
|
||||
echo "🔍 Поиск хардкод-текстов в проекте"
|
||||
echo "========================================="
|
||||
echo ""
|
||||
|
||||
# Директория для поиска
|
||||
SEARCH_DIR="src"
|
||||
|
||||
# Подсчет общего количества хардкод-текстов
|
||||
echo "📊 Общая статистика:"
|
||||
echo "-------------------"
|
||||
|
||||
single_quotes=$(grep -rn "'[А-Яа-яЁё]" $SEARCH_DIR --include="*.ts" | wc -l)
|
||||
double_quotes=$(grep -rn '"[А-Яа-яЁё]' $SEARCH_DIR --include="*.ts" | wc -l)
|
||||
total=$((single_quotes + double_quotes))
|
||||
|
||||
echo "Тексты в одинарных кавычках: $single_quotes"
|
||||
echo "Тексты в двойных кавычках: $double_quotes"
|
||||
echo "ВСЕГО: $total"
|
||||
echo ""
|
||||
|
||||
# Топ-10 файлов с наибольшим количеством хардкода
|
||||
echo "📁 Топ-10 файлов с хардкод-текстами:"
|
||||
echo "-----------------------------------"
|
||||
grep -rn "'[А-Яа-яЁё]\|\"[А-Яа-яЁё]" $SEARCH_DIR --include="*.ts" | \
|
||||
cut -d: -f1 | \
|
||||
sort | \
|
||||
uniq -c | \
|
||||
sort -rn | \
|
||||
head -10 | \
|
||||
awk '{printf "%3d тексто в: %s\n", $1, $2}'
|
||||
echo ""
|
||||
|
||||
# Детальная информация по каждому файлу
|
||||
echo "📄 Детальная статистика по файлам:"
|
||||
echo "----------------------------------"
|
||||
|
||||
for file in $(grep -rl "'[А-Яа-яЁё]\|\"[А-Яа-яЁё]" $SEARCH_DIR --include="*.ts" | sort); do
|
||||
count=$(grep -n "'[А-Яа-яЁё]\|\"[А-Яа-яЁё]" "$file" | wc -l)
|
||||
if [ $count -gt 0 ]; then
|
||||
printf "%-60s %3d текстов\n" "$file" "$count"
|
||||
fi
|
||||
done
|
||||
|
||||
echo ""
|
||||
echo "========================================="
|
||||
echo "✅ Анализ завершен"
|
||||
echo "========================================="
|
||||
echo ""
|
||||
echo "Рекомендации:"
|
||||
echo "1. Начните с файлов, содержащих больше всего текстов"
|
||||
echo "2. Используйте команду для просмотра конкретных строк:"
|
||||
echo " grep -n \"'[А-Яа-яЁё]\\|\\\"[А-Яа-яЁё]\" <файл>"
|
||||
echo "3. Замените тексты на локализационные ключи"
|
||||
echo "4. Добавьте переводы в файлы src/locales/*.json"
|
||||
echo ""
|
||||
100
bin/fix_docker.bat
Normal file
100
bin/fix_docker.bat
Normal file
@@ -0,0 +1,100 @@
|
||||
@echo off
|
||||
REM fix_docker.bat - Скрипт для устранения проблемы ContainerConfig в Windows
|
||||
|
||||
echo 🔧 Устранение проблемы с Docker контейнерами...
|
||||
|
||||
REM Остановка всех контейнеров проекта
|
||||
echo 📥 Остановка всех контейнеров проекта...
|
||||
docker-compose down -v
|
||||
|
||||
REM Принудительное удаление контейнеров по имени
|
||||
echo 🗑️ Принудительное удаление оставшихся контейнеров...
|
||||
docker rm -f postgres-tinder adminer-tinder telegram-tinder-bot 2>NUL
|
||||
|
||||
REM Очистка неиспользуемых томов и сетей
|
||||
echo 🧹 Очистка неиспользуемых томов и сетей...
|
||||
docker system prune -f --volumes
|
||||
|
||||
REM Очистка кеша Docker
|
||||
echo 🧼 Очистка кеша Docker...
|
||||
docker builder prune -f
|
||||
|
||||
REM Исправление docker-compose.yml
|
||||
echo 📝 Создание обновленного docker-compose.yml...
|
||||
|
||||
REM Создаем обновленный docker-compose.yml с использованием PowerShell
|
||||
powershell -Command "& {
|
||||
$content = @'
|
||||
version: '3.8'
|
||||
|
||||
services:
|
||||
bot:
|
||||
build: .
|
||||
container_name: telegram-tinder-bot
|
||||
restart: unless-stopped
|
||||
env_file: .env
|
||||
environment:
|
||||
- NODE_ENV=production
|
||||
- DB_HOST=${DB_HOST:-db}
|
||||
- DB_PORT=${DB_PORT:-5432}
|
||||
- DB_NAME=${DB_NAME:-telegram_tinder_bot}
|
||||
- DB_USERNAME=${DB_USERNAME:-postgres}
|
||||
- DB_PASSWORD=${DB_PASSWORD:-postgres}
|
||||
volumes:
|
||||
- ./uploads:/app/uploads:rw
|
||||
- ./logs:/app/logs:rw
|
||||
networks:
|
||||
- bot-network
|
||||
healthcheck:
|
||||
test: [\"CMD\", \"wget\", \"--no-verbose\", \"--tries=1\", \"--spider\", \"http://localhost:3000/health\"]
|
||||
interval: 30s
|
||||
timeout: 5s
|
||||
retries: 3
|
||||
start_period: 10s
|
||||
|
||||
db:
|
||||
image: postgres:15-alpine
|
||||
container_name: postgres-tinder
|
||||
restart: unless-stopped
|
||||
environment:
|
||||
- POSTGRES_DB=${DB_NAME:-telegram_tinder_bot}
|
||||
- POSTGRES_USER=${DB_USERNAME:-postgres}
|
||||
- POSTGRES_PASSWORD=${DB_PASSWORD:-postgres}
|
||||
volumes:
|
||||
- postgres_data:/var/lib/postgresql/data
|
||||
ports:
|
||||
- \"5433:5432\"
|
||||
networks:
|
||||
- bot-network
|
||||
healthcheck:
|
||||
test: [\"CMD-SHELL\", \"pg_isready -U ${DB_USERNAME:-postgres} -d ${DB_NAME:-telegram_tinder_bot}\"]
|
||||
interval: 10s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
start_period: 10s
|
||||
|
||||
adminer:
|
||||
image: adminer:latest
|
||||
container_name: adminer-tinder
|
||||
restart: unless-stopped
|
||||
ports:
|
||||
- \"8080:8080\"
|
||||
networks:
|
||||
- bot-network
|
||||
|
||||
volumes:
|
||||
postgres_data:
|
||||
|
||||
networks:
|
||||
bot-network:
|
||||
driver: bridge
|
||||
'@
|
||||
Set-Content -Path 'docker-compose.yml' -Value $content
|
||||
}"
|
||||
|
||||
echo ✅ docker-compose.yml обновлен!
|
||||
|
||||
echo 🚀 Готово! Теперь вы можете запустить контейнеры снова с помощью команды:
|
||||
echo docker-compose up -d
|
||||
|
||||
pause
|
||||
117
bin/fix_docker.sh
Normal file
117
bin/fix_docker.sh
Normal file
@@ -0,0 +1,117 @@
|
||||
#!/bin/bash
|
||||
# fix_docker.sh - Скрипт для устранения проблемы ContainerConfig
|
||||
|
||||
echo "🔧 Устранение проблемы с Docker контейнерами..."
|
||||
|
||||
# Остановка всех контейнеров проекта
|
||||
echo "📥 Остановка всех контейнеров проекта..."
|
||||
docker-compose down -v
|
||||
|
||||
# Принудительное удаление контейнеров по имени
|
||||
echo "🗑️ Принудительное удаление оставшихся контейнеров..."
|
||||
docker rm -f postgres-tinder adminer-tinder telegram-tinder-bot 2>/dev/null || true
|
||||
|
||||
# Очистка неиспользуемых томов и сетей
|
||||
echo "🧹 Очистка неиспользуемых томов и сетей..."
|
||||
docker system prune -f --volumes
|
||||
|
||||
# Очистка кеша Docker
|
||||
echo "🧼 Очистка кеша Docker..."
|
||||
docker builder prune -f
|
||||
|
||||
# Исправление docker-compose.yml
|
||||
echo "📝 Создание обновленного docker-compose.yml..."
|
||||
|
||||
cat > docker-compose.yml << EOL
|
||||
version: '3.8'
|
||||
|
||||
services:
|
||||
bot:
|
||||
build: .
|
||||
container_name: telegram-tinder-bot
|
||||
restart: unless-stopped
|
||||
env_file: .env
|
||||
environment:
|
||||
- NODE_ENV=production
|
||||
- DB_HOST=${DB_HOST:-db}
|
||||
- DB_PORT=${DB_PORT:-5432}
|
||||
- DB_NAME=${DB_NAME:-telegram_tinder_bot}
|
||||
- DB_USERNAME=${DB_USERNAME:-postgres}
|
||||
- DB_PASSWORD=${DB_PASSWORD:-postgres}
|
||||
volumes:
|
||||
- ./uploads:/app/uploads:rw
|
||||
- ./logs:/app/logs:rw
|
||||
networks:
|
||||
- bot-network
|
||||
healthcheck:
|
||||
test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://localhost:3000/health"]
|
||||
interval: 30s
|
||||
timeout: 5s
|
||||
retries: 3
|
||||
start_period: 10s
|
||||
EOL
|
||||
|
||||
# Если используем внешнюю базу данных, добавляем только adminer
|
||||
if [ "${DB_HOST:-db}" != "db" ]; then
|
||||
cat >> docker-compose.yml << EOL
|
||||
|
||||
adminer:
|
||||
image: adminer:latest
|
||||
container_name: adminer-tinder
|
||||
restart: unless-stopped
|
||||
ports:
|
||||
- "8080:8080"
|
||||
networks:
|
||||
- bot-network
|
||||
EOL
|
||||
else
|
||||
# Если используем локальную базу данных, добавляем PostgreSQL и adminer
|
||||
cat >> docker-compose.yml << EOL
|
||||
|
||||
db:
|
||||
image: postgres:15-alpine
|
||||
container_name: postgres-tinder
|
||||
restart: unless-stopped
|
||||
environment:
|
||||
- POSTGRES_DB=\${DB_NAME:-telegram_tinder_bot}
|
||||
- POSTGRES_USER=\${DB_USERNAME:-postgres}
|
||||
- POSTGRES_PASSWORD=\${DB_PASSWORD:-postgres}
|
||||
volumes:
|
||||
- postgres_data:/var/lib/postgresql/data
|
||||
ports:
|
||||
- "5433:5432"
|
||||
networks:
|
||||
- bot-network
|
||||
healthcheck:
|
||||
test: ["CMD-SHELL", "pg_isready -U \${DB_USERNAME:-postgres} -d \${DB_NAME:-telegram_tinder_bot}"]
|
||||
interval: 10s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
start_period: 10s
|
||||
|
||||
adminer:
|
||||
image: adminer:latest
|
||||
container_name: adminer-tinder
|
||||
restart: unless-stopped
|
||||
ports:
|
||||
- "8080:8080"
|
||||
networks:
|
||||
- bot-network
|
||||
EOL
|
||||
fi
|
||||
|
||||
# Завершаем файл docker-compose.yml
|
||||
cat >> docker-compose.yml << EOL
|
||||
|
||||
volumes:
|
||||
postgres_data:
|
||||
|
||||
networks:
|
||||
bot-network:
|
||||
driver: bridge
|
||||
EOL
|
||||
|
||||
echo "✅ docker-compose.yml обновлен!"
|
||||
|
||||
echo "🚀 Готово! Теперь вы можете запустить контейнеры снова с помощью команды:"
|
||||
echo "docker-compose up -d"
|
||||
15
bin/fix_line_endings.sh
Normal file
15
bin/fix_line_endings.sh
Normal file
@@ -0,0 +1,15 @@
|
||||
#!/bin/bash
|
||||
# fix_line_endings.sh - Script to fix line endings in shell scripts
|
||||
|
||||
echo "🔧 Fixing line endings in shell scripts..."
|
||||
|
||||
# Fix shell scripts
|
||||
for file in $(find . -name "*.sh"); do
|
||||
echo "📄 Processing $file..."
|
||||
tr -d '\r' < "$file" > "$file.fixed"
|
||||
mv "$file.fixed" "$file"
|
||||
chmod +x "$file"
|
||||
echo "✅ Fixed $file"
|
||||
done
|
||||
|
||||
echo "🚀 All shell scripts fixed!"
|
||||
17
bin/fix_permissions.sh
Normal file
17
bin/fix_permissions.sh
Normal file
@@ -0,0 +1,17 @@
|
||||
#!/bin/bash
|
||||
# fix_permissions.sh - Устанавливает права на выполнение для всех скриптов
|
||||
|
||||
echo "🔧 Установка прав на выполнение для всех скриптов..."
|
||||
|
||||
# Находим все .sh файлы и устанавливаем права на выполнение
|
||||
find . -name "*.sh" -type f -exec chmod +x {} \;
|
||||
|
||||
echo "✅ Права на выполнение установлены!"
|
||||
|
||||
# Исправление переносов строк
|
||||
echo "🔧 Исправление переносов строк..."
|
||||
find . -name "*.sh" -type f -exec sh -c 'tr -d "\r" < "$1" > "$1.fixed" && mv "$1.fixed" "$1"' -- {} \;
|
||||
|
||||
echo "✅ Переносы строк исправлены!"
|
||||
|
||||
echo "🚀 Готово! Все скрипты готовы к использованию."
|
||||
20
bin/init_database.sh
Normal file
20
bin/init_database.sh
Normal file
@@ -0,0 +1,20 @@
|
||||
#!/bin/bash
|
||||
# init_database.sh - Initialize database with schema
|
||||
|
||||
set -e
|
||||
|
||||
echo "🗄️ Initializing database..."
|
||||
|
||||
# Wait for database to be ready
|
||||
echo "⏳ Waiting for database..."
|
||||
sleep 3
|
||||
|
||||
# Create UUID extension
|
||||
echo "📦 Creating UUID extension..."
|
||||
docker compose exec -T db psql -U postgres -d telegram_tinder_bot -c "CREATE EXTENSION IF NOT EXISTS \"uuid-ossp\";" || true
|
||||
|
||||
# Apply consolidated schema
|
||||
echo "📋 Applying database schema..."
|
||||
docker compose exec -T db psql -U postgres -d telegram_tinder_bot < sql/consolidated.sql
|
||||
|
||||
echo "✅ Database initialized successfully!"
|
||||
58
bin/install_docker.sh
Normal file
58
bin/install_docker.sh
Normal file
@@ -0,0 +1,58 @@
|
||||
#!/bin/bash
|
||||
# install_docker.sh - Script for installing Docker and Docker Compose
|
||||
|
||||
echo "🚀 Installing Docker and Docker Compose..."
|
||||
|
||||
# Check if script is run as root
|
||||
if [ "$(id -u)" -ne 0 ]; then
|
||||
echo "❌ This script must be run as root. Please run with sudo."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Update package lists
|
||||
echo "📦 Updating package lists..."
|
||||
apt update
|
||||
|
||||
# Install required packages
|
||||
echo "📦 Installing required packages..."
|
||||
apt install -y apt-transport-https ca-certificates curl software-properties-common
|
||||
|
||||
# Add Docker GPG key
|
||||
echo "🔑 Adding Docker GPG key..."
|
||||
curl -fsSL https://download.docker.com/linux/ubuntu/gpg | apt-key add -
|
||||
|
||||
# Add Docker repository
|
||||
echo "📁 Adding Docker repository..."
|
||||
add-apt-repository "deb [arch=amd64] https://download.docker.com/linux/ubuntu $(lsb_release -cs) stable"
|
||||
|
||||
# Update package lists again
|
||||
apt update
|
||||
|
||||
# Install Docker
|
||||
echo "🐳 Installing Docker..."
|
||||
apt install -y docker-ce docker-ce-cli containerd.io
|
||||
|
||||
# Enable and start Docker service
|
||||
systemctl enable docker
|
||||
systemctl start docker
|
||||
|
||||
# Install Docker Compose
|
||||
echo "🐳 Installing Docker Compose..."
|
||||
curl -L "https://github.com/docker/compose/releases/download/v2.24.6/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose
|
||||
chmod +x /usr/local/bin/docker-compose
|
||||
|
||||
# Check versions
|
||||
echo "✅ Installation complete!"
|
||||
echo "Docker version:"
|
||||
docker --version
|
||||
echo "Docker Compose version:"
|
||||
docker-compose --version
|
||||
|
||||
# Add current user to docker group if not root
|
||||
if [ -n "$SUDO_USER" ]; then
|
||||
echo "👤 Adding user $SUDO_USER to docker group..."
|
||||
usermod -aG docker $SUDO_USER
|
||||
echo "⚠️ Please log out and log back in for group changes to take effect."
|
||||
fi
|
||||
|
||||
echo "🎉 Docker installation completed successfully!"
|
||||
37
bin/run_full_migration.sh
Normal file
37
bin/run_full_migration.sh
Normal file
@@ -0,0 +1,37 @@
|
||||
#!/bin/bash
|
||||
# run_full_migration.sh - Полный процесс миграции с компиляцией TypeScript
|
||||
|
||||
echo "🚀 Запуск полного процесса миграции..."
|
||||
|
||||
# Проверка наличия файлов TS
|
||||
if find migrations -name "*.ts" -quit; then
|
||||
echo "📋 Обнаружены TypeScript миграции. Компилируем их..."
|
||||
|
||||
# Компиляция TS файлов
|
||||
./bin/compile_ts_migrations.sh
|
||||
|
||||
# Проверка результата
|
||||
if [ $? -ne 0 ]; then
|
||||
echo "❌ Ошибка компиляции TS миграций!"
|
||||
exit 1
|
||||
fi
|
||||
else
|
||||
echo "ℹ️ TypeScript миграции не обнаружены, пропускаем компиляцию."
|
||||
mkdir -p temp_migrations
|
||||
fi
|
||||
|
||||
# Копирование JS миграций
|
||||
echo "📂 Копирование JS-миграций..."
|
||||
find migrations -name "*.js" -exec cp {} temp_migrations/ \;
|
||||
|
||||
# Запуск миграций
|
||||
echo "🔄 Применение всех миграций..."
|
||||
./bin/apply_migrations.sh
|
||||
|
||||
# Проверка результата
|
||||
if [ $? -eq 0 ]; then
|
||||
echo "✅ Процесс миграции успешно завершен!"
|
||||
else
|
||||
echo "❌ Ошибка в процессе миграции."
|
||||
exit 1
|
||||
fi
|
||||
101
bin/run_sql_migrations.sh
Normal file
101
bin/run_sql_migrations.sh
Normal file
@@ -0,0 +1,101 @@
|
||||
#!/bin/bash
|
||||
# run_sql_migrations.sh - Ручное применение SQL-миграций
|
||||
|
||||
echo "🚀 Запуск SQL-миграций..."
|
||||
|
||||
# Загрузка переменных окружения из .env
|
||||
if [ -f .env ]; then
|
||||
echo "📝 Загрузка переменных окружения из .env..."
|
||||
set -o allexport
|
||||
source .env
|
||||
set +o allexport
|
||||
else
|
||||
echo "⚠️ Файл .env не найден, используем значения по умолчанию"
|
||||
export DB_HOST="localhost"
|
||||
export DB_PORT="5432"
|
||||
export DB_NAME="telegram_tinder_bot"
|
||||
export DB_USERNAME="postgres"
|
||||
export DB_PASSWORD="postgres"
|
||||
fi
|
||||
|
||||
# Вывод информации о подключении
|
||||
echo "🔍 Используемые параметры подключения:"
|
||||
echo "DB_HOST: $DB_HOST"
|
||||
echo "DB_PORT: $DB_PORT"
|
||||
echo "DB_NAME: $DB_NAME"
|
||||
echo "DB_USERNAME: $DB_USERNAME"
|
||||
echo "DB_PASSWORD: ********"
|
||||
|
||||
# Функция для применения SQL файлов из директории
|
||||
apply_sql_files() {
|
||||
local directory=$1
|
||||
echo "🔍 Ищем SQL-файлы в директории $directory..."
|
||||
|
||||
if [ -d "$directory" ]; then
|
||||
# Получаем список файлов в порядке времени создания
|
||||
files=$(find "$directory" -name "*.sql" | sort)
|
||||
|
||||
if [ -z "$files" ]; then
|
||||
echo "⚠️ SQL-файлы не найдены в $directory"
|
||||
return 0
|
||||
fi
|
||||
|
||||
for sql_file in $files; do
|
||||
echo "🔄 Применение миграции $sql_file..."
|
||||
|
||||
# Проверяем, есть ли уже запись о миграции в таблице migrations
|
||||
filename=$(basename "$sql_file")
|
||||
exists=$(PGPASSWORD=$DB_PASSWORD psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -t -c "SELECT EXISTS(SELECT 1 FROM migrations WHERE name='$filename')" 2>/dev/null)
|
||||
|
||||
# Если таблицы migrations не существует, создаем её
|
||||
if [ $? -ne 0 ]; then
|
||||
echo "📝 Таблица migrations не найдена. Создаем..."
|
||||
PGPASSWORD=$DB_PASSWORD psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -c "
|
||||
CREATE TABLE IF NOT EXISTS migrations (
|
||||
id SERIAL PRIMARY KEY,
|
||||
name VARCHAR(255) NOT NULL,
|
||||
executed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
)
|
||||
" 2>/dev/null
|
||||
exists=" f"
|
||||
fi
|
||||
|
||||
# Если миграция уже применена, пропускаем
|
||||
if [[ "$exists" == *"t"* ]]; then
|
||||
echo "⏭️ Миграция $filename уже применена, пропускаем"
|
||||
continue
|
||||
fi
|
||||
|
||||
# Применяем миграцию
|
||||
PGPASSWORD=$DB_PASSWORD psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -f "$sql_file"
|
||||
|
||||
if [ $? -eq 0 ]; then
|
||||
echo "✅ Миграция $filename успешно применена"
|
||||
# Записываем в таблицу migrations
|
||||
PGPASSWORD=$DB_PASSWORD psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -c "
|
||||
INSERT INTO migrations (name) VALUES ('$filename')
|
||||
" 2>/dev/null
|
||||
else
|
||||
echo "❌ Ошибка при применении миграции $filename"
|
||||
return 1
|
||||
fi
|
||||
done
|
||||
else
|
||||
echo "⚠️ Директория $directory не найдена"
|
||||
return 0
|
||||
fi
|
||||
|
||||
return 0
|
||||
}
|
||||
|
||||
# Применяем SQL миграции из всех возможных папок
|
||||
echo "🔄 Применение SQL-миграций из src/database/migrations..."
|
||||
apply_sql_files "src/database/migrations"
|
||||
|
||||
echo "🔄 Применение SQL-миграций из migrations/sql..."
|
||||
apply_sql_files "migrations/sql"
|
||||
|
||||
echo "🔄 Применение SQL-миграций из migrations (если есть)..."
|
||||
apply_sql_files "migrations"
|
||||
|
||||
echo "✅ Процесс применения SQL-миграций завершен!"
|
||||
@@ -1,14 +0,0 @@
|
||||
import { query } from './src/database/connection';
|
||||
|
||||
async function checkSchema() {
|
||||
try {
|
||||
const result = await query('SELECT column_name, data_type FROM information_schema.columns WHERE table_name = $1', ['messages']);
|
||||
console.log(result.rows);
|
||||
process.exit(0);
|
||||
} catch (error) {
|
||||
console.error(error);
|
||||
process.exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
checkSchema();
|
||||
217
deploy.sh
Normal file
217
deploy.sh
Normal file
@@ -0,0 +1,217 @@
|
||||
#!/bin/bash
|
||||
# deploy.sh - Улучшенный скрипт для деплоя Telegram Tinder Bot
|
||||
|
||||
set -e # Выход при ошибке
|
||||
|
||||
# Определение цветов для вывода
|
||||
GREEN='\033[0;32m'
|
||||
BLUE='\033[0;34m'
|
||||
YELLOW='\033[0;33m'
|
||||
RED='\033[0;31m'
|
||||
NC='\033[0m' # No Color
|
||||
|
||||
echo -e "${BLUE}========================================${NC}"
|
||||
echo -e "${BLUE} Telegram Tinder Bot Deploy ${NC}"
|
||||
echo -e "${BLUE}========================================${NC}"
|
||||
|
||||
# Определяем рабочую директорию
|
||||
SCRIPT_DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )"
|
||||
cd "$SCRIPT_DIR"
|
||||
|
||||
# Функция для проверки наличия команды
|
||||
check_command() {
|
||||
if ! command -v "$1" &> /dev/null; then
|
||||
echo -e "${RED}❌ Команда $1 не найдена!${NC}"
|
||||
return 1
|
||||
else
|
||||
echo -e "${GREEN}✓ Команда $1 найдена${NC}"
|
||||
return 0
|
||||
fi
|
||||
}
|
||||
|
||||
# Шаг 1: Проверка и установка зависимостей
|
||||
echo -e "\n${BLUE}Шаг 1: Проверка и установка зависимостей...${NC}"
|
||||
|
||||
# Проверяем наличие Docker и Docker Compose
|
||||
if ! check_command docker || ! check_command docker-compose; then
|
||||
echo -e "${YELLOW}Установка Docker и Docker Compose...${NC}"
|
||||
|
||||
# Проверяем, запущен ли скрипт от имени root
|
||||
if [ "$(id -u)" -ne 0 ]; then
|
||||
echo -e "${RED}❌ Этот скрипт должен быть запущен с правами root для установки Docker.${NC}"
|
||||
echo -e "Пожалуйста, запустите: ${YELLOW}sudo $0${NC}"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Устанавливаем Docker и Docker Compose
|
||||
if [ -f bin/install_docker.sh ]; then
|
||||
bash bin/install_docker.sh
|
||||
else
|
||||
echo -e "${YELLOW}Установка Docker с помощью apt...${NC}"
|
||||
apt update
|
||||
apt install -y docker.io docker-compose
|
||||
fi
|
||||
fi
|
||||
|
||||
# Проверяем наличие Git
|
||||
if ! check_command git; then
|
||||
echo -e "${YELLOW}Установка Git...${NC}"
|
||||
apt update && apt install -y git
|
||||
fi
|
||||
|
||||
# Проверяем наличие Node.js (для локальных операций)
|
||||
if ! check_command node || ! check_command npm; then
|
||||
echo -e "${YELLOW}Установка Node.js...${NC}"
|
||||
apt update
|
||||
apt install -y curl
|
||||
curl -fsSL https://deb.nodesource.com/setup_20.x | bash -
|
||||
apt install -y nodejs
|
||||
fi
|
||||
|
||||
# Шаг 2: Получение последних изменений из репозитория
|
||||
echo -e "\n${BLUE}Шаг 2: Получение последних изменений из репозитория...${NC}"
|
||||
|
||||
# Сохраняем локальные изменения, если они есть
|
||||
git stash save "Auto-stash before deploy: $(date)" || true
|
||||
|
||||
# Получаем последние изменения
|
||||
git fetch --all
|
||||
git checkout main || git checkout master
|
||||
git pull origin "$(git rev-parse --abbrev-ref HEAD)"
|
||||
echo -e "${GREEN}✓ Получены последние изменения${NC}"
|
||||
|
||||
# Шаг 3: Проверка и создание файлов конфигурации
|
||||
echo -e "\n${BLUE}Шаг 3: Проверка и настройка конфигурационных файлов...${NC}"
|
||||
|
||||
# Проверяем наличие .env файла
|
||||
if [ ! -f .env ]; then
|
||||
echo -e "${YELLOW}⚠️ Файл .env не найден!${NC}"
|
||||
|
||||
# Пытаемся найти шаблон .env файла
|
||||
if [ -f .env.production ]; then
|
||||
echo -e "${YELLOW}Создание .env файла из .env.production...${NC}"
|
||||
cp .env.production .env
|
||||
elif [ -f .env.example ]; then
|
||||
echo -e "${YELLOW}Создание .env файла из .env.example...${NC}"
|
||||
cp .env.example .env
|
||||
else
|
||||
echo -e "${RED}❌ Шаблон .env файла не найден! Создаем базовый .env файл...${NC}"
|
||||
cat > .env << EOL
|
||||
# Базовый .env файл
|
||||
NODE_ENV=production
|
||||
PORT=3000
|
||||
DB_HOST=db
|
||||
DB_PORT=5432
|
||||
DB_NAME=telegram_tinder_bot
|
||||
DB_USERNAME=postgres
|
||||
DB_PASSWORD=postgres
|
||||
TELEGRAM_BOT_TOKEN=YOUR_BOT_TOKEN
|
||||
EOL
|
||||
fi
|
||||
|
||||
echo -e "${YELLOW}⚠️ Пожалуйста, отредактируйте файл .env и укажите свои настройки!${NC}"
|
||||
echo -e "${YELLOW}⚠️ Особенно важно указать TELEGRAM_BOT_TOKEN${NC}"
|
||||
read -p "Продолжить деплой? (y/n): " continue_deploy
|
||||
if [[ ! $continue_deploy =~ ^[Yy]$ ]]; then
|
||||
echo -e "${RED}Деплой отменен. Пожалуйста, настройте .env файл и запустите скрипт снова.${NC}"
|
||||
exit 1
|
||||
fi
|
||||
fi
|
||||
|
||||
# Проверяем наличие docker-compose.override.yml
|
||||
if [ ! -f docker-compose.override.yml ] && [ -f docker-compose.override.yml.example ]; then
|
||||
echo -e "${YELLOW}Создание docker-compose.override.yml из примера...${NC}"
|
||||
cp docker-compose.override.yml.example docker-compose.override.yml
|
||||
fi
|
||||
|
||||
# Шаг 4: Исправление проблем с Docker
|
||||
echo -e "\n${BLUE}Шаг 4: Проверка и исправление проблем с Docker...${NC}"
|
||||
|
||||
# Исправляем проблему с командой сборки в Dockerfile
|
||||
if [ -f Dockerfile ] && grep -q "RUN npm run build" Dockerfile; then
|
||||
echo -e "${YELLOW}⚠️ Исправление команды сборки в Dockerfile для совместимости с Linux...${NC}"
|
||||
sed -i 's/RUN npm run build/RUN npm run build:linux/g' Dockerfile
|
||||
echo -e "${GREEN}✓ Dockerfile обновлен${NC}"
|
||||
fi
|
||||
|
||||
# Исправление прав доступа к файлам в Unix-системах
|
||||
if [ -f bin/fix_permissions.sh ]; then
|
||||
echo -e "${YELLOW}Исправление прав доступа к файлам...${NC}"
|
||||
bash bin/fix_permissions.sh
|
||||
fi
|
||||
|
||||
# Шаг 5: Запуск с Docker Compose
|
||||
echo -e "\n${BLUE}Шаг 5: Сборка и запуск Docker контейнеров...${NC}"
|
||||
|
||||
# Остановка и удаление старых контейнеров
|
||||
echo -e "${YELLOW}Остановка и удаление старых контейнеров...${NC}"
|
||||
docker-compose down || true
|
||||
|
||||
# Проверка наличия скрипта для исправления Docker
|
||||
if [ -f bin/fix_docker.sh ]; then
|
||||
echo -e "${YELLOW}Запуск скрипта исправления Docker...${NC}"
|
||||
bash bin/fix_docker.sh
|
||||
fi
|
||||
|
||||
# Создание необходимых директорий с правильными правами доступа
|
||||
echo -e "${YELLOW}Создание необходимых директорий...${NC}"
|
||||
mkdir -p logs uploads
|
||||
chmod -R 777 logs uploads
|
||||
|
||||
# Сборка и запуск контейнеров
|
||||
echo -e "${YELLOW}Сборка контейнеров...${NC}"
|
||||
docker-compose build
|
||||
|
||||
echo -e "${YELLOW}Запуск контейнеров...${NC}"
|
||||
docker-compose up -d
|
||||
|
||||
# Шаг 6: Применение миграций
|
||||
echo -e "\n${BLUE}Шаг 6: Применение миграций базы данных...${NC}"
|
||||
|
||||
# Ждем инициализации базы данных
|
||||
echo -e "${YELLOW}Ожидание инициализации базы данных...${NC}"
|
||||
sleep 10
|
||||
|
||||
# Выбор способа миграции
|
||||
if [ -f bin/run_full_migration.sh ]; then
|
||||
echo -e "${YELLOW}Запуск полной миграции базы данных...${NC}"
|
||||
docker-compose exec bot bash -c "cd /app && ./bin/run_full_migration.sh" || true
|
||||
elif [ -f bin/apply_migrations.sh ]; then
|
||||
echo -e "${YELLOW}Применение миграций базы данных...${NC}"
|
||||
docker-compose exec bot bash -c "cd /app && ./bin/apply_migrations.sh" || true
|
||||
else
|
||||
echo -e "${YELLOW}Миграционные скрипты не найдены, пропускаем этап миграции${NC}"
|
||||
fi
|
||||
|
||||
# Шаг 7: Проверка работоспособности
|
||||
echo -e "\n${BLUE}Шаг 7: Проверка работоспособности...${NC}"
|
||||
|
||||
# Проверяем статус контейнеров
|
||||
echo -e "${YELLOW}Проверка статуса контейнеров...${NC}"
|
||||
docker-compose ps
|
||||
|
||||
# Ждем запуска API
|
||||
echo -e "${YELLOW}Ожидание запуска API...${NC}"
|
||||
sleep 5
|
||||
docker-compose exec bot curl -s http://localhost:3000/health || echo "⚠️ Сервис не отвечает на проверку здоровья"
|
||||
|
||||
# Вывод информации о деплое
|
||||
echo -e "\n${GREEN}✅ Деплой успешно завершен!${NC}"
|
||||
echo -e "${GREEN}✅ Бот должен быть доступен через Telegram.${NC}"
|
||||
echo ""
|
||||
echo -e "${BLUE}📊 Полезные команды:${NC}"
|
||||
echo -e "- ${YELLOW}Просмотр логов:${NC} docker-compose logs -f bot"
|
||||
echo -e "- ${YELLOW}Перезапуск сервисов:${NC} docker-compose restart"
|
||||
echo -e "- ${YELLOW}Остановка всех сервисов:${NC} docker-compose down"
|
||||
echo -e "- ${YELLOW}Доступ к базе данных:${NC} docker-compose exec db psql -U postgres -d telegram_tinder_bot"
|
||||
echo -e "- ${YELLOW}Проверка состояния бота:${NC} curl http://localhost:3000/health"
|
||||
echo ""
|
||||
echo -e "${BLUE}🌟 Для администрирования базы данных:${NC}"
|
||||
echo -e "Adminer доступен по адресу: http://ваш_сервер:8080"
|
||||
echo -e " - ${YELLOW}Система:${NC} PostgreSQL"
|
||||
echo -e " - ${YELLOW}Сервер:${NC} db"
|
||||
echo -e " - ${YELLOW}Пользователь:${NC} postgres"
|
||||
echo -e " - ${YELLOW}Пароль:${NC} (из переменной DB_PASSWORD в .env)"
|
||||
echo -e " - ${YELLOW}База данных:${NC} telegram_tinder_bot"
|
||||
echo ""
|
||||
echo -e "${YELLOW}⚠️ При возникновении проблем проверьте файлы в директории bin/ для дополнительных утилит исправления.${NC}"
|
||||
@@ -1,49 +1,55 @@
|
||||
version: '3.8'
|
||||
|
||||
services:
|
||||
bot:
|
||||
build: .
|
||||
container_name: telegram-tinder-bot
|
||||
restart: unless-stopped
|
||||
depends_on:
|
||||
- db
|
||||
env_file: .env
|
||||
environment:
|
||||
- NODE_ENV=production
|
||||
- DB_HOST=db
|
||||
- DB_PORT=5432
|
||||
- DB_NAME=telegram_tinder_bot
|
||||
- DB_USERNAME=postgres
|
||||
- DB_PASSWORD=${DB_PASSWORD}
|
||||
- TELEGRAM_BOT_TOKEN=${TELEGRAM_BOT_TOKEN}
|
||||
- DB_HOST=${DB_HOST:-db}
|
||||
- DB_PORT=${DB_PORT:-5432}
|
||||
- DB_NAME=${DB_NAME:-telegram_tinder_bot}
|
||||
- DB_USERNAME=${DB_USERNAME:-postgres}
|
||||
- DB_PASSWORD=${DB_PASSWORD:-postgres}
|
||||
volumes:
|
||||
- ./uploads:/app/uploads
|
||||
- ./uploads:/app/uploads:rw
|
||||
- ./logs:/app/logs:rw
|
||||
networks:
|
||||
- bot-network
|
||||
healthcheck:
|
||||
test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://localhost:3000/health"]
|
||||
interval: 30s
|
||||
timeout: 5s
|
||||
retries: 3
|
||||
start_period: 10s
|
||||
|
||||
db:
|
||||
image: postgres:15-alpine
|
||||
container_name: postgres-tinder
|
||||
restart: unless-stopped
|
||||
environment:
|
||||
- POSTGRES_DB=telegram_tinder_bot
|
||||
- POSTGRES_USER=postgres
|
||||
- POSTGRES_PASSWORD=password123
|
||||
- POSTGRES_DB=${DB_NAME:-telegram_tinder_bot}
|
||||
- POSTGRES_USER=${DB_USERNAME:-postgres}
|
||||
- POSTGRES_PASSWORD=${DB_PASSWORD:-postgres}
|
||||
volumes:
|
||||
- postgres_data:/var/lib/postgresql/data
|
||||
- ./src/database/migrations/init.sql:/docker-entrypoint-initdb.d/init.sql
|
||||
ports:
|
||||
- "5433:5432"
|
||||
networks:
|
||||
- bot-network
|
||||
|
||||
healthcheck:
|
||||
test: ["CMD-SHELL", "pg_isready -U ${DB_USERNAME:-postgres} -d ${DB_NAME:-telegram_tinder_bot}"]
|
||||
interval: 10s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
start_period: 10s
|
||||
|
||||
adminer:
|
||||
image: adminer:latest
|
||||
container_name: adminer-tinder
|
||||
restart: unless-stopped
|
||||
ports:
|
||||
- "8080:8080"
|
||||
depends_on:
|
||||
- db
|
||||
networks:
|
||||
- bot-network
|
||||
|
||||
|
||||
176
docs/DATABASE_FIXES.md
Normal file
176
docs/DATABASE_FIXES.md
Normal file
@@ -0,0 +1,176 @@
|
||||
# Database Schema Fixes
|
||||
|
||||
## Обзор исправлений
|
||||
|
||||
Этот документ описывает исправления схемы базы данных, примененные к проекту Telegram Tinder Bot для устранения критических ошибок и предупреждений.
|
||||
|
||||
## Дата последнего обновления: 2025-11-06
|
||||
|
||||
---
|
||||
|
||||
## Исправление 1: Колонка `looking_for` и триггер создания профиля
|
||||
|
||||
### Проблема
|
||||
```
|
||||
ERROR: null value in column "looking_for" of relation "profiles" violates not-null constraint
|
||||
```
|
||||
|
||||
### Причина
|
||||
Триггер `create_initial_profile()` не устанавливал значение для обязательного поля `looking_for` при автоматическом создании профиля.
|
||||
|
||||
### Решение
|
||||
Применен патч: `sql/fix_looking_for_column.sql`
|
||||
|
||||
**Изменения:**
|
||||
1. Обновлен триггер для включения `looking_for = 'both'` и `interested_in = 'both'`
|
||||
2. Сделана колонка `looking_for` необязательной (nullable) с DEFAULT 'both'
|
||||
3. Добавлена колонка `interested_in` как современный синоним для `looking_for`
|
||||
4. Создан индекс для поиска: `idx_profiles_interested_in`
|
||||
|
||||
**Применение:**
|
||||
```bash
|
||||
PGPASSWORD='your_password' psql -h host -p 5432 -U user -d db_name -f sql/fix_looking_for_column.sql
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Исправление 2: Колонка `job` и `state`
|
||||
|
||||
### Проблема 1: Column "job" does not exist
|
||||
```
|
||||
ERROR: column "job" of relation "profiles" does not exist
|
||||
```
|
||||
|
||||
**Причина:** Код использует `job`, но в БД создана колонка `occupation`.
|
||||
|
||||
### Проблема 2: State column does not exist
|
||||
```
|
||||
WARNING: State column does not exist in users table. Skipping state check.
|
||||
```
|
||||
|
||||
**Причина:** Таблица `users` не содержит колонку `state` для отслеживания состояния диалога.
|
||||
|
||||
### Решение
|
||||
Применен патч: `sql/add_job_and_state_columns.sql`
|
||||
|
||||
**Изменения:**
|
||||
1. Добавлена колонка `job VARCHAR(255)` в таблицу `profiles`
|
||||
2. Скопированы данные из `occupation` → `job` для обратной совместимости
|
||||
3. Добавлена колонка `state VARCHAR(50)` в таблицу `users`
|
||||
4. Созданы индексы: `idx_profiles_job`, `idx_users_state`
|
||||
|
||||
**Применение:**
|
||||
```bash
|
||||
PGPASSWORD='your_password' psql -h host -p 5432 -U user -d db_name -f sql/add_job_and_state_columns.sql
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Полная последовательность применения патчей
|
||||
|
||||
Для нового развертывания применяйте патчи в следующем порядке:
|
||||
|
||||
```bash
|
||||
# 1. Основная схема (если еще не применена)
|
||||
PGPASSWORD='your_password' psql -h host -p 5432 -U user -d db_name -f sql/consolidated.sql
|
||||
|
||||
# 2. Исправление looking_for
|
||||
PGPASSWORD='your_password' psql -h host -p 5432 -U user -d db_name -f sql/fix_looking_for_column.sql
|
||||
|
||||
# 3. Добавление job и state
|
||||
PGPASSWORD='your_password' psql -h host -p 5432 -U user -d db_name -f sql/add_job_and_state_columns.sql
|
||||
```
|
||||
|
||||
Или используйте автоматизированную команду:
|
||||
```bash
|
||||
make migrate
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Проверка применения исправлений
|
||||
|
||||
### Проверка триггера looking_for
|
||||
```sql
|
||||
SELECT proname, prosrc
|
||||
FROM pg_proc
|
||||
WHERE proname = 'create_initial_profile';
|
||||
```
|
||||
|
||||
Должен содержать: `looking_for = 'both', interested_in = 'both'`
|
||||
|
||||
### Проверка колонок
|
||||
```sql
|
||||
-- Проверка profiles
|
||||
SELECT column_name, data_type, is_nullable, column_default
|
||||
FROM information_schema.columns
|
||||
WHERE table_name = 'profiles'
|
||||
AND column_name IN ('job', 'occupation', 'looking_for', 'interested_in')
|
||||
ORDER BY column_name;
|
||||
|
||||
-- Проверка users
|
||||
SELECT column_name, data_type, is_nullable
|
||||
FROM information_schema.columns
|
||||
WHERE table_name = 'users'
|
||||
AND column_name = 'state';
|
||||
```
|
||||
|
||||
### Проверка индексов
|
||||
```sql
|
||||
SELECT indexname, tablename, indexdef
|
||||
FROM pg_indexes
|
||||
WHERE tablename IN ('profiles', 'users')
|
||||
AND indexname IN ('idx_profiles_job', 'idx_users_state', 'idx_profiles_interested_in')
|
||||
ORDER BY tablename, indexname;
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Известные предупреждения (неопасные)
|
||||
|
||||
### ES Module в миграциях
|
||||
```
|
||||
SyntaxError: Unexpected token 'export'
|
||||
```
|
||||
|
||||
**Статус:** Не критично. Миграции применяются через psql напрямую, а не через node-pg-migrate.
|
||||
|
||||
**Причина:** Файлы миграций в `/migrations` используют ES6 синтаксис, несовместимый с node-pg-migrate в режиме CommonJS.
|
||||
|
||||
**Решение:** Используйте `make migrate` или применяйте SQL патчи напрямую через psql.
|
||||
|
||||
### DEEPSEEK_API_KEY not found
|
||||
```
|
||||
⚠️ DEEPSEEK_API_KEY not found in environment variables
|
||||
```
|
||||
|
||||
**Статус:** Не критично. Это опциональная AI-функция.
|
||||
|
||||
**Решение:** Добавьте `DEEPSEEK_API_KEY=your_key` в `.env` если хотите использовать AI-фичи.
|
||||
|
||||
---
|
||||
|
||||
## Mapping колонок (для справки)
|
||||
|
||||
| Код (TypeScript) | База данных | Комментарий |
|
||||
|------------------|-------------|-------------|
|
||||
| `job` | `job` | Основная колонка (новая) |
|
||||
| `job` | `occupation` | Устаревшая, оставлена для совместимости |
|
||||
| `interestedIn` | `interested_in` | Основная колонка (новая) |
|
||||
| `lookingFor` | `looking_for` | Устаревшая, nullable |
|
||||
| - | `state` | Новая колонка для users.state |
|
||||
|
||||
---
|
||||
|
||||
## Контакты для поддержки
|
||||
|
||||
При возникновении проблем с миграциями:
|
||||
|
||||
1. Проверьте логи бота: `docker compose logs bot --tail 50`
|
||||
2. Проверьте применение всех патчей (см. раздел "Проверка")
|
||||
3. Убедитесь, что `.env` содержит правильные DB_* переменные
|
||||
4. Попробуйте применить патчи вручную через psql
|
||||
|
||||
**Версия документа:** 1.0
|
||||
**Автор:** GitHub Copilot
|
||||
**Дата:** 2025-11-06
|
||||
173
docs/FIXES_SUMMARY_2025-11-06.md
Normal file
173
docs/FIXES_SUMMARY_2025-11-06.md
Normal file
@@ -0,0 +1,173 @@
|
||||
# Сводка исправлений от 2025-11-06
|
||||
|
||||
## 🎯 Цель
|
||||
Устранить критические ошибки базы данных, блокирующие создание профилей пользователей.
|
||||
|
||||
---
|
||||
|
||||
## 🐛 Исправленные ошибки
|
||||
|
||||
### 1. ❌ Column "job" does not exist
|
||||
**Ошибка:**
|
||||
```
|
||||
ERROR: column "job" of relation "profiles" does not exist
|
||||
Code: 42703
|
||||
```
|
||||
|
||||
**Причина:** Код использует поле `job`, но в БД существует только `occupation`.
|
||||
|
||||
**Решение:** Добавлена колонка `job` в таблицу `profiles`.
|
||||
|
||||
**Файл патча:** `sql/add_job_and_state_columns.sql`
|
||||
|
||||
**Статус:** ✅ ИСПРАВЛЕНО
|
||||
|
||||
---
|
||||
|
||||
### 2. ⚠️ State column does not exist in users table
|
||||
**Предупреждение:**
|
||||
```
|
||||
State column does not exist in users table. Skipping state check.
|
||||
```
|
||||
|
||||
**Причина:** Код пытается проверить состояние диалога пользователя через колонку `state`, которой нет в таблице `users`.
|
||||
|
||||
**Решение:** Добавлена колонка `state VARCHAR(50)` в таблицу `users`.
|
||||
|
||||
**Файл патча:** `sql/add_job_and_state_columns.sql`
|
||||
|
||||
**Статус:** ✅ ИСПРАВЛЕНО
|
||||
|
||||
---
|
||||
|
||||
### 3. ❌ null value in column "looking_for" violates not-null constraint
|
||||
**Ошибка:**
|
||||
```
|
||||
ERROR: null value in column "looking_for" of relation "profiles" violates not-null constraint
|
||||
Code: 23502
|
||||
```
|
||||
|
||||
**Причина:** Триггер `create_initial_profile()` не устанавливал значение для обязательного поля `looking_for`.
|
||||
|
||||
**Решение:**
|
||||
- Обновлен триггер для включения `looking_for = 'both'`
|
||||
- Колонка сделана nullable с DEFAULT 'both'
|
||||
- Добавлена колонка `interested_in` как современный синоним
|
||||
|
||||
**Файл патча:** `sql/fix_looking_for_column.sql`
|
||||
|
||||
**Статус:** ✅ ИСПРАВЛЕНО
|
||||
|
||||
---
|
||||
|
||||
## 📦 Применённые патчи
|
||||
|
||||
| # | Файл | Описание | Дата |
|
||||
|---|------|----------|------|
|
||||
| 1 | `sql/fix_looking_for_column.sql` | Исправление триггера и колонки looking_for | 2025-11-06 |
|
||||
| 2 | `sql/add_job_and_state_columns.sql` | Добавление колонок job и state | 2025-11-06 |
|
||||
|
||||
---
|
||||
|
||||
## 🔧 Применение патчей
|
||||
|
||||
### Автоматически (рекомендуется):
|
||||
```bash
|
||||
./bin/apply_all_patches.sh
|
||||
```
|
||||
|
||||
### Вручную:
|
||||
```bash
|
||||
# Патч 1: looking_for
|
||||
PGPASSWORD='password' psql -h host -p 5432 -U user -d db_name \
|
||||
-f sql/fix_looking_for_column.sql
|
||||
|
||||
# Патч 2: job и state
|
||||
PGPASSWORD='password' psql -h host -p 5432 -U user -d db_name \
|
||||
-f sql/add_job_and_state_columns.sql
|
||||
|
||||
# Перезапуск бота
|
||||
docker compose restart bot
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## ✅ Результаты после исправлений
|
||||
|
||||
### Проверка логов (нет критичных ошибок):
|
||||
```bash
|
||||
docker compose logs bot --since 5m | grep -E "(State column|column.*job|does not exist)"
|
||||
```
|
||||
**Результат:** Пусто (0 строк) ✅
|
||||
|
||||
### Проверка структуры БД:
|
||||
```
|
||||
table_name | column_name | data_type | nullable | column_default
|
||||
------------+---------------+-------------------+----------+--------------------
|
||||
profiles | interested_in | character varying | NULL | 'both'
|
||||
profiles | job | character varying | NULL |
|
||||
profiles | looking_for | character varying | NULL | 'both'
|
||||
profiles | occupation | character varying | NULL |
|
||||
users | state | character varying | NULL |
|
||||
```
|
||||
**Результат:** Все колонки присутствуют ✅
|
||||
|
||||
### Статус бота:
|
||||
```
|
||||
🎉 Bot initialized successfully!
|
||||
🤖 Bot is running and ready to match people!
|
||||
📱 Bot username: @seoulmate_officialbot
|
||||
```
|
||||
**Результат:** Бот запущен успешно ✅
|
||||
|
||||
---
|
||||
|
||||
## 📊 Статистика изменений
|
||||
|
||||
- **Добавлено колонок:** 3 (`job`, `state`, `interested_in`)
|
||||
- **Обновлено триггеров:** 1 (`create_initial_profile`)
|
||||
- **Создано индексов:** 3 (`idx_profiles_job`, `idx_users_state`, `idx_profiles_interested_in`)
|
||||
- **Файлов патчей:** 2
|
||||
- **Создано документации:** 3 файла (DATABASE_FIXES.md, HEALTH_CHECK.md, этот файл)
|
||||
- **Создано утилит:** 1 (`bin/apply_all_patches.sh`)
|
||||
|
||||
---
|
||||
|
||||
## 🚀 Дальнейшие действия
|
||||
|
||||
### Обязательно:
|
||||
- ✅ Протестировать создание профиля через бота
|
||||
- ✅ Проверить обновление профиля (поле job)
|
||||
- ✅ Убедиться что свайпы работают
|
||||
|
||||
### Опционально:
|
||||
- ⚠️ Рассмотреть объединение `job` и `occupation` в одну колонку
|
||||
- ⚠️ Рассмотреть объединение `looking_for` и `interested_in` в одну колонку
|
||||
- ⚠️ Исправить ES module warnings в миграциях (низкий приоритет)
|
||||
- ⚠️ Настроить DEEPSEEK_API_KEY если нужны AI-фичи
|
||||
|
||||
---
|
||||
|
||||
## 📚 Связанная документация
|
||||
|
||||
- `/docs/DATABASE_FIXES.md` - Подробное описание всех исправлений БД
|
||||
- `/docs/HEALTH_CHECK.md` - Чеклист проверки здоровья бота
|
||||
- `/bin/README.md` - Описание утилит и скриптов
|
||||
- `/bin/apply_all_patches.sh` - Скрипт автоматического применения патчей
|
||||
|
||||
---
|
||||
|
||||
## 👤 Авторство
|
||||
|
||||
**Дата:** 2025-11-06
|
||||
**Автор:** GitHub Copilot
|
||||
**Проект:** Telegram Tinder Bot
|
||||
**Версия:** 2.0
|
||||
|
||||
---
|
||||
|
||||
## ✨ Заключение
|
||||
|
||||
Все критические ошибки устранены. Бот готов к работе в production-среде с внешним PostgreSQL сервером (192.168.0.102:5432).
|
||||
|
||||
**Статус проекта:** 🟢 РАБОТОСПОСОБЕН
|
||||
199
docs/HEALTH_CHECK.md
Normal file
199
docs/HEALTH_CHECK.md
Normal file
@@ -0,0 +1,199 @@
|
||||
# Быстрая проверка здоровья бота
|
||||
|
||||
Используйте этот чеклист после развёртывания или обновления бота.
|
||||
|
||||
## ✅ Чеклист проверки
|
||||
|
||||
### 1. Проверка контейнера
|
||||
```bash
|
||||
docker compose ps
|
||||
# Ожидается: telegram-tinder-bot в состоянии "running" (healthy)
|
||||
```
|
||||
|
||||
### 2. Проверка логов (нет критичных ошибок)
|
||||
```bash
|
||||
docker compose logs bot --tail 50
|
||||
# ✅ Должно быть: "Bot initialized successfully"
|
||||
# ✅ Должно быть: "Bot username: @your_bot_name"
|
||||
# ❌ НЕ должно быть: "column X does not exist" (критическая ошибка)
|
||||
```
|
||||
|
||||
### 3. Проверка схемы БД
|
||||
```bash
|
||||
# Проверка критичных колонок
|
||||
PGPASSWORD='your_password' psql -h host -p 5432 -U user -d db_name << 'EOF'
|
||||
SELECT
|
||||
table_name,
|
||||
column_name,
|
||||
data_type,
|
||||
is_nullable
|
||||
FROM information_schema.columns
|
||||
WHERE table_name IN ('users', 'profiles')
|
||||
AND column_name IN ('state', 'job', 'looking_for', 'interested_in', 'occupation')
|
||||
ORDER BY table_name, column_name;
|
||||
EOF
|
||||
```
|
||||
|
||||
**Ожидаемый результат:**
|
||||
```
|
||||
table_name | column_name | data_type | is_nullable
|
||||
------------+----------------+--------------------+-------------
|
||||
profiles | interested_in | character varying | YES
|
||||
profiles | job | character varying | YES
|
||||
profiles | looking_for | character varying | YES
|
||||
profiles | occupation | character varying | YES
|
||||
users | state | character varying | YES
|
||||
```
|
||||
|
||||
### 4. Проверка триггера
|
||||
```bash
|
||||
PGPASSWORD='your_password' psql -h host -p 5432 -U user -d db_name -c \
|
||||
"SELECT proname FROM pg_proc WHERE proname = 'create_initial_profile';"
|
||||
```
|
||||
|
||||
**Ожидается:** `create_initial_profile` (1 строка)
|
||||
|
||||
### 5. Проверка индексов
|
||||
```bash
|
||||
PGPASSWORD='your_password' psql -h host -p 5432 -U user -d db_name << 'EOF'
|
||||
SELECT indexname
|
||||
FROM pg_indexes
|
||||
WHERE indexname IN ('idx_profiles_job', 'idx_users_state', 'idx_profiles_interested_in');
|
||||
EOF
|
||||
```
|
||||
|
||||
**Ожидается:** 3 строки с названиями индексов
|
||||
|
||||
---
|
||||
|
||||
## 🔧 Типичные проблемы и решения
|
||||
|
||||
### Проблема: "column job does not exist"
|
||||
**Решение:**
|
||||
```bash
|
||||
./bin/apply_all_patches.sh
|
||||
docker compose restart bot
|
||||
```
|
||||
|
||||
### Проблема: "State column does not exist" (много раз)
|
||||
**Решение:**
|
||||
```bash
|
||||
PGPASSWORD='password' psql -h host -p 5432 -U user -d db_name \
|
||||
-c "ALTER TABLE users ADD COLUMN IF NOT EXISTS state VARCHAR(50);"
|
||||
docker compose restart bot
|
||||
```
|
||||
|
||||
### Проблема: "looking_for violates not-null constraint"
|
||||
**Решение:**
|
||||
```bash
|
||||
PGPASSWORD='password' psql -h host -p 5432 -U user -d db_name \
|
||||
-f sql/fix_looking_for_column.sql
|
||||
docker compose restart bot
|
||||
```
|
||||
|
||||
### Проблема: Бот не запускается (exit code 1)
|
||||
**Диагностика:**
|
||||
```bash
|
||||
docker compose logs bot --tail 100
|
||||
# Ищите строки с ERROR или "does not exist"
|
||||
```
|
||||
|
||||
**Решения:**
|
||||
1. Проверьте `.env` - все переменные DB_* заданы?
|
||||
2. Проверьте подключение к БД: `PGPASSWORD='password' psql -h host -p 5432 -U user -d db_name -c "SELECT 1;"`
|
||||
3. Примените все патчи: `./bin/apply_all_patches.sh`
|
||||
4. Пересоберите контейнер: `make update`
|
||||
|
||||
---
|
||||
|
||||
## 📊 Быстрая диагностика одной командой
|
||||
|
||||
```bash
|
||||
# Создайте alias для удобства
|
||||
alias bot-health='docker compose ps && echo "=== LOGS ===" && docker compose logs bot --tail 20'
|
||||
|
||||
# Использование
|
||||
bot-health
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 🚀 Команды для разработки
|
||||
|
||||
```bash
|
||||
# Полное обновление (pull + rebuild + migrate + restart)
|
||||
make update
|
||||
|
||||
# Применение миграций
|
||||
make migrate
|
||||
|
||||
# Только перезапуск
|
||||
docker compose restart bot
|
||||
|
||||
# Пересборка с нуля
|
||||
make clean && make install
|
||||
|
||||
# Проверка синтаксиса TypeScript
|
||||
npm run build
|
||||
|
||||
# Запуск в режиме разработки (локально)
|
||||
npm run dev
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📝 Переменные окружения (.env)
|
||||
|
||||
Обязательные:
|
||||
```env
|
||||
DB_HOST=192.168.0.102
|
||||
DB_PORT=5432
|
||||
DB_NAME=telegram_tinder_bot
|
||||
DB_USERNAME=trevor
|
||||
DB_PASSWORD=your_secure_password
|
||||
|
||||
TELEGRAM_BOT_TOKEN=your_bot_token
|
||||
JWT_SECRET=your_jwt_secret
|
||||
APP_SECRET=your_app_secret
|
||||
|
||||
NODE_ENV=production
|
||||
PORT=3000
|
||||
```
|
||||
|
||||
Опциональные:
|
||||
```env
|
||||
DEEPSEEK_API_KEY=your_deepseek_key # Для AI фич
|
||||
LOG_LEVEL=info # debug | info | warn | error
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 🆘 Экстренное восстановление
|
||||
|
||||
Если бот полностью сломан:
|
||||
|
||||
```bash
|
||||
# 1. Остановить всё
|
||||
docker compose down
|
||||
|
||||
# 2. Сделать бэкап БД
|
||||
./bin/backup_db.sh
|
||||
|
||||
# 3. Откатить к последнему коммиту
|
||||
git reset --hard HEAD
|
||||
|
||||
# 4. Применить все патчи заново
|
||||
./bin/apply_all_patches.sh
|
||||
|
||||
# 5. Пересобрать
|
||||
make install
|
||||
|
||||
# 6. Запустить
|
||||
docker compose up -d
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
**Версия:** 1.0
|
||||
**Дата:** 2025-11-06
|
||||
**Для:** Telegram Tinder Bot v2.0
|
||||
225
docs/LOCALIZATION_CHECKLIST.md
Normal file
225
docs/LOCALIZATION_CHECKLIST.md
Normal file
@@ -0,0 +1,225 @@
|
||||
# Чеклист локализации
|
||||
|
||||
## Фаза 1: Инфраструктура ✅ ЗАВЕРШЕНО
|
||||
|
||||
- [x] Добавить колонку `lang` в таблицу `users`
|
||||
- [x] Создать миграцию `sql/add_user_language.sql`
|
||||
- [x] Применить миграцию к БД
|
||||
- [x] Создать `LanguageHandlers`
|
||||
- [x] Добавить методы в `ProfileService` (getUserLanguage, updateUserLanguage)
|
||||
- [x] Интегрировать выбор языка в `/start`
|
||||
- [x] Добавить обработку callback `set_lang_*`
|
||||
- [x] Обновить `ru.json` (секция language)
|
||||
- [x] Обновить `en.json` (секция language)
|
||||
- [x] Создать документацию
|
||||
- [x] Создать скрипт поиска хардкода
|
||||
- [x] Протестировать выбор языка
|
||||
|
||||
## Фаза 2: Замена хардкод-текстов ⚠️ В ПРОЦЕССЕ
|
||||
|
||||
### Приоритет 1: Обработчики (HIGH)
|
||||
|
||||
#### callbackHandlers.ts (90 текстов)
|
||||
- [ ] Просмотр профилей (showProfile, showNextCandidate) - ~15 текстов
|
||||
- [ ] Редактирование профиля (edit_name, edit_age, edit_bio) - ~20 текстов
|
||||
- [ ] Лайки и матчи (handleLike, handleMatch) - ~10 текстов
|
||||
- [ ] VIP функции (handleVIPSearch, translateProfile) - ~15 текстов
|
||||
- [ ] Меню и кнопки - ~20 текстов
|
||||
- [ ] Прочие обработчики - ~10 текстов
|
||||
|
||||
**Прогресс:** 0/90 (0%)
|
||||
|
||||
#### messageHandlers.ts (21 текст)
|
||||
- [ ] Создание профиля (handleCreateProfile) - ~8 текстов
|
||||
- [ ] Ввод данных профиля (name, age, city, bio) - ~8 текстов
|
||||
- [ ] Валидация ввода - ~5 текстов
|
||||
|
||||
**Прогресс:** 0/21 (0%)
|
||||
|
||||
#### notificationHandlers.ts (31 текст)
|
||||
- [ ] Настройки уведомлений - ~15 текстов
|
||||
- [ ] Меню уведомлений - ~10 текстов
|
||||
- [ ] Обработка включения/выключения - ~6 текстов
|
||||
|
||||
**Прогресс:** 0/31 (0%)
|
||||
|
||||
### Приоритет 2: Сервисы (MEDIUM)
|
||||
|
||||
#### notificationService.ts (22 текста)
|
||||
- [ ] Уведомления о лайках - ~8 текстов
|
||||
- [ ] Уведомления о матчах - ~8 текстов
|
||||
- [ ] Уведомления о сообщениях - ~6 текстов
|
||||
|
||||
**Прогресс:** 0/22 (0%)
|
||||
|
||||
### Приоритет 3: Контроллеры (MEDIUM)
|
||||
|
||||
#### vipController.ts (21 текст)
|
||||
- [ ] VIP поиск - ~10 текстов
|
||||
- [ ] Фильтры - ~8 текстов
|
||||
- [ ] Перевод анкет - ~3 текста
|
||||
|
||||
**Прогресс:** 0/21 (0%)
|
||||
|
||||
#### profileEditController.ts (21 текст)
|
||||
- [ ] Редактирование полей - ~15 текстов
|
||||
- [ ] Валидация - ~6 текстов
|
||||
|
||||
**Прогресс:** 0/21 (0%)
|
||||
|
||||
### Приоритет 4: Команды (LOW)
|
||||
|
||||
#### commandHandlers.ts (6 текстов)
|
||||
- [ ] Справка (/help) - ~3 текста
|
||||
- [ ] Команды - ~3 текста
|
||||
|
||||
**Прогресс:** 0/6 (0%)
|
||||
|
||||
### Прочие файлы (LOW)
|
||||
|
||||
- [ ] enhancedChatHandlers.ts (4 текста)
|
||||
- [ ] likeBackHandler.ts (2 текста)
|
||||
|
||||
**Прогресс:** 0/6 (0%)
|
||||
|
||||
**ИТОГО ФАЗА 2:** 0/239 (0%)
|
||||
|
||||
## Фаза 3: Переводы ⏳ НЕ НАЧАТО
|
||||
|
||||
### Базовые секции (для всех 9 языков)
|
||||
|
||||
- [ ] **language.*** - Секция управления языком
|
||||
- [ ] es (Español)
|
||||
- [ ] fr (Français)
|
||||
- [ ] de (Deutsch)
|
||||
- [ ] it (Italiano)
|
||||
- [ ] pt (Português)
|
||||
- [ ] ko (한국어)
|
||||
- [ ] zh (中文)
|
||||
- [ ] ja (日本語)
|
||||
- [ ] kk (Қазақша)
|
||||
- [ ] uz (O'zbek)
|
||||
|
||||
### Полные переводы
|
||||
|
||||
После завершения Фазы 2, перевести все новые ключи:
|
||||
|
||||
- [ ] **es.json** (Español) - 0% готовности
|
||||
- [ ] **fr.json** (Français) - 0% готовности
|
||||
- [ ] **de.json** (Deutsch) - 0% готовности
|
||||
- [ ] **it.json** (Italiano) - 0% готовности
|
||||
- [ ] **pt.json** (Português) - 0% готовности
|
||||
- [ ] **ko.json** (한국어) - 0% готовности
|
||||
- [ ] **zh.json** (中文) - 0% готовности
|
||||
- [ ] **ja.json** (日本語) - 0% готовности
|
||||
- [ ] **kk.json** (Қазақша) - 0% готовности
|
||||
- [ ] **uz.json** (O'zbek) - 0% готовности
|
||||
|
||||
**Примечание:** Нанять native speakers или использовать профессиональные сервисы перевода.
|
||||
|
||||
## Фаза 4: Дополнительные функции ⏳ НЕ НАЧАТО
|
||||
|
||||
- [ ] Добавить кнопку "🌍 Язык" в настройки
|
||||
- [ ] Создать команду `/language`
|
||||
- [ ] Добавить автоопределение языка по `msg.from.language_code` (опционально)
|
||||
- [ ] Написать тесты для локализации
|
||||
- [ ] Создать админ-панель для управления переводами (опционально)
|
||||
|
||||
## Прогресс по файлам
|
||||
|
||||
| Файл | Текстов | Заменено | % | Статус |
|
||||
|------|---------|----------|---|--------|
|
||||
| callbackHandlers.ts | 90 | 0 | 0% | ⏳ Не начато |
|
||||
| notificationHandlers.ts | 31 | 0 | 0% | ⏳ Не начато |
|
||||
| notificationService.ts | 22 | 0 | 0% | ⏳ Не начато |
|
||||
| messageHandlers.ts | 21 | 0 | 0% | ⏳ Не начато |
|
||||
| vipController.ts | 21 | 0 | 0% | ⏳ Не начато |
|
||||
| profileEditController.ts | 21 | 0 | 0% | ⏳ Не начато |
|
||||
| commandHandlers.ts | 6 | 0 | 0% | ⏳ Не начато |
|
||||
| enhancedChatHandlers.ts | 4 | 0 | 0% | ⏳ Не начато |
|
||||
| likeBackHandler.ts | 2 | 0 | 0% | ⏳ Не начато |
|
||||
|
||||
**ОБЩИЙ ПРОГРЕСС:** 0/218 (0%)
|
||||
|
||||
*(Исключены скрипты: cleanDb.ts, createTestData.ts, getDatabaseInfo.ts, enhanceNotifications.ts, add-premium-columns.ts)*
|
||||
|
||||
## Оценка времени
|
||||
|
||||
| Фаза | Задача | Время | Статус |
|
||||
|------|--------|-------|--------|
|
||||
| 1 | Инфраструктура | 4-6 ч | ✅ Завершено |
|
||||
| 2 | Замена хардкода | 15-22 ч | ⏳ 0% |
|
||||
| 3 | Переводы (9 языков) | 18-27 ч | ⏳ 0% |
|
||||
| 4 | Доп. функции | 3-5 ч | ⏳ 0% |
|
||||
|
||||
**ИТОГО:** 40-60 часов работы
|
||||
|
||||
**ВЫПОЛНЕНО:** ~5 часов (инфраструктура)
|
||||
**ОСТАЛОСЬ:** ~35-55 часов
|
||||
|
||||
## Еженедельные цели
|
||||
|
||||
### Неделя 1 (текущая)
|
||||
- [x] Создать инфраструктуру локализации
|
||||
- [ ] Заменить тексты в callbackHandlers.ts (90 текстов)
|
||||
- [ ] Заменить тексты в messageHandlers.ts (21 текст)
|
||||
|
||||
**Цель:** Завершить 111 замен (46% от общего)
|
||||
|
||||
### Неделя 2
|
||||
- [ ] Заменить тексты в notificationHandlers.ts (31 текст)
|
||||
- [ ] Заменить тексты в notificationService.ts (22 текста)
|
||||
- [ ] Заменить тексты в контроллерах (42 текста)
|
||||
|
||||
**Цель:** Завершить 95 замен (40% от общего)
|
||||
|
||||
### Неделя 3
|
||||
- [ ] Заменить оставшиеся тексты (12 текстов)
|
||||
- [ ] Начать переводы базовых секций (language.*)
|
||||
- [ ] Перевести 3-4 языка
|
||||
|
||||
**Цель:** Завершить замены (100%), переводы (40%)
|
||||
|
||||
### Неделя 4
|
||||
- [ ] Завершить переводы всех 9 языков
|
||||
- [ ] Добавить кнопку смены языка в настройки
|
||||
- [ ] Финальное тестирование
|
||||
- [ ] Деплой в production
|
||||
|
||||
**Цель:** 100% готовности системы локализации
|
||||
|
||||
## Метрики качества
|
||||
|
||||
- [ ] Все хардкод-тексты заменены (0/218)
|
||||
- [ ] Все языковые файлы содержат одинаковые ключи (0/10)
|
||||
- [ ] Нет TypeScript ошибок
|
||||
- [ ] Нет runtime ошибок при переключении языков
|
||||
- [ ] Все кнопки и меню работают на всех языках
|
||||
- [ ] Документация обновлена
|
||||
- [ ] Написаны тесты (опционально)
|
||||
|
||||
## Как обновлять этот файл
|
||||
|
||||
После замены текстов в файле, обновите прогресс:
|
||||
|
||||
```markdown
|
||||
#### callbackHandlers.ts (90 текстов)
|
||||
- [x] Просмотр профилей (showProfile, showNextCandidate) - ~15 текстов ✅
|
||||
- [ ] Редактирование профиля (edit_name, edit_age, edit_bio) - ~20 текстов
|
||||
...
|
||||
|
||||
**Прогресс:** 15/90 (17%) ⚠️ В процессе
|
||||
```
|
||||
|
||||
## Примечания
|
||||
|
||||
- **⏳** = Не начато
|
||||
- **⚠️** = В процессе
|
||||
- **✅** = Завершено
|
||||
- **❌** = Заблокировано
|
||||
|
||||
---
|
||||
|
||||
**Последнее обновление:** 06.11.2025
|
||||
**Статус проекта:** Фаза 1 завершена, Фаза 2 готова к старту
|
||||
**Общий прогресс:** ~10% (инфраструктура готова)
|
||||
430
docs/LOCALIZATION_MIGRATION_PLAN.md
Normal file
430
docs/LOCALIZATION_MIGRATION_PLAN.md
Normal file
@@ -0,0 +1,430 @@
|
||||
# План замены хардкод-текстов на локализационные ключи
|
||||
|
||||
## Текущее состояние
|
||||
|
||||
✅ **Реализовано:**
|
||||
- Система локализации с i18next
|
||||
- Выбор языка при первом запуске
|
||||
- 10 поддерживаемых языков
|
||||
- Сохранение языка в БД
|
||||
|
||||
⚠️ **Требуется:**
|
||||
- Извлечение и замена ~500+ хардкод-текстов в коде
|
||||
- Дополнение языковых файлов
|
||||
|
||||
## Стратегия замены
|
||||
|
||||
### Фаза 1: Критически важные пользовательские тексты (СНАЧАЛА)
|
||||
|
||||
#### Приоритет: HIGH
|
||||
Файлы с наибольшим количеством пользовательских сообщений:
|
||||
|
||||
1. **src/handlers/messageHandlers.ts** (~150 текстов)
|
||||
- Создание профиля
|
||||
- Ввод данных (имя, возраст, город, био)
|
||||
- Валидация ввода
|
||||
- Сообщения об ошибках
|
||||
|
||||
2. **src/handlers/callbackHandlers.ts** (~200 текстов)
|
||||
- Кнопки меню
|
||||
- Просмотр профилей
|
||||
- Лайки/дислайки
|
||||
- Настройки профиля
|
||||
- VIP функции
|
||||
|
||||
3. **src/handlers/commandHandlers.ts** (~50 текстов)
|
||||
- Команды бота
|
||||
- Главное меню
|
||||
- Справка
|
||||
|
||||
### Фаза 2: Второстепенные тексты
|
||||
|
||||
#### Приоритет: MEDIUM
|
||||
|
||||
4. **src/services/notificationService.ts** (~30 текстов)
|
||||
- Уведомления о лайках
|
||||
- Уведомления о матчах
|
||||
- Уведомления о сообщениях
|
||||
|
||||
5. **src/handlers/notificationHandlers.ts** (~20 текстов)
|
||||
- Настройки уведомлений
|
||||
|
||||
### Фаза 3: Служебные тексты
|
||||
|
||||
#### Приоритет: LOW
|
||||
|
||||
6. **src/services/profileService.ts** (~10 текстов)
|
||||
- Сообщения об ошибках валидации
|
||||
|
||||
7. **src/services/matchingService.ts** (~5 текстов)
|
||||
- Логирование и отладка
|
||||
|
||||
## Процесс замены (пошаговый)
|
||||
|
||||
### Шаг 1: Анализ файла
|
||||
|
||||
```bash
|
||||
# Найти все хардкод-тексты
|
||||
grep -n "'[А-Яа-яЁё]" src/handlers/messageHandlers.ts
|
||||
grep -n '"[А-Яа-яЁё]' src/handlers/messageHandlers.ts
|
||||
```
|
||||
|
||||
### Шаг 2: Создание ключей локализации
|
||||
|
||||
Для каждого найденного текста:
|
||||
|
||||
1. **Определить категорию:**
|
||||
- `profile.*` - профиль
|
||||
- `buttons.*` - кнопки
|
||||
- `errors.*` - ошибки
|
||||
- `messages.*` - сообщения
|
||||
- `commands.*` - команды
|
||||
- `search.*` - поиск
|
||||
- `matches.*` - матчи
|
||||
- `settings.*` - настройки
|
||||
- `notifications.*` - уведомления
|
||||
|
||||
2. **Создать понятный ключ:**
|
||||
```
|
||||
Плохо: "text1", "msg2"
|
||||
Хорошо: "profile.namePrompt", "errors.invalidAge"
|
||||
```
|
||||
|
||||
3. **Добавить в ru.json:**
|
||||
```json
|
||||
{
|
||||
"profile": {
|
||||
"namePrompt": "👤 Введите ваше имя:",
|
||||
"agePrompt": "🎂 Сколько вам лет?",
|
||||
"cityPrompt": "🌍 В каком городе вы находитесь?"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### Шаг 3: Замена в коде
|
||||
|
||||
#### Было:
|
||||
```typescript
|
||||
await bot.sendMessage(chatId, '👤 Введите ваше имя:');
|
||||
```
|
||||
|
||||
#### Стало:
|
||||
```typescript
|
||||
const userId = msg.from?.id.toString();
|
||||
const lang = await profileService.getUserLanguage(userId);
|
||||
localizationService.setLanguage(lang);
|
||||
|
||||
await bot.sendMessage(chatId, localizationService.t('profile.namePrompt'));
|
||||
```
|
||||
|
||||
#### Оптимизация (для методов класса):
|
||||
|
||||
```typescript
|
||||
// В начале метода
|
||||
private async sendLocalizedMessage(
|
||||
chatId: number,
|
||||
userId: string,
|
||||
key: string,
|
||||
options?: any
|
||||
): Promise<void> {
|
||||
const lang = await this.profileService.getUserLanguage(userId);
|
||||
this.localizationService.setLanguage(lang);
|
||||
const text = this.localizationService.t(key, options);
|
||||
await this.bot.sendMessage(chatId, text);
|
||||
}
|
||||
|
||||
// Использование
|
||||
await this.sendLocalizedMessage(chatId, userId, 'profile.namePrompt');
|
||||
```
|
||||
|
||||
### Шаг 4: Перевод на другие языки
|
||||
|
||||
После добавления ключа в `ru.json`, добавить переводы:
|
||||
|
||||
**en.json:**
|
||||
```json
|
||||
{
|
||||
"profile": {
|
||||
"namePrompt": "👤 Enter your name:",
|
||||
"agePrompt": "🎂 How old are you?",
|
||||
"cityPrompt": "🌍 What city are you in?"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
**ko.json:**
|
||||
```json
|
||||
{
|
||||
"profile": {
|
||||
"namePrompt": "👤 이름을 입력하세요:",
|
||||
"agePrompt": "🎂 나이가 어떻게 되세요?",
|
||||
"cityPrompt": "🌍 어느 도시에 계세요?"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
И так для всех 10 языков.
|
||||
|
||||
## Примеры типичных замен
|
||||
|
||||
### 1. Простое сообщение
|
||||
|
||||
**Было:**
|
||||
```typescript
|
||||
await bot.sendMessage(chatId, 'Анкеты закончились! Попробуйте позже.');
|
||||
```
|
||||
|
||||
**Стало:**
|
||||
```typescript
|
||||
await bot.sendMessage(chatId, localizationService.t('search.noProfiles'));
|
||||
```
|
||||
|
||||
### 2. Сообщение с параметрами
|
||||
|
||||
**Было:**
|
||||
```typescript
|
||||
await bot.sendMessage(chatId, `Привет, ${name}! С возвращением!`);
|
||||
```
|
||||
|
||||
**Стало:**
|
||||
```json
|
||||
// ru.json
|
||||
{
|
||||
"welcome": {
|
||||
"greeting": "Привет, {{name}}! С возвращением!"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
```typescript
|
||||
await bot.sendMessage(
|
||||
chatId,
|
||||
localizationService.t('welcome.greeting', { name })
|
||||
);
|
||||
```
|
||||
|
||||
### 3. Кнопки
|
||||
|
||||
**Было:**
|
||||
```typescript
|
||||
const keyboard = {
|
||||
inline_keyboard: [
|
||||
[{ text: '❤️ Нравится', callback_data: 'like' }],
|
||||
[{ text: '👎 Не нравится', callback_data: 'dislike' }]
|
||||
]
|
||||
};
|
||||
```
|
||||
|
||||
**Стало:**
|
||||
```json
|
||||
// ru.json
|
||||
{
|
||||
"buttons": {
|
||||
"like": "❤️ Нравится",
|
||||
"dislike": "👎 Не нравится"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
```typescript
|
||||
const keyboard = {
|
||||
inline_keyboard: [
|
||||
[{
|
||||
text: localizationService.t('buttons.like'),
|
||||
callback_data: 'like'
|
||||
}],
|
||||
[{
|
||||
text: localizationService.t('buttons.dislike'),
|
||||
callback_data: 'dislike'
|
||||
}]
|
||||
]
|
||||
};
|
||||
```
|
||||
|
||||
### 4. Множественное число (плюрализация)
|
||||
|
||||
**Было:**
|
||||
```typescript
|
||||
const text = count === 1
|
||||
? `У вас ${count} новый матч`
|
||||
: `У вас ${count} новых матчей`;
|
||||
```
|
||||
|
||||
**Стало:**
|
||||
```json
|
||||
// ru.json
|
||||
{
|
||||
"matches": {
|
||||
"newCount_one": "У вас {{count}} новый матч",
|
||||
"newCount_few": "У вас {{count}} новых матча",
|
||||
"newCount_many": "У вас {{count}} новых матчей"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
```typescript
|
||||
await bot.sendMessage(
|
||||
chatId,
|
||||
localizationService.t('matches.newCount', { count })
|
||||
);
|
||||
```
|
||||
|
||||
## Инструменты для автоматизации
|
||||
|
||||
### Скрипт поиска хардкод-текстов
|
||||
|
||||
```bash
|
||||
#!/bin/bash
|
||||
# find_hardcoded_texts.sh
|
||||
|
||||
echo "Поиск русских текстов в кавычках..."
|
||||
grep -rn "'[А-Яа-яЁё]" src/ --include="*.ts" | wc -l
|
||||
grep -rn '"[А-Яа-яЁё]' src/ --include="*.ts" | wc -l
|
||||
|
||||
echo "Топ-10 файлов с наибольшим количеством хардкода:"
|
||||
grep -rn "'[А-Яа-яЁё]\|\"[А-Яа-яЁё]" src/ --include="*.ts" | \
|
||||
cut -d: -f1 | \
|
||||
sort | \
|
||||
uniq -c | \
|
||||
sort -rn | \
|
||||
head -10
|
||||
```
|
||||
|
||||
### Скрипт проверки покрытия переводами
|
||||
|
||||
```typescript
|
||||
// scripts/check-translations.ts
|
||||
import * as fs from 'fs';
|
||||
import * as path from 'path';
|
||||
|
||||
const localesPath = path.join(__dirname, '..', 'src', 'locales');
|
||||
const ruFile = JSON.parse(fs.readFileSync(path.join(localesPath, 'ru.json'), 'utf8'));
|
||||
|
||||
function getAllKeys(obj: any, prefix = ''): string[] {
|
||||
let keys: string[] = [];
|
||||
for (const key in obj) {
|
||||
const fullKey = prefix ? `${prefix}.${key}` : key;
|
||||
if (typeof obj[key] === 'object' && !Array.isArray(obj[key])) {
|
||||
keys = keys.concat(getAllKeys(obj[key], fullKey));
|
||||
} else {
|
||||
keys.push(fullKey);
|
||||
}
|
||||
}
|
||||
return keys;
|
||||
}
|
||||
|
||||
const ruKeys = getAllKeys(ruFile);
|
||||
const languages = ['en', 'es', 'fr', 'de', 'it', 'pt', 'zh', 'ja', 'ko'];
|
||||
|
||||
languages.forEach(lang => {
|
||||
const langFile = JSON.parse(fs.readFileSync(path.join(localesPath, `${lang}.json`), 'utf8'));
|
||||
const langKeys = getAllKeys(langFile);
|
||||
|
||||
const missing = ruKeys.filter(key => !langKeys.includes(key));
|
||||
|
||||
console.log(`\n${lang}.json:`);
|
||||
console.log(` Всего ключей: ${langKeys.length}/${ruKeys.length}`);
|
||||
if (missing.length > 0) {
|
||||
console.log(` Отсутствуют: ${missing.length}`);
|
||||
console.log(` Пример: ${missing.slice(0, 5).join(', ')}`);
|
||||
} else {
|
||||
console.log(` ✅ Все ключи присутствуют`);
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
## Контрольный список (Checklist)
|
||||
|
||||
### Перед началом замены файла:
|
||||
|
||||
- [ ] Сделать backup файла или создать новую ветку в Git
|
||||
- [ ] Прочитать весь файл, понять структуру
|
||||
- [ ] Составить список всех текстов для замены
|
||||
|
||||
### В процессе замены:
|
||||
|
||||
- [ ] Заменять по 10-20 текстов за раз
|
||||
- [ ] Тестировать после каждой замены
|
||||
- [ ] Проверять TypeScript ошибки: `npm run build`
|
||||
- [ ] Коммитить изменения: `git commit -m "localize: messageHandlers profile section"`
|
||||
|
||||
### После замены файла:
|
||||
|
||||
- [ ] Убедиться, что нет TypeScript ошибок
|
||||
- [ ] Протестировать все функции файла в боте
|
||||
- [ ] Обновить переводы для всех 10 языков
|
||||
- [ ] Запустить скрипт проверки покрытия
|
||||
- [ ] Создать Pull Request для review
|
||||
|
||||
## Рекомендации
|
||||
|
||||
1. **Начинайте с самого используемого функционала:**
|
||||
- Регистрация (messageHandlers.ts - createProfile)
|
||||
- Просмотр анкет (callbackHandlers.ts - showNextCandidate)
|
||||
- Главное меню (commandHandlers.ts - handleStart)
|
||||
|
||||
2. **Группируйте ключи логически:**
|
||||
```json
|
||||
{
|
||||
"profile": {
|
||||
"prompts": {
|
||||
"name": "...",
|
||||
"age": "...",
|
||||
"city": "..."
|
||||
},
|
||||
"validation": {
|
||||
"nameLength": "...",
|
||||
"ageRange": "...",
|
||||
"cityRequired": "..."
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
3. **Используйте консистентную нотацию:**
|
||||
- Всегда camelCase для ключей
|
||||
- Всегда точки для разделения уровней
|
||||
- Prefix для категории (profile, button, error)
|
||||
|
||||
4. **Не переводите:**
|
||||
- Технические логи
|
||||
- Callback_data значения
|
||||
- Имена переменных и функций
|
||||
|
||||
5. **Делайте переводы качественными:**
|
||||
- Нанимайте native speakers для перевода
|
||||
- Используйте контекст культуры (эмодзи, формальность)
|
||||
- Учитывайте длину текста (для кнопок)
|
||||
|
||||
## Оценка объема работ
|
||||
|
||||
### Время на замену (приблизительно):
|
||||
|
||||
- **messageHandlers.ts**: 4-6 часов
|
||||
- **callbackHandlers.ts**: 6-8 часов
|
||||
- **commandHandlers.ts**: 2-3 часа
|
||||
- **notificationService.ts**: 1-2 часа
|
||||
- **Прочие файлы**: 2-3 часа
|
||||
|
||||
**Итого на замену:** ~15-22 часа
|
||||
|
||||
### Время на переводы:
|
||||
|
||||
- **1 язык (native speaker)**: 2-3 часа
|
||||
- **9 языков**: 18-27 часов
|
||||
|
||||
**ОБЩИЙ ОБЪЕМ:** ~33-49 часов
|
||||
|
||||
## Следующий шаг
|
||||
|
||||
Начните с файла **src/handlers/messageHandlers.ts**, секция создания профиля:
|
||||
|
||||
```bash
|
||||
# Создайте ветку для работы
|
||||
git checkout -b localization-phase1-message-handlers
|
||||
|
||||
# Начните замену
|
||||
code src/handlers/messageHandlers.ts
|
||||
```
|
||||
|
||||
Удачи! 🚀
|
||||
226
docs/LOCALIZATION_QUICKSTART.md
Normal file
226
docs/LOCALIZATION_QUICKSTART.md
Normal file
@@ -0,0 +1,226 @@
|
||||
# Быстрый старт: Система локализации
|
||||
|
||||
## Что уже работает ✅
|
||||
|
||||
1. **Выбор языка при первом запуске** - новые пользователи видят меню из 10 языков
|
||||
2. **Сохранение языка в БД** - колонка `users.lang` хранит выбор пользователя
|
||||
3. **10 поддерживаемых языков** - ru, en, es, fr, de, it, pt, ko, zh, ja
|
||||
4. **Инфраструктура i18next** - готова к использованию
|
||||
|
||||
## Что нужно сделать ⚠️
|
||||
|
||||
### ГЛАВНАЯ ЗАДАЧА: Заменить 255 хардкод-текстов
|
||||
|
||||
**Файлы по приоритету:**
|
||||
1. `src/handlers/callbackHandlers.ts` - 90 текстов (кнопки, меню)
|
||||
2. `src/handlers/notificationHandlers.ts` - 31 текст (уведомления)
|
||||
3. `src/services/notificationService.ts` - 22 текста (сервис уведомлений)
|
||||
4. `src/handlers/messageHandlers.ts` - 21 текст (создание профиля)
|
||||
5. Остальные файлы - ~91 текст
|
||||
|
||||
## Как использовать локализацию в коде
|
||||
|
||||
### Вариант 1: Через LocalizationService
|
||||
|
||||
```typescript
|
||||
import LocalizationService from '../services/localizationService';
|
||||
|
||||
// В методе класса:
|
||||
const locService = LocalizationService.getInstance();
|
||||
const userId = msg.from?.id.toString();
|
||||
const lang = await this.profileService.getUserLanguage(userId);
|
||||
|
||||
locService.setLanguage(lang);
|
||||
const text = locService.t('profile.namePrompt');
|
||||
|
||||
await this.bot.sendMessage(chatId, text);
|
||||
```
|
||||
|
||||
### Вариант 2: Через getUserTranslation (рекомендуется)
|
||||
|
||||
```typescript
|
||||
import { getUserTranslation } from '../services/localizationService';
|
||||
|
||||
const userId = msg.from?.id.toString();
|
||||
const text = await getUserTranslation(userId, 'profile.namePrompt');
|
||||
|
||||
await this.bot.sendMessage(chatId, text);
|
||||
```
|
||||
|
||||
### Вариант 3: С параметрами
|
||||
|
||||
```typescript
|
||||
// В ru.json:
|
||||
{
|
||||
"welcome": {
|
||||
"greeting": "Привет, {{name}}! Добро пожаловать!"
|
||||
}
|
||||
}
|
||||
|
||||
// В коде:
|
||||
const text = await getUserTranslation(userId, 'welcome.greeting', { name: userName });
|
||||
```
|
||||
|
||||
## Процесс замены текста
|
||||
|
||||
### ШАГ 1: Найти хардкод-текст
|
||||
|
||||
```bash
|
||||
# Найти все тексты в файле
|
||||
grep -n "'[А-Яа-яЁё]\|\"[А-Яа-яЁё]" src/handlers/callbackHandlers.ts
|
||||
```
|
||||
|
||||
### ШАГ 2: Добавить ключ в ru.json
|
||||
|
||||
**Было в коде:**
|
||||
```typescript
|
||||
await bot.sendMessage(chatId, '👤 Введите ваше имя:');
|
||||
```
|
||||
|
||||
**Добавляем в ru.json:**
|
||||
```json
|
||||
{
|
||||
"profile": {
|
||||
"prompts": {
|
||||
"name": "👤 Введите ваше имя:"
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### ШАГ 3: Заменить в коде
|
||||
|
||||
```typescript
|
||||
const text = await getUserTranslation(userId, 'profile.prompts.name');
|
||||
await bot.sendMessage(chatId, text);
|
||||
```
|
||||
|
||||
### ШАГ 4: Добавить перевод в en.json
|
||||
|
||||
```json
|
||||
{
|
||||
"profile": {
|
||||
"prompts": {
|
||||
"name": "👤 Enter your name:"
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### ШАГ 5: Протестировать
|
||||
|
||||
```bash
|
||||
# Пересобрать
|
||||
docker compose up -d --build bot
|
||||
|
||||
# Проверить
|
||||
docker compose logs bot --tail 20
|
||||
```
|
||||
|
||||
## Полезные команды
|
||||
|
||||
```bash
|
||||
# Найти все хардкод-тексты
|
||||
./bin/find_hardcoded_texts.sh
|
||||
|
||||
# Посмотреть тексты в конкретном файле
|
||||
grep -n "'[А-Яа-яЁё]\|\"[А-Яа-яЁё]" src/handlers/callbackHandlers.ts
|
||||
|
||||
# Собрать и запустить бота
|
||||
docker compose up -d --build bot
|
||||
|
||||
# Проверить логи
|
||||
docker compose logs bot --tail 50 -f
|
||||
|
||||
# Применить миграцию БД (если еще не применена)
|
||||
PGPASSWORD='Cl0ud_1985!' psql -h 192.168.0.102 -p 5432 -U trevor \
|
||||
-d telegram_tinder_bot -f sql/add_user_language.sql
|
||||
```
|
||||
|
||||
## Структура ключей (рекомендуется)
|
||||
|
||||
```json
|
||||
{
|
||||
"language": { ... }, // Управление языком
|
||||
"welcome": { ... }, // Приветствия
|
||||
"profile": {
|
||||
"prompts": { ... }, // Запросы ввода
|
||||
"validation": { ... }, // Ошибки валидации
|
||||
"labels": { ... } // Метки полей
|
||||
},
|
||||
"buttons": { ... }, // Кнопки
|
||||
"errors": { ... }, // Общие ошибки
|
||||
"commands": { ... }, // Команды бота
|
||||
"search": { ... }, // Поиск анкет
|
||||
"matches": { ... }, // Матчи
|
||||
"notifications": { ... }, // Уведомления
|
||||
"settings": { ... }, // Настройки
|
||||
"vip": { ... } // VIP функции
|
||||
}
|
||||
```
|
||||
|
||||
## Пример: Замена кнопок
|
||||
|
||||
**Было:**
|
||||
```typescript
|
||||
const keyboard = {
|
||||
inline_keyboard: [
|
||||
[{ text: '❤️ Нравится', callback_data: 'like' }],
|
||||
[{ text: '👎 Не нравится', callback_data: 'dislike' }]
|
||||
]
|
||||
};
|
||||
```
|
||||
|
||||
**Добавили в ru.json:**
|
||||
```json
|
||||
{
|
||||
"buttons": {
|
||||
"like": "❤️ Нравится",
|
||||
"dislike": "👎 Не нравится"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
**Стало:**
|
||||
```typescript
|
||||
const userId = msg.from?.id.toString();
|
||||
const lang = await this.profileService.getUserLanguage(userId);
|
||||
this.localizationService.setLanguage(lang);
|
||||
|
||||
const keyboard = {
|
||||
inline_keyboard: [
|
||||
[{
|
||||
text: this.localizationService.t('buttons.like'),
|
||||
callback_data: 'like'
|
||||
}],
|
||||
[{
|
||||
text: this.localizationService.t('buttons.dislike'),
|
||||
callback_data: 'dislike'
|
||||
}]
|
||||
]
|
||||
};
|
||||
```
|
||||
|
||||
## Документация
|
||||
|
||||
- **docs/LOCALIZATION_SYSTEM.md** - Полное описание системы
|
||||
- **docs/LOCALIZATION_MIGRATION_PLAN.md** - Детальный план замены текстов
|
||||
- **docs/LOCALIZATION_REPORT.md** - Отчет о выполненной работе
|
||||
|
||||
## Следующий шаг
|
||||
|
||||
**Начните с самого крупного файла:**
|
||||
|
||||
```bash
|
||||
# Посмотреть все тексты
|
||||
grep -n "'[А-Яа-яЁё]\|\"[А-Яа-яЁё]" src/handlers/callbackHandlers.ts
|
||||
|
||||
# Открыть файл
|
||||
code src/handlers/callbackHandlers.ts
|
||||
```
|
||||
|
||||
**Заменяйте по 10-20 текстов за раз, тестируйте после каждой замены!**
|
||||
|
||||
---
|
||||
|
||||
Удачи! 🚀
|
||||
377
docs/LOCALIZATION_REPORT.md
Normal file
377
docs/LOCALIZATION_REPORT.md
Normal file
@@ -0,0 +1,377 @@
|
||||
# Отчет о реализации системы локализации
|
||||
|
||||
**Дата:** 06.11.2025
|
||||
**Ветка:** localization
|
||||
**Статус:** ✅ Система локализации внедрена и работает
|
||||
|
||||
## Выполненные задачи
|
||||
|
||||
### 1. ✅ База данных
|
||||
|
||||
**Файл:** `sql/add_user_language.sql`
|
||||
|
||||
- Добавлена колонка `lang VARCHAR(5) DEFAULT 'ru' NOT NULL` в таблицу `users`
|
||||
- Создан индекс `idx_users_lang` для оптимизации запросов
|
||||
- Все существующие пользователи получили язык `ru` по умолчанию
|
||||
- Миграция успешно применена к production БД
|
||||
|
||||
**Результат:**
|
||||
```sql
|
||||
SELECT COUNT(*), lang FROM users GROUP BY lang;
|
||||
-- 2 пользователя с lang='ru'
|
||||
```
|
||||
|
||||
### 2. ✅ Обработчик языков
|
||||
|
||||
**Файл:** `src/handlers/languageHandlers.ts` (НОВЫЙ)
|
||||
|
||||
Реализован класс `LanguageHandlers` с методами:
|
||||
- `showLanguageSelection()` - показать меню из 10 языков с флагами
|
||||
- `handleSetLanguage()` - обработать выбор языка пользователем
|
||||
- `checkAndShowLanguageSelection()` - автоматически показывать выбор новым пользователям
|
||||
|
||||
**Функционал:**
|
||||
- Интеграция с `ProfileService` для сохранения языка
|
||||
- Интеграция с `LocalizationService` для смены языка
|
||||
- Автоматическое удаление меню выбора после установки языка
|
||||
- Показ приветственного сообщения на выбранном языке
|
||||
|
||||
### 3. ✅ Расширение ProfileService
|
||||
|
||||
**Файл:** `src/services/profileService.ts` (ОБНОВЛЕН)
|
||||
|
||||
Добавлены методы:
|
||||
```typescript
|
||||
async ensureUser(telegramId, userData, language = 'ru'): Promise<string>
|
||||
async updateUserLanguage(telegramId, language): Promise<void>
|
||||
async getUserLanguage(telegramId): Promise<string>
|
||||
```
|
||||
|
||||
**Изменения:**
|
||||
- `INSERT INTO users` теперь включает колонку `lang`
|
||||
- UPSERT сохраняет существующий язык пользователя (не перезаписывает)
|
||||
|
||||
### 4. ✅ Интеграция в основной бот
|
||||
|
||||
**Файл:** `src/bot.ts` (ОБНОВЛЕН)
|
||||
|
||||
- Добавлен import `LanguageHandlers`
|
||||
- Создан экземпляр `this.languageHandlers = new LanguageHandlers(this.bot)`
|
||||
- Инициализация происходит при старте бота
|
||||
|
||||
**Файл:** `src/handlers/commandHandlers.ts` (ОБНОВЛЕН)
|
||||
|
||||
- В метод `handleStart()` добавлена проверка:
|
||||
```typescript
|
||||
const languageSelectionShown = await this.languageHandlers.checkAndShowLanguageSelection(userId, chatId);
|
||||
if (languageSelectionShown) {
|
||||
return; // Показываем выбор языка и выходим
|
||||
}
|
||||
```
|
||||
- Новым пользователям сначала показывается выбор языка, затем приветствие
|
||||
|
||||
**Файл:** `src/handlers/callbackHandlers.ts` (ОБНОВЛЕН)
|
||||
|
||||
- Добавлена обработка callback `set_lang_{код}`:
|
||||
```typescript
|
||||
if (data.startsWith('set_lang_')) {
|
||||
const languageHandlers = new LanguageHandlers(this.bot);
|
||||
await languageHandlers.handleSetLanguage(query);
|
||||
return;
|
||||
}
|
||||
```
|
||||
|
||||
### 5. ✅ Локализационные файлы
|
||||
|
||||
Обновлены файлы:
|
||||
- `src/locales/ru.json` - добавлена секция `language`
|
||||
- `src/locales/en.json` - добавлена секция `language`
|
||||
|
||||
**Структура секции:**
|
||||
```json
|
||||
{
|
||||
"language": {
|
||||
"select": "🌍 Выберите язык интерфейса:...",
|
||||
"changed": "✅ Язык изменен на Русский",
|
||||
"ru": "🇷🇺 Русский",
|
||||
"en": "🇬🇧 English",
|
||||
"es": "🇪🇸 Español",
|
||||
"fr": "🇫🇷 Français",
|
||||
"de": "🇩🇪 Deutsch",
|
||||
"it": "🇮🇹 Italiano",
|
||||
"pt": "🇵🇹 Português",
|
||||
"zh": "🇨🇳 中文",
|
||||
"ja": "🇯🇵 日本語",
|
||||
"ko": "🇰🇷 한국어"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 6. ✅ Документация
|
||||
|
||||
Созданы документы:
|
||||
1. **docs/LOCALIZATION_SYSTEM.md** - полное описание системы локализации
|
||||
2. **docs/LOCALIZATION_MIGRATION_PLAN.md** - план миграции хардкод-текстов
|
||||
3. **bin/find_hardcoded_texts.sh** - скрипт поиска хардкод-текстов
|
||||
|
||||
### 7. ✅ Тестирование
|
||||
|
||||
- Docker build: успешно ✅
|
||||
- Запуск бота: успешно ✅
|
||||
- Логи: `✅ Localization service initialized successfully`
|
||||
- Бот работает: @seoulmate_officialbot
|
||||
|
||||
## Поддерживаемые языки (10)
|
||||
|
||||
| Код | Язык | Файл | Статус |
|
||||
|-----|-----------|-----------|--------|
|
||||
| ru | Русский | ru.json | ✅ Базовые ключи |
|
||||
| en | English | en.json | ✅ Базовые ключи |
|
||||
| es | Español | es.json | ⚠️ Требуется дополнение |
|
||||
| fr | Français | fr.json | ⚠️ Требуется дополнение |
|
||||
| de | Deutsch | de.json | ⚠️ Требуется дополнение |
|
||||
| it | Italiano | it.json | ⚠️ Требуется дополнение |
|
||||
| pt | Português | pt.json | ⚠️ Требуется дополнение |
|
||||
| ko | 한국어 | ko.json | ⚠️ Требуется дополнение |
|
||||
| zh | 中文 | zh.json | ⚠️ Требуется дополнение |
|
||||
| ja | 日本語 | ja.json | ⚠️ Требуется дополнение |
|
||||
|
||||
## Пользовательский опыт (UX Flow)
|
||||
|
||||
### Новый пользователь:
|
||||
|
||||
```
|
||||
Пользователь → /start
|
||||
↓
|
||||
Бот проверяет: есть ли профиль?
|
||||
↓ НЕТ
|
||||
Показывает меню выбора из 10 языков
|
||||
↓
|
||||
Пользователь нажимает, например, "🇰🇷 한국어"
|
||||
↓
|
||||
Callback: set_lang_ko
|
||||
↓
|
||||
UPDATE users SET lang='ko' WHERE telegram_id=...
|
||||
↓
|
||||
Localization сервис переключается на корейский
|
||||
↓
|
||||
Приветственное сообщение на корейском
|
||||
↓
|
||||
Кнопка "Создать профиль" на корейском
|
||||
```
|
||||
|
||||
### Существующий пользователь:
|
||||
|
||||
```
|
||||
Пользователь → /start
|
||||
↓
|
||||
Бот загружает язык из БД (например, 'en')
|
||||
↓
|
||||
Устанавливает язык в LocalizationService
|
||||
↓
|
||||
Показывает главное меню на английском
|
||||
```
|
||||
|
||||
## Статистика хардкод-текстов
|
||||
|
||||
**Результат анализа (`./bin/find_hardcoded_texts.sh`):**
|
||||
|
||||
```
|
||||
Тексты в одинарных кавычках: 217
|
||||
Тексты в двойных кавычках: 38
|
||||
ВСЕГО: 255
|
||||
```
|
||||
|
||||
**Топ-5 файлов для замены:**
|
||||
|
||||
| Файл | Количество текстов |
|
||||
|------|-------------------|
|
||||
| callbackHandlers.ts | 90 |
|
||||
| notificationHandlers.ts | 31 |
|
||||
| notificationService.ts | 22 |
|
||||
| messageHandlers.ts | 21 |
|
||||
| vipController.ts | 21 |
|
||||
|
||||
## Следующие шаги
|
||||
|
||||
### Фаза 2: Замена хардкод-текстов (ПРИОРИТЕТ)
|
||||
|
||||
**Оценка времени:** 15-22 часа
|
||||
|
||||
1. **messageHandlers.ts** (21 текст) - 2-3 часа
|
||||
- Регистрация пользователя
|
||||
- Создание профиля
|
||||
- Валидация ввода
|
||||
|
||||
2. **callbackHandlers.ts** (90 текстов) - 6-8 часов
|
||||
- Кнопки меню
|
||||
- Просмотр профилей
|
||||
- Лайки/дислайки
|
||||
- Настройки
|
||||
|
||||
3. **notificationHandlers.ts + notificationService.ts** (53 текста) - 3-4 часа
|
||||
- Уведомления о лайках
|
||||
- Уведомления о матчах
|
||||
- Настройки уведомлений
|
||||
|
||||
4. **commandHandlers.ts** (6 текстов) - 1 час
|
||||
- Команды бота
|
||||
- Справка
|
||||
|
||||
5. **Контроллеры** (42 текста) - 3-4 часа
|
||||
- vipController.ts
|
||||
- profileEditController.ts
|
||||
|
||||
### Фаза 3: Переводы (ПОСЛЕ ЗАМЕНЫ)
|
||||
|
||||
**Оценка времени:** 18-27 часов (2-3 часа на язык × 9 языков)
|
||||
|
||||
Необходимо перевести все новые ключи на 9 языков:
|
||||
- es, fr, de, it, pt - Европейские языки
|
||||
- ko, zh, ja - Азиатские языки
|
||||
|
||||
**Рекомендация:** Нанять native speakers или использовать профессиональные переводческие сервисы.
|
||||
|
||||
### Фаза 4: Дополнительные функции
|
||||
|
||||
1. Добавить кнопку "🌍 Язык / Language" в настройки
|
||||
2. Добавить команду `/language` для быстрой смены языка
|
||||
3. Автоопределение языка по `msg.from.language_code` (опционально)
|
||||
|
||||
## Технические детали
|
||||
|
||||
### Database Schema
|
||||
|
||||
```sql
|
||||
-- Колонка добавлена в таблицу users
|
||||
lang VARCHAR(5) DEFAULT 'ru' NOT NULL
|
||||
|
||||
-- Индекс создан для оптимизации
|
||||
CREATE INDEX idx_users_lang ON users(lang);
|
||||
```
|
||||
|
||||
### Callback Data Format
|
||||
|
||||
Все callback для выбора языка имеют формат:
|
||||
```
|
||||
set_lang_{код_ISO_639-1}
|
||||
```
|
||||
|
||||
Примеры:
|
||||
- `set_lang_ru` → Русский
|
||||
- `set_lang_en` → English
|
||||
- `set_lang_ko` → 한국어
|
||||
|
||||
### Localization Keys Structure
|
||||
|
||||
```json
|
||||
{
|
||||
"language.*": "Управление языком",
|
||||
"welcome.*": "Приветствия",
|
||||
"profile.*": "Профиль",
|
||||
"buttons.*": "Кнопки",
|
||||
"errors.*": "Ошибки",
|
||||
"commands.*": "Команды",
|
||||
"search.*": "Поиск",
|
||||
"matches.*": "Матчи",
|
||||
"notifications.*": "Уведомления",
|
||||
"settings.*": "Настройки",
|
||||
"vip.*": "VIP функции"
|
||||
}
|
||||
```
|
||||
|
||||
## Проблемы и решения
|
||||
|
||||
### Проблема 1: Инициализация LanguageHandlers
|
||||
**Проблема:** TypeScript ошибка "свойство не имеет инициализатора"
|
||||
**Решение:** Добавлена инициализация в конструктор `this.languageHandlers = new LanguageHandlers(bot)`
|
||||
|
||||
### Проблема 2: Новый пользователь vs существующий
|
||||
**Проблема:** Как определить, когда показывать выбор языка?
|
||||
**Решение:** Метод `checkAndShowLanguageSelection()` проверяет наличие профиля
|
||||
|
||||
### Проблема 3: Сохранение выбранного языка
|
||||
**Проблема:** Где хранить язык пользователя?
|
||||
**Решение:** Колонка `lang` в таблице `users`, методы в `ProfileService`
|
||||
|
||||
## Выводы
|
||||
|
||||
### Что работает ✅
|
||||
|
||||
1. **Автоматический выбор языка для новых пользователей**
|
||||
- Показывается меню из 10 языков
|
||||
- Язык сохраняется в БД
|
||||
- Приветствие показывается на выбранном языке
|
||||
|
||||
2. **Сохранение языка пользователя**
|
||||
- Язык хранится в колонке `users.lang`
|
||||
- Загружается при каждом запросе
|
||||
- Используется для всех сообщений
|
||||
|
||||
3. **Инфраструктура локализации**
|
||||
- `LocalizationService` работает с i18next
|
||||
- 10 языковых файлов готовы
|
||||
- Методы `t()`, `setLanguage()`, `getCurrentLanguage()` работают
|
||||
|
||||
### Что требует доработки ⚠️
|
||||
|
||||
1. **Замена 255 хардкод-текстов**
|
||||
- Основная работа впереди
|
||||
- Требуется систематическая замена
|
||||
- Оценка: ~20 часов работы
|
||||
|
||||
2. **Переводы для 9 языков**
|
||||
- Только `ru.json` и `en.json` содержат секцию `language`
|
||||
- Остальные 8 языков требуют перевода
|
||||
- Оценка: ~20 часов (с переводчиками)
|
||||
|
||||
3. **Кнопка смены языка в настройках**
|
||||
- Пока можно сменить только через `/start` (для новых)
|
||||
- Нужна кнопка в меню настроек
|
||||
- Оценка: 1-2 часа
|
||||
|
||||
## Команды для работы
|
||||
|
||||
### Применить миграцию БД:
|
||||
```bash
|
||||
PGPASSWORD='Cl0ud_1985!' psql -h 192.168.0.102 -p 5432 -U trevor -d telegram_tinder_bot -f sql/add_user_language.sql
|
||||
```
|
||||
|
||||
### Найти хардкод-тексты:
|
||||
```bash
|
||||
./bin/find_hardcoded_texts.sh
|
||||
```
|
||||
|
||||
### Посмотреть тексты в файле:
|
||||
```bash
|
||||
grep -n "'[А-Яа-яЁё]\|\"[А-Яа-яЁё]" src/handlers/callbackHandlers.ts
|
||||
```
|
||||
|
||||
### Собрать и запустить бота:
|
||||
```bash
|
||||
docker compose up -d --build bot
|
||||
```
|
||||
|
||||
### Проверить логи:
|
||||
```bash
|
||||
docker compose logs bot --tail 50
|
||||
```
|
||||
|
||||
## Итог
|
||||
|
||||
✅ **Система локализации полностью внедрена и работает!**
|
||||
|
||||
Бот теперь:
|
||||
- Спрашивает язык у новых пользователей
|
||||
- Сохраняет язык в базе данных
|
||||
- Поддерживает 10 языков
|
||||
- Готов к замене всех хардкод-текстов
|
||||
|
||||
**Следующий шаг:** Начать систематическую замену хардкод-текстов, начиная с `callbackHandlers.ts` (90 текстов).
|
||||
|
||||
---
|
||||
|
||||
**Разработчик:** GitHub Copilot
|
||||
**Заказчик:** Trevor
|
||||
**Дата завершения:** 06.11.2025
|
||||
**Статус:** ✅ ГОТОВО К ИСПОЛЬЗОВАНИЮ
|
||||
329
docs/LOCALIZATION_SYSTEM.md
Normal file
329
docs/LOCALIZATION_SYSTEM.md
Normal file
@@ -0,0 +1,329 @@
|
||||
# Система локализации Telegram Tinder Bot
|
||||
|
||||
## Обзор
|
||||
|
||||
Реализована полноценная система мультиязычной поддержки бота с возможностью выбора языка интерфейса.
|
||||
|
||||
## Реализованные функции
|
||||
|
||||
### 1. База данных
|
||||
|
||||
**Миграция:** `sql/add_user_language.sql`
|
||||
|
||||
Добавлена колонка `lang` в таблицу `users`:
|
||||
- Тип: `VARCHAR(5)`
|
||||
- Значение по умолчанию: `'ru'` (Русский)
|
||||
- NOT NULL constraint
|
||||
- Индекс для быстрого поиска: `idx_users_lang`
|
||||
|
||||
```sql
|
||||
ALTER TABLE users
|
||||
ADD COLUMN IF NOT EXISTS lang VARCHAR(5) DEFAULT 'ru' NOT NULL;
|
||||
|
||||
CREATE INDEX IF NOT EXISTS idx_users_lang ON users(lang);
|
||||
```
|
||||
|
||||
### 2. Поддерживаемые языки
|
||||
|
||||
Бот поддерживает **10 языков**:
|
||||
|
||||
| Код | Язык | Флаг | Файл локализации |
|
||||
|------|-----------|------|------------------|
|
||||
| `ru` | Русский | 🇷🇺 | `ru.json` |
|
||||
| `en` | English | 🇬🇧 | `en.json` |
|
||||
| `es` | Español | 🇪🇸 | `es.json` |
|
||||
| `fr` | Français | 🇫🇷 | `fr.json` |
|
||||
| `de` | Deutsch | 🇩🇪 | `de.json` |
|
||||
| `it` | Italiano | 🇮🇹 | `it.json` |
|
||||
| `pt` | Português | 🇵🇹 | `pt.json` |
|
||||
| `ko` | 한국어 | 🇰🇷 | `ko.json` |
|
||||
| `zh` | 中文 | 🇨🇳 | `zh.json` |
|
||||
| `ja` | 日本語 | 🇯🇵 | `ja.json` |
|
||||
|
||||
### 3. Архитектура
|
||||
|
||||
#### LocalizationService (`src/services/localizationService.ts`)
|
||||
|
||||
Сервис на базе `i18next`:
|
||||
- Singleton pattern
|
||||
- Автоматическая загрузка всех языковых файлов при инициализации
|
||||
- Методы:
|
||||
- `initialize()` - инициализация сервиса
|
||||
- `t(key, options)` - получение перевода по ключу
|
||||
- `setLanguage(lang)` - смена языка
|
||||
- `getCurrentLanguage()` - получение текущего языка
|
||||
- `getSupportedLanguages()` - список поддерживаемых языков
|
||||
- `getTranslation(key, lang, options)` - получить перевод для конкретного языка без смены текущего
|
||||
|
||||
#### LanguageHandlers (`src/handlers/languageHandlers.ts`)
|
||||
|
||||
Обработчик выбора и управления языком:
|
||||
- `showLanguageSelection(chatId, messageId?)` - показать меню выбора языка
|
||||
- `handleSetLanguage(query)` - обработать установку языка
|
||||
- `checkAndShowLanguageSelection(userId, chatId)` - проверить, нужно ли показывать выбор языка
|
||||
|
||||
#### ProfileService - Расширение (`src/services/profileService.ts`)
|
||||
|
||||
Добавлены методы для работы с языком пользователя:
|
||||
- `ensureUser(telegramId, userData, language)` - создание/обновление пользователя с сохранением языка
|
||||
- `updateUserLanguage(telegramId, language)` - обновление языка пользователя
|
||||
- `getUserLanguage(telegramId)` - получение языка пользователя
|
||||
|
||||
### 4. Пользовательский опыт (UX)
|
||||
|
||||
#### Новый пользователь
|
||||
|
||||
1. Пользователь отправляет `/start`
|
||||
2. **Автоматически показывается меню выбора языка** (10 кнопок с флагами)
|
||||
3. После выбора языка:
|
||||
- Язык сохраняется в БД
|
||||
- Показывается приветственное сообщение на выбранном языке
|
||||
- Предлагается создать профиль
|
||||
|
||||
#### Существующий пользователь
|
||||
|
||||
1. Пользователь отправляет `/start`
|
||||
2. Бот использует сохраненный язык из БД
|
||||
3. Показывается главное меню на выбранном языке
|
||||
|
||||
#### Изменение языка в настройках
|
||||
|
||||
Запланировано: добавить кнопку "🌍 Язык / Language" в раздел "⚙️ Настройки"
|
||||
|
||||
### 5. Структура локализационных файлов
|
||||
|
||||
Каждый файл `src/locales/{lang}.json` содержит:
|
||||
|
||||
```json
|
||||
{
|
||||
"language": {
|
||||
"select": "🌍 Выберите язык...",
|
||||
"changed": "✅ Язык изменен на...",
|
||||
"ru": "🇷🇺 Русский",
|
||||
"en": "🇬🇧 English",
|
||||
...
|
||||
},
|
||||
"welcome": {
|
||||
"greeting": "Добро пожаловать...",
|
||||
"description": "...",
|
||||
"getStarted": "..."
|
||||
},
|
||||
"profile": { ... },
|
||||
"search": { ... },
|
||||
"buttons": { ... },
|
||||
"errors": { ... },
|
||||
...
|
||||
}
|
||||
```
|
||||
|
||||
### 6. Интеграция в код
|
||||
|
||||
#### Импорт функции перевода
|
||||
|
||||
```typescript
|
||||
import { getUserTranslation } from '../services/localizationService';
|
||||
```
|
||||
|
||||
#### Использование в коде
|
||||
|
||||
```typescript
|
||||
// Асинхронный вызов
|
||||
const text = await getUserTranslation(userId, 'welcome.greeting');
|
||||
|
||||
// Или через сервис
|
||||
const locService = LocalizationService.getInstance();
|
||||
const text = locService.t('welcome.greeting');
|
||||
```
|
||||
|
||||
#### Callback для выбора языка
|
||||
|
||||
Все callback_data для выбора языка имеют формат:
|
||||
```
|
||||
set_lang_{код_языка}
|
||||
```
|
||||
|
||||
Например:
|
||||
- `set_lang_ru` - установить русский
|
||||
- `set_lang_en` - установить английский
|
||||
- `set_lang_ko` - установить корейский
|
||||
|
||||
### 7. Запуск миграции
|
||||
|
||||
```bash
|
||||
# Применить миграцию добавления колонки lang
|
||||
PGPASSWORD='Cl0ud_1985!' psql -h 192.168.0.102 -p 5432 -U trevor -d telegram_tinder_bot -f sql/add_user_language.sql
|
||||
```
|
||||
|
||||
### 8. Тестирование
|
||||
|
||||
#### Проверка выбора языка для нового пользователя:
|
||||
|
||||
1. Удалите свой профиль из БД:
|
||||
```sql
|
||||
DELETE FROM profiles WHERE user_id IN (
|
||||
SELECT id FROM users WHERE telegram_id = YOUR_TELEGRAM_ID
|
||||
);
|
||||
DELETE FROM users WHERE telegram_id = YOUR_TELEGRAM_ID;
|
||||
```
|
||||
|
||||
2. Отправьте `/start` боту
|
||||
3. Должно появиться меню выбора из 10 языков
|
||||
4. Выберите любой язык
|
||||
5. Проверьте, что приветствие отображается на выбранном языке
|
||||
|
||||
#### Проверка сохранения языка:
|
||||
|
||||
```sql
|
||||
SELECT telegram_id, username, lang FROM users;
|
||||
```
|
||||
|
||||
## Следующие шаги
|
||||
|
||||
### Приоритет 1: Замена всех хардкод-текстов
|
||||
|
||||
Необходимо заменить все хардкод-тексты в следующих файлах:
|
||||
|
||||
1. **`src/handlers/messageHandlers.ts`** (профиль, регистрация)
|
||||
2. **`src/handlers/callbackHandlers.ts`** (кнопки, меню)
|
||||
3. **`src/handlers/commandHandlers.ts`** (команды)
|
||||
4. **`src/services/notificationService.ts`** (уведомления)
|
||||
|
||||
### Приоритет 2: Дополнение языковых файлов
|
||||
|
||||
Текущие файлы содержат только базовые ключи. Нужно:
|
||||
1. Извлечь все существующие тексты из кода
|
||||
2. Добавить ключи в `ru.json` (эталонный файл)
|
||||
3. Перевести ключи для остальных 9 языков
|
||||
|
||||
### Приоритет 3: Кнопка смены языка в настройках
|
||||
|
||||
Добавить в меню "⚙️ Настройки" кнопку:
|
||||
```typescript
|
||||
{ text: '🌍 Язык / Language', callback_data: 'change_language' }
|
||||
```
|
||||
|
||||
## Состояние реализации
|
||||
|
||||
✅ **Выполнено:**
|
||||
- Добавлена колонка `lang` в таблицу `users`
|
||||
- Создан `LanguageHandlers` для управления языком
|
||||
- Интегрирован выбор языка в `/start` для новых пользователей
|
||||
- Обновлен `LocalizationService`
|
||||
- Добавлены секции `language` в `ru.json` и `en.json`
|
||||
- Методы работы с языком в `ProfileService`
|
||||
|
||||
⚠️ **В процессе:**
|
||||
- Замена хардкод-текстов на локализационные ключи
|
||||
- Дополнение всех языковых файлов
|
||||
|
||||
📋 **Планируется:**
|
||||
- Кнопка смены языка в настройках
|
||||
- Полный перевод всех 10 языков
|
||||
- Тесты локализации
|
||||
|
||||
## Технические детали
|
||||
|
||||
### Callback Query Flow
|
||||
|
||||
```
|
||||
Пользователь нажимает кнопку "🇷🇺 Русский"
|
||||
↓
|
||||
callback_data: 'set_lang_ru'
|
||||
↓
|
||||
callbackHandlers.handleCallback() перехватывает
|
||||
↓
|
||||
if (data.startsWith('set_lang_'))
|
||||
↓
|
||||
languageHandlers.handleSetLanguage(query)
|
||||
↓
|
||||
profileService.updateUserLanguage(userId, 'ru')
|
||||
↓
|
||||
localizationService.setLanguage('ru')
|
||||
↓
|
||||
Показ приветственного сообщения на русском
|
||||
```
|
||||
|
||||
### Database Schema
|
||||
|
||||
```sql
|
||||
CREATE TABLE users (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
telegram_id BIGINT UNIQUE NOT NULL,
|
||||
username VARCHAR(255),
|
||||
first_name VARCHAR(255),
|
||||
last_name VARCHAR(255),
|
||||
lang VARCHAR(5) DEFAULT 'ru' NOT NULL, -- ← НОВАЯ КОЛОНКА
|
||||
is_active BOOLEAN DEFAULT true,
|
||||
created_at TIMESTAMP DEFAULT NOW(),
|
||||
last_active_at TIMESTAMP DEFAULT NOW()
|
||||
);
|
||||
|
||||
CREATE INDEX idx_users_lang ON users(lang);
|
||||
```
|
||||
|
||||
## Примеры использования
|
||||
|
||||
### Пример 1: Приветственное сообщение
|
||||
|
||||
```typescript
|
||||
const lang = await profileService.getUserLanguage(userId);
|
||||
localizationService.setLanguage(lang);
|
||||
|
||||
const greeting = localizationService.t('welcome.greeting');
|
||||
const description = localizationService.t('welcome.description');
|
||||
|
||||
await bot.sendMessage(chatId, `${greeting}\n\n${description}`);
|
||||
```
|
||||
|
||||
### Пример 2: Кнопки с переводом
|
||||
|
||||
```typescript
|
||||
const keyboard: InlineKeyboardMarkup = {
|
||||
inline_keyboard: [
|
||||
[{
|
||||
text: localizationService.t('buttons.save'),
|
||||
callback_data: 'save_profile'
|
||||
}],
|
||||
[{
|
||||
text: localizationService.t('buttons.cancel'),
|
||||
callback_data: 'cancel'
|
||||
}]
|
||||
]
|
||||
};
|
||||
```
|
||||
|
||||
### Пример 3: Ошибки
|
||||
|
||||
```typescript
|
||||
try {
|
||||
// ... код
|
||||
} catch (error) {
|
||||
const errorMsg = localizationService.t('errors.serverError');
|
||||
await bot.sendMessage(chatId, errorMsg);
|
||||
}
|
||||
```
|
||||
|
||||
## Поддержка
|
||||
|
||||
Если нужно добавить новый язык:
|
||||
|
||||
1. Создайте файл `src/locales/{код}.json`
|
||||
2. Скопируйте структуру из `ru.json`
|
||||
3. Переведите все ключи
|
||||
4. Добавьте язык в `LocalizationService.initialize()`:
|
||||
```typescript
|
||||
const newLangTranslations = JSON.parse(
|
||||
fs.readFileSync(path.join(localesPath, 'новый_код.json'), 'utf8')
|
||||
);
|
||||
```
|
||||
5. Добавьте в `resources` объект
|
||||
6. Добавьте в `getSupportedLanguages()`
|
||||
7. Добавьте кнопку в `LanguageHandlers.showLanguageSelection()`
|
||||
|
||||
---
|
||||
|
||||
**Статус:** ✅ Система локализации активна и работает
|
||||
|
||||
**Версия:** 1.0.0
|
||||
|
||||
**Дата:** 06.11.2025
|
||||
264
docs/PRODUCTION_DEPLOYMENT.md
Normal file
264
docs/PRODUCTION_DEPLOYMENT.md
Normal file
@@ -0,0 +1,264 @@
|
||||
# Инструкция по развертыванию Telegram Tinder Bot в Production
|
||||
|
||||
Это подробное руководство по развертыванию Telegram Tinder Bot в production-окружении с использованием Docker и Docker Compose.
|
||||
|
||||
## 📋 Требования
|
||||
|
||||
- **Операционная система**: Ubuntu 20.04 или выше (рекомендуется) / Windows Server с Docker
|
||||
- **Программное обеспечение**:
|
||||
- Docker (последняя версия)
|
||||
- Docker Compose (последняя версия)
|
||||
- Git
|
||||
|
||||
## 🚀 Быстрое развертывание
|
||||
|
||||
### 1. Клонирование репозитория
|
||||
|
||||
```bash
|
||||
git clone https://github.com/your-username/telegram-tinder-bot.git
|
||||
cd telegram-tinder-bot
|
||||
```
|
||||
|
||||
### 2. Настройка конфигурации
|
||||
|
||||
```bash
|
||||
# Создание файла конфигурации из шаблона
|
||||
cp .env.production .env
|
||||
|
||||
# Редактирование конфигурационного файла
|
||||
nano .env
|
||||
```
|
||||
|
||||
Важно указать следующие параметры:
|
||||
- `TELEGRAM_BOT_TOKEN`: токен от @BotFather
|
||||
- `DB_PASSWORD`: надежный пароль для базы данных
|
||||
- `JWT_SECRET`: случайная строка для JWT
|
||||
- `ENCRYPTION_KEY`: случайная строка для шифрования
|
||||
|
||||
### 3. Запуск деплоя
|
||||
|
||||
```bash
|
||||
# Сделайте скрипт исполняемым
|
||||
chmod +x deploy.sh
|
||||
|
||||
# Запустите деплой
|
||||
./deploy.sh
|
||||
```
|
||||
|
||||
## 🔧 Подробное руководство по установке
|
||||
|
||||
### Подготовка сервера Ubuntu
|
||||
|
||||
```bash
|
||||
# Обновление системы
|
||||
sudo apt update && sudo apt upgrade -y
|
||||
|
||||
# Установка необходимых пакетов
|
||||
sudo apt install -y apt-transport-https ca-certificates curl software-properties-common git
|
||||
|
||||
# Установка Docker
|
||||
curl -fsSL https://get.docker.com -o get-docker.sh
|
||||
sudo sh get-docker.sh
|
||||
|
||||
# Добавление текущего пользователя в группу docker
|
||||
sudo usermod -aG docker ${USER}
|
||||
|
||||
# Установка Docker Compose
|
||||
sudo apt install -y docker-compose
|
||||
```
|
||||
|
||||
### Клонирование и настройка проекта
|
||||
|
||||
```bash
|
||||
# Создание директории для проекта
|
||||
mkdir -p /opt/telegram-tinder
|
||||
cd /opt/telegram-tinder
|
||||
|
||||
# Клонирование репозитория
|
||||
git clone https://github.com/your-username/telegram-tinder-bot.git .
|
||||
|
||||
# Настройка .env файла
|
||||
cp .env.production .env
|
||||
nano .env
|
||||
|
||||
# Создание директорий для данных и логов
|
||||
mkdir -p uploads logs
|
||||
chmod 777 uploads logs
|
||||
```
|
||||
|
||||
### Запуск проекта
|
||||
|
||||
```bash
|
||||
# Запуск в фоновом режиме
|
||||
docker-compose up -d
|
||||
|
||||
# Проверка статуса контейнеров
|
||||
docker-compose ps
|
||||
|
||||
# Просмотр логов
|
||||
docker-compose logs -f
|
||||
```
|
||||
|
||||
## 🔄 Обновление бота
|
||||
|
||||
Для обновления бота выполните:
|
||||
|
||||
```bash
|
||||
cd /путь/к/telegram-tinder-bot
|
||||
./deploy.sh
|
||||
```
|
||||
|
||||
Скрипт автоматически выполнит:
|
||||
1. Получение последних изменений из репозитория
|
||||
2. Перезапуск контейнеров с новой версией кода
|
||||
3. Применение миграций базы данных
|
||||
|
||||
## 🛡️ Обеспечение безопасности
|
||||
|
||||
### Настройка файрвола
|
||||
|
||||
```bash
|
||||
# Разрешение только необходимых портов
|
||||
sudo ufw allow 22/tcp
|
||||
sudo ufw allow 80/tcp
|
||||
sudo ufw allow 443/tcp
|
||||
sudo ufw enable
|
||||
```
|
||||
|
||||
### Настройка HTTPS с Let's Encrypt (опционально)
|
||||
|
||||
Для использования HTTPS с Let's Encrypt и Nginx:
|
||||
|
||||
```bash
|
||||
# Установка Certbot
|
||||
sudo apt install -y certbot python3-certbot-nginx
|
||||
|
||||
# Получение SSL-сертификата
|
||||
sudo certbot --nginx -d your-domain.com
|
||||
```
|
||||
|
||||
## 📊 Мониторинг и управление
|
||||
|
||||
### Просмотр логов
|
||||
|
||||
```bash
|
||||
# Логи всех контейнеров
|
||||
docker-compose logs -f
|
||||
|
||||
# Логи конкретного контейнера (например, бота)
|
||||
docker-compose logs -f bot
|
||||
|
||||
# Последние 100 строк логов
|
||||
docker-compose logs --tail=100 bot
|
||||
```
|
||||
|
||||
### Управление сервисами
|
||||
|
||||
```bash
|
||||
# Остановка всех контейнеров
|
||||
docker-compose down
|
||||
|
||||
# Перезапуск всех контейнеров
|
||||
docker-compose restart
|
||||
|
||||
# Перезапуск только бота
|
||||
docker-compose restart bot
|
||||
```
|
||||
|
||||
### Доступ к базе данных
|
||||
|
||||
```bash
|
||||
# Вход в консоль PostgreSQL
|
||||
docker-compose exec db psql -U postgres -d telegram_tinder_bot
|
||||
|
||||
# Резервное копирование базы данных
|
||||
docker-compose exec db pg_dump -U postgres telegram_tinder_bot > backup_$(date +%Y%m%d).sql
|
||||
|
||||
# Восстановление базы из резервной копии
|
||||
cat backup.sql | docker-compose exec -T db psql -U postgres -d telegram_tinder_bot
|
||||
```
|
||||
|
||||
## 🔍 Устранение неполадок
|
||||
|
||||
### Проверка работоспособности
|
||||
|
||||
```bash
|
||||
# Проверка API бота
|
||||
curl http://localhost:3000/health
|
||||
|
||||
# Проверка подключения к базе данных
|
||||
docker-compose exec bot node -e "const { Client } = require('pg'); const client = new Client({ host: 'db', port: 5432, database: 'telegram_tinder_bot', user: 'postgres', password: process.env.DB_PASSWORD }); client.connect().then(() => { console.log('Connected to DB!'); client.end(); }).catch(e => console.error(e));"
|
||||
```
|
||||
|
||||
### Общие проблемы и решения
|
||||
|
||||
**Проблема**: Бот не отвечает в Telegram
|
||||
**Решение**:
|
||||
- Проверьте валидность токена бота
|
||||
- Проверьте логи на наличие ошибок: `docker-compose logs -f bot`
|
||||
|
||||
**Проблема**: Ошибки подключения к базе данных
|
||||
**Решение**:
|
||||
- Проверьте настройки подключения в `.env`
|
||||
- Убедитесь, что контейнер с базой данных запущен: `docker-compose ps`
|
||||
- Проверьте логи базы данных: `docker-compose logs db`
|
||||
|
||||
**Проблема**: Недостаточно свободного места на диске
|
||||
**Решение**:
|
||||
- Очистите неиспользуемые Docker образы: `docker image prune -a`
|
||||
- Очистите неиспользуемые Docker тома: `docker volume prune`
|
||||
|
||||
## 🔁 Настройка автоматического обновления
|
||||
|
||||
### Настройка автообновления через Cron
|
||||
|
||||
```bash
|
||||
# Редактирование crontab
|
||||
crontab -e
|
||||
|
||||
# Добавление задачи (обновление каждую ночь в 3:00)
|
||||
0 3 * * * cd /путь/к/telegram-tinder-bot && ./deploy.sh > /tmp/tg-tinder-update.log 2>&1
|
||||
```
|
||||
|
||||
## 📝 Рекомендации по обслуживанию
|
||||
|
||||
1. **Регулярное резервное копирование**:
|
||||
```bash
|
||||
# Ежедневное резервное копирование через cron
|
||||
0 2 * * * docker-compose exec -T db pg_dump -U postgres telegram_tinder_bot > /path/to/backups/tg_$(date +\%Y\%m\%d).sql
|
||||
```
|
||||
|
||||
2. **Мониторинг использования ресурсов**:
|
||||
```bash
|
||||
# Просмотр использования ресурсов контейнерами
|
||||
docker stats
|
||||
```
|
||||
|
||||
3. **Обновление Docker образов**:
|
||||
```bash
|
||||
# Обновление образов
|
||||
docker-compose pull
|
||||
docker-compose up -d
|
||||
```
|
||||
|
||||
4. **Проверка журналов на наличие ошибок**:
|
||||
```bash
|
||||
# Поиск ошибок в логах
|
||||
docker-compose logs | grep -i error
|
||||
docker-compose logs | grep -i exception
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📋 Контрольный список деплоя
|
||||
|
||||
- [ ] Установлены Docker и Docker Compose
|
||||
- [ ] Клонирован репозиторий
|
||||
- [ ] Настроен файл .env с реальными данными
|
||||
- [ ] Запущены контейнеры через docker-compose
|
||||
- [ ] Проверено подключение бота к Telegram API
|
||||
- [ ] Настроено резервное копирование
|
||||
- [ ] Настроен файрвол и безопасность сервера
|
||||
- [ ] Проверены и настроены логи
|
||||
- [ ] (Опционально) Настроен SSL для веб-интерфейса
|
||||
- [ ] (Опционально) Настроено автоматическое обновление
|
||||
80
docs/docker_fix.md
Normal file
80
docs/docker_fix.md
Normal file
@@ -0,0 +1,80 @@
|
||||
# Решение проблемы с Docker-контейнерами
|
||||
|
||||
## Проблема
|
||||
При запуске контейнеров через Docker Compose возникает ошибка `KeyError: 'ContainerConfig'`. Эта ошибка появляется из-за несовместимости между версиями Docker, Docker Compose и структурой docker-compose.yml.
|
||||
|
||||
## Решение
|
||||
|
||||
### 1. Очистка окружения Docker
|
||||
|
||||
На сервере выполните следующие команды, чтобы полностью очистить окружение Docker:
|
||||
|
||||
```bash
|
||||
# Остановка и удаление контейнеров
|
||||
docker-compose down -v
|
||||
|
||||
# Принудительное удаление контейнеров по имени
|
||||
docker rm -f postgres-tinder adminer-tinder telegram-tinder-bot
|
||||
|
||||
# Очистка неиспользуемых томов и сетей
|
||||
docker system prune -f --volumes
|
||||
|
||||
# Очистка кеша Docker
|
||||
docker builder prune -f
|
||||
```
|
||||
|
||||
### 2. Исправление проблем с переносами строк
|
||||
|
||||
Файлы, созданные в Windows и перенесенные в Linux, могут содержать неправильные символы переноса строки.
|
||||
|
||||
```bash
|
||||
# Исправление переносов строк в shell-скриптах
|
||||
find . -name "*.sh" -type f -exec sh -c 'tr -d "\r" < "$1" > "$1.fixed" && mv "$1.fixed" "$1" && chmod +x "$1"' -- {} \;
|
||||
```
|
||||
|
||||
### 3. Обновление docker-compose.yml
|
||||
|
||||
Создайте новый docker-compose.yml с исправленной структурой:
|
||||
|
||||
```bash
|
||||
# Запустите скрипт для исправления проблем с Docker
|
||||
./bin/fix_docker.sh
|
||||
```
|
||||
|
||||
### 4. Запуск с полностью чистым окружением
|
||||
|
||||
После выполнения всех исправлений запустите контейнеры заново:
|
||||
|
||||
```bash
|
||||
docker-compose up -d
|
||||
```
|
||||
|
||||
## Альтернативное решение
|
||||
|
||||
Если проблема сохраняется, можно попробовать запустить контейнеры по отдельности:
|
||||
|
||||
```bash
|
||||
# Сначала запустить базу данных (если она нужна)
|
||||
docker-compose up -d db
|
||||
|
||||
# Дождаться запуска базы данных
|
||||
sleep 10
|
||||
|
||||
# Запустить бота
|
||||
docker-compose up -d bot
|
||||
|
||||
# Запустить adminer
|
||||
docker-compose up -d adminer
|
||||
```
|
||||
|
||||
## Проверка работы миграций
|
||||
|
||||
После запуска контейнеров проверьте, что миграции базы данных применяются правильно:
|
||||
|
||||
```bash
|
||||
# Просмотр логов контейнера бота
|
||||
docker logs telegram-tinder-bot
|
||||
|
||||
# Если миграции не применяются, можно запустить их вручную внутри контейнера
|
||||
docker exec -it telegram-tinder-bot sh -c "DATABASE_URL=postgres://$DB_USERNAME:$DB_PASSWORD@$DB_HOST:$DB_PORT/$DB_NAME npx node-pg-migrate up"
|
||||
```
|
||||
88
docs/migrations_fix.md
Normal file
88
docs/migrations_fix.md
Normal file
@@ -0,0 +1,88 @@
|
||||
# Решение проблемы с миграциями базы данных
|
||||
|
||||
## Проблемы
|
||||
|
||||
При попытке применить миграции были обнаружены следующие проблемы:
|
||||
|
||||
1. **Ошибка с TypeScript файлами**: Node.js не может напрямую выполнять файлы `.ts` без компиляции их в JavaScript.
|
||||
2. **Предупреждения о ES модулях**: Файлы используют синтаксис ES модулей, но не имеют расширения `.mjs` или настроек в package.json.
|
||||
3. **Неверный порядок миграций**: Миграции могут выполняться в неправильном порядке.
|
||||
|
||||
## Решения
|
||||
|
||||
### Для быстрого применения миграций
|
||||
|
||||
Используйте один из следующих сценариев:
|
||||
|
||||
```bash
|
||||
# Полный процесс миграции с компиляцией TypeScript
|
||||
./bin/run_full_migration.sh
|
||||
|
||||
# Только SQL-миграции (минуя node-pg-migrate)
|
||||
./bin/run_sql_migrations.sh
|
||||
```
|
||||
|
||||
### Пошаговое решение
|
||||
|
||||
1. **Компиляция TypeScript миграций в JavaScript**:
|
||||
```bash
|
||||
./bin/compile_ts_migrations.sh
|
||||
```
|
||||
|
||||
2. **Применение JS-миграций**:
|
||||
```bash
|
||||
./bin/apply_migrations.sh
|
||||
```
|
||||
|
||||
3. **Ручное применение SQL-миграций**:
|
||||
```bash
|
||||
./bin/run_sql_migrations.sh
|
||||
```
|
||||
|
||||
## Описание скриптов
|
||||
|
||||
- **run_full_migration.sh**: Полный процесс миграции, включающий компиляцию TypeScript и применение всех миграций.
|
||||
- **compile_ts_migrations.sh**: Только компиляция TypeScript миграций в JavaScript.
|
||||
- **apply_migrations.sh**: Применение JS-миграций через node-pg-migrate.
|
||||
- **run_sql_migrations.sh**: Прямое применение SQL-миграций через psql.
|
||||
|
||||
## Проверка результатов
|
||||
|
||||
После выполнения миграций проверьте состояние базы данных:
|
||||
|
||||
```bash
|
||||
# Подключение к базе данных
|
||||
PGPASSWORD=$DB_PASSWORD psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME
|
||||
|
||||
# Проверка таблиц
|
||||
\dt
|
||||
|
||||
# Проверка примененных миграций
|
||||
SELECT * FROM migrations ORDER BY executed_at;
|
||||
```
|
||||
|
||||
## Если проблемы сохраняются
|
||||
|
||||
1. **Очистить директорию миграций**:
|
||||
```bash
|
||||
# Создание резервной копии
|
||||
mkdir -p backup_migrations
|
||||
cp -r migrations/* backup_migrations/
|
||||
|
||||
# Оставить только JS-миграции
|
||||
rm -f migrations/*.ts
|
||||
```
|
||||
|
||||
2. **Инициализировать миграции заново**:
|
||||
```bash
|
||||
npx node-pg-migrate init
|
||||
```
|
||||
|
||||
3. **Применить специальную консолидированную миграцию**:
|
||||
```bash
|
||||
# Создание консолидированной миграции
|
||||
cat src/database/migrations/*.sql > consolidated.sql
|
||||
|
||||
# Применение консолидированной миграции
|
||||
PGPASSWORD=$DB_PASSWORD psql -h $DB_HOST -p $DB_PORT -U $DB_USERNAME -d $DB_NAME -f consolidated.sql
|
||||
```
|
||||
19
fix_dockerfile.sh
Executable file
19
fix_dockerfile.sh
Executable file
@@ -0,0 +1,19 @@
|
||||
#!/bin/bash
|
||||
# Скрипт для исправления Dockerfile на production сервере
|
||||
|
||||
echo "Исправление Dockerfile..."
|
||||
|
||||
# Замена всех вариантов build:linux:* на просто build:linux
|
||||
sed -i 's/build:linux:linux:linux:linux:linux:linux/build:linux/g' Dockerfile
|
||||
sed -i 's/build:linux:linux:linux:linux:linux/build:linux/g' Dockerfile
|
||||
sed -i 's/build:linux:linux:linux:linux/build:linux/g' Dockerfile
|
||||
sed -i 's/build:linux:linux:linux/build:linux/g' Dockerfile
|
||||
sed -i 's/build:linux:linux/build:linux/g' Dockerfile
|
||||
|
||||
echo "✅ Dockerfile исправлен"
|
||||
echo ""
|
||||
echo "Проверка строки 19:"
|
||||
grep -n "build:linux" Dockerfile | head -1
|
||||
|
||||
echo ""
|
||||
echo "Теперь можно запустить: make run"
|
||||
44
migrations/1631980000000_add_profile_views_table.ts
Normal file
44
migrations/1631980000000_add_profile_views_table.ts
Normal file
@@ -0,0 +1,44 @@
|
||||
import { MigrationBuilder, ColumnDefinitions } from 'node-pg-migrate';
|
||||
|
||||
export const shorthands: ColumnDefinitions | undefined = undefined;
|
||||
|
||||
export async function up(pgm: MigrationBuilder): Promise<void> {
|
||||
// Создание таблицы profile_views для хранения информации о просмотренных профилях
|
||||
pgm.createTable('profile_views', {
|
||||
id: { type: 'uuid', primaryKey: true, default: pgm.func('uuid_generate_v4()') },
|
||||
viewer_id: {
|
||||
type: 'uuid',
|
||||
notNull: true,
|
||||
references: 'users',
|
||||
onDelete: 'CASCADE'
|
||||
},
|
||||
viewed_profile_id: {
|
||||
type: 'uuid',
|
||||
notNull: true,
|
||||
references: 'profiles(user_id)',
|
||||
onDelete: 'CASCADE'
|
||||
},
|
||||
view_date: { type: 'timestamp', notNull: true, default: pgm.func('now()') },
|
||||
view_type: { type: 'varchar(20)', notNull: true, default: 'browse' }, // browse, match, like, etc.
|
||||
});
|
||||
|
||||
// Создание индекса для быстрого поиска по паре (просмотревший - просмотренный)
|
||||
pgm.createIndex('profile_views', ['viewer_id', 'viewed_profile_id'], {
|
||||
unique: true,
|
||||
name: 'profile_views_viewer_viewed_idx'
|
||||
});
|
||||
|
||||
// Индекс для быстрого поиска по viewer_id
|
||||
pgm.createIndex('profile_views', ['viewer_id'], {
|
||||
name: 'profile_views_viewer_idx'
|
||||
});
|
||||
|
||||
// Индекс для быстрого поиска по viewed_profile_id
|
||||
pgm.createIndex('profile_views', ['viewed_profile_id'], {
|
||||
name: 'profile_views_viewed_idx'
|
||||
});
|
||||
}
|
||||
|
||||
export async function down(pgm: MigrationBuilder): Promise<void> {
|
||||
pgm.dropTable('profile_views', { cascade: true });
|
||||
}
|
||||
@@ -0,0 +1,52 @@
|
||||
/* eslint-disable camelcase */
|
||||
|
||||
exports.shorthands = undefined;
|
||||
|
||||
exports.up = pgm => {
|
||||
// Проверяем существование таблицы scheduled_notifications
|
||||
pgm.sql(`
|
||||
DO $$
|
||||
BEGIN
|
||||
IF EXISTS (
|
||||
SELECT 1 FROM information_schema.tables
|
||||
WHERE table_name = 'scheduled_notifications'
|
||||
) THEN
|
||||
-- Проверяем, нет ли уже столбца processed
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'scheduled_notifications' AND column_name = 'processed'
|
||||
) THEN
|
||||
-- Добавляем столбец processed
|
||||
ALTER TABLE scheduled_notifications ADD COLUMN processed BOOLEAN DEFAULT FALSE;
|
||||
END IF;
|
||||
ELSE
|
||||
-- Создаем таблицу, если она не существует
|
||||
CREATE TABLE scheduled_notifications (
|
||||
id UUID PRIMARY KEY,
|
||||
user_id UUID REFERENCES users(id),
|
||||
type VARCHAR(50) NOT NULL,
|
||||
data JSONB,
|
||||
scheduled_at TIMESTAMP NOT NULL,
|
||||
processed BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP DEFAULT NOW()
|
||||
);
|
||||
END IF;
|
||||
END
|
||||
$$;
|
||||
`);
|
||||
};
|
||||
|
||||
exports.down = pgm => {
|
||||
pgm.sql(`
|
||||
DO $$
|
||||
BEGIN
|
||||
IF EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'scheduled_notifications' AND column_name = 'processed'
|
||||
) THEN
|
||||
ALTER TABLE scheduled_notifications DROP COLUMN processed;
|
||||
END IF;
|
||||
END
|
||||
$$;
|
||||
`);
|
||||
};
|
||||
14
migrations/add_user_state_columns.sql
Normal file
14
migrations/add_user_state_columns.sql
Normal file
@@ -0,0 +1,14 @@
|
||||
-- Добавление столбцов state и state_data в таблицу users для обработки состояний пользователя
|
||||
|
||||
-- Добавляем столбец state для хранения текущего состояния пользователя
|
||||
ALTER TABLE users ADD COLUMN IF NOT EXISTS state VARCHAR(255) NULL;
|
||||
|
||||
-- Добавляем столбец state_data для хранения дополнительных данных о состоянии
|
||||
ALTER TABLE users ADD COLUMN IF NOT EXISTS state_data JSONB DEFAULT '{}'::jsonb;
|
||||
|
||||
-- Добавляем индекс для быстрого поиска по state
|
||||
CREATE INDEX IF NOT EXISTS idx_users_state ON users(state);
|
||||
|
||||
-- Комментарий к столбцам
|
||||
COMMENT ON COLUMN users.state IS 'Текущее состояние пользователя (например, ожидание ввода)';
|
||||
COMMENT ON COLUMN users.state_data IS 'Дополнительные данные о состоянии пользователя в формате JSON';
|
||||
BIN
new_docker-keyring.gpg
Normal file
BIN
new_docker-keyring.gpg
Normal file
Binary file not shown.
@@ -22,6 +22,7 @@
|
||||
"premium:direct": "ts-node src/scripts/setPremiumDirectConnect.ts",
|
||||
"db:info": "ts-node src/scripts/getDatabaseInfo.ts",
|
||||
"db:test-data": "ts-node src/scripts/createTestData.ts",
|
||||
"enhance-notifications": "ts-node src/scripts/enhanceNotifications.ts",
|
||||
"update": "bash ./bin/update.sh",
|
||||
"update:win": ".\\bin\\update.bat",
|
||||
"start:sh": "bash ./bin/start_bot.sh"
|
||||
|
||||
49
scripts/README.md
Normal file
49
scripts/README.md
Normal file
@@ -0,0 +1,49 @@
|
||||
# Структура скриптов в директории `/scripts`
|
||||
|
||||
Эта директория содержит вспомогательные скрипты для работы с Telegram Tinder Bot.
|
||||
|
||||
## Основные скрипты
|
||||
|
||||
- `startup.sh` - Скрипт запуска бота в Docker-контейнере
|
||||
- `migrate-sync.js` - Синхронизация миграций базы данных
|
||||
- `createNotificationTables.js` - Создание таблиц для системы уведомлений
|
||||
- `add-hobbies-column.js` - Добавление колонки интересов в профиль
|
||||
- `create_profile_fix.js` - Исправление профилей пользователей
|
||||
- `createProfileViewsTable.js` - Создание таблицы для учета просмотров профилей
|
||||
- `update_bot_with_notifications.js` - Обновление бота с поддержкой уведомлений
|
||||
|
||||
## Директории
|
||||
|
||||
- `/legacy` - Устаревшие и тестовые скрипты, сохраненные для истории
|
||||
|
||||
## Использование скриптов
|
||||
|
||||
Скрипты JavaScript можно запускать с помощью Node.js:
|
||||
|
||||
```bash
|
||||
node scripts/script-name.js
|
||||
```
|
||||
|
||||
Bash скрипты должны быть сделаны исполняемыми:
|
||||
|
||||
```bash
|
||||
chmod +x scripts/script-name.sh
|
||||
./scripts/script-name.sh
|
||||
```
|
||||
|
||||
## Добавление новых скриптов
|
||||
|
||||
При добавлении новых скриптов соблюдайте следующие правила:
|
||||
1. Используйте понятное имя файла, отражающее его назначение
|
||||
2. Добавьте комментарии в начало файла с описанием его функциональности
|
||||
3. Добавьте запись об этом скрипте в текущий файл README.md
|
||||
|
||||
## Скрипты миграций
|
||||
|
||||
Миграции базы данных следует создавать с помощью команды:
|
||||
|
||||
```bash
|
||||
npm run migrate:create your_migration_name
|
||||
```
|
||||
|
||||
Это создаст файл миграции в директории `/migrations`.
|
||||
@@ -1,15 +1,16 @@
|
||||
// add-hobbies-column.js
|
||||
// Скрипт для добавления колонки hobbies в таблицу profiles
|
||||
|
||||
require('dotenv').config();
|
||||
const { Pool } = require('pg');
|
||||
|
||||
// Настройки подключения к базе данных
|
||||
// Настройки подключения к базе данных из переменных окружения
|
||||
const pool = new Pool({
|
||||
host: '192.168.0.102',
|
||||
port: 5432,
|
||||
database: 'telegram_tinder_bot',
|
||||
user: 'trevor',
|
||||
password: 'Cl0ud_1985!'
|
||||
host: process.env.DB_HOST || 'localhost',
|
||||
port: parseInt(process.env.DB_PORT) || 5432,
|
||||
database: process.env.DB_NAME || 'telegram_tinder_bot',
|
||||
user: process.env.DB_USERNAME || 'postgres',
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
async function addHobbiesColumn() {
|
||||
|
||||
259
scripts/createNotificationTables.js
Normal file
259
scripts/createNotificationTables.js
Normal file
@@ -0,0 +1,259 @@
|
||||
const { Pool } = require('pg');
|
||||
const dotenv = require('dotenv');
|
||||
const uuid = require('uuid');
|
||||
|
||||
dotenv.config();
|
||||
|
||||
const pool = new Pool({
|
||||
connectionString: process.env.DATABASE_URL,
|
||||
});
|
||||
|
||||
async function createNotificationTables() {
|
||||
const client = await pool.connect();
|
||||
|
||||
try {
|
||||
await client.query('BEGIN');
|
||||
|
||||
console.log('Creating UUID extension if not exists...');
|
||||
await client.query(`
|
||||
CREATE EXTENSION IF NOT EXISTS "uuid-ossp";
|
||||
`);
|
||||
|
||||
// Проверяем существование таблицы notifications
|
||||
const notificationsExists = await client.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.tables
|
||||
WHERE table_name = 'notifications'
|
||||
) as exists
|
||||
`);
|
||||
|
||||
if (!notificationsExists.rows[0].exists) {
|
||||
console.log('Creating notifications table...');
|
||||
await client.query(`
|
||||
CREATE TABLE notifications (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
data JSONB,
|
||||
is_read BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP DEFAULT NOW()
|
||||
)
|
||||
`);
|
||||
|
||||
console.log('Creating index on notifications...');
|
||||
await client.query(`
|
||||
CREATE INDEX idx_notifications_user_id ON notifications (user_id);
|
||||
CREATE INDEX idx_notifications_type ON notifications (type);
|
||||
CREATE INDEX idx_notifications_created_at ON notifications (created_at);
|
||||
`);
|
||||
} else {
|
||||
console.log('Notifications table already exists.');
|
||||
}
|
||||
|
||||
// Проверяем существование таблицы scheduled_notifications
|
||||
const scheduledExists = await client.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.tables
|
||||
WHERE table_name = 'scheduled_notifications'
|
||||
) as exists
|
||||
`);
|
||||
|
||||
if (!scheduledExists.rows[0].exists) {
|
||||
console.log('Creating scheduled_notifications table...');
|
||||
await client.query(`
|
||||
CREATE TABLE scheduled_notifications (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
data JSONB,
|
||||
scheduled_at TIMESTAMP NOT NULL,
|
||||
processed BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP DEFAULT NOW()
|
||||
)
|
||||
`);
|
||||
|
||||
console.log('Creating index on scheduled_notifications...');
|
||||
await client.query(`
|
||||
CREATE INDEX idx_scheduled_notifications_user_id ON scheduled_notifications (user_id);
|
||||
CREATE INDEX idx_scheduled_notifications_scheduled_at ON scheduled_notifications (scheduled_at);
|
||||
CREATE INDEX idx_scheduled_notifications_processed ON scheduled_notifications (processed);
|
||||
`);
|
||||
} else {
|
||||
console.log('Scheduled_notifications table already exists.');
|
||||
}
|
||||
|
||||
// Проверяем существование таблицы notification_templates
|
||||
const templatesExists = await client.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.tables
|
||||
WHERE table_name = 'notification_templates'
|
||||
) as exists
|
||||
`);
|
||||
|
||||
if (!templatesExists.rows[0].exists) {
|
||||
console.log('Creating notification_templates table...');
|
||||
await client.query(`
|
||||
CREATE TABLE notification_templates (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
type VARCHAR(50) NOT NULL UNIQUE,
|
||||
title TEXT NOT NULL,
|
||||
message_template TEXT NOT NULL,
|
||||
button_template JSONB,
|
||||
created_at TIMESTAMP DEFAULT NOW(),
|
||||
updated_at TIMESTAMP DEFAULT NOW()
|
||||
)
|
||||
`);
|
||||
} else {
|
||||
console.log('Notification_templates table already exists.');
|
||||
}
|
||||
|
||||
// Проверяем наличие колонки notification_settings в таблице users
|
||||
const settingsColumnExists = await client.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.columns
|
||||
WHERE table_name = 'users' AND column_name = 'notification_settings'
|
||||
) as exists
|
||||
`);
|
||||
|
||||
if (!settingsColumnExists.rows[0].exists) {
|
||||
console.log('Adding notification_settings column to users table...');
|
||||
await client.query(`
|
||||
ALTER TABLE users
|
||||
ADD COLUMN notification_settings JSONB DEFAULT '{
|
||||
"newMatches": true,
|
||||
"newMessages": true,
|
||||
"newLikes": true,
|
||||
"reminders": true,
|
||||
"dailySummary": true,
|
||||
"timePreference": "evening",
|
||||
"doNotDisturb": false
|
||||
}'::jsonb
|
||||
`);
|
||||
} else {
|
||||
console.log('Notification_settings column already exists in users table.');
|
||||
}
|
||||
|
||||
// Заполнение таблицы шаблонов уведомлений базовыми шаблонами
|
||||
if (!templatesExists.rows[0].exists) {
|
||||
console.log('Populating notification templates...');
|
||||
|
||||
const templates = [
|
||||
{
|
||||
type: 'new_like',
|
||||
title: 'Новый лайк!',
|
||||
message_template: '❤️ *{{name}}* поставил(а) вам лайк!\n\nВозраст: {{age}}\n{{city}}\n\nОтветьте взаимностью или посмотрите профиль.',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '👀 Посмотреть профиль', callback_data: 'view_profile:{{userId}}' }],
|
||||
[
|
||||
{ text: '❤️ Лайк в ответ', callback_data: 'like_back:{{userId}}' },
|
||||
{ text: '⛔️ Пропустить', callback_data: 'dislike_profile:{{userId}}' }
|
||||
],
|
||||
[{ text: '💕 Открыть все лайки', callback_data: 'view_likes' }]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'super_like',
|
||||
title: 'Супер-лайк!',
|
||||
message_template: '⭐️ *{{name}}* отправил(а) вам супер-лайк!\n\nВозраст: {{age}}\n{{city}}\n\nВы произвели особое впечатление! Ответьте взаимностью или посмотрите профиль.',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '👀 Посмотреть профиль', callback_data: 'view_profile:{{userId}}' }],
|
||||
[
|
||||
{ text: '❤️ Лайк в ответ', callback_data: 'like_back:{{userId}}' },
|
||||
{ text: '⛔️ Пропустить', callback_data: 'dislike_profile:{{userId}}' }
|
||||
],
|
||||
[{ text: '💕 Открыть все лайки', callback_data: 'view_likes' }]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'new_match',
|
||||
title: 'Новый матч!',
|
||||
message_template: '🎊 *Ура! Это взаимно!* 🎊\n\nВы и *{{name}}* понравились друг другу!\nВозраст: {{age}}\n{{city}}\n\nСделайте первый шаг - напишите сообщение!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '💬 Начать общение', callback_data: 'open_chat:{{matchId}}' }],
|
||||
[
|
||||
{ text: '👀 Посмотреть профиль', callback_data: 'view_profile:{{userId}}' },
|
||||
{ text: '📋 Все матчи', callback_data: 'view_matches' }
|
||||
]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'new_message',
|
||||
title: 'Новое сообщение!',
|
||||
message_template: '💌 *Новое сообщение!*\n\nОт: *{{name}}*\n\n"{{message}}"\n\nОтветьте на сообщение прямо сейчас!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '📩 Ответить', callback_data: 'open_chat:{{matchId}}' }],
|
||||
[
|
||||
{ text: '👤 Профиль', callback_data: 'view_profile:{{userId}}' },
|
||||
{ text: '📋 Все чаты', callback_data: 'view_matches' }
|
||||
]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'match_reminder',
|
||||
title: 'Напоминание о матче',
|
||||
message_template: '💕 У вас есть матч с *{{name}}*, но вы еще не начали общение!\n\nНе упустите шанс познакомиться поближе!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '💬 Начать общение', callback_data: 'open_chat:{{matchId}}' }],
|
||||
[{ text: '👀 Посмотреть профиль', callback_data: 'view_profile:{{userId}}' }]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'inactive_matches',
|
||||
title: 'Неактивные матчи',
|
||||
message_template: '⏰ У вас {{count}} неактивных матчей!\n\nПродолжите общение, чтобы не упустить интересные знакомства!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '📋 Открыть матчи', callback_data: 'view_matches' }],
|
||||
[{ text: '💕 Смотреть новые анкеты', callback_data: 'start_browsing' }]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'like_summary',
|
||||
title: 'Сводка лайков',
|
||||
message_template: '💖 У вас {{count}} новых лайков!\n\nПосмотрите, кто проявил к вам интерес сегодня!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '👀 Посмотреть лайки', callback_data: 'view_likes' }],
|
||||
[{ text: '💕 Начать знакомиться', callback_data: 'start_browsing' }]
|
||||
]
|
||||
}
|
||||
}
|
||||
];
|
||||
|
||||
for (const template of templates) {
|
||||
await client.query(`
|
||||
INSERT INTO notification_templates (id, type, title, message_template, button_template)
|
||||
VALUES ($1, $2, $3, $4, $5)
|
||||
`, [
|
||||
uuid.v4(),
|
||||
template.type,
|
||||
template.title,
|
||||
template.message_template,
|
||||
JSON.stringify(template.button_template)
|
||||
]);
|
||||
}
|
||||
}
|
||||
|
||||
await client.query('COMMIT');
|
||||
console.log('Successfully created notification tables');
|
||||
} catch (err) {
|
||||
await client.query('ROLLBACK');
|
||||
console.error('Error creating notification tables:', err);
|
||||
} finally {
|
||||
client.release();
|
||||
pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
createNotificationTables().catch(err => console.error('Failed to create notification tables:', err));
|
||||
86
scripts/createProfileViewsTable.js
Normal file
86
scripts/createProfileViewsTable.js
Normal file
@@ -0,0 +1,86 @@
|
||||
const { Pool } = require('pg');
|
||||
require('dotenv').config();
|
||||
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST,
|
||||
port: parseInt(process.env.DB_PORT || '5432'),
|
||||
database: process.env.DB_NAME,
|
||||
user: process.env.DB_USERNAME,
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
async function createProfileViewsTable() {
|
||||
const client = await pool.connect();
|
||||
try {
|
||||
console.log('Creating profile_views table...');
|
||||
await client.query('BEGIN');
|
||||
|
||||
// Включаем расширение uuid-ossp, если оно еще не включено
|
||||
await client.query(`CREATE EXTENSION IF NOT EXISTS "uuid-ossp"`);
|
||||
|
||||
// Создаем таблицу profile_views, если она не существует
|
||||
await client.query(`
|
||||
CREATE TABLE IF NOT EXISTS profile_views (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
viewer_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
viewed_profile_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
view_date TIMESTAMP NOT NULL DEFAULT NOW(),
|
||||
view_type VARCHAR(20) NOT NULL DEFAULT 'browse'
|
||||
)
|
||||
`);
|
||||
|
||||
// Создаем уникальный индекс для пары (просмотревший - просмотренный)
|
||||
await client.query(`
|
||||
DO $$
|
||||
BEGIN
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM pg_indexes
|
||||
WHERE indexname = 'profile_views_viewer_viewed_idx'
|
||||
) THEN
|
||||
CREATE UNIQUE INDEX profile_views_viewer_viewed_idx
|
||||
ON profile_views (viewer_id, viewed_profile_id);
|
||||
END IF;
|
||||
END $$;
|
||||
`);
|
||||
|
||||
// Создаем индекс для быстрого поиска по viewer_id
|
||||
await client.query(`
|
||||
DO $$
|
||||
BEGIN
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM pg_indexes
|
||||
WHERE indexname = 'profile_views_viewer_idx'
|
||||
) THEN
|
||||
CREATE INDEX profile_views_viewer_idx
|
||||
ON profile_views (viewer_id);
|
||||
END IF;
|
||||
END $$;
|
||||
`);
|
||||
|
||||
// Создаем индекс для быстрого поиска по viewed_profile_id
|
||||
await client.query(`
|
||||
DO $$
|
||||
BEGIN
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM pg_indexes
|
||||
WHERE indexname = 'profile_views_viewed_idx'
|
||||
) THEN
|
||||
CREATE INDEX profile_views_viewed_idx
|
||||
ON profile_views (viewed_profile_id);
|
||||
END IF;
|
||||
END $$;
|
||||
`);
|
||||
|
||||
await client.query('COMMIT');
|
||||
console.log('Table profile_views created successfully');
|
||||
} catch (e) {
|
||||
await client.query('ROLLBACK');
|
||||
console.error('Error creating table:', e);
|
||||
} finally {
|
||||
client.release();
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем функцию создания таблицы
|
||||
createProfileViewsTable();
|
||||
@@ -1,4 +1,5 @@
|
||||
// Исправленный код для создания профиля
|
||||
require('dotenv').config();
|
||||
const { Client } = require('pg');
|
||||
const { v4: uuidv4 } = require('uuid');
|
||||
|
||||
@@ -18,13 +19,13 @@ if (!telegramId || !name || !age || !gender || !city || !bio || !photoFileId) {
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
// Устанавливаем соединение с базой данных
|
||||
// Устанавливаем соединение с базой данных из переменных окружения
|
||||
const client = new Client({
|
||||
host: '192.168.0.102',
|
||||
port: 5432,
|
||||
user: 'trevor',
|
||||
password: 'Cl0ud_1985!',
|
||||
database: 'telegram_tinder_bot'
|
||||
host: process.env.DB_HOST || 'localhost',
|
||||
port: parseInt(process.env.DB_PORT) || 5432,
|
||||
user: process.env.DB_USERNAME || 'postgres',
|
||||
password: process.env.DB_PASSWORD,
|
||||
database: process.env.DB_NAME || 'telegram_tinder_bot'
|
||||
});
|
||||
|
||||
async function createProfile() {
|
||||
|
||||
88
scripts/legacy/checkCallbackHandlers.js
Normal file
88
scripts/legacy/checkCallbackHandlers.js
Normal file
@@ -0,0 +1,88 @@
|
||||
// Скрипт для анализа и отладки проблем с обработчиками коллбэков
|
||||
require('dotenv').config();
|
||||
const fs = require('fs');
|
||||
const path = require('path');
|
||||
|
||||
function analyzeCallbackHandlers() {
|
||||
const filePath = path.join(__dirname, '..', 'src', 'handlers', 'callbackHandlers.ts');
|
||||
const content = fs.readFileSync(filePath, 'utf-8');
|
||||
|
||||
// Проверяем наличие реализаций методов
|
||||
const methodsToCheck = [
|
||||
'handleCreateProfile',
|
||||
'handleGenderSelection',
|
||||
'handleViewMyProfile',
|
||||
'handleEditProfile',
|
||||
'handleManagePhotos',
|
||||
'handleStartBrowsing',
|
||||
'handleSettings'
|
||||
];
|
||||
|
||||
const issues = [];
|
||||
let debugInfo = [];
|
||||
|
||||
methodsToCheck.forEach(method => {
|
||||
debugInfo.push(`Проверяем метод: ${method}`);
|
||||
|
||||
// Проверяем наличие полной реализации метода (не только сигнатуры)
|
||||
const methodSignatureRegex = new RegExp(`async\\s+${method}\\s*\\([^)]*\\)\\s*:\\s*Promise<void>\\s*{`, 'g');
|
||||
const hasSignature = methodSignatureRegex.test(content);
|
||||
|
||||
const methodBodyRegex = new RegExp(`async\\s+${method}\\s*\\([^)]*\\)\\s*:\\s*Promise<void>\\s*{[\\s\\S]+?}`, 'g');
|
||||
const methodMatch = content.match(methodBodyRegex);
|
||||
|
||||
debugInfo.push(` Сигнатура найдена: ${hasSignature}`);
|
||||
debugInfo.push(` Реализация найдена: ${methodMatch !== null}`);
|
||||
|
||||
if (methodMatch) {
|
||||
const methodContent = methodMatch[0];
|
||||
debugInfo.push(` Длина метода: ${methodContent.length} символов`);
|
||||
|
||||
// Проверяем, содержит ли метод только заглушку
|
||||
const isStub = methodContent.includes('// Заглушка метода') ||
|
||||
(!methodContent.includes('await') && methodContent.split('\n').length <= 3);
|
||||
|
||||
if (isStub) {
|
||||
issues.push(`❌ Метод ${method} содержит только заглушку, нет реальной реализации`);
|
||||
} else {
|
||||
debugInfo.push(` Метод ${method} имеет полную реализацию`);
|
||||
}
|
||||
} else if (hasSignature) {
|
||||
issues.push(`❌ Метод ${method} имеет только сигнатуру, но нет реализации`);
|
||||
} else {
|
||||
issues.push(`❌ Метод ${method} не найден в файле`);
|
||||
}
|
||||
});
|
||||
|
||||
// Проверяем регистрацию обработчиков для NotificationHandlers
|
||||
const notificationHandlersRegex = /this\.notificationHandlers\s*=\s*new\s+NotificationHandlers\(bot\);/g;
|
||||
const hasNotificationHandlers = notificationHandlersRegex.test(content);
|
||||
debugInfo.push(`NotificationHandlers инициализирован: ${hasNotificationHandlers}`);
|
||||
|
||||
// Проверяем обработку коллбэка notifications
|
||||
const notificationsCallbackRegex = /if\s*\(data\s*===\s*['"]notifications['"].*?\)/g;
|
||||
const hasNotificationsCallback = notificationsCallbackRegex.test(content);
|
||||
debugInfo.push(`Обработчик для callback 'notifications' найден: ${hasNotificationsCallback}`);
|
||||
|
||||
// Выводим результаты
|
||||
console.log('\n=== Анализ CallbackHandlers.ts ===\n');
|
||||
if (issues.length > 0) {
|
||||
console.log('НАЙДЕНЫ ПРОБЛЕМЫ:');
|
||||
issues.forEach(issue => console.log(issue));
|
||||
console.log('\nРЕКОМЕНДАЦИИ:');
|
||||
console.log('1. Восстановите оригинальные реализации методов вместо заглушек');
|
||||
console.log('2. Убедитесь, что методы содержат необходимую бизнес-логику');
|
||||
console.log('3. Проверьте, что все коллбэки правильно обрабатываются');
|
||||
} else {
|
||||
console.log('✅ Проблем не обнаружено');
|
||||
}
|
||||
|
||||
console.log('\n=== Отладочная информация ===\n');
|
||||
debugInfo.forEach(info => console.log(info));
|
||||
|
||||
// Проверяем количество методов в файле
|
||||
const asyncMethodsCount = (content.match(/async\s+handle[A-Za-z]+\s*\(/g) || []).length;
|
||||
console.log(`\nВсего async методов в файле: ${asyncMethodsCount}`);
|
||||
}
|
||||
|
||||
analyzeCallbackHandlers();
|
||||
66
scripts/legacy/checkDatabase.js
Normal file
66
scripts/legacy/checkDatabase.js
Normal file
@@ -0,0 +1,66 @@
|
||||
const { Pool } = require('pg');
|
||||
require('dotenv').config();
|
||||
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST,
|
||||
port: parseInt(process.env.DB_PORT || '5432'),
|
||||
database: process.env.DB_NAME,
|
||||
user: process.env.DB_USERNAME,
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
async function checkDatabase() {
|
||||
const client = await pool.connect();
|
||||
try {
|
||||
console.log('\n===== ПРОВЕРКА СОСТОЯНИЯ БАЗЫ ДАННЫХ =====');
|
||||
|
||||
// Проверка таблицы users
|
||||
const usersResult = await client.query('SELECT COUNT(*) as count FROM users');
|
||||
console.log(`Пользователей в БД: ${usersResult.rows[0].count}`);
|
||||
if (parseInt(usersResult.rows[0].count) > 0) {
|
||||
const users = await client.query('SELECT id, telegram_id, username, first_name FROM users LIMIT 10');
|
||||
console.log('Последние пользователи:');
|
||||
users.rows.forEach(user => {
|
||||
console.log(` - ID: ${user.id.substring(0, 8)}... | Telegram: ${user.telegram_id} | Имя: ${user.first_name || user.username}`);
|
||||
});
|
||||
}
|
||||
|
||||
// Проверка таблицы profiles
|
||||
const profilesResult = await client.query('SELECT COUNT(*) as count FROM profiles');
|
||||
console.log(`\nПрофилей в БД: ${profilesResult.rows[0].count}`);
|
||||
if (parseInt(profilesResult.rows[0].count) > 0) {
|
||||
const profiles = await client.query(`
|
||||
SELECT p.id, p.user_id, p.name, p.age, p.gender, p.interested_in, p.is_visible
|
||||
FROM profiles p
|
||||
ORDER BY p.created_at DESC
|
||||
LIMIT 10
|
||||
`);
|
||||
console.log('Последние профили:');
|
||||
profiles.rows.forEach(profile => {
|
||||
console.log(` - ID: ${profile.id.substring(0, 8)}... | UserID: ${profile.user_id.substring(0, 8)}... | Имя: ${profile.name} | Возраст: ${profile.age} | Пол: ${profile.gender} | Интересы: ${profile.interested_in} | Виден: ${profile.is_visible}`);
|
||||
});
|
||||
}
|
||||
|
||||
// Проверка таблицы swipes
|
||||
const swipesResult = await client.query('SELECT COUNT(*) as count FROM swipes');
|
||||
console.log(`\nСвайпов в БД: ${swipesResult.rows[0].count}`);
|
||||
|
||||
// Проверка таблицы profile_views
|
||||
const viewsResult = await client.query('SELECT COUNT(*) as count FROM profile_views');
|
||||
console.log(`Просмотров профилей в БД: ${viewsResult.rows[0].count}`);
|
||||
|
||||
// Проверка таблицы matches
|
||||
const matchesResult = await client.query('SELECT COUNT(*) as count FROM matches');
|
||||
console.log(`Матчей в БД: ${matchesResult.rows[0].count}`);
|
||||
|
||||
console.log('\n===== ПРОВЕРКА ЗАВЕРШЕНА =====\n');
|
||||
} catch (e) {
|
||||
console.error('Ошибка при проверке базы данных:', e);
|
||||
} finally {
|
||||
client.release();
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем проверку
|
||||
checkDatabase();
|
||||
64
scripts/legacy/checkProfileViews.js
Normal file
64
scripts/legacy/checkProfileViews.js
Normal file
@@ -0,0 +1,64 @@
|
||||
// Скрипт для проверки таблицы profile_views
|
||||
const { Pool } = require('pg');
|
||||
require('dotenv').config();
|
||||
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST,
|
||||
port: parseInt(process.env.DB_PORT || '5432'),
|
||||
database: process.env.DB_NAME,
|
||||
user: process.env.DB_USERNAME,
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
async function checkProfileViewsTable() {
|
||||
const client = await pool.connect();
|
||||
try {
|
||||
console.log('Проверка таблицы profile_views...');
|
||||
|
||||
// Проверяем наличие таблицы
|
||||
const tableCheck = await client.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.tables
|
||||
WHERE table_schema = 'public'
|
||||
AND table_name = 'profile_views'
|
||||
);
|
||||
`);
|
||||
|
||||
const tableExists = tableCheck.rows[0].exists;
|
||||
console.log(`Таблица profile_views ${tableExists ? 'существует' : 'не существует'}`);
|
||||
|
||||
if (tableExists) {
|
||||
// Проверяем количество записей в таблице
|
||||
const countResult = await client.query('SELECT COUNT(*) FROM profile_views');
|
||||
console.log(`Количество записей в таблице: ${countResult.rows[0].count}`);
|
||||
|
||||
// Получаем данные из таблицы
|
||||
const dataResult = await client.query(`
|
||||
SELECT pv.*,
|
||||
v.telegram_id as viewer_telegram_id,
|
||||
vp.telegram_id as viewed_telegram_id
|
||||
FROM profile_views pv
|
||||
LEFT JOIN users v ON pv.viewer_id = v.id
|
||||
LEFT JOIN users vp ON pv.viewed_profile_id = vp.id
|
||||
LIMIT 10
|
||||
`);
|
||||
|
||||
if (dataResult.rows.length > 0) {
|
||||
console.log('Данные из таблицы profile_views:');
|
||||
dataResult.rows.forEach((row, index) => {
|
||||
console.log(`${index + 1}. Просмотр: ${row.viewer_telegram_id || 'Неизвестно'} → ${row.viewed_telegram_id || 'Неизвестно'}, дата: ${row.view_date}`);
|
||||
});
|
||||
} else {
|
||||
console.log('Таблица profile_views пуста');
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка при проверке таблицы profile_views:', error);
|
||||
} finally {
|
||||
client.release();
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем проверку
|
||||
checkProfileViewsTable();
|
||||
74
scripts/legacy/checkUserTable.js
Normal file
74
scripts/legacy/checkUserTable.js
Normal file
@@ -0,0 +1,74 @@
|
||||
const { Pool } = require('pg');
|
||||
require('dotenv').config();
|
||||
|
||||
const pool = new Pool({
|
||||
user: process.env.DB_USER || 'postgres',
|
||||
host: process.env.DB_HOST || 'localhost',
|
||||
database: process.env.DB_NAME || 'telegram_tinder_db',
|
||||
password: process.env.DB_PASSWORD || 'postgres',
|
||||
port: parseInt(process.env.DB_PORT || '5432')
|
||||
});
|
||||
|
||||
async function checkUserTableStructure() {
|
||||
try {
|
||||
// Получаем информацию о структуре таблицы users
|
||||
const result = await pool.query(`
|
||||
SELECT column_name, data_type, is_nullable
|
||||
FROM information_schema.columns
|
||||
WHERE table_name = 'users'
|
||||
ORDER BY ordinal_position;
|
||||
`);
|
||||
|
||||
console.log('=== Структура таблицы users ===');
|
||||
console.table(result.rows);
|
||||
|
||||
// Проверяем наличие столбцов state и state_data
|
||||
const stateColumn = result.rows.find(row => row.column_name === 'state');
|
||||
const stateDataColumn = result.rows.find(row => row.column_name === 'state_data');
|
||||
|
||||
if (!stateColumn) {
|
||||
console.log('❌ Столбец state отсутствует в таблице users');
|
||||
} else {
|
||||
console.log('✅ Столбец state присутствует в таблице users');
|
||||
}
|
||||
|
||||
if (!stateDataColumn) {
|
||||
console.log('❌ Столбец state_data отсутствует в таблице users');
|
||||
} else {
|
||||
console.log('✅ Столбец state_data присутствует в таблице users');
|
||||
}
|
||||
|
||||
// Добавляем эти столбцы, если их нет
|
||||
if (!stateColumn || !stateDataColumn) {
|
||||
console.log('🔄 Добавление отсутствующих столбцов...');
|
||||
|
||||
await pool.query(`
|
||||
ALTER TABLE users
|
||||
ADD COLUMN IF NOT EXISTS state VARCHAR(255) NULL,
|
||||
ADD COLUMN IF NOT EXISTS state_data JSONB DEFAULT '{}'::jsonb;
|
||||
`);
|
||||
|
||||
console.log('✅ Столбцы успешно добавлены');
|
||||
}
|
||||
|
||||
// Проверяем наличие других таблиц, связанных с уведомлениями
|
||||
const tablesResult = await pool.query(`
|
||||
SELECT tablename
|
||||
FROM pg_catalog.pg_tables
|
||||
WHERE schemaname = 'public'
|
||||
AND tablename IN ('notifications', 'notification_settings', 'scheduled_notifications');
|
||||
`);
|
||||
|
||||
console.log('\n=== Таблицы для уведомлений ===');
|
||||
console.table(tablesResult.rows);
|
||||
|
||||
// Закрываем соединение
|
||||
await pool.end();
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка при проверке структуры таблицы:', error);
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
checkUserTableStructure();
|
||||
55
scripts/legacy/cleanDatabase.js
Normal file
55
scripts/legacy/cleanDatabase.js
Normal file
@@ -0,0 +1,55 @@
|
||||
const { Pool } = require('pg');
|
||||
require('dotenv').config();
|
||||
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST,
|
||||
port: parseInt(process.env.DB_PORT || '5432'),
|
||||
database: process.env.DB_NAME,
|
||||
user: process.env.DB_USERNAME,
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
async function cleanDatabase() {
|
||||
const client = await pool.connect();
|
||||
try {
|
||||
console.log('Очистка базы данных...');
|
||||
await client.query('BEGIN');
|
||||
|
||||
// Отключаем временно foreign key constraints
|
||||
await client.query('SET CONSTRAINTS ALL DEFERRED');
|
||||
|
||||
// Очищаем таблицы в правильном порядке
|
||||
console.log('Очистка таблицы messages...');
|
||||
await client.query('DELETE FROM messages');
|
||||
|
||||
console.log('Очистка таблицы profile_views...');
|
||||
await client.query('DELETE FROM profile_views');
|
||||
|
||||
console.log('Очистка таблицы matches...');
|
||||
await client.query('DELETE FROM matches');
|
||||
|
||||
console.log('Очистка таблицы swipes...');
|
||||
await client.query('DELETE FROM swipes');
|
||||
|
||||
console.log('Очистка таблицы profiles...');
|
||||
await client.query('DELETE FROM profiles');
|
||||
|
||||
console.log('Очистка таблицы users...');
|
||||
await client.query('DELETE FROM users');
|
||||
|
||||
// Возвращаем foreign key constraints
|
||||
await client.query('SET CONSTRAINTS ALL IMMEDIATE');
|
||||
|
||||
await client.query('COMMIT');
|
||||
console.log('✅ База данных успешно очищена');
|
||||
} catch (e) {
|
||||
await client.query('ROLLBACK');
|
||||
console.error('❌ Ошибка при очистке базы данных:', e);
|
||||
} finally {
|
||||
client.release();
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем функцию очистки
|
||||
cleanDatabase();
|
||||
66
scripts/legacy/clearDatabase.js
Normal file
66
scripts/legacy/clearDatabase.js
Normal file
@@ -0,0 +1,66 @@
|
||||
// Скрипт для очистки всех таблиц в базе данных
|
||||
import { Pool } from 'pg';
|
||||
import dotenv from 'dotenv';
|
||||
|
||||
// Загружаем переменные окружения из .env файла
|
||||
dotenv.config();
|
||||
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST,
|
||||
port: parseInt(process.env.DB_PORT || '5432'),
|
||||
database: process.env.DB_NAME,
|
||||
user: process.env.DB_USERNAME,
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
async function clearDatabase() {
|
||||
const client = await pool.connect();
|
||||
try {
|
||||
console.log('Начинаем очистку базы данных...');
|
||||
|
||||
// Начинаем транзакцию
|
||||
await client.query('BEGIN');
|
||||
|
||||
// Отключаем внешние ключи на время выполнения (если они используются)
|
||||
// await client.query('SET session_replication_role = \'replica\'');
|
||||
|
||||
// Очистка таблиц в порядке, учитывающем зависимости
|
||||
console.log('Очистка таблицы сообщений...');
|
||||
await client.query('TRUNCATE TABLE messages CASCADE');
|
||||
|
||||
console.log('Очистка таблицы просмотров профилей...');
|
||||
await client.query('TRUNCATE TABLE profile_views CASCADE');
|
||||
|
||||
console.log('Очистка таблицы свайпов...');
|
||||
await client.query('TRUNCATE TABLE swipes CASCADE');
|
||||
|
||||
console.log('Очистка таблицы матчей...');
|
||||
await client.query('TRUNCATE TABLE matches CASCADE');
|
||||
|
||||
console.log('Очистка таблицы профилей...');
|
||||
await client.query('TRUNCATE TABLE profiles CASCADE');
|
||||
|
||||
console.log('Очистка таблицы пользователей...');
|
||||
await client.query('TRUNCATE TABLE users CASCADE');
|
||||
|
||||
// Возвращаем внешние ключи (если они использовались)
|
||||
// await client.query('SET session_replication_role = \'origin\'');
|
||||
|
||||
// Фиксируем транзакцию
|
||||
await client.query('COMMIT');
|
||||
|
||||
console.log('Все таблицы успешно очищены!');
|
||||
} catch (error) {
|
||||
// В случае ошибки откатываем транзакцию
|
||||
await client.query('ROLLBACK');
|
||||
console.error('Произошла ошибка при очистке базы данных:', error);
|
||||
} finally {
|
||||
// Освобождаем клиента
|
||||
client.release();
|
||||
// Закрываем пул соединений
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем функцию очистки
|
||||
clearDatabase();
|
||||
81
scripts/legacy/clearDatabase.mjs
Normal file
81
scripts/legacy/clearDatabase.mjs
Normal file
@@ -0,0 +1,81 @@
|
||||
// Скрипт для очистки всех таблиц в базе данных
|
||||
import { Pool } from 'pg';
|
||||
import dotenv from 'dotenv';
|
||||
|
||||
// Загружаем переменные окружения из .env файла
|
||||
dotenv.config();
|
||||
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST,
|
||||
port: parseInt(process.env.DB_PORT || '5432'),
|
||||
database: process.env.DB_NAME,
|
||||
user: process.env.DB_USERNAME,
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
async function clearDatabase() {
|
||||
const client = await pool.connect();
|
||||
try {
|
||||
console.log('Начинаем очистку базы данных...');
|
||||
|
||||
// Начинаем транзакцию
|
||||
await client.query('BEGIN');
|
||||
|
||||
// Получаем список существующих таблиц
|
||||
const tablesResult = await client.query(`
|
||||
SELECT table_name
|
||||
FROM information_schema.tables
|
||||
WHERE table_schema = 'public'
|
||||
AND table_type = 'BASE TABLE'
|
||||
`);
|
||||
|
||||
const tables = tablesResult.rows.map(row => row.table_name);
|
||||
console.log('Найдены таблицы:', tables.join(', '));
|
||||
|
||||
// Очистка таблиц в порядке, учитывающем зависимости
|
||||
if (tables.includes('messages')) {
|
||||
console.log('Очистка таблицы messages...');
|
||||
await client.query('TRUNCATE TABLE messages CASCADE');
|
||||
}
|
||||
|
||||
if (tables.includes('swipes')) {
|
||||
console.log('Очистка таблицы swipes...');
|
||||
await client.query('TRUNCATE TABLE swipes CASCADE');
|
||||
}
|
||||
|
||||
if (tables.includes('matches')) {
|
||||
console.log('Очистка таблицы matches...');
|
||||
await client.query('TRUNCATE TABLE matches CASCADE');
|
||||
}
|
||||
|
||||
if (tables.includes('profiles')) {
|
||||
console.log('Очистка таблицы profiles...');
|
||||
await client.query('TRUNCATE TABLE profiles CASCADE');
|
||||
}
|
||||
|
||||
if (tables.includes('users')) {
|
||||
console.log('Очистка таблицы users...');
|
||||
await client.query('TRUNCATE TABLE users CASCADE');
|
||||
}
|
||||
|
||||
// Возвращаем внешние ключи (если они использовались)
|
||||
// await client.query('SET session_replication_role = \'origin\'');
|
||||
|
||||
// Фиксируем транзакцию
|
||||
await client.query('COMMIT');
|
||||
|
||||
console.log('Все таблицы успешно очищены!');
|
||||
} catch (error) {
|
||||
// В случае ошибки откатываем транзакцию
|
||||
await client.query('ROLLBACK');
|
||||
console.error('Произошла ошибка при очистке базы данных:', error);
|
||||
} finally {
|
||||
// Освобождаем клиента
|
||||
client.release();
|
||||
// Закрываем пул соединений
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем функцию очистки
|
||||
clearDatabase();
|
||||
26
scripts/legacy/clear_database.sql
Normal file
26
scripts/legacy/clear_database.sql
Normal file
@@ -0,0 +1,26 @@
|
||||
-- Скрипт для очистки всех таблиц в базе данных
|
||||
-- Важно: таблицы очищаются в порядке, учитывающем зависимости между ними
|
||||
|
||||
-- Отключаем внешние ключи на время выполнения (если они используются)
|
||||
-- SET session_replication_role = 'replica';
|
||||
|
||||
-- Очистка таблицы сообщений
|
||||
TRUNCATE TABLE messages CASCADE;
|
||||
|
||||
-- Очистка таблицы просмотров профилей
|
||||
TRUNCATE TABLE profile_views CASCADE;
|
||||
|
||||
-- Очистка таблицы свайпов
|
||||
TRUNCATE TABLE swipes CASCADE;
|
||||
|
||||
-- Очистка таблицы матчей
|
||||
TRUNCATE TABLE matches CASCADE;
|
||||
|
||||
-- Очистка таблицы профилей
|
||||
TRUNCATE TABLE profiles CASCADE;
|
||||
|
||||
-- Очистка таблицы пользователей
|
||||
TRUNCATE TABLE users CASCADE;
|
||||
|
||||
-- Возвращаем внешние ключи (если они использовались)
|
||||
-- SET session_replication_role = 'origin';
|
||||
70
scripts/legacy/createProfileViewsTable.ts
Normal file
70
scripts/legacy/createProfileViewsTable.ts
Normal file
@@ -0,0 +1,70 @@
|
||||
// Скрипт для создания таблицы profile_views
|
||||
|
||||
// Функция для ручного запуска создания таблицы profile_views
|
||||
async function createProfileViewsTable() {
|
||||
const client = await require('../database/connection').pool.connect();
|
||||
try {
|
||||
console.log('Создание таблицы profile_views...');
|
||||
|
||||
// Проверяем, существует ли уже таблица profile_views
|
||||
const tableCheck = await client.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.tables
|
||||
WHERE table_schema = 'public'
|
||||
AND table_name = 'profile_views'
|
||||
);
|
||||
`);
|
||||
|
||||
if (tableCheck.rows[0].exists) {
|
||||
console.log('Таблица profile_views уже существует, пропускаем создание');
|
||||
return;
|
||||
}
|
||||
|
||||
// Начинаем транзакцию
|
||||
await client.query('BEGIN');
|
||||
|
||||
// Создаем таблицу profile_views
|
||||
await client.query(`
|
||||
CREATE TABLE profile_views (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
viewer_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
viewed_profile_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
view_date TIMESTAMP NOT NULL DEFAULT NOW(),
|
||||
view_type VARCHAR(20) NOT NULL DEFAULT 'browse'
|
||||
);
|
||||
`);
|
||||
|
||||
// Создаем индекс для быстрого поиска по паре (просмотревший - просмотренный)
|
||||
await client.query(`
|
||||
CREATE UNIQUE INDEX profile_views_viewer_viewed_idx ON profile_views (viewer_id, viewed_profile_id);
|
||||
`);
|
||||
|
||||
// Индекс для быстрого поиска по viewer_id
|
||||
await client.query(`
|
||||
CREATE INDEX profile_views_viewer_idx ON profile_views (viewer_id);
|
||||
`);
|
||||
|
||||
// Индекс для быстрого поиска по viewed_profile_id
|
||||
await client.query(`
|
||||
CREATE INDEX profile_views_viewed_idx ON profile_views (viewed_profile_id);
|
||||
`);
|
||||
|
||||
// Фиксируем транзакцию
|
||||
await client.query('COMMIT');
|
||||
|
||||
console.log('Таблица profile_views успешно создана!');
|
||||
} catch (error) {
|
||||
// В случае ошибки откатываем транзакцию
|
||||
await client.query('ROLLBACK');
|
||||
console.error('Произошла ошибка при создании таблицы profile_views:', error);
|
||||
} finally {
|
||||
// Освобождаем клиента
|
||||
client.release();
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем функцию создания таблицы
|
||||
createProfileViewsTable()
|
||||
.then(() => console.log('Скрипт выполнен'))
|
||||
.catch(err => console.error('Ошибка выполнения скрипта:', err))
|
||||
.finally(() => process.exit());
|
||||
142
scripts/legacy/fixCallbackHandlers.js
Normal file
142
scripts/legacy/fixCallbackHandlers.js
Normal file
@@ -0,0 +1,142 @@
|
||||
// Скрипт для восстановления оригинальной функциональности callbackHandlers.ts
|
||||
const fs = require('fs');
|
||||
const path = require('path');
|
||||
|
||||
// Находим самую последнюю версию файла callbackHandlers.ts в репозитории
|
||||
const { execSync } = require('child_process');
|
||||
|
||||
try {
|
||||
console.log('Поиск оригинальной версии CallbackHandlers.ts с полной функциональностью...');
|
||||
|
||||
// Находим коммиты, содержащие значительные изменения в файле (более 1000 символов)
|
||||
const commits = execSync('git log --format="%H" -- src/handlers/callbackHandlers.ts')
|
||||
.toString()
|
||||
.trim()
|
||||
.split('\n');
|
||||
|
||||
console.log(`Найдено ${commits.length} коммитов с изменениями файла`);
|
||||
|
||||
// Пробуем разные коммиты, начиная с последнего, чтобы найти полную реализацию
|
||||
let foundFullImplementation = false;
|
||||
let fullImplementationContent = '';
|
||||
|
||||
for (const commit of commits) {
|
||||
console.log(`Проверяем коммит ${commit.substring(0, 8)}...`);
|
||||
|
||||
try {
|
||||
const fileContent = execSync(`git show ${commit}:src/handlers/callbackHandlers.ts`).toString();
|
||||
|
||||
// Проверяем, содержит ли файл полные реализации методов
|
||||
const hasFullImplementations = !fileContent.includes('// Заглушка метода') &&
|
||||
fileContent.includes('await this.bot.sendMessage');
|
||||
|
||||
if (hasFullImplementations) {
|
||||
console.log(`✅ Найдена полная реализация в коммите ${commit.substring(0, 8)}`);
|
||||
fullImplementationContent = fileContent;
|
||||
foundFullImplementation = true;
|
||||
break;
|
||||
} else {
|
||||
console.log(`❌ Коммит ${commit.substring(0, 8)} не содержит полной реализации`);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error(`Ошибка при проверке коммита ${commit}:`, error.message);
|
||||
}
|
||||
}
|
||||
|
||||
if (!foundFullImplementation) {
|
||||
console.error('❌ Не удалось найти полную реализацию в истории коммитов');
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
// Теперь получаем текущую версию файла с поддержкой уведомлений
|
||||
console.log('Получаем текущую версию с поддержкой уведомлений...');
|
||||
const currentFilePath = path.join(__dirname, '..', 'src', 'handlers', 'callbackHandlers.ts');
|
||||
const currentContent = fs.readFileSync(currentFilePath, 'utf-8');
|
||||
|
||||
// Сначала создаем бэкап текущего файла
|
||||
const backupPath = currentFilePath + '.backup-' + Date.now();
|
||||
fs.writeFileSync(backupPath, currentContent);
|
||||
console.log(`✅ Создан бэкап текущей версии: ${path.basename(backupPath)}`);
|
||||
|
||||
// Извлекаем код для поддержки уведомлений из текущей версии
|
||||
console.log('Извлекаем код для поддержки уведомлений...');
|
||||
|
||||
// Находим импорт NotificationHandlers
|
||||
const notificationImportRegex = /import\s+{\s*NotificationHandlers\s*}\s*from\s*['"]\.\/notificationHandlers['"]\s*;/;
|
||||
const notificationImport = currentContent.match(notificationImportRegex)?.[0] || '';
|
||||
|
||||
// Находим объявление поля notificationHandlers
|
||||
const notificationFieldRegex = /private\s+notificationHandlers\?\s*:\s*NotificationHandlers\s*;/;
|
||||
const notificationField = currentContent.match(notificationFieldRegex)?.[0] || '';
|
||||
|
||||
// Находим инициализацию notificationHandlers в конструкторе
|
||||
const notificationInitRegex = /\/\/\s*Создаем экземпляр NotificationHandlers[\s\S]*?try\s*{[\s\S]*?this\.notificationHandlers\s*=\s*new\s*NotificationHandlers[\s\S]*?}\s*catch[\s\S]*?}/;
|
||||
const notificationInit = currentContent.match(notificationInitRegex)?.[0] || '';
|
||||
|
||||
// Находим метод handleNotificationSettings
|
||||
const notificationSettingsMethodRegex = /async\s+handleNotificationSettings[\s\S]*?}\s*}/;
|
||||
const notificationSettingsMethod = currentContent.match(notificationSettingsMethodRegex)?.[0] || '';
|
||||
|
||||
// Находим обработку callback для notifications в handleCallback
|
||||
const notificationCallbackRegex = /\/\/\s*Настройки уведомлений[\s\S]*?else\s+if\s*\(data\s*===\s*['"]notifications['"][\s\S]*?}\s*}/;
|
||||
const notificationCallback = currentContent.match(notificationCallbackRegex)?.[0] || '';
|
||||
|
||||
// Получаем часть обработки коллбэков для уведомлений
|
||||
const notificationToggleRegex = /\/\/\s*Обработка переключения настроек уведомлений[\s\S]*?else\s+if[\s\S]*?notif_[\s\S]*?}\s*}/;
|
||||
const notificationToggle = currentContent.match(notificationToggleRegex)?.[0] || '';
|
||||
|
||||
console.log(`✅ Извлечены блоки кода для уведомлений`);
|
||||
|
||||
// Интегрируем код уведомлений в оригинальную версию
|
||||
console.log('Интегрируем код уведомлений в оригинальную версию...');
|
||||
|
||||
// 1. Добавляем импорт
|
||||
let newContent = fullImplementationContent;
|
||||
if (notificationImport) {
|
||||
newContent = newContent.replace(/import\s*{[^}]*}\s*from\s*['"]\.\/messageHandlers['"]\s*;/,
|
||||
match => match + '\n' + notificationImport);
|
||||
}
|
||||
|
||||
// 2. Добавляем объявление поля
|
||||
if (notificationField) {
|
||||
newContent = newContent.replace(/private\s+translationController\s*:\s*TranslationController\s*;/,
|
||||
match => match + '\n ' + notificationField);
|
||||
}
|
||||
|
||||
// 3. Добавляем инициализацию в конструкторе
|
||||
if (notificationInit) {
|
||||
newContent = newContent.replace(/this\.translationController\s*=\s*new\s*TranslationController\(\);/,
|
||||
match => match + '\n ' + notificationInit);
|
||||
}
|
||||
|
||||
// 4. Добавляем обработку коллбэков для уведомлений
|
||||
if (notificationCallback) {
|
||||
newContent = newContent.replace(/else\s+{\s*await\s+this\.bot\.answerCallbackQuery\(query\.id[\s\S]*?return;/,
|
||||
match => notificationCallback + '\n ' + match);
|
||||
}
|
||||
|
||||
// 5. Добавляем обработку переключения настроек уведомлений
|
||||
if (notificationToggle) {
|
||||
newContent = newContent.replace(/else\s+{\s*await\s+this\.bot\.answerCallbackQuery\(query\.id[\s\S]*?return;/,
|
||||
match => notificationToggle + '\n ' + match);
|
||||
}
|
||||
|
||||
// 6. Добавляем метод handleNotificationSettings в конец класса
|
||||
if (notificationSettingsMethod) {
|
||||
newContent = newContent.replace(/}(\s*)$/, notificationSettingsMethod + '\n}$1');
|
||||
}
|
||||
|
||||
// Сохраняем обновленный файл
|
||||
const outputPath = currentFilePath + '.fixed';
|
||||
fs.writeFileSync(outputPath, newContent);
|
||||
console.log(`✅ Создана исправленная версия файла: ${path.basename(outputPath)}`);
|
||||
|
||||
console.log('\nИнструкция по восстановлению:');
|
||||
console.log(`1. Проверьте файл ${path.basename(outputPath)}`);
|
||||
console.log('2. Если все выглядит правильно, выполните команду:');
|
||||
console.log(` Move-Item -Force "${path.basename(outputPath)}" "${path.basename(currentFilePath)}"`);
|
||||
console.log('3. Перезапустите бота');
|
||||
|
||||
} catch (error) {
|
||||
console.error('Произошла ошибка:', error);
|
||||
}
|
||||
170
scripts/legacy/fixDatabaseStructure.js
Normal file
170
scripts/legacy/fixDatabaseStructure.js
Normal file
@@ -0,0 +1,170 @@
|
||||
// Скрипт для исправления проблемы с ботом
|
||||
require('dotenv').config();
|
||||
const { Pool } = require('pg');
|
||||
|
||||
// Получаем данные подключения из .env
|
||||
console.log('Параметры подключения к БД:');
|
||||
console.log('DB_USERNAME:', process.env.DB_USERNAME);
|
||||
console.log('DB_HOST:', process.env.DB_HOST);
|
||||
console.log('DB_NAME:', process.env.DB_NAME);
|
||||
console.log('DB_PASSWORD:', process.env.DB_PASSWORD ? '[указан]' : '[не указан]');
|
||||
console.log('DB_PORT:', process.env.DB_PORT);
|
||||
|
||||
// Создаем пул соединений
|
||||
const pool = new Pool({
|
||||
user: process.env.DB_USERNAME,
|
||||
host: process.env.DB_HOST,
|
||||
database: process.env.DB_NAME,
|
||||
password: process.env.DB_PASSWORD,
|
||||
port: parseInt(process.env.DB_PORT || '5432')
|
||||
});
|
||||
|
||||
async function fixDatabase() {
|
||||
try {
|
||||
console.log('Начинаем исправление базы данных...');
|
||||
|
||||
// Проверяем существование таблицы users
|
||||
const tableResult = await pool.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.tables
|
||||
WHERE table_schema = 'public'
|
||||
AND table_name = 'users'
|
||||
);
|
||||
`);
|
||||
|
||||
if (!tableResult.rows[0].exists) {
|
||||
console.error('Таблица users не найдена!');
|
||||
return;
|
||||
}
|
||||
|
||||
console.log('✅ Таблица users существует');
|
||||
|
||||
// Проверяем и добавляем столбцы state и state_data, если они отсутствуют
|
||||
console.log('Проверяем наличие столбцов state и state_data...');
|
||||
|
||||
const stateColumnResult = await pool.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.columns
|
||||
WHERE table_schema = 'public'
|
||||
AND table_name = 'users'
|
||||
AND column_name = 'state'
|
||||
);
|
||||
`);
|
||||
|
||||
const stateDataColumnResult = await pool.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.columns
|
||||
WHERE table_schema = 'public'
|
||||
AND table_name = 'users'
|
||||
AND column_name = 'state_data'
|
||||
);
|
||||
`);
|
||||
|
||||
if (!stateColumnResult.rows[0].exists) {
|
||||
console.log('🔄 Добавляем столбец state...');
|
||||
await pool.query(`ALTER TABLE users ADD COLUMN state VARCHAR(255) NULL;`);
|
||||
console.log('✅ Столбец state успешно добавлен');
|
||||
} else {
|
||||
console.log('✅ Столбец state уже существует');
|
||||
}
|
||||
|
||||
if (!stateDataColumnResult.rows[0].exists) {
|
||||
console.log('🔄 Добавляем столбец state_data...');
|
||||
await pool.query(`ALTER TABLE users ADD COLUMN state_data JSONB DEFAULT '{}'::jsonb;`);
|
||||
console.log('✅ Столбец state_data успешно добавлен');
|
||||
} else {
|
||||
console.log('✅ Столбец state_data уже существует');
|
||||
}
|
||||
|
||||
// Проверка наличия таблиц для уведомлений
|
||||
console.log('Проверяем наличие таблиц для уведомлений...');
|
||||
|
||||
const tablesCheck = await Promise.all([
|
||||
checkTableExists('notifications'),
|
||||
checkTableExists('notification_settings'),
|
||||
checkTableExists('scheduled_notifications')
|
||||
]);
|
||||
|
||||
// Создаем отсутствующие таблицы
|
||||
if (!tablesCheck[0]) {
|
||||
console.log('🔄 Создаем таблицу notifications...');
|
||||
await pool.query(`
|
||||
CREATE TABLE notifications (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
content JSONB NOT NULL DEFAULT '{}',
|
||||
is_read BOOLEAN DEFAULT false,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||
);
|
||||
CREATE INDEX idx_notifications_user_id ON notifications(user_id);
|
||||
CREATE INDEX idx_notifications_type ON notifications(type);
|
||||
CREATE INDEX idx_notifications_created_at ON notifications(created_at);
|
||||
`);
|
||||
console.log('✅ Таблица notifications успешно создана');
|
||||
}
|
||||
|
||||
if (!tablesCheck[1]) {
|
||||
console.log('🔄 Создаем таблицу notification_settings...');
|
||||
await pool.query(`
|
||||
CREATE TABLE notification_settings (
|
||||
user_id UUID PRIMARY KEY REFERENCES users(id) ON DELETE CASCADE,
|
||||
new_matches BOOLEAN DEFAULT true,
|
||||
new_messages BOOLEAN DEFAULT true,
|
||||
new_likes BOOLEAN DEFAULT true,
|
||||
reminders BOOLEAN DEFAULT true,
|
||||
daily_summary BOOLEAN DEFAULT false,
|
||||
time_preference VARCHAR(20) DEFAULT 'evening',
|
||||
do_not_disturb BOOLEAN DEFAULT false,
|
||||
do_not_disturb_start TIME,
|
||||
do_not_disturb_end TIME,
|
||||
updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||
);
|
||||
`);
|
||||
console.log('✅ Таблица notification_settings успешно создана');
|
||||
}
|
||||
|
||||
if (!tablesCheck[2]) {
|
||||
console.log('🔄 Создаем таблицу scheduled_notifications...');
|
||||
await pool.query(`
|
||||
CREATE TABLE scheduled_notifications (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
content JSONB NOT NULL DEFAULT '{}',
|
||||
scheduled_at TIMESTAMP WITH TIME ZONE NOT NULL,
|
||||
processed BOOLEAN DEFAULT false,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||
);
|
||||
CREATE INDEX idx_scheduled_notifications_user_id ON scheduled_notifications(user_id);
|
||||
CREATE INDEX idx_scheduled_notifications_scheduled_at ON scheduled_notifications(scheduled_at);
|
||||
CREATE INDEX idx_scheduled_notifications_processed ON scheduled_notifications(processed);
|
||||
`);
|
||||
console.log('✅ Таблица scheduled_notifications успешно создана');
|
||||
}
|
||||
|
||||
console.log('✅ Исправление базы данных завершено успешно');
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка при исправлении базы данных:', error);
|
||||
} finally {
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
async function checkTableExists(tableName) {
|
||||
const result = await pool.query(`
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.tables
|
||||
WHERE table_schema = 'public'
|
||||
AND table_name = $1
|
||||
);
|
||||
`, [tableName]);
|
||||
|
||||
const exists = result.rows[0].exists;
|
||||
console.log(`${exists ? '✅' : '❌'} Таблица ${tableName} ${exists ? 'существует' : 'отсутствует'}`);
|
||||
|
||||
return exists;
|
||||
}
|
||||
|
||||
fixDatabase();
|
||||
48
scripts/legacy/fix_all_notifications.js
Normal file
48
scripts/legacy/fix_all_notifications.js
Normal file
@@ -0,0 +1,48 @@
|
||||
/**
|
||||
* Комплексный скрипт для исправления всех проблем с уведомлениями
|
||||
* Запускает последовательно оба скрипта исправления
|
||||
*/
|
||||
|
||||
const { exec } = require('child_process');
|
||||
const path = require('path');
|
||||
|
||||
console.log('🔧 Запуск комплексного исправления проблем с уведомлениями...');
|
||||
|
||||
// Путь к скриптам
|
||||
const fixNotificationCallbacksScript = path.join(__dirname, 'fix_notification_callbacks.js');
|
||||
const updateBotWithNotificationsScript = path.join(__dirname, 'update_bot_with_notifications.js');
|
||||
|
||||
// Запуск первого скрипта для исправления таблиц и колонок
|
||||
console.log('\n📊 Шаг 1/2: Проверка и исправление таблиц базы данных...');
|
||||
exec(`node ${fixNotificationCallbacksScript}`, (error, stdout, stderr) => {
|
||||
if (error) {
|
||||
console.error(`❌ Ошибка при запуске скрипта исправления таблиц: ${error}`);
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(stdout);
|
||||
|
||||
if (stderr) {
|
||||
console.error(`❌ Ошибки при выполнении скрипта: ${stderr}`);
|
||||
}
|
||||
|
||||
// Запуск второго скрипта для обновления bot.ts
|
||||
console.log('\n📝 Шаг 2/2: Обновление файла bot.ts для регистрации обработчиков уведомлений...');
|
||||
exec(`node ${updateBotWithNotificationsScript}`, (error2, stdout2, stderr2) => {
|
||||
if (error2) {
|
||||
console.error(`❌ Ошибка при запуске скрипта обновления bot.ts: ${error2}`);
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(stdout2);
|
||||
|
||||
if (stderr2) {
|
||||
console.error(`❌ Ошибки при выполнении скрипта: ${stderr2}`);
|
||||
}
|
||||
|
||||
console.log('\n✅ Все исправления успешно выполнены!');
|
||||
console.log('🔄 Пожалуйста, перезапустите бота для применения изменений:');
|
||||
console.log(' npm run start');
|
||||
console.log('\n💡 Уведомления должны теперь работать корректно!');
|
||||
});
|
||||
});
|
||||
332
scripts/legacy/fix_notification_callbacks.js
Normal file
332
scripts/legacy/fix_notification_callbacks.js
Normal file
@@ -0,0 +1,332 @@
|
||||
/**
|
||||
* Скрипт для проверки и исправления проблем с обработчиками уведомлений в боте
|
||||
*/
|
||||
|
||||
const { Client } = require('pg');
|
||||
const fs = require('fs');
|
||||
|
||||
// Конфигурация базы данных
|
||||
const dbConfig = {
|
||||
host: 'localhost',
|
||||
port: 5432,
|
||||
database: 'telegram_tinder',
|
||||
user: 'postgres',
|
||||
password: 'postgres'
|
||||
};
|
||||
|
||||
// Подключение к базе данных
|
||||
const client = new Client(dbConfig);
|
||||
|
||||
async function main() {
|
||||
try {
|
||||
console.log('Подключение к базе данных...');
|
||||
await client.connect();
|
||||
console.log('Успешно подключено к базе данных');
|
||||
|
||||
// Шаг 1: Проверка существования необходимых таблиц для уведомлений
|
||||
console.log('\n=== Проверка таблиц для уведомлений ===');
|
||||
|
||||
// Проверяем таблицу notifications
|
||||
let notificationsTableExists = await checkTableExists('notifications');
|
||||
if (!notificationsTableExists) {
|
||||
console.log('Таблица notifications не найдена. Создаем...');
|
||||
await createNotificationsTable();
|
||||
console.log('Таблица notifications успешно создана');
|
||||
} else {
|
||||
console.log('Таблица notifications уже существует');
|
||||
}
|
||||
|
||||
// Проверяем таблицу scheduled_notifications
|
||||
let scheduledNotificationsTableExists = await checkTableExists('scheduled_notifications');
|
||||
if (!scheduledNotificationsTableExists) {
|
||||
console.log('Таблица scheduled_notifications не найдена. Создаем...');
|
||||
await createScheduledNotificationsTable();
|
||||
console.log('Таблица scheduled_notifications успешно создана');
|
||||
} else {
|
||||
console.log('Таблица scheduled_notifications уже существует');
|
||||
}
|
||||
|
||||
// Проверяем таблицу notification_templates
|
||||
let notificationTemplatesTableExists = await checkTableExists('notification_templates');
|
||||
if (!notificationTemplatesTableExists) {
|
||||
console.log('Таблица notification_templates не найдена. Создаем...');
|
||||
await createNotificationTemplatesTable();
|
||||
console.log('Таблица notification_templates успешно создана');
|
||||
console.log('Заполняем таблицу базовыми шаблонами...');
|
||||
await populateDefaultTemplates();
|
||||
console.log('Шаблоны успешно добавлены');
|
||||
} else {
|
||||
console.log('Таблица notification_templates уже существует');
|
||||
}
|
||||
|
||||
// Шаг 2: Проверка существования столбца notification_settings в таблице users
|
||||
console.log('\n=== Проверка столбца notification_settings в таблице users ===');
|
||||
|
||||
const notificationSettingsColumnExists = await checkColumnExists('users', 'notification_settings');
|
||||
if (!notificationSettingsColumnExists) {
|
||||
console.log('Столбец notification_settings не найден. Добавляем...');
|
||||
await addNotificationSettingsColumn();
|
||||
console.log('Столбец notification_settings успешно добавлен');
|
||||
} else {
|
||||
console.log('Столбец notification_settings уже существует');
|
||||
}
|
||||
|
||||
// Шаг 3: Проверка существования столбцов state и state_data в таблице users
|
||||
console.log('\n=== Проверка столбцов state и state_data в таблице users ===');
|
||||
|
||||
const stateColumnExists = await checkColumnExists('users', 'state');
|
||||
if (!stateColumnExists) {
|
||||
console.log('Столбец state не найден. Добавляем...');
|
||||
await addStateColumn();
|
||||
console.log('Столбец state успешно добавлен');
|
||||
} else {
|
||||
console.log('Столбец state уже существует');
|
||||
}
|
||||
|
||||
const stateDataColumnExists = await checkColumnExists('users', 'state_data');
|
||||
if (!stateDataColumnExists) {
|
||||
console.log('Столбец state_data не найден. Добавляем...');
|
||||
await addStateDataColumn();
|
||||
console.log('Столбец state_data успешно добавлен');
|
||||
} else {
|
||||
console.log('Столбец state_data уже существует');
|
||||
}
|
||||
|
||||
console.log('\nВсе таблицы и столбцы успешно проверены и созданы при необходимости.');
|
||||
console.log('Механизм уведомлений должен работать корректно.');
|
||||
|
||||
console.log('\n=== Проверка регистрации обработчиков уведомлений ===');
|
||||
console.log('Подсказка: убедитесь, что в файле bot.ts создается экземпляр NotificationHandlers и регистрируются его обработчики:');
|
||||
console.log(`
|
||||
// Настройка обработчиков уведомлений
|
||||
const notificationHandlers = new NotificationHandlers(bot);
|
||||
notificationHandlers.register();
|
||||
|
||||
// Запуск обработчика запланированных уведомлений
|
||||
setInterval(() => {
|
||||
const notificationService = new NotificationService(bot);
|
||||
notificationService.processScheduledNotifications();
|
||||
}, 60000); // Проверяем каждую минуту
|
||||
`);
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка выполнения скрипта:', error);
|
||||
} finally {
|
||||
await client.end();
|
||||
console.log('\nСоединение с базой данных закрыто.');
|
||||
}
|
||||
}
|
||||
|
||||
async function checkTableExists(tableName) {
|
||||
const query = `
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.tables
|
||||
WHERE table_name = $1
|
||||
) as exists
|
||||
`;
|
||||
|
||||
const result = await client.query(query, [tableName]);
|
||||
return result.rows[0].exists;
|
||||
}
|
||||
|
||||
async function checkColumnExists(tableName, columnName) {
|
||||
const query = `
|
||||
SELECT EXISTS (
|
||||
SELECT FROM information_schema.columns
|
||||
WHERE table_name = $1 AND column_name = $2
|
||||
) as exists
|
||||
`;
|
||||
|
||||
const result = await client.query(query, [tableName, columnName]);
|
||||
return result.rows[0].exists;
|
||||
}
|
||||
|
||||
async function createNotificationsTable() {
|
||||
await client.query(`
|
||||
CREATE EXTENSION IF NOT EXISTS "uuid-ossp";
|
||||
|
||||
CREATE TABLE notifications (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
data JSONB,
|
||||
is_read BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP DEFAULT NOW()
|
||||
)
|
||||
`);
|
||||
}
|
||||
|
||||
async function createScheduledNotificationsTable() {
|
||||
await client.query(`
|
||||
CREATE EXTENSION IF NOT EXISTS "uuid-ossp";
|
||||
|
||||
CREATE TABLE scheduled_notifications (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
data JSONB,
|
||||
scheduled_at TIMESTAMP NOT NULL,
|
||||
processed BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP DEFAULT NOW()
|
||||
)
|
||||
`);
|
||||
}
|
||||
|
||||
async function createNotificationTemplatesTable() {
|
||||
await client.query(`
|
||||
CREATE EXTENSION IF NOT EXISTS "uuid-ossp";
|
||||
|
||||
CREATE TABLE notification_templates (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
type VARCHAR(50) NOT NULL UNIQUE,
|
||||
title TEXT NOT NULL,
|
||||
message_template TEXT NOT NULL,
|
||||
button_template JSONB,
|
||||
created_at TIMESTAMP DEFAULT NOW(),
|
||||
updated_at TIMESTAMP DEFAULT NOW()
|
||||
)
|
||||
`);
|
||||
}
|
||||
|
||||
async function addNotificationSettingsColumn() {
|
||||
await client.query(`
|
||||
ALTER TABLE users
|
||||
ADD COLUMN notification_settings JSONB DEFAULT '{
|
||||
"newMatches": true,
|
||||
"newMessages": true,
|
||||
"newLikes": true,
|
||||
"reminders": true,
|
||||
"dailySummary": true,
|
||||
"timePreference": "evening",
|
||||
"doNotDisturb": false
|
||||
}'::jsonb
|
||||
`);
|
||||
}
|
||||
|
||||
async function addStateColumn() {
|
||||
await client.query(`
|
||||
ALTER TABLE users ADD COLUMN state VARCHAR(255) NULL
|
||||
`);
|
||||
}
|
||||
|
||||
async function addStateDataColumn() {
|
||||
await client.query(`
|
||||
ALTER TABLE users ADD COLUMN state_data JSONB DEFAULT '{}'::jsonb
|
||||
`);
|
||||
}
|
||||
|
||||
async function populateDefaultTemplates() {
|
||||
const templates = [
|
||||
{
|
||||
type: 'new_like',
|
||||
title: 'Новый лайк!',
|
||||
message_template: '❤️ *{{name}}* поставил(а) вам лайк!\n\nВозраст: {{age}}\n{{city}}\n\nОтветьте взаимностью или посмотрите профиль.',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '👀 Посмотреть профиль', callback_data: 'view_profile:{{userId}}' }],
|
||||
[
|
||||
{ text: '❤️ Лайк в ответ', callback_data: 'like_back:{{userId}}' },
|
||||
{ text: '⛔️ Пропустить', callback_data: 'dislike_profile:{{userId}}' }
|
||||
],
|
||||
[{ text: '💕 Открыть все лайки', callback_data: 'view_likes' }]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'super_like',
|
||||
title: 'Супер-лайк!',
|
||||
message_template: '⭐️ *{{name}}* отправил(а) вам супер-лайк!\n\nВозраст: {{age}}\n{{city}}\n\nВы произвели особое впечатление! Ответьте взаимностью или посмотрите профиль.',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '👀 Посмотреть профиль', callback_data: 'view_profile:{{userId}}' }],
|
||||
[
|
||||
{ text: '❤️ Лайк в ответ', callback_data: 'like_back:{{userId}}' },
|
||||
{ text: '⛔️ Пропустить', callback_data: 'dislike_profile:{{userId}}' }
|
||||
],
|
||||
[{ text: '💕 Открыть все лайки', callback_data: 'view_likes' }]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'new_match',
|
||||
title: 'Новый матч!',
|
||||
message_template: '🎊 *Ура! Это взаимно!* 🎊\n\nВы и *{{name}}* понравились друг другу!\nВозраст: {{age}}\n{{city}}\n\nСделайте первый шаг - напишите сообщение!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '💬 Начать общение', callback_data: 'open_chat:{{matchId}}' }],
|
||||
[
|
||||
{ text: '👀 Посмотреть профиль', callback_data: 'view_profile:{{userId}}' },
|
||||
{ text: '📋 Все матчи', callback_data: 'view_matches' }
|
||||
]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'new_message',
|
||||
title: 'Новое сообщение!',
|
||||
message_template: '💌 *Новое сообщение!*\n\nОт: *{{name}}*\n\n"{{message}}"\n\nОтветьте на сообщение прямо сейчас!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '📩 Ответить', callback_data: 'open_chat:{{matchId}}' }],
|
||||
[
|
||||
{ text: '👤 Профиль', callback_data: 'view_profile:{{userId}}' },
|
||||
{ text: '📋 Все чаты', callback_data: 'view_matches' }
|
||||
]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'match_reminder',
|
||||
title: 'Напоминание о матче',
|
||||
message_template: '💕 У вас есть матч с *{{name}}*, но вы еще не начали общение!\n\nНе упустите шанс познакомиться поближе!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '💬 Начать общение', callback_data: 'open_chat:{{matchId}}' }],
|
||||
[{ text: '👀 Посмотреть профиль', callback_data: 'view_profile:{{userId}}' }]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'inactive_matches',
|
||||
title: 'Неактивные матчи',
|
||||
message_template: '⏰ У вас {{count}} неактивных матчей!\n\nПродолжите общение, чтобы не упустить интересные знакомства!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '📋 Открыть матчи', callback_data: 'view_matches' }],
|
||||
[{ text: '💕 Смотреть новые анкеты', callback_data: 'start_browsing' }]
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
type: 'like_summary',
|
||||
title: 'Сводка лайков',
|
||||
message_template: '💖 У вас {{count}} новых лайков!\n\nПосмотрите, кто проявил к вам интерес сегодня!',
|
||||
button_template: {
|
||||
inline_keyboard: [
|
||||
[{ text: '👀 Посмотреть лайки', callback_data: 'view_likes' }],
|
||||
[{ text: '💕 Начать знакомиться', callback_data: 'start_browsing' }]
|
||||
]
|
||||
}
|
||||
}
|
||||
];
|
||||
|
||||
for (const template of templates) {
|
||||
await client.query(`
|
||||
INSERT INTO notification_templates (type, title, message_template, button_template)
|
||||
VALUES ($1, $2, $3, $4)
|
||||
ON CONFLICT (type) DO UPDATE
|
||||
SET title = EXCLUDED.title,
|
||||
message_template = EXCLUDED.message_template,
|
||||
button_template = EXCLUDED.button_template,
|
||||
updated_at = NOW()
|
||||
`, [
|
||||
template.type,
|
||||
template.title,
|
||||
template.message_template,
|
||||
JSON.stringify(template.button_template)
|
||||
]);
|
||||
}
|
||||
}
|
||||
|
||||
// Запуск скрипта
|
||||
main();
|
||||
85
scripts/legacy/testCallbacks.js
Normal file
85
scripts/legacy/testCallbacks.js
Normal file
@@ -0,0 +1,85 @@
|
||||
// Скрипт для проверки работы callback-хэндлеров и уведомлений
|
||||
require('dotenv').config();
|
||||
const TelegramBot = require('node-telegram-bot-api');
|
||||
const { Pool } = require('pg');
|
||||
|
||||
// Создаем пул соединений
|
||||
const pool = new Pool({
|
||||
user: process.env.DB_USERNAME,
|
||||
host: process.env.DB_HOST,
|
||||
database: process.env.DB_NAME,
|
||||
password: process.env.DB_PASSWORD,
|
||||
port: parseInt(process.env.DB_PORT || '5432')
|
||||
});
|
||||
|
||||
// Функция для имитации callback-запроса к боту
|
||||
async function testCallback() {
|
||||
try {
|
||||
console.log('Начинаем тестирование callback-хэндлеров и уведомлений...');
|
||||
|
||||
// Используем последнего пользователя из базы данных
|
||||
const userResult = await pool.query(`
|
||||
SELECT * FROM users ORDER BY last_active_at DESC NULLS LAST LIMIT 1
|
||||
`);
|
||||
|
||||
if (userResult.rows.length === 0) {
|
||||
console.error('❌ Пользователи не найдены в базе данных');
|
||||
return;
|
||||
}
|
||||
|
||||
const user = userResult.rows[0];
|
||||
console.log(`Выбран тестовый пользователь: ${user.first_name || 'Без имени'} (ID: ${user.telegram_id})`);
|
||||
|
||||
// Получаем токен бота из переменных окружения
|
||||
const token = process.env.TELEGRAM_BOT_TOKEN;
|
||||
if (!token) {
|
||||
console.error('❌ Токен бота не найден в переменных окружения');
|
||||
return;
|
||||
}
|
||||
|
||||
// Создаем экземпляр бота
|
||||
const bot = new TelegramBot(token);
|
||||
|
||||
// Отправляем тестовое уведомление пользователю
|
||||
console.log(`Отправляем тестовое уведомление пользователю ID: ${user.telegram_id}...`);
|
||||
|
||||
try {
|
||||
const result = await bot.sendMessage(
|
||||
user.telegram_id,
|
||||
`🔔 *Тестовое уведомление*\n\nЭто проверка работы уведомлений и callback-хэндлеров.\n\nВаш премиум-статус: ${user.premium ? '✅ Активен' : '❌ Не активен'}`,
|
||||
{
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '🔔 Уведомления', callback_data: 'notification_settings' },
|
||||
{ text: '❤️ Профиль', callback_data: 'view_profile' }
|
||||
],
|
||||
[
|
||||
{ text: '⚙️ Настройки', callback_data: 'settings' }
|
||||
]
|
||||
]
|
||||
}
|
||||
}
|
||||
);
|
||||
|
||||
console.log('✅ Тестовое сообщение успешно отправлено!');
|
||||
console.log('Информация о сообщении:', JSON.stringify(result, null, 2));
|
||||
} catch (error) {
|
||||
console.error('❌ Ошибка при отправке тестового сообщения:', error.message);
|
||||
if (error.response && error.response.body) {
|
||||
console.error('Детали ошибки:', JSON.stringify(error.response.body, null, 2));
|
||||
}
|
||||
}
|
||||
|
||||
} catch (error) {
|
||||
console.error('❌ Ошибка при тестировании:', error);
|
||||
} finally {
|
||||
await pool.end();
|
||||
console.log('Соединение с базой данных закрыто');
|
||||
console.log('Тестирование завершено!');
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем тестирование
|
||||
testCallback();
|
||||
102
scripts/legacy/testMatching.js
Normal file
102
scripts/legacy/testMatching.js
Normal file
@@ -0,0 +1,102 @@
|
||||
require('dotenv').config();
|
||||
const { MatchingService } = require('../dist/services/matchingService');
|
||||
const { ProfileService } = require('../dist/services/profileService');
|
||||
|
||||
// Функция для создания тестовых пользователей
|
||||
async function createTestUsers() {
|
||||
const profileService = new ProfileService();
|
||||
|
||||
console.log('Создание тестовых пользователей...');
|
||||
|
||||
// Создаем мужской профиль
|
||||
const maleUserId = await profileService.ensureUser('123456', {
|
||||
username: 'test_male',
|
||||
first_name: 'Иван',
|
||||
last_name: 'Тестов'
|
||||
});
|
||||
|
||||
await profileService.createProfile(maleUserId, {
|
||||
name: 'Иван',
|
||||
age: 30,
|
||||
gender: 'male',
|
||||
interestedIn: 'female',
|
||||
bio: 'Тестовый мужской профиль',
|
||||
photos: ['photo1.jpg'],
|
||||
city: 'Москва',
|
||||
searchPreferences: {
|
||||
minAge: 18,
|
||||
maxAge: 45,
|
||||
maxDistance: 50
|
||||
}
|
||||
});
|
||||
console.log(`Создан мужской профиль: userId=${maleUserId}, telegramId=123456`);
|
||||
|
||||
// Создаем женский профиль
|
||||
const femaleUserId = await profileService.ensureUser('654321', {
|
||||
username: 'test_female',
|
||||
first_name: 'Анна',
|
||||
last_name: 'Тестова'
|
||||
});
|
||||
|
||||
await profileService.createProfile(femaleUserId, {
|
||||
name: 'Анна',
|
||||
age: 28,
|
||||
gender: 'female',
|
||||
interestedIn: 'male',
|
||||
bio: 'Тестовый женский профиль',
|
||||
photos: ['photo2.jpg'],
|
||||
city: 'Москва',
|
||||
searchPreferences: {
|
||||
minAge: 25,
|
||||
maxAge: 40,
|
||||
maxDistance: 30
|
||||
}
|
||||
});
|
||||
console.log(`Создан женский профиль: userId=${femaleUserId}, telegramId=654321`);
|
||||
|
||||
console.log('Тестовые пользователи созданы успешно');
|
||||
}
|
||||
|
||||
// Функция для тестирования подбора анкет
|
||||
async function testMatching() {
|
||||
console.log('\n===== ТЕСТИРОВАНИЕ ПОДБОРА АНКЕТ =====');
|
||||
|
||||
const matchingService = new MatchingService();
|
||||
|
||||
console.log('\nТест 1: Получение анкеты для мужского профиля (должна вернуться женская анкета)');
|
||||
const femaleProfile = await matchingService.getNextCandidate('123456', true);
|
||||
if (femaleProfile) {
|
||||
console.log(`✓ Получена анкета: ${femaleProfile.name}, возраст: ${femaleProfile.age}, пол: ${femaleProfile.gender}`);
|
||||
} else {
|
||||
console.log('✗ Анкета не найдена');
|
||||
}
|
||||
|
||||
console.log('\nТест 2: Получение анкеты для женского профиля (должна вернуться мужская анкета)');
|
||||
const maleProfile = await matchingService.getNextCandidate('654321', true);
|
||||
if (maleProfile) {
|
||||
console.log(`✓ Получена анкета: ${maleProfile.name}, возраст: ${maleProfile.age}, пол: ${maleProfile.gender}`);
|
||||
} else {
|
||||
console.log('✗ Анкета не найдена');
|
||||
}
|
||||
|
||||
console.log('\n===== ТЕСТИРОВАНИЕ ЗАВЕРШЕНО =====\n');
|
||||
|
||||
// Завершение работы скрипта
|
||||
process.exit(0);
|
||||
}
|
||||
|
||||
// Главная функция
|
||||
async function main() {
|
||||
try {
|
||||
// Создаем тестовых пользователей
|
||||
await createTestUsers();
|
||||
|
||||
// Тестируем подбор анкет
|
||||
await testMatching();
|
||||
} catch (error) {
|
||||
console.error('Ошибка при выполнении тестов:', error);
|
||||
process.exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
main();
|
||||
98
scripts/legacy/testProfileViews.js
Normal file
98
scripts/legacy/testProfileViews.js
Normal file
@@ -0,0 +1,98 @@
|
||||
// Тестирование работы с таблицей profile_views
|
||||
const { Pool } = require('pg');
|
||||
require('dotenv').config();
|
||||
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST,
|
||||
port: parseInt(process.env.DB_PORT || '5432'),
|
||||
database: process.env.DB_NAME,
|
||||
user: process.env.DB_USERNAME,
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
// Функция для тестирования записи просмотра профиля
|
||||
async function testRecordProfileView(viewerId, viewedProfileId) {
|
||||
const client = await pool.connect();
|
||||
try {
|
||||
console.log(`Запись просмотра профиля: ${viewerId} просмотрел ${viewedProfileId}`);
|
||||
|
||||
// Получаем UUID пользователей
|
||||
const viewerResult = await client.query('SELECT id FROM users WHERE telegram_id = $1', [viewerId]);
|
||||
if (viewerResult.rows.length === 0) {
|
||||
console.log(`Пользователь с telegram_id ${viewerId} не найден, создаём нового пользователя`);
|
||||
const newUserResult = await client.query(`
|
||||
INSERT INTO users (telegram_id, username, first_name, last_name)
|
||||
VALUES ($1, $2, $3, $4) RETURNING id
|
||||
`, [viewerId, `user_${viewerId}`, `Имя ${viewerId}`, `Фамилия ${viewerId}`]);
|
||||
|
||||
var viewerUuid = newUserResult.rows[0].id;
|
||||
} else {
|
||||
var viewerUuid = viewerResult.rows[0].id;
|
||||
}
|
||||
|
||||
const viewedResult = await client.query('SELECT id FROM users WHERE telegram_id = $1', [viewedProfileId]);
|
||||
if (viewedResult.rows.length === 0) {
|
||||
console.log(`Пользователь с telegram_id ${viewedProfileId} не найден, создаём нового пользователя`);
|
||||
const newUserResult = await client.query(`
|
||||
INSERT INTO users (telegram_id, username, first_name, last_name)
|
||||
VALUES ($1, $2, $3, $4) RETURNING id
|
||||
`, [viewedProfileId, `user_${viewedProfileId}`, `Имя ${viewedProfileId}`, `Фамилия ${viewedProfileId}`]);
|
||||
|
||||
var viewedUuid = newUserResult.rows[0].id;
|
||||
} else {
|
||||
var viewedUuid = viewedResult.rows[0].id;
|
||||
}
|
||||
|
||||
console.log(`UUID просматривающего: ${viewerUuid}`);
|
||||
console.log(`UUID просматриваемого: ${viewedUuid}`);
|
||||
|
||||
// Записываем просмотр
|
||||
await client.query(`
|
||||
INSERT INTO profile_views (viewer_id, viewed_profile_id, view_type, view_date)
|
||||
VALUES ($1, $2, $3, NOW())
|
||||
ON CONFLICT (viewer_id, viewed_profile_id) DO UPDATE
|
||||
SET view_date = NOW(), view_type = $3
|
||||
`, [viewerUuid, viewedUuid, 'browse']);
|
||||
|
||||
console.log('Просмотр профиля успешно записан');
|
||||
|
||||
// Получаем список просмотренных профилей
|
||||
const viewedProfiles = await client.query(`
|
||||
SELECT v.viewed_profile_id, v.view_date, u.telegram_id
|
||||
FROM profile_views v
|
||||
JOIN users u ON u.id = v.viewed_profile_id
|
||||
WHERE v.viewer_id = $1
|
||||
ORDER BY v.view_date DESC
|
||||
`, [viewerUuid]);
|
||||
|
||||
console.log('Список просмотренных профилей:');
|
||||
viewedProfiles.rows.forEach((row, index) => {
|
||||
console.log(`${index + 1}. ID: ${row.telegram_id}, просмотрен: ${row.view_date}`);
|
||||
});
|
||||
|
||||
return true;
|
||||
} catch (error) {
|
||||
console.error('Ошибка записи просмотра профиля:', error);
|
||||
return false;
|
||||
} finally {
|
||||
client.release();
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем тест
|
||||
async function runTest() {
|
||||
try {
|
||||
// Тестируем запись просмотра профиля
|
||||
await testRecordProfileView(123456, 789012);
|
||||
await testRecordProfileView(123456, 345678);
|
||||
await testRecordProfileView(789012, 123456);
|
||||
|
||||
console.log('Тесты завершены успешно');
|
||||
} catch (error) {
|
||||
console.error('Ошибка при выполнении тестов:', error);
|
||||
} finally {
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
runTest();
|
||||
81
scripts/legacy/testVipMethod.js
Normal file
81
scripts/legacy/testVipMethod.js
Normal file
@@ -0,0 +1,81 @@
|
||||
// Скрипт для тестирования метода checkPremiumStatus
|
||||
require('dotenv').config();
|
||||
const { Pool } = require('pg');
|
||||
|
||||
// Создаем пул соединений
|
||||
const pool = new Pool({
|
||||
user: process.env.DB_USERNAME,
|
||||
host: process.env.DB_HOST,
|
||||
database: process.env.DB_NAME,
|
||||
password: process.env.DB_PASSWORD,
|
||||
port: parseInt(process.env.DB_PORT || '5432')
|
||||
});
|
||||
|
||||
async function testCheckPremiumMethod() {
|
||||
try {
|
||||
console.log('Тестирование метода checkPremiumStatus...');
|
||||
|
||||
// Получаем пользователя для тестирования
|
||||
const userResult = await pool.query(`
|
||||
SELECT id, telegram_id, first_name, username, premium
|
||||
FROM users
|
||||
ORDER BY last_active_at DESC NULLS LAST
|
||||
LIMIT 1
|
||||
`);
|
||||
|
||||
if (userResult.rows.length === 0) {
|
||||
console.error('❌ Пользователи не найдены в базе данных');
|
||||
return;
|
||||
}
|
||||
|
||||
const user = userResult.rows[0];
|
||||
console.log(`Выбран тестовый пользователь: ${user.first_name || user.username || 'Без имени'} (Telegram ID: ${user.telegram_id})`);
|
||||
console.log(`Текущий премиум-статус: ${user.premium ? '✅ Активен' : '❌ Не активен'}`);
|
||||
|
||||
// Проверка работы метода checkPremiumStatus
|
||||
console.log('\nЭмулируем вызов метода checkPremiumStatus из vipService:');
|
||||
const result = await pool.query(`
|
||||
SELECT id, premium
|
||||
FROM users
|
||||
WHERE telegram_id = $1
|
||||
`, [user.telegram_id]);
|
||||
|
||||
if (result.rows.length === 0) {
|
||||
console.log('❌ Пользователь не найден');
|
||||
} else {
|
||||
const isPremium = result.rows[0].premium || false;
|
||||
console.log(`Результат метода: isPremium = ${isPremium ? '✅ true' : '❌ false'}`);
|
||||
|
||||
if (!isPremium) {
|
||||
console.log('\nПремиум-статус отсутствует. Устанавливаем премиум...');
|
||||
await pool.query(`
|
||||
UPDATE users
|
||||
SET premium = true
|
||||
WHERE telegram_id = $1
|
||||
`, [user.telegram_id]);
|
||||
|
||||
// Проверяем обновление
|
||||
const updatedResult = await pool.query(`
|
||||
SELECT premium
|
||||
FROM users
|
||||
WHERE telegram_id = $1
|
||||
`, [user.telegram_id]);
|
||||
|
||||
const updatedPremium = updatedResult.rows[0].premium;
|
||||
console.log(`Обновленный статус: isPremium = ${updatedPremium ? '✅ true' : '❌ false'}`);
|
||||
}
|
||||
}
|
||||
|
||||
console.log('\n✅ Тестирование завершено');
|
||||
console.log('🔧 Теперь проверьте функциональность VIP поиска в боте');
|
||||
|
||||
} catch (error) {
|
||||
console.error('❌ Ошибка при тестировании:', error);
|
||||
} finally {
|
||||
await pool.end();
|
||||
console.log('Соединение с базой данных закрыто');
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем тест
|
||||
testCheckPremiumMethod();
|
||||
75
scripts/legacy/testVipStatus.js
Normal file
75
scripts/legacy/testVipStatus.js
Normal file
@@ -0,0 +1,75 @@
|
||||
// Скрипт для тестирования VIP функционала
|
||||
require('dotenv').config();
|
||||
const { Pool } = require('pg');
|
||||
|
||||
// Создаем пул соединений
|
||||
const pool = new Pool({
|
||||
user: process.env.DB_USERNAME,
|
||||
host: process.env.DB_HOST,
|
||||
database: process.env.DB_NAME,
|
||||
password: process.env.DB_PASSWORD,
|
||||
port: parseInt(process.env.DB_PORT || '5432')
|
||||
});
|
||||
|
||||
async function testVipStatus() {
|
||||
try {
|
||||
console.log('Тестирование функционала VIP статуса...');
|
||||
|
||||
// Получаем список пользователей с информацией о premium статусе
|
||||
const users = await pool.query(`
|
||||
SELECT id, telegram_id, username, first_name, premium
|
||||
FROM users
|
||||
ORDER BY last_active_at DESC
|
||||
LIMIT 5
|
||||
`);
|
||||
|
||||
console.log('Список пользователей и их премиум статус:');
|
||||
users.rows.forEach(user => {
|
||||
console.log(`ID: ${user.id.substr(0, 8)}... | Telegram ID: ${user.telegram_id} | Имя: ${user.first_name || user.username || 'Не указано'} | Premium: ${user.premium ? '✅' : '❌'}`);
|
||||
});
|
||||
|
||||
// Если premium у всех false, устанавливаем premium = true
|
||||
const nonPremiumUsers = users.rows.filter(user => !user.premium);
|
||||
if (nonPremiumUsers.length > 0) {
|
||||
console.log('\nОбнаружены пользователи без премиум статуса. Устанавливаем премиум...');
|
||||
|
||||
for (const user of nonPremiumUsers) {
|
||||
await pool.query(`
|
||||
UPDATE users
|
||||
SET premium = true
|
||||
WHERE id = $1
|
||||
RETURNING id, telegram_id, premium
|
||||
`, [user.id]);
|
||||
|
||||
console.log(`✅ Установлен премиум для пользователя ${user.first_name || user.username || user.telegram_id}`);
|
||||
}
|
||||
} else {
|
||||
console.log('\nВсе пользователи уже имеют премиум-статус!');
|
||||
}
|
||||
|
||||
// Проверяем результат
|
||||
const updatedUsers = await pool.query(`
|
||||
SELECT id, telegram_id, username, first_name, premium
|
||||
FROM users
|
||||
ORDER BY last_active_at DESC
|
||||
LIMIT 5
|
||||
`);
|
||||
|
||||
console.log('\nОбновленный список пользователей и их премиум статус:');
|
||||
updatedUsers.rows.forEach(user => {
|
||||
console.log(`ID: ${user.id.substr(0, 8)}... | Telegram ID: ${user.telegram_id} | Имя: ${user.first_name || user.username || 'Не указано'} | Premium: ${user.premium ? '✅' : '❌'}`);
|
||||
});
|
||||
|
||||
console.log('\n✅ Тестирование VIP функционала завершено');
|
||||
console.log('🔧 Проверьте доступность VIP поиска в боте через меню или команды');
|
||||
|
||||
} catch (error) {
|
||||
console.error('❌ Ошибка при тестировании VIP статуса:', error);
|
||||
} finally {
|
||||
await pool.end();
|
||||
console.log('Соединение с базой данных закрыто');
|
||||
}
|
||||
}
|
||||
|
||||
// Запускаем тест
|
||||
testVipStatus();
|
||||
@@ -2,17 +2,18 @@
|
||||
// Этот скрипт создает записи о миграциях в таблице pgmigrations без применения изменений
|
||||
// Используется для синхронизации существующей базы с миграциями
|
||||
|
||||
require('dotenv').config();
|
||||
const { Client } = require('pg');
|
||||
const fs = require('fs');
|
||||
const path = require('path');
|
||||
|
||||
// Подключение к базе данных
|
||||
// Подключение к базе данных из переменных окружения
|
||||
const client = new Client({
|
||||
host: '192.168.0.102',
|
||||
port: 5432,
|
||||
database: 'telegram_tinder_bot',
|
||||
user: 'trevor',
|
||||
password: 'Cl0ud_1985!'
|
||||
host: process.env.DB_HOST || 'localhost',
|
||||
port: parseInt(process.env.DB_PORT) || 5432,
|
||||
database: process.env.DB_NAME || 'telegram_tinder_bot',
|
||||
user: process.env.DB_USERNAME || 'postgres',
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
|
||||
async function syncMigrations() {
|
||||
|
||||
83
scripts/startup.sh
Normal file
83
scripts/startup.sh
Normal file
@@ -0,0 +1,83 @@
|
||||
#!/bin/sh
|
||||
# startup.sh - Script to run migrations and start the bot
|
||||
|
||||
echo "🚀 Starting Telegram Tinder Bot..."
|
||||
|
||||
# Check for locales directory
|
||||
if [ ! -d "dist/locales" ]; then
|
||||
echo "⚠️ Locales directory not found in dist/locales"
|
||||
echo "🔍 Checking source directory structure..."
|
||||
ls -la dist/ || echo "Error listing dist directory"
|
||||
|
||||
# If src/locales exists, copy it to dist/locales
|
||||
if [ -d "src/locales" ]; then
|
||||
echo "📂 Found src/locales directory. Copying to dist/locales..."
|
||||
mkdir -p dist/locales
|
||||
cp -R src/locales/* dist/locales/
|
||||
echo "✅ Locales copied successfully"
|
||||
else
|
||||
echo "❌ src/locales directory not found either. Creating empty locales directory..."
|
||||
mkdir -p dist/locales
|
||||
fi
|
||||
fi
|
||||
|
||||
# Wait for database to be ready
|
||||
echo "⏳ Waiting for database to be ready..."
|
||||
sleep 5
|
||||
|
||||
# Run database migrations
|
||||
echo "🔄 Running database migrations..."
|
||||
|
||||
# Create migrations directory structure
|
||||
mkdir -p dist/database/migrations
|
||||
|
||||
# Copy any available migrations
|
||||
if [ -d "src/database/migrations" ]; then
|
||||
echo "<22> Found SQL migrations. Copying..."
|
||||
cp -R src/database/migrations/* dist/database/migrations/ 2>/dev/null || echo "No SQL migrations to copy"
|
||||
fi
|
||||
|
||||
# Copy JS migrations if available
|
||||
if [ -d "migrations" ]; then
|
||||
echo "📂 Found JS migrations. Copying..."
|
||||
mkdir -p migrations-temp
|
||||
cp migrations/*.js migrations-temp/ 2>/dev/null || echo "No JS migrations to copy"
|
||||
# Move JS migrations to dist/database/migrations
|
||||
cp migrations-temp/*.js dist/database/migrations/ 2>/dev/null || echo "No JS migrations to copy to dist"
|
||||
fi
|
||||
|
||||
# Display environment variables for debugging (without passwords)
|
||||
echo "🔍 Environment variables for database connection:"
|
||||
echo "DB_HOST: $DB_HOST"
|
||||
echo "DB_PORT: $DB_PORT"
|
||||
echo "DB_NAME: $DB_NAME"
|
||||
echo "DB_USERNAME: $DB_USERNAME"
|
||||
|
||||
# Run migrations using node-pg-migrate
|
||||
echo "🔄 Running migrations with node-pg-migrate..."
|
||||
DATABASE_URL="postgres://$DB_USERNAME:$DB_PASSWORD@$DB_HOST:$DB_PORT/$DB_NAME" npx node-pg-migrate up
|
||||
|
||||
# Verify connection to database
|
||||
echo "🔍 Verifying database connection..."
|
||||
node -e "
|
||||
const { Pool } = require('pg');
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST,
|
||||
port: process.env.DB_PORT,
|
||||
database: process.env.DB_NAME,
|
||||
user: process.env.DB_USERNAME,
|
||||
password: process.env.DB_PASSWORD
|
||||
});
|
||||
pool.query('SELECT NOW()', (err, res) => {
|
||||
if (err) {
|
||||
console.error('❌ Database connection failed:', err.message);
|
||||
process.exit(1);
|
||||
} else {
|
||||
console.log('✅ Database connection successful:', res.rows[0].now);
|
||||
pool.end();
|
||||
}
|
||||
});" || echo "❌ Failed to verify database connection"
|
||||
|
||||
# Start the bot
|
||||
echo "✅ Starting the bot..."
|
||||
node dist/bot.js
|
||||
104
scripts/update_bot_with_notifications.js
Normal file
104
scripts/update_bot_with_notifications.js
Normal file
@@ -0,0 +1,104 @@
|
||||
/**
|
||||
* Скрипт для проверки и исправления регистрации NotificationHandlers в bot.ts
|
||||
*/
|
||||
|
||||
const fs = require('fs');
|
||||
const path = require('path');
|
||||
|
||||
const botFilePath = path.join(__dirname, '../src/bot.ts');
|
||||
|
||||
// Проверка существования файла bot.ts
|
||||
if (!fs.existsSync(botFilePath)) {
|
||||
console.error(`❌ Файл ${botFilePath} не найден`);
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
// Чтение содержимого файла bot.ts
|
||||
let botContent = fs.readFileSync(botFilePath, 'utf8');
|
||||
|
||||
// Проверка импорта NotificationHandlers
|
||||
if (!botContent.includes('import { NotificationHandlers }')) {
|
||||
console.log('Добавляем импорт NotificationHandlers в bot.ts...');
|
||||
|
||||
// Находим последний импорт
|
||||
const importRegex = /^import.*?;/gms;
|
||||
const matches = [...botContent.matchAll(importRegex)];
|
||||
|
||||
if (matches.length > 0) {
|
||||
const lastImport = matches[matches.length - 1][0];
|
||||
const lastImportIndex = botContent.lastIndexOf(lastImport) + lastImport.length;
|
||||
|
||||
// Добавляем импорт NotificationHandlers
|
||||
botContent =
|
||||
botContent.slice(0, lastImportIndex) +
|
||||
'\nimport { NotificationHandlers } from \'./handlers/notificationHandlers\';\n' +
|
||||
botContent.slice(lastImportIndex);
|
||||
|
||||
console.log('✅ Импорт NotificationHandlers добавлен');
|
||||
} else {
|
||||
console.error('❌ Не удалось найти место для добавления импорта');
|
||||
}
|
||||
}
|
||||
|
||||
// Проверка объявления NotificationHandlers в классе
|
||||
if (!botContent.includes('private notificationHandlers')) {
|
||||
console.log('Добавляем объявление notificationHandlers в класс...');
|
||||
|
||||
const classPropertiesRegex = /class TelegramTinderBot {([^}]+?)constructor/s;
|
||||
const classPropertiesMatch = botContent.match(classPropertiesRegex);
|
||||
|
||||
if (classPropertiesMatch) {
|
||||
const classProperties = classPropertiesMatch[1];
|
||||
const updatedProperties = classProperties + ' private notificationHandlers: NotificationHandlers;\n ';
|
||||
|
||||
botContent = botContent.replace(classPropertiesRegex, `class TelegramTinderBot {${updatedProperties}constructor`);
|
||||
|
||||
console.log('✅ Объявление notificationHandlers добавлено');
|
||||
} else {
|
||||
console.error('❌ Не удалось найти место для добавления объявления notificationHandlers');
|
||||
}
|
||||
}
|
||||
|
||||
// Проверка создания экземпляра NotificationHandlers в конструкторе
|
||||
if (!botContent.includes('this.notificationHandlers = new NotificationHandlers')) {
|
||||
console.log('Добавляем инициализацию notificationHandlers в конструктор...');
|
||||
|
||||
const initializationRegex = /(this\.callbackHandlers = new CallbackHandlers[^;]+;)/;
|
||||
const initializationMatch = botContent.match(initializationRegex);
|
||||
|
||||
if (initializationMatch) {
|
||||
const callbackHandlersInit = initializationMatch[1];
|
||||
const updatedInit = callbackHandlersInit + '\n this.notificationHandlers = new NotificationHandlers(this.bot);';
|
||||
|
||||
botContent = botContent.replace(initializationRegex, updatedInit);
|
||||
|
||||
console.log('✅ Инициализация notificationHandlers добавлена');
|
||||
} else {
|
||||
console.error('❌ Не удалось найти место для добавления инициализации notificationHandlers');
|
||||
}
|
||||
}
|
||||
|
||||
// Проверка регистрации notificationHandlers в методе registerHandlers
|
||||
if (!botContent.includes('this.notificationHandlers.register()')) {
|
||||
console.log('Добавляем регистрацию notificationHandlers...');
|
||||
|
||||
const registerHandlersRegex = /(private registerHandlers\(\): void {[^}]+?)}/s;
|
||||
const registerHandlersMatch = botContent.match(registerHandlersRegex);
|
||||
|
||||
if (registerHandlersMatch) {
|
||||
const registerHandlersBody = registerHandlersMatch[1];
|
||||
const updatedBody = registerHandlersBody + '\n // Обработчики уведомлений\n this.notificationHandlers.register();\n }';
|
||||
|
||||
botContent = botContent.replace(registerHandlersRegex, updatedBody);
|
||||
|
||||
console.log('✅ Регистрация notificationHandlers добавлена');
|
||||
} else {
|
||||
console.error('❌ Не удалось найти место для добавления регистрации notificationHandlers');
|
||||
}
|
||||
}
|
||||
|
||||
// Запись обновленного содержимого в файл
|
||||
fs.writeFileSync(botFilePath, botContent, 'utf8');
|
||||
|
||||
console.log('✅ Файл bot.ts успешно обновлен');
|
||||
console.log('🔔 Перезапустите бота для применения изменений');
|
||||
BIN
set-premium.js
BIN
set-premium.js
Binary file not shown.
36
sql/add_job_and_state_columns.sql
Normal file
36
sql/add_job_and_state_columns.sql
Normal file
@@ -0,0 +1,36 @@
|
||||
-- Добавление колонок job и state
|
||||
-- Дата: 2025-11-06
|
||||
-- Исправляет ошибки: "column job does not exist" и "State column does not exist"
|
||||
|
||||
-- 1. Добавляем колонку job в таблицу profiles (синоним для occupation)
|
||||
ALTER TABLE profiles ADD COLUMN IF NOT EXISTS job VARCHAR(255);
|
||||
|
||||
-- 2. Копируем существующие данные из occupation в job
|
||||
UPDATE profiles SET job = occupation WHERE occupation IS NOT NULL AND job IS NULL;
|
||||
|
||||
-- 3. Добавляем колонку state в таблицу users для отслеживания состояния диалога
|
||||
ALTER TABLE users ADD COLUMN IF NOT EXISTS state VARCHAR(50);
|
||||
|
||||
-- 4. Создаём индексы для производительности
|
||||
CREATE INDEX IF NOT EXISTS idx_profiles_job ON profiles(job);
|
||||
CREATE INDEX IF NOT EXISTS idx_users_state ON users(state);
|
||||
|
||||
-- 5. Добавляем комментарии для документации
|
||||
COMMENT ON COLUMN profiles.job IS 'Профессия/работа пользователя (синоним для occupation)';
|
||||
COMMENT ON COLUMN profiles.occupation IS 'Профессия/работа пользователя (устаревшее, используйте job)';
|
||||
COMMENT ON COLUMN users.state IS 'Текущее состояние пользователя в диалоге с ботом';
|
||||
|
||||
-- Проверка результата
|
||||
SELECT
|
||||
'profiles.job' as column_name,
|
||||
CASE WHEN EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'profiles' AND column_name = 'job'
|
||||
) THEN '✅ Существует' ELSE '❌ Не найдена' END as status
|
||||
UNION ALL
|
||||
SELECT
|
||||
'users.state' as column_name,
|
||||
CASE WHEN EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'users' AND column_name = 'state'
|
||||
) THEN '✅ Существует' ELSE '❌ Не найдена' END as status;
|
||||
17
sql/add_location_coordinates.sql
Normal file
17
sql/add_location_coordinates.sql
Normal file
@@ -0,0 +1,17 @@
|
||||
-- Миграция: Добавление колонок для хранения координат местоположения
|
||||
-- Дата: 2025-01-20
|
||||
-- Описание: Добавляет location_lat и location_lon для хранения GPS-координат,
|
||||
-- полученных через Kakao Maps API, для расчета расстояния между пользователями
|
||||
|
||||
-- Добавляем колонки для широты и долготы
|
||||
ALTER TABLE profiles ADD COLUMN IF NOT EXISTS location_lat DECIMAL(10, 8);
|
||||
ALTER TABLE profiles ADD COLUMN IF NOT EXISTS location_lon DECIMAL(11, 8);
|
||||
|
||||
-- Создаем индекс для быстрого поиска по координатам
|
||||
CREATE INDEX IF NOT EXISTS idx_profiles_location
|
||||
ON profiles(location_lat, location_lon)
|
||||
WHERE location_lat IS NOT NULL AND location_lon IS NOT NULL;
|
||||
|
||||
-- Комментарии для документации
|
||||
COMMENT ON COLUMN profiles.location_lat IS 'Широта местоположения пользователя (из Kakao Maps)';
|
||||
COMMENT ON COLUMN profiles.location_lon IS 'Долгота местоположения пользователя (из Kakao Maps)';
|
||||
24
sql/add_user_language.sql
Normal file
24
sql/add_user_language.sql
Normal file
@@ -0,0 +1,24 @@
|
||||
-- Добавление колонки lang в таблицу users
|
||||
-- Эта миграция добавляет поддержку мультиязычности
|
||||
|
||||
-- Добавляем колонку lang с дефолтным значением 'ru'
|
||||
ALTER TABLE users
|
||||
ADD COLUMN IF NOT EXISTS lang VARCHAR(5) DEFAULT 'ru' NOT NULL;
|
||||
|
||||
-- Создаем индекс для быстрого поиска по языку
|
||||
CREATE INDEX IF NOT EXISTS idx_users_lang ON users(lang);
|
||||
|
||||
-- Обновляем всех существующих пользователей, устанавливая русский язык
|
||||
UPDATE users SET lang = 'ru' WHERE lang IS NULL OR lang = '';
|
||||
|
||||
-- Добавляем комментарий к колонке
|
||||
COMMENT ON COLUMN users.lang IS 'User interface language (ISO 639-1 code)';
|
||||
|
||||
-- Проверка результата
|
||||
SELECT
|
||||
COUNT(*) as total_users,
|
||||
lang,
|
||||
COUNT(*) * 100.0 / SUM(COUNT(*)) OVER() as percentage
|
||||
FROM users
|
||||
GROUP BY lang
|
||||
ORDER BY total_users DESC;
|
||||
42
sql/clear_interactions.sql
Normal file
42
sql/clear_interactions.sql
Normal file
@@ -0,0 +1,42 @@
|
||||
-- Скрипт для очистки всех взаимодействий между пользователями
|
||||
-- Удаляет матчи, сообщения, свайпы и показы анкет
|
||||
-- Оставляет только пользователей и их профили
|
||||
|
||||
-- Начало транзакции
|
||||
BEGIN;
|
||||
|
||||
-- Удаление всех сообщений в чатах
|
||||
DELETE FROM messages;
|
||||
|
||||
-- Удаление всех матчей
|
||||
DELETE FROM matches;
|
||||
|
||||
-- Удаление всех просмотров профилей
|
||||
DELETE FROM profile_views;
|
||||
|
||||
-- Удаление всех свайпов (лайки, дизлайки, суперлайки)
|
||||
DELETE FROM swipes;
|
||||
|
||||
-- Удаление всех уведомлений
|
||||
DELETE FROM notifications;
|
||||
|
||||
-- Фиксация транзакции
|
||||
COMMIT;
|
||||
|
||||
-- Вывод статистики после очистки
|
||||
SELECT
|
||||
'messages' as table_name,
|
||||
COUNT(*) as remaining_records
|
||||
FROM messages
|
||||
UNION ALL
|
||||
SELECT 'matches', COUNT(*) FROM matches
|
||||
UNION ALL
|
||||
SELECT 'profile_views', COUNT(*) FROM profile_views
|
||||
UNION ALL
|
||||
SELECT 'swipes', COUNT(*) FROM swipes
|
||||
UNION ALL
|
||||
SELECT 'notifications', COUNT(*) FROM notifications
|
||||
UNION ALL
|
||||
SELECT 'users', COUNT(*) FROM users
|
||||
UNION ALL
|
||||
SELECT 'profiles', COUNT(*) FROM profiles;
|
||||
404
sql/consolidated.sql
Normal file
404
sql/consolidated.sql
Normal file
@@ -0,0 +1,404 @@
|
||||
# Consolidated SQL файл для миграции базы данных Telegram Tinder Bot
|
||||
# Этот файл содержит все необходимые SQL-запросы для создания базы данных с нуля
|
||||
|
||||
-- Создание расширения для UUID
|
||||
CREATE EXTENSION IF NOT EXISTS "uuid-ossp";
|
||||
|
||||
-- Создание перечислений
|
||||
CREATE TYPE gender_type AS ENUM ('male', 'female', 'other');
|
||||
CREATE TYPE swipe_action AS ENUM ('like', 'dislike', 'superlike');
|
||||
|
||||
-- Создание таблицы пользователей
|
||||
CREATE TABLE IF NOT EXISTS users (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
telegram_id BIGINT UNIQUE NOT NULL,
|
||||
username VARCHAR(255),
|
||||
first_name VARCHAR(255) NOT NULL,
|
||||
last_name VARCHAR(255),
|
||||
language_code VARCHAR(10),
|
||||
is_premium BOOLEAN DEFAULT FALSE,
|
||||
is_active BOOLEAN DEFAULT TRUE,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
-- Создание таблицы профилей
|
||||
CREATE TABLE IF NOT EXISTS profiles (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
name VARCHAR(255) NOT NULL,
|
||||
age INTEGER NOT NULL CHECK (age >= 18),
|
||||
gender gender_type NOT NULL,
|
||||
bio TEXT,
|
||||
photos TEXT[], -- JSON array of photo file_ids
|
||||
location VARCHAR(255),
|
||||
job VARCHAR(255),
|
||||
interests TEXT[], -- JSON array of interests
|
||||
last_active TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
is_completed BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
CONSTRAINT unique_user_profile UNIQUE (user_id)
|
||||
);
|
||||
|
||||
-- Создание индекса для поиска по возрасту и полу
|
||||
CREATE INDEX idx_profiles_age_gender ON profiles(age, gender);
|
||||
|
||||
-- Создание таблицы предпочтений поиска
|
||||
CREATE TABLE IF NOT EXISTS search_preferences (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
age_min INTEGER NOT NULL DEFAULT 18 CHECK (age_min >= 18),
|
||||
age_max INTEGER NOT NULL DEFAULT 99 CHECK (age_max >= age_min),
|
||||
looking_for gender_type NOT NULL,
|
||||
distance_max INTEGER, -- max distance in km, NULL means no limit
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
CONSTRAINT unique_user_preferences UNIQUE (user_id)
|
||||
);
|
||||
|
||||
-- Создание таблицы действий (лайки/дизлайки)
|
||||
CREATE TABLE IF NOT EXISTS swipes (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
target_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
action swipe_action NOT NULL,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
CONSTRAINT unique_swipe UNIQUE (user_id, target_id)
|
||||
);
|
||||
|
||||
-- Создание индекса для быстрого поиска матчей
|
||||
CREATE INDEX idx_swipes_user_target ON swipes(user_id, target_id);
|
||||
|
||||
-- Создание таблицы матчей
|
||||
CREATE TABLE IF NOT EXISTS matches (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id_1 UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
user_id_2 UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
is_active BOOLEAN DEFAULT TRUE,
|
||||
CONSTRAINT unique_match UNIQUE (user_id_1, user_id_2)
|
||||
);
|
||||
|
||||
-- Создание индекса для быстрого поиска матчей по пользователю
|
||||
CREATE INDEX idx_matches_user_id_1 ON matches(user_id_1);
|
||||
CREATE INDEX idx_matches_user_id_2 ON matches(user_id_2);
|
||||
|
||||
-- Создание таблицы блокировок
|
||||
CREATE TABLE IF NOT EXISTS blocks (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
blocker_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
blocked_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
CONSTRAINT unique_block UNIQUE (blocker_id, blocked_id)
|
||||
);
|
||||
|
||||
-- Создание таблицы сообщений
|
||||
CREATE TABLE IF NOT EXISTS messages (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
match_id UUID NOT NULL REFERENCES matches(id) ON DELETE CASCADE,
|
||||
sender_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
text TEXT NOT NULL,
|
||||
is_read BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
-- Создание индекса для быстрого поиска сообщений
|
||||
CREATE INDEX idx_messages_match_id ON messages(match_id);
|
||||
|
||||
-- Создание таблицы уведомлений
|
||||
CREATE TABLE IF NOT EXISTS notifications (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL, -- new_match, new_message, etc.
|
||||
content TEXT NOT NULL,
|
||||
is_read BOOLEAN DEFAULT FALSE,
|
||||
reference_id UUID, -- Can reference a match_id, message_id, etc.
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
-- Создание индекса для быстрого поиска уведомлений
|
||||
CREATE INDEX idx_notifications_user_id ON notifications(user_id);
|
||||
|
||||
-- Создание таблицы настроек
|
||||
CREATE TABLE IF NOT EXISTS settings (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
notifications_enabled BOOLEAN DEFAULT TRUE,
|
||||
show_online_status BOOLEAN DEFAULT TRUE,
|
||||
visibility BOOLEAN DEFAULT TRUE, -- whether profile is visible in search
|
||||
theme VARCHAR(20) DEFAULT 'light',
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
CONSTRAINT unique_user_settings UNIQUE (user_id)
|
||||
);
|
||||
|
||||
-- Создание таблицы просмотров профиля
|
||||
CREATE TABLE IF NOT EXISTS profile_views (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
viewer_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
viewed_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
view_count INTEGER DEFAULT 1,
|
||||
last_viewed_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
CONSTRAINT unique_view UNIQUE (viewer_id, viewed_id)
|
||||
);
|
||||
|
||||
-- Создание таблицы для премиум-пользователей
|
||||
CREATE TABLE IF NOT EXISTS premium_features (
|
||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
is_premium BOOLEAN DEFAULT FALSE,
|
||||
superlike_quota INTEGER DEFAULT 1,
|
||||
spotlight_quota INTEGER DEFAULT 0,
|
||||
see_likes BOOLEAN DEFAULT FALSE, -- Can see who liked their profile
|
||||
unlimited_likes BOOLEAN DEFAULT FALSE,
|
||||
expires_at TIMESTAMP WITH TIME ZONE,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
updated_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||
CONSTRAINT unique_user_premium UNIQUE (user_id)
|
||||
);
|
||||
|
||||
-- Функция для обновления поля updated_at
|
||||
CREATE OR REPLACE FUNCTION update_updated_at()
|
||||
RETURNS TRIGGER AS $$
|
||||
BEGIN
|
||||
NEW.updated_at = CURRENT_TIMESTAMP;
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Триггеры для обновления поля updated_at
|
||||
CREATE TRIGGER update_users_updated_at
|
||||
BEFORE UPDATE ON users
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_updated_at();
|
||||
|
||||
CREATE TRIGGER update_profiles_updated_at
|
||||
BEFORE UPDATE ON profiles
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_updated_at();
|
||||
|
||||
CREATE TRIGGER update_search_preferences_updated_at
|
||||
BEFORE UPDATE ON search_preferences
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_updated_at();
|
||||
|
||||
CREATE TRIGGER update_settings_updated_at
|
||||
BEFORE UPDATE ON settings
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_updated_at();
|
||||
|
||||
CREATE TRIGGER update_premium_features_updated_at
|
||||
BEFORE UPDATE ON premium_features
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_updated_at();
|
||||
|
||||
-- Индекс для поиска пользователей по Telegram ID (часто используемый запрос)
|
||||
CREATE INDEX IF NOT EXISTS idx_users_telegram_id ON users(telegram_id);
|
||||
|
||||
-- Индекс для статуса профиля (активный/неактивный, завершенный/незавершенный)
|
||||
CREATE INDEX IF NOT EXISTS idx_profiles_is_completed ON profiles(is_completed);
|
||||
|
||||
-- Представление для статистики
|
||||
CREATE OR REPLACE VIEW user_statistics AS
|
||||
SELECT
|
||||
u.id,
|
||||
u.telegram_id,
|
||||
(SELECT COUNT(*) FROM swipes WHERE user_id = u.id AND action = 'like') AS likes_given,
|
||||
(SELECT COUNT(*) FROM swipes WHERE user_id = u.id AND action = 'dislike') AS dislikes_given,
|
||||
(SELECT COUNT(*) FROM swipes WHERE target_id = u.id AND action = 'like') AS likes_received,
|
||||
(SELECT COUNT(*) FROM matches WHERE user_id_1 = u.id OR user_id_2 = u.id) AS matches_count,
|
||||
(SELECT COUNT(*) FROM messages WHERE sender_id = u.id) AS messages_sent,
|
||||
(SELECT COUNT(*) FROM profile_views WHERE viewed_id = u.id) AS profile_views
|
||||
FROM users u;
|
||||
|
||||
-- Функция для создания матча при взаимных лайках
|
||||
CREATE OR REPLACE FUNCTION create_match_on_mutual_like()
|
||||
RETURNS TRIGGER AS $$
|
||||
DECLARE
|
||||
reverse_like_exists BOOLEAN;
|
||||
BEGIN
|
||||
-- Check if there is a reverse like
|
||||
SELECT EXISTS (
|
||||
SELECT 1
|
||||
FROM swipes
|
||||
WHERE user_id = NEW.target_id
|
||||
AND target_id = NEW.user_id
|
||||
AND action = 'like'
|
||||
) INTO reverse_like_exists;
|
||||
|
||||
-- If there is a reverse like, create a match
|
||||
IF reverse_like_exists AND NEW.action = 'like' THEN
|
||||
INSERT INTO matches (user_id_1, user_id_2)
|
||||
VALUES (
|
||||
LEAST(NEW.user_id, NEW.target_id),
|
||||
GREATEST(NEW.user_id, NEW.target_id)
|
||||
)
|
||||
ON CONFLICT (user_id_1, user_id_2) DO NOTHING;
|
||||
|
||||
-- Create notifications for both users
|
||||
INSERT INTO notifications (user_id, type, content, reference_id)
|
||||
VALUES (
|
||||
NEW.user_id,
|
||||
'new_match',
|
||||
'У вас новый матч!',
|
||||
(SELECT id FROM matches WHERE
|
||||
(user_id_1 = LEAST(NEW.user_id, NEW.target_id) AND user_id_2 = GREATEST(NEW.user_id, NEW.target_id))
|
||||
)
|
||||
);
|
||||
|
||||
INSERT INTO notifications (user_id, type, content, reference_id)
|
||||
VALUES (
|
||||
NEW.target_id,
|
||||
'new_match',
|
||||
'У вас новый матч!',
|
||||
(SELECT id FROM matches WHERE
|
||||
(user_id_1 = LEAST(NEW.user_id, NEW.target_id) AND user_id_2 = GREATEST(NEW.user_id, NEW.target_id))
|
||||
)
|
||||
);
|
||||
END IF;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Триггер для создания матча при взаимных лайках
|
||||
CREATE TRIGGER create_match_trigger
|
||||
AFTER INSERT ON swipes
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION create_match_on_mutual_like();
|
||||
|
||||
-- Функция для создания уведомления о новом сообщении
|
||||
CREATE OR REPLACE FUNCTION notify_new_message()
|
||||
RETURNS TRIGGER AS $$
|
||||
DECLARE
|
||||
recipient_id UUID;
|
||||
match_record RECORD;
|
||||
BEGIN
|
||||
-- Get the match record
|
||||
SELECT * INTO match_record FROM matches WHERE id = NEW.match_id;
|
||||
|
||||
-- Determine the recipient
|
||||
IF match_record.user_id_1 = NEW.sender_id THEN
|
||||
recipient_id := match_record.user_id_2;
|
||||
ELSE
|
||||
recipient_id := match_record.user_id_1;
|
||||
END IF;
|
||||
|
||||
-- Create notification
|
||||
INSERT INTO notifications (user_id, type, content, reference_id)
|
||||
VALUES (
|
||||
recipient_id,
|
||||
'new_message',
|
||||
'У вас новое сообщение!',
|
||||
NEW.id
|
||||
);
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Триггер для создания уведомления о новом сообщении
|
||||
CREATE TRIGGER notify_new_message_trigger
|
||||
AFTER INSERT ON messages
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION notify_new_message();
|
||||
|
||||
-- Функция для обновления времени последней активности пользователя
|
||||
CREATE OR REPLACE FUNCTION update_last_active()
|
||||
RETURNS TRIGGER AS $$
|
||||
BEGIN
|
||||
UPDATE profiles
|
||||
SET last_active = CURRENT_TIMESTAMP
|
||||
WHERE user_id = NEW.user_id;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Триггер для обновления времени последней активности при свайпах
|
||||
CREATE TRIGGER update_last_active_on_swipe
|
||||
AFTER INSERT ON swipes
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_last_active();
|
||||
|
||||
-- Триггер для обновления времени последней активности при отправке сообщений
|
||||
CREATE TRIGGER update_last_active_on_message
|
||||
AFTER INSERT ON messages
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_last_active();
|
||||
|
||||
-- Создание функции для автоматического создания профиля при создании пользователя
|
||||
CREATE OR REPLACE FUNCTION create_initial_profile()
|
||||
RETURNS TRIGGER AS $$
|
||||
BEGIN
|
||||
INSERT INTO profiles (user_id, name, age, gender)
|
||||
VALUES (NEW.id, COALESCE(NEW.first_name, 'User'), 18, 'other');
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Триггер для автоматического создания профиля при создании пользователя
|
||||
CREATE TRIGGER create_profile_trigger
|
||||
AFTER INSERT ON users
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION create_initial_profile();
|
||||
|
||||
-- Индексы для оптимизации частых запросов
|
||||
CREATE INDEX IF NOT EXISTS idx_profiles_last_active ON profiles(last_active);
|
||||
CREATE INDEX IF NOT EXISTS idx_swipes_action ON swipes(action);
|
||||
CREATE INDEX IF NOT EXISTS idx_notifications_is_read ON notifications(is_read);
|
||||
CREATE INDEX IF NOT EXISTS idx_messages_is_read ON messages(is_read);
|
||||
|
||||
-- Добавление ограничений для проверки возраста
|
||||
ALTER TABLE profiles DROP CONSTRAINT IF EXISTS age_check;
|
||||
ALTER TABLE profiles ADD CONSTRAINT age_check CHECK (age >= 18 AND age <= 99);
|
||||
|
||||
-- Добавление ограничений для предпочтений
|
||||
ALTER TABLE search_preferences DROP CONSTRAINT IF EXISTS age_range_check;
|
||||
ALTER TABLE search_preferences ADD CONSTRAINT age_range_check CHECK (age_min >= 18 AND age_max >= age_min AND age_max <= 99);
|
||||
|
||||
-- Комментарии к таблицам для документации
|
||||
COMMENT ON TABLE users IS 'Таблица пользователей Telegram';
|
||||
COMMENT ON TABLE profiles IS 'Профили пользователей для знакомств';
|
||||
COMMENT ON TABLE search_preferences IS 'Предпочтения поиска пользователей';
|
||||
COMMENT ON TABLE swipes IS 'История лайков/дислайков';
|
||||
COMMENT ON TABLE matches IS 'Совпадения (матчи) между пользователями';
|
||||
COMMENT ON TABLE messages IS 'Сообщения между пользователями';
|
||||
COMMENT ON TABLE notifications IS 'Уведомления для пользователей';
|
||||
COMMENT ON TABLE settings IS 'Настройки пользователей';
|
||||
COMMENT ON TABLE profile_views IS 'История просмотров профилей';
|
||||
COMMENT ON TABLE premium_features IS 'Премиум-функции для пользователей';
|
||||
|
||||
-- Представление для быстрого получения активных матчей с информацией о пользователе
|
||||
CREATE OR REPLACE VIEW active_matches AS
|
||||
SELECT
|
||||
m.id AS match_id,
|
||||
m.created_at AS match_date,
|
||||
CASE
|
||||
WHEN m.user_id_1 = u1.id THEN u2.id
|
||||
ELSE u1.id
|
||||
END AS partner_id,
|
||||
CASE
|
||||
WHEN m.user_id_1 = u1.id THEN u2.telegram_id
|
||||
ELSE u1.telegram_id
|
||||
END AS partner_telegram_id,
|
||||
CASE
|
||||
WHEN m.user_id_1 = u1.id THEN p2.name
|
||||
ELSE p1.name
|
||||
END AS partner_name,
|
||||
CASE
|
||||
WHEN m.user_id_1 = u1.id THEN p2.photos[1]
|
||||
ELSE p1.photos[1]
|
||||
END AS partner_photo,
|
||||
(SELECT COUNT(*) FROM messages WHERE match_id = m.id) AS message_count,
|
||||
(SELECT COUNT(*) FROM messages WHERE match_id = m.id AND is_read = false AND sender_id != u1.id) AS unread_count,
|
||||
m.is_active
|
||||
FROM matches m
|
||||
JOIN users u1 ON m.user_id_1 = u1.id
|
||||
JOIN users u2 ON m.user_id_2 = u2.id
|
||||
JOIN profiles p1 ON u1.id = p1.user_id
|
||||
JOIN profiles p2 ON u2.id = p2.user_id
|
||||
WHERE m.is_active = true;
|
||||
43
sql/fix_looking_for_column.sql
Normal file
43
sql/fix_looking_for_column.sql
Normal file
@@ -0,0 +1,43 @@
|
||||
-- Исправление триггера create_initial_profile и колонки looking_for
|
||||
-- Дата: 2025-11-06
|
||||
|
||||
-- 1. Удаляем старую функцию триггера
|
||||
DROP FUNCTION IF EXISTS create_initial_profile() CASCADE;
|
||||
|
||||
-- 2. Создаём исправленную функцию с полем looking_for
|
||||
CREATE OR REPLACE FUNCTION create_initial_profile()
|
||||
RETURNS TRIGGER AS $$
|
||||
BEGIN
|
||||
INSERT INTO profiles (user_id, name, age, gender, looking_for, interested_in)
|
||||
VALUES (NEW.id, COALESCE(NEW.first_name, 'User'), 18, 'other', 'both', 'both');
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- 3. Пересоздаём триггер
|
||||
DROP TRIGGER IF EXISTS create_profile_on_user_insert ON users;
|
||||
DROP TRIGGER IF EXISTS create_profile_trigger ON users;
|
||||
|
||||
CREATE TRIGGER create_profile_on_user_insert
|
||||
AFTER INSERT ON users
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION create_initial_profile();
|
||||
|
||||
-- 4. Делаем looking_for необязательным с дефолтным значением
|
||||
ALTER TABLE profiles ALTER COLUMN looking_for DROP NOT NULL;
|
||||
ALTER TABLE profiles ALTER COLUMN looking_for SET DEFAULT 'both';
|
||||
|
||||
-- 5. Добавляем interested_in как синоним для looking_for
|
||||
ALTER TABLE profiles ADD COLUMN IF NOT EXISTS interested_in VARCHAR(20) DEFAULT 'both'
|
||||
CHECK (interested_in IN ('male', 'female', 'both'));
|
||||
|
||||
-- 6. Обновляем существующие записи
|
||||
UPDATE profiles SET looking_for = 'both' WHERE looking_for IS NULL;
|
||||
UPDATE profiles SET interested_in = COALESCE(looking_for, 'both') WHERE interested_in IS NULL;
|
||||
|
||||
-- 7. Создаём индекс для поиска по interested_in
|
||||
CREATE INDEX IF NOT EXISTS idx_profiles_interested_in ON profiles(interested_in);
|
||||
|
||||
COMMENT ON COLUMN profiles.looking_for IS 'Предпочитаемый пол для знакомства (устаревшее, используйте interested_in)';
|
||||
COMMENT ON COLUMN profiles.interested_in IS 'Предпочитаемый пол для знакомства: male, female, both';
|
||||
82
sql/fix_match_trigger.sql
Normal file
82
sql/fix_match_trigger.sql
Normal file
@@ -0,0 +1,82 @@
|
||||
-- Исправление триггера create_match_on_mutual_like
|
||||
-- Дата: 2025-11-06
|
||||
-- Проблемы:
|
||||
-- 1. Использовались неправильные имена полей: target_id вместо target_user_id, action вместо type
|
||||
-- 2. Использовались несуществующие колонки в notifications: content и reference_id вместо data
|
||||
|
||||
-- Удаляем старую функцию
|
||||
DROP FUNCTION IF EXISTS create_match_on_mutual_like() CASCADE;
|
||||
|
||||
-- Создаем исправленную функцию
|
||||
CREATE OR REPLACE FUNCTION create_match_on_mutual_like()
|
||||
RETURNS TRIGGER AS $$
|
||||
DECLARE
|
||||
reverse_like_exists BOOLEAN;
|
||||
match_id_var UUID;
|
||||
BEGIN
|
||||
-- Проверяем только лайки и суперлайки (игнорируем pass)
|
||||
IF NEW.type != 'like' AND NEW.type != 'superlike' THEN
|
||||
RETURN NEW;
|
||||
END IF;
|
||||
|
||||
-- Проверяем есть ли обратный лайк (правильные имена полей: target_user_id, type)
|
||||
SELECT EXISTS (
|
||||
SELECT 1
|
||||
FROM swipes
|
||||
WHERE user_id = NEW.target_user_id
|
||||
AND target_user_id = NEW.user_id
|
||||
AND type IN ('like', 'superlike')
|
||||
) INTO reverse_like_exists;
|
||||
|
||||
-- Если есть взаимный лайк, создаем матч
|
||||
IF reverse_like_exists THEN
|
||||
-- Создаем матч и получаем его ID
|
||||
INSERT INTO matches (user_id_1, user_id_2, created_at, is_active)
|
||||
VALUES (
|
||||
LEAST(NEW.user_id, NEW.target_user_id),
|
||||
GREATEST(NEW.user_id, NEW.target_user_id),
|
||||
NOW(),
|
||||
true
|
||||
)
|
||||
ON CONFLICT (user_id_1, user_id_2) DO UPDATE SET is_active = true
|
||||
RETURNING id INTO match_id_var;
|
||||
|
||||
-- Создаем уведомления для обоих пользователей
|
||||
-- Используем data (jsonb) вместо content и reference_id
|
||||
INSERT INTO notifications (user_id, type, data, is_read, created_at)
|
||||
VALUES (
|
||||
NEW.user_id,
|
||||
'new_match',
|
||||
jsonb_build_object('message', 'У вас новый матч!', 'match_id', match_id_var),
|
||||
false,
|
||||
NOW()
|
||||
);
|
||||
|
||||
INSERT INTO notifications (user_id, type, data, is_read, created_at)
|
||||
VALUES (
|
||||
NEW.target_user_id,
|
||||
'new_match',
|
||||
jsonb_build_object('message', 'У вас новый матч!', 'match_id', match_id_var),
|
||||
false,
|
||||
NOW()
|
||||
);
|
||||
END IF;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Пересоздаем триггер
|
||||
DROP TRIGGER IF EXISTS create_match_trigger ON swipes;
|
||||
CREATE TRIGGER create_match_trigger
|
||||
AFTER INSERT ON swipes
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION create_match_on_mutual_like();
|
||||
|
||||
-- Проверка
|
||||
SELECT 'Триггер create_match_on_mutual_like успешно исправлен!' as status;
|
||||
|
||||
COMMENT ON FUNCTION create_match_on_mutual_like() IS
|
||||
'Триггер автоматически создает матч при взаимном лайке.
|
||||
Исправлено: использование правильных имен полей (target_user_id, type)
|
||||
и правильной структуры notifications (data jsonb).';
|
||||
49
sql/fix_notify_message_trigger.sql
Normal file
49
sql/fix_notify_message_trigger.sql
Normal file
@@ -0,0 +1,49 @@
|
||||
-- Исправление триггера notify_new_message для использования правильной схемы notifications
|
||||
-- Проблема: триггер использует content и reference_id вместо data (jsonb)
|
||||
|
||||
-- Удаляем старый триггер и функцию
|
||||
DROP TRIGGER IF EXISTS notify_new_message_trigger ON messages;
|
||||
DROP FUNCTION IF EXISTS notify_new_message();
|
||||
|
||||
-- Создаём новую функцию с правильной схемой
|
||||
CREATE OR REPLACE FUNCTION notify_new_message()
|
||||
RETURNS TRIGGER AS $$
|
||||
DECLARE
|
||||
recipient_id UUID;
|
||||
BEGIN
|
||||
-- Определяем получателя сообщения (второго участника матча)
|
||||
SELECT CASE
|
||||
WHEN m.user_id_1 = NEW.sender_id THEN m.user_id_2
|
||||
ELSE m.user_id_1
|
||||
END INTO recipient_id
|
||||
FROM matches m
|
||||
WHERE m.id = NEW.match_id;
|
||||
|
||||
-- Создаём уведомление с правильной структурой (data jsonb)
|
||||
IF recipient_id IS NOT NULL THEN
|
||||
INSERT INTO notifications (user_id, type, data, created_at)
|
||||
VALUES (
|
||||
recipient_id,
|
||||
'new_message',
|
||||
jsonb_build_object(
|
||||
'message_id', NEW.id,
|
||||
'match_id', NEW.match_id,
|
||||
'sender_id', NEW.sender_id,
|
||||
'content_preview', LEFT(NEW.content, 50)
|
||||
),
|
||||
NOW()
|
||||
);
|
||||
END IF;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Создаём триггер заново
|
||||
CREATE TRIGGER notify_new_message_trigger
|
||||
AFTER INSERT ON messages
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION notify_new_message();
|
||||
|
||||
-- Проверка
|
||||
SELECT 'Trigger notify_new_message fixed successfully' AS status;
|
||||
46
sql/fix_update_last_active_trigger.sql
Normal file
46
sql/fix_update_last_active_trigger.sql
Normal file
@@ -0,0 +1,46 @@
|
||||
-- Исправление триггера update_last_active для работы с messages и swipes
|
||||
-- Проблема: в messages есть sender_id, а в swipes есть user_id
|
||||
|
||||
-- Удаляем старые триггеры
|
||||
DROP TRIGGER IF EXISTS update_last_active_on_message ON messages;
|
||||
DROP TRIGGER IF EXISTS update_last_active_on_swipe ON swipes;
|
||||
DROP FUNCTION IF EXISTS update_last_active();
|
||||
|
||||
-- Создаём функцию для обновления last_active для отправителя сообщения
|
||||
CREATE OR REPLACE FUNCTION update_last_active_on_message()
|
||||
RETURNS TRIGGER AS $$
|
||||
BEGIN
|
||||
UPDATE profiles
|
||||
SET last_active = CURRENT_TIMESTAMP
|
||||
WHERE user_id = NEW.sender_id;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Создаём функцию для обновления last_active при свайпе
|
||||
CREATE OR REPLACE FUNCTION update_last_active_on_swipe()
|
||||
RETURNS TRIGGER AS $$
|
||||
BEGIN
|
||||
UPDATE profiles
|
||||
SET last_active = CURRENT_TIMESTAMP
|
||||
WHERE user_id = NEW.user_id;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
-- Создаём триггер для messages
|
||||
CREATE TRIGGER update_last_active_on_message
|
||||
AFTER INSERT ON messages
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_last_active_on_message();
|
||||
|
||||
-- Создаём триггер для swipes
|
||||
CREATE TRIGGER update_last_active_on_swipe
|
||||
AFTER INSERT ON swipes
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION update_last_active_on_swipe();
|
||||
|
||||
-- Проверка
|
||||
SELECT 'Triggers update_last_active fixed successfully' AS status;
|
||||
56
src/bot.ts
56
src/bot.ts
@@ -8,6 +8,9 @@ import LocalizationService from './services/localizationService';
|
||||
import { CommandHandlers } from './handlers/commandHandlers';
|
||||
import { CallbackHandlers } from './handlers/callbackHandlers';
|
||||
import { MessageHandlers } from './handlers/messageHandlers';
|
||||
import { NotificationHandlers } from './handlers/notificationHandlers';
|
||||
import { LanguageHandlers } from './handlers/languageHandlers';
|
||||
|
||||
|
||||
class TelegramTinderBot {
|
||||
private bot: TelegramBot;
|
||||
@@ -18,7 +21,8 @@ class TelegramTinderBot {
|
||||
private commandHandlers: CommandHandlers;
|
||||
private callbackHandlers: CallbackHandlers;
|
||||
private messageHandlers: MessageHandlers;
|
||||
|
||||
private notificationHandlers: NotificationHandlers;
|
||||
private languageHandlers: LanguageHandlers;
|
||||
constructor() {
|
||||
const token = process.env.TELEGRAM_BOT_TOKEN;
|
||||
if (!token) {
|
||||
@@ -32,8 +36,10 @@ class TelegramTinderBot {
|
||||
this.localizationService = LocalizationService.getInstance();
|
||||
|
||||
this.commandHandlers = new CommandHandlers(this.bot);
|
||||
this.messageHandlers = new MessageHandlers(this.bot);
|
||||
this.messageHandlers = new MessageHandlers(this.bot, this.notificationService);
|
||||
this.callbackHandlers = new CallbackHandlers(this.bot, this.messageHandlers);
|
||||
this.notificationHandlers = new NotificationHandlers(this.bot);
|
||||
this.languageHandlers = new LanguageHandlers(this.bot);
|
||||
|
||||
this.setupErrorHandling();
|
||||
this.setupPeriodicTasks();
|
||||
@@ -78,6 +84,7 @@ class TelegramTinderBot {
|
||||
{ command: 'browse', description: '💕 Смотреть анкеты' },
|
||||
{ command: 'matches', description: '💖 Мои матчи' },
|
||||
{ command: 'settings', description: '⚙️ Настройки' },
|
||||
{ command: 'notifications', description: '🔔 Настройки уведомлений' },
|
||||
{ command: 'help', description: '❓ Помощь' }
|
||||
];
|
||||
|
||||
@@ -94,6 +101,9 @@ class TelegramTinderBot {
|
||||
|
||||
// Сообщения
|
||||
this.messageHandlers.register();
|
||||
|
||||
// Обработчики уведомлений
|
||||
this.notificationHandlers.register();
|
||||
}
|
||||
|
||||
// Обработка ошибок
|
||||
@@ -137,14 +147,31 @@ class TelegramTinderBot {
|
||||
}
|
||||
}, 5 * 60 * 1000);
|
||||
|
||||
// Очистка старых данных каждый день
|
||||
// Планирование периодических уведомлений раз в день в 00:05
|
||||
setInterval(async () => {
|
||||
try {
|
||||
await this.cleanupOldData();
|
||||
const now = new Date();
|
||||
if (now.getHours() === 0 && now.getMinutes() >= 5 && now.getMinutes() < 10) {
|
||||
console.log('🔔 Scheduling periodic notifications...');
|
||||
await this.notificationService.schedulePeriodicNotifications();
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error scheduling periodic notifications:', error);
|
||||
}
|
||||
}, 5 * 60 * 1000);
|
||||
|
||||
// Очистка старых данных каждый день в 03:00
|
||||
setInterval(async () => {
|
||||
try {
|
||||
const now = new Date();
|
||||
if (now.getHours() === 3 && now.getMinutes() < 5) {
|
||||
console.log('🧹 Running scheduled cleanup...');
|
||||
await this.cleanupOldData();
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error cleaning up old data:', error);
|
||||
}
|
||||
}, 24 * 60 * 60 * 1000);
|
||||
}, 5 * 60 * 1000);
|
||||
}
|
||||
|
||||
// Очистка старых данных
|
||||
@@ -152,11 +179,18 @@ class TelegramTinderBot {
|
||||
console.log('🧹 Running cleanup tasks...');
|
||||
|
||||
try {
|
||||
// Очистка старых уведомлений (старше 30 дней)
|
||||
const notificationsResult = await query(`
|
||||
// Очистка старых запланированных уведомлений (старше 30 дней или обработанных)
|
||||
const scheduledNotificationsResult = await query(`
|
||||
DELETE FROM scheduled_notifications
|
||||
WHERE processed = true
|
||||
AND created_at < CURRENT_TIMESTAMP - INTERVAL '30 days'
|
||||
WHERE (processed = true AND created_at < CURRENT_TIMESTAMP - INTERVAL '30 days')
|
||||
OR (scheduled_at < CURRENT_TIMESTAMP - INTERVAL '7 days')
|
||||
`);
|
||||
console.log(`🗑️ Cleaned up ${scheduledNotificationsResult.rowCount} old scheduled notifications`);
|
||||
|
||||
// Очистка старых уведомлений (старше 90 дней)
|
||||
const notificationsResult = await query(`
|
||||
DELETE FROM notifications
|
||||
WHERE created_at < CURRENT_TIMESTAMP - INTERVAL '90 days'
|
||||
`);
|
||||
console.log(`🗑️ Cleaned up ${notificationsResult.rowCount} old notifications`);
|
||||
|
||||
@@ -186,7 +220,7 @@ class TelegramTinderBot {
|
||||
console.log(`💬 Cleaned up ${messagesResult.rowCount} old messages`);
|
||||
|
||||
// Обновление статистики таблиц после очистки
|
||||
await query('VACUUM ANALYZE scheduled_notifications, profile_views, swipes, messages');
|
||||
await query('VACUUM ANALYZE notifications, scheduled_notifications, profile_views, swipes, messages');
|
||||
|
||||
console.log('✅ Cleanup completed successfully');
|
||||
} catch (error) {
|
||||
@@ -229,4 +263,4 @@ if (require.main === module) {
|
||||
});
|
||||
}
|
||||
|
||||
export { TelegramTinderBot };
|
||||
export { TelegramTinderBot };
|
||||
|
||||
108
src/database/migrateOnStartup.ts
Normal file
108
src/database/migrateOnStartup.ts
Normal file
@@ -0,0 +1,108 @@
|
||||
// Script to run migrations on startup
|
||||
import { Pool } from 'pg';
|
||||
import * as fs from 'fs';
|
||||
import * as path from 'path';
|
||||
import 'dotenv/config';
|
||||
|
||||
async function runMigrations() {
|
||||
console.log('Starting database migration...');
|
||||
|
||||
// Create a connection pool
|
||||
const pool = new Pool({
|
||||
host: process.env.DB_HOST || 'localhost',
|
||||
port: parseInt(process.env.DB_PORT || '5432'),
|
||||
database: process.env.DB_NAME || 'telegram_tinder_bot',
|
||||
user: process.env.DB_USERNAME || 'postgres',
|
||||
password: process.env.DB_PASSWORD,
|
||||
max: 20,
|
||||
idleTimeoutMillis: 30000,
|
||||
connectionTimeoutMillis: 2000,
|
||||
});
|
||||
|
||||
try {
|
||||
// Test connection
|
||||
const testRes = await pool.query('SELECT NOW()');
|
||||
console.log(`Database connection successful at ${testRes.rows[0].now}`);
|
||||
|
||||
// Create migrations table if not exists
|
||||
await pool.query(`
|
||||
CREATE TABLE IF NOT EXISTS migrations (
|
||||
id SERIAL PRIMARY KEY,
|
||||
name VARCHAR(255) NOT NULL,
|
||||
executed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||
)
|
||||
`);
|
||||
|
||||
// Get list of executed migrations
|
||||
const migrationRes = await pool.query('SELECT name FROM migrations');
|
||||
const executedMigrations = migrationRes.rows.map(row => row.name);
|
||||
console.log(`Found ${executedMigrations.length} executed migrations`);
|
||||
|
||||
// Get migration files
|
||||
const migrationsPath = path.join(__dirname, 'migrations');
|
||||
let migrationFiles = [];
|
||||
|
||||
try {
|
||||
migrationFiles = fs.readdirSync(migrationsPath)
|
||||
.filter(file => file.endsWith('.sql'))
|
||||
.sort();
|
||||
console.log(`Found ${migrationFiles.length} migration files`);
|
||||
} catch (error: any) {
|
||||
console.error(`Error reading migrations directory: ${error.message}`);
|
||||
console.log('Continuing with built-in consolidated migration...');
|
||||
|
||||
// If no external files found, use consolidated.sql
|
||||
const consolidatedSQL = fs.readFileSync(path.join(__dirname, 'migrations', 'consolidated.sql'), 'utf8');
|
||||
|
||||
console.log('Executing consolidated migration...');
|
||||
await pool.query(consolidatedSQL);
|
||||
|
||||
if (!executedMigrations.includes('consolidated.sql')) {
|
||||
await pool.query(
|
||||
'INSERT INTO migrations (name) VALUES ($1)',
|
||||
['consolidated.sql']
|
||||
);
|
||||
}
|
||||
|
||||
console.log('Consolidated migration completed successfully');
|
||||
return;
|
||||
}
|
||||
|
||||
// Run each migration that hasn't been executed yet
|
||||
for (const file of migrationFiles) {
|
||||
if (!executedMigrations.includes(file)) {
|
||||
console.log(`Executing migration: ${file}`);
|
||||
const sql = fs.readFileSync(path.join(migrationsPath, file), 'utf8');
|
||||
|
||||
try {
|
||||
await pool.query('BEGIN');
|
||||
await pool.query(sql);
|
||||
await pool.query(
|
||||
'INSERT INTO migrations (name) VALUES ($1)',
|
||||
[file]
|
||||
);
|
||||
await pool.query('COMMIT');
|
||||
console.log(`Migration ${file} completed successfully`);
|
||||
} catch (error: any) {
|
||||
await pool.query('ROLLBACK');
|
||||
console.error(`Error executing migration ${file}: ${error.message}`);
|
||||
throw error;
|
||||
}
|
||||
} else {
|
||||
console.log(`Migration ${file} already executed, skipping`);
|
||||
}
|
||||
}
|
||||
|
||||
console.log('All migrations completed successfully!');
|
||||
} catch (error: any) {
|
||||
console.error(`Migration failed: ${error.message}`);
|
||||
process.exit(1);
|
||||
} finally {
|
||||
await pool.end();
|
||||
}
|
||||
}
|
||||
|
||||
runMigrations().catch((error: any) => {
|
||||
console.error('Unhandled error during migration:', error);
|
||||
process.exit(1);
|
||||
});
|
||||
182
src/database/migrations/consolidated.sql
Normal file
182
src/database/migrations/consolidated.sql
Normal file
@@ -0,0 +1,182 @@
|
||||
-- Consolidated migrations for Telegram Tinder Bot
|
||||
|
||||
-- Create extension for UUID if not exists
|
||||
CREATE EXTENSION IF NOT EXISTS "uuid-ossp";
|
||||
CREATE EXTENSION IF NOT EXISTS "pgcrypto";
|
||||
|
||||
----------------------------------------------
|
||||
-- Core Tables
|
||||
----------------------------------------------
|
||||
|
||||
-- Users table
|
||||
CREATE TABLE IF NOT EXISTS users (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
telegram_id BIGINT UNIQUE NOT NULL,
|
||||
username VARCHAR(255),
|
||||
first_name VARCHAR(255),
|
||||
last_name VARCHAR(255),
|
||||
language_code VARCHAR(10) DEFAULT 'ru',
|
||||
is_active BOOLEAN DEFAULT TRUE,
|
||||
created_at TIMESTAMP DEFAULT NOW(),
|
||||
last_active_at TIMESTAMP DEFAULT NOW(),
|
||||
premium BOOLEAN DEFAULT FALSE,
|
||||
state VARCHAR(255),
|
||||
state_data JSONB DEFAULT '{}'::jsonb
|
||||
);
|
||||
|
||||
-- Profiles table
|
||||
CREATE TABLE IF NOT EXISTS profiles (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
user_id UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
name VARCHAR(255) NOT NULL,
|
||||
age INTEGER NOT NULL CHECK (age >= 18 AND age <= 100),
|
||||
gender VARCHAR(10) NOT NULL CHECK (gender IN ('male', 'female', 'other')),
|
||||
interested_in VARCHAR(10) NOT NULL CHECK (interested_in IN ('male', 'female', 'both')),
|
||||
bio TEXT,
|
||||
photos JSONB DEFAULT '[]',
|
||||
interests JSONB DEFAULT '[]',
|
||||
city VARCHAR(255),
|
||||
education VARCHAR(255),
|
||||
job VARCHAR(255),
|
||||
height INTEGER,
|
||||
location_lat DECIMAL(10, 8),
|
||||
location_lon DECIMAL(11, 8),
|
||||
search_min_age INTEGER DEFAULT 18,
|
||||
search_max_age INTEGER DEFAULT 50,
|
||||
search_max_distance INTEGER DEFAULT 50,
|
||||
is_verified BOOLEAN DEFAULT FALSE,
|
||||
is_visible BOOLEAN DEFAULT TRUE,
|
||||
created_at TIMESTAMP DEFAULT NOW(),
|
||||
updated_at TIMESTAMP DEFAULT NOW(),
|
||||
religion VARCHAR(255),
|
||||
dating_goal VARCHAR(50),
|
||||
smoking VARCHAR(20),
|
||||
drinking VARCHAR(20),
|
||||
has_kids BOOLEAN DEFAULT FALSE
|
||||
);
|
||||
|
||||
-- Swipes table
|
||||
CREATE TABLE IF NOT EXISTS swipes (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
user_id UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
target_user_id UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(20) NOT NULL CHECK (type IN ('like', 'pass', 'superlike')),
|
||||
created_at TIMESTAMP DEFAULT NOW(),
|
||||
is_match BOOLEAN DEFAULT FALSE,
|
||||
UNIQUE(user_id, target_user_id)
|
||||
);
|
||||
|
||||
-- Matches table
|
||||
CREATE TABLE IF NOT EXISTS matches (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
user_id_1 UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
user_id_2 UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
created_at TIMESTAMP DEFAULT NOW(),
|
||||
last_message_at TIMESTAMP,
|
||||
is_active BOOLEAN DEFAULT TRUE,
|
||||
is_super_match BOOLEAN DEFAULT FALSE,
|
||||
unread_count_1 INTEGER DEFAULT 0,
|
||||
unread_count_2 INTEGER DEFAULT 0,
|
||||
UNIQUE(user_id_1, user_id_2)
|
||||
);
|
||||
|
||||
-- Messages table
|
||||
CREATE TABLE IF NOT EXISTS messages (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
match_id UUID REFERENCES matches(id) ON DELETE CASCADE,
|
||||
sender_id UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
receiver_id UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
content TEXT NOT NULL,
|
||||
message_type VARCHAR(20) DEFAULT 'text' CHECK (message_type IN ('text', 'photo', 'gif', 'sticker')),
|
||||
created_at TIMESTAMP DEFAULT NOW(),
|
||||
is_read BOOLEAN DEFAULT FALSE
|
||||
);
|
||||
|
||||
----------------------------------------------
|
||||
-- Profile Views Table
|
||||
----------------------------------------------
|
||||
|
||||
-- Table for tracking profile views
|
||||
CREATE TABLE IF NOT EXISTS profile_views (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
viewer_id UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
viewed_id UUID REFERENCES users(id) ON DELETE CASCADE,
|
||||
view_type VARCHAR(20) DEFAULT 'browse' CHECK (view_type IN ('browse', 'search', 'recommended')),
|
||||
viewed_at TIMESTAMP DEFAULT NOW(),
|
||||
CONSTRAINT unique_profile_view UNIQUE (viewer_id, viewed_id, view_type)
|
||||
);
|
||||
|
||||
-- Index for profile views
|
||||
CREATE INDEX IF NOT EXISTS idx_profile_views_viewer ON profile_views(viewer_id);
|
||||
CREATE INDEX IF NOT EXISTS idx_profile_views_viewed ON profile_views(viewed_id);
|
||||
|
||||
----------------------------------------------
|
||||
-- Notification Tables
|
||||
----------------------------------------------
|
||||
|
||||
-- Notifications table
|
||||
CREATE TABLE IF NOT EXISTS notifications (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
content JSONB NOT NULL DEFAULT '{}',
|
||||
is_read BOOLEAN DEFAULT FALSE,
|
||||
processed BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||
);
|
||||
|
||||
-- Notification settings table
|
||||
CREATE TABLE IF NOT EXISTS notification_settings (
|
||||
user_id UUID PRIMARY KEY REFERENCES users(id) ON DELETE CASCADE,
|
||||
new_matches BOOLEAN DEFAULT TRUE,
|
||||
new_messages BOOLEAN DEFAULT TRUE,
|
||||
new_likes BOOLEAN DEFAULT TRUE,
|
||||
reminders BOOLEAN DEFAULT TRUE,
|
||||
daily_summary BOOLEAN DEFAULT FALSE,
|
||||
time_preference VARCHAR(20) DEFAULT 'evening',
|
||||
do_not_disturb BOOLEAN DEFAULT FALSE,
|
||||
do_not_disturb_start TIME,
|
||||
do_not_disturb_end TIME,
|
||||
updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||
);
|
||||
|
||||
-- Scheduled notifications table
|
||||
CREATE TABLE IF NOT EXISTS scheduled_notifications (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||
type VARCHAR(50) NOT NULL,
|
||||
content JSONB NOT NULL DEFAULT '{}',
|
||||
scheduled_at TIMESTAMP WITH TIME ZONE NOT NULL,
|
||||
processed BOOLEAN DEFAULT FALSE,
|
||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||
);
|
||||
|
||||
----------------------------------------------
|
||||
-- Indexes for better performance
|
||||
----------------------------------------------
|
||||
|
||||
-- User Indexes
|
||||
CREATE INDEX IF NOT EXISTS idx_users_telegram_id ON users(telegram_id);
|
||||
|
||||
-- Profile Indexes
|
||||
CREATE INDEX IF NOT EXISTS idx_profiles_user_id ON profiles(user_id);
|
||||
CREATE INDEX IF NOT EXISTS idx_profiles_location ON profiles(location_lat, location_lon)
|
||||
WHERE location_lat IS NOT NULL AND location_lon IS NOT NULL;
|
||||
CREATE INDEX IF NOT EXISTS idx_profiles_age_gender ON profiles(age, gender, interested_in);
|
||||
|
||||
-- Swipe Indexes
|
||||
CREATE INDEX IF NOT EXISTS idx_swipes_user ON swipes(user_id, target_user_id);
|
||||
|
||||
-- Match Indexes
|
||||
CREATE INDEX IF NOT EXISTS idx_matches_users ON matches(user_id_1, user_id_2);
|
||||
|
||||
-- Message Indexes
|
||||
CREATE INDEX IF NOT EXISTS idx_messages_match ON messages(match_id, created_at);
|
||||
|
||||
-- Notification Indexes
|
||||
CREATE INDEX IF NOT EXISTS idx_notifications_user_id ON notifications(user_id);
|
||||
CREATE INDEX IF NOT EXISTS idx_notifications_type ON notifications(type);
|
||||
CREATE INDEX IF NOT EXISTS idx_notifications_created_at ON notifications(created_at);
|
||||
CREATE INDEX IF NOT EXISTS idx_scheduled_notifications_user_id ON scheduled_notifications(user_id);
|
||||
CREATE INDEX IF NOT EXISTS idx_scheduled_notifications_scheduled_at ON scheduled_notifications(scheduled_at);
|
||||
CREATE INDEX IF NOT EXISTS idx_scheduled_notifications_processed ON scheduled_notifications(processed);
|
||||
@@ -4,12 +4,14 @@ import { MatchingService } from '../services/matchingService';
|
||||
import { ChatService } from '../services/chatService';
|
||||
import { Profile } from '../models/Profile';
|
||||
import { MessageHandlers } from './messageHandlers';
|
||||
import { NotificationHandlers } from './notificationHandlers';
|
||||
import { ProfileEditController } from '../controllers/profileEditController';
|
||||
import { EnhancedChatHandlers } from './enhancedChatHandlers';
|
||||
import { VipController } from '../controllers/vipController';
|
||||
import { VipService } from '../services/vipService';
|
||||
import { TranslationController } from '../controllers/translationController';
|
||||
import { t } from '../services/localizationService';
|
||||
import { LikeBackHandler } from './likeBackHandler';
|
||||
|
||||
export class CallbackHandlers {
|
||||
private bot: TelegramBot;
|
||||
@@ -22,18 +24,44 @@ export class CallbackHandlers {
|
||||
private vipController: VipController;
|
||||
private vipService: VipService;
|
||||
private translationController: TranslationController;
|
||||
private notificationHandlers?: NotificationHandlers;
|
||||
private likeBackHandler: LikeBackHandler;
|
||||
|
||||
constructor(bot: TelegramBot, messageHandlers: MessageHandlers) {
|
||||
this.bot = bot;
|
||||
this.profileService = new ProfileService();
|
||||
this.matchingService = new MatchingService();
|
||||
this.chatService = new ChatService();
|
||||
// Получаем notificationService из messageHandlers (если есть)
|
||||
const notificationService = (messageHandlers as any).notificationService;
|
||||
this.chatService = new ChatService(notificationService);
|
||||
this.messageHandlers = messageHandlers;
|
||||
this.profileEditController = new ProfileEditController(this.profileService);
|
||||
this.enhancedChatHandlers = new EnhancedChatHandlers(bot);
|
||||
this.vipController = new VipController(bot);
|
||||
this.vipService = new VipService();
|
||||
this.translationController = new TranslationController();
|
||||
// Создаем экземпляр NotificationHandlers
|
||||
try {
|
||||
this.notificationHandlers = new NotificationHandlers(bot);
|
||||
} catch (error) {
|
||||
console.error('Failed to initialize NotificationHandlers:', error);
|
||||
}
|
||||
this.likeBackHandler = new LikeBackHandler(bot);
|
||||
}
|
||||
|
||||
// Вспомогательный метод для получения перевода с учетом языка пользователя
|
||||
private async getTranslation(userId: string, key: string, options?: any): Promise<string> {
|
||||
try {
|
||||
const lang = await this.profileService.getUserLanguage(userId);
|
||||
const LocalizationService = require('../services/localizationService').default;
|
||||
const locService = LocalizationService.getInstance();
|
||||
locService.setLanguage(lang);
|
||||
return locService.t(key, options);
|
||||
} catch (error) {
|
||||
console.error('Translation error:', error);
|
||||
// Возвращаем ключ как fallback
|
||||
return key;
|
||||
}
|
||||
}
|
||||
|
||||
register(): void {
|
||||
@@ -48,6 +76,14 @@ export class CallbackHandlers {
|
||||
const data = query.data;
|
||||
|
||||
try {
|
||||
// Обработка выбора языка
|
||||
if (data.startsWith('set_lang_')) {
|
||||
const LanguageHandlers = require('./languageHandlers').LanguageHandlers;
|
||||
const languageHandlers = new LanguageHandlers(this.bot);
|
||||
await languageHandlers.handleSetLanguage(query);
|
||||
return;
|
||||
}
|
||||
|
||||
// Основные действия профиля
|
||||
if (data === 'create_profile') {
|
||||
await this.handleCreateProfile(chatId, telegramId);
|
||||
@@ -75,6 +111,119 @@ export class CallbackHandlers {
|
||||
await this.handleEditHobbies(chatId, telegramId);
|
||||
} else if (data === 'edit_city') {
|
||||
await this.handleEditCity(chatId, telegramId);
|
||||
} else if (data === 'confirm_city') {
|
||||
try {
|
||||
const states = (this.messageHandlers as any).userStates as Map<string, any>;
|
||||
const userState = states ? states.get(telegramId) : null;
|
||||
if (userState && userState.step === 'confirm_city') {
|
||||
// Подтверждаем город и переводим пользователя к вводу био
|
||||
userState.step = 'waiting_bio';
|
||||
console.log(`User ${telegramId} confirmed city: ${userState.data.city}`);
|
||||
// Убираем inline-кнопки из сообщения с подтверждением
|
||||
try {
|
||||
// clear inline keyboard
|
||||
await this.bot.editMessageReplyMarkup({ inline_keyboard: [] } as any, { chat_id: chatId, message_id: query.message?.message_id });
|
||||
} catch (e) {
|
||||
// ignore
|
||||
}
|
||||
await this.bot.sendMessage(chatId, `✅ Город подтверждён: *${userState.data.city}*\n\n📝 Теперь расскажите немного о себе (био):`, { parse_mode: 'Markdown' });
|
||||
} else {
|
||||
const errorText = await this.getTranslation(telegramId, 'errors.contextNotFound');
|
||||
await this.bot.answerCallbackQuery(query.id, { text: errorText });
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error confirming city via callback:', error);
|
||||
const errorText = await this.getTranslation(telegramId, 'errors.cityConfirmError');
|
||||
await this.bot.answerCallbackQuery(query.id, { text: errorText });
|
||||
}
|
||||
} else if (data === 'edit_city_manual') {
|
||||
try {
|
||||
const states = (this.messageHandlers as any).userStates as Map<string, any>;
|
||||
const userState = states ? states.get(telegramId) : null;
|
||||
if (userState) {
|
||||
userState.step = 'waiting_city';
|
||||
console.log(`User ${telegramId} chose to enter city manually`);
|
||||
try {
|
||||
// clear inline keyboard
|
||||
await this.bot.editMessageReplyMarkup({ inline_keyboard: [] } as any, { chat_id: chatId, message_id: query.message?.message_id });
|
||||
} catch (e) { }
|
||||
await this.bot.sendMessage(chatId, '✏️ Введите название вашего города вручную:', { reply_markup: { remove_keyboard: true } });
|
||||
} else {
|
||||
const errorText = await this.getTranslation(telegramId, 'errors.contextNotFound');
|
||||
await this.bot.answerCallbackQuery(query.id, { text: errorText });
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error switching to manual city input via callback:', error);
|
||||
const errorText = await this.getTranslation(telegramId, 'errors.generalError');
|
||||
await this.bot.answerCallbackQuery(query.id, { text: errorText });
|
||||
}
|
||||
} else if (data === 'confirm_city_edit') {
|
||||
try {
|
||||
const editState = this.messageHandlers.profileEditStates.get(telegramId);
|
||||
if (editState && editState.field === 'city' && editState.tempCity) {
|
||||
console.log(`User ${telegramId} confirmed city edit: ${editState.tempCity}`);
|
||||
// Обновляем город в профиле
|
||||
await this.messageHandlers.updateProfileField(telegramId, 'city', editState.tempCity);
|
||||
// Обновляем координаты, если они есть
|
||||
if (editState.tempLocation) {
|
||||
console.log(`User ${telegramId} updating location: lat=${editState.tempLocation.latitude}, lon=${editState.tempLocation.longitude}`);
|
||||
await this.messageHandlers.updateProfileField(telegramId, 'location', editState.tempLocation);
|
||||
}
|
||||
// Очищаем состояние
|
||||
this.messageHandlers.clearProfileEditState(telegramId);
|
||||
// Убираем inline-кнопки
|
||||
try {
|
||||
await this.bot.editMessageReplyMarkup({ inline_keyboard: [] } as any, { chat_id: chatId, message_id: query.message?.message_id });
|
||||
} catch (e) { }
|
||||
await this.bot.sendMessage(chatId, '✅ Город успешно обновлён!');
|
||||
setTimeout(async () => {
|
||||
const keyboard = {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '✏️ Продолжить редактирование', callback_data: 'edit_profile' },
|
||||
{ text: '👀 Предпросмотр', callback_data: 'preview_profile' }
|
||||
],
|
||||
[{ text: '🏠 Главное меню', callback_data: 'main_menu' }]
|
||||
]
|
||||
};
|
||||
const selectActionText = await this.getTranslation(telegramId, 'buttons.selectAction');
|
||||
await this.bot.sendMessage(chatId, selectActionText, { reply_markup: keyboard });
|
||||
}, 500);
|
||||
} else {
|
||||
const errorText = await this.getTranslation(telegramId, 'errors.contextNotFound');
|
||||
await this.bot.answerCallbackQuery(query.id, { text: errorText });
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error confirming city edit via callback:', error);
|
||||
const errorText = await this.getTranslation(telegramId, 'errors.cityConfirmError');
|
||||
await this.bot.answerCallbackQuery(query.id, { text: errorText });
|
||||
}
|
||||
} else if (data === 'edit_city_manual_edit') {
|
||||
try {
|
||||
const editState = this.messageHandlers.profileEditStates.get(telegramId);
|
||||
if (editState && editState.field === 'city') {
|
||||
console.log(`User ${telegramId} chose to re-enter city during edit`);
|
||||
// Очищаем временный город, но оставляем состояние редактирования
|
||||
delete editState.tempCity;
|
||||
try {
|
||||
await this.bot.editMessageReplyMarkup({ inline_keyboard: [] } as any, { chat_id: chatId, message_id: query.message?.message_id });
|
||||
} catch (e) { }
|
||||
await this.bot.sendMessage(chatId, '✏️ Введите название вашего города вручную или отправьте геолокацию:', {
|
||||
reply_markup: {
|
||||
keyboard: [
|
||||
[{ text: '📍 Отправить геолокацию', request_location: true }]
|
||||
],
|
||||
resize_keyboard: true,
|
||||
one_time_keyboard: true
|
||||
}
|
||||
});
|
||||
} else {
|
||||
await this.bot.answerCallbackQuery(query.id, { text: 'Контекст не найден. Повторите, пожалуйста.' });
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error switching to re-enter city during edit via callback:', error);
|
||||
await this.bot.answerCallbackQuery(query.id, { text: 'Ошибка' });
|
||||
}
|
||||
} else if (data === 'edit_job') {
|
||||
await this.handleEditJob(chatId, telegramId);
|
||||
} else if (data === 'edit_education') {
|
||||
@@ -167,6 +316,18 @@ export class CallbackHandlers {
|
||||
await this.handleMorePhotos(chatId, telegramId, targetUserId);
|
||||
}
|
||||
|
||||
// Обработка лайков и ответных лайков из уведомлений
|
||||
else if (data.startsWith('like_back:')) {
|
||||
const targetUserId = data.replace('like_back:', '');
|
||||
await this.likeBackHandler.handleLikeBack(chatId, telegramId, targetUserId);
|
||||
}
|
||||
|
||||
// Быстрый переход в чат из уведомлений
|
||||
else if (data.startsWith('open_chat:')) {
|
||||
const matchId = data.replace('open_chat:', '');
|
||||
await this.enhancedChatHandlers.openNativeChat(chatId, telegramId, matchId);
|
||||
}
|
||||
|
||||
// Матчи и чаты
|
||||
else if (data === 'view_matches') {
|
||||
await this.handleViewMatches(chatId, telegramId);
|
||||
@@ -263,9 +424,46 @@ export class CallbackHandlers {
|
||||
await this.handleSettings(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Настройки уведомлений
|
||||
else if (data === 'notifications') {
|
||||
if (this.notificationHandlers) {
|
||||
const userId = await this.profileService.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.answerCallbackQuery(query.id, { text: '❌ Вы не зарегистрированы.' });
|
||||
return;
|
||||
}
|
||||
|
||||
const settings = await this.notificationHandlers.getNotificationService().getNotificationSettings(userId);
|
||||
await this.notificationHandlers.sendNotificationSettings(chatId, settings);
|
||||
} else {
|
||||
await this.handleNotificationSettings(chatId, telegramId);
|
||||
}
|
||||
}
|
||||
// Обработка переключения настроек уведомлений
|
||||
else if (data.startsWith('notif_toggle:') ||
|
||||
data === 'notif_time' ||
|
||||
data.startsWith('notif_time_set:') ||
|
||||
data === 'notif_dnd' ||
|
||||
data.startsWith('notif_dnd_set:') ||
|
||||
data === 'notif_dnd_time' ||
|
||||
data.startsWith('notif_dnd_time_set:') ||
|
||||
data === 'notif_dnd_time_custom') {
|
||||
// Делегируем обработку в NotificationHandlers, если он доступен
|
||||
if (this.notificationHandlers) {
|
||||
// Эти коллбэки уже обрабатываются в NotificationHandlers, поэтому здесь ничего не делаем
|
||||
// NotificationHandlers уже зарегистрировал свои обработчики в register()
|
||||
} else {
|
||||
const errorText = await this.getTranslation(telegramId, 'notifications.unavailable');
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: errorText,
|
||||
show_alert: true
|
||||
});
|
||||
}
|
||||
}
|
||||
else {
|
||||
const devText = await this.getTranslation(telegramId, 'notifications.inDevelopment');
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: 'Функция в разработке!',
|
||||
text: devText,
|
||||
show_alert: false
|
||||
});
|
||||
return;
|
||||
@@ -275,8 +473,9 @@ export class CallbackHandlers {
|
||||
|
||||
} catch (error) {
|
||||
console.error('Callback handler error:', error);
|
||||
const errorText = await this.getTranslation(telegramId, 'errors.tryAgain');
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: 'Произошла ошибка. Попробуйте еще раз.',
|
||||
text: errorText,
|
||||
show_alert: true
|
||||
});
|
||||
}
|
||||
@@ -374,20 +573,27 @@ export class CallbackHandlers {
|
||||
]
|
||||
};
|
||||
|
||||
const matchText = await this.getTranslation(telegramId, 'matches.mutualLike', {
|
||||
name: targetProfile?.name || await this.getTranslation(telegramId, 'common.thisUser')
|
||||
});
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'🎉 ЭТО МАТЧ! 💕\n\n' +
|
||||
'Вы понравились друг другу с ' + (targetProfile?.name || 'этим пользователем') + '!\n\n' +
|
||||
'Теперь вы можете начать общение!',
|
||||
'🎉 ЭТО МАТЧ! 💕\n\n' + matchText,
|
||||
{ reply_markup: keyboard }
|
||||
);
|
||||
} else {
|
||||
await this.bot.sendMessage(chatId, '👍 Лайк отправлен!');
|
||||
await this.showNextCandidate(chatId, telegramId);
|
||||
}
|
||||
} catch (error) {
|
||||
await this.bot.sendMessage(chatId, '❌ Ошибка при отправке лайка');
|
||||
console.error('Like error:', error);
|
||||
} catch (error: any) {
|
||||
// Проверяем, связана ли ошибка с уже существующим свайпом
|
||||
if (error.message === 'Already swiped this profile' || error.code === 'ALREADY_SWIPED') {
|
||||
await this.bot.sendMessage(chatId, '❓ Вы уже оценивали этот профиль ранее');
|
||||
await this.showNextCandidate(chatId, telegramId);
|
||||
} else {
|
||||
await this.bot.sendMessage(chatId, '❌ Ошибка при отправке лайка');
|
||||
console.error('Like error:', error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -402,9 +608,15 @@ export class CallbackHandlers {
|
||||
|
||||
await this.matchingService.performSwipe(telegramId, targetTelegramId, 'pass');
|
||||
await this.showNextCandidate(chatId, telegramId);
|
||||
} catch (error) {
|
||||
await this.bot.sendMessage(chatId, '❌ Ошибка при отправке дизлайка');
|
||||
console.error('Dislike error:', error);
|
||||
} catch (error: any) {
|
||||
// Проверяем, связана ли ошибка с уже существующим свайпом
|
||||
if (error.message === 'Already swiped this profile' || error.code === 'ALREADY_SWIPED') {
|
||||
await this.bot.sendMessage(chatId, '❓ Вы уже оценивали этот профиль ранее');
|
||||
await this.showNextCandidate(chatId, telegramId);
|
||||
} else {
|
||||
await this.bot.sendMessage(chatId, '❌ Ошибка при отправке дизлайка');
|
||||
console.error('Dislike error:', error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -432,20 +644,85 @@ export class CallbackHandlers {
|
||||
]
|
||||
};
|
||||
|
||||
const superMatchText = await this.getTranslation(telegramId, 'matches.superLikeMatch', {
|
||||
name: targetProfile?.name || await this.getTranslation(telegramId, 'common.thisUser')
|
||||
});
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'💖 СУПЕР МАТЧ! ⭐\n\n' +
|
||||
'Ваш супер лайк произвел впечатление на ' + (targetProfile?.name || 'этого пользователя') + '!\n\n' +
|
||||
'Начните общение первыми!',
|
||||
'💖 СУПЕР МАТЧ! ⭐\n\n' + superMatchText,
|
||||
{ reply_markup: keyboard }
|
||||
);
|
||||
} else {
|
||||
await this.bot.sendMessage(chatId, '💖 Супер лайк отправлен!');
|
||||
await this.showNextCandidate(chatId, telegramId);
|
||||
}
|
||||
} catch (error: any) {
|
||||
// Проверяем, связана ли ошибка с уже существующим свайпом
|
||||
if (error.message === 'Already swiped this profile' || error.code === 'ALREADY_SWIPED') {
|
||||
await this.bot.sendMessage(chatId, '❓ Вы уже оценивали этот профиль ранее');
|
||||
await this.showNextCandidate(chatId, telegramId);
|
||||
} else {
|
||||
await this.bot.sendMessage(chatId, '❌ Ошибка при отправке супер лайка');
|
||||
console.error('Superlike error:', error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Обработка обратного лайка из уведомления
|
||||
async handleLikeBack(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
try {
|
||||
// Получаем информацию о пользователях
|
||||
const targetProfile = await this.profileService.getProfileByUserId(targetUserId);
|
||||
|
||||
if (!targetProfile) {
|
||||
await this.bot.sendMessage(chatId, '❌ Не удалось найти профиль');
|
||||
return;
|
||||
}
|
||||
|
||||
// Получаем telegram ID целевого пользователя для свайпа
|
||||
const targetTelegramId = await this.profileService.getTelegramIdByUserId(targetUserId);
|
||||
if (!targetTelegramId) {
|
||||
await this.bot.sendMessage(chatId, '❌ Не удалось найти пользователя');
|
||||
return;
|
||||
}
|
||||
|
||||
// Выполняем свайп
|
||||
const result = await this.matchingService.performSwipe(telegramId, targetTelegramId, 'like');
|
||||
|
||||
if (result.isMatch) {
|
||||
// Это матч!
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'🎉 *Поздравляем! Это взаимно!*\n\n' +
|
||||
`Вы и *${targetProfile.name}* понравились друг другу!\n` +
|
||||
'Теперь вы можете начать общение.',
|
||||
{
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[{ text: '💬 Начать общение', callback_data: `start_chat:${targetUserId}` }],
|
||||
[{ text: '👀 Посмотреть профиль', callback_data: `view_profile:${targetUserId}` }]
|
||||
]
|
||||
}
|
||||
}
|
||||
);
|
||||
} else {
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'❤️ Вам понравился профиль ' + targetProfile.name + '!\n\n' +
|
||||
'Если вы также понравитесь этому пользователю, будет создан матч.',
|
||||
{
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[{ text: '🔍 Продолжить поиск', callback_data: 'start_browsing' }]
|
||||
]
|
||||
}
|
||||
}
|
||||
);
|
||||
}
|
||||
} catch (error) {
|
||||
await this.bot.sendMessage(chatId, '❌ Ошибка при отправке супер лайка');
|
||||
console.error('Superlike error:', error);
|
||||
console.error('Error in handleLikeBack:', error);
|
||||
await this.bot.sendMessage(chatId, '❌ Произошла ошибка при обработке лайка');
|
||||
}
|
||||
}
|
||||
|
||||
@@ -476,9 +753,28 @@ export class CallbackHandlers {
|
||||
return;
|
||||
}
|
||||
|
||||
for (let i = 1; i < targetProfile.photos.length; i++) {
|
||||
const photoFileId = targetProfile.photos[i];
|
||||
await this.bot.sendPhoto(chatId, photoFileId);
|
||||
// Отправляем фотографии в виде медиа-группы (коллажа)
|
||||
// Создаем массив объектов медиа для группового отправления
|
||||
const mediaGroup = targetProfile.photos.slice(1).map((photoFileId, index) => ({
|
||||
type: 'photo' as const,
|
||||
media: photoFileId,
|
||||
caption: index === 0 ? `📸 Дополнительные фото ${targetProfile.name}` : undefined
|
||||
}));
|
||||
|
||||
try {
|
||||
// Отправляем все фото одним сообщением (медиа-группой)
|
||||
await this.bot.sendMediaGroup(chatId, mediaGroup);
|
||||
} catch (error) {
|
||||
console.error('Error sending media group:', error);
|
||||
|
||||
// Если не получилось отправить медиа-группой, отправляем по одной
|
||||
for (let i = 1; i < targetProfile.photos.length; i++) {
|
||||
try {
|
||||
await this.bot.sendPhoto(chatId, targetProfile.photos[i]);
|
||||
} catch (photoError) {
|
||||
console.error(`Error sending photo ${i}:`, photoError);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
const keyboard: InlineKeyboardMarkup = {
|
||||
@@ -766,13 +1062,7 @@ export class CallbackHandlers {
|
||||
);
|
||||
}
|
||||
|
||||
// Настройки уведомлений
|
||||
async handleNotificationSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'🔔 Настройки уведомлений будут доступны в следующем обновлении!'
|
||||
);
|
||||
}
|
||||
// Настройки уведомлений - реализация перенесена в расширенную версию
|
||||
|
||||
// Как это работает
|
||||
async handleHowItWorks(chatId: number): Promise<void> {
|
||||
@@ -807,10 +1097,24 @@ export class CallbackHandlers {
|
||||
|
||||
// Вспомогательные методы
|
||||
async showProfile(chatId: number, profile: Profile, isOwner: boolean = false, viewerId?: string): Promise<void> {
|
||||
const hasMultiplePhotos = profile.photos.length > 1;
|
||||
const mainPhotoFileId = profile.photos[0]; // Первое фото - главное
|
||||
|
||||
let profileText = '👤 ' + profile.name + ', ' + profile.age + '\n';
|
||||
profileText += '📍 ' + (profile.city || 'Не указан') + '\n';
|
||||
profileText += '📍 ' + (profile.city || 'Не указан');
|
||||
|
||||
// Добавляем расстояние, если это не владелец профиля и есть viewerId
|
||||
if (!isOwner && viewerId) {
|
||||
const viewerProfile = await this.profileService.getProfileByTelegramId(viewerId);
|
||||
if (viewerProfile && viewerProfile.location && profile.location) {
|
||||
const distance = viewerProfile.getDistanceTo(profile);
|
||||
if (distance !== null) {
|
||||
profileText += ` (${Math.round(distance)} км)`;
|
||||
}
|
||||
}
|
||||
}
|
||||
profileText += '\n';
|
||||
|
||||
if (profile.job) profileText += '💼 ' + profile.job + '\n';
|
||||
if (profile.education) profileText += '🎓 ' + profile.education + '\n';
|
||||
if (profile.height) profileText += '📏 ' + profile.height + ' см\n';
|
||||
@@ -833,14 +1137,22 @@ export class CallbackHandlers {
|
||||
profileText += '\n📝 ' + (profile.bio || 'Описание не указано') + '\n';
|
||||
|
||||
// Хобби с хэштегами
|
||||
if (profile.hobbies && profile.hobbies.trim()) {
|
||||
const hobbiesArray = profile.hobbies.split(',').map(hobby => hobby.trim()).filter(hobby => hobby);
|
||||
const formattedHobbies = hobbiesArray.map(hobby => '#' + hobby).join(' ');
|
||||
profileText += '\n🎯 ' + formattedHobbies + '\n';
|
||||
if (profile.hobbies) {
|
||||
let hobbiesArray: string[] = [];
|
||||
if (typeof profile.hobbies === 'string') {
|
||||
hobbiesArray = profile.hobbies.split(',').map(hobby => hobby.trim()).filter(hobby => hobby);
|
||||
} else if (Array.isArray(profile.hobbies)) {
|
||||
hobbiesArray = (profile.hobbies as string[]).filter((hobby: string) => hobby && hobby.trim());
|
||||
}
|
||||
|
||||
if (hobbiesArray.length > 0) {
|
||||
const formattedHobbies = hobbiesArray.map(hobby => '#' + hobby).join(' ');
|
||||
profileText += '\n🎯 ' + formattedHobbies + '\n';
|
||||
}
|
||||
}
|
||||
|
||||
if (profile.interests.length > 0) {
|
||||
profileText += '\n<EFBFBD> Интересы: ' + profile.interests.join(', ');
|
||||
profileText += '\n💡 Интересы: ' + profile.interests.join(', ');
|
||||
}
|
||||
|
||||
let keyboard: InlineKeyboardMarkup;
|
||||
@@ -876,20 +1188,43 @@ export class CallbackHandlers {
|
||||
|
||||
if (hasValidPhoto) {
|
||||
try {
|
||||
await this.bot.sendPhoto(chatId, mainPhotoFileId, {
|
||||
caption: profileText,
|
||||
reply_markup: keyboard
|
||||
});
|
||||
if (hasMultiplePhotos) {
|
||||
// Если есть несколько фото, отправляем их как медиа-группу (коллаж)
|
||||
const mediaGroup = profile.photos.map((photoFileId, index) => ({
|
||||
type: 'photo' as const,
|
||||
media: photoFileId,
|
||||
caption: index === 0 ? profileText : undefined,
|
||||
parse_mode: index === 0 ? 'Markdown' as const : undefined
|
||||
}));
|
||||
|
||||
// Сначала отправляем медиа-группу
|
||||
await this.bot.sendMediaGroup(chatId, mediaGroup);
|
||||
|
||||
// Затем отправляем отдельное сообщение с кнопками
|
||||
await this.bot.sendMessage(chatId, '📸 Выберите действие:', {
|
||||
reply_markup: keyboard
|
||||
});
|
||||
} else {
|
||||
// Если только одно фото, отправляем его с текстом
|
||||
await this.bot.sendPhoto(chatId, mainPhotoFileId, {
|
||||
caption: profileText,
|
||||
reply_markup: keyboard,
|
||||
parse_mode: 'Markdown'
|
||||
});
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error sending profile photos:', error);
|
||||
// Если не удалось отправить фото, отправляем текст
|
||||
await this.bot.sendMessage(chatId, '🖼 Фото недоступно\n\n' + profileText, {
|
||||
reply_markup: keyboard
|
||||
reply_markup: keyboard,
|
||||
parse_mode: 'Markdown'
|
||||
});
|
||||
}
|
||||
} else {
|
||||
// Отправляем как текстовое сообщение
|
||||
await this.bot.sendMessage(chatId, profileText, {
|
||||
reply_markup: keyboard
|
||||
reply_markup: keyboard,
|
||||
parse_mode: 'Markdown'
|
||||
});
|
||||
}
|
||||
}
|
||||
@@ -922,8 +1257,21 @@ export class CallbackHandlers {
|
||||
|
||||
const candidatePhotoFileId = candidate.photos[0]; // Первое фото - главное
|
||||
|
||||
// Получаем профиль текущего пользователя для вычисления расстояния
|
||||
const userProfile = await this.profileService.getProfileByTelegramId(telegramId);
|
||||
|
||||
let candidateText = candidate.name + ', ' + candidate.age + '\n';
|
||||
candidateText += '📍 ' + (candidate.city || 'Не указан') + '\n';
|
||||
candidateText += '📍 ' + (candidate.city || 'Не указан');
|
||||
|
||||
// Добавляем расстояние, если есть координаты у обоих пользователей
|
||||
if (userProfile && userProfile.location && candidate.location) {
|
||||
const distance = userProfile.getDistanceTo(candidate);
|
||||
if (distance !== null) {
|
||||
candidateText += ` (${Math.round(distance)} км)`;
|
||||
}
|
||||
}
|
||||
candidateText += '\n';
|
||||
|
||||
if (candidate.job) candidateText += '💼 ' + candidate.job + '\n';
|
||||
if (candidate.education) candidateText += '🎓 ' + candidate.education + '\n';
|
||||
if (candidate.height) candidateText += '📏 ' + candidate.height + ' см\n';
|
||||
@@ -1027,8 +1375,15 @@ export class CallbackHandlers {
|
||||
// Редактирование города
|
||||
async handleEditCity(chatId: number, telegramId: string): Promise<void> {
|
||||
this.messageHandlers.setWaitingForInput(parseInt(telegramId), 'city');
|
||||
await this.bot.sendMessage(chatId, '🏙️ *Введите ваш город:*\n\nНапример: Москва', {
|
||||
parse_mode: 'Markdown'
|
||||
await this.bot.sendMessage(chatId, '🏙️ *Укажите ваш город:*\n\nВыберите один из вариантов:', {
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
keyboard: [
|
||||
[{ text: '📍 Отправить геолокацию', request_location: true }]
|
||||
],
|
||||
resize_keyboard: true,
|
||||
one_time_keyboard: true
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@@ -1450,7 +1805,7 @@ export class CallbackHandlers {
|
||||
try {
|
||||
// Проверяем VIP статус пользователя
|
||||
const user = await this.profileService.getUserByTelegramId(telegramId);
|
||||
if (!user || !user.isPremium) {
|
||||
if (!user || !user.premium) { // Изменено с isPremium на premium, чтобы соответствовать названию колонки в базе данных
|
||||
const keyboard = {
|
||||
inline_keyboard: [
|
||||
[
|
||||
@@ -1597,10 +1952,18 @@ export class CallbackHandlers {
|
||||
candidateText += '\n📝 ' + (candidate.bio || 'Описание отсутствует') + '\n';
|
||||
|
||||
// Хобби с хэштегами
|
||||
if (candidate.hobbies && candidate.hobbies.trim()) {
|
||||
const hobbiesArray = candidate.hobbies.split(',').map(hobby => hobby.trim()).filter(hobby => hobby);
|
||||
const formattedHobbies = hobbiesArray.map(hobby => '#' + hobby).join(' ');
|
||||
candidateText += '\n🎯 ' + formattedHobbies + '\n';
|
||||
if (candidate.hobbies) {
|
||||
let hobbiesArray: string[] = [];
|
||||
if (typeof candidate.hobbies === 'string') {
|
||||
hobbiesArray = candidate.hobbies.split(',').map(hobby => hobby.trim()).filter(hobby => hobby);
|
||||
} else if (Array.isArray(candidate.hobbies)) {
|
||||
hobbiesArray = (candidate.hobbies as string[]).filter((hobby: string) => hobby && hobby.trim());
|
||||
}
|
||||
|
||||
if (hobbiesArray.length > 0) {
|
||||
const formattedHobbies = hobbiesArray.map(hobby => '#' + hobby).join(' ');
|
||||
candidateText += '\n🎯 ' + formattedHobbies + '\n';
|
||||
}
|
||||
}
|
||||
|
||||
if (candidate.interests.length > 0) {
|
||||
@@ -2112,4 +2475,27 @@ export class CallbackHandlers {
|
||||
await this.bot.sendMessage(chatId, t('translation.error'));
|
||||
}
|
||||
}
|
||||
|
||||
async handleNotificationSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
try {
|
||||
if (this.notificationHandlers) {
|
||||
const userId = await this.profileService.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.sendMessage(chatId, '❌ Вы не зарегистрированы. Используйте команду /start для регистрации.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Вызываем метод из notificationHandlers для получения настроек и отображения меню
|
||||
const settings = await this.notificationHandlers.getNotificationService().getNotificationSettings(userId);
|
||||
await this.notificationHandlers.sendNotificationSettings(chatId, settings);
|
||||
} else {
|
||||
// Если NotificationHandlers недоступен, показываем сообщение об ошибке
|
||||
await this.bot.sendMessage(chatId, '⚙️ Настройки уведомлений временно недоступны. Попробуйте позже.');
|
||||
await this.handleSettings(chatId, telegramId);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Notification settings error:', error);
|
||||
await this.bot.sendMessage(chatId, '❌ Произошла ошибка при загрузке настроек уведомлений. Попробуйте позже.');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
606
src/handlers/callbackHandlers.ts.backup-1758166633763
Normal file
606
src/handlers/callbackHandlers.ts.backup-1758166633763
Normal file
@@ -0,0 +1,606 @@
|
||||
import TelegramBot, { CallbackQuery, InlineKeyboardMarkup } from 'node-telegram-bot-api';
|
||||
import { ProfileService } from '../services/profileService';
|
||||
import { MatchingService } from '../services/matchingService';
|
||||
import { ChatService } from '../services/chatService';
|
||||
import { Profile } from '../models/Profile';
|
||||
import { MessageHandlers } from './messageHandlers';
|
||||
import { ProfileEditController } from '../controllers/profileEditController';
|
||||
import { EnhancedChatHandlers } from './enhancedChatHandlers';
|
||||
import { VipController } from '../controllers/vipController';
|
||||
import { VipService } from '../services/vipService';
|
||||
import { TranslationController } from '../controllers/translationController';
|
||||
import { t } from '../services/localizationService';
|
||||
import { LikeBackHandler } from './likeBackHandler';
|
||||
import { NotificationHandlers } from './notificationHandlers';
|
||||
|
||||
export class CallbackHandlers {
|
||||
private bot: TelegramBot;
|
||||
private profileService: ProfileService;
|
||||
private matchingService: MatchingService;
|
||||
private chatService: ChatService;
|
||||
private messageHandlers: MessageHandlers;
|
||||
private profileEditController: ProfileEditController;
|
||||
private enhancedChatHandlers: EnhancedChatHandlers;
|
||||
private vipController: VipController;
|
||||
private vipService: VipService;
|
||||
private translationController: TranslationController;
|
||||
private likeBackHandler: LikeBackHandler;
|
||||
private notificationHandlers?: NotificationHandlers;
|
||||
|
||||
constructor(bot: TelegramBot, messageHandlers: MessageHandlers) {
|
||||
this.bot = bot;
|
||||
this.profileService = new ProfileService();
|
||||
this.matchingService = new MatchingService();
|
||||
this.chatService = new ChatService();
|
||||
this.messageHandlers = messageHandlers;
|
||||
this.profileEditController = new ProfileEditController(this.profileService);
|
||||
this.enhancedChatHandlers = new EnhancedChatHandlers(bot);
|
||||
this.vipController = new VipController(bot);
|
||||
this.vipService = new VipService();
|
||||
this.translationController = new TranslationController();
|
||||
this.likeBackHandler = new LikeBackHandler(bot);
|
||||
|
||||
// Создаем экземпляр NotificationHandlers
|
||||
try {
|
||||
this.notificationHandlers = new NotificationHandlers(bot);
|
||||
} catch (error) {
|
||||
console.error('Failed to initialize NotificationHandlers:', error);
|
||||
}
|
||||
}
|
||||
|
||||
register(): void {
|
||||
this.bot.on('callback_query', (query) => this.handleCallback(query));
|
||||
}
|
||||
|
||||
async handleCallback(query: CallbackQuery): Promise<void> {
|
||||
if (!query.data || !query.from || !query.message) return;
|
||||
|
||||
const telegramId = query.from.id.toString();
|
||||
const chatId = query.message.chat.id;
|
||||
const data = query.data;
|
||||
|
||||
try {
|
||||
// Основные действия профиля
|
||||
if (data === 'create_profile') {
|
||||
await this.handleCreateProfile(chatId, telegramId);
|
||||
} else if (data.startsWith('gender_')) {
|
||||
const gender = data.replace('gender_', '');
|
||||
await this.handleGenderSelection(chatId, telegramId, gender);
|
||||
} else if (data === 'view_my_profile') {
|
||||
await this.handleViewMyProfile(chatId, telegramId);
|
||||
} else if (data === 'edit_profile') {
|
||||
await this.handleEditProfile(chatId, telegramId);
|
||||
} else if (data === 'manage_photos') {
|
||||
await this.handleManagePhotos(chatId, telegramId);
|
||||
} else if (data === 'preview_profile') {
|
||||
await this.handlePreviewProfile(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Редактирование полей профиля
|
||||
else if (data === 'edit_name') {
|
||||
await this.handleEditName(chatId, telegramId);
|
||||
} else if (data === 'edit_age') {
|
||||
await this.handleEditAge(chatId, telegramId);
|
||||
} else if (data === 'edit_bio') {
|
||||
await this.handleEditBio(chatId, telegramId);
|
||||
} else if (data === 'edit_hobbies') {
|
||||
await this.handleEditHobbies(chatId, telegramId);
|
||||
} else if (data === 'edit_city') {
|
||||
await this.handleEditCity(chatId, telegramId);
|
||||
} else if (data === 'edit_job') {
|
||||
await this.handleEditJob(chatId, telegramId);
|
||||
} else if (data === 'edit_education') {
|
||||
await this.handleEditEducation(chatId, telegramId);
|
||||
} else if (data === 'edit_height') {
|
||||
await this.handleEditHeight(chatId, telegramId);
|
||||
} else if (data === 'edit_religion') {
|
||||
await this.handleEditReligion(chatId, telegramId);
|
||||
} else if (data === 'edit_dating_goal') {
|
||||
await this.handleEditDatingGoal(chatId, telegramId);
|
||||
} else if (data === 'edit_lifestyle') {
|
||||
await this.handleEditLifestyle(chatId, telegramId);
|
||||
} else if (data === 'edit_search_preferences') {
|
||||
await this.handleEditSearchPreferences(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Управление фотографиями
|
||||
else if (data === 'add_photo') {
|
||||
await this.handleAddPhoto(chatId, telegramId);
|
||||
} else if (data === 'delete_photo') {
|
||||
await this.handleDeletePhoto(chatId, telegramId);
|
||||
} else if (data === 'set_main_photo') {
|
||||
await this.handleSetMainPhoto(chatId, telegramId);
|
||||
} else if (data.startsWith('delete_photo_')) {
|
||||
const photoIndex = parseInt(data.replace('delete_photo_', ''));
|
||||
await this.handleDeletePhotoByIndex(chatId, telegramId, photoIndex);
|
||||
} else if (data.startsWith('set_main_photo_')) {
|
||||
const photoIndex = parseInt(data.replace('set_main_photo_', ''));
|
||||
await this.handleSetMainPhotoByIndex(chatId, telegramId, photoIndex);
|
||||
}
|
||||
|
||||
// Цели знакомства
|
||||
else if (data.startsWith('set_dating_goal_')) {
|
||||
const goal = data.replace('set_dating_goal_', '');
|
||||
await this.handleSetDatingGoal(chatId, telegramId, goal);
|
||||
}
|
||||
|
||||
// Образ жизни
|
||||
else if (data === 'edit_smoking') {
|
||||
await this.handleEditSmoking(chatId, telegramId);
|
||||
} else if (data === 'edit_drinking') {
|
||||
await this.handleEditDrinking(chatId, telegramId);
|
||||
} else if (data === 'edit_kids') {
|
||||
await this.handleEditKids(chatId, telegramId);
|
||||
} else if (data.startsWith('set_smoking_')) {
|
||||
const value = data.replace('set_smoking_', '');
|
||||
await this.handleSetLifestyle(chatId, telegramId, 'smoking', value);
|
||||
} else if (data.startsWith('set_drinking_')) {
|
||||
const value = data.replace('set_drinking_', '');
|
||||
await this.handleSetLifestyle(chatId, telegramId, 'drinking', value);
|
||||
} else if (data.startsWith('set_kids_')) {
|
||||
const value = data.replace('set_kids_', '');
|
||||
await this.handleSetLifestyle(chatId, telegramId, 'kids', value);
|
||||
}
|
||||
|
||||
// Настройки поиска
|
||||
else if (data === 'edit_age_range') {
|
||||
await this.handleEditAgeRange(chatId, telegramId);
|
||||
} else if (data === 'edit_distance') {
|
||||
await this.handleEditDistance(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Просмотр анкет и свайпы
|
||||
else if (data === 'start_browsing') {
|
||||
await this.handleStartBrowsing(chatId, telegramId, false);
|
||||
} else if (data === 'start_browsing_first') {
|
||||
// Показываем всех пользователей для нового пользователя
|
||||
await this.handleStartBrowsing(chatId, telegramId, true);
|
||||
} else if (data === 'vip_search') {
|
||||
await this.handleVipSearch(chatId, telegramId);
|
||||
} else if (data.startsWith('search_by_goal_')) {
|
||||
const goal = data.replace('search_by_goal_', '');
|
||||
await this.handleSearchByGoal(chatId, telegramId, goal);
|
||||
} else if (data === 'next_candidate') {
|
||||
await this.handleNextCandidate(chatId, telegramId);
|
||||
} else if (data.startsWith('like_')) {
|
||||
const targetUserId = data.replace('like_', '');
|
||||
await this.handleLike(chatId, telegramId, targetUserId);
|
||||
} else if (data.startsWith('dislike_')) {
|
||||
const targetUserId = data.replace('dislike_', '');
|
||||
await this.handleDislike(chatId, telegramId, targetUserId);
|
||||
} else if (data.startsWith('superlike_')) {
|
||||
const targetUserId = data.replace('superlike_', '');
|
||||
await this.handleSuperlike(chatId, telegramId, targetUserId);
|
||||
} else if (data.startsWith('view_profile_')) {
|
||||
const targetUserId = data.replace('view_profile_', '');
|
||||
await this.handleViewProfile(chatId, telegramId, targetUserId);
|
||||
} else if (data.startsWith('more_photos_')) {
|
||||
const targetUserId = data.replace('more_photos_', '');
|
||||
await this.handleMorePhotos(chatId, telegramId, targetUserId);
|
||||
}
|
||||
|
||||
// Обработка лайков и ответных лайков из уведомлений
|
||||
else if (data.startsWith('like_back:')) {
|
||||
const targetUserId = data.replace('like_back:', '');
|
||||
await this.likeBackHandler.handleLikeBack(chatId, telegramId, targetUserId);
|
||||
}
|
||||
|
||||
// Матчи и чаты
|
||||
else if (data === 'view_matches') {
|
||||
await this.handleViewMatches(chatId, telegramId);
|
||||
} else if (data === 'open_chats') {
|
||||
await this.handleOpenChats(chatId, telegramId);
|
||||
} else if (data === 'native_chats') {
|
||||
await this.enhancedChatHandlers.showChatsNative(chatId, telegramId);
|
||||
} else if (data.startsWith('open_native_chat_')) {
|
||||
const matchId = data.replace('open_native_chat_', '');
|
||||
await this.enhancedChatHandlers.openNativeChat(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('chat_history_')) {
|
||||
const matchId = data.replace('chat_history_', '');
|
||||
await this.enhancedChatHandlers.showChatHistory(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('chat_')) {
|
||||
const matchId = data.replace('chat_', '');
|
||||
await this.handleOpenChat(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('send_message_')) {
|
||||
const matchId = data.replace('send_message_', '');
|
||||
await this.handleSendMessage(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('view_chat_profile_')) {
|
||||
const matchId = data.replace('view_chat_profile_', '');
|
||||
await this.handleViewChatProfile(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('unmatch_')) {
|
||||
const matchId = data.replace('unmatch_', '');
|
||||
await this.handleUnmatch(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('confirm_unmatch_')) {
|
||||
const matchId = data.replace('confirm_unmatch_', '');
|
||||
await this.handleConfirmUnmatch(chatId, telegramId, matchId);
|
||||
}
|
||||
|
||||
// Настройки
|
||||
else if (data === 'settings') {
|
||||
await this.handleSettings(chatId, telegramId);
|
||||
} else if (data === 'search_settings') {
|
||||
await this.handleSearchSettings(chatId, telegramId);
|
||||
} else if (data === 'notification_settings') {
|
||||
await this.handleNotificationSettings(chatId, telegramId);
|
||||
} else if (data === 'view_stats') {
|
||||
await this.handleViewStats(chatId, telegramId);
|
||||
} else if (data === 'view_profile_viewers') {
|
||||
await this.handleViewProfileViewers(chatId, telegramId);
|
||||
} else if (data === 'hide_profile') {
|
||||
await this.handleHideProfile(chatId, telegramId);
|
||||
} else if (data === 'delete_profile') {
|
||||
await this.handleDeleteProfile(chatId, telegramId);
|
||||
} else if (data === 'main_menu') {
|
||||
await this.handleMainMenu(chatId, telegramId);
|
||||
} else if (data === 'confirm_delete_profile') {
|
||||
await this.handleConfirmDeleteProfile(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Информация
|
||||
else if (data === 'how_it_works') {
|
||||
await this.handleHowItWorks(chatId);
|
||||
} else if (data === 'back_to_browsing') {
|
||||
await this.handleStartBrowsing(chatId, telegramId);
|
||||
} else if (data === 'get_vip') {
|
||||
await this.vipController.showVipSearch(chatId, telegramId);
|
||||
}
|
||||
|
||||
// VIP функции
|
||||
else if (data === 'vip_search') {
|
||||
await this.vipController.showVipSearch(chatId, telegramId);
|
||||
} else if (data === 'vip_quick_search') {
|
||||
await this.vipController.performQuickVipSearch(chatId, telegramId);
|
||||
} else if (data === 'vip_advanced_search') {
|
||||
await this.vipController.startAdvancedSearch(chatId, telegramId);
|
||||
} else if (data === 'vip_dating_goal_search') {
|
||||
await this.vipController.showDatingGoalSearch(chatId, telegramId);
|
||||
} else if (data.startsWith('vip_goal_')) {
|
||||
const goal = data.replace('vip_goal_', '');
|
||||
await this.vipController.performDatingGoalSearch(chatId, telegramId, goal);
|
||||
} else if (data.startsWith('vip_like_')) {
|
||||
const targetTelegramId = data.replace('vip_like_', '');
|
||||
await this.handleVipLike(chatId, telegramId, targetTelegramId);
|
||||
} else if (data.startsWith('vip_superlike_')) {
|
||||
const targetTelegramId = data.replace('vip_superlike_', '');
|
||||
await this.handleVipSuperlike(chatId, telegramId, targetTelegramId);
|
||||
} else if (data.startsWith('vip_dislike_')) {
|
||||
const targetTelegramId = data.replace('vip_dislike_', '');
|
||||
await this.handleVipDislike(chatId, telegramId, targetTelegramId);
|
||||
}
|
||||
|
||||
// Настройки языка и переводы
|
||||
else if (data === 'language_settings') {
|
||||
await this.handleLanguageSettings(chatId, telegramId);
|
||||
} else if (data.startsWith('set_language_')) {
|
||||
const languageCode = data.replace('set_language_', '');
|
||||
await this.handleSetLanguage(chatId, telegramId, languageCode);
|
||||
} else if (data.startsWith('translate_profile_')) {
|
||||
const profileUserId = parseInt(data.replace('translate_profile_', ''));
|
||||
await this.handleTranslateProfile(chatId, telegramId, profileUserId);
|
||||
} else if (data === 'back_to_settings') {
|
||||
await this.handleSettings(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Настройки уведомлений
|
||||
else if (data === 'notifications') {
|
||||
if (this.notificationHandlers) {
|
||||
const userId = await this.profileService.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.answerCallbackQuery(query.id, { text: '❌ Вы не зарегистрированы.' });
|
||||
return;
|
||||
}
|
||||
|
||||
const settings = await this.notificationHandlers.getNotificationService().getNotificationSettings(userId);
|
||||
await this.notificationHandlers.sendNotificationSettings(chatId, settings);
|
||||
} else {
|
||||
await this.handleNotificationSettings(chatId, telegramId);
|
||||
}
|
||||
}
|
||||
// Обработка переключения настроек уведомлений
|
||||
else if (data.startsWith('notif_toggle:') ||
|
||||
data === 'notif_time' ||
|
||||
data.startsWith('notif_time_set:') ||
|
||||
data === 'notif_dnd' ||
|
||||
data.startsWith('notif_dnd_set:') ||
|
||||
data === 'notif_dnd_time' ||
|
||||
data.startsWith('notif_dnd_time_set:') ||
|
||||
data === 'notif_dnd_time_custom') {
|
||||
// Делегируем обработку в NotificationHandlers, если он доступен
|
||||
if (this.notificationHandlers) {
|
||||
// Эти коллбэки уже обрабатываются в NotificationHandlers, поэтому здесь ничего не делаем
|
||||
// NotificationHandlers уже зарегистрировал свои обработчики в register()
|
||||
} else {
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: 'Функция настройки уведомлений недоступна.',
|
||||
show_alert: true
|
||||
});
|
||||
}
|
||||
}
|
||||
else {
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: 'Функция в разработке!',
|
||||
show_alert: false
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
await this.bot.answerCallbackQuery(query.id);
|
||||
|
||||
} catch (error) {
|
||||
console.error('Callback handler error:', error);
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: 'Произошла ошибка. Попробуйте еще раз.',
|
||||
show_alert: true
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
// Добавим все необходимые методы для обработки коллбэков
|
||||
async handleCreateProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleGenderSelection(chatId: number, telegramId: string, gender: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewMyProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleManagePhotos(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handlePreviewProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditName(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditAge(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditBio(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditHobbies(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditCity(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditJob(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditEducation(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditHeight(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditReligion(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditDatingGoal(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditLifestyle(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditSearchPreferences(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleAddPhoto(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleDeletePhoto(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetMainPhoto(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleDeletePhotoByIndex(chatId: number, telegramId: string, photoIndex: number): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetMainPhotoByIndex(chatId: number, telegramId: string, photoIndex: number): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetDatingGoal(chatId: number, telegramId: string, goal: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditSmoking(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditDrinking(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditKids(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetLifestyle(chatId: number, telegramId: string, type: string, value: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditAgeRange(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditDistance(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleStartBrowsing(chatId: number, telegramId: string, showAll: boolean = false): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleVipSearch(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSearchByGoal(chatId: number, telegramId: string, goal: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleNextCandidate(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleLike(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleDislike(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSuperlike(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewProfile(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleMorePhotos(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewMatches(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleOpenChats(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleOpenChat(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSendMessage(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewChatProfile(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleUnmatch(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleConfirmUnmatch(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSearchSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewStats(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewProfileViewers(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleHideProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleDeleteProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleMainMenu(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleConfirmDeleteProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleHowItWorks(chatId: number): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleVipLike(chatId: number, telegramId: string, targetTelegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleVipSuperlike(chatId: number, telegramId: string, targetTelegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleVipDislike(chatId: number, telegramId: string, targetTelegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleLanguageSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetLanguage(chatId: number, telegramId: string, languageCode: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleTranslateProfile(chatId: number, telegramId: string, profileUserId: number): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
// Добавим новый метод для настроек уведомлений (вызывает NotificationHandlers)
|
||||
async handleNotificationSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
try {
|
||||
if (this.notificationHandlers) {
|
||||
const userId = await this.profileService.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.sendMessage(chatId, '❌ Вы не зарегистрированы. Используйте команду /start для регистрации.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Вызываем метод из notificationHandlers для получения настроек и отображения меню
|
||||
const settings = await this.notificationHandlers.getNotificationService().getNotificationSettings(userId);
|
||||
await this.notificationHandlers.sendNotificationSettings(chatId, settings);
|
||||
} else {
|
||||
// Если NotificationHandlers недоступен, показываем сообщение об ошибке
|
||||
await this.bot.sendMessage(chatId, '⚙️ Настройки уведомлений временно недоступны. Попробуйте позже.');
|
||||
await this.handleSettings(chatId, telegramId);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error handling notification settings:', error);
|
||||
await this.bot.sendMessage(chatId, '❌ Произошла ошибка при загрузке настроек уведомлений.');
|
||||
}
|
||||
}
|
||||
}
|
||||
BIN
src/handlers/callbackHandlers.ts.original
Normal file
BIN
src/handlers/callbackHandlers.ts.original
Normal file
Binary file not shown.
606
src/handlers/callbackHandlers.ts.stub
Normal file
606
src/handlers/callbackHandlers.ts.stub
Normal file
@@ -0,0 +1,606 @@
|
||||
import TelegramBot, { CallbackQuery, InlineKeyboardMarkup } from 'node-telegram-bot-api';
|
||||
import { ProfileService } from '../services/profileService';
|
||||
import { MatchingService } from '../services/matchingService';
|
||||
import { ChatService } from '../services/chatService';
|
||||
import { Profile } from '../models/Profile';
|
||||
import { MessageHandlers } from './messageHandlers';
|
||||
import { ProfileEditController } from '../controllers/profileEditController';
|
||||
import { EnhancedChatHandlers } from './enhancedChatHandlers';
|
||||
import { VipController } from '../controllers/vipController';
|
||||
import { VipService } from '../services/vipService';
|
||||
import { TranslationController } from '../controllers/translationController';
|
||||
import { t } from '../services/localizationService';
|
||||
import { LikeBackHandler } from './likeBackHandler';
|
||||
import { NotificationHandlers } from './notificationHandlers';
|
||||
|
||||
export class CallbackHandlers {
|
||||
private bot: TelegramBot;
|
||||
private profileService: ProfileService;
|
||||
private matchingService: MatchingService;
|
||||
private chatService: ChatService;
|
||||
private messageHandlers: MessageHandlers;
|
||||
private profileEditController: ProfileEditController;
|
||||
private enhancedChatHandlers: EnhancedChatHandlers;
|
||||
private vipController: VipController;
|
||||
private vipService: VipService;
|
||||
private translationController: TranslationController;
|
||||
private likeBackHandler: LikeBackHandler;
|
||||
private notificationHandlers?: NotificationHandlers;
|
||||
|
||||
constructor(bot: TelegramBot, messageHandlers: MessageHandlers) {
|
||||
this.bot = bot;
|
||||
this.profileService = new ProfileService();
|
||||
this.matchingService = new MatchingService();
|
||||
this.chatService = new ChatService();
|
||||
this.messageHandlers = messageHandlers;
|
||||
this.profileEditController = new ProfileEditController(this.profileService);
|
||||
this.enhancedChatHandlers = new EnhancedChatHandlers(bot);
|
||||
this.vipController = new VipController(bot);
|
||||
this.vipService = new VipService();
|
||||
this.translationController = new TranslationController();
|
||||
this.likeBackHandler = new LikeBackHandler(bot);
|
||||
|
||||
// Создаем экземпляр NotificationHandlers
|
||||
try {
|
||||
this.notificationHandlers = new NotificationHandlers(bot);
|
||||
} catch (error) {
|
||||
console.error('Failed to initialize NotificationHandlers:', error);
|
||||
}
|
||||
}
|
||||
|
||||
register(): void {
|
||||
this.bot.on('callback_query', (query) => this.handleCallback(query));
|
||||
}
|
||||
|
||||
async handleCallback(query: CallbackQuery): Promise<void> {
|
||||
if (!query.data || !query.from || !query.message) return;
|
||||
|
||||
const telegramId = query.from.id.toString();
|
||||
const chatId = query.message.chat.id;
|
||||
const data = query.data;
|
||||
|
||||
try {
|
||||
// Основные действия профиля
|
||||
if (data === 'create_profile') {
|
||||
await this.handleCreateProfile(chatId, telegramId);
|
||||
} else if (data.startsWith('gender_')) {
|
||||
const gender = data.replace('gender_', '');
|
||||
await this.handleGenderSelection(chatId, telegramId, gender);
|
||||
} else if (data === 'view_my_profile') {
|
||||
await this.handleViewMyProfile(chatId, telegramId);
|
||||
} else if (data === 'edit_profile') {
|
||||
await this.handleEditProfile(chatId, telegramId);
|
||||
} else if (data === 'manage_photos') {
|
||||
await this.handleManagePhotos(chatId, telegramId);
|
||||
} else if (data === 'preview_profile') {
|
||||
await this.handlePreviewProfile(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Редактирование полей профиля
|
||||
else if (data === 'edit_name') {
|
||||
await this.handleEditName(chatId, telegramId);
|
||||
} else if (data === 'edit_age') {
|
||||
await this.handleEditAge(chatId, telegramId);
|
||||
} else if (data === 'edit_bio') {
|
||||
await this.handleEditBio(chatId, telegramId);
|
||||
} else if (data === 'edit_hobbies') {
|
||||
await this.handleEditHobbies(chatId, telegramId);
|
||||
} else if (data === 'edit_city') {
|
||||
await this.handleEditCity(chatId, telegramId);
|
||||
} else if (data === 'edit_job') {
|
||||
await this.handleEditJob(chatId, telegramId);
|
||||
} else if (data === 'edit_education') {
|
||||
await this.handleEditEducation(chatId, telegramId);
|
||||
} else if (data === 'edit_height') {
|
||||
await this.handleEditHeight(chatId, telegramId);
|
||||
} else if (data === 'edit_religion') {
|
||||
await this.handleEditReligion(chatId, telegramId);
|
||||
} else if (data === 'edit_dating_goal') {
|
||||
await this.handleEditDatingGoal(chatId, telegramId);
|
||||
} else if (data === 'edit_lifestyle') {
|
||||
await this.handleEditLifestyle(chatId, telegramId);
|
||||
} else if (data === 'edit_search_preferences') {
|
||||
await this.handleEditSearchPreferences(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Управление фотографиями
|
||||
else if (data === 'add_photo') {
|
||||
await this.handleAddPhoto(chatId, telegramId);
|
||||
} else if (data === 'delete_photo') {
|
||||
await this.handleDeletePhoto(chatId, telegramId);
|
||||
} else if (data === 'set_main_photo') {
|
||||
await this.handleSetMainPhoto(chatId, telegramId);
|
||||
} else if (data.startsWith('delete_photo_')) {
|
||||
const photoIndex = parseInt(data.replace('delete_photo_', ''));
|
||||
await this.handleDeletePhotoByIndex(chatId, telegramId, photoIndex);
|
||||
} else if (data.startsWith('set_main_photo_')) {
|
||||
const photoIndex = parseInt(data.replace('set_main_photo_', ''));
|
||||
await this.handleSetMainPhotoByIndex(chatId, telegramId, photoIndex);
|
||||
}
|
||||
|
||||
// Цели знакомства
|
||||
else if (data.startsWith('set_dating_goal_')) {
|
||||
const goal = data.replace('set_dating_goal_', '');
|
||||
await this.handleSetDatingGoal(chatId, telegramId, goal);
|
||||
}
|
||||
|
||||
// Образ жизни
|
||||
else if (data === 'edit_smoking') {
|
||||
await this.handleEditSmoking(chatId, telegramId);
|
||||
} else if (data === 'edit_drinking') {
|
||||
await this.handleEditDrinking(chatId, telegramId);
|
||||
} else if (data === 'edit_kids') {
|
||||
await this.handleEditKids(chatId, telegramId);
|
||||
} else if (data.startsWith('set_smoking_')) {
|
||||
const value = data.replace('set_smoking_', '');
|
||||
await this.handleSetLifestyle(chatId, telegramId, 'smoking', value);
|
||||
} else if (data.startsWith('set_drinking_')) {
|
||||
const value = data.replace('set_drinking_', '');
|
||||
await this.handleSetLifestyle(chatId, telegramId, 'drinking', value);
|
||||
} else if (data.startsWith('set_kids_')) {
|
||||
const value = data.replace('set_kids_', '');
|
||||
await this.handleSetLifestyle(chatId, telegramId, 'kids', value);
|
||||
}
|
||||
|
||||
// Настройки поиска
|
||||
else if (data === 'edit_age_range') {
|
||||
await this.handleEditAgeRange(chatId, telegramId);
|
||||
} else if (data === 'edit_distance') {
|
||||
await this.handleEditDistance(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Просмотр анкет и свайпы
|
||||
else if (data === 'start_browsing') {
|
||||
await this.handleStartBrowsing(chatId, telegramId, false);
|
||||
} else if (data === 'start_browsing_first') {
|
||||
// Показываем всех пользователей для нового пользователя
|
||||
await this.handleStartBrowsing(chatId, telegramId, true);
|
||||
} else if (data === 'vip_search') {
|
||||
await this.handleVipSearch(chatId, telegramId);
|
||||
} else if (data.startsWith('search_by_goal_')) {
|
||||
const goal = data.replace('search_by_goal_', '');
|
||||
await this.handleSearchByGoal(chatId, telegramId, goal);
|
||||
} else if (data === 'next_candidate') {
|
||||
await this.handleNextCandidate(chatId, telegramId);
|
||||
} else if (data.startsWith('like_')) {
|
||||
const targetUserId = data.replace('like_', '');
|
||||
await this.handleLike(chatId, telegramId, targetUserId);
|
||||
} else if (data.startsWith('dislike_')) {
|
||||
const targetUserId = data.replace('dislike_', '');
|
||||
await this.handleDislike(chatId, telegramId, targetUserId);
|
||||
} else if (data.startsWith('superlike_')) {
|
||||
const targetUserId = data.replace('superlike_', '');
|
||||
await this.handleSuperlike(chatId, telegramId, targetUserId);
|
||||
} else if (data.startsWith('view_profile_')) {
|
||||
const targetUserId = data.replace('view_profile_', '');
|
||||
await this.handleViewProfile(chatId, telegramId, targetUserId);
|
||||
} else if (data.startsWith('more_photos_')) {
|
||||
const targetUserId = data.replace('more_photos_', '');
|
||||
await this.handleMorePhotos(chatId, telegramId, targetUserId);
|
||||
}
|
||||
|
||||
// Обработка лайков и ответных лайков из уведомлений
|
||||
else if (data.startsWith('like_back:')) {
|
||||
const targetUserId = data.replace('like_back:', '');
|
||||
await this.likeBackHandler.handleLikeBack(chatId, telegramId, targetUserId);
|
||||
}
|
||||
|
||||
// Матчи и чаты
|
||||
else if (data === 'view_matches') {
|
||||
await this.handleViewMatches(chatId, telegramId);
|
||||
} else if (data === 'open_chats') {
|
||||
await this.handleOpenChats(chatId, telegramId);
|
||||
} else if (data === 'native_chats') {
|
||||
await this.enhancedChatHandlers.showChatsNative(chatId, telegramId);
|
||||
} else if (data.startsWith('open_native_chat_')) {
|
||||
const matchId = data.replace('open_native_chat_', '');
|
||||
await this.enhancedChatHandlers.openNativeChat(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('chat_history_')) {
|
||||
const matchId = data.replace('chat_history_', '');
|
||||
await this.enhancedChatHandlers.showChatHistory(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('chat_')) {
|
||||
const matchId = data.replace('chat_', '');
|
||||
await this.handleOpenChat(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('send_message_')) {
|
||||
const matchId = data.replace('send_message_', '');
|
||||
await this.handleSendMessage(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('view_chat_profile_')) {
|
||||
const matchId = data.replace('view_chat_profile_', '');
|
||||
await this.handleViewChatProfile(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('unmatch_')) {
|
||||
const matchId = data.replace('unmatch_', '');
|
||||
await this.handleUnmatch(chatId, telegramId, matchId);
|
||||
} else if (data.startsWith('confirm_unmatch_')) {
|
||||
const matchId = data.replace('confirm_unmatch_', '');
|
||||
await this.handleConfirmUnmatch(chatId, telegramId, matchId);
|
||||
}
|
||||
|
||||
// Настройки
|
||||
else if (data === 'settings') {
|
||||
await this.handleSettings(chatId, telegramId);
|
||||
} else if (data === 'search_settings') {
|
||||
await this.handleSearchSettings(chatId, telegramId);
|
||||
} else if (data === 'notification_settings') {
|
||||
await this.handleNotificationSettings(chatId, telegramId);
|
||||
} else if (data === 'view_stats') {
|
||||
await this.handleViewStats(chatId, telegramId);
|
||||
} else if (data === 'view_profile_viewers') {
|
||||
await this.handleViewProfileViewers(chatId, telegramId);
|
||||
} else if (data === 'hide_profile') {
|
||||
await this.handleHideProfile(chatId, telegramId);
|
||||
} else if (data === 'delete_profile') {
|
||||
await this.handleDeleteProfile(chatId, telegramId);
|
||||
} else if (data === 'main_menu') {
|
||||
await this.handleMainMenu(chatId, telegramId);
|
||||
} else if (data === 'confirm_delete_profile') {
|
||||
await this.handleConfirmDeleteProfile(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Информация
|
||||
else if (data === 'how_it_works') {
|
||||
await this.handleHowItWorks(chatId);
|
||||
} else if (data === 'back_to_browsing') {
|
||||
await this.handleStartBrowsing(chatId, telegramId);
|
||||
} else if (data === 'get_vip') {
|
||||
await this.vipController.showVipSearch(chatId, telegramId);
|
||||
}
|
||||
|
||||
// VIP функции
|
||||
else if (data === 'vip_search') {
|
||||
await this.vipController.showVipSearch(chatId, telegramId);
|
||||
} else if (data === 'vip_quick_search') {
|
||||
await this.vipController.performQuickVipSearch(chatId, telegramId);
|
||||
} else if (data === 'vip_advanced_search') {
|
||||
await this.vipController.startAdvancedSearch(chatId, telegramId);
|
||||
} else if (data === 'vip_dating_goal_search') {
|
||||
await this.vipController.showDatingGoalSearch(chatId, telegramId);
|
||||
} else if (data.startsWith('vip_goal_')) {
|
||||
const goal = data.replace('vip_goal_', '');
|
||||
await this.vipController.performDatingGoalSearch(chatId, telegramId, goal);
|
||||
} else if (data.startsWith('vip_like_')) {
|
||||
const targetTelegramId = data.replace('vip_like_', '');
|
||||
await this.handleVipLike(chatId, telegramId, targetTelegramId);
|
||||
} else if (data.startsWith('vip_superlike_')) {
|
||||
const targetTelegramId = data.replace('vip_superlike_', '');
|
||||
await this.handleVipSuperlike(chatId, telegramId, targetTelegramId);
|
||||
} else if (data.startsWith('vip_dislike_')) {
|
||||
const targetTelegramId = data.replace('vip_dislike_', '');
|
||||
await this.handleVipDislike(chatId, telegramId, targetTelegramId);
|
||||
}
|
||||
|
||||
// Настройки языка и переводы
|
||||
else if (data === 'language_settings') {
|
||||
await this.handleLanguageSettings(chatId, telegramId);
|
||||
} else if (data.startsWith('set_language_')) {
|
||||
const languageCode = data.replace('set_language_', '');
|
||||
await this.handleSetLanguage(chatId, telegramId, languageCode);
|
||||
} else if (data.startsWith('translate_profile_')) {
|
||||
const profileUserId = parseInt(data.replace('translate_profile_', ''));
|
||||
await this.handleTranslateProfile(chatId, telegramId, profileUserId);
|
||||
} else if (data === 'back_to_settings') {
|
||||
await this.handleSettings(chatId, telegramId);
|
||||
}
|
||||
|
||||
// Настройки уведомлений
|
||||
else if (data === 'notifications') {
|
||||
if (this.notificationHandlers) {
|
||||
const userId = await this.profileService.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.answerCallbackQuery(query.id, { text: '❌ Вы не зарегистрированы.' });
|
||||
return;
|
||||
}
|
||||
|
||||
const settings = await this.notificationHandlers.getNotificationService().getNotificationSettings(userId);
|
||||
await this.notificationHandlers.sendNotificationSettings(chatId, settings);
|
||||
} else {
|
||||
await this.handleNotificationSettings(chatId, telegramId);
|
||||
}
|
||||
}
|
||||
// Обработка переключения настроек уведомлений
|
||||
else if (data.startsWith('notif_toggle:') ||
|
||||
data === 'notif_time' ||
|
||||
data.startsWith('notif_time_set:') ||
|
||||
data === 'notif_dnd' ||
|
||||
data.startsWith('notif_dnd_set:') ||
|
||||
data === 'notif_dnd_time' ||
|
||||
data.startsWith('notif_dnd_time_set:') ||
|
||||
data === 'notif_dnd_time_custom') {
|
||||
// Делегируем обработку в NotificationHandlers, если он доступен
|
||||
if (this.notificationHandlers) {
|
||||
// Эти коллбэки уже обрабатываются в NotificationHandlers, поэтому здесь ничего не делаем
|
||||
// NotificationHandlers уже зарегистрировал свои обработчики в register()
|
||||
} else {
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: 'Функция настройки уведомлений недоступна.',
|
||||
show_alert: true
|
||||
});
|
||||
}
|
||||
}
|
||||
else {
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: 'Функция в разработке!',
|
||||
show_alert: false
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
await this.bot.answerCallbackQuery(query.id);
|
||||
|
||||
} catch (error) {
|
||||
console.error('Callback handler error:', error);
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: 'Произошла ошибка. Попробуйте еще раз.',
|
||||
show_alert: true
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
// Добавим все необходимые методы для обработки коллбэков
|
||||
async handleCreateProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleGenderSelection(chatId: number, telegramId: string, gender: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewMyProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleManagePhotos(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handlePreviewProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditName(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditAge(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditBio(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditHobbies(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditCity(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditJob(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditEducation(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditHeight(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditReligion(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditDatingGoal(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditLifestyle(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditSearchPreferences(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleAddPhoto(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleDeletePhoto(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetMainPhoto(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleDeletePhotoByIndex(chatId: number, telegramId: string, photoIndex: number): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetMainPhotoByIndex(chatId: number, telegramId: string, photoIndex: number): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetDatingGoal(chatId: number, telegramId: string, goal: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditSmoking(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditDrinking(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditKids(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetLifestyle(chatId: number, telegramId: string, type: string, value: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditAgeRange(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleEditDistance(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleStartBrowsing(chatId: number, telegramId: string, showAll: boolean = false): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleVipSearch(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSearchByGoal(chatId: number, telegramId: string, goal: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleNextCandidate(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleLike(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleDislike(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSuperlike(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewProfile(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleMorePhotos(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewMatches(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleOpenChats(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleOpenChat(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSendMessage(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewChatProfile(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleUnmatch(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleConfirmUnmatch(chatId: number, telegramId: string, matchId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSearchSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewStats(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleViewProfileViewers(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleHideProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleDeleteProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleMainMenu(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleConfirmDeleteProfile(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleHowItWorks(chatId: number): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleVipLike(chatId: number, telegramId: string, targetTelegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleVipSuperlike(chatId: number, telegramId: string, targetTelegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleVipDislike(chatId: number, telegramId: string, targetTelegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleLanguageSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleSetLanguage(chatId: number, telegramId: string, languageCode: string): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
async handleTranslateProfile(chatId: number, telegramId: string, profileUserId: number): Promise<void> {
|
||||
// Заглушка метода
|
||||
}
|
||||
|
||||
// Добавим новый метод для настроек уведомлений (вызывает NotificationHandlers)
|
||||
async handleNotificationSettings(chatId: number, telegramId: string): Promise<void> {
|
||||
try {
|
||||
if (this.notificationHandlers) {
|
||||
const userId = await this.profileService.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.sendMessage(chatId, '❌ Вы не зарегистрированы. Используйте команду /start для регистрации.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Вызываем метод из notificationHandlers для получения настроек и отображения меню
|
||||
const settings = await this.notificationHandlers.getNotificationService().getNotificationSettings(userId);
|
||||
await this.notificationHandlers.sendNotificationSettings(chatId, settings);
|
||||
} else {
|
||||
// Если NotificationHandlers недоступен, показываем сообщение об ошибке
|
||||
await this.bot.sendMessage(chatId, '⚙️ Настройки уведомлений временно недоступны. Попробуйте позже.');
|
||||
await this.handleSettings(chatId, telegramId);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error handling notification settings:', error);
|
||||
await this.bot.sendMessage(chatId, '❌ Произошла ошибка при загрузке настроек уведомлений.');
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -3,16 +3,22 @@ import { ProfileService } from '../services/profileService';
|
||||
import { MatchingService } from '../services/matchingService';
|
||||
import { Profile } from '../models/Profile';
|
||||
import { getUserTranslation } from '../services/localizationService';
|
||||
import { NotificationHandlers } from './notificationHandlers';
|
||||
import { LanguageHandlers } from './languageHandlers';
|
||||
|
||||
export class CommandHandlers {
|
||||
private bot: TelegramBot;
|
||||
private profileService: ProfileService;
|
||||
private languageHandlers: LanguageHandlers;
|
||||
private matchingService: MatchingService;
|
||||
private notificationHandlers: NotificationHandlers;
|
||||
|
||||
constructor(bot: TelegramBot) {
|
||||
this.bot = bot;
|
||||
this.profileService = new ProfileService();
|
||||
this.matchingService = new MatchingService();
|
||||
this.notificationHandlers = new NotificationHandlers(bot);
|
||||
this.languageHandlers = new LanguageHandlers(bot);
|
||||
}
|
||||
|
||||
register(): void {
|
||||
@@ -23,12 +29,24 @@ export class CommandHandlers {
|
||||
this.bot.onText(/\/matches/, (msg: Message) => this.handleMatches(msg));
|
||||
this.bot.onText(/\/settings/, (msg: Message) => this.handleSettings(msg));
|
||||
this.bot.onText(/\/create_profile/, (msg: Message) => this.handleCreateProfile(msg));
|
||||
|
||||
// Регистрация обработчика настроек уведомлений
|
||||
this.bot.onText(/\/notifications/, (msg: Message) => this.notificationHandlers.handleNotificationsCommand(msg));
|
||||
|
||||
// Регистрируем обработчики для уведомлений
|
||||
this.notificationHandlers.register();
|
||||
}
|
||||
|
||||
async handleStart(msg: Message): Promise<void> {
|
||||
const userId = msg.from?.id.toString();
|
||||
if (!userId) return;
|
||||
|
||||
// Проверяем, нужно ли показать выбор языка новому пользователю
|
||||
const languageSelectionShown = await this.languageHandlers.checkAndShowLanguageSelection(userId, msg.chat.id);
|
||||
if (languageSelectionShown) {
|
||||
return; // Показали выбор языка, ждем ответа пользователя
|
||||
}
|
||||
|
||||
// Проверяем есть ли у пользователя профиль
|
||||
const existingProfile = await this.profileService.getProfileByTelegramId(userId);
|
||||
|
||||
@@ -44,7 +62,8 @@ export class CommandHandlers {
|
||||
{ text: '⭐ VIP поиск', callback_data: 'vip_search' }
|
||||
],
|
||||
[
|
||||
{ text: '⚙️ Настройки', callback_data: 'settings' }
|
||||
{ text: '⚙️ Настройки', callback_data: 'settings' },
|
||||
{ text: '🔔 Уведомления', callback_data: 'notifications' }
|
||||
]
|
||||
]
|
||||
};
|
||||
@@ -84,6 +103,7 @@ export class CommandHandlers {
|
||||
/browse - Просмотр анкет
|
||||
/matches - Ваши матчи
|
||||
/settings - Настройки
|
||||
/notifications - Настройки уведомлений
|
||||
/help - Эта справка
|
||||
|
||||
<EFBFBD> Как использовать:
|
||||
@@ -191,7 +211,7 @@ export class CommandHandlers {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '🔍 Настройки поиска', callback_data: 'search_settings' },
|
||||
{ text: '🔔 Уведомления', callback_data: 'notification_settings' }
|
||||
{ text: '🔔 Уведомления', callback_data: 'notifications' }
|
||||
],
|
||||
[
|
||||
{ text: '🚫 Скрыть профиль', callback_data: 'hide_profile' },
|
||||
@@ -242,7 +262,10 @@ export class CommandHandlers {
|
||||
{ text: '✏️ Редактировать', callback_data: 'edit_profile' },
|
||||
{ text: '📸 Фото', callback_data: 'manage_photos' }
|
||||
],
|
||||
[{ text: '🔍 Начать поиск', callback_data: 'start_browsing' }]
|
||||
[
|
||||
{ text: '🔍 Начать поиск', callback_data: 'start_browsing' },
|
||||
{ text: '🔔 Уведомления', callback_data: 'notifications' }
|
||||
]
|
||||
]
|
||||
} : {
|
||||
inline_keyboard: [
|
||||
|
||||
@@ -11,9 +11,9 @@ export class EnhancedChatHandlers {
|
||||
|
||||
constructor(bot: TelegramBot) {
|
||||
this.bot = bot;
|
||||
this.chatService = new ChatService();
|
||||
this.profileService = new ProfileService();
|
||||
this.notificationService = new NotificationService(bot);
|
||||
this.chatService = new ChatService(this.notificationService);
|
||||
}
|
||||
|
||||
// ===== НАТИВНЫЙ ИНТЕРФЕЙС ЧАТОВ =====
|
||||
@@ -168,25 +168,24 @@ export class EnhancedChatHandlers {
|
||||
|
||||
// ===== СИСТЕМА УВЕДОМЛЕНИЙ =====
|
||||
|
||||
// Отправить уведомление о новом сообщении
|
||||
// Отправить уведомление о новом сообщении - теперь используем NotificationService
|
||||
async sendMessageNotification(receiverTelegramId: string, senderName: string, messagePreview: string, matchId: string): Promise<void> {
|
||||
try {
|
||||
const receiverChatId = parseInt(receiverTelegramId);
|
||||
// Получаем идентификаторы пользователей для использования в NotificationService
|
||||
const receiverUserId = await this.profileService.getUserIdByTelegramId(receiverTelegramId);
|
||||
const sender = await this.chatService.getMatchInfo(matchId, receiverTelegramId);
|
||||
|
||||
await this.bot.sendMessage(
|
||||
receiverChatId,
|
||||
`💌 *Новое сообщение от ${senderName}*\n\n` +
|
||||
`"${this.escapeMarkdown(messagePreview)}"\n\n` +
|
||||
'👆 Нажмите "Открыть чат" для ответа',
|
||||
{
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[{ text: '💬 Открыть чат', callback_data: `open_native_chat_${matchId}` }],
|
||||
[{ text: '📱 Все чаты', callback_data: 'native_chats' }]
|
||||
]
|
||||
}
|
||||
}
|
||||
if (!receiverUserId || !sender?.otherUserId) {
|
||||
console.error('Failed to get user IDs for notification');
|
||||
return;
|
||||
}
|
||||
|
||||
// Используем сервис уведомлений для отправки более красивого уведомления
|
||||
await this.notificationService.sendMessageNotification(
|
||||
receiverUserId,
|
||||
sender.otherUserId,
|
||||
messagePreview,
|
||||
matchId
|
||||
);
|
||||
} catch (error) {
|
||||
console.error('Error sending message notification:', error);
|
||||
|
||||
167
src/handlers/languageHandlers.ts
Normal file
167
src/handlers/languageHandlers.ts
Normal file
@@ -0,0 +1,167 @@
|
||||
import TelegramBot, { CallbackQuery, InlineKeyboardMarkup } from 'node-telegram-bot-api';
|
||||
import { ProfileService } from '../services/profileService';
|
||||
import LocalizationService from '../services/localizationService';
|
||||
|
||||
export class LanguageHandlers {
|
||||
private bot: TelegramBot;
|
||||
private profileService: ProfileService;
|
||||
private localizationService: LocalizationService;
|
||||
|
||||
constructor(bot: TelegramBot) {
|
||||
this.bot = bot;
|
||||
this.profileService = new ProfileService();
|
||||
this.localizationService = LocalizationService.getInstance();
|
||||
}
|
||||
|
||||
/**
|
||||
* Показать меню выбора языка
|
||||
*/
|
||||
async showLanguageSelection(chatId: number, messageId?: number): Promise<void> {
|
||||
const keyboard: InlineKeyboardMarkup = {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '🇷🇺 Русский', callback_data: 'set_lang_ru' },
|
||||
{ text: '🇬🇧 English', callback_data: 'set_lang_en' }
|
||||
],
|
||||
[
|
||||
{ text: '🇪🇸 Español', callback_data: 'set_lang_es' },
|
||||
{ text: '🇫🇷 Français', callback_data: 'set_lang_fr' }
|
||||
],
|
||||
[
|
||||
{ text: '🇩🇪 Deutsch', callback_data: 'set_lang_de' },
|
||||
{ text: '🇮🇹 Italiano', callback_data: 'set_lang_it' }
|
||||
],
|
||||
[
|
||||
{ text: '🇵🇹 Português', callback_data: 'set_lang_pt' },
|
||||
{ text: '🇰🇷 한국어', callback_data: 'set_lang_ko' }
|
||||
],
|
||||
[
|
||||
{ text: '🇨🇳 中文', callback_data: 'set_lang_zh' },
|
||||
{ text: '🇯🇵 日本語', callback_data: 'set_lang_ja' }
|
||||
]
|
||||
]
|
||||
};
|
||||
|
||||
const text =
|
||||
'🌍 Choose your language / Выберите язык:\n\n' +
|
||||
'🇷🇺 Русский\n' +
|
||||
'🇬🇧 English\n' +
|
||||
'🇪🇸 Español\n' +
|
||||
'🇫🇷 Français\n' +
|
||||
'🇩🇪 Deutsch\n' +
|
||||
'🇮🇹 Italiano\n' +
|
||||
'🇵🇹 Português\n' +
|
||||
'🇰🇷 한국어\n' +
|
||||
'🇨🇳 中文\n' +
|
||||
'🇯🇵 日本語';
|
||||
|
||||
if (messageId) {
|
||||
// Обновляем существующее сообщение
|
||||
await this.bot.editMessageText(text, {
|
||||
chat_id: chatId,
|
||||
message_id: messageId,
|
||||
reply_markup: keyboard
|
||||
});
|
||||
} else {
|
||||
// Отправляем новое сообщение
|
||||
await this.bot.sendMessage(chatId, text, { reply_markup: keyboard });
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Обработать установку языка
|
||||
*/
|
||||
async handleSetLanguage(query: CallbackQuery): Promise<void> {
|
||||
const chatId = query.message?.chat.id;
|
||||
const userId = query.from.id.toString();
|
||||
const messageId = query.message?.message_id;
|
||||
|
||||
if (!chatId || !userId) return;
|
||||
|
||||
// Извлекаем код языка из callback_data (например, 'set_lang_ru' -> 'ru')
|
||||
const langCode = query.data?.replace('set_lang_', '');
|
||||
if (!langCode) return;
|
||||
|
||||
try {
|
||||
// Проверяем, поддерживается ли язык
|
||||
const supportedLanguages = this.localizationService.getSupportedLanguages();
|
||||
if (!supportedLanguages.includes(langCode)) {
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: '❌ Unsupported language / Язык не поддерживается'
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
// Обновляем язык пользователя в базе данных
|
||||
await this.profileService.updateUserLanguage(userId, langCode);
|
||||
|
||||
// Устанавливаем язык в сервисе локализации
|
||||
this.localizationService.setLanguage(langCode);
|
||||
|
||||
// Получаем переведенное сообщение об успехе
|
||||
const successMessage = this.localizationService.t('language.changed');
|
||||
|
||||
// Показываем уведомление об успехе
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: successMessage,
|
||||
show_alert: false
|
||||
});
|
||||
|
||||
// Удаляем сообщение с выбором языка
|
||||
if (messageId) {
|
||||
await this.bot.deleteMessage(chatId, messageId);
|
||||
}
|
||||
|
||||
// Показываем приветственное сообщение на выбранном языке
|
||||
const keyboard: InlineKeyboardMarkup = {
|
||||
inline_keyboard: [
|
||||
[{ text: this.localizationService.t('start.createProfile'), callback_data: 'create_profile' }],
|
||||
[{ text: this.localizationService.t('start.howItWorks'), callback_data: 'how_it_works' }]
|
||||
]
|
||||
};
|
||||
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
this.localizationService.t('start.welcomeNew'),
|
||||
{ reply_markup: keyboard }
|
||||
);
|
||||
|
||||
} catch (error) {
|
||||
console.error('Error setting language:', error);
|
||||
await this.bot.answerCallbackQuery(query.id, {
|
||||
text: '❌ Error / Ошибка'
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Проверить, нужно ли показать выбор языка новому пользователю
|
||||
*/
|
||||
async checkAndShowLanguageSelection(userId: string, chatId: number): Promise<boolean> {
|
||||
try {
|
||||
// Получаем текущий язык пользователя
|
||||
const currentLang = await this.profileService.getUserLanguage(userId);
|
||||
|
||||
// Если язык уже установлен, не показываем выбор
|
||||
if (currentLang && currentLang !== 'ru') {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Проверяем, есть ли у пользователя профиль
|
||||
const profile = await this.profileService.getProfileByTelegramId(userId);
|
||||
|
||||
// Показываем выбор языка только новым пользователям без профиля
|
||||
if (!profile) {
|
||||
await this.showLanguageSelection(chatId);
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
} catch (error) {
|
||||
console.error('Error checking language selection:', error);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export default LanguageHandlers;
|
||||
76
src/handlers/likeBackHandler.ts
Normal file
76
src/handlers/likeBackHandler.ts
Normal file
@@ -0,0 +1,76 @@
|
||||
import TelegramBot from 'node-telegram-bot-api';
|
||||
import { ProfileService } from '../services/profileService';
|
||||
import { MatchingService } from '../services/matchingService';
|
||||
|
||||
export class LikeBackHandler {
|
||||
private bot: TelegramBot;
|
||||
private profileService: ProfileService;
|
||||
private matchingService: MatchingService;
|
||||
|
||||
constructor(bot: TelegramBot) {
|
||||
this.bot = bot;
|
||||
this.profileService = new ProfileService();
|
||||
this.matchingService = new MatchingService();
|
||||
}
|
||||
|
||||
// Функция для обработки обратного лайка из уведомления
|
||||
async handleLikeBack(chatId: number, telegramId: string, targetUserId: string): Promise<void> {
|
||||
try {
|
||||
// Получаем информацию о пользователях
|
||||
const [userId, targetProfile] = await Promise.all([
|
||||
this.profileService.getUserIdByTelegramId(telegramId),
|
||||
this.profileService.getProfileByUserId(targetUserId)
|
||||
]);
|
||||
|
||||
if (!userId || !targetProfile) {
|
||||
await this.bot.sendMessage(chatId, '❌ Не удалось найти профиль');
|
||||
return;
|
||||
}
|
||||
|
||||
// Проверяем, есть ли уже свайп
|
||||
const existingSwipe = await this.matchingService.getSwipeBetweenUsers(userId, targetUserId);
|
||||
if (existingSwipe) {
|
||||
await this.bot.sendMessage(chatId, '❓ Вы уже оценили этот профиль ранее.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Создаем свайп (лайк)
|
||||
const result = await this.matchingService.createSwipe(userId, targetUserId, 'like');
|
||||
|
||||
if (result.isMatch) {
|
||||
// Это матч!
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'🎉 *Поздравляем! Это взаимно!*\n\n' +
|
||||
`Вы и *${targetProfile.name}* понравились друг другу!\n` +
|
||||
'Теперь вы можете начать общение.',
|
||||
{
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[{ text: '💬 Начать общение', callback_data: `start_chat:${targetUserId}` }],
|
||||
[{ text: '👀 Посмотреть профиль', callback_data: `view_profile:${targetUserId}` }]
|
||||
]
|
||||
}
|
||||
}
|
||||
);
|
||||
} else {
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'❤️ Вам понравился профиль ' + targetProfile.name + '!\n\n' +
|
||||
'Если вы также понравитесь этому пользователю, будет создан матч.',
|
||||
{
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[{ text: '🔍 Продолжить поиск', callback_data: 'start_browsing' }]
|
||||
]
|
||||
}
|
||||
}
|
||||
);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error in handleLikeBack:', error);
|
||||
await this.bot.sendMessage(chatId, '❌ Произошла ошибка при обработке лайка');
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -2,6 +2,7 @@ import TelegramBot, { Message, InlineKeyboardMarkup } from 'node-telegram-bot-ap
|
||||
import { ProfileService } from '../services/profileService';
|
||||
import { ChatService } from '../services/chatService';
|
||||
import { EnhancedChatHandlers } from './enhancedChatHandlers';
|
||||
import { KakaoMapService } from '../services/kakaoMapService';
|
||||
|
||||
// Состояния пользователей для создания профилей
|
||||
interface UserState {
|
||||
@@ -19,6 +20,8 @@ interface ChatState {
|
||||
interface ProfileEditState {
|
||||
waitingForInput: boolean;
|
||||
field: string;
|
||||
tempCity?: string; // Временное хранение города для подтверждения
|
||||
tempLocation?: { latitude: number; longitude: number }; // Временное хранение координат
|
||||
}
|
||||
|
||||
export class MessageHandlers {
|
||||
@@ -26,15 +29,27 @@ export class MessageHandlers {
|
||||
private profileService: ProfileService;
|
||||
private chatService: ChatService;
|
||||
private enhancedChatHandlers: EnhancedChatHandlers;
|
||||
private notificationService: any;
|
||||
private kakaoMapService: KakaoMapService | null = null;
|
||||
private userStates: Map<string, UserState> = new Map();
|
||||
private chatStates: Map<string, ChatState> = new Map();
|
||||
private profileEditStates: Map<string, ProfileEditState> = new Map();
|
||||
public profileEditStates: Map<string, ProfileEditState> = new Map();
|
||||
|
||||
constructor(bot: TelegramBot) {
|
||||
constructor(bot: TelegramBot, notificationService?: any) {
|
||||
this.bot = bot;
|
||||
this.profileService = new ProfileService();
|
||||
this.chatService = new ChatService();
|
||||
this.notificationService = notificationService;
|
||||
this.chatService = new ChatService(notificationService);
|
||||
this.enhancedChatHandlers = new EnhancedChatHandlers(bot);
|
||||
|
||||
// Инициализируем Kakao Maps, если есть API ключ
|
||||
const kakaoApiKey = process.env.KAKAO_REST_API_KEY || process.env.KAKAO_MAP_REST_KEY;
|
||||
if (kakaoApiKey) {
|
||||
this.kakaoMapService = new KakaoMapService(kakaoApiKey);
|
||||
console.log('✅ Kakao Maps service initialized');
|
||||
} else {
|
||||
console.warn('⚠️ KAKAO_REST_API_KEY or KAKAO_MAP_REST_KEY not found, location features will be limited');
|
||||
}
|
||||
}
|
||||
|
||||
register(): void {
|
||||
@@ -55,7 +70,7 @@ export class MessageHandlers {
|
||||
const profileEditState = this.profileEditStates.get(userId);
|
||||
|
||||
// Проверяем на нативные чаты (прямые сообщения в контексте чата)
|
||||
if (msg.text && await this.enhancedChatHandlers.handleIncomingChatMessage(msg.chat.id, msg.text)) {
|
||||
if (await this.enhancedChatHandlers.handleIncomingChatMessage(msg, userId)) {
|
||||
return; // Сообщение обработано как сообщение в чате
|
||||
}
|
||||
|
||||
@@ -127,22 +142,68 @@ export class MessageHandlers {
|
||||
userState.data.age = age;
|
||||
userState.step = 'waiting_city';
|
||||
|
||||
await this.bot.sendMessage(chatId, '📍 Прекрасно! В каком городе вы живете?');
|
||||
// Запрашиваем геолокацию или текст
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'📍 Прекрасно! В каком городе вы живете?\n\n' +
|
||||
'💡 Вы можете:\n' +
|
||||
'• Отправить геолокацию 📍 (кнопка ниже)\n' +
|
||||
'• Написать название города вручную',
|
||||
{
|
||||
reply_markup: {
|
||||
keyboard: [
|
||||
[{ text: '📍 Отправить геолокацию', request_location: true }]
|
||||
],
|
||||
one_time_keyboard: true,
|
||||
resize_keyboard: true
|
||||
}
|
||||
}
|
||||
);
|
||||
break;
|
||||
|
||||
case 'waiting_city':
|
||||
if (!msg.text) {
|
||||
await this.bot.sendMessage(chatId, '❌ Пожалуйста, отправьте название города');
|
||||
// Обработка геолокации
|
||||
if (msg.location) {
|
||||
await this.handleLocationForCity(msg, userId, userState);
|
||||
return;
|
||||
}
|
||||
|
||||
userState.data.city = msg.text.trim();
|
||||
userState.step = 'waiting_bio';
|
||||
// Обработка текста
|
||||
if (!msg.text) {
|
||||
await this.bot.sendMessage(chatId, '❌ Пожалуйста, отправьте название города или геолокацию');
|
||||
return;
|
||||
}
|
||||
|
||||
const cityInput = msg.text.trim();
|
||||
console.log(`User ${userId} entered city manually: ${cityInput}`);
|
||||
|
||||
// Временно сохраняем город и запрашиваем подтверждение
|
||||
userState.data.city = cityInput;
|
||||
userState.step = 'confirm_city';
|
||||
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'📝 Теперь расскажите немного о себе (био):\n\n' +
|
||||
'💡 Например: хобби, интересы, что ищете в отношениях и т.д.'
|
||||
chatId,
|
||||
`<EFBFBD> Вы указали город: *${cityInput}*\n\n` +
|
||||
'Подтвердите или введите заново.',
|
||||
{
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '✅ Подтвердить', callback_data: 'confirm_city' },
|
||||
{ text: '✏️ Ввести заново', callback_data: 'edit_city_manual' }
|
||||
]
|
||||
]
|
||||
}
|
||||
}
|
||||
);
|
||||
break;
|
||||
|
||||
case 'confirm_city':
|
||||
// Этот случай обрабатывается через callback_data, но на случай если пользователь напишет текст
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'⚠️ Пожалуйста, используйте кнопки для подтверждения города.'
|
||||
);
|
||||
break;
|
||||
|
||||
@@ -208,6 +269,7 @@ export class MessageHandlers {
|
||||
interestedIn: interestedIn,
|
||||
bio: profileData.bio,
|
||||
city: profileData.city,
|
||||
location: profileData.location, // Добавляем координаты
|
||||
photos: profileData.photos,
|
||||
interests: [],
|
||||
searchPreferences: {
|
||||
@@ -428,6 +490,47 @@ export class MessageHandlers {
|
||||
}
|
||||
value = distance;
|
||||
break;
|
||||
|
||||
case 'hobbies':
|
||||
// Разбиваем строку с запятыми на массив
|
||||
if (typeof value === 'string') {
|
||||
value = value.split(',').map(hobby => hobby.trim()).filter(hobby => hobby.length > 0);
|
||||
}
|
||||
break;
|
||||
|
||||
case 'city':
|
||||
// Обработка города: поддержка геолокации и текстового ввода
|
||||
if (msg.location) {
|
||||
// Обработка геолокации
|
||||
await this.handleLocationForCityEdit(msg, userId);
|
||||
return; // Выходим из функции, так как требуется подтверждение
|
||||
} else if (msg.text) {
|
||||
// Обработка текстового ввода города
|
||||
const cityInput = msg.text.trim();
|
||||
console.log(`User ${userId} entered city manually during edit: ${cityInput}`);
|
||||
// Сохраняем временно в состояние редактирования
|
||||
const editState = this.profileEditStates.get(userId);
|
||||
if (editState) {
|
||||
editState.tempCity = cityInput;
|
||||
}
|
||||
// Требуем подтверждения
|
||||
await this.bot.sendMessage(chatId, `📍 Вы указали город: *${cityInput}*\n\nПодтвердите или введите заново.`, {
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '✅ Подтвердить', callback_data: 'confirm_city_edit' },
|
||||
{ text: '✏️ Ввести заново', callback_data: 'edit_city_manual_edit' }
|
||||
]
|
||||
]
|
||||
}
|
||||
});
|
||||
return; // Выходим, ждем подтверждения
|
||||
} else {
|
||||
isValid = false;
|
||||
errorMessage = '❌ Пожалуйста, отправьте название города или геолокацию';
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!isValid) {
|
||||
@@ -497,6 +600,10 @@ export class MessageHandlers {
|
||||
// В БД поле называется 'city' (не 'location')
|
||||
updates.city = value;
|
||||
break;
|
||||
case 'location':
|
||||
// Обновляем координаты
|
||||
updates.location = value;
|
||||
break;
|
||||
case 'job':
|
||||
// В БД поле называется 'occupation', но мы используем job в модели
|
||||
updates.job = value;
|
||||
@@ -537,4 +644,202 @@ export class MessageHandlers {
|
||||
await this.profileService.updateProfile(profile.userId, updates);
|
||||
}
|
||||
}
|
||||
|
||||
// Обработка геолокации для определения города
|
||||
private async handleLocationForCity(msg: Message, userId: string, userState: UserState): Promise<void> {
|
||||
const chatId = msg.chat.id;
|
||||
|
||||
if (!msg.location) {
|
||||
await this.bot.sendMessage(chatId, '❌ Не удалось получить геолокацию');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
// Показываем индикатор загрузки
|
||||
await this.bot.sendChatAction(chatId, 'typing');
|
||||
|
||||
if (!this.kakaoMapService) {
|
||||
// Если Kakao Maps не настроен, используем координаты как есть
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'⚠️ Автоматическое определение города недоступно.\n' +
|
||||
'Пожалуйста, введите название вашего города вручную.',
|
||||
{ reply_markup: { remove_keyboard: true } }
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
// Показываем индикатор загрузки
|
||||
await this.bot.sendChatAction(chatId, 'typing');
|
||||
|
||||
if (!this.kakaoMapService) {
|
||||
// Если Kakao Maps не настроен, используем координаты как есть
|
||||
console.warn(`KakaoMaps not configured - user ${userId} sent coords: ${msg.location.latitude},${msg.location.longitude}`);
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'⚠️ Автоматическое определение города недоступно.\n' +
|
||||
'Пожалуйста, введите название вашего города вручную.',
|
||||
{ reply_markup: { remove_keyboard: true } }
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
// Логируем входные координаты
|
||||
console.log(`Processing coordinates for user ${userId}: lat=${msg.location.latitude}, lon=${msg.location.longitude}`);
|
||||
|
||||
// Получаем адрес через Kakao Maps
|
||||
const address = await this.kakaoMapService.getAddressFromCoordinates(
|
||||
msg.location.latitude,
|
||||
msg.location.longitude
|
||||
);
|
||||
|
||||
if (!address) {
|
||||
console.warn(`KakaoMaps returned no address for user ${userId} coords: ${msg.location.latitude},${msg.location.longitude}`);
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'❌ Не удалось определить город по вашей геолокации.\n' +
|
||||
'Пожалуйста, введите название города вручную.',
|
||||
{ reply_markup: { remove_keyboard: true } }
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
// Получаем название города для сохранения
|
||||
const cityName = this.kakaoMapService.getCityNameForProfile(address);
|
||||
const displayAddress = this.kakaoMapService.formatAddressForDisplay(address);
|
||||
|
||||
// Логируем результат
|
||||
console.log(`KakaoMaps resolved for user ${userId}: city=${cityName}, address=${displayAddress}`);
|
||||
|
||||
// Временно сохраняем город И координаты (пока не подтверждены пользователем)
|
||||
userState.data.city = cityName;
|
||||
userState.data.location = {
|
||||
latitude: msg.location.latitude,
|
||||
longitude: msg.location.longitude
|
||||
};
|
||||
userState.step = 'confirm_city';
|
||||
|
||||
// Отправляем пользователю информацию с кнопками подтверждения
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
`✅ Мы распознали местоположение: *${displayAddress}*\n\n` +
|
||||
'Пожалуйста, подтвердите город проживания или введите название вручную.',
|
||||
{
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '✅ Подтвердить', callback_data: 'confirm_city' },
|
||||
{ text: '✏️ Ввести вручную', callback_data: 'edit_city_manual' }
|
||||
]
|
||||
]
|
||||
}
|
||||
}
|
||||
);
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'📝 Теперь расскажите немного о себе (био):\n\n' +
|
||||
'💡 Например: хобби, интересы, что ищете в отношениях и т.д.'
|
||||
);
|
||||
|
||||
} catch (error) {
|
||||
console.error('Error handling location for city:', error);
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'❌ Произошла ошибка при определении города.\n' +
|
||||
'Пожалуйста, введите название города вручную.',
|
||||
{ reply_markup: { remove_keyboard: true } }
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// Обработка геолокации для определения города при редактировании профиля
|
||||
private async handleLocationForCityEdit(msg: Message, userId: string): Promise<void> {
|
||||
const chatId = msg.chat.id;
|
||||
|
||||
if (!msg.location) {
|
||||
await this.bot.sendMessage(chatId, '❌ Не удалось получить геолокацию');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
// Показываем индикатор загрузки
|
||||
await this.bot.sendChatAction(chatId, 'typing');
|
||||
|
||||
if (!this.kakaoMapService) {
|
||||
console.warn(`KakaoMaps not configured - user ${userId} sent coords during edit: ${msg.location.latitude},${msg.location.longitude}`);
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'⚠️ Автоматическое определение города недоступно.\n' +
|
||||
'Пожалуйста, введите название вашего города вручную.',
|
||||
{ reply_markup: { remove_keyboard: true } }
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
// Логируем входные координаты
|
||||
console.log(`Processing coordinates for user ${userId} during edit: lat=${msg.location.latitude}, lon=${msg.location.longitude}`);
|
||||
|
||||
// Получаем адрес через Kakao Maps
|
||||
const address = await this.kakaoMapService.getAddressFromCoordinates(
|
||||
msg.location.latitude,
|
||||
msg.location.longitude
|
||||
);
|
||||
|
||||
if (!address) {
|
||||
console.warn(`KakaoMaps returned no address for user ${userId} during edit coords: ${msg.location.latitude},${msg.location.longitude}`);
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'❌ Не удалось определить город по вашей геолокации.\n' +
|
||||
'Пожалуйста, введите название города вручную.',
|
||||
{ reply_markup: { remove_keyboard: true } }
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
// Получаем название города для сохранения
|
||||
const cityName = this.kakaoMapService.getCityNameForProfile(address);
|
||||
const displayAddress = this.kakaoMapService.formatAddressForDisplay(address);
|
||||
|
||||
// Логируем результат
|
||||
console.log(`KakaoMaps resolved for user ${userId} during edit: city=${cityName}, address=${displayAddress}`);
|
||||
|
||||
// Временно сохраняем город И координаты в состояние редактирования
|
||||
const editState = this.profileEditStates.get(userId);
|
||||
if (editState) {
|
||||
editState.tempCity = cityName;
|
||||
editState.tempLocation = {
|
||||
latitude: msg.location.latitude,
|
||||
longitude: msg.location.longitude
|
||||
};
|
||||
}
|
||||
|
||||
// Отправляем пользователю информацию с кнопками подтверждения
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
`✅ Мы распознали местоположение: *${displayAddress}*\n\n` +
|
||||
'Пожалуйста, подтвердите город проживания или введите название вручную.',
|
||||
{
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '✅ Подтвердить', callback_data: 'confirm_city_edit' },
|
||||
{ text: '✏️ Ввести вручную', callback_data: 'edit_city_manual_edit' }
|
||||
]
|
||||
]
|
||||
}
|
||||
}
|
||||
);
|
||||
|
||||
} catch (error) {
|
||||
console.error('Error handling location for city during edit:', error);
|
||||
await this.bot.sendMessage(
|
||||
chatId,
|
||||
'❌ Произошла ошибка при определении города.\n' +
|
||||
'Пожалуйста, введите название города вручную.',
|
||||
{ reply_markup: { remove_keyboard: true } }
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
644
src/handlers/notificationHandlers.ts
Normal file
644
src/handlers/notificationHandlers.ts
Normal file
@@ -0,0 +1,644 @@
|
||||
import TelegramBot from 'node-telegram-bot-api';
|
||||
import { v4 as uuidv4 } from 'uuid';
|
||||
import { query } from '../database/connection';
|
||||
import { NotificationService } from '../services/notificationService';
|
||||
|
||||
interface NotificationSettings {
|
||||
newMatches: boolean;
|
||||
newMessages: boolean;
|
||||
newLikes: boolean;
|
||||
reminders: boolean;
|
||||
dailySummary: boolean;
|
||||
timePreference: 'morning' | 'afternoon' | 'evening' | 'night';
|
||||
doNotDisturb: boolean;
|
||||
doNotDisturbStart?: string;
|
||||
doNotDisturbEnd?: string;
|
||||
}
|
||||
|
||||
export class NotificationHandlers {
|
||||
private bot: TelegramBot;
|
||||
private notificationService: NotificationService;
|
||||
|
||||
constructor(bot: TelegramBot) {
|
||||
this.bot = bot;
|
||||
this.notificationService = new NotificationService(bot);
|
||||
}
|
||||
|
||||
// Метод для получения экземпляра сервиса уведомлений
|
||||
getNotificationService(): NotificationService {
|
||||
return this.notificationService;
|
||||
}
|
||||
|
||||
// Обработка команды /notifications
|
||||
async handleNotificationsCommand(msg: TelegramBot.Message): Promise<void> {
|
||||
const telegramId = msg.from?.id.toString();
|
||||
if (!telegramId) return;
|
||||
|
||||
try {
|
||||
const userId = await this.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.sendMessage(msg.chat.id, '❌ Вы не зарегистрированы. Используйте команду /start для регистрации.');
|
||||
return;
|
||||
}
|
||||
|
||||
const settings = await this.notificationService.getNotificationSettings(userId);
|
||||
await this.sendNotificationSettings(msg.chat.id, settings as NotificationSettings);
|
||||
} catch (error) {
|
||||
console.error('Error handling notifications command:', error);
|
||||
await this.bot.sendMessage(msg.chat.id, '❌ Произошла ошибка при загрузке настроек уведомлений.');
|
||||
}
|
||||
}
|
||||
|
||||
// Отправка меню настроек уведомлений
|
||||
async sendNotificationSettings(chatId: number, settings: NotificationSettings): Promise<void> {
|
||||
const message = `
|
||||
🔔 *Настройки уведомлений*
|
||||
|
||||
Выберите, какие уведомления вы хотите получать:
|
||||
|
||||
${settings.newMatches ? '✅' : '❌'} Новые матчи
|
||||
${settings.newMessages ? '✅' : '❌'} Новые сообщения
|
||||
${settings.newLikes ? '✅' : '❌'} Новые лайки
|
||||
${settings.reminders ? '✅' : '❌'} Напоминания
|
||||
${settings.dailySummary ? '✅' : '❌'} Ежедневные сводки
|
||||
|
||||
⏰ Предпочтительное время: ${this.getTimePreferenceText(settings.timePreference)}
|
||||
|
||||
${settings.doNotDisturb ? '🔕' : '🔔'} Режим "Не беспокоить": ${settings.doNotDisturb ? 'Включен' : 'Выключен'}
|
||||
${settings.doNotDisturb && settings.doNotDisturbStart && settings.doNotDisturbEnd ?
|
||||
`с ${settings.doNotDisturbStart} до ${settings.doNotDisturbEnd}` : ''}
|
||||
|
||||
Нажмите на кнопку, чтобы изменить настройку:
|
||||
`;
|
||||
|
||||
await this.bot.sendMessage(chatId, message, {
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: `${settings.newMatches ? '✅' : '❌'} Новые матчи`, callback_data: 'notif_toggle:newMatches' },
|
||||
{ text: `${settings.newMessages ? '✅' : '❌'} Новые сообщения`, callback_data: 'notif_toggle:newMessages' }
|
||||
],
|
||||
[
|
||||
{ text: `${settings.newLikes ? '✅' : '❌'} Новые лайки`, callback_data: 'notif_toggle:newLikes' },
|
||||
{ text: `${settings.reminders ? '✅' : '❌'} Напоминания`, callback_data: 'notif_toggle:reminders' }
|
||||
],
|
||||
[
|
||||
{ text: `${settings.dailySummary ? '✅' : '❌'} Ежедневные сводки`, callback_data: 'notif_toggle:dailySummary' }
|
||||
],
|
||||
[
|
||||
{ text: `⏰ Время: ${this.getTimePreferenceText(settings.timePreference)}`, callback_data: 'notif_time' }
|
||||
],
|
||||
[
|
||||
{ text: `${settings.doNotDisturb ? '🔕' : '🔔'} Режим "Не беспокоить"`, callback_data: 'notif_dnd' }
|
||||
],
|
||||
[
|
||||
{ text: '↩️ Назад', callback_data: 'settings' }
|
||||
]
|
||||
]
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Обработка переключения настройки уведомления
|
||||
async handleNotificationToggle(callbackQuery: TelegramBot.CallbackQuery): Promise<void> {
|
||||
const telegramId = callbackQuery.from?.id.toString();
|
||||
if (!telegramId || !callbackQuery.message) return;
|
||||
|
||||
try {
|
||||
const userId = await this.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Вы не зарегистрированы.' });
|
||||
return;
|
||||
}
|
||||
|
||||
// notif_toggle:settingName
|
||||
const settingName = callbackQuery.data?.split(':')[1];
|
||||
if (!settingName) return;
|
||||
|
||||
const settings = await this.notificationService.getNotificationSettings(userId);
|
||||
let updatedSettings: Partial<NotificationSettings> = { ...settings };
|
||||
|
||||
// Инвертируем значение настройки
|
||||
if (settingName in updatedSettings) {
|
||||
switch(settingName) {
|
||||
case 'newMatches':
|
||||
updatedSettings.newMatches = !updatedSettings.newMatches;
|
||||
break;
|
||||
case 'newMessages':
|
||||
updatedSettings.newMessages = !updatedSettings.newMessages;
|
||||
break;
|
||||
case 'newLikes':
|
||||
updatedSettings.newLikes = !updatedSettings.newLikes;
|
||||
break;
|
||||
case 'reminders':
|
||||
updatedSettings.reminders = !updatedSettings.reminders;
|
||||
break;
|
||||
case 'dailySummary':
|
||||
updatedSettings.dailySummary = !updatedSettings.dailySummary;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Обновляем настройки
|
||||
await this.notificationService.updateNotificationSettings(userId, updatedSettings);
|
||||
|
||||
// Отправляем обновленные настройки
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, {
|
||||
text: `✅ Настройка "${this.getSettingName(settingName)}" ${updatedSettings[settingName as keyof NotificationSettings] ? 'включена' : 'отключена'}`
|
||||
});
|
||||
|
||||
await this.sendNotificationSettings(callbackQuery.message.chat.id, updatedSettings as NotificationSettings);
|
||||
} catch (error) {
|
||||
console.error('Error handling notification toggle:', error);
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Произошла ошибка при обновлении настроек.' });
|
||||
}
|
||||
}
|
||||
|
||||
// Обработка выбора времени для уведомлений
|
||||
async handleTimePreference(callbackQuery: TelegramBot.CallbackQuery): Promise<void> {
|
||||
if (!callbackQuery.message) return;
|
||||
|
||||
await this.bot.editMessageText('⏰ *Выберите предпочтительное время для уведомлений:*', {
|
||||
chat_id: callbackQuery.message.chat.id,
|
||||
message_id: callbackQuery.message.message_id,
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '🌅 Утро (9:00)', callback_data: 'notif_time_set:morning' },
|
||||
{ text: '☀️ День (13:00)', callback_data: 'notif_time_set:afternoon' }
|
||||
],
|
||||
[
|
||||
{ text: '🌆 Вечер (19:00)', callback_data: 'notif_time_set:evening' },
|
||||
{ text: '🌙 Ночь (22:00)', callback_data: 'notif_time_set:night' }
|
||||
],
|
||||
[
|
||||
{ text: '↩️ Назад', callback_data: 'notifications' }
|
||||
]
|
||||
]
|
||||
}
|
||||
});
|
||||
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id);
|
||||
}
|
||||
|
||||
// Обработка установки времени для уведомлений
|
||||
async handleTimePreferenceSet(callbackQuery: TelegramBot.CallbackQuery): Promise<void> {
|
||||
const telegramId = callbackQuery.from?.id.toString();
|
||||
if (!telegramId || !callbackQuery.message) return;
|
||||
|
||||
try {
|
||||
const userId = await this.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Вы не зарегистрированы.' });
|
||||
return;
|
||||
}
|
||||
|
||||
// notif_time_set:timePreference
|
||||
const timePreference = callbackQuery.data?.split(':')[1] as 'morning' | 'afternoon' | 'evening' | 'night';
|
||||
if (!timePreference) return;
|
||||
|
||||
const settings = await this.notificationService.getNotificationSettings(userId);
|
||||
// Копируем существующие настройки и обновляем нужные поля
|
||||
const existingSettings = settings as NotificationSettings;
|
||||
const updatedSettings: NotificationSettings = {
|
||||
...existingSettings,
|
||||
timePreference
|
||||
};
|
||||
|
||||
// Обновляем настройки
|
||||
await this.notificationService.updateNotificationSettings(userId, updatedSettings);
|
||||
|
||||
// Отправляем обновленные настройки
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, {
|
||||
text: `✅ Время уведомлений установлено на ${this.getTimePreferenceText(timePreference)}`
|
||||
});
|
||||
|
||||
await this.sendNotificationSettings(callbackQuery.message.chat.id, updatedSettings);
|
||||
} catch (error) {
|
||||
console.error('Error handling time preference set:', error);
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Произошла ошибка при обновлении времени уведомлений.' });
|
||||
}
|
||||
}
|
||||
|
||||
// Обработка режима "Не беспокоить"
|
||||
async handleDndMode(callbackQuery: TelegramBot.CallbackQuery): Promise<void> {
|
||||
if (!callbackQuery.message) return;
|
||||
|
||||
await this.bot.editMessageText('🔕 *Режим "Не беспокоить"*\n\nВыберите действие:', {
|
||||
chat_id: callbackQuery.message.chat.id,
|
||||
message_id: callbackQuery.message.message_id,
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '✅ Включить', callback_data: 'notif_dnd_set:on' },
|
||||
{ text: '❌ Выключить', callback_data: 'notif_dnd_set:off' }
|
||||
],
|
||||
[
|
||||
{ text: '⏰ Настроить время', callback_data: 'notif_dnd_time' }
|
||||
],
|
||||
[
|
||||
{ text: '↩️ Назад', callback_data: 'notifications' }
|
||||
]
|
||||
]
|
||||
}
|
||||
});
|
||||
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id);
|
||||
}
|
||||
|
||||
// Обработка установки режима "Не беспокоить"
|
||||
async handleDndModeSet(callbackQuery: TelegramBot.CallbackQuery): Promise<void> {
|
||||
const telegramId = callbackQuery.from?.id.toString();
|
||||
if (!telegramId || !callbackQuery.message) return;
|
||||
|
||||
try {
|
||||
const userId = await this.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Вы не зарегистрированы.' });
|
||||
return;
|
||||
}
|
||||
|
||||
// notif_dnd_set:on/off
|
||||
const mode = callbackQuery.data?.split(':')[1];
|
||||
if (!mode) return;
|
||||
|
||||
const settings = await this.notificationService.getNotificationSettings(userId);
|
||||
// Копируем существующие настройки и обновляем нужное поле
|
||||
const existingSettings = settings as NotificationSettings;
|
||||
let updatedSettings: NotificationSettings = {
|
||||
...existingSettings,
|
||||
doNotDisturb: mode === 'on'
|
||||
};
|
||||
|
||||
// Если включаем режим "Не беспокоить", но не задано время, ставим дефолтные значения
|
||||
if (mode === 'on' && (!updatedSettings.doNotDisturbStart || !updatedSettings.doNotDisturbEnd)) {
|
||||
updatedSettings.doNotDisturbStart = '23:00';
|
||||
updatedSettings.doNotDisturbEnd = '08:00';
|
||||
}
|
||||
|
||||
// Обновляем настройки
|
||||
await this.notificationService.updateNotificationSettings(userId, updatedSettings);
|
||||
|
||||
// Отправляем обновленные настройки
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, {
|
||||
text: `✅ Режим "Не беспокоить" ${mode === 'on' ? 'включен' : 'выключен'}`
|
||||
});
|
||||
|
||||
await this.sendNotificationSettings(callbackQuery.message.chat.id, updatedSettings);
|
||||
} catch (error) {
|
||||
console.error('Error handling DND mode set:', error);
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Произошла ошибка при обновлении режима "Не беспокоить".' });
|
||||
}
|
||||
}
|
||||
|
||||
// Настройка времени для режима "Не беспокоить"
|
||||
async handleDndTimeSetup(callbackQuery: TelegramBot.CallbackQuery): Promise<void> {
|
||||
if (!callbackQuery.message) return;
|
||||
|
||||
await this.bot.editMessageText('⏰ *Настройка времени для режима "Не беспокоить"*\n\nВыберите один из предустановленных вариантов или введите свой:', {
|
||||
chat_id: callbackQuery.message.chat.id,
|
||||
message_id: callbackQuery.message.message_id,
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '🌙 23:00 - 08:00', callback_data: 'notif_dnd_time_set:23:00:08:00' }
|
||||
],
|
||||
[
|
||||
{ text: '🌙 22:00 - 07:00', callback_data: 'notif_dnd_time_set:22:00:07:00' }
|
||||
],
|
||||
[
|
||||
{ text: '🌙 00:00 - 09:00', callback_data: 'notif_dnd_time_set:00:00:09:00' }
|
||||
],
|
||||
[
|
||||
{ text: '✏️ Ввести свой вариант', callback_data: 'notif_dnd_time_custom' }
|
||||
],
|
||||
[
|
||||
{ text: '↩️ Назад', callback_data: 'notif_dnd' }
|
||||
]
|
||||
]
|
||||
}
|
||||
});
|
||||
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id);
|
||||
}
|
||||
|
||||
// Установка предустановленного времени для режима "Не беспокоить"
|
||||
async handleDndTimeSet(callbackQuery: TelegramBot.CallbackQuery): Promise<void> {
|
||||
const telegramId = callbackQuery.from?.id.toString();
|
||||
if (!telegramId || !callbackQuery.message) return;
|
||||
|
||||
try {
|
||||
const userId = await this.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Вы не зарегистрированы.' });
|
||||
return;
|
||||
}
|
||||
|
||||
// notif_dnd_time_set:startTime:endTime
|
||||
const parts = callbackQuery.data?.split(':');
|
||||
if (parts && parts.length >= 4) {
|
||||
const startTime = `${parts[2]}:${parts[3]}`;
|
||||
const endTime = `${parts[4]}:${parts[5]}`;
|
||||
|
||||
const settings = await this.notificationService.getNotificationSettings(userId);
|
||||
// Копируем существующие настройки и обновляем нужные поля
|
||||
const existingSettings = settings as NotificationSettings;
|
||||
const updatedSettings: NotificationSettings = {
|
||||
...existingSettings,
|
||||
doNotDisturb: true,
|
||||
doNotDisturbStart: startTime,
|
||||
doNotDisturbEnd: endTime
|
||||
};
|
||||
|
||||
// Обновляем настройки
|
||||
await this.notificationService.updateNotificationSettings(userId, updatedSettings);
|
||||
|
||||
// Отправляем обновленные настройки
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, {
|
||||
text: `✅ Время "Не беспокоить" установлено с ${startTime} до ${endTime}`
|
||||
});
|
||||
|
||||
await this.sendNotificationSettings(callbackQuery.message.chat.id, updatedSettings);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error handling DND time set:', error);
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Произошла ошибка при настройке времени "Не беспокоить".' });
|
||||
}
|
||||
}
|
||||
|
||||
// Запрос пользовательского времени для режима "Не беспокоить"
|
||||
async handleDndTimeCustom(callbackQuery: TelegramBot.CallbackQuery): Promise<void> {
|
||||
if (!callbackQuery.message) return;
|
||||
|
||||
// Устанавливаем ожидание пользовательского ввода
|
||||
const userId = callbackQuery.from?.id.toString();
|
||||
if (userId) {
|
||||
await this.setUserState(userId, 'waiting_dnd_time');
|
||||
}
|
||||
|
||||
await this.bot.editMessageText('⏰ *Введите время для режима "Не беспокоить"*\n\nУкажите время в формате:\n`с [ЧЧ:ММ] до [ЧЧ:ММ]`\n\nНапример: `с 23:30 до 07:00`', {
|
||||
chat_id: callbackQuery.message.chat.id,
|
||||
message_id: callbackQuery.message.message_id,
|
||||
parse_mode: 'Markdown',
|
||||
reply_markup: {
|
||||
inline_keyboard: [
|
||||
[
|
||||
{ text: '↩️ Отмена', callback_data: 'notif_dnd_time' }
|
||||
]
|
||||
]
|
||||
}
|
||||
});
|
||||
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id);
|
||||
}
|
||||
|
||||
// Обработка пользовательского ввода времени для режима "Не беспокоить"
|
||||
async handleDndTimeInput(msg: TelegramBot.Message): Promise<void> {
|
||||
const telegramId = msg.from?.id.toString();
|
||||
if (!telegramId) return;
|
||||
|
||||
try {
|
||||
const userId = await this.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.sendMessage(msg.chat.id, '❌ Вы не зарегистрированы. Используйте команду /start для регистрации.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Очищаем состояние ожидания
|
||||
await this.clearUserState(telegramId);
|
||||
|
||||
// Парсим введенное время
|
||||
const timeRegex = /с\s+(\d{1,2}[:\.]\d{2})\s+до\s+(\d{1,2}[:\.]\d{2})/i;
|
||||
const match = msg.text?.match(timeRegex);
|
||||
|
||||
if (match && match.length >= 3) {
|
||||
let startTime = match[1].replace('.', ':');
|
||||
let endTime = match[2].replace('.', ':');
|
||||
|
||||
// Проверяем и форматируем время
|
||||
if (this.isValidTime(startTime) && this.isValidTime(endTime)) {
|
||||
startTime = this.formatTime(startTime);
|
||||
endTime = this.formatTime(endTime);
|
||||
|
||||
const settings = await this.notificationService.getNotificationSettings(userId);
|
||||
// Копируем существующие настройки и обновляем нужные поля
|
||||
const existingSettings = settings as NotificationSettings;
|
||||
const updatedSettings: NotificationSettings = {
|
||||
...existingSettings,
|
||||
doNotDisturb: true,
|
||||
doNotDisturbStart: startTime,
|
||||
doNotDisturbEnd: endTime
|
||||
};
|
||||
|
||||
// Обновляем настройки
|
||||
await this.notificationService.updateNotificationSettings(userId, updatedSettings);
|
||||
|
||||
await this.bot.sendMessage(msg.chat.id, `✅ Время "Не беспокоить" установлено с ${startTime} до ${endTime}`);
|
||||
await this.sendNotificationSettings(msg.chat.id, updatedSettings);
|
||||
} else {
|
||||
await this.bot.sendMessage(msg.chat.id, '❌ Неверный формат времени. Пожалуйста, используйте формат ЧЧ:ММ (например, 23:30).');
|
||||
}
|
||||
} else {
|
||||
await this.bot.sendMessage(msg.chat.id, '❌ Неверный формат ввода. Пожалуйста, введите время в формате "с [ЧЧ:ММ] до [ЧЧ:ММ]" (например, "с 23:30 до 07:00").');
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error handling DND time input:', error);
|
||||
await this.bot.sendMessage(msg.chat.id, '❌ Произошла ошибка при настройке времени "Не беспокоить".');
|
||||
}
|
||||
}
|
||||
|
||||
// Проверка валидности времени
|
||||
private isValidTime(time: string): boolean {
|
||||
const regex = /^(\d{1,2}):(\d{2})$/;
|
||||
const match = time.match(regex);
|
||||
|
||||
if (match) {
|
||||
const hours = parseInt(match[1]);
|
||||
const minutes = parseInt(match[2]);
|
||||
return hours >= 0 && hours <= 23 && minutes >= 0 && minutes <= 59;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
// Форматирование времени в формат ЧЧ:ММ
|
||||
private formatTime(time: string): string {
|
||||
const [hours, minutes] = time.split(':').map(Number);
|
||||
return `${hours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}`;
|
||||
}
|
||||
|
||||
// Получение текстового представления времени
|
||||
private getTimePreferenceText(preference: string): string {
|
||||
switch (preference) {
|
||||
case 'morning': return 'Утро (9:00)';
|
||||
case 'afternoon': return 'День (13:00)';
|
||||
case 'evening': return 'Вечер (19:00)';
|
||||
case 'night': return 'Ночь (22:00)';
|
||||
default: return 'Вечер (19:00)';
|
||||
}
|
||||
}
|
||||
|
||||
// Получение названия настройки
|
||||
private getSettingName(setting: string): string {
|
||||
switch (setting) {
|
||||
case 'newMatches': return 'Новые матчи';
|
||||
case 'newMessages': return 'Новые сообщения';
|
||||
case 'newLikes': return 'Новые лайки';
|
||||
case 'reminders': return 'Напоминания';
|
||||
case 'dailySummary': return 'Ежедневные сводки';
|
||||
default: return setting;
|
||||
}
|
||||
}
|
||||
|
||||
// Получение ID пользователя по Telegram ID
|
||||
private async getUserIdByTelegramId(telegramId: string): Promise<string | null> {
|
||||
try {
|
||||
const result = await query(
|
||||
'SELECT id FROM users WHERE telegram_id = $1',
|
||||
[parseInt(telegramId)]
|
||||
);
|
||||
return result.rows.length > 0 ? result.rows[0].id : null;
|
||||
} catch (error) {
|
||||
console.error('Error getting user by telegram ID:', error);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
// Установка состояния ожидания пользователя
|
||||
private async setUserState(telegramId: string, state: string): Promise<void> {
|
||||
try {
|
||||
const userId = await this.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) return;
|
||||
|
||||
// Сначала проверяем, существуют ли столбцы state и state_data
|
||||
const checkColumnResult = await query(`
|
||||
SELECT column_name
|
||||
FROM information_schema.columns
|
||||
WHERE table_name = 'users' AND column_name = 'state'
|
||||
`);
|
||||
|
||||
if (checkColumnResult.rows.length === 0) {
|
||||
console.log('Adding state and state_data columns to users table...');
|
||||
// Добавляем столбцы, если их нет
|
||||
await query(`
|
||||
ALTER TABLE users ADD COLUMN IF NOT EXISTS state VARCHAR(255) NULL;
|
||||
ALTER TABLE users ADD COLUMN IF NOT EXISTS state_data JSONB DEFAULT '{}'::jsonb;
|
||||
`);
|
||||
}
|
||||
|
||||
// Теперь устанавливаем состояние
|
||||
await query(
|
||||
`UPDATE users
|
||||
SET state = $1,
|
||||
state_data = jsonb_set(COALESCE(state_data, '{}'::jsonb), '{timestamp}', to_jsonb(NOW()))
|
||||
WHERE telegram_id = $2`,
|
||||
[state, parseInt(telegramId)]
|
||||
);
|
||||
} catch (error) {
|
||||
console.error('Error setting user state:', error);
|
||||
}
|
||||
}
|
||||
|
||||
// Очистка состояния ожидания пользователя
|
||||
private async clearUserState(telegramId: string): Promise<void> {
|
||||
try {
|
||||
await query(
|
||||
'UPDATE users SET state = NULL WHERE telegram_id = $1',
|
||||
[parseInt(telegramId)]
|
||||
);
|
||||
} catch (error) {
|
||||
console.error('Error clearing user state:', error);
|
||||
}
|
||||
}
|
||||
|
||||
// Регистрация обработчиков уведомлений
|
||||
register(): void {
|
||||
// Команда настройки уведомлений
|
||||
this.bot.onText(/\/notifications/, this.handleNotificationsCommand.bind(this));
|
||||
|
||||
// Обработчик для кнопки настроек уведомлений в меню настроек
|
||||
this.bot.on('callback_query', async (callbackQuery) => {
|
||||
if (callbackQuery.data === 'notifications') {
|
||||
const telegramId = callbackQuery.from?.id.toString();
|
||||
if (!telegramId || !callbackQuery.message) return;
|
||||
|
||||
try {
|
||||
const userId = await this.getUserIdByTelegramId(telegramId);
|
||||
if (!userId) {
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Вы не зарегистрированы.' });
|
||||
return;
|
||||
}
|
||||
|
||||
const settings = await this.notificationService.getNotificationSettings(userId);
|
||||
await this.sendNotificationSettings(callbackQuery.message.chat.id, settings as NotificationSettings);
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id);
|
||||
} catch (error) {
|
||||
console.error('Error handling notifications callback:', error);
|
||||
await this.bot.answerCallbackQuery(callbackQuery.id, { text: '❌ Произошла ошибка при загрузке настроек уведомлений.' });
|
||||
}
|
||||
}
|
||||
else if (callbackQuery.data?.startsWith('notif_toggle:')) {
|
||||
await this.handleNotificationToggle(callbackQuery);
|
||||
}
|
||||
else if (callbackQuery.data === 'notif_time') {
|
||||
await this.handleTimePreference(callbackQuery);
|
||||
}
|
||||
else if (callbackQuery.data?.startsWith('notif_time_set:')) {
|
||||
await this.handleTimePreferenceSet(callbackQuery);
|
||||
}
|
||||
else if (callbackQuery.data === 'notif_dnd') {
|
||||
await this.handleDndMode(callbackQuery);
|
||||
}
|
||||
else if (callbackQuery.data?.startsWith('notif_dnd_set:')) {
|
||||
await this.handleDndModeSet(callbackQuery);
|
||||
}
|
||||
else if (callbackQuery.data === 'notif_dnd_time') {
|
||||
await this.handleDndTimeSetup(callbackQuery);
|
||||
}
|
||||
else if (callbackQuery.data?.startsWith('notif_dnd_time_set:')) {
|
||||
await this.handleDndTimeSet(callbackQuery);
|
||||
}
|
||||
else if (callbackQuery.data === 'notif_dnd_time_custom') {
|
||||
await this.handleDndTimeCustom(callbackQuery);
|
||||
}
|
||||
});
|
||||
|
||||
// Обработчик пользовательского ввода для времени "Не беспокоить"
|
||||
this.bot.on('message', async (msg) => {
|
||||
if (!msg.text) return;
|
||||
|
||||
const telegramId = msg.from?.id.toString();
|
||||
if (!telegramId) return;
|
||||
|
||||
try {
|
||||
// Сначала проверяем, существует ли столбец state
|
||||
const checkColumnResult = await query(`
|
||||
SELECT column_name
|
||||
FROM information_schema.columns
|
||||
WHERE table_name = 'users' AND column_name = 'state'
|
||||
`);
|
||||
|
||||
if (checkColumnResult.rows.length === 0) {
|
||||
console.log('State column does not exist in users table. Skipping state check.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Теперь проверяем состояние пользователя
|
||||
const result = await query(
|
||||
'SELECT state FROM users WHERE telegram_id = $1',
|
||||
[parseInt(telegramId)]
|
||||
);
|
||||
|
||||
if (result.rows.length > 0 && result.rows[0].state === 'waiting_dnd_time') {
|
||||
await this.handleDndTimeInput(msg);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error checking user state:', error);
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
@@ -1,4 +1,18 @@
|
||||
{
|
||||
"language": {
|
||||
"select": "🌍 Select interface language:\n\nYou can change the language later in settings.",
|
||||
"changed": "✅ Language changed to English",
|
||||
"ru": "🇷🇺 Русский",
|
||||
"en": "🇬🇧 English",
|
||||
"es": "🇪🇸 Español",
|
||||
"fr": "🇫🇷 Français",
|
||||
"de": "🇩🇪 Deutsch",
|
||||
"it": "🇮🇹 Italiano",
|
||||
"pt": "🇵🇹 Português",
|
||||
"zh": "🇨🇳 中文",
|
||||
"ja": "🇯🇵 日本語",
|
||||
"ko": "🇰🇷 한국어"
|
||||
},
|
||||
"welcome": {
|
||||
"greeting": "Welcome to Telegram Tinder Bot! 💕",
|
||||
"description": "Find your soulmate right here!",
|
||||
|
||||
@@ -1,4 +1,18 @@
|
||||
{
|
||||
"language": {
|
||||
"select": "🌍 Выберите язык интерфейса:\n\nВы сможете изменить язык позже в настройках.",
|
||||
"changed": "✅ Язык изменен на Русский",
|
||||
"ru": "🇷🇺 Русский",
|
||||
"en": "🇬🇧 English",
|
||||
"es": "🇪🇸 Español",
|
||||
"fr": "🇫🇷 Français",
|
||||
"de": "🇩🇪 Deutsch",
|
||||
"it": "🇮🇹 Italiano",
|
||||
"pt": "🇵🇹 Português",
|
||||
"zh": "🇨🇳 中文",
|
||||
"ja": "🇯🇵 日本語",
|
||||
"ko": "🇰🇷 한국어"
|
||||
},
|
||||
"welcome": {
|
||||
"greeting": "Добро пожаловать в Telegram Tinder Bot! 💕",
|
||||
"description": "Найди свою вторую половинку прямо здесь!",
|
||||
@@ -83,7 +97,12 @@
|
||||
"matches": "Взаимности",
|
||||
"premium": "Премиум",
|
||||
"settings": "Настройки",
|
||||
"help": "Помощь"
|
||||
"help": "Помощь",
|
||||
"notifications": "Уведомления"
|
||||
},
|
||||
"notifications": {
|
||||
"unavailable": "Функция настройки уведомлений недоступна.",
|
||||
"inDevelopment": "Функция в разработке!"
|
||||
},
|
||||
"buttons": {
|
||||
"back": "« Назад",
|
||||
@@ -94,7 +113,8 @@
|
||||
"edit": "Редактировать",
|
||||
"delete": "Удалить",
|
||||
"yes": "Да",
|
||||
"no": "Нет"
|
||||
"no": "Нет",
|
||||
"selectAction": "Выберите действие:"
|
||||
},
|
||||
"errors": {
|
||||
"profileNotFound": "Анкета не найдена",
|
||||
@@ -102,13 +122,29 @@
|
||||
"ageInvalid": "Введите корректный возраст (18-100)",
|
||||
"photoRequired": "Добавьте хотя бы одну фотографию",
|
||||
"networkError": "Ошибка сети. Попробуйте позже.",
|
||||
"serverError": "Ошибка сервера. Попробуйте позже."
|
||||
"serverError": "Ошибка сервера. Попробуйте позже.",
|
||||
"contextNotFound": "Контекст не найден. Повторите, пожалуйста.",
|
||||
"cityConfirmError": "Ошибка при подтверждении города",
|
||||
"generalError": "Ошибка",
|
||||
"tryAgain": "Произошла ошибка. Попробуйте еще раз."
|
||||
},
|
||||
"common": {
|
||||
"back": "👈 Назад"
|
||||
"back": "👈 Назад",
|
||||
"thisUser": "этим пользователем"
|
||||
},
|
||||
"matches": {
|
||||
"noMatches": "✨ У вас пока нет матчей.\n\n🔍 Попробуйте просмотреть больше анкет!\nИспользуйте /browse для поиска."
|
||||
"noMatches": "✨ У вас пока нет матчей.\n\n🔍 Попробуйте просмотреть больше анкет!\nИспользуйте /browse для поиска.",
|
||||
"title": "Ваши матчи ({count})",
|
||||
"mutualLike": "Вы понравились друг другу с {name}!\n\nТеперь вы можете начать общение!",
|
||||
"superLikeMatch": "Ваш супер лайк произвел впечатление на {name}!\n\nНачните общение первыми!",
|
||||
"likeBackMatch": "Теперь вы можете начать общение.",
|
||||
"likeNotification": "Если вы также понравитесь этому пользователю, будет создан матч.",
|
||||
"tryMoreProfiles": "Попробуйте просмотреть больше анкет!",
|
||||
"startBrowsing": "Начните просматривать анкеты и получите первые матчи!",
|
||||
"newMatch": "Новый матч",
|
||||
"youSaid": "Вы",
|
||||
"unmatchConfirm": "Вы больше не увидите этого пользователя в своих матчах.",
|
||||
"bioMissing": "Описание отсутствует"
|
||||
},
|
||||
"start": {
|
||||
"welcomeBack": "🎉 С возвращением, {name}!\n\n💖 Telegram Tinder Bot готов к работе!\n\nЧто хотите сделать?",
|
||||
|
||||
62
src/premium/README.md
Normal file
62
src/premium/README.md
Normal file
@@ -0,0 +1,62 @@
|
||||
# Модуль премиум-функций Telegram Tinder Bot
|
||||
|
||||
Этот каталог содержит модули и скрипты для управления премиум-функциями бота.
|
||||
|
||||
## Содержимое
|
||||
|
||||
- `add-premium-columns.js` - Добавление колонок для премиум-функций в базу данных (версия JavaScript)
|
||||
- `add-premium-columns.ts` - Добавление колонок для премиум-функций в базу данных (версия TypeScript)
|
||||
- `add-premium-columns-direct.js` - Прямое добавление премиум-колонок без миграций
|
||||
- `addPremiumColumn.js` - Добавление отдельной колонки премиум в таблицу пользователей
|
||||
- `setPremiumStatus.js` - Обновление статуса премиум для пользователей
|
||||
|
||||
## Премиум-функции
|
||||
|
||||
В боте реализованы следующие премиум-функции:
|
||||
|
||||
1. **Неограниченные лайки** - снятие дневного лимита на количество лайков
|
||||
2. **Супер-лайки** - возможность отправлять супер-лайки (повышенный приоритет)
|
||||
3. **Просмотр лайков** - возможность видеть, кто поставил лайк вашему профилю
|
||||
4. **Скрытый режим** - возможность скрывать свою активность
|
||||
5. **Расширенные фильтры** - дополнительные параметры для поиска
|
||||
|
||||
## Использование
|
||||
|
||||
### Добавление премиум-колонок в базу данных
|
||||
|
||||
```bash
|
||||
node src/premium/add-premium-columns.js
|
||||
```
|
||||
|
||||
### Изменение премиум-статуса пользователя
|
||||
|
||||
```typescript
|
||||
import { PremiumService } from '../services/premiumService';
|
||||
|
||||
// Установка премиум-статуса для пользователя
|
||||
const premiumService = new PremiumService();
|
||||
await premiumService.setPremiumStatus(userId, true, 30); // 30 дней премиума
|
||||
```
|
||||
|
||||
## Интеграция в основной код
|
||||
|
||||
Проверка премиум-статуса должна выполняться следующим образом:
|
||||
|
||||
```typescript
|
||||
// В классах контроллеров
|
||||
const isPremium = await this.premiumService.checkUserPremium(userId);
|
||||
|
||||
if (isPremium) {
|
||||
// Предоставить премиум-функцию
|
||||
} else {
|
||||
// Сообщить о необходимости премиум-подписки
|
||||
}
|
||||
```
|
||||
|
||||
## Период действия премиум-статуса
|
||||
|
||||
По умолчанию премиум-статус устанавливается на 30 дней. Для изменения срока используйте третий параметр в методе `setPremiumStatus`.
|
||||
|
||||
## Дополнительная информация
|
||||
|
||||
Более подробная информация о премиум-функциях содержится в документации проекта в каталоге `docs/VIP_FUNCTIONS.md`.
|
||||
@@ -3,13 +3,15 @@
|
||||
|
||||
const { Pool } = require('pg');
|
||||
|
||||
// Настройки подключения к базе данных - используем те же настройки, что и раньше
|
||||
// Настройки подключения к базе данных из переменных окружения
|
||||
require('dotenv').config();
|
||||
|
||||
const pool = new Pool({
|
||||
host: '192.168.0.102',
|
||||
port: 5432,
|
||||
database: 'telegram_tinder_bot',
|
||||
user: 'trevor',
|
||||
password: 'Cl0ud_1985!'
|
||||
host: process.env.DB_HOST || 'localhost',
|
||||
port: process.env.DB_PORT || 5432,
|
||||
database: process.env.DB_NAME || 'telegram_tinder_bot',
|
||||
user: process.env.DB_USERNAME || 'postgres',
|
||||
password: process.env.DB_PASSWORD || ''
|
||||
});
|
||||
|
||||
async function addPremiumColumns() {
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user