185 lines
6.2 KiB
Python
185 lines
6.2 KiB
Python
from typing import List
|
|
from ninja import Router
|
|
from ninja.errors import HttpError
|
|
from django.shortcuts import get_object_or_404
|
|
from django.db.models import Sum, Prefetch
|
|
from django.db.models.functions import Coalesce
|
|
from django.db import transaction
|
|
|
|
from market.models import Market, MarketOption, UserBet, UserPointChange
|
|
from market.schemas import (
|
|
MarketListSchema,
|
|
ResolveMarketSchema,
|
|
UserBetCreateSchema,
|
|
UserBetSchema,
|
|
)
|
|
|
|
|
|
router = Router(tags=["market"])
|
|
|
|
|
|
@router.get("/", response=List[MarketListSchema])
|
|
def list_markets(request):
|
|
"""List all markets (excludes draft markets)."""
|
|
markets = Market.objects.exclude(status=Market.Status.DRAFT)
|
|
# Prefetch options with total_bets annotation sorted by total_bets desc, then text asc
|
|
options_queryset = MarketOption.objects.annotate(
|
|
total_bets=Coalesce(Sum("user_bets__amount"), 0)
|
|
).order_by("-total_bets", "text")
|
|
|
|
return markets.prefetch_related(Prefetch("options", queryset=options_queryset))
|
|
|
|
|
|
@router.get("/user/bets", response=List[UserBetSchema])
|
|
def list_user_bets(request):
|
|
"""List all bets placed by the current user."""
|
|
if not request.user.is_authenticated:
|
|
raise HttpError(401, "Authentication required")
|
|
|
|
return (
|
|
UserBet.objects.filter(user=request.user)
|
|
.select_related("option__market")
|
|
.prefetch_related("option__market__options")
|
|
)
|
|
|
|
|
|
@router.post("/{market_uuid}/actions/close")
|
|
def close_market(request, market_uuid: str):
|
|
"""Close a market. Admin only."""
|
|
if not request.user.is_staff:
|
|
raise HttpError(403, "Permission denied")
|
|
|
|
market = get_object_or_404(Market, uuid=market_uuid)
|
|
market.status = Market.Status.CLOSED
|
|
market.save(update_fields=["status", "updated_at"])
|
|
return {"status": Market.Status.CLOSED}
|
|
|
|
|
|
@router.post("/{market_uuid}/actions/resolve", response=MarketListSchema)
|
|
def resolve_market(request, market_uuid: str, payload: ResolveMarketSchema):
|
|
"""Resolve a market with a winning option. Admin only."""
|
|
if not request.user.is_staff:
|
|
raise HttpError(403, "Permission denied")
|
|
|
|
market = get_object_or_404(Market, uuid=market_uuid)
|
|
winning_option = get_object_or_404(MarketOption, uuid=payload.winning_option_uuid)
|
|
|
|
if winning_option.market_id != market.id:
|
|
raise HttpError(400, "Option does not belong to this market")
|
|
|
|
market.winning_option = winning_option
|
|
market.status = Market.Status.RESOLVED
|
|
market.save(update_fields=["winning_option", "status", "updated_at"])
|
|
|
|
# Calculate and distribute winnings
|
|
all_bets = list(
|
|
UserBet.objects.filter(option__market=market).select_related("user")
|
|
)
|
|
|
|
# Calculate total pot
|
|
total_pot = sum(bet.amount for bet in all_bets)
|
|
if total_pot == 0:
|
|
return market
|
|
|
|
# Separate winning and losing bets
|
|
winning_bets = [bet for bet in all_bets if bet.option_id == winning_option.id]
|
|
losing_bets = [bet for bet in all_bets if bet.option_id != winning_option.id]
|
|
|
|
total_winning = sum(bet.amount for bet in winning_bets)
|
|
|
|
point_changes = []
|
|
users_to_update = []
|
|
|
|
with transaction.atomic():
|
|
# Award payouts to winners with multiplier
|
|
if total_winning > 0:
|
|
for bet in winning_bets:
|
|
payout = round(
|
|
bet.amount / total_winning * total_pot * market.multiplier
|
|
)
|
|
bet.user.points += payout
|
|
users_to_update.append(bet.user)
|
|
point_changes.append(
|
|
UserPointChange(
|
|
user=bet.user,
|
|
market=market,
|
|
amount=payout,
|
|
reason=UserPointChange.Reason.BET_WON,
|
|
)
|
|
)
|
|
|
|
# Record losing bets (points already deducted)
|
|
for bet in losing_bets:
|
|
point_changes.append(
|
|
UserPointChange(
|
|
user=bet.user,
|
|
market=market,
|
|
amount=-bet.amount,
|
|
reason=UserPointChange.Reason.BET_LOST,
|
|
)
|
|
)
|
|
|
|
# Bulk update users
|
|
for user in users_to_update:
|
|
user.save(update_fields=["points"])
|
|
|
|
# Bulk create point changes
|
|
UserPointChange.objects.bulk_create(point_changes)
|
|
|
|
return market
|
|
|
|
|
|
@router.post("/{market_uuid}/bets", response=UserBetSchema)
|
|
def create_bet(request, market_uuid: str, payload: UserBetCreateSchema):
|
|
"""Place a bet on a market option."""
|
|
if not request.user.is_authenticated:
|
|
raise HttpError(401, "Authentication required")
|
|
|
|
market = get_object_or_404(Market, uuid=market_uuid)
|
|
option = get_object_or_404(MarketOption, uuid=payload.option_uuid)
|
|
|
|
if option.market_id != market.id:
|
|
raise HttpError(400, "Option does not belong to this market")
|
|
|
|
if market.status != Market.Status.OPEN:
|
|
raise HttpError(400, "Market is not open for betting")
|
|
|
|
# Check if user already has a bet on a different option in this market
|
|
existing_bet_on_market = (
|
|
UserBet.objects.filter(user=request.user, option__market=market)
|
|
.exclude(option=option)
|
|
.first()
|
|
)
|
|
if existing_bet_on_market:
|
|
raise HttpError(400, "You can only bet on one option per market")
|
|
|
|
# Check if user already has a bet on this option
|
|
existing_bet = UserBet.objects.filter(user=request.user, option=option).first()
|
|
if existing_bet and payload.amount < existing_bet.amount:
|
|
raise HttpError(400, "Cannot decrease bet amount. You can only increase it.")
|
|
|
|
# Calculate delta (amount to deduct from user's points)
|
|
delta = payload.amount - (existing_bet.amount if existing_bet else 0)
|
|
|
|
# Check if user has enough points
|
|
if request.user.points < delta:
|
|
raise HttpError(400, "Insufficient points for this bet")
|
|
|
|
user_bet, created = UserBet.objects.update_or_create(
|
|
user=request.user,
|
|
option=option,
|
|
defaults={"amount": payload.amount},
|
|
)
|
|
|
|
# Deduct points and record the change
|
|
request.user.points -= delta
|
|
request.user.save(update_fields=["points"])
|
|
UserPointChange.objects.create(
|
|
user=request.user,
|
|
market=market,
|
|
amount=-delta,
|
|
reason=UserPointChange.Reason.BET_PLACED,
|
|
)
|
|
|
|
return user_bet
|