Upgrade check command - add support for 3rd party checks
CLI command "neutron-status upgrade check" now can load checks from 3rd party and stadium projects using entry_points. Such additional checks should be available under "neutron.status.upgrade.checks" namespace in entry_points. Change-Id: I416abbcdd7b397ec6acd2827129d7a56095ea63d Story: 2003657
This commit is contained in:
parent
5ad2bfdbfe
commit
f5b01e0e12
@ -65,6 +65,7 @@ the developer guide includes information about Neutron testing infrastructure.
|
|||||||
neutron_api
|
neutron_api
|
||||||
client_command_extensions
|
client_command_extensions
|
||||||
alembic_migrations
|
alembic_migrations
|
||||||
|
upgrade_checks
|
||||||
testing/index
|
testing/index
|
||||||
|
|
||||||
Neutron Internals
|
Neutron Internals
|
||||||
|
59
doc/source/contributor/upgrade_checks.rst
Normal file
59
doc/source/contributor/upgrade_checks.rst
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
..
|
||||||
|
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.
|
||||||
|
|
||||||
|
|
||||||
|
Convention for heading levels in Neutron devref:
|
||||||
|
======= Heading 0 (reserved for the title in a document)
|
||||||
|
------- Heading 1
|
||||||
|
~~~~~~~ Heading 2
|
||||||
|
+++++++ Heading 3
|
||||||
|
''''''' Heading 4
|
||||||
|
(Avoid deeper levels because they do not render well.)
|
||||||
|
|
||||||
|
.. _upgrade_checks:
|
||||||
|
|
||||||
|
Upgrade checks
|
||||||
|
==============
|
||||||
|
|
||||||
|
Introduction
|
||||||
|
------------
|
||||||
|
|
||||||
|
CLI tool ``neutron-status upgrade check`` contains checks which perform a
|
||||||
|
release-specific readiness check before restarting services with new code.
|
||||||
|
For more details see `neutron-status command-line client
|
||||||
|
</cli/neutron-status.html>`_ page.
|
||||||
|
|
||||||
|
3rd party plugins checks
|
||||||
|
------------------------
|
||||||
|
|
||||||
|
Neutron upgrade checks script allows to add checks by stadium and 3rd party
|
||||||
|
projects.
|
||||||
|
The ``neutron-status`` script detects which sub-projects have been installed by
|
||||||
|
enumerating the ``neutron.status.upgrade.checks`` entrypoints. For more details
|
||||||
|
see the `Entry Points section of Contributing extensions to Neutron
|
||||||
|
<contribute.html#entry-points>`_.
|
||||||
|
Checks can be run in random order and should be independent from each other.
|
||||||
|
|
||||||
|
The recommended entry point name is a repository name: For example,
|
||||||
|
'neutron-fwaas' for FWaaS and 'networking-sfc' for SFC:
|
||||||
|
|
||||||
|
.. code-block:: ini
|
||||||
|
|
||||||
|
neutron.status.upgrade.checks =
|
||||||
|
neutron-fwaas = neutron_fwaas.upgrade.checks:Checks
|
||||||
|
|
||||||
|
Entrypoint should be class which inherits from
|
||||||
|
``neutron.cmd.upgrade_checks.base.BaseChecks``.
|
||||||
|
|
||||||
|
An example of a checks class can be found in
|
||||||
|
``neutron.cmd.upgrade_checks.checks.CoreChecks``.
|
@ -12,13 +12,35 @@
|
|||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
|
from neutron_lib.utils import runtime
|
||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
|
from oslo_log import log as logging
|
||||||
from oslo_upgradecheck import upgradecheck
|
from oslo_upgradecheck import upgradecheck
|
||||||
|
|
||||||
from neutron._i18n import _
|
CHECKS_ENTRYPOINTS = 'neutron.status.upgrade.checks'
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class Checks(upgradecheck.UpgradeCommands):
|
def load_checks():
|
||||||
|
checks = []
|
||||||
|
ns_plugin = runtime.NamespacedPlugins(CHECKS_ENTRYPOINTS)
|
||||||
|
# TODO(slaweq): stop using private attribute of runtime.NamespacedPlugins
|
||||||
|
# class when it will provide some better way to access extensions
|
||||||
|
for module_name, module in ns_plugin._extensions.items():
|
||||||
|
try:
|
||||||
|
project_checks_class = module.entry_point.load()
|
||||||
|
project_checks = project_checks_class().get_checks()
|
||||||
|
if project_checks:
|
||||||
|
checks.append(project_checks)
|
||||||
|
except Exception as e:
|
||||||
|
LOG.exception("Checks class %(entrypoint)s failed to load. "
|
||||||
|
"Error: %(err)s",
|
||||||
|
{'entrypoint': module_name, 'err': e})
|
||||||
|
continue
|
||||||
|
return tuple(checks)
|
||||||
|
|
||||||
|
|
||||||
|
class Checker(upgradecheck.UpgradeCommands):
|
||||||
|
|
||||||
"""Various upgrade checks should be added as separate methods in this class
|
"""Various upgrade checks should be added as separate methods in this class
|
||||||
and added to _upgrade_checks tuple.
|
and added to _upgrade_checks tuple.
|
||||||
@ -29,11 +51,6 @@ class Checks(upgradecheck.UpgradeCommands):
|
|||||||
like the database schema migrations.
|
like the database schema migrations.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def _check_nothing(self):
|
|
||||||
# NOTE(slaweq) This is only example Noop check, it can be removed when
|
|
||||||
# some real check methods will be added
|
|
||||||
return upgradecheck.Result(upgradecheck.Code.SUCCESS)
|
|
||||||
|
|
||||||
# The format of the check functions is to return an
|
# The format of the check functions is to return an
|
||||||
# oslo_upgradecheck.upgradecheck.Result
|
# oslo_upgradecheck.upgradecheck.Result
|
||||||
# object with the appropriate
|
# object with the appropriate
|
||||||
@ -41,13 +58,9 @@ class Checks(upgradecheck.UpgradeCommands):
|
|||||||
# If the check hits warnings or failures then those should be stored
|
# If the check hits warnings or failures then those should be stored
|
||||||
# in the returned Result's "details" attribute. The
|
# in the returned Result's "details" attribute. The
|
||||||
# summary will be rolled up at the end of the check() method.
|
# summary will be rolled up at the end of the check() method.
|
||||||
_upgrade_checks = (
|
_upgrade_checks = load_checks()
|
||||||
# Check nothing
|
|
||||||
# In the future there should be some real checks added here
|
|
||||||
(_('Check nothing'), _check_nothing),
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
return upgradecheck.main(
|
return upgradecheck.main(
|
||||||
cfg.CONF, project='neutron', upgrade_command=Checks())
|
cfg.CONF, project='neutron', upgrade_command=Checker())
|
||||||
|
0
neutron/cmd/upgrade_checks/__init__.py
Normal file
0
neutron/cmd/upgrade_checks/__init__.py
Normal file
36
neutron/cmd/upgrade_checks/base.py
Normal file
36
neutron/cmd/upgrade_checks/base.py
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
# Copyright 2018 Red Hat Inc.
|
||||||
|
#
|
||||||
|
# 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 abc
|
||||||
|
|
||||||
|
import six
|
||||||
|
|
||||||
|
|
||||||
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
|
class BaseChecks(object):
|
||||||
|
|
||||||
|
"""Base class providing upgrade checks.
|
||||||
|
|
||||||
|
Stadium projects which want to provide their own upgrade checks to
|
||||||
|
neutron-status CLI tool should inherit from this class.
|
||||||
|
|
||||||
|
Each check method have to accept neutron.cmd.status.Checker
|
||||||
|
class as an argument because all checkes will be run in context of
|
||||||
|
this class.
|
||||||
|
"""
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
def get_checks(self):
|
||||||
|
"""Get tuple with check methods and check names to run."""
|
||||||
|
pass
|
33
neutron/cmd/upgrade_checks/checks.py
Normal file
33
neutron/cmd/upgrade_checks/checks.py
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
# Copyright 2018 Red Hat Inc.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
from oslo_upgradecheck import upgradecheck
|
||||||
|
|
||||||
|
from neutron._i18n import _
|
||||||
|
from neutron.cmd.upgrade_checks import base
|
||||||
|
|
||||||
|
|
||||||
|
class CoreChecks(base.BaseChecks):
|
||||||
|
|
||||||
|
def get_checks(self):
|
||||||
|
return (
|
||||||
|
(_("Check nothing"), self.noop_check)
|
||||||
|
)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def noop_check(checker):
|
||||||
|
# NOTE(slaweq) This is only example Noop check, it can be removed when
|
||||||
|
# some real check methods will be added
|
||||||
|
return upgradecheck.Result(
|
||||||
|
upgradecheck.Code.SUCCESS, _("Always succeed (placeholder)"))
|
@ -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.
|
||||||
|
|
||||||
from oslo_upgradecheck.upgradecheck import Code
|
import mock
|
||||||
|
|
||||||
from neutron.cmd import status
|
from neutron.cmd import status
|
||||||
from neutron.tests import base
|
from neutron.tests import base
|
||||||
@ -20,10 +20,19 @@ from neutron.tests import base
|
|||||||
|
|
||||||
class TestUpgradeChecks(base.BaseTestCase):
|
class TestUpgradeChecks(base.BaseTestCase):
|
||||||
|
|
||||||
def setUp(self):
|
def test_load_checks(self):
|
||||||
super(TestUpgradeChecks, self).setUp()
|
expected_checks = tuple(
|
||||||
self.cmd = status.Checks()
|
("test check", "test_check_method"))
|
||||||
|
checks_class_1 = mock.MagicMock()
|
||||||
def test__check_nothing(self):
|
checks_class_1.entry_point.load()().get_checks.return_value = (
|
||||||
check_result = self.cmd._check_nothing()
|
expected_checks)
|
||||||
self.assertEqual(Code.SUCCESS, check_result.code)
|
checks_class_2 = mock.MagicMock()
|
||||||
|
checks_class_2.entry_point.load()().get_checks.return_value = None
|
||||||
|
with mock.patch(
|
||||||
|
"neutron_lib.utils.runtime.NamespacedPlugins"
|
||||||
|
) as namespace_plugins_mock:
|
||||||
|
namespace_plugins = namespace_plugins_mock.return_value
|
||||||
|
namespace_plugins._extensions = {
|
||||||
|
"tests": checks_class_1,
|
||||||
|
"no-checks-class": checks_class_2}
|
||||||
|
self.assertEqual((expected_checks,), status.load_checks())
|
||||||
|
0
neutron/tests/unit/cmd/upgrade_checks/__init__.py
Normal file
0
neutron/tests/unit/cmd/upgrade_checks/__init__.py
Normal file
33
neutron/tests/unit/cmd/upgrade_checks/test_checks.py
Normal file
33
neutron/tests/unit/cmd/upgrade_checks/test_checks.py
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
# Copyright 2018 Red Hat Inc.
|
||||||
|
#
|
||||||
|
# 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 mock
|
||||||
|
from oslo_upgradecheck.upgradecheck import Code
|
||||||
|
|
||||||
|
from neutron.cmd.upgrade_checks import checks
|
||||||
|
from neutron.tests import base
|
||||||
|
|
||||||
|
|
||||||
|
class TestChecks(base.BaseTestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestChecks, self).setUp()
|
||||||
|
self.checks = checks.CoreChecks()
|
||||||
|
|
||||||
|
def test_get_checks_list(self):
|
||||||
|
self.assertIsInstance(self.checks.get_checks(), tuple)
|
||||||
|
|
||||||
|
def test_noop_check(self):
|
||||||
|
check_result = checks.CoreChecks.noop_check(mock.Mock())
|
||||||
|
self.assertEqual(Code.SUCCESS, check_result.code)
|
@ -6,6 +6,9 @@ features:
|
|||||||
New framework for ``neutron-status upgrade check`` command is added.
|
New framework for ``neutron-status upgrade check`` command is added.
|
||||||
This framework allows adding various checks which can be run before a
|
This framework allows adding various checks which can be run before a
|
||||||
Neutron upgrade to ensure if the upgrade can be performed safely.
|
Neutron upgrade to ensure if the upgrade can be performed safely.
|
||||||
|
Stadium and 3rd party projects can register their own checks to this new
|
||||||
|
neutron-status CLI tool using entrypoints in
|
||||||
|
``neutron.status.upgrade.checks`` namespace.
|
||||||
upgrade:
|
upgrade:
|
||||||
- |
|
- |
|
||||||
Operator can now use new CLI tool ``neutron-status upgrade check``
|
Operator can now use new CLI tool ``neutron-status upgrade check``
|
||||||
|
@ -231,7 +231,8 @@ neutron.objects =
|
|||||||
SubnetServiceType = neutron.objects.subnet:SubnetServiceType
|
SubnetServiceType = neutron.objects.subnet:SubnetServiceType
|
||||||
Tag = neutron.objects.tag:Tag
|
Tag = neutron.objects.tag:Tag
|
||||||
Trunk = neutron.objects.trunk:Trunk
|
Trunk = neutron.objects.trunk:Trunk
|
||||||
|
neutron.status.upgrade.checks =
|
||||||
|
neutron = neutron.cmd.upgrade_checks.checks:CoreChecks
|
||||||
|
|
||||||
[extract_messages]
|
[extract_messages]
|
||||||
keywords = _ gettext ngettext l_ lazy_gettext
|
keywords = _ gettext ngettext l_ lazy_gettext
|
||||||
|
Loading…
Reference in New Issue
Block a user