4 stable releases
1.1.3 | Nov 23, 2024 |
---|---|
1.0.1 | Nov 22, 2024 |
#1514 in Database interfaces
31KB
571 lines
cache-any
A cache library for Rust.
This library provides a trait Cache
and some implementations of it. It defines the basic operations of a cache, for example, caches::Cache::get
, caches::Cache::set
. All functions are async, because we may use async storage backends. All caches are key-value based.
By default, it provides a simple memory cache as example. See caches::MemoryCache
.
Features
redis
: Use redis as storage backend. Seecaches::RedisCache
.mysql
: Use mysql as storage backend. Seecaches::MySqlCache
.
Usage
Add cache-any
to your Cargo.toml
:
[dependencies]
cache-any = { version = "1", features = ["full"] }
Concepts
- Key: Specified by the cache implementation. Usually it is a string-like type (&str, String, ...).
- Value: The value of a cache is a
Cacheable
value.
Cacheable
is a trait that describes how to convert a value
to bytes and vice versa.
A cache can store any value that implements Cacheable
. That is, you can store usize and string (or any other types) at the same time. But you need to know the exact type when you retrieve the value.
Basic Usage
We use caches::MemoryCache
as example:
let cache = MemoryCache::default();
// The cache is empty, so get returns None.
assert!(cache.get::<()>("non-existent-key").await.unwrap().is_none());
// [SET a -> 1]
cache.set("a", 1).await.unwrap();
// [GET a] -> Some(1)
let a_value: u8 = cache.get("a").await.unwrap().unwrap();
assert_eq!(a_value, 1);
// you can do type casting, using u16 instead of u8 as an example.
let a_value: u16 = cache.get("a").await.unwrap().unwrap();
assert_eq!(a_value, 1);
// you can also store String in the same cache:
cache.set("b", String::from("hello")).await.unwrap();
let b_value: String = cache.get("b").await.unwrap().unwrap();
assert_eq!(b_value, String::from("hello"));
More examples: GitHub
Extend Cacheable
You can extend Cacheable
for your own types. For example, you can define a struct and implement Cacheable
for it:
#[derive(serde::Serialize, serde::Deserialize)] // for json
struct MyStruct {
a: u8,
b: String,
}
In this case, we use serde_json
to convert the struct to bytes and vice versa:
impl Cacheable for MyStruct {
fn to_bytes(&self) -> Vec<u8> {
serde_json::to_vec(self).unwrap()
}
fn from_bytes(bytes: &[u8]) -> anyhow::Result<Self> {
let ret = serde_json::from_slice(bytes)?;
Ok(ret)
}
}
Then you can store MyStruct
in the cache:
cache.set("my-struct", MyStruct { a: 1, b: String::from("hello") }).await.unwrap();
Work in Progress
- Add examples for each cache implementation.
Contributing
Any contributions are welcome.
If you find any useful cache implementation, feel free to open an issue or a pull request at Github.
If bugs are found, just file an issue at Github, and I will fix it ASAP.
License
This project is licensed under the MIT License.
Dependencies
~2–16MB
~241K SLoC