Source code for ironic.command.conductor
#
# Copyright 2013 Hewlett-Packard Development Company, L.P.
# All Rights Reserved.
#
# 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.
"""
The Ironic Management Service
"""
import sys
from oslo_config import cfg
from oslo_log import log
from oslo_service import service
from ironic.command import utils as command_utils
from ironic.common import service as ironic_service
from ironic.common import utils
from ironic.conductor import rpc_service
CONF = cfg.CONF
LOG = log.getLogger(__name__)
[docs]
def warn_about_unsafe_shred_parameters(conf):
iterations = conf.deploy.shred_random_overwrite_iterations
overwrite_with_zeros = conf.deploy.shred_final_overwrite_with_zeros
if iterations == 0 and overwrite_with_zeros is False:
LOG.warning('With shred_random_overwrite_iterations set to 0 and '
'shred_final_overwrite_with_zeros set to False, disks '
'may NOT be shredded at all, unless they support ATA '
'Secure Erase. This is a possible SECURITY ISSUE!')
[docs]
def warn_about_sqlite():
# We are intentionally calling the helper here to ensure it caches
# for all future calls.
if utils.is_ironic_using_sqlite():
LOG.warning('Ironic has been configured to utilize SQLite. '
'This has some restrictions and impacts. You must run '
'as as a single combined ironic process, and some '
'internal mechanisms do not execute such as the hash '
'ring will remain static and the conductor\'s '
'``last_updated`` field will also not update. This is '
'in order to minimize database locking issues present '
'as a result of SQLAlchemy 2.0 and the removal of '
'autocommit support.')
[docs]
def warn_about_max_wait_parameters(conf):
max_wait = conf.conductor.max_conductor_wait_step_seconds
max_deploy_timeout = conf.conductor.deploy_callback_timeout
max_clean_timeout = conf.conductor.clean_callback_timeout
error_with = None
if max_wait >= max_deploy_timeout:
error_with = 'deploy_callback_timeout'
if max_wait >= max_clean_timeout:
error_with = 'clean_callback_timeout'
if error_with:
LOG.warning('The [conductor]max_conductor_wait_step_seconds '
'configuration parameter exceeds the value of '
'[conductor]%s, which could create a condition where '
'tasks may timeout. Ironic recommends a low default '
'value for [conductor]max_conductor_wait_step_seconds '
'please re-evaluate your configuration.', error_with)
[docs]
def warn_about_inconsistent_kernel_ramdisk(conf):
"""Check consistency of configurations around kernels and ramdisks
This method logs a warning if any of the paired
CONF.conductor.*_ramdisk_by_arch and CONF.conductor.*_kernel_by_arch
config dictionaries are inconsistent -- such as having a kernel configured
for aarch64 without having a ramdisk configured.
:param conf: an ironic.conf.CONF oslo.config object
:returns: None
"""
config_pairs = [
('deploy_kernel_by_arch', 'deploy_ramdisk_by_arch', 'provisioning'),
('rescue_kernel_by_arch', 'rescue_ramdisk_by_arch', 'rescue'),
]
for kernel_opt, ramdisk_opt, operation in config_pairs:
kernel_dict = getattr(conf.conductor, kernel_opt)
ramdisk_dict = getattr(conf.conductor, ramdisk_opt)
kernel_arches = set(kernel_dict.keys())
ramdisk_arches = set(ramdisk_dict.keys())
kernel_only = kernel_arches - ramdisk_arches
ramdisk_only = ramdisk_arches - kernel_arches
if kernel_only:
LOG.warning('The [conductor]%s configuration has entries for '
'architectures %s that are missing from '
'[conductor]%s. This may result in failed %s '
'operations for these architectures.',
kernel_opt, ', '.join(sorted(kernel_only)),
ramdisk_opt, operation)
if ramdisk_only:
LOG.warning('The [conductor]%s configuration has entries for '
'architectures %s that are missing from '
'[conductor]%s. This may result in failed %s '
'operations for these architectures.',
ramdisk_opt, ', '.join(sorted(ramdisk_only)),
kernel_opt, operation)
[docs]
def issue_startup_warnings(conf):
warn_about_unsafe_shred_parameters(conf)
warn_about_sqlite()
warn_about_max_wait_parameters(conf)
warn_about_inconsistent_kernel_ramdisk(conf)
[docs]
def main():
# NOTE(lucasagomes): Safeguard to prevent 'ironic.conductor.manager'
# from being imported prior to the configuration options being loaded.
# If this happened, the periodic decorators would always use the
# default values of the options instead of the configured ones. For
# more information see: https://bugs.launchpad.net/ironic/+bug/1562258
# and https://bugs.launchpad.net/ironic/+bug/1279774.
assert 'ironic.conductor.manager' not in sys.modules
# Parse config file and command line options, then start logging
ironic_service.prepare_service('ironic_conductor', sys.argv)
ironic_service.ensure_rpc_transport(CONF)
mgr = rpc_service.RPCService(CONF.host,
'ironic.conductor.manager',
'ConductorManager')
issue_startup_warnings(CONF)
# Ultimately this returns a ServiceLauncher class which has a _manager
# object (cotyledon), and where launch_service has been invoked which
# adds an instance of the service to the _manager object.
launcher = service.launch(CONF, mgr, restart_method='mutate')
# The approach above also is for a single application where we then start a
# worker process.
# TODO(TheJulia): At this location, we're missing signaling, but where
# the signaling needs to be is past a spawn() call triggered by wait().
# What signaling here would make sense is signal handling to force this
# process to exit.
# TODO(TheJulia): So, the tl;dr of execution is wait() triggers the
# process launch, fork, and then other actions.
# This is in cotyledon where the Service Manager run() method is called
# https://github.com/sileht/cotyledon/blob/main/cotyledon/_service_manager.py#L240
# which then will wait forever and never return.
# Set override signals.
command_utils.handle_signal()
# Start the processes!
sys.exit(launcher.wait())
if __name__ == '__main__':
sys.exit(main())