GenCo

Build Status crates.io

GenCo is an even simpler code generator for Rust, written for use in [reproto].

Note: We currently depend on #![feature(proc_macro_hygiene)], which is set to be stabilized in Rust 1.45.

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:

Examples

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

![feature(procmacrohygiene)]

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(())

} ```

Indentation Rules

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

![feature(procmacrohygiene)]

let _: genco::Tokens = genco::quote!(fn test() {}); ```

Becomes:

rust fn test() {}

More that two line breaks are collapsed.

So:

```rust

![feature(procmacrohygiene)]

let _: genco::Tokens = genco::quote! { fn test() { println!("Hello...");

    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

![feature(procmacrohygiene)]

let _: genco::Tokens = genco::quote! { fn test() { println!("Hello..."); println!("... World!"); } }; ```

Becomes:

rust fn test() { println!("Hello..."); println!("... World!"); }