mirror of
https://github.com/htrefil/rkvm.git
synced 2025-01-13 20:01:29 +01:00
Make timeouts configurable
This commit is contained in:
parent
347dafeae7
commit
16f4a72b30
15 changed files with 255 additions and 38 deletions
90
Cargo.lock
generated
90
Cargo.lock
generated
|
@ -299,6 +299,12 @@ dependencies = [
|
||||||
"termcolor",
|
"termcolor",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "equivalent"
|
||||||
|
version = "1.0.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "errno"
|
name = "errno"
|
||||||
version = "0.3.9"
|
version = "0.3.9"
|
||||||
|
@ -437,6 +443,12 @@ version = "0.3.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b"
|
checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "hashbrown"
|
||||||
|
version = "0.14.5"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "heck"
|
name = "heck"
|
||||||
version = "0.5.0"
|
version = "0.5.0"
|
||||||
|
@ -482,6 +494,16 @@ version = "2.1.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4"
|
checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "indexmap"
|
||||||
|
version = "2.2.6"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "168fb715dda47215e360912c096649d23d58bf392ac62f73919e831745e40f26"
|
||||||
|
dependencies = [
|
||||||
|
"equivalent",
|
||||||
|
"hashbrown",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "inotify"
|
name = "inotify"
|
||||||
version = "0.10.2"
|
version = "0.10.2"
|
||||||
|
@ -804,6 +826,7 @@ version = "0.6.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"clap",
|
"clap",
|
||||||
"env_logger",
|
"env_logger",
|
||||||
|
"rkvm-config",
|
||||||
"rkvm-input",
|
"rkvm-input",
|
||||||
"rkvm-net",
|
"rkvm-net",
|
||||||
"rustls-pemfile",
|
"rustls-pemfile",
|
||||||
|
@ -811,11 +834,20 @@ dependencies = [
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-rustls",
|
"tokio-rustls",
|
||||||
"toml",
|
"toml 0.5.11",
|
||||||
"tracing",
|
"tracing",
|
||||||
"tracing-subscriber",
|
"tracing-subscriber",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rkvm-config"
|
||||||
|
version = "0.1.0"
|
||||||
|
dependencies = [
|
||||||
|
"humantime",
|
||||||
|
"serde",
|
||||||
|
"toml 0.8.14",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rkvm-input"
|
name = "rkvm-input"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
|
@ -854,7 +886,9 @@ version = "0.6.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"clap",
|
"clap",
|
||||||
"env_logger",
|
"env_logger",
|
||||||
|
"humantime",
|
||||||
"rand",
|
"rand",
|
||||||
|
"rkvm-config",
|
||||||
"rkvm-input",
|
"rkvm-input",
|
||||||
"rkvm-net",
|
"rkvm-net",
|
||||||
"rustls-pemfile",
|
"rustls-pemfile",
|
||||||
|
@ -863,7 +897,7 @@ dependencies = [
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-rustls",
|
"tokio-rustls",
|
||||||
"toml",
|
"toml 0.5.11",
|
||||||
"tracing",
|
"tracing",
|
||||||
"tracing-subscriber",
|
"tracing-subscriber",
|
||||||
]
|
]
|
||||||
|
@ -954,6 +988,15 @@ dependencies = [
|
||||||
"syn",
|
"syn",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "serde_spanned"
|
||||||
|
version = "0.6.6"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "79e674e01f999af37c49f70a6ede167a8a60b2503e56c5599532a65baa5969a0"
|
||||||
|
dependencies = [
|
||||||
|
"serde",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "sha2"
|
name = "sha2"
|
||||||
version = "0.10.8"
|
version = "0.10.8"
|
||||||
|
@ -1145,6 +1188,40 @@ dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "toml"
|
||||||
|
version = "0.8.14"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "6f49eb2ab21d2f26bd6db7bf383edc527a7ebaee412d17af4d40fdccd442f335"
|
||||||
|
dependencies = [
|
||||||
|
"serde",
|
||||||
|
"serde_spanned",
|
||||||
|
"toml_datetime",
|
||||||
|
"toml_edit",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "toml_datetime"
|
||||||
|
version = "0.6.6"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "4badfd56924ae69bcc9039335b2e017639ce3f9b001c393c1b2d1ef846ce2cbf"
|
||||||
|
dependencies = [
|
||||||
|
"serde",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "toml_edit"
|
||||||
|
version = "0.22.15"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d59a3a72298453f564e2b111fa896f8d07fabb36f51f06d7e875fc5e0b5a3ef1"
|
||||||
|
dependencies = [
|
||||||
|
"indexmap",
|
||||||
|
"serde",
|
||||||
|
"serde_spanned",
|
||||||
|
"toml_datetime",
|
||||||
|
"winnow",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "tracing"
|
name = "tracing"
|
||||||
version = "0.1.40"
|
version = "0.1.40"
|
||||||
|
@ -1429,3 +1506,12 @@ name = "windows_x86_64_msvc"
|
||||||
version = "0.52.6"
|
version = "0.52.6"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec"
|
checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "winnow"
|
||||||
|
version = "0.6.13"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "59b5e5f6c299a3c7890b876a2a587f3115162487e704907d9b6cd29473052ba1"
|
||||||
|
dependencies = [
|
||||||
|
"memchr",
|
||||||
|
]
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
[workspace]
|
[workspace]
|
||||||
resolver = "2"
|
resolver = "2"
|
||||||
members = ["rkvm-client", "rkvm-server", "rkvm-input", "rkvm-net", "rkvm-certificate-gen"]
|
members = ["rkvm-client", "rkvm-server", "rkvm-input", "rkvm-net", "rkvm-certificate-gen", "rkvm-config"]
|
||||||
|
|
|
@ -6,3 +6,13 @@ certificate = "/etc/rkvm/certificate.pem"
|
||||||
#
|
#
|
||||||
# Change this to your own value before deploying rkvm.
|
# Change this to your own value before deploying rkvm.
|
||||||
password = "123456789"
|
password = "123456789"
|
||||||
|
|
||||||
|
# Optional values.
|
||||||
|
# Make sure these match what you have in your server's config.
|
||||||
|
|
||||||
|
# Message read timeout.
|
||||||
|
# read-timeout = "500ms"
|
||||||
|
# Message write timeout.
|
||||||
|
# write-timeout = "500ms"
|
||||||
|
# TLS handshake timeout.
|
||||||
|
# tls-timeout = "500ms"
|
|
@ -12,3 +12,13 @@ key = "/etc/rkvm/key.pem"
|
||||||
#
|
#
|
||||||
# Change this to your own value before deploying rkvm.
|
# Change this to your own value before deploying rkvm.
|
||||||
password = "123456789"
|
password = "123456789"
|
||||||
|
|
||||||
|
# Optional values.
|
||||||
|
# Make sure these match what you have in your client's config.
|
||||||
|
|
||||||
|
# Message write timeout. Increase this if you are getting timeout errors.
|
||||||
|
# read-timeout = "500ms"
|
||||||
|
# Message write timeout. Increase this if you are getting timeout errors.
|
||||||
|
# write-timeout = "500ms"
|
||||||
|
# TLS handshake timeout.
|
||||||
|
# tls-timeout = "500ms"
|
|
@ -10,7 +10,6 @@ edition = "2021"
|
||||||
[dependencies]
|
[dependencies]
|
||||||
tokio = { version = "1.0.1", features = ["macros", "time", "fs", "net", "signal", "rt-multi-thread", "sync"] }
|
tokio = { version = "1.0.1", features = ["macros", "time", "fs", "net", "signal", "rt-multi-thread", "sync"] }
|
||||||
rkvm-input = { path = "../rkvm-input" }
|
rkvm-input = { path = "../rkvm-input" }
|
||||||
rkvm-net = { path = "../rkvm-net" }
|
|
||||||
serde = { version = "1.0.117", features = ["derive"] }
|
serde = { version = "1.0.117", features = ["derive"] }
|
||||||
toml = "0.5.7"
|
toml = "0.5.7"
|
||||||
env_logger = "0.8.1"
|
env_logger = "0.8.1"
|
||||||
|
@ -20,6 +19,8 @@ tokio-rustls = "0.24.0"
|
||||||
rustls-pemfile = "1.0.2"
|
rustls-pemfile = "1.0.2"
|
||||||
tracing = "0.1.37"
|
tracing = "0.1.37"
|
||||||
tracing-subscriber = { version = "0.3.17", features = ["env-filter"] }
|
tracing-subscriber = { version = "0.3.17", features = ["env-filter"] }
|
||||||
|
rkvm-net = { path = "../rkvm-net" }
|
||||||
|
rkvm-config = { path = "../rkvm-config" }
|
||||||
|
|
||||||
[package.metadata.rpm]
|
[package.metadata.rpm]
|
||||||
package = "rkvm-client"
|
package = "rkvm-client"
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
use rkvm_config::Timeout;
|
||||||
use rkvm_input::writer::Writer;
|
use rkvm_input::writer::Writer;
|
||||||
use rkvm_net::auth::{AuthChallenge, AuthStatus};
|
use rkvm_net::auth::{AuthChallenge, AuthStatus};
|
||||||
use rkvm_net::message::Message;
|
use rkvm_net::message::Message;
|
||||||
|
@ -31,6 +32,7 @@ pub async fn run(
|
||||||
port: u16,
|
port: u16,
|
||||||
connector: TlsConnector,
|
connector: TlsConnector,
|
||||||
password: &str,
|
password: &str,
|
||||||
|
timeout: Timeout,
|
||||||
) -> Result<(), Error> {
|
) -> Result<(), Error> {
|
||||||
// Intentionally don't impose any timeout for TCP connect.
|
// Intentionally don't impose any timeout for TCP connect.
|
||||||
let stream = match hostname {
|
let stream = match hostname {
|
||||||
|
@ -42,18 +44,15 @@ pub async fn run(
|
||||||
|
|
||||||
tracing::info!("Connected to server");
|
tracing::info!("Connected to server");
|
||||||
|
|
||||||
let stream = rkvm_net::timeout(
|
let stream = rkvm_net::timeout(timeout.tls, connector.connect(hostname.clone(), stream))
|
||||||
rkvm_net::TLS_TIMEOUT,
|
.await
|
||||||
connector.connect(hostname.clone(), stream),
|
.map_err(Error::Network)?;
|
||||||
)
|
|
||||||
.await
|
|
||||||
.map_err(Error::Network)?;
|
|
||||||
|
|
||||||
tracing::info!("TLS connected");
|
tracing::info!("TLS connected");
|
||||||
|
|
||||||
let mut stream = BufStream::with_capacity(1024, 1024, stream);
|
let mut stream = BufStream::with_capacity(1024, 1024, stream);
|
||||||
|
|
||||||
rkvm_net::timeout(rkvm_net::WRITE_TIMEOUT, async {
|
rkvm_net::timeout(timeout.write, async {
|
||||||
Version::CURRENT.encode(&mut stream).await?;
|
Version::CURRENT.encode(&mut stream).await?;
|
||||||
stream.flush().await?;
|
stream.flush().await?;
|
||||||
|
|
||||||
|
@ -62,7 +61,7 @@ pub async fn run(
|
||||||
.await
|
.await
|
||||||
.map_err(Error::Network)?;
|
.map_err(Error::Network)?;
|
||||||
|
|
||||||
let version = rkvm_net::timeout(rkvm_net::READ_TIMEOUT, Version::decode(&mut stream))
|
let version = rkvm_net::timeout(timeout.read, Version::decode(&mut stream))
|
||||||
.await
|
.await
|
||||||
.map_err(Error::Network)?;
|
.map_err(Error::Network)?;
|
||||||
|
|
||||||
|
@ -73,13 +72,13 @@ pub async fn run(
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
let challenge = rkvm_net::timeout(rkvm_net::READ_TIMEOUT, AuthChallenge::decode(&mut stream))
|
let challenge = rkvm_net::timeout(timeout.read, AuthChallenge::decode(&mut stream))
|
||||||
.await
|
.await
|
||||||
.map_err(Error::Network)?;
|
.map_err(Error::Network)?;
|
||||||
|
|
||||||
let response = challenge.respond(password);
|
let response = challenge.respond(password);
|
||||||
|
|
||||||
rkvm_net::timeout(rkvm_net::WRITE_TIMEOUT, async {
|
rkvm_net::timeout(timeout.write, async {
|
||||||
response.encode(&mut stream).await?;
|
response.encode(&mut stream).await?;
|
||||||
stream.flush().await?;
|
stream.flush().await?;
|
||||||
|
|
||||||
|
@ -88,7 +87,7 @@ pub async fn run(
|
||||||
.await
|
.await
|
||||||
.map_err(Error::Network)?;
|
.map_err(Error::Network)?;
|
||||||
|
|
||||||
let status = rkvm_net::timeout(rkvm_net::READ_TIMEOUT, AuthStatus::decode(&mut stream))
|
let status = rkvm_net::timeout(timeout.read, AuthStatus::decode(&mut stream))
|
||||||
.await
|
.await
|
||||||
.map_err(Error::Network)?;
|
.map_err(Error::Network)?;
|
||||||
|
|
||||||
|
@ -101,7 +100,7 @@ pub async fn run(
|
||||||
|
|
||||||
let mut start = Instant::now();
|
let mut start = Instant::now();
|
||||||
|
|
||||||
let mut interval = time::interval(rkvm_net::PING_INTERVAL + rkvm_net::READ_TIMEOUT);
|
let mut interval = time::interval(rkvm_net::PING_INTERVAL + timeout.read);
|
||||||
let mut writers = HashMap::new();
|
let mut writers = HashMap::new();
|
||||||
|
|
||||||
// Interval ticks immediately after creation.
|
// Interval ticks immediately after creation.
|
||||||
|
@ -191,7 +190,7 @@ pub async fn run(
|
||||||
start = Instant::now();
|
start = Instant::now();
|
||||||
interval.reset();
|
interval.reset();
|
||||||
|
|
||||||
rkvm_net::timeout(rkvm_net::WRITE_TIMEOUT, async {
|
rkvm_net::timeout(timeout.write, async {
|
||||||
Pong.encode(&mut stream).await?;
|
Pong.encode(&mut stream).await?;
|
||||||
stream.flush().await?;
|
stream.flush().await?;
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
use rkvm_config::Timeout;
|
||||||
use serde::de::{self, Visitor};
|
use serde::de::{self, Visitor};
|
||||||
use serde::{Deserialize, Deserializer};
|
use serde::{Deserialize, Deserializer};
|
||||||
use std::fmt::{self, Formatter};
|
use std::fmt::{self, Formatter};
|
||||||
|
@ -12,6 +13,8 @@ pub struct Config {
|
||||||
pub server: Server,
|
pub server: Server,
|
||||||
pub certificate: PathBuf,
|
pub certificate: PathBuf,
|
||||||
pub password: String,
|
pub password: String,
|
||||||
|
#[serde(flatten)]
|
||||||
|
pub timeout: Timeout,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Server {
|
pub struct Server {
|
||||||
|
|
|
@ -57,7 +57,7 @@ async fn main() -> ExitCode {
|
||||||
};
|
};
|
||||||
|
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
result = client::run(&config.server.hostname, config.server.port, connector, &config.password) => {
|
result = client::run(&config.server.hostname, config.server.port, connector, &config.password, config.timeout) => {
|
||||||
if let Err(err) = result {
|
if let Err(err) = result {
|
||||||
tracing::error!("Error: {}", err);
|
tracing::error!("Error: {}", err);
|
||||||
return ExitCode::FAILURE;
|
return ExitCode::FAILURE;
|
||||||
|
|
13
rkvm-config/Cargo.toml
Normal file
13
rkvm-config/Cargo.toml
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
[package]
|
||||||
|
name = "rkvm-config"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
humantime = "2.1.0"
|
||||||
|
serde = { version = "1.0.204", features = ["derive"] }
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
toml = "0.8.14"
|
97
rkvm-config/src/lib.rs
Normal file
97
rkvm-config/src/lib.rs
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
use serde::de::{Deserializer, Error, Visitor};
|
||||||
|
use serde::Deserialize;
|
||||||
|
use std::fmt::{self, Formatter};
|
||||||
|
use std::str::FromStr;
|
||||||
|
use std::time::Duration;
|
||||||
|
|
||||||
|
#[derive(Deserialize, Clone, Copy, PartialEq, Eq, Debug)]
|
||||||
|
pub struct Timeout {
|
||||||
|
#[serde(
|
||||||
|
rename = "read-timeout",
|
||||||
|
default = "default_timeout",
|
||||||
|
deserialize_with = "deserialize_duration"
|
||||||
|
)]
|
||||||
|
pub read: Duration,
|
||||||
|
#[serde(
|
||||||
|
rename = "write-timeout",
|
||||||
|
default = "default_timeout",
|
||||||
|
deserialize_with = "deserialize_duration"
|
||||||
|
)]
|
||||||
|
pub write: Duration,
|
||||||
|
#[serde(
|
||||||
|
rename = "tls-timeout",
|
||||||
|
default = "default_timeout",
|
||||||
|
deserialize_with = "deserialize_duration"
|
||||||
|
)]
|
||||||
|
pub tls: Duration,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn default_timeout() -> Duration {
|
||||||
|
Duration::from_millis(500)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_duration<'de, D>(deserializer: D) -> Result<Duration, D::Error>
|
||||||
|
where
|
||||||
|
D: Deserializer<'de>,
|
||||||
|
{
|
||||||
|
struct DurationVisitor;
|
||||||
|
|
||||||
|
impl Visitor<'_> for DurationVisitor {
|
||||||
|
type Value = Duration;
|
||||||
|
|
||||||
|
fn expecting(&self, formatter: &mut Formatter) -> fmt::Result {
|
||||||
|
write!(formatter, "a duration of time (for example \"500ms\")")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
|
||||||
|
where
|
||||||
|
E: Error,
|
||||||
|
{
|
||||||
|
humantime::Duration::from_str(v)
|
||||||
|
.map_err(E::custom)
|
||||||
|
.map(Into::into)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
deserializer.deserialize_str(DurationVisitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn timeout_deserialize() {
|
||||||
|
let parsed = toml::from_str::<Timeout>(
|
||||||
|
r#"
|
||||||
|
read-timeout = "1s"
|
||||||
|
write-timeout = "200ms"
|
||||||
|
tls-timeout = "500ms"
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
parsed,
|
||||||
|
Timeout {
|
||||||
|
read: Duration::from_secs(1),
|
||||||
|
write: Duration::from_millis(200),
|
||||||
|
tls: Duration::from_millis(500),
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn timeout_missing_values() {
|
||||||
|
let parsed = toml::from_str::<Timeout>("").unwrap();
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
parsed,
|
||||||
|
Timeout {
|
||||||
|
read: default_timeout(),
|
||||||
|
write: default_timeout(),
|
||||||
|
tls: default_timeout(),
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,15 +19,6 @@ use tokio::time;
|
||||||
|
|
||||||
pub const PING_INTERVAL: Duration = Duration::from_secs(1);
|
pub const PING_INTERVAL: Duration = Duration::from_secs(1);
|
||||||
|
|
||||||
// Message read timeout (does not apply to updates, only auth negotiation and replies).
|
|
||||||
pub const READ_TIMEOUT: Duration = Duration::from_millis(500);
|
|
||||||
|
|
||||||
// Message write timeout (applies to all messages).
|
|
||||||
pub const WRITE_TIMEOUT: Duration = Duration::from_millis(500);
|
|
||||||
|
|
||||||
// TLS negotiation timeout.
|
|
||||||
pub const TLS_TIMEOUT: Duration = Duration::from_millis(500);
|
|
||||||
|
|
||||||
#[derive(Deserialize, Serialize, Debug)]
|
#[derive(Deserialize, Serialize, Debug)]
|
||||||
pub enum Update {
|
pub enum Update {
|
||||||
CreateDevice {
|
CreateDevice {
|
||||||
|
|
|
@ -20,8 +20,10 @@ slab = "0.4.8"
|
||||||
rand = "0.8.5"
|
rand = "0.8.5"
|
||||||
tracing-subscriber = { version = "0.3.17", features = ["env-filter"] }
|
tracing-subscriber = { version = "0.3.17", features = ["env-filter"] }
|
||||||
tracing = "0.1.37"
|
tracing = "0.1.37"
|
||||||
|
humantime = "2.1.0"
|
||||||
rkvm-net = { path = "../rkvm-net" }
|
rkvm-net = { path = "../rkvm-net" }
|
||||||
rkvm-input = { path = "../rkvm-input" }
|
rkvm-input = { path = "../rkvm-input" }
|
||||||
|
rkvm-config = { path = "../rkvm-config" }
|
||||||
|
|
||||||
[package.metadata.rpm]
|
[package.metadata.rpm]
|
||||||
package = "rkvm-server"
|
package = "rkvm-server"
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
use rkvm_config::Timeout;
|
||||||
use rkvm_input::key::{Button, Key, Keyboard};
|
use rkvm_input::key::{Button, Key, Keyboard};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use std::collections::HashSet;
|
use std::collections::HashSet;
|
||||||
|
@ -13,6 +14,8 @@ pub struct Config {
|
||||||
pub password: String,
|
pub password: String,
|
||||||
pub switch_keys: HashSet<SwitchKey>,
|
pub switch_keys: HashSet<SwitchKey>,
|
||||||
pub propagate_switch_keys: Option<bool>,
|
pub propagate_switch_keys: Option<bool>,
|
||||||
|
#[serde(flatten)]
|
||||||
|
pub timeout: Timeout,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Deserialize, Clone, Copy, PartialEq, Eq, Hash)]
|
#[derive(Deserialize, Clone, Copy, PartialEq, Eq, Hash)]
|
||||||
|
|
|
@ -71,7 +71,7 @@ async fn main() -> ExitCode {
|
||||||
let propagate_switch_keys = config.propagate_switch_keys.unwrap_or(true);
|
let propagate_switch_keys = config.propagate_switch_keys.unwrap_or(true);
|
||||||
|
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
result = server::run(config.listen, acceptor, &config.password, &switch_keys, propagate_switch_keys) => {
|
result = server::run(config.listen, acceptor, &config.password, &switch_keys, propagate_switch_keys, config.timeout) => {
|
||||||
if let Err(err) = result {
|
if let Err(err) = result {
|
||||||
tracing::error!("Error: {}", err);
|
tracing::error!("Error: {}", err);
|
||||||
return ExitCode::FAILURE;
|
return ExitCode::FAILURE;
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
use rkvm_config::Timeout;
|
||||||
use rkvm_input::abs::{AbsAxis, AbsInfo};
|
use rkvm_input::abs::{AbsAxis, AbsInfo};
|
||||||
use rkvm_input::event::Event;
|
use rkvm_input::event::Event;
|
||||||
use rkvm_input::key::{Key, KeyEvent};
|
use rkvm_input::key::{Key, KeyEvent};
|
||||||
|
@ -39,6 +40,7 @@ pub async fn run(
|
||||||
password: &str,
|
password: &str,
|
||||||
switch_keys: &HashSet<Key>,
|
switch_keys: &HashSet<Key>,
|
||||||
propagate_switch_keys: bool,
|
propagate_switch_keys: bool,
|
||||||
|
timeout: Timeout,
|
||||||
) -> Result<(), Error> {
|
) -> Result<(), Error> {
|
||||||
let listener = TcpListener::bind(&listen).await.map_err(Error::Network)?;
|
let listener = TcpListener::bind(&listen).await.map_err(Error::Network)?;
|
||||||
tracing::info!("Listening on {}", listen);
|
tracing::info!("Listening on {}", listen);
|
||||||
|
@ -92,7 +94,7 @@ pub async fn run(
|
||||||
async move {
|
async move {
|
||||||
tracing::info!("Connected");
|
tracing::info!("Connected");
|
||||||
|
|
||||||
match client(init_updates, receiver, stream, acceptor, &password).await {
|
match client(init_updates, receiver, stream, acceptor, &password, timeout).await {
|
||||||
Ok(()) => tracing::info!("Disconnected"),
|
Ok(()) => tracing::info!("Disconnected"),
|
||||||
Err(err) => tracing::error!("Disconnected: {}", err),
|
Err(err) => tracing::error!("Disconnected: {}", err),
|
||||||
}
|
}
|
||||||
|
@ -308,13 +310,14 @@ async fn client(
|
||||||
stream: TcpStream,
|
stream: TcpStream,
|
||||||
acceptor: TlsAcceptor,
|
acceptor: TlsAcceptor,
|
||||||
password: &str,
|
password: &str,
|
||||||
|
timeout: Timeout,
|
||||||
) -> Result<(), ClientError> {
|
) -> Result<(), ClientError> {
|
||||||
let stream = rkvm_net::timeout(rkvm_net::TLS_TIMEOUT, acceptor.accept(stream)).await?;
|
let stream = rkvm_net::timeout(timeout.tls, acceptor.accept(stream)).await?;
|
||||||
tracing::info!("TLS connected");
|
tracing::info!("TLS connected");
|
||||||
|
|
||||||
let mut stream = BufStream::with_capacity(1024, 1024, stream);
|
let mut stream = BufStream::with_capacity(1024, 1024, stream);
|
||||||
|
|
||||||
rkvm_net::timeout(rkvm_net::WRITE_TIMEOUT, async {
|
rkvm_net::timeout(timeout.write, async {
|
||||||
Version::CURRENT.encode(&mut stream).await?;
|
Version::CURRENT.encode(&mut stream).await?;
|
||||||
stream.flush().await?;
|
stream.flush().await?;
|
||||||
|
|
||||||
|
@ -322,7 +325,7 @@ async fn client(
|
||||||
})
|
})
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
let version = rkvm_net::timeout(rkvm_net::READ_TIMEOUT, Version::decode(&mut stream)).await?;
|
let version = rkvm_net::timeout(timeout.read, Version::decode(&mut stream)).await?;
|
||||||
if version != Version::CURRENT {
|
if version != Version::CURRENT {
|
||||||
return Err(ClientError::Version {
|
return Err(ClientError::Version {
|
||||||
server: Version::CURRENT,
|
server: Version::CURRENT,
|
||||||
|
@ -332,7 +335,7 @@ async fn client(
|
||||||
|
|
||||||
let challenge = AuthChallenge::generate().await?;
|
let challenge = AuthChallenge::generate().await?;
|
||||||
|
|
||||||
rkvm_net::timeout(rkvm_net::WRITE_TIMEOUT, async {
|
rkvm_net::timeout(timeout.write, async {
|
||||||
challenge.encode(&mut stream).await?;
|
challenge.encode(&mut stream).await?;
|
||||||
stream.flush().await?;
|
stream.flush().await?;
|
||||||
|
|
||||||
|
@ -340,14 +343,13 @@ async fn client(
|
||||||
})
|
})
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
let response =
|
let response = rkvm_net::timeout(timeout.read, AuthResponse::decode(&mut stream)).await?;
|
||||||
rkvm_net::timeout(rkvm_net::READ_TIMEOUT, AuthResponse::decode(&mut stream)).await?;
|
|
||||||
let status = match response.verify(&challenge, password) {
|
let status = match response.verify(&challenge, password) {
|
||||||
true => AuthStatus::Passed,
|
true => AuthStatus::Passed,
|
||||||
false => AuthStatus::Failed,
|
false => AuthStatus::Failed,
|
||||||
};
|
};
|
||||||
|
|
||||||
rkvm_net::timeout(rkvm_net::WRITE_TIMEOUT, async {
|
rkvm_net::timeout(timeout.write, async {
|
||||||
status.encode(&mut stream).await?;
|
status.encode(&mut stream).await?;
|
||||||
stream.flush().await?;
|
stream.flush().await?;
|
||||||
|
|
||||||
|
@ -386,7 +388,7 @@ async fn client(
|
||||||
};
|
};
|
||||||
|
|
||||||
let start = Instant::now();
|
let start = Instant::now();
|
||||||
rkvm_net::timeout(rkvm_net::WRITE_TIMEOUT, async {
|
rkvm_net::timeout(timeout.write, async {
|
||||||
update.encode(&mut stream).await?;
|
update.encode(&mut stream).await?;
|
||||||
stream.flush().await?;
|
stream.flush().await?;
|
||||||
|
|
||||||
|
@ -400,7 +402,7 @@ async fn client(
|
||||||
tracing::debug!(duration = ?duration, "Sent ping");
|
tracing::debug!(duration = ?duration, "Sent ping");
|
||||||
|
|
||||||
let start = Instant::now();
|
let start = Instant::now();
|
||||||
rkvm_net::timeout(rkvm_net::READ_TIMEOUT, Pong::decode(&mut stream)).await?;
|
rkvm_net::timeout(timeout.read, Pong::decode(&mut stream)).await?;
|
||||||
let duration = start.elapsed();
|
let duration = start.elapsed();
|
||||||
|
|
||||||
tracing::debug!(duration = ?duration, "Received pong");
|
tracing::debug!(duration = ?duration, "Received pong");
|
||||||
|
|
Loading…
Reference in a new issue