2022-02-16 21:54:00 +00:00
|
|
|
use crate::{
|
|
|
|
http::{HttpClient, Request, Response, ServerResponse},
|
|
|
|
Client, Connection, Credentials, EstablishConnectionError, UserStore,
|
|
|
|
};
|
|
|
|
use anyhow::{anyhow, Result};
|
2022-01-12 17:26:00 +00:00
|
|
|
use futures::{future::BoxFuture, stream::BoxStream, Future, StreamExt};
|
2022-02-16 21:54:00 +00:00
|
|
|
use gpui::{executor, ModelHandle, TestAppContext};
|
2021-10-04 19:07:35 +00:00
|
|
|
use parking_lot::Mutex;
|
2021-10-04 19:28:00 +00:00
|
|
|
use rpc::{proto, ConnectionId, Peer, Receipt, TypedEnvelope};
|
2022-02-16 21:54:00 +00:00
|
|
|
use std::{fmt, rc::Rc, sync::Arc};
|
2021-10-04 19:07:35 +00:00
|
|
|
|
|
|
|
pub struct FakeServer {
|
|
|
|
peer: Arc<Peer>,
|
2022-02-16 21:54:00 +00:00
|
|
|
state: Arc<Mutex<FakeServerState>>,
|
2021-10-04 19:07:35 +00:00
|
|
|
user_id: u64,
|
2022-02-16 21:54:00 +00:00
|
|
|
executor: Rc<executor::Foreground>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
struct FakeServerState {
|
|
|
|
incoming: Option<BoxStream<'static, Box<dyn proto::AnyTypedEnvelope>>>,
|
|
|
|
connection_id: Option<ConnectionId>,
|
|
|
|
forbid_connections: bool,
|
|
|
|
auth_count: usize,
|
|
|
|
access_token: usize,
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl FakeServer {
|
|
|
|
pub async fn for_client(
|
|
|
|
client_user_id: u64,
|
|
|
|
client: &mut Arc<Client>,
|
|
|
|
cx: &TestAppContext,
|
2022-02-16 21:54:00 +00:00
|
|
|
) -> Self {
|
|
|
|
let server = Self {
|
2021-10-04 19:07:35 +00:00
|
|
|
peer: Peer::new(),
|
2022-02-16 21:54:00 +00:00
|
|
|
state: Default::default(),
|
2021-10-04 19:07:35 +00:00
|
|
|
user_id: client_user_id,
|
2022-02-16 21:54:00 +00:00
|
|
|
executor: cx.foreground(),
|
|
|
|
};
|
2021-10-04 19:07:35 +00:00
|
|
|
|
|
|
|
Arc::get_mut(client)
|
|
|
|
.unwrap()
|
|
|
|
.override_authenticate({
|
2022-02-16 21:54:00 +00:00
|
|
|
let state = server.state.clone();
|
2021-10-04 19:07:35 +00:00
|
|
|
move |cx| {
|
2022-02-16 21:54:00 +00:00
|
|
|
let mut state = state.lock();
|
|
|
|
state.auth_count += 1;
|
|
|
|
let access_token = state.access_token.to_string();
|
2021-10-04 19:07:35 +00:00
|
|
|
cx.spawn(move |_| async move {
|
|
|
|
Ok(Credentials {
|
|
|
|
user_id: client_user_id,
|
|
|
|
access_token,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.override_establish_connection({
|
2022-02-16 21:54:00 +00:00
|
|
|
let peer = server.peer.clone();
|
|
|
|
let state = server.state.clone();
|
2021-10-04 19:07:35 +00:00
|
|
|
move |credentials, cx| {
|
2022-02-16 21:54:00 +00:00
|
|
|
let peer = peer.clone();
|
|
|
|
let state = state.clone();
|
2021-10-04 19:07:35 +00:00
|
|
|
let credentials = credentials.clone();
|
2022-02-16 21:54:00 +00:00
|
|
|
cx.spawn(move |cx| async move {
|
|
|
|
assert_eq!(credentials.user_id, client_user_id);
|
|
|
|
|
|
|
|
if state.lock().forbid_connections {
|
|
|
|
Err(EstablishConnectionError::Other(anyhow!(
|
|
|
|
"server is forbidding connections"
|
|
|
|
)))?
|
|
|
|
}
|
|
|
|
|
|
|
|
if credentials.access_token != state.lock().access_token.to_string() {
|
|
|
|
Err(EstablishConnectionError::Unauthorized)?
|
|
|
|
}
|
|
|
|
|
2022-04-08 14:04:03 +00:00
|
|
|
let (client_conn, server_conn, _) = Connection::in_memory(cx.background());
|
2022-03-04 21:53:40 +00:00
|
|
|
let (connection_id, io, incoming) =
|
|
|
|
peer.add_test_connection(server_conn, cx.background()).await;
|
2022-02-16 21:54:00 +00:00
|
|
|
cx.background().spawn(io).detach();
|
|
|
|
let mut state = state.lock();
|
|
|
|
state.connection_id = Some(connection_id);
|
|
|
|
state.incoming = Some(incoming);
|
|
|
|
Ok(client_conn)
|
2021-10-04 19:07:35 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
client
|
2022-03-23 18:15:36 +00:00
|
|
|
.authenticate_and_connect(false, &cx.to_async())
|
2021-10-04 19:07:35 +00:00
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
server
|
|
|
|
}
|
|
|
|
|
2022-01-12 17:02:41 +00:00
|
|
|
pub fn disconnect(&self) {
|
|
|
|
self.peer.disconnect(self.connection_id());
|
2022-02-16 21:54:00 +00:00
|
|
|
let mut state = self.state.lock();
|
|
|
|
state.connection_id.take();
|
|
|
|
state.incoming.take();
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn auth_count(&self) -> usize {
|
2022-02-16 21:54:00 +00:00
|
|
|
self.state.lock().auth_count
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn roll_access_token(&self) {
|
2022-02-16 21:54:00 +00:00
|
|
|
self.state.lock().access_token += 1;
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn forbid_connections(&self) {
|
2022-02-16 21:54:00 +00:00
|
|
|
self.state.lock().forbid_connections = true;
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn allow_connections(&self) {
|
2022-02-16 21:54:00 +00:00
|
|
|
self.state.lock().forbid_connections = false;
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 20:27:13 +00:00
|
|
|
pub fn send<T: proto::EnvelopedMessage>(&self, message: T) {
|
|
|
|
self.peer.send(self.connection_id(), message).unwrap();
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn receive<M: proto::EnvelopedMessage>(&self) -> Result<TypedEnvelope<M>> {
|
2022-02-16 21:54:00 +00:00
|
|
|
self.executor.start_waiting();
|
2021-10-04 19:07:35 +00:00
|
|
|
let message = self
|
2022-02-16 21:54:00 +00:00
|
|
|
.state
|
2021-10-04 19:07:35 +00:00
|
|
|
.lock()
|
2022-02-16 21:54:00 +00:00
|
|
|
.incoming
|
2021-10-04 19:07:35 +00:00
|
|
|
.as_mut()
|
|
|
|
.expect("not connected")
|
2022-01-12 17:26:00 +00:00
|
|
|
.next()
|
2021-10-04 19:07:35 +00:00
|
|
|
.await
|
|
|
|
.ok_or_else(|| anyhow!("other half hung up"))?;
|
2022-02-16 21:54:00 +00:00
|
|
|
self.executor.finish_waiting();
|
2021-10-04 19:07:35 +00:00
|
|
|
let type_name = message.payload_type_name();
|
|
|
|
Ok(*message
|
|
|
|
.into_any()
|
|
|
|
.downcast::<TypedEnvelope<M>>()
|
|
|
|
.unwrap_or_else(|_| {
|
|
|
|
panic!(
|
|
|
|
"fake server received unexpected message type: {:?}",
|
|
|
|
type_name
|
|
|
|
);
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn respond<T: proto::RequestMessage>(
|
|
|
|
&self,
|
|
|
|
receipt: Receipt<T>,
|
|
|
|
response: T::Response,
|
|
|
|
) {
|
2022-02-07 20:27:13 +00:00
|
|
|
self.peer.respond(receipt, response).unwrap()
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn connection_id(&self) -> ConnectionId {
|
2022-02-16 21:54:00 +00:00
|
|
|
self.state.lock().connection_id.expect("not connected")
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
2021-11-27 03:35:50 +00:00
|
|
|
|
|
|
|
pub async fn build_user_store(
|
|
|
|
&self,
|
|
|
|
client: Arc<Client>,
|
|
|
|
cx: &mut TestAppContext,
|
|
|
|
) -> ModelHandle<UserStore> {
|
|
|
|
let http_client = FakeHttpClient::with_404_response();
|
|
|
|
let user_store = cx.add_model(|cx| UserStore::new(client, http_client, cx));
|
|
|
|
assert_eq!(
|
|
|
|
self.receive::<proto::GetUsers>()
|
|
|
|
.await
|
|
|
|
.unwrap()
|
|
|
|
.payload
|
|
|
|
.user_ids,
|
|
|
|
&[self.user_id]
|
|
|
|
);
|
|
|
|
user_store
|
|
|
|
}
|
2021-10-04 19:07:35 +00:00
|
|
|
}
|
2021-10-05 00:30:11 +00:00
|
|
|
|
|
|
|
pub struct FakeHttpClient {
|
|
|
|
handler:
|
|
|
|
Box<dyn 'static + Send + Sync + Fn(Request) -> BoxFuture<'static, Result<ServerResponse>>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FakeHttpClient {
|
|
|
|
pub fn new<Fut, F>(handler: F) -> Arc<dyn HttpClient>
|
|
|
|
where
|
|
|
|
Fut: 'static + Send + Future<Output = Result<ServerResponse>>,
|
|
|
|
F: 'static + Send + Sync + Fn(Request) -> Fut,
|
|
|
|
{
|
|
|
|
Arc::new(Self {
|
|
|
|
handler: Box::new(move |req| Box::pin(handler(req))),
|
|
|
|
})
|
|
|
|
}
|
2021-11-27 02:12:12 +00:00
|
|
|
|
|
|
|
pub fn with_404_response() -> Arc<dyn HttpClient> {
|
|
|
|
Self::new(|_| async move { Ok(ServerResponse::new(404)) })
|
|
|
|
}
|
2021-10-05 00:30:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Debug for FakeHttpClient {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
f.debug_struct("FakeHttpClient").finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HttpClient for FakeHttpClient {
|
|
|
|
fn send<'a>(&'a self, req: Request) -> BoxFuture<'a, Result<Response>> {
|
|
|
|
let future = (self.handler)(req);
|
|
|
|
Box::pin(async move { future.await.map(Into::into) })
|
|
|
|
}
|
|
|
|
}
|