linkerd2/proxy/tests/support/proxy.rs

186 lines
5.3 KiB
Rust

use support::*;
use std::sync::{Arc, Mutex};
use convert::TryFrom;
pub fn new() -> Proxy {
Proxy::new()
}
#[derive(Debug)]
pub struct Proxy {
controller: Option<controller::Listening>,
inbound: Option<server::Listening>,
outbound: Option<server::Listening>,
metrics_flush_interval: Option<Duration>,
}
#[derive(Debug)]
pub struct Listening {
pub control: SocketAddr,
pub inbound: SocketAddr,
pub outbound: SocketAddr,
pub outbound_server: Option<server::Listening>,
pub inbound_server: Option<server::Listening>,
shutdown: Shutdown,
}
impl Proxy {
pub fn new() -> Self {
Proxy {
controller: None,
inbound: None,
outbound: None,
metrics_flush_interval: None,
}
}
pub fn controller(mut self, c: controller::Listening) -> Self {
self.controller = Some(c);
self
}
pub fn inbound(mut self, s: server::Listening) -> Self {
self.inbound = Some(s);
self
}
pub fn outbound(mut self, s: server::Listening) -> Self {
self.outbound = Some(s);
self
}
pub fn metrics_flush_interval(mut self, dur: Duration) -> Self {
self.metrics_flush_interval = Some(dur);
self
}
pub fn run(self) -> Listening {
self.run_with_test_env(config::TestEnv::new())
}
pub fn run_with_test_env(self, mut env: config::TestEnv) -> Listening {
run(self, env)
}
}
#[derive(Clone, Debug)]
struct MockOriginalDst(Arc<Mutex<DstInner>>);
#[derive(Debug, Default)]
struct DstInner {
inbound_orig_addr: Option<SocketAddr>,
inbound_local_addr: Option<SocketAddr>,
outbound_orig_addr: Option<SocketAddr>,
outbound_local_addr: Option<SocketAddr>,
}
impl conduit_proxy::GetOriginalDst for MockOriginalDst {
fn get_original_dst(&self, sock: &TcpStream) -> Option<SocketAddr> {
sock.local_addr()
.ok()
.and_then(|local| {
let inner = self.0.lock().unwrap();
if inner.inbound_local_addr == Some(local) {
inner.inbound_orig_addr
} else if inner.outbound_local_addr == Some(local) {
inner.outbound_orig_addr
} else {
None
}
})
}
}
fn run(proxy: Proxy, mut env: config::TestEnv) -> Listening {
use self::conduit_proxy::config;
let controller = proxy.controller.expect("proxy controller missing");
let inbound = proxy.inbound;
let outbound = proxy.outbound;
let mut mock_orig_dst = DstInner::default();
env.put(config::ENV_CONTROL_URL, format!("tcp://{}", controller.addr));
env.put(config::ENV_PRIVATE_LISTENER, "tcp://127.0.0.1:0".to_owned());
if let Some(ref inbound) = inbound {
env.put(config::ENV_PRIVATE_FORWARD, format!("tcp://{}", inbound.addr));
mock_orig_dst.inbound_orig_addr = Some(inbound.addr);
}
if let Some(ref outbound) = outbound {
mock_orig_dst.outbound_orig_addr = Some(outbound.addr);
}
env.put(config::ENV_PUBLIC_LISTENER, "tcp://127.0.0.1:0".to_owned());
env.put(config::ENV_CONTROL_LISTENER, "tcp://127.0.0.1:0".to_owned());
env.put(config::ENV_POD_NAMESPACE, "test".to_owned());
let mut config = config::Config::try_from(&env).unwrap();
// TODO: We currently can't use `config::ENV_METRICS_FLUSH_INTERVAL_SECS`
// because we need to be able to set the flush interval to a fraction of a
// second. We should change config::ENV_METRICS_FLUSH_INTERVAL_SECS so that
// it can support this.
if let Some(dur) = proxy.metrics_flush_interval {
config.metrics_flush_interval = dur;
}
let (running_tx, running_rx) = oneshot::channel();
let (tx, mut rx) = shutdown_signal();
::std::thread::Builder::new()
.name("support proxy".into())
.spawn(move || {
let _c = controller;
let mock_orig_dst = MockOriginalDst(Arc::new(Mutex::new(mock_orig_dst)));
let main = conduit_proxy::Main::new(config, mock_orig_dst.clone());
let control_addr = main.control_addr();
let inbound_addr = main.inbound_addr();
let outbound_addr = main.outbound_addr();
{
let mut inner = mock_orig_dst.0.lock().unwrap();
inner.inbound_local_addr = Some(inbound_addr);
inner.outbound_local_addr = Some(outbound_addr);
}
// slip the running tx into the shutdown future, since the first time
// the shutdown future is polled, that means all of the proxy is now
// running.
let addrs = (control_addr, inbound_addr, outbound_addr);
let mut running = Some((running_tx, addrs));
let on_shutdown = future::poll_fn(move || {
if let Some((tx, addrs)) = running.take() {
let _ = tx.send(addrs);
}
rx.poll()
});
main.run_until(on_shutdown);
})
.unwrap();
let (control_addr, inbound_addr, outbound_addr) = running_rx.wait().unwrap();
Listening {
control: control_addr,
inbound: inbound_addr,
outbound: outbound_addr,
outbound_server: outbound,
inbound_server: inbound,
shutdown: tx,
}
}