rust-sdk: Add a method to get the ougtoing requests

This commit is contained in:
Damir Jelić 2021-02-11 11:11:25 +01:00
parent 0b9be11d85
commit 628f530633
3 changed files with 165 additions and 25 deletions

View File

@ -15,12 +15,16 @@ tokio = { version = "1.1.1", default_features = false, features = ["rt-multi-thr
serde_json = "1.0.61" serde_json = "1.0.61"
thiserror = "1.0.23" thiserror = "1.0.23"
http = "0.2.3" http = "0.2.3"
uniffi = "0.7.0"
[dependencies.matrix-sdk-crypto] [dependencies.matrix-sdk-crypto]
git = "https://github.com/matrix-org/matrix-rust-sdk/" git = "https://github.com/matrix-org/matrix-rust-sdk/"
features = ["sled_cryptostore"] features = ["sled_cryptostore"]
[dependencies.uniffi]
version = "0.7.0"
git = "https://github.com/mozilla/uniffi-rs"
branch = "tagged-unions"
[dependencies.ruma] [dependencies.ruma]
version = "0.0.2" version = "0.0.2"
git = "https://github.com/ruma/ruma" git = "https://github.com/ruma/ruma"

View File

@ -5,17 +5,25 @@ use std::{
}; };
use futures::executor::block_on; use futures::executor::block_on;
use http::Response;
use serde_json::json;
use tokio::{runtime::Runtime, time::sleep}; use tokio::{runtime::Runtime, time::sleep};
use matrix_sdk_common::{ use matrix_sdk_common::{
api::r0::sync::sync_events::{ api::r0::{
DeviceLists as RumaDeviceLists, ToDevice, keys::{
claim_keys::Response as KeysClaimResponse, get_keys::Response as KeysQueryResponse,
upload_keys::Response as KeysUploadResponse,
},
sync::sync_events::{DeviceLists as RumaDeviceLists, ToDevice},
}, },
identifiers::{DeviceKeyAlgorithm, Error as RumaIdentifierError, UserId}, identifiers::{DeviceKeyAlgorithm, Error as RumaIdentifierError, UserId},
uuid::Uuid,
UInt, UInt,
}; };
use matrix_sdk_crypto::{ use matrix_sdk_crypto::{
store::CryptoStoreError as InnerStoreError, OlmMachine as InnerMachine, ToDeviceRequest, store::CryptoStoreError as InnerStoreError, IncomingResponse, OlmError,
OlmMachine as InnerMachine, OutgoingRequest, ToDeviceRequest,
}; };
pub struct OlmMachine { pub struct OlmMachine {
@ -45,6 +53,40 @@ impl Into<RumaDeviceLists> for DeviceLists {
} }
} }
enum OwnedResponse {
KeysClaim(KeysClaimResponse),
KeysUpload(KeysUploadResponse),
KeysQuery(KeysQueryResponse),
}
impl From<KeysClaimResponse> for OwnedResponse {
fn from(response: KeysClaimResponse) -> Self {
OwnedResponse::KeysClaim(response)
}
}
impl From<KeysQueryResponse> for OwnedResponse {
fn from(response: KeysQueryResponse) -> Self {
OwnedResponse::KeysQuery(response)
}
}
impl From<KeysUploadResponse> for OwnedResponse {
fn from(response: KeysUploadResponse) -> Self {
OwnedResponse::KeysUpload(response)
}
}
impl<'a> Into<IncomingResponse<'a>> for &'a OwnedResponse {
fn into(self) -> IncomingResponse<'a> {
match self {
OwnedResponse::KeysClaim(r) => IncomingResponse::KeysClaim(r),
OwnedResponse::KeysQuery(r) => IncomingResponse::KeysQuery(r),
OwnedResponse::KeysUpload(r) => IncomingResponse::KeysUpload(r),
}
}
}
#[derive(Debug, thiserror::Error)] #[derive(Debug, thiserror::Error)]
pub enum MachineCreationError { pub enum MachineCreationError {
#[error(transparent)] #[error(transparent)]
@ -57,6 +99,8 @@ pub enum MachineCreationError {
pub enum CryptoStoreError { pub enum CryptoStoreError {
#[error(transparent)] #[error(transparent)]
CryptoStore(#[from] InnerStoreError), CryptoStore(#[from] InnerStoreError),
#[error(transparent)]
OlmError(#[from] OlmError),
} }
pub enum RequestType { pub enum RequestType {
@ -78,28 +122,81 @@ pub struct Sas {
pub request: Request, pub request: Request,
} }
pub struct Request { pub enum Request {
pub request_id: String, ToDevice {
pub request_type: RequestType, request_id: String,
pub request_body: String, event_type: String,
body: String,
},
KeysUpload {
request_id: String,
body: String,
},
KeysQuery {
request_id: String,
body: String,
},
} }
impl From<ToDeviceRequest> for Request { impl From<OutgoingRequest> for Request {
fn from(r: ToDeviceRequest) -> Self { fn from(r: OutgoingRequest) -> Self {
Request { use matrix_sdk_crypto::OutgoingRequests::*;
request_id: r.txn_id_string(),
request_type: RequestType::ToDevice, match r.request() {
request_body: serde_json::to_string(&r.messages).unwrap(), KeysUpload(u) => {
let body = json!({
"device_keys": u.device_keys,
"one_time_keys": u.one_time_keys,
});
Request::KeysUpload {
request_id: r.request_id().to_string(),
body: serde_json::to_string(&body)
.expect("Can't serialize keys upload request"),
}
}
KeysQuery(k) => {
let body = json!({
"device_keys": k.device_keys,
});
Request::KeysQuery {
request_id: r.request_id().to_string(),
body: serde_json::to_string(&body).expect("Can't serialize keys query request"),
}
}
ToDeviceRequest(t) => Request::from(t),
SignatureUpload(_) => todo!(),
RoomMessage(_) => todo!(),
} }
} }
} }
// fn response_from_string(body: &str) -> Response<Vec<u8>> { impl From<ToDeviceRequest> for Request {
// Response::builder() fn from(r: ToDeviceRequest) -> Self {
// .status(200) Request::ToDevice {
// .body(body.as_bytes().to_vec()) request_id: r.txn_id_string(),
// .expect("Can't create HTTP response") event_type: r.event_type.to_string(),
// } body: serde_json::to_string(&r.messages).unwrap(),
}
}
}
impl From<&ToDeviceRequest> for Request {
fn from(r: &ToDeviceRequest) -> Self {
Request::ToDevice {
request_id: r.txn_id_string(),
event_type: r.event_type.to_string(),
body: serde_json::to_string(&r.messages).unwrap(),
}
}
}
fn response_from_string(body: &str) -> Response<Vec<u8>> {
Response::builder()
.status(200)
.body(body.as_bytes().to_vec())
.expect("Can't create HTTP response")
}
impl OlmMachine { impl OlmMachine {
pub fn new(user_id: &str, device_id: &str, path: &str) -> Result<Self, MachineCreationError> { pub fn new(user_id: &str, device_id: &str, path: &str) -> Result<Self, MachineCreationError> {
@ -175,6 +272,28 @@ impl OlmMachine {
}) })
} }
pub fn mark_request_as_sent(
&self,
request_id: &str,
request_type: RequestType,
response_body: &str,
) -> Result<(), CryptoStoreError> {
let id = Uuid::parse_str(request_id).expect("Can't parse request id");
let response = response_from_string(&response_body);
let response: OwnedResponse = match request_type {
RequestType::KeysUpload => KeysUploadResponse::try_from(response).map(Into::into),
RequestType::KeysQuery => KeysQueryResponse::try_from(response).map(Into::into),
RequestType::ToDevice => KeysClaimResponse::try_from(response).map(Into::into),
}
.expect("Can't convert json string to response");
block_on(self.inner.mark_request_as_sent(&id, &response))?;
Ok(())
}
pub fn start_verification(&self, device: &Device) -> Result<Sas, CryptoStoreError> { pub fn start_verification(&self, device: &Device) -> Result<Sas, CryptoStoreError> {
let user_id = UserId::try_from(device.user_id.clone()).unwrap(); let user_id = UserId::try_from(device.user_id.clone()).unwrap();
let device_id = device.device_id.as_str().into(); let device_id = device.device_id.as_str().into();
@ -190,6 +309,13 @@ impl OlmMachine {
}) })
} }
pub fn outgoing_requests(&self) -> Vec<Request> {
block_on(self.inner.outgoing_requests())
.into_iter()
.map(|r| r.into())
.collect()
}
pub fn receive_sync_changes( pub fn receive_sync_changes(
&self, &self,
events: &str, events: &str,

View File

@ -9,6 +9,7 @@ enum MachineCreationError {
[Error] [Error]
enum CryptoStoreError { enum CryptoStoreError {
"CryptoStore", "CryptoStore",
"OlmError",
}; };
dictionary DeviceLists { dictionary DeviceLists {
@ -29,10 +30,11 @@ dictionary Sas {
Request request; Request request;
}; };
dictionary Request { [TaggedUnion]
string request_id; interface Request {
RequestType request_type; ToDevice(string request_id, string event_type, string body);
string request_body; KeysUpload(string request_id, string body);
KeysQuery(string request_id, string body);
}; };
enum RequestType { enum RequestType {
@ -41,6 +43,7 @@ enum RequestType {
"ToDevice", "ToDevice",
}; };
[Threadsafe]
interface OlmMachine { interface OlmMachine {
[Throws=MachineCreationError] [Throws=MachineCreationError]
constructor([ByRef] string user_id, [ByRef] string device_id, [ByRef] string path); constructor([ByRef] string user_id, [ByRef] string device_id, [ByRef] string path);
@ -52,11 +55,18 @@ interface OlmMachine {
record<DOMString, string> identity_keys(); record<DOMString, string> identity_keys();
string user_id(); string user_id();
string slow_user_id();
string device_id(); string device_id();
Device? get_device([ByRef] string user_id, [ByRef] string device_id); Device? get_device([ByRef] string user_id, [ByRef] string device_id);
sequence<Device> get_user_devices([ByRef] string user_id); sequence<Device> get_user_devices([ByRef] string user_id);
sequence<Request> outgoing_requests();
[Throws=CryptoStoreError]
void mark_request_as_sent(
[ByRef] string request_id,
RequestType request_type,
[ByRef] string response
);
[Throws=CryptoStoreError] [Throws=CryptoStoreError]
Sas start_verification([ByRef] Device device); Sas start_verification([ByRef] Device device);