32
37
from ivle import (util, studpath, chat, console)
39
from ivle.webapp.base.views import JSONRESTView, named_operation
41
# XXX: Should be RPC view, with actions in URL?
42
class ConsoleServiceRESTView(JSONRESTView):
43
'''An RPC interface to a Python console.'''
45
def start(self, req, cwd=''):
46
working_dir = os.path.join("/home", req.user.login, cwd)
51
jail_path = os.path.join(ivle.conf.jail_base, req.user.login)
52
cons = console.Console(uid, jail_path, working_dir)
54
# Assemble the key and return it. Yes, it is double-encoded.
55
return {'key': cjson.encode({"host": cons.host,
57
"magic": cons.magic}).encode('hex')}
60
def chat(self, req, key, text='', kind="chat"):
61
# The request *should* have the following four fields:
62
# key: Hex JSON dict of host and port where the console server lives,
63
# and the secret to use to digitally sign the communication with the
65
# text: Fields to pass along to the console server
66
# It simply acts as a proxy to the console server
69
keydict = cjson.decode(key.decode('hex'))
70
host = keydict['host']
71
port = keydict['port']
72
magic = keydict['magic']
74
raise BadRequest("Invalid console key.")
76
jail_path = os.path.join(ivle.conf.jail_base, req.user.login)
77
working_dir = os.path.join("/home", req.user.login) # Within jail
80
msg = {'cmd':kind, 'text':text}
82
json_response = chat.chat(host, port, msg, magic, decode = False)
84
# Snoop the response from python-console to check that it's valid
86
response = cjson.decode(json_response)
87
except cjson.DecodeError:
88
# Could not decode the reply from the python-console server
89
response = {"terminate":
90
"Communication to console process lost"}
91
if "terminate" in response:
92
response = restart_console(uid, jail_path, working_dir,
93
response["terminate"])
94
except socket.error, (enumber, estring):
95
if enumber == errno.ECONNREFUSED:
96
# Timeout: Restart the session
97
response = restart_console(uid, jail_path, working_dir,
98
"The IVLE console has timed out due to inactivity")
99
elif enumber == errno.ECONNRESET:
100
# Communication issue: Restart the session
101
response = restart_console(uid, jail_path, working_dir,
102
"Connection with the console has been reset")
104
# Some other error - probably serious
105
raise socket.error, (enumber, estring)
37
109
"""Handler for the Console Service AJAX backend application."""
58
130
req.throw_error(req.HTTP_BAD_REQUEST)
60
def handle_start(req):
61
# Changes the state on the server - must be POST
62
if req.method != "POST":
63
req.throw_error(req.HTTP_BAD_REQUEST)
65
# See if we have been given extra params
66
fields = req.get_fieldstorage()
68
startdir = fields.getfirst("startdir").value
69
working_dir = os.path.join("/home", req.user.login, startdir)
70
except AttributeError:
71
working_dir = os.path.join("/home", req.user.login)
73
# Get the UID of the logged-in user
76
# Set request attributes
77
req.content_type = "text/plain"
78
req.write_html_head_foot = False
81
jail_path = os.path.join(ivle.conf.jail_base, req.user.login)
82
cons = console.Console(uid, jail_path, working_dir)
84
# Assemble the key and return it.
86
{"host": cons.host, "port": cons.port, "magic": cons.magic})
87
req.write(cjson.encode({"key": key.encode("hex")}))
89
def handle_chat(req, kind = "chat"):
90
# The request *should* have the following four fields:
91
# host, port, magic: Host and port where the console server lives,
92
# and the secret to use to digitally sign the communication with the
94
# text: Fields to pass along to the console server
95
# It simply acts as a proxy to the console server
96
if req.method != "POST":
97
req.throw_error(req.HTTP_BAD_REQUEST)
98
jail_path = os.path.join(ivle.conf.jail_base, req.user.login)
99
working_dir = os.path.join("/home", req.user.login) # Within jail
100
uid = req.user.unixid
101
fields = req.get_fieldstorage()
103
key = cjson.decode(fields.getfirst("key").value.decode("hex"))
107
except AttributeError:
108
# Any of the getfirsts returned None
109
req.throw_error(req.HTTP_BAD_REQUEST)
110
# If text is None, it was probably just an empty line
112
text = fields.getfirst("text").value.decode('utf-8')
113
except AttributeError:
116
msg = {'cmd':kind, 'text':text}
118
response = chat.chat(host, port, msg, magic, decode = False)
120
# Snoop the response from python-console to check that it's valid
122
decoded_response = cjson.decode(response)
123
except cjson.DecodeError:
124
# Could not decode the reply from the python-console server
125
decoded_response = {"terminate":
126
"Communication to console process lost"}
127
if "terminate" in decoded_response:
128
response = restart_console(uid, jail_path, working_dir,
129
decoded_response["terminate"])
131
except socket.error, (enumber, estring):
132
if enumber == errno.ECONNREFUSED:
133
# Timeout: Restart the session
134
response = restart_console(uid, jail_path, working_dir,
135
"The IVLE console has timed out due to inactivity")
136
elif enumber == errno.ECONNRESET:
137
# Communication issue: Restart the session
138
response = restart_console(uid, jail_path, working_dir,
139
"Connection with the console has been reset")
141
# Some other error - probably serious
142
raise socket.error, (enumber, estring)
144
req.content_type = "text/plain"
147
134
def restart_console(uid, jail_path, working_dir, reason):
148
135
"""Tells the client that it must be issued a new console since the old