From 432fb3e8dc3e2ced4a642a9dd4e0742a65a49357 Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sun, 3 Mar 2024 04:36:25 +0000 Subject: [PATCH 1/6] Add questionably correct REST client code --- backend/Cargo.toml | 4 ++- backend/src/cache_client.rs | 72 +++++++++++++++++++++++++++++++++++++ backend/src/lib.rs | 2 ++ backend/src/recipe.rs | 12 +++++++ 4 files changed, 89 insertions(+), 1 deletion(-) create mode 100644 backend/src/cache_client.rs create mode 100644 backend/src/recipe.rs diff --git a/backend/Cargo.toml b/backend/Cargo.toml index d5f3402..1096e14 100644 --- a/backend/Cargo.toml +++ b/backend/Cargo.toml @@ -18,4 +18,6 @@ warp = "0.3" serde = { version = "1.0", features = ["derive"] } serde_json = "1.0" futures = { version = "0.3", default-features=false} -serde_repr = "0.1" \ No newline at end of file +serde_repr = "0.1" +reqwest = "0.11.24" +url = "2.5.0" diff --git a/backend/src/cache_client.rs b/backend/src/cache_client.rs new file mode 100644 index 0000000..d123d7c --- /dev/null +++ b/backend/src/cache_client.rs @@ -0,0 +1,72 @@ +use crate::kube::{Kube, KubeId}; +use crate::recipe::Recipe; +use serde::Deserialize; +use thiserror::Error; +use reqwest::Url; +use uuid::Uuid; + +#[derive(Debug, Error)] +pub enum EnvVarParseError { + #[error("Error while getting environment variable. Have you included an equals sign?")] + EnvironmentError(#[from] std::env::VarError), + #[error("Failed to parse URL, is it valid?")] + UrlParseError(#[from] url::ParseError) +} + +#[derive(Debug, Error)] +pub enum RestError { + #[error("Error while trying to reach REST server.")] + ReqwestError(#[from] reqwest::Error), + #[error("Could not parse JSON.")] + SerdeError(#[from] serde_json::Error), +} + +#[derive(Debug)] +pub struct CacheClient { + url: Url, +} +impl CacheClient { + pub fn new() -> Result { + let url_string = std::env::var("ENDPOINT")?; + Ok(CacheClient { + url: Url::parse(&url_string)?, + }) + } + pub fn from_url(url: &str) -> Result { + Ok(CacheClient { + url: Url::parse(url)?, + }) + } + pub async fn get_kubes(&self) -> Result, RestError> { + let res = reqwest::get(self.url.join("kubes").unwrap()).await?.text().await?; + Ok(serde_json::from_str(&res)?) + } + pub async fn get_recipes(&self) -> Result, RestError> { + let res = reqwest::get(self.url.join("kubeRecipes").unwrap()).await?.text().await?; + Ok(serde_json::from_str(&res)?) + } + pub async fn get_kube_by_id(&self, id: KubeId) -> Result { + let res = reqwest::get(self.url.join(format!("kube/:{}", id.uuid()).as_str()).unwrap()).await?.text().await?; + Ok(serde_json::from_str(&res)?) + } + pub async fn get_recipe_by_id(&self, id1: Uuid, id2: Uuid) -> Result { + let res = reqwest::get(self.url.join(format!("kubeRecipeByIds/:{}/:{}", id1, id2).as_str()).unwrap()).await?.text().await?; + Ok(serde_json::from_str(&res)?) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[tokio::test] + async fn get_kubes() { + let client = CacheClient::from_url("https://hack.djpiper28.co.uk/cache/").unwrap(); + let kubes = client.get_kubes().await.unwrap(); + let control_kubes_string = String::from( +"[{\"name\":\"hydrogen\",\"id\":\"153227c6-c069-4748-b5aa-aafac8abef00\"},{\"name\":\"oxygen\",\"id\":\"3ffe4c9e-5d35-42c9-a70e-1d80c544bdbb\"},{\"name\":\"nitrogen\",\"id\":\"bb155bf1-7200-45e7-b126-f2882f7aecaa\"},{\"name\":\"calcium\",\"id\":\"5d75a6e7-b45b-4c5f-ae14-042ad17b3156\"},{\"name\":\"iron\",\"id\":\"a0e89ba5-41cb-4a8d-895e-866f0b2004f2\"},{\"name\":\"aluminium\",\"id\":\"f475d09e-75e7-4096-af5b-a00de9ae5e48\"},{\"name\":\"uranium\",\"id\":\"615b14af-d4a8-4e57-8a6c-190110114ad1\"},{\"name\":\"sodium\",\"id\":\"485cbd9d-5608-4b6a-afbe-52829663090d\"},{\"name\":\"chlorine\",\"id\":\"dbe34fef-e969-4f4d-b148-7898d6de699c\"},{\"name\":\"light\",\"id\":\"0652bcac-e87b-4323-9d05-939ddfd1c726\"},{\"name\":\"time\",\"id\":\"efd60b9a-dcc4-4690-a087-c1aba4a7d16e\"},{\"name\":\"silicon\",\"id\":\"c37997c9-e7ba-44d6-b726-97aec89545f8\"},{\"name\":\"water\",\"id\":\"bf1ed678-3be7-42d7-8544-82124b6c6111\"},{\"name\":\"tap water\",\"id\":\"7bc0ee7b-6772-4315-84dc-93887e007bd6\"},{\"name\":\"salt\",\"id\":\"d24a2af2-eed2-40af-a064-001de3545e84\"},{\"name\":\"sea water\",\"id\":\"c622c644-1590-4e4f-b7ae-42fb358ad004\"},{\"name\":\"air\",\"id\":\"ad2ed248-7bc8-469e-b9bb-7b1bc4878cad\"},{\"name\":\"rust\",\"id\":\"1c213d9d-b060-45c5-a8e0-48c45b01c30b\"},{\"name\":\"feldspar\",\"id\":\"9796d796-d166-4a7e-895a-fd9bf65dd3ee\"},{\"name\":\"sand\",\"id\":\"2d727d5b-a71c-49c3-84ef-cffa2fe47f9a\"},{\"name\":\"dirt\",\"id\":\"b4a9dc42-273a-4dc6-9c76-c51a847b743c\"},{\"name\":\"beach\",\"id\":\"e7dcaab2-1c0a-4af2-a5a5-38995f4c0167\"},{\"name\":\"earth\",\"id\":\"39807784-9d57-48fc-8e7a-e877e975eb9c\"},{\"name\":\"life\",\"id\":\"dc383720-1649-4119-ad0a-68f209b237d5\"},{\"name\":\"age\",\"id\":\"181ec200-54aa-4dca-821d-b88efe6b129d\"},{\"name\":\"energy\",\"id\":\"abc9bcf7-47d2-46f0-bf6e-f70a126820b7\"},{\"name\":\"rock\",\"id\":\"0b3ab01f-4fc6-4bc8-8acc-5ae2a7fb640f\"},{\"name\":\"fire\",\"id\":\"bbee222a-4f4a-477a-8be8-2e6f314b5a36\"},{\"name\":\"glass\",\"id\":\"f84b34f7-5406-4795-a56d-9d07c4b3c9d5\"},{\"name\":\"steam\",\"id\":\"a38fd90e-c2ff-4955-9161-462018dfaf52\"},{\"name\":\"radioactive water\",\"id\":\"6bffa307-4114-40ff-bf43-6934945b1753\"},{\"name\":\"nuclear reactor\",\"id\":\"021b0835-e16e-4163-ac66-4f7c7b78ce7e\"}]" + ); + let control_kubes: Vec = serde_json::from_str(&control_kubes_string).unwrap(); + assert_eq!(control_kubes, kubes) + } +} diff --git a/backend/src/lib.rs b/backend/src/lib.rs index ba733cf..1dc5894 100644 --- a/backend/src/lib.rs +++ b/backend/src/lib.rs @@ -4,5 +4,7 @@ pub mod kube; pub mod player; pub mod llm; pub mod local_grid; +pub mod cache_client; +pub mod recipe; type Coordinate = [u64; 2]; diff --git a/backend/src/recipe.rs b/backend/src/recipe.rs new file mode 100644 index 0000000..f12e7d8 --- /dev/null +++ b/backend/src/recipe.rs @@ -0,0 +1,12 @@ +use serde::{Deserialize, Serialize}; +use uuid::Uuid; +use crate::kube::{Kube, KubeId}; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Recipe { + id: Uuid, + outputId: Uuid, + outputKube: Kube, + kube1_id: KubeId, + kube2_id: KubeId, +} \ No newline at end of file From 5ce3cbc2b799427b61e82d936ab93c1cfc80682a Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sun, 3 Mar 2024 08:22:03 +0000 Subject: [PATCH 2/6] Fix errors in cache client and remove KubeId --- backend/src/cache_client.rs | 26 +++++++++++------ backend/src/kube.rs | 57 +++++++++++++++++++++---------------- backend/src/recipe.rs | 10 +++---- 3 files changed, 55 insertions(+), 38 deletions(-) diff --git a/backend/src/cache_client.rs b/backend/src/cache_client.rs index d123d7c..02f504e 100644 --- a/backend/src/cache_client.rs +++ b/backend/src/cache_client.rs @@ -1,4 +1,4 @@ -use crate::kube::{Kube, KubeId}; +use crate::kube::{Kube}; use crate::recipe::Recipe; use serde::Deserialize; use thiserror::Error; @@ -45,12 +45,12 @@ impl CacheClient { let res = reqwest::get(self.url.join("kubeRecipes").unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } - pub async fn get_kube_by_id(&self, id: KubeId) -> Result { - let res = reqwest::get(self.url.join(format!("kube/:{}", id.uuid()).as_str()).unwrap()).await?.text().await?; + pub async fn get_kube_by_id(&self, id: Uuid) -> Result { + let res = reqwest::get(self.url.join(format!("kubeById/{}", id).as_str()).unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } pub async fn get_recipe_by_id(&self, id1: Uuid, id2: Uuid) -> Result { - let res = reqwest::get(self.url.join(format!("kubeRecipeByIds/:{}/:{}", id1, id2).as_str()).unwrap()).await?.text().await?; + let res = reqwest::get(self.url.join(format!("kubeRecipeByIds/{}/{}", id1, id2).as_str()).unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } } @@ -60,13 +60,23 @@ mod tests { use super::*; #[tokio::test] - async fn get_kubes() { + async fn try_get_kubes() { let client = CacheClient::from_url("https://hack.djpiper28.co.uk/cache/").unwrap(); - let kubes = client.get_kubes().await.unwrap(); + let mut kubes = client.get_kubes().await.unwrap(); let control_kubes_string = String::from( -"[{\"name\":\"hydrogen\",\"id\":\"153227c6-c069-4748-b5aa-aafac8abef00\"},{\"name\":\"oxygen\",\"id\":\"3ffe4c9e-5d35-42c9-a70e-1d80c544bdbb\"},{\"name\":\"nitrogen\",\"id\":\"bb155bf1-7200-45e7-b126-f2882f7aecaa\"},{\"name\":\"calcium\",\"id\":\"5d75a6e7-b45b-4c5f-ae14-042ad17b3156\"},{\"name\":\"iron\",\"id\":\"a0e89ba5-41cb-4a8d-895e-866f0b2004f2\"},{\"name\":\"aluminium\",\"id\":\"f475d09e-75e7-4096-af5b-a00de9ae5e48\"},{\"name\":\"uranium\",\"id\":\"615b14af-d4a8-4e57-8a6c-190110114ad1\"},{\"name\":\"sodium\",\"id\":\"485cbd9d-5608-4b6a-afbe-52829663090d\"},{\"name\":\"chlorine\",\"id\":\"dbe34fef-e969-4f4d-b148-7898d6de699c\"},{\"name\":\"light\",\"id\":\"0652bcac-e87b-4323-9d05-939ddfd1c726\"},{\"name\":\"time\",\"id\":\"efd60b9a-dcc4-4690-a087-c1aba4a7d16e\"},{\"name\":\"silicon\",\"id\":\"c37997c9-e7ba-44d6-b726-97aec89545f8\"},{\"name\":\"water\",\"id\":\"bf1ed678-3be7-42d7-8544-82124b6c6111\"},{\"name\":\"tap water\",\"id\":\"7bc0ee7b-6772-4315-84dc-93887e007bd6\"},{\"name\":\"salt\",\"id\":\"d24a2af2-eed2-40af-a064-001de3545e84\"},{\"name\":\"sea water\",\"id\":\"c622c644-1590-4e4f-b7ae-42fb358ad004\"},{\"name\":\"air\",\"id\":\"ad2ed248-7bc8-469e-b9bb-7b1bc4878cad\"},{\"name\":\"rust\",\"id\":\"1c213d9d-b060-45c5-a8e0-48c45b01c30b\"},{\"name\":\"feldspar\",\"id\":\"9796d796-d166-4a7e-895a-fd9bf65dd3ee\"},{\"name\":\"sand\",\"id\":\"2d727d5b-a71c-49c3-84ef-cffa2fe47f9a\"},{\"name\":\"dirt\",\"id\":\"b4a9dc42-273a-4dc6-9c76-c51a847b743c\"},{\"name\":\"beach\",\"id\":\"e7dcaab2-1c0a-4af2-a5a5-38995f4c0167\"},{\"name\":\"earth\",\"id\":\"39807784-9d57-48fc-8e7a-e877e975eb9c\"},{\"name\":\"life\",\"id\":\"dc383720-1649-4119-ad0a-68f209b237d5\"},{\"name\":\"age\",\"id\":\"181ec200-54aa-4dca-821d-b88efe6b129d\"},{\"name\":\"energy\",\"id\":\"abc9bcf7-47d2-46f0-bf6e-f70a126820b7\"},{\"name\":\"rock\",\"id\":\"0b3ab01f-4fc6-4bc8-8acc-5ae2a7fb640f\"},{\"name\":\"fire\",\"id\":\"bbee222a-4f4a-477a-8be8-2e6f314b5a36\"},{\"name\":\"glass\",\"id\":\"f84b34f7-5406-4795-a56d-9d07c4b3c9d5\"},{\"name\":\"steam\",\"id\":\"a38fd90e-c2ff-4955-9161-462018dfaf52\"},{\"name\":\"radioactive water\",\"id\":\"6bffa307-4114-40ff-bf43-6934945b1753\"},{\"name\":\"nuclear reactor\",\"id\":\"021b0835-e16e-4163-ac66-4f7c7b78ce7e\"}]" +"[{\"name\":\"hydrogen\",\"id\":\"cdede93f-d0d7-4b4a-9fde-2a909444c58d\"},{\"name\":\"oxygen\",\"id\":\"8ddcf7ad-61f6-47ff-a49c-4abcec21d6a1\"},{\"name\":\"nitrogen\",\"id\":\"59a64f5b-bcf4-4d2d-bb7f-bc4eceaf41e5\"},{\"name\":\"calcium\",\"id\":\"2b006956-063d-4ca2-b75d-f6e5d67455c9\"},{\"name\":\"iron\",\"id\":\"5e930e14-4597-49b3-95fa-3e6dcc40b1ae\"},{\"name\":\"aluminium\",\"id\":\"d076033a-c583-4d38-8094-249a7fe2972b\"},{\"name\":\"uranium\",\"id\":\"82ac0ed4-62e3-4c5e-af3e-024c38285227\"},{\"name\":\"sodium\",\"id\":\"1c7fda1b-af90-411d-8162-8fd04c4890d3\"},{\"name\":\"chlorine\",\"id\":\"061f6efd-0067-4d71-92b8-a0b58562b906\"},{\"name\":\"light\",\"id\":\"e38ba705-58c1-469d-8eff-7cc01b94fd46\"},{\"name\":\"time\",\"id\":\"6991989a-f347-48eb-8c67-ade4cdc010d0\"},{\"name\":\"silicon\",\"id\":\"72650f96-011b-404d-aba0-2c8aa2f17aeb\"},{\"name\":\"water\",\"id\":\"8cf89e77-bf8b-4cf4-9941-46b853df4480\"},{\"name\":\"tap water\",\"id\":\"5ca230bc-135e-4be8-8be3-7c7c1b3e5484\"},{\"name\":\"salt\",\"id\":\"540710d4-5d7d-42f6-b9b7-aad181affbcf\"},{\"name\":\"sea water\",\"id\":\"74102256-00b5-42aa-9665-1bc81f13c18b\"},{\"name\":\"air\",\"id\":\"88bb179c-0d91-4322-a4e5-d3862bf83a31\"},{\"name\":\"rust\",\"id\":\"ad92587b-1643-469e-8357-1d6ec5ab6380\"},{\"name\":\"feldspar\",\"id\":\"2adfa430-faa4-4ecd-95e1-c6cc8c85f3b5\"},{\"name\":\"sand\",\"id\":\"1db355de-1404-4e7c-bf8b-a6b3355b9dc4\"},{\"name\":\"dirt\",\"id\":\"649e8325-530b-4abb-8fe0-5b79b395e84f\"},{\"name\":\"beach\",\"id\":\"03f9f164-be03-4de3-b5b4-c7549c1ef9e4\"},{\"name\":\"earth\",\"id\":\"66744f80-ccec-4c8b-9025-9edbb75df0a6\"},{\"name\":\"life\",\"id\":\"abd2f9a5-34cd-4b3f-941f-8cf934f6b967\"},{\"name\":\"age\",\"id\":\"bda3c6c5-3b79-418d-8d24-cc533a509065\"},{\"name\":\"energy\",\"id\":\"b4eba917-2179-4cd4-a64e-316fe005f11e\"},{\"name\":\"rock\",\"id\":\"3c545935-1382-4c3d-9771-1fa8492f1b77\"},{\"name\":\"fire\",\"id\":\"1e2e14df-f36b-43a4-9a2a-5112f84abb52\"},{\"name\":\"glass\",\"id\":\"12ec3f8d-0986-42d7-acc2-e6af8ba1842a\"}]" ); - let control_kubes: Vec = serde_json::from_str(&control_kubes_string).unwrap(); + let mut control_kubes: Vec = serde_json::from_str(&control_kubes_string).unwrap(); + kubes.sort(); + control_kubes.sort(); assert_eq!(control_kubes, kubes) } + #[tokio::test] + async fn try_kube_by_id() { + let client = CacheClient::from_url("https://hack.djpiper28.co.uk/cache/").unwrap(); + let kube = client.get_kube_by_id(Uuid::parse_str("5e930e14-4597-49b3-95fa-3e6dcc40b1ae").unwrap()).await.unwrap(); + let expected_string = String::from("{\"name\":\"iron\",\"id\":\"5e930e14-4597-49b3-95fa-3e6dcc40b1ae\"}"); + let expected_kube: Kube = serde_json::from_str(&expected_string).unwrap(); + assert_eq!(expected_kube, kube); + } } diff --git a/backend/src/kube.rs b/backend/src/kube.rs index e5f755b..fc6ffd8 100644 --- a/backend/src/kube.rs +++ b/backend/src/kube.rs @@ -1,41 +1,48 @@ +use std::str::FromStr; + use uuid::Uuid; use serde::{Deserialize, Serialize}; -#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone, Serialize, Deserialize)] -pub struct KubeId { - uuid: Uuid, -} - -impl KubeId { - pub fn new(name: &str) -> Self { - let mut name = name.to_string(); - name.push_str("kube"); - KubeId { - uuid: Uuid::new_v5(&Uuid::NAMESPACE_DNS, name.as_bytes()), - } - } - - pub fn as_u128(&self) -> u128 { - self.uuid.as_u128() - } - - pub fn uuid(&self) -> Uuid { - self.uuid - } -} - #[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)] pub struct Kube { - pub id: KubeId, + pub id: Uuid, pub name: String, } impl Kube { pub fn new(name: String) -> Kube { + let mut name_uuid = name.clone(); + name_uuid.push_str("kube"); Kube { - id: KubeId::new(name.as_str()), + id: Uuid::new_v5(&Uuid::NAMESPACE_DNS, name_uuid.as_bytes()), name, } } + pub fn from_name_uuid(name: &str, uuid: &str) -> Result::Err> { + Ok(Kube { + id: Uuid::from_str(uuid)?, + name: name.to_string() + }) + } } // we should have a placeholder ''loading'' cube we can send over if api is slow + +#[cfg(test)] +mod tests { + use super::*; + #[test] + fn parse_kubes() { + let input_json = String::from( +"[{\"name\":\"hydrogen\",\"id\":\"153227c6-c069-4748-b5aa-aafac8abef00\"},{\"name\":\"oxygen\",\"id\":\"3ffe4c9e-5d35-42c9-a70e-1d80c544bdbb\"},{\"name\":\"nitrogen\",\"id\":\"bb155bf1-7200-45e7-b126-f2882f7aecaa\"}]" + ); + let mut expected: Vec = vec![ + Kube::from_name_uuid("hydrogen", "153227c6-c069-4748-b5aa-aafac8abef00").unwrap(), + Kube::from_name_uuid("oxygen", "3ffe4c9e-5d35-42c9-a70e-1d80c544bdbb").unwrap(), + Kube::from_name_uuid("nitrogen", "bb155bf1-7200-45e7-b126-f2882f7aecaa").unwrap(), + ]; + let mut kubes: Vec = serde_json::from_str(&input_json).unwrap(); + expected.sort(); + kubes.sort(); + assert_eq!(expected, kubes); + } +} diff --git a/backend/src/recipe.rs b/backend/src/recipe.rs index f12e7d8..9c333e3 100644 --- a/backend/src/recipe.rs +++ b/backend/src/recipe.rs @@ -1,12 +1,12 @@ use serde::{Deserialize, Serialize}; use uuid::Uuid; -use crate::kube::{Kube, KubeId}; +use crate::kube::Kube; #[derive(Debug, Clone, Serialize, Deserialize)] pub struct Recipe { id: Uuid, - outputId: Uuid, - outputKube: Kube, - kube1_id: KubeId, - kube2_id: KubeId, + output_id: Uuid, + output_kube: Kube, + kube1_id: Uuid, + kube2_id: Uuid, } \ No newline at end of file From f65d05f5c1be580ad82f13c37b5be78858d34c46 Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sun, 3 Mar 2024 08:29:33 +0000 Subject: [PATCH 3/6] Add docs to the cache client --- backend/src/cache_client.rs | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/backend/src/cache_client.rs b/backend/src/cache_client.rs index 02f504e..b7e1667 100644 --- a/backend/src/cache_client.rs +++ b/backend/src/cache_client.rs @@ -21,34 +21,65 @@ pub enum RestError { SerdeError(#[from] serde_json::Error), } +/// A client which interacts with the cache server. #[derive(Debug)] pub struct CacheClient { + /// The endpoint URL. url: Url, } impl CacheClient { + /// Create a new client using an environment variable. + /// + /// # Errors + /// Will error when the environment variable isn't defined correctly, or if the URL couldn't be parsed. pub fn new() -> Result { let url_string = std::env::var("ENDPOINT")?; Ok(CacheClient { url: Url::parse(&url_string)?, }) } + /// Create a new client using a given endpoint URL. + /// + /// # Errors + /// + /// Will error if the URL couldn't be parsed. pub fn from_url(url: &str) -> Result { Ok(CacheClient { url: Url::parse(url)?, }) } + /// Gets all the Kubes from the cache server. + /// + /// # Errors + /// + /// If the REST server connection fails, or if the output is an error (like 404), then this will return an Err. pub async fn get_kubes(&self) -> Result, RestError> { let res = reqwest::get(self.url.join("kubes").unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } + /// Gets all the recipes from the cache server. + /// + /// # Errors + /// + /// If the REST server connection fails, or if the output is an error (like 404), then this will return an Err. pub async fn get_recipes(&self) -> Result, RestError> { let res = reqwest::get(self.url.join("kubeRecipes").unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } + /// Gets a Kube by its ID from the cache server. + /// + /// # Errors + /// + /// If the REST server connection fails, or if the output is an error (like 404), then this will return an Err. pub async fn get_kube_by_id(&self, id: Uuid) -> Result { let res = reqwest::get(self.url.join(format!("kubeById/{}", id).as_str()).unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } + /// Gets a recipe by its ID from the cache server. + /// + /// # Errors + /// + /// If the REST server connection fails, or if the output is an error (like 404), then this will return an Err. pub async fn get_recipe_by_id(&self, id1: Uuid, id2: Uuid) -> Result { let res = reqwest::get(self.url.join(format!("kubeRecipeByIds/{}/{}", id1, id2).as_str()).unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) From 134ea31617cfc6a5e37db115ce512139d20d86e1 Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sun, 3 Mar 2024 08:31:48 +0000 Subject: [PATCH 4/6] Do some tidying of the code --- backend/src/cache_client.rs | 25 ++++++++++++------------- backend/src/grid.rs | 4 ++-- backend/src/ws.rs | 2 +- 3 files changed, 15 insertions(+), 16 deletions(-) diff --git a/backend/src/cache_client.rs b/backend/src/cache_client.rs index b7e1667..8d6bb15 100644 --- a/backend/src/cache_client.rs +++ b/backend/src/cache_client.rs @@ -1,6 +1,5 @@ -use crate::kube::{Kube}; +use crate::kube::Kube; use crate::recipe::Recipe; -use serde::Deserialize; use thiserror::Error; use reqwest::Url; use uuid::Uuid; @@ -29,7 +28,7 @@ pub struct CacheClient { } impl CacheClient { /// Create a new client using an environment variable. - /// + /// /// # Errors /// Will error when the environment variable isn't defined correctly, or if the URL couldn't be parsed. pub fn new() -> Result { @@ -39,9 +38,9 @@ impl CacheClient { }) } /// Create a new client using a given endpoint URL. - /// + /// /// # Errors - /// + /// /// Will error if the URL couldn't be parsed. pub fn from_url(url: &str) -> Result { Ok(CacheClient { @@ -49,36 +48,36 @@ impl CacheClient { }) } /// Gets all the Kubes from the cache server. - /// + /// /// # Errors - /// + /// /// If the REST server connection fails, or if the output is an error (like 404), then this will return an Err. pub async fn get_kubes(&self) -> Result, RestError> { let res = reqwest::get(self.url.join("kubes").unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } /// Gets all the recipes from the cache server. - /// + /// /// # Errors - /// + /// /// If the REST server connection fails, or if the output is an error (like 404), then this will return an Err. pub async fn get_recipes(&self) -> Result, RestError> { let res = reqwest::get(self.url.join("kubeRecipes").unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } /// Gets a Kube by its ID from the cache server. - /// + /// /// # Errors - /// + /// /// If the REST server connection fails, or if the output is an error (like 404), then this will return an Err. pub async fn get_kube_by_id(&self, id: Uuid) -> Result { let res = reqwest::get(self.url.join(format!("kubeById/{}", id).as_str()).unwrap()).await?.text().await?; Ok(serde_json::from_str(&res)?) } /// Gets a recipe by its ID from the cache server. - /// + /// /// # Errors - /// + /// /// If the REST server connection fails, or if the output is an error (like 404), then this will return an Err. pub async fn get_recipe_by_id(&self, id1: Uuid, id2: Uuid) -> Result { let res = reqwest::get(self.url.join(format!("kubeRecipeByIds/{}/{}", id1, id2).as_str()).unwrap()).await?.text().await?; diff --git a/backend/src/grid.rs b/backend/src/grid.rs index 4c9e115..c127269 100644 --- a/backend/src/grid.rs +++ b/backend/src/grid.rs @@ -67,7 +67,7 @@ impl Grid { /// ```rust /// use cosmic_kube::grid::Grid; /// use cosmic_kube::space::{ Space, SpaceKind }; - /// + /// /// let grid = Grid::from_spaces( /// vec![ /// Space::new([0, 2], SpaceKind::EmptySpace), @@ -206,7 +206,7 @@ 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. pub fn expand_grid(&mut self, rings_to_add: u64) -> Result<(), ResizeError> { self.change_grid_by_rings(rings_to_add, GrowDirection::Expand) diff --git a/backend/src/ws.rs b/backend/src/ws.rs index d9152e8..219964a 100644 --- a/backend/src/ws.rs +++ b/backend/src/ws.rs @@ -58,7 +58,7 @@ pub async fn client_connection(ws: WebSocket, clients: Clients) { // ->recieve client game info <- send back client game state -// wwwwwwwwwwwwwwwwwwwww i am so tired +// wwwwwwwwwwwwwwwwwwwww i am so tired async fn client_msg(client_id: &str, msg: Message, clients: &Clients) { println!("received message from {}: {:?}", client_id, msg); //debug From ca185f253aa6655c3b7942978dfabe8283c0f52c Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sun, 3 Mar 2024 08:54:02 +0000 Subject: [PATCH 5/6] Add more documentation --- backend/src/cache_client.rs | 6 ++++++ backend/src/grid.rs | 32 +++++++++++++++++++++++++++----- 2 files changed, 33 insertions(+), 5 deletions(-) diff --git a/backend/src/cache_client.rs b/backend/src/cache_client.rs index 8d6bb15..fca0d7c 100644 --- a/backend/src/cache_client.rs +++ b/backend/src/cache_client.rs @@ -5,18 +5,24 @@ use reqwest::Url; use uuid::Uuid; #[derive(Debug, Error)] +/// Errors while parsing the environment variable for the endpoint URL. pub enum EnvVarParseError { #[error("Error while getting environment variable. Have you included an equals sign?")] + /// Used when the environment variable hasn't been set properly (such as a missing equals sign). EnvironmentError(#[from] std::env::VarError), #[error("Failed to parse URL, is it valid?")] + /// Used when the URL can't be parsed, usually meaning that it's not been written correctly. UrlParseError(#[from] url::ParseError) } #[derive(Debug, Error)] +/// Errors while communicating with the REST cache server. pub enum RestError { #[error("Error while trying to reach REST server.")] + /// Failed to establish connection to the REST server, or got another error (e.g. 404). ReqwestError(#[from] reqwest::Error), #[error("Could not parse JSON.")] + /// Couldn't parse the JSON, usually because something else went wrong when trying to establish a connection to the server. SerdeError(#[from] serde_json::Error), } diff --git a/backend/src/grid.rs b/backend/src/grid.rs index c127269..64ced36 100644 --- a/backend/src/grid.rs +++ b/backend/src/grid.rs @@ -7,6 +7,7 @@ fn distance([a, b]: Coordinate, [x, y]: Coordinate) -> u64 { (a.abs_diff(x)).max(b.abs_diff(y)) } +#[derive(Debug, Clone, Copy)] /// The direction in which to grow. For example, going from 3 to 9 would give a `GrowDirection::Expand`. enum GrowDirection { Shrink, @@ -56,6 +57,10 @@ impl Grid { pub fn insert(&mut self, space: Space) { self.spaces.insert(space.coordinate, space); } + /// Removes the space at the given coordinate. If there is no recorded space there, then this returns [`std::option::Option::None`]. + pub fn remove(&mut self, coordinate: Coordinate) -> Option { + self.spaces.remove(&coordinate) + } /// Checks that a coordinate is not beyond the bounds of the grid. pub fn in_bounds(&self, coordinate: [u64; 2]) -> bool { coordinate[0] < self.width && coordinate[1] < self.height @@ -83,6 +88,9 @@ impl Grid { self.spaces.get(&coordinate) } + /// Gets the neighbours that are *n* squares away. + /// + /// In other words, this will look at all the rings which are 1, 2, …, n squares away from the coordinate. It will return any squares found in these rings. pub fn get_neighbours_n_away(&self, coordinate: Coordinate, n: u64) -> Vec<&Space> { let mut coords: Vec = Vec::new(); let mut stack: Vec = self.neighbour_coords_in_bounds(coordinate); @@ -97,18 +105,18 @@ impl Grid { .filter(|c| distance(coordinate, **c) <= n && distance(coordinate, **c) > distance(coord, **c) ) - ) + ); } let mut to_return: Vec<&Space> = Vec::new(); for coord in coords { - match self.spaces.get(&coord) { - Some(space) => to_return.push(space), - None => (), + if let Some(space) = self.spaces.get(&coord) { + to_return.push(space) } } to_return } + /// Gets all the neighbours (directly adjacent squares incl. diagonally) of the coordinate if they're in bounds. fn neighbour_coords_in_bounds(&self, coordinate: Coordinate) -> Vec { let coordinates: [[Option; 2]; 8] = [ [coordinate[0].checked_add(1), Some(coordinate[1])], @@ -129,7 +137,7 @@ impl Grid { .collect::>() } - /// Returns neighbours which are in the grid *and* which aren't [`crate::space::SpaceKind::EmptySpace`]s. + /// Returns neighbours (adjacent incl. diagonally) 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)) @@ -208,11 +216,25 @@ 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. + /// + ///
Warning! This function may use a lot of memory! When resizing the grid, the program needs to copy all of the grid's contents to a new `Vec`. If the grid is densely populated then take care when calling it.
+ /// + /// # Errors + /// + /// Will fail if the resize request is invalid, if the new grid will be too big, (bigger than 2^64 - 1). pub fn expand_grid(&mut self, rings_to_add: u64) -> Result<(), ResizeError> { self.change_grid_by_rings(rings_to_add, GrowDirection::Expand) } /// Like [`crate::grid::Grid::expand_grid`], but instead of expanding, this shrinks. + /// + ///
Warning! This function may use a lot of memory! When resizing the grid, the program needs to copy all of the grid's contents to a new `Vec`. If the grid is densely populated then take care when calling it.
+ /// + /// # Errors + /// + /// Will fail if the resize request is invalid. + /// - If the new grid will be too small, (smaller than 1). + /// - If there is a Kube near the edge such that it would be "cut off" when the rings are removed. 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 9ff124e70f5d4288692e0fe545bedc964c3e1dfc Mon Sep 17 00:00:00 2001 From: ettolrach Date: Sun, 3 Mar 2024 09:30:26 +0000 Subject: [PATCH 6/6] Remove unused LLM Rust file --- backend/src/llm.rs | 52 ---------------------------------------------- 1 file changed, 52 deletions(-) delete mode 100644 backend/src/llm.rs diff --git a/backend/src/llm.rs b/backend/src/llm.rs deleted file mode 100644 index c94aecb..0000000 --- a/backend/src/llm.rs +++ /dev/null @@ -1,52 +0,0 @@ -use crate::kube::Kube; -use async_trait::async_trait; - -/// Trait for interacting with an LLM. -#[async_trait] -pub trait LLM { - /// Send a query to the LLM and get a [`std::string::String`] response. - async fn query(input: &str) -> String; - /// Ask the LLM to combine the given Kubes and return a new Kube. - async fn combine(&self, kubes: &[Kube]) -> Kube; -} - -/// A fake LLM that functions very basically, not processing the input in any meaningful way. This is most useful for testing functionality of other features which use LLMs. -pub struct FakeLLM { -} -impl FakeLLM { - fn new() -> FakeLLM { - FakeLLM { } - } -} - -#[async_trait] -impl LLM for FakeLLM { - async fn query(input: &str) -> String { - format!("This is a response to: {input}") - } - async fn combine(&self, kubes: &[Kube]) -> Kube { - let mut new_string = String::new(); - for kube in kubes { - new_string.push_str(kube.name.as_str()); - } - Kube::new(new_string) - } -} - -#[cfg(test)] -mod tests { - use super::*; - #[tokio::test] - async fn fake_combine_test() { - let kubes = vec![ - Kube::new(String::from("water")), - Kube::new(String::from("glass")), - ]; - let fake_llm = FakeLLM::new(); - let response_kube = fake_llm.combine(&kubes).await; - assert_eq!( - String::from("waterglass"), - response_kube.name, - ); - } -}