blob: b3943ff39324c27b8fe2e3697c803ed7fdf36fa7 [file] [log] [blame]
Guido van Rossum07a272d1995-04-10 11:40:52 +00001"""RPC Server module."""
2
3import sys
4import socket
5import pickle
6from fnmatch import fnmatch
Alexandre Vassalotti1f2ba4b2008-05-16 07:12:44 +00007from reprlib import repr
Guido van Rossum07a272d1995-04-10 11:40:52 +00008
9
10# Default verbosity (0 = silent, 1 = print connections, 2 = print requests too)
11VERBOSE = 1
12
13
14class Server:
Tim Peterse6ddc8b2004-07-18 05:56:09 +000015
16 """RPC Server class. Derive a class to implement a particular service."""
17
18 def __init__(self, address, verbose = VERBOSE):
19 if type(address) == type(0):
20 address = ('', address)
21 self._address = address
22 self._verbose = verbose
23 self._socket = None
24 self._socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
25 self._socket.bind(address)
26 self._socket.listen(1)
27 self._listening = 1
28
29 def _setverbose(self, verbose):
30 self._verbose = verbose
31
32 def __del__(self):
33 self._close()
34
35 def _close(self):
36 self._listening = 0
37 if self._socket:
38 self._socket.close()
39 self._socket = None
40
41 def _serverloop(self):
42 while self._listening:
43 self._serve()
44
45 def _serve(self):
Collin Winter6f2df4d2007-07-17 20:59:35 +000046 if self._verbose: print("Wait for connection ...")
Tim Peterse6ddc8b2004-07-18 05:56:09 +000047 conn, address = self._socket.accept()
Collin Winter6f2df4d2007-07-17 20:59:35 +000048 if self._verbose: print("Accepted connection from %s" % repr(address))
Tim Peterse6ddc8b2004-07-18 05:56:09 +000049 if not self._verify(conn, address):
Collin Winter6f2df4d2007-07-17 20:59:35 +000050 print("*** Connection from %s refused" % repr(address))
Tim Peterse6ddc8b2004-07-18 05:56:09 +000051 conn.close()
52 return
53 rf = conn.makefile('r')
54 wf = conn.makefile('w')
55 ok = 1
56 while ok:
57 wf.flush()
Collin Winter6f2df4d2007-07-17 20:59:35 +000058 if self._verbose > 1: print("Wait for next request ...")
Tim Peterse6ddc8b2004-07-18 05:56:09 +000059 ok = self._dorequest(rf, wf)
60
61 _valid = ['192.16.201.*', '192.16.197.*', '132.151.1.*', '129.6.64.*']
62
63 def _verify(self, conn, address):
64 host, port = address
65 for pat in self._valid:
66 if fnmatch(host, pat): return 1
67 return 0
68
69 def _dorequest(self, rf, wf):
70 rp = pickle.Unpickler(rf)
71 try:
72 request = rp.load()
73 except EOFError:
74 return 0
Collin Winter6f2df4d2007-07-17 20:59:35 +000075 if self._verbose > 1: print("Got request: %s" % repr(request))
Tim Peterse6ddc8b2004-07-18 05:56:09 +000076 try:
77 methodname, args, id = request
78 if '.' in methodname:
79 reply = (None, self._special(methodname, args), id)
80 elif methodname[0] == '_':
Collin Winter6f2df4d2007-07-17 20:59:35 +000081 raise NameError("illegal method name %s" % repr(methodname))
Tim Peterse6ddc8b2004-07-18 05:56:09 +000082 else:
83 method = getattr(self, methodname)
Neal Norwitzd9108552006-03-17 08:00:19 +000084 reply = (None, method(*args), id)
Tim Peterse6ddc8b2004-07-18 05:56:09 +000085 except:
Neal Norwitzac3625f2006-03-17 05:49:33 +000086 reply = (sys.exc_info()[:2], id)
Tim Peterse6ddc8b2004-07-18 05:56:09 +000087 if id < 0 and reply[:2] == (None, None):
Collin Winter6f2df4d2007-07-17 20:59:35 +000088 if self._verbose > 1: print("Suppress reply")
Tim Peterse6ddc8b2004-07-18 05:56:09 +000089 return 1
Collin Winter6f2df4d2007-07-17 20:59:35 +000090 if self._verbose > 1: print("Send reply: %s" % repr(reply))
Tim Peterse6ddc8b2004-07-18 05:56:09 +000091 wp = pickle.Pickler(wf)
92 wp.dump(reply)
93 return 1
94
95 def _special(self, methodname, args):
96 if methodname == '.methods':
97 if not hasattr(self, '_methods'):
98 self._methods = tuple(self._listmethods())
99 return self._methods
Collin Winter6f2df4d2007-07-17 20:59:35 +0000100 raise NameError("unrecognized special method name %s" % repr(methodname))
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000101
102 def _listmethods(self, cl=None):
103 if not cl: cl = self.__class__
Skip Montanaro1e8ce582007-08-06 21:07:53 +0000104 names = sorted([x for x in cl.__dict__.keys() if x[0] != '_'])
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000105 for base in cl.__bases__:
106 basenames = self._listmethods(base)
Collin Winter6f2df4d2007-07-17 20:59:35 +0000107 basenames = list(filter(lambda x, names=names: x not in names, basenames))
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000108 names[len(names):] = basenames
109 return names
Guido van Rossum45babef1995-06-21 01:00:17 +0000110
111
112from security import Security
113
114
115class SecureServer(Server, Security):
116
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000117 def __init__(self, *args):
Neal Norwitzd9108552006-03-17 08:00:19 +0000118 Server.__init__(self, *args)
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000119 Security.__init__(self)
Guido van Rossum45babef1995-06-21 01:00:17 +0000120
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000121 def _verify(self, conn, address):
122 import string
123 challenge = self._generate_challenge()
124 conn.send("%d\n" % challenge)
125 response = ""
126 while "\n" not in response and len(response) < 100:
127 data = conn.recv(100)
128 if not data:
129 break
130 response = response + data
131 try:
132 response = string.atol(string.strip(response))
133 except string.atol_error:
134 if self._verbose > 0:
Collin Winter6f2df4d2007-07-17 20:59:35 +0000135 print("Invalid response syntax", repr(response))
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000136 return 0
137 if not self._compare_challenge_response(challenge, response):
138 if self._verbose > 0:
Collin Winter6f2df4d2007-07-17 20:59:35 +0000139 print("Invalid response value", repr(response))
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000140 return 0
141 if self._verbose > 1:
Collin Winter6f2df4d2007-07-17 20:59:35 +0000142 print("Response matches challenge. Go ahead!")
Tim Peterse6ddc8b2004-07-18 05:56:09 +0000143 return 1