From ebe04bfa8fdf33ef0b31d1da1ba723a2071478d1 Mon Sep 17 00:00:00 2001 From: Paul-Louis NECH Date: Sat, 4 Apr 2020 19:04:58 +0200 Subject: [PATCH] refactor: Move server to subfolder --- .gitignore | 144 ------------------------------------------------------------------------------------------------------------------------------------------------ model/__init__.py | 0 model/animals.py | 93 --------------------------------------------------------------------------------------------- model/card.py | 52 ---------------------------------------------------- model/data.py | 125 ----------------------------------------------------------------------------------------------------------------------------- model/deck.py | 25 ------------------------- model/hand.py | 90 ------------------------------------------------------------------------------------------ model/hands.py | 59 ----------------------------------------------------------- model/known.py | 13 ------------- requirements.txt | 2 -- server/.gitignore | 146 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ server/model/__init__.py | 0 server/model/animals.py | 93 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ server/model/card.py | 52 ++++++++++++++++++++++++++++++++++++++++++++++++++++ server/model/data.py | 125 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ server/model/deck.py | 25 +++++++++++++++++++++++++ server/model/hand.py | 90 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ server/model/hands.py | 59 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ server/model/known.py | 13 +++++++++++++ server/requirements.txt | 2 ++ server/test/__init__.py | 0 server/test/test_data.py | 204 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ server/test/test_game.py | 36 ++++++++++++++++++++++++++++++++++++ test/__init__.py | 0 test/test_data.py | 204 ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ test/test_game.py | 36 ------------------------------------ 26 files changed, 845 insertions(+), 843 deletions(-) delete mode 100644 model/__init__.py delete mode 100644 model/animals.py delete mode 100644 model/card.py delete mode 100644 model/data.py delete mode 100644 model/deck.py delete mode 100644 model/hand.py delete mode 100644 model/hands.py delete mode 100644 model/known.py delete mode 100644 requirements.txt create mode 100644 server/.gitignore create mode 100644 server/model/__init__.py create mode 100644 server/model/animals.py create mode 100644 server/model/card.py create mode 100644 server/model/data.py create mode 100644 server/model/deck.py create mode 100644 server/model/hand.py create mode 100644 server/model/hands.py create mode 100644 server/model/known.py create mode 100644 server/requirements.txt create mode 100644 server/test/__init__.py create mode 100644 server/test/test_data.py create mode 100644 server/test/test_game.py delete mode 100644 test/__init__.py delete mode 100644 test/test_data.py delete mode 100644 test/test_game.py diff --git a/.gitignore b/.gitignore index 32df70c..1fd1dae 100644 --- a/.gitignore +++ b/.gitignore @@ -1,146 +1,2 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -pip-wheel-metadata/ -share/python-wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.nox/ -.coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -*.py,cover -.hypothesis/ -.pytest_cache/ -cover/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -.pybuilder/ -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# IPython -profile_default/ -ipython_config.py - -# pyenv -# For a library or package, you might want to ignore these files since the code is -# intended to run in multiple environments; otherwise, check them in: -# .python-version - -# pipenv -# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. -# However, in case of collaboration, if having platform-specific dependencies or dependencies -# having no cross-platform support, pipenv may install dependencies that don't work, or not -# install all needed dependencies. -#Pipfile.lock - -# PEP 582; used by e.g. github.com/David-OConnor/pyflow -__pypackages__/ - -# Celery stuff -celerybeat-schedule -celerybeat.pid - -# SageMath parsed files -*.sage.py - -# Environments -.env -.venv -env/ -venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ -.dmypy.json -dmypy.json - -# Pyre type checker -.pyre/ - -# pytype static type analyzer -.pytype/ - -# Cython debug symbols -cython_debug/ - -# static files generated from Django application using `collectstatic` -media -static - # IDE .idea/ diff --git a/model/__init__.py b/model/__init__.py deleted file mode 100644 index e69de29..0000000 --- a/model/__init__.py +++ /dev/null diff --git a/model/animals.py b/model/animals.py deleted file mode 100644 index a25fb2d..0000000 --- a/model/animals.py +++ /dev/null @@ -1,93 +0,0 @@ -import random - - -def random_animal_name(): - names = ["Aardvark", "Abyssinian", "Adelie Penguin", "Affenpinscher", "Afghan Hound", "African Bush Elephant", - "African Civet", "African Clawed Frog", "African Forest Elephant", "African Palm Civet", "African Penguin", - "African Tree Toad", "African Wild Dog", "Ainu Dog", "Airedale Terrier", "Akbash", "Akita", - "Alaskan Malamute", "Albatross", "Aldabra Giant Tortoise", "Alligator", "Alpine Dachsbracke", - "American Bulldog", "American Cocker Spaniel", "American Coonhound", "American Eskimo Dog", - "American Foxhound", "American Pit Bull Terrier", "American Staffordshire Terrier", - "American Water Spaniel", "Amur Leopard", "Anatolian Shepherd Dog", "Angelfish", "Ant", "Anteater", - "Antelope", "Appenzeller Dog", "Arctic Fox", "Arctic Hare", "Arctic Wolf", "Armadillo", "Asian Elephant", - "Asian Giant Hornet", "Asian Palm Civet", "Asiatic Black Bear", "Australian Cattle Dog", - "Australian Kelpie Dog", "Australian Mist", "Australian Shepherd", "Australian Terrier", "Avocet", - "Axolotl", "Aye Aye " "Baboon", "Bactrian Camel", "Badger", "Balinese", "Banded Palm Civet", - "Bandicoot", "Barb", "Barn Owl", "Barnacle", "Barracuda", "Basenji Dog", "Basking Shark", "Basset Hound", - "Bat", "Bavarian Mountain Hound", "Beagle", "Bear", "Bearded Collie", "Bearded Dragon", "Beaver", - "Bedlington Terrier", "Beetle", "Bengal Tiger", "Bernese Mountain Dog", "Bichon Frise", "Binturong", - "Bird", "Birds Of Paradise", "Birman", "Bison", "Black Rhinoceros", "Black Russian Terrier", - "Black Widow Spider", "Bloodhound", "Blue Lacy Dog", "Blue Whale", "Bluetick Coonhound", "Bobcat", - "Bolognese Dog", "Bombay", "Bongo", "Bonobo", "Booby", "Border Collie", "Border Terrier", - "Bornean Orang-utan", "Borneo Elephant", "Boston Terrier", "Bottlenose Dolphin", "Boxer Dog", - "Boykin Spaniel", "Brazilian Terrier", "Brown Bear", "Budgerigar", "Buffalo", "Bull Mastiff", "Bull Shark", - "Bull Terrier", "Bulldog", "Bullfrog", "Bumble Bee", "Burmese", "Burrowing Frog", "Butterfly", - "Butterfly Fish", "Caiman", "Caiman Lizard", "Cairn Terrier", "Camel", "Canaan Dog", "Capybara", "Caracal", - "Carolina Dog", "Cassowary", "Cat", "Caterpillar", "Catfish", "Cavalier King Charles Spaniel", "Centipede", - "Cesky Fousek", "Chameleon", "Chamois", "Cheetah", "Chesapeake Bay Retriever", "Chicken", "Chihuahua", - "Chimpanzee", "Chinchilla", "Chinese Crested Dog", "Chinook", "Chinstrap Penguin", "Chipmunk", "Chow Chow", - "Cichlid", "Clouded Leopard", "Clown Fish", "Clumber Spaniel", "Coati", "Cockroach", "Collared Peccary", - "Collie", "Common Buzzard", "Common Frog", "Common Loon", "Common Toad", "Coral", "Cottontop Tamarin", - "Cougar", "Cow", "Coyote", "Crab", "Crab-Eating Macaque", "Crane", "Crested Penguin", "Crocodile", - "Cross River Gorilla", "Curly Coated Retriever", "Cuscus", "Cuttlefish", "Dachshund", "Dalmatian", - "Darwin's Frog", "Deer", "Desert Tortoise", "Deutsche Bracke", "Dhole", "Dingo", "Discus", - "Doberman Pinscher", "Dodo", "Dog", "Dogo Argentino", "Dogue De Bordeaux", "Dolphin", "Donkey", "Dormouse", - "Dragonfly", "Drever", "Duck", "Dugong", "Dunker", "Dusky Dolphin", "Dwarf Crocodile", "Eagle", "Earwig", - "Eastern Gorilla", "Eastern Lowland Gorilla", "Echidna", "Edible Frog", "Egyptian Mau", "Electric Eel", - "Elephant", "Elephant Seal", "Elephant Shrew", "Emperor Penguin", "Emperor Tamarin", "Emu", - "English Cocker Spaniel", "English Shepherd", "English Springer Spaniel", "Entlebucher Mountain Dog", - "Epagneul Pont Audemer", "Eskimo Dog", "Estrela Mountain Dog", "Falcon", "Fennec Fox", "Ferret", - "Field Spaniel", "Fin Whale", "Finnish Spitz", "Fire-Bellied Toad", "Fish", "Fishing Cat", "Flamingo", - "Flat Coat Retriever", "Flounder", "Fly", "Flying Squirrel", "Fossa", "Fox", "Fox Terrier", - "French Bulldog", "Frigatebird", "Frilled Lizard", "Frog", "Fur Seal", "Galapagos Penguin", - "Galapagos Tortoise", "Gar", "Gecko", "Gentoo Penguin", "Geoffroys Tamarin", "Gerbil", "German Pinscher", - "German Shepherd", "Gharial", "Giant African Land Snail", "Giant Clam", "Giant Panda Bear", - "Giant Schnauzer", "Gibbon", "Gila Monster", "Giraffe", "Glass Lizard", "Glow Worm", "Goat", - "Golden Lion Tamarin", "Golden Oriole", "Golden Retriever", "Goose", "Gopher", "Gorilla", "Grasshopper", - "Great Dane", "Great White Shark", "Greater Swiss Mountain Dog", "Green Bee-Eater", "Greenland Dog", - "Grey Mouse Lemur", "Grey Reef Shark", "Grey Seal", "Greyhound", "Grizzly Bear", "Grouse", "Guinea Fowl", - "Guinea Pig", "Guppy", "Hammerhead Shark", "Hamster", "Hare", "Harrier", "Havanese", "Hedgehog", - "Hercules Beetle", "Hermit Crab", "Heron", "Highland Cattle", "Himalayan", "Hippopotamus", "Honey Bee", - "Horn Shark", "Horned Frog", "Horse", "Horseshoe Crab", "Howler Monkey", "Human", "Humboldt Penguin", - "Hummingbird", "Humpback Whale", "Hyena", "Ibis", "Ibizan Hound", "Iguana", "Impala", "Indian Elephant", - "Indian Palm Squirrel", "Indian Rhinoceros", "Indian Star Tortoise", "Indochinese Tiger", "Indri", - "Insect", "Irish Setter", "Irish WolfHound", "Jack Russel", "Jackal", "Jaguar", "Japanese Chin", - "Japanese Macaque", "Javan Rhinoceros", "Javanese", "Jellyfish", "Kakapo", "Kangaroo", - "Keel Billed Toucan", "Killer Whale", "King Crab", "King Penguin", "Kingfisher", "Kiwi", "Koala", - "Komodo Dragon", "Kudu", "Labradoodle", "Labrador Retriever", "Ladybug", "Leaf-Tailed Gecko", "Lemming", - "Lemur", "Leopard", "Leopard Cat", "Leopard Seal", "Leopard Tortoise", "Liger", "Lion", "Lionfish", - "Little Penguin", "Lizard", "Llama", "Lobster", "Long-Eared Owl", "Lynx", "Macaroni Penguin", "Macaw", - "Magellanic Penguin", "Magpie", "Maine Coon", "Malayan Civet", "Malayan Tiger", "Maltese", "Manatee", - "Mandrill", "Manta Ray", "Marine Toad", "Markhor", "Marsh Frog", "Masked Palm Civet", "Mastiff", "Mayfly", - "Meerkat", "Millipede", "Minke Whale", "Mole", "Molly", "Mongoose", "Mongrel", "Monitor Lizard", "Monkey", - "Monte Iberia Eleuth", "Moorhen", "Moose", "Moray Eel", "Moth", "Mountain Gorilla", "Mountain Lion", - "Mouse", "Mule", "Neanderthal", "Neapolitan Mastiff", "Newfoundland", "Newt", "Nightingale", - "Norfolk Terrier", "North American Black Bear", "Norwegian Forest", "Numbat", "Nurse Shark", "Ocelot", - "Octopus", "Okapi", "Old English Sheepdog", "Olm", "Opossum", "Orang-utan", "Ostrich", "Otter", "Oyster", - "Pademelon", "Panther", "Parrot", "Patas Monkey", "Peacock", "Pekingese", "Pelican", "Penguin", "Persian", - "Pheasant", "Pied Tamarin", "Pig", "Pika", "Pike", "Pink Fairy Armadillo", "Piranha", "Platypus", - "Pointer", "Poison Dart Frog", "Polar Bear", "Pond Skater", "Poodle", "Pool Frog", "Porcupine", "Possum", - "Prawn", "Proboscis Monkey", "Puffer Fish", "Puffin", "Pug", "Puma", "Purple Emperor", "Puss Moth", - "Pygmy Hippopotamus", "Pygmy Marmoset", "Quail", "Quetzal", "Quokka", "Quoll", "Rabbit", "Raccoon", - "Raccoon Dog", "Radiated Tortoise", "Ragdoll", "Rat", "Rattlesnake", "Red Knee Tarantula", "Red Panda", - "Red Wolf", "Red-handed Tamarin", "Reindeer", "Rhinoceros", "River Dolphin", "River Turtle", "Robin", - "Rock Hyrax", "Rockhopper Penguin", "Roseate Spoonbill", "Rottweiler", "Royal Penguin", "Russian Blue", - "Sabre-Toothed Tiger", "Saint Bernard", "Salamander", "Sand Lizard", "Saola", "Scorpion", "Scorpion Fish", - "Sea Dragon", "Sea Lion", "Sea Otter", "Sea Slug", "Sea Squirt", "Sea Turtle", "Sea Urchin", "Seahorse", - "Seal", "Serval", "Sheep", "Shih Tzu", "Shrimp", "Siamese", "Siamese Fighting Fish", "Siberian", - "Siberian Husky", "Siberian Tiger", "Silver Dollar", "Skunk", "Sloth", "Slow Worm", "Snail", "Snake", - "Snapping Turtle", "Snowshoe", "Snowy Owl", "Somali", "South China Tiger", "Spadefoot Toad", "Sparrow", - "Spectacled Bear", "Sperm Whale", "Spider Monkey", "Spiny Dogfish", "Sponge", "Squid", "Squirrel", - "Squirrel Monkey", "Sri Lankan Elephant", "Staffordshire Bull Terrier", "Stag Beetle", "Starfish", - "Stellers Sea Cow", "Stick Insect", "Stingray", "Stoat", "Striped Rocket Frog", "Sumatran Elephant", - "Sumatran Orang-utan" "Sumatran Rhinoceros", "Sumatran Tiger", "Sun Bear", "Swan", "Tang", - "Tapanuli Orang-utan", "Tapir", "Tarsier", "Tasmanian Devil", "Tawny Owl", "Termite", "Tetra", - "Thorny Devil", "Tibetan Mastiff", "Tiffany", "Tiger", "Tiger Salamander", "Tiger Shark", "Tortoise", - "Toucan", "Tree Frog", "Tropicbird", "Tuatara", "Turkey", "Turkish Angora", "Uakari", "Uguisu", - "Umbrellabird", "Vampire Bat", "Vervet Monkey", "Vulture", "Wallaby", "Walrus", "Warthog", "Wasp", - "Water Buffalo", "Water Dragon", "Water Vole", "Weasel", "Welsh Corgi", "West Highland Terrier", - "Western Gorilla", "Western Lowland Gorilla", "Whale Shark", "Whippet", "White Faced Capuchin", - "White Rhinoceros", "White Tiger", "Wild Boar", "Wildebeest", "Wolf", "Wolverine", "Wombat", "Woodlouse", - "Woodpecker", "Woolly Mammoth", "Woolly Monkey", "Wrasse", "X-Ray Tetra", "Yak", - "Yellow-Eyed Penguin" "Yorkshire Terrier", "Zebra", "Zebra Shark", "Zebu", "Zonkey", "Zorse"] - return random.choice(names) \ No newline at end of file diff --git a/model/card.py b/model/card.py deleted file mode 100644 index ad09ab1..0000000 --- a/model/card.py +++ /dev/null @@ -1,52 +0,0 @@ -from dataclasses import dataclass -from enum import Enum -from typing import Optional - - -class Value(Enum): - Two = 1 - Three = 2 - Four = 3 - Five = 4 - Six = 5 - Seven = 6 - Eight = 7 - Nine = 8 - Ten = 9 - Jack = 10 - Queen = 11 - King = 12 - Ace = 13 - - -class Color(Enum): - Hearts = "♥" - Spades = "♠" - Clubs = "♣" - Diamonds = "♦" - - -@dataclass(frozen=True) -class Card: - value: Value - color: Optional[Color] = None - - def __cmp__(self, other: "Card"): - my = self.score() - their = other.score() - return (my > their) - (my < their) - - def __lt__(self, other: "Card"): - return self.score() < other.score() - - def __gt__(self, other: "Card"): - return self.score() > other.score() - - def __eq__(self, other: "Card"): - return self.score() == other.score() - - def __str__(self) -> str: - return f"{self.value.name} of {self.color.value if self.color else '?'}" - - def score(self) -> int: - return int(self.value.value) \ No newline at end of file diff --git a/model/data.py b/model/data.py deleted file mode 100644 index 4a6d1ac..0000000 --- a/model/data.py +++ /dev/null @@ -1,125 +0,0 @@ -import itertools -from collections import defaultdict -from dataclasses import dataclass -from typing import List, Dict, Optional - -from model.animals import random_animal_name -from model.card import Card -from model.deck import Deck -from model.hand import Hand - - -@dataclass -class Announce: - bet: Optional[Hand] = None - - @property - def menteur(self): - return not self.bet - - -class Player: - def __init__(self, name: str = None): - if not name: - name = random_animal_name() - self.name: str = name - self.hand: Hand = Hand() - - def __str__(self): - return f"Player {self.name}" - - def give(self, card: Card): - self.hand.give(card) - - def announce(self, current_bet: Optional[Hand]) -> Announce: - """ A naive player that only trusts what they sees: - bets his hand, or menteur if his hand is lower. """ - if not current_bet or self.hand.value() > current_bet.value(): - return Announce(self.hand) - else: - return Announce() - - -class Game: - def __init__(self, players=None, deck: Deck = Deck()): - if players is None: - players = [] - - self.deck: Deck = deck - self.players: List[Player] = players - self.defeats: Dict[Player, int] = defaultdict(lambda: 0) - - @property - def global_hand(self) -> Hand: - return Hand([c for p in self.players for c in p.hand]) - - def new_game(self): - self.deck.reset() - - while len(self.players) > 1: - loser = self.new_turn() - - if self.defeats[loser] == 5: - print(f"{loser} is eliminated!") - self.players.remove(loser) - else: - print(f"{loser} lost the round, now playing with {self.defeats[loser] + 1} cards!") - - winner = self.players[0] - print(f"Game over - {winner.name} wins with {len(winner.hand)} cards!") - - def new_turn(self) -> Player: - """ - Runs a turn of the game. - - :return: the player that lost this turn. - """ - # Distribution - self.deck.reset() - for current_player in self.players: - current_player.give(self.deck.random_card()) - - # Tour - current_bet: Optional[Hand] = None - last_player = None - for current_player in itertools.cycle(self.players): - print(f"| {current_player}'s turn >") - if not current_bet: # First player, has to bet something - while not current_bet: - announce = current_player.announce(current_bet) - if announce.bet: - current_bet = announce.bet - print(f"{current_player} starts the round: {current_bet}") - else: # Next player, announce or menteur - announce = current_player.announce(current_bet) - if announce.bet: - while announce.bet.value() < current_bet.value(): # Bad announce! - print(f"Invalid bet, {announce.bet} < {current_bet}!") - announce = current_player.announce(current_bet) - current_bet = announce.bet - print(f" {current_player} bets {current_bet}.") - else: # Menteur! Who lost the round? - menteur = self.is_menteur(current_bet) - print(f"{current_player} says Menteur... {'Bravo!' if menteur else 'FAIL!'}") - loser = last_player if menteur else current_player - print(f"{loser} lost the round!") - self.count_defeat(loser) - return loser - last_player = current_player - - # TODO: Put next first player first of list - - def is_menteur(self, bet: Hand): - to_scan = [Card(c.value) for c in self.global_hand.cards.copy()] # Keep only values - to_find = [Card(c.value) for c in bet.cards] - for card in to_find: - if card in to_scan: - to_scan.remove(card) - continue - else: - print(f"Missing a {card}!") - print(f"Didn't find {bet} in {self.global_hand}: MENTEUR!") - return False # - - def count_defeat(self, loser): - self.defeats[loser] += 1 diff --git a/model/deck.py b/model/deck.py deleted file mode 100644 index 4cdf951..0000000 --- a/model/deck.py +++ /dev/null @@ -1,25 +0,0 @@ -from random import randrange, shuffle - -from model.card import Card, Value, Color - - -class Deck: - def __init__(self): - self.cards = [Card(v, c) for v in Value for c in Color] - self.defausse = [] - - def __len__(self): - return len(self.cards) - - def random_card(self) -> Card: - if not self.cards: - self.reset() - - card = self.cards.pop(randrange(len(self.cards))) - self.defausse.append(card) - return card - - def reset(self): - self.cards.extend(self.defausse) - self.defausse.clear() - shuffle(self.cards) \ No newline at end of file diff --git a/model/hand.py b/model/hand.py deleted file mode 100644 index ecc6699..0000000 --- a/model/hand.py +++ /dev/null @@ -1,90 +0,0 @@ -from collections import Counter -from typing import List - -from model.card import Card - - -class Hand: - def __init__(self, cards: List[Card] = None): - if cards is None: - cards = [] - self.cards: List[Card] = cards - - def __contains__(self, item: Card): - return item in self.cards - - def __len__(self) -> int: - return len(self.cards) - - def __getitem__(self, item): - return self.cards[item] - - def __repr__(self): - return "|".join([str(c) for c in self.cards]) - - def give(self, other: Card): - self.cards.append(other) - - def value(self): - counter = Counter([c.value for c in self.cards]) - - has_pair = False - has_double_pair = False - has_brelan = False - has_full = False - has_carre = False - - highest_card = None - pair = None - double_pair = None - brelan = None - carre = None - - for element, count in counter.items(): - element_cards = [c for c in self.cards if c.value == element] - card = element_cards[0] # Note we take a random color - highest_card = max(highest_card, card) if highest_card else card - - if count == 2: - if has_pair: - has_double_pair = True - double_pair = max(pair, card) - pair = min(pair, card) - else: - has_pair = True - pair = max(pair, card) if pair else card - if count == 3: - has_brelan = True - brelan = max(brelan, card) if brelan else card - if has_brelan and has_pair: - has_full = True - if count == 4: - has_carre = True - carre = max(carre, card) if carre else card - - analysis_log = " | ".join([ - f"ANALYSIS", - f"Carre[{carre}]" if has_carre else "no carre", - f"Full[{brelan}|{pair}]" if has_full else "no full", - f"Brelan[{brelan}]" if has_brelan else "no carre", - f"Double paire[{double_pair}|{pair}]" if has_double_pair else "no Dpaire", - f"Paire[{pair}]" if has_pair else "no paire", - f"Card[{highest_card}]" - ]) - # Finished counting, let's return scores - if has_carre: - score = (20 ** 5) * carre.score() - elif has_full: - score = (20 ** 4) * brelan.score() + 20 * pair.score() - elif has_brelan: - score = (20 ** 3) * brelan.score() - elif has_double_pair: - score = (20 ** 2) * double_pair.score() + pair.score() - elif has_pair: - score = 20 * pair.score() - else: - score = highest_card.score() - - analysis_log += f"\t-> score=\t{score}" - # print(analysis_log) - return score \ No newline at end of file diff --git a/model/hands.py b/model/hands.py deleted file mode 100644 index 34a4666..0000000 --- a/model/hands.py +++ /dev/null @@ -1,59 +0,0 @@ -from model.hand import Hand -from model.card import Value, Color, Card - - -def carre(value) -> Hand: - return Hand([ - Card(value, Color.Hearts), - Card(value, Color.Clubs), - Card(value, Color.Diamonds), - Card(value, Color.Spades) - ]) - - -def full(value_aux: Value, - value_par: Value) -> Hand: - return Hand([ - Card(value_aux, Color.Hearts), - Card(value_aux, Color.Clubs), - Card(value_aux, Color.Spades), - Card(value_par, Color.Hearts), - Card(value_par, Color.Clubs) - ]) - - -def brelan(value) -> Hand: - return Hand([ - Card(value, Color.Hearts), - Card(value, Color.Clubs), - Card(value, Color.Diamonds) - ]) - - -def pair(value) -> Hand: - return Hand([ - Card(value, Color.Hearts), - Card(value, Color.Clubs) - ]) - - -def single(low_value) -> Hand: - return Hand([ - Card(low_value, Color.Hearts) - ]) - - -def double_pair(value: Value, other: Value = None): - # Ensure no carre - if not other or other == value: - other = Value.Two if value == Value.Three else Value.Three - assert other != value - - return Hand([ - # Pair of value - Card(value, Color.Hearts), - Card(value, Color.Clubs), - # And pair of twos or threes - Card(other, Color.Hearts), - Card(other, Color.Clubs) - ]) diff --git a/model/known.py b/model/known.py deleted file mode 100644 index d43183d..0000000 --- a/model/known.py +++ /dev/null @@ -1,13 +0,0 @@ -from model.card import Value, Color, Card -from model.hands import pair, single, double_pair, brelan, full, carre - -ACE_OF_HEARTS = Card(Value.Ace, Color.Hearts) -ACE_OF_SPADES = Card(Value.Ace, Color.Spades) -ACE_OF_DIAMONDS = Card(Value.Ace, Color.Diamonds) -ACE_OF_CLUBS = Card(Value.Ace, Color.Clubs) -PAIR_ACE = pair(Value.Ace) -SINGLE_ACE = single(Value.Ace) -DOUBLE_PAIR_ACE = double_pair(Value.Ace) -BRELAN_ACE = brelan(Value.Ace) -FULL_ACE = full(Value.Ace, Value.King) -CARRE_ACE = carre(Value.Ace) diff --git a/requirements.txt b/requirements.txt deleted file mode 100644 index f536c03..0000000 --- a/requirements.txt +++ /dev/null @@ -1,2 +0,0 @@ -fastapi==0.53.2 -uvicorn==0.11.3 diff --git a/server/.gitignore b/server/.gitignore new file mode 100644 index 0000000..8df6e1b --- /dev/null +++ b/server/.gitignore @@ -0,0 +1,146 @@ +# Byte-compiled / optimized / DLL files +__pycache__/ +*.py[cod] +*$py.class + +# C extensions +*.so + +# Distribution / packaging +.Python +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +pip-wheel-metadata/ +share/python-wheels/ +*.egg-info/ +.installed.cfg +*.egg +MANIFEST + +# PyInstaller +# Usually these files are written by a python script from a template +# before PyInstaller builds the exe, so as to inject date/other infos into it. +*.manifest +*.spec + +# Installer logs +pip-log.txt +pip-delete-this-directory.txt + +# Unit test / coverage reports +htmlcov/ +.tox/ +.nox/ +.coverage +.coverage.* +.cache +nosetests.xml +coverage.xml +*.cover +*.py,cover +.hypothesis/ +.pytest_cache/ +cover/ + +# Translations +*.mo +*.pot + +# Django stuff: +*.log +local_settings.py +db.sqlite3 +db.sqlite3-journal + +# Flask stuff: +instance/ +.webassets-cache + +# Scrapy stuff: +.scrapy + +# Sphinx documentation +docs/_build/ + +# PyBuilder +.pybuilder/ +target/ + +# Jupyter Notebook +.ipynb_checkpoints + +# IPython +profile_default/ +ipython_config.py + +# pyenv +# For a library or package, you might want to ignore these files since the code is +# intended to run in multiple environments; otherwise, check them in: +# .python-version + +# pipenv +# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. +# However, in case of collaboration, if having platform-specific dependencies or dependencies +# having no cross-platform support, pipenv may install dependencies that don't work, or not +# install all needed dependencies. +#Pipfile.lock + +# PEP 582; used by e.g. github.com/David-OConnor/pyflow +__pypackages__/ + +# Celery stuff +celerybeat-schedule +celerybeat.pid + +# SageMath parsed files +*.sage.py + +# Environments +.env +.venv +env/ +venv/ +ENV/ +env.bak/ +venv.bak/ + +# Spyder project settings +.spyderproject +.spyproject + +# Rope project settings +.ropeproject + +# mkdocs documentation +/site + +# mypy +.mypy_cache/ +.dmypy.json +dmypy.json + +# Pyre type checker +.pyre/ + +# pytype static type analyzer +.pytype/ + +# Cython debug symbols +cython_debug/ + +# static files generated from Django application using `collectstatic` +media +static + +# IDE +../.idea/ diff --git a/server/model/__init__.py b/server/model/__init__.py new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/server/model/__init__.py diff --git a/server/model/animals.py b/server/model/animals.py new file mode 100644 index 0000000..a25fb2d --- /dev/null +++ b/server/model/animals.py @@ -0,0 +1,93 @@ +import random + + +def random_animal_name(): + names = ["Aardvark", "Abyssinian", "Adelie Penguin", "Affenpinscher", "Afghan Hound", "African Bush Elephant", + "African Civet", "African Clawed Frog", "African Forest Elephant", "African Palm Civet", "African Penguin", + "African Tree Toad", "African Wild Dog", "Ainu Dog", "Airedale Terrier", "Akbash", "Akita", + "Alaskan Malamute", "Albatross", "Aldabra Giant Tortoise", "Alligator", "Alpine Dachsbracke", + "American Bulldog", "American Cocker Spaniel", "American Coonhound", "American Eskimo Dog", + "American Foxhound", "American Pit Bull Terrier", "American Staffordshire Terrier", + "American Water Spaniel", "Amur Leopard", "Anatolian Shepherd Dog", "Angelfish", "Ant", "Anteater", + "Antelope", "Appenzeller Dog", "Arctic Fox", "Arctic Hare", "Arctic Wolf", "Armadillo", "Asian Elephant", + "Asian Giant Hornet", "Asian Palm Civet", "Asiatic Black Bear", "Australian Cattle Dog", + "Australian Kelpie Dog", "Australian Mist", "Australian Shepherd", "Australian Terrier", "Avocet", + "Axolotl", "Aye Aye " "Baboon", "Bactrian Camel", "Badger", "Balinese", "Banded Palm Civet", + "Bandicoot", "Barb", "Barn Owl", "Barnacle", "Barracuda", "Basenji Dog", "Basking Shark", "Basset Hound", + "Bat", "Bavarian Mountain Hound", "Beagle", "Bear", "Bearded Collie", "Bearded Dragon", "Beaver", + "Bedlington Terrier", "Beetle", "Bengal Tiger", "Bernese Mountain Dog", "Bichon Frise", "Binturong", + "Bird", "Birds Of Paradise", "Birman", "Bison", "Black Rhinoceros", "Black Russian Terrier", + "Black Widow Spider", "Bloodhound", "Blue Lacy Dog", "Blue Whale", "Bluetick Coonhound", "Bobcat", + "Bolognese Dog", "Bombay", "Bongo", "Bonobo", "Booby", "Border Collie", "Border Terrier", + "Bornean Orang-utan", "Borneo Elephant", "Boston Terrier", "Bottlenose Dolphin", "Boxer Dog", + "Boykin Spaniel", "Brazilian Terrier", "Brown Bear", "Budgerigar", "Buffalo", "Bull Mastiff", "Bull Shark", + "Bull Terrier", "Bulldog", "Bullfrog", "Bumble Bee", "Burmese", "Burrowing Frog", "Butterfly", + "Butterfly Fish", "Caiman", "Caiman Lizard", "Cairn Terrier", "Camel", "Canaan Dog", "Capybara", "Caracal", + "Carolina Dog", "Cassowary", "Cat", "Caterpillar", "Catfish", "Cavalier King Charles Spaniel", "Centipede", + "Cesky Fousek", "Chameleon", "Chamois", "Cheetah", "Chesapeake Bay Retriever", "Chicken", "Chihuahua", + "Chimpanzee", "Chinchilla", "Chinese Crested Dog", "Chinook", "Chinstrap Penguin", "Chipmunk", "Chow Chow", + "Cichlid", "Clouded Leopard", "Clown Fish", "Clumber Spaniel", "Coati", "Cockroach", "Collared Peccary", + "Collie", "Common Buzzard", "Common Frog", "Common Loon", "Common Toad", "Coral", "Cottontop Tamarin", + "Cougar", "Cow", "Coyote", "Crab", "Crab-Eating Macaque", "Crane", "Crested Penguin", "Crocodile", + "Cross River Gorilla", "Curly Coated Retriever", "Cuscus", "Cuttlefish", "Dachshund", "Dalmatian", + "Darwin's Frog", "Deer", "Desert Tortoise", "Deutsche Bracke", "Dhole", "Dingo", "Discus", + "Doberman Pinscher", "Dodo", "Dog", "Dogo Argentino", "Dogue De Bordeaux", "Dolphin", "Donkey", "Dormouse", + "Dragonfly", "Drever", "Duck", "Dugong", "Dunker", "Dusky Dolphin", "Dwarf Crocodile", "Eagle", "Earwig", + "Eastern Gorilla", "Eastern Lowland Gorilla", "Echidna", "Edible Frog", "Egyptian Mau", "Electric Eel", + "Elephant", "Elephant Seal", "Elephant Shrew", "Emperor Penguin", "Emperor Tamarin", "Emu", + "English Cocker Spaniel", "English Shepherd", "English Springer Spaniel", "Entlebucher Mountain Dog", + "Epagneul Pont Audemer", "Eskimo Dog", "Estrela Mountain Dog", "Falcon", "Fennec Fox", "Ferret", + "Field Spaniel", "Fin Whale", "Finnish Spitz", "Fire-Bellied Toad", "Fish", "Fishing Cat", "Flamingo", + "Flat Coat Retriever", "Flounder", "Fly", "Flying Squirrel", "Fossa", "Fox", "Fox Terrier", + "French Bulldog", "Frigatebird", "Frilled Lizard", "Frog", "Fur Seal", "Galapagos Penguin", + "Galapagos Tortoise", "Gar", "Gecko", "Gentoo Penguin", "Geoffroys Tamarin", "Gerbil", "German Pinscher", + "German Shepherd", "Gharial", "Giant African Land Snail", "Giant Clam", "Giant Panda Bear", + "Giant Schnauzer", "Gibbon", "Gila Monster", "Giraffe", "Glass Lizard", "Glow Worm", "Goat", + "Golden Lion Tamarin", "Golden Oriole", "Golden Retriever", "Goose", "Gopher", "Gorilla", "Grasshopper", + "Great Dane", "Great White Shark", "Greater Swiss Mountain Dog", "Green Bee-Eater", "Greenland Dog", + "Grey Mouse Lemur", "Grey Reef Shark", "Grey Seal", "Greyhound", "Grizzly Bear", "Grouse", "Guinea Fowl", + "Guinea Pig", "Guppy", "Hammerhead Shark", "Hamster", "Hare", "Harrier", "Havanese", "Hedgehog", + "Hercules Beetle", "Hermit Crab", "Heron", "Highland Cattle", "Himalayan", "Hippopotamus", "Honey Bee", + "Horn Shark", "Horned Frog", "Horse", "Horseshoe Crab", "Howler Monkey", "Human", "Humboldt Penguin", + "Hummingbird", "Humpback Whale", "Hyena", "Ibis", "Ibizan Hound", "Iguana", "Impala", "Indian Elephant", + "Indian Palm Squirrel", "Indian Rhinoceros", "Indian Star Tortoise", "Indochinese Tiger", "Indri", + "Insect", "Irish Setter", "Irish WolfHound", "Jack Russel", "Jackal", "Jaguar", "Japanese Chin", + "Japanese Macaque", "Javan Rhinoceros", "Javanese", "Jellyfish", "Kakapo", "Kangaroo", + "Keel Billed Toucan", "Killer Whale", "King Crab", "King Penguin", "Kingfisher", "Kiwi", "Koala", + "Komodo Dragon", "Kudu", "Labradoodle", "Labrador Retriever", "Ladybug", "Leaf-Tailed Gecko", "Lemming", + "Lemur", "Leopard", "Leopard Cat", "Leopard Seal", "Leopard Tortoise", "Liger", "Lion", "Lionfish", + "Little Penguin", "Lizard", "Llama", "Lobster", "Long-Eared Owl", "Lynx", "Macaroni Penguin", "Macaw", + "Magellanic Penguin", "Magpie", "Maine Coon", "Malayan Civet", "Malayan Tiger", "Maltese", "Manatee", + "Mandrill", "Manta Ray", "Marine Toad", "Markhor", "Marsh Frog", "Masked Palm Civet", "Mastiff", "Mayfly", + "Meerkat", "Millipede", "Minke Whale", "Mole", "Molly", "Mongoose", "Mongrel", "Monitor Lizard", "Monkey", + "Monte Iberia Eleuth", "Moorhen", "Moose", "Moray Eel", "Moth", "Mountain Gorilla", "Mountain Lion", + "Mouse", "Mule", "Neanderthal", "Neapolitan Mastiff", "Newfoundland", "Newt", "Nightingale", + "Norfolk Terrier", "North American Black Bear", "Norwegian Forest", "Numbat", "Nurse Shark", "Ocelot", + "Octopus", "Okapi", "Old English Sheepdog", "Olm", "Opossum", "Orang-utan", "Ostrich", "Otter", "Oyster", + "Pademelon", "Panther", "Parrot", "Patas Monkey", "Peacock", "Pekingese", "Pelican", "Penguin", "Persian", + "Pheasant", "Pied Tamarin", "Pig", "Pika", "Pike", "Pink Fairy Armadillo", "Piranha", "Platypus", + "Pointer", "Poison Dart Frog", "Polar Bear", "Pond Skater", "Poodle", "Pool Frog", "Porcupine", "Possum", + "Prawn", "Proboscis Monkey", "Puffer Fish", "Puffin", "Pug", "Puma", "Purple Emperor", "Puss Moth", + "Pygmy Hippopotamus", "Pygmy Marmoset", "Quail", "Quetzal", "Quokka", "Quoll", "Rabbit", "Raccoon", + "Raccoon Dog", "Radiated Tortoise", "Ragdoll", "Rat", "Rattlesnake", "Red Knee Tarantula", "Red Panda", + "Red Wolf", "Red-handed Tamarin", "Reindeer", "Rhinoceros", "River Dolphin", "River Turtle", "Robin", + "Rock Hyrax", "Rockhopper Penguin", "Roseate Spoonbill", "Rottweiler", "Royal Penguin", "Russian Blue", + "Sabre-Toothed Tiger", "Saint Bernard", "Salamander", "Sand Lizard", "Saola", "Scorpion", "Scorpion Fish", + "Sea Dragon", "Sea Lion", "Sea Otter", "Sea Slug", "Sea Squirt", "Sea Turtle", "Sea Urchin", "Seahorse", + "Seal", "Serval", "Sheep", "Shih Tzu", "Shrimp", "Siamese", "Siamese Fighting Fish", "Siberian", + "Siberian Husky", "Siberian Tiger", "Silver Dollar", "Skunk", "Sloth", "Slow Worm", "Snail", "Snake", + "Snapping Turtle", "Snowshoe", "Snowy Owl", "Somali", "South China Tiger", "Spadefoot Toad", "Sparrow", + "Spectacled Bear", "Sperm Whale", "Spider Monkey", "Spiny Dogfish", "Sponge", "Squid", "Squirrel", + "Squirrel Monkey", "Sri Lankan Elephant", "Staffordshire Bull Terrier", "Stag Beetle", "Starfish", + "Stellers Sea Cow", "Stick Insect", "Stingray", "Stoat", "Striped Rocket Frog", "Sumatran Elephant", + "Sumatran Orang-utan" "Sumatran Rhinoceros", "Sumatran Tiger", "Sun Bear", "Swan", "Tang", + "Tapanuli Orang-utan", "Tapir", "Tarsier", "Tasmanian Devil", "Tawny Owl", "Termite", "Tetra", + "Thorny Devil", "Tibetan Mastiff", "Tiffany", "Tiger", "Tiger Salamander", "Tiger Shark", "Tortoise", + "Toucan", "Tree Frog", "Tropicbird", "Tuatara", "Turkey", "Turkish Angora", "Uakari", "Uguisu", + "Umbrellabird", "Vampire Bat", "Vervet Monkey", "Vulture", "Wallaby", "Walrus", "Warthog", "Wasp", + "Water Buffalo", "Water Dragon", "Water Vole", "Weasel", "Welsh Corgi", "West Highland Terrier", + "Western Gorilla", "Western Lowland Gorilla", "Whale Shark", "Whippet", "White Faced Capuchin", + "White Rhinoceros", "White Tiger", "Wild Boar", "Wildebeest", "Wolf", "Wolverine", "Wombat", "Woodlouse", + "Woodpecker", "Woolly Mammoth", "Woolly Monkey", "Wrasse", "X-Ray Tetra", "Yak", + "Yellow-Eyed Penguin" "Yorkshire Terrier", "Zebra", "Zebra Shark", "Zebu", "Zonkey", "Zorse"] + return random.choice(names) \ No newline at end of file diff --git a/server/model/card.py b/server/model/card.py new file mode 100644 index 0000000..ad09ab1 --- /dev/null +++ b/server/model/card.py @@ -0,0 +1,52 @@ +from dataclasses import dataclass +from enum import Enum +from typing import Optional + + +class Value(Enum): + Two = 1 + Three = 2 + Four = 3 + Five = 4 + Six = 5 + Seven = 6 + Eight = 7 + Nine = 8 + Ten = 9 + Jack = 10 + Queen = 11 + King = 12 + Ace = 13 + + +class Color(Enum): + Hearts = "♥" + Spades = "♠" + Clubs = "♣" + Diamonds = "♦" + + +@dataclass(frozen=True) +class Card: + value: Value + color: Optional[Color] = None + + def __cmp__(self, other: "Card"): + my = self.score() + their = other.score() + return (my > their) - (my < their) + + def __lt__(self, other: "Card"): + return self.score() < other.score() + + def __gt__(self, other: "Card"): + return self.score() > other.score() + + def __eq__(self, other: "Card"): + return self.score() == other.score() + + def __str__(self) -> str: + return f"{self.value.name} of {self.color.value if self.color else '?'}" + + def score(self) -> int: + return int(self.value.value) \ No newline at end of file diff --git a/server/model/data.py b/server/model/data.py new file mode 100644 index 0000000..b68d4b8 --- /dev/null +++ b/server/model/data.py @@ -0,0 +1,125 @@ +import itertools +from collections import defaultdict +from dataclasses import dataclass +from typing import List, Dict, Optional + +from server.model.hand import Hand +from server.model.animals import random_animal_name +from server.model.card import Card +from server.model.deck import Deck + + +@dataclass +class Announce: + bet: Optional[Hand] = None + + @property + def menteur(self): + return not self.bet + + +class Player: + def __init__(self, name: str = None): + if not name: + name = random_animal_name() + self.name: str = name + self.hand: Hand = Hand() + + def __str__(self): + return f"Player {self.name}" + + def give(self, card: Card): + self.hand.give(card) + + def announce(self, current_bet: Optional[Hand]) -> Announce: + """ A naive player that only trusts what they sees: + bets his hand, or menteur if his hand is lower. """ + if not current_bet or self.hand.value() > current_bet.value(): + return Announce(self.hand) + else: + return Announce() + + +class Game: + def __init__(self, players=None, deck: Deck = Deck()): + if players is None: + players = [] + + self.deck: Deck = deck + self.players: List[Player] = players + self.defeats: Dict[Player, int] = defaultdict(lambda: 0) + + @property + def global_hand(self) -> Hand: + return Hand([c for p in self.players for c in p.hand]) + + def new_game(self): + self.deck.reset() + + while len(self.players) > 1: + loser = self.new_turn() + + if self.defeats[loser] == 5: + print(f"{loser} is eliminated!") + self.players.remove(loser) + else: + print(f"{loser} lost the round, now playing with {self.defeats[loser] + 1} cards!") + + winner = self.players[0] + print(f"Game over - {winner.name} wins with {len(winner.hand)} cards!") + + def new_turn(self) -> Player: + """ + Runs a turn of the game. + + :return: the player that lost this turn. + """ + # Distribution + self.deck.reset() + for current_player in self.players: + current_player.give(self.deck.random_card()) + + # Tour + current_bet: Optional[Hand] = None + last_player = None + for current_player in itertools.cycle(self.players): + print(f"| {current_player}'s turn >") + if not current_bet: # First player, has to bet something + while not current_bet: + announce = current_player.announce(current_bet) + if announce.bet: + current_bet = announce.bet + print(f"{current_player} starts the round: {current_bet}") + else: # Next player, announce or menteur + announce = current_player.announce(current_bet) + if announce.bet: + while announce.bet.value() < current_bet.value(): # Bad announce! + print(f"Invalid bet, {announce.bet} < {current_bet}!") + announce = current_player.announce(current_bet) + current_bet = announce.bet + print(f" {current_player} bets {current_bet}.") + else: # Menteur! Who lost the round? + menteur = self.is_menteur(current_bet) + print(f"{current_player} says Menteur... {'Bravo!' if menteur else 'FAIL!'}") + loser = last_player if menteur else current_player + print(f"{loser} lost the round!") + self.count_defeat(loser) + return loser + last_player = current_player + + # TODO: Put next first player first of list + + def is_menteur(self, bet: Hand): + to_scan = [Card(c.value) for c in self.global_hand.cards.copy()] # Keep only values + to_find = [Card(c.value) for c in bet.cards] + for card in to_find: + if card in to_scan: + to_scan.remove(card) + continue + else: + print(f"Missing a {card}!") + print(f"Didn't find {bet} in {self.global_hand}: MENTEUR!") + return False # + + def count_defeat(self, loser): + self.defeats[loser] += 1 diff --git a/server/model/deck.py b/server/model/deck.py new file mode 100644 index 0000000..0a67670 --- /dev/null +++ b/server/model/deck.py @@ -0,0 +1,25 @@ +from random import randrange, shuffle + +from server.model.card import Card, Value, Color + + +class Deck: + def __init__(self): + self.cards = [Card(v, c) for v in Value for c in Color] + self.defausse = [] + + def __len__(self): + return len(self.cards) + + def random_card(self) -> Card: + if not self.cards: + self.reset() + + card = self.cards.pop(randrange(len(self.cards))) + self.defausse.append(card) + return card + + def reset(self): + self.cards.extend(self.defausse) + self.defausse.clear() + shuffle(self.cards) \ No newline at end of file diff --git a/server/model/hand.py b/server/model/hand.py new file mode 100644 index 0000000..58b8a20 --- /dev/null +++ b/server/model/hand.py @@ -0,0 +1,90 @@ +from collections import Counter +from typing import List + +from server.model.card import Card + + +class Hand: + def __init__(self, cards: List[Card] = None): + if cards is None: + cards = [] + self.cards: List[Card] = cards + + def __contains__(self, item: Card): + return item in self.cards + + def __len__(self) -> int: + return len(self.cards) + + def __getitem__(self, item): + return self.cards[item] + + def __repr__(self): + return "|".join([str(c) for c in self.cards]) + + def give(self, other: Card): + self.cards.append(other) + + def value(self): + counter = Counter([c.value for c in self.cards]) + + has_pair = False + has_double_pair = False + has_brelan = False + has_full = False + has_carre = False + + highest_card = None + pair = None + double_pair = None + brelan = None + carre = None + + for element, count in counter.items(): + element_cards = [c for c in self.cards if c.value == element] + card = element_cards[0] # Note we take a random color + highest_card = max(highest_card, card) if highest_card else card + + if count == 2: + if has_pair: + has_double_pair = True + double_pair = max(pair, card) + pair = min(pair, card) + else: + has_pair = True + pair = max(pair, card) if pair else card + if count == 3: + has_brelan = True + brelan = max(brelan, card) if brelan else card + if has_brelan and has_pair: + has_full = True + if count == 4: + has_carre = True + carre = max(carre, card) if carre else card + + analysis_log = " | ".join([ + f"ANALYSIS", + f"Carre[{carre}]" if has_carre else "no carre", + f"Full[{brelan}|{pair}]" if has_full else "no full", + f"Brelan[{brelan}]" if has_brelan else "no carre", + f"Double paire[{double_pair}|{pair}]" if has_double_pair else "no Dpaire", + f"Paire[{pair}]" if has_pair else "no paire", + f"Card[{highest_card}]" + ]) + # Finished counting, let's return scores + if has_carre: + score = (20 ** 5) * carre.score() + elif has_full: + score = (20 ** 4) * brelan.score() + 20 * pair.score() + elif has_brelan: + score = (20 ** 3) * brelan.score() + elif has_double_pair: + score = (20 ** 2) * double_pair.score() + pair.score() + elif has_pair: + score = 20 * pair.score() + else: + score = highest_card.score() + + analysis_log += f"\t-> score=\t{score}" + # print(analysis_log) + return score \ No newline at end of file diff --git a/server/model/hands.py b/server/model/hands.py new file mode 100644 index 0000000..7904faa --- /dev/null +++ b/server/model/hands.py @@ -0,0 +1,59 @@ +from server.model.hand import Hand +from server.model.card import Value, Color, Card + + +def carre(value) -> Hand: + return Hand([ + Card(value, Color.Hearts), + Card(value, Color.Clubs), + Card(value, Color.Diamonds), + Card(value, Color.Spades) + ]) + + +def full(value_aux: Value, + value_par: Value) -> Hand: + return Hand([ + Card(value_aux, Color.Hearts), + Card(value_aux, Color.Clubs), + Card(value_aux, Color.Spades), + Card(value_par, Color.Hearts), + Card(value_par, Color.Clubs) + ]) + + +def brelan(value) -> Hand: + return Hand([ + Card(value, Color.Hearts), + Card(value, Color.Clubs), + Card(value, Color.Diamonds) + ]) + + +def pair(value) -> Hand: + return Hand([ + Card(value, Color.Hearts), + Card(value, Color.Clubs) + ]) + + +def single(low_value) -> Hand: + return Hand([ + Card(low_value, Color.Hearts) + ]) + + +def double_pair(value: Value, other: Value = None): + # Ensure no carre + if not other or other == value: + other = Value.Two if value == Value.Three else Value.Three + assert other != value + + return Hand([ + # Pair of value + Card(value, Color.Hearts), + Card(value, Color.Clubs), + # And pair of twos or threes + Card(other, Color.Hearts), + Card(other, Color.Clubs) + ]) diff --git a/server/model/known.py b/server/model/known.py new file mode 100644 index 0000000..d0acae6 --- /dev/null +++ b/server/model/known.py @@ -0,0 +1,13 @@ +from server.model.card import Value, Color, Card +from server.model.hands import pair, single, double_pair, brelan, full, carre + +ACE_OF_HEARTS = Card(Value.Ace, Color.Hearts) +ACE_OF_SPADES = Card(Value.Ace, Color.Spades) +ACE_OF_DIAMONDS = Card(Value.Ace, Color.Diamonds) +ACE_OF_CLUBS = Card(Value.Ace, Color.Clubs) +PAIR_ACE = pair(Value.Ace) +SINGLE_ACE = single(Value.Ace) +DOUBLE_PAIR_ACE = double_pair(Value.Ace) +BRELAN_ACE = brelan(Value.Ace) +FULL_ACE = full(Value.Ace, Value.King) +CARRE_ACE = carre(Value.Ace) diff --git a/server/requirements.txt b/server/requirements.txt new file mode 100644 index 0000000..f536c03 --- /dev/null +++ b/server/requirements.txt @@ -0,0 +1,2 @@ +fastapi==0.53.2 +uvicorn==0.11.3 diff --git a/server/test/__init__.py b/server/test/__init__.py new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/server/test/__init__.py diff --git a/server/test/test_data.py b/server/test/test_data.py new file mode 100644 index 0000000..22d56f5 --- /dev/null +++ b/server/test/test_data.py @@ -0,0 +1,204 @@ +from unittest import TestCase + +from server.model.data import Player +from server.model.deck import Deck +from server.model.hand import Hand +from server.model.card import Value, Color, Card +from server.model.hands import full, brelan, pair, single, double_pair, carre +from server.model.known import ACE_OF_HEARTS, PAIR_ACE, SINGLE_ACE, DOUBLE_PAIR_ACE, BRELAN_ACE, FULL_ACE + + +class TestDeck(TestCase): + + def setUp(self) -> None: + super().setUp() + self.deck = Deck() + + def testInitDeck(self): + self.assertEqual(52, len(self.deck), "52 cards") + self.assertEqual(52, len(set(self.deck.cards)), "unique cards") + + def testRandomCard(self): + cards = [] + for i in range(len(self.deck)): + cards.append(self.deck.random_card()) + + self.assertEqual(52, len(set(cards)), "draw each card once") + + +class TestPlayer(TestCase): + + def setUp(self) -> None: + super().setUp() + self.player = Player() + + def testHand(self): + self.assertEqual(0, len(self.player.hand), "Begin no cards") + self.assertTrue(len(self.player.name), "Has a name") + + self.player.give(ACE_OF_HEARTS) + + self.assertEqual(1, len(self.player.hand), "Gave one card") + self.assertEqual(Value.Ace, self.player.hand[0].value, "Is Ace") + self.assertEqual(Color.Hearts, self.player.hand[0].color, "of Hearts") + + def testDefeats(self): + pass + + +def lowest_value_and_rest(): + lowValue: Value = Value.Two + otherValues = list(Value) + otherValues.remove(lowValue) + + return lowValue, otherValues + + +class TestHand(TestCase): + def setUp(self) -> None: + self.hand = Hand() + + def testSimple(self): + low_value, other_values = lowest_value_and_rest() + + for value in other_values: + high_hand = single(value) + low_hand = single(low_value) + + self.assertGreater(high_hand.value(), low_hand.value()) + + def testPair(self): + low_value, other_values = lowest_value_and_rest() + + for value in other_values: + high_hand: Hand = pair(value) + low_hand: Hand = pair(low_value) + single_hand: Hand = single(Value.Ace) + + high_score = high_hand.value() + low_score = low_hand.value() + single_score = single_hand.value() + + self.assertGreater(high_score, low_score, f"Pair[{high_hand}] > Pair[{low_hand}]]") + self.assertGreater(high_score, single_score, f"Pair[{high_hand}] > Ace") + + def testDoublePair(self): + low_value, other_values = lowest_value_and_rest() + + for value in other_values: + high_hand: Hand = double_pair(value) + low_hand: Hand = double_pair(low_value) + pair_hand: Hand = PAIR_ACE + single_hand: Hand = SINGLE_ACE + + high_score = high_hand.value() + low_score = low_hand.value() + pair_score = pair_hand.value() + single_score = single_hand.value() + + # Consider case when we compare equal double pairs + low_cards = [c.value for c in low_hand.cards] + if all([card.value in low_cards for card in high_hand.cards]): + self.assertEqual(high_score, low_score, f"DoublePair[{high_hand}] == DoublePair[{low_hand}]") + else: + self.assertGreater(high_score, low_score, f"DoublePair[{high_hand}] > DoublePair[{low_hand}]") + self.assertGreater(high_score, pair_score, f"DoublePair[{high_hand}] > Pair[Ace]") + self.assertGreater(high_score, single_score, f"DoublePair[{high_hand}] > Ace") + + def testBrelan(self): + low_value, other_values = lowest_value_and_rest() + + for value in other_values: + high_hand: Hand = brelan(value) + low_hand: Hand = brelan(low_value) + double_pair_hand: Hand = DOUBLE_PAIR_ACE + pair_hand: Hand = PAIR_ACE + single_hand: Hand = SINGLE_ACE + + high_score = high_hand.value() + low_score = low_hand.value() + double_pair_score = double_pair_hand.value() + pair_score = pair_hand.value() + single_score = single_hand.value() + + self.assertGreater(high_score, low_score, f"Brelan[{high_hand}] > Brelan[{low_hand}]]") + self.assertGreater(high_score, double_pair_score, f"Brelan[{high_hand}] > Pair[Ace]") + self.assertGreater(high_score, pair_score, f"Brelan[{high_hand}] > Pair[Ace]") + self.assertGreater(high_score, single_score, f"Brelan[{high_hand}] > Ace") + + def testFulls(self): + # AssertionError: 24060 not greater than 24060 : + # Full[Three of ♥|Three of ♣|Three of ♠|Four of ♥|Four of ♣] + # > Full[Two of ♥|Two of ♣|Two of ♠|Four of ♥|Four of ♣]] + + full1 = Hand([Card(v) for v in [ + Value.Three, + Value.Three, + Value.Three, + Value.Four, + Value.Four, + ]]) + full2 = Hand([Card(v) for v in [ + Value.Two, + Value.Two, + Value.Two, + Value.Four, + Value.Four, + ]]) + self.assertGreater(full1.value(), full2.value(), "Full1 > full2 (threes >> twos)") + + def testFull(self): + low_value, other_values = lowest_value_and_rest() + + for full_aux in other_values: # Full_aux: brelan + full_options = other_values.copy() + full_options.remove(full_aux) + full_options.append(low_value) + + for full_par in full_options: # Full_des: paire + high_hand: Hand = full(full_aux, full_par) + low_hand: Hand = full(low_value, full_par) + brelan_hand: Hand = BRELAN_ACE + double_pair_hand: Hand = DOUBLE_PAIR_ACE + pair_hand: Hand = PAIR_ACE + single_hand: Hand = SINGLE_ACE + + high_score = high_hand.value() + low_score = low_hand.value() + brelan_score = brelan_hand.value() + double_pair_score = double_pair_hand.value() + pair_score = pair_hand.value() + single_score = single_hand.value() + + self.assertGreater(high_score, low_score, f"Full[{high_hand}] > Full[{low_hand}]]") + self.assertGreater(high_score, brelan_score, f"Full[{high_hand}] > Brelan[Ace]") + self.assertGreater(high_score, double_pair_score, f"Full[{high_hand}] > Pair[Ace]") + self.assertGreater(high_score, pair_score, f"Full[{high_hand}] > Pair[Ace]") + self.assertGreater(high_score, single_score, f"Full[{high_hand}] > Ace") + + def testCarre(self): + low_value, other_values = lowest_value_and_rest() + + for value in other_values: + high_hand: Hand = carre(value) + low_hand: Hand = carre(low_value) + full_hand: Hand = FULL_ACE + brelan_hand: Hand = BRELAN_ACE + double_pair_hand: Hand = DOUBLE_PAIR_ACE + pair_hand: Hand = PAIR_ACE + single_hand: Hand = SINGLE_ACE + + high_score = high_hand.value() + low_score = low_hand.value() + full_score = full_hand.value() + brelan_score = brelan_hand.value() + double_pair_score = double_pair_hand.value() + pair_score = pair_hand.value() + single_score = single_hand.value() + + self.assertGreater(high_score, low_score, f"Carre[{high_hand}] > Carre[{low_hand}]]") + self.assertGreater(high_score, full_score, f"Carre[{high_hand}] > Full[Ace]]") + self.assertGreater(high_score, brelan_score, f"Carre[{high_hand}] > Brelan[Ace]") + self.assertGreater(high_score, double_pair_score, f"Carre[{high_hand}] > Pair[Ace]") + self.assertGreater(high_score, pair_score, f"Carre[{high_hand}] > Pair[Ace]") + self.assertGreater(high_score, single_score, f"Carre[{high_hand}] > Ace") diff --git a/server/test/test_game.py b/server/test/test_game.py new file mode 100644 index 0000000..2dbfe19 --- /dev/null +++ b/server/test/test_game.py @@ -0,0 +1,36 @@ +from unittest import TestCase + +from server.model.data import Game, Player + + +class TestGame(TestCase): + def setUp(self) -> None: + super().setUp() + self.player1 = Player("PLN") + self.player2 = Player("Nassim") + self.game = Game([self.player1, self.player2]) + + def test_global_hand(self) -> None: + card1 = self.game.deck.random_card() + self.player1.give(card1) + + card2 = self.game.deck.random_card() + card3 = self.game.deck.random_card() + self.player2.give(card2) + self.player2.give(card3) + + self.assertTrue(card1 in self.game.global_hand, "Global hand should contain player1's card") + self.assertTrue(card2 in self.game.global_hand, "Global hand should contain player2's first card") + self.assertTrue(card3 in self.game.global_hand, "Global hand should contain player2's second card") + + def test_turn(self): + self.game.new_turn() + + self.assertEqual(1, len(self.game.defeats.values()), "There should have been one defeat.") + self.assertTrue(1 in self.game.defeats.values(), "A player should have one defeat.") + + def test_full_game(self): + self.game.new_game() + + self.assertGreater(len(self.game.defeats.values()), 0, "There should be at least one player with defeats.") + self.assertTrue(5 in self.game.defeats.values(), "A player should have lost five times.") diff --git a/test/__init__.py b/test/__init__.py deleted file mode 100644 index e69de29..0000000 --- a/test/__init__.py +++ /dev/null diff --git a/test/test_data.py b/test/test_data.py deleted file mode 100644 index c4ef4b1..0000000 --- a/test/test_data.py +++ /dev/null @@ -1,204 +0,0 @@ -from unittest import TestCase - -from model.data import Player -from model.deck import Deck -from model.hand import Hand -from model.card import Value, Color, Card -from model.hands import full, brelan, pair, single, double_pair, carre -from model.known import ACE_OF_HEARTS, PAIR_ACE, SINGLE_ACE, DOUBLE_PAIR_ACE, BRELAN_ACE, FULL_ACE - - -class TestDeck(TestCase): - - def setUp(self) -> None: - super().setUp() - self.deck = Deck() - - def testInitDeck(self): - self.assertEqual(52, len(self.deck), "52 cards") - self.assertEqual(52, len(set(self.deck.cards)), "unique cards") - - def testRandomCard(self): - cards = [] - for i in range(len(self.deck)): - cards.append(self.deck.random_card()) - - self.assertEqual(52, len(set(cards)), "draw each card once") - - -class TestPlayer(TestCase): - - def setUp(self) -> None: - super().setUp() - self.player = Player() - - def testHand(self): - self.assertEqual(0, len(self.player.hand), "Begin no cards") - self.assertTrue(len(self.player.name), "Has a name") - - self.player.give(ACE_OF_HEARTS) - - self.assertEqual(1, len(self.player.hand), "Gave one card") - self.assertEqual(Value.Ace, self.player.hand[0].value, "Is Ace") - self.assertEqual(Color.Hearts, self.player.hand[0].color, "of Hearts") - - def testDefeats(self): - pass - - -def lowest_value_and_rest(): - lowValue: Value = Value.Two - otherValues = list(Value) - otherValues.remove(lowValue) - - return lowValue, otherValues - - -class TestHand(TestCase): - def setUp(self) -> None: - self.hand = Hand() - - def testSimple(self): - low_value, other_values = lowest_value_and_rest() - - for value in other_values: - high_hand = single(value) - low_hand = single(low_value) - - self.assertGreater(high_hand.value(), low_hand.value()) - - def testPair(self): - low_value, other_values = lowest_value_and_rest() - - for value in other_values: - high_hand: Hand = pair(value) - low_hand: Hand = pair(low_value) - single_hand: Hand = single(Value.Ace) - - high_score = high_hand.value() - low_score = low_hand.value() - single_score = single_hand.value() - - self.assertGreater(high_score, low_score, f"Pair[{high_hand}] > Pair[{low_hand}]]") - self.assertGreater(high_score, single_score, f"Pair[{high_hand}] > Ace") - - def testDoublePair(self): - low_value, other_values = lowest_value_and_rest() - - for value in other_values: - high_hand: Hand = double_pair(value) - low_hand: Hand = double_pair(low_value) - pair_hand: Hand = PAIR_ACE - single_hand: Hand = SINGLE_ACE - - high_score = high_hand.value() - low_score = low_hand.value() - pair_score = pair_hand.value() - single_score = single_hand.value() - - # Consider case when we compare equal double pairs - low_cards = [c.value for c in low_hand.cards] - if all([card.value in low_cards for card in high_hand.cards]): - self.assertEqual(high_score, low_score, f"DoublePair[{high_hand}] == DoublePair[{low_hand}]") - else: - self.assertGreater(high_score, low_score, f"DoublePair[{high_hand}] > DoublePair[{low_hand}]") - self.assertGreater(high_score, pair_score, f"DoublePair[{high_hand}] > Pair[Ace]") - self.assertGreater(high_score, single_score, f"DoublePair[{high_hand}] > Ace") - - def testBrelan(self): - low_value, other_values = lowest_value_and_rest() - - for value in other_values: - high_hand: Hand = brelan(value) - low_hand: Hand = brelan(low_value) - double_pair_hand: Hand = DOUBLE_PAIR_ACE - pair_hand: Hand = PAIR_ACE - single_hand: Hand = SINGLE_ACE - - high_score = high_hand.value() - low_score = low_hand.value() - double_pair_score = double_pair_hand.value() - pair_score = pair_hand.value() - single_score = single_hand.value() - - self.assertGreater(high_score, low_score, f"Brelan[{high_hand}] > Brelan[{low_hand}]]") - self.assertGreater(high_score, double_pair_score, f"Brelan[{high_hand}] > Pair[Ace]") - self.assertGreater(high_score, pair_score, f"Brelan[{high_hand}] > Pair[Ace]") - self.assertGreater(high_score, single_score, f"Brelan[{high_hand}] > Ace") - - def testFulls(self): - # AssertionError: 24060 not greater than 24060 : - # Full[Three of ♥|Three of ♣|Three of ♠|Four of ♥|Four of ♣] - # > Full[Two of ♥|Two of ♣|Two of ♠|Four of ♥|Four of ♣]] - - full1 = Hand([Card(v) for v in [ - Value.Three, - Value.Three, - Value.Three, - Value.Four, - Value.Four, - ]]) - full2 = Hand([Card(v) for v in [ - Value.Two, - Value.Two, - Value.Two, - Value.Four, - Value.Four, - ]]) - self.assertGreater(full1.value(), full2.value(), "Full1 > full2 (threes >> twos)") - - def testFull(self): - low_value, other_values = lowest_value_and_rest() - - for full_aux in other_values: # Full_aux: brelan - full_options = other_values.copy() - full_options.remove(full_aux) - full_options.append(low_value) - - for full_par in full_options: # Full_des: paire - high_hand: Hand = full(full_aux, full_par) - low_hand: Hand = full(low_value, full_par) - brelan_hand: Hand = BRELAN_ACE - double_pair_hand: Hand = DOUBLE_PAIR_ACE - pair_hand: Hand = PAIR_ACE - single_hand: Hand = SINGLE_ACE - - high_score = high_hand.value() - low_score = low_hand.value() - brelan_score = brelan_hand.value() - double_pair_score = double_pair_hand.value() - pair_score = pair_hand.value() - single_score = single_hand.value() - - self.assertGreater(high_score, low_score, f"Full[{high_hand}] > Full[{low_hand}]]") - self.assertGreater(high_score, brelan_score, f"Full[{high_hand}] > Brelan[Ace]") - self.assertGreater(high_score, double_pair_score, f"Full[{high_hand}] > Pair[Ace]") - self.assertGreater(high_score, pair_score, f"Full[{high_hand}] > Pair[Ace]") - self.assertGreater(high_score, single_score, f"Full[{high_hand}] > Ace") - - def testCarre(self): - low_value, other_values = lowest_value_and_rest() - - for value in other_values: - high_hand: Hand = carre(value) - low_hand: Hand = carre(low_value) - full_hand: Hand = FULL_ACE - brelan_hand: Hand = BRELAN_ACE - double_pair_hand: Hand = DOUBLE_PAIR_ACE - pair_hand: Hand = PAIR_ACE - single_hand: Hand = SINGLE_ACE - - high_score = high_hand.value() - low_score = low_hand.value() - full_score = full_hand.value() - brelan_score = brelan_hand.value() - double_pair_score = double_pair_hand.value() - pair_score = pair_hand.value() - single_score = single_hand.value() - - self.assertGreater(high_score, low_score, f"Carre[{high_hand}] > Carre[{low_hand}]]") - self.assertGreater(high_score, full_score, f"Carre[{high_hand}] > Full[Ace]]") - self.assertGreater(high_score, brelan_score, f"Carre[{high_hand}] > Brelan[Ace]") - self.assertGreater(high_score, double_pair_score, f"Carre[{high_hand}] > Pair[Ace]") - self.assertGreater(high_score, pair_score, f"Carre[{high_hand}] > Pair[Ace]") - self.assertGreater(high_score, single_score, f"Carre[{high_hand}] > Ace") diff --git a/test/test_game.py b/test/test_game.py deleted file mode 100644 index 023a1b7..0000000 --- a/test/test_game.py +++ /dev/null @@ -1,36 +0,0 @@ -from unittest import TestCase - -from model.data import Game, Player - - -class TestGame(TestCase): - def setUp(self) -> None: - super().setUp() - self.player1 = Player("PLN") - self.player2 = Player("Nassim") - self.game = Game([self.player1, self.player2]) - - def test_global_hand(self) -> None: - card1 = self.game.deck.random_card() - self.player1.give(card1) - - card2 = self.game.deck.random_card() - card3 = self.game.deck.random_card() - self.player2.give(card2) - self.player2.give(card3) - - self.assertTrue(card1 in self.game.global_hand, "Global hand should contain player1's card") - self.assertTrue(card2 in self.game.global_hand, "Global hand should contain player2's first card") - self.assertTrue(card3 in self.game.global_hand, "Global hand should contain player2's second card") - - def test_turn(self): - self.game.new_turn() - - self.assertEqual(1, len(self.game.defeats.values()), "There should have been one defeat.") - self.assertTrue(1 in self.game.defeats.values(), "A player should have one defeat.") - - def test_full_game(self): - self.game.new_game() - - self.assertGreater(len(self.game.defeats.values()), 0, "There should be at least one player with defeats.") - self.assertTrue(5 in self.game.defeats.values(), "A player should have lost five times.") -- libgit2 0.27.0