use crate::cli::CF; use crate::dbs::DB; use crate::err::Error; use crate::net::input::bytes_to_utf8; use crate::net::output; use crate::net::params::{Param, Params}; use crate::net::session; use bytes::Bytes; use serde::Deserialize; use std::str; use surrealdb::dbs::Session; use surrealdb::sql::Value; use warp::path; use warp::Filter; const MAX: u64 = 1024 * 16; // 16 KiB #[derive(Default, Deserialize, Debug, Clone)] struct Query { pub limit: Option, pub start: Option, } #[allow(opaque_hidden_inferred_bound)] pub fn config() -> impl Filter + Clone { // ------------------------------ // Routes for OPTIONS // ------------------------------ let base = warp::path("key"); // Set opts method let opts = base.and(warp::options()).map(warp::reply); // ------------------------------ // Routes for a table // ------------------------------ // Set select method let select = warp::any() .and(warp::get()) .and(warp::header::(http::header::ACCEPT.as_str())) .and(path!("key" / Param).and(warp::path::end())) .and(warp::query()) .and(session::build()) .and_then(select_all); // Set create method let create = warp::any() .and(warp::post()) .and(warp::header::(http::header::ACCEPT.as_str())) .and(path!("key" / Param).and(warp::path::end())) .and(warp::body::content_length_limit(MAX)) .and(warp::body::bytes()) .and(warp::query()) .and(session::build()) .and_then(create_all); // Set delete method let delete = warp::any() .and(warp::delete()) .and(warp::header::(http::header::ACCEPT.as_str())) .and(path!("key" / Param).and(warp::path::end())) .and(warp::query()) .and(session::build()) .and_then(delete_all); // Specify route let all = select.or(create).or(delete); // ------------------------------ // Routes for a thing // ------------------------------ // Set select method let select = warp::any() .and(warp::get()) .and(warp::header::(http::header::ACCEPT.as_str())) .and(path!("key" / Param / Param).and(warp::path::end())) .and(session::build()) .and_then(select_one); // Set create method let create = warp::any() .and(warp::post()) .and(warp::header::(http::header::ACCEPT.as_str())) .and(path!("key" / Param / Param).and(warp::path::end())) .and(warp::body::content_length_limit(MAX)) .and(warp::body::bytes()) .and(warp::query()) .and(session::build()) .and_then(create_one); // Set update method let update = warp::any() .and(warp::put()) .and(warp::header::(http::header::ACCEPT.as_str())) .and(path!("key" / Param / Param).and(warp::path::end())) .and(warp::body::content_length_limit(MAX)) .and(warp::body::bytes()) .and(warp::query()) .and(session::build()) .and_then(update_one); // Set modify method let modify = warp::any() .and(warp::patch()) .and(warp::header::(http::header::ACCEPT.as_str())) .and(path!("key" / Param / Param).and(warp::path::end())) .and(warp::body::content_length_limit(MAX)) .and(warp::body::bytes()) .and(warp::query()) .and(session::build()) .and_then(modify_one); // Set delete method let delete = warp::any() .and(warp::delete()) .and(warp::header::(http::header::ACCEPT.as_str())) .and(path!("key" / Param / Param).and(warp::path::end())) .and(warp::query()) .and(session::build()) .and_then(delete_one); // Specify route let one = select.or(create).or(update).or(modify).or(delete); // ------------------------------ // All routes // ------------------------------ // Specify route opts.or(all).or(one) } // ------------------------------ // Routes for a table // ------------------------------ async fn select_all( output: String, table: Param, query: Query, session: Session, ) -> Result { // Get the datastore reference let db = DB.get().unwrap(); // Get local copy of options let opt = CF.get().unwrap(); // Specify the request statement let sql = format!( "SELECT * FROM type::table($table) LIMIT {l} START {s}", l = query.limit.unwrap_or_else(|| String::from("100")), s = query.start.unwrap_or_else(|| String::from("0")), ); // Specify the request variables let vars = map! { String::from("table") => Value::from(table), }; // Execute the query and return the result match db.execute(sql.as_str(), &session, Some(vars), opt.strict).await { Ok(ref res) => match output.as_ref() { // Simple serialization "application/json" => Ok(output::json(&res)), "application/cbor" => Ok(output::cbor(&res)), "application/pack" => Ok(output::pack(&res)), // Internal serialization "application/cork" => Ok(output::cork(&res)), // An incorrect content-type was requested _ => Err(warp::reject::custom(Error::InvalidType)), }, // There was an error when executing the query Err(err) => Err(warp::reject::custom(Error::from(err))), } } async fn create_all( output: String, table: Param, body: Bytes, params: Params, session: Session, ) -> Result { // Get the datastore reference let db = DB.get().unwrap(); // Get local copy of options let opt = CF.get().unwrap(); // Convert the HTTP request body let data = bytes_to_utf8(&body)?; // Parse the request body as JSON match surrealdb::sql::json(data) { Ok(data) => { // Specify the request statement let sql = "CREATE type::table($table) CONTENT $data"; // Specify the request variables let vars = map! { String::from("table") => Value::from(table), String::from("data") => data, => params.parse() }; // Execute the query and return the result match db.execute(sql, &session, Some(vars), opt.strict).await { Ok(res) => match output.as_ref() { // Simple serialization "application/json" => Ok(output::json(&res)), "application/cbor" => Ok(output::cbor(&res)), "application/pack" => Ok(output::pack(&res)), // Internal serialization "application/cork" => Ok(output::cork(&res)), // An incorrect content-type was requested _ => Err(warp::reject::custom(Error::InvalidType)), }, // There was an error when executing the query Err(err) => Err(warp::reject::custom(Error::from(err))), } } Err(_) => Err(warp::reject::custom(Error::Request)), } } async fn delete_all( output: String, table: Param, params: Params, session: Session, ) -> Result { // Get the datastore reference let db = DB.get().unwrap(); // Get local copy of options let opt = CF.get().unwrap(); // Specify the request statement let sql = "DELETE type::table($table) RETURN BEFORE"; // Specify the request variables let vars = map! { String::from("table") => Value::from(table), => params.parse() }; // Execute the query and return the result match db.execute(sql, &session, Some(vars), opt.strict).await { Ok(res) => match output.as_ref() { // Simple serialization "application/json" => Ok(output::json(&res)), "application/cbor" => Ok(output::cbor(&res)), "application/pack" => Ok(output::pack(&res)), // Internal serialization "application/cork" => Ok(output::cork(&res)), // An incorrect content-type was requested _ => Err(warp::reject::custom(Error::InvalidType)), }, // There was an error when executing the query Err(err) => Err(warp::reject::custom(Error::from(err))), } } // ------------------------------ // Routes for a thing // ------------------------------ async fn select_one( output: String, table: Param, id: Param, session: Session, ) -> Result { // Get the datastore reference let db = DB.get().unwrap(); // Get local copy of options let opt = CF.get().unwrap(); // Specify the request statement let sql = "SELECT * FROM type::thing($table, $id)"; // Parse the Record ID as a SurrealQL value let rid = match surrealdb::sql::json(&id) { Ok(id) => id, Err(_) => Value::from(id), }; // Specify the request variables let vars = map! { String::from("table") => Value::from(table), String::from("id") => rid, }; // Execute the query and return the result match db.execute(sql, &session, Some(vars), opt.strict).await { Ok(res) => match output.as_ref() { // Simple serialization "application/json" => Ok(output::json(&res)), "application/cbor" => Ok(output::cbor(&res)), "application/pack" => Ok(output::pack(&res)), // Internal serialization "application/cork" => Ok(output::cork(&res)), // An incorrect content-type was requested _ => Err(warp::reject::custom(Error::InvalidType)), }, // There was an error when executing the query Err(err) => Err(warp::reject::custom(Error::from(err))), } } async fn create_one( output: String, table: Param, id: Param, body: Bytes, params: Params, session: Session, ) -> Result { // Get the datastore reference let db = DB.get().unwrap(); // Get local copy of options let opt = CF.get().unwrap(); // Convert the HTTP request body let data = bytes_to_utf8(&body)?; // Parse the Record ID as a SurrealQL value let rid = match surrealdb::sql::json(&id) { Ok(id) => id, Err(_) => Value::from(id), }; // Parse the request body as JSON match surrealdb::sql::json(data) { Ok(data) => { // Specify the request statement let sql = "CREATE type::thing($table, $id) CONTENT $data"; // Specify the request variables let vars = map! { String::from("table") => Value::from(table), String::from("id") => rid, String::from("data") => data, => params.parse() }; // Execute the query and return the result match db.execute(sql, &session, Some(vars), opt.strict).await { Ok(res) => match output.as_ref() { // Simple serialization "application/json" => Ok(output::json(&res)), "application/cbor" => Ok(output::cbor(&res)), "application/pack" => Ok(output::pack(&res)), // Internal serialization "application/cork" => Ok(output::cork(&res)), // An incorrect content-type was requested _ => Err(warp::reject::custom(Error::InvalidType)), }, // There was an error when executing the query Err(err) => Err(warp::reject::custom(Error::from(err))), } } Err(_) => Err(warp::reject::custom(Error::Request)), } } async fn update_one( output: String, table: Param, id: Param, body: Bytes, params: Params, session: Session, ) -> Result { // Get the datastore reference let db = DB.get().unwrap(); // Get local copy of options let opt = CF.get().unwrap(); // Convert the HTTP request body let data = bytes_to_utf8(&body)?; // Parse the Record ID as a SurrealQL value let rid = match surrealdb::sql::json(&id) { Ok(id) => id, Err(_) => Value::from(id), }; // Parse the request body as JSON match surrealdb::sql::json(data) { Ok(data) => { // Specify the request statement let sql = "UPDATE type::thing($table, $id) CONTENT $data"; // Specify the request variables let vars = map! { String::from("table") => Value::from(table), String::from("id") => rid, String::from("data") => data, => params.parse() }; // Execute the query and return the result match db.execute(sql, &session, Some(vars), opt.strict).await { Ok(res) => match output.as_ref() { // Simple serialization "application/json" => Ok(output::json(&res)), "application/cbor" => Ok(output::cbor(&res)), "application/pack" => Ok(output::pack(&res)), // Internal serialization "application/cork" => Ok(output::cork(&res)), // An incorrect content-type was requested _ => Err(warp::reject::custom(Error::InvalidType)), }, // There was an error when executing the query Err(err) => Err(warp::reject::custom(Error::from(err))), } } Err(_) => Err(warp::reject::custom(Error::Request)), } } async fn modify_one( output: String, table: Param, id: Param, body: Bytes, params: Params, session: Session, ) -> Result { // Get the datastore reference let db = DB.get().unwrap(); // Get local copy of options let opt = CF.get().unwrap(); // Convert the HTTP request body let data = bytes_to_utf8(&body)?; // Parse the Record ID as a SurrealQL value let rid = match surrealdb::sql::json(&id) { Ok(id) => id, Err(_) => Value::from(id), }; // Parse the request body as JSON match surrealdb::sql::json(data) { Ok(data) => { // Specify the request statement let sql = "UPDATE type::thing($table, $id) MERGE $data"; // Specify the request variables let vars = map! { String::from("table") => Value::from(table), String::from("id") => rid, String::from("data") => data, => params.parse() }; // Execute the query and return the result match db.execute(sql, &session, Some(vars), opt.strict).await { Ok(res) => match output.as_ref() { // Simple serialization "application/json" => Ok(output::json(&res)), "application/cbor" => Ok(output::cbor(&res)), "application/pack" => Ok(output::pack(&res)), // Internal serialization "application/cork" => Ok(output::cork(&res)), // An incorrect content-type was requested _ => Err(warp::reject::custom(Error::InvalidType)), }, // There was an error when executing the query Err(err) => Err(warp::reject::custom(Error::from(err))), } } Err(_) => Err(warp::reject::custom(Error::Request)), } } async fn delete_one( output: String, table: Param, id: Param, params: Params, session: Session, ) -> Result { // Get the datastore reference let db = DB.get().unwrap(); // Get local copy of options let opt = CF.get().unwrap(); // Specify the request statement let sql = "DELETE type::thing($table, $id) RETURN BEFORE"; // Parse the Record ID as a SurrealQL value let rid = match surrealdb::sql::json(&id) { Ok(id) => id, Err(_) => Value::from(id), }; // Specify the request variables let vars = map! { String::from("table") => Value::from(table), String::from("id") => rid, => params.parse() }; // Execute the query and return the result match db.execute(sql, &session, Some(vars), opt.strict).await { Ok(res) => match output.as_ref() { // Simple serialization "application/json" => Ok(output::json(&res)), "application/cbor" => Ok(output::cbor(&res)), "application/pack" => Ok(output::pack(&res)), // Internal serialization "application/cork" => Ok(output::cork(&res)), // An incorrect content-type was requested _ => Err(warp::reject::custom(Error::InvalidType)), }, // There was an error when executing the query Err(err) => Err(warp::reject::custom(Error::from(err))), } }