Add description field to node
This patch implements the feature of storing informational free-form text into ironic node, via the "description" field. Operators can do simple queries on the context of description. Change-Id: I787fb0df34566aff30dea4c4a3ba0e1ec820d044 Story: 2003089 Task: 23178
This commit is contained in:
parent
680e5b5687
commit
d30d814956
@ -2,6 +2,12 @@
|
|||||||
REST API Version History
|
REST API Version History
|
||||||
========================
|
========================
|
||||||
|
|
||||||
|
1.51 (Stein, master)
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
Added ``description`` field to the node object to enable operators to store
|
||||||
|
any information relates to the node. The field is up to 4096 characters.
|
||||||
|
|
||||||
1.50 (Stein, master)
|
1.50 (Stein, master)
|
||||||
--------------------
|
--------------------
|
||||||
|
|
||||||
|
@ -110,6 +110,8 @@ ALLOWED_TARGET_POWER_STATES = (ir_states.POWER_ON,
|
|||||||
ir_states.SOFT_REBOOT,
|
ir_states.SOFT_REBOOT,
|
||||||
ir_states.SOFT_POWER_OFF)
|
ir_states.SOFT_POWER_OFF)
|
||||||
|
|
||||||
|
_NODE_DESCRIPTION_MAX_LENGTH = 4096
|
||||||
|
|
||||||
|
|
||||||
def get_nodes_controller_reserved_names():
|
def get_nodes_controller_reserved_names():
|
||||||
global _NODES_CONTROLLER_RESERVED_WORDS
|
global _NODES_CONTROLLER_RESERVED_WORDS
|
||||||
@ -1078,6 +1080,9 @@ class Node(base.APIBase):
|
|||||||
owner = wsme.wsattr(wtypes.text)
|
owner = wsme.wsattr(wtypes.text)
|
||||||
"""Field for storage of physical node owner"""
|
"""Field for storage of physical node owner"""
|
||||||
|
|
||||||
|
description = wsme.wsattr(wtypes.text)
|
||||||
|
"""Field for node description"""
|
||||||
|
|
||||||
# NOTE(deva): "conductor_affinity" shouldn't be presented on the
|
# NOTE(deva): "conductor_affinity" shouldn't be presented on the
|
||||||
# API because it's an internal value. Don't add it here.
|
# API because it's an internal value. Don't add it here.
|
||||||
|
|
||||||
@ -1603,7 +1608,8 @@ class NodesController(rest.RestController):
|
|||||||
sort_key, sort_dir, driver=None,
|
sort_key, sort_dir, driver=None,
|
||||||
resource_class=None, resource_url=None,
|
resource_class=None, resource_url=None,
|
||||||
fields=None, fault=None, conductor_group=None,
|
fields=None, fault=None, conductor_group=None,
|
||||||
detail=None, conductor=None, owner=None):
|
detail=None, conductor=None, owner=None,
|
||||||
|
description_contains=None):
|
||||||
if self.from_chassis and not chassis_uuid:
|
if self.from_chassis and not chassis_uuid:
|
||||||
raise exception.MissingParameterValue(
|
raise exception.MissingParameterValue(
|
||||||
_("Chassis id not specified."))
|
_("Chassis id not specified."))
|
||||||
@ -1646,6 +1652,7 @@ class NodesController(rest.RestController):
|
|||||||
'fault': fault,
|
'fault': fault,
|
||||||
'conductor_group': conductor_group,
|
'conductor_group': conductor_group,
|
||||||
'owner': owner,
|
'owner': owner,
|
||||||
|
'description_contains': description_contains,
|
||||||
}
|
}
|
||||||
filters = {}
|
filters = {}
|
||||||
for key, value in possible_filters.items():
|
for key, value in possible_filters.items():
|
||||||
@ -1763,13 +1770,13 @@ class NodesController(rest.RestController):
|
|||||||
types.boolean, wtypes.text, types.uuid, int, wtypes.text,
|
types.boolean, wtypes.text, types.uuid, int, wtypes.text,
|
||||||
wtypes.text, wtypes.text, types.listtype, wtypes.text,
|
wtypes.text, wtypes.text, types.listtype, wtypes.text,
|
||||||
wtypes.text, wtypes.text, types.boolean, wtypes.text,
|
wtypes.text, wtypes.text, types.boolean, wtypes.text,
|
||||||
wtypes.text)
|
wtypes.text, wtypes.text)
|
||||||
def get_all(self, chassis_uuid=None, instance_uuid=None, associated=None,
|
def get_all(self, chassis_uuid=None, instance_uuid=None, associated=None,
|
||||||
maintenance=None, provision_state=None, marker=None,
|
maintenance=None, provision_state=None, marker=None,
|
||||||
limit=None, sort_key='id', sort_dir='asc', driver=None,
|
limit=None, sort_key='id', sort_dir='asc', driver=None,
|
||||||
fields=None, resource_class=None, fault=None,
|
fields=None, resource_class=None, fault=None,
|
||||||
conductor_group=None, detail=None, conductor=None,
|
conductor_group=None, detail=None, conductor=None,
|
||||||
owner=None):
|
owner=None, description_contains=None):
|
||||||
"""Retrieve a list of nodes.
|
"""Retrieve a list of nodes.
|
||||||
|
|
||||||
:param chassis_uuid: Optional UUID of a chassis, to get only nodes for
|
:param chassis_uuid: Optional UUID of a chassis, to get only nodes for
|
||||||
@ -1804,6 +1811,9 @@ class NodesController(rest.RestController):
|
|||||||
:param fields: Optional, a list with a specified set of fields
|
:param fields: Optional, a list with a specified set of fields
|
||||||
of the resource to be returned.
|
of the resource to be returned.
|
||||||
:param fault: Optional string value to get only nodes with that fault.
|
:param fault: Optional string value to get only nodes with that fault.
|
||||||
|
:param description_contains: Optional string value to get only nodes
|
||||||
|
with description field contains matching
|
||||||
|
value.
|
||||||
"""
|
"""
|
||||||
cdict = pecan.request.context.to_policy_values()
|
cdict = pecan.request.context.to_policy_values()
|
||||||
policy.authorize('baremetal:node:get', cdict, cdict)
|
policy.authorize('baremetal:node:get', cdict, cdict)
|
||||||
@ -1822,6 +1832,7 @@ class NodesController(rest.RestController):
|
|||||||
fields = api_utils.get_request_return_fields(fields, detail,
|
fields = api_utils.get_request_return_fields(fields, detail,
|
||||||
_DEFAULT_RETURN_FIELDS)
|
_DEFAULT_RETURN_FIELDS)
|
||||||
|
|
||||||
|
extra_args = {'description_contains': description_contains}
|
||||||
return self._get_nodes_collection(chassis_uuid, instance_uuid,
|
return self._get_nodes_collection(chassis_uuid, instance_uuid,
|
||||||
associated, maintenance,
|
associated, maintenance,
|
||||||
provision_state, marker,
|
provision_state, marker,
|
||||||
@ -1832,18 +1843,19 @@ class NodesController(rest.RestController):
|
|||||||
conductor_group=conductor_group,
|
conductor_group=conductor_group,
|
||||||
detail=detail,
|
detail=detail,
|
||||||
conductor=conductor,
|
conductor=conductor,
|
||||||
owner=owner)
|
owner=owner,
|
||||||
|
**extra_args)
|
||||||
|
|
||||||
@METRICS.timer('NodesController.detail')
|
@METRICS.timer('NodesController.detail')
|
||||||
@expose.expose(NodeCollection, types.uuid, types.uuid, types.boolean,
|
@expose.expose(NodeCollection, types.uuid, types.uuid, types.boolean,
|
||||||
types.boolean, wtypes.text, types.uuid, int, wtypes.text,
|
types.boolean, wtypes.text, types.uuid, int, wtypes.text,
|
||||||
wtypes.text, wtypes.text, wtypes.text, wtypes.text,
|
wtypes.text, wtypes.text, wtypes.text, wtypes.text,
|
||||||
wtypes.text, wtypes.text, wtypes.text)
|
wtypes.text, wtypes.text, wtypes.text, wtypes.text)
|
||||||
def detail(self, chassis_uuid=None, instance_uuid=None, associated=None,
|
def detail(self, chassis_uuid=None, instance_uuid=None, associated=None,
|
||||||
maintenance=None, provision_state=None, marker=None,
|
maintenance=None, provision_state=None, marker=None,
|
||||||
limit=None, sort_key='id', sort_dir='asc', driver=None,
|
limit=None, sort_key='id', sort_dir='asc', driver=None,
|
||||||
resource_class=None, fault=None, conductor_group=None,
|
resource_class=None, fault=None, conductor_group=None,
|
||||||
conductor=None, owner=None):
|
conductor=None, owner=None, description_contains=None):
|
||||||
"""Retrieve a list of nodes with detail.
|
"""Retrieve a list of nodes with detail.
|
||||||
|
|
||||||
:param chassis_uuid: Optional UUID of a chassis, to get only nodes for
|
:param chassis_uuid: Optional UUID of a chassis, to get only nodes for
|
||||||
@ -1874,6 +1886,9 @@ class NodesController(rest.RestController):
|
|||||||
that conductor_group.
|
that conductor_group.
|
||||||
:param owner: Optional string value that set the owner whose nodes
|
:param owner: Optional string value that set the owner whose nodes
|
||||||
are to be retrurned.
|
are to be retrurned.
|
||||||
|
:param description_contains: Optional string value to get only nodes
|
||||||
|
with description field contains matching
|
||||||
|
value.
|
||||||
"""
|
"""
|
||||||
cdict = pecan.request.context.to_policy_values()
|
cdict = pecan.request.context.to_policy_values()
|
||||||
policy.authorize('baremetal:node:get', cdict, cdict)
|
policy.authorize('baremetal:node:get', cdict, cdict)
|
||||||
@ -1893,6 +1908,7 @@ class NodesController(rest.RestController):
|
|||||||
api_utils.check_allow_filter_by_conductor(conductor)
|
api_utils.check_allow_filter_by_conductor(conductor)
|
||||||
|
|
||||||
resource_url = '/'.join(['nodes', 'detail'])
|
resource_url = '/'.join(['nodes', 'detail'])
|
||||||
|
extra_args = {'description_contains': description_contains}
|
||||||
return self._get_nodes_collection(chassis_uuid, instance_uuid,
|
return self._get_nodes_collection(chassis_uuid, instance_uuid,
|
||||||
associated, maintenance,
|
associated, maintenance,
|
||||||
provision_state, marker,
|
provision_state, marker,
|
||||||
@ -1903,7 +1919,8 @@ class NodesController(rest.RestController):
|
|||||||
fault=fault,
|
fault=fault,
|
||||||
conductor_group=conductor_group,
|
conductor_group=conductor_group,
|
||||||
conductor=conductor,
|
conductor=conductor,
|
||||||
owner=owner)
|
owner=owner,
|
||||||
|
**extra_args)
|
||||||
|
|
||||||
@METRICS.timer('NodesController.validate')
|
@METRICS.timer('NodesController.validate')
|
||||||
@expose.expose(wtypes.text, types.uuid_or_name, types.uuid)
|
@expose.expose(wtypes.text, types.uuid_or_name, types.uuid)
|
||||||
@ -1984,6 +2001,12 @@ class NodesController(rest.RestController):
|
|||||||
"creation. These fields can only be set for active nodes")
|
"creation. These fields can only be set for active nodes")
|
||||||
raise exception.Invalid(msg)
|
raise exception.Invalid(msg)
|
||||||
|
|
||||||
|
if (node.description is not wtypes.Unset and
|
||||||
|
len(node.description) > _NODE_DESCRIPTION_MAX_LENGTH):
|
||||||
|
msg = _("Cannot create node with description exceeds %s "
|
||||||
|
"characters") % _NODE_DESCRIPTION_MAX_LENGTH
|
||||||
|
raise exception.Invalid(msg)
|
||||||
|
|
||||||
# NOTE(deva): get_topic_for checks if node.driver is in the hash ring
|
# NOTE(deva): get_topic_for checks if node.driver is in the hash ring
|
||||||
# and raises NoValidHost if it is not.
|
# and raises NoValidHost if it is not.
|
||||||
# We need to ensure that node has a UUID before it can
|
# We need to ensure that node has a UUID before it can
|
||||||
@ -2040,6 +2063,12 @@ class NodesController(rest.RestController):
|
|||||||
"changing the node's driver.")
|
"changing the node's driver.")
|
||||||
raise exception.Invalid(msg)
|
raise exception.Invalid(msg)
|
||||||
|
|
||||||
|
description = api_utils.get_patch_values(patch, '/description')
|
||||||
|
if description and len(description[0]) > _NODE_DESCRIPTION_MAX_LENGTH:
|
||||||
|
msg = _("Cannot create node with description exceeds %s "
|
||||||
|
"characters") % _NODE_DESCRIPTION_MAX_LENGTH
|
||||||
|
raise exception.Invalid(msg)
|
||||||
|
|
||||||
@METRICS.timer('NodesController.patch')
|
@METRICS.timer('NodesController.patch')
|
||||||
@wsme.validate(types.uuid, types.boolean, [NodePatchType])
|
@wsme.validate(types.uuid, types.boolean, [NodePatchType])
|
||||||
@expose.expose(Node, types.uuid_or_name, types.boolean,
|
@expose.expose(Node, types.uuid_or_name, types.boolean,
|
||||||
|
@ -380,6 +380,7 @@ VERSIONED_FIELDS = {
|
|||||||
'protected_reason': versions.MINOR_48_NODE_PROTECTED,
|
'protected_reason': versions.MINOR_48_NODE_PROTECTED,
|
||||||
'conductor': versions.MINOR_49_CONDUCTORS,
|
'conductor': versions.MINOR_49_CONDUCTORS,
|
||||||
'owner': versions.MINOR_50_NODE_OWNER,
|
'owner': versions.MINOR_50_NODE_OWNER,
|
||||||
|
'description': versions.MINOR_51_NODE_DESCRIPTION,
|
||||||
}
|
}
|
||||||
|
|
||||||
for field in V31_FIELDS:
|
for field in V31_FIELDS:
|
||||||
|
@ -88,6 +88,7 @@ BASE_VERSION = 1
|
|||||||
# v1.48: Add protected to the node object.
|
# v1.48: Add protected to the node object.
|
||||||
# v1.49: Exposes current conductor on the node object.
|
# v1.49: Exposes current conductor on the node object.
|
||||||
# v1.50: Add owner to the node object.
|
# v1.50: Add owner to the node object.
|
||||||
|
# v1.51: Add description to the node object.
|
||||||
|
|
||||||
MINOR_0_JUNO = 0
|
MINOR_0_JUNO = 0
|
||||||
MINOR_1_INITIAL_VERSION = 1
|
MINOR_1_INITIAL_VERSION = 1
|
||||||
@ -140,6 +141,7 @@ MINOR_47_NODE_AUTOMATED_CLEAN = 47
|
|||||||
MINOR_48_NODE_PROTECTED = 48
|
MINOR_48_NODE_PROTECTED = 48
|
||||||
MINOR_49_CONDUCTORS = 49
|
MINOR_49_CONDUCTORS = 49
|
||||||
MINOR_50_NODE_OWNER = 50
|
MINOR_50_NODE_OWNER = 50
|
||||||
|
MINOR_51_NODE_DESCRIPTION = 51
|
||||||
|
|
||||||
# When adding another version, update:
|
# When adding another version, update:
|
||||||
# - MINOR_MAX_VERSION
|
# - MINOR_MAX_VERSION
|
||||||
@ -147,7 +149,7 @@ MINOR_50_NODE_OWNER = 50
|
|||||||
# explanation of what changed in the new version
|
# explanation of what changed in the new version
|
||||||
# - common/release_mappings.py, RELEASE_MAPPING['master']['api']
|
# - common/release_mappings.py, RELEASE_MAPPING['master']['api']
|
||||||
|
|
||||||
MINOR_MAX_VERSION = MINOR_50_NODE_OWNER
|
MINOR_MAX_VERSION = MINOR_51_NODE_DESCRIPTION
|
||||||
|
|
||||||
# String representations of the minor and maximum versions
|
# String representations of the minor and maximum versions
|
||||||
_MIN_VERSION_STRING = '{}.{}'.format(BASE_VERSION, MINOR_1_INITIAL_VERSION)
|
_MIN_VERSION_STRING = '{}.{}'.format(BASE_VERSION, MINOR_1_INITIAL_VERSION)
|
||||||
|
@ -131,11 +131,11 @@ RELEASE_MAPPING = {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
'master': {
|
'master': {
|
||||||
'api': '1.50',
|
'api': '1.51',
|
||||||
'rpc': '1.48',
|
'rpc': '1.48',
|
||||||
'objects': {
|
'objects': {
|
||||||
'Allocation': ['1.0'],
|
'Allocation': ['1.0'],
|
||||||
'Node': ['1.31', '1.30', '1.29', '1.28'],
|
'Node': ['1.32', '1.31', '1.30', '1.29', '1.28'],
|
||||||
'Conductor': ['1.3'],
|
'Conductor': ['1.3'],
|
||||||
'Chassis': ['1.3'],
|
'Chassis': ['1.3'],
|
||||||
'Port': ['1.9'],
|
'Port': ['1.9'],
|
||||||
|
@ -0,0 +1,31 @@
|
|||||||
|
# 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.
|
||||||
|
|
||||||
|
"""add node description
|
||||||
|
|
||||||
|
Revision ID: 28c44432c9c3
|
||||||
|
Revises: dd67b91a1981
|
||||||
|
Create Date: 2019-01-23 13:54:08.850421
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = '28c44432c9c3'
|
||||||
|
down_revision = '9cbeefa3763f'
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
op.add_column('nodes', sa.Column('description', sa.Text(),
|
||||||
|
nullable=True))
|
@ -225,7 +225,7 @@ class Connection(api.Connection):
|
|||||||
def __init__(self):
|
def __init__(self):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def _add_nodes_filters(self, query, filters):
|
def _validate_nodes_filters(self, filters):
|
||||||
if filters is None:
|
if filters is None:
|
||||||
filters = dict()
|
filters = dict()
|
||||||
supported_filters = {'console_enabled', 'maintenance', 'driver',
|
supported_filters = {'console_enabled', 'maintenance', 'driver',
|
||||||
@ -233,13 +233,17 @@ class Connection(api.Connection):
|
|||||||
'chassis_uuid', 'associated', 'reserved',
|
'chassis_uuid', 'associated', 'reserved',
|
||||||
'reserved_by_any_of', 'provisioned_before',
|
'reserved_by_any_of', 'provisioned_before',
|
||||||
'inspection_started_before', 'fault',
|
'inspection_started_before', 'fault',
|
||||||
'conductor_group', 'owner',
|
'conductor_group', 'owner', 'uuid_in',
|
||||||
'uuid_in', 'with_power_state'}
|
'with_power_state', 'description_contains'}
|
||||||
unsupported_filters = set(filters).difference(supported_filters)
|
unsupported_filters = set(filters).difference(supported_filters)
|
||||||
if unsupported_filters:
|
if unsupported_filters:
|
||||||
msg = _("SqlAlchemy API does not support "
|
msg = _("SqlAlchemy API does not support "
|
||||||
"filtering by %s") % ', '.join(unsupported_filters)
|
"filtering by %s") % ', '.join(unsupported_filters)
|
||||||
raise ValueError(msg)
|
raise ValueError(msg)
|
||||||
|
return filters
|
||||||
|
|
||||||
|
def _add_nodes_filters(self, query, filters):
|
||||||
|
filters = self._validate_nodes_filters(filters)
|
||||||
for field in ['console_enabled', 'maintenance', 'driver',
|
for field in ['console_enabled', 'maintenance', 'driver',
|
||||||
'resource_class', 'provision_state', 'uuid', 'id',
|
'resource_class', 'provision_state', 'uuid', 'id',
|
||||||
'fault', 'conductor_group', 'owner']:
|
'fault', 'conductor_group', 'owner']:
|
||||||
@ -280,6 +284,11 @@ class Connection(api.Connection):
|
|||||||
query = query.filter(models.Node.power_state != sql.null())
|
query = query.filter(models.Node.power_state != sql.null())
|
||||||
else:
|
else:
|
||||||
query = query.filter(models.Node.power_state == sql.null())
|
query = query.filter(models.Node.power_state == sql.null())
|
||||||
|
if 'description_contains' in filters:
|
||||||
|
keyword = filters['description_contains']
|
||||||
|
if keyword is not None:
|
||||||
|
query = query.filter(
|
||||||
|
models.Node.description.like(r'%{}%'.format(keyword)))
|
||||||
|
|
||||||
return query
|
return query
|
||||||
|
|
||||||
|
@ -182,6 +182,7 @@ class Node(Base):
|
|||||||
owner = Column(String(255), nullable=True)
|
owner = Column(String(255), nullable=True)
|
||||||
allocation_id = Column(Integer, ForeignKey('allocations.id'),
|
allocation_id = Column(Integer, ForeignKey('allocations.id'),
|
||||||
nullable=True)
|
nullable=True)
|
||||||
|
description = Column(Text, nullable=True)
|
||||||
|
|
||||||
bios_interface = Column(String(255), nullable=True)
|
bios_interface = Column(String(255), nullable=True)
|
||||||
boot_interface = Column(String(255), nullable=True)
|
boot_interface = Column(String(255), nullable=True)
|
||||||
|
@ -68,7 +68,8 @@ class Node(base.IronicObject, object_base.VersionedObjectDictCompat):
|
|||||||
# Version 1.29: Add protected and protected_reason fields
|
# Version 1.29: Add protected and protected_reason fields
|
||||||
# Version 1.30: Add owner field
|
# Version 1.30: Add owner field
|
||||||
# Version 1.31: Add allocation_id field
|
# Version 1.31: Add allocation_id field
|
||||||
VERSION = '1.31'
|
# Version 1.32: Add description field
|
||||||
|
VERSION = '1.32'
|
||||||
|
|
||||||
dbapi = db_api.get_instance()
|
dbapi = db_api.get_instance()
|
||||||
|
|
||||||
@ -153,6 +154,7 @@ class Node(base.IronicObject, object_base.VersionedObjectDictCompat):
|
|||||||
'vendor_interface': object_fields.StringField(nullable=True),
|
'vendor_interface': object_fields.StringField(nullable=True),
|
||||||
'traits': object_fields.ObjectField('TraitList', nullable=True),
|
'traits': object_fields.ObjectField('TraitList', nullable=True),
|
||||||
'owner': object_fields.StringField(nullable=True),
|
'owner': object_fields.StringField(nullable=True),
|
||||||
|
'description': object_fields.StringField(nullable=True),
|
||||||
}
|
}
|
||||||
|
|
||||||
def as_dict(self, secure=False):
|
def as_dict(self, secure=False):
|
||||||
@ -577,6 +579,8 @@ class Node(base.IronicObject, object_base.VersionedObjectDictCompat):
|
|||||||
set to None or removed.
|
set to None or removed.
|
||||||
Version 1.31: allocation_id was added. For versions prior to this, it
|
Version 1.31: allocation_id was added. For versions prior to this, it
|
||||||
should be set to None (or removed).
|
should be set to None (or removed).
|
||||||
|
Version 1.32: description was added. For versions prior to this, it
|
||||||
|
should be set to None (or removed).
|
||||||
|
|
||||||
:param target_version: the desired version of the object
|
:param target_version: the desired version of the object
|
||||||
:param remove_unavailable_fields: True to remove fields that are
|
:param remove_unavailable_fields: True to remove fields that are
|
||||||
@ -590,7 +594,7 @@ class Node(base.IronicObject, object_base.VersionedObjectDictCompat):
|
|||||||
fields = [('rescue_interface', 22), ('traits', 23),
|
fields = [('rescue_interface', 22), ('traits', 23),
|
||||||
('bios_interface', 24), ('fault', 25),
|
('bios_interface', 24), ('fault', 25),
|
||||||
('automated_clean', 28), ('protected_reason', 29),
|
('automated_clean', 28), ('protected_reason', 29),
|
||||||
('owner', 30), ('allocation_id', 31)]
|
('owner', 30), ('allocation_id', 31), ('description', 32)]
|
||||||
for name, minor in fields:
|
for name, minor in fields:
|
||||||
self._adjust_field_to_version(name, None, target_version,
|
self._adjust_field_to_version(name, None, target_version,
|
||||||
1, minor, remove_unavailable_fields)
|
1, minor, remove_unavailable_fields)
|
||||||
@ -622,6 +626,7 @@ class NodePayload(notification.NotificationPayloadBase):
|
|||||||
'console_enabled': ('node', 'console_enabled'),
|
'console_enabled': ('node', 'console_enabled'),
|
||||||
'created_at': ('node', 'created_at'),
|
'created_at': ('node', 'created_at'),
|
||||||
'deploy_step': ('node', 'deploy_step'),
|
'deploy_step': ('node', 'deploy_step'),
|
||||||
|
'description': ('node', 'description'),
|
||||||
'driver': ('node', 'driver'),
|
'driver': ('node', 'driver'),
|
||||||
'extra': ('node', 'extra'),
|
'extra': ('node', 'extra'),
|
||||||
'inspection_finished_at': ('node', 'inspection_finished_at'),
|
'inspection_finished_at': ('node', 'inspection_finished_at'),
|
||||||
@ -672,13 +677,15 @@ class NodePayload(notification.NotificationPayloadBase):
|
|||||||
# Version 1.10: Add conductor_group field exposed via API.
|
# Version 1.10: Add conductor_group field exposed via API.
|
||||||
# Version 1.11: Add protected and protected_reason fields exposed via API.
|
# Version 1.11: Add protected and protected_reason fields exposed via API.
|
||||||
# Version 1.12: Add node owner field.
|
# Version 1.12: Add node owner field.
|
||||||
VERSION = '1.12'
|
# Version 1.13: Add description field.
|
||||||
|
VERSION = '1.13'
|
||||||
fields = {
|
fields = {
|
||||||
'clean_step': object_fields.FlexibleDictField(nullable=True),
|
'clean_step': object_fields.FlexibleDictField(nullable=True),
|
||||||
'conductor_group': object_fields.StringField(nullable=True),
|
'conductor_group': object_fields.StringField(nullable=True),
|
||||||
'console_enabled': object_fields.BooleanField(nullable=True),
|
'console_enabled': object_fields.BooleanField(nullable=True),
|
||||||
'created_at': object_fields.DateTimeField(nullable=True),
|
'created_at': object_fields.DateTimeField(nullable=True),
|
||||||
'deploy_step': object_fields.FlexibleDictField(nullable=True),
|
'deploy_step': object_fields.FlexibleDictField(nullable=True),
|
||||||
|
'description': object_fields.StringField(nullable=True),
|
||||||
'driver': object_fields.StringField(nullable=True),
|
'driver': object_fields.StringField(nullable=True),
|
||||||
'extra': object_fields.FlexibleDictField(nullable=True),
|
'extra': object_fields.FlexibleDictField(nullable=True),
|
||||||
'inspection_finished_at': object_fields.DateTimeField(nullable=True),
|
'inspection_finished_at': object_fields.DateTimeField(nullable=True),
|
||||||
@ -754,7 +761,8 @@ class NodeSetPowerStatePayload(NodePayload):
|
|||||||
# Version 1.10: Parent NodePayload version 1.10
|
# Version 1.10: Parent NodePayload version 1.10
|
||||||
# Version 1.11: Parent NodePayload version 1.11
|
# Version 1.11: Parent NodePayload version 1.11
|
||||||
# Version 1.12: Parent NodePayload version 1.12
|
# Version 1.12: Parent NodePayload version 1.12
|
||||||
VERSION = '1.12'
|
# Version 1.13: Parent NodePayload version 1.13
|
||||||
|
VERSION = '1.13'
|
||||||
|
|
||||||
fields = {
|
fields = {
|
||||||
# "to_power" indicates the future target_power_state of the node. A
|
# "to_power" indicates the future target_power_state of the node. A
|
||||||
@ -807,7 +815,8 @@ class NodeCorrectedPowerStatePayload(NodePayload):
|
|||||||
# Version 1.10: Parent NodePayload version 1.10
|
# Version 1.10: Parent NodePayload version 1.10
|
||||||
# Version 1.11: Parent NodePayload version 1.11
|
# Version 1.11: Parent NodePayload version 1.11
|
||||||
# Version 1.12: Parent NodePayload version 1.12
|
# Version 1.12: Parent NodePayload version 1.12
|
||||||
VERSION = '1.12'
|
# Version 1.13: Parent NodePayload version 1.13
|
||||||
|
VERSION = '1.13'
|
||||||
|
|
||||||
fields = {
|
fields = {
|
||||||
'from_power': object_fields.StringField(nullable=True)
|
'from_power': object_fields.StringField(nullable=True)
|
||||||
@ -844,7 +853,8 @@ class NodeSetProvisionStatePayload(NodePayload):
|
|||||||
# Version 1.10: Parent NodePayload version 1.10
|
# Version 1.10: Parent NodePayload version 1.10
|
||||||
# Version 1.11: Parent NodePayload version 1.11
|
# Version 1.11: Parent NodePayload version 1.11
|
||||||
# Version 1.12: Parent NodePayload version 1.12
|
# Version 1.12: Parent NodePayload version 1.12
|
||||||
VERSION = '1.12'
|
# Version 1.13: Parent NodePayload version 1.13
|
||||||
|
VERSION = '1.13'
|
||||||
|
|
||||||
SCHEMA = dict(NodePayload.SCHEMA,
|
SCHEMA = dict(NodePayload.SCHEMA,
|
||||||
**{'instance_info': ('node', 'instance_info')})
|
**{'instance_info': ('node', 'instance_info')})
|
||||||
@ -888,7 +898,8 @@ class NodeCRUDPayload(NodePayload):
|
|||||||
# Version 1.8: Parent NodePayload version 1.10
|
# Version 1.8: Parent NodePayload version 1.10
|
||||||
# Version 1.9: Parent NodePayload version 1.11
|
# Version 1.9: Parent NodePayload version 1.11
|
||||||
# Version 1.10: Parent NodePayload version 1.12
|
# Version 1.10: Parent NodePayload version 1.12
|
||||||
VERSION = '1.10'
|
# Version 1.11: Parent NodePayload version 1.13
|
||||||
|
VERSION = '1.11'
|
||||||
|
|
||||||
SCHEMA = dict(NodePayload.SCHEMA,
|
SCHEMA = dict(NodePayload.SCHEMA,
|
||||||
**{'instance_info': ('node', 'instance_info'),
|
**{'instance_info': ('node', 'instance_info'),
|
||||||
|
@ -345,6 +345,12 @@ class TestListNodes(test_api_base.BaseApiTest):
|
|||||||
headers={api_base.Version.string: '1.50'})
|
headers={api_base.Version.string: '1.50'})
|
||||||
self.assertEqual(data['owner'], "akindofmagic")
|
self.assertEqual(data['owner'], "akindofmagic")
|
||||||
|
|
||||||
|
def test_node_description_null_field(self):
|
||||||
|
node = obj_utils.create_test_node(self.context, description=None)
|
||||||
|
data = self.get_json('/nodes/%s' % node.uuid,
|
||||||
|
headers={api_base.Version.string: '1.51'})
|
||||||
|
self.assertIsNone(data['description'])
|
||||||
|
|
||||||
def test_get_one_custom_fields(self):
|
def test_get_one_custom_fields(self):
|
||||||
node = obj_utils.create_test_node(self.context,
|
node = obj_utils.create_test_node(self.context,
|
||||||
chassis_id=self.chassis.id)
|
chassis_id=self.chassis.id)
|
||||||
@ -543,6 +549,14 @@ class TestListNodes(test_api_base.BaseApiTest):
|
|||||||
headers={api_base.Version.string: '1.50'})
|
headers={api_base.Version.string: '1.50'})
|
||||||
self.assertIn('owner', response)
|
self.assertIn('owner', response)
|
||||||
|
|
||||||
|
def test_get_description_field(self):
|
||||||
|
node = obj_utils.create_test_node(self.context,
|
||||||
|
description='useful piece')
|
||||||
|
fields = 'description'
|
||||||
|
response = self.get_json('/nodes/%s?fields=%s' % (node.uuid, fields),
|
||||||
|
headers={api_base.Version.string: '1.51'})
|
||||||
|
self.assertIn('description', response)
|
||||||
|
|
||||||
def test_detail(self):
|
def test_detail(self):
|
||||||
node = obj_utils.create_test_node(self.context,
|
node = obj_utils.create_test_node(self.context,
|
||||||
chassis_id=self.chassis.id)
|
chassis_id=self.chassis.id)
|
||||||
@ -790,6 +804,17 @@ class TestListNodes(test_api_base.BaseApiTest):
|
|||||||
'/nodes/detail', headers={api_base.Version.string: '1.37'})
|
'/nodes/detail', headers={api_base.Version.string: '1.37'})
|
||||||
self.assertEqual(['CUSTOM_1'], new_data['nodes'][0]["traits"])
|
self.assertEqual(['CUSTOM_1'], new_data['nodes'][0]["traits"])
|
||||||
|
|
||||||
|
def test_hide_fields_in_newer_versions_description(self):
|
||||||
|
node = obj_utils.create_test_node(self.context,
|
||||||
|
description="useful piece")
|
||||||
|
data = self.get_json('/nodes/%s' % node.uuid,
|
||||||
|
headers={api_base.Version.string: "1.50"})
|
||||||
|
self.assertNotIn('description', data)
|
||||||
|
|
||||||
|
data = self.get_json('/nodes/%s' % node.uuid,
|
||||||
|
headers={api_base.Version.string: "1.51"})
|
||||||
|
self.assertEqual('useful piece', data['description'])
|
||||||
|
|
||||||
def test_many(self):
|
def test_many(self):
|
||||||
nodes = []
|
nodes = []
|
||||||
for id in range(5):
|
for id in range(5):
|
||||||
@ -1690,6 +1715,25 @@ class TestListNodes(test_api_base.BaseApiTest):
|
|||||||
self.assertEqual(http_client.NOT_ACCEPTABLE, response.status_code)
|
self.assertEqual(http_client.NOT_ACCEPTABLE, response.status_code)
|
||||||
self.assertTrue(response.json['error_message'])
|
self.assertTrue(response.json['error_message'])
|
||||||
|
|
||||||
|
def test_get_nodes_by_description(self):
|
||||||
|
node1 = obj_utils.create_test_node(self.context,
|
||||||
|
uuid=uuidutils.generate_uuid(),
|
||||||
|
description='some cats here')
|
||||||
|
node2 = obj_utils.create_test_node(self.context,
|
||||||
|
uuid=uuidutils.generate_uuid(),
|
||||||
|
description='some dogs there')
|
||||||
|
data = self.get_json('/nodes?description_contains=cat',
|
||||||
|
headers={api_base.Version.string: '1.51'})
|
||||||
|
uuids = [n['uuid'] for n in data['nodes']]
|
||||||
|
self.assertIn(node1.uuid, uuids)
|
||||||
|
self.assertNotIn(node2.uuid, uuids)
|
||||||
|
|
||||||
|
data = self.get_json('/nodes?description_contains=dog',
|
||||||
|
headers={api_base.Version.string: '1.51'})
|
||||||
|
uuids = [n['uuid'] for n in data['nodes']]
|
||||||
|
self.assertIn(node2.uuid, uuids)
|
||||||
|
self.assertNotIn(node1.uuid, uuids)
|
||||||
|
|
||||||
def test_get_console_information(self):
|
def test_get_console_information(self):
|
||||||
node = obj_utils.create_test_node(self.context)
|
node = obj_utils.create_test_node(self.context)
|
||||||
expected_console_info = {'test': 'test-data'}
|
expected_console_info = {'test': 'test-data'}
|
||||||
@ -2924,6 +2968,34 @@ class TestPatch(test_api_base.BaseApiTest):
|
|||||||
self.assertEqual('application/json', response.content_type)
|
self.assertEqual('application/json', response.content_type)
|
||||||
self.assertEqual(http_client.NOT_ACCEPTABLE, response.status_code)
|
self.assertEqual(http_client.NOT_ACCEPTABLE, response.status_code)
|
||||||
|
|
||||||
|
def test_update_description(self):
|
||||||
|
node = obj_utils.create_test_node(self.context,
|
||||||
|
uuid=uuidutils.generate_uuid())
|
||||||
|
self.mock_update_node.return_value = node
|
||||||
|
headers = {api_base.Version.string: '1.51'}
|
||||||
|
response = self.patch_json('/nodes/%s' % node.uuid,
|
||||||
|
[{'path': '/description',
|
||||||
|
'value': 'meow',
|
||||||
|
'op': 'replace'}],
|
||||||
|
headers=headers)
|
||||||
|
self.assertEqual('application/json', response.content_type)
|
||||||
|
self.assertEqual(http_client.OK, response.status_code)
|
||||||
|
|
||||||
|
def test_update_description_oversize(self):
|
||||||
|
node = obj_utils.create_test_node(self.context,
|
||||||
|
uuid=uuidutils.generate_uuid())
|
||||||
|
desc = '12345678' * 512 + 'last weed'
|
||||||
|
self.mock_update_node.return_value = node
|
||||||
|
headers = {api_base.Version.string: '1.51'}
|
||||||
|
response = self.patch_json('/nodes/%s' % node.uuid,
|
||||||
|
[{'path': '/description',
|
||||||
|
'value': desc,
|
||||||
|
'op': 'replace'}],
|
||||||
|
headers=headers,
|
||||||
|
expect_errors=True)
|
||||||
|
self.assertEqual('application/json', response.content_type)
|
||||||
|
self.assertEqual(http_client.BAD_REQUEST, response.status_code)
|
||||||
|
|
||||||
|
|
||||||
def _create_node_locally(node):
|
def _create_node_locally(node):
|
||||||
driver_factory.check_and_update_node_interfaces(node)
|
driver_factory.check_and_update_node_interfaces(node)
|
||||||
@ -3550,6 +3622,27 @@ class TestPost(test_api_base.BaseApiTest):
|
|||||||
self.assertEqual('application/json', response.content_type)
|
self.assertEqual('application/json', response.content_type)
|
||||||
self.assertEqual(http_client.NOT_ACCEPTABLE, response.status_int)
|
self.assertEqual(http_client.NOT_ACCEPTABLE, response.status_int)
|
||||||
|
|
||||||
|
def test_create_node_description(self):
|
||||||
|
node = test_api_utils.post_get_test_node(description='useful stuff')
|
||||||
|
response = self.post_json('/nodes', node,
|
||||||
|
headers={api_base.Version.string:
|
||||||
|
str(api_v1.max_version())})
|
||||||
|
self.assertEqual(http_client.CREATED, response.status_int)
|
||||||
|
result = self.get_json('/nodes/%s' % node['uuid'],
|
||||||
|
headers={api_base.Version.string:
|
||||||
|
str(api_v1.max_version())})
|
||||||
|
self.assertEqual('useful stuff', result['description'])
|
||||||
|
|
||||||
|
def test_create_node_description_oversize(self):
|
||||||
|
desc = '12345678' * 512 + 'last weed'
|
||||||
|
node = test_api_utils.post_get_test_node(description=desc)
|
||||||
|
response = self.post_json('/nodes', node,
|
||||||
|
headers={api_base.Version.string:
|
||||||
|
str(api_v1.max_version())},
|
||||||
|
expect_errors=True)
|
||||||
|
self.assertEqual('application/json', response.content_type)
|
||||||
|
self.assertEqual(http_client.BAD_REQUEST, response.status_int)
|
||||||
|
|
||||||
|
|
||||||
class TestDelete(test_api_base.BaseApiTest):
|
class TestDelete(test_api_base.BaseApiTest):
|
||||||
|
|
||||||
|
@ -851,6 +851,13 @@ class MigrationCheckersMixin(object):
|
|||||||
(sqlalchemy.types.Boolean,
|
(sqlalchemy.types.Boolean,
|
||||||
sqlalchemy.types.Integer))
|
sqlalchemy.types.Integer))
|
||||||
|
|
||||||
|
def _check_28c44432c9c3(self, engine, data):
|
||||||
|
nodes_tbl = db_utils.get_table(engine, 'nodes')
|
||||||
|
col_names = [column.name for column in nodes_tbl.c]
|
||||||
|
self.assertIn('description', col_names)
|
||||||
|
self.assertIsInstance(nodes_tbl.c.description.type,
|
||||||
|
sqlalchemy.types.TEXT)
|
||||||
|
|
||||||
def test_upgrade_and_version(self):
|
def test_upgrade_and_version(self):
|
||||||
with patch_with_engine(self.engine):
|
with patch_with_engine(self.engine):
|
||||||
self.migration_api.upgrade('head')
|
self.migration_api.upgrade('head')
|
||||||
|
@ -273,6 +273,19 @@ class DbNodeTestCase(base.DbTestCase):
|
|||||||
states.INSPECTING})
|
states.INSPECTING})
|
||||||
self.assertEqual([node2.id], [r[0] for r in res])
|
self.assertEqual([node2.id], [r[0] for r in res])
|
||||||
|
|
||||||
|
def test_get_nodeinfo_list_description(self):
|
||||||
|
node1 = utils.create_test_node(uuid=uuidutils.generate_uuid(),
|
||||||
|
description='Hello')
|
||||||
|
node2 = utils.create_test_node(uuid=uuidutils.generate_uuid(),
|
||||||
|
description='World!')
|
||||||
|
res = self.dbapi.get_nodeinfo_list(
|
||||||
|
filters={'description_contains': 'Hello'})
|
||||||
|
self.assertEqual([node1.id], [r[0] for r in res])
|
||||||
|
|
||||||
|
res = self.dbapi.get_nodeinfo_list(filters={'description_contains':
|
||||||
|
'World!'})
|
||||||
|
self.assertEqual([node2.id], [r[0] for r in res])
|
||||||
|
|
||||||
def test_get_node_list(self):
|
def test_get_node_list(self):
|
||||||
uuids = []
|
uuids = []
|
||||||
for i in range(1, 6):
|
for i in range(1, 6):
|
||||||
@ -382,6 +395,19 @@ class DbNodeTestCase(base.DbTestCase):
|
|||||||
self.dbapi.get_node_list,
|
self.dbapi.get_node_list,
|
||||||
filters=filters)
|
filters=filters)
|
||||||
|
|
||||||
|
def test_get_node_list_description(self):
|
||||||
|
node1 = utils.create_test_node(uuid=uuidutils.generate_uuid(),
|
||||||
|
description='Hello')
|
||||||
|
node2 = utils.create_test_node(uuid=uuidutils.generate_uuid(),
|
||||||
|
description='World!')
|
||||||
|
res = self.dbapi.get_node_list(filters={
|
||||||
|
'description_contains': 'Hello'})
|
||||||
|
self.assertEqual([node1.id], [r.id for r in res])
|
||||||
|
|
||||||
|
res = self.dbapi.get_node_list(filters={
|
||||||
|
'description_contains': 'World!'})
|
||||||
|
self.assertEqual([node2.id], [r.id for r in res])
|
||||||
|
|
||||||
def test_get_node_list_chassis_not_found(self):
|
def test_get_node_list_chassis_not_found(self):
|
||||||
self.assertRaises(exception.ChassisNotFound,
|
self.assertRaises(exception.ChassisNotFound,
|
||||||
self.dbapi.get_node_list,
|
self.dbapi.get_node_list,
|
||||||
|
@ -222,6 +222,7 @@ def get_test_node(**kw):
|
|||||||
'conductor': kw.get('conductor'),
|
'conductor': kw.get('conductor'),
|
||||||
'owner': kw.get('owner', None),
|
'owner': kw.get('owner', None),
|
||||||
'allocation_id': kw.get('allocation_id'),
|
'allocation_id': kw.get('allocation_id'),
|
||||||
|
'description': kw.get('description'),
|
||||||
}
|
}
|
||||||
|
|
||||||
for iface in drivers_base.ALL_INTERFACES:
|
for iface in drivers_base.ALL_INTERFACES:
|
||||||
|
@ -949,6 +949,68 @@ class TestConvertToVersion(db_base.DbTestCase):
|
|||||||
self.assertIsNone(node.allocation_id)
|
self.assertIsNone(node.allocation_id)
|
||||||
self.assertEqual({}, node.obj_get_changes())
|
self.assertEqual({}, node.obj_get_changes())
|
||||||
|
|
||||||
|
def test_description_supported_missing(self):
|
||||||
|
# description not set, should be set to default.
|
||||||
|
node = obj_utils.get_test_node(self.ctxt, **self.fake_node)
|
||||||
|
delattr(node, 'description')
|
||||||
|
node.obj_reset_changes()
|
||||||
|
node._convert_to_version("1.32")
|
||||||
|
self.assertIsNone(node.description)
|
||||||
|
self.assertEqual({'description': None},
|
||||||
|
node.obj_get_changes())
|
||||||
|
|
||||||
|
def test_description_supported_set(self):
|
||||||
|
# description set, no change required.
|
||||||
|
node = obj_utils.get_test_node(self.ctxt, **self.fake_node)
|
||||||
|
|
||||||
|
node.description = "Useful information relates to this node"
|
||||||
|
node.obj_reset_changes()
|
||||||
|
node._convert_to_version("1.32")
|
||||||
|
self.assertEqual("Useful information relates to this node",
|
||||||
|
node.description)
|
||||||
|
self.assertEqual({}, node.obj_get_changes())
|
||||||
|
|
||||||
|
def test_description_unsupported_missing(self):
|
||||||
|
# description not set, no change required.
|
||||||
|
node = obj_utils.get_test_node(self.ctxt, **self.fake_node)
|
||||||
|
|
||||||
|
delattr(node, 'description')
|
||||||
|
node.obj_reset_changes()
|
||||||
|
node._convert_to_version("1.31")
|
||||||
|
self.assertNotIn('description', node)
|
||||||
|
self.assertEqual({}, node.obj_get_changes())
|
||||||
|
|
||||||
|
def test_description_unsupported_set_remove(self):
|
||||||
|
# description set, should be removed.
|
||||||
|
node = obj_utils.get_test_node(self.ctxt, **self.fake_node)
|
||||||
|
|
||||||
|
node.description = "Useful piece"
|
||||||
|
node.obj_reset_changes()
|
||||||
|
node._convert_to_version("1.31")
|
||||||
|
self.assertNotIn('description', node)
|
||||||
|
self.assertEqual({}, node.obj_get_changes())
|
||||||
|
|
||||||
|
def test_description_unsupported_set_no_remove_non_default(self):
|
||||||
|
# description set, should be set to default.
|
||||||
|
node = obj_utils.get_test_node(self.ctxt, **self.fake_node)
|
||||||
|
|
||||||
|
node.description = "Useful piece"
|
||||||
|
node.obj_reset_changes()
|
||||||
|
node._convert_to_version("1.31", False)
|
||||||
|
self.assertIsNone(node.description)
|
||||||
|
self.assertEqual({'description': None},
|
||||||
|
node.obj_get_changes())
|
||||||
|
|
||||||
|
def test_description_unsupported_set_no_remove_default(self):
|
||||||
|
# description set, no change required.
|
||||||
|
node = obj_utils.get_test_node(self.ctxt, **self.fake_node)
|
||||||
|
|
||||||
|
node.description = None
|
||||||
|
node.obj_reset_changes()
|
||||||
|
node._convert_to_version("1.31", False)
|
||||||
|
self.assertIsNone(node.description)
|
||||||
|
self.assertEqual({}, node.obj_get_changes())
|
||||||
|
|
||||||
|
|
||||||
class TestNodePayloads(db_base.DbTestCase):
|
class TestNodePayloads(db_base.DbTestCase):
|
||||||
|
|
||||||
|
@ -677,7 +677,7 @@ class TestObject(_LocalTest, _TestObject):
|
|||||||
# version bump. It is an MD5 hash of the object fields and remotable methods.
|
# version bump. It is an MD5 hash of the object fields and remotable methods.
|
||||||
# The fingerprint values should only be changed if there is a version bump.
|
# The fingerprint values should only be changed if there is a version bump.
|
||||||
expected_object_fingerprints = {
|
expected_object_fingerprints = {
|
||||||
'Node': '1.31-1b77c11e94f971a71c76f5f44fb5b3f4',
|
'Node': '1.32-525750e76f07b62142ed5297334b7832',
|
||||||
'MyObj': '1.5-9459d30d6954bffc7a9afd347a807ca6',
|
'MyObj': '1.5-9459d30d6954bffc7a9afd347a807ca6',
|
||||||
'Chassis': '1.3-d656e039fd8ae9f34efc232ab3980905',
|
'Chassis': '1.3-d656e039fd8ae9f34efc232ab3980905',
|
||||||
'Port': '1.9-0cb9202a4ec442e8c0d87a324155eaaf',
|
'Port': '1.9-0cb9202a4ec442e8c0d87a324155eaaf',
|
||||||
@ -685,21 +685,21 @@ expected_object_fingerprints = {
|
|||||||
'Conductor': '1.3-d3f53e853b4d58cae5bfbd9a8341af4a',
|
'Conductor': '1.3-d3f53e853b4d58cae5bfbd9a8341af4a',
|
||||||
'EventType': '1.1-aa2ba1afd38553e3880c267404e8d370',
|
'EventType': '1.1-aa2ba1afd38553e3880c267404e8d370',
|
||||||
'NotificationPublisher': '1.0-51a09397d6c0687771fb5be9a999605d',
|
'NotificationPublisher': '1.0-51a09397d6c0687771fb5be9a999605d',
|
||||||
'NodePayload': '1.12-7d650c2a024357275990681f020512e4',
|
'NodePayload': '1.13-18a34d461ef7d5dbc1c3e5a55fcb867a',
|
||||||
'NodeSetPowerStateNotification': '1.0-59acc533c11d306f149846f922739c15',
|
'NodeSetPowerStateNotification': '1.0-59acc533c11d306f149846f922739c15',
|
||||||
'NodeSetPowerStatePayload': '1.12-703d110d571cc95b2947bb6bd153fcb8',
|
'NodeSetPowerStatePayload': '1.13-4f96e52568e058e3fd6ffc9b0cf15764',
|
||||||
'NodeCorrectedPowerStateNotification':
|
'NodeCorrectedPowerStateNotification':
|
||||||
'1.0-59acc533c11d306f149846f922739c15',
|
'1.0-59acc533c11d306f149846f922739c15',
|
||||||
'NodeCorrectedPowerStatePayload': '1.12-29cbb6b20a0aeea9e0ab9e17302e9e16',
|
'NodeCorrectedPowerStatePayload': '1.13-929af354e7c3474520ce6162ee794717',
|
||||||
'NodeSetProvisionStateNotification':
|
'NodeSetProvisionStateNotification':
|
||||||
'1.0-59acc533c11d306f149846f922739c15',
|
'1.0-59acc533c11d306f149846f922739c15',
|
||||||
'NodeSetProvisionStatePayload': '1.12-a302ce357ad39a0a4d1ca3c0ee44f0e0',
|
'NodeSetProvisionStatePayload': '1.13-fa15d2954961d8edcaba9d737a1cad91',
|
||||||
'VolumeConnector': '1.0-3e0252c0ab6e6b9d158d09238a577d97',
|
'VolumeConnector': '1.0-3e0252c0ab6e6b9d158d09238a577d97',
|
||||||
'VolumeTarget': '1.0-0b10d663d8dae675900b2c7548f76f5e',
|
'VolumeTarget': '1.0-0b10d663d8dae675900b2c7548f76f5e',
|
||||||
'ChassisCRUDNotification': '1.0-59acc533c11d306f149846f922739c15',
|
'ChassisCRUDNotification': '1.0-59acc533c11d306f149846f922739c15',
|
||||||
'ChassisCRUDPayload': '1.0-dce63895d8186279a7dd577cffccb202',
|
'ChassisCRUDPayload': '1.0-dce63895d8186279a7dd577cffccb202',
|
||||||
'NodeCRUDNotification': '1.0-59acc533c11d306f149846f922739c15',
|
'NodeCRUDNotification': '1.0-59acc533c11d306f149846f922739c15',
|
||||||
'NodeCRUDPayload': '1.10-49590dee863c5ed1193f5deae0a0a2f2',
|
'NodeCRUDPayload': '1.11-f1c6a6b099e8e28f55378c448c033de0',
|
||||||
'PortCRUDNotification': '1.0-59acc533c11d306f149846f922739c15',
|
'PortCRUDNotification': '1.0-59acc533c11d306f149846f922739c15',
|
||||||
'PortCRUDPayload': '1.3-21235916ed54a91b2a122f59571194e7',
|
'PortCRUDPayload': '1.3-21235916ed54a91b2a122f59571194e7',
|
||||||
'NodeMaintenanceNotification': '1.0-59acc533c11d306f149846f922739c15',
|
'NodeMaintenanceNotification': '1.0-59acc533c11d306f149846f922739c15',
|
||||||
|
@ -0,0 +1,6 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- |
|
||||||
|
Adds a ``description`` field to the node object to enable operators to
|
||||||
|
store any information relates to the node. The field is up to 4096
|
||||||
|
characters.
|
Loading…
Reference in New Issue
Block a user