refactor(tvix/castore/directorysvc): factor out gRPC client gen

Move this code into a helper function, which we'll use in other places
in a bit.

Change-Id: Icae6f6dd2d4b2fa86fd2b836ddd7a4ca0e0354e7
Reviewed-on: https://cl.tvl.fyi/c/depot/+/9559
Autosubmit: flokli <flokli@flokli.de>
Reviewed-by: Connor Brewster <cbrewster@hey.com>
Tested-by: BuildkiteCI
This commit is contained in:
Florian Klink 2023-10-07 08:35:31 +02:00 committed by flokli
parent 31f28b6105
commit a77914db73
2 changed files with 180 additions and 175 deletions

View file

@ -338,87 +338,22 @@ impl DirectoryPutter for GRPCPutter {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use core::time; use core::time;
use std::thread;
use futures::StreamExt; use futures::StreamExt;
use tempfile::TempDir; use tempfile::TempDir;
use tokio::net::{UnixListener, UnixStream};
use tokio_stream::wrappers::UnixListenerStream;
use tonic::transport::{Endpoint, Server, Uri};
use crate::{ use crate::{
directoryservice::DirectoryService, directoryservice::DirectoryService,
fixtures::{DIRECTORY_A, DIRECTORY_B}, fixtures::{DIRECTORY_A, DIRECTORY_B},
proto, utils::gen_directorysvc_grpc_client,
proto::{directory_service_server::DirectoryServiceServer, GRPCDirectoryServiceWrapper},
utils::gen_directory_service,
}; };
#[test] #[tokio::test]
fn test() { async fn test() {
let tmpdir = TempDir::new().unwrap(); let tempdir = TempDir::new().expect("must succeed");
let socket_path = tmpdir.path().join("socket"); // create the GrpcDirectoryService
let directory_service = super::GRPCDirectoryService::from_client(
// Spin up a server, in a thread far away, which spawns its own tokio runtime, gen_directorysvc_grpc_client(tempdir.path()).await,
// and blocks on the task.
let socket_path_clone = socket_path.clone();
thread::spawn(move || {
// Create the runtime
let rt = tokio::runtime::Runtime::new().unwrap();
// Get a handle from this runtime
let handle = rt.handle();
let task = handle.spawn(async {
let uds = UnixListener::bind(socket_path_clone).unwrap();
let uds_stream = UnixListenerStream::new(uds);
// spin up a new DirectoryService
let mut server = Server::builder();
let router = server.add_service(DirectoryServiceServer::new(
GRPCDirectoryServiceWrapper::from(gen_directory_service()),
));
router.serve_with_incoming(uds_stream).await
});
handle.block_on(task)
});
// set up the local client runtime. This is similar to what the [tokio:test] macro desugars to.
let tester_runtime = tokio::runtime::Builder::new_current_thread()
.enable_all()
.build()
.unwrap();
// wait for the socket to be created
{
let mut socket_created = false;
for _try in 1..20 {
if socket_path.exists() {
socket_created = true;
break;
}
std::thread::sleep(time::Duration::from_millis(20))
}
assert!(
socket_created,
"expected socket path to eventually get created, but never happened"
); );
}
tester_runtime.block_on(async move {
// Create a channel, connecting to the uds at socket_path.
// The URI is unused.
let channel = Endpoint::try_from("http://[::]:50051")
.unwrap()
.connect_with_connector_lazy(tower::service_fn(move |_: Uri| {
UnixStream::connect(socket_path.clone())
}));
let grpc_client = proto::directory_service_client::DirectoryServiceClient::new(channel);
// create the GrpcDirectoryService, using the tester_runtime.
let directory_service = super::GRPCDirectoryService::from_client(grpc_client);
// try to get DIRECTORY_A should return Ok(None) // try to get DIRECTORY_A should return Ok(None)
assert_eq!( assert_eq!(
@ -534,6 +469,5 @@ mod tests {
.await .await
.expect_err("must fail"); .expect_err("must fail");
} }
});
} }
} }

View file

@ -3,11 +3,20 @@
//! Only used for testing purposes, but across crates. //! Only used for testing purposes, but across crates.
//! Should be removed once we have a better concept of a "Service registry". //! Should be removed once we have a better concept of a "Service registry".
use std::sync::Arc; use core::time;
use std::{path::Path, sync::Arc, thread};
use tokio::net::{UnixListener, UnixStream};
use tokio_stream::wrappers::UnixListenerStream;
use tonic::transport::{Channel, Endpoint, Server, Uri};
use crate::{ use crate::{
blobservice::{BlobService, MemoryBlobService}, blobservice::{BlobService, MemoryBlobService},
directoryservice::{DirectoryService, MemoryDirectoryService}, directoryservice::{DirectoryService, MemoryDirectoryService},
proto::{
directory_service_client::DirectoryServiceClient,
directory_service_server::DirectoryServiceServer, GRPCDirectoryServiceWrapper,
},
}; };
pub fn gen_blob_service() -> Arc<dyn BlobService> { pub fn gen_blob_service() -> Arc<dyn BlobService> {
@ -17,3 +26,65 @@ pub fn gen_blob_service() -> Arc<dyn BlobService> {
pub fn gen_directory_service() -> Arc<dyn DirectoryService> { pub fn gen_directory_service() -> Arc<dyn DirectoryService> {
Arc::new(MemoryDirectoryService::default()) Arc::new(MemoryDirectoryService::default())
} }
/// This will spawn a separate thread, with its own tokio runtime, and start a gRPC server there.
/// Once it's listening, it'll start a gRPC client from the original thread, and return it.
/// FUTUREWORK: accept a closure to create the service, so we can test this with different ones.
#[allow(dead_code)]
pub(crate) async fn gen_directorysvc_grpc_client(tmpdir: &Path) -> DirectoryServiceClient<Channel> {
let socket_path = tmpdir.join("socket");
// Spin up a server, in a thread far away, which spawns its own tokio runtime,
// and blocks on the task.
let socket_path_clone = socket_path.clone();
thread::spawn(move || {
// Create the runtime
let rt = tokio::runtime::Runtime::new().unwrap();
// Get a handle from this runtime
let handle = rt.handle();
let task = handle.spawn(async {
let uds = UnixListener::bind(socket_path_clone).unwrap();
let uds_stream = UnixListenerStream::new(uds);
// spin up a new DirectoryService
let mut server = Server::builder();
let router = server.add_service(DirectoryServiceServer::new(
GRPCDirectoryServiceWrapper::from(gen_directory_service()),
));
router.serve_with_incoming(uds_stream).await
});
handle.block_on(task)
});
// wait for the socket to be created
// TODO: pass around FDs instead?
{
let mut socket_created = false;
for _try in 1..20 {
if socket_path.exists() {
socket_created = true;
break;
}
tokio::time::sleep(time::Duration::from_millis(20)).await;
}
assert!(
socket_created,
"expected socket path to eventually get created, but never happened"
);
}
// Create a channel, connecting to the uds at socket_path.
// The URI is unused.
let channel = Endpoint::try_from("http://[::]:50051")
.unwrap()
.connect_with_connector_lazy(tower::service_fn(move |_: Uri| {
UnixStream::connect(socket_path.clone())
}));
let grpc_client = DirectoryServiceClient::new(channel);
grpc_client
}