Create initial wordle methods

pull/124/head
stijndcl 2022-07-25 22:58:02 +02:00
parent 2f4c2c347f
commit cbd3030565
4 changed files with 126 additions and 3 deletions

View File

@ -0,0 +1,23 @@
from typing import Optional
from database.mongo_types import MongoCollection
from database.schemas.mongo import WordleGame
__all__ = ["get_active_wordle_game", "make_wordle_guess", "start_new_wordle_game"]
async def get_active_wordle_game(collection: MongoCollection, user_id: int) -> Optional[WordleGame]:
"""Find a player's active game"""
return await collection.find_one({"user_id": user_id})
async def start_new_wordle_game(collection: MongoCollection, user_id: int, word: str) -> WordleGame:
"""Start a new game"""
game = WordleGame(user_id=user_id, word=word)
await collection.insert_one(game.dict(by_alias=True))
return game
async def make_wordle_guess(collection: MongoCollection, user_id: int, guess: str):
"""Make a guess in your current game"""
await collection.update_one({"user_id": user_id}, {"$push": {"guesses": guess}})

View File

@ -0,0 +1,6 @@
import motor.motor_asyncio
# Type aliases for the Motor types, which are way too long
MongoClient = motor.motor_asyncio.AsyncIOMotorClient
MongoDatabase = motor.motor_asyncio.AsyncIOMotorDatabase
MongoCollection = motor.motor_asyncio.AsyncIOMotorCollection

View File

@ -1,7 +1,12 @@
from bson import ObjectId import datetime
from pydantic import BaseModel, Field from abc import ABC, abstractmethod
from typing import Optional
__all__ = ["MongoBase"] from bson import ObjectId
from overrides import overrides
from pydantic import BaseModel, Field, conlist
__all__ = ["MongoBase", "WordleGame"]
class PyObjectId(str): class PyObjectId(str):
@ -36,3 +41,50 @@ class MongoBase(BaseModel):
arbitrary_types_allowed = True arbitrary_types_allowed = True
json_encoders = {ObjectId: str, PyObjectId: str} json_encoders = {ObjectId: str, PyObjectId: str}
use_enum_values = True use_enum_values = True
class MongoCollection(MongoBase, ABC):
"""Base model for the 'main class' in a collection
This field stores the name of the collection to avoid making typos against it
"""
@staticmethod
@abstractmethod
def collection() -> str:
raise NotImplementedError
class WordleStats(BaseModel):
"""Model that holds stats about a player's Wordle performance"""
guess_distribution: conlist(int, min_items=6, max_items=6) = Field(default_factory=lambda: [0, 0, 0, 0, 0, 0])
last_guess: Optional[datetime.date] = None
win_rate: float = 0
current_streak: int = 0
max_streak: int = 0
class GameStats(MongoCollection):
"""Collection that holds stats about how well a user has performed in games"""
user_id: int
wordle: Optional[WordleStats] = None
@staticmethod
@overrides
def collection() -> str:
return "game_stats"
class WordleGame(MongoCollection):
"""Collection that holds people's active Wordle games"""
user_id: int
word: str
guesses: conlist(str, min_items=0, max_items=6) = Field(default_factory=list)
@staticmethod
@overrides
def collection() -> str:
return "wordle"

View File

@ -0,0 +1,42 @@
import pytest
from database.crud import wordle as crud
from database.mongo_types import MongoCollection, MongoDatabase
from database.schemas.mongo import WordleGame
@pytest.fixture
async def wordle_collection(mongodb: MongoDatabase) -> MongoCollection:
"""Fixture to get a reference to the wordle collection"""
yield mongodb[WordleGame.collection()]
@pytest.fixture
async def wordle_game(wordle_collection: MongoCollection, test_user_id: int) -> WordleGame:
"""Fixture to create a new game"""
game = WordleGame(user_id=test_user_id, word="test")
await wordle_collection.insert_one(game.dict(by_alias=True))
yield game
async def test_start_new_game(wordle_collection: MongoCollection, test_user_id: int):
"""Test starting a new game"""
result = await wordle_collection.find_one({"user_id": test_user_id})
assert result is None
await crud.start_new_wordle_game(wordle_collection, test_user_id, "test")
result = await wordle_collection.find_one({"user_id": test_user_id})
assert result is not None
async def test_get_active_wordle_game_none(wordle_collection: MongoCollection, test_user_id: int):
"""Test getting an active game when there is none"""
result = await crud.get_active_wordle_game(wordle_collection, test_user_id)
assert result is None
async def test_get_active_wordle_game(wordle_collection: MongoCollection, wordle_game: WordleGame):
"""Test getting an active game when there is none"""
result = await crud.get_active_wordle_game(wordle_collection, wordle_game.user_id)
assert result == wordle_game.dict(by_alias=True)