aboutsummaryrefslogtreecommitdiff
path: root/src/server/mod.rs
blob: 4f95378da1e1327f88cba6ba00b13a045ae5bda8 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
//! Contains the necessary ingredients to start a graf karto world server.

use crate::net::server::ConnectionManager;
use crate::net::Cargo;
use crate::world::World;
use std::io;
use std::net::{Ipv4Addr, Ipv6Addr, SocketAddr};
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::Arc;
use std::thread::{self, JoinHandle};

/// The default port the dedicated graf karto server runs on.
pub const DEFAULT_PORT: u16 = 44309;

fn unspecified(port: u16, ipv4: bool) -> SocketAddr {
    if ipv4 {
        SocketAddr::new(Ipv4Addr::UNSPECIFIED.into(), port)
    } else {
        SocketAddr::new(Ipv6Addr::UNSPECIFIED.into(), port)
    }
}

/// Starts a thread for the server and tries to bind to the specified port. If the
/// port cannot be bound to it fails returning an error, otherwise the join handle
/// for the server thread is returned.
pub fn start_with_port(port: u16, ipv4: bool) -> Result<JoinHandle<()>, io::Error> {
    let addr = unspecified(port, ipv4);
    let conn_man = ConnectionManager::start(addr)?;

    Ok(start(conn_man).0)
}

/// Starts a thread on any free system port. Returns an error in case that's not
/// possible.
pub fn start_any_port(ipv4: bool) -> Result<(JoinHandle<()>, Arc<AtomicBool>, u16), io::Error> {
    let addr = unspecified(0, ipv4);

    let conn_man = ConnectionManager::start(addr)?;
    let port = conn_man.port();

    let (handle, running) = start(conn_man);
    Ok((handle, running, port))
}

fn start(conn_man: ConnectionManager<Cargo>) -> (JoinHandle<()>, Arc<AtomicBool>) {
    info!("Server started on port {}", conn_man.port());

    let running = Arc::new(AtomicBool::new(true));
    let running_cl = running.clone();
    let mut world = World::new();
    let handle = thread::spawn(move || {
        while running_cl.load(Ordering::Relaxed) {
            if let Some((_user, cargo)) = conn_man.next_packet() {
                match cargo {
                    Cargo::AddRoom(room) => {
                        let room_id = world.push_room(room.clone());
                        conn_man.broadcast(Cargo::SetRoom((room_id, room)));
                    }
                    Cargo::AddIcon(icon) => {
                        let icon_id = world.push_icon(icon.clone());
                        conn_man.broadcast(Cargo::SetIcon((icon_id, icon)));
                    }
                    Cargo::AddWall(wall) => {
                        let wall_id = world.push_wall(wall.clone());
                        conn_man.broadcast(Cargo::SetWall((wall_id, wall)));
                    }
                    Cargo::SetRoom((id, new)) => {
                        if let Some(old) = world.get_room_mut(id) {
                            *old = new;
                        } else {
                            error!(
                                "Unable to change room. Room with id `{}` does not exist.",
                                id
                            );
                        }
                    }
                    Cargo::SetIcon((id, new)) => {
                        if let Some(old) = world.get_icon_mut(id) {
                            *old = new;
                        } else {
                            error!(
                                "Unable to change icon. Icon with id `{}` does not exist.",
                                id
                            );
                        }
                    }
                    Cargo::SetWall((id, new)) => {
                        if let Some(old) = world.get_wall_mut(id) {
                            *old = new;
                        } else {
                            error!(
                                "Unable to change wall. Wall with id `{}` does not exist.",
                                id
                            );
                        }
                    }
                    Cargo::ClearAll => {
                        world.clear();
                        conn_man.broadcast(Cargo::ClearAll);
                    }
                    Cargo::Remove(id) => {
                        if world.remove(id) {
                            conn_man.broadcast(Cargo::Remove(id));
                        } else {
                            error!("Unable to remove item. No item with id `{}` found.", id);
                        }
                    }
                    Cargo::ApplyMatrix((_id, _matrix)) => unimplemented!(),
                    Cargo::AddMapData(_) | Cargo::UpdateMapData(_) => {
                        error!("This packet is not allowed in the client -> server direction");
                    }
                }
            }
        }
    });

    (handle, running)
}