2023-10-05 16:19:57 +00:00
|
|
|
use std::{io::Read, path::PathBuf};
|
2023-07-20 04:55:24 +00:00
|
|
|
|
2024-03-23 18:14:53 +00:00
|
|
|
use sgf::{parse_sgf, GameRecord};
|
2023-07-20 04:55:24 +00:00
|
|
|
use thiserror::Error;
|
|
|
|
|
|
|
|
#[derive(Error, Debug)]
|
|
|
|
pub enum Error {
|
2023-10-05 16:19:57 +00:00
|
|
|
/*
|
2023-07-20 04:55:24 +00:00
|
|
|
#[error("Database permission denied")]
|
|
|
|
PermissionDenied,
|
2023-10-05 16:19:57 +00:00
|
|
|
*/
|
2023-07-20 04:55:24 +00:00
|
|
|
#[error("An IO error occurred: {0}")]
|
|
|
|
IOError(std::io::Error),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<std::io::Error> for Error {
|
|
|
|
fn from(err: std::io::Error) -> Self {
|
|
|
|
Error::IOError(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-26 02:49:43 +00:00
|
|
|
#[derive(Debug)]
|
2023-07-20 04:55:24 +00:00
|
|
|
pub struct Database {
|
2024-03-23 18:14:53 +00:00
|
|
|
games: Vec<GameRecord>,
|
2023-07-20 04:55:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Database {
|
|
|
|
pub fn open_path(path: PathBuf) -> Result<Database, Error> {
|
2024-03-23 18:14:53 +00:00
|
|
|
let mut games: Vec<GameRecord> = Vec::new();
|
2023-07-20 04:55:24 +00:00
|
|
|
|
|
|
|
let extension = PathBuf::from("sgf").into_os_string();
|
|
|
|
|
|
|
|
let path_iter = std::fs::read_dir(path.clone())?;
|
|
|
|
for entry in path_iter {
|
|
|
|
match entry {
|
|
|
|
Ok(entry) => {
|
|
|
|
if entry.path().extension() == Some(&extension) {
|
|
|
|
let mut buffer = String::new();
|
|
|
|
std::fs::File::open(entry.path())
|
|
|
|
.unwrap()
|
|
|
|
.read_to_string(&mut buffer)
|
|
|
|
.unwrap();
|
2023-08-23 19:57:09 +00:00
|
|
|
match parse_sgf(&buffer) {
|
|
|
|
Ok(sgfs) => {
|
2024-04-28 17:48:52 +00:00
|
|
|
let mut sgfs =
|
|
|
|
sgfs.into_iter().flatten().collect::<Vec<sgf::GameRecord>>();
|
2024-03-23 18:41:35 +00:00
|
|
|
games.append(&mut sgfs);
|
2023-07-27 00:29:12 +00:00
|
|
|
}
|
2024-03-23 18:41:35 +00:00
|
|
|
Err(err) => println!("Error parsing {:?}: {:?}", entry.path(), err),
|
2023-07-27 00:29:12 +00:00
|
|
|
}
|
2023-07-20 04:55:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(err) => println!("failed entry: {:?}", err),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-05 16:19:57 +00:00
|
|
|
Ok(Database { games })
|
2023-07-26 02:49:43 +00:00
|
|
|
}
|
|
|
|
|
2024-03-23 18:14:53 +00:00
|
|
|
pub fn all_games(&self) -> impl Iterator<Item = &GameRecord> {
|
2023-07-20 04:55:24 +00:00
|
|
|
self.games.iter()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::*;
|
2023-07-21 14:47:30 +00:00
|
|
|
use cool_asserts::assert_matches;
|
2023-07-27 00:29:12 +00:00
|
|
|
use sgf::Date;
|
2023-07-20 04:55:24 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn it_reads_empty_database() {
|
|
|
|
let db = Database::open_path(PathBuf::from("fixtures/empty_database/"))
|
|
|
|
.expect("database to open");
|
|
|
|
assert_eq!(db.all_games().count(), 0);
|
|
|
|
}
|
|
|
|
|
2023-10-19 06:43:08 +00:00
|
|
|
#[ignore]
|
2023-07-20 04:55:24 +00:00
|
|
|
#[test]
|
|
|
|
fn it_reads_five_games_from_database() {
|
|
|
|
let db =
|
|
|
|
Database::open_path(PathBuf::from("fixtures/five_games/")).expect("database to open");
|
|
|
|
assert_eq!(db.all_games().count(), 5);
|
2023-07-21 14:47:30 +00:00
|
|
|
|
2024-03-23 18:14:53 +00:00
|
|
|
assert_matches!(db.all_games().find(|g| g.black_player.name == Some("Steve".to_owned())),
|
2023-07-21 14:47:30 +00:00
|
|
|
Some(game) => {
|
2024-03-23 18:14:53 +00:00
|
|
|
assert_eq!(game.black_player.name, Some("Steve".to_owned()));
|
|
|
|
assert_eq!(game.white_player.name, Some("Savanni".to_owned()));
|
|
|
|
assert_eq!(game.dates, vec![Date::Date(chrono::NaiveDate::from_ymd_opt(2023, 4, 19).unwrap())]);
|
2023-10-05 16:47:37 +00:00
|
|
|
// assert_eq!(game.info.komi, Some(6.5));
|
2023-07-21 14:47:30 +00:00
|
|
|
}
|
|
|
|
);
|
2023-07-20 04:55:24 +00:00
|
|
|
}
|
|
|
|
}
|