-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver.py
executable file
·92 lines (88 loc) · 3.77 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
#!/usr/bin/env python
############### Imports and globals #################
from server_db import ServerDBAgent
db_agent = ServerDBAgent('users.sqlite',
['''create table users
(username text,
fullname text,
password text)'''])
PORT = 8888
from message import *
import locale
import logging
logging.basicConfig(level=logging.DEBUG,
format='%(name)s: %(message)s')
logger = logging.getLogger('Server')
######################################################
from twisted.cred import portal, credentials
from auth import *
from twisted.internet import reactor, protocol, defer
from twisted.protocols import basic
class IMProtocol(basic.LineReceiver):
def connectionMade(self):
logger.debug("Client -- %s -- connected", self.transport.getPeer())
def connectionLost(self, reason):
logger.debug("Client -- %s -- disconnected", self.transport.getPeer())
def __forward_message(self,msg):
# check sender identity
try: sender = self.factory.clients[msg.src_id][0]
except KeyError: self.transport.loseConnection()
if sender != self:
self.transport.loseConnection()
return
# check if the recipient is logged in
try: recipient = self.factory.clients[msg.dst_id][0]
except KeyError: return
# forward message
msg = Message(Message.private,msg.src_id,msg.msg,msg.dst_id)
recipient.transport.write(str(msg))
def __login_user(self, req):
creds = credentials.UsernamePassword(req.src_id,req.msg)
self.factory.portal.login(creds,None,INamedUserAvatar).addCallback(
self.__login_succeeded).addErrback(self.__login_failed)
def __login_succeeded(self,avatar_info):
# add avatar to logged in clients list
avatar_interface, avatar, logout = avatar_info
self.factory.clients[avatar.username] = (self,avatar_info)
self.transport.write(str(Message(Message.login,0,locale.Login.succ_msg,
avatar.username)))
def __login_failed(self, failure):
logger.debug("failure: %s", str(failure))
self.transport.write(str(Message(Message.login,0,
locale.Login.failed_msg,
0)))
self.transport.loseConnection()
def __logout_user(self, req):
pass
#logger.debug("Client -- %s -- logged out", self.transport.getPeer())
#self.factory.clients.remove(self)
#resp = Message(SERVER_ID, req.src_id, LOGOUT, locale.Login.success)
#self.transport.write(resp.to_s())
#self.transport.loseConnection()
def __create_user(self, req): pass
def lineReceived(self, line):
logger.debug("Received: %s | from %s",
repr(line), self.transport.getPeer())
# parse received packet
try: req = Message(*line.split(','))
except TypeError, msg:
raise MessageFormatInvalid("Invalid message format: "+str(msg))
# react accordingly
if req.msg_type == Message.private:
self.__forward_message(req)
elif req.msg_type == Message.login:
self.__login_user(req)
elif req.msg_type == Message.logout:
self.__logout_user(req)
elif req.msg_type == Message.create:
self.__create_user(req)
class IMServerFactory(protocol.ServerFactory):
protocol = IMProtocol
clients = {}
def __init__(self, portal):
self.portal = portal
if __name__ == '__main__':
p = portal.Portal(DBRealm(db_agent))
p.registerChecker(DBPasswordChecker(db_agent))
reactor.listenTCP(PORT,IMServerFactory(p))
reactor.run()