From 2744af8de06f0986c4c48223ef5c0f05265da6b8 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 13:58:16 +0000 Subject: [PATCH 01/11] Update cgp crate --- Cargo.lock | 43 +++++++++++++++++++++++--------------- Cargo.toml | 1 + content/field-accessors.md | 12 +++++------ 3 files changed, 33 insertions(+), 23 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 81a76a9..426cdcf 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -101,7 +101,7 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" [[package]] name = "cgp" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-async", "cgp-core", @@ -111,7 +111,7 @@ dependencies = [ [[package]] name = "cgp-async" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-async-macro", "cgp-sync", @@ -120,7 +120,7 @@ dependencies = [ [[package]] name = "cgp-async-macro" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "proc-macro2", "quote", @@ -130,12 +130,12 @@ dependencies = [ [[package]] name = "cgp-component" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" [[package]] name = "cgp-component-macro" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-component-macro-lib", ] @@ -143,7 +143,7 @@ dependencies = [ [[package]] name = "cgp-component-macro-lib" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "itertools", "prettyplease", @@ -155,7 +155,7 @@ dependencies = [ [[package]] name = "cgp-core" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-async", "cgp-component", @@ -169,7 +169,7 @@ dependencies = [ [[package]] name = "cgp-error" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-async", "cgp-component", @@ -180,17 +180,26 @@ dependencies = [ [[package]] name = "cgp-error-anyhow" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "anyhow", "cgp-core", ] +[[package]] +name = "cgp-error-extra" +version = "0.2.0" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +dependencies = [ + "cgp-error", +] + [[package]] name = "cgp-extra" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ + "cgp-error-extra", "cgp-inner", "cgp-run", "cgp-runtime", @@ -199,7 +208,7 @@ dependencies = [ [[package]] name = "cgp-field" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-component", "cgp-type", @@ -208,7 +217,7 @@ dependencies = [ [[package]] name = "cgp-field-macro" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-field-macro-lib", "proc-macro2", @@ -217,7 +226,7 @@ dependencies = [ [[package]] name = "cgp-field-macro-lib" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "prettyplease", "proc-macro2", @@ -228,7 +237,7 @@ dependencies = [ [[package]] name = "cgp-inner" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-component", "cgp-component-macro", @@ -252,7 +261,7 @@ dependencies = [ [[package]] name = "cgp-run" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-async", "cgp-component", @@ -263,7 +272,7 @@ dependencies = [ [[package]] name = "cgp-runtime" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-core", ] @@ -280,7 +289,7 @@ dependencies = [ [[package]] name = "cgp-type" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#3dc5190a4d687c0a2792e5b3a1426d40c0254118" +source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" dependencies = [ "cgp-component", "cgp-component-macro", diff --git a/Cargo.toml b/Cargo.toml index b724e79..8358d79 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -29,6 +29,7 @@ cgp-field = { git = "https://github.com/contextgeneric/cgp.git cgp-field-macro = { git = "https://github.com/contextgeneric/cgp.git" } cgp-field-macro-lib = { git = "https://github.com/contextgeneric/cgp.git" } cgp-error = { git = "https://github.com/contextgeneric/cgp.git" } +cgp-error-extra = { git = "https://github.com/contextgeneric/cgp.git" } cgp-error-anyhow = { git = "https://github.com/contextgeneric/cgp.git" } cgp-run = { git = "https://github.com/contextgeneric/cgp.git" } cgp-runtime = { git = "https://github.com/contextgeneric/cgp.git" } diff --git a/content/field-accessors.md b/content/field-accessors.md index b61a7f7..432813a 100644 --- a/content/field-accessors.md +++ b/content/field-accessors.md @@ -552,7 +552,7 @@ as follows: # use core::fmt::Display; # # use cgp::core::component::UseDelegate; -# use cgp::core::error::impls::RaiseFrom; +# use cgp::extra::error::RaiseFrom; # use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; # use cgp::prelude::*; # use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; @@ -980,7 +980,7 @@ Using `UseField`, we can implement the providers as follows: # use core::marker::PhantomData; # # use cgp::prelude::*; -# use cgp::core::field::impls::use_field::UseField; +# use cgp::core::field::UseField; # # #[cgp_component { # provider: ApiBaseUrlGetter, @@ -1044,9 +1044,9 @@ wire up the accessor components directly inside `delegate_components!`: # use core::marker::PhantomData; # # use cgp::core::component::UseDelegate; -# use cgp::core::error::impls::RaiseFrom; +# use cgp::extra::error::RaiseFrom; # use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; -# use cgp::core::field::impls::use_field::UseField; +# use cgp::core::field::UseField; # use cgp::prelude::*; # use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; # use reqwest::blocking::Client; @@ -1428,9 +1428,9 @@ Using `UseProductionApiUrl`, we can now define a production `ApiClient` context # use std::sync::OnceLock; # # use cgp::core::component::UseDelegate; -# use cgp::core::error::impls::RaiseFrom; +# use cgp::extra::error::RaiseFrom; # use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; -# use cgp::core::field::impls::use_field::UseField; +# use cgp::core::field::UseField; # use cgp::prelude::*; # use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; # use reqwest::blocking::Client; From 85097a7f34aca2a6e2742cc5b4d2b899f6453682 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 16:14:16 +0000 Subject: [PATCH 02/11] Test using cgp_type! inside chapter --- Cargo.lock | 36 ++++++++++++++++++------------------ content/associated-types.md | 24 ++++-------------------- content/error-raisers.md | 1 - 3 files changed, 22 insertions(+), 39 deletions(-) delete mode 100644 content/error-raisers.md diff --git a/Cargo.lock b/Cargo.lock index 426cdcf..c656fa3 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -101,7 +101,7 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" [[package]] name = "cgp" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-async", "cgp-core", @@ -111,7 +111,7 @@ dependencies = [ [[package]] name = "cgp-async" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-async-macro", "cgp-sync", @@ -120,7 +120,7 @@ dependencies = [ [[package]] name = "cgp-async-macro" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "proc-macro2", "quote", @@ -130,12 +130,12 @@ dependencies = [ [[package]] name = "cgp-component" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" [[package]] name = "cgp-component-macro" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-component-macro-lib", ] @@ -143,7 +143,7 @@ dependencies = [ [[package]] name = "cgp-component-macro-lib" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "itertools", "prettyplease", @@ -155,7 +155,7 @@ dependencies = [ [[package]] name = "cgp-core" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-async", "cgp-component", @@ -169,7 +169,7 @@ dependencies = [ [[package]] name = "cgp-error" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-async", "cgp-component", @@ -180,7 +180,7 @@ dependencies = [ [[package]] name = "cgp-error-anyhow" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "anyhow", "cgp-core", @@ -189,7 +189,7 @@ dependencies = [ [[package]] name = "cgp-error-extra" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-error", ] @@ -197,7 +197,7 @@ dependencies = [ [[package]] name = "cgp-extra" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-error-extra", "cgp-inner", @@ -208,7 +208,7 @@ dependencies = [ [[package]] name = "cgp-field" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-component", "cgp-type", @@ -217,7 +217,7 @@ dependencies = [ [[package]] name = "cgp-field-macro" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-field-macro-lib", "proc-macro2", @@ -226,7 +226,7 @@ dependencies = [ [[package]] name = "cgp-field-macro-lib" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "prettyplease", "proc-macro2", @@ -237,7 +237,7 @@ dependencies = [ [[package]] name = "cgp-inner" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-component", "cgp-component-macro", @@ -261,7 +261,7 @@ dependencies = [ [[package]] name = "cgp-run" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-async", "cgp-component", @@ -272,7 +272,7 @@ dependencies = [ [[package]] name = "cgp-runtime" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-core", ] @@ -289,7 +289,7 @@ dependencies = [ [[package]] name = "cgp-type" version = "0.2.0" -source = "git+https://github.com/contextgeneric/cgp.git#9d8cf040ac9373284d7f19c1cecc9ccbf937edde" +source = "git+https://github.com/contextgeneric/cgp.git#8fe487c4c17f6857d4a30c451516e8a3876e838f" dependencies = [ "cgp-component", "cgp-component-macro", diff --git a/content/associated-types.md b/content/associated-types.md index c343e4e..762d6da 100644 --- a/content/associated-types.md +++ b/content/associated-types.md @@ -697,21 +697,8 @@ pub mod traits { use anyhow::Error; use cgp::prelude::*; - #[cgp_component { - name: TimeTypeComponent, - provider: ProvideTimeType, - }] - pub trait HasTimeType { - type Time; - } - - #[cgp_component { - name: AuthTokenTypeComponent, - provider: ProvideAuthTokenType, - }] - pub trait HasAuthTokenType { - type AuthToken; - } + cgp_type!( Time ); + cgp_type!( AuthToken ); #[cgp_component { provider: AuthTokenValidator, @@ -740,6 +727,7 @@ pub mod traits { pub mod impls { use anyhow::{anyhow, Error}; + use cgp::prelude::*; use datetime::LocalDateTime; use super::traits::*; @@ -782,11 +770,7 @@ pub mod impls { } } - pub struct UseStringAuthToken; - - impl ProvideAuthTokenType for UseStringAuthToken { - type AuthToken = String; - } + pub type UseStringAuthToken = UseType; } pub mod contexts { diff --git a/content/error-raisers.md b/content/error-raisers.md deleted file mode 100644 index 310595c..0000000 --- a/content/error-raisers.md +++ /dev/null @@ -1 +0,0 @@ -# Error Raisers \ No newline at end of file From b0c1329c91b38b5c2d7b1ce7e00f2571f7068e57 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 18:24:25 +0000 Subject: [PATCH 03/11] Use AI to improve the section Trait Minimalism --- content/associated-types.md | 22 +++++++++------------- 1 file changed, 9 insertions(+), 13 deletions(-) diff --git a/content/associated-types.md b/content/associated-types.md index 762d6da..b6a3b2d 100644 --- a/content/associated-types.md +++ b/content/associated-types.md @@ -352,10 +352,9 @@ of associated types. ## Trait Minimalism -It may look overly verbose to define multiple type traits and require -the exact type trait to to be included as the supertrait of a method -interface. For example, one may be tempted to define just one trait -that contains methods and types, such as: +At first glance, it might seem overly verbose to define multiple type traits and require +each to be explicitly included as a supertrait of a method interface. For instance, +you might be tempted to consolidate the methods and types into a single trait, like this: ```rust # extern crate cgp; @@ -380,21 +379,18 @@ pub trait AppTrait { } ``` -However, doing so introduces unnecessary _coupling_ between unrelated types and methods. -For example, an application may want to implement the token validation by forwarding the -validation to an external _microservice_. In such case, it would be redundant to require -the application to choose a time type that it won't actually use. +While this approach might seem simpler, it introduces unnecessary _coupling_ between +potentially unrelated types and methods. For example, an application implementing +token validation might delegate this functionality to an external microservice. +In such a case, it is redundant to require the application to specify a Time type that +it doesn’t actually use. In practice, we find the practical benefits of defining many _minimal_ traits often outweight any theoretical advantages of combining multiple items into one trait. As we will demonstrate in later chapters, having traits that contain only one type or method would also enable more advanced CGP patterns to be applied to such traits. -Because of this, we encourage readers to follow our advice and be _encouraged_ -to use as many minimal traits without worrying about any theoretical overhead. -That said, this advice is _non-binding_, so readers are free to add as many items -as they prefer into a trait, and perhaps go through the hard way of learning why the -alternative is better. +We encourage readers to embrace minimal traits without concern for theoretical overhead. However, during the early phases of a project, you might prefer to consolidate items to reduce cognitive overload while learning or prototyping. As the project matures, you can always refactor and decompose larger traits into smaller, more focused ones, following the techniques outlined in this book. ## Impl-Side Associated Type Constraints From 1e61a61f2d23cf1611c48f0328eedee92f3ae8b7 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 18:53:23 +0000 Subject: [PATCH 04/11] Revise TypeProviders section --- content/associated-types.md | 122 +++++++++++++----------------------- 1 file changed, 45 insertions(+), 77 deletions(-) diff --git a/content/associated-types.md b/content/associated-types.md index b6a3b2d..e5eecd3 100644 --- a/content/associated-types.md +++ b/content/associated-types.md @@ -350,6 +350,33 @@ Compared to regular generic programming, instead of specifying all generic param by position, we are able to parameterize the abstract types using _names_, in the form of associated types. +## Defining Abstract Type Traits using `cgp_type!` + +The type traits `HasTimeType` and `HasAuthTokenType` follows similar boilerplate, +and it may quickly become tedious as we define more abstract types. To help with +defining such type traits, the `cgp` crate provides the `cgp_type!` macro that +allows us to have much shorter definition as follows: + + +```rust +# extern crate cgp; +# +use cgp::prelude::*; + +cgp_type!( Time: Eq + Ord ); +cgp_type!( AuthToken ); +``` + +The `cgp_type!` macro accepts the name of an abstract type `$name`, together with any +applicable constraint for that type. It then derives the same implementation as the +`cgp_component` macro, with a consumer trait named `Has{$name}Type`, a provider trait +named `Provide{$name}Type`, and a component name `${name}TypeComponent`. +Inside the traits, there is one associated type defined with `type $name: $constraints;`. + +In addition to the standard derivation from `cgp_component`, `cgp_type!` also +derives some additional implementations, which we will cover the usage in later chapters. + + ## Trait Minimalism At first glance, it might seem overly verbose to define multiple type traits and require @@ -388,7 +415,8 @@ it doesn’t actually use. In practice, we find the practical benefits of defining many _minimal_ traits often outweight any theoretical advantages of combining multiple items into one trait. As we will demonstrate in later chapters, having traits that contain only one type -or method would also enable more advanced CGP patterns to be applied to such traits. +or method would also enable more advanced CGP patterns to be applied, including +the use of `cgp_type!` that we have just covered. We encourage readers to embrace minimal traits without concern for theoretical overhead. However, during the early phases of a project, you might prefer to consolidate items to reduce cognitive overload while learning or prototyping. As the project matures, you can always refactor and decompose larger traits into smaller, more focused ones, following the techniques outlined in this book. @@ -403,13 +431,7 @@ Looking back at the definition of `HasTimeType`: # # use cgp::prelude::*; # -#[cgp_component { - name: TimeTypeComponent, - provider: ProvideTimeType, -}] -pub trait HasTimeType { - type Time: Eq + Ord; -} +cgp_type!( Time: Eq + Ord ); ``` The associated `Time` type has the constraint `Eq + Ord` specified. With this, the constraints @@ -432,21 +454,9 @@ them on the associated type constraints: # use anyhow::{anyhow, Error}; # use cgp::prelude::*; # -#[cgp_component { - name: TimeTypeComponent, - provider: ProvideTimeType, -}] -pub trait HasTimeType { - type Time; -} +cgp_type!( Time ); -# #[cgp_component { -# name: AuthTokenTypeComponent, -# provider: ProvideAuthTokenType, -# }] -# pub trait HasAuthTokenType { -# type AuthToken; -# } +# cgp_type!( AuthToken ); # # #[cgp_component { # provider: AuthTokenValidator, @@ -518,10 +528,7 @@ types such as `Debug` and `Eq`. ## Type Providers -With the type abstraction in place, we can define different context-generic -providers for the `Time` and `AuthToken` abstract types. -For instance, we can define a provider that provides `std::time::Instant` -as the `Time` type: +With type abstraction in place, we can define context-generic providers for the `Time` and `AuthToken` abstract types. For example, we can create a provider that uses `std::time::Instant` as the `Time` type: ```rust # extern crate cgp; @@ -563,26 +570,11 @@ where } ``` -Our context-generic provider `UseInstant` can be used to implement -`ProvideTimeType` for any `Context` type, by setting the associated -type `Time` to be `Instant`. -Additionally, `UseInstant` also implements `CurrentTimeGetter` -for any `Context` type, _provided_ that `Context::Time` is the -same as `Instant`. -The type equality constraint works similar to how regular impl-side -dependencies work, and would be used frequently for scope-limited -access to the underlying concrete type for an abstract type. - -Note that this type equality constraint is required in this case, -because a context may _not_ necessary choose `UseInstant` as -the provider for `ProvideTimeType`. As a result, there is an -additional constraint that `UseInstant` can only implement -`CurrentTimeGetter`, if `Context` uses it or a different -provider that also uses `Instant` to implement `Time`. - -Aside from `Instant`, we can also implement separate time providers that -make use of a different time type, such as -[`datetime::LocalDateTime`](https://docs.rs/datetime/latest/datetime/struct.LocalDateTime.html): +Here, the `UseInstant` provider implements `ProvideTimeType` for any `Context` type by setting the associated type `Time` to `Instant`. Additionally, it implements `CurrentTimeGetter` for any `Context`, _provided_ that `Context::Time` is `Instant`. This type equality constraint works similarly to regular implementation-side dependencies and is frequently used for scope-limited access to a concrete type associated with an abstract type. + +The type equality constraint is necessary because a given context might not always use `UseInstant` as the provider for `ProvideTimeType`. Instead, the context could choose a different provider that uses another type to represent `Time`. Consequently, `UseInstant` can only implement `CurrentTimeGetter` if the `Context` uses it or another provider that also uses `Instant` as its `Time` type. + +Aside from `Instant`, we can also define alternative providers for `Time`, using other types like [`datetime::LocalDateTime`](https://docs.rs/datetime/latest/datetime/struct.LocalDateTime.html): ```rust # extern crate cgp; @@ -624,18 +616,9 @@ where } ``` -Since our application only require `Time` to implement `Ord`, -and the ability to get the current time, we can easily swap between different -time providers if they satisfy all the dependencies we need. -As the application grows, there may be additional constraints imposed on -the time type, which may restrict the available choice of concrete time types. -But with CGP, we can incrementally introduce new dependencies according -the needs of the application, so that we do not prematurely restrict -our choices based on dependencies that are not used by the application. +Since our application only requires the `Time` type to implement `Ord` and the ability to retrieve the current time, we can easily swap between different time providers, as long as they meet these dependencies. As the application evolves, additional constraints might be introduced on the Time type, potentially limiting the available concrete time types. However, with CGP, we can incrementally introduce new dependencies based on the application’s needs, avoiding premature restrictions caused by unused requirements. -Similar to the abstract `Time` type, we can also define a context-generic -provider for `ProvideAuthTokenType`, which implements `AuthToken` using -`String`: +Similarly, for the abstract `AuthToken` type, we can define a context-generic provider `ProvideAuthTokenType` that uses `String` as its implementation: ```rust # extern crate cgp; @@ -657,26 +640,11 @@ impl ProvideAuthTokenType for UseStringAuthToken { } ``` -Notice that compared to the newtype pattern, we can opt to use plain old `String` -_without_ wrapping it around a newtype struct. Contradicting to common wisdom, -with CGP we do not put as much emphasis of requiring newtype wrapping every -abstract type used by the application. This is particularly the case if the -majority of the application is written as context-generic code. The reason -for this is because the abstract types and their accompanying interfaces -already serve the same purpose as newtypes, and so there are less needs -to "protect" the raw values by wrapping them inside newtypes. - -That being said, readers are free to define newtypes and use them together with -abstract types. This would be helpful at least for beginners, as there are -different approaches that we will discuss in later chapters on how to properly -restrict the access of underlying concrete types inside context-generic code. -Newtypes would also still be useful, if the values are also accessed by -non-trival non-context-generic code, which would have unrestricted access to -the concrete type. - -In this book, we will continue using the pattern of implementing abstract types -using plain types without additional newtype wrapping. We will revisit the topic -of comparing newtypes and abstract types in later chapters. +Compared to the newtype pattern, we can use plain `String` values directly, without wrapping them in a newtype struct. Contrary to common wisdom, in CGP, we place less emphasis on wrapping every domain type in a newtype. This is particularly true when most of the application is written in a context-generic style. The rationale is that abstract types and their accompanying interfaces already fulfill the role of newtypes by encapsulating and "protecting" raw values, reducing the need for additional wrapping. + +That said, readers are free to define newtypes and use them alongside abstract types. For beginners, this can be especially useful, as later chapters will explore methods to properly restrict access to underlying concrete types in context-generic code. Additionally, newtypes remain valuable when the raw values are also used in non-context-generic code, where access to the concrete types is unrestricted. + +Throughout this book, we will primarily use plain types to implement abstract types, without additional newtype wrapping. However, we will revisit the comparison between newtypes and abstract types in later chapters, providing further guidance on when each approach is most appropriate. ## Putting It Altogether From 8279adbee19e62734bb4e33c117eb4305b4c9614 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 19:13:01 +0000 Subject: [PATCH 05/11] Add section for UseType pattern --- content/associated-types.md | 57 +++++++++++++++++++++++++++++++++++-- src/lib.rs | 1 - 2 files changed, 54 insertions(+), 4 deletions(-) diff --git a/content/associated-types.md b/content/associated-types.md index e5eecd3..db71c81 100644 --- a/content/associated-types.md +++ b/content/associated-types.md @@ -646,6 +646,59 @@ That said, readers are free to define newtypes and use them alongside abstract t Throughout this book, we will primarily use plain types to implement abstract types, without additional newtype wrapping. However, we will revisit the comparison between newtypes and abstract types in later chapters, providing further guidance on when each approach is most appropriate. +## The `UseType` Pattern + +The way we implement type providers can quickly become tedious, as the number of abstract types that we need to implement grows. For instance, just to use `String` as `AuthToken`, we need to first define a new struct `UseStringAuthToken`, and then implement `ProvideAuthTokenType` on it. To simplify the implementation of abstract types, the `cgp_type!` macro also generates a provider implementation that uses the `UseType` pattern. The generated implementation is as follows: + +```rust +# extern crate cgp; +# +# use core::marker::PhantomData; +# +# use cgp::prelude::*; +# +# #[cgp_component { +# name: AuthTokenTypeComponent, +# provider: ProvideAuthTokenType, +# }] +# pub trait HasAuthTokenType { +# type AuthToken; +# } +# +pub struct UseType(pub PhantomData); + +impl ProvideAuthTokenType for UseType { + type AuthToken = AuthToken; +} +``` + +The `UseType` struct is a _marker_ type that is used for implementing the `UseType` +pattern. It has a generic `Type` parameter, which is used to mark the type that +we want to use with a given type trait. With `PhantomData` in its body, the `UseType` +type is not intended to be used as a value anywhere in any code. + +We then have a generic implementation of `ProvideAuthTokenType` for `UseType`, +by implementing the `AuthToken` type with the type that is provided to `UseType`. + +With the blanket implementation, we can redefine `UseStringAuthToken` simply +as an alias to `UseType`: + +```rust +# use core::marker::PhantomData; +# +# pub struct UseType(pub PhantomData); +# +type UseStringAuthToken = UseType; +``` + +In fact, we can also skip defining a type alias, and use `UseType` directly inside +`delegate_components` when wiring the type providers. + +The `UseType` struct is provided by the `cgp` crate. When an abstract type is defined +using `cgp_type!`, the generic implementation for `UseType` is also automatically +implemented. Hence, can make use of `UseType` directly to simplify our component wiring. + + ## Putting It Altogether With all pieces in place, we can put together everything we learn, and refactor @@ -733,8 +786,6 @@ pub mod impls { Ok(LocalDateTime::now()) } } - - pub type UseStringAuthToken = UseType; } pub mod contexts { @@ -763,7 +814,7 @@ pub mod contexts { TimeTypeComponent, CurrentTimeGetterComponent, ]: UseLocalDateTime, - AuthTokenTypeComponent: UseStringAuthToken, + AuthTokenTypeComponent: UseType, AuthTokenValidatorComponent: ValidateTokenIsNotExpired, } } diff --git a/src/lib.rs b/src/lib.rs index 8b13789..e69de29 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1 +0,0 @@ - From efe121e5c0b8c09a8901be725429aa5ce97ca807 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 20:17:00 +0000 Subject: [PATCH 06/11] AI-revise Associated Types chapter --- content/associated-types.md | 185 ++++++++---------------------------- 1 file changed, 40 insertions(+), 145 deletions(-) diff --git a/content/associated-types.md b/content/associated-types.md index db71c81..f0a068e 100644 --- a/content/associated-types.md +++ b/content/associated-types.md @@ -1,19 +1,10 @@ # Associated Types -In the first part of this book, we have learned about how CGP makes use of -Rust's trait system to wire up components using blanket implementations. -Since CGP works within Rust's trait system, we can make use of advanced -Rust features together with CGP to form new design patterns. -In this chapter, we will learn about how to make use of _associated types_ -with CGP to define context-generic providers that are generic over multiple -_abstract_ types. +In the first part of this book, we explored how CGP leverages Rust's trait system to wire up components using blanket implementations. Because CGP operates within Rust's trait system, it allows us to incorporate advanced Rust features to create new design patterns. In this chapter, we will focus on using _associated types_ with CGP to define context-generic providers that are generic over multiple _abstract_ types. ## Building Authentication Components -Supposed that we want to build a simple authentication system using _bearer tokens_ -with expiry time. To build such system, we would need to fetch the expiry time of -a valid token, and ensure that the time is not in the past. A naive attempt of -implementing the authentication would be as follows: +Suppose we want to build a simple authentication system using _bearer tokens_ with an expiration time. To achieve this, we need to fetch the expiration time of a valid token and ensure that it is not in the past. A naive approach to implementing the authentication might look like the following: ```rust # extern crate cgp; @@ -133,36 +124,17 @@ pub mod contexts { # } ``` -We first define `CanValidateAuthToken`, which would be used as the main API for validating an -auth token. In order to help implementing the validator, we also define -`CanFetchAuthTokenExpiry` used for fetching the expiry time of an auth token, if the token is valid. -Finally, we also define `HasCurrentTime` which is used for fetching the current time. +In this example, we first define the `CanValidateAuthToken` trait, which serves as the primary API for validating authentication tokens. To facilitate the implementation of the validator, we also define the `CanFetchAuthTokenExpiry` trait, which is responsible for fetching the expiration time of an authentication token — assuming the token is valid. Finally, the `HasCurrentTime` trait is introduced to retrieve the current time. -We then define a context-generic provider `ValidateTokenIsNotExpired`, which validates auth tokens -by fetching the token's expiry time and the current time, and ensure that the token's expiry time -does not exceed the current time. We also define a context-generic provider `GetSystemTimestamp`, -which gets the current time using `std::time::System::now()`. +Next, we define a context-generic provider, `ValidateTokenIsNotExpired`, which validates authentication tokens by comparing their expiration time with the current time. The provider fetches both the token’s expiration time and the current time, and ensure that the token is still valid. Additionally, we define another context-generic provider, `GetSystemTimestamp`, which retrieves the current time using `std::time::SystemTime::now()`. -For the purpose of this demo, we also define a concrete context `MockApp`, which contains a -`auth_tokens_store` field with mocked collection of auth tokens with respective expiry time -stored in a `BTreeMap`. -We then implement a context-specific provider of `AuthTokenExpiryFetcher` for `MockApp`, -which reads from the mocked `auth_tokens_store`. -We also define a check trait `CanUseMockApp`, to check that `MockApp` correctly implements -`CanValidateAuthToken` with the wiring provided. +For this demonstration, we introduce a concrete context, `MockApp`, which includes an `auth_tokens_store` field. This store is a mocked collection of authentication tokens with their respective expiration times, stored in a `BTreeMap`. We also implement the `AuthTokenExpiryFetcher` trait specifically for the `MockApp` context, which retrieves expiration times from the mocked `auth_tokens_store`. Lastly, we define the `CanUseMockApp` trait, ensuring that `MockApp` properly implements the `CanValidateAuthToken` trait through the provided wiring. ## Abstract Types -The naive example above makes use of basic CGP techniques to implement a reusable provider -`ValidateTokenIsNotExpired`, which can be used with different concrete contexts. -However, we can see that the method signatures are tied to specific types. -In particular, we used `String` to represent the auth token, and `u64` to -represent the unix timestamp in milliseconds. +The previous example demonstrates basic CGP techniques for implementing a reusable provider, `ValidateTokenIsNotExpired`, which can work with different concrete contexts. However, the method signatures are tied to specific types. For instance, we use `String` to represent the authentication token and `u64` to represent the Unix timestamp in milliseconds. -Common wisdom tells us that we should use distinct types to distinguish values -from specific domains, so that we do not accidentally mix up values from different -domains. A common approach in Rust is to make use of the _newtype pattern_ to -define wrapper types such as follows: +Common practice suggests that we should use distinct types to differentiate values from different domains, reducing the chance of mixing them up. A common approach in Rust is to use the _newtype pattern_ to define wrapper types, like so: ```rust pub struct AuthToken { @@ -174,16 +146,9 @@ pub struct Time { } ``` -Although the newtype pattern abstracts over the underlying value, it does not allow -our code to be generalized over distinct types. For example, instead of defining -our own `Time` type with Unix timestamp semantics, we may want to use a datetime -library such as `chrono` or `datetime`. However, the exact choice of a datetime -library may depend on the specific use case of a concrete application. +While the newtype pattern helps abstract over underlying values, it doesn't fully generalize the code to work with different types. For example, instead of defining our own `Time` type with Unix timestamp semantics, we may want to use a datetime library such as `datetime` or `chrono`. The choice of library could depend on the specific use case of a concrete application. -A better approach would be to define an _abstract_ time type, so that we can -implement context-generic providers that can work with _any_ time type that -the concrete context chooses. We can do this in CGP by defining _type traits_ -that contain _associated types_: +A more flexible approach is to define an _abstract_ `Time` type that allows us to implement context-generic providers compatible with _any_ `Time` type chosen by the concrete context. This can be achieved in CGP by defining _type traits_ that contain _associated types_: ```rust # extern crate cgp; @@ -207,19 +172,11 @@ pub trait HasAuthTokenType { } ``` -We first introduce a `HasTimeType` trait, which contains only an associated type -`Time`. We also have additional constraints that the abstract `Time` type must -implement `Eq` and `Ord`, so that we compare between two time values. -Similarly, we also introduce a `HasAuthTokenType` trait, which contains an `AuthToken` -associated type, but without any extra constraint. +Here, we define the `HasTimeType` trait with an associated type `Time`, which is constrained to types that implement `Eq` and `Ord` so that they can be compared. Similarly, the `HasAuthTokenType` trait defines an associated type `AuthToken`, without any additional constraints. -Similar to trait methods, we can use CGP to auto derive blanket implementations -that delegate the implementation associated types to providers using `HasComponents` -and `DelegateComponent`. As such, we can use `#[cgp_component]` also on traits that -contain associated types. +Similar to regular trait methods, CGP allows us to auto-derive blanket implementations that delegate the associated types to providers using `HasComponents` and `DelegateComponent`. Therefore, we can use `#[cgp_component]` on traits containing associated types as well. -With the type traits defined, we can update our authentication components to make -use of the abstract types inside the trait methods: +With these type traits in place, we can now update our authentication components to leverage abstract types within the trait methods: ```rust # extern crate cgp; @@ -268,14 +225,9 @@ pub trait HasCurrentTime: HasTimeType { } ``` -The trait `CanValidateAuthToken` is updated to include `HasAuthTokenType` -as a supertrait, so that it can accept the abstract type `Self::AuthToken` -inside the method parameter `validate_auth_token`. -Similarly, `CanFetchAuthTokenExpiry` requires both `HasAuthTokenType` -and `HasTimeType`, while `HasCurrentTime` only requires `HasTimeType`. +Here, we modify the `CanValidateAuthToken` trait to include `HasAuthTokenType` as a supertrait, allowing it to accept the abstract type `Self::AuthToken` as a method parameter. Likewise, `CanFetchAuthTokenExpiry` requires both `HasAuthTokenType` and `HasTimeType`, while `HasCurrentTime` only requires `HasTimeType`. -With the abstract types in place, we can now redefine `ValidateTokenIsNotExpired` -to be implemented generically over _any_ abstract `Time` and `AuthToken` types. +With the abstract types defined, we can now update `ValidateTokenIsNotExpired` to work generically with any `Time` and `AuthToken` types: ```rust # extern crate cgp; @@ -344,19 +296,13 @@ where } ``` -Through this example, we can see that CGP allows us to define context-generic providers -that are not only generic over the main context, but also all its associated types. -Compared to regular generic programming, instead of specifying all generic parameters -by position, we are able to parameterize the abstract types using _names_, in the form -of associated types. +This example shows how CGP enables us to define context-generic providers that are not just generic over the context itself, but also over its associated types. Unlike traditional generic programming, where all generic parameters are specified positionally, CGP allows us to parameterize abstract types using _names_ via associated types. -## Defining Abstract Type Traits using `cgp_type!` +## Defining Abstract Type Traits with `cgp_type!` -The type traits `HasTimeType` and `HasAuthTokenType` follows similar boilerplate, -and it may quickly become tedious as we define more abstract types. To help with -defining such type traits, the `cgp` crate provides the `cgp_type!` macro that -allows us to have much shorter definition as follows: +The type traits `HasTimeType` and `HasAuthTokenType` share a similar structure, and as you define more abstract types, this boilerplate can become tedious. To streamline the process, the `cgp` crate provides the `cgp_type!` macro, which simplifies type trait definitions. +Here's how you can define the same types with `cgp_type!`: ```rust # extern crate cgp; @@ -367,21 +313,12 @@ cgp_type!( Time: Eq + Ord ); cgp_type!( AuthToken ); ``` -The `cgp_type!` macro accepts the name of an abstract type `$name`, together with any -applicable constraint for that type. It then derives the same implementation as the -`cgp_component` macro, with a consumer trait named `Has{$name}Type`, a provider trait -named `Provide{$name}Type`, and a component name `${name}TypeComponent`. -Inside the traits, there is one associated type defined with `type $name: $constraints;`. - -In addition to the standard derivation from `cgp_component`, `cgp_type!` also -derives some additional implementations, which we will cover the usage in later chapters. - +The `cgp_type!` macro accepts the name of an abstract type, `$name`, along with any applicable constraints for that type. It then automatically generates the same implementation as the `cgp_component` macro: a consumer trait named `Has{$name}Type`, a provider trait named `Provide{$name}Type`, and a component name type named `${name}TypeComponent`. Each of the generated traits includes an associated type defined as `type $name: $constraints;`. +In addition, `cgp_type!` also derives some other implementations, which we'll explore in later chapters. ## Trait Minimalism -At first glance, it might seem overly verbose to define multiple type traits and require -each to be explicitly included as a supertrait of a method interface. For instance, -you might be tempted to consolidate the methods and types into a single trait, like this: +At first glance, it might seem overly verbose to define multiple type traits and require each to be explicitly included as a supertrait of a method interface. For instance, you might be tempted to consolidate the methods and types into a single trait, like this: ```rust # extern crate cgp; @@ -422,9 +359,7 @@ We encourage readers to embrace minimal traits without concern for theoretical o ## Impl-Side Associated Type Constraints -The minimalism philosophy for CGP also extends to the constraints specified -on the associated type inside a type trait. -Looking back at the definition of `HasTimeType`: +The minimalism philosophy of CGP extends to the constraints placed on associated types within type traits. Consider the earlier definition of `HasTimeType`: ```rust # extern crate cgp; @@ -434,18 +369,11 @@ Looking back at the definition of `HasTimeType`: cgp_type!( Time: Eq + Ord ); ``` -The associated `Time` type has the constraint `Eq + Ord` specified. With this, the constraints -are imposed on _all_ concrete time types, regardless of whether they are actually used by -the providers. In fact, if we revisit our previous code, we could notice that the `Eq` -constraint is not reallying being used anywhere. +Here, the associated `Time` type is constrained by `Eq + Ord`. This means that all concrete implementations of `Time` must satisfy these constraints, regardless of whether they are actually required by the providers. In fact, if we revisit our previous examples, we notice that the `Eq` constraint isn’t used anywhere. -For this reason, the constraints specified on the associated type often become a bottleneck -that significantly restricts how the application can evolve. For example, as the application -grows more complex, it is not uncommon to now require `Time` to implement many additional traits, -such as `Debug + Display + Clone + Hash + Serialize + Deserialize` and so on. +Such overly restrictive constraints can become a bottleneck as the application evolves. As complexity increases, it’s common to require additional traits on `Time`, such as `Debug + Display + Clone + Hash + Serialize + Deserialize` and so on. Imposing these constraints globally limits flexibility and makes it harder to adapt to changing requirements. -Fortunately with CGP, we can reuse the same techniques as impl-side dependencies, and apply -them on the associated type constraints: +Fortunately, CGP allows us to apply the same principle of impl-side dependencies to associated type constraints. Consider the following example: ```rust # extern crate cgp; @@ -503,28 +431,13 @@ where } ``` -In the above example, we redefine `HasTimeType::Time` to _not_ have any constraint. -Then in the provider implementation of `ValidateTokenIsNotExpired`, we add an -additional constraint that requires `Context::Time: Ord`. This way, -`ValidateTokenIsNotExpired` is able to compare the token expiry time, even -when `Ord` is not specified on `HasTimeType::Time`. - -With this approach, we can _conditionally_ require `HasTimeType::Time` to implement -`Ord`, only when `ValidateTokenIsNotExpired` is used as the provider. -This essentially allows the abstract types to scale in the same way as the generic -context types, and allows us to make use of the same CGP patterns also on abstract types. - -That said, in some cases it is still convenient to directly include constraints -such as `Debug` on an associated type, especially if the constraint is used in -almost all providers. With the current state of error reporting, including -all constraints on the associated type also tend to provide better error messages, -when there is any unsatisfied constraint. - -As a guideline, we encourage readers to first try to define type traits without -including any constraint on the associated type, and try to include the constraints -on the impl-side as often as possible. However readers are free to include default -constraints to associated types as they see fit, at least for relatively trivial -types such as `Debug` and `Eq`. +In this example, we redefine `HasTimeType::Time` _without_ any constraints. Instead, we specify the constraint `Context::Time: Ord` in the provider implementation for `ValidateTokenIsNotExpired`. This ensures that the `ValidateTokenIsNotExpired` provider can compare the token expiry time using `Ord`, while avoiding unnecessary global constraints on `Time`. + +By applying constraints on the implementation side, we can conditionally require `HasTimeType::Time` to implement `Ord`, but only when the `ValidateTokenIsNotExpired` provider is in use. This approach allows abstract types to scale flexibly alongside generic context types, enabling the same CGP patterns to be applied to abstract types. + +In some cases, it can still be convenient to include constraints (e.g., `Debug`) directly on an associated type, especially if those constraints are nearly universal across providers. Additionally, current Rust error reporting often produces clearer error messages when constraints are defined at the associated type level, as opposed to being deferred to the implementation. + +As a guideline, we recommend that readers begin by defining type traits without placing constraints on associated types, relying instead on implementation-side constraints wherever possible. However, readers may choose to apply global constraints to associated types when appropriate, particularly for simple and widely applicable traits like `Debug` and `Eq`. ## Type Providers @@ -648,7 +561,7 @@ Throughout this book, we will primarily use plain types to implement abstract ty ## The `UseType` Pattern -The way we implement type providers can quickly become tedious, as the number of abstract types that we need to implement grows. For instance, just to use `String` as `AuthToken`, we need to first define a new struct `UseStringAuthToken`, and then implement `ProvideAuthTokenType` on it. To simplify the implementation of abstract types, the `cgp_type!` macro also generates a provider implementation that uses the `UseType` pattern. The generated implementation is as follows: +Implementing type providers can quickly become repetitive as the number of abstract types grows. For example, to use `String` as the `AuthToken` type, we first need to define a new struct, `UseStringAuthToken`, and then implement `ProvideAuthTokenType` for it. To streamline this process, the `cgp_type!` macro simplifies the implementation by automatically generating a provider using the _`UseType`_ pattern. The generated implementation looks like this: ```rust # extern crate cgp; @@ -672,16 +585,9 @@ impl ProvideAuthTokenType for UseType { } ``` -The `UseType` struct is a _marker_ type that is used for implementing the `UseType` -pattern. It has a generic `Type` parameter, which is used to mark the type that -we want to use with a given type trait. With `PhantomData` in its body, the `UseType` -type is not intended to be used as a value anywhere in any code. - -We then have a generic implementation of `ProvideAuthTokenType` for `UseType`, -by implementing the `AuthToken` type with the type that is provided to `UseType`. +Here, `UseType` is a _marker_ type with a generic parameter `Type`, representing the type to be used for a given type trait. Since `PhantomData` is its only field, `UseType` is never intended to be used as a runtime value. The generic implementation of `ProvideAuthTokenType` for `UseType` ensures that the AuthToken type is directly set to the `Type` parameter of `UseType`. -With the blanket implementation, we can redefine `UseStringAuthToken` simply -as an alias to `UseType`: +With this generic implementation, we can redefine `UseStringAuthToken` as a simple type alias for `UseType`: ```rust # use core::marker::PhantomData; @@ -691,18 +597,13 @@ as an alias to `UseType`: type UseStringAuthToken = UseType; ``` -In fact, we can also skip defining a type alias, and use `UseType` directly inside -`delegate_components` when wiring the type providers. - -The `UseType` struct is provided by the `cgp` crate. When an abstract type is defined -using `cgp_type!`, the generic implementation for `UseType` is also automatically -implemented. Hence, can make use of `UseType` directly to simplify our component wiring. +In fact, we can even skip defining type aliases altogether and use `UseType` directly in the `delegate_components` macro when wiring type providers. +The `UseType` struct is included in the `cgp` crate, and when you define an abstract type using the `cgp_type!` macro, the corresponding generic `UseType` implementation is automatically derived. This makes `UseType` a powerful tool for simplifying component wiring and reducing boilerplate in your code. ## Putting It Altogether -With all pieces in place, we can put together everything we learn, and refactor -our naive authentication components to make use of abstract types as follows: +With all the pieces in place, we can now apply what we've learned and refactor our naive authentication components to utilize abstract types, as shown below: ```rust # extern crate cgp; @@ -840,12 +741,6 @@ pub mod contexts { # } ``` -Compared to before, it is now much easier for us to update the `MockApp` context to -use different time and auth token providers. In case if we need to use different -concrete types for different use cases, we can also easily define additional -context types with different wirings, without having to duplicate the core logic. +Compared to our earlier approach, it is now much easier to update the `MockApp` context to use different time and auth token providers. If different use cases require distinct concrete types, we can easily define additional context types with different configurations, all without duplicating the core logic. -At this point, we have make use of abstract types on the time and auth token types, -but we are still using a concrete `anyhow::Error` type. In the next chapter, we -will look into the topic of error handling, and learn how to make use of -abstract error types to better handle application errors. \ No newline at end of file +So far, we have applied abstract types to the `Time` and `AuthToken` types, but we are still relying on the concrete `anyhow::Error` type. In the next chapter, we will explore error handling in depth and learn how to use abstract error types to improve the way application errors are managed. From 2c87e3a563705e04b46bf7b5f54a7d190856beb5 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 20:31:15 +0000 Subject: [PATCH 07/11] Add suggestion for aggregated common field struct --- content/field-accessors.md | 26 +++++++++++++++++++++++++- 1 file changed, 25 insertions(+), 1 deletion(-) diff --git a/content/field-accessors.md b/content/field-accessors.md index 432813a..cede044 100644 --- a/content/field-accessors.md +++ b/content/field-accessors.md @@ -480,7 +480,31 @@ programming practices for decades. As a result, readers are encourage to feel free to experiment around, and include as many types and methods in a CGP trait as they prefer. -On the other hand, for the purpose of this book, we will continue to make use +As an alternative to defining multiple accessor methods, you may also consider defining an inner struct that contains all the common fields that you might want to use with most of your providers: + +```rust +# extern crate cgp; +# +# use cgp::prelude::*; +# +pub struct ApiClientFields { + pub api_base_url: String, + pub auth_token: String, +} + +#[cgp_component { + provider: ApiClientFieldsGetter, +}] +pub trait HasApiClientFields { + fn api_client_fields(&self) -> &ApiClientFields; +} +``` + +In the example above, we define an `ApiClientFields` struct that contains both `api_base_url` and `auth_token` fields. With that, we can redefine the `HasApiClientFields` trait to have only one getter method which returns `ApiClientFields`. + +Note that a downside of this approach is that we can no longer make use of any abstract type inside the struct. As shown, the `ApiClientFields` field stores the `auth_token` as a concrete `String`, rather than an abstract `AuthToken` type. Because of this, this approach may only work if your providers make no use of fields made of abstract types. + +For the purpose of this book, we will continue to make use of minimal traits, since the book serves as reference materials that should encourage best practices to its readers. From 93581db41972e899a93a8c195f66b119ce389cb0 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 20:34:43 +0000 Subject: [PATCH 08/11] Split out sub-chapter for context-generic accessor providers --- content/SUMMARY.md | 1 + content/field-accessors.md | 958 -------------------------- content/generic-accessor-providers.md | 957 +++++++++++++++++++++++++ 3 files changed, 958 insertions(+), 958 deletions(-) create mode 100644 content/generic-accessor-providers.md diff --git a/content/SUMMARY.md b/content/SUMMARY.md index 4f58faa..9d3108f 100644 --- a/content/SUMMARY.md +++ b/content/SUMMARY.md @@ -26,6 +26,7 @@ - [Error Reporting](error-reporting.md) - [Error Wrapping](error-wrapping.md) - [Field Accessors](field-accessors.md) + - [Generic Accessor Providers](generic-accessor-providers.md) - [Component Presets]() - [Trait-Generic Providers]() - [`WithProvider`]() diff --git a/content/field-accessors.md b/content/field-accessors.md index cede044..f5c507b 100644 --- a/content/field-accessors.md +++ b/content/field-accessors.md @@ -744,961 +744,3 @@ The `ApiClient` context is defined with the fields that we need to implement the We then have context-specific implementation of `ApiBaseUrlGetter` and `AuthTokenGetter` to work directly with `ApiClient`. With that, our wiring is completed, and we can check that `ApiClient` implements `CanQueryMessage`. - -## Context-Generic Accessor Provider - -Although the previous accessor implementation for `ApiClient` works, we have to have explicit and -concrete access to the `ApiClient` context in order to implement the accessors. -While this is not too bad with only two accessor methods, it can quickly become tedious once -the application grows, and we need to implement many accessors across many contexts. -It would be more efficient if we can implement _context-generic_ providers for field accessors, -and then use them for any context that contains a given field. - -To make the implementation of context-generic accessors possible, the `cgp` crate offers a derivable -`HasField` trait that can be used as a proxy to access the fields in a concrete context. -The trait is defined as follows: - -```rust -# use core::marker::PhantomData; -# -pub trait HasField { - type Value; - - fn get_field(&self, tag: PhantomData) -> &Self::Value; -} -``` - -For each of the field inside a concrete context, we can implement a `HasField` instance -with the `Tag` type representing the field _name_, and the associated type `Value` -representing the field _type_. -There is also a `get_field` method, which gets a reference of the field value from -the context. The `get_field` method accepts an additional `tag` parameter, -which is just a `PhantomData` with the field name `Tag` as the type. -This phantom parameter is mainly used to help type inference in Rust, -as otherwise Rust would not be able to infer which field `Tag` we are trying to access. - -We can automatically derive `HasField` instances for a context like `ApiClient` -by using the derive macro as follows: - -```rust -# extern crate cgp; -# -# use cgp::prelude::*; -# -#[derive(HasField)] -pub struct ApiClient { - pub api_base_url: String, - pub auth_token: String, -} -``` - -The derive macro would then generate the following `HasField` instances for -`ApiClient`: - -```rust -# extern crate cgp; -# -# use core::marker::PhantomData; -# -# use cgp::prelude::*; -# -# pub struct ApiClient { -# pub api_base_url: String, -# pub auth_token: String, -# } -impl HasField for ApiClient { - type Value = String; - - fn get_field(&self, _tag: PhantomData) -> &String { - &self.api_base_url - } -} - -impl HasField for ApiClient { - type Value = String; - - fn get_field(&self, _tag: PhantomData) -> &String { - &self.auth_token - } -} -``` - -## Symbols - -In the derived `HasField` instances, we can see the use of `symbol!("api_base_url")` -and `symbol!("auth_token")` at the position of the `Tag` generic type. -Recall that a string like `"api_base_url"` is a _value_ of type `&str`, -but we want to use the string as a _type_. -To do that, we use the `symbol!` macro to "lift" a string value into a unique -type, so that we get a _type_ that uniquely identifies the string `"api_base_url"`. -Basically, this means that if the string content in two different uses of `symbol!` -are the same, then they would be treated as the same type. - -Behind the scene, `symbol!` first use the `Char` type to "lift" individual characters -into types. The `Char` type is defined as follows: - -```rust -pub struct Char; -``` - -We make use of the [_const generics_](https://blog.rust-lang.org/2021/02/26/const-generics-mvp-beta.html) -feature in Rust to parameterize `Char` with a constant `CHAR` of type `char`. -The `Char` struct itself has an empty body, because we only want to use it like -a `char` at the type level. - -Note that although we can use const generics to lift individual characters, we can't -yet use a type like `String` or `&str` inside const generics. -So until we can use strings inside const generics, we need a different workaround -to lift strings into types. - -We workaround that by constructing a _type-level list_ of characters. So a type like -`symbol!("abc")` would be desugared to something like: - -```rust,ignore -(Char<'a'>, (Char<'b'>, (Char<'c'>, ()))) -``` - -In `cgp`, instead of using the native Rust tuple, we define the `Cons` and `Nil` -types to help identifying type level lists: - -```rust -pub struct Nil; - -pub struct Cons(pub Head, pub Tail); -``` - -Similar to the linked list concepts in Lisp, the `Nil` type is used to represent -an empty type-level list, and the `Cons` type is used to "add" an element to the -front of the type-level list. - -With that, the actual desugaring of a type like `symbol!("abc")` looks like follows: - -```rust,ignore -Cons, Cons, Cons, Nil>>> -``` - -Although the type make look complicated, it has a pretty compact representation from the -perspective of the Rust compiler. And since we never construct a value out of the symbol -type at runtime, we don't need to worry about any runtime overhead on using symbol types. -Aside from that, since we will mostly only use `HasField` to implement context-generic -accessors, there is negligible compile-time overhead of using `HasField` inside large -codebases. - -It is also worth noting that the current representation of symbols is a temporary -workaround. Once Rust supports the use of strings inside const generics, we can -migrate the desugaring of `symbol!` to make use of that to simplify the type -representation. - -## Using `HasField` in Accessor Providers - -Using `HasField`, we can then implement a context-generic provider for `ApiUrlGetter` -like follows: - -```rust -# extern crate cgp; -# -# use core::marker::PhantomData; -# -# use cgp::prelude::*; -# -# #[cgp_component { -# provider: ApiBaseUrlGetter, -# }] -# pub trait HasApiBaseUrl { -# fn api_base_url(&self) -> &String; -# } -# -pub struct GetApiUrl; - -impl ApiBaseUrlGetter for GetApiUrl -where - Context: HasField, -{ - fn api_base_url(context: &Context) -> &String { - context.get_field(PhantomData) - } -} -``` - -The provider `GetApiUrl` is implemented for any `Context` type that implements -`HasField`. This means that as long as the -context uses `#[derive(HasField)]` has an `api_url` field with `String` type, -then we can use `GetApiUrl` with it. - -Similarly, we can implement a context-generic provider for `AuthTokenGetter` as follows: - -```rust -# extern crate cgp; -# -# use core::marker::PhantomData; -# -# use cgp::prelude::*; -# -# #[cgp_component { -# name: AuthTokenTypeComponent, -# provider: ProvideAuthTokenType, -# }] -# pub trait HasAuthTokenType { -# type AuthToken; -# } -# -# #[cgp_component { -# provider: AuthTokenGetter, -# }] -# pub trait HasAuthToken: HasAuthTokenType { -# fn auth_token(&self) -> &Self::AuthToken; -# } -# -pub struct GetAuthToken; - -impl AuthTokenGetter for GetAuthToken -where - Context: HasAuthTokenType + HasField, -{ - fn auth_token(context: &Context) -> &Context::AuthToken { - context.get_field(PhantomData) - } -} -``` - -The provider `GetAuthToken` is slightly more complicated, because the `auth_token()` method -returns an abstract `Context::AuthToken` type. -To work with that, we first need `Context` to implement `HasAuthTokenType`, and then -require the `Value` associated type to be the same as `Context::AuthToken`. -This means that `GetAuthToken` can be used with a context, if it uses -`#[derive(HasField)]` and has an `auth_token` field with the same type as -the `AuthToken` type that it implements. - -## The `UseField` Pattern - -In the previous section, we managed to implement the context-generic accessor providers -`GetApiUrl` and `GetAuthToken`, without access to the concrete context. However, the field names -`api_url` and `auth_token` are hardcoded into the provider implementation. This means that -a concrete context cannot choose different _field names_ for the specific fields, unless -they manually re-implement the accessors. - -There may be different reasons why a context may want to use different names to store the -field values. For example, there could be two independent accessor providers that happen -to choose the same field name for different types. A context may also have multiple similar -fields that serve similar purposes but with slightly different names. -Whatever the reason is, it would be nice if we can allow the contexts to customize the -field names, instead of letting the providers to pick fixed field names. - -For this purpose, the `cgp` crate provides the `UseField` type that we can use to -implement accessor providers: - -```rust -# use core::marker::PhantomData; -# -pub struct UseField(pub PhantomData); -``` - -Similar to the [`UseDelegate` pattern](./delegated-error-raiser.md), the `UseField` type -is used as a label for accessor implementations following the `UseField` pattern. -Using `UseField`, we can implement the providers as follows: - - -```rust -# extern crate cgp; -# -# use core::marker::PhantomData; -# -# use cgp::prelude::*; -# use cgp::core::field::UseField; -# -# #[cgp_component { -# provider: ApiBaseUrlGetter, -# }] -# pub trait HasApiBaseUrl { -# fn api_base_url(&self) -> &String; -# } -# -# #[cgp_component { -# name: AuthTokenTypeComponent, -# provider: ProvideAuthTokenType, -# }] -# pub trait HasAuthTokenType { -# type AuthToken; -# } -# -# #[cgp_component { -# provider: AuthTokenGetter, -# }] -# pub trait HasAuthToken: HasAuthTokenType { -# fn auth_token(&self) -> &Self::AuthToken; -# } -# -impl ApiBaseUrlGetter for UseField -where - Context: HasField, -{ - fn api_base_url(context: &Context) -> &String { - context.get_field(PhantomData) - } -} - -impl AuthTokenGetter for UseField -where - Context: HasAuthTokenType + HasField, -{ - fn auth_token(context: &Context) -> &Context::AuthToken { - context.get_field(PhantomData) - } -} -``` - -Compared to the explicit providers `GetApiUrl` and `GetAuthToken`, we implement -the traits `ApiBaseUrlGetter` and `AuthTokenGetter` directly on the `UseField` -type provided by the `cgp` crate. -The implementation is also parameterized by an additional `Tag` type, to represent -the name of the field we want to use. -We can see that the implementation is almost the same as before, except that -we no longer use `symbol!` to directly refer to the field names. - -Using `UseField`, we get to simplify the implementation of `ApiClient` and -wire up the accessor components directly inside `delegate_components!`: - -```rust -# extern crate cgp; -# extern crate cgp_error_anyhow; -# extern crate reqwest; -# extern crate serde; -# -# use core::fmt::Display; -# use core::marker::PhantomData; -# -# use cgp::core::component::UseDelegate; -# use cgp::extra::error::RaiseFrom; -# use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; -# use cgp::core::field::UseField; -# use cgp::prelude::*; -# use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; -# use reqwest::blocking::Client; -# use reqwest::StatusCode; -# use serde::Deserialize; -# -# #[cgp_component { -# name: MessageIdTypeComponent, -# provider: ProvideMessageIdType, -# }] -# pub trait HasMessageIdType { -# type MessageId; -# } -# -# #[cgp_component { -# name: MessageTypeComponent, -# provider: ProvideMessageType, -# }] -# pub trait HasMessageType { -# type Message; -# } -# -# #[cgp_component { -# provider: MessageQuerier, -# }] -# pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { -# fn query_message(&self, message_id: &Self::MessageId) -> Result; -# } -# -# #[cgp_component { -# provider: ApiBaseUrlGetter, -# }] -# pub trait HasApiBaseUrl { -# fn api_base_url(&self) -> &String; -# } -# -# #[cgp_component { -# name: AuthTokenTypeComponent, -# provider: ProvideAuthTokenType, -# }] -# pub trait HasAuthTokenType { -# type AuthToken; -# } -# -# #[cgp_component { -# provider: AuthTokenGetter, -# }] -# pub trait HasAuthToken: HasAuthTokenType { -# fn auth_token(&self) -> &Self::AuthToken; -# } -# -# pub struct ReadMessageFromApi; -# -# #[derive(Debug)] -# pub struct ErrStatusCode { -# pub status_code: StatusCode, -# } -# -# #[derive(Deserialize)] -# pub struct ApiMessageResponse { -# pub message: String, -# } -# -# impl MessageQuerier for ReadMessageFromApi -# where -# Context: HasMessageIdType -# + HasMessageType -# + HasApiBaseUrl -# + HasAuthToken -# + CanRaiseError -# + CanRaiseError, -# Context::AuthToken: Display, -# { -# fn query_message(context: &Context, message_id: &u64) -> Result { -# let client = Client::new(); -# -# let url = format!("{}/api/messages/{}", context.api_base_url(), message_id); -# -# let response = client -# .get(url) -# .bearer_auth(context.auth_token()) -# .send() -# .map_err(Context::raise_error)?; -# -# let status_code = response.status(); -# -# if !status_code.is_success() { -# return Err(Context::raise_error(ErrStatusCode { status_code })); -# } -# -# let message_response: ApiMessageResponse = response.json().map_err(Context::raise_error)?; -# -# Ok(message_response.message) -# } -# } -# -# pub struct UseStringAuthToken; -# -# impl ProvideAuthTokenType for UseStringAuthToken { -# type AuthToken = String; -# } -# -# pub struct UseU64MessageId; -# -# impl ProvideMessageIdType for UseU64MessageId { -# type MessageId = u64; -# } -# -# pub struct UseStringMessage; -# -# impl ProvideMessageType for UseStringMessage { -# type Message = String; -# } -# -# impl ApiBaseUrlGetter for UseField -# where -# Context: HasField, -# { -# fn api_base_url(context: &Context) -> &String { -# context.get_field(PhantomData) -# } -# } -# -# impl AuthTokenGetter for UseField -# where -# Context: HasAuthTokenType + HasField, -# { -# fn auth_token(context: &Context) -> &Context::AuthToken { -# context.get_field(PhantomData) -# } -# } -# -# #[derive(HasField)] -# pub struct ApiClient { -# pub api_base_url: String, -# pub auth_token: String, -# } -# -# pub struct ApiClientComponents; -# -# pub struct RaiseApiErrors; -# -# impl HasComponents for ApiClient { -# type Components = ApiClientComponents; -# } -# -delegate_components! { - ApiClientComponents { - ErrorTypeComponent: UseAnyhowError, - ErrorRaiserComponent: UseDelegate, - MessageIdTypeComponent: UseU64MessageId, - MessageTypeComponent: UseStringMessage, - AuthTokenTypeComponent: UseStringAuthToken, - ApiBaseUrlGetterComponent: UseField, - AuthTokenGetterComponent: UseField, - MessageQuerierComponent: ReadMessageFromApi, - } -} -# -# delegate_components! { -# RaiseApiErrors { -# reqwest::Error: RaiseFrom, -# ErrStatusCode: DebugAnyhowError, -# } -# } -# -# pub trait CanUseApiClient: CanQueryMessage {} -# -# impl CanUseApiClient for ApiClient {} -``` - -The wiring above uses `UseField` to implement `ApiBaseUrlGetterComponent`, -and `UseField` to implement `AuthTokenGetterComponent`. -With the field names specified explicitly in the wiring, we can easily change the field -names in the `ApiClient` context, and update the wiring accordingly. - -## Using `HasField` Directly Inside Providers - -Since the `HasField` trait can be automatically derived by contexts, some readers may be -tempted to not define any accessor trait, and instead make use of `HasField` directly -inside the providers. For example, we can in principle remove `HasApiBaseUrl` and -`HasAuthToken`, and re-implement `ReadMessageFromApi` as follows: - -```rust -# extern crate cgp; -# extern crate reqwest; -# extern crate serde; -# -# use core::fmt::Display; -# use core::marker::PhantomData; -# -# use cgp::prelude::*; -# use reqwest::blocking::Client; -# use reqwest::StatusCode; -# use serde::Deserialize; -# -# #[cgp_component { -# name: MessageIdTypeComponent, -# provider: ProvideMessageIdType, -# }] -# pub trait HasMessageIdType { -# type MessageId; -# } -# -# #[cgp_component { -# name: MessageTypeComponent, -# provider: ProvideMessageType, -# }] -# pub trait HasMessageType { -# type Message; -# } -# -# #[cgp_component { -# provider: MessageQuerier, -# }] -# pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { -# fn query_message(&self, message_id: &Self::MessageId) -> Result; -# } -# -# #[cgp_component { -# name: AuthTokenTypeComponent, -# provider: ProvideAuthTokenType, -# }] -# pub trait HasAuthTokenType { -# type AuthToken; -# } -# -# pub struct ReadMessageFromApi; -# -# #[derive(Debug)] -# pub struct ErrStatusCode { -# pub status_code: StatusCode, -# } -# -# #[derive(Deserialize)] -# pub struct ApiMessageResponse { -# pub message: String, -# } -# -impl MessageQuerier for ReadMessageFromApi -where - Context: HasMessageIdType - + HasMessageType - + HasAuthTokenType - + HasField - + HasField - + CanRaiseError - + CanRaiseError, - Context::AuthToken: Display, -{ - fn query_message(context: &Context, message_id: &u64) -> Result { - let client = Client::new(); - - let url = format!( - "{}/api/messages/{}", - context.get_field(PhantomData::), - message_id - ); - - let response = client - .get(url) - .bearer_auth(context.get_field(PhantomData::)) - .send() - .map_err(Context::raise_error)?; - - let status_code = response.status(); - - if !status_code.is_success() { - return Err(Context::raise_error(ErrStatusCode { status_code })); - } - - let message_response: ApiMessageResponse = response.json().map_err(Context::raise_error)?; - - Ok(message_response.message) - } -} -``` - -In the implementation above, the provider `ReadMessageFromApi` requires the context to implement -`HasField` and `HasField`. -To preserve the original behavior, we also have additional constraints that the field `api_base_url` -needs to be of `String` type, and the field `auth_token` needs to have the same type as -`Context::AuthToken`. -When using `get_field`, since there are two instances of `HasField` implemented in scope, -we need to fully qualify the call to specify the field name that we want to access, -such as `context.get_field(PhantomData::)`. - -As we can see, the direct use of `HasField` may not necessary make the code simpler, and instead -require more verbose specification of the fields. The direct use of `HasFields` also requires -explicit specification of what the field types should be. -Whereas in accessor traits like `HasAuthToken`, we can better specify that the method always -return the abstract type `Self::AuthToken`, so one cannot accidentally read from different -fields that happen to have the same underlying concrete type. - -By using `HasField` directly, the provider also makes it less flexible for the context to have -custom ways of getting the field value. For example, instead of putting the `api_url` field -directly in the context, we may want to put it inside another `ApiConfig` struct such as follows: - -```rust -pub struct Config { - pub api_base_url: String, - // other fields -} - -pub struct ApiClient { - pub config: Config, - pub auth_token: String, - // other fields -} -``` - -In such cases, with an accessor trait like `HasApiUrl`, the context can easily make use of -custom accessor providers to implement such indirect access. But with direct use of -`HasFields`, it would be more tedious to implement the indirect access. - -That said, similar to other shortcut methods, the direct use of `HasField` can be convenient -during initial development, as it helps to significantly reduce the number of traits the -developer needs to keep track of. As a result, we encourage readers to feel free to make -use of `HasField` as they see fit, and then slowly migrate to proper accessor traits -when the need arise. - -## Static Accessors - -One benefit of defining minimal accessor traits is that we get to implement custom -accessor providers that do not necessarily need to read the field values from the context. -For example, we can implement _static accessor_ providers that always return a global -constant value. - -The use of static accessors can be useful when we want to hard code some values for a -specific context. For instance, we may want to define a production `ApiClient` context -that always use a hard-coded API URL: - -```rust -# extern crate cgp; -# -# use core::marker::PhantomData; -use std::sync::OnceLock; - -# use cgp::prelude::*; -# -# #[cgp_component { -# provider: ApiBaseUrlGetter, -# }] -# pub trait HasApiBaseUrl { -# fn api_base_url(&self) -> &String; -# } -# -pub struct UseProductionApiUrl; - -impl ApiBaseUrlGetter for UseProductionApiUrl { - fn api_base_url(_context: &Context) -> &String { - static BASE_URL: OnceLock = OnceLock::new(); - - BASE_URL.get_or_init(|| "https://api.example.com".into()) - } -} -``` - -The provider `UseProductionApiUrl` implements `ApiBaseUrlGetter` for any context type. -Inside the `api_base_url` method, we first define a static `BASE_URL` value with the -type `OnceLock`. The use of [`OnceLock`](https://doc.rust-lang.org/std/sync/struct.OnceLock.html) -allows us to define a global variable in Rust that is initialized exactly once, and -then remain constant throughout the application. -This is mainly useful because constructors like `String::from` are not currently `const fn`, -so we have to make use of `OnceLock::get_or_init` to run the non-const constructor. -By defining the static variable inside the method, we ensure that the variable can only be -accessed and initialized by the provider. - -Using `UseProductionApiUrl`, we can now define a production `ApiClient` context such as follows: - -```rust -# extern crate cgp; -# extern crate cgp_error_anyhow; -# extern crate reqwest; -# extern crate serde; -# -# use core::fmt::Display; -# use core::marker::PhantomData; -# use std::sync::OnceLock; -# -# use cgp::core::component::UseDelegate; -# use cgp::extra::error::RaiseFrom; -# use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; -# use cgp::core::field::UseField; -# use cgp::prelude::*; -# use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; -# use reqwest::blocking::Client; -# use reqwest::StatusCode; -# use serde::Deserialize; -# -# #[cgp_component { -# name: MessageIdTypeComponent, -# provider: ProvideMessageIdType, -# }] -# pub trait HasMessageIdType { -# type MessageId; -# } -# -# #[cgp_component { -# name: MessageTypeComponent, -# provider: ProvideMessageType, -# }] -# pub trait HasMessageType { -# type Message; -# } -# -# #[cgp_component { -# provider: MessageQuerier, -# }] -# pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { -# fn query_message(&self, message_id: &Self::MessageId) -> Result; -# } -# -# #[cgp_component { -# provider: ApiBaseUrlGetter, -# }] -# pub trait HasApiBaseUrl { -# fn api_base_url(&self) -> &String; -# } -# -# #[cgp_component { -# name: AuthTokenTypeComponent, -# provider: ProvideAuthTokenType, -# }] -# pub trait HasAuthTokenType { -# type AuthToken; -# } -# -# #[cgp_component { -# provider: AuthTokenGetter, -# }] -# pub trait HasAuthToken: HasAuthTokenType { -# fn auth_token(&self) -> &Self::AuthToken; -# } -# -# pub struct ReadMessageFromApi; -# -# #[derive(Debug)] -# pub struct ErrStatusCode { -# pub status_code: StatusCode, -# } -# -# #[derive(Deserialize)] -# pub struct ApiMessageResponse { -# pub message: String, -# } -# -# impl MessageQuerier for ReadMessageFromApi -# where -# Context: HasMessageIdType -# + HasMessageType -# + HasApiBaseUrl -# + HasAuthToken -# + CanRaiseError -# + CanRaiseError, -# Context::AuthToken: Display, -# { -# fn query_message(context: &Context, message_id: &u64) -> Result { -# let client = Client::new(); -# -# let url = format!("{}/api/messages/{}", context.api_base_url(), message_id); -# -# let response = client -# .get(url) -# .bearer_auth(context.auth_token()) -# .send() -# .map_err(Context::raise_error)?; -# -# let status_code = response.status(); -# -# if !status_code.is_success() { -# return Err(Context::raise_error(ErrStatusCode { status_code })); -# } -# -# let message_response: ApiMessageResponse = response.json().map_err(Context::raise_error)?; -# -# Ok(message_response.message) -# } -# } -# -# pub struct UseStringAuthToken; -# -# impl ProvideAuthTokenType for UseStringAuthToken { -# type AuthToken = String; -# } -# -# pub struct UseU64MessageId; -# -# impl ProvideMessageIdType for UseU64MessageId { -# type MessageId = u64; -# } -# -# pub struct UseStringMessage; -# -# impl ProvideMessageType for UseStringMessage { -# type Message = String; -# } -# -# impl AuthTokenGetter for UseField -# where -# Context: HasAuthTokenType + HasField, -# { -# fn auth_token(context: &Context) -> &Context::AuthToken { -# context.get_field(PhantomData) -# } -# } -# -# pub struct UseProductionApiUrl; -# -# impl ApiBaseUrlGetter for UseProductionApiUrl { -# fn api_base_url(_context: &Context) -> &String { -# static BASE_URL: OnceLock = OnceLock::new(); -# -# BASE_URL.get_or_init(|| "https://api.example.com".into()) -# } -# } -# -#[derive(HasField)] -pub struct ApiClient { - pub auth_token: String, -} - -pub struct ApiClientComponents; - -# pub struct RaiseApiErrors; -# -impl HasComponents for ApiClient { - type Components = ApiClientComponents; -} - -delegate_components! { - ApiClientComponents { - ErrorTypeComponent: UseAnyhowError, - ErrorRaiserComponent: UseDelegate, - MessageIdTypeComponent: UseU64MessageId, - MessageTypeComponent: UseStringMessage, - AuthTokenTypeComponent: UseStringAuthToken, - ApiBaseUrlGetterComponent: UseProductionApiUrl, - AuthTokenGetterComponent: UseField, - MessageQuerierComponent: ReadMessageFromApi, - } -} -# -# delegate_components! { -# RaiseApiErrors { -# reqwest::Error: RaiseFrom, -# ErrStatusCode: DebugAnyhowError, -# } -# } -# -# pub trait CanUseApiClient: CanQueryMessage {} -# -# impl CanUseApiClient for ApiClient {} -``` - -Inside the component wiring, we choose `UseProductionApiUrl` to be the provider -for `ApiBaseUrlGetterComponent`. -Notice that now the `ApiClient` context no longer contain any `api_base_url` field. - -The use of static accessors can be useful to implement specialized contexts -that keep the values constant for certain fields. -With this approach, the constant values no longer needs to be passed around -as part of the context during runtime, and we no longer need to worry -about keeping the field private or preventing the wrong value being assigned -at runtime. -Thanks to the compile-time wiring, we may even get some performance advantage -as compared to passing around dynamic values at runtime. - -## Auto Accessor Traits - -The need to define and wire up many CGP components may overwhelm a developer who -is new to CGP. -At least during the beginning phase, a project don't usually that much flexibility -in customizing how fields are accessed. -As such, some may consider the full use of field accessors introduced in this chapter -being unnecessarily complicated. - -One intermediate way to simplify use of accessor traits is to define them _not_ -as CGP components, but as regular Rust traits with blanket implementations that -use `HasField`. For example, we can re-define the `HasApiUrl` trait as follows: - -```rust -# extern crate cgp; -# -# use core::marker::PhantomData; -# -# use cgp::prelude::*; -# -pub trait HasApiBaseUrl { - fn api_base_url(&self) -> &String; -} - -impl HasApiBaseUrl for Context -where - Context: HasField, -{ - fn api_base_url(&self) -> &String { - self.get_field(PhantomData) - } -} -``` - -This way, the `HasApiBaseUrl` will always be implemented for any context -that derive `HasField` and have the relevant field, and -there is no need to have explicit wiring of `ApiBaseUrlGetterComponent` -inside the wiring of the context components. - -With this, providers like `ReadMessageFromApi` can still use traits like `HasApiBaseUrl` -to simplify the access of fields. And the context implementors can just use -`#[derive(HasField)]` without having to worry about the wiring. - -The main downside of this approach is that the context cannot easily override the -implementation of `HaswApiBaseUrl`, unless they don't implement `HasField` at all. -Nevertheless, it will be straightforward to refactor the trait in the future -to turn it into a full CGP component. - -As a result, this may be an appealing option for readers who want to have a simpler -experience of using CGP and not use its full power. - -## Conclusion - -In this chapter, we have learned about different ways to define accessor traits, -and to implement the accessor providers. The use of a derivable `HasField` trait -makes it possible to implement context-generic accessor providers without -requiring direct access to the concrete context. The use of the `UseField` pattern -unifies the convention of implementing field accessors, and allows contexts -to choose different field names for the accessors. - -As we will see in later chapters, the use of context-generic accessor providers -make it possible to implement almost everything as context-generic providers, -and leaving almost no code tied to specific concrete contexts. \ No newline at end of file diff --git a/content/generic-accessor-providers.md b/content/generic-accessor-providers.md new file mode 100644 index 0000000..1d76169 --- /dev/null +++ b/content/generic-accessor-providers.md @@ -0,0 +1,957 @@ +# Context-Generic Accessor Providers + +Although the previous accessor implementation for `ApiClient` works, we have to have explicit and +concrete access to the `ApiClient` context in order to implement the accessors. +While this is not too bad with only two accessor methods, it can quickly become tedious once +the application grows, and we need to implement many accessors across many contexts. +It would be more efficient if we can implement _context-generic_ providers for field accessors, +and then use them for any context that contains a given field. + +To make the implementation of context-generic accessors possible, the `cgp` crate offers a derivable +`HasField` trait that can be used as a proxy to access the fields in a concrete context. +The trait is defined as follows: + +```rust +# use core::marker::PhantomData; +# +pub trait HasField { + type Value; + + fn get_field(&self, tag: PhantomData) -> &Self::Value; +} +``` + +For each of the field inside a concrete context, we can implement a `HasField` instance +with the `Tag` type representing the field _name_, and the associated type `Value` +representing the field _type_. +There is also a `get_field` method, which gets a reference of the field value from +the context. The `get_field` method accepts an additional `tag` parameter, +which is just a `PhantomData` with the field name `Tag` as the type. +This phantom parameter is mainly used to help type inference in Rust, +as otherwise Rust would not be able to infer which field `Tag` we are trying to access. + +We can automatically derive `HasField` instances for a context like `ApiClient` +by using the derive macro as follows: + +```rust +# extern crate cgp; +# +# use cgp::prelude::*; +# +#[derive(HasField)] +pub struct ApiClient { + pub api_base_url: String, + pub auth_token: String, +} +``` + +The derive macro would then generate the following `HasField` instances for +`ApiClient`: + +```rust +# extern crate cgp; +# +# use core::marker::PhantomData; +# +# use cgp::prelude::*; +# +# pub struct ApiClient { +# pub api_base_url: String, +# pub auth_token: String, +# } +impl HasField for ApiClient { + type Value = String; + + fn get_field(&self, _tag: PhantomData) -> &String { + &self.api_base_url + } +} + +impl HasField for ApiClient { + type Value = String; + + fn get_field(&self, _tag: PhantomData) -> &String { + &self.auth_token + } +} +``` + +## Symbols + +In the derived `HasField` instances, we can see the use of `symbol!("api_base_url")` +and `symbol!("auth_token")` at the position of the `Tag` generic type. +Recall that a string like `"api_base_url"` is a _value_ of type `&str`, +but we want to use the string as a _type_. +To do that, we use the `symbol!` macro to "lift" a string value into a unique +type, so that we get a _type_ that uniquely identifies the string `"api_base_url"`. +Basically, this means that if the string content in two different uses of `symbol!` +are the same, then they would be treated as the same type. + +Behind the scene, `symbol!` first use the `Char` type to "lift" individual characters +into types. The `Char` type is defined as follows: + +```rust +pub struct Char; +``` + +We make use of the [_const generics_](https://blog.rust-lang.org/2021/02/26/const-generics-mvp-beta.html) +feature in Rust to parameterize `Char` with a constant `CHAR` of type `char`. +The `Char` struct itself has an empty body, because we only want to use it like +a `char` at the type level. + +Note that although we can use const generics to lift individual characters, we can't +yet use a type like `String` or `&str` inside const generics. +So until we can use strings inside const generics, we need a different workaround +to lift strings into types. + +We workaround that by constructing a _type-level list_ of characters. So a type like +`symbol!("abc")` would be desugared to something like: + +```rust,ignore +(Char<'a'>, (Char<'b'>, (Char<'c'>, ()))) +``` + +In `cgp`, instead of using the native Rust tuple, we define the `Cons` and `Nil` +types to help identifying type level lists: + +```rust +pub struct Nil; + +pub struct Cons(pub Head, pub Tail); +``` + +Similar to the linked list concepts in Lisp, the `Nil` type is used to represent +an empty type-level list, and the `Cons` type is used to "add" an element to the +front of the type-level list. + +With that, the actual desugaring of a type like `symbol!("abc")` looks like follows: + +```rust,ignore +Cons, Cons, Cons, Nil>>> +``` + +Although the type make look complicated, it has a pretty compact representation from the +perspective of the Rust compiler. And since we never construct a value out of the symbol +type at runtime, we don't need to worry about any runtime overhead on using symbol types. +Aside from that, since we will mostly only use `HasField` to implement context-generic +accessors, there is negligible compile-time overhead of using `HasField` inside large +codebases. + +It is also worth noting that the current representation of symbols is a temporary +workaround. Once Rust supports the use of strings inside const generics, we can +migrate the desugaring of `symbol!` to make use of that to simplify the type +representation. + +## Using `HasField` in Accessor Providers + +Using `HasField`, we can then implement a context-generic provider for `ApiUrlGetter` +like follows: + +```rust +# extern crate cgp; +# +# use core::marker::PhantomData; +# +# use cgp::prelude::*; +# +# #[cgp_component { +# provider: ApiBaseUrlGetter, +# }] +# pub trait HasApiBaseUrl { +# fn api_base_url(&self) -> &String; +# } +# +pub struct GetApiUrl; + +impl ApiBaseUrlGetter for GetApiUrl +where + Context: HasField, +{ + fn api_base_url(context: &Context) -> &String { + context.get_field(PhantomData) + } +} +``` + +The provider `GetApiUrl` is implemented for any `Context` type that implements +`HasField`. This means that as long as the +context uses `#[derive(HasField)]` has an `api_url` field with `String` type, +then we can use `GetApiUrl` with it. + +Similarly, we can implement a context-generic provider for `AuthTokenGetter` as follows: + +```rust +# extern crate cgp; +# +# use core::marker::PhantomData; +# +# use cgp::prelude::*; +# +# #[cgp_component { +# name: AuthTokenTypeComponent, +# provider: ProvideAuthTokenType, +# }] +# pub trait HasAuthTokenType { +# type AuthToken; +# } +# +# #[cgp_component { +# provider: AuthTokenGetter, +# }] +# pub trait HasAuthToken: HasAuthTokenType { +# fn auth_token(&self) -> &Self::AuthToken; +# } +# +pub struct GetAuthToken; + +impl AuthTokenGetter for GetAuthToken +where + Context: HasAuthTokenType + HasField, +{ + fn auth_token(context: &Context) -> &Context::AuthToken { + context.get_field(PhantomData) + } +} +``` + +The provider `GetAuthToken` is slightly more complicated, because the `auth_token()` method +returns an abstract `Context::AuthToken` type. +To work with that, we first need `Context` to implement `HasAuthTokenType`, and then +require the `Value` associated type to be the same as `Context::AuthToken`. +This means that `GetAuthToken` can be used with a context, if it uses +`#[derive(HasField)]` and has an `auth_token` field with the same type as +the `AuthToken` type that it implements. + +## The `UseField` Pattern + +In the previous section, we managed to implement the context-generic accessor providers +`GetApiUrl` and `GetAuthToken`, without access to the concrete context. However, the field names +`api_url` and `auth_token` are hardcoded into the provider implementation. This means that +a concrete context cannot choose different _field names_ for the specific fields, unless +they manually re-implement the accessors. + +There may be different reasons why a context may want to use different names to store the +field values. For example, there could be two independent accessor providers that happen +to choose the same field name for different types. A context may also have multiple similar +fields that serve similar purposes but with slightly different names. +Whatever the reason is, it would be nice if we can allow the contexts to customize the +field names, instead of letting the providers to pick fixed field names. + +For this purpose, the `cgp` crate provides the `UseField` type that we can use to +implement accessor providers: + +```rust +# use core::marker::PhantomData; +# +pub struct UseField(pub PhantomData); +``` + +Similar to the [`UseDelegate` pattern](./delegated-error-raiser.md), the `UseField` type +is used as a label for accessor implementations following the `UseField` pattern. +Using `UseField`, we can implement the providers as follows: + + +```rust +# extern crate cgp; +# +# use core::marker::PhantomData; +# +# use cgp::prelude::*; +# use cgp::core::field::UseField; +# +# #[cgp_component { +# provider: ApiBaseUrlGetter, +# }] +# pub trait HasApiBaseUrl { +# fn api_base_url(&self) -> &String; +# } +# +# #[cgp_component { +# name: AuthTokenTypeComponent, +# provider: ProvideAuthTokenType, +# }] +# pub trait HasAuthTokenType { +# type AuthToken; +# } +# +# #[cgp_component { +# provider: AuthTokenGetter, +# }] +# pub trait HasAuthToken: HasAuthTokenType { +# fn auth_token(&self) -> &Self::AuthToken; +# } +# +impl ApiBaseUrlGetter for UseField +where + Context: HasField, +{ + fn api_base_url(context: &Context) -> &String { + context.get_field(PhantomData) + } +} + +impl AuthTokenGetter for UseField +where + Context: HasAuthTokenType + HasField, +{ + fn auth_token(context: &Context) -> &Context::AuthToken { + context.get_field(PhantomData) + } +} +``` + +Compared to the explicit providers `GetApiUrl` and `GetAuthToken`, we implement +the traits `ApiBaseUrlGetter` and `AuthTokenGetter` directly on the `UseField` +type provided by the `cgp` crate. +The implementation is also parameterized by an additional `Tag` type, to represent +the name of the field we want to use. +We can see that the implementation is almost the same as before, except that +we no longer use `symbol!` to directly refer to the field names. + +Using `UseField`, we get to simplify the implementation of `ApiClient` and +wire up the accessor components directly inside `delegate_components!`: + +```rust +# extern crate cgp; +# extern crate cgp_error_anyhow; +# extern crate reqwest; +# extern crate serde; +# +# use core::fmt::Display; +# use core::marker::PhantomData; +# +# use cgp::core::component::UseDelegate; +# use cgp::extra::error::RaiseFrom; +# use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; +# use cgp::core::field::UseField; +# use cgp::prelude::*; +# use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; +# use reqwest::blocking::Client; +# use reqwest::StatusCode; +# use serde::Deserialize; +# +# #[cgp_component { +# name: MessageIdTypeComponent, +# provider: ProvideMessageIdType, +# }] +# pub trait HasMessageIdType { +# type MessageId; +# } +# +# #[cgp_component { +# name: MessageTypeComponent, +# provider: ProvideMessageType, +# }] +# pub trait HasMessageType { +# type Message; +# } +# +# #[cgp_component { +# provider: MessageQuerier, +# }] +# pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { +# fn query_message(&self, message_id: &Self::MessageId) -> Result; +# } +# +# #[cgp_component { +# provider: ApiBaseUrlGetter, +# }] +# pub trait HasApiBaseUrl { +# fn api_base_url(&self) -> &String; +# } +# +# #[cgp_component { +# name: AuthTokenTypeComponent, +# provider: ProvideAuthTokenType, +# }] +# pub trait HasAuthTokenType { +# type AuthToken; +# } +# +# #[cgp_component { +# provider: AuthTokenGetter, +# }] +# pub trait HasAuthToken: HasAuthTokenType { +# fn auth_token(&self) -> &Self::AuthToken; +# } +# +# pub struct ReadMessageFromApi; +# +# #[derive(Debug)] +# pub struct ErrStatusCode { +# pub status_code: StatusCode, +# } +# +# #[derive(Deserialize)] +# pub struct ApiMessageResponse { +# pub message: String, +# } +# +# impl MessageQuerier for ReadMessageFromApi +# where +# Context: HasMessageIdType +# + HasMessageType +# + HasApiBaseUrl +# + HasAuthToken +# + CanRaiseError +# + CanRaiseError, +# Context::AuthToken: Display, +# { +# fn query_message(context: &Context, message_id: &u64) -> Result { +# let client = Client::new(); +# +# let url = format!("{}/api/messages/{}", context.api_base_url(), message_id); +# +# let response = client +# .get(url) +# .bearer_auth(context.auth_token()) +# .send() +# .map_err(Context::raise_error)?; +# +# let status_code = response.status(); +# +# if !status_code.is_success() { +# return Err(Context::raise_error(ErrStatusCode { status_code })); +# } +# +# let message_response: ApiMessageResponse = response.json().map_err(Context::raise_error)?; +# +# Ok(message_response.message) +# } +# } +# +# pub struct UseStringAuthToken; +# +# impl ProvideAuthTokenType for UseStringAuthToken { +# type AuthToken = String; +# } +# +# pub struct UseU64MessageId; +# +# impl ProvideMessageIdType for UseU64MessageId { +# type MessageId = u64; +# } +# +# pub struct UseStringMessage; +# +# impl ProvideMessageType for UseStringMessage { +# type Message = String; +# } +# +# impl ApiBaseUrlGetter for UseField +# where +# Context: HasField, +# { +# fn api_base_url(context: &Context) -> &String { +# context.get_field(PhantomData) +# } +# } +# +# impl AuthTokenGetter for UseField +# where +# Context: HasAuthTokenType + HasField, +# { +# fn auth_token(context: &Context) -> &Context::AuthToken { +# context.get_field(PhantomData) +# } +# } +# +# #[derive(HasField)] +# pub struct ApiClient { +# pub api_base_url: String, +# pub auth_token: String, +# } +# +# pub struct ApiClientComponents; +# +# pub struct RaiseApiErrors; +# +# impl HasComponents for ApiClient { +# type Components = ApiClientComponents; +# } +# +delegate_components! { + ApiClientComponents { + ErrorTypeComponent: UseAnyhowError, + ErrorRaiserComponent: UseDelegate, + MessageIdTypeComponent: UseU64MessageId, + MessageTypeComponent: UseStringMessage, + AuthTokenTypeComponent: UseStringAuthToken, + ApiBaseUrlGetterComponent: UseField, + AuthTokenGetterComponent: UseField, + MessageQuerierComponent: ReadMessageFromApi, + } +} +# +# delegate_components! { +# RaiseApiErrors { +# reqwest::Error: RaiseFrom, +# ErrStatusCode: DebugAnyhowError, +# } +# } +# +# pub trait CanUseApiClient: CanQueryMessage {} +# +# impl CanUseApiClient for ApiClient {} +``` + +The wiring above uses `UseField` to implement `ApiBaseUrlGetterComponent`, +and `UseField` to implement `AuthTokenGetterComponent`. +With the field names specified explicitly in the wiring, we can easily change the field +names in the `ApiClient` context, and update the wiring accordingly. + +## Using `HasField` Directly Inside Providers + +Since the `HasField` trait can be automatically derived by contexts, some readers may be +tempted to not define any accessor trait, and instead make use of `HasField` directly +inside the providers. For example, we can in principle remove `HasApiBaseUrl` and +`HasAuthToken`, and re-implement `ReadMessageFromApi` as follows: + +```rust +# extern crate cgp; +# extern crate reqwest; +# extern crate serde; +# +# use core::fmt::Display; +# use core::marker::PhantomData; +# +# use cgp::prelude::*; +# use reqwest::blocking::Client; +# use reqwest::StatusCode; +# use serde::Deserialize; +# +# #[cgp_component { +# name: MessageIdTypeComponent, +# provider: ProvideMessageIdType, +# }] +# pub trait HasMessageIdType { +# type MessageId; +# } +# +# #[cgp_component { +# name: MessageTypeComponent, +# provider: ProvideMessageType, +# }] +# pub trait HasMessageType { +# type Message; +# } +# +# #[cgp_component { +# provider: MessageQuerier, +# }] +# pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { +# fn query_message(&self, message_id: &Self::MessageId) -> Result; +# } +# +# #[cgp_component { +# name: AuthTokenTypeComponent, +# provider: ProvideAuthTokenType, +# }] +# pub trait HasAuthTokenType { +# type AuthToken; +# } +# +# pub struct ReadMessageFromApi; +# +# #[derive(Debug)] +# pub struct ErrStatusCode { +# pub status_code: StatusCode, +# } +# +# #[derive(Deserialize)] +# pub struct ApiMessageResponse { +# pub message: String, +# } +# +impl MessageQuerier for ReadMessageFromApi +where + Context: HasMessageIdType + + HasMessageType + + HasAuthTokenType + + HasField + + HasField + + CanRaiseError + + CanRaiseError, + Context::AuthToken: Display, +{ + fn query_message(context: &Context, message_id: &u64) -> Result { + let client = Client::new(); + + let url = format!( + "{}/api/messages/{}", + context.get_field(PhantomData::), + message_id + ); + + let response = client + .get(url) + .bearer_auth(context.get_field(PhantomData::)) + .send() + .map_err(Context::raise_error)?; + + let status_code = response.status(); + + if !status_code.is_success() { + return Err(Context::raise_error(ErrStatusCode { status_code })); + } + + let message_response: ApiMessageResponse = response.json().map_err(Context::raise_error)?; + + Ok(message_response.message) + } +} +``` + +In the implementation above, the provider `ReadMessageFromApi` requires the context to implement +`HasField` and `HasField`. +To preserve the original behavior, we also have additional constraints that the field `api_base_url` +needs to be of `String` type, and the field `auth_token` needs to have the same type as +`Context::AuthToken`. +When using `get_field`, since there are two instances of `HasField` implemented in scope, +we need to fully qualify the call to specify the field name that we want to access, +such as `context.get_field(PhantomData::)`. + +As we can see, the direct use of `HasField` may not necessary make the code simpler, and instead +require more verbose specification of the fields. The direct use of `HasFields` also requires +explicit specification of what the field types should be. +Whereas in accessor traits like `HasAuthToken`, we can better specify that the method always +return the abstract type `Self::AuthToken`, so one cannot accidentally read from different +fields that happen to have the same underlying concrete type. + +By using `HasField` directly, the provider also makes it less flexible for the context to have +custom ways of getting the field value. For example, instead of putting the `api_url` field +directly in the context, we may want to put it inside another `ApiConfig` struct such as follows: + +```rust +pub struct Config { + pub api_base_url: String, + // other fields +} + +pub struct ApiClient { + pub config: Config, + pub auth_token: String, + // other fields +} +``` + +In such cases, with an accessor trait like `HasApiUrl`, the context can easily make use of +custom accessor providers to implement such indirect access. But with direct use of +`HasFields`, it would be more tedious to implement the indirect access. + +That said, similar to other shortcut methods, the direct use of `HasField` can be convenient +during initial development, as it helps to significantly reduce the number of traits the +developer needs to keep track of. As a result, we encourage readers to feel free to make +use of `HasField` as they see fit, and then slowly migrate to proper accessor traits +when the need arise. + +## Static Accessors + +One benefit of defining minimal accessor traits is that we get to implement custom +accessor providers that do not necessarily need to read the field values from the context. +For example, we can implement _static accessor_ providers that always return a global +constant value. + +The use of static accessors can be useful when we want to hard code some values for a +specific context. For instance, we may want to define a production `ApiClient` context +that always use a hard-coded API URL: + +```rust +# extern crate cgp; +# +# use core::marker::PhantomData; +use std::sync::OnceLock; + +# use cgp::prelude::*; +# +# #[cgp_component { +# provider: ApiBaseUrlGetter, +# }] +# pub trait HasApiBaseUrl { +# fn api_base_url(&self) -> &String; +# } +# +pub struct UseProductionApiUrl; + +impl ApiBaseUrlGetter for UseProductionApiUrl { + fn api_base_url(_context: &Context) -> &String { + static BASE_URL: OnceLock = OnceLock::new(); + + BASE_URL.get_or_init(|| "https://api.example.com".into()) + } +} +``` + +The provider `UseProductionApiUrl` implements `ApiBaseUrlGetter` for any context type. +Inside the `api_base_url` method, we first define a static `BASE_URL` value with the +type `OnceLock`. The use of [`OnceLock`](https://doc.rust-lang.org/std/sync/struct.OnceLock.html) +allows us to define a global variable in Rust that is initialized exactly once, and +then remain constant throughout the application. +This is mainly useful because constructors like `String::from` are not currently `const fn`, +so we have to make use of `OnceLock::get_or_init` to run the non-const constructor. +By defining the static variable inside the method, we ensure that the variable can only be +accessed and initialized by the provider. + +Using `UseProductionApiUrl`, we can now define a production `ApiClient` context such as follows: + +```rust +# extern crate cgp; +# extern crate cgp_error_anyhow; +# extern crate reqwest; +# extern crate serde; +# +# use core::fmt::Display; +# use core::marker::PhantomData; +# use std::sync::OnceLock; +# +# use cgp::core::component::UseDelegate; +# use cgp::extra::error::RaiseFrom; +# use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; +# use cgp::core::field::UseField; +# use cgp::prelude::*; +# use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; +# use reqwest::blocking::Client; +# use reqwest::StatusCode; +# use serde::Deserialize; +# +# #[cgp_component { +# name: MessageIdTypeComponent, +# provider: ProvideMessageIdType, +# }] +# pub trait HasMessageIdType { +# type MessageId; +# } +# +# #[cgp_component { +# name: MessageTypeComponent, +# provider: ProvideMessageType, +# }] +# pub trait HasMessageType { +# type Message; +# } +# +# #[cgp_component { +# provider: MessageQuerier, +# }] +# pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { +# fn query_message(&self, message_id: &Self::MessageId) -> Result; +# } +# +# #[cgp_component { +# provider: ApiBaseUrlGetter, +# }] +# pub trait HasApiBaseUrl { +# fn api_base_url(&self) -> &String; +# } +# +# #[cgp_component { +# name: AuthTokenTypeComponent, +# provider: ProvideAuthTokenType, +# }] +# pub trait HasAuthTokenType { +# type AuthToken; +# } +# +# #[cgp_component { +# provider: AuthTokenGetter, +# }] +# pub trait HasAuthToken: HasAuthTokenType { +# fn auth_token(&self) -> &Self::AuthToken; +# } +# +# pub struct ReadMessageFromApi; +# +# #[derive(Debug)] +# pub struct ErrStatusCode { +# pub status_code: StatusCode, +# } +# +# #[derive(Deserialize)] +# pub struct ApiMessageResponse { +# pub message: String, +# } +# +# impl MessageQuerier for ReadMessageFromApi +# where +# Context: HasMessageIdType +# + HasMessageType +# + HasApiBaseUrl +# + HasAuthToken +# + CanRaiseError +# + CanRaiseError, +# Context::AuthToken: Display, +# { +# fn query_message(context: &Context, message_id: &u64) -> Result { +# let client = Client::new(); +# +# let url = format!("{}/api/messages/{}", context.api_base_url(), message_id); +# +# let response = client +# .get(url) +# .bearer_auth(context.auth_token()) +# .send() +# .map_err(Context::raise_error)?; +# +# let status_code = response.status(); +# +# if !status_code.is_success() { +# return Err(Context::raise_error(ErrStatusCode { status_code })); +# } +# +# let message_response: ApiMessageResponse = response.json().map_err(Context::raise_error)?; +# +# Ok(message_response.message) +# } +# } +# +# pub struct UseStringAuthToken; +# +# impl ProvideAuthTokenType for UseStringAuthToken { +# type AuthToken = String; +# } +# +# pub struct UseU64MessageId; +# +# impl ProvideMessageIdType for UseU64MessageId { +# type MessageId = u64; +# } +# +# pub struct UseStringMessage; +# +# impl ProvideMessageType for UseStringMessage { +# type Message = String; +# } +# +# impl AuthTokenGetter for UseField +# where +# Context: HasAuthTokenType + HasField, +# { +# fn auth_token(context: &Context) -> &Context::AuthToken { +# context.get_field(PhantomData) +# } +# } +# +# pub struct UseProductionApiUrl; +# +# impl ApiBaseUrlGetter for UseProductionApiUrl { +# fn api_base_url(_context: &Context) -> &String { +# static BASE_URL: OnceLock = OnceLock::new(); +# +# BASE_URL.get_or_init(|| "https://api.example.com".into()) +# } +# } +# +#[derive(HasField)] +pub struct ApiClient { + pub auth_token: String, +} + +pub struct ApiClientComponents; + +# pub struct RaiseApiErrors; +# +impl HasComponents for ApiClient { + type Components = ApiClientComponents; +} + +delegate_components! { + ApiClientComponents { + ErrorTypeComponent: UseAnyhowError, + ErrorRaiserComponent: UseDelegate, + MessageIdTypeComponent: UseU64MessageId, + MessageTypeComponent: UseStringMessage, + AuthTokenTypeComponent: UseStringAuthToken, + ApiBaseUrlGetterComponent: UseProductionApiUrl, + AuthTokenGetterComponent: UseField, + MessageQuerierComponent: ReadMessageFromApi, + } +} +# +# delegate_components! { +# RaiseApiErrors { +# reqwest::Error: RaiseFrom, +# ErrStatusCode: DebugAnyhowError, +# } +# } +# +# pub trait CanUseApiClient: CanQueryMessage {} +# +# impl CanUseApiClient for ApiClient {} +``` + +Inside the component wiring, we choose `UseProductionApiUrl` to be the provider +for `ApiBaseUrlGetterComponent`. +Notice that now the `ApiClient` context no longer contain any `api_base_url` field. + +The use of static accessors can be useful to implement specialized contexts +that keep the values constant for certain fields. +With this approach, the constant values no longer needs to be passed around +as part of the context during runtime, and we no longer need to worry +about keeping the field private or preventing the wrong value being assigned +at runtime. +Thanks to the compile-time wiring, we may even get some performance advantage +as compared to passing around dynamic values at runtime. + +## Auto Accessor Traits + +The need to define and wire up many CGP components may overwhelm a developer who +is new to CGP. +At least during the beginning phase, a project don't usually that much flexibility +in customizing how fields are accessed. +As such, some may consider the full use of field accessors introduced in this chapter +being unnecessarily complicated. + +One intermediate way to simplify use of accessor traits is to define them _not_ +as CGP components, but as regular Rust traits with blanket implementations that +use `HasField`. For example, we can re-define the `HasApiUrl` trait as follows: + +```rust +# extern crate cgp; +# +# use core::marker::PhantomData; +# +# use cgp::prelude::*; +# +pub trait HasApiBaseUrl { + fn api_base_url(&self) -> &String; +} + +impl HasApiBaseUrl for Context +where + Context: HasField, +{ + fn api_base_url(&self) -> &String { + self.get_field(PhantomData) + } +} +``` + +This way, the `HasApiBaseUrl` will always be implemented for any context +that derive `HasField` and have the relevant field, and +there is no need to have explicit wiring of `ApiBaseUrlGetterComponent` +inside the wiring of the context components. + +With this, providers like `ReadMessageFromApi` can still use traits like `HasApiBaseUrl` +to simplify the access of fields. And the context implementors can just use +`#[derive(HasField)]` without having to worry about the wiring. + +The main downside of this approach is that the context cannot easily override the +implementation of `HaswApiBaseUrl`, unless they don't implement `HasField` at all. +Nevertheless, it will be straightforward to refactor the trait in the future +to turn it into a full CGP component. + +As a result, this may be an appealing option for readers who want to have a simpler +experience of using CGP and not use its full power. + +## Conclusion + +In this chapter, we have learned about different ways to define accessor traits, +and to implement the accessor providers. The use of a derivable `HasField` trait +makes it possible to implement context-generic accessor providers without +requiring direct access to the concrete context. The use of the `UseField` pattern +unifies the convention of implementing field accessors, and allows contexts +to choose different field names for the accessors. + +As we will see in later chapters, the use of context-generic accessor providers +make it possible to implement almost everything as context-generic providers, +and leaving almost no code tied to specific concrete contexts. \ No newline at end of file From 797be196f4bb8c90e59cef2f72c32e963646e22b Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 21:17:17 +0000 Subject: [PATCH 09/11] AI-revising field accessors chapter --- content/field-accessors.md | 160 ++++++-------------------- content/generic-accessor-providers.md | 126 +++++--------------- 2 files changed, 64 insertions(+), 222 deletions(-) diff --git a/content/field-accessors.md b/content/field-accessors.md index f5c507b..9e9550b 100644 --- a/content/field-accessors.md +++ b/content/field-accessors.md @@ -1,18 +1,10 @@ # Field Accessors -Using impl-side dependencies, CGP provides a way to inject dependencies into providers -without polluting the public interfaces with additional constraints. A common use of -dependency injection is for the provider to retrieve some values from the context. -More commonly, we call this pattern field _accessor_ or _getter_, since we are getting -or accessing field values from the context. -In this chapter, we will walk through how to effectively define and use field accessors -with CGP. +With impl-side dependencies, CGP offers a way to inject dependencies into providers without cluttering the public interfaces with extra constraints. One common use of this dependency injection is for a provider to retrieve values from the context. This pattern is often referred to as a field _accessor_ or _getter_, since it involves accessing field values from the context. In this chapter, we'll explore how to define and use field accessors effectively with CGP. ## Example: API Call -Supposed that our application needs to make API calls to an external services to read -messages by message ID. To abstract away the details of the API call, we would define -CGP traits such as follows: +Suppose our application needs to make API calls to an external service to read messages by their message ID. To abstract away the details of the API call, we can define CGP traits as follows: ```rust # extern crate cgp; @@ -43,15 +35,9 @@ pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { } ``` -Following the patterns for [associated types](./associated-types.md), we define the -type traits `HasMessageIdType` and `HasMessageType` to abstract away the detailed structures -of the message ID and messages. -Following the patterns for [error handling](./error-handling.md), we define the -`CanQueryMessage` trait to accept an abstract `MessageId` value, and return -either an abstract `Message` or an abstract `Error`. +Following the patterns for [associated types](./associated-types.md), we define the type traits `HasMessageIdType` and `HasMessageType` to abstract away the details of the message ID and message structures. Additionally, the `CanQueryMessage` trait accepts an abstract `MessageId` and returns either an abstract `Message` or an abstract `Error`, following the patterns for [error handling](./error-handling.md). -With the interfaces defined, we will then try and implement a naive API client provider -that queries the message as HTTP request: +With the interfaces defined, we now implement a simple API client provider that queries the message via an HTTP request. ```rust # extern crate cgp; @@ -126,34 +112,19 @@ where } ``` -For the purpose of the examples here, we will use the [`reqwest`](https://docs.rs/reqwest) -library to make the HTTP calls. We will also use the _blocking_ version of the APIs -in this chapter, as we will only cover about doing asynchronous programming in CGP in -later chapters. - -In the above example, we implement `MessageQuerier` for the provider `ReadMessageFromApi`. -For simplicity, we require the additional constraint that `MessageId` needs to be -`u64`, and the `Message` type is just a simple `String`. -We also make use of the context to raise the `reqwest::Error` returned from calling -`reqwest` methods, and also a custom `ErrStatusCode` error in case if the server -returns error HTTP response. - -Inside the method body, we first build a reqwest `Client`, and then use it to issue -a HTTP GET request to the URL `"http://localhost:8000/api/messages/{message_id}"`. -If the returned HTTP status is not successful, we raise the error `ErrStatusCode`. -Otherwise, we parse the response body as JSON using the `ApiMessageResponse` struct, -which expects the response body to contain a `message` string field. - -We may quickly notice that the naive provider has several things hard coded. -For start, it has the hardcoded API base URL `http://localhost:8000`, which should -be made configurable. We will next walk through how to define _accessor_ traits -to access these configurable values from the context. +For the purposes of the examples in this chapter, we will use the [`reqwest`](https://docs.rs/reqwest) library to make HTTP calls. We will also use the _blocking_ version of the API in this chapter, as asynchronous programming in CGP will be covered in later chapters. + +In the example above, we implement `MessageQuerier` for the `ReadMessageFromApi` provider. For simplicity, we add the constraint that `MessageId` must be of type `u64` and the `Message` type is a basic `String`. + +We also use the context to handle errors. Specifically, we raise the `reqwest::Error` returned by the `reqwest` methods, as well as a custom `ErrStatusCode` error if the server responds with an error HTTP status. + +Within the method, we first create a `reqwest::Client`, and then use it to send an HTTP GET request to the URL `"http://localhost:8000/api/messages/{message_id}"`. If the returned HTTP status is unsuccessful, we raise the `ErrStatusCode`. Otherwise, we parse the response body as JSON into the `ApiMessageResponse` struct, which expects the response to contain a `message` field. + +It's clear that the naive provider has some hard-coded values. For instance, the API base URL `http://localhost:8000` is fixed, but it should be configurable. In the next section, we will explore how to define _accessor_ traits to retrieve these configurable values from the context. ## Getting the Base API URL -Using CGP, it is pretty straightforward to define an accessor trait for getting -values from the context. To make the base API URL configurable, we would define -a `HasApiBaseUrl` trait as follows: +In CGP, defining an accessor trait to retrieve values from the context is straightforward. To make the base API URL configurable, we define a `HasApiBaseUrl` trait as follows: ```rust # extern crate cgp; @@ -168,13 +139,9 @@ pub trait HasApiBaseUrl { } ``` -The trait `HasApiBaseUrl` provides a method `api_base_url`, which returns a `&String` -from the context. In production applications, we may want the method to return a -[`Url`](https://docs.rs/url/latest/url/struct.Url.html), or even an abstract `Url` type. -But we will use strings here to keep the example simple. +The `HasApiBaseUrl` trait defines a method, `api_base_url`, which returns a reference to a `String` from the context. In production applications, you might prefer to return a [`url::Url`](https://docs.rs/url/latest/url/struct.Url.html) or even an abstract `Url` type instead of a `String`. However, for simplicity, we use a `String` in this example. -We can then include `HasApiBaseUrl` inside `ReadMessageFromApi`, so that we can -construct the HTTP request using the base API URL provided by the context: +Next, we can include the `HasApiBaseUrl` trait within `ReadMessageFromApi`, allowing us to construct the HTTP request using the base API URL provided by the context: ```rust # extern crate cgp; @@ -258,12 +225,9 @@ where ## Getting the Auth Token -Aside from the base API URL, it is common for API services to require some kind of authentication -to protect the API resource from being accessed by unauthorized party. -For the purpose of this example, we will make use of simple _bearer tokens_ to access the API. +In addition to the base API URL, many API services require authentication to protect their resources from unauthorized access. For this example, we’ll use simple _bearer tokens_ for API access. -Similar to `HasApiBaseUrl`, we will define a `HasAuthToken` getter to get the -auth token as follows: +Just as we did with `HasApiBaseUrl`, we can define a `HasAuthToken` trait to retrieve the authentication token as follows: ```rust # extern crate cgp; @@ -286,15 +250,9 @@ pub trait HasAuthToken: HasAuthTokenType { } ``` -Similar to the [earlier chapter](./associated-types.md), we first define `HasAuthTokenType` -to keep the `AuthToken` type abstract. In fact, the same `HasAuthTokenType` trait -and their respective providers could be reused across the chapters. This also shows -that having minimal CGP traits make it easier to reuse the same interface across -different applications. +Similar to the pattern used in the [earlier chapter](./associated-types.md), we first define `HasAuthTokenType` to keep the `AuthToken` type abstract. In fact, this `HasAuthTokenType` trait and its associated providers can be reused across different chapters or applications. This demonstrates how minimal CGP traits facilitate the reuse of interfaces in multiple contexts. -We then define a getter trait `HasAuthToken`, to get an abstract `AuthToken` value from -the context. We can then update `ReadMessageFromApi` to include the auth token -inside the `Authorization` HTTP header: +Next, we define a getter trait, `HasAuthToken`, which provides access to an abstract `AuthToken` value from the context. With this in place, we can now update `ReadMessageFromApi` to include the authentication token in the `Authorization` HTTP header: ```rust # extern crate cgp; @@ -399,17 +357,11 @@ where } ``` -In the updated code, we make use of reqwest's -[`bearer_auth`](https://docs.rs/reqwest/latest/reqwest/blocking/struct.RequestBuilder.html#method.bearer_auth) -method to include the auth token into the HTTP header. -In this case, the provider only require `Context::AuthToken` to implement `Display`, -making it possible to be used with custom `AuthToken` types other than `String`. +In this updated code, we use the [`bearer_auth`](https://docs.rs/reqwest/latest/reqwest/blocking/struct.RequestBuilder.html#method.bearer_auth) method from the `reqwest` library to include the authentication token in the HTTP header. In this case, the provider only requires that `Context::AuthToken` implement the `Display` trait, allowing it to work with custom `AuthToken` types, not limited to `String`. ## Accessor Method Minimalism -Given that it is common for providers like `ReadMessageFromApi` to use both `HasApiBaseUrl` and -`HasAuthToken` together, it may be tempting to merge the two traits and define a single trait -that contains both accessor methods: +When creating providers like `ReadMessageFromApi`, which often need to use both `HasApiBaseUrl` and `HasAuthToken`, it might seem tempting to combine these two traits into a single one, containing both accessor methods: ```rust # extern crate cgp; @@ -434,53 +386,19 @@ pub trait HasApiClientFields: HasAuthTokenType { } ``` -Although this approach also works, it introduces unnecessary coupling between -the `api_base_url` field and the `auth_token` field. -If a provider only needs `api_base_url` but not `auth_token`, it would still -have to include the dependencies that it don't need. -Similarly, we can no longer implement separate providers for `ApiClientFieldsGetter` -to separately provide the fields `api_base_url` and `auth_token` in different ways. - -The coupling of unrelated fields also makes it more challenging to evolve the -application in the future. For example, if we switch to a different authentication -method like public key cryptography, we now need to remove the `auth_token` -method and replace it with a different method, which would affect all code -that depend on `HasApiClientFields`. On the other hand, it is much simpler -to add an additional getter trait, and gradually deprecate and transition -providers to use the new trait while still keeping the old trait around. - -As an application grows more complex, it would also be common to require -dozens of accessor methods, which would make a trait like `HasApiClientFields` -quickly become the bottleneck, and making it difficult for the application -to further evolve. In general, it is not possible to know up front which -of the accessor methods are related, and it can be a distraction to -attempt to make up theories of why it "makes sense" to group accessor -methods in certain ways. - -With the experience of using CGP in real world applications, we find that -one accessor method per accessor trait is the most effective way to -quickly iterate on the application implementation. -This makes it easy to add or remove accessor methods, and it removes a lot of -cognitive overload on having to think, decide and debate about which trait -an accessor method should belong or not belong to. -With the passage of time, it is almost inevitable that an accessor trait -that contains multiple accessor methods will need to be broken up, -because some of the accessor methods are no longer applicable to some -part of the application. - -As we will see in later sections and chapters, breaking the accessor methods -down to individual traits also allows us to introduce new design patterns -that can work when the trait contains only one accessor method. - -Nevertheless, CGP does not prevent developers to define accessor traits that contain -multiple types and accessor methods. -In terms of comfort, it would also make sense for developers who are new to CGP -to want to define non-minimal traits, since it has been in the mainstream -programming practices for decades. -As a result, readers are encourage to feel free to experiment around, and -include as many types and methods in a CGP trait as they prefer. - -As an alternative to defining multiple accessor methods, you may also consider defining an inner struct that contains all the common fields that you might want to use with most of your providers: +While this approach works, it introduces unnecessary coupling between the `api_base_url` and `auth_token` fields. If a provider only requires `api_base_url` but not `auth_token`, it would still need to include the unnecessary `auth_token` dependency. Additionally, this design prevents us from implementing separate providers that could provide the `api_base_url` and `auth_token` fields independently, each with its own logic. + +This coupling also makes future changes more challenging. For example, if we switch to a different authentication method, like public key cryptography, we would need to remove the auth_token method and replace it with a new one. This change would affect all code dependent on `HasApiClientFields`. Instead, it's much easier to add a new getter trait and gradually transition providers to the new trait while keeping the old one intact. + +As applications grow in complexity, it’s common to need many accessor methods. A trait like `HasApiClientFields`, with dozens of methods, could quickly become a bottleneck, making the application harder to evolve. Moreover, it's often unclear upfront which accessor methods are related, and trying to theorize about logical groupings can be a distraction. + +From real-world experience using CGP, we’ve found that defining one accessor method per trait is the most effective approach for rapidly iterating on application development. This method simplifies the process of adding or removing accessor methods and reduces cognitive overload, as developers don’t need to spend time deciding or debating which method should belong to which trait. Over time, it's almost inevitable that a multi-method accessor trait will need to be broken up as some methods become irrelevant to parts of the application. + +In future chapters, we’ll explore how breaking accessor methods down into individual traits can enable new design patterns that work well with single-method traits. + +However, CGP doesn’t prevent developers from creating accessor traits with multiple methods and types. For those new to CGP, it might feel more comfortable to define non-minimal traits, as this has been a mainstream practice in programming for decades. So, feel free to experiment and include as many types and methods in a CGP trait as you prefer. + +As an alternative to defining multiple accessor methods, you could define an inner struct containing all the common fields you’ll use across most providers: ```rust # extern crate cgp; @@ -500,13 +418,11 @@ pub trait HasApiClientFields { } ``` -In the example above, we define an `ApiClientFields` struct that contains both `api_base_url` and `auth_token` fields. With that, we can redefine the `HasApiClientFields` trait to have only one getter method which returns `ApiClientFields`. +In this example, we define an `ApiClientFields` struct that groups both the `api_base_url` and `auth_token` fields. The `HasApiClientFields` trait now only needs one getter method, returning the `ApiClientFields` struct. -Note that a downside of this approach is that we can no longer make use of any abstract type inside the struct. As shown, the `ApiClientFields` field stores the `auth_token` as a concrete `String`, rather than an abstract `AuthToken` type. Because of this, this approach may only work if your providers make no use of fields made of abstract types. +One downside to this approach is that we can no longer use abstract types within the struct. For instance, the `ApiClientFields` struct stores the `auth_token` as a concrete `String` rather than as an abstract `AuthToken` type. As a result, this approach works best when your providers don’t rely on abstract types for their fields. -For the purpose of this book, we will continue to make use -of minimal traits, since the book serves as reference materials that should -encourage best practices to its readers. +For the purposes of this book, we will continue to use minimal traits, as this encourages best practices and provides readers with a clear reference for idiomatic CGP usage. ## Implementing Accessor Providers diff --git a/content/generic-accessor-providers.md b/content/generic-accessor-providers.md index 1d76169..8bc92ee 100644 --- a/content/generic-accessor-providers.md +++ b/content/generic-accessor-providers.md @@ -1,15 +1,8 @@ # Context-Generic Accessor Providers -Although the previous accessor implementation for `ApiClient` works, we have to have explicit and -concrete access to the `ApiClient` context in order to implement the accessors. -While this is not too bad with only two accessor methods, it can quickly become tedious once -the application grows, and we need to implement many accessors across many contexts. -It would be more efficient if we can implement _context-generic_ providers for field accessors, -and then use them for any context that contains a given field. +While the previous accessor implementation for `ApiClient` works, it requires explicit and concrete access to the `ApiClient` context to implement the accessors. While this approach is manageable with only a couple of accessor methods, it can quickly become cumbersome as the application grows and requires numerous accessors across multiple contexts. A more efficient approach would be to implement _context-generic_ providers for field accessors, allowing us to reuse them across any context that contains the relevant field. -To make the implementation of context-generic accessors possible, the `cgp` crate offers a derivable -`HasField` trait that can be used as a proxy to access the fields in a concrete context. -The trait is defined as follows: +To enable the implementation of context-generic accessors, the `cgp` crate provides a derivable `HasField` trait. This trait acts as a _proxy_, allowing access to fields in a concrete context. The trait is defined as follows: ```rust # use core::marker::PhantomData; @@ -21,17 +14,9 @@ pub trait HasField { } ``` -For each of the field inside a concrete context, we can implement a `HasField` instance -with the `Tag` type representing the field _name_, and the associated type `Value` -representing the field _type_. -There is also a `get_field` method, which gets a reference of the field value from -the context. The `get_field` method accepts an additional `tag` parameter, -which is just a `PhantomData` with the field name `Tag` as the type. -This phantom parameter is mainly used to help type inference in Rust, -as otherwise Rust would not be able to infer which field `Tag` we are trying to access. +For each field within a concrete context, we can implement a `HasField` instance by associating a `Tag` type with the field's _name_ and an associated type `Value` representing the field's _type_. Additionally, the `HasField` trait includes a `get_field` method, which retrieves a reference to the field value from the context. The `get_field` method accepts an additional `tag` parameter, which is a `PhantomData` type parameter tied to the field's name `Tag`. This phantom parameter helps with type inference in Rust, as without it, Rust would not be able to deduce which field associated with `Tag` is being accessed. -We can automatically derive `HasField` instances for a context like `ApiClient` -by using the derive macro as follows: +We can automatically derive `HasField` instances for a context like `ApiClient` using the derive macro, as shown below: ```rust # extern crate cgp; @@ -45,7 +30,7 @@ pub struct ApiClient { } ``` -The derive macro would then generate the following `HasField` instances for +The derive macro would then generate the corresponding `HasField` instances for `ApiClient`: ```rust @@ -78,41 +63,23 @@ impl HasField for ApiClient { ## Symbols -In the derived `HasField` instances, we can see the use of `symbol!("api_base_url")` -and `symbol!("auth_token")` at the position of the `Tag` generic type. -Recall that a string like `"api_base_url"` is a _value_ of type `&str`, -but we want to use the string as a _type_. -To do that, we use the `symbol!` macro to "lift" a string value into a unique -type, so that we get a _type_ that uniquely identifies the string `"api_base_url"`. -Basically, this means that if the string content in two different uses of `symbol!` -are the same, then they would be treated as the same type. +In the derived `HasField` instances, we observe the use of `symbol!("api_base_url")` and `symbol!("auth_token")` for the `Tag` generic type. While a string like `"api_base_url"` is a value of type `&str`, we need to use it as a _type_ within the `Tag` parameter. To achieve this, we use the `symbol!` macro to "lift" a string value into a unique type, which allows us to treat the string `"api_base_url"` as a _type_. Essentially, this means that if the string content is the same across two uses of `symbol!`, the types will be treated as equivalent. -Behind the scene, `symbol!` first use the `Char` type to "lift" individual characters -into types. The `Char` type is defined as follows: +Behind the scenes, the `symbol!` macro first uses the `Char` type to "lift" individual characters into types. The `Char` type is defined as follows: ```rust pub struct Char; ``` -We make use of the [_const generics_](https://blog.rust-lang.org/2021/02/26/const-generics-mvp-beta.html) -feature in Rust to parameterize `Char` with a constant `CHAR` of type `char`. -The `Char` struct itself has an empty body, because we only want to use it like -a `char` at the type level. +This makes use of Rust's [_const generics_](https://blog.rust-lang.org/2021/02/26/const-generics-mvp-beta.html) feature to parameterize `Char` with a constant `CHAR` of type `char`. The `Char` struct itself is empty, as we only use it for type-level manipulation. -Note that although we can use const generics to lift individual characters, we can't -yet use a type like `String` or `&str` inside const generics. -So until we can use strings inside const generics, we need a different workaround -to lift strings into types. - -We workaround that by constructing a _type-level list_ of characters. So a type like -`symbol!("abc")` would be desugared to something like: +Although we can use const generics to lift individual characters, we currently cannot use a type like `String` or `&str` within const generics. As a workaround, we construct a _type-level list_ of characters. For example, `symbol!("abc")` is desugared to a type-level list of characters like: ```rust,ignore (Char<'a'>, (Char<'b'>, (Char<'c'>, ()))) ``` -In `cgp`, instead of using the native Rust tuple, we define the `Cons` and `Nil` -types to help identifying type level lists: +In `cgp`, instead of using Rust’s native tuple, we define the `Cons` and `Nil` types to represent type-level lists: ```rust pub struct Nil; @@ -120,32 +87,21 @@ pub struct Nil; pub struct Cons(pub Head, pub Tail); ``` -Similar to the linked list concepts in Lisp, the `Nil` type is used to represent -an empty type-level list, and the `Cons` type is used to "add" an element to the -front of the type-level list. +The `Nil` type represents an empty type-level list, while `Cons` is used to prepend an element to the front of the list, similar to how linked lists work in Lisp. -With that, the actual desugaring of a type like `symbol!("abc")` looks like follows: +Thus, the actual desugaring of `symbol!("abc")` looks like this: ```rust,ignore Cons, Cons, Cons, Nil>>> ``` -Although the type make look complicated, it has a pretty compact representation from the -perspective of the Rust compiler. And since we never construct a value out of the symbol -type at runtime, we don't need to worry about any runtime overhead on using symbol types. -Aside from that, since we will mostly only use `HasField` to implement context-generic -accessors, there is negligible compile-time overhead of using `HasField` inside large -codebases. +While this type may seem complex, it has a compact representation from the perspective of the Rust compiler. Furthermore, since we don’t construct values from symbol types at runtime, there is no runtime overhead associated with them. The use of `HasField` to implement context-generic accessors introduces negligible compile-time overhead, even in large codebases. -It is also worth noting that the current representation of symbols is a temporary -workaround. Once Rust supports the use of strings inside const generics, we can -migrate the desugaring of `symbol!` to make use of that to simplify the type -representation. +It’s important to note that the current representation of symbols is a temporary workaround. Once Rust supports using strings in const generics, we can simplify the desugaring process and adjust our implementation accordingly. ## Using `HasField` in Accessor Providers -Using `HasField`, we can then implement a context-generic provider for `ApiUrlGetter` -like follows: +With `HasField`, we can implement context-generic providers like `ApiUrlGetter`. Here's an example: ```rust # extern crate cgp; @@ -173,10 +129,7 @@ where } ``` -The provider `GetApiUrl` is implemented for any `Context` type that implements -`HasField`. This means that as long as the -context uses `#[derive(HasField)]` has an `api_url` field with `String` type, -then we can use `GetApiUrl` with it. +In this implementation, `GetApiUrl` is defined for any `Context` type that implements `HasField`. This means that as long as the context uses `#[derive(HasField)]`, and has a field named `api_url` of type `String`, the `GetApiUrl` provider can be used with it. Similarly, we can implement a context-generic provider for `AuthTokenGetter` as follows: @@ -214,31 +167,15 @@ where } ``` -The provider `GetAuthToken` is slightly more complicated, because the `auth_token()` method -returns an abstract `Context::AuthToken` type. -To work with that, we first need `Context` to implement `HasAuthTokenType`, and then -require the `Value` associated type to be the same as `Context::AuthToken`. -This means that `GetAuthToken` can be used with a context, if it uses -`#[derive(HasField)]` and has an `auth_token` field with the same type as -the `AuthToken` type that it implements. +The `GetAuthToken` provider is slightly more complex since the `auth_token` method returns an abstract `Context::AuthToken` type. To handle this, we require the `Context` to implement `HasAuthTokenType` and for the `Value` associated type to match `Context::AuthToken`. This ensures that `GetAuthToken` can be used with any context that has an `auth_token` field of the same type as the `AuthToken` defined in `HasAuthTokenType`. ## The `UseField` Pattern -In the previous section, we managed to implement the context-generic accessor providers -`GetApiUrl` and `GetAuthToken`, without access to the concrete context. However, the field names -`api_url` and `auth_token` are hardcoded into the provider implementation. This means that -a concrete context cannot choose different _field names_ for the specific fields, unless -they manually re-implement the accessors. +In the previous section, we were able to implement context-generic accessor providers like `GetApiUrl` and `GetAuthToken` without directly referencing the concrete context. However, the field names, such as `api_url` and `auth_token`, were hardcoded into the provider implementation. This means that a concrete context cannot choose different _field names_ for these specific fields unless it manually re-implements the accessors. -There may be different reasons why a context may want to use different names to store the -field values. For example, there could be two independent accessor providers that happen -to choose the same field name for different types. A context may also have multiple similar -fields that serve similar purposes but with slightly different names. -Whatever the reason is, it would be nice if we can allow the contexts to customize the -field names, instead of letting the providers to pick fixed field names. +There are various reasons why a context might want to use different names for the field values. For instance, two independent accessor providers might choose the same field name for different types, or a context might have multiple similar fields with slightly different names. In these cases, it would be beneficial to allow the context to customize the field names instead of having the providers pick fixed field names. -For this purpose, the `cgp` crate provides the `UseField` type that we can use to -implement accessor providers: +To address this, the `cgp` crate provides the `UseField` type, which we can leverage to implement flexible accessor providers: ```rust # use core::marker::PhantomData; @@ -246,10 +183,7 @@ implement accessor providers: pub struct UseField(pub PhantomData); ``` -Similar to the [`UseDelegate` pattern](./delegated-error-raiser.md), the `UseField` type -is used as a label for accessor implementations following the `UseField` pattern. -Using `UseField`, we can implement the providers as follows: - +Similar to the [`UseDelegate` pattern](./delegated-error-raiser.md), the `UseField` type acts as a marker for accessor implementations that follow the UseField pattern. Using `UseField`, we can define the providers as follows: ```rust # extern crate cgp; @@ -300,16 +234,11 @@ where } ``` -Compared to the explicit providers `GetApiUrl` and `GetAuthToken`, we implement -the traits `ApiBaseUrlGetter` and `AuthTokenGetter` directly on the `UseField` -type provided by the `cgp` crate. -The implementation is also parameterized by an additional `Tag` type, to represent -the name of the field we want to use. -We can see that the implementation is almost the same as before, except that -we no longer use `symbol!` to directly refer to the field names. +In contrast to the explicit providers `GetApiUrl` and `GetAuthToken`, we now implement the `ApiBaseUrlGetter` and `AuthTokenGetter` traits directly on the `UseField` type provided by the `cgp` crate. The implementation is parameterized by an additional `Tag` type, which represents the field name we want to access. + +The structure of the implementation is almost the same as before, but instead of using `symbol!` to directly reference the field names, we rely on the `Tag` type to abstract the field names. -Using `UseField`, we get to simplify the implementation of `ApiClient` and -wire up the accessor components directly inside `delegate_components!`: +By using `UseField`, we can simplify the implementation of `ApiClient` and wire up the accessor components directly within `delegate_components!`: ```rust # extern crate cgp; @@ -495,10 +424,7 @@ delegate_components! { # impl CanUseApiClient for ApiClient {} ``` -The wiring above uses `UseField` to implement `ApiBaseUrlGetterComponent`, -and `UseField` to implement `AuthTokenGetterComponent`. -With the field names specified explicitly in the wiring, we can easily change the field -names in the `ApiClient` context, and update the wiring accordingly. +In this wiring example, `UseField` is used to implement the `ApiBaseUrlGetterComponent`, and `UseField` is used for the `AuthTokenGetterComponent`. By explicitly specifying the field names in the wiring, we can easily change the field names in the `ApiClient` context and update the wiring accordingly. ## Using `HasField` Directly Inside Providers From 36b208d983446d3964945fb1ef29415943875b09 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 21:24:08 +0000 Subject: [PATCH 10/11] Reorder field accessor sections --- content/SUMMARY.md | 1 + content/generic-accessor-providers.md | 471 ++++++-------------------- content/use-field-pattern.md | 256 ++++++++++++++ 3 files changed, 365 insertions(+), 363 deletions(-) create mode 100644 content/use-field-pattern.md diff --git a/content/SUMMARY.md b/content/SUMMARY.md index 9d3108f..6a01cb9 100644 --- a/content/SUMMARY.md +++ b/content/SUMMARY.md @@ -27,6 +27,7 @@ - [Error Wrapping](error-wrapping.md) - [Field Accessors](field-accessors.md) - [Generic Accessor Providers](generic-accessor-providers.md) + - [The `UseField` Pattern](use-field-pattern.md) - [Component Presets]() - [Trait-Generic Providers]() - [`WithProvider`]() diff --git a/content/generic-accessor-providers.md b/content/generic-accessor-providers.md index 8bc92ee..ab0d6ea 100644 --- a/content/generic-accessor-providers.md +++ b/content/generic-accessor-providers.md @@ -169,29 +169,75 @@ where The `GetAuthToken` provider is slightly more complex since the `auth_token` method returns an abstract `Context::AuthToken` type. To handle this, we require the `Context` to implement `HasAuthTokenType` and for the `Value` associated type to match `Context::AuthToken`. This ensures that `GetAuthToken` can be used with any context that has an `auth_token` field of the same type as the `AuthToken` defined in `HasAuthTokenType`. -## The `UseField` Pattern - -In the previous section, we were able to implement context-generic accessor providers like `GetApiUrl` and `GetAuthToken` without directly referencing the concrete context. However, the field names, such as `api_url` and `auth_token`, were hardcoded into the provider implementation. This means that a concrete context cannot choose different _field names_ for these specific fields unless it manually re-implements the accessors. +## Auto Accessor Traits -There are various reasons why a context might want to use different names for the field values. For instance, two independent accessor providers might choose the same field name for different types, or a context might have multiple similar fields with slightly different names. In these cases, it would be beneficial to allow the context to customize the field names instead of having the providers pick fixed field names. +The need to define and wire up many CGP components may overwhelm a developer who +is new to CGP. +At least during the beginning phase, a project don't usually that much flexibility +in customizing how fields are accessed. +As such, some may consider the full use of field accessors introduced in this chapter +being unnecessarily complicated. -To address this, the `cgp` crate provides the `UseField` type, which we can leverage to implement flexible accessor providers: +One intermediate way to simplify use of accessor traits is to define them _not_ +as CGP components, but as regular Rust traits with blanket implementations that +use `HasField`. For example, we can re-define the `HasApiUrl` trait as follows: ```rust +# extern crate cgp; +# # use core::marker::PhantomData; # -pub struct UseField(pub PhantomData); +# use cgp::prelude::*; +# +pub trait HasApiBaseUrl { + fn api_base_url(&self) -> &String; +} + +impl HasApiBaseUrl for Context +where + Context: HasField, +{ + fn api_base_url(&self) -> &String { + self.get_field(PhantomData) + } +} ``` -Similar to the [`UseDelegate` pattern](./delegated-error-raiser.md), the `UseField` type acts as a marker for accessor implementations that follow the UseField pattern. Using `UseField`, we can define the providers as follows: +This way, the `HasApiBaseUrl` will always be implemented for any context +that derive `HasField` and have the relevant field, and +there is no need to have explicit wiring of `ApiBaseUrlGetterComponent` +inside the wiring of the context components. + +With this, providers like `ReadMessageFromApi` can still use traits like `HasApiBaseUrl` +to simplify the access of fields. And the context implementors can just use +`#[derive(HasField)]` without having to worry about the wiring. + +The main downside of this approach is that the context cannot easily override the +implementation of `HaswApiBaseUrl`, unless they don't implement `HasField` at all. +Nevertheless, it will be straightforward to refactor the trait in the future +to turn it into a full CGP component. + +As a result, this may be an appealing option for readers who want to have a simpler +experience of using CGP and not use its full power. + +## Static Accessors + +One benefit of defining minimal accessor traits is that we get to implement custom +accessor providers that do not necessarily need to read the field values from the context. +For example, we can implement _static accessor_ providers that always return a global +constant value. + +The use of static accessors can be useful when we want to hard code some values for a +specific context. For instance, we may want to define a production `ApiClient` context +that always use a hard-coded API URL: ```rust # extern crate cgp; # # use core::marker::PhantomData; -# +use std::sync::OnceLock; + # use cgp::prelude::*; -# use cgp::core::field::UseField; # # #[cgp_component { # provider: ApiBaseUrlGetter, @@ -200,45 +246,28 @@ Similar to the [`UseDelegate` pattern](./delegated-error-raiser.md), the `UseFie # fn api_base_url(&self) -> &String; # } # -# #[cgp_component { -# name: AuthTokenTypeComponent, -# provider: ProvideAuthTokenType, -# }] -# pub trait HasAuthTokenType { -# type AuthToken; -# } -# -# #[cgp_component { -# provider: AuthTokenGetter, -# }] -# pub trait HasAuthToken: HasAuthTokenType { -# fn auth_token(&self) -> &Self::AuthToken; -# } -# -impl ApiBaseUrlGetter for UseField -where - Context: HasField, -{ - fn api_base_url(context: &Context) -> &String { - context.get_field(PhantomData) - } -} +pub struct UseProductionApiUrl; -impl AuthTokenGetter for UseField -where - Context: HasAuthTokenType + HasField, -{ - fn auth_token(context: &Context) -> &Context::AuthToken { - context.get_field(PhantomData) +impl ApiBaseUrlGetter for UseProductionApiUrl { + fn api_base_url(_context: &Context) -> &String { + static BASE_URL: OnceLock = OnceLock::new(); + + BASE_URL.get_or_init(|| "https://api.example.com".into()) } } ``` -In contrast to the explicit providers `GetApiUrl` and `GetAuthToken`, we now implement the `ApiBaseUrlGetter` and `AuthTokenGetter` traits directly on the `UseField` type provided by the `cgp` crate. The implementation is parameterized by an additional `Tag` type, which represents the field name we want to access. - -The structure of the implementation is almost the same as before, but instead of using `symbol!` to directly reference the field names, we rely on the `Tag` type to abstract the field names. +The provider `UseProductionApiUrl` implements `ApiBaseUrlGetter` for any context type. +Inside the `api_base_url` method, we first define a static `BASE_URL` value with the +type `OnceLock`. The use of [`OnceLock`](https://doc.rust-lang.org/std/sync/struct.OnceLock.html) +allows us to define a global variable in Rust that is initialized exactly once, and +then remain constant throughout the application. +This is mainly useful because constructors like `String::from` are not currently `const fn`, +so we have to make use of `OnceLock::get_or_init` to run the non-const constructor. +By defining the static variable inside the method, we ensure that the variable can only be +accessed and initialized by the provider. -By using `UseField`, we can simplify the implementation of `ApiClient` and wire up the accessor components directly within `delegate_components!`: +Using `UseProductionApiUrl`, we can now define a production `ApiClient` context such as follows: ```rust # extern crate cgp; @@ -248,6 +277,7 @@ By using `UseField`, we can simplify the implementation of `ApiClient` and wire # # use core::fmt::Display; # use core::marker::PhantomData; +# use std::sync::OnceLock; # # use cgp::core::component::UseDelegate; # use cgp::extra::error::RaiseFrom; @@ -367,38 +397,40 @@ By using `UseField`, we can simplify the implementation of `ApiClient` and wire # type Message = String; # } # -# impl ApiBaseUrlGetter for UseField -# where -# Context: HasField, -# { -# fn api_base_url(context: &Context) -> &String { -# context.get_field(PhantomData) +# pub struct UseProductionApiUrl; +# +# impl ApiBaseUrlGetter for UseProductionApiUrl { +# fn api_base_url(_context: &Context) -> &String { +# static BASE_URL: OnceLock = OnceLock::new(); +# +# BASE_URL.get_or_init(|| "https://api.example.com".into()) # } # } # -# impl AuthTokenGetter for UseField +# pub struct GetAuthToken; +# +# impl AuthTokenGetter for GetAuthToken # where -# Context: HasAuthTokenType + HasField, +# Context: HasAuthTokenType + HasField, # { # fn auth_token(context: &Context) -> &Context::AuthToken { # context.get_field(PhantomData) # } # } # -# #[derive(HasField)] -# pub struct ApiClient { -# pub api_base_url: String, -# pub auth_token: String, -# } -# -# pub struct ApiClientComponents; -# +#[derive(HasField)] +pub struct ApiClient { + pub auth_token: String, +} + +pub struct ApiClientComponents; + # pub struct RaiseApiErrors; # -# impl HasComponents for ApiClient { -# type Components = ApiClientComponents; -# } -# +impl HasComponents for ApiClient { + type Components = ApiClientComponents; +} + delegate_components! { ApiClientComponents { ErrorTypeComponent: UseAnyhowError, @@ -406,8 +438,8 @@ delegate_components! { MessageIdTypeComponent: UseU64MessageId, MessageTypeComponent: UseStringMessage, AuthTokenTypeComponent: UseStringAuthToken, - ApiBaseUrlGetterComponent: UseField, - AuthTokenGetterComponent: UseField, + ApiBaseUrlGetterComponent: UseProductionApiUrl, + AuthTokenGetterComponent: GetAuthToken, MessageQuerierComponent: ReadMessageFromApi, } } @@ -424,7 +456,18 @@ delegate_components! { # impl CanUseApiClient for ApiClient {} ``` -In this wiring example, `UseField` is used to implement the `ApiBaseUrlGetterComponent`, and `UseField` is used for the `AuthTokenGetterComponent`. By explicitly specifying the field names in the wiring, we can easily change the field names in the `ApiClient` context and update the wiring accordingly. +Inside the component wiring, we choose `UseProductionApiUrl` to be the provider +for `ApiBaseUrlGetterComponent`. +Notice that now the `ApiClient` context no longer contain any `api_base_url` field. + +The use of static accessors can be useful to implement specialized contexts +that keep the values constant for certain fields. +With this approach, the constant values no longer needs to be passed around +as part of the context during runtime, and we no longer need to worry +about keeping the field private or preventing the wrong value being assigned +at runtime. +Thanks to the compile-time wiring, we may even get some performance advantage +as compared to passing around dynamic values at runtime. ## Using `HasField` Directly Inside Providers @@ -571,304 +614,6 @@ developer needs to keep track of. As a result, we encourage readers to feel free use of `HasField` as they see fit, and then slowly migrate to proper accessor traits when the need arise. -## Static Accessors - -One benefit of defining minimal accessor traits is that we get to implement custom -accessor providers that do not necessarily need to read the field values from the context. -For example, we can implement _static accessor_ providers that always return a global -constant value. - -The use of static accessors can be useful when we want to hard code some values for a -specific context. For instance, we may want to define a production `ApiClient` context -that always use a hard-coded API URL: - -```rust -# extern crate cgp; -# -# use core::marker::PhantomData; -use std::sync::OnceLock; - -# use cgp::prelude::*; -# -# #[cgp_component { -# provider: ApiBaseUrlGetter, -# }] -# pub trait HasApiBaseUrl { -# fn api_base_url(&self) -> &String; -# } -# -pub struct UseProductionApiUrl; - -impl ApiBaseUrlGetter for UseProductionApiUrl { - fn api_base_url(_context: &Context) -> &String { - static BASE_URL: OnceLock = OnceLock::new(); - - BASE_URL.get_or_init(|| "https://api.example.com".into()) - } -} -``` - -The provider `UseProductionApiUrl` implements `ApiBaseUrlGetter` for any context type. -Inside the `api_base_url` method, we first define a static `BASE_URL` value with the -type `OnceLock`. The use of [`OnceLock`](https://doc.rust-lang.org/std/sync/struct.OnceLock.html) -allows us to define a global variable in Rust that is initialized exactly once, and -then remain constant throughout the application. -This is mainly useful because constructors like `String::from` are not currently `const fn`, -so we have to make use of `OnceLock::get_or_init` to run the non-const constructor. -By defining the static variable inside the method, we ensure that the variable can only be -accessed and initialized by the provider. - -Using `UseProductionApiUrl`, we can now define a production `ApiClient` context such as follows: - -```rust -# extern crate cgp; -# extern crate cgp_error_anyhow; -# extern crate reqwest; -# extern crate serde; -# -# use core::fmt::Display; -# use core::marker::PhantomData; -# use std::sync::OnceLock; -# -# use cgp::core::component::UseDelegate; -# use cgp::extra::error::RaiseFrom; -# use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; -# use cgp::core::field::UseField; -# use cgp::prelude::*; -# use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; -# use reqwest::blocking::Client; -# use reqwest::StatusCode; -# use serde::Deserialize; -# -# #[cgp_component { -# name: MessageIdTypeComponent, -# provider: ProvideMessageIdType, -# }] -# pub trait HasMessageIdType { -# type MessageId; -# } -# -# #[cgp_component { -# name: MessageTypeComponent, -# provider: ProvideMessageType, -# }] -# pub trait HasMessageType { -# type Message; -# } -# -# #[cgp_component { -# provider: MessageQuerier, -# }] -# pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { -# fn query_message(&self, message_id: &Self::MessageId) -> Result; -# } -# -# #[cgp_component { -# provider: ApiBaseUrlGetter, -# }] -# pub trait HasApiBaseUrl { -# fn api_base_url(&self) -> &String; -# } -# -# #[cgp_component { -# name: AuthTokenTypeComponent, -# provider: ProvideAuthTokenType, -# }] -# pub trait HasAuthTokenType { -# type AuthToken; -# } -# -# #[cgp_component { -# provider: AuthTokenGetter, -# }] -# pub trait HasAuthToken: HasAuthTokenType { -# fn auth_token(&self) -> &Self::AuthToken; -# } -# -# pub struct ReadMessageFromApi; -# -# #[derive(Debug)] -# pub struct ErrStatusCode { -# pub status_code: StatusCode, -# } -# -# #[derive(Deserialize)] -# pub struct ApiMessageResponse { -# pub message: String, -# } -# -# impl MessageQuerier for ReadMessageFromApi -# where -# Context: HasMessageIdType -# + HasMessageType -# + HasApiBaseUrl -# + HasAuthToken -# + CanRaiseError -# + CanRaiseError, -# Context::AuthToken: Display, -# { -# fn query_message(context: &Context, message_id: &u64) -> Result { -# let client = Client::new(); -# -# let url = format!("{}/api/messages/{}", context.api_base_url(), message_id); -# -# let response = client -# .get(url) -# .bearer_auth(context.auth_token()) -# .send() -# .map_err(Context::raise_error)?; -# -# let status_code = response.status(); -# -# if !status_code.is_success() { -# return Err(Context::raise_error(ErrStatusCode { status_code })); -# } -# -# let message_response: ApiMessageResponse = response.json().map_err(Context::raise_error)?; -# -# Ok(message_response.message) -# } -# } -# -# pub struct UseStringAuthToken; -# -# impl ProvideAuthTokenType for UseStringAuthToken { -# type AuthToken = String; -# } -# -# pub struct UseU64MessageId; -# -# impl ProvideMessageIdType for UseU64MessageId { -# type MessageId = u64; -# } -# -# pub struct UseStringMessage; -# -# impl ProvideMessageType for UseStringMessage { -# type Message = String; -# } -# -# impl AuthTokenGetter for UseField -# where -# Context: HasAuthTokenType + HasField, -# { -# fn auth_token(context: &Context) -> &Context::AuthToken { -# context.get_field(PhantomData) -# } -# } -# -# pub struct UseProductionApiUrl; -# -# impl ApiBaseUrlGetter for UseProductionApiUrl { -# fn api_base_url(_context: &Context) -> &String { -# static BASE_URL: OnceLock = OnceLock::new(); -# -# BASE_URL.get_or_init(|| "https://api.example.com".into()) -# } -# } -# -#[derive(HasField)] -pub struct ApiClient { - pub auth_token: String, -} - -pub struct ApiClientComponents; - -# pub struct RaiseApiErrors; -# -impl HasComponents for ApiClient { - type Components = ApiClientComponents; -} - -delegate_components! { - ApiClientComponents { - ErrorTypeComponent: UseAnyhowError, - ErrorRaiserComponent: UseDelegate, - MessageIdTypeComponent: UseU64MessageId, - MessageTypeComponent: UseStringMessage, - AuthTokenTypeComponent: UseStringAuthToken, - ApiBaseUrlGetterComponent: UseProductionApiUrl, - AuthTokenGetterComponent: UseField, - MessageQuerierComponent: ReadMessageFromApi, - } -} -# -# delegate_components! { -# RaiseApiErrors { -# reqwest::Error: RaiseFrom, -# ErrStatusCode: DebugAnyhowError, -# } -# } -# -# pub trait CanUseApiClient: CanQueryMessage {} -# -# impl CanUseApiClient for ApiClient {} -``` - -Inside the component wiring, we choose `UseProductionApiUrl` to be the provider -for `ApiBaseUrlGetterComponent`. -Notice that now the `ApiClient` context no longer contain any `api_base_url` field. - -The use of static accessors can be useful to implement specialized contexts -that keep the values constant for certain fields. -With this approach, the constant values no longer needs to be passed around -as part of the context during runtime, and we no longer need to worry -about keeping the field private or preventing the wrong value being assigned -at runtime. -Thanks to the compile-time wiring, we may even get some performance advantage -as compared to passing around dynamic values at runtime. - -## Auto Accessor Traits - -The need to define and wire up many CGP components may overwhelm a developer who -is new to CGP. -At least during the beginning phase, a project don't usually that much flexibility -in customizing how fields are accessed. -As such, some may consider the full use of field accessors introduced in this chapter -being unnecessarily complicated. - -One intermediate way to simplify use of accessor traits is to define them _not_ -as CGP components, but as regular Rust traits with blanket implementations that -use `HasField`. For example, we can re-define the `HasApiUrl` trait as follows: - -```rust -# extern crate cgp; -# -# use core::marker::PhantomData; -# -# use cgp::prelude::*; -# -pub trait HasApiBaseUrl { - fn api_base_url(&self) -> &String; -} - -impl HasApiBaseUrl for Context -where - Context: HasField, -{ - fn api_base_url(&self) -> &String { - self.get_field(PhantomData) - } -} -``` - -This way, the `HasApiBaseUrl` will always be implemented for any context -that derive `HasField` and have the relevant field, and -there is no need to have explicit wiring of `ApiBaseUrlGetterComponent` -inside the wiring of the context components. - -With this, providers like `ReadMessageFromApi` can still use traits like `HasApiBaseUrl` -to simplify the access of fields. And the context implementors can just use -`#[derive(HasField)]` without having to worry about the wiring. - -The main downside of this approach is that the context cannot easily override the -implementation of `HaswApiBaseUrl`, unless they don't implement `HasField` at all. -Nevertheless, it will be straightforward to refactor the trait in the future -to turn it into a full CGP component. - -As a result, this may be an appealing option for readers who want to have a simpler -experience of using CGP and not use its full power. - ## Conclusion In this chapter, we have learned about different ways to define accessor traits, diff --git a/content/use-field-pattern.md b/content/use-field-pattern.md new file mode 100644 index 0000000..402d56f --- /dev/null +++ b/content/use-field-pattern.md @@ -0,0 +1,256 @@ +# The `UseField` Pattern + +In the previous section, we were able to implement context-generic accessor providers like `GetApiUrl` and `GetAuthToken` without directly referencing the concrete context. However, the field names, such as `api_url` and `auth_token`, were hardcoded into the provider implementation. This means that a concrete context cannot choose different _field names_ for these specific fields unless it manually re-implements the accessors. + +There are various reasons why a context might want to use different names for the field values. For instance, two independent accessor providers might choose the same field name for different types, or a context might have multiple similar fields with slightly different names. In these cases, it would be beneficial to allow the context to customize the field names instead of having the providers pick fixed field names. + +To address this, the `cgp` crate provides the `UseField` type, which we can leverage to implement flexible accessor providers: + +```rust +# use core::marker::PhantomData; +# +pub struct UseField(pub PhantomData); +``` + +Similar to the [`UseDelegate` pattern](./delegated-error-raiser.md), the `UseField` type acts as a marker for accessor implementations that follow the UseField pattern. Using `UseField`, we can define the providers as follows: + +```rust +# extern crate cgp; +# +# use core::marker::PhantomData; +# +# use cgp::prelude::*; +# use cgp::core::field::UseField; +# +# #[cgp_component { +# provider: ApiBaseUrlGetter, +# }] +# pub trait HasApiBaseUrl { +# fn api_base_url(&self) -> &String; +# } +# +# #[cgp_component { +# name: AuthTokenTypeComponent, +# provider: ProvideAuthTokenType, +# }] +# pub trait HasAuthTokenType { +# type AuthToken; +# } +# +# #[cgp_component { +# provider: AuthTokenGetter, +# }] +# pub trait HasAuthToken: HasAuthTokenType { +# fn auth_token(&self) -> &Self::AuthToken; +# } +# +impl ApiBaseUrlGetter for UseField +where + Context: HasField, +{ + fn api_base_url(context: &Context) -> &String { + context.get_field(PhantomData) + } +} + +impl AuthTokenGetter for UseField +where + Context: HasAuthTokenType + HasField, +{ + fn auth_token(context: &Context) -> &Context::AuthToken { + context.get_field(PhantomData) + } +} +``` + +In contrast to the explicit providers `GetApiUrl` and `GetAuthToken`, we now implement the `ApiBaseUrlGetter` and `AuthTokenGetter` traits directly on the `UseField` type provided by the `cgp` crate. The implementation is parameterized by an additional `Tag` type, which represents the field name we want to access. + +The structure of the implementation is almost the same as before, but instead of using `symbol!` to directly reference the field names, we rely on the `Tag` type to abstract the field names. + +By using `UseField`, we can simplify the implementation of `ApiClient` and wire up the accessor components directly within `delegate_components!`: + +```rust +# extern crate cgp; +# extern crate cgp_error_anyhow; +# extern crate reqwest; +# extern crate serde; +# +# use core::fmt::Display; +# use core::marker::PhantomData; +# +# use cgp::core::component::UseDelegate; +# use cgp::extra::error::RaiseFrom; +# use cgp::core::error::{ErrorRaiserComponent, ErrorTypeComponent}; +# use cgp::core::field::UseField; +# use cgp::prelude::*; +# use cgp_error_anyhow::{DebugAnyhowError, UseAnyhowError}; +# use reqwest::blocking::Client; +# use reqwest::StatusCode; +# use serde::Deserialize; +# +# #[cgp_component { +# name: MessageIdTypeComponent, +# provider: ProvideMessageIdType, +# }] +# pub trait HasMessageIdType { +# type MessageId; +# } +# +# #[cgp_component { +# name: MessageTypeComponent, +# provider: ProvideMessageType, +# }] +# pub trait HasMessageType { +# type Message; +# } +# +# #[cgp_component { +# provider: MessageQuerier, +# }] +# pub trait CanQueryMessage: HasMessageIdType + HasMessageType + HasErrorType { +# fn query_message(&self, message_id: &Self::MessageId) -> Result; +# } +# +# #[cgp_component { +# provider: ApiBaseUrlGetter, +# }] +# pub trait HasApiBaseUrl { +# fn api_base_url(&self) -> &String; +# } +# +# #[cgp_component { +# name: AuthTokenTypeComponent, +# provider: ProvideAuthTokenType, +# }] +# pub trait HasAuthTokenType { +# type AuthToken; +# } +# +# #[cgp_component { +# provider: AuthTokenGetter, +# }] +# pub trait HasAuthToken: HasAuthTokenType { +# fn auth_token(&self) -> &Self::AuthToken; +# } +# +# pub struct ReadMessageFromApi; +# +# #[derive(Debug)] +# pub struct ErrStatusCode { +# pub status_code: StatusCode, +# } +# +# #[derive(Deserialize)] +# pub struct ApiMessageResponse { +# pub message: String, +# } +# +# impl MessageQuerier for ReadMessageFromApi +# where +# Context: HasMessageIdType +# + HasMessageType +# + HasApiBaseUrl +# + HasAuthToken +# + CanRaiseError +# + CanRaiseError, +# Context::AuthToken: Display, +# { +# fn query_message(context: &Context, message_id: &u64) -> Result { +# let client = Client::new(); +# +# let url = format!("{}/api/messages/{}", context.api_base_url(), message_id); +# +# let response = client +# .get(url) +# .bearer_auth(context.auth_token()) +# .send() +# .map_err(Context::raise_error)?; +# +# let status_code = response.status(); +# +# if !status_code.is_success() { +# return Err(Context::raise_error(ErrStatusCode { status_code })); +# } +# +# let message_response: ApiMessageResponse = response.json().map_err(Context::raise_error)?; +# +# Ok(message_response.message) +# } +# } +# +# pub struct UseStringAuthToken; +# +# impl ProvideAuthTokenType for UseStringAuthToken { +# type AuthToken = String; +# } +# +# pub struct UseU64MessageId; +# +# impl ProvideMessageIdType for UseU64MessageId { +# type MessageId = u64; +# } +# +# pub struct UseStringMessage; +# +# impl ProvideMessageType for UseStringMessage { +# type Message = String; +# } +# +# impl ApiBaseUrlGetter for UseField +# where +# Context: HasField, +# { +# fn api_base_url(context: &Context) -> &String { +# context.get_field(PhantomData) +# } +# } +# +# impl AuthTokenGetter for UseField +# where +# Context: HasAuthTokenType + HasField, +# { +# fn auth_token(context: &Context) -> &Context::AuthToken { +# context.get_field(PhantomData) +# } +# } +# +# #[derive(HasField)] +# pub struct ApiClient { +# pub api_base_url: String, +# pub auth_token: String, +# } +# +# pub struct ApiClientComponents; +# +# pub struct RaiseApiErrors; +# +# impl HasComponents for ApiClient { +# type Components = ApiClientComponents; +# } +# +delegate_components! { + ApiClientComponents { + ErrorTypeComponent: UseAnyhowError, + ErrorRaiserComponent: UseDelegate, + MessageIdTypeComponent: UseU64MessageId, + MessageTypeComponent: UseStringMessage, + AuthTokenTypeComponent: UseStringAuthToken, + ApiBaseUrlGetterComponent: UseField, + AuthTokenGetterComponent: UseField, + MessageQuerierComponent: ReadMessageFromApi, + } +} +# +# delegate_components! { +# RaiseApiErrors { +# reqwest::Error: RaiseFrom, +# ErrStatusCode: DebugAnyhowError, +# } +# } +# +# pub trait CanUseApiClient: CanQueryMessage {} +# +# impl CanUseApiClient for ApiClient {} +``` + +In this wiring example, `UseField` is used to implement the `ApiBaseUrlGetterComponent`, and `UseField` is used for the `AuthTokenGetterComponent`. By explicitly specifying the field names in the wiring, we can easily change the field names in the `ApiClient` context and update the wiring accordingly. From 68e1786c285a70296e843d1d028db9cfb2043861 Mon Sep 17 00:00:00 2001 From: Soares Chen Date: Tue, 7 Jan 2025 21:38:17 +0000 Subject: [PATCH 11/11] AI-revision --- content/generic-accessor-providers.md | 118 ++++++-------------------- content/use-field-pattern.md | 6 ++ 2 files changed, 30 insertions(+), 94 deletions(-) diff --git a/content/generic-accessor-providers.md b/content/generic-accessor-providers.md index ab0d6ea..f8e3c8f 100644 --- a/content/generic-accessor-providers.md +++ b/content/generic-accessor-providers.md @@ -171,16 +171,9 @@ The `GetAuthToken` provider is slightly more complex since the `auth_token` meth ## Auto Accessor Traits -The need to define and wire up many CGP components may overwhelm a developer who -is new to CGP. -At least during the beginning phase, a project don't usually that much flexibility -in customizing how fields are accessed. -As such, some may consider the full use of field accessors introduced in this chapter -being unnecessarily complicated. +The process of defining and wiring many CGP components can be overwhelming for developers who are new to CGP. In the early stages of a project, there is typically not much need for customizing how fields are accessed. As a result, some developers may find the full use of field accessors introduced in this chapter unnecessarily complex. -One intermediate way to simplify use of accessor traits is to define them _not_ -as CGP components, but as regular Rust traits with blanket implementations that -use `HasField`. For example, we can re-define the `HasApiUrl` trait as follows: +To simplify the use of accessor traits, one approach is to define them not as CGP components, but as regular Rust traits with blanket implementations that leverage `HasField`. For example, we can redefine the `HasApiBaseUrl` trait as follows: ```rust # extern crate cgp; @@ -203,33 +196,19 @@ where } ``` -This way, the `HasApiBaseUrl` will always be implemented for any context -that derive `HasField` and have the relevant field, and -there is no need to have explicit wiring of `ApiBaseUrlGetterComponent` -inside the wiring of the context components. +With this approach, the `HasApiBaseUrl` trait will be automatically implemented for any context that derives `HasField` and contains the relevant field. There is no longer need for explicit wiring of the `ApiBaseUrlGetterComponent` within the context components. -With this, providers like `ReadMessageFromApi` can still use traits like `HasApiBaseUrl` -to simplify the access of fields. And the context implementors can just use -`#[derive(HasField)]` without having to worry about the wiring. +This approach allows providers, such as `ReadMessageFromApi`, to still use accessor traits like `HasApiBaseUrl` to simplify field access. Meanwhile, context implementers can simply use `#[derive(HasField)]` without having to worry about manual wiring. -The main downside of this approach is that the context cannot easily override the -implementation of `HaswApiBaseUrl`, unless they don't implement `HasField` at all. -Nevertheless, it will be straightforward to refactor the trait in the future -to turn it into a full CGP component. +The main drawback of this approach is that the context cannot easily override the implementation of `HasApiBaseUrl`, unless it opts not to implement `HasField`. However, it would be straightforward to refactor the trait in the future to convert it into a full CGP component. -As a result, this may be an appealing option for readers who want to have a simpler -experience of using CGP and not use its full power. +Overall, this approach may be an appealing option for developers who want a simpler experience with CGP without fully utilizing its advanced features. ## Static Accessors -One benefit of defining minimal accessor traits is that we get to implement custom -accessor providers that do not necessarily need to read the field values from the context. -For example, we can implement _static accessor_ providers that always return a global -constant value. +One advantage of defining minimal accessor traits is that it allows the implementation of custom accessor providers that do not necessarily read field values from the context. For instance, we can create _static accessor_ providers that always return a global constant value. -The use of static accessors can be useful when we want to hard code some values for a -specific context. For instance, we may want to define a production `ApiClient` context -that always use a hard-coded API URL: +Static accessors are useful when we want to hard-code values for a specific context. For example, we might define a production `ApiClient` context that always uses a fixed API URL: ```rust # extern crate cgp; @@ -257,17 +236,11 @@ impl ApiBaseUrlGetter for UseProductionApiUrl { } ``` -The provider `UseProductionApiUrl` implements `ApiBaseUrlGetter` for any context type. -Inside the `api_base_url` method, we first define a static `BASE_URL` value with the -type `OnceLock`. The use of [`OnceLock`](https://doc.rust-lang.org/std/sync/struct.OnceLock.html) -allows us to define a global variable in Rust that is initialized exactly once, and -then remain constant throughout the application. -This is mainly useful because constructors like `String::from` are not currently `const fn`, -so we have to make use of `OnceLock::get_or_init` to run the non-const constructor. -By defining the static variable inside the method, we ensure that the variable can only be -accessed and initialized by the provider. +In this example, the `UseProductionApiUrl` provider implements `ApiBaseUrlGetter` for any context type. Inside the `api_base_url` method, we define a `static` variable `BASE_URL` using `OnceLock`. This allows us to initialize the global variable exactly once, and it remains constant throughout the application. -Using `UseProductionApiUrl`, we can now define a production `ApiClient` context such as follows: +[`OnceLock`](https://doc.rust-lang.org/std/sync/struct.OnceLock.html) is especially useful since constructors like `String::from` are not `const` fn in Rust. By using `OnceLock::get_or_init`, we can run non-const constructors at runtime while still benefiting from compile-time guarantees. The static variable is scoped within the method, so it is only accessible and initialized by the provider. + +With `UseProductionApiUrl`, we can now define a production `ApiClient` context, as shown below: ```rust # extern crate cgp; @@ -456,25 +429,13 @@ delegate_components! { # impl CanUseApiClient for ApiClient {} ``` -Inside the component wiring, we choose `UseProductionApiUrl` to be the provider -for `ApiBaseUrlGetterComponent`. -Notice that now the `ApiClient` context no longer contain any `api_base_url` field. +In the component wiring, we specify `UseProductionApiUrl` as the provider for `ApiBaseUrlGetterComponent`. Notably, the `ApiClient` context no longer contains the `api_base_url` field. -The use of static accessors can be useful to implement specialized contexts -that keep the values constant for certain fields. -With this approach, the constant values no longer needs to be passed around -as part of the context during runtime, and we no longer need to worry -about keeping the field private or preventing the wrong value being assigned -at runtime. -Thanks to the compile-time wiring, we may even get some performance advantage -as compared to passing around dynamic values at runtime. +Static accessors are particularly useful for implementing specialized contexts where certain fields must remain constant. With this approach, constant values don't need to be passed around as part of the context during runtime, and there's no concern about incorrect values being assigned at runtime. Additionally, because of the compile-time wiring, this method may offer performance benefits compared to passing dynamic values during execution. ## Using `HasField` Directly Inside Providers -Since the `HasField` trait can be automatically derived by contexts, some readers may be -tempted to not define any accessor trait, and instead make use of `HasField` directly -inside the providers. For example, we can in principle remove `HasApiBaseUrl` and -`HasAuthToken`, and re-implement `ReadMessageFromApi` as follows: +Since the `HasField` trait can be automatically derived by contexts, some developers may be tempted to forgo defining accessor traits and instead use `HasField` directly within the providers. For example, one could remove `HasApiBaseUrl` and `HasAuthToken` and implement `ReadMessageFromApi` as follows: ```rust # extern crate cgp; @@ -571,25 +532,13 @@ where } ``` -In the implementation above, the provider `ReadMessageFromApi` requires the context to implement -`HasField` and `HasField`. -To preserve the original behavior, we also have additional constraints that the field `api_base_url` -needs to be of `String` type, and the field `auth_token` needs to have the same type as -`Context::AuthToken`. -When using `get_field`, since there are two instances of `HasField` implemented in scope, -we need to fully qualify the call to specify the field name that we want to access, -such as `context.get_field(PhantomData::)`. - -As we can see, the direct use of `HasField` may not necessary make the code simpler, and instead -require more verbose specification of the fields. The direct use of `HasFields` also requires -explicit specification of what the field types should be. -Whereas in accessor traits like `HasAuthToken`, we can better specify that the method always -return the abstract type `Self::AuthToken`, so one cannot accidentally read from different -fields that happen to have the same underlying concrete type. - -By using `HasField` directly, the provider also makes it less flexible for the context to have -custom ways of getting the field value. For example, instead of putting the `api_url` field -directly in the context, we may want to put it inside another `ApiConfig` struct such as follows: +In the example above, the provider `ReadMessageFromApi` requires the context to implement `HasField` and `HasField`. To preserve the original behavior, we add constraints ensuring that the `api_base_url` field is of type `String` and that the `auth_token` field matches the type of `Context::AuthToken`. + +When using `get_field`, since there are multiple `HasField` instances in scope, we need to fully qualify the field access to specify which field we want to retrieve. For example, we call `context.get_field(PhantomData::)` to access the `api_base_url` field. + +However, while the direct use of `HasField` is possible, it does not necessarily simplify the code. In fact, it often requires more verbose specifications for each field. Additionally, using `HasField` directly necessitates explicitly defining the field types. In contrast, with custom accessor traits like `HasAuthToken`, we can specify that a method returns an abstract type like `Self::AuthToken, which prevents accidental access to fields with the same underlying concrete type. + +Using `HasField` directly also makes the provider less flexible if the context requires custom access methods. For instance, if we wanted to put the `api_base_url` field inside a separate `ApiConfig` struct, we would run into difficulties with `HasField`: ```rust pub struct Config { @@ -604,25 +553,6 @@ pub struct ApiClient { } ``` -In such cases, with an accessor trait like `HasApiUrl`, the context can easily make use of -custom accessor providers to implement such indirect access. But with direct use of -`HasFields`, it would be more tedious to implement the indirect access. - -That said, similar to other shortcut methods, the direct use of `HasField` can be convenient -during initial development, as it helps to significantly reduce the number of traits the -developer needs to keep track of. As a result, we encourage readers to feel free to make -use of `HasField` as they see fit, and then slowly migrate to proper accessor traits -when the need arise. - -## Conclusion - -In this chapter, we have learned about different ways to define accessor traits, -and to implement the accessor providers. The use of a derivable `HasField` trait -makes it possible to implement context-generic accessor providers without -requiring direct access to the concrete context. The use of the `UseField` pattern -unifies the convention of implementing field accessors, and allows contexts -to choose different field names for the accessors. +In this case, an accessor trait like `HasApiUrl` would allow the context to easily use a custom accessor provider. With direct use of `HasField`, however, indirect access would be more cumbersome to implement. -As we will see in later chapters, the use of context-generic accessor providers -make it possible to implement almost everything as context-generic providers, -and leaving almost no code tied to specific concrete contexts. \ No newline at end of file +That said, using `HasField` directly can be convenient during the initial development stages, as it reduces the number of traits a developer needs to manage. Therefore, we encourage readers to use `HasField` where appropriate and gradually migrate to more specific accessor traits when necessary. diff --git a/content/use-field-pattern.md b/content/use-field-pattern.md index 402d56f..70c2f51 100644 --- a/content/use-field-pattern.md +++ b/content/use-field-pattern.md @@ -254,3 +254,9 @@ delegate_components! { ``` In this wiring example, `UseField` is used to implement the `ApiBaseUrlGetterComponent`, and `UseField` is used for the `AuthTokenGetterComponent`. By explicitly specifying the field names in the wiring, we can easily change the field names in the `ApiClient` context and update the wiring accordingly. + +## Conclusion + +In this chapter, we explored various ways to define accessor traits and implement accessor providers. The `HasField` trait, being derivable, offers a way to create context-generic accessor providers without directly accessing the context's concrete fields. The `UseField` pattern standardizes how field accessors are implemented, enabling contexts to customize field names for the accessors. + +As we will see in later chapters, context-generic accessor providers allow us to implement a wide range of functionality without tying code to specific concrete contexts. This approach makes it possible to maintain flexibility and reusability across different contexts.