12 releases (5 breaking)

0.6.0 Jul 22, 2024
0.4.5 Apr 30, 2024
0.4.4 Mar 7, 2024
0.4.0 Dec 4, 2023
0.3.0 Oct 27, 2023

#1369 in Procedural macros

Download history 257/week @ 2024-07-01 105/week @ 2024-07-08 52/week @ 2024-07-15 391/week @ 2024-07-22 224/week @ 2024-07-29 65/week @ 2024-08-05 93/week @ 2024-08-12 75/week @ 2024-08-19 19/week @ 2024-08-26 54/week @ 2024-09-02 97/week @ 2024-09-09 150/week @ 2024-09-16 121/week @ 2024-09-23 34/week @ 2024-09-30 118/week @ 2024-10-07 65/week @ 2024-10-14

347 downloads per month
Used in memory-serve

Apache-2.0 OR MIT

16KB
213 lines

Memory serve

memory-serve enables fast static file serving for axum web applications, by keeping all assets in memory.

It loads static web assets like HTML, stylesheets, images and scripts into the rust binary at compile time and exposes them as an axum Router. It automatically adds cache headers and handles file compression.

During development (debug builds) files are served dynamically, they are read and compressed at request time.

Text-based files like HTML or javascript are compressed using brotli at compile time and decompressed at startup, to minimize the binary size.

All files are served with an etag header and If-None-Match requests are handled accordingly.

Text-based files are served in plain or with gzip or brotli compression based on the abilities and preferences of the client.

Routing can be configured in a flexible manner, for instance to accommodate an SPA.

Compatibility

memory-serve is designed to work with axum

Usage

Provide a relative path to the directory containing your static assets to the load_assets! macro. This macro creates a data structure intended to be consumed by MemoryServe::new. Calling [MemoryServe::into_router()] on the resulting instance produces a axum Router that can either be merged in another Router or used directly in a server by calling Router::into_make_service().

Example

use axum::{response::Html, routing::get, Router};
use memory_serve::{load_assets, MemoryServe};
use std::net::SocketAddr;

#[tokio::main]
async fn main() {
    let memory_router = MemoryServe::new(load_assets!("static"))
        .index_file(Some("/index.html"))
        .into_router();

    // possible other routes can be added at this point, like API routes
    let app = Router::new()
        .merge(memory_router);

    let addr = SocketAddr::from(([127, 0, 0, 1], 3000));
    let listener = tokio::net::TcpListener::bind(addr).await.unwrap();
    axum::serve(listener, app).await.unwrap();
}

Configuration options

An instance of the MemoryServe struct can be configured by calling the following configuration methods:

method Default value Description
MemoryServe::index_file Some("/index.html") Which file to serve on the route "/"
MemoryServe::index_on_subdirectories false Whether to serve the corresponding index in subdirectories
MemoryServe::fallback None Which file to serve if no routed matched the request
MemoryServe::fallback_status StatusCode::NOT_FOUND The HTTP status code to routes that did not match
MemoryServe::enable_gzip true Allow to serve gzip encoded files
MemoryServe::enable_brotli true Allow to serve brotli encoded files
MemoryServe::html_cache_control CacheControl::Short Cache control header to serve on HTML files
MemoryServe::cache_control CacheControl::Medium Cache control header to serve on other files
MemoryServe::add_alias [] Create a route / file alias
MemoryServe::enable_clean_url false Enable clean URLs

See Cache control for the cache control options.

Logging

During compilation, problems that occur with the inclusion or compression of assets are logged to stdout, for instance:

WARN skipping file "static/empty.txt": file empty

When running the resulting executable, all registered routes and asset sizes are logged using the tracing crate. To print or log them, use tracing-subscriber. Example output:

 INFO memory_serve: serving /assets/icon.jpg 1366 bytes
 INFO memory_serve: serving /assets/index.css 1552 bytes
 INFO memory_serve: serving /assets/index.css (brotli compressed) 509 bytes
 INFO memory_serve: serving /assets/index.css (gzip compressed) 624 bytes
 INFO memory_serve: serving /assets/index.js 20 bytes
 INFO memory_serve: serving /assets/stars.svg 2255 bytes
 INFO memory_serve: serving /assets/stars.svg (brotli compressed) 907 bytes
 INFO memory_serve: serving /assets/stars.svg (gzip compressed) 1048 bytes
 INFO memory_serve: serving /index.html 437 bytes
 INFO memory_serve: serving /index.html (brotli compressed) 178 bytes
 INFO memory_serve: serving /index.html (gzip compressed) 274 bytes
 INFO memory_serve: serving /index.html as index on /

Cache control

There are 5 different values to choose from for the cache-control settings:

Option Description Value
CacheControl::Long clients can keep assets that have cache busting for a year max-age=31536000, immutable
CacheControl::Medium assets without cache busting are revalidated after a day and can be kept for a week max-age=604800, stale-while-revalidate=86400
CacheControl::Short cache kept for max 5 minutes, only at the client (not in a proxy) max-age:300, private
CacheControl::NoCache do not cache if freshness is really vital no-cache
CacheControl::Custom Custom value user defined

Dependencies

~9–16MB
~359K SLoC