use std::net::{SocketAddr, ToSocketAddrs};
use std::sync::Arc;
use roa::{App, Endpoint, Executor, Server, State};
use super::TcpIncoming;
pub trait Listener {
type Server;
fn bind(self, addr: impl ToSocketAddrs) -> std::io::Result<(SocketAddr, Self::Server)>;
fn listen(
self,
addr: impl ToSocketAddrs,
callback: impl Fn(SocketAddr),
) -> std::io::Result<Self::Server>;
fn run(self) -> std::io::Result<(SocketAddr, Self::Server)>;
}
impl<S, E> Listener for App<S, Arc<E>>
where
S: State,
E: for<'a> Endpoint<'a, S>,
{
type Server = Server<TcpIncoming, Self, Executor>;
fn bind(self, addr: impl ToSocketAddrs) -> std::io::Result<(SocketAddr, Self::Server)> {
let incoming = TcpIncoming::bind(addr)?;
let local_addr = incoming.local_addr();
Ok((local_addr, self.accept(incoming)))
}
fn listen(
self,
addr: impl ToSocketAddrs,
callback: impl Fn(SocketAddr),
) -> std::io::Result<Self::Server> {
let (addr, server) = self.bind(addr)?;
callback(addr);
Ok(server)
}
fn run(self) -> std::io::Result<(SocketAddr, Self::Server)> {
self.bind("127.0.0.1:0")
}
}
#[cfg(test)]
mod tests {
use std::error::Error;
use roa::http::StatusCode;
use roa::App;
use super::Listener;
use crate::Exec;
#[tokio::test]
async fn incoming() -> Result<(), Box<dyn Error>> {
let (addr, server) = App::with_exec((), Exec).end(()).run()?;
tokio::task::spawn(server);
let resp = reqwest::get(&format!("http://{}", addr)).await?;
assert_eq!(StatusCode::OK, resp.status());
Ok(())
}
}