Merge pull request #1 from gnuoy/add-unit-tests

Add first set of unit tests
This commit is contained in:
Dmitrii Shcherbakov 2020-04-24 19:52:17 +03:00 committed by GitHub
commit 2ef02f14d6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 285 additions and 5 deletions

3
.gitignore vendored
View File

@ -1 +1,4 @@
.tox .tox
.stestr/
__pycache__

3
.stestr.conf Normal file
View File

@ -0,0 +1,3 @@
[DEFAULT]
test_path=./unit_tests
top_dir=./

View File

@ -87,7 +87,8 @@ class OSBaseCharm(CharmBase):
return return
missing_relations = [] missing_relations = []
for relation in self.REQUIRED_RELATIONS: for relation in self.REQUIRED_RELATIONS:
if not self.model.get_relation(relation): rel = self.model.get_relation(relation)
if rel is None:
missing_relations.append(relation) missing_relations.append(relation)
if missing_relations: if missing_relations:
self.unit.status = BlockedStatus( self.unit.status = BlockedStatus(
@ -98,7 +99,7 @@ class OSBaseCharm(CharmBase):
# If the check failed the custom check will have set the status. # If the check failed the custom check will have set the status.
if not self.custom_status_check(): if not self.custom_status_check():
return return
except NotImplementedError: except NotImplementedError:
pass pass
if self.state.is_started: if self.state.is_started:
self.unit.status = ActiveStatus('Unit is ready') self.unit.status = ActiveStatus('Unit is ready')
@ -113,7 +114,7 @@ class OSBaseCharm(CharmBase):
_svcs = [] _svcs = []
for svc in self.RESTART_MAP.values(): for svc in self.RESTART_MAP.values():
_svcs.extend(svc) _svcs.extend(svc)
return list(set(_svcs)) return sorted(list(set(_svcs)))
def on_pre_series_upgrade(self, event): def on_pre_series_upgrade(self, event):
_, messages = os_utils.manage_payload_services( _, messages = os_utils.manage_payload_services(

View File

@ -1,5 +1,9 @@
# Lint and unit test requirements # Lint and unit test requirements
flake8>=2.2.4,<=2.4.1 flake8
stestr>=2.2.0
mock>=1.2 mock>=1.2
nose>=1.3.7
coverage>=3.6 coverage>=3.6
# Install netifaces as its a horrible charmhelpers lazy import
netifaces
charmhelpers
git+https://github.com/canonical/operator.git#egg=ops

0
unit_tests/__init__.py Normal file
View File

View File

@ -0,0 +1,269 @@
#!/usr/bin/env python3
# Copyright 2020 Canonical Ltd.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import unittest
from mock import patch, MagicMock
from ops.testing import Harness, _TestingModelBackend
from ops.model import (
ActiveStatus,
BlockedStatus,
MaintenanceStatus,
WaitingStatus,
)
from ops import framework, model
import ops_openstack
class OpenStackTestAPICharm(ops_openstack.OSBaseCharm):
PACKAGES = ['keystone-common']
REQUIRED_RELATIONS = ['shared-db']
RESTART_MAP = {
'/etc/f1.conf': ['apache2'],
'/etc/f2.conf': ['apache2', 'ks-api'],
'/etc/f3.conf': []}
def custom_status_check(self):
if self.model.config.get('custom-check-fail', 'False') == 'True':
self.unit.status = MaintenanceStatus(
'Custom check failed')
return False
else:
return True
class CharmTestCase(unittest.TestCase):
def setUp(self, obj, patches):
super().setUp()
self.patches = patches
self.obj = obj
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 TestOSBaseCharm(CharmTestCase):
PATCHES = [
'add_source',
'apt_update',
'apt_install',
'os_utils']
def setUp(self):
super().setUp(ops_openstack, self.PATCHES)
self.os_utils.manage_payload_services = MagicMock()
self.harness = Harness(
OpenStackTestAPICharm,
meta='''
name: client
requires:
shared-db:
interface: mysql-shared
provides:
ceph-client:
interface: ceph-client
''',
actions='''
pause:
description: pause action
resume:
description: resume action
''')
# BEGIN: Workaround until
# https://github.com/canonical/operator/pull/196 lands
class _TestingOPSModelBackend(_TestingModelBackend):
def relation_ids(self, relation_name):
return self._relation_ids_map.get(relation_name, [])
self.harness._backend = _TestingOPSModelBackend(
self.harness._unit_name)
self.harness._model = model.Model(
self.harness._unit_name,
self.harness._meta,
self.harness._backend)
self.harness._framework = framework.Framework(
":memory:",
self.harness._charm_dir,
self.harness._meta,
self.harness._model)
# END Workaround
def test_init(self):
self.harness.begin()
self.assertFalse(self.harness.charm.state.is_started)
self.assertFalse(self.harness.charm.state.is_paused)
self.assertFalse(self.harness.charm.state.series_upgrade)
def test_install(self):
print(self.harness._backend)
self.harness.begin()
self.harness.charm.on.install.emit()
self.assertFalse(self.add_source.called)
self.apt_update.assert_called_once_with(fatal=True)
self.apt_install.assert_called_once_with(
['keystone-common'],
fatal=True)
def test_install_ppa(self):
self.harness.update_config(
key_values={
'source': 'cloud:myppa',
'key': 'akey'})
self.harness.begin()
self.harness.charm.on.install.emit()
self.add_source.assert_called_once_with('cloud:myppa', 'akey')
self.apt_update.assert_called_once_with(fatal=True)
self.apt_install.assert_called_once_with(
['keystone-common'],
fatal=True)
def test_update_status(self):
self.harness.add_relation('shared-db', 'mysql')
self.harness.begin()
self.harness.charm.state.is_started = True
self.harness.charm.on.update_status.emit()
self.assertEqual(
self.harness.charm.unit.status.message,
'Unit is ready')
self.assertIsInstance(
self.harness.charm.unit.status,
ActiveStatus)
def test_update_status_custom_check_fail(self):
self.harness.update_config(
key_values={
'custom-check-fail': 'True'})
self.harness.add_relation('shared-db', 'mysql')
self.harness.begin()
self.harness.charm.state.is_started = True
self.harness.charm.on.update_status.emit()
self.assertEqual(
self.harness.charm.unit.status.message,
'Custom check failed')
self.assertIsInstance(
self.harness.charm.unit.status,
MaintenanceStatus)
def test_update_status_not_started(self):
self.harness.add_relation('shared-db', 'mysql')
self.harness.begin()
self.harness.charm.on.update_status.emit()
self.assertEqual(
self.harness.charm.unit.status.message,
'Charm configuration in progress')
self.assertIsInstance(
self.harness.charm.unit.status,
WaitingStatus)
def test_update_status_series_upgrade(self):
self.harness.begin()
self.harness.charm.state.series_upgrade = True
self.harness.charm.on_update_status('An Event')
self.assertEqual(
self.harness.charm.unit.status.message,
('Ready for do-release-upgrade and reboot. Set complete when '
'finished.'))
self.assertIsInstance(
self.harness.charm.unit.status,
BlockedStatus)
def test_update_status_series_paused(self):
self.harness.begin()
self.harness.charm.state.is_paused = True
self.harness.charm.on.update_status.emit()
self.assertEqual(
self.harness.charm.unit.status.message,
"Paused. Use 'resume' action to resume normal service.")
self.assertIsInstance(
self.harness.charm.unit.status,
MaintenanceStatus)
def test_update_status_missing_relation(self):
self.harness.begin()
self.harness.charm.on.update_status.emit()
self.assertEqual(
self.harness.charm.unit.status.message,
'Missing relations: shared-db')
self.assertIsInstance(
self.harness.charm.unit.status,
BlockedStatus)
def test_services(self):
self.harness.begin()
self.assertEqual(
self.harness.charm.services(),
['apache2', 'ks-api'])
def test_pre_series_upgrade(self):
self.os_utils.manage_payload_services.return_value = ('a', 'b')
self.harness.begin()
self.assertFalse(self.harness.charm.state.series_upgrade)
self.assertFalse(self.harness.charm.state.is_paused)
self.harness.charm.on.pre_series_upgrade.emit()
self.assertTrue(self.harness.charm.state.series_upgrade)
self.assertTrue(self.harness.charm.state.is_paused)
self.os_utils.manage_payload_services.assert_called_once_with(
'pause',
services=['apache2', 'ks-api'],
charm_func=None)
def test_post_series_upgrade(self):
self.os_utils.manage_payload_services.return_value = ('a', 'b')
self.harness.begin()
self.harness.charm.state.series_upgrade = True
self.harness.charm.state.is_paused = True
self.harness.charm.on.post_series_upgrade.emit()
self.assertFalse(self.harness.charm.state.series_upgrade)
self.assertFalse(self.harness.charm.state.is_paused)
self.os_utils.manage_payload_services.assert_called_once_with(
'resume',
services=['apache2', 'ks-api'],
charm_func=None)
def test_pause(self):
self.os_utils.manage_payload_services.return_value = ('a', 'b')
self.harness.begin()
self.assertFalse(self.harness.charm.state.is_paused)
self.harness.charm.on_pause_action('An Event')
self.assertTrue(self.harness.charm.state.is_paused)
self.os_utils.manage_payload_services.assert_called_once_with(
'pause',
services=['apache2', 'ks-api'],
charm_func=None)
def test_resume(self):
self.os_utils.manage_payload_services.return_value = ('a', 'b')
self.harness.begin()
self.harness.charm.state.is_paused = True
self.harness.charm.on_resume_action('An Event')
self.assertFalse(self.harness.charm.state.is_paused)
self.os_utils.manage_payload_services.assert_called_once_with(
'resume',
services=['apache2', 'ks-api'],
charm_func=None)