# Kyler Olsen # CS 2450 Final Project # Apr 2025 from __future__ import annotations from typing import NamedTuple, Self, TYPE_CHECKING import select import datetime import network_utilities if TYPE_CHECKING: from socket import socket from library import Library class URL(NamedTuple): volume: str book: str chapter: int verse: int def __str__(self) -> str: if not self.volume: return "/" if not self.book: return f"/{self.volume}" if not self.chapter: return f"/{self.volume}/{self.book}" if not self.verse: return f"/{self.volume}/{self.book}/{self.chapter}" else: return f"/{self.volume}/{self.book}/{self.chapter}/{self.verse}" @classmethod def from_str(cls, url: str) -> Self: parts = url.strip('/').split('/') if len(parts) >= 1: volume = parts[0] else: volume = "" if len(parts) >= 2: book = parts[1] else: book = "" if len(parts) >= 3: chapter = int(parts[2]) else: chapter = 0 if len(parts) >= 4: verse = int(parts[3]) else: verse = 0 return cls(volume, book, chapter, verse) class Game: __library: Library __current_url: URL __clients: list[Player] __round_points: list[int] __total_scores: list[int] __active: bool __finished: bool __created: datetime.datetime def __init__(self, library: Library): self.__library = library self.__current_url = URL.from_str('') self.__clients = [] self.__round_points = [] self.__total_scores = [] self.__active = False self.__finished = False self.__created = datetime.datetime.now() @property def active(self) -> bool: return self.__active @property def finished(self) -> bool: return self.__finished def add_player(self, name: str, conn: socket): if not self.__active: self.__clients.append(Player(name, self, conn)) self.__total_scores.append(0) for player in self.__clients: player.player_joined(name) def start_game(self): self.__active = True def start_round(self, difficulty: int): if self.__active and not self.__finished: self.__round_points = [0] * len(self.__clients) self.__library.get_verse(difficulty, self) def new_verse(self, url: str, text: str): for player in self.__clients: self.__current_url = URL.from_str(url) player.new_verse(text) def guess_reference(self, url: str, player: Player): if self.__active and not self.__finished: if url == str(self.__current_url): player.guess_correct() for i, points in enumerate(self.__round_points): self.__total_scores[i] += points self.__clients[i].verse_guessed( points, str(self.__current_url), player.name) else: player.guess_incorrect() def end_game(self): self.__finished = True for player in self.__clients: player.game_over( [i.name for i in self.__clients], self.__total_scores) def update(self): if not self.__active and ( datetime.datetime.now() - self.__created >= datetime.timedelta(minutes=10) ) or ( datetime.datetime.now() - self.__created >= datetime.timedelta(1) ): self.__finished = True if not self.__finished: for player in self.__clients: player.update() class Player: __name: str __game: Game __client: socket def __init__(self, name: str, game: Game, conn: socket): self.__name = name self.__game = game self.__client = conn @property def name(self) -> str: return self.__name def player_joined(self, name: str): data = network_utilities.pack_varint(1) data += network_utilities.pack_string(name) self.__client.send(data) def new_verse(self, text: str): data = network_utilities.pack_varint(2) data += network_utilities.pack_string(text) self.__client.send(data) def guess_incorrect(self): data = network_utilities.pack_varint(3) self.__client.send(data) def guess_correct(self): data = network_utilities.pack_varint(4) self.__client.send(data) def verse_guessed(self, points: int, url: str, player: str): data = network_utilities.pack_varint(5) data += network_utilities.pack_varint(points) data += network_utilities.pack_string(url) data += network_utilities.pack_string(player) self.__client.send(data) def game_over(self, players: list[str], scores: list[int]): data = network_utilities.pack_varint(6) data += network_utilities.pack_string_array(players) data += network_utilities.pack_varint_array(scores) self.__client.send(data) self.__client.close() def update(self): ready_to_read, _, _ = select.select([self.__client], [], [], 0) if ready_to_read: packet_id = network_utilities.unpack_varint(self.__client) if packet_id == 2: self.__game.start_game() elif packet_id == 3: difficulty = network_utilities.unpack_varint(self.__client) self.__game.start_round(difficulty) elif packet_id == 4: url = network_utilities.unpack_string(self.__client) self.__game.guess_reference(url, self) elif packet_id == 5: self.__game.end_game()