rlevo-environments 0.1.0

RL benchmark environments and landscapes for rlevo (internal crate — use `rlevo` for the full API)
Documentation
//! Toy-text tabular RL environments.
//!
//! Faithful Rust implementations of the canonical Sutton & Barto tabular testbeds:
//! [`blackjack`], [`taxi`], [`cliff_walking`], and [`frozen_lake`].
//! All environments are discrete MDPs with no physics.

pub mod blackjack;
pub mod cliff_walking;
pub mod frozen_lake;
pub mod taxi;

/// Error returned when a custom [`frozen_lake::FrozenMapSpec::Custom`] map fails validation,
/// or when random map generation exhausts its retry budget.
#[derive(Debug)]
pub enum MapError {
    /// A row has a different number of tiles than the first row.
    RowLengthMismatch { row: usize, got: usize, expected: usize },
    /// The map does not contain exactly one `'S'` start tile.
    WrongStartCount(usize),
    /// The map contains no `'G'` goal tiles.
    NoGoal(usize),
    /// BFS from start cannot reach any goal tile.
    GoalUnreachable,
    /// A tile character is not one of `'S'`, `'F'`, `'H'`, `'G'`.
    InvalidTile { row: usize, col: usize, ch: char },
    /// Random map generation failed to produce a solvable map within the retry budget.
    MaxRetriesExceeded,
}

impl std::fmt::Display for MapError {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            MapError::RowLengthMismatch { row, got, expected } => {
                write!(f, "row {row} has length {got}, expected {expected}")
            }
            MapError::WrongStartCount(n) => {
                write!(f, "map must contain exactly one 'S' tile, found {n}")
            }
            MapError::NoGoal(n) => {
                write!(f, "map must contain at least one 'G' tile, found {n}")
            }
            MapError::GoalUnreachable => write!(f, "goal is unreachable from start under BFS"),
            MapError::InvalidTile { row, col, ch } => {
                write!(f, "invalid tile {ch:?} at ({row}, {col})")
            }
            MapError::MaxRetriesExceeded => {
                write!(f, "failed to generate solvable random map within retry limit")
            }
        }
    }
}

impl std::error::Error for MapError {}