rust/hedgewars-server/src/main.rs
author unc0rr
Wed, 23 Jun 2021 23:41:51 +0200
changeset 15804 747278149393
parent 15800 6af892a0a4b8
child 15831 7d0f747afcb8
permissions -rw-r--r--
Extract network protocol into a separate crate

#![allow(unused_imports)]
#![deny(bare_trait_objects)]

use getopts::Options;
use log::*;
use mio::{net::*, *};
use std::{
    env,
    str::FromStr as _,
    time::{Duration, Instant},
};

mod core;
mod handlers;
mod protocol;
mod server;
mod utils;

use crate::server::network::{NetworkLayer, NetworkLayerBuilder};

const PROGRAM_NAME: &'_ str = "Hedgewars Game Server";

fn main() {
    env_logger::init();

    info!("Hedgewars game server, protocol {}", utils::SERVER_VERSION);

    let args: Vec<String> = env::args().collect();
    let mut opts = Options::new();

    opts.optopt("p", "port", "port - defaults to 46631", "PORT");
    opts.optflag("h", "help", "help");
    let matches = match opts.parse(&args[1..]) {
        Ok(m) => m,
        Err(e) => {
            println!("{}\n{}", e, opts.short_usage(""));
            return;
        }
    };
    if matches.opt_present("h") {
        println!("{}", opts.usage(PROGRAM_NAME));
        return;
    }

    let port = matches
        .opt_str("p")
        .and_then(|s| u16::from_str(&s).ok())
        .unwrap_or(46631);
    let address = format!("0.0.0.0:{}", port).parse().unwrap();

    let listener = TcpListener::bind(address).unwrap();

    let mut poll = Poll::new().unwrap();
    let mut hw_builder = NetworkLayerBuilder::default().with_listener(listener);

    #[cfg(feature = "tls-connections")]
    {
        let address = format!("0.0.0.0:{}", port + 1).parse().unwrap();
        hw_builder = hw_builder.with_secure_listener(TcpListener::bind(address).unwrap());
    }

    let mut hw_network = hw_builder.build(&poll);
    hw_network.register(&poll).unwrap();

    let mut events = Events::with_capacity(1024);

    let mut time = Instant::now();

    loop {
        let timeout = if hw_network.has_pending_operations() {
            Some(Duration::from_millis(1))
        } else {
            None
        };

        poll.poll(&mut events, timeout).unwrap();

        for event in events.iter() {
            if event.is_readable() {
                match event.token() {
                    token @ (utils::SERVER_TOKEN | utils::SECURE_SERVER_TOKEN) => {
                        match hw_network.accept_client(&poll, token) {
                            Ok(()) => (),
                            Err(e) => debug!("Error accepting client: {}", e),
                        }
                    }
                    #[cfg(feature = "official-server")]
                    utils::IO_TOKEN => match hw_network.handle_io_result(&poll) {
                        Ok(()) => (),
                        Err(e) => debug!("Error in IO task: {}", e),
                    },
                    Token(token) => match hw_network.client_readable(&poll, token) {
                        Ok(()) => (),
                        Err(e) => debug!("Error reading from client socket {}: {}", token, e),
                    },
                }
            }
            if event.is_writable() {
                match event.token() {
                    utils::SERVER_TOKEN | utils::SECURE_SERVER_TOKEN | utils::IO_TOKEN => {
                        unreachable!()
                    }
                    Token(token) => match hw_network.client_writable(&poll, token) {
                        Ok(()) => (),
                        Err(e) => debug!("Error writing to client socket {}: {}", token, e),
                    },
                }
            }
        }

        match hw_network.on_idle(&poll) {
            Ok(()) => (),
            Err(e) => debug!("Error in idle handler: {}", e),
        };

        if time.elapsed() > Duration::from_secs(1) {
            time = Instant::now();
            match hw_network.handle_timeout(&mut poll) {
                Ok(()) => (),
                Err(e) => debug!("Error in timer event: {}", e),
            }
        }
    }
}