mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
Update Network Protocol
- now last digit version is compatible 0.6.0 will connect to 0.6.1 - the TCP DATA Frames no longer contain START field, as it's not needed - the TCP OPENSTREAM Frames will now contain the BANDWIDTH field - MID is not Protocol internal Update network - update API with Bandwidth Update veloren - introduce better runtime and `async` things that are IO bound. - Remove `uvth` and instead use `tokio::runtime::Runtime::spawn_blocking` - remove futures_execute from client and server use tokio::runtime::Runtime instead - give threads a Name
This commit is contained in:
parent
cee08c7012
commit
514d5db038
63
Cargo.lock
generated
63
Cargo.lock
generated
@ -233,7 +233,7 @@ checksum = "dac94eeee6ebd1165959e440836a452109f9f839d6cfde12974d75a5b4222406"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"ahash 0.6.3",
|
"ahash 0.6.3",
|
||||||
"bincode",
|
"bincode",
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"log",
|
"log",
|
||||||
"notify 4.0.15",
|
"notify 4.0.15",
|
||||||
"parking_lot 0.11.1",
|
"parking_lot 0.11.1",
|
||||||
@ -1093,22 +1093,13 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "fd01a6eb3daaafa260f6fc94c3a6c36390abc2080e38e3e34ced87393fb77d80"
|
checksum = "fd01a6eb3daaafa260f6fc94c3a6c36390abc2080e38e3e34ced87393fb77d80"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if 1.0.0",
|
"cfg-if 1.0.0",
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"crossbeam-deque 0.8.0",
|
"crossbeam-deque 0.8.0",
|
||||||
"crossbeam-epoch 0.9.1",
|
"crossbeam-epoch 0.9.1",
|
||||||
"crossbeam-queue",
|
"crossbeam-queue",
|
||||||
"crossbeam-utils 0.8.1",
|
"crossbeam-utils 0.8.1",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crossbeam-channel"
|
|
||||||
version = "0.3.9"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "c8ec7fcd21571dc78f96cc96243cab8d8f035247c3efd16c687be154c3fa9efa"
|
|
||||||
dependencies = [
|
|
||||||
"crossbeam-utils 0.6.6",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crossbeam-channel"
|
name = "crossbeam-channel"
|
||||||
version = "0.5.0"
|
version = "0.5.0"
|
||||||
@ -1180,16 +1171,6 @@ dependencies = [
|
|||||||
"crossbeam-utils 0.8.1",
|
"crossbeam-utils 0.8.1",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crossbeam-utils"
|
|
||||||
version = "0.6.6"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "04973fa96e96579258a5091af6003abde64af786b860f18622b82e026cca60e6"
|
|
||||||
dependencies = [
|
|
||||||
"cfg-if 0.1.10",
|
|
||||||
"lazy_static",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crossbeam-utils"
|
name = "crossbeam-utils"
|
||||||
version = "0.7.2"
|
version = "0.7.2"
|
||||||
@ -1829,12 +1810,6 @@ dependencies = [
|
|||||||
"once_cell",
|
"once_cell",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "futures-timer"
|
|
||||||
version = "3.0.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "e64b03909df88034c26dc1547e8970b91f98bdb65165d6a4e9110d94263dbb2c"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "futures-util"
|
name = "futures-util"
|
||||||
version = "0.3.12"
|
version = "0.3.12"
|
||||||
@ -2162,7 +2137,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "ac2c82074cafb68b9e459c50c655f7eedcb92d6ee7166813802934bc6fc29fa3"
|
checksum = "ac2c82074cafb68b9e459c50c655f7eedcb92d6ee7166813802934bc6fc29fa3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"ab_glyph",
|
"ab_glyph",
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"crossbeam-deque 0.8.0",
|
"crossbeam-deque 0.8.0",
|
||||||
"linked-hash-map",
|
"linked-hash-map",
|
||||||
"rayon",
|
"rayon",
|
||||||
@ -3302,7 +3277,7 @@ checksum = "58e54552360d7b89a698eca6de3927205a8e03e8080dc13d779de5c7876e098b"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"anymap",
|
"anymap",
|
||||||
"bitflags",
|
"bitflags",
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"filetime",
|
"filetime",
|
||||||
"fsevent 2.0.2",
|
"fsevent 2.0.2",
|
||||||
"fsevent-sys 3.0.2",
|
"fsevent-sys 3.0.2",
|
||||||
@ -4142,7 +4117,7 @@ version = "1.9.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9ab346ac5921dc62ffa9f89b7a773907511cdfa5490c572ae9be1be33e8afa4a"
|
checksum = "9ab346ac5921dc62ffa9f89b7a773907511cdfa5490c572ae9be1be33e8afa4a"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"crossbeam-deque 0.8.0",
|
"crossbeam-deque 0.8.0",
|
||||||
"crossbeam-utils 0.8.1",
|
"crossbeam-utils 0.8.1",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
@ -5241,7 +5216,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "9965507e507f12c8901432a33e31131222abac31edd90cabbcf85cf544b7127a"
|
checksum = "9965507e507f12c8901432a33e31131222abac31edd90cabbcf85cf544b7127a"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"chrono",
|
"chrono",
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"tracing-subscriber",
|
"tracing-subscriber",
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -5524,17 +5499,6 @@ dependencies = [
|
|||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "uvth"
|
|
||||||
version = "3.1.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "e59a167890d173eb0fcd7a1b99b84dc05c521ae8d76599130b8e19bef287abbf"
|
|
||||||
dependencies = [
|
|
||||||
"crossbeam-channel 0.3.9",
|
|
||||||
"log",
|
|
||||||
"num_cpus",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "vcpkg"
|
name = "vcpkg"
|
||||||
version = "0.2.11"
|
version = "0.2.11"
|
||||||
@ -5593,19 +5557,15 @@ version = "0.8.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"authc",
|
"authc",
|
||||||
"byteorder",
|
"byteorder",
|
||||||
"futures-executor",
|
|
||||||
"futures-timer",
|
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"hashbrown 0.9.1",
|
"hashbrown 0.9.1",
|
||||||
"image",
|
"image",
|
||||||
"num 0.3.1",
|
"num 0.3.1",
|
||||||
"num_cpus",
|
|
||||||
"rayon",
|
"rayon",
|
||||||
"specs",
|
"specs",
|
||||||
"tokio 1.2.0",
|
"tokio 1.2.0",
|
||||||
"tracing",
|
"tracing",
|
||||||
"tracing-subscriber",
|
"tracing-subscriber",
|
||||||
"uvth",
|
|
||||||
"vek 0.12.0",
|
"vek 0.12.0",
|
||||||
"veloren-common",
|
"veloren-common",
|
||||||
"veloren-common-net",
|
"veloren-common-net",
|
||||||
@ -5621,7 +5581,7 @@ dependencies = [
|
|||||||
"arraygen",
|
"arraygen",
|
||||||
"assets_manager",
|
"assets_manager",
|
||||||
"criterion",
|
"criterion",
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"crossbeam-utils 0.8.1",
|
"crossbeam-utils 0.8.1",
|
||||||
"csv",
|
"csv",
|
||||||
"directories-next",
|
"directories-next",
|
||||||
@ -5702,7 +5662,7 @@ dependencies = [
|
|||||||
"bytes 1.0.1",
|
"bytes 1.0.1",
|
||||||
"clap",
|
"clap",
|
||||||
"criterion",
|
"criterion",
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"futures-core",
|
"futures-core",
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
@ -5767,13 +5727,10 @@ version = "0.8.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"authc",
|
"authc",
|
||||||
"chrono",
|
"chrono",
|
||||||
"crossbeam-channel 0.5.0",
|
"crossbeam-channel",
|
||||||
"diesel",
|
"diesel",
|
||||||
"diesel_migrations",
|
"diesel_migrations",
|
||||||
"dotenv",
|
"dotenv",
|
||||||
"futures-channel",
|
|
||||||
"futures-executor",
|
|
||||||
"futures-timer",
|
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"hashbrown 0.9.1",
|
"hashbrown 0.9.1",
|
||||||
"itertools 0.9.0",
|
"itertools 0.9.0",
|
||||||
@ -5793,7 +5750,6 @@ dependencies = [
|
|||||||
"specs-idvs",
|
"specs-idvs",
|
||||||
"tokio 1.2.0",
|
"tokio 1.2.0",
|
||||||
"tracing",
|
"tracing",
|
||||||
"uvth",
|
|
||||||
"vek 0.12.0",
|
"vek 0.12.0",
|
||||||
"veloren-common",
|
"veloren-common",
|
||||||
"veloren-common-net",
|
"veloren-common-net",
|
||||||
@ -5880,7 +5836,6 @@ dependencies = [
|
|||||||
"tracing-subscriber",
|
"tracing-subscriber",
|
||||||
"tracing-tracy",
|
"tracing-tracy",
|
||||||
"treeculler",
|
"treeculler",
|
||||||
"uvth",
|
|
||||||
"vek 0.12.0",
|
"vek 0.12.0",
|
||||||
"veloren-client",
|
"veloren-client",
|
||||||
"veloren-common",
|
"veloren-common",
|
||||||
|
@ -17,14 +17,10 @@ common-net = { package = "veloren-common-net", path = "../common/net" }
|
|||||||
network = { package = "veloren-network", path = "../network", features = ["compression"], default-features = false }
|
network = { package = "veloren-network", path = "../network", features = ["compression"], default-features = false }
|
||||||
|
|
||||||
byteorder = "1.3.2"
|
byteorder = "1.3.2"
|
||||||
uvth = "3.1.1"
|
|
||||||
futures-util = "0.3.7"
|
futures-util = "0.3.7"
|
||||||
futures-executor = "0.3"
|
|
||||||
futures-timer = "3.0"
|
|
||||||
tokio = { version = "1", default-features = false, features = ["rt-multi-thread"] }
|
tokio = { version = "1", default-features = false, features = ["rt-multi-thread"] }
|
||||||
image = { version = "0.23.12", default-features = false, features = ["png"] }
|
image = { version = "0.23.12", default-features = false, features = ["png"] }
|
||||||
num = "0.3.1"
|
num = "0.3.1"
|
||||||
num_cpus = "1.10.1"
|
|
||||||
tracing = { version = "0.1", default-features = false }
|
tracing = { version = "0.1", default-features = false }
|
||||||
rayon = "1.5"
|
rayon = "1.5"
|
||||||
specs = { git = "https://github.com/amethyst/specs.git", rev = "d4435bdf496cf322c74886ca09dd8795984919b4" }
|
specs = { git = "https://github.com/amethyst/specs.git", rev = "d4435bdf496cf322c74886ca09dd8795984919b4" }
|
||||||
|
@ -45,27 +45,29 @@ fn main() {
|
|||||||
let password = read_input();
|
let password = read_input();
|
||||||
|
|
||||||
let runtime = Arc::new(Runtime::new().unwrap());
|
let runtime = Arc::new(Runtime::new().unwrap());
|
||||||
|
let runtime2 = Arc::clone(&runtime);
|
||||||
|
|
||||||
// Create a client.
|
// Create a client.
|
||||||
let mut client = Client::new(
|
let mut client = runtime
|
||||||
|
.block_on(Client::new(
|
||||||
server_addr
|
server_addr
|
||||||
.to_socket_addrs()
|
.to_socket_addrs()
|
||||||
.expect("Invalid server address")
|
.expect("Invalid server address")
|
||||||
.next()
|
.next()
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
None,
|
None,
|
||||||
runtime,
|
runtime2,
|
||||||
)
|
))
|
||||||
.expect("Failed to create client instance");
|
.expect("Failed to create client instance");
|
||||||
|
|
||||||
println!("Server info: {:?}", client.server_info());
|
println!("Server info: {:?}", client.server_info());
|
||||||
|
|
||||||
println!("Players online: {:?}", client.get_players());
|
println!("Players online: {:?}", client.get_players());
|
||||||
|
|
||||||
client
|
runtime
|
||||||
.register(username, password, |provider| {
|
.block_on(client.register(username, password, |provider| {
|
||||||
provider == "https://auth.veloren.net"
|
provider == "https://auth.veloren.net"
|
||||||
})
|
}))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
let (tx, rx) = mpsc::channel();
|
let (tx, rx) = mpsc::channel();
|
||||||
|
@ -48,9 +48,7 @@ use common_net::{
|
|||||||
};
|
};
|
||||||
use common_sys::state::State;
|
use common_sys::state::State;
|
||||||
use comp::BuffKind;
|
use comp::BuffKind;
|
||||||
use futures_executor::block_on;
|
use futures_util::FutureExt;
|
||||||
use futures_timer::Delay;
|
|
||||||
use futures_util::{select, FutureExt};
|
|
||||||
use hashbrown::{HashMap, HashSet};
|
use hashbrown::{HashMap, HashSet};
|
||||||
use image::DynamicImage;
|
use image::DynamicImage;
|
||||||
use network::{Network, Participant, Pid, ProtocolAddr, Stream};
|
use network::{Network, Participant, Pid, ProtocolAddr, Stream};
|
||||||
@ -63,9 +61,8 @@ use std::{
|
|||||||
sync::Arc,
|
sync::Arc,
|
||||||
time::{Duration, Instant},
|
time::{Duration, Instant},
|
||||||
};
|
};
|
||||||
use tokio::runtime::Runtime;
|
use tokio::{runtime::Runtime, select};
|
||||||
use tracing::{debug, error, trace, warn};
|
use tracing::{debug, error, trace, warn};
|
||||||
use uvth::{ThreadPool, ThreadPoolBuilder};
|
|
||||||
use vek::*;
|
use vek::*;
|
||||||
|
|
||||||
const PING_ROLLING_AVERAGE_SECS: usize = 10;
|
const PING_ROLLING_AVERAGE_SECS: usize = 10;
|
||||||
@ -131,7 +128,6 @@ pub struct Client {
|
|||||||
registered: bool,
|
registered: bool,
|
||||||
presence: Option<PresenceKind>,
|
presence: Option<PresenceKind>,
|
||||||
runtime: Arc<Runtime>,
|
runtime: Arc<Runtime>,
|
||||||
thread_pool: ThreadPool,
|
|
||||||
server_info: ServerInfo,
|
server_info: ServerInfo,
|
||||||
world_data: WorldData,
|
world_data: WorldData,
|
||||||
player_list: HashMap<Uid, PlayerInfo>,
|
player_list: HashMap<Uid, PlayerInfo>,
|
||||||
@ -187,28 +183,22 @@ pub struct CharacterList {
|
|||||||
|
|
||||||
impl Client {
|
impl Client {
|
||||||
/// Create a new `Client`.
|
/// Create a new `Client`.
|
||||||
pub fn new<A: Into<SocketAddr>>(
|
pub async fn new<A: Into<SocketAddr>>(
|
||||||
addr: A,
|
addr: A,
|
||||||
view_distance: Option<u32>,
|
view_distance: Option<u32>,
|
||||||
runtime: Arc<Runtime>,
|
runtime: Arc<Runtime>,
|
||||||
) -> Result<Self, Error> {
|
) -> Result<Self, Error> {
|
||||||
let mut thread_pool = ThreadPoolBuilder::new()
|
|
||||||
.name("veloren-worker".into())
|
|
||||||
.build();
|
|
||||||
// We reduce the thread count by 1 to keep rendering smooth
|
|
||||||
thread_pool.set_num_threads((num_cpus::get() - 1).max(1));
|
|
||||||
|
|
||||||
let network = Network::new(Pid::new(), Arc::clone(&runtime));
|
let network = Network::new(Pid::new(), Arc::clone(&runtime));
|
||||||
|
|
||||||
let participant = block_on(network.connect(ProtocolAddr::Tcp(addr.into())))?;
|
let participant = network.connect(ProtocolAddr::Tcp(addr.into())).await?;
|
||||||
let stream = block_on(participant.opened())?;
|
let stream = participant.opened().await?;
|
||||||
let mut ping_stream = block_on(participant.opened())?;
|
let mut ping_stream = participant.opened().await?;
|
||||||
let mut register_stream = block_on(participant.opened())?;
|
let mut register_stream = participant.opened().await?;
|
||||||
let character_screen_stream = block_on(participant.opened())?;
|
let character_screen_stream = participant.opened().await?;
|
||||||
let in_game_stream = block_on(participant.opened())?;
|
let in_game_stream = participant.opened().await?;
|
||||||
|
|
||||||
register_stream.send(ClientType::Game)?;
|
register_stream.send(ClientType::Game)?;
|
||||||
let server_info: ServerInfo = block_on(register_stream.recv())?;
|
let server_info: ServerInfo = register_stream.recv().await?;
|
||||||
|
|
||||||
// TODO: Display that versions don't match in Voxygen
|
// TODO: Display that versions don't match in Voxygen
|
||||||
if server_info.git_hash != *common::util::GIT_HASH {
|
if server_info.git_hash != *common::util::GIT_HASH {
|
||||||
@ -236,7 +226,7 @@ impl Client {
|
|||||||
recipe_book,
|
recipe_book,
|
||||||
max_group_size,
|
max_group_size,
|
||||||
client_timeout,
|
client_timeout,
|
||||||
) = match block_on(register_stream.recv())? {
|
) = match register_stream.recv().await? {
|
||||||
ServerInit::GameSync {
|
ServerInit::GameSync {
|
||||||
entity_package,
|
entity_package,
|
||||||
time_of_day,
|
time_of_day,
|
||||||
@ -411,19 +401,12 @@ impl Client {
|
|||||||
}?;
|
}?;
|
||||||
ping_stream.send(PingMsg::Ping)?;
|
ping_stream.send(PingMsg::Ping)?;
|
||||||
|
|
||||||
let mut thread_pool = ThreadPoolBuilder::new()
|
|
||||||
.name("veloren-worker".into())
|
|
||||||
.build();
|
|
||||||
// We reduce the thread count by 1 to keep rendering smooth
|
|
||||||
thread_pool.set_num_threads((num_cpus::get() - 1).max(1));
|
|
||||||
|
|
||||||
debug!("Initial sync done");
|
debug!("Initial sync done");
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
registered: false,
|
registered: false,
|
||||||
presence: None,
|
presence: None,
|
||||||
runtime,
|
runtime,
|
||||||
thread_pool,
|
|
||||||
server_info,
|
server_info,
|
||||||
world_data: WorldData {
|
world_data: WorldData {
|
||||||
lod_base,
|
lod_base,
|
||||||
@ -470,13 +453,8 @@ impl Client {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_thread_pool(mut self, thread_pool: ThreadPool) -> Self {
|
|
||||||
self.thread_pool = thread_pool;
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Request a state transition to `ClientState::Registered`.
|
/// Request a state transition to `ClientState::Registered`.
|
||||||
pub fn register(
|
pub async fn register(
|
||||||
&mut self,
|
&mut self,
|
||||||
username: String,
|
username: String,
|
||||||
password: String,
|
password: String,
|
||||||
@ -496,7 +474,7 @@ impl Client {
|
|||||||
|
|
||||||
self.send_msg_err(ClientRegister { token_or_username })?;
|
self.send_msg_err(ClientRegister { token_or_username })?;
|
||||||
|
|
||||||
match block_on(self.register_stream.recv::<ServerRegisterAnswer>())? {
|
match self.register_stream.recv::<ServerRegisterAnswer>().await? {
|
||||||
Err(RegisterError::AlreadyLoggedIn) => Err(Error::AlreadyLoggedIn),
|
Err(RegisterError::AlreadyLoggedIn) => Err(Error::AlreadyLoggedIn),
|
||||||
Err(RegisterError::AuthError(err)) => Err(Error::AuthErr(err)),
|
Err(RegisterError::AuthError(err)) => Err(Error::AuthErr(err)),
|
||||||
Err(RegisterError::InvalidCharacter) => Err(Error::InvalidCharacter),
|
Err(RegisterError::InvalidCharacter) => Err(Error::InvalidCharacter),
|
||||||
@ -1688,10 +1666,11 @@ impl Client {
|
|||||||
|
|
||||||
let mut handles_msg = 0;
|
let mut handles_msg = 0;
|
||||||
|
|
||||||
block_on(async {
|
let runtime = Arc::clone(&self.runtime);
|
||||||
|
runtime.block_on(async {
|
||||||
//TIMEOUT 0.01 ms for msg handling
|
//TIMEOUT 0.01 ms for msg handling
|
||||||
select!(
|
select!(
|
||||||
_ = Delay::new(std::time::Duration::from_micros(10)).fuse() => Ok(()),
|
_ = tokio::time::sleep(std::time::Duration::from_micros(10)).fuse() => Ok(()),
|
||||||
err = self.handle_messages(&mut frontend_events, &mut handles_msg).fuse() => err,
|
err = self.handle_messages(&mut frontend_events, &mut handles_msg).fuse() => err,
|
||||||
)
|
)
|
||||||
})?;
|
})?;
|
||||||
@ -1733,12 +1712,10 @@ impl Client {
|
|||||||
* 1000.0
|
* 1000.0
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get a reference to the client's worker thread pool. This pool should be
|
/// Get a reference to the client's runtime thread pool. This pool should be
|
||||||
/// used for any computationally expensive operations that run outside
|
/// used for any computationally expensive operations that run outside
|
||||||
/// of the main thread (i.e., threads that block on I/O operations are
|
/// of the main thread (i.e., threads that block on I/O operations are
|
||||||
/// exempt).
|
/// exempt).
|
||||||
pub fn thread_pool(&self) -> &ThreadPool { &self.thread_pool }
|
|
||||||
|
|
||||||
pub fn runtime(&self) -> &Arc<Runtime> { &self.runtime }
|
pub fn runtime(&self) -> &Arc<Runtime> { &self.runtime }
|
||||||
|
|
||||||
/// Get a reference to the client's game state.
|
/// Get a reference to the client's game state.
|
||||||
@ -2042,7 +2019,10 @@ impl Drop for Client {
|
|||||||
} else {
|
} else {
|
||||||
trace!("no disconnect msg necessary as client wasn't registered")
|
trace!("no disconnect msg necessary as client wasn't registered")
|
||||||
}
|
}
|
||||||
if let Err(e) = block_on(self.participant.take().unwrap().disconnect()) {
|
if let Err(e) = self
|
||||||
|
.runtime
|
||||||
|
.block_on(self.participant.take().unwrap().disconnect())
|
||||||
|
{
|
||||||
warn!(?e, "error when disconnecting, couldn't send all data");
|
warn!(?e, "error when disconnecting, couldn't send all data");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2067,7 +2047,9 @@ mod tests {
|
|||||||
let socket = SocketAddr::new(IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), 9000);
|
let socket = SocketAddr::new(IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), 9000);
|
||||||
let view_distance: Option<u32> = None;
|
let view_distance: Option<u32> = None;
|
||||||
let runtime = Arc::new(Runtime::new().unwrap());
|
let runtime = Arc::new(Runtime::new().unwrap());
|
||||||
let veloren_client: Result<Client, Error> = Client::new(socket, view_distance, runtime);
|
let runtime2 = Arc::clone(&runtime);
|
||||||
|
let veloren_client: Result<Client, Error> =
|
||||||
|
runtime.block_on(Client::new(socket, view_distance, runtime2));
|
||||||
|
|
||||||
let _ = veloren_client.map(|mut client| {
|
let _ = veloren_client.map(|mut client| {
|
||||||
//register
|
//register
|
||||||
@ -2075,9 +2057,9 @@ mod tests {
|
|||||||
let password: String = "Bar".to_string();
|
let password: String = "Bar".to_string();
|
||||||
let auth_server: String = "auth.veloren.net".to_string();
|
let auth_server: String = "auth.veloren.net".to_string();
|
||||||
let _result: Result<(), Error> =
|
let _result: Result<(), Error> =
|
||||||
client.register(username, password, |suggestion: &str| {
|
runtime.block_on(client.register(username, password, |suggestion: &str| {
|
||||||
suggestion == auth_server
|
suggestion == auth_server
|
||||||
});
|
}));
|
||||||
|
|
||||||
//clock
|
//clock
|
||||||
let mut clock = Clock::new(Duration::from_secs_f64(SPT));
|
let mut clock = Clock::new(Duration::from_secs_f64(SPT));
|
||||||
|
@ -28,7 +28,7 @@ tracing = { version = "0.1", default-features = false, features = ["attributes"]
|
|||||||
prometheus = { version = "0.11", default-features = false, optional = true }
|
prometheus = { version = "0.11", default-features = false, optional = true }
|
||||||
#async
|
#async
|
||||||
futures-core = { version = "0.3", default-features = false }
|
futures-core = { version = "0.3", default-features = false }
|
||||||
futures-util = { version = "0.3", default-features = false, features = ["std"] }
|
futures-util = { version = "0.3.7", default-features = false, features = ["std"] }
|
||||||
async-channel = "1.5.1" #use for .close() channels
|
async-channel = "1.5.1" #use for .close() channels
|
||||||
#mpsc channel registry
|
#mpsc channel registry
|
||||||
lazy_static = { version = "1.4", default-features = false }
|
lazy_static = { version = "1.4", default-features = false }
|
||||||
@ -43,7 +43,7 @@ bytes = "^1"
|
|||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
tracing-subscriber = { version = "0.2.3", default-features = false, features = ["env-filter", "fmt", "chrono", "ansi", "smallvec"] }
|
tracing-subscriber = { version = "0.2.3", default-features = false, features = ["env-filter", "fmt", "chrono", "ansi", "smallvec"] }
|
||||||
tokio = { version = "1.2", default-features = false, features = ["io-std", "fs", "rt-multi-thread"] }
|
tokio = { version = "1.2", default-features = false, features = ["io-std", "fs", "rt-multi-thread"] }
|
||||||
futures-util = { version = "0.3", default-features = false, features = ["sink", "std"] }
|
futures-util = { version = "0.3.7", default-features = false, features = ["sink", "std"] }
|
||||||
clap = { version = "2.33", default-features = false }
|
clap = { version = "2.33", default-features = false }
|
||||||
shellexpand = "2.0.0"
|
shellexpand = "2.0.0"
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
|
@ -31,7 +31,7 @@ fn criterion_util(c: &mut Criterion) {
|
|||||||
|
|
||||||
let (r, _n_a, p_a, s1_a, _n_b, _p_b, _s1_b) =
|
let (r, _n_a, p_a, s1_a, _n_b, _p_b, _s1_b) =
|
||||||
network_participant_stream(ProtocolAddr::Mpsc(5000));
|
network_participant_stream(ProtocolAddr::Mpsc(5000));
|
||||||
let s2_a = r.block_on(p_a.open(4, Promises::COMPRESSED)).unwrap();
|
let s2_a = r.block_on(p_a.open(4, Promises::COMPRESSED, 0)).unwrap();
|
||||||
|
|
||||||
c.throughput(Throughput::Bytes(1000))
|
c.throughput(Throughput::Bytes(1000))
|
||||||
.bench_function("message_serialize", |b| {
|
.bench_function("message_serialize", |b| {
|
||||||
@ -134,7 +134,7 @@ pub fn network_participant_stream(
|
|||||||
let p1_b = n_b.connect(addr).await.unwrap();
|
let p1_b = n_b.connect(addr).await.unwrap();
|
||||||
let p1_a = n_a.connected().await.unwrap();
|
let p1_a = n_a.connected().await.unwrap();
|
||||||
|
|
||||||
let s1_a = p1_a.open(4, Promises::empty()).await.unwrap();
|
let s1_a = p1_a.open(4, Promises::empty(), 0).await.unwrap();
|
||||||
let s1_b = p1_b.opened().await.unwrap();
|
let s1_b = p1_b.opened().await.unwrap();
|
||||||
|
|
||||||
(n_a, p1_a, s1_a, n_b, p1_b, s1_b)
|
(n_a, p1_a, s1_a, n_b, p1_b, s1_b)
|
||||||
|
@ -130,7 +130,10 @@ async fn client_connection(
|
|||||||
Ok(msg) => {
|
Ok(msg) => {
|
||||||
println!("[{}]: {}", username, msg);
|
println!("[{}]: {}", username, msg);
|
||||||
for p in participants.read().await.iter() {
|
for p in participants.read().await.iter() {
|
||||||
match p.open(4, Promises::ORDERED | Promises::CONSISTENCY).await {
|
match p
|
||||||
|
.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0)
|
||||||
|
.await
|
||||||
|
{
|
||||||
Err(_) => info!("error talking to client, //TODO drop it"),
|
Err(_) => info!("error talking to client, //TODO drop it"),
|
||||||
Ok(mut s) => s.send((username.clone(), msg.clone())).unwrap(),
|
Ok(mut s) => s.send((username.clone(), msg.clone())).unwrap(),
|
||||||
};
|
};
|
||||||
@ -148,7 +151,7 @@ fn client(address: ProtocolAddr) {
|
|||||||
r.block_on(async {
|
r.block_on(async {
|
||||||
let p1 = client.connect(address.clone()).await.unwrap(); //remote representation of p1
|
let p1 = client.connect(address.clone()).await.unwrap(); //remote representation of p1
|
||||||
let mut s1 = p1
|
let mut s1 = p1
|
||||||
.open(4, Promises::ORDERED | Promises::CONSISTENCY)
|
.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0)
|
||||||
.await
|
.await
|
||||||
.unwrap(); //remote representation of s1
|
.unwrap(); //remote representation of s1
|
||||||
let mut input_lines = io::BufReader::new(io::stdin());
|
let mut input_lines = io::BufReader::new(io::stdin());
|
||||||
|
@ -121,8 +121,9 @@ impl Server {
|
|||||||
#[allow(clippy::eval_order_dependence)]
|
#[allow(clippy::eval_order_dependence)]
|
||||||
async fn loop_participant(&self, p: Participant) {
|
async fn loop_participant(&self, p: Participant) {
|
||||||
if let (Ok(cmd_out), Ok(file_out), Ok(cmd_in), Ok(file_in)) = (
|
if let (Ok(cmd_out), Ok(file_out), Ok(cmd_in), Ok(file_in)) = (
|
||||||
p.open(3, Promises::ORDERED | Promises::CONSISTENCY).await,
|
p.open(3, Promises::ORDERED | Promises::CONSISTENCY, 0)
|
||||||
p.open(6, Promises::CONSISTENCY).await,
|
.await,
|
||||||
|
p.open(6, Promises::CONSISTENCY, 0).await,
|
||||||
p.opened().await,
|
p.opened().await,
|
||||||
p.opened().await,
|
p.opened().await,
|
||||||
) {
|
) {
|
||||||
|
@ -164,7 +164,7 @@ fn client(address: ProtocolAddr, runtime: Arc<Runtime>) {
|
|||||||
|
|
||||||
let p1 = runtime.block_on(client.connect(address)).unwrap(); //remote representation of p1
|
let p1 = runtime.block_on(client.connect(address)).unwrap(); //remote representation of p1
|
||||||
let mut s1 = runtime
|
let mut s1 = runtime
|
||||||
.block_on(p1.open(4, Promises::ORDERED | Promises::CONSISTENCY))
|
.block_on(p1.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0))
|
||||||
.unwrap(); //remote representation of s1
|
.unwrap(); //remote representation of s1
|
||||||
let mut last = Instant::now();
|
let mut last = Instant::now();
|
||||||
let mut id = 0u64;
|
let mut id = 0u64;
|
||||||
|
@ -27,7 +27,7 @@ bytes = "^1"
|
|||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
async-channel = "1.5.1"
|
async-channel = "1.5.1"
|
||||||
tokio = { version = "1.2", default-features = false, features = ["rt", "macros"] }
|
tokio = { version = "^1", default-features = false, features = ["rt", "macros"] }
|
||||||
criterion = { version = "0.3.4", features = ["default", "async_tokio"] }
|
criterion = { version = "0.3.4", features = ["default", "async_tokio"] }
|
||||||
|
|
||||||
[[bench]]
|
[[bench]]
|
||||||
|
@ -47,7 +47,6 @@ async fn send_msg<T: SendProtocol>(mut s: T, data: Bytes, cnt: usize) {
|
|||||||
for i in 0..cnt {
|
for i in 0..cnt {
|
||||||
s.send(ProtocolEvent::Message {
|
s.send(ProtocolEvent::Message {
|
||||||
sid: Sid::new(12),
|
sid: Sid::new(12),
|
||||||
mid: i as u64,
|
|
||||||
data: data.clone(),
|
data: data.clone(),
|
||||||
})
|
})
|
||||||
.await
|
.await
|
||||||
@ -93,7 +92,6 @@ fn criterion_util(c: &mut Criterion) {
|
|||||||
let mut buffer = BytesMut::with_capacity(1500);
|
let mut buffer = BytesMut::with_capacity(1500);
|
||||||
let frame = OTFrame::Data {
|
let frame = OTFrame::Data {
|
||||||
mid: 65,
|
mid: 65,
|
||||||
start: 89u64,
|
|
||||||
data: Bytes::from(&b"hello_world"[..]),
|
data: Bytes::from(&b"hello_world"[..]),
|
||||||
};
|
};
|
||||||
b.iter_with_setup(
|
b.iter_with_setup(
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
frame::OTFrame,
|
frame::OTFrame,
|
||||||
types::{Bandwidth, Mid, Prio, Promises, Sid},
|
types::{Bandwidth, Prio, Promises, Sid},
|
||||||
};
|
};
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
|
|
||||||
@ -23,7 +23,6 @@ pub enum ProtocolEvent {
|
|||||||
},
|
},
|
||||||
Message {
|
Message {
|
||||||
data: Bytes,
|
data: Bytes,
|
||||||
mid: Mid,
|
|
||||||
sid: Sid,
|
sid: Sid,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@ -36,11 +35,12 @@ impl ProtocolEvent {
|
|||||||
sid,
|
sid,
|
||||||
prio,
|
prio,
|
||||||
promises,
|
promises,
|
||||||
guaranteed_bandwidth: _,
|
guaranteed_bandwidth,
|
||||||
} => OTFrame::OpenStream {
|
} => OTFrame::OpenStream {
|
||||||
sid: *sid,
|
sid: *sid,
|
||||||
prio: *prio,
|
prio: *prio,
|
||||||
promises: *promises,
|
promises: *promises,
|
||||||
|
guaranteed_bandwidth: *guaranteed_bandwidth,
|
||||||
},
|
},
|
||||||
ProtocolEvent::CloseStream { sid } => OTFrame::CloseStream { sid: *sid },
|
ProtocolEvent::CloseStream { sid } => OTFrame::CloseStream { sid: *sid },
|
||||||
ProtocolEvent::Message { .. } => {
|
ProtocolEvent::Message { .. } => {
|
||||||
@ -68,7 +68,6 @@ mod tests {
|
|||||||
fn test_msg_buffer_panic() {
|
fn test_msg_buffer_panic() {
|
||||||
let _ = ProtocolEvent::Message {
|
let _ = ProtocolEvent::Message {
|
||||||
data: Bytes::new(),
|
data: Bytes::new(),
|
||||||
mid: 0,
|
|
||||||
sid: Sid::new(23),
|
sid: Sid::new(23),
|
||||||
}
|
}
|
||||||
.to_frame();
|
.to_frame();
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use crate::types::{Mid, Pid, Prio, Promises, Sid};
|
use crate::types::{Bandwidth, Mid, Pid, Prio, Promises, Sid};
|
||||||
use bytes::{Buf, BufMut, Bytes, BytesMut};
|
use bytes::{Buf, BufMut, Bytes, BytesMut};
|
||||||
|
|
||||||
// const FRAME_RESERVED_1: u8 = 0;
|
// const FRAME_RESERVED_1: u8 = 0;
|
||||||
@ -38,6 +38,7 @@ pub enum OTFrame {
|
|||||||
sid: Sid,
|
sid: Sid,
|
||||||
prio: Prio,
|
prio: Prio,
|
||||||
promises: Promises,
|
promises: Promises,
|
||||||
|
guaranteed_bandwidth: Bandwidth,
|
||||||
},
|
},
|
||||||
CloseStream {
|
CloseStream {
|
||||||
sid: Sid,
|
sid: Sid,
|
||||||
@ -49,7 +50,6 @@ pub enum OTFrame {
|
|||||||
},
|
},
|
||||||
Data {
|
Data {
|
||||||
mid: Mid,
|
mid: Mid,
|
||||||
start: u64, /* remove */
|
|
||||||
data: Bytes,
|
data: Bytes,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@ -63,6 +63,7 @@ pub enum ITFrame {
|
|||||||
sid: Sid,
|
sid: Sid,
|
||||||
prio: Prio,
|
prio: Prio,
|
||||||
promises: Promises,
|
promises: Promises,
|
||||||
|
guaranteed_bandwidth: Bandwidth,
|
||||||
},
|
},
|
||||||
CloseStream {
|
CloseStream {
|
||||||
sid: Sid,
|
sid: Sid,
|
||||||
@ -74,7 +75,6 @@ pub enum ITFrame {
|
|||||||
},
|
},
|
||||||
Data {
|
Data {
|
||||||
mid: Mid,
|
mid: Mid,
|
||||||
start: u64, /* remove */
|
|
||||||
data: BytesMut,
|
data: BytesMut,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@ -161,9 +161,9 @@ impl InitFrame {
|
|||||||
|
|
||||||
pub(crate) const TCP_CLOSE_STREAM_CNS: usize = 8;
|
pub(crate) const TCP_CLOSE_STREAM_CNS: usize = 8;
|
||||||
/// const part of the DATA frame, actual size is variable
|
/// const part of the DATA frame, actual size is variable
|
||||||
pub(crate) const TCP_DATA_CNS: usize = 18;
|
pub(crate) const TCP_DATA_CNS: usize = 10;
|
||||||
pub(crate) const TCP_DATA_HEADER_CNS: usize = 24;
|
pub(crate) const TCP_DATA_HEADER_CNS: usize = 24;
|
||||||
pub(crate) const TCP_OPEN_STREAM_CNS: usize = 10;
|
pub(crate) const TCP_OPEN_STREAM_CNS: usize = 18;
|
||||||
// Size WITHOUT the 1rst indicating byte
|
// Size WITHOUT the 1rst indicating byte
|
||||||
pub(crate) const TCP_SHUTDOWN_CNS: usize = 0;
|
pub(crate) const TCP_SHUTDOWN_CNS: usize = 0;
|
||||||
|
|
||||||
@ -177,11 +177,13 @@ impl OTFrame {
|
|||||||
sid,
|
sid,
|
||||||
prio,
|
prio,
|
||||||
promises,
|
promises,
|
||||||
|
guaranteed_bandwidth,
|
||||||
} => {
|
} => {
|
||||||
bytes.put_u8(FRAME_OPEN_STREAM);
|
bytes.put_u8(FRAME_OPEN_STREAM);
|
||||||
sid.to_bytes(bytes);
|
sid.to_bytes(bytes);
|
||||||
bytes.put_u8(prio);
|
bytes.put_u8(prio);
|
||||||
bytes.put_u8(promises.to_le_bytes()[0]);
|
bytes.put_u8(promises.to_le_bytes()[0]);
|
||||||
|
bytes.put_u64_le(guaranteed_bandwidth);
|
||||||
},
|
},
|
||||||
Self::CloseStream { sid } => {
|
Self::CloseStream { sid } => {
|
||||||
bytes.put_u8(FRAME_CLOSE_STREAM);
|
bytes.put_u8(FRAME_CLOSE_STREAM);
|
||||||
@ -193,10 +195,9 @@ impl OTFrame {
|
|||||||
sid.to_bytes(bytes);
|
sid.to_bytes(bytes);
|
||||||
bytes.put_u64_le(length);
|
bytes.put_u64_le(length);
|
||||||
},
|
},
|
||||||
Self::Data { mid, start, data } => {
|
Self::Data { mid, data } => {
|
||||||
bytes.put_u8(FRAME_DATA);
|
bytes.put_u8(FRAME_DATA);
|
||||||
bytes.put_u64_le(mid);
|
bytes.put_u64_le(mid);
|
||||||
bytes.put_u64_le(start);
|
|
||||||
bytes.put_u16_le(data.len() as u16);
|
bytes.put_u16_le(data.len() as u16);
|
||||||
bytes.put_slice(&data);
|
bytes.put_slice(&data);
|
||||||
},
|
},
|
||||||
@ -216,10 +217,10 @@ impl ITFrame {
|
|||||||
FRAME_CLOSE_STREAM => TCP_CLOSE_STREAM_CNS,
|
FRAME_CLOSE_STREAM => TCP_CLOSE_STREAM_CNS,
|
||||||
FRAME_DATA_HEADER => TCP_DATA_HEADER_CNS,
|
FRAME_DATA_HEADER => TCP_DATA_HEADER_CNS,
|
||||||
FRAME_DATA => {
|
FRAME_DATA => {
|
||||||
if bytes.len() < 17 + 1 + 1 {
|
if bytes.len() < 9 + 1 + 1 {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
u16::from_le_bytes([bytes[16 + 1], bytes[17 + 1]]) as usize + TCP_DATA_CNS
|
u16::from_le_bytes([bytes[8 + 1], bytes[9 + 1]]) as usize + TCP_DATA_CNS
|
||||||
},
|
},
|
||||||
_ => return None,
|
_ => return None,
|
||||||
};
|
};
|
||||||
@ -240,6 +241,7 @@ impl ITFrame {
|
|||||||
sid: Sid::from_bytes(&mut bytes),
|
sid: Sid::from_bytes(&mut bytes),
|
||||||
prio: bytes.get_u8(),
|
prio: bytes.get_u8(),
|
||||||
promises: Promises::from_bits_truncate(bytes.get_u8()),
|
promises: Promises::from_bits_truncate(bytes.get_u8()),
|
||||||
|
guaranteed_bandwidth: bytes.get_u64_le(),
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
FRAME_CLOSE_STREAM => {
|
FRAME_CLOSE_STREAM => {
|
||||||
@ -261,11 +263,10 @@ impl ITFrame {
|
|||||||
FRAME_DATA => {
|
FRAME_DATA => {
|
||||||
bytes.advance(1);
|
bytes.advance(1);
|
||||||
let mid = bytes.get_u64_le();
|
let mid = bytes.get_u64_le();
|
||||||
let start = bytes.get_u64_le();
|
|
||||||
let length = bytes.get_u16_le();
|
let length = bytes.get_u16_le();
|
||||||
debug_assert_eq!(length as usize, size - TCP_DATA_CNS);
|
debug_assert_eq!(length as usize, size - TCP_DATA_CNS);
|
||||||
let data = bytes.split_to(length as usize);
|
let data = bytes.split_to(length as usize);
|
||||||
Self::Data { mid, start, data }
|
Self::Data { mid, data }
|
||||||
},
|
},
|
||||||
_ => unreachable!("Frame::to_frame should be handled before!"),
|
_ => unreachable!("Frame::to_frame should be handled before!"),
|
||||||
};
|
};
|
||||||
@ -282,16 +283,18 @@ impl PartialEq<ITFrame> for OTFrame {
|
|||||||
sid,
|
sid,
|
||||||
prio,
|
prio,
|
||||||
promises,
|
promises,
|
||||||
|
guaranteed_bandwidth,
|
||||||
} => matches!(other, ITFrame::OpenStream {
|
} => matches!(other, ITFrame::OpenStream {
|
||||||
sid,
|
sid,
|
||||||
prio,
|
prio,
|
||||||
promises
|
promises,
|
||||||
|
guaranteed_bandwidth,
|
||||||
}),
|
}),
|
||||||
Self::CloseStream { sid } => matches!(other, ITFrame::CloseStream { sid }),
|
Self::CloseStream { sid } => matches!(other, ITFrame::CloseStream { sid }),
|
||||||
Self::DataHeader { mid, sid, length } => {
|
Self::DataHeader { mid, sid, length } => {
|
||||||
matches!(other, ITFrame::DataHeader { mid, sid, length })
|
matches!(other, ITFrame::DataHeader { mid, sid, length })
|
||||||
},
|
},
|
||||||
Self::Data { mid, start, data } => matches!(other, ITFrame::Data { mid, start, data }),
|
Self::Data { mid, data } => matches!(other, ITFrame::Data { mid, data }),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -321,6 +324,7 @@ mod tests {
|
|||||||
sid: Sid::new(1337),
|
sid: Sid::new(1337),
|
||||||
prio: 14,
|
prio: 14,
|
||||||
promises: Promises::GUARANTEED_DELIVERY,
|
promises: Promises::GUARANTEED_DELIVERY,
|
||||||
|
guaranteed_bandwidth: 1_000_000,
|
||||||
},
|
},
|
||||||
OTFrame::DataHeader {
|
OTFrame::DataHeader {
|
||||||
sid: Sid::new(1337),
|
sid: Sid::new(1337),
|
||||||
@ -329,12 +333,10 @@ mod tests {
|
|||||||
},
|
},
|
||||||
OTFrame::Data {
|
OTFrame::Data {
|
||||||
mid: 0,
|
mid: 0,
|
||||||
start: 0,
|
|
||||||
data: Bytes::from(&[77u8; 20][..]),
|
data: Bytes::from(&[77u8; 20][..]),
|
||||||
},
|
},
|
||||||
OTFrame::Data {
|
OTFrame::Data {
|
||||||
mid: 0,
|
mid: 0,
|
||||||
start: 20,
|
|
||||||
data: Bytes::from(&[42u8; 16][..]),
|
data: Bytes::from(&[42u8; 16][..]),
|
||||||
},
|
},
|
||||||
OTFrame::CloseStream {
|
OTFrame::CloseStream {
|
||||||
@ -496,6 +498,7 @@ mod tests {
|
|||||||
sid: Sid::new(88),
|
sid: Sid::new(88),
|
||||||
promises: Promises::ENCRYPTED,
|
promises: Promises::ENCRYPTED,
|
||||||
prio: 88,
|
prio: 88,
|
||||||
|
guaranteed_bandwidth: 1_000_000,
|
||||||
};
|
};
|
||||||
OTFrame::write_bytes(frame1, &mut buffer);
|
OTFrame::write_bytes(frame1, &mut buffer);
|
||||||
}
|
}
|
||||||
@ -508,6 +511,7 @@ mod tests {
|
|||||||
sid: Sid::new(88),
|
sid: Sid::new(88),
|
||||||
promises: Promises::ENCRYPTED,
|
promises: Promises::ENCRYPTED,
|
||||||
prio: 88,
|
prio: 88,
|
||||||
|
guaranteed_bandwidth: 1_000_000,
|
||||||
};
|
};
|
||||||
OTFrame::write_bytes(frame1, &mut buffer);
|
OTFrame::write_bytes(frame1, &mut buffer);
|
||||||
buffer.truncate(6); // simulate partial retrieve
|
buffer.truncate(6); // simulate partial retrieve
|
||||||
@ -527,12 +531,11 @@ mod tests {
|
|||||||
|
|
||||||
let frame1 = OTFrame::Data {
|
let frame1 = OTFrame::Data {
|
||||||
mid: 7u64,
|
mid: 7u64,
|
||||||
start: 1u64,
|
|
||||||
data: Bytes::from(&b"foobar"[..]),
|
data: Bytes::from(&b"foobar"[..]),
|
||||||
};
|
};
|
||||||
|
|
||||||
OTFrame::write_bytes(frame1, &mut buffer);
|
OTFrame::write_bytes(frame1, &mut buffer);
|
||||||
buffer[17] = 255;
|
buffer[9] = 255;
|
||||||
let framed = ITFrame::read_frame(&mut buffer);
|
let framed = ITFrame::read_frame(&mut buffer);
|
||||||
assert_eq!(framed, None);
|
assert_eq!(framed, None);
|
||||||
}
|
}
|
||||||
@ -543,18 +546,16 @@ mod tests {
|
|||||||
|
|
||||||
let frame1 = OTFrame::Data {
|
let frame1 = OTFrame::Data {
|
||||||
mid: 7u64,
|
mid: 7u64,
|
||||||
start: 1u64,
|
|
||||||
data: Bytes::from(&b"foobar"[..]),
|
data: Bytes::from(&b"foobar"[..]),
|
||||||
};
|
};
|
||||||
|
|
||||||
OTFrame::write_bytes(frame1, &mut buffer);
|
OTFrame::write_bytes(frame1, &mut buffer);
|
||||||
buffer[17] = 3;
|
buffer[9] = 3;
|
||||||
let framed = ITFrame::read_frame(&mut buffer);
|
let framed = ITFrame::read_frame(&mut buffer);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
framed,
|
framed,
|
||||||
Some(ITFrame::Data {
|
Some(ITFrame::Data {
|
||||||
mid: 7u64,
|
mid: 7u64,
|
||||||
start: 1u64,
|
|
||||||
data: BytesMut::from(&b"foo"[..]),
|
data: BytesMut::from(&b"foo"[..]),
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
@ -80,7 +80,9 @@ where
|
|||||||
.send(InitFrame::Raw(WRONG_NUMBER.as_bytes().to_vec()))
|
.send(InitFrame::Raw(WRONG_NUMBER.as_bytes().to_vec()))
|
||||||
.await?;
|
.await?;
|
||||||
Err(InitProtocolError::WrongMagicNumber(magic_number))
|
Err(InitProtocolError::WrongMagicNumber(magic_number))
|
||||||
} else if version != VELOREN_NETWORK_VERSION {
|
} else if version[0] != VELOREN_NETWORK_VERSION[0]
|
||||||
|
|| version[1] != VELOREN_NETWORK_VERSION[1]
|
||||||
|
{
|
||||||
error!(?version, "Connection with wrong network version");
|
error!(?version, "Connection with wrong network version");
|
||||||
#[cfg(debug_assertions)]
|
#[cfg(debug_assertions)]
|
||||||
drain
|
drain
|
||||||
|
@ -65,9 +65,7 @@ pub use metrics::ProtocolMetricCache;
|
|||||||
pub use metrics::ProtocolMetrics;
|
pub use metrics::ProtocolMetrics;
|
||||||
pub use mpsc::{MpscMsg, MpscRecvProtocol, MpscSendProtocol};
|
pub use mpsc::{MpscMsg, MpscRecvProtocol, MpscSendProtocol};
|
||||||
pub use tcp::{TcpRecvProtocol, TcpSendProtocol};
|
pub use tcp::{TcpRecvProtocol, TcpSendProtocol};
|
||||||
pub use types::{
|
pub use types::{Bandwidth, Cid, Pid, Prio, Promises, Sid, HIGHEST_PRIO, VELOREN_NETWORK_VERSION};
|
||||||
Bandwidth, Cid, Mid, Pid, Prio, Promises, Sid, HIGHEST_PRIO, VELOREN_NETWORK_VERSION,
|
|
||||||
};
|
|
||||||
|
|
||||||
///use at own risk, might change any time, for internal benchmarks
|
///use at own risk, might change any time, for internal benchmarks
|
||||||
pub mod _internal {
|
pub mod _internal {
|
||||||
|
@ -57,12 +57,10 @@ impl OTMessage {
|
|||||||
fn get_next_data(&mut self) -> OTFrame {
|
fn get_next_data(&mut self) -> OTFrame {
|
||||||
let to_send = std::cmp::min(self.data.len(), Self::FRAME_DATA_SIZE as usize);
|
let to_send = std::cmp::min(self.data.len(), Self::FRAME_DATA_SIZE as usize);
|
||||||
let data = self.data.split_to(to_send);
|
let data = self.data.split_to(to_send);
|
||||||
let start = self.start;
|
|
||||||
self.start += Self::FRAME_DATA_SIZE;
|
self.start += Self::FRAME_DATA_SIZE;
|
||||||
|
|
||||||
OTFrame::Data {
|
OTFrame::Data {
|
||||||
mid: self.mid,
|
mid: self.mid,
|
||||||
start,
|
|
||||||
data,
|
data,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -78,7 +78,6 @@ where
|
|||||||
match &event {
|
match &event {
|
||||||
ProtocolEvent::Message {
|
ProtocolEvent::Message {
|
||||||
data: _data,
|
data: _data,
|
||||||
mid: _,
|
|
||||||
sid: _sid,
|
sid: _sid,
|
||||||
} => {
|
} => {
|
||||||
#[cfg(feature = "metrics")]
|
#[cfg(feature = "metrics")]
|
||||||
@ -118,7 +117,7 @@ where
|
|||||||
MpscMsg::Event(e) => {
|
MpscMsg::Event(e) => {
|
||||||
#[cfg(feature = "metrics")]
|
#[cfg(feature = "metrics")]
|
||||||
{
|
{
|
||||||
if let ProtocolEvent::Message { data, mid: _, sid } = &e {
|
if let ProtocolEvent::Message { data, sid } = &e {
|
||||||
let sid = *sid;
|
let sid = *sid;
|
||||||
let bytes = data.len() as u64;
|
let bytes = data.len() as u64;
|
||||||
let line = self.metrics.init_sid(sid);
|
let line = self.metrics.init_sid(sid);
|
||||||
|
@ -28,6 +28,7 @@ where
|
|||||||
{
|
{
|
||||||
buffer: BytesMut,
|
buffer: BytesMut,
|
||||||
store: PrioManager,
|
store: PrioManager,
|
||||||
|
next_mid: Mid,
|
||||||
closing_streams: Vec<Sid>,
|
closing_streams: Vec<Sid>,
|
||||||
notify_closing_streams: Vec<Sid>,
|
notify_closing_streams: Vec<Sid>,
|
||||||
pending_shutdown: bool,
|
pending_shutdown: bool,
|
||||||
@ -59,6 +60,7 @@ where
|
|||||||
Self {
|
Self {
|
||||||
buffer: BytesMut::new(),
|
buffer: BytesMut::new(),
|
||||||
store: PrioManager::new(metrics.clone()),
|
store: PrioManager::new(metrics.clone()),
|
||||||
|
next_mid: 0u64,
|
||||||
closing_streams: vec![],
|
closing_streams: vec![],
|
||||||
notify_closing_streams: vec![],
|
notify_closing_streams: vec![],
|
||||||
pending_shutdown: false,
|
pending_shutdown: false,
|
||||||
@ -146,9 +148,10 @@ where
|
|||||||
self.pending_shutdown = true;
|
self.pending_shutdown = true;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
ProtocolEvent::Message { data, mid, sid } => {
|
ProtocolEvent::Message { data, sid } => {
|
||||||
self.metrics.smsg_ib(sid, data.len() as u64);
|
self.metrics.smsg_ib(sid, data.len() as u64);
|
||||||
self.store.add(data, mid, sid);
|
self.store.add(data, self.next_mid, sid);
|
||||||
|
self.next_mid += 1;
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -160,12 +163,7 @@ where
|
|||||||
let mut data_frames = 0;
|
let mut data_frames = 0;
|
||||||
let mut data_bandwidth = 0;
|
let mut data_bandwidth = 0;
|
||||||
for frame in frames {
|
for frame in frames {
|
||||||
if let OTFrame::Data {
|
if let OTFrame::Data { mid: _, data } = &frame {
|
||||||
mid: _,
|
|
||||||
start: _,
|
|
||||||
data,
|
|
||||||
} = &frame
|
|
||||||
{
|
|
||||||
data_bandwidth += data.len();
|
data_bandwidth += data.len();
|
||||||
data_frames += 1;
|
data_frames += 1;
|
||||||
}
|
}
|
||||||
@ -228,12 +226,13 @@ where
|
|||||||
sid,
|
sid,
|
||||||
prio,
|
prio,
|
||||||
promises,
|
promises,
|
||||||
|
guaranteed_bandwidth,
|
||||||
} => {
|
} => {
|
||||||
break 'outer Ok(ProtocolEvent::OpenStream {
|
break 'outer Ok(ProtocolEvent::OpenStream {
|
||||||
sid,
|
sid,
|
||||||
prio: prio.min(crate::types::HIGHEST_PRIO),
|
prio: prio.min(crate::types::HIGHEST_PRIO),
|
||||||
promises,
|
promises,
|
||||||
guaranteed_bandwidth: 1_000_000,
|
guaranteed_bandwidth,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
ITFrame::CloseStream { sid } => {
|
ITFrame::CloseStream { sid } => {
|
||||||
@ -244,11 +243,7 @@ where
|
|||||||
self.metrics.rmsg_ib(sid, length);
|
self.metrics.rmsg_ib(sid, length);
|
||||||
self.incoming.insert(mid, m);
|
self.incoming.insert(mid, m);
|
||||||
},
|
},
|
||||||
ITFrame::Data {
|
ITFrame::Data { mid, data } => {
|
||||||
mid,
|
|
||||||
start: _,
|
|
||||||
data,
|
|
||||||
} => {
|
|
||||||
self.metrics.rdata_frames_b(data.len() as u64);
|
self.metrics.rdata_frames_b(data.len() as u64);
|
||||||
let m = match self.incoming.get_mut(&mid) {
|
let m = match self.incoming.get_mut(&mid) {
|
||||||
Some(m) => m,
|
Some(m) => m,
|
||||||
@ -271,7 +266,6 @@ where
|
|||||||
);
|
);
|
||||||
break 'outer Ok(ProtocolEvent::Message {
|
break 'outer Ok(ProtocolEvent::Message {
|
||||||
sid: m.sid,
|
sid: m.sid,
|
||||||
mid,
|
|
||||||
data: m.data.freeze(),
|
data: m.data.freeze(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -431,7 +425,6 @@ mod tests {
|
|||||||
let _ = r.recv().await.unwrap();
|
let _ = r.recv().await.unwrap();
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid: Sid::new(10),
|
sid: Sid::new(10),
|
||||||
mid: 0,
|
|
||||||
data: Bytes::from(&[188u8; 600][..]),
|
data: Bytes::from(&[188u8; 600][..]),
|
||||||
};
|
};
|
||||||
s.send(event.clone()).await.unwrap();
|
s.send(event.clone()).await.unwrap();
|
||||||
@ -441,7 +434,6 @@ mod tests {
|
|||||||
// 2nd short message
|
// 2nd short message
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid: Sid::new(10),
|
sid: Sid::new(10),
|
||||||
mid: 1,
|
|
||||||
data: Bytes::from(&[7u8; 30][..]),
|
data: Bytes::from(&[7u8; 30][..]),
|
||||||
};
|
};
|
||||||
s.send(event.clone()).await.unwrap();
|
s.send(event.clone()).await.unwrap();
|
||||||
@ -467,7 +459,6 @@ mod tests {
|
|||||||
let _ = r.recv().await.unwrap();
|
let _ = r.recv().await.unwrap();
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid,
|
sid,
|
||||||
mid: 77,
|
|
||||||
data: Bytes::from(&[99u8; 500_000][..]),
|
data: Bytes::from(&[99u8; 500_000][..]),
|
||||||
};
|
};
|
||||||
s.send(event.clone()).await.unwrap();
|
s.send(event.clone()).await.unwrap();
|
||||||
@ -495,7 +486,6 @@ mod tests {
|
|||||||
let _ = r.recv().await.unwrap();
|
let _ = r.recv().await.unwrap();
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid,
|
sid,
|
||||||
mid: 77,
|
|
||||||
data: Bytes::from(&[99u8; 500_000][..]),
|
data: Bytes::from(&[99u8; 500_000][..]),
|
||||||
};
|
};
|
||||||
s.send(event).await.unwrap();
|
s.send(event).await.unwrap();
|
||||||
@ -524,7 +514,6 @@ mod tests {
|
|||||||
let _ = r.recv().await.unwrap();
|
let _ = r.recv().await.unwrap();
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid,
|
sid,
|
||||||
mid: 77,
|
|
||||||
data: Bytes::from(&[99u8; 500_000][..]),
|
data: Bytes::from(&[99u8; 500_000][..]),
|
||||||
};
|
};
|
||||||
s.send(event).await.unwrap();
|
s.send(event).await.unwrap();
|
||||||
@ -556,14 +545,12 @@ mod tests {
|
|||||||
s.send(event).await.unwrap();
|
s.send(event).await.unwrap();
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid,
|
sid,
|
||||||
mid: 77,
|
|
||||||
data: Bytes::from(&[99u8; 500_000][..]),
|
data: Bytes::from(&[99u8; 500_000][..]),
|
||||||
};
|
};
|
||||||
s.send(event).await.unwrap();
|
s.send(event).await.unwrap();
|
||||||
s.flush(1_000_000, Duration::from_secs(1)).await.unwrap();
|
s.flush(1_000_000, Duration::from_secs(1)).await.unwrap();
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid,
|
sid,
|
||||||
mid: 78,
|
|
||||||
data: Bytes::from(&[100u8; 500_000][..]),
|
data: Bytes::from(&[100u8; 500_000][..]),
|
||||||
};
|
};
|
||||||
s.send(event).await.unwrap();
|
s.send(event).await.unwrap();
|
||||||
@ -593,6 +580,7 @@ mod tests {
|
|||||||
sid,
|
sid,
|
||||||
prio: 5u8,
|
prio: 5u8,
|
||||||
promises: Promises::COMPRESSED,
|
promises: Promises::COMPRESSED,
|
||||||
|
guaranteed_bandwidth: 1_000_000,
|
||||||
}
|
}
|
||||||
.write_bytes(&mut bytes);
|
.write_bytes(&mut bytes);
|
||||||
OTFrame::DataHeader {
|
OTFrame::DataHeader {
|
||||||
@ -605,13 +593,11 @@ mod tests {
|
|||||||
|
|
||||||
OTFrame::Data {
|
OTFrame::Data {
|
||||||
mid: 99,
|
mid: 99,
|
||||||
start: 0,
|
|
||||||
data: Bytes::from(&DATA1[..]),
|
data: Bytes::from(&DATA1[..]),
|
||||||
}
|
}
|
||||||
.write_bytes(&mut bytes);
|
.write_bytes(&mut bytes);
|
||||||
OTFrame::Data {
|
OTFrame::Data {
|
||||||
mid: 99,
|
mid: 99,
|
||||||
start: DATA1.len() as u64,
|
|
||||||
data: Bytes::from(&DATA2[..]),
|
data: Bytes::from(&DATA2[..]),
|
||||||
}
|
}
|
||||||
.write_bytes(&mut bytes);
|
.write_bytes(&mut bytes);
|
||||||
@ -641,6 +627,7 @@ mod tests {
|
|||||||
sid,
|
sid,
|
||||||
prio: 5u8,
|
prio: 5u8,
|
||||||
promises: Promises::COMPRESSED,
|
promises: Promises::COMPRESSED,
|
||||||
|
guaranteed_bandwidth: 1_000_000,
|
||||||
}
|
}
|
||||||
.write_bytes(&mut bytes);
|
.write_bytes(&mut bytes);
|
||||||
s.send(bytes.split()).await.unwrap();
|
s.send(bytes.split()).await.unwrap();
|
||||||
@ -670,7 +657,6 @@ mod tests {
|
|||||||
let _ = p2.1.recv().await.unwrap();
|
let _ = p2.1.recv().await.unwrap();
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid: Sid::new(10),
|
sid: Sid::new(10),
|
||||||
mid: 0,
|
|
||||||
data: Bytes::from(&[188u8; 600][..]),
|
data: Bytes::from(&[188u8; 600][..]),
|
||||||
};
|
};
|
||||||
p2.0.send(event.clone()).await.unwrap();
|
p2.0.send(event.clone()).await.unwrap();
|
||||||
@ -695,7 +681,6 @@ mod tests {
|
|||||||
p2.0.notify_from_recv(e);
|
p2.0.notify_from_recv(e);
|
||||||
let event = ProtocolEvent::Message {
|
let event = ProtocolEvent::Message {
|
||||||
sid: Sid::new(10),
|
sid: Sid::new(10),
|
||||||
mid: 0,
|
|
||||||
data: Bytes::from(&[188u8; 600][..]),
|
data: Bytes::from(&[188u8; 600][..]),
|
||||||
};
|
};
|
||||||
p2.0.send(event.clone()).await.unwrap();
|
p2.0.send(event.clone()).await.unwrap();
|
||||||
|
@ -49,7 +49,7 @@ impl Promises {
|
|||||||
|
|
||||||
pub(crate) const VELOREN_MAGIC_NUMBER: [u8; 7] = *b"VELOREN";
|
pub(crate) const VELOREN_MAGIC_NUMBER: [u8; 7] = *b"VELOREN";
|
||||||
/// When this semver differs, 2 Networks can't communicate.
|
/// When this semver differs, 2 Networks can't communicate.
|
||||||
pub const VELOREN_NETWORK_VERSION: [u32; 3] = [0, 5, 0];
|
pub const VELOREN_NETWORK_VERSION: [u32; 3] = [0, 6, 0];
|
||||||
pub(crate) const STREAM_ID_OFFSET1: Sid = Sid::new(0);
|
pub(crate) const STREAM_ID_OFFSET1: Sid = Sid::new(0);
|
||||||
pub(crate) const STREAM_ID_OFFSET2: Sid = Sid::new(u64::MAX / 2);
|
pub(crate) const STREAM_ID_OFFSET2: Sid = Sid::new(u64::MAX / 2);
|
||||||
/// Maximal possible Prio to choose (for performance reasons)
|
/// Maximal possible Prio to choose (for performance reasons)
|
||||||
|
@ -438,6 +438,9 @@ impl Participant {
|
|||||||
/// link for further documentation. You can combine them, e.g.
|
/// link for further documentation. You can combine them, e.g.
|
||||||
/// `Promises::ORDERED | Promises::CONSISTENCY` The Stream will then
|
/// `Promises::ORDERED | Promises::CONSISTENCY` The Stream will then
|
||||||
/// guarantee that those promises are met.
|
/// guarantee that those promises are met.
|
||||||
|
/// * `bandwidth` - sets a guaranteed bandwidth which is reserved for this
|
||||||
|
/// stream. When excess bandwidth is available it will be used. See
|
||||||
|
/// [`Bandwidth`] for details.
|
||||||
///
|
///
|
||||||
/// A [`ParticipantError`] might be thrown if the `Participant` is already
|
/// A [`ParticipantError`] might be thrown if the `Participant` is already
|
||||||
/// closed. [`Streams`] can be created without a answer from the remote
|
/// closed. [`Streams`] can be created without a answer from the remote
|
||||||
@ -460,7 +463,7 @@ impl Participant {
|
|||||||
/// .connect(ProtocolAddr::Tcp("127.0.0.1:2100".parse().unwrap()))
|
/// .connect(ProtocolAddr::Tcp("127.0.0.1:2100".parse().unwrap()))
|
||||||
/// .await?;
|
/// .await?;
|
||||||
/// let _s1 = p1
|
/// let _s1 = p1
|
||||||
/// .open(4, Promises::ORDERED | Promises::CONSISTENCY)
|
/// .open(4, Promises::ORDERED | Promises::CONSISTENCY, 1000)
|
||||||
/// .await?;
|
/// .await?;
|
||||||
/// # Ok(())
|
/// # Ok(())
|
||||||
/// })
|
/// })
|
||||||
@ -468,16 +471,22 @@ impl Participant {
|
|||||||
/// ```
|
/// ```
|
||||||
///
|
///
|
||||||
/// [`Prio`]: network_protocol::Prio
|
/// [`Prio`]: network_protocol::Prio
|
||||||
|
/// [`Bandwidth`]: network_protocol::Bandwidth
|
||||||
/// [`Promises`]: network_protocol::Promises
|
/// [`Promises`]: network_protocol::Promises
|
||||||
/// [`Streams`]: crate::api::Stream
|
/// [`Streams`]: crate::api::Stream
|
||||||
#[instrument(name="network", skip(self, prio, promises), fields(p = %self.local_pid))]
|
#[instrument(name="network", skip(self, prio, promises), fields(p = %self.local_pid))]
|
||||||
pub async fn open(&self, prio: u8, promises: Promises) -> Result<Stream, ParticipantError> {
|
pub async fn open(
|
||||||
|
&self,
|
||||||
|
prio: u8,
|
||||||
|
promises: Promises,
|
||||||
|
bandwidth: Bandwidth,
|
||||||
|
) -> Result<Stream, ParticipantError> {
|
||||||
debug_assert!(prio <= network_protocol::HIGHEST_PRIO, "invalid prio");
|
debug_assert!(prio <= network_protocol::HIGHEST_PRIO, "invalid prio");
|
||||||
let (p2a_return_stream_s, p2a_return_stream_r) = oneshot::channel::<Stream>();
|
let (p2a_return_stream_s, p2a_return_stream_r) = oneshot::channel::<Stream>();
|
||||||
if let Err(e) = self.a2b_open_stream_s.lock().await.send((
|
if let Err(e) = self.a2b_open_stream_s.lock().await.send((
|
||||||
prio,
|
prio,
|
||||||
promises,
|
promises,
|
||||||
1_000_000,
|
bandwidth,
|
||||||
p2a_return_stream_s,
|
p2a_return_stream_s,
|
||||||
)) {
|
)) {
|
||||||
debug!(?e, "bParticipant is already closed, notifying");
|
debug!(?e, "bParticipant is already closed, notifying");
|
||||||
@ -519,7 +528,7 @@ impl Participant {
|
|||||||
/// # remote.listen(ProtocolAddr::Tcp("127.0.0.1:2110".parse().unwrap())).await?;
|
/// # remote.listen(ProtocolAddr::Tcp("127.0.0.1:2110".parse().unwrap())).await?;
|
||||||
/// let p1 = network.connect(ProtocolAddr::Tcp("127.0.0.1:2110".parse().unwrap())).await?;
|
/// let p1 = network.connect(ProtocolAddr::Tcp("127.0.0.1:2110".parse().unwrap())).await?;
|
||||||
/// # let p2 = remote.connected().await?;
|
/// # let p2 = remote.connected().await?;
|
||||||
/// # p2.open(4, Promises::ORDERED | Promises::CONSISTENCY).await?;
|
/// # p2.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0).await?;
|
||||||
/// let _s1 = p1.opened().await?;
|
/// let _s1 = p1.opened().await?;
|
||||||
/// # Ok(())
|
/// # Ok(())
|
||||||
/// })
|
/// })
|
||||||
@ -704,7 +713,7 @@ impl Stream {
|
|||||||
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2200".parse().unwrap())).await?;
|
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2200".parse().unwrap())).await?;
|
||||||
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2200".parse().unwrap())).await?;
|
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2200".parse().unwrap())).await?;
|
||||||
/// # // keep it alive
|
/// # // keep it alive
|
||||||
/// # let _stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY).await?;
|
/// # let _stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0).await?;
|
||||||
/// let participant_a = network.connected().await?;
|
/// let participant_a = network.connected().await?;
|
||||||
/// let mut stream_a = participant_a.opened().await?;
|
/// let mut stream_a = participant_a.opened().await?;
|
||||||
/// //Send Message
|
/// //Send Message
|
||||||
@ -746,8 +755,8 @@ impl Stream {
|
|||||||
/// # let remote1_p = remote1.connect(ProtocolAddr::Tcp("127.0.0.1:2210".parse().unwrap())).await?;
|
/// # let remote1_p = remote1.connect(ProtocolAddr::Tcp("127.0.0.1:2210".parse().unwrap())).await?;
|
||||||
/// # let remote2_p = remote2.connect(ProtocolAddr::Tcp("127.0.0.1:2210".parse().unwrap())).await?;
|
/// # let remote2_p = remote2.connect(ProtocolAddr::Tcp("127.0.0.1:2210".parse().unwrap())).await?;
|
||||||
/// # assert_eq!(remote1_p.remote_pid(), remote2_p.remote_pid());
|
/// # assert_eq!(remote1_p.remote_pid(), remote2_p.remote_pid());
|
||||||
/// # remote1_p.open(4, Promises::ORDERED | Promises::CONSISTENCY).await?;
|
/// # remote1_p.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0).await?;
|
||||||
/// # remote2_p.open(4, Promises::ORDERED | Promises::CONSISTENCY).await?;
|
/// # remote2_p.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0).await?;
|
||||||
/// let participant_a = network.connected().await?;
|
/// let participant_a = network.connected().await?;
|
||||||
/// let participant_b = network.connected().await?;
|
/// let participant_b = network.connected().await?;
|
||||||
/// let mut stream_a = participant_a.opened().await?;
|
/// let mut stream_a = participant_a.opened().await?;
|
||||||
@ -801,7 +810,7 @@ impl Stream {
|
|||||||
/// runtime.block_on(async {
|
/// runtime.block_on(async {
|
||||||
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2220".parse().unwrap())).await?;
|
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2220".parse().unwrap())).await?;
|
||||||
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2220".parse().unwrap())).await?;
|
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2220".parse().unwrap())).await?;
|
||||||
/// # let mut stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY).await?;
|
/// # let mut stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0).await?;
|
||||||
/// # stream_p.send("Hello World");
|
/// # stream_p.send("Hello World");
|
||||||
/// let participant_a = network.connected().await?;
|
/// let participant_a = network.connected().await?;
|
||||||
/// let mut stream_a = participant_a.opened().await?;
|
/// let mut stream_a = participant_a.opened().await?;
|
||||||
@ -834,7 +843,7 @@ impl Stream {
|
|||||||
/// runtime.block_on(async {
|
/// runtime.block_on(async {
|
||||||
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2230".parse().unwrap())).await?;
|
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2230".parse().unwrap())).await?;
|
||||||
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2230".parse().unwrap())).await?;
|
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2230".parse().unwrap())).await?;
|
||||||
/// # let mut stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY).await?;
|
/// # let mut stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0).await?;
|
||||||
/// # stream_p.send("Hello World");
|
/// # stream_p.send("Hello World");
|
||||||
/// let participant_a = network.connected().await?;
|
/// let participant_a = network.connected().await?;
|
||||||
/// let mut stream_a = participant_a.opened().await?;
|
/// let mut stream_a = participant_a.opened().await?;
|
||||||
@ -889,7 +898,7 @@ impl Stream {
|
|||||||
/// runtime.block_on(async {
|
/// runtime.block_on(async {
|
||||||
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2240".parse().unwrap())).await?;
|
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2240".parse().unwrap())).await?;
|
||||||
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2240".parse().unwrap())).await?;
|
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2240".parse().unwrap())).await?;
|
||||||
/// # let mut stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY).await?;
|
/// # let mut stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0).await?;
|
||||||
/// # stream_p.send("Hello World");
|
/// # stream_p.send("Hello World");
|
||||||
/// # std::thread::sleep(std::time::Duration::from_secs(1));
|
/// # std::thread::sleep(std::time::Duration::from_secs(1));
|
||||||
/// let participant_a = network.connected().await?;
|
/// let participant_a = network.connected().await?;
|
||||||
|
@ -222,7 +222,6 @@ mod tests {
|
|||||||
s.send(event.clone()).await.unwrap();
|
s.send(event.clone()).await.unwrap();
|
||||||
s.send(ProtocolEvent::Message {
|
s.send(ProtocolEvent::Message {
|
||||||
sid: Sid::new(1),
|
sid: Sid::new(1),
|
||||||
mid: 0,
|
|
||||||
data: Bytes::from(&[8u8; 8][..]),
|
data: Bytes::from(&[8u8; 8][..]),
|
||||||
})
|
})
|
||||||
.await
|
.await
|
||||||
|
@ -51,7 +51,7 @@
|
|||||||
//! .connect(ProtocolAddr::Tcp("127.0.0.1:12345".parse().unwrap()))
|
//! .connect(ProtocolAddr::Tcp("127.0.0.1:12345".parse().unwrap()))
|
||||||
//! .await?;
|
//! .await?;
|
||||||
//! let mut stream = server
|
//! let mut stream = server
|
||||||
//! .open(4, Promises::ORDERED | Promises::CONSISTENCY)
|
//! .open(4, Promises::ORDERED | Promises::CONSISTENCY, 0)
|
||||||
//! .await?;
|
//! .await?;
|
||||||
//! stream.send("Hello World")?;
|
//! stream.send("Hello World")?;
|
||||||
//! Ok(())
|
//! Ok(())
|
||||||
|
@ -83,7 +83,7 @@ impl Message {
|
|||||||
/// # runtime.block_on(async {
|
/// # runtime.block_on(async {
|
||||||
/// # network.listen(ProtocolAddr::Tcp("127.0.0.1:2300".parse().unwrap())).await?;
|
/// # network.listen(ProtocolAddr::Tcp("127.0.0.1:2300".parse().unwrap())).await?;
|
||||||
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2300".parse().unwrap())).await?;
|
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2300".parse().unwrap())).await?;
|
||||||
/// # let mut stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY).await?;
|
/// # let mut stream_p = remote_p.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0).await?;
|
||||||
/// # stream_p.send("Hello World");
|
/// # stream_p.send("Hello World");
|
||||||
/// # let participant_a = network.connected().await?;
|
/// # let participant_a = network.connected().await?;
|
||||||
/// let mut stream_a = participant_a.opened().await?;
|
/// let mut stream_a = participant_a.opened().await?;
|
||||||
|
@ -51,12 +51,6 @@ struct ControlChannels {
|
|||||||
s2b_shutdown_bparticipant_r: oneshot::Receiver<S2bShutdownBparticipant>, /* own */
|
s2b_shutdown_bparticipant_r: oneshot::Receiver<S2bShutdownBparticipant>, /* own */
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
struct ShutdownInfo {
|
|
||||||
b2b_close_stream_opened_sender_s: Option<oneshot::Sender<()>>,
|
|
||||||
error: Option<ParticipantError>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct OpenStreamInfo {
|
struct OpenStreamInfo {
|
||||||
a2b_msg_s: crossbeam_channel::Sender<(Sid, Bytes)>,
|
a2b_msg_s: crossbeam_channel::Sender<(Sid, Bytes)>,
|
||||||
@ -74,7 +68,6 @@ pub struct BParticipant {
|
|||||||
run_channels: Option<ControlChannels>,
|
run_channels: Option<ControlChannels>,
|
||||||
shutdown_barrier: AtomicI32,
|
shutdown_barrier: AtomicI32,
|
||||||
metrics: Arc<NetworkMetrics>,
|
metrics: Arc<NetworkMetrics>,
|
||||||
no_channel_error_info: RwLock<(Instant, u64)>,
|
|
||||||
open_stream_channels: Arc<Mutex<Option<OpenStreamInfo>>>,
|
open_stream_channels: Arc<Mutex<Option<OpenStreamInfo>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -84,7 +77,7 @@ impl BParticipant {
|
|||||||
const BARR_RECV: i32 = 4;
|
const BARR_RECV: i32 = 4;
|
||||||
const BARR_SEND: i32 = 2;
|
const BARR_SEND: i32 = 2;
|
||||||
const TICK_TIME: Duration = Duration::from_millis(Self::TICK_TIME_MS);
|
const TICK_TIME: Duration = Duration::from_millis(Self::TICK_TIME_MS);
|
||||||
const TICK_TIME_MS: u64 = 10;
|
const TICK_TIME_MS: u64 = 5;
|
||||||
|
|
||||||
#[allow(clippy::type_complexity)]
|
#[allow(clippy::type_complexity)]
|
||||||
pub(crate) fn new(
|
pub(crate) fn new(
|
||||||
@ -124,7 +117,6 @@ impl BParticipant {
|
|||||||
),
|
),
|
||||||
run_channels,
|
run_channels,
|
||||||
metrics,
|
metrics,
|
||||||
no_channel_error_info: RwLock::new((Instant::now(), 0)),
|
|
||||||
open_stream_channels: Arc::new(Mutex::new(None)),
|
open_stream_channels: Arc::new(Mutex::new(None)),
|
||||||
},
|
},
|
||||||
a2b_open_stream_s,
|
a2b_open_stream_s,
|
||||||
@ -203,7 +195,6 @@ impl BParticipant {
|
|||||||
let mut interval = tokio::time::interval(Self::TICK_TIME);
|
let mut interval = tokio::time::interval(Self::TICK_TIME);
|
||||||
let mut last_instant = Instant::now();
|
let mut last_instant = Instant::now();
|
||||||
let mut stream_ids = self.offset_sid;
|
let mut stream_ids = self.offset_sid;
|
||||||
let mut fake_mid = 0; //TODO: move MID to protocol, should be inc per stream ? or ?
|
|
||||||
trace!("workaround, actively wait for first protocol");
|
trace!("workaround, actively wait for first protocol");
|
||||||
b2b_add_protocol_r
|
b2b_add_protocol_r
|
||||||
.recv()
|
.recv()
|
||||||
@ -267,13 +258,8 @@ impl BParticipant {
|
|||||||
|
|
||||||
// get all messages and assign it to a channel
|
// get all messages and assign it to a channel
|
||||||
for (sid, buffer) in a2b_msg_r.try_iter() {
|
for (sid, buffer) in a2b_msg_r.try_iter() {
|
||||||
fake_mid += 1;
|
|
||||||
active
|
active
|
||||||
.send(ProtocolEvent::Message {
|
.send(ProtocolEvent::Message { data: buffer, sid })
|
||||||
data: buffer,
|
|
||||||
mid: fake_mid,
|
|
||||||
sid,
|
|
||||||
})
|
|
||||||
.await?
|
.await?
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -416,7 +402,7 @@ impl BParticipant {
|
|||||||
self.delete_stream(sid).await;
|
self.delete_stream(sid).await;
|
||||||
retrigger(cid, p, &mut recv_protocols);
|
retrigger(cid, p, &mut recv_protocols);
|
||||||
},
|
},
|
||||||
Ok(ProtocolEvent::Message { data, mid: _, sid }) => {
|
Ok(ProtocolEvent::Message { data, sid }) => {
|
||||||
let lock = self.streams.read().await;
|
let lock = self.streams.read().await;
|
||||||
match lock.get(&sid) {
|
match lock.get(&sid) {
|
||||||
Some(stream) => {
|
Some(stream) => {
|
||||||
|
@ -230,7 +230,7 @@ fn close_network_then_disconnect_part() {
|
|||||||
fn opened_stream_before_remote_part_is_closed() {
|
fn opened_stream_before_remote_part_is_closed() {
|
||||||
let (_, _) = helper::setup(false, 0);
|
let (_, _) = helper::setup(false, 0);
|
||||||
let (r, _n_a, p_a, _, _n_b, p_b, _) = network_participant_stream(tcp());
|
let (r, _n_a, p_a, _, _n_b, p_b, _) = network_participant_stream(tcp());
|
||||||
let mut s2_a = r.block_on(p_a.open(4, Promises::empty())).unwrap();
|
let mut s2_a = r.block_on(p_a.open(4, Promises::empty(), 0)).unwrap();
|
||||||
s2_a.send("HelloWorld").unwrap();
|
s2_a.send("HelloWorld").unwrap();
|
||||||
let mut s2_b = r.block_on(p_b.opened()).unwrap();
|
let mut s2_b = r.block_on(p_b.opened()).unwrap();
|
||||||
drop(p_a);
|
drop(p_a);
|
||||||
@ -243,7 +243,7 @@ fn opened_stream_before_remote_part_is_closed() {
|
|||||||
fn opened_stream_after_remote_part_is_closed() {
|
fn opened_stream_after_remote_part_is_closed() {
|
||||||
let (_, _) = helper::setup(false, 0);
|
let (_, _) = helper::setup(false, 0);
|
||||||
let (r, _n_a, p_a, _, _n_b, p_b, _) = network_participant_stream(tcp());
|
let (r, _n_a, p_a, _, _n_b, p_b, _) = network_participant_stream(tcp());
|
||||||
let mut s2_a = r.block_on(p_a.open(3, Promises::empty())).unwrap();
|
let mut s2_a = r.block_on(p_a.open(3, Promises::empty(), 0)).unwrap();
|
||||||
s2_a.send("HelloWorld").unwrap();
|
s2_a.send("HelloWorld").unwrap();
|
||||||
drop(p_a);
|
drop(p_a);
|
||||||
std::thread::sleep(std::time::Duration::from_millis(1000));
|
std::thread::sleep(std::time::Duration::from_millis(1000));
|
||||||
@ -260,14 +260,14 @@ fn opened_stream_after_remote_part_is_closed() {
|
|||||||
fn open_stream_after_remote_part_is_closed() {
|
fn open_stream_after_remote_part_is_closed() {
|
||||||
let (_, _) = helper::setup(false, 0);
|
let (_, _) = helper::setup(false, 0);
|
||||||
let (r, _n_a, p_a, _, _n_b, p_b, _) = network_participant_stream(tcp());
|
let (r, _n_a, p_a, _, _n_b, p_b, _) = network_participant_stream(tcp());
|
||||||
let mut s2_a = r.block_on(p_a.open(4, Promises::empty())).unwrap();
|
let mut s2_a = r.block_on(p_a.open(4, Promises::empty(), 0)).unwrap();
|
||||||
s2_a.send("HelloWorld").unwrap();
|
s2_a.send("HelloWorld").unwrap();
|
||||||
drop(p_a);
|
drop(p_a);
|
||||||
std::thread::sleep(std::time::Duration::from_millis(1000));
|
std::thread::sleep(std::time::Duration::from_millis(1000));
|
||||||
let mut s2_b = r.block_on(p_b.opened()).unwrap();
|
let mut s2_b = r.block_on(p_b.opened()).unwrap();
|
||||||
assert_eq!(r.block_on(s2_b.recv()), Ok("HelloWorld".to_string()));
|
assert_eq!(r.block_on(s2_b.recv()), Ok("HelloWorld".to_string()));
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
r.block_on(p_b.open(5, Promises::empty())).unwrap_err(),
|
r.block_on(p_b.open(5, Promises::empty(), 0)).unwrap_err(),
|
||||||
ParticipantError::ParticipantDisconnected
|
ParticipantError::ParticipantDisconnected
|
||||||
);
|
);
|
||||||
drop((_n_a, _n_b, p_b)); //clean teardown
|
drop((_n_a, _n_b, p_b)); //clean teardown
|
||||||
@ -294,7 +294,7 @@ fn open_participant_before_remote_part_is_closed() {
|
|||||||
let addr = tcp();
|
let addr = tcp();
|
||||||
r.block_on(n_a.listen(addr.clone())).unwrap();
|
r.block_on(n_a.listen(addr.clone())).unwrap();
|
||||||
let p_b = r.block_on(n_b.connect(addr)).unwrap();
|
let p_b = r.block_on(n_b.connect(addr)).unwrap();
|
||||||
let mut s1_b = r.block_on(p_b.open(4, Promises::empty())).unwrap();
|
let mut s1_b = r.block_on(p_b.open(4, Promises::empty(), 0)).unwrap();
|
||||||
s1_b.send("HelloWorld").unwrap();
|
s1_b.send("HelloWorld").unwrap();
|
||||||
let p_a = r.block_on(n_a.connected()).unwrap();
|
let p_a = r.block_on(n_a.connected()).unwrap();
|
||||||
drop(s1_b);
|
drop(s1_b);
|
||||||
@ -314,7 +314,7 @@ fn open_participant_after_remote_part_is_closed() {
|
|||||||
let addr = tcp();
|
let addr = tcp();
|
||||||
r.block_on(n_a.listen(addr.clone())).unwrap();
|
r.block_on(n_a.listen(addr.clone())).unwrap();
|
||||||
let p_b = r.block_on(n_b.connect(addr)).unwrap();
|
let p_b = r.block_on(n_b.connect(addr)).unwrap();
|
||||||
let mut s1_b = r.block_on(p_b.open(4, Promises::empty())).unwrap();
|
let mut s1_b = r.block_on(p_b.open(4, Promises::empty(), 0)).unwrap();
|
||||||
s1_b.send("HelloWorld").unwrap();
|
s1_b.send("HelloWorld").unwrap();
|
||||||
drop(s1_b);
|
drop(s1_b);
|
||||||
drop(p_b);
|
drop(p_b);
|
||||||
@ -334,7 +334,7 @@ fn close_network_scheduler_completely() {
|
|||||||
let addr = tcp();
|
let addr = tcp();
|
||||||
r.block_on(n_a.listen(addr.clone())).unwrap();
|
r.block_on(n_a.listen(addr.clone())).unwrap();
|
||||||
let p_b = r.block_on(n_b.connect(addr)).unwrap();
|
let p_b = r.block_on(n_b.connect(addr)).unwrap();
|
||||||
let mut s1_b = r.block_on(p_b.open(4, Promises::empty())).unwrap();
|
let mut s1_b = r.block_on(p_b.open(4, Promises::empty(), 0)).unwrap();
|
||||||
s1_b.send("HelloWorld").unwrap();
|
s1_b.send("HelloWorld").unwrap();
|
||||||
|
|
||||||
let p_a = r.block_on(n_a.connected()).unwrap();
|
let p_a = r.block_on(n_a.connected()).unwrap();
|
||||||
|
@ -67,7 +67,7 @@ pub fn network_participant_stream(
|
|||||||
let p1_b = n_b.connect(addr).await.unwrap();
|
let p1_b = n_b.connect(addr).await.unwrap();
|
||||||
let p1_a = n_a.connected().await.unwrap();
|
let p1_a = n_a.connected().await.unwrap();
|
||||||
|
|
||||||
let s1_a = p1_a.open(4, Promises::empty()).await.unwrap();
|
let s1_a = p1_a.open(4, Promises::empty(), 0).await.unwrap();
|
||||||
let s1_b = p1_b.opened().await.unwrap();
|
let s1_b = p1_b.opened().await.unwrap();
|
||||||
|
|
||||||
(n_a, p1_a, s1_a, n_b, p1_b, s1_b)
|
(n_a, p1_a, s1_a, n_b, p1_b, s1_b)
|
||||||
|
@ -177,7 +177,7 @@ fn api_stream_send_main() -> std::result::Result<(), Box<dyn std::error::Error>>
|
|||||||
.await?;
|
.await?;
|
||||||
// keep it alive
|
// keep it alive
|
||||||
let _stream_p = remote_p
|
let _stream_p = remote_p
|
||||||
.open(4, Promises::ORDERED | Promises::CONSISTENCY)
|
.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0)
|
||||||
.await?;
|
.await?;
|
||||||
let participant_a = network.connected().await?;
|
let participant_a = network.connected().await?;
|
||||||
let mut stream_a = participant_a.opened().await?;
|
let mut stream_a = participant_a.opened().await?;
|
||||||
@ -205,7 +205,7 @@ fn api_stream_recv_main() -> std::result::Result<(), Box<dyn std::error::Error>>
|
|||||||
.connect(ProtocolAddr::Tcp("127.0.0.1:1220".parse().unwrap()))
|
.connect(ProtocolAddr::Tcp("127.0.0.1:1220".parse().unwrap()))
|
||||||
.await?;
|
.await?;
|
||||||
let mut stream_p = remote_p
|
let mut stream_p = remote_p
|
||||||
.open(4, Promises::ORDERED | Promises::CONSISTENCY)
|
.open(4, Promises::ORDERED | Promises::CONSISTENCY, 0)
|
||||||
.await?;
|
.await?;
|
||||||
stream_p.send("Hello World")?;
|
stream_p.send("Hello World")?;
|
||||||
let participant_a = network.connected().await?;
|
let participant_a = network.connected().await?;
|
||||||
|
@ -16,7 +16,6 @@ pub fn init(basic: bool) {
|
|||||||
let base_exceptions = |env: EnvFilter| {
|
let base_exceptions = |env: EnvFilter| {
|
||||||
env.add_directive("veloren_world::sim=info".parse().unwrap())
|
env.add_directive("veloren_world::sim=info".parse().unwrap())
|
||||||
.add_directive("veloren_world::civ=info".parse().unwrap())
|
.add_directive("veloren_world::civ=info".parse().unwrap())
|
||||||
.add_directive("uvth=warn".parse().unwrap())
|
|
||||||
.add_directive("hyper=info".parse().unwrap())
|
.add_directive("hyper=info".parse().unwrap())
|
||||||
.add_directive("prometheus_hyper=info".parse().unwrap())
|
.add_directive("prometheus_hyper=info".parse().unwrap())
|
||||||
.add_directive("mio::pool=info".parse().unwrap())
|
.add_directive("mio::pool=info".parse().unwrap())
|
||||||
|
@ -23,11 +23,7 @@ specs-idvs = { git = "https://gitlab.com/veloren/specs-idvs.git", rev = "9fab7b3
|
|||||||
|
|
||||||
tracing = "0.1"
|
tracing = "0.1"
|
||||||
vek = { version = "0.12.0", features = ["serde"] }
|
vek = { version = "0.12.0", features = ["serde"] }
|
||||||
uvth = "3.1.1"
|
|
||||||
futures-util = "0.3.7"
|
futures-util = "0.3.7"
|
||||||
futures-executor = "0.3"
|
|
||||||
futures-timer = "3.0"
|
|
||||||
futures-channel = "0.3"
|
|
||||||
tokio = { version = "1", default-features = false, features = ["rt"] }
|
tokio = { version = "1", default-features = false, features = ["rt"] }
|
||||||
prometheus-hyper = "0.1.1"
|
prometheus-hyper = "0.1.1"
|
||||||
itertools = "0.9"
|
itertools = "0.9"
|
||||||
|
@ -8,6 +8,7 @@ use std::sync::{
|
|||||||
atomic::{AtomicBool, Ordering},
|
atomic::{AtomicBool, Ordering},
|
||||||
Arc,
|
Arc,
|
||||||
};
|
};
|
||||||
|
use tokio::runtime::Runtime;
|
||||||
use vek::*;
|
use vek::*;
|
||||||
#[cfg(feature = "worldgen")]
|
#[cfg(feature = "worldgen")]
|
||||||
use world::{IndexOwned, World};
|
use world::{IndexOwned, World};
|
||||||
@ -39,7 +40,7 @@ impl ChunkGenerator {
|
|||||||
&mut self,
|
&mut self,
|
||||||
entity: Option<EcsEntity>,
|
entity: Option<EcsEntity>,
|
||||||
key: Vec2<i32>,
|
key: Vec2<i32>,
|
||||||
thread_pool: &mut uvth::ThreadPool,
|
runtime: &mut Arc<Runtime>,
|
||||||
world: Arc<World>,
|
world: Arc<World>,
|
||||||
index: IndexOwned,
|
index: IndexOwned,
|
||||||
) {
|
) {
|
||||||
@ -52,7 +53,7 @@ impl ChunkGenerator {
|
|||||||
v.insert(Arc::clone(&cancel));
|
v.insert(Arc::clone(&cancel));
|
||||||
let chunk_tx = self.chunk_tx.clone();
|
let chunk_tx = self.chunk_tx.clone();
|
||||||
self.metrics.chunks_requested.inc();
|
self.metrics.chunks_requested.inc();
|
||||||
thread_pool.execute(move || {
|
runtime.spawn_blocking(move || {
|
||||||
let index = index.as_index_ref();
|
let index = index.as_index_ref();
|
||||||
let payload = world
|
let payload = world
|
||||||
.generate_chunk(index, key, || cancel.load(Ordering::Relaxed))
|
.generate_chunk(index, key, || cancel.load(Ordering::Relaxed))
|
||||||
|
@ -1,11 +1,9 @@
|
|||||||
use crate::{Client, ClientType, ServerInfo};
|
use crate::{Client, ClientType, ServerInfo};
|
||||||
use crossbeam_channel::{bounded, unbounded, Receiver, Sender};
|
use crossbeam_channel::{bounded, unbounded, Receiver, Sender};
|
||||||
use futures_channel::oneshot;
|
use futures_util::future::FutureExt;
|
||||||
use futures_executor::block_on;
|
|
||||||
use futures_timer::Delay;
|
|
||||||
use futures_util::{select, FutureExt};
|
|
||||||
use network::{Network, Participant, Promises};
|
use network::{Network, Participant, Promises};
|
||||||
use std::{sync::Arc, thread, time::Duration};
|
use std::{sync::Arc, time::Duration};
|
||||||
|
use tokio::{runtime::Runtime, select, sync::oneshot};
|
||||||
use tracing::{debug, error, trace, warn};
|
use tracing::{debug, error, trace, warn};
|
||||||
|
|
||||||
pub(crate) struct ServerInfoPacket {
|
pub(crate) struct ServerInfoPacket {
|
||||||
@ -17,7 +15,7 @@ pub(crate) type IncomingClient = Client;
|
|||||||
|
|
||||||
pub(crate) struct ConnectionHandler {
|
pub(crate) struct ConnectionHandler {
|
||||||
_network: Arc<Network>,
|
_network: Arc<Network>,
|
||||||
thread_handle: Option<thread::JoinHandle<()>>,
|
thread_handle: Option<tokio::task::JoinHandle<()>>,
|
||||||
pub client_receiver: Receiver<IncomingClient>,
|
pub client_receiver: Receiver<IncomingClient>,
|
||||||
pub info_requester_receiver: Receiver<Sender<ServerInfoPacket>>,
|
pub info_requester_receiver: Receiver<Sender<ServerInfoPacket>>,
|
||||||
stop_sender: Option<oneshot::Sender<()>>,
|
stop_sender: Option<oneshot::Sender<()>>,
|
||||||
@ -28,7 +26,7 @@ pub(crate) struct ConnectionHandler {
|
|||||||
/// to the Server main thread sometimes though to get the current server_info
|
/// to the Server main thread sometimes though to get the current server_info
|
||||||
/// and time
|
/// and time
|
||||||
impl ConnectionHandler {
|
impl ConnectionHandler {
|
||||||
pub fn new(network: Network) -> Self {
|
pub fn new(network: Network, runtime: Arc<Runtime>) -> Self {
|
||||||
let network = Arc::new(network);
|
let network = Arc::new(network);
|
||||||
let network_clone = Arc::clone(&network);
|
let network_clone = Arc::clone(&network);
|
||||||
let (stop_sender, stop_receiver) = oneshot::channel();
|
let (stop_sender, stop_receiver) = oneshot::channel();
|
||||||
@ -37,14 +35,12 @@ impl ConnectionHandler {
|
|||||||
let (info_requester_sender, info_requester_receiver) =
|
let (info_requester_sender, info_requester_receiver) =
|
||||||
bounded::<Sender<ServerInfoPacket>>(1);
|
bounded::<Sender<ServerInfoPacket>>(1);
|
||||||
|
|
||||||
let thread_handle = Some(thread::spawn(|| {
|
let thread_handle = Some(runtime.spawn(Self::work(
|
||||||
block_on(Self::work(
|
|
||||||
network_clone,
|
network_clone,
|
||||||
client_sender,
|
client_sender,
|
||||||
info_requester_sender,
|
info_requester_sender,
|
||||||
stop_receiver,
|
stop_receiver,
|
||||||
));
|
)));
|
||||||
}));
|
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
_network: network,
|
_network: network,
|
||||||
@ -64,7 +60,7 @@ impl ConnectionHandler {
|
|||||||
let mut stop_receiver = stop_receiver.fuse();
|
let mut stop_receiver = stop_receiver.fuse();
|
||||||
loop {
|
loop {
|
||||||
let participant = match select!(
|
let participant = match select!(
|
||||||
_ = stop_receiver => None,
|
_ = &mut stop_receiver => None,
|
||||||
p = network.connected().fuse() => Some(p),
|
p = network.connected().fuse() => Some(p),
|
||||||
) {
|
) {
|
||||||
None => break,
|
None => break,
|
||||||
@ -82,7 +78,7 @@ impl ConnectionHandler {
|
|||||||
let info_requester_sender = info_requester_sender.clone();
|
let info_requester_sender = info_requester_sender.clone();
|
||||||
|
|
||||||
match select!(
|
match select!(
|
||||||
_ = stop_receiver => None,
|
_ = &mut stop_receiver => None,
|
||||||
e = Self::init_participant(participant, client_sender, info_requester_sender).fuse() => Some(e),
|
e = Self::init_participant(participant, client_sender, info_requester_sender).fuse() => Some(e),
|
||||||
) {
|
) {
|
||||||
None => break,
|
None => break,
|
||||||
@ -104,11 +100,11 @@ impl ConnectionHandler {
|
|||||||
let reliable = Promises::ORDERED | Promises::CONSISTENCY;
|
let reliable = Promises::ORDERED | Promises::CONSISTENCY;
|
||||||
let reliablec = reliable | Promises::COMPRESSED;
|
let reliablec = reliable | Promises::COMPRESSED;
|
||||||
|
|
||||||
let general_stream = participant.open(3, reliablec).await?;
|
let general_stream = participant.open(3, reliablec, 500).await?;
|
||||||
let ping_stream = participant.open(2, reliable).await?;
|
let ping_stream = participant.open(2, reliable, 500).await?;
|
||||||
let mut register_stream = participant.open(3, reliablec).await?;
|
let mut register_stream = participant.open(3, reliablec, 0).await?;
|
||||||
let character_screen_stream = participant.open(3, reliablec).await?;
|
let character_screen_stream = participant.open(3, reliablec, 0).await?;
|
||||||
let in_game_stream = participant.open(3, reliablec).await?;
|
let in_game_stream = participant.open(3, reliablec, 400_000).await?;
|
||||||
|
|
||||||
let server_data = receiver.recv()?;
|
let server_data = receiver.recv()?;
|
||||||
|
|
||||||
@ -116,7 +112,7 @@ impl ConnectionHandler {
|
|||||||
|
|
||||||
const TIMEOUT: Duration = Duration::from_secs(5);
|
const TIMEOUT: Duration = Duration::from_secs(5);
|
||||||
let client_type = match select!(
|
let client_type = match select!(
|
||||||
_ = Delay::new(TIMEOUT).fuse() => None,
|
_ = tokio::time::sleep(TIMEOUT).fuse() => None,
|
||||||
t = register_stream.recv::<ClientType>().fuse() => Some(t),
|
t = register_stream.recv::<ClientType>().fuse() => Some(t),
|
||||||
) {
|
) {
|
||||||
None => {
|
None => {
|
||||||
@ -145,12 +141,8 @@ impl ConnectionHandler {
|
|||||||
impl Drop for ConnectionHandler {
|
impl Drop for ConnectionHandler {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
let _ = self.stop_sender.take().unwrap().send(());
|
let _ = self.stop_sender.take().unwrap().send(());
|
||||||
trace!("blocking till ConnectionHandler is closed");
|
trace!("aborting ConnectionHandler");
|
||||||
self.thread_handle
|
self.thread_handle.take().unwrap().abort();
|
||||||
.take()
|
trace!("aborted ConnectionHandler!");
|
||||||
.unwrap()
|
|
||||||
.join()
|
|
||||||
.expect("There was an error in ConnectionHandler, clean shutdown impossible");
|
|
||||||
trace!("gracefully closed ConnectionHandler!");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -11,9 +11,8 @@ use common::{
|
|||||||
};
|
};
|
||||||
use common_net::msg::{PlayerListUpdate, PresenceKind, ServerGeneral};
|
use common_net::msg::{PlayerListUpdate, PresenceKind, ServerGeneral};
|
||||||
use common_sys::state::State;
|
use common_sys::state::State;
|
||||||
use futures_executor::block_on;
|
|
||||||
use specs::{saveload::MarkerAllocator, Builder, Entity as EcsEntity, WorldExt};
|
use specs::{saveload::MarkerAllocator, Builder, Entity as EcsEntity, WorldExt};
|
||||||
use tracing::{debug, error, trace, warn};
|
use tracing::*;
|
||||||
|
|
||||||
pub fn handle_exit_ingame(server: &mut Server, entity: EcsEntity) {
|
pub fn handle_exit_ingame(server: &mut Server, entity: EcsEntity) {
|
||||||
span!(_guard, "handle_exit_ingame");
|
span!(_guard, "handle_exit_ingame");
|
||||||
@ -107,26 +106,26 @@ pub fn handle_client_disconnect(server: &mut Server, entity: EcsEntity) -> Event
|
|||||||
{
|
{
|
||||||
let participant = client.participant.take().unwrap();
|
let participant = client.participant.take().unwrap();
|
||||||
let pid = participant.remote_pid();
|
let pid = participant.remote_pid();
|
||||||
std::thread::spawn(move || {
|
server.runtime.spawn(
|
||||||
let span = tracing::span!(tracing::Level::DEBUG, "client_disconnect", ?pid, ?entity);
|
async {
|
||||||
let _enter = span.enter();
|
|
||||||
let now = std::time::Instant::now();
|
let now = std::time::Instant::now();
|
||||||
debug!(?pid, ?entity, "Start handle disconnect of client");
|
debug!("Start handle disconnect of client");
|
||||||
if let Err(e) = block_on(participant.disconnect()) {
|
if let Err(e) = participant.disconnect().await {
|
||||||
debug!(
|
debug!(
|
||||||
?e,
|
?e,
|
||||||
?pid,
|
|
||||||
"Error when disconnecting client, maybe the pipe already broke"
|
"Error when disconnecting client, maybe the pipe already broke"
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
trace!(?pid, "finished disconnect");
|
trace!("finished disconnect");
|
||||||
let elapsed = now.elapsed();
|
let elapsed = now.elapsed();
|
||||||
if elapsed.as_millis() > 100 {
|
if elapsed.as_millis() > 100 {
|
||||||
warn!(?elapsed, ?pid, "disconnecting took quite long");
|
warn!(?elapsed, "disconnecting took quite long");
|
||||||
} else {
|
} else {
|
||||||
debug!(?elapsed, ?pid, "disconnecting took");
|
debug!(?elapsed, "disconnecting took");
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
.instrument(tracing::debug_span!("client_disconnect", ?pid, ?entity)),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
let state = server.state_mut();
|
let state = server.state_mut();
|
||||||
|
@ -75,7 +75,6 @@ use common_net::{
|
|||||||
#[cfg(feature = "plugins")]
|
#[cfg(feature = "plugins")]
|
||||||
use common_sys::plugin::PluginMgr;
|
use common_sys::plugin::PluginMgr;
|
||||||
use common_sys::state::State;
|
use common_sys::state::State;
|
||||||
use futures_executor::block_on;
|
|
||||||
use metrics::{PhysicsMetrics, StateTickMetrics, TickMetrics};
|
use metrics::{PhysicsMetrics, StateTickMetrics, TickMetrics};
|
||||||
use network::{Network, Pid, ProtocolAddr};
|
use network::{Network, Pid, ProtocolAddr};
|
||||||
use persistence::{
|
use persistence::{
|
||||||
@ -95,7 +94,6 @@ use std::{
|
|||||||
use test_world::{IndexOwned, World};
|
use test_world::{IndexOwned, World};
|
||||||
use tokio::{runtime::Runtime, sync::Notify};
|
use tokio::{runtime::Runtime, sync::Notify};
|
||||||
use tracing::{debug, error, info, trace};
|
use tracing::{debug, error, info, trace};
|
||||||
use uvth::{ThreadPool, ThreadPoolBuilder};
|
|
||||||
use vek::*;
|
use vek::*;
|
||||||
|
|
||||||
#[cfg(feature = "worldgen")]
|
#[cfg(feature = "worldgen")]
|
||||||
@ -123,8 +121,7 @@ pub struct Server {
|
|||||||
|
|
||||||
connection_handler: ConnectionHandler,
|
connection_handler: ConnectionHandler,
|
||||||
|
|
||||||
_runtime: Arc<Runtime>,
|
runtime: Arc<Runtime>,
|
||||||
thread_pool: ThreadPool,
|
|
||||||
|
|
||||||
metrics_shutdown: Arc<Notify>,
|
metrics_shutdown: Arc<Notify>,
|
||||||
tick_metrics: TickMetrics,
|
tick_metrics: TickMetrics,
|
||||||
@ -366,9 +363,6 @@ impl Server {
|
|||||||
registry_state(®istry).expect("failed to register state metrics");
|
registry_state(®istry).expect("failed to register state metrics");
|
||||||
registry_physics(®istry).expect("failed to register state metrics");
|
registry_physics(®istry).expect("failed to register state metrics");
|
||||||
|
|
||||||
let thread_pool = ThreadPoolBuilder::new()
|
|
||||||
.name("veloren-worker".to_string())
|
|
||||||
.build();
|
|
||||||
let network = Network::new_with_registry(Pid::new(), Arc::clone(&runtime), ®istry);
|
let network = Network::new_with_registry(Pid::new(), Arc::clone(&runtime), ®istry);
|
||||||
let metrics_shutdown = Arc::new(Notify::new());
|
let metrics_shutdown = Arc::new(Notify::new());
|
||||||
let metrics_shutdown_clone = Arc::clone(&metrics_shutdown);
|
let metrics_shutdown_clone = Arc::clone(&metrics_shutdown);
|
||||||
@ -381,8 +375,8 @@ impl Server {
|
|||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
});
|
});
|
||||||
block_on(network.listen(ProtocolAddr::Tcp(settings.gameserver_address)))?;
|
runtime.block_on(network.listen(ProtocolAddr::Tcp(settings.gameserver_address)))?;
|
||||||
let connection_handler = ConnectionHandler::new(network);
|
let connection_handler = ConnectionHandler::new(network, Arc::clone(&runtime));
|
||||||
|
|
||||||
// Initiate real-time world simulation
|
// Initiate real-time world simulation
|
||||||
#[cfg(feature = "worldgen")]
|
#[cfg(feature = "worldgen")]
|
||||||
@ -397,9 +391,7 @@ impl Server {
|
|||||||
map,
|
map,
|
||||||
|
|
||||||
connection_handler,
|
connection_handler,
|
||||||
|
runtime,
|
||||||
_runtime: runtime,
|
|
||||||
thread_pool,
|
|
||||||
|
|
||||||
metrics_shutdown,
|
metrics_shutdown,
|
||||||
tick_metrics,
|
tick_metrics,
|
||||||
@ -431,11 +423,6 @@ impl Server {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_thread_pool(mut self, thread_pool: ThreadPool) -> Self {
|
|
||||||
self.thread_pool = thread_pool;
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get a reference to the server's settings
|
/// Get a reference to the server's settings
|
||||||
pub fn settings(&self) -> impl Deref<Target = Settings> + '_ {
|
pub fn settings(&self) -> impl Deref<Target = Settings> + '_ {
|
||||||
self.state.ecs().fetch::<Settings>()
|
self.state.ecs().fetch::<Settings>()
|
||||||
@ -676,7 +663,7 @@ impl Server {
|
|||||||
// only work we do here on the fast path is perform a relaxed read on an atomic.
|
// only work we do here on the fast path is perform a relaxed read on an atomic.
|
||||||
// boolean.
|
// boolean.
|
||||||
let index = &mut self.index;
|
let index = &mut self.index;
|
||||||
let thread_pool = &mut self.thread_pool;
|
let runtime = &mut self.runtime;
|
||||||
let world = &mut self.world;
|
let world = &mut self.world;
|
||||||
let ecs = self.state.ecs_mut();
|
let ecs = self.state.ecs_mut();
|
||||||
|
|
||||||
@ -697,7 +684,7 @@ impl Server {
|
|||||||
chunk_generator.generate_chunk(
|
chunk_generator.generate_chunk(
|
||||||
None,
|
None,
|
||||||
pos,
|
pos,
|
||||||
thread_pool,
|
runtime,
|
||||||
Arc::clone(&world),
|
Arc::clone(&world),
|
||||||
index.clone(),
|
index.clone(),
|
||||||
);
|
);
|
||||||
@ -1021,7 +1008,7 @@ impl Server {
|
|||||||
.generate_chunk(
|
.generate_chunk(
|
||||||
Some(entity),
|
Some(entity),
|
||||||
key,
|
key,
|
||||||
&mut self.thread_pool,
|
&mut self.runtime,
|
||||||
Arc::clone(&self.world),
|
Arc::clone(&self.world),
|
||||||
self.index.clone(),
|
self.index.clone(),
|
||||||
);
|
);
|
||||||
|
@ -73,12 +73,13 @@ impl CharacterLoader {
|
|||||||
|
|
||||||
let mut conn = establish_connection(db_dir)?;
|
let mut conn = establish_connection(db_dir)?;
|
||||||
|
|
||||||
std::thread::spawn(move || {
|
let builder = std::thread::Builder::new().name("persistence_loader".into());
|
||||||
|
builder
|
||||||
|
.spawn(move || {
|
||||||
for request in internal_rx {
|
for request in internal_rx {
|
||||||
let (entity, kind) = request;
|
let (entity, kind) = request;
|
||||||
|
|
||||||
if let Err(e) =
|
if let Err(e) = internal_tx.send(CharacterLoaderResponse {
|
||||||
internal_tx.send(CharacterLoaderResponse {
|
|
||||||
entity,
|
entity,
|
||||||
result: match kind {
|
result: match kind {
|
||||||
CharacterLoaderRequestKind::CreateCharacter {
|
CharacterLoaderRequestKind::CreateCharacter {
|
||||||
@ -123,12 +124,12 @@ impl CharacterLoader {
|
|||||||
CharacterLoaderResponseKind::CharacterData(Box::new(result))
|
CharacterLoaderResponseKind::CharacterData(Box::new(result))
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
})
|
}) {
|
||||||
{
|
|
||||||
error!(?e, "Could not send send persistence request");
|
error!(?e, "Could not send send persistence request");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
update_tx,
|
update_tx,
|
||||||
|
@ -24,13 +24,16 @@ impl CharacterUpdater {
|
|||||||
|
|
||||||
let mut conn = establish_connection(db_dir)?;
|
let mut conn = establish_connection(db_dir)?;
|
||||||
|
|
||||||
let handle = std::thread::spawn(move || {
|
let builder = std::thread::Builder::new().name("persistence_updater".into());
|
||||||
|
let handle = builder
|
||||||
|
.spawn(move || {
|
||||||
while let Ok(updates) = update_rx.recv() {
|
while let Ok(updates) = update_rx.recv() {
|
||||||
trace!("Persistence batch update starting");
|
trace!("Persistence batch update starting");
|
||||||
execute_batch_update(updates, &mut conn);
|
execute_batch_update(updates, &mut conn);
|
||||||
trace!("Persistence batch update finished");
|
trace!("Persistence batch update finished");
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
update_tx: Some(update_tx),
|
update_tx: Some(update_tx),
|
||||||
|
@ -81,7 +81,6 @@ rodio = {version = "0.13", default-features = false, features = ["wav", "vorbis"
|
|||||||
ron = {version = "0.6", default-features = false}
|
ron = {version = "0.6", default-features = false}
|
||||||
serde = {version = "1.0", features = [ "rc", "derive" ]}
|
serde = {version = "1.0", features = [ "rc", "derive" ]}
|
||||||
treeculler = "0.1.0"
|
treeculler = "0.1.0"
|
||||||
uvth = "3.1.1"
|
|
||||||
tokio = { version = "1", default-features = false, features = ["rt-multi-thread"] }
|
tokio = { version = "1", default-features = false, features = ["rt-multi-thread"] }
|
||||||
num_cpus = "1.0"
|
num_cpus = "1.0"
|
||||||
# vec_map = { version = "0.8.2" }
|
# vec_map = { version = "0.8.2" }
|
||||||
|
@ -22,14 +22,14 @@ const RUST_LOG_ENV: &str = "RUST_LOG";
|
|||||||
/// `RUST_LOG="veloren_voxygen=trace"`
|
/// `RUST_LOG="veloren_voxygen=trace"`
|
||||||
///
|
///
|
||||||
/// more complex tracing can be done by concatenating with a `,` as seperator:
|
/// more complex tracing can be done by concatenating with a `,` as seperator:
|
||||||
/// - warn for `uvth`, `tiny_http`, `dot_vox`, `gfx_device_gl::factory,
|
/// - warn for `prometheus_hyper`, `dot_vox`, `gfx_device_gl::factory,
|
||||||
/// `gfx_device_gl::shade` trace for `veloren_voxygen`, info for everything
|
/// `gfx_device_gl::shade` trace for `veloren_voxygen`, info for everything
|
||||||
/// else
|
/// else
|
||||||
/// `RUST_LOG="uvth=warn,tiny_http=warn,dot_vox::parser=warn,gfx_device_gl::
|
/// `RUST_LOG="prometheus_hyper=warn,dot_vox::parser=warn,gfx_device_gl::
|
||||||
/// factory=warn,gfx_device_gl::shade=warn,veloren_voxygen=trace,info"`
|
/// factory=warn,gfx_device_gl::shade=warn,veloren_voxygen=trace,info"`
|
||||||
///
|
///
|
||||||
/// By default a few directives are set to `warn` by default, until explicitly
|
/// By default a few directives are set to `warn` by default, until explicitly
|
||||||
/// overwritten! e.g. `RUST_LOG="uvth=debug"`
|
/// overwritten! e.g. `RUST_LOG="gfx_device_gl=debug"`
|
||||||
pub fn init(settings: &Settings) -> Vec<impl Drop> {
|
pub fn init(settings: &Settings) -> Vec<impl Drop> {
|
||||||
// To hold the guards that we create, they will cause the logs to be
|
// To hold the guards that we create, they will cause the logs to be
|
||||||
// flushed when they're dropped.
|
// flushed when they're dropped.
|
||||||
@ -42,8 +42,7 @@ pub fn init(settings: &Settings) -> Vec<impl Drop> {
|
|||||||
let base_exceptions = |env: EnvFilter| {
|
let base_exceptions = |env: EnvFilter| {
|
||||||
env.add_directive("dot_vox::parser=warn".parse().unwrap())
|
env.add_directive("dot_vox::parser=warn".parse().unwrap())
|
||||||
.add_directive("gfx_device_gl=warn".parse().unwrap())
|
.add_directive("gfx_device_gl=warn".parse().unwrap())
|
||||||
.add_directive("uvth=warn".parse().unwrap())
|
.add_directive("prometheus_hyper=warn".parse().unwrap())
|
||||||
.add_directive("tiny_http=warn".parse().unwrap())
|
|
||||||
.add_directive("mio::sys::windows=debug".parse().unwrap())
|
.add_directive("mio::sys::windows=debug".parse().unwrap())
|
||||||
.add_directive("veloren_network_protocol=info".parse().unwrap())
|
.add_directive("veloren_network_protocol=info".parse().unwrap())
|
||||||
.add_directive(
|
.add_directive(
|
||||||
|
@ -124,11 +124,16 @@ fn main() {
|
|||||||
|
|
||||||
// On windows we need to spawn a thread as the msg doesn't work otherwise
|
// On windows we need to spawn a thread as the msg doesn't work otherwise
|
||||||
#[cfg(target_os = "windows")]
|
#[cfg(target_os = "windows")]
|
||||||
std::thread::spawn(move || {
|
{
|
||||||
|
let builder = std::thread::Builder::new().name("shutdown".into());
|
||||||
|
builder
|
||||||
|
.spawn(move || {
|
||||||
mbox();
|
mbox();
|
||||||
})
|
})
|
||||||
|
.unwrap()
|
||||||
.join()
|
.join()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(not(target_os = "windows"))]
|
#[cfg(not(target_os = "windows"))]
|
||||||
mbox();
|
mbox();
|
||||||
|
@ -147,7 +147,7 @@ impl PlayState for CharSelectionState {
|
|||||||
time: client.state().get_time(),
|
time: client.state().get_time(),
|
||||||
delta_time: client.state().ecs().read_resource::<DeltaTime>().0,
|
delta_time: client.state().ecs().read_resource::<DeltaTime>().0,
|
||||||
tick: client.get_tick(),
|
tick: client.get_tick(),
|
||||||
thread_pool: client.thread_pool(),
|
runtime: client.runtime(),
|
||||||
body: humanoid_body,
|
body: humanoid_body,
|
||||||
gamma: global_state.settings.graphics.gamma,
|
gamma: global_state.settings.graphics.gamma,
|
||||||
exposure: global_state.settings.graphics.exposure,
|
exposure: global_state.settings.graphics.exposure,
|
||||||
|
@ -4,14 +4,14 @@ use client::{
|
|||||||
};
|
};
|
||||||
use crossbeam::channel::{unbounded, Receiver, Sender, TryRecvError};
|
use crossbeam::channel::{unbounded, Receiver, Sender, TryRecvError};
|
||||||
use std::{
|
use std::{
|
||||||
net::ToSocketAddrs,
|
net::SocketAddr,
|
||||||
sync::{
|
sync::{
|
||||||
atomic::{AtomicBool, Ordering},
|
atomic::{AtomicBool, Ordering},
|
||||||
Arc,
|
Arc,
|
||||||
},
|
},
|
||||||
thread,
|
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
use tokio::{net::lookup_host, runtime};
|
||||||
use tracing::{trace, warn};
|
use tracing::{trace, warn};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
@ -40,6 +40,7 @@ pub struct ClientInit {
|
|||||||
rx: Receiver<Msg>,
|
rx: Receiver<Msg>,
|
||||||
trust_tx: Sender<AuthTrust>,
|
trust_tx: Sender<AuthTrust>,
|
||||||
cancel: Arc<AtomicBool>,
|
cancel: Arc<AtomicBool>,
|
||||||
|
_runtime: Arc<runtime::Runtime>,
|
||||||
}
|
}
|
||||||
impl ClientInit {
|
impl ClientInit {
|
||||||
#[allow(clippy::op_ref)] // TODO: Pending review in #587
|
#[allow(clippy::op_ref)] // TODO: Pending review in #587
|
||||||
@ -50,50 +51,47 @@ impl ClientInit {
|
|||||||
view_distance: Option<u32>,
|
view_distance: Option<u32>,
|
||||||
password: String,
|
password: String,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let (server_address, default_port, prefer_ipv6) = connection_args;
|
let (server_address, port, prefer_ipv6) = connection_args;
|
||||||
|
|
||||||
let (tx, rx) = unbounded();
|
let (tx, rx) = unbounded();
|
||||||
let (trust_tx, trust_rx) = unbounded();
|
let (trust_tx, trust_rx) = unbounded();
|
||||||
let cancel = Arc::new(AtomicBool::new(false));
|
let cancel = Arc::new(AtomicBool::new(false));
|
||||||
let cancel2 = Arc::clone(&cancel);
|
let cancel2 = Arc::clone(&cancel);
|
||||||
|
|
||||||
thread::spawn(move || {
|
|
||||||
// Parse ip address or resolves hostname.
|
|
||||||
// Note: if you use an ipv6 address, the number after the last colon will be
|
|
||||||
// used as the port unless you use [] around the address.
|
|
||||||
match server_address
|
|
||||||
.to_socket_addrs()
|
|
||||||
.or((server_address.as_ref(), default_port).to_socket_addrs())
|
|
||||||
{
|
|
||||||
Ok(socket_address) => {
|
|
||||||
let (first_addrs, second_addrs) =
|
|
||||||
socket_address.partition::<Vec<_>, _>(|a| a.is_ipv6() == prefer_ipv6);
|
|
||||||
|
|
||||||
let mut last_err = None;
|
|
||||||
|
|
||||||
let cores = num_cpus::get();
|
let cores = num_cpus::get();
|
||||||
|
|
||||||
let runtime = Arc::new(
|
let runtime = Arc::new(
|
||||||
tokio::runtime::Builder::new_multi_thread()
|
runtime::Builder::new_multi_thread()
|
||||||
.enable_all()
|
.enable_all()
|
||||||
.worker_threads(if cores > 4 { cores - 1 } else { cores })
|
.worker_threads(if cores > 4 { cores - 1 } else { cores })
|
||||||
.build()
|
.build()
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
);
|
);
|
||||||
|
let runtime2 = Arc::clone(&runtime);
|
||||||
|
|
||||||
|
runtime.spawn(async move {
|
||||||
|
let addresses = match Self::resolve(server_address, port, prefer_ipv6).await {
|
||||||
|
Ok(a) => a,
|
||||||
|
Err(e) => {
|
||||||
|
let _ = tx.send(Msg::Done(Err(Error::BadAddress(e))));
|
||||||
|
return;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
let mut last_err = None;
|
||||||
|
|
||||||
const FOUR_MINUTES_RETRIES: u64 = 48;
|
const FOUR_MINUTES_RETRIES: u64 = 48;
|
||||||
'tries: for _ in 0..FOUR_MINUTES_RETRIES {
|
'tries: for _ in 0..FOUR_MINUTES_RETRIES {
|
||||||
if cancel2.load(Ordering::Relaxed) {
|
if cancel2.load(Ordering::Relaxed) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
for socket_addr in
|
for socket_addr in &addresses {
|
||||||
first_addrs.clone().into_iter().chain(second_addrs.clone())
|
match Client::new(socket_addr.clone(), view_distance, Arc::clone(&runtime2))
|
||||||
|
.await
|
||||||
{
|
{
|
||||||
match Client::new(socket_addr, view_distance, Arc::clone(&runtime)) {
|
|
||||||
Ok(mut client) => {
|
Ok(mut client) => {
|
||||||
if let Err(e) =
|
if let Err(e) = client
|
||||||
client.register(username, password, |auth_server| {
|
.register(username, password, |auth_server| {
|
||||||
let _ = tx
|
let _ = tx.send(Msg::IsAuthTrusted(auth_server.to_string()));
|
||||||
.send(Msg::IsAuthTrusted(auth_server.to_string()));
|
|
||||||
trust_rx
|
trust_rx
|
||||||
.recv()
|
.recv()
|
||||||
.map(|AuthTrust(server, trust)| {
|
.map(|AuthTrust(server, trust)| {
|
||||||
@ -101,6 +99,7 @@ impl ClientInit {
|
|||||||
})
|
})
|
||||||
.unwrap_or(false)
|
.unwrap_or(false)
|
||||||
})
|
})
|
||||||
|
.await
|
||||||
{
|
{
|
||||||
last_err = Some(Error::ClientError(e));
|
last_err = Some(Error::ClientError(e));
|
||||||
break 'tries;
|
break 'tries;
|
||||||
@ -111,9 +110,9 @@ impl ClientInit {
|
|||||||
Err(ClientError::NetworkErr(NetworkError::ConnectFailed(e))) => {
|
Err(ClientError::NetworkErr(NetworkError::ConnectFailed(e))) => {
|
||||||
if e.kind() == std::io::ErrorKind::PermissionDenied {
|
if e.kind() == std::io::ErrorKind::PermissionDenied {
|
||||||
warn!(?e, "Cannot connect to server: Incompatible version");
|
warn!(?e, "Cannot connect to server: Incompatible version");
|
||||||
last_err = Some(Error::ClientError(
|
last_err = Some(Error::ClientError(ClientError::NetworkErr(
|
||||||
ClientError::NetworkErr(NetworkError::ConnectFailed(e)),
|
NetworkError::ConnectFailed(e),
|
||||||
));
|
)));
|
||||||
break 'tries;
|
break 'tries;
|
||||||
} else {
|
} else {
|
||||||
warn!(?e, "Failed to connect to the server. Retrying...");
|
warn!(?e, "Failed to connect to the server. Retrying...");
|
||||||
@ -126,25 +125,55 @@ impl ClientInit {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
thread::sleep(Duration::from_secs(5));
|
tokio::time::sleep(Duration::from_secs(5)).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parsing/host name resolution successful but no connection succeeded.
|
// Parsing/host name resolution successful but no connection succeeded.
|
||||||
let _ = tx.send(Msg::Done(Err(last_err.unwrap_or(Error::NoAddress))));
|
let _ = tx.send(Msg::Done(Err(last_err.unwrap_or(Error::NoAddress))));
|
||||||
},
|
|
||||||
Err(err) => {
|
|
||||||
// Error parsing input string or error resolving host name.
|
|
||||||
let _ = tx.send(Msg::Done(Err(Error::BadAddress(err))));
|
|
||||||
},
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
ClientInit {
|
ClientInit {
|
||||||
rx,
|
rx,
|
||||||
trust_tx,
|
trust_tx,
|
||||||
cancel,
|
cancel,
|
||||||
|
_runtime: runtime,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn resolve(
|
||||||
|
server_address: String,
|
||||||
|
port: u16,
|
||||||
|
prefer_ipv6: bool,
|
||||||
|
) -> Result<Vec<SocketAddr>, std::io::Error> {
|
||||||
|
//1. try if server_address already contains a port
|
||||||
|
if let Ok(addr) = server_address.parse::<SocketAddr>() {
|
||||||
|
warn!("please don't add port directly to server_address");
|
||||||
|
return Ok(vec![addr]);
|
||||||
|
}
|
||||||
|
|
||||||
|
//2, try server_address and port
|
||||||
|
if let Ok(addr) = format!("{}:{}", server_address, port).parse::<SocketAddr>() {
|
||||||
|
return Ok(vec![addr]);
|
||||||
|
}
|
||||||
|
|
||||||
|
//3. do DNS call
|
||||||
|
let (mut first_addrs, mut second_addrs) = match lookup_host(server_address).await {
|
||||||
|
Ok(s) => s.partition::<Vec<_>, _>(|a| a.is_ipv6() == prefer_ipv6),
|
||||||
|
Err(e) => {
|
||||||
|
return Err(e);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(
|
||||||
|
std::iter::Iterator::chain(first_addrs.drain(..), second_addrs.drain(..))
|
||||||
|
.map(|mut addr| {
|
||||||
|
addr.set_port(port);
|
||||||
|
addr
|
||||||
|
})
|
||||||
|
.collect(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
/// Poll if the thread is complete.
|
/// Poll if the thread is complete.
|
||||||
/// Returns None if the thread is still running, otherwise returns the
|
/// Returns None if the thread is still running, otherwise returns the
|
||||||
/// Result of client creation.
|
/// Result of client creation.
|
||||||
|
@ -27,6 +27,7 @@ use core::{hash::Hash, ops::Range};
|
|||||||
use crossbeam::atomic;
|
use crossbeam::atomic;
|
||||||
use hashbrown::{hash_map::Entry, HashMap};
|
use hashbrown::{hash_map::Entry, HashMap};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
use tokio::runtime::Runtime;
|
||||||
use vek::*;
|
use vek::*;
|
||||||
|
|
||||||
/// A type produced by mesh worker threads corresponding to the information
|
/// A type produced by mesh worker threads corresponding to the information
|
||||||
@ -311,7 +312,7 @@ where
|
|||||||
tick: u64,
|
tick: u64,
|
||||||
camera_mode: CameraMode,
|
camera_mode: CameraMode,
|
||||||
character_state: Option<&CharacterState>,
|
character_state: Option<&CharacterState>,
|
||||||
thread_pool: &uvth::ThreadPool,
|
runtime: &Arc<Runtime>,
|
||||||
) -> (FigureModelEntryLod<'c>, &'c Skel::Attr)
|
) -> (FigureModelEntryLod<'c>, &'c Skel::Attr)
|
||||||
where
|
where
|
||||||
for<'a> &'a Skel::Body: Into<Skel::Attr>,
|
for<'a> &'a Skel::Body: Into<Skel::Attr>,
|
||||||
@ -377,7 +378,7 @@ where
|
|||||||
let manifests = self.manifests;
|
let manifests = self.manifests;
|
||||||
let slot_ = Arc::clone(&slot);
|
let slot_ = Arc::clone(&slot);
|
||||||
|
|
||||||
thread_pool.execute(move || {
|
runtime.spawn_blocking(move || {
|
||||||
// First, load all the base vertex data.
|
// First, load all the base vertex data.
|
||||||
let manifests = &*manifests.read();
|
let manifests = &*manifests.read();
|
||||||
let meshes = <Skel::Body as BodySpec>::bone_meshes(&key, manifests);
|
let meshes = <Skel::Body as BodySpec>::bone_meshes(&key, manifests);
|
||||||
|
@ -719,7 +719,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -1445,7 +1445,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -1653,7 +1653,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -1976,7 +1976,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -2329,7 +2329,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -2435,7 +2435,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -2520,7 +2520,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state =
|
let state =
|
||||||
@ -2610,7 +2610,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -2762,7 +2762,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -2849,7 +2849,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -2934,7 +2934,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state = self
|
let state = self
|
||||||
@ -3344,7 +3344,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state =
|
let state =
|
||||||
@ -3480,7 +3480,7 @@ impl FigureMgr {
|
|||||||
tick,
|
tick,
|
||||||
player_camera_mode,
|
player_camera_mode,
|
||||||
player_character_state,
|
player_character_state,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
let state =
|
let state =
|
||||||
|
@ -36,6 +36,8 @@ use common_sys::state::State;
|
|||||||
use comp::item::Reagent;
|
use comp::item::Reagent;
|
||||||
use num::traits::{Float, FloatConst};
|
use num::traits::{Float, FloatConst};
|
||||||
use specs::{Entity as EcsEntity, Join, WorldExt};
|
use specs::{Entity as EcsEntity, Join, WorldExt};
|
||||||
|
use std::sync::Arc;
|
||||||
|
use tokio::runtime::Runtime;
|
||||||
use vek::*;
|
use vek::*;
|
||||||
|
|
||||||
// TODO: Don't hard-code this.
|
// TODO: Don't hard-code this.
|
||||||
@ -114,7 +116,7 @@ pub struct SceneData<'a> {
|
|||||||
pub loaded_distance: f32,
|
pub loaded_distance: f32,
|
||||||
pub view_distance: u32,
|
pub view_distance: u32,
|
||||||
pub tick: u64,
|
pub tick: u64,
|
||||||
pub thread_pool: &'a uvth::ThreadPool,
|
pub runtime: &'a Arc<Runtime>,
|
||||||
pub gamma: f32,
|
pub gamma: f32,
|
||||||
pub exposure: f32,
|
pub exposure: f32,
|
||||||
pub ambiance: f32,
|
pub ambiance: f32,
|
||||||
|
@ -29,6 +29,8 @@ use common::{
|
|||||||
terrain::BlockKind,
|
terrain::BlockKind,
|
||||||
vol::{BaseVol, ReadVol},
|
vol::{BaseVol, ReadVol},
|
||||||
};
|
};
|
||||||
|
use std::sync::Arc;
|
||||||
|
use tokio::runtime::Runtime;
|
||||||
use tracing::error;
|
use tracing::error;
|
||||||
use vek::*;
|
use vek::*;
|
||||||
use winit::event::MouseButton;
|
use winit::event::MouseButton;
|
||||||
@ -96,7 +98,7 @@ pub struct SceneData<'a> {
|
|||||||
pub time: f64,
|
pub time: f64,
|
||||||
pub delta_time: f32,
|
pub delta_time: f32,
|
||||||
pub tick: u64,
|
pub tick: u64,
|
||||||
pub thread_pool: &'a uvth::ThreadPool,
|
pub runtime: &'a Arc<Runtime>,
|
||||||
pub body: Option<humanoid::Body>,
|
pub body: Option<humanoid::Body>,
|
||||||
pub gamma: f32,
|
pub gamma: f32,
|
||||||
pub exposure: f32,
|
pub exposure: f32,
|
||||||
@ -350,7 +352,7 @@ impl Scene {
|
|||||||
scene_data.tick,
|
scene_data.tick,
|
||||||
CameraMode::default(),
|
CameraMode::default(),
|
||||||
None,
|
None,
|
||||||
scene_data.thread_pool,
|
scene_data.runtime,
|
||||||
)
|
)
|
||||||
.0;
|
.0;
|
||||||
let mut buf = [Default::default(); anim::MAX_BONE_COUNT];
|
let mut buf = [Default::default(); anim::MAX_BONE_COUNT];
|
||||||
|
@ -27,7 +27,10 @@ use enum_iterator::IntoEnumIterator;
|
|||||||
use guillotiere::AtlasAllocator;
|
use guillotiere::AtlasAllocator;
|
||||||
use hashbrown::HashMap;
|
use hashbrown::HashMap;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use std::sync::Arc;
|
use std::sync::{
|
||||||
|
atomic::{AtomicU64, Ordering},
|
||||||
|
Arc,
|
||||||
|
};
|
||||||
use tracing::warn;
|
use tracing::warn;
|
||||||
use treeculler::{BVol, Frustum, AABB};
|
use treeculler::{BVol, Frustum, AABB};
|
||||||
use vek::*;
|
use vek::*;
|
||||||
@ -259,6 +262,7 @@ pub struct Terrain<V: RectRasterableVol = TerrainChunk> {
|
|||||||
mesh_send_tmp: channel::Sender<MeshWorkerResponse>,
|
mesh_send_tmp: channel::Sender<MeshWorkerResponse>,
|
||||||
mesh_recv: channel::Receiver<MeshWorkerResponse>,
|
mesh_recv: channel::Receiver<MeshWorkerResponse>,
|
||||||
mesh_todo: HashMap<Vec2<i32>, ChunkMeshState>,
|
mesh_todo: HashMap<Vec2<i32>, ChunkMeshState>,
|
||||||
|
mesh_todos_active: Arc<AtomicU64>,
|
||||||
|
|
||||||
// GPU data
|
// GPU data
|
||||||
sprite_data: Arc<HashMap<(SpriteKind, usize), Vec<SpriteData>>>,
|
sprite_data: Arc<HashMap<(SpriteKind, usize), Vec<SpriteData>>>,
|
||||||
@ -406,6 +410,7 @@ impl<V: RectRasterableVol> Terrain<V> {
|
|||||||
mesh_send_tmp: send,
|
mesh_send_tmp: send,
|
||||||
mesh_recv: recv,
|
mesh_recv: recv,
|
||||||
mesh_todo: HashMap::default(),
|
mesh_todo: HashMap::default(),
|
||||||
|
mesh_todos_active: Arc::new(AtomicU64::new(0)),
|
||||||
sprite_data: Arc::new(sprite_data),
|
sprite_data: Arc::new(sprite_data),
|
||||||
sprite_col_lights,
|
sprite_col_lights,
|
||||||
waves: renderer
|
waves: renderer
|
||||||
@ -633,6 +638,7 @@ impl<V: RectRasterableVol> Terrain<V> {
|
|||||||
|
|
||||||
// Limit ourselves to u16::MAX even if larger textures are supported.
|
// Limit ourselves to u16::MAX even if larger textures are supported.
|
||||||
let max_texture_size = renderer.max_texture_size();
|
let max_texture_size = renderer.max_texture_size();
|
||||||
|
let cores = num_cpus::get();
|
||||||
|
|
||||||
span!(guard, "Queue meshing from todo list");
|
span!(guard, "Queue meshing from todo list");
|
||||||
for (todo, chunk) in self
|
for (todo, chunk) in self
|
||||||
@ -649,8 +655,7 @@ impl<V: RectRasterableVol> Terrain<V> {
|
|||||||
.cloned()?))
|
.cloned()?))
|
||||||
})
|
})
|
||||||
{
|
{
|
||||||
// TODO: find a alternative!
|
if self.mesh_todos_active.load(Ordering::Relaxed) > (cores as u64 - 1).max(1) {
|
||||||
if scene_data.thread_pool.queued_jobs() > 0 {
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -701,7 +706,9 @@ impl<V: RectRasterableVol> Terrain<V> {
|
|||||||
let started_tick = todo.started_tick;
|
let started_tick = todo.started_tick;
|
||||||
let sprite_data = Arc::clone(&self.sprite_data);
|
let sprite_data = Arc::clone(&self.sprite_data);
|
||||||
let sprite_config = Arc::clone(&self.sprite_config);
|
let sprite_config = Arc::clone(&self.sprite_config);
|
||||||
scene_data.thread_pool.execute(move || {
|
let cnt = Arc::clone(&self.mesh_todos_active);
|
||||||
|
cnt.fetch_add(1, Ordering::Relaxed);
|
||||||
|
scene_data.runtime.spawn_blocking(move || {
|
||||||
let sprite_data = sprite_data;
|
let sprite_data = sprite_data;
|
||||||
let _ = send.send(mesh_worker(
|
let _ = send.send(mesh_worker(
|
||||||
pos,
|
pos,
|
||||||
@ -714,6 +721,7 @@ impl<V: RectRasterableVol> Terrain<V> {
|
|||||||
&sprite_data,
|
&sprite_data,
|
||||||
&sprite_config,
|
&sprite_config,
|
||||||
));
|
));
|
||||||
|
cnt.fetch_sub(1, Ordering::Relaxed);
|
||||||
});
|
});
|
||||||
todo.is_worker_active = true;
|
todo.is_worker_active = true;
|
||||||
}
|
}
|
||||||
|
@ -1307,7 +1307,7 @@ impl PlayState for SessionState {
|
|||||||
loaded_distance: client.loaded_distance(),
|
loaded_distance: client.loaded_distance(),
|
||||||
view_distance: client.view_distance().unwrap_or(1),
|
view_distance: client.view_distance().unwrap_or(1),
|
||||||
tick: client.get_tick(),
|
tick: client.get_tick(),
|
||||||
thread_pool: client.thread_pool(),
|
runtime: &client.runtime(),
|
||||||
gamma: global_state.settings.graphics.gamma,
|
gamma: global_state.settings.graphics.gamma,
|
||||||
exposure: global_state.settings.graphics.exposure,
|
exposure: global_state.settings.graphics.exposure,
|
||||||
ambiance: global_state.settings.graphics.ambiance,
|
ambiance: global_state.settings.graphics.ambiance,
|
||||||
@ -1375,7 +1375,7 @@ impl PlayState for SessionState {
|
|||||||
loaded_distance: client.loaded_distance(),
|
loaded_distance: client.loaded_distance(),
|
||||||
view_distance: client.view_distance().unwrap_or(1),
|
view_distance: client.view_distance().unwrap_or(1),
|
||||||
tick: client.get_tick(),
|
tick: client.get_tick(),
|
||||||
thread_pool: client.thread_pool(),
|
runtime: &client.runtime(),
|
||||||
gamma: settings.graphics.gamma,
|
gamma: settings.graphics.gamma,
|
||||||
exposure: settings.graphics.exposure,
|
exposure: settings.graphics.exposure,
|
||||||
ambiance: settings.graphics.ambiance,
|
ambiance: settings.graphics.ambiance,
|
||||||
|
@ -10,7 +10,7 @@ use std::{
|
|||||||
thread::{self, JoinHandle},
|
thread::{self, JoinHandle},
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
use tracing::{error, info, warn};
|
use tracing::{error, info, warn, trace, debug};
|
||||||
|
|
||||||
const TPS: u64 = 30;
|
const TPS: u64 = 30;
|
||||||
|
|
||||||
@ -81,15 +81,19 @@ impl Singleplayer {
|
|||||||
let settings = server::Settings::singleplayer(&server_data_dir);
|
let settings = server::Settings::singleplayer(&server_data_dir);
|
||||||
let editable_settings = server::EditableSettings::singleplayer(&server_data_dir);
|
let editable_settings = server::EditableSettings::singleplayer(&server_data_dir);
|
||||||
|
|
||||||
let thread_pool = client.map(|c| c.thread_pool().clone());
|
let runtime = if let Some(c) = client {
|
||||||
|
Arc::clone(&c.runtime())
|
||||||
|
} else {
|
||||||
let cores = num_cpus::get();
|
let cores = num_cpus::get();
|
||||||
let runtime = Arc::new(
|
debug!("creating a new runtime for server");
|
||||||
|
Arc::new(
|
||||||
tokio::runtime::Builder::new_multi_thread()
|
tokio::runtime::Builder::new_multi_thread()
|
||||||
.enable_all()
|
.enable_all()
|
||||||
.worker_threads(if cores > 4 { cores - 1 } else { cores })
|
.worker_threads(if cores > 4 { cores - 1 } else { cores })
|
||||||
.build()
|
.build()
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
);
|
)
|
||||||
|
};
|
||||||
let settings2 = settings.clone();
|
let settings2 = settings.clone();
|
||||||
|
|
||||||
let paused = Arc::new(AtomicBool::new(false));
|
let paused = Arc::new(AtomicBool::new(false));
|
||||||
@ -97,7 +101,10 @@ impl Singleplayer {
|
|||||||
|
|
||||||
let (result_sender, result_receiver) = bounded(1);
|
let (result_sender, result_receiver) = bounded(1);
|
||||||
|
|
||||||
let thread = thread::spawn(move || {
|
let builder = thread::Builder::new().name("singleplayer-server-thread".into());
|
||||||
|
let thread = builder
|
||||||
|
.spawn(move || {
|
||||||
|
trace!("starting singleplayer server thread");
|
||||||
let mut server = None;
|
let mut server = None;
|
||||||
if let Err(e) = result_sender.send(
|
if let Err(e) = result_sender.send(
|
||||||
match Server::new(settings2, editable_settings, &server_data_dir, runtime) {
|
match Server::new(settings2, editable_settings, &server_data_dir, runtime) {
|
||||||
@ -110,8 +117,8 @@ impl Singleplayer {
|
|||||||
) {
|
) {
|
||||||
warn!(
|
warn!(
|
||||||
?e,
|
?e,
|
||||||
"Failed to send singleplayer server initialization result. Most likely the \
|
"Failed to send singleplayer server initialization result. Most likely \
|
||||||
channel was closed by cancelling server creation. Stopping Server"
|
the channel was closed by cancelling server creation. Stopping Server"
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
@ -121,13 +128,10 @@ impl Singleplayer {
|
|||||||
None => return,
|
None => return,
|
||||||
};
|
};
|
||||||
|
|
||||||
let server = match thread_pool {
|
|
||||||
Some(pool) => server.with_thread_pool(pool),
|
|
||||||
None => server,
|
|
||||||
};
|
|
||||||
|
|
||||||
run_server(server, receiver, paused1);
|
run_server(server, receiver, paused1);
|
||||||
});
|
trace!("ending singleplayer server thread");
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
Singleplayer {
|
Singleplayer {
|
||||||
_server_thread: thread,
|
_server_thread: thread,
|
||||||
|
@ -1319,14 +1319,16 @@ impl Window {
|
|||||||
let mut path = settings.screenshots_path.clone();
|
let mut path = settings.screenshots_path.clone();
|
||||||
let sender = self.message_sender.clone();
|
let sender = self.message_sender.clone();
|
||||||
|
|
||||||
std::thread::spawn(move || {
|
let builder = std::thread::Builder::new().name("screenshot".into());
|
||||||
|
builder
|
||||||
|
.spawn(move || {
|
||||||
use std::time::SystemTime;
|
use std::time::SystemTime;
|
||||||
// Check if folder exists and create it if it does not
|
// Check if folder exists and create it if it does not
|
||||||
if !path.exists() {
|
if !path.exists() {
|
||||||
if let Err(e) = std::fs::create_dir_all(&path) {
|
if let Err(e) = std::fs::create_dir_all(&path) {
|
||||||
warn!(?e, "Couldn't create folder for screenshot");
|
warn!(?e, "Couldn't create folder for screenshot");
|
||||||
let _result =
|
let _result = sender
|
||||||
sender.send(String::from("Couldn't create folder for screenshot"));
|
.send(String::from("Couldn't create folder for screenshot"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
path.push(format!(
|
path.push(format!(
|
||||||
@ -1340,10 +1342,11 @@ impl Window {
|
|||||||
warn!(?e, "Couldn't save screenshot");
|
warn!(?e, "Couldn't save screenshot");
|
||||||
let _result = sender.send(String::from("Couldn't save screenshot"));
|
let _result = sender.send(String::from("Couldn't save screenshot"));
|
||||||
} else {
|
} else {
|
||||||
let _result =
|
let _result = sender
|
||||||
sender.send(format!("Screenshot saved to {}", path.to_string_lossy()));
|
.send(format!("Screenshot saved to {}", path.to_string_lossy()));
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
|
.unwrap();
|
||||||
},
|
},
|
||||||
Err(e) => error!(?e, "Couldn't create screenshot due to renderer error"),
|
Err(e) => error!(?e, "Couldn't create screenshot due to renderer error"),
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user