This commit is contained in:
nora 2022-12-05 11:05:23 +01:00
parent 6172250ff5
commit 5b0c3106c0
No known key found for this signature in database
11 changed files with 560 additions and 185 deletions

View file

@ -1,15 +1,103 @@
#![allow(unused_imports)]
#![feature(let_chains)]
use std::{fmt::Display, time::SystemTime};
use std::{fmt::Display, str::FromStr, time::SystemTime};
use clap::{Parser, ValueEnum};
use minmax::{
connect4::{self, board::Connect4},
tic_tac_toe::{GreedyPlayer, HumanPlayer, PerfectPlayer, TicTacToe},
Game, GamePlayer, Player,
tic_tac_toe::{self, TicTacToe},
Game, GamePlayer, PerfectPlayer, Player,
};
#[derive(Debug, Clone)]
enum PlayerConfig {
Human,
Perfect { depth: Option<usize> },
}
impl FromStr for PlayerConfig {
type Err = String;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let mut parts = s.split(":");
let mut player = match parts
.next()
.ok_or_else(|| "No player name provided".to_owned())?
{
"human" | "h" => Self::Human,
"perfect" | "p" | "ai" | "minmax" => Self::Perfect { depth: None },
string => {
return Err(format!(
"Invalid player: {string}. Available players: human,perfect"
))
}
};
if let Some(depth) = parts.next()
&& let Self::Perfect { depth: player_depth } = &mut player
{
match depth.parse() {
Ok(depth) => *player_depth = Some(depth),
Err(err) => return Err(format!("Invalid depth: {depth}. {err}")),
}
}
Ok(player)
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, ValueEnum)]
enum GameType {
TicTacToe,
Connect4,
}
#[derive(Debug, Parser)]
#[command(author, version, about)]
struct Args {
#[arg(short, long)]
game: GameType,
#[arg(short)]
x: PlayerConfig,
#[arg(short)]
o: PlayerConfig,
}
fn main() {
play::<connect4::HumanPlayer, connect4::HumanPlayer, _>(true);
let args = Args::parse();
match args.game {
GameType::Connect4 => {
let get_player = |player| -> Box<dyn GamePlayer<Connect4>> {
match player {
PlayerConfig::Human => Box::new(connect4::HumanPlayer),
PlayerConfig::Perfect { depth } => {
Box::new(PerfectPlayer::new().with_max_depth(depth))
}
}
};
let player_a = get_player(args.o);
let player_b = get_player(args.x);
play_with_players(player_a, player_b);
}
GameType::TicTacToe => {
let get_player = |player| -> Box<dyn GamePlayer<TicTacToe>> {
match player {
PlayerConfig::Human => Box::new(tic_tac_toe::HumanPlayer),
PlayerConfig::Perfect { depth } => {
Box::new(PerfectPlayer::new().with_max_depth(depth))
}
}
};
let player_a = get_player(args.o);
let player_b = get_player(args.x);
play_with_players(player_a, player_b);
}
}
}
#[allow(dead_code)]
@ -19,7 +107,7 @@ fn tic_tac_toe_stats() {
let start = SystemTime::now();
for _ in 0..100 {
let result = play::<PerfectPlayer, GreedyPlayer, _>(false);
let result = play::<PerfectPlayer<TicTacToe>, tic_tac_toe::GreedyPlayer, _>(false);
let idx = Player::as_u8(result);
results[idx as usize] += 1;
}
@ -34,7 +122,16 @@ fn tic_tac_toe_stats() {
println!("Completed in {}ms", time.as_millis());
}
fn play<X: GamePlayer<G>, O: GamePlayer<G>, G: Game>(print: bool) -> Option<Player> {
fn play_with_players<G: Game, X: GamePlayer<G>, O: GamePlayer<G>>(mut x: X, mut o: O) {
let mut board = G::empty();
let result = board.play(&mut x, &mut o);
print_result(result, board);
}
fn play<X: GamePlayer<G> + Default, O: GamePlayer<G> + Default, G: Game>(
print: bool,
) -> Option<Player> {
let mut board = G::empty();
let result = board.play(&mut X::default(), &mut O::default());
if print {