#serialization #serde #build-time #macro #no-std

no-std sucrose

Rust crate for embedding static data from files at build time

4 stable releases

1.1.1 Aug 13, 2023
1.0.1 Aug 12, 2023

#796 in Encoding

46 downloads per month

MIT license

12KB
164 lines

Sucrose

A Rust crate for embedding static data from files at build time.

Resource

Sucrose works by generating mirrors of existing structures with entirely static fields.

For example, this structure:

#[derive(sucrose::Resource)]
struct Character {
	name: String
}

...would be translated into this:

struct Character {
	name: &'static str
}

Furthermore, dynamic instances can be converted to their static counterparts as well. This:

# struct Character { name: String };
Character {
	name: String::from("Hello, world!"),
}
# ;

...becomes this:

# struct Character { name: &'static str };
Character {
	name: "Hello, world!",
}
# ;

The conversion may look trivial at first, but you can also nest resources (types which implement Resource), and create conversions for your own types using the ToStatic trait.

ToStatic

Certain types aren't necessarily resources, but need to be members of a resource regardless. For example, consider a simple integer type: i32. Sucrose needs to know how to trasform this type from a dynamic to static representation. In the case of integer, that's simple: both representations are the same.

The ToStatic trait can be used to express that a type may be a member of a resource (in fact, #derive(Resource also generates an implementation of ToStatic). An implementation for i32 may look like this:

# use sucrose::*;
impl ToStatic for i32 {
	fn static_type() -> TokenStream {
		quote!(i32)
	}
	fn static_value(&self) -> TokenStream {
		quote!(#self)
	}
}

This implementation says that the static equivalent of i32 is, well, i32, and that its value would be written in Rust as a plain integer literal (which is what quote!(#self) generates).

Sucrose provides implementations for all integer types, as well as String, Vec, bool, Option, and the NonZero integer types. More implementations may be added as needed

Note that implementations on types from other libraries can only be written within sucrose or the library owning the type. To get around this, you can use "newtype" wrappers to give your code ownership of the type. As a more long term solution, open an issue requesting an implementation of the type. See the Rust documentation for more information.

Dependencies

~1.5–2.2MB
~43K SLoC