Rust Custom Derive Macro which loads files into the rust binary at compile time during release and loads the file from the fs during dev.
You can use this to embed your css, js and images into a single executable which can be deployed to your servers. Also it makes it easy to build a very small docker image for you to deploy.
[dependencies]
rust-embed="6.3.0"
You need to add the custom derive macro RustEmbed to your struct with an attribute folder
which is the path to your static folder.
The path resolution works as follows:
- In
debug
and whendebug-embed
feature is not enabled, the folder path is resolved relative to where the binary is run from. - In
release
or whendebug-embed
feature is enabled, the folder path is resolved relative to whereCargo.toml
is.
#[derive(RustEmbed)]
#[folder = "examples/public/"]
struct Asset;
The macro will generate the following code:
impl Asset {
pub fn get(file_path: &str) -> Option<rust_embed::EmbeddedFile> {
...
}
pub fn iter() -> impl Iterator<Item = Cow<'static, str>> {
...
}
}
impl RustEmbed for Asset {
fn get(file_path: &str) -> Option<rust_embed::EmbeddedFile> {
...
}
fn iter() -> impl Iterator<Item = Cow<'static, str>> {
...
}
}
// Where EmbeddedFile contains these fields,
pub struct EmbeddedFile {
pub data: Cow<'static, [u8]>,
pub metadata: Metadata,
}
pub struct Metadata {
hash: [u8; 32],
last_modified: Option<u64>,
}
Given a relative path from the assets folder returns the EmbeddedFile
if found.
If the feature debug-embed
is enabled or the binary compiled in release mode the bytes have been embeded in the binary and a Option<rust_embed::EmbeddedFile>
is returned.
Otherwise the bytes are read from the file system on each call and a Option<rust_embed::EmbeddedFile>
is returned.
Iterates the files in this assets folder.
If the feature debug-embed
is enabled or the binary compiled in release mode a static array to the list of relative paths to the files is returned.
Otherwise the files are listed from the file system on each call.
You can add #[prefix = "my_prefix/"]
to the RustEmbed
struct to add a prefix
to all of the file paths. This prefix will be required on get
calls, and will
be included in the file paths returned by iter
.
Always embed the files in the binary, even in debug mode.
Allow environment variables to be used in the folder
path. Example:
#[derive(RustEmbed)]
#[folder = "$CARGO_MANIFEST_DIR/foo"]
struct Asset;
This will pull the foo
directory relative to your Cargo.toml
file.
Compress each file when embedding into the binary. Compression is done via include-flate
.
Filter files to be embedded with multiple #[include = "*.txt"]
and #[exclude = "*.jpg"]
attributes.
Matching is done on relative file paths, via globset
.
exclude
attributes have higher priority than include
attributes.
Example:
#[derive(RustEmbed)]
#[folder = "examples/public/"]
#[include = "*.html"]
#[include = "images/*"]
#[exclude = "*.txt"]
struct Asset;
use rust_embed::RustEmbed;
#[derive(RustEmbed)]
#[folder = "examples/public/"]
#[prefix = "prefix/"]
struct Asset;
fn main() {
let index_html = Asset::get("prefix/index.html").unwrap();
println!("{:?}", std::str::from_utf8(index_html.data.as_ref()));
for file in Asset::iter() {
println!("{}", file.as_ref());
}
}
To run the example in dev mode where it reads from the fs,
cargo run --example basic
To run the example in release mode where it reads from binary,
cargo run --example basic --release
Note: To run the actix-web
example:
cargo run --example actix --features actix
Note: To run the rocket
example:
cargo run --example rocket --features rocket
Note: To run the warp
example:
cargo run --example warp --features warp-ex
Note: To run the axum
example:
cargo run --example axum --features axum-ex
Note: To run the poem example:
cargo run --example poem --features poem-ex
Note: To run the salvo example:
cargo run --example salvo --features salvo-ex
debug: cargo test --test lib
release: cargo test --test lib --release
Go Rusketeers! The power is yours!