kuh-handel-lib 1.0.0

A package for implementing a bot for the Kuh-Handel game
Documentation
use crate::messages::actions::{FromActionMessage, InitialTrade};

use crate::animals::{Animal, AnimalSet};
use crate::messages::game_updates::Points;
use crate::messages::message_protocol::StateMessage;

use crate::player::player_actions::PlayerActions;
use crate::player::player_error::PlayerError;
use crate::player::random_player::RandomPlayerActions;
use crate::player::simple_player::SimplePlayer;
use crate::player::wallet::Wallet;
use pyo3::prelude::*;
use std::collections::BTreeMap;
use std::fmt;
use std::fmt::Display;
use std::sync::Arc;
use tokio::sync::Mutex;

pub type PlayerId = String;

#[pyclass()]
pub struct Player {
    id: PlayerId,
    wallet: Wallet,
    owned_animals: BTreeMap<Animal, usize>,
    game_stack: Vec<Arc<Mutex<AnimalSet>>>,
    player_actions: Box<dyn PlayerActions + Send + Sync>,
}

impl Player {
    pub fn new(
        id: String,
        wallet: Wallet,
        game_stack: Vec<Arc<Mutex<AnimalSet>>>,
        player_actions: Box<dyn PlayerActions + Send + Sync>,
    ) -> Self {
        Player {
            id: id,
            wallet: wallet,
            game_stack: game_stack,
            owned_animals: BTreeMap::new(),
            player_actions,
        }
    }

    pub fn calculate_points(&self) -> Points {
        let mut full_stacks: usize = 0;
        let mut total_points: usize = 0;
        for (animal, current_animal_count) in self.owned_animals.iter() {
            let animal_set = self
                .game_stack
                .iter()
                .find(|set| set.blocking_lock().animal() == animal);
            if let Some(animal_set) = animal_set {
                if animal_set.blocking_lock().occurrences() == *current_animal_count {
                    total_points += animal.value();
                    full_stacks += 1;
                }
            }
        }
        full_stacks * total_points
    }

    pub fn can_trade(&self, opponents: Vec<Arc<Mutex<Player>>>) -> Option<InitialTrade> {
        for opponent in opponents.iter() {
            let possible_trade = self.can_trade_against(Arc::clone(opponent));
            match possible_trade {
                Some(trade) => {
                    return Some(trade);
                }
                None => {
                    continue;
                }
            }
        }
        None
    }

    pub fn can_trade_animal(
        &self,
        animal: &Animal,
        opponents: &Vec<Arc<Mutex<Player>>>,
    ) -> Option<InitialTrade> {
        if let Some(&animal_count) = self.owned_animals.get(animal) {
            for opponent in opponents.iter() {
                let binding = opponent.blocking_lock();
                if let Some(&opponent_animal_count) = binding.owned_animals().get(animal) {
                    let max_trade_count = std::cmp::min(animal_count, opponent_animal_count);

                    return Some(InitialTrade {
                        opponent: binding.id().clone(),
                        animal: animal.clone(),
                        animal_count: max_trade_count,
                        amount: Vec::new(),
                    });
                }
            }
        }
        None
    }

    pub fn can_trade_against(&self, opponent: Arc<Mutex<Player>>) -> Option<InitialTrade> {
        for (&animal, &animal_count) in self.owned_animals.iter() {
            if animal_count
                < self
                    .game_stack
                    .iter()
                    .find(|set| set.blocking_lock().animal() == &animal)
                    .map(|set| set.blocking_lock().occurrences())
                    .unwrap_or(0)
            {
                let binding = opponent.blocking_lock();
                let opponent_animals = binding.owned_animals();
                if let Some(&opponent_animal_count) = opponent_animals.get(&animal) {
                    let max_trade_count = std::cmp::min(animal_count, opponent_animal_count);
                    return Some(InitialTrade {
                        opponent: binding.id().clone(),
                        animal: animal,
                        animal_count: max_trade_count,
                        amount: Vec::new(),
                    });
                }
            }
        }

        None
    }

    pub fn map_to_action_inner<T: FromActionMessage>(
        &mut self,
        state_msg: StateMessage,
    ) -> Option<T> {
        let action_msg = self.player_actions.map_to_action(state_msg);
        T::extract(action_msg)
    }

    pub fn player_actions(&mut self) -> &mut dyn PlayerActions {
        self.player_actions.as_mut()
    }

    pub fn wallet(&self) -> &Wallet {
        &self.wallet
    }

    pub fn wallet_mut(&mut self) -> &mut Wallet {
        &mut self.wallet
    }

    pub fn owned_animals(&self) -> &BTreeMap<Animal, usize> {
        &self.owned_animals
    }

    pub fn add_animals(&mut self, animal: &Animal, count: usize) {
        self.owned_animals
            .entry(*animal)
            .and_modify(|current| *current += count)
            .or_insert(count);
    }

    pub fn remove_animals(&mut self, animal: &Animal, count: usize) -> Result<(), PlayerError> {
        let backup_animals = self.owned_animals.clone();
        let current_count = self.owned_animals.get_mut(animal);
        match current_count {
            Some(current_count) => {
                let res: isize = *current_count as isize - count as isize;
                if res > 0 {
                    *current_count -= count;
                } else if *current_count == 0 || res == 0 {
                    self.owned_animals.remove(animal);
                } else {
                    self.owned_animals = backup_animals;
                    return Result::Err(PlayerError::AnimalsNotAvailable);
                }
            }
            None => {
                self.owned_animals = backup_animals;
                return Result::Err(PlayerError::AnimalsNotAvailable);
            }
        }

        Ok(())
    }
}

impl Display for Player {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "{}", self.id)
    }
}

#[pymethods]
impl Player {
    #[new]
    pub fn new_py(id: String, wallet: Wallet, game_stack: Vec<AnimalSet>) -> Self {
        let dummy_action = SimplePlayer::new_from_seed(id.clone(), 0);
        let game_stack = game_stack
            .iter()
            .map(|set| Arc::new(Mutex::new(set.clone())))
            .collect();
        Player::new(id, wallet, game_stack, Box::new(dummy_action))
    }

    #[getter]
    pub fn id(&self) -> &String {
        &self.id
    }
}