game.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. from fastapi import APIRouter, Depends, HTTPException, BackgroundTasks
  2. from fastapi_camelcase import CamelModel
  3. from pydantic import conint, constr
  4. from sqlalchemy.orm import Session
  5. from .. import scoring
  6. from ..schemas import GameConfig, Guess
  7. from ..db import get_db, queries, models
  8. from ..gen import generate_points, restock_source
  9. router = APIRouter()
  10. class LinkedGame(CamelModel):
  11. linked_game: str
  12. class JoinGame(CamelModel):
  13. player_name: constr(min_length=1)
  14. class GuessResult(CamelModel):
  15. distance: int = None
  16. score: int = 0
  17. total_score: int
  18. @router.put("/")
  19. def create_game(config: GameConfig, bg: BackgroundTasks, db: Session = Depends(get_db)):
  20. coords = generate_points(config)
  21. game_id = queries.create_game(db, config, coords)
  22. bg.add_task(restock_source, config)
  23. return { "gameId": game_id }
  24. def get_game(game_id: str, db: Session = Depends(get_db)) -> models.Game:
  25. game = queries.get_game(db, game_id)
  26. if game is None:
  27. raise HTTPException(status_code=404, detail="Game not found")
  28. return game
  29. @router.get("/{game_id}/config", response_model=GameConfig)
  30. def get_game_config(game: models.Game = Depends(get_game)):
  31. return game
  32. @router.get("/{game_id}/coords")
  33. def get_game_coords(game: models.Game = Depends(get_game)):
  34. return {
  35. str(coord.round_number): {
  36. "lat": coord.latitude,
  37. "lng": coord.longitude,
  38. }
  39. for coord in game.coordinates
  40. }
  41. @router.post("/{game_id}/join")
  42. def join_game(game_id: str, join: JoinGame, db: Session = Depends(get_db)):
  43. get_game(game_id, db) # confirm this game exists
  44. player_id = queries.join_game(db, game_id, join.player_name)
  45. if player_id is None:
  46. raise HTTPException(status_code=409, detail="Player name in use")
  47. return { "playerId": player_id }
  48. @router.get("/{game_id}/players")
  49. def get_players(game: models.Game = Depends(get_game)):
  50. return { "players": [
  51. {
  52. "name": p.player_name,
  53. "currentRound": queries.get_next_round_number(p),
  54. "totalScore": queries.get_total_score(p),
  55. "guesses": {
  56. str(g.round_number): {
  57. "lat": g.latitude,
  58. "lng": g.longitude,
  59. "score": g.round_score,
  60. "timeRemaining": g.time_remaining,
  61. } for g in p.guesses
  62. }
  63. } for p in game.players
  64. ] }
  65. def get_player(game_id: str, player_id: str, db: Session = Depends(get_db)) -> models.Player:
  66. player = queries.get_player(db, player_id)
  67. if player is None:
  68. raise HTTPException(status_code=404, detail="Player not found")
  69. if player.game_id != game_id:
  70. raise HTTPException(status_code=404, detail="Player not in game")
  71. return player
  72. @router.get("/{game_id}/players/{player_id}/current")
  73. def get_current_round(db: Session = Depends(get_db), player: models.Player = Depends(get_player)):
  74. coord = queries.get_next_coordinate(db, player)
  75. if coord is None:
  76. return {
  77. "currentRound": None,
  78. "coord": None,
  79. "timer": None,
  80. }
  81. return {
  82. "currentRound": coord.round_number,
  83. "coord": {
  84. "lat": coord.latitude,
  85. "lng": coord.longitude,
  86. },
  87. "timer": queries.get_next_round_time(player),
  88. }
  89. @router.put("/{game_id}/linked", status_code=204)
  90. def set_linked_game(game_id: str, linked_game: LinkedGame, db: Session = Depends(get_db)):
  91. queries.link_game(db, game_id, linked_game.linked_game)
  92. @router.get("/{game_id}/linked")
  93. def get_linked_game(game: models.Game = Depends(get_game)):
  94. return { "linkedGame": game.linked_game }
  95. @router.get("/{game_id}/round/{round_number}/first")
  96. def get_first_submitter(game_id: str, round_number: conint(gt=0), db: Session = Depends(get_db)):
  97. return { "first": queries.get_first_submitter(db, game_id, round_number) }
  98. @router.post("/{game_id}/round/{round_number}/guess/{player_id}", response_model=GuessResult)
  99. def submit_guess(round_number: conint(gt=0), guess: Guess, db: Session = Depends(get_db), player: models.Player = Depends(get_player)):
  100. target = queries.get_coordinate(db, player.game_id, round_number)
  101. score, distance = scoring.score((target.latitude, target.longitude), (guess.lat, guess.lng))
  102. added = queries.add_guess(db, guess, player, round_number, score)
  103. if not added:
  104. raise HTTPException(status_code=409, detail="Already submitted guess for this round")
  105. total_score = queries.get_total_score(player)
  106. return GuessResult(distance=distance, score=score, total_score=total_score)
  107. @router.post("/{game_id}/round/{round_number}/timeout/{player_id}", response_model=GuessResult, response_model_include={"total_score"})
  108. def submit_timeout(round_number: conint(gt=0), db: Session = Depends(get_db), player: models.Player = Depends(get_player)):
  109. added = queries.add_timeout(db, player, round_number)
  110. if not added:
  111. raise HTTPException(status_code=409, detail="Already submitted guess for this round")
  112. total_score = queries.get_total_score(player)
  113. return GuessResult(total_score=total_score)