loguserdata: handle exceptions from running the userdata

- log the exceptions so the user can diagnose issues.

bug 1154641
Change-Id: Ic085c9f062255a9fa44b3e31b464c9ebd19a947c
Signed-off-by: Angus Salkeld <asalkeld@redhat.com>
This commit is contained in:
Angus Salkeld 2013-03-14 14:21:55 +11:00
parent fbea931550
commit 655e377d45
2 changed files with 66 additions and 7 deletions

View File

@ -1,5 +1,6 @@
#!/usr/bin/env python #!/usr/bin/env python
import errno
import datetime import datetime
import pkg_resources import pkg_resources
import os import os
@ -25,8 +26,19 @@ def create_log(log_path):
def call(args, logger): def call(args, logger):
logger.write('%s\n' % ' '.join(args)) logger.write('%s\n' % ' '.join(args))
logger.flush() logger.flush()
p = subprocess.Popen(args, stdout=logger, stderr=logger) try:
p.wait() p = subprocess.Popen(args, stdout=logger, stderr=logger)
p.wait()
except OSError as ex:
if ex.errno == errno.ENOEXEC:
logger.write('Userdata empty or not executable: %s\n' % str(ex))
return os.EX_OK
else:
logger.write('OS error running userdata: %s\n' % str(ex))
return os.EX_OSERR
except Exception as ex:
logger.write('Unknown error running userdata: %s\n' % str(ex))
return os.EX_SOFTWARE
return p.returncode return p.returncode

View File

@ -12,7 +12,7 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import errno
import mox import mox
import os import os
import pkg_resources import pkg_resources
@ -23,7 +23,7 @@ import StringIO
from nose.plugins.attrib import attr from nose.plugins.attrib import attr
import heat.cloudinit.loguserdata as loguserdata from heat.cloudinit import loguserdata
class FakeCiVersion(): class FakeCiVersion():
@ -118,14 +118,62 @@ class LoguserdataTest(unittest.TestCase):
loguserdata.main(log) loguserdata.main(log)
self.m.VerifyAll() self.m.VerifyAll()
def test_main_fails(self): def test_main_script_empty(self):
log = StringIO.StringIO() log = StringIO.StringIO()
pkg_resources.get_distribution('cloud-init').AndReturn(
FakeCiVersion('0.7.0'))
os.chmod('/var/lib/heat-cfntools/cfn-userdata', 0700).AndReturn(None)
subprocess.Popen(
['/var/lib/heat-cfntools/cfn-userdata'],
stderr=log,
stdout=log).AndRaise(OSError(errno.ENOEXEC, "empty script"))
self.m.ReplayAll()
self.assertEqual(None, loguserdata.main(log))
self.m.VerifyAll()
def test_main_os_error(self):
log = StringIO.StringIO()
pkg_resources.get_distribution('cloud-init').AndReturn(
FakeCiVersion('0.7.0'))
os.chmod('/var/lib/heat-cfntools/cfn-userdata', 0700).AndReturn(None)
subprocess.Popen(
['/var/lib/heat-cfntools/cfn-userdata'],
stderr=log,
stdout=log).AndRaise(OSError(errno.ENOENT, "no such file"))
self.m.ReplayAll()
self.assertEqual(os.EX_OSERR, loguserdata.main(log))
self.m.VerifyAll()
def test_main_error_other(self):
log = StringIO.StringIO()
pkg_resources.get_distribution('cloud-init').AndReturn(
FakeCiVersion('0.7.0'))
os.chmod('/var/lib/heat-cfntools/cfn-userdata', 0700).AndReturn(None)
subprocess.Popen(
['/var/lib/heat-cfntools/cfn-userdata'],
stderr=log,
stdout=log).AndRaise(IOError("read failed"))
self.m.ReplayAll()
self.assertEqual(os.EX_SOFTWARE, loguserdata.main(log))
self.m.VerifyAll()
def test_main_fails(self):
log = StringIO.StringIO()
#fail on ci version #fail on ci version
pkg_resources.get_distribution('cloud-init').AndReturn( pkg_resources.get_distribution('cloud-init').AndReturn(
FakeCiVersion('0.5.0')) FakeCiVersion('0.5.0'))
#fail on execute cfn-userdata #fail on execute cfn-userdata
pkg_resources.get_distribution('cloud-init').AndReturn( pkg_resources.get_distribution('cloud-init').AndReturn(
FakeCiVersion('0.7.0')) FakeCiVersion('0.7.0'))
@ -139,5 +187,4 @@ class LoguserdataTest(unittest.TestCase):
self.m.ReplayAll() self.m.ReplayAll()
self.assertEqual(-1, loguserdata.main(log)) self.assertEqual(-1, loguserdata.main(log))
self.assertEqual(-2, loguserdata.main(log)) self.assertEqual(-2, loguserdata.main(log))
self.m.VerifyAll() self.m.VerifyAll()