#compile-time #directory-tree #macro #compression #proc-macro #embedding #gzip

macro build rz-embed

rz-embed implements a simple proc-macro for embedding directory trees in rust binaries

2 releases

0.1.1 Oct 5, 2024
0.1.0 May 20, 2024

#360 in Build Utils

MIT/Apache

23KB
451 lines

rz-embed

rz-embed implements a simple proc-macro for embedding directory trees in rust binaries. All files are gzip compressed at compile (or rather, macro expansion) time. lazy_static is used to decompress resources at runtime.

I originally wrote this to bundle frontend code (svelte, angular..) as gzip-compressed data embedded in rocket.rs binaries. adding rocket = true to the macro call will automatically generate the appropriate handlers, as well as a routes() function to easily mount them.

Usage

rz_embed::include_as_compressed!(
    "src/some-data",
    module_name = "some_data", // access files via some_data::FILENAME_SLUG
    rocket = true // generate routes for each file 
);

Rocket route URLs are relative to the source directory, i.e.

  • some-data/ipsum.md -> /ipsum.md
  • some-data/foo/ipsum.md -> /foo/ipsum.md

The macro will output compressed files (and the compression rate) to stdout:

❯ cargo run --release --bin example-rocket
   Compiling example-app v0.1.0 (rz-embed/example-app)
[~] rz-embed/example-app/src/frontend/index.html already compressed 388 -> 230 bytes (40.72%)
[~] rz-embed/example-app/src/frontend/index.css already compressed 33 -> 56 bytes (-69.70%)
[*] total: 421 -> 286 bytes (32.07%)
[+] rz-embed/example-app/src/some-data/ipsum.txt: 591 -> 207 bytes (64.97%)
[+] rz-embed/example-app/src/some-data/ipsum.md: 606 -> 215 bytes (64.52%)
[~] rz-embed/example-app/src/some-data/ipsum.xml already compressed 648 -> 255 bytes (60.65%)
[*] total: 1845 -> 677 bytes (63.31%)
    Finished release [optimized] target(s) in 5.65s
     Running `target/release/example-rocket`
🚀 Rocket has launched from http://127.0.0.1:8000

Generated routes can be collected with <module-name>::routes():

>> (serve_index) GET /index
>> (serve_index_css) GET /index.css
>> (serve_index_html) GET /index.html
>> (serve_ipsum_md) GET /data/ipsum.md
>> (serve_ipsum_txt) GET /data/ipsum.txt
>> (serve_ipsum_xml) GET /data/ipsum.xml

See example-app for generic and rocket examples.

Compression

Compressed files are stored in target/rz-embed/<input path slug>/:

example-app/target/rz-embed
├── [...]_rz_embed_example_app_src_frontend
│  ├── index_css.gz
│  └── index_html.gz
└── [...]_rz_embed_example_app_src_some_data
   ├── ipsum_md.gz
   ├── ipsum_txt.gz
   └── ipsum_xml.gz

Files are only re-compressed if they changed after compression, when the macro is expanded. This is done via file metadata/mtime - currently only tested on Linux/ext4.

TODOs

  • skip compression if it would increase file size (i.e. dont compress small files)
  • skip compression for known compressed formats (i.e. jpeg,zip,...)

Dependencies

~3–11MB
~110K SLoC