Skip to content

Commit

Permalink
CASMCMS-8998: v2: Added more checks to avoid operating on empty lists
Browse files Browse the repository at this point in the history
  • Loading branch information
mharding-hpe committed May 16, 2024
1 parent 6e3060e commit bc8c94f
Show file tree
Hide file tree
Showing 9 changed files with 87 additions and 19 deletions.
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).

## [Unreleased]
### Changed
- Added more checks to avoid operating on empty lists

## [2.17.6] - 2024-04-19
### Fixed
Expand Down
16 changes: 16 additions & 0 deletions src/bos/operators/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -140,6 +140,10 @@ def _get_components(self) -> List[dict]:

def _handle_failed_components(self, components: List[dict]) -> List[dict]:
""" Marks components failed if the retry limits are exceeded """
if not components:
# If we have been passed an empty list, there is nothing to do.
LOGGER.debug("_handle_failed_components: No components to handle")
return []
failed_components = []
good_components = [] # Any component that isn't determined to be in a failed state
for component in components:
Expand All @@ -163,6 +167,10 @@ def _update_database(self, components: List[dict], additional_fields: dict=None)
Updates the BOS database for all components acted on by the operator
Includes updating the last action, attempt count and error
"""
if not components:
# If we have been passed an empty list, there is nothing to do.
LOGGER.debug("_update_database: No components require database updates")
return
data = []
for component in components:
patch = {
Expand Down Expand Up @@ -200,6 +208,10 @@ def _preset_last_action(self, components: List[dict]) -> None:
# e.g. nodes could be powered-on without the correct power-on last action, causing status problems
if not self.name:
return
if not components:
# If we have been passed an empty list, there is nothing to do.
LOGGER.debug("_preset_last_action: No components require database updates")
return
data = []
for component in components:
patch = {
Expand All @@ -221,6 +233,10 @@ def _update_database_for_failure(self, components: List[dict]) -> None:
"""
Updates the BOS database for all components the operator believes have failed
"""
if not components:
# If we have been passed an empty list, there is nothing to do.
LOGGER.debug("_update_database_for_failure: No components require database updates")
return
data = []
for component in components:
patch = {
Expand Down
5 changes: 3 additions & 2 deletions src/bos/operators/configuration.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
#
# MIT License
#
# (C) Copyright 2022 Hewlett Packard Enterprise Development LP
# (C) Copyright 2022, 2024 Hewlett Packard Enterprise Development LP
#
# Permission is hereby granted, free of charge, to any person obtaining a
# copy of this software and associated documentation files (the "Software"),
Expand Down Expand Up @@ -57,7 +57,8 @@ def filters(self):
]

def _act(self, components):
set_cfs(components, enabled=True)
if components:
set_cfs(components, enabled=True)
return components


Expand Down
7 changes: 4 additions & 3 deletions src/bos/operators/power_off_forceful.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
#
# MIT License
#
# (C) Copyright 2021-2023 Hewlett Packard Enterprise Development LP
# (C) Copyright 2021-2024 Hewlett Packard Enterprise Development LP
#
# Permission is hereby granted, free of charge, to any person obtaining a
# copy of this software and associated documentation files (the "Software"),
Expand Down Expand Up @@ -57,8 +57,9 @@ def filters(self):
]

def _act(self, components):
component_ids = [component['id'] for component in components]
pcs.force_off(nodes=component_ids)
if components:
component_ids = [component['id'] for component in components]
pcs.force_off(nodes=component_ids)
return components


Expand Down
7 changes: 4 additions & 3 deletions src/bos/operators/power_off_graceful.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
#
# MIT License
#
# (C) Copyright 2021-2023 Hewlett Packard Enterprise Development LP
# (C) Copyright 2021-2024 Hewlett Packard Enterprise Development LP
#
# Permission is hereby granted, free of charge, to any person obtaining a
# copy of this software and associated documentation files (the "Software"),
Expand Down Expand Up @@ -53,8 +53,9 @@ def filters(self):
]

def _act(self, components):
component_ids = [component['id'] for component in components]
pcs.soft_off(component_ids)
if components:
component_ids = [component['id'] for component in components]
pcs.soft_off(component_ids)
return components


Expand Down
9 changes: 8 additions & 1 deletion src/bos/operators/power_on.py
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,8 @@ def filters(self):
]

def _act(self, components):
if not components:
return components
self._preset_last_action(components)
try:
self._set_bss(components)
Expand All @@ -85,6 +87,10 @@ def _set_bss(self, components, retries=5):
Because the connection to the BSS tokens database can be lost due to
infrequent use, retry up to retries number of times.
"""
if not components:
# If we have been passed an empty list, there is nothing to do.
LOGGER.debug("_set_bss: No components to act on")
return
parameters = defaultdict(set)
sessions = {}
for component in components:
Expand Down Expand Up @@ -128,6 +134,8 @@ def _set_bss(self, components, retries=5):
"desired_state": {"bss_token": token},
"session": sessions[node]})
LOGGER.info('Found %d components that require BSS token updates', len(bss_tokens))
if not bss_tokens:
return
redacted_component_updates = [
{ "id": comp["id"],
"session": comp["session"]
Expand All @@ -136,7 +144,6 @@ def _set_bss(self, components, retries=5):
LOGGER.debug('Updated components (minus desired_state data): {}'.format(redacted_component_updates))
self.bos_client.components.update_components(bss_tokens)


if __name__ == '__main__':
main(PowerOnOperator)

Expand Down
50 changes: 43 additions & 7 deletions src/bos/operators/session_setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -166,20 +166,26 @@ def _get_boot_set_component_list(self, boot_set) -> Set[str]:
self._log(LOGGER.warning, f"No hardware matching role {role_name}")
continue
nodes |= self.inventory.roles[role_name]
if not nodes:
self._log(LOGGER.warning, "After populating node list, before any filtering, no nodes to act upon.")
return nodes
self._log(LOGGER.debug, "Before any limiting or filtering, %d nodes to act upon.", len(nodes))
# Filter out any nodes that do not match the boot set architecture desired; boot sets that do not have a
# specified arch are considered 'X86' nodes.
arch = boot_set.get('arch', 'X86')
nodes = self._apply_arch(nodes, arch)
if not nodes:
return nodes
# Filter to nodes defined by limit
nodes = self._apply_limit(nodes)
if not nodes:
return nodes
# Exclude disabled nodes
include_disabled = self.session_data.get("include_disabled", False)
if not include_disabled:
hsmfilter = HSMState(enabled=True)
nodes = set(hsmfilter._filter(list(nodes)))
nodes = self._apply_tenant_limit(nodes)
nodes = self._apply_include_disabled(nodes)
if not nodes:
self._log(LOGGER.warning, "No nodes were found to act upon.")
return nodes
# If this session is for a tenant, filter out nodes not belonging to this tenant
nodes = self._apply_tenant_limit(nodes)
return nodes

def _apply_arch(self, nodes, arch):
Expand All @@ -204,7 +210,29 @@ def _apply_arch(self, nodes, arch):
if arch == 'X86':
valid_archs.add('UNKNOWN')
hsm_filter = HSMState()
return set(hsm_filter.filter_by_arch(nodes, valid_archs))
nodes = set(hsm_filter.filter_by_arch(nodes, valid_archs))
if not nodes:
self._log(LOGGER.warning, "After filtering for architecture, no nodes remain to act upon.")
else:
self._log(LOGGER.debug, "After filtering for architecture, %d nodes remain to act upon.", len(nodes))
return nodes

def _apply_include_disabled(self, nodes):
"""
If include_disabled is False for this session, filter out any nodes which are disabled in HSM.
If include_disabled is True, return the node list unchanged.
"""
include_disabled = self.session_data.get("include_disabled", False)
if include_disabled:
# Nodes disabled in HSM may be included, so no filtering is required
return nodes
hsmfilter = HSMState(enabled=True)
nodes = set(hsmfilter._filter(list(nodes)))
if not nodes:
self._log(LOGGER.warning, "After removing disabled nodes, no nodes remain to act upon.")
else:
self._log(LOGGER.debug, "After removing disabled nodes, %d nodes remain to act upon.", len(nodes))
return nodes

def _apply_limit(self, nodes):
session_limit = self.session_data.get('limit')
Expand All @@ -230,6 +258,10 @@ def _apply_limit(self, nodes):
limit_nodes = self.inventory[limit]
limit_node_set = op(limit_nodes)
nodes = nodes.intersection(limit_node_set)
if not nodes:
self._log(LOGGER.warning, "After applying limit, no nodes remain to act upon.")
else:
self._log(LOGGER.debug, "After applying limit, %d nodes remain to act upon.", len(nodes))
return nodes

def _apply_tenant_limit(self, nodes):
Expand All @@ -241,6 +273,10 @@ def _apply_tenant_limit(self, nodes):
except InvalidTenantException as e:
raise SessionSetupException(str(e)) from e
nodes = nodes.intersection(tenant_limit)
if not nodes:
self._log(LOGGER.warning, "After applying tenant limit, no nodes remain to act upon.")
else:
self._log(LOGGER.debug, "After applying tenant limit, %d nodes remain to act upon.", len(nodes))
return nodes

def _mark_running(self, component_ids):
Expand Down
9 changes: 6 additions & 3 deletions src/bos/operators/utils/clients/bss.py
Original file line number Diff line number Diff line change
Expand Up @@ -58,13 +58,16 @@ def set_bss(node_set, kernel_params, kernel, initrd, session=None):
communicating with the
Hardware State Manager
'''
if not node_set:
# Cannot simply return if no nodes are specified, as this function
# is intended to return the response object from BSS.
# Accordingly, an Exception is raised.
raise Exception("set_bss called with empty node_set")

session = session or requests_retry_session()
LOGGER.info("Params: {}".format(kernel_params))
url = "%s/bootparameters" % (ENDPOINT)

if not node_set:
return

# Assignment payload
payload = {"hosts": list(node_set),
"params": kernel_params,
Expand Down
1 change: 1 addition & 0 deletions src/bos/operators/utils/clients/hsm.py
Original file line number Diff line number Diff line change
Expand Up @@ -121,6 +121,7 @@ def get_components(node_list, enabled=None) -> dict[str,list[dict]]:
}
"""
if not node_list:
LOGGER.warning("hsm.get_components called with empty node list")
return {'Components': []}
session = requests_retry_session()
try:
Expand Down

0 comments on commit bc8c94f

Please sign in to comment.