Compare commits

...

5 Commits

Author SHA1 Message Date
6a64527022 version bump
now with str and bytes mode
2025-07-28 21:08:53 +02:00
7b110a43b4 fixed typos & formatting 2025-07-28 21:07:27 +02:00
1a81ce8376 updated README with new modes 2025-07-28 20:50:54 +02:00
aac8787410 added tests for str mode 2025-07-28 20:44:50 +02:00
d025c3c120 added str and bytes mode 2025-07-28 20:44:32 +02:00
6 changed files with 139 additions and 28 deletions

2
Cargo.lock generated
View File

@@ -4,7 +4,7 @@ version = 4
[[package]]
name = "dir-embed"
version = "0.1.1"
version = "0.2.0"
dependencies = [
"proc-macro2",
"quote",

View File

@@ -1,6 +1,6 @@
[package]
name = "dir-embed"
version = "0.1.1"
version = "0.2.0"
edition = "2024"
license = "MIT"
description = "Like include_bytes! for directories"

View File

@@ -1,11 +1,17 @@
Simple way to use [`include_bytes!`](https://doc.rust-lang.org/std/macro.include_bytes.html) for directories.
# Example
You can embed files two ways.
## Bytes mode
```rust
use dir_embed::Embed;
#[derive(Embed)]
#[dir = "../web/static"] // Path is relativ to the current file
#[mode = "bytes"] // Is the default. Can be omitted.
pub struct Assets;
fn main(){
@@ -17,3 +23,19 @@ fn main(){
}
```
## Str mode
```rust
use dir_embed::Embed;
#[derive(Embed)]
#[dir = "../web/static"] // Path is relativ to the current file
#[mode = "str"]
pub struct Assets;
fn main(){
let file: &str = Assets::get("css/style.css").expect("Can't find file");
println!("{file}");
}
```

View File

@@ -6,32 +6,41 @@ use syn::{Attribute, DeriveInput, Expr, Ident, Lit, Meta, parse_macro_input};
use std::fs;
use std::path::{Path, PathBuf};
#[proc_macro_derive(Embed, attributes(dir))]
#[derive(Debug, Clone, Copy)]
enum EmbedMode {
Bytes,
Str,
}
#[proc_macro_derive(Embed, attributes(dir, mode))]
pub fn embed(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
let input = parse_macro_input!(input as DeriveInput);
let struct_name = &input.ident;
let attr = input
let dir_attr = input
.attrs
.iter()
.find(|e| e.path().is_ident("dir"))
.expect("No #[dir = \"...\"] attribute found");
let base_path = PathBuf::from(extract_dir_path(attr));
let mode_attr = input.attrs.iter().find(|e| e.path().is_ident("mode"));
let mode = mode_attr.map(extract_mode).unwrap_or(EmbedMode::Bytes);
let base_path = PathBuf::from(extract_dir_path(dir_attr));
let source_file = PathBuf::from(input.span().unwrap().file());
let source_dir = if let Some(parent) = source_file.parent() {
parent
} else {
// HACK: when running in rust-analyzer i can't seem to get the parent dir.
return TokenStream::from(generate_impl(struct_name, Vec::new(), Vec::new()));
return TokenStream::from(generate_byte_impl(struct_name, Vec::new()));
};
let absolue_path = source_dir.join(&base_path);
let mut match_arms = Vec::new();
let mut entries = Vec::new();
for entry in collect_files(&absolue_path) {
let rel_path = entry
@@ -42,18 +51,20 @@ pub fn embed(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
.replace("\\", "/");
let include_path = base_path.join(&rel_path);
let include_string = include_path.to_str();
let include_string = include_path.to_str().unwrap();
match_arms.push(quote! {
#rel_path => Some(include_bytes!(#include_string) as &'static [u8]),
});
let arm = match mode {
EmbedMode::Bytes => generate_byte_arm(&rel_path, include_string),
EmbedMode::Str => generate_str_arm(&rel_path, include_string),
};
entries.push(quote! {
(#rel_path, include_bytes!(#include_string) as &'static [u8])
});
match_arms.push(arm);
}
let expanded = generate_impl(struct_name, match_arms, entries);
let expanded = match mode {
EmbedMode::Bytes => generate_byte_impl(struct_name, match_arms),
EmbedMode::Str => generate_str_impl(struct_name, match_arms),
};
proc_macro::TokenStream::from(expanded)
}
@@ -88,10 +99,36 @@ fn extract_dir_path(attr: &Attribute) -> String {
}
}
fn generate_impl(
fn extract_mode(attr: &Attribute) -> EmbedMode {
let meta = match &attr.meta {
Meta::NameValue(meta) => meta,
_ => panic!("Expected #[mode = \"bytes\"|\"str\"] as a name-value attribute."),
};
let expr_lit = match &meta.value {
Expr::Lit(expr_lit) => expr_lit,
_ => panic!("Expected #[mode = \"bytes\"|\"str\"] with a string literal."),
};
match &expr_lit.lit {
Lit::Str(str) => match str.value().as_str() {
"bytes" => EmbedMode::Bytes,
"str" => EmbedMode::Str,
other => panic!("Unknown mode: {other}. Use `bytes` or `str`."),
},
_ => panic!("Expected #[mode = \"bytes\"|\"str\"] to be a string."),
}
}
fn generate_byte_arm(rel: &str, include: &str) -> proc_macro2::TokenStream {
quote! {
#rel => Some(include_bytes!(#include)),
}
}
fn generate_byte_impl(
struct_name: &Ident,
match_arms: Vec<proc_macro2::TokenStream>,
entries: Vec<proc_macro2::TokenStream>,
) -> proc_macro2::TokenStream {
quote! {
impl #struct_name {
@@ -101,9 +138,27 @@ fn generate_impl(
_ => None,
}
}
}
}
}
pub fn iter() -> impl Iterator<Item = (&'static str, &'static [u8])> {
[#(#entries),*].into_iter()
fn generate_str_arm(rel: &str, include: &str) -> proc_macro2::TokenStream {
quote! {
#rel => Some(include_str!(#include)),
}
}
fn generate_str_impl(
struct_name: &Ident,
match_arms: Vec<proc_macro2::TokenStream>,
) -> proc_macro2::TokenStream {
quote! {
impl #struct_name {
pub fn get(name: &str) -> Option<&'static str> {
match name {
#(#match_arms)*
_ => None,
}
}
}
}

View File

@@ -2,43 +2,42 @@ use dir_embed::Embed;
#[derive(Embed)]
#[dir = "./../testdata/"]
#[mode = "bytes"]
pub struct Assets;
#[test]
fn get() {
fn byte_get() {
assert!(Assets::get("file1.txt").is_some());
}
#[test]
fn get_missing() {
fn byte_get_missing() {
assert!(Assets::get("missing.txt").is_none());
}
#[test]
fn read_content() {
fn byte_read_content() {
let content_should = "file1".as_bytes();
let content_is = Assets::get("file1.txt").unwrap();
assert_eq!(*content_is, *content_should);
}
#[test]
fn parse_string() {
fn byte_parse_string() {
let file: &[u8] = Assets::get("file1.txt").expect("Can't find file");
let string = str::from_utf8(file).expect("Failed to parse file");
println!("{string}");
assert_eq!(string, "file1");
}
#[test]
fn sub_directories_get() {
fn byte_sub_directories_get() {
assert!(Assets::get("sub/file2.txt").is_some());
}
#[test]
fn sub_directories_content() {
fn byte_sub_directories_content() {
let content_should = "file2".as_bytes();
let content_is = Assets::get("sub/file2.txt").unwrap();
assert_eq!(*content_is, *content_should);
}

35
tests/str.rs Normal file
View File

@@ -0,0 +1,35 @@
use dir_embed::Embed;
#[derive(Embed)]
#[dir = "./../testdata/"]
#[mode = "str"]
pub struct Assets;
#[test]
fn str_get() {
assert!(Assets::get("file1.txt").is_some());
}
#[test]
fn str_get_missing() {
assert!(Assets::get("missing.txt").is_none());
}
#[test]
fn str_read_content() {
let content_should = "file1";
let content_is = Assets::get("file1.txt").unwrap();
assert_eq!(content_is, content_should);
}
#[test]
fn str_sub_directories_get() {
assert!(Assets::get("sub/file2.txt").is_some());
}
#[test]
fn str_sub_directories_content() {
let content_should = "file2";
let content_is = Assets::get("sub/file2.txt").unwrap();
assert_eq!(content_is, content_should);
}