From 54138b044ac3607f242af918c4dbd31edb10b5ac Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sat, 2 Mar 2024 22:31:02 +0000 Subject: [PATCH 1/3] Make Grid be able to change size --- backend/Cargo.toml | 1 + backend/src/grid.rs | 150 +++++++++++++++++++++++++++++++++++++++++- backend/src/kube.rs | 4 +- backend/src/player.rs | 2 +- backend/src/space.rs | 4 +- 5 files changed, 155 insertions(+), 6 deletions(-) diff --git a/backend/Cargo.toml b/backend/Cargo.toml index da37338..aaeb3fd 100644 --- a/backend/Cargo.toml +++ b/backend/Cargo.toml @@ -12,3 +12,4 @@ dotenvy = "0.15" uuid = { version = "1.7.0", features = ["v5", "fast-rng", "macro-diagnostics"] } tokio = { version = "1", features = ["full"] } async-trait = "0.1.77" +thiserror = "1.0.57" diff --git a/backend/src/grid.rs b/backend/src/grid.rs index a3393d3..1f25395 100644 --- a/backend/src/grid.rs +++ b/backend/src/grid.rs @@ -1,8 +1,27 @@ use std::collections::BTreeMap; -use crate::space::{ Space, SpaceKind }; +use crate::space::Space; use crate::Coordinate; +use thiserror::Error; + +/// The direction in which to grow. For example, going from 3 to 9 would give a `GrowDirection::Expand`. +enum GrowDirection { + Shrink, + Expand, +} + +/// An error when trying to expand or shrink the grid. +#[derive(Error, Debug)] +pub enum ResizeError { + #[error("At least one Kube which was formerly in the grid would not be in the grid after shrinking.")] + KubeWillBeOutOfBounds, + #[error("Grid width or height will exceed `u64::MAX`.")] + GridTooBig, + #[error("Grid width or height will be less than zero.")] + GridTooSmall, +} /// A grid to represent the playing area. Only saves spaces which aren't empty. +#[derive(Debug, PartialEq)] pub struct Grid { spaces: BTreeMap, width: u64, @@ -85,12 +104,103 @@ impl Grid { .flatten() .collect() } + + /// Returns the grid size in the format `[width, height]`. + fn get_grid_size(&self) -> [u64; 2] { + [self.width, self.height] + } + + fn change_grid_by_rings(&mut self, rings_to_change_by: u64, direction: GrowDirection) -> Result<(), ResizeError> { + // If the specified size is the same as the current size, no need to do anything. + if rings_to_change_by == 0 { + return Ok(()); + } + let Some(double_rings) = rings_to_change_by.checked_mul(2) else { + return Err(ResizeError::GridTooBig) + }; + let Some(new_width) = (match direction { + GrowDirection::Expand => self.width.checked_add(double_rings), + GrowDirection::Shrink => self.width.checked_sub(double_rings), + }) else { + return Err(ResizeError::GridTooBig); + }; + let Some(new_height) = (match direction { + GrowDirection::Expand => self.height.checked_add(double_rings), + GrowDirection::Shrink => self.height.checked_sub(double_rings), + }) else { + return Err(ResizeError::GridTooBig); + }; + + // Careful! If the map is big, this may use a lot of memory! + let mut spaces_in_map: Vec = self.spaces.clone().into_values().collect(); + for space in &mut spaces_in_map { + space.coordinate[0] = match direction { + GrowDirection::Expand => { + match space.coordinate[0].checked_add(rings_to_change_by) { + Some(n) => n, + None => return Err(ResizeError::KubeWillBeOutOfBounds), + } + }, + GrowDirection::Shrink => { + match space.coordinate[0].checked_sub(rings_to_change_by) { + Some(n) => n, + None => return Err(ResizeError::KubeWillBeOutOfBounds), + } + } + }; + space.coordinate[1] = match direction { + GrowDirection::Expand => { + match space.coordinate[1].checked_add(rings_to_change_by) { + Some(n) => n, + None => return Err(ResizeError::KubeWillBeOutOfBounds), + } + }, + GrowDirection::Shrink => { + match space.coordinate[1].checked_sub(rings_to_change_by) { + Some(n) => n, + None => return Err(ResizeError::KubeWillBeOutOfBounds), + } + } + }; + } + self.spaces = BTreeMap::new(); + for space in spaces_in_map { + self.insert(space); + } + self.width = new_width; + self.height = new_height; + Ok(()) + + } + + /// This will expand the grid size and change the coordinates of the respective kubes. + /// + /// The change in size can be thought of as "rings of squares" to be added around the outside of the grid. So if the grid used to be a 2×2 grid, adding a ring of squares around the outside will give a 4×4 square. + fn expand_grid(&mut self, rings_to_add: u64) -> Result<(), ResizeError> { + self.change_grid_by_rings(rings_to_add, GrowDirection::Expand) + } + + fn shrink_grid(&mut self, rings_to_shrink_by: u64) -> Result<(), ResizeError> { + self.change_grid_by_rings(rings_to_shrink_by, GrowDirection::Shrink) + } } #[cfg(test)] mod tests { use super::*; use crate::space::{Space, SpaceKind}; + fn grid_2x2() -> Grid { + let mut grid = Grid::new(2, 2); + let spaces = [ + Space::new([0, 0], SpaceKind::EmptySpace), + Space::new([0, 1], SpaceKind::EmptySpace), + Space::new([1, 1], SpaceKind::EmptySpace), + ]; + for space in spaces { + grid.insert(space); + } + grid + } fn grid_3x3() -> Grid { let mut grid = Grid::new(3, 3); let spaces = [ @@ -111,4 +221,42 @@ mod tests { let neighbours = grid.get_nonempty_neighbours([0, 2]); assert_eq!(vec![&Space::new([0, 1], SpaceKind::EmptySpace)], neighbours); } + + #[test] + fn increase_2_to_4() { + let mut grid = grid_2x2(); + let _ = grid.expand_grid(1); + let mut expected_grid = Grid::new(4, 4); + let spaces = [ + Space::new([1, 1], SpaceKind::EmptySpace), + Space::new([1, 2], SpaceKind::EmptySpace), + Space::new([2, 2], SpaceKind::EmptySpace), + ]; + for space in spaces { + expected_grid.insert(space); + } + assert_eq!(expected_grid, grid); + } + #[test] + fn decrease_from_3_to_1() { + let mut grid = Grid::new(3, 3); + grid.insert(Space::new([1, 1], SpaceKind::EmptySpace)); + let _ = grid.shrink_grid(1); + let mut expected_grid = Grid::new(1, 1); + let spaces = [ + Space::new([0, 0], SpaceKind::EmptySpace), + ]; + for space in spaces { + expected_grid.insert(space); + } + assert_eq!(expected_grid, grid); + } + #[test] + fn fail_on_invalid_grid_size_change() { + let mut grid = grid_3x3(); + let shrink_res = grid.shrink_grid(1); + assert!(shrink_res.is_err()); + let expand_res = grid.expand_grid(u64::MAX - 1); + assert!(expand_res.is_err()); + } } diff --git a/backend/src/kube.rs b/backend/src/kube.rs index e7554b2..b6abe71 100644 --- a/backend/src/kube.rs +++ b/backend/src/kube.rs @@ -1,6 +1,6 @@ use uuid::Uuid; -#[derive(PartialEq, Eq, PartialOrd, Ord, Debug)] +#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone)] pub struct KubeId { uuid: Uuid, } @@ -19,7 +19,7 @@ impl KubeId { } } -#[derive(PartialEq, Debug)] +#[derive(PartialEq, Debug, Clone)] pub struct Kube { pub id: KubeId, pub name: String, diff --git a/backend/src/player.rs b/backend/src/player.rs index e64640b..6ce81b0 100644 --- a/backend/src/player.rs +++ b/backend/src/player.rs @@ -1,4 +1,4 @@ -#[derive(Debug, PartialEq)] +#[derive(Debug, PartialEq, Clone)] pub struct Player { uuid: String, username: String, diff --git a/backend/src/space.rs b/backend/src/space.rs index 0af1509..228ba8a 100644 --- a/backend/src/space.rs +++ b/backend/src/space.rs @@ -2,7 +2,7 @@ use crate::kube; use crate::player; use crate::Coordinate; -#[derive(Debug, PartialEq)] +#[derive(Debug, PartialEq, Clone)] pub struct Space { pub coordinate: Coordinate, pub contains: SpaceKind, @@ -16,7 +16,7 @@ impl Space { } } -#[derive(Debug, PartialEq)] +#[derive(Debug, PartialEq, Clone)] #[allow(clippy::module_name_repetitions)] pub enum SpaceKind { Kube(kube::Kube), From 6211c4fd6d208f2861f0135585923694aad19ee1 Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sat, 2 Mar 2024 22:49:23 +0000 Subject: [PATCH 2/3] Add more docs to Grid --- backend/src/grid.rs | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/backend/src/grid.rs b/backend/src/grid.rs index 1f25395..f4219d1 100644 --- a/backend/src/grid.rs +++ b/backend/src/grid.rs @@ -47,6 +47,8 @@ impl Grid { } } + + /// Adds a new space to the grid. In the future, this may return an [`std::result::Result::Err`] if the space is of type [`space::SpaceKind::EmptySpace`]. pub fn insert(&mut self, space: Space) { self.spaces.insert(space.coordinate, space); } @@ -176,11 +178,12 @@ impl Grid { /// This will expand the grid size and change the coordinates of the respective kubes. /// /// The change in size can be thought of as "rings of squares" to be added around the outside of the grid. So if the grid used to be a 2×2 grid, adding a ring of squares around the outside will give a 4×4 square. - fn expand_grid(&mut self, rings_to_add: u64) -> Result<(), ResizeError> { + pub fn expand_grid(&mut self, rings_to_add: u64) -> Result<(), ResizeError> { self.change_grid_by_rings(rings_to_add, GrowDirection::Expand) } - fn shrink_grid(&mut self, rings_to_shrink_by: u64) -> Result<(), ResizeError> { + /// Like [`grid::expand_grid`], but instead of expanding, this shrinks. + pub fn shrink_grid(&mut self, rings_to_shrink_by: u64) -> Result<(), ResizeError> { self.change_grid_by_rings(rings_to_shrink_by, GrowDirection::Shrink) } } From a41418e9f56c3df00746317e727b6168686a5b62 Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sat, 2 Mar 2024 22:51:36 +0000 Subject: [PATCH 3/3] Fix name issue in docs --- backend/src/grid.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/backend/src/grid.rs b/backend/src/grid.rs index f4219d1..ef20ebf 100644 --- a/backend/src/grid.rs +++ b/backend/src/grid.rs @@ -48,7 +48,7 @@ impl Grid { } - /// Adds a new space to the grid. In the future, this may return an [`std::result::Result::Err`] if the space is of type [`space::SpaceKind::EmptySpace`]. + /// Adds a new space to the grid. In the future, this may return an [`std::result::Result::Err`] if the space is of type [`crate::space::SpaceKind::EmptySpace`]. pub fn insert(&mut self, space: Space) { self.spaces.insert(space.coordinate, space); } @@ -99,7 +99,7 @@ impl Grid { .collect::>() } - /// Returns neighbours which are in the grid *and* which aren't [`SpaceKind::EmptySpace`]s. + /// Returns neighbours which are in the grid *and* which aren't [`crate::space::SpaceKind::EmptySpace`]s. pub fn get_nonempty_neighbours(&self, coordinate: Coordinate) -> Vec<&Space> { self.neighbour_coords_in_bounds(coordinate).iter() .map(|coord| self.get_space(*coord)) @@ -182,7 +182,7 @@ impl Grid { self.change_grid_by_rings(rings_to_add, GrowDirection::Expand) } - /// Like [`grid::expand_grid`], but instead of expanding, this shrinks. + /// Like [`crate::grid::Grid::expand_grid`], but instead of expanding, this shrinks. pub fn shrink_grid(&mut self, rings_to_shrink_by: u64) -> Result<(), ResizeError> { self.change_grid_by_rings(rings_to_shrink_by, GrowDirection::Shrink) }