GenCo is an even simpler code generator for Rust, written for use in [reproto].
We depend on (Yet to be released) 1.45 and some of its proc_macro_hygiene
stabilizations. Until then, you must build with the beta
branch.
For example:
bash
cargo +beta run --example rust
The workhorse of GenCo is the [quote!] macro. While tokens can be constructed manually, [quote!] makes this process much easier.
GenCo only minimally deals with language-specific syntax, but primarily deals with solving the following:
<stmt>.quoted()
].The following are language specific examples for GenCo using the [quote!] macro.
You can run one of the examples above using:
bash
cargo run --example go
The following is the included example Rust program.
```rust use genco::rust::{imported, Config}; use genco::{quote, FormatterConfig, Rust}; use rand::Rng;
use std::fmt;
fn main() -> fmt::Result { // Import the LittleEndian item, without referencing it through the last // module component it is part of. let littleendian = imported("byteorder", "LittleEndian").qualified(); let bigendian = imported("byteorder", "BigEndian");
// This is a trait, so only import it into the scope (unless we intent to
// implement it).
let write_bytes_ext = imported("byteorder", "WriteBytesExt").alias("_");
let read_bytes_ext = imported("byteorder", "ReadBytesExt").alias("_");
// Iterators can be tokenized using `tokenize_iter`, as long as they contain
// something which can be converted into a stream of tokens.
let numbers = (0..10)
.map(|_| quote!(#(rand::thread_rng().gen::<i16>())));
let tokens = quote! {
// Markup used for imports without an immediate use.
#@(write_bytes_ext)
#@(read_bytes_ext)
fn test() {
let mut wtr = vec![];
wtr.write_u16::<#little_endian>(517).unwrap();
wtr.write_u16::<#big_endian>(768).unwrap();
assert_eq!(wtr, vec![#numbers,*]);
}
};
// Simpler printing with default indentation:
// println!("{}", tokens.to_file_string()?);
tokens.to_io_writer_with(
std::io::stdout().lock(),
Config::default(),
FormatterConfig::from_lang::<Rust>().with_indentation(2),
)?;
Ok(())
} ```
The quote!
macro has the following rules for dealing with indentation and
spacing.
Two tokens that are separated, are spaced. Regardless of how many spaces there are between them.
So:
rust
let _: genco::Tokens<genco::Rust> = genco::quote!(fn test() {});
Becomes:
rust
fn test() {}
More that two line breaks are collapsed.
So:
```rust
let _: genco::Tokens
println!("... World!");
}
}; ```
Becomes:
```rust fn test() { println!("Hello...");
println!("... World!");
} ```
Indentation is determined on a row-by-row basis. If a column is further in than the one on the preceeding row, it is indented one level deeper.
Like wise if a column starts before the previous rows column, it is indended one level shallower.
So:
rust
let _: genco::Tokens<genco::Rust> = genco::quote! {
fn test() {
println!("Hello...");
println!("... World!");
}
};
Becomes:
rust
fn test() {
println!("Hello...");
println!("... World!");
}