Cache
提供缓存功能的中间件.
Cache 中间件可以对 Response
中的 StatusCode
, Headers
, Body
提供缓存功能. 对于已经缓存的内容, 当下次处理请求时, Cache 中间件会直接把缓存在内存中的内容发送给客户端.
注意, 此插件不会缓存 Body
是 ResBody::Stream
的 Response
. 如果应用到了这一类型的 Response
, Cache 不会处理这些请求, 也不会引起错误.
主要功能
CacheIssuer
提供了对分配的缓存键值的抽象.RequestIssuer
是它的一个实现, 可以定义依据请求的 URL 的哪些部分以及请求的Method
生成缓存的键. 你也可以定义你自己的缓存键生成的逻辑. 缓存的键不一定是字符串类型, 任何满足Hash + Eq + Send + Sync + 'static
约束的类型都可以作为键.CacheStore
提供对数据的存取操作.MemoryStore
是内置的基于moka
的一个内存的缓存实现. 你也可以定义自己的实现方式.Cache
是实现了Handler
的结构体, 内部还有一个skipper
字段, 可以指定跳过某些不需要缓存的请求. 默认情况下, 会使用MethodSkipper
跳过除了Method::GET
以外的所有请求.内部实现示例代码:
impl<S, I> Cache<S, I> { pub fn new(store: S, issuer: I) -> Self { let skipper = MethodSkipper::new().skip_all().skip_get(false); Cache { store, issuer, skipper: Box::new(skipper), } } }
示例代码
use std::time::Duration;
use salvo::cache::{Cache, MemoryStore, RequestIssuer};
use salvo::prelude::*;
use salvo::writing::Text;
use time::OffsetDateTime;
#[handler]
async fn home() -> Text<&'static str> {
Text::Html(HOME_HTML)
}
#[handler]
async fn short() -> String {
format!(
"Hello World, my birth time is {}",
OffsetDateTime::now_utc()
)
}
#[handler]
async fn long() -> String {
format!(
"Hello World, my birth time is {}",
OffsetDateTime::now_utc()
)
}
#[tokio::main]
async fn main() {
tracing_subscriber::fmt().init();
let short_cache = Cache::new(
MemoryStore::builder()
.time_to_live(Duration::from_secs(5))
.build(),
RequestIssuer::default(),
);
let long_cache = Cache::new(
MemoryStore::builder()
.time_to_live(Duration::from_secs(60))
.build(),
RequestIssuer::default(),
);
let router = Router::new()
.get(home)
.push(Router::with_path("short").hoop(short_cache).get(short))
.push(Router::with_path("long").hoop(long_cache).get(long));
let acceptor = TcpListener::new("127.0.0.1:5800").bind().await;
Server::new(acceptor).serve(router).await;
}
static HOME_HTML: &str = r#"
<!DOCTYPE html>
<html>
<head>
<title>Cache Example</title>
</head>
<body>
<h2>Cache Example</h2>
<p>
This examples shows how to use cache middleware.
</p>
<p>
<a href="/short" target="_blank">Cache 5 seconds</a>
</p>
<p>
<a href="/long" target="_blank">Cache 1 minute</a>
</p>
</body>
</html>
"#;
[package]
name = "example-cache-simple"
version = "0.1.0"
edition = "2021"
publish = false
[dependencies]
salvo = {workspace = true, features = ["cache"] }
tokio = { version = "1", features = ["macros"] }
time = "0.3"
tracing = "0.1"
tracing-subscriber = "0.3"