Commit 75cb3572 authored by Martín Ferrari's avatar Martín Ferrari

Proper propagation of exceptions

parent 9fc3b3ec
...@@ -9,8 +9,6 @@ except ImportError: ...@@ -9,8 +9,6 @@ except ImportError:
import base64, os, passfd, re, signal, sys, traceback, unshare, yaml import base64, os, passfd, re, signal, sys, traceback, unshare, yaml
import netns.subprocess_, netns.iproute, netns.interface import netns.subprocess_, netns.iproute, netns.interface
# FIXME: proper and uniform handling of errors
# ============================================================================ # ============================================================================
# Server-side protocol implementation # Server-side protocol implementation
# #
...@@ -220,7 +218,13 @@ class Server(object): ...@@ -220,7 +218,13 @@ class Server(object):
cmd = self.readcmd() cmd = self.readcmd()
if cmd == None: if cmd == None:
continue continue
cmd[0](cmd[1], *cmd[2]) try:
cmd[0](cmd[1], *cmd[2])
except:
(t, v, tb) = sys.exc_info()
v.child_traceback = "".join(
traceback.format_exception(t, v, tb))
self.reply(550, yaml.dump(v))
try: try:
self._rfd.close() self._rfd.close()
self._wfd.close() self._wfd.close()
...@@ -291,28 +295,20 @@ class Server(object): ...@@ -291,28 +295,20 @@ class Server(object):
do_PROC_SOUT = do_PROC_SERR = do_PROC_SIN do_PROC_SOUT = do_PROC_SERR = do_PROC_SIN
def do_PROC_RUN(self, cmdname): def do_PROC_RUN(self, cmdname):
try: params = self._proc
self._proc['close_fds'] = True # forced params['close_fds'] = True # forced
chld = netns.subprocess_.spawn(**self._proc) self._proc = None
except:
(t, v, tb) = sys.exc_info()
r = ["Failure starting process: %s" % str(v)]
if self.debug:
r += traceback.format_exception(t, v, tb)
self.reply(500, r)
self._proc = None
self.commands = _proto_commands
return
self._children.add(chld)
self.commands = _proto_commands self.commands = _proto_commands
# I can close the fds now try:
for d in ('stdin', 'stdout', 'stderr'): chld = netns.subprocess_.spawn(**params)
if d in self._proc: finally:
os.close(self._proc[d]) # I can close the fds now
for d in ('stdin', 'stdout', 'stderr'):
if d in params:
os.close(params[d])
self._proc = None self._children.add(chld)
self.reply(200, "%d running." % chld) self.reply(200, "%d running." % chld)
def do_PROC_ABRT(self, cmdname): def do_PROC_ABRT(self, cmdname):
...@@ -351,9 +347,6 @@ class Server(object): ...@@ -351,9 +347,6 @@ class Server(object):
def do_IF_LIST(self, cmdname, ifnr = None): def do_IF_LIST(self, cmdname, ifnr = None):
ifdata = netns.iproute.get_if_data()[0] ifdata = netns.iproute.get_if_data()[0]
if ifnr != None: if ifnr != None:
if ifnr not in ifdata:
self.reply(500, "Interface not found.")
return
ifdata = ifdata[ifnr] ifdata = ifdata[ifnr]
self.reply(200, ["# Interface data follows."] + self.reply(200, ["# Interface data follows."] +
yaml.dump(ifdata).split("\n")) yaml.dump(ifdata).split("\n"))
...@@ -367,33 +360,17 @@ class Server(object): ...@@ -367,33 +360,17 @@ class Server(object):
for i in range(len(args) / 2): for i in range(len(args) / 2):
d[str(args[i * 2])] = args[i * 2 + 1] d[str(args[i * 2])] = args[i * 2 + 1]
try: iface = netns.interface.interface(**d)
iface = netns.interface.interface(**d) netns.iproute.set_if(iface)
except:
self.reply(500, "Invalid parameters.")
return
try:
netns.iproute.set_if(iface)
except BaseException, e:
self.reply(500, "Error setting interface: %s." % str(e))
return
self.reply(200, "Done.") self.reply(200, "Done.")
def do_IF_RTRN(self, cmdname, ifnr, netns): def do_IF_RTRN(self, cmdname, ifnr, netns):
try: netns.iproute.change_netns(ifnr, netns)
netns.iproute.change_netns(ifnr, netns)
except BaseException, e:
self.reply(500, "Error returning interface: %s." % str(e))
return
self.reply(200, "Done.") self.reply(200, "Done.")
def do_ADDR_LIST(self, cmdname, ifnr = None): def do_ADDR_LIST(self, cmdname, ifnr = None):
addrdata = netns.iproute.get_addr_data()[0] addrdata = netns.iproute.get_addr_data()[0]
if ifnr != None: if ifnr != None:
if ifnr not in addrdata:
self.reply(500, "Interface not found.")
return
addrdata = addrdata[ifnr] addrdata = addrdata[ifnr]
self.reply(200, ["# Address data follows."] + self.reply(200, ["# Address data follows."] +
yaml.dump(addrdata).split("\n")) yaml.dump(addrdata).split("\n"))
...@@ -458,6 +435,9 @@ class Client(object): ...@@ -458,6 +435,9 @@ class Client(object):
code is not the expected value. If expected is not specified, it code is not the expected value. If expected is not specified, it
defaults to 2.""" defaults to 2."""
code, text = self._read_reply() code, text = self._read_reply()
if code == 550: # exception
e = yaml.load(text)
raise e
if code / 100 != expected: if code / 100 != expected:
# FIXME: shuld try to save and re-create exceptions # FIXME: shuld try to save and re-create exceptions
raise RuntimeError("Error from slave: %d %s" % (code, text)) raise RuntimeError("Error from slave: %d %s" % (code, text))
...@@ -610,8 +590,8 @@ class Client(object): ...@@ -610,8 +590,8 @@ class Client(object):
def _b64(text): def _b64(text):
text = str(text) text = str(text)
if filter(lambda x: ord(x) <= ord(" ") or ord(x) > ord("z") if len(text) == 0 or filter(lambda x: ord(x) <= ord(" ") or
or x == "=", text): ord(x) > ord("z") or x == "=", text):
return "=" + base64.b64encode(text) return "=" + base64.b64encode(text)
else: else:
return text return text
......
...@@ -123,26 +123,26 @@ class TestSubprocess(unittest.TestCase): ...@@ -123,26 +123,26 @@ class TestSubprocess(unittest.TestCase):
self.assertEquals(wait(p), 0) self.assertEquals(wait(p), 0)
def test_Subprocess_basic(self): def test_Subprocess_basic(self):
node = netns.Node(nonetns = True) node = netns.Node(nonetns = True, debug = 0)
# User does not exist # User does not exist
self.assertRaises(RuntimeError, Subprocess, node, self.assertRaises(ValueError, Subprocess, node,
['/bin/sleep', '1000'], user = self.nouser) ['/bin/sleep', '1000'], user = self.nouser)
self.assertRaises(RuntimeError, Subprocess, node, self.assertRaises(ValueError, Subprocess, node,
['/bin/sleep', '1000'], user = self.nouid) ['/bin/sleep', '1000'], user = self.nouid)
# Invalid CWD: it is a file # Invalid CWD: it is a file
self.assertRaises(RuntimeError, Subprocess, node, self.assertRaises(OSError, Subprocess, node,
'/bin/sleep', cwd = '/bin/sleep') '/bin/sleep', cwd = '/bin/sleep')
# Invalid CWD: does not exist # Invalid CWD: does not exist
self.assertRaises(RuntimeError, Subprocess, node, self.assertRaises(OSError, Subprocess, node,
'/bin/sleep', cwd = self.nofile) '/bin/sleep', cwd = self.nofile)
# Exec failure # Exec failure
self.assertRaises(RuntimeError, Subprocess, node, self.nofile) self.assertRaises(OSError, Subprocess, node, self.nofile)
# Test that the environment is cleared: sleep should not be found # Test that the environment is cleared: sleep should not be found
self.assertRaises(RuntimeError, Subprocess, node, self.assertRaises(OSError, Subprocess, node,
'sleep', env = {'PATH': ''}) 'sleep', env = {'PATH': ''})
# Argv # Argv
self.assertRaises(RuntimeError, Subprocess, node, 'true; false') self.assertRaises(OSError, Subprocess, node, 'true; false')
self.assertEquals(Subprocess(node, 'true').wait(), 0) self.assertEquals(Subprocess(node, 'true').wait(), 0)
self.assertEquals(Subprocess(node, 'true; false', shell = True).wait(), self.assertEquals(Subprocess(node, 'true; false', shell = True).wait(),
1) 1)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment