1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
use std::marker::PhantomData;
#[cfg(any(feature = "websockets", feature = "acme"))]
use std::net::{Ipv6Addr, SocketAddr, SocketAddrV6};
#[cfg(feature = "acme")]
use std::time::Duration;
use clap::Args;
use crate::{Backend, CustomServer, Error, TcpService};
#[derive(Args, Debug)]
pub struct Serve<B: Backend> {
#[clap(short = 'l', long = "listen-on")]
pub listen_on: Option<u16>,
#[cfg(any(feature = "websockets", feature = "acme"))]
#[clap(long = "http")]
pub http_port: Option<SocketAddr>,
#[cfg(any(feature = "websockets", feature = "acme"))]
#[clap(long = "https")]
pub https_port: Option<SocketAddr>,
#[clap(skip)]
_backend: PhantomData<B>,
}
impl<B: Backend> Serve<B> {
pub async fn execute(&self, server: &CustomServer<B>) -> Result<(), Error> {
self.execute_with(server, ()).await
}
#[cfg_attr(
not(any(feature = "websockets", feature = "acme")),
allow(unused_variables)
)]
pub async fn execute_with<S: TcpService>(
&self,
server: &CustomServer<B>,
service: S,
) -> Result<(), Error> {
drop(env_logger::try_init());
let listen_on = self.listen_on.unwrap_or(5645);
#[cfg(any(feature = "websockets", feature = "acme"))]
{
let listen_address = self.http_port.unwrap_or_else(|| {
SocketAddr::V6(SocketAddrV6::new(Ipv6Addr::UNSPECIFIED, 80, 0, 0))
});
let task_server = server.clone();
let task_service = service.clone();
tokio::task::spawn(async move {
task_server
.listen_for_tcp_on(listen_address, task_service)
.await
});
let listen_address = self.https_port.unwrap_or_else(|| {
SocketAddr::V6(SocketAddrV6::new(Ipv6Addr::UNSPECIFIED, 443, 0, 0))
});
let task_server = server.clone();
tokio::task::spawn(async move {
task_server
.listen_for_secure_tcp_on(listen_address, service)
.await
});
#[cfg(feature = "acme")]
if server.certificate_chain().await.is_err() {
log::warn!("Server has no certificate chain. Because acme is enabled, waiting for certificate to be acquired.");
while server.certificate_chain().await.is_err() {
tokio::time::sleep(Duration::from_secs(1)).await;
}
log::info!("Server certificate acquired. Listening for certificate");
}
}
let task_server = server.clone();
tokio::task::spawn(async move { task_server.listen_on(listen_on).await });
server.listen_for_shutdown().await?;
Ok(())
}
}