mirror of
https://github.com/Noratrieb/minmax.git
synced 2026-01-16 08:15:04 +01:00
cli
This commit is contained in:
parent
6172250ff5
commit
5b0c3106c0
11 changed files with 560 additions and 185 deletions
111
src/main.rs
111
src/main.rs
|
|
@ -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 {
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue