mirror of
https://opendev.org/openstack/ironic.git
synced 2026-01-11 19:57:20 +00:00
Implements the foundational infrastructure for a new standalone networking service that can operate independently of the main ironic conductor. This commit establishes the service skeleton with: - RPC API layer with oslo.messaging integration for remote calls - Public API interface for conductor/API to interact with the service - RPC service implementation for handling network requests - Stub networking manager with method signatures (implementation added in subsequent commit) - Service entry point (ironic-networking command) for deployment - Configuration options for service behavior and networking backend - Infrastructure and packaging changes for the new service The manager includes stub implementations that raise NetworkError, with the full implementation of network operations, driver framework and switch drivers are added in subsequence commits. Related-Bug: 2113769 Assisted-by: Claude/sonnet-4.5 Change-Id: I351c7afe96cbcebd6b2e2bb5f0b4f17b5d804ceb Signed-off-by: Allain Legacy <alegacy@redhat.com>
156 lines
5.5 KiB
Python
156 lines
5.5 KiB
Python
# Copyright 2014 Rackspace, 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_log import log
|
|
|
|
from ironic.common import exception
|
|
|
|
LOG = log.getLogger(__name__)
|
|
|
|
# Network Types
|
|
IDLE_NETWORK = 'idle'
|
|
RESCUING_NETWORK = 'rescuing'
|
|
CLEANING_NETWORK = 'cleaning'
|
|
SERVICING_NETWORK = 'servicing'
|
|
INSPECTION_NETWORK = 'inspection'
|
|
PROVISIONING_NETWORK = 'provisioning'
|
|
TENANT_NETWORK = 'tenant'
|
|
|
|
|
|
def get_node_vif_ids(task):
|
|
"""Get all VIF ids for a node.
|
|
|
|
This function does not handle multi node operations.
|
|
|
|
:param task: a TaskManager instance.
|
|
:returns: A dict of Node's neutron ports where keys are
|
|
'ports' & 'portgroups' and the values are dict of UUIDs
|
|
and their associated VIFs, e.g.
|
|
|
|
::
|
|
|
|
{'ports': {'port.uuid': vif.id},
|
|
'portgroups': {'portgroup.uuid': vif.id}}
|
|
"""
|
|
vifs = {}
|
|
portgroup_vifs = {}
|
|
port_vifs = {}
|
|
for portgroup in task.portgroups:
|
|
vif = task.driver.network.get_current_vif(task, portgroup)
|
|
if vif:
|
|
portgroup_vifs[portgroup.uuid] = vif
|
|
vifs['portgroups'] = portgroup_vifs
|
|
for port in task.ports:
|
|
vif = task.driver.network.get_current_vif(task, port)
|
|
if vif:
|
|
port_vifs[port.uuid] = vif
|
|
vifs['ports'] = port_vifs
|
|
return vifs
|
|
|
|
|
|
def get_portgroup_by_id(task, portgroup_id):
|
|
"""Lookup a portgroup by ID on a task object.
|
|
|
|
:param task: a TaskManager instance
|
|
:param portgroup_id: ID of the portgroup.
|
|
:returns: A Portgroup object or None.
|
|
"""
|
|
for portgroup in task.portgroups:
|
|
if portgroup.id == portgroup_id:
|
|
return portgroup
|
|
|
|
|
|
def get_ports_by_portgroup_id(task, portgroup_id):
|
|
"""Lookup ports by their portgroup ID on a task object.
|
|
|
|
:param task: a TaskManager instance
|
|
:param portgroup_id: ID of the portgroup.
|
|
:returns: A list of Port objects.
|
|
"""
|
|
return [port for port in task.ports if port.portgroup_id == portgroup_id]
|
|
|
|
|
|
def get_physnets_for_node(task):
|
|
"""Return the set of physical networks for a node.
|
|
|
|
Returns the set of physical networks associated with a node's ports. The
|
|
physical network None is excluded from the set.
|
|
|
|
:param task: a TaskManager instance
|
|
:returns: A set of physical networks.
|
|
"""
|
|
return set(port.physical_network
|
|
for port in task.ports
|
|
if port.physical_network is not None)
|
|
|
|
|
|
def get_physnets_by_portgroup_id(task, portgroup_id, exclude_port=None):
|
|
"""Return the set of physical networks associated with a portgroup.
|
|
|
|
:param task: a TaskManager instance.
|
|
:param portgroup_id: ID of the portgroup.
|
|
:param exclude_port: A Port object to exclude from the determination of the
|
|
portgroup's physical network, or None.
|
|
:returns: The set of physical networks associated with the portgroup. The
|
|
set will contain zero or one physical networks.
|
|
:raises: PortgroupPhysnetInconsistent if the portgroup's ports are not
|
|
assigned the same physical network.
|
|
"""
|
|
pg_ports = get_ports_by_portgroup_id(task, portgroup_id)
|
|
if exclude_port is not None and 'id' in exclude_port:
|
|
exclude_port_id = exclude_port.id
|
|
else:
|
|
exclude_port_id = None
|
|
pg_physnets = set(port.physical_network
|
|
for port in pg_ports
|
|
if port.id != exclude_port_id)
|
|
# Sanity check: all ports should have the same physical network.
|
|
if len(pg_physnets) > 1:
|
|
portgroup = get_portgroup_by_id(task, portgroup_id)
|
|
raise exception.PortgroupPhysnetInconsistent(
|
|
portgroup=portgroup.uuid, physical_networks=", ".join(pg_physnets))
|
|
return pg_physnets
|
|
|
|
|
|
def remove_vifs_from_node(task):
|
|
"""Remove all vif attachment records from a node.
|
|
|
|
:param task: a TaskManager instance.
|
|
"""
|
|
vifs = task.driver.network.vif_list(task)
|
|
for vif_entry in vifs:
|
|
vif = vif_entry.get('id')
|
|
if not vif:
|
|
LOG.warning('Node %(node)s has an incorrect VIF entry: %(found)s. '
|
|
'This entry lacks an "id" field and is thus '
|
|
'unsupported.',
|
|
{'node': task.node.uuid,
|
|
'found': vif_entry})
|
|
continue
|
|
try:
|
|
task.driver.network.vif_detach(task, vif)
|
|
except exception.VifNotAttached:
|
|
LOG.warning('While removing records of VIF attachments from node '
|
|
'%(node)s, we received indication that VIF %(vif)s is '
|
|
'no longer attached. This should not happen under '
|
|
'normal circumstances.',
|
|
{'node': task.node.uuid,
|
|
'vif': vif})
|
|
|
|
except exception.NetworkError as e:
|
|
LOG.error('For node %(node)s, an error occurred while removing a '
|
|
'VIF record for VIF %(vif)s. Error: %(error)s',
|
|
{'node': task.node.uuid,
|
|
'vif': vif,
|
|
'error': e})
|