daemon: avoid raising UnboundLocalError to callers

If something in the daemon_start() function fails before server variable
is initialized, we get the following exception:

UnboundLocalError: local variable 'server' referenced before assignment

We should not attempt to close connections or kill all threads for a
daemon that failed to start (or that hasn't even reached the moment of
the start).

Closes-Bug: #1465350
Change-Id: I7769e40c13e3bd740d5b8a949a61d1bcc127f137
This commit is contained in:
Ihar Hrachyshka 2015-06-15 18:53:54 +02:00
parent 293def2d23
commit e0bf7767da
2 changed files with 57 additions and 38 deletions

View File

@ -90,6 +90,7 @@ def daemon_start(config, filters):
manager_cls = get_manager_class(config, filters) manager_cls = get_manager_class(config, filters)
manager = manager_cls(address=socket_path) manager = manager_cls(address=socket_path)
server = manager.get_server() server = manager.get_server()
try:
# allow everybody to connect to the socket # allow everybody to connect to the socket
rw_rw_rw_ = (stat.S_IRUSR | stat.S_IWUSR | rw_rw_rw_ = (stat.S_IRUSR | stat.S_IWUSR |
stat.S_IRGRP | stat.S_IWGRP | stat.S_IRGRP | stat.S_IWGRP |
@ -114,11 +115,11 @@ def daemon_start(config, filters):
server.serve_forever() server.serve_forever()
finally: finally:
conn = server.listener conn = server.listener
# This will break accept() loop with EOFError if it was not in the main # This will break accept() loop with EOFError if it was not in the
# thread (as in Python 3.x) # main thread (as in Python 3.x)
conn.close() conn.close()
# Closing all currently connected client sockets for reading to break # Closing all currently connected client sockets for reading to
# worker threads blocked on recv() # break worker threads blocked on recv()
for cl_conn in conn.get_accepted(): for cl_conn in conn.get_accepted():
try: try:
cl_conn.half_close() cl_conn.half_close()
@ -130,6 +131,7 @@ def daemon_start(config, filters):
if thread.daemon: if thread.daemon:
LOG.debug("Joining thread %s", thread) LOG.debug("Joining thread %s", thread)
thread.join() thread.join()
finally:
LOG.debug("Removing temporary directory %s", temp_dir) LOG.debug("Removing temporary directory %s", temp_dir)
shutil.rmtree(temp_dir) shutil.rmtree(temp_dir)

View File

@ -24,6 +24,7 @@ from six import moves
import testtools import testtools
from oslo_rootwrap import cmd from oslo_rootwrap import cmd
from oslo_rootwrap import daemon
from oslo_rootwrap import filters from oslo_rootwrap import filters
from oslo_rootwrap import wrapper from oslo_rootwrap import wrapper
@ -583,3 +584,19 @@ class RunOneCommandTestCase(testtools.TestCase):
def test_negative_returncode(self): def test_negative_returncode(self):
self._test_returncode_helper(-1, 129) self._test_returncode_helper(-1, 129)
class DaemonCleanupException(Exception):
pass
class DaemonCleanupTestCase(testtools.TestCase):
@mock.patch('os.chmod')
@mock.patch('shutil.rmtree')
@mock.patch('tempfile.mkdtemp')
@mock.patch('multiprocessing.managers.BaseManager.get_server',
side_effect=DaemonCleanupException)
def test_daemon_no_cleanup_for_uninitialized_server(self, gs, *args):
self.assertRaises(DaemonCleanupException, daemon.daemon_start,
config=None, filters=None)