move report to struct so that users of the lib can pull
cargo devel CI / cargo CI (push) Failing after 1m16s Details

This commit is contained in:
Christoph J. Scherr 2024-04-05 20:00:33 +02:00
parent fb59537cf1
commit e6934bfb08
No known key found for this signature in database
GPG Key ID: 7CDD0B14851A08EF
6 changed files with 83 additions and 33 deletions

View File

@ -28,6 +28,7 @@ chrono = { version = "0.4.37" }
clap = { version = "4.5.3", features = ["derive"], optional = true }
colored = { version = "2.1.0", optional = false }
libpt = "0.4.2"
num_cpus = "1.16.0"
rand = "0.8.5"
rayon = "1.10.0"
regex = "1.10.3"

View File

@ -1,14 +1,20 @@
use std::sync::{Arc, Mutex};
use libpt::log::info;
use crate::error::{BenchError, Error, WResult};
use crate::game::{Game, GameBuilder};
use crate::solve::Solver;
use crate::wlist::WordList;
use super::Benchmark;
use super::{Benchmark, Report};
#[derive(Debug, Clone)]
pub struct BuiltinBenchmark<'wl, WL: WordList, SL: Solver<'wl, WL>> {
wordlist: &'wl WL,
solver: SL,
builder: GameBuilder<'wl, WL>,
report: Arc<Mutex<Report>>,
}
impl<'wl, WL, SL> Benchmark<'wl, WL, SL> for BuiltinBenchmark<'wl, WL, SL>
@ -22,10 +28,14 @@ where
wordlist: &'wl WL,
solver: SL,
builder: GameBuilder<'wl, WL>,
threads: usize
) -> crate::error::WResult<Self> {
info!("using {threads} threads for benchmarking");
rayon::ThreadPoolBuilder::new().num_threads(threads).build_global().unwrap();
Ok(Self {
wordlist,
solver,
report: Arc::new(Mutex::new(Report::new(builder.build()?))),
builder,
})
}
@ -35,4 +45,12 @@ where
fn builder(&'wl self) -> &'wl crate::game::GameBuilder<'wl, WL> {
&self.builder
}
fn report_mutex(&'wl self) -> Arc<Mutex<Report>> {
self.report.clone()
}
fn report(&'wl self) -> super::Report {
self.report.lock().expect("lock is poisoned").clone()
}
}

View File

@ -30,6 +30,7 @@ where
wordlist: &'wl WL,
solver: SL,
builder: GameBuilder<'wl, WL>,
threads: usize
) -> crate::error::WResult<Self>;
fn builder(&'wl self) -> &'wl GameBuilder<'wl, WL>;
fn make_game(&'wl self) -> WResult<Game<'wl, WL>> {
@ -42,11 +43,7 @@ where
// TODO: add some interface to get reports while the benchmark runs
// TODO: make the benchmark optionally multithreaded
fn bench(&'wl self, n: usize) -> WResult<Report> {
let part = match n / 20 {
0 => 19,
other => other,
};
let report = Arc::new(Mutex::new(Report::new(self.make_game()?)));
let report = self.report_mutex();
let this = std::sync::Arc::new(self);
(0..n)
@ -59,11 +56,12 @@ where
report.lock().expect("lock is poisoned").add(r);
});
// FIXME: find some way to take the Report from the Mutex
// Mutex::into_inner() does not work
let mut report: Report = report.lock().unwrap().clone();
report.finalize();
report.lock().expect("lock is poisoned").finalize();
drop(report);
Ok(report)
Ok(self.report())
}
// PERF: Somehow returning &Report would be better as we don't need to clone then
fn report(&'wl self) -> Report;
fn report_mutex(&'wl self) -> Arc<Mutex<Report>>;
}

View File

@ -3,6 +3,7 @@ use core::panic;
use libpt::log::debug;
use std::fmt::Display;
use std::ops::Div;
use rayon::prelude::*;
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
@ -69,9 +70,11 @@ impl Report {
self.total_steps() as f64 / self.n() as f64
}
pub fn avg_time(&self) -> Option<TimeDelta> {
let av = self.benchtime()? / self.n() as i32;
Some(av)
pub fn avg_time(&self) -> TimeDelta {
if self.n() == 0 {
return TimeDelta::new(0, 0).unwrap();
}
self.benchtime() / self.n() as i32
}
fn rating_steps(&self) -> f64 {
@ -83,20 +86,25 @@ impl Report {
WEIGHTING_WIN * (1.0 - self.avg_win())
}
fn rating_time(&self) -> Option<f64> {
let n = 1.0 / (1.0 + (self.avg_time()?.num_nanoseconds()? as f64).exp());
Some(WEIGHTING_TIME * (1.0 - n))
fn rating_time(&self) -> f64 {
let n = 1.0
/ (1.0
+ (self
.avg_time()
.num_nanoseconds()
.expect("nanoseconds overflow") as f64)
.exp());
WEIGHTING_TIME * (1.0 - n)
}
pub fn rating(&self) -> Option<f64> {
pub fn rating(&self) -> f64 {
let rating_steps: f64 = self.rating_steps();
let rating_win: f64 = self.rating_win();
let rating_time: f64 = self.rating_time()?;
let rating_time: f64 = self.rating_time();
debug!("partial rating - steps: {}", rating_steps);
debug!("partial rating - win: {}", rating_win);
debug!("partial rating - time: {:?}", rating_time);
let r = rating_win + rating_time + rating_steps;
Some(r)
rating_win + rating_time + rating_steps
}
/// finalize the record
@ -116,8 +124,8 @@ impl Report {
self.finished
}
pub fn benchtime(&self) -> Option<TimeDelta> {
self.benchtime
pub fn benchtime(&self) -> TimeDelta {
chrono::Local::now().naive_local() - self.start
}
pub fn max_steps(&self) -> usize {
@ -132,9 +140,6 @@ impl Display for Report {
///
/// This will panic if the [Report] is not finished
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
if !self.finished {
panic!("can only display finished reports");
}
write!(
f,
"n: {}, win_ratio: {:.2}%, avg_score: {:.4} steps until finish, avgerage time per game: {}μs, \n\
@ -143,9 +148,9 @@ impl Display for Report {
self.n(),
self.avg_win() * 100.0,
self.avg_steps(),
self.avg_time().unwrap().num_microseconds().expect("overflow when converting to micrseconds"),
self.rating().unwrap(),
self.benchtime().unwrap().num_milliseconds()
self.avg_time(),
self.rating(),
self.benchtime()
)
}
}

View File

@ -2,11 +2,15 @@
// #![warn(missing_docs)]
#![warn(missing_debug_implementations)]
use std::sync::Arc;
use clap::Parser;
use libpt::log::*;
use wordle_analyzer::bench::builtin::BuiltinBenchmark;
use wordle_analyzer::bench::report::Report;
use wordle_analyzer::bench::{Benchmark, DEFAULT_N};
use wordle_analyzer::error::WResult;
use wordle_analyzer::solve::{BuiltinSolverNames, Solver};
use wordle_analyzer::wlist::builtin::BuiltinWList;
@ -33,6 +37,11 @@ struct Cli {
/// how many games to play for the benchmark
#[arg(short, long, default_value_t = DEFAULT_N)]
n: usize,
/// how many threads to use for benchmarking
///
/// Note that the application as the whole will use at least one more thread.
#[arg(short, long, default_value_t = num_cpus::get())]
threads: usize,
}
fn main() -> anyhow::Result<()> {
@ -50,11 +59,19 @@ fn main() -> anyhow::Result<()> {
.max_steps(cli.max_steps)
.precompute(cli.precompute);
let solver = cli.solver.to_solver(&wl);
let bench = BuiltinBenchmark::build(&wl, solver, builder)?;
let bench = Arc::new(BuiltinBenchmark::build(&wl, solver, builder, cli.threads)?);
let bench_running = bench.clone();
trace!("{bench:#?}");
let report = bench.bench(cli.n)?;
let n = cli.n;
let bench_th: std::thread::JoinHandle<WResult<Report>> =
std::thread::spawn(move || bench_running.bench(n));
println!("{report}");
while !bench_th.is_finished() {
println!("{}", bench.report());
}
// finished report
println!("{}", bench_th.join().expect("thread go boom")?);
Ok(())
}

View File

@ -8,11 +8,16 @@ pub type GameResult<T> = std::result::Result<T, GameError>;
#[derive(Debug, Error)]
pub enum Error {
#[error("GameError")]
#[error("Game Error")]
GameError {
#[from]
source: GameError,
},
#[error("Benchmark Error")]
BenchError {
#[from]
source: BenchError,
},
#[error(transparent)]
Other {
#[from]
@ -42,3 +47,9 @@ pub enum GameError {
#[error("Tried to guess a word that is not in the wordlist ({0})")]
WordNotInWordlist(Word),
}
#[derive(Debug, Clone, Error)]
pub enum BenchError {
#[error("Trying to modify a finished report")]
ModifyFinishedReport
}