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:
parent
31f28b6105
commit
a77914db73
2 changed files with 180 additions and 175 deletions
|
@ -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");
|
||||||
}
|
}
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue