2019-10-23 20:47:00 +00:00
|
|
|
use std::convert::{TryFrom, TryInto};
|
|
|
|
use std::future::Future;
|
|
|
|
use std::pin::Pin;
|
2019-10-30 22:26:26 +00:00
|
|
|
use std::rc::Rc;
|
2019-11-10 10:44:03 +00:00
|
|
|
use std::sync::atomic::{AtomicU64, Ordering};
|
2019-11-10 17:33:06 +00:00
|
|
|
use std::sync::{Arc, RwLock};
|
2019-10-23 20:47:00 +00:00
|
|
|
|
|
|
|
use http::Method as HttpMethod;
|
|
|
|
use http::Response as HttpResponse;
|
|
|
|
use js_int::UInt;
|
2019-10-24 20:34:58 +00:00
|
|
|
use reqwest::header::{HeaderValue, InvalidHeaderValue};
|
2019-10-23 20:47:00 +00:00
|
|
|
use url::Url;
|
|
|
|
|
|
|
|
use ruma_api::Endpoint;
|
|
|
|
use ruma_events::collections::all::RoomEvent;
|
2019-11-10 10:44:03 +00:00
|
|
|
use ruma_events::room::message::MessageEvent;
|
|
|
|
use ruma_events::room::message::MessageEventContent;
|
2019-10-23 20:47:00 +00:00
|
|
|
use ruma_events::Event;
|
|
|
|
pub use ruma_events::EventType;
|
2019-11-10 17:33:06 +00:00
|
|
|
use ruma_identifiers::RoomId;
|
2019-10-23 20:47:00 +00:00
|
|
|
|
|
|
|
use crate::api;
|
|
|
|
use crate::base_client::Client as BaseClient;
|
2019-10-30 18:30:55 +00:00
|
|
|
use crate::base_client::Room;
|
2019-10-23 20:47:00 +00:00
|
|
|
use crate::error::{Error, InnerError};
|
|
|
|
use crate::session::Session;
|
|
|
|
|
2019-11-10 17:33:06 +00:00
|
|
|
type RoomEventCallback = Box<dyn FnMut(&Room, &RoomEvent)>;
|
|
|
|
type RoomEventCallbackF = Box<
|
|
|
|
dyn FnMut(Arc<RwLock<Room>>, Arc<RoomEvent>) -> Pin<Box<dyn Future<Output = ()> + Send + Sync>>
|
|
|
|
+ Send
|
|
|
|
+ Sync,
|
|
|
|
>;
|
2019-10-30 18:30:55 +00:00
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
#[derive(Clone)]
|
2019-10-23 20:47:00 +00:00
|
|
|
pub struct AsyncClient {
|
|
|
|
/// The URL of the homeserver to connect to.
|
|
|
|
homeserver: Url,
|
|
|
|
/// The underlying HTTP client.
|
|
|
|
http_client: reqwest::Client,
|
|
|
|
/// User session data.
|
2019-11-10 10:44:03 +00:00
|
|
|
base_client: Arc<RwLock<BaseClient>>,
|
|
|
|
/// The transaction id.
|
|
|
|
transaction_id: Arc<AtomicU64>,
|
2019-10-31 08:17:13 +00:00
|
|
|
// /// Event callbacks
|
|
|
|
// event_callbacks: Vec<RoomEventCallback>,
|
2019-11-10 10:44:03 +00:00
|
|
|
// /// Event futures
|
|
|
|
// event_futures: Vec<RoomEventCallbackF>,
|
2019-10-23 20:47:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Default, Debug)]
|
|
|
|
pub struct AsyncClientConfig {
|
|
|
|
proxy: Option<reqwest::Proxy>,
|
|
|
|
use_sys_proxy: bool,
|
2019-10-24 20:34:58 +00:00
|
|
|
user_agent: Option<HeaderValue>,
|
2019-10-23 20:47:00 +00:00
|
|
|
disable_ssl_verification: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsyncClientConfig {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Default::default()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn proxy(mut self, proxy: &str) -> Result<Self, Error> {
|
|
|
|
if self.use_sys_proxy {
|
|
|
|
return Err(Error(InnerError::ConfigurationError(
|
|
|
|
"Using the system proxy has been previously configured.".to_string(),
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
self.proxy = Some(reqwest::Proxy::all(proxy)?);
|
|
|
|
Ok(self)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn use_sys_proxy(mut self) -> Result<Self, Error> {
|
|
|
|
if self.proxy.is_some() {
|
|
|
|
return Err(Error(InnerError::ConfigurationError(
|
|
|
|
"A proxy has already been configured.".to_string(),
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
self.use_sys_proxy = true;
|
|
|
|
Ok(self)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn disable_ssl_verification(mut self) -> Self {
|
|
|
|
self.disable_ssl_verification = true;
|
|
|
|
self
|
|
|
|
}
|
2019-10-24 20:34:58 +00:00
|
|
|
|
|
|
|
pub fn user_agent(mut self, user_agent: &str) -> Result<Self, InvalidHeaderValue> {
|
|
|
|
self.user_agent = Some(HeaderValue::from_str(user_agent)?);
|
|
|
|
Ok(self)
|
|
|
|
}
|
2019-10-23 20:47:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Default)]
|
|
|
|
pub struct SyncSettings {
|
|
|
|
pub(crate) timeout: Option<UInt>,
|
|
|
|
pub(crate) token: Option<String>,
|
|
|
|
pub(crate) full_state: Option<bool>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SyncSettings {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Default::default()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn token<S: Into<String>>(mut self, token: S) -> Self {
|
|
|
|
self.token = Some(token.into());
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn timeout<T: TryInto<UInt>>(mut self, timeout: T) -> Result<Self, js_int::TryFromIntError>
|
|
|
|
where
|
|
|
|
js_int::TryFromIntError:
|
|
|
|
std::convert::From<<T as std::convert::TryInto<js_int::UInt>>::Error>,
|
|
|
|
{
|
|
|
|
self.timeout = Some(timeout.try_into()?);
|
|
|
|
Ok(self)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn full_state(mut self, full_state: bool) -> Self {
|
|
|
|
self.full_state = Some(full_state);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-10 17:33:06 +00:00
|
|
|
use api::r0::send::send_message_event;
|
2019-10-23 20:47:00 +00:00
|
|
|
use api::r0::session::login;
|
|
|
|
use api::r0::sync::sync_events;
|
|
|
|
|
|
|
|
impl AsyncClient {
|
|
|
|
/// Creates a new client for making HTTP requests to the given homeserver.
|
2019-11-10 17:33:06 +00:00
|
|
|
pub fn new<U: TryInto<Url>>(
|
|
|
|
homeserver_url: U,
|
|
|
|
session: Option<Session>,
|
|
|
|
) -> Result<Self, Error> {
|
2019-10-24 20:34:58 +00:00
|
|
|
let config = AsyncClientConfig::new();
|
|
|
|
AsyncClient::new_with_config(homeserver_url, session, config)
|
2019-10-23 20:47:00 +00:00
|
|
|
}
|
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
pub fn new_with_config<U: TryInto<Url>>(
|
|
|
|
homeserver_url: U,
|
2019-10-23 20:47:00 +00:00
|
|
|
session: Option<Session>,
|
|
|
|
config: AsyncClientConfig,
|
2019-10-24 20:34:58 +00:00
|
|
|
) -> Result<Self, Error> {
|
2019-11-10 10:44:03 +00:00
|
|
|
let homeserver: Url = match homeserver_url.try_into() {
|
|
|
|
Ok(u) => u,
|
2019-11-10 17:33:06 +00:00
|
|
|
Err(e) => panic!("Error parsing homeserver url"),
|
2019-11-10 10:44:03 +00:00
|
|
|
};
|
|
|
|
|
2019-10-23 20:47:00 +00:00
|
|
|
let http_client = reqwest::Client::builder();
|
|
|
|
|
|
|
|
let http_client = if config.disable_ssl_verification {
|
|
|
|
http_client.danger_accept_invalid_certs(true)
|
|
|
|
} else {
|
|
|
|
http_client
|
|
|
|
};
|
|
|
|
|
|
|
|
let http_client = match config.proxy {
|
|
|
|
Some(p) => http_client.proxy(p),
|
|
|
|
None => http_client,
|
|
|
|
};
|
|
|
|
|
|
|
|
let http_client = if config.use_sys_proxy {
|
|
|
|
http_client.use_sys_proxy()
|
|
|
|
} else {
|
|
|
|
http_client
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut headers = reqwest::header::HeaderMap::new();
|
|
|
|
|
2019-10-24 20:34:58 +00:00
|
|
|
let user_agent = match config.user_agent {
|
|
|
|
Some(a) => a,
|
|
|
|
None => HeaderValue::from_static("nio-rust"),
|
|
|
|
};
|
|
|
|
|
2019-11-10 17:33:06 +00:00
|
|
|
headers.insert(reqwest::header::USER_AGENT, user_agent);
|
2019-10-23 20:47:00 +00:00
|
|
|
|
|
|
|
let http_client = http_client.default_headers(headers).build().unwrap();
|
|
|
|
|
|
|
|
Ok(Self {
|
|
|
|
homeserver,
|
|
|
|
http_client,
|
2019-11-10 10:44:03 +00:00
|
|
|
base_client: Arc::new(RwLock::new(BaseClient::new(session))),
|
|
|
|
transaction_id: Arc::new(AtomicU64::new(0)),
|
2019-10-31 08:17:13 +00:00
|
|
|
// event_callbacks: Vec::new(),
|
2019-11-10 10:44:03 +00:00
|
|
|
// event_futures: Vec::new(),
|
2019-10-23 20:47:00 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-10-31 08:17:13 +00:00
|
|
|
// pub fn add_event_callback(
|
|
|
|
// &mut self,
|
|
|
|
// event_type: EventType,
|
|
|
|
// callback: RoomEventCallback,
|
|
|
|
// ) {
|
|
|
|
// self.event_callbacks.push(callback);
|
|
|
|
// }
|
2019-10-23 20:47:00 +00:00
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
// pub fn add_event_future(
|
|
|
|
// &mut self,
|
|
|
|
// event_type: EventType,
|
|
|
|
// callback: RoomEventCallbackF,
|
|
|
|
// ) {
|
|
|
|
// self.event_futures.push(callback);
|
|
|
|
// }
|
2019-10-30 22:26:26 +00:00
|
|
|
|
2019-10-23 20:47:00 +00:00
|
|
|
pub async fn login<S: Into<String>>(
|
|
|
|
&mut self,
|
|
|
|
user: S,
|
|
|
|
password: S,
|
|
|
|
device_id: Option<S>,
|
|
|
|
) -> Result<login::Response, Error> {
|
|
|
|
let request = login::Request {
|
|
|
|
address: None,
|
|
|
|
login_type: login::LoginType::Password,
|
|
|
|
medium: None,
|
|
|
|
device_id: device_id.map(|d| d.into()),
|
|
|
|
password: password.into(),
|
|
|
|
user: user.into(),
|
|
|
|
};
|
|
|
|
|
2019-10-30 18:30:55 +00:00
|
|
|
let response = self.send(request).await?;
|
2019-11-10 10:44:03 +00:00
|
|
|
let mut client = self.base_client.write().unwrap();
|
|
|
|
client.receive_login_response(&response);
|
2019-10-23 20:47:00 +00:00
|
|
|
|
|
|
|
Ok(response)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn sync(
|
|
|
|
&mut self,
|
|
|
|
sync_settings: SyncSettings,
|
|
|
|
) -> Result<sync_events::Response, Error> {
|
|
|
|
let request = sync_events::Request {
|
|
|
|
filter: None,
|
|
|
|
since: sync_settings.token,
|
|
|
|
full_state: sync_settings.full_state,
|
|
|
|
set_presence: None,
|
|
|
|
timeout: sync_settings.timeout,
|
|
|
|
};
|
|
|
|
|
2019-10-30 18:30:55 +00:00
|
|
|
let response = self.send(request).await?;
|
2019-10-23 20:47:00 +00:00
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
let mut client = self.base_client.write().unwrap();
|
|
|
|
|
2019-10-23 21:36:57 +00:00
|
|
|
for (room_id, room) in &response.rooms.join {
|
|
|
|
let room_id = room_id.to_string();
|
|
|
|
|
|
|
|
for event in &room.state.events {
|
2019-10-30 18:30:55 +00:00
|
|
|
let event = match event.clone().into_result() {
|
|
|
|
Ok(e) => e,
|
2019-11-10 17:33:06 +00:00
|
|
|
Err(e) => continue,
|
2019-10-30 18:30:55 +00:00
|
|
|
};
|
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
client.receive_joined_state_event(&room_id, &event);
|
2019-10-23 21:36:57 +00:00
|
|
|
}
|
|
|
|
|
2019-10-23 20:47:00 +00:00
|
|
|
for event in &room.timeline.events {
|
2019-10-30 18:30:55 +00:00
|
|
|
let event = match event.clone().into_result() {
|
|
|
|
Ok(e) => e,
|
2019-11-10 17:33:06 +00:00
|
|
|
Err(e) => continue,
|
2019-10-23 20:47:00 +00:00
|
|
|
};
|
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
client.receive_joined_timeline_event(&room_id, &event);
|
2019-10-30 18:30:55 +00:00
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
let room = client.joined_rooms.get(&room_id).unwrap();
|
2019-10-30 18:30:55 +00:00
|
|
|
|
2019-10-31 08:17:13 +00:00
|
|
|
// for mut cb in &mut self.event_callbacks {
|
|
|
|
// cb(&room.lock().unwrap(), &event);
|
|
|
|
// }
|
2019-10-30 18:30:55 +00:00
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
// for mut cb in &mut self.event_futures {
|
|
|
|
// cb(room.clone(), Arc::new(event.clone())).await;
|
|
|
|
// }
|
2019-10-23 20:47:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(response)
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn send<Request: Endpoint>(&self, request: Request) -> Result<Request::Response, Error> {
|
|
|
|
let request: http::Request<Vec<u8>> = request.try_into()?;
|
|
|
|
let url = request.uri();
|
2019-11-10 17:33:06 +00:00
|
|
|
let url = self
|
|
|
|
.homeserver
|
|
|
|
.join(url.path_and_query().unwrap().as_str())
|
|
|
|
.unwrap();
|
2019-10-23 20:47:00 +00:00
|
|
|
|
|
|
|
let request_builder = match Request::METADATA.method {
|
|
|
|
HttpMethod::GET => self.http_client.get(url),
|
|
|
|
HttpMethod::POST => {
|
|
|
|
let body = request.body().clone();
|
|
|
|
self.http_client.post(url).body(body)
|
|
|
|
}
|
2019-11-10 10:44:03 +00:00
|
|
|
HttpMethod::PUT => {
|
|
|
|
let body = request.body().clone();
|
|
|
|
self.http_client.put(url).body(body)
|
|
|
|
}
|
2019-10-23 20:47:00 +00:00
|
|
|
HttpMethod::DELETE => unimplemented!(),
|
|
|
|
_ => panic!("Unsuported method"),
|
|
|
|
};
|
|
|
|
|
2019-11-10 10:44:03 +00:00
|
|
|
// let request_builder = if let Some(query) = request.uri().query() {
|
|
|
|
// request_builder.query(query)
|
|
|
|
// } else {
|
|
|
|
// request_builder
|
|
|
|
// };
|
|
|
|
|
2019-10-23 20:47:00 +00:00
|
|
|
let request_builder = if Request::METADATA.requires_authentication {
|
2019-11-10 10:44:03 +00:00
|
|
|
let client = self.base_client.read().unwrap();
|
|
|
|
|
|
|
|
if let Some(ref session) = client.session {
|
2019-10-23 20:47:00 +00:00
|
|
|
request_builder.bearer_auth(&session.access_token)
|
|
|
|
} else {
|
|
|
|
return Err(Error(InnerError::AuthenticationRequired));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
request_builder
|
|
|
|
};
|
|
|
|
|
|
|
|
let response = request_builder.send().await?;
|
|
|
|
|
|
|
|
let status = response.status();
|
|
|
|
let body = response.bytes().await?.as_ref().to_owned();
|
|
|
|
let response = HttpResponse::builder().status(status).body(body).unwrap();
|
|
|
|
let response = Request::Response::try_from(response)?;
|
|
|
|
|
|
|
|
Ok(response)
|
|
|
|
}
|
2019-11-10 10:44:44 +00:00
|
|
|
|
|
|
|
fn transaction_id(&self) -> u64 {
|
|
|
|
self.transaction_id.fetch_add(1, Ordering::SeqCst)
|
|
|
|
}
|
|
|
|
|
2019-11-10 17:33:06 +00:00
|
|
|
pub async fn room_send(
|
|
|
|
&mut self,
|
|
|
|
room_id: &str,
|
|
|
|
data: MessageEventContent,
|
|
|
|
) -> Result<send_message_event::Response, Error> {
|
2019-11-10 10:44:44 +00:00
|
|
|
let request = send_message_event::Request {
|
|
|
|
room_id: RoomId::try_from(room_id).unwrap(),
|
|
|
|
event_type: EventType::RoomMessage,
|
|
|
|
txn_id: self.transaction_id().to_string(),
|
|
|
|
data,
|
|
|
|
};
|
|
|
|
|
|
|
|
let response = self.send(request).await?;
|
|
|
|
Ok(response)
|
|
|
|
}
|
2019-10-23 20:47:00 +00:00
|
|
|
}
|