15
15
# along with this program; if not, write to the Free Software
16
16
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19
# Author: Matt Giuca, Tom Conway
18
# Author: Matt Giuca, Tom Conway, Will Grant
20
'''Python console RPC service.
22
Provides an HTTP RPC interface to a Python console process.
26
from common import (util, studpath)
30
"""Handler for the Console Service AJAX backend application."""
32
# Set request attributes
33
req.content_type = "text/plain"
34
req.write_html_head_foot = False
36
# TODO: Figure out the host name the console server is running on.
39
# Find an available port on the server.
47
# Start the console server (port, magic)
48
# TODO: Trampoline into the jail, and run it as a background process
49
jail = os.path.join(conf.jail_base, req.username)
50
console_dir = os.path.join(jail, "opt/ivle/console/")
51
console_path = os.path.join(console_dir, "python-console")
52
os.system("cd " + console_dir + "; "
53
+ console_path + " " + str(port) + " " + str(magic)
54
+ " > /home/mgiuca/Desktop/python_console_log 2>&1 &")
57
req.write(cjson.encode({"host": host, "port": port, "magic": magic}))
34
from ivle.webapp.base.rest import JSONRESTView, write_operation
35
from ivle.webapp.errors import BadRequest
37
# XXX: Should be RPC view, with actions in URL?
38
class ConsoleServiceRESTView(JSONRESTView):
39
'''An RPC interface to a Python console.'''
40
def get_permissions(self, user, config):
46
@write_operation('use')
47
def start(self, req, cwd=''):
48
working_dir = os.path.join("/home", req.user.login, cwd)
51
jail_path = os.path.join(req.config['paths']['jails']['mounts'],
53
cons = ivle.console.Console(req.config, req.user, jail_path,
56
# Assemble the key and return it. Yes, it is double-encoded.
57
return {'key': cjson.encode({"host": cons.host,
59
"magic": cons.magic}).encode('hex')}
61
@write_operation('use')
62
def chat(self, req, key, text='', cwd='', kind="chat"):
63
# The request *should* have the following four fields:
64
# key: Hex JSON dict of host and port where the console server lives,
65
# and the secret to use to digitally sign the communication with the
67
# text: Fields to pass along to the console server
68
# It simply acts as a proxy to the console server
71
keydict = cjson.decode(key.decode('hex'))
72
host = keydict['host']
73
port = keydict['port']
74
magic = keydict['magic']
76
raise BadRequest("Invalid console key.")
78
jail_path = os.path.join(req.config['paths']['jails']['mounts'],
81
working_dir = os.path.join("/home", req.user.login, cwd)
83
# XXX: JSONRESTView should do this for us.
84
text = text.decode('utf-8')
86
msg = {'cmd':kind, 'text':text}
89
json_response = ivle.chat.chat(host, port, msg, magic,decode=False)
90
# Snoop the response from python-console to check that it's valid
91
response = cjson.decode(json_response)
92
except (cjson.DecodeError, ivle.chat.ProtocolError):
93
# Could not decode the reply from the python-console server
94
response = {"terminate":
96
if "terminate" in response:
97
response = restart_console(req.config, req.user, jail_path,
98
working_dir, response["terminate"])
99
except socket.error, (enumber, estring):
100
if enumber == errno.ECONNREFUSED:
101
# Timeout: Restart the session
102
response = restart_console(req.config, req.user, jail_path,
104
"Timed out due to inactivity")
105
elif enumber == errno.ECONNRESET:
106
# Communication issue: Restart the session
107
response = restart_console(req.config, req.user, jail_path,
111
# Some other error - probably serious
112
raise socket.error, (enumber, estring)
116
def restart_console(config, user, jail_path, working_dir, reason):
117
"""Tells the client that it must be issued a new console since the old
118
console is no longer availible. The client must accept the new key.
119
Returns the JSON response to be given to the client.
121
# Start a new console server console
122
cons = ivle.console.Console(config, user, jail_path, working_dir)
124
# Make a JSON object to tell the browser to restart its console client
125
new_key = cjson.encode(
126
{"host": cons.host, "port": cons.port, "magic": cons.magic})
128
return {"restart": reason, "key": new_key.encode("hex")}