1
# IVLE - Informatics Virtual Learning Environment
2
# Copyright (C) 2007-2008 The University of Melbourne
4
# This program is free software; you can redistribute it and/or modify
5
# it under the terms of the GNU General Public License as published by
6
# the Free Software Foundation; either version 2 of the License, or
7
# (at your option) any later version.
9
# This program is distributed in the hope that it will be useful,
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12
# GNU General Public License for more details.
14
# You should have received a copy of the GNU General Public License
15
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19
# Author: Thomas Conway
33
class Terminate(Exception):
34
"""Exception thrown when server is to be shut down. It will attempt to
35
send the final_response to the client and then exits"""
36
def __init__(self, final_response=None):
37
self.final_response = final_response
40
return repr(self.final_response)
43
def start_server(port, magic, daemon_mode, handler, initializer = None):
44
# Attempt to open the socket.
45
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
49
# Excellent! It worked. Let's turn ourself into a daemon,
50
# then get on with the job of being a python interpreter.
52
if os.fork(): # launch child and...
53
os._exit(0) # kill off parent
55
if os.fork(): # launch child and...
56
os._exit(0) # kill off parent again.
60
MAXFD = os.sysconf("SC_OPEN_MAX")
64
# Close all file descriptors, except the socket.
65
for i in xrange(MAXFD):
73
si = os.open(os.devnull, os.O_RDONLY)
74
os.dup2(si, sys.stdin.fileno())
76
so = os.open(os.devnull, os.O_WRONLY)
77
os.dup2(so, sys.stdout.fileno())
79
se = os.open(os.devnull, os.O_WRONLY)
80
os.dup2(se, sys.stderr.fileno())
86
(conn, addr) = s.accept()
87
conn.settimeout(SOCKETTIMEOUT)
90
inp = recv_netstring(conn)
91
env = cjson.decode(inp)
93
# Check that the message is
94
digest = hashlib.md5(env['content'] + magic).hexdigest()
95
if env['digest'] != digest:
99
content = cjson.decode(env['content'])
101
response = handler(content)
103
send_netstring(conn, cjson.encode(response))
108
# Try and send final response and then terminate
110
send_netstring(conn, cjson.encode(t.final_response))
114
# Make a JSON object full of exceptional goodness
115
tb_dump = cStringIO.StringIO()
116
e_type, e_val, e_tb = sys.exc_info()
117
traceback.print_tb(e_tb, file=tb_dump)
119
"type": e_type.__name__,
121
"traceback": tb_dump.getvalue()
123
send_netstring(conn, cjson.encode(json_exc))
127
def chat(host, port, msg, magic, decode = True):
128
sok = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
129
sok.connect((host, port))
130
sok.settimeout(SOCKETTIMEOUT)
131
content = cjson.encode(msg)
132
digest = hashlib.md5(content + magic).hexdigest()
133
env = {'digest':digest,'content':content}
134
json = cjson.encode(env)
136
send_netstring(sok, json)
137
inp = recv_netstring(sok)
142
return cjson.decode(inp)
147
def send_netstring(sok, data):
148
netstring = "%d:%s,"%(len(data),data)
149
sok.sendall(netstring)
152
def recv_netstring(sok):
157
# Limit the Netstring to less than 10^10 bytes (~1GB).
158
if len(size_buffer) >= 10:
159
raise ValueError("Not valid Netstring: More than 10^10 bytes")
160
size_buffer.append(c)
163
# Message should be length plus ',' terminator
164
recv_expected = int(''.join(size_buffer)) + 1
165
except ValueError, e:
166
raise ValueError("Not valid Netstring: '%s'"%blk)
170
recv_data = sok.recv(min(recv_expected, BLOCKSIZE))
171
recv_size = len(recv_data)
172
while recv_size < recv_expected:
173
buf.append(recv_data)
174
recv_data = sok.recv(min(recv_expected-recv_size, 1024))
175
recv_size = recv_size + len(recv_data)
176
assert(recv_size == recv_expected)
178
# Did we receive the correct amount?
179
if recv_data[-1] != ',':
180
raise ValueError("Not valid Netstring: Did not end with ','")
181
buf.append(recv_data[:-1])