code: side-feature: exploring string into solution. will probably be moved away
This commit is contained in:
		
							parent
							
								
									7923d0a957
								
							
						
					
					
						commit
						2c088d03ce
					
				
							
								
								
									
										9
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @ -1258,10 +1258,19 @@ dependencies = [ | |||||||
|  "regex", |  "regex", | ||||||
|  "rodio", |  "rodio", | ||||||
|  "serde", |  "serde", | ||||||
|  |  "strinto", | ||||||
|  "tokio", |  "tokio", | ||||||
|  "xxhash-rust", |  "xxhash-rust", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "strinto" | ||||||
|  | version = "0.1.0" | ||||||
|  | dependencies = [ | ||||||
|  |  "quote", | ||||||
|  |  "syn 2.0.26", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "symphonia" | name = "symphonia" | ||||||
| version = "0.5.3" | version = "0.5.3" | ||||||
|  | |||||||
| @ -21,3 +21,4 @@ rodio = "0.17.1" | |||||||
| serde = { version = "1.0.171", features = ["derive"] } | serde = { version = "1.0.171", features = ["derive"] } | ||||||
| tokio = { version = "1.29.1", features = ["full"] } | tokio = { version = "1.29.1", features = ["full"] } | ||||||
| xxhash-rust = { version = "0.8.6", features = ["xxh3", "const_xxh3"] } | xxhash-rust = { version = "0.8.6", features = ["xxh3", "const_xxh3"] } | ||||||
|  | strinto = { path = "./strinto" } | ||||||
|  | |||||||
							
								
								
									
										30
									
								
								src/bin/into.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/bin/into.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,30 @@ | |||||||
|  | #[macro_export] | ||||||
|  | macro_rules! into { | ||||||
|  |     ($struct:ident { $($field:ident : $value:expr),* $(,)? }) => { | ||||||
|  |         $struct { | ||||||
|  |             $( | ||||||
|  |                 $field: $value.into(), | ||||||
|  |             )* | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct SomeStruct { | ||||||
|  |     first_name: String, | ||||||
|  |     last_name: String, | ||||||
|  |     age: usize, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub fn main() { | ||||||
|  |     let last_name: String = "Last".to_string(); | ||||||
|  |     let age: usize = 30; | ||||||
|  | 
 | ||||||
|  |     let x = into!(SomeStruct { | ||||||
|  |         first_name: "First", | ||||||
|  |         last_name: last_name, | ||||||
|  |         age: age, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // see cargo expand to see how this function changes all struct inits to .into().
 | ||||||
|  | // see also strinto, if you think "Why could it not be done in a macro as condition?"
 | ||||||
							
								
								
									
										68
									
								
								src/bin/strinto_declarative.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								src/bin/strinto_declarative.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,68 @@ | |||||||
|  | ///
 | ||||||
|  | /// Trying to get around "".into() for String values.
 | ||||||
|  | /// Or "".to_owned().
 | ||||||
|  | /// Or String::from("").
 | ||||||
|  | /// Or "".to_string().
 | ||||||
|  | /// Choose your church.
 | ||||||
|  | ///
 | ||||||
|  | /// This is as far as you will get declaratively.
 | ||||||
|  | 
 | ||||||
|  | #[macro_export] | ||||||
|  | macro_rules! strinto { | ||||||
|  |     ($struct:ident { $($field:ident : $value:expr),* $(,)? }) => { | ||||||
|  |         $struct { | ||||||
|  |             $( | ||||||
|  |                 $field: $crate::strinto!(@convert $value), | ||||||
|  |             )* | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     (@convert $value:literal) => { | ||||||
|  |         match () { | ||||||
|  |             _ if stringify!($value).starts_with("\"") => { | ||||||
|  |                 $value.to_string() | ||||||
|  |             }, | ||||||
|  |             _ => $value.into(), // <-- no getting rid of the into!
 | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct SomeStruct { | ||||||
|  |     first_name: String, | ||||||
|  |     //last_name: String, // NOPE because of @convert.
 | ||||||
|  |     //age: usize, // reason of .into() in the first place.
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub fn main() { | ||||||
|  |     let x = strinto!(SomeStruct { | ||||||
|  |         first_name: "First", | ||||||
|  |         //last_name: String::from("Last"), // NOPE 2.
 | ||||||
|  |         //age: 1, // NOPE 1. But I went further.
 | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // while this compiles for only &str "", it is also useless compared to into!
 | ||||||
|  | // the reason is, that you cannot type check in the declarative macros.
 | ||||||
|  | // while yes, you would conditionally run stringify!($value) in the match, but the match expansion
 | ||||||
|  | // will always lead you to type errors if you don't do an .into() in the other arm.
 | ||||||
|  | // also in the end it fails to compile for anything but all members of the struct being Strings.
 | ||||||
|  | 
 | ||||||
|  | // the last idea was going with a helper trait, but that will bleed into the runtime code, and yet again only work on pure String structs.
 | ||||||
|  | 
 | ||||||
|  | // I guess I have to embrace String::from(), .to_string(), .to_owned(), .into() madness, for something every human reader can deduce in a second,
 | ||||||
|  | // if you would just implicitly convert &str to String if literally written in the code.
 | ||||||
|  | 
 | ||||||
|  | // It is kinda amusing, that the other solution, to use new() kind of turns me off, because I cannot be explicit about the parameter name in the call.
 | ||||||
|  | // I would literally love the option to be more explicit in function param names.
 | ||||||
|  | 
 | ||||||
|  | // while builder patterns feel a bit bloated for static runtime options, i will probably look into automations for that.
 | ||||||
|  | 
 | ||||||
|  | // this is probably the only aesthetic decision in rust I probably will hate forever.
 | ||||||
|  | // It does not make sense, Strings as literals already are special in your code.
 | ||||||
|  | // Because numbers are as well, you dont have to write 123.into() either.
 | ||||||
|  | // I know I probably made some really harsh logical mistakes in my opinion here, and maybe it can be proven, that I am wrong, and I would love to hear that
 | ||||||
|  | // However it kind of feels like an excuse to not simplify assigning declaratively written &str to Strings in the code.
 | ||||||
|  | 
 | ||||||
|  | // And it makes sense to be explicit about creating a String Buffer sometimes, but it does not make sense mostly.
 | ||||||
|  | 
 | ||||||
|  | // Anyway, I will still try a procedural macro for this, just for fun.
 | ||||||
							
								
								
									
										1
									
								
								strinto/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								strinto/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | /target | ||||||
							
								
								
									
										47
									
								
								strinto/Cargo.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								strinto/Cargo.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @ -0,0 +1,47 @@ | |||||||
|  | # This file is automatically @generated by Cargo. | ||||||
|  | # It is not intended for manual editing. | ||||||
|  | version = 3 | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "proc-macro2" | ||||||
|  | version = "1.0.78" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e2422ad645d89c99f8f3e6b88a9fdeca7fabeac836b1002371c4367c8f984aae" | ||||||
|  | dependencies = [ | ||||||
|  |  "unicode-ident", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "quote" | ||||||
|  | version = "1.0.35" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "291ec9ab5efd934aaf503a6466c5d5251535d108ee747472c3977cc5acc868ef" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "strinto" | ||||||
|  | version = "0.1.0" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "syn" | ||||||
|  | version = "2.0.48" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "0f3531638e407dfc0814761abb7c00a5b54992b849452a0646b7f65c9f770f3f" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "unicode-ident", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "unicode-ident" | ||||||
|  | version = "1.0.12" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b" | ||||||
							
								
								
									
										14
									
								
								strinto/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								strinto/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | |||||||
|  | [package] | ||||||
|  | name = "strinto" | ||||||
|  | version = "0.1.0" | ||||||
|  | edition = "2021" | ||||||
|  | 
 | ||||||
|  | [lib] | ||||||
|  | proc-macro = true | ||||||
|  | 
 | ||||||
|  | [dependencies] | ||||||
|  | syn = { version = "2.0", features = ["full"] } | ||||||
|  | quote = "1.0" | ||||||
|  | 
 | ||||||
|  | [dev-dependencies] | ||||||
|  | proc-macro2 = "1.0" | ||||||
							
								
								
									
										62
									
								
								strinto/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								strinto/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,62 @@ | |||||||
|  | extern crate proc_macro; | ||||||
|  | use proc_macro::TokenStream; | ||||||
|  | use quote::quote; | ||||||
|  | use syn::{parse_macro_input, Expr, ExprStruct}; | ||||||
|  | 
 | ||||||
|  | /// # literal strings in a struct insantiation are converted .into().
 | ||||||
|  | ///
 | ||||||
|  | /// ```rust
 | ||||||
|  | /// use strinto::strinto;
 | ||||||
|  | ///    #[derive(Debug)]
 | ||||||
|  | ///    struct TestStruct {
 | ||||||
|  | ///        name: String,
 | ||||||
|  | ///        title: String,
 | ||||||
|  | ///        description: String,
 | ||||||
|  | ///        age: usize,
 | ||||||
|  | ///    }
 | ||||||
|  | ///
 | ||||||
|  | ///    let descr = "description";
 | ||||||
|  | ///
 | ||||||
|  | ///    let output = strinto!(TestStruct {
 | ||||||
|  | ///        name: "John", // Literal string.
 | ||||||
|  | ///        title: String::from("Wicked"),
 | ||||||
|  | ///        description: descr.to_string(),
 | ||||||
|  | ///        age: 30,
 | ||||||
|  | ///    });
 | ||||||
|  | ///
 | ||||||
|  | ///    let output_string = format!("{:?}", output);
 | ||||||
|  | ///    assert_eq!(
 | ||||||
|  | ///        output_string,
 | ||||||
|  | ///        "TestStruct { name: \"John\", title: \"Wicked\", description: \"description\", age: 30 }"
 | ||||||
|  | ///    );
 | ||||||
|  | /// ```
 | ||||||
|  | 
 | ||||||
|  | #[proc_macro] | ||||||
|  | pub fn strinto(input: TokenStream) -> TokenStream { | ||||||
|  |     let expr_struct = parse_macro_input!(input as ExprStruct); | ||||||
|  | 
 | ||||||
|  |     // Extract struct name and fields
 | ||||||
|  |     let struct_name = &expr_struct.path; | ||||||
|  |     let fields = expr_struct.fields.iter().map(|field| { | ||||||
|  |         let field_name = field.member.clone(); | ||||||
|  |         let field_value = &field.expr; | ||||||
|  |         // Determine if the field value is a string literal and transform it
 | ||||||
|  |         if let Expr::Lit(expr_lit) = field_value { | ||||||
|  |             if let syn::Lit::Str(_) = expr_lit.lit { | ||||||
|  |                 quote! { #field_name: #field_value.into() } | ||||||
|  |             } else { | ||||||
|  |                 quote! { #field_name: #field_value } | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             quote! { #field_name: #field_value } | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     let expanded = quote! { | ||||||
|  |         #struct_name { | ||||||
|  |             #(#fields,)* | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     expanded.into() | ||||||
|  | } | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user