forked from CreggHancock/HolePuncher
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Server.py
153 lines (118 loc) · 4.12 KB
/
Server.py
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
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
"""UDP hole punching server."""
from twisted.internet.protocol import DatagramProtocol
from twisted.internet import reactor
from time import sleep
import sys
def address_to_string(address):
ip, port = address
return ':'.join([ip, str(port)])
class ServerProtocol(DatagramProtocol):
def __init__(self):
self.active_sessions = {}
self.registered_clients = {}
def name_is_registered(self, name):
return name in self.registered_clients
def create_session(self, s_id, client_list):
if s_id in self.active_sessions:
print("Tried to create existing session")
return
self.active_sessions[s_id] = Session(s_id, client_list, self)
def remove_session(self, s_id):
try:
del self.active_sessions[s_id]
except KeyError:
print("Tried to terminate non-existing session")
def register_client(self, c_name, c_session, c_ip, c_port):
if self.name_is_registered(c_name):
print("Client %s is already registered." % [c_name])
return
if not c_session in self.active_sessions:
print("Client registered for non-existing session")
else:
new_client = Client(c_name, c_session, c_ip, c_port)
self.registered_clients[c_name] = new_client
self.active_sessions[c_session].client_registered(new_client)
def exchange_info(self, c_session):
if not c_session in self.active_sessions:
return
self.active_sessions[c_session].exchange_peer_info()
def client_checkout(self, name):
try:
del self.registered_clients[name]
except KeyError:
print("Tried to checkout unregistered client")
def datagramReceived(self, datagram, address):
"""Handle incoming datagram messages."""
print(datagram)
data_string = datagram.decode("utf-8")
msg_type = data_string[:2]
if msg_type == "rs":
# register session
c_ip, c_port = address
self.transport.write(bytes('ok:'+str(c_port),"utf-8"), address)
split = data_string.split(":")
session = split[1]
max_clients = split[2]
self.create_session(session, max_clients)
elif msg_type == "rc":
# register client
split = data_string.split(":")
c_name = split[1]
c_session = split[2]
c_ip, c_port = address
self.transport.write(bytes('ok:'+str(c_port),"utf-8"), address)
self.register_client(c_name, c_session, c_ip, c_port)
elif msg_type == "ep":
# exchange peers
split = data_string.split(":")
c_session = split[1]
self.exchange_info(c_session)
elif msg_type == "cc":
# checkout client
split = data_string.split(":")
c_name = split[1]
self.client_checkout(c_name)
class Session:
def __init__(self, session_id, max_clients, server):
self.id = session_id
self.client_max = max_clients
self.server = server
self.registered_clients = []
def client_registered(self, client):
if client in self.registered_clients: return
# print("Client %c registered for Session %s" % client.name, self.id)
self.registered_clients.append(client)
if len(self.registered_clients) == int(self.client_max):
sleep(5)
print("waited for OK message to send, sending out info to peers")
self.exchange_peer_info()
def exchange_peer_info(self):
for addressed_client in self.registered_clients:
address_list = []
for client in self.registered_clients:
if not client.name == addressed_client.name:
address_list.append(client.name + ":" + address_to_string((client.ip, client.port)))
address_string = ",".join(address_list)
message = bytes( "peers:" + address_string, "utf-8")
self.server.transport.write(message, (addressed_client.ip, addressed_client.port))
print("Peer info has been sent. Terminating Session")
for client in self.registered_clients:
self.server.client_checkout(client.name)
self.server.remove_session(self.id)
class Client:
def confirmation_received(self):
self.received_peer_info = True
def __init__(self, c_name, c_session, c_ip, c_port):
self.name = c_name
self.session_id = c_session
self.ip = c_ip
self.port = c_port
self.received_peer_info = False
if __name__ == '__main__':
if len(sys.argv) < 2:
print("Usage: ./server.py PORT")
sys.exit(1)
port = int(sys.argv[1])
reactor.listenUDP(port, ServerProtocol())
print('Listening on *:%d' % (port))
reactor.run()