Files
remnawave-bedolaga-telegram…/app/webapi/routes/broadcasts.py
Fringg eb18994b7d fix: complete datetime.utcnow() → datetime.now(UTC) migration
- Migrate 660+ datetime.utcnow() across 153 files to datetime.now(UTC)
- Migrate 30+ datetime.now() without UTC to datetime.now(UTC)
- Convert all 170 DateTime columns to DateTime(timezone=True)
- Add migrate_datetime_to_timestamptz() in universal_migration with SET LOCAL timezone='UTC' safety
- Remove 70+ .replace(tzinfo=None) workarounds
- Fix utcfromtimestamp → fromtimestamp(..., tz=UTC)
- Fix fromtimestamp() without tz= (system_logs, backup_service, referral_diagnostics)
- Fix fromisoformat/isoparse to ensure aware output (platega, yookassa, wata, miniapp, nalogo)
- Fix strptime() to add .replace(tzinfo=UTC) (backup_service, referral_diagnostics)
- Fix datetime.combine() to include tzinfo=UTC (remnawave_sync, traffic_monitoring)
- Fix datetime.max/datetime.min sentinels with .replace(tzinfo=UTC)
- Rename panel_datetime_to_naive_utc → panel_datetime_to_utc
- Remove DTZ003 from ruff ignore list
2026-02-17 04:45:40 +03:00

147 lines
4.6 KiB
Python

from __future__ import annotations
from datetime import UTC, datetime
from typing import Any
from fastapi import APIRouter, Depends, HTTPException, Query, status
from sqlalchemy import func, select
from sqlalchemy.ext.asyncio import AsyncSession
from app.database.models import BroadcastHistory
from app.services.broadcast_service import (
BroadcastConfig,
BroadcastMediaConfig,
broadcast_service,
)
from ..dependencies import get_db_session, require_api_token
from ..schemas.broadcasts import (
BroadcastCreateRequest,
BroadcastListResponse,
BroadcastResponse,
)
router = APIRouter()
def _serialize_broadcast(broadcast: BroadcastHistory) -> BroadcastResponse:
return BroadcastResponse(
id=broadcast.id,
target_type=broadcast.target_type,
message_text=broadcast.message_text,
has_media=broadcast.has_media,
media_type=broadcast.media_type,
media_file_id=broadcast.media_file_id,
media_caption=broadcast.media_caption,
total_count=broadcast.total_count,
sent_count=broadcast.sent_count,
failed_count=broadcast.failed_count,
status=broadcast.status,
admin_id=broadcast.admin_id,
admin_name=broadcast.admin_name,
created_at=broadcast.created_at,
completed_at=broadcast.completed_at,
)
@router.post('', response_model=BroadcastResponse, status_code=status.HTTP_201_CREATED)
async def create_broadcast(
payload: BroadcastCreateRequest,
token: Any = Depends(require_api_token),
db: AsyncSession = Depends(get_db_session),
) -> BroadcastResponse:
message_text = payload.message_text.strip()
if not message_text:
raise HTTPException(status.HTTP_400_BAD_REQUEST, 'Message text must not be empty')
media_payload = payload.media
broadcast = BroadcastHistory(
target_type=payload.target,
message_text=message_text,
has_media=media_payload is not None,
media_type=media_payload.type if media_payload else None,
media_file_id=media_payload.file_id if media_payload else None,
media_caption=media_payload.caption if media_payload else None,
total_count=0,
sent_count=0,
failed_count=0,
status='queued',
admin_id=None,
admin_name=getattr(token, 'name', None) or getattr(token, 'created_by', None),
)
db.add(broadcast)
await db.commit()
await db.refresh(broadcast)
media_config = None
if media_payload:
media_config = BroadcastMediaConfig(
type=media_payload.type,
file_id=media_payload.file_id,
caption=media_payload.caption or message_text,
)
config = BroadcastConfig(
target=payload.target,
message_text=message_text,
selected_buttons=payload.selected_buttons,
media=media_config,
initiator_name=getattr(token, 'name', None) or getattr(token, 'created_by', None),
)
await broadcast_service.start_broadcast(broadcast.id, config)
await db.refresh(broadcast)
return _serialize_broadcast(broadcast)
@router.get('', response_model=BroadcastListResponse)
async def list_broadcasts(
_: Any = Depends(require_api_token),
db: AsyncSession = Depends(get_db_session),
limit: int = Query(50, ge=1, le=200),
offset: int = Query(0, ge=0),
) -> BroadcastListResponse:
total = await db.scalar(select(func.count(BroadcastHistory.id))) or 0
result = await db.execute(
select(BroadcastHistory).order_by(BroadcastHistory.created_at.desc()).offset(offset).limit(limit)
)
broadcasts = result.scalars().all()
return BroadcastListResponse(
items=[_serialize_broadcast(item) for item in broadcasts],
total=int(total),
limit=limit,
offset=offset,
)
@router.post('/{broadcast_id}/stop', response_model=BroadcastResponse)
async def stop_broadcast(
broadcast_id: int,
_: Any = Depends(require_api_token),
db: AsyncSession = Depends(get_db_session),
) -> BroadcastResponse:
broadcast = await db.get(BroadcastHistory, broadcast_id)
if not broadcast:
raise HTTPException(status.HTTP_404_NOT_FOUND, 'Broadcast not found')
if broadcast.status not in {'queued', 'in_progress', 'cancelling'}:
raise HTTPException(status.HTTP_400_BAD_REQUEST, 'Broadcast is not running')
is_running = await broadcast_service.request_stop(broadcast_id)
if is_running:
broadcast.status = 'cancelling'
else:
broadcast.status = 'cancelled'
broadcast.completed_at = datetime.now(UTC)
await db.commit()
await db.refresh(broadcast)
return _serialize_broadcast(broadcast)