session-open-group-server/src/rpc.rs

116 lines
5 KiB
Rust
Raw Normal View History

2021-03-12 05:11:12 +01:00
use serde::Deserialize;
2021-03-12 05:46:06 +01:00
use warp::{Rejection, reply::Response};
2021-03-12 05:11:12 +01:00
2021-03-12 06:40:24 +01:00
use super::errors::Error;
2021-03-12 05:11:12 +01:00
use super::handlers;
use super::lsrpc;
use super::storage;
#[derive(Debug, Deserialize)]
pub struct QueryOptions {
pub limit: Option<u16>,
pub from_server_id: Option<i64>
}
2021-03-12 05:46:06 +01:00
pub async fn handle_rpc_call(rpc_call: lsrpc::RpcCall, pool: &storage::DatabaseConnectionPool) -> Result<Response, Rejection> {
2021-03-12 05:11:12 +01:00
// Check that the endpoint is a valid URI
let uri = match rpc_call.endpoint.parse::<http::Uri>() {
Ok(uri) => uri,
Err(e) => {
println!("Couldn't parse URI from: {:?} due to error: {:?}.", rpc_call.endpoint, e);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
};
// Switch on the HTTP method
match rpc_call.method.as_ref() {
2021-03-12 05:32:41 +01:00
"GET" => return handle_get_request(rpc_call, uri, pool).await,
"POST" => return handle_post_request(rpc_call, uri, pool).await,
"DELETE" => return handle_delete_request(rpc_call, uri, pool).await,
2021-03-12 05:11:12 +01:00
_ => {
println!("Ignoring RPC call with invalid or unused HTTP method: {:?}.", rpc_call.method);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
}
}
2021-03-12 05:46:06 +01:00
async fn handle_get_request(rpc_call: lsrpc::RpcCall, uri: http::Uri, pool: &storage::DatabaseConnectionPool) -> Result<Response, Rejection> {
2021-03-12 05:11:12 +01:00
// Parse query options if needed
let mut query_options = QueryOptions { limit : None, from_server_id : None };
if let Some(query) = uri.query() {
query_options = match serde_json::from_str(&query) {
Ok(query_options) => query_options,
Err(e) => {
println!("Couldn't parse query options from: {:?} due to error: {:?}.", query, e);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
};
}
// Switch on the path
match uri.path() {
"/messages" => return handlers::get_messages(query_options, pool).await,
"/deleted_messages" => return handlers::get_deleted_messages(query_options, pool).await,
"/moderators" => return handlers::get_moderators(pool).await,
"/block_list" => return handlers::get_banned_public_keys(pool).await,
"/member_count" => return handlers::get_member_count(pool).await,
_ => {
println!("Ignoring RPC call with invalid or unused endpoint: {:?}.", rpc_call.endpoint);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
}
}
2021-03-12 05:46:06 +01:00
async fn handle_post_request(rpc_call: lsrpc::RpcCall, uri: http::Uri, pool: &storage::DatabaseConnectionPool) -> Result<Response, Rejection> {
2021-03-12 05:11:12 +01:00
match uri.path() {
"/messages" => {
let message = match serde_json::from_str(&rpc_call.body) {
Ok(query_options) => query_options,
Err(e) => {
println!("Couldn't parse message from: {:?} due to error: {:?}.", rpc_call.body, e);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
};
return handlers::insert_message(message, pool).await;
},
2021-03-12 05:32:41 +01:00
"/block_list" => {
let public_key = rpc_call.body;
return handlers::ban(public_key, pool).await;
},
2021-03-12 05:11:12 +01:00
_ => {
println!("Ignoring RPC call with invalid or unused endpoint: {:?}.", rpc_call.endpoint);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
}
}
2021-03-12 05:46:06 +01:00
async fn handle_delete_request(rpc_call: lsrpc::RpcCall, uri: http::Uri, pool: &storage::DatabaseConnectionPool) -> Result<Response, Rejection> {
2021-03-12 05:11:12 +01:00
// DELETE /messages/:server_id
if uri.path().starts_with("/messages") {
let components: Vec<&str> = uri.path()[1..].split("/").collect(); // Drop the leading slash and split on subsequent slashes
if components.len() != 2 {
println!("Invalid endpoint: {:?}.", rpc_call.endpoint);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
let server_id: i64 = match components[1].parse() {
Ok(server_id) => server_id,
2021-03-12 05:46:06 +01:00
Err(_) => {
2021-03-12 05:11:12 +01:00
println!("Invalid endpoint: {:?}.", rpc_call.endpoint);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
};
return handlers::delete_message(server_id, pool).await;
}
// DELETE /block_list/:public_key
if uri.path().starts_with("/block_list") {
let components: Vec<&str> = uri.path()[1..].split("/").collect(); // Drop the leading slash and split on subsequent slashes
if components.len() != 2 {
println!("Invalid endpoint: {:?}.", rpc_call.endpoint);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}
let public_key = components[1].to_string();
return handlers::unban(public_key, pool).await;
}
// Unrecognized endpoint
2021-03-12 05:32:41 +01:00
println!("Ignoring RPC call with invalid or unused endpoint: {:?}.", rpc_call.endpoint);
2021-03-12 06:40:24 +01:00
return Err(warp::reject::custom(Error::InvalidRequest));
2021-03-12 05:11:12 +01:00
}