Add unit tests for service status
This commit is contained in:
parent
94fd9f7bc1
commit
286111285f
7
.coveragerc
Normal file
7
.coveragerc
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
[report]
|
||||||
|
# Regexes for lines to exclude from consideration
|
||||||
|
exclude_lines =
|
||||||
|
if __name__ == .__main__.:
|
||||||
|
include=
|
||||||
|
hooks/hooks.py
|
||||||
|
hooks/ceph*.py
|
@ -4,5 +4,6 @@
|
|||||||
<pydev_property name="org.python.pydev.PYTHON_PROJECT_INTERPRETER">Default</pydev_property>
|
<pydev_property name="org.python.pydev.PYTHON_PROJECT_INTERPRETER">Default</pydev_property>
|
||||||
<pydev_pathproperty name="org.python.pydev.PROJECT_SOURCE_PATH">
|
<pydev_pathproperty name="org.python.pydev.PROJECT_SOURCE_PATH">
|
||||||
<path>/ceph-osd/hooks</path>
|
<path>/ceph-osd/hooks</path>
|
||||||
|
<path>/ceph-osd/unit_tests</path>
|
||||||
</pydev_pathproperty>
|
</pydev_pathproperty>
|
||||||
</pydev_project>
|
</pydev_project>
|
||||||
|
9
Makefile
9
Makefile
@ -3,9 +3,14 @@ PYTHON := /usr/bin/env python
|
|||||||
|
|
||||||
lint:
|
lint:
|
||||||
@flake8 --exclude hooks/charmhelpers,tests/charmhelpers \
|
@flake8 --exclude hooks/charmhelpers,tests/charmhelpers \
|
||||||
hooks tests unit_tests
|
hooks tests unit_tests
|
||||||
@charm proof
|
@charm proof
|
||||||
|
|
||||||
|
test:
|
||||||
|
@# Bundletester expects unit tests here.
|
||||||
|
@echo Starting unit tests...
|
||||||
|
@$(PYTHON) /usr/bin/nosetests --nologcapture --with-coverage unit_tests
|
||||||
|
|
||||||
functional_test:
|
functional_test:
|
||||||
@echo Starting Amulet tests...
|
@echo Starting Amulet tests...
|
||||||
@juju test -v -p AMULET_HTTP_PROXY,AMULET_OS_VIP --timeout 2700
|
@juju test -v -p AMULET_HTTP_PROXY,AMULET_OS_VIP --timeout 2700
|
||||||
@ -13,7 +18,7 @@ functional_test:
|
|||||||
bin/charm_helpers_sync.py:
|
bin/charm_helpers_sync.py:
|
||||||
@mkdir -p bin
|
@mkdir -p bin
|
||||||
@bzr cat lp:charm-helpers/tools/charm_helpers_sync/charm_helpers_sync.py \
|
@bzr cat lp:charm-helpers/tools/charm_helpers_sync/charm_helpers_sync.py \
|
||||||
> bin/charm_helpers_sync.py
|
> bin/charm_helpers_sync.py
|
||||||
|
|
||||||
sync: bin/charm_helpers_sync.py
|
sync: bin/charm_helpers_sync.py
|
||||||
$(PYTHON) bin/charm_helpers_sync.py -c charm-helpers-hooks.yaml
|
$(PYTHON) bin/charm_helpers_sync.py -c charm-helpers-hooks.yaml
|
||||||
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
1
hooks/update-status
Symbolic link
1
hooks/update-status
Symbolic link
@ -0,0 +1 @@
|
|||||||
|
ceph_hooks.py
|
@ -1 +1 @@
|
|||||||
hooks.py
|
ceph_hooks.py
|
5
setup.cfg
Normal file
5
setup.cfg
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
[nosetests]
|
||||||
|
verbosity=2
|
||||||
|
with-coverage=1
|
||||||
|
cover-erase=1
|
||||||
|
cover-package=hooks
|
@ -0,0 +1,2 @@
|
|||||||
|
import sys
|
||||||
|
sys.path.append('hooks')
|
56
unit_tests/test_status.py
Normal file
56
unit_tests/test_status.py
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
import mock
|
||||||
|
import test_utils
|
||||||
|
|
||||||
|
with mock.patch('utils.get_unit_hostname'):
|
||||||
|
import ceph_hooks as hooks
|
||||||
|
|
||||||
|
TO_PATCH = [
|
||||||
|
'status_set',
|
||||||
|
'config',
|
||||||
|
'ceph',
|
||||||
|
'relation_ids',
|
||||||
|
'relation_get',
|
||||||
|
'related_units',
|
||||||
|
'get_conf',
|
||||||
|
]
|
||||||
|
|
||||||
|
CEPH_MONS = [
|
||||||
|
'ceph/0',
|
||||||
|
'ceph/1',
|
||||||
|
'ceph/2',
|
||||||
|
]
|
||||||
|
|
||||||
|
class ServiceStatusTestCase(test_utils.CharmTestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(ServiceStatusTestCase, self).setUp(hooks, TO_PATCH)
|
||||||
|
self.config.side_effect = self.test_config.get
|
||||||
|
|
||||||
|
def test_assess_status_no_monitor_relation(self):
|
||||||
|
self.relation_ids.return_value = []
|
||||||
|
hooks.assess_status()
|
||||||
|
self.status_set.assert_called_with('blocked', mock.ANY)
|
||||||
|
|
||||||
|
def test_assess_status_monitor_relation_incomplete(self):
|
||||||
|
self.relation_ids.return_value = ['mon:1']
|
||||||
|
self.related_units.return_value = CEPH_MONS
|
||||||
|
self.get_conf.return_value = None
|
||||||
|
hooks.assess_status()
|
||||||
|
self.status_set.assert_called_with('waiting', mock.ANY)
|
||||||
|
|
||||||
|
def test_assess_status_monitor_complete_no_disks(self):
|
||||||
|
self.relation_ids.return_value = ['mon:1']
|
||||||
|
self.related_units.return_value = CEPH_MONS
|
||||||
|
self.get_conf.return_value = 'monitor-bootstrap-key'
|
||||||
|
self.ceph.get_running_osds.return_value = []
|
||||||
|
hooks.assess_status()
|
||||||
|
self.status_set.assert_called_with('blocked', mock.ANY)
|
||||||
|
|
||||||
|
def test_assess_status_monitor_complete_disks(self):
|
||||||
|
self.relation_ids.return_value = ['mon:1']
|
||||||
|
self.related_units.return_value = CEPH_MONS
|
||||||
|
self.get_conf.return_value = 'monitor-bootstrap-key'
|
||||||
|
self.ceph.get_running_osds.return_value = ['12345',
|
||||||
|
'67890']
|
||||||
|
hooks.assess_status()
|
||||||
|
self.status_set.assert_called_with('active', mock.ANY)
|
121
unit_tests/test_utils.py
Normal file
121
unit_tests/test_utils.py
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
import logging
|
||||||
|
import unittest
|
||||||
|
import os
|
||||||
|
import yaml
|
||||||
|
|
||||||
|
from contextlib import contextmanager
|
||||||
|
from mock import patch, MagicMock
|
||||||
|
|
||||||
|
|
||||||
|
def load_config():
|
||||||
|
'''
|
||||||
|
Walk backwords from __file__ looking for config.yaml, load and return the
|
||||||
|
'options' section'
|
||||||
|
'''
|
||||||
|
config = None
|
||||||
|
f = __file__
|
||||||
|
while config is None:
|
||||||
|
d = os.path.dirname(f)
|
||||||
|
if os.path.isfile(os.path.join(d, 'config.yaml')):
|
||||||
|
config = os.path.join(d, 'config.yaml')
|
||||||
|
break
|
||||||
|
f = d
|
||||||
|
|
||||||
|
if not config:
|
||||||
|
logging.error('Could not find config.yaml in any parent directory '
|
||||||
|
'of %s. ' % f)
|
||||||
|
raise Exception
|
||||||
|
|
||||||
|
return yaml.safe_load(open(config).read())['options']
|
||||||
|
|
||||||
|
|
||||||
|
def get_default_config():
|
||||||
|
'''
|
||||||
|
Load default charm config from config.yaml return as a dict.
|
||||||
|
If no default is set in config.yaml, its value is None.
|
||||||
|
'''
|
||||||
|
default_config = {}
|
||||||
|
config = load_config()
|
||||||
|
for k, v in config.iteritems():
|
||||||
|
if 'default' in v:
|
||||||
|
default_config[k] = v['default']
|
||||||
|
else:
|
||||||
|
default_config[k] = None
|
||||||
|
return default_config
|
||||||
|
|
||||||
|
|
||||||
|
class CharmTestCase(unittest.TestCase):
|
||||||
|
|
||||||
|
def setUp(self, obj, patches):
|
||||||
|
super(CharmTestCase, self).setUp()
|
||||||
|
self.patches = patches
|
||||||
|
self.obj = obj
|
||||||
|
self.test_config = TestConfig()
|
||||||
|
self.test_relation = TestRelation()
|
||||||
|
self.patch_all()
|
||||||
|
|
||||||
|
def patch(self, method):
|
||||||
|
_m = patch.object(self.obj, method)
|
||||||
|
mock = _m.start()
|
||||||
|
self.addCleanup(_m.stop)
|
||||||
|
return mock
|
||||||
|
|
||||||
|
def patch_all(self):
|
||||||
|
for method in self.patches:
|
||||||
|
setattr(self, method, self.patch(method))
|
||||||
|
|
||||||
|
|
||||||
|
class TestConfig(object):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self.config = get_default_config()
|
||||||
|
|
||||||
|
def get(self, attr=None):
|
||||||
|
if not attr:
|
||||||
|
return self.get_all()
|
||||||
|
try:
|
||||||
|
return self.config[attr]
|
||||||
|
except KeyError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
def get_all(self):
|
||||||
|
return self.config
|
||||||
|
|
||||||
|
def set(self, attr, value):
|
||||||
|
if attr not in self.config:
|
||||||
|
raise KeyError
|
||||||
|
self.config[attr] = value
|
||||||
|
|
||||||
|
|
||||||
|
class TestRelation(object):
|
||||||
|
|
||||||
|
def __init__(self, relation_data={}):
|
||||||
|
self.relation_data = relation_data
|
||||||
|
|
||||||
|
def set(self, relation_data):
|
||||||
|
self.relation_data = relation_data
|
||||||
|
|
||||||
|
def get(self, attr=None, unit=None, rid=None):
|
||||||
|
if attr is None:
|
||||||
|
return self.relation_data
|
||||||
|
elif attr in self.relation_data:
|
||||||
|
return self.relation_data[attr]
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def patch_open():
|
||||||
|
'''Patch open() to allow mocking both open() itself and the file that is
|
||||||
|
yielded.
|
||||||
|
|
||||||
|
Yields the mock for "open" and "file", respectively.'''
|
||||||
|
mock_open = MagicMock(spec=open)
|
||||||
|
mock_file = MagicMock(spec=file)
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def stub_open(*args, **kwargs):
|
||||||
|
mock_open(*args, **kwargs)
|
||||||
|
yield mock_file
|
||||||
|
|
||||||
|
with patch('__builtin__.open', stub_open):
|
||||||
|
yield mock_open, mock_file
|
Loading…
Reference in New Issue
Block a user