33
from ivle import chat, interpret
36
from ivle import (chat, util)
39
trampoline_path = os.path.join(ivle.conf.lib_path, "trampoline")
41
python_path = "/usr/bin/python"
42
console_dir = os.path.join(ivle.conf.share_path, 'services')
43
console_path = os.path.join(console_dir, 'python-console')
35
45
class ConsoleError(Exception):
36
46
""" The console failed in some way. This is bad. """
47
def __init__(self, value):
50
return repr(self.value)
39
52
class ConsoleException(Exception):
40
53
""" The code being exectuted on the console returned an exception.
55
def __init__(self, value):
58
return repr(self.value)
44
60
class TruncateStringIO(StringIO.StringIO):
45
61
""" A class that wraps around StringIO and truncates the buffer when the
106
122
class Console(object):
107
123
""" Provides a nice python interface to the console
109
def __init__(self, config, user, jail_path, working_dir):
125
def __init__(self, uid, jail_path, working_dir):
110
126
"""Starts up a console service for user uid, inside chroot jail
111
127
jail_path with work directory of working_dir
113
129
super(Console, self).__init__()
117
132
self.jail_path = jail_path
118
133
self.working_dir = working_dir
142
self.magic = hashlib.md5(uuid.uuid4().bytes).hexdigest()
157
self.magic = md5.new(uuid.uuid4().bytes).digest().encode('hex')
144
159
# Try to find a free port on the server.
145
160
# Just try some random ports in the range [3000,8000)
150
165
# is (k / N) ** t (e.g. 3.2*10e-9).
155
169
self.port = int(random.uniform(3000, 8000))
157
python_console = os.path.join(self.config['paths']['share'],
158
'services/python-console')
159
args = [python_console, str(self.port), str(self.magic)]
171
# Start the console server (port, magic)
172
# trampoline usage: tramp uid jail_dir working_dir script_path args
173
# console usage: python-console port magic
174
res = os.spawnv(os.P_WAIT, trampoline_path, [
162
interpret.execute_raw(self.config, self.user, self.jail_path,
163
self.working_dir, "/usr/bin/python", args)
166
except interpret.ExecutionError, e:
170
# If we can't start the console after 5 attemps (can't find a free
171
# port during random probing, syntax errors, segfaults) throw an
192
# If we can't start the console after 5 attemps (can't find a free port
193
# during random probing, syntax errors, segfaults) throw an exception.
174
raise ConsoleError('Unable to start console service: %s'%error)
195
raise ConsoleError("Unable to start console service!")
176
197
def __chat(self, cmd, args):
177
198
""" A wrapper around chat.chat to comunicate directly with the
189
210
# Some other error - probably serious
190
211
raise socket.error, (enumber, estring)
212
except cjson.DecodeError:
192
213
# Couldn't decode the JSON
193
214
raise ConsoleError(
194
215
"Could not understand the python console response")
195
except chat.ProtocolError, e:
196
raise ConsoleError(*e.args)
279
298
# Unpickle any exceptions
280
299
if 'exception' in execute:
281
execute['exception'] = cPickle.loads(str(execute['exception']))
300
return cPickle.loads(execute['exception'])
285
305
def set_vars(self, variables):