use crate::api::helpers::with_db;
use rapiddb::traits::IAsyncDatabase;
use warp::{Filter, Rejection, Reply};
pub fn get(
db: std::sync::Arc<tokio::sync::RwLock<impl IAsyncDatabase + ?Sized>>,
) -> impl Filter<Extract = impl Reply, Error = Rejection> + Clone {
warp::path!("api" / "v0" / "sensors" / "latest" / usize)
.and(warp::get())
.and(with_db(db))
.and_then(_get)
}
pub async fn _get(
limit: usize,
db: std::sync::Arc<tokio::sync::RwLock<impl IAsyncDatabase + ?Sized>>,
) -> Result<impl warp::Reply, std::convert::Infallible> {
let mut lock = db.write().await;
let data = lock.get_all_latest_with_limit(limit).await;
if !data.is_empty() {
let mut result: String = Default::default();
result += "{";
for (key, value_arr) in data {
result += &format!("\"{key}\":[");
for value in value_arr {
result +=
&format!("{},", std::str::from_utf8(&value).unwrap_or_default());
}
result.pop();
result += "],";
}
result.pop();
result += "}";
return Ok(
warp::hyper::Response::builder()
.status(warp::http::StatusCode::OK)
.body(result),
);
}
Ok(
warp::hyper::Response::builder()
.status(warp::http::StatusCode::NOT_FOUND)
.body(Default::default()),
)
}
#[tokio::test]
async fn test_get() {
let database_test_factory = rapiddb::db::DatabaseTestFactory::new(
".temp/test/sensors_latest_limit/test_get",
);
for db in database_test_factory.get_instance().values() {
let api = super::endpoints((*db).clone());
let id = "test-0";
let id0 = "test-1";
let id1 = "test-2";
let limit = 10;
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/sensors/latest/{limit}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 404);
db.write()
.await
.post(id, serde_json::json!({ "id": &id }).to_string().as_bytes())
.await;
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/sensors/latest/{limit}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 200);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body()).unwrap(),
serde_json::json!({id: [{"id": &id}]})
);
let resp = warp::test::request()
.method("GET")
.path("/api/v0/sensors/latest/0")
.reply(&api)
.await;
assert_eq!(resp.status(), 404);
db.write()
.await
.post(id, serde_json::json!({ "id1": &id }).to_string().as_bytes())
.await;
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/sensors/latest/{limit}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 200);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.len(),
1
);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id)
.unwrap()
.as_array()
.unwrap()
.len(),
2
);
for _ in 0..8 {
db.write()
.await
.post(id0, serde_json::json!({ "id2": &id }).to_string().as_bytes())
.await;
}
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/sensors/latest/{limit}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 200);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.len(),
2
);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id)
.unwrap()
.as_array()
.unwrap()
.len()
+ serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id0)
.unwrap()
.as_array()
.unwrap()
.len(),
10
);
for _ in 0..8 {
db.write()
.await
.post(id0, serde_json::json!({ "id2": &id }).to_string().as_bytes())
.await;
}
for _ in 0..7 {
db.write()
.await
.post(id, serde_json::json!({ "id2": &id }).to_string().as_bytes())
.await;
}
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/sensors/latest/{limit}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 200);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.len(),
2
);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id)
.unwrap()
.as_array()
.unwrap()
.len()
+ serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id0)
.unwrap()
.as_array()
.unwrap()
.len(),
19
);
db.write()
.await
.post(id, serde_json::json!({ "id2": &id }).to_string().as_bytes())
.await;
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/sensors/latest/{limit}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 200);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.len(),
2
);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id)
.unwrap()
.as_array()
.unwrap()
.len()
+ serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id0)
.unwrap()
.as_array()
.unwrap()
.len(),
20
);
for _ in 0..8 {
db.write()
.await
.post(id, serde_json::json!({ "id2": &id }).to_string().as_bytes())
.await;
}
for _ in 0..8 {
db.write()
.await
.post(id0, serde_json::json!({ "id2": &id }).to_string().as_bytes())
.await;
}
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/sensors/latest/{limit}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 200);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.len(),
2
);
assert_eq!(
serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id)
.unwrap()
.as_array()
.unwrap()
.len()
+ serde_json::from_slice::<serde_json::Value>(resp.body())
.unwrap()
.as_object()
.unwrap()
.get(id0)
.unwrap()
.as_array()
.unwrap()
.len(),
20
);
db.write()
.await
.post_meta(
id1,
serde_json::json!({ "id1": &id1 }).to_string().as_bytes().to_vec(),
)
.await;
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/sensors/latest/{limit}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 200);
}
}