2021-06-04 16:09:20 +00:00
|
|
|
// Copyright 2021 The Matrix.org Foundation C.I.C.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
use std::sync::Arc;
|
|
|
|
|
|
|
|
use dashmap::DashMap;
|
2021-06-07 12:47:59 +00:00
|
|
|
use matrix_sdk_common::uuid::Uuid;
|
|
|
|
use ruma::{DeviceId, UserId};
|
2021-06-04 16:09:20 +00:00
|
|
|
|
|
|
|
use super::{event_enums::OutgoingContent, sas::content_to_request, Sas, Verification};
|
|
|
|
use crate::{OutgoingRequest, RoomMessageRequest};
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub struct VerificationCache {
|
2021-06-08 14:13:14 +00:00
|
|
|
verification: Arc<DashMap<UserId, DashMap<String, Verification>>>,
|
2021-06-04 16:09:20 +00:00
|
|
|
outgoing_requests: Arc<DashMap<Uuid, OutgoingRequest>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl VerificationCache {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Self { verification: DashMap::new().into(), outgoing_requests: DashMap::new().into() }
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
2021-06-04 16:49:08 +00:00
|
|
|
#[allow(dead_code)]
|
2021-06-04 16:09:20 +00:00
|
|
|
pub fn is_empty(&self) -> bool {
|
2021-06-08 14:13:14 +00:00
|
|
|
self.verification.iter().all(|m| m.is_empty())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn insert(&self, verification: impl Into<Verification>) {
|
|
|
|
let verification = verification.into();
|
|
|
|
|
|
|
|
self.verification
|
|
|
|
.entry(verification.other_user().to_owned())
|
|
|
|
.or_insert_with(DashMap::new)
|
|
|
|
.insert(verification.flow_id().to_owned(), verification);
|
2021-06-04 16:09:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn insert_sas(&self, sas: Sas) {
|
2021-06-08 14:13:14 +00:00
|
|
|
self.insert(sas);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get(&self, sender: &UserId, flow_id: &str) -> Option<Verification> {
|
|
|
|
self.verification.get(sender).and_then(|m| m.get(flow_id).map(|v| v.clone()))
|
2021-06-04 16:09:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn outgoing_requests(&self) -> Vec<OutgoingRequest> {
|
|
|
|
self.outgoing_requests.iter().map(|r| (*r).clone()).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn garbage_collect(&self) -> Vec<OutgoingRequest> {
|
2021-06-08 14:13:14 +00:00
|
|
|
for user_verification in self.verification.iter() {
|
|
|
|
user_verification.retain(|_, s| !(s.is_done() || s.is_cancelled()));
|
|
|
|
}
|
|
|
|
|
|
|
|
self.verification.retain(|_, m| !m.is_empty());
|
2021-06-04 16:09:20 +00:00
|
|
|
|
|
|
|
self.verification
|
|
|
|
.iter()
|
2021-06-08 14:13:14 +00:00
|
|
|
.flat_map(|v| {
|
|
|
|
let requests: Vec<OutgoingRequest> = v
|
|
|
|
.value()
|
|
|
|
.iter()
|
|
|
|
.filter_map(|s| {
|
|
|
|
if let Verification::SasV1(s) = s.value() {
|
|
|
|
s.cancel_if_timed_out().map(|r| OutgoingRequest {
|
|
|
|
request_id: r.request_id(),
|
|
|
|
request: Arc::new(r.into()),
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2021-06-04 16:09:20 +00:00
|
|
|
})
|
2021-06-08 14:13:14 +00:00
|
|
|
.collect();
|
|
|
|
|
|
|
|
requests
|
2021-06-04 16:09:20 +00:00
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
2021-06-08 14:13:14 +00:00
|
|
|
pub fn get_sas(&self, user_id: &UserId, flow_id: &str) -> Option<Sas> {
|
|
|
|
self.get(user_id, flow_id).and_then(|v| {
|
|
|
|
if let Verification::SasV1(sas) = v {
|
2021-06-04 16:09:20 +00:00
|
|
|
Some(sas.clone())
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_request(&self, request: OutgoingRequest) {
|
|
|
|
self.outgoing_requests.insert(request.request_id, request);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn queue_up_content(
|
|
|
|
&self,
|
|
|
|
recipient: &UserId,
|
|
|
|
recipient_device: &DeviceId,
|
|
|
|
content: OutgoingContent,
|
|
|
|
) {
|
|
|
|
match content {
|
|
|
|
OutgoingContent::ToDevice(c) => {
|
|
|
|
let request = content_to_request(recipient, recipient_device.to_owned(), c);
|
|
|
|
let request_id = request.txn_id;
|
|
|
|
|
|
|
|
let request = OutgoingRequest { request_id, request: Arc::new(request.into()) };
|
|
|
|
|
|
|
|
self.outgoing_requests.insert(request_id, request);
|
|
|
|
}
|
|
|
|
|
|
|
|
OutgoingContent::Room(r, c) => {
|
|
|
|
let request_id = Uuid::new_v4();
|
|
|
|
|
|
|
|
let request = OutgoingRequest {
|
|
|
|
request: Arc::new(
|
|
|
|
RoomMessageRequest { room_id: r, txn_id: request_id, content: c }.into(),
|
|
|
|
),
|
|
|
|
request_id,
|
|
|
|
};
|
|
|
|
|
|
|
|
self.outgoing_requests.insert(request_id, request);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn mark_request_as_sent(&self, uuid: &Uuid) {
|
|
|
|
self.outgoing_requests.remove(uuid);
|
|
|
|
}
|
|
|
|
}
|