73
73
# (Note that paths "relative" to the jail actually begin with a '/' as
74
74
# they are absolute in the jailspace)
76
return interpreter(owner, jail_dir, working_dir, filename_abs, req,
76
return interpreter(owner.unixid, jail_dir, working_dir, filename_abs, req,
89
89
self.headers = {} # Header names : values
91
def execute_cgi(interpreter, owner, jail_dir, working_dir, script_path,
91
def execute_cgi(interpreter, trampoline, uid, jail_dir, working_dir,
92
script_path, req, gentle):
94
94
trampoline: Full path on the local system to the CGI wrapper program
96
owner: User object of the owner of the file.
96
uid: User ID of the owner of the file.
97
97
jail_dir: Absolute path of owner's jail directory.
98
98
working_dir: Directory containing the script file relative to owner's
136
134
del os.environ[k]
137
135
for (k,v) in req.get_cgi_environ().items():
138
136
os.environ[k] = v
139
fixup_environ(req, script_path, owner)
137
fixup_environ(req, script_path)
141
139
# usage: tramp uid jail_dir working_dir script_path
142
cmd_line = [trampoline, str(owner.unixid),
143
req.config['paths']['jails']['mounts'],
144
req.config['paths']['jails']['src'],
145
req.config['paths']['jails']['template'],
146
jail_dir, working_dir, interpreter, script_path]
147
# Popen doesn't like unicode strings. It hateses them.
148
cmd_line = [(s.encode('utf-8') if isinstance(s, unicode) else s)
150
pid = subprocess.Popen(cmd_line,
140
pid = subprocess.Popen(
141
[trampoline, str(uid), ivle.conf.jail_base, ivle.conf.jail_src_base,
142
ivle.conf.jail_system, jail_dir, working_dir, interpreter,
151
144
stdin=f, stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
350
343
""" % (warning, text))
345
location_cgi_python = os.path.join(ivle.conf.lib_path, "trampoline")
352
347
# Mapping of interpreter names (as given in conf/app/server.py) to
353
348
# interpreter functions.
355
350
interpreter_objects = {
357
: functools.partial(execute_cgi, "/usr/bin/python"),
352
: functools.partial(execute_cgi, "/usr/bin/python",
353
location_cgi_python),
359
: functools.partial(execute_cgi, None),
355
: functools.partial(execute_cgi, None,
356
location_cgi_python),
360
357
# Should also have:
362
359
# python-server-page
365
def fixup_environ(req, script_path, user):
362
def fixup_environ(req, script_path):
366
363
"""Assuming os.environ has been written with the CGI variables from
367
364
apache, make a few changes for security and correctness.
419
416
env['SERVER_SOFTWARE'] = "IVLE/" + ivle.__version__
421
418
# Additional environment variables
422
username = user.login
419
username = split_path(req.path)[0]
423
420
env['HOME'] = os.path.join('/home', username)
425
422
class ExecutionError(Exception):
428
def execute_raw(config, user, jail_dir, working_dir, binary, args):
425
def execute_raw(user, jail_dir, working_dir, binary, args):
429
426
'''Execute a binary in a user's jail, returning the raw output.
431
428
The binary is executed in the given working directory with the given
432
429
args. A tuple of (stdout, stderr) is returned.
435
tramp = os.path.join(config['paths']['lib'], 'trampoline')
436
tramp_dir = os.path.split(tramp)[0]
432
tramp = location_cgi_python
433
tramp_dir = os.path.split(location_cgi_python)[0]
438
435
# Fire up trampoline. Vroom, vroom.
439
cmd_line = [tramp, str(user.unixid), config['paths']['jails']['mounts'],
440
config['paths']['jails']['src'],
441
config['paths']['jails']['template'],
442
jail_dir, working_dir, binary] + args
443
# Popen doesn't like unicode strings. It hateses them.
444
cmd_line = [(s.encode('utf-8') if isinstance(s, unicode) else s)
446
proc = subprocess.Popen(cmd_line,
436
proc = subprocess.Popen(
437
[tramp, str(user.unixid), ivle.conf.jail_base,
438
ivle.conf.jail_src_base, ivle.conf.jail_system, jail_dir,
439
working_dir, binary] + args,
447
440
stdin=subprocess.PIPE, stdout=subprocess.PIPE,
448
stderr=subprocess.PIPE, cwd=tramp_dir, close_fds=True,
449
env={'HOME': os.path.join('/home', user.login),
450
'PATH': os.environ['PATH'],
452
'LOGNAME': user.login})
441
stderr=subprocess.PIPE, cwd=tramp_dir, close_fds=True)
454
443
(stdout, stderr) = proc.communicate()
455
444
exitcode = proc.returncode
457
446
if exitcode != 0:
458
raise ExecutionError('subprocess ended with code %d, stderr: "%s"' %
447
raise ExecutionError('subprocess ended with code %d, stderr %s' %
448
(exitcode, proc.stderr.read()))
460
449
return (stdout, stderr)