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.
29
from common import (util, studpath)
32
trampoline_path = os.path.join(conf.ivle_install_dir, "bin/trampoline")
33
python_path = "/usr/bin/python" # Within jail
34
console_dir = "/opt/ivle/console" # Within jail
35
console_path = "/opt/ivle/console/python-console" # Within jail
38
"""Handler for the Console Service AJAX backend application."""
39
if len(req.path) > 0 and req.path[-1] == os.sep:
43
# The path determines which "command" we are receiving
44
if req.path == "start":
46
elif req.path == "chat":
49
req.throw_error(req.HTTP_BAD_REQUEST)
51
def handle_start(req):
52
jail_path = os.path.join(conf.jail_base, req.username)
53
working_dir = os.path.join("/home", req.username) # Within jail
55
# Get the UID of the logged-in user
57
(_,_,uid,_,_,_,_) = pwd.getpwnam(req.username)
59
# The user does not exist. This should have already failed the
61
req.throw_error(req.HTTP_INTERNAL_SERVER_ERROR)
63
# Set request attributes
64
req.content_type = "text/plain"
65
req.write_html_head_foot = False
67
# TODO: Figure out the host name the console server is running on.
70
# Find an available port on the server.
78
# Start the console server (port, magic)
79
# trampoline usage: tramp uid jail_dir working_dir script_path args
80
# console usage: python-console port magic
81
# TODO: Cleanup (don't use os.system)
82
# TODO: Pass working_dir as argument, let console cd to it
83
# Use "&" to run as a background process
84
cmd = ' '.join([trampoline_path, str(uid), jail_path, console_dir,
85
python_path, console_path, str(port), str(magic), "&"])
86
#req.write(cmd + '\n')
90
req.write(cjson.encode({"host": host, "port": port, "magic": magic}))
93
# The request *should* have the following four fields:
94
# host, port: Host and port where the console server apparently lives
95
# digest, text: Fields to pass along to the console server
96
# It simply acts as a proxy to the console server
97
if req.method != "POST":
98
req.throw_error(req.HTTP_BAD_REQUEST)
99
fields = req.get_fieldstorage()
101
host = fields.getfirst("host").value
102
port = fields.getfirst("port").value
103
digest = fields.getfirst("digest").value
104
except AttributeError:
105
# Any of the getfirsts returned None
106
req.throw_error(req.HTTP_BAD_REQUEST)
107
# If text is None, it was probably just an empty line
109
text = fields.getfirst("text").value
110
except AttributeError:
113
# Open an HTTP connection
114
url = ("http://" + urllib.quote(host) + ":" + urllib.quote(port)
116
body = ("digest=" + urllib.quote(digest)
117
+ "&text=" + urllib.quote(text))
118
headers = {"Content-Type": "application/x-www-form-urlencoded"}
120
conn = httplib.HTTPConnection(host, port)
122
req.throw_error(req.HTTP_BAD_REQUEST)
123
conn.request("POST", url, body, headers)
125
response = conn.getresponse()
127
req.status = response.status
128
# NOTE: Ignoring arbitrary headers returned by the server
129
# Probably not necessary to proxy them
130
req.content_type = response.getheader("Content-Type", "text/plain")
131
req.write(response.read())
35
from ivle.webapp.base.rest import JSONRESTView, named_operation
37
# XXX: Should be RPC view, with actions in URL?
38
class ConsoleServiceRESTView(JSONRESTView):
39
'''An RPC interface to a Python console.'''
41
def start(self, req, cwd=''):
42
working_dir = os.path.join("/home", req.user.login, cwd)
47
jail_path = os.path.join(ivle.conf.jail_base, req.user.login)
48
cons = ivle.console.Console(uid, jail_path, working_dir)
50
# Assemble the key and return it. Yes, it is double-encoded.
51
return {'key': cjson.encode({"host": cons.host,
53
"magic": cons.magic}).encode('hex')}
56
def chat(self, req, key, text='', kind="chat"):
57
# The request *should* have the following four fields:
58
# key: Hex JSON dict of host and port where the console server lives,
59
# and the secret to use to digitally sign the communication with the
61
# text: Fields to pass along to the console server
62
# It simply acts as a proxy to the console server
65
keydict = cjson.decode(key.decode('hex'))
66
host = keydict['host']
67
port = keydict['port']
68
magic = keydict['magic']
70
raise BadRequest("Invalid console key.")
72
jail_path = os.path.join(ivle.conf.jail_base, req.user.login)
73
working_dir = os.path.join("/home", req.user.login) # Within jail
76
msg = {'cmd':kind, 'text':text}
78
json_response = ivle.chat.chat(host, port, msg, magic,decode=False)
80
# Snoop the response from python-console to check that it's valid
82
response = cjson.decode(json_response)
83
except cjson.DecodeError:
84
# Could not decode the reply from the python-console server
85
response = {"terminate":
86
"Communication to console process lost"}
87
if "terminate" in response:
88
response = restart_console(uid, jail_path, working_dir,
89
response["terminate"])
90
except socket.error, (enumber, estring):
91
if enumber == errno.ECONNREFUSED:
92
# Timeout: Restart the session
93
response = restart_console(uid, jail_path, working_dir,
94
"The IVLE console has timed out due to inactivity")
95
elif enumber == errno.ECONNRESET:
96
# Communication issue: Restart the session
97
response = restart_console(uid, jail_path, working_dir,
98
"Connection with the console has been reset")
100
# Some other error - probably serious
101
raise socket.error, (enumber, estring)
105
def restart_console(uid, jail_path, working_dir, reason):
106
"""Tells the client that it must be issued a new console since the old
107
console is no longer availible. The client must accept the new key.
108
Returns the JSON response to be given to the client.
110
# Start a new console server console
111
cons = ivle.console.Console(uid, jail_path, working_dir)
113
# Make a JSON object to tell the browser to restart its console client
114
new_key = cjson.encode(
115
{"host": cons.host, "port": cons.port, "magic": cons.magic})
117
return {"restart": reason, "key": new_key.encode("hex")}