cid 48f50248c2 Self-Service via Runbooks
Adds runbooks; the new API feature that makes it possible for
project members to self-serve maintenance tasks through curated step
lists associated with target nodes via traits.

In addition to basic CRUD support, runbook extends current API flow for
performing manual cleaning and servicing to support runbooks in lieu of
an explicit/arbitrary ``clean_steps`` and ``service_steps`` user-defined
lists.

Demo Video: https://youtu.be/00PJS4SXFYQ
Closes-Bug: #2027690

Change-Id: I43555ef72cb882adcada2ed875fda40eed0dd034
2024-08-02 05:44:29 +01:00

227 lines
7.8 KiB
Python

# -*- encoding: utf-8 -*-
#
# 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.
"""
Utils for testing the API service.
"""
import datetime
import hashlib
import json
from oslo_utils import timeutils
from ironic.api.controllers.v1 import allocation as al_controller
from ironic.api.controllers.v1 import chassis as chassis_controller
from ironic.api.controllers.v1 import deploy_template as dt_controller
from ironic.api.controllers.v1 import node as node_controller
from ironic.api.controllers.v1 import port as port_controller
from ironic.api.controllers.v1 import portgroup as portgroup_controller
from ironic.api.controllers.v1 import runbook as rb_controller
from ironic.api.controllers.v1 import utils as api_utils
from ironic.api.controllers.v1 import volume_connector as vc_controller
from ironic.api.controllers.v1 import volume_target as vt_controller
from ironic.tests.unit.db import utils as db_utils
ADMIN_TOKEN = '4562138218392831'
MEMBER_TOKEN = '4562138218392832'
ADMIN_BODY = {
'access': {
'token': {'id': ADMIN_TOKEN,
'expires': '2100-09-11T00:00:00'},
'user': {'id': 'user_id1',
'name': 'user_name1',
'tenantId': '123i2910',
'tenantName': 'mytenant',
'roles': [{'name': 'admin'}]},
}
}
MEMBER_BODY = {
'access': {
'token': {'id': MEMBER_TOKEN,
'expires': '2100-09-11T00:00:00'},
'user': {'id': 'user_id2',
'name': 'user-good',
'tenantId': 'project-good',
'tenantName': 'goodies',
'roles': [{'name': 'Member'}]},
}
}
DEFAULT_CACHE_VALUES = {
ADMIN_TOKEN: ADMIN_BODY,
MEMBER_TOKEN: MEMBER_BODY
}
class FakeMemcache(object):
"""Fake cache that is used for keystone tokens lookup."""
def __init__(self, cache_values=None):
if not cache_values:
cache_values = DEFAULT_CACHE_VALUES
self._cache = {}
for k, v in cache_values.items():
self._cache['tokens/%s' % k] = v
# NOTE(lucasagomes): keystonemiddleware >= 2.0.0 the token cache
# keys are sha256 hashes of the token key. This was introduced in
# https://review.opendev.org/#/c/186971
self._cache['tokens/%s' %
hashlib.sha256(k.encode()).hexdigest()] = v
self.set_key = None
self.set_value = None
self.token_expiration = None
def get(self, key):
dt = timeutils.utcnow() + datetime.timedelta(minutes=5)
return json.dumps((self._cache.get(key), dt.isoformat()))
def set(self, key, value, time=0, min_compress_len=0):
self.set_value = value
self.set_key = key
def remove_internal(values, internal):
# NOTE(yuriyz): internal attributes should not be posted, except uuid
int_attr = [attr.lstrip('/') for attr in internal if attr != '/uuid']
return {k: v for (k, v) in values.items() if k not in int_attr}
def remove_other_fields(values, allowed_fields):
return {k: v for (k, v) in values.items() if k in allowed_fields}
def node_post_data(**kw):
node = db_utils.get_test_node(**kw)
# NOTE(jroll): pop out fields that were introduced in later API versions,
# unless explicitly requested. Otherwise, these will cause tests using
# older API versions to fail.
for field in api_utils.VERSIONED_FIELDS:
if field not in kw:
node.pop(field, None)
return remove_other_fields(
node, node_controller.node_schema()['properties'])
def port_post_data(**kw):
port = db_utils.get_test_port(**kw)
return remove_other_fields(port,
port_controller.PORT_SCHEMA['properties'])
def volume_connector_post_data(**kw):
connector = db_utils.get_test_volume_connector(**kw)
return remove_other_fields(connector,
vc_controller.CONNECTOR_SCHEMA['properties'])
def volume_target_post_data(**kw):
target = db_utils.get_test_volume_target(**kw)
return remove_other_fields(target,
vt_controller.TARGET_SCHEMA['properties'])
def chassis_post_data(**kw):
chassis = db_utils.get_test_chassis(**kw)
return remove_other_fields(
chassis, chassis_controller.CHASSIS_SCHEMA['properties'])
def post_get_test_node(**kw):
# NOTE(lucasagomes): When creating a node via API (POST)
# we have to use chassis_uuid
node = node_post_data(**kw)
chassis = db_utils.get_test_chassis()
node['chassis_uuid'] = kw.get('chassis_uuid', chassis['uuid'])
return node
def portgroup_post_data(**kw):
"""Return a Portgroup object without internal attributes."""
portgroup = db_utils.get_test_portgroup(**kw)
# These values are not part of the API object
portgroup.pop('version')
portgroup.pop('node_id')
# NOTE(jroll): pop out fields that were introduced in later API versions,
# unless explicitly requested. Otherwise, these will cause tests using
# older API versions to fail.
new_api_ver_arguments = ['mode', 'properties']
for arg in new_api_ver_arguments:
if arg not in kw:
portgroup.pop(arg)
return remove_other_fields(
portgroup, portgroup_controller.PORTGROUP_SCHEMA['properties'])
def post_get_test_portgroup(**kw):
"""Return a Portgroup object with appropriate attributes."""
portgroup = portgroup_post_data(**kw)
node = db_utils.get_test_node()
portgroup['node_uuid'] = kw.get('node_uuid', node['uuid'])
return portgroup
def allocation_post_data(node=None, **kw):
"""Return an Allocation object without internal attributes."""
allocation = db_utils.get_test_allocation(**kw)
if node:
# This is not a database field, so it has to be handled explicitly
allocation['node'] = node
return remove_other_fields(
allocation, al_controller.ALLOCATION_SCHEMA['properties'])
def deploy_template_post_data(**kw):
"""Return a DeployTemplate object without internal attributes."""
template = db_utils.get_test_deploy_template(**kw)
# These values are not part of the API object
template.pop('version')
# Remove internal attributes from each step.
step_internal = api_utils.DEPLOY_STEP_SCHEMA['properties']
template['steps'] = [remove_other_fields(step, step_internal)
for step in template['steps']]
# Remove internal attributes from the template.
return remove_other_fields(
template, dt_controller.TEMPLATE_SCHEMA['properties'])
def runbook_post_data(**kw):
"""Return a Runbook object without internal attributes."""
runbook = db_utils.get_test_runbook(**kw)
# These values are not part of the API object
runbook.pop('version')
# Remove internal attributes from each step.
step_internal = api_utils.RUNBOOK_STEP_SCHEMA['properties']
runbook['steps'] = [remove_other_fields(step, step_internal)
for step in runbook['steps']]
# Remove internal attributes from the runbook.
return remove_other_fields(
runbook, rb_controller.RUNBOOK_SCHEMA['properties'])
def post_get_test_deploy_template(**kw):
"""Return a DeployTemplate object with appropriate attributes."""
return deploy_template_post_data(**kw)
def post_get_test_runbook(**kw):
"""Return a Runbook object with appropriate attributes."""
return runbook_post_data(**kw)