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" / String)
.and(warp::get())
.and(with_db(db))
.and_then(_get)
}
pub async fn _get(
id: String,
db: std::sync::Arc<tokio::sync::RwLock<impl IAsyncDatabase + ?Sized>>,
) -> Result<impl warp::Reply, std::convert::Infallible> {
if db.write().await.contains(&id).await {
return Ok(warp::hyper::Response::builder()
.status(warp::http::StatusCode::OK)
.body(format!("{}", &serde_json::json!({
"resources": [
{"endpoint": format!("/api/v0/{id}/latest"), "description": format!("GET latest measurment from {id}")},
{"endpoint": format!("/api/v0/{id}/latest/:count"), "description": format!("GET latest :count measurments from {id}")},
{"endpoint": format!("/api/v0/{id}/:id"), "description": format!("GET measurment by id from {id}")},
{"endpoint": format!("/api/v0/{id}/:start/:end"), "description": format!("GET measurment by id in range :start to :end from {id}")},
{"endpoint": format!("/api/v0/{id}/meta"), "description": format!("GET metadata from {id}")},
{"endpoint": format!("/api/v0/{id}/aggregates"), "description": format!("GET aggregates from {id}")},
{"endpoint": format!("/api/v0/{id}"), "description": format!("POST data to {id}")},
{"endpoint": format!("/api/v0/{id}/meta"), "description": format!("POST metadata to {id}")},
],
"description": format!("Discover resources available under {id}"),
}))));
}
Ok(
warp::hyper::Response::builder()
.status(warp::http::StatusCode::NOT_FOUND)
.body(Default::default()),
)
}
pub fn post(
db: std::sync::Arc<tokio::sync::RwLock<impl IAsyncDatabase + ?Sized>>,
) -> impl Filter<Extract = impl Reply, Error = Rejection> + Clone {
warp::path!("api" / "v0" / String)
.and(warp::post())
.and(warp::body::content_length_limit(1024 * 16))
.and(warp::body::bytes())
.and(with_db(db))
.and_then(_post)
}
pub async fn _post(
id: String,
data: warp::hyper::body::Bytes,
db: std::sync::Arc<tokio::sync::RwLock<impl IAsyncDatabase + ?Sized>>,
) -> Result<impl warp::Reply, std::convert::Infallible> {
db.write().await.post(&id, &data).await;
Ok(
warp::hyper::Response::builder()
.status(warp::http::StatusCode::ACCEPTED)
.body(""),
)
}
#[tokio::test]
async fn test_get() {
let database_test_factory =
rapiddb::db::DatabaseTestFactory::new(".temp/test/sensor/test_get");
for db in database_test_factory.get_instance().values() {
let api = super::endpoints((*db).clone());
let id = "test-0";
let resp = warp::test::request()
.method("GET")
.path(&format!("/api/v0/{id}"))
.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/{id}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 200);
}
}
#[tokio::test]
async fn test_post() {
let database_test_factory =
rapiddb::db::DatabaseTestFactory::new(".temp/test/sensor/test_post");
for db in database_test_factory.get_instance().values() {
let api = super::endpoints((*db).clone());
let id = "test-0";
let resp = warp::test::request()
.method("POST")
.path(&format!("/api/v0/{id}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 411);
let resp = warp::test::request()
.method("POST")
.body(id)
.path(&format!("/api/v0/{id}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 202);
let resp = warp::test::request()
.method("POST")
.json(&serde_json::json!({ "id": &id }))
.path(&format!("/api/v0/{id}"))
.reply(&api)
.await;
assert_eq!(resp.status(), 202);
assert_eq!(resp.body().len(), 0);
let id_db = db.write().await.get_latest(id).await;
assert_eq!(
id_db,
serde_json::json!({ "id": &id }).to_string().as_bytes().to_vec()
);
}
}