pcs/store-clone-instance-id-in-resource-status-dtos.patch

813 lines
32 KiB
Diff
Raw Normal View History

2024-03-21 08:42:11 +08:00
From 7c56001aa76c4a5f69f29b328061c419c7ce856b Mon Sep 17 00:00:00 2001
From: Peter Romancik <promanci@redhat.com>
Date: Thu, 1 Feb 2024 17:17:40 +0100
Subject: [PATCH 1/2] further fixes after review
---
pcs/common/reports/codes.py | 2 +-
pcs/common/reports/messages.py | 8 +-
pcs/lib/pacemaker/status.py | 85 ++++++++++---------
.../tier0/common/reports/test_messages.py | 14 +--
pcs_test/tier0/lib/commands/test_status.py | 4 +-
pcs_test/tier0/lib/pacemaker/test_status.py | 68 ++++++++-------
6 files changed, 97 insertions(+), 84 deletions(-)
diff --git a/pcs/common/reports/codes.py b/pcs/common/reports/codes.py
index f9614331..e967d0b1 100644
--- a/pcs/common/reports/codes.py
+++ b/pcs/common/reports/codes.py
@@ -50,7 +50,7 @@ AGENT_SELF_VALIDATION_SKIPPED_UPDATED_RESOURCE_MISCONFIGURED = M(
)
AGENT_SELF_VALIDATION_RESULT = M("AGENT_SELF_VALIDATION_RESULT")
BAD_CLUSTER_STATE_FORMAT = M("BAD_CLUSTER_STATE_FORMAT")
-BAD_CLUSTER_STATE = M("BAD_CLUSTER_STATE")
+BAD_CLUSTER_STATE_DATA = M("BAD_CLUSTER_STATE_DATA")
BOOTH_ADDRESS_DUPLICATION = M("BOOTH_ADDRESS_DUPLICATION")
BOOTH_ALREADY_IN_CIB = M("BOOTH_ALREADY_IN_CIB")
BOOTH_AUTHFILE_NOT_USED = M("BOOTH_AUTHFILE_NOT_USED")
diff --git a/pcs/common/reports/messages.py b/pcs/common/reports/messages.py
index 8b9bc63e..53f15170 100644
--- a/pcs/common/reports/messages.py
+++ b/pcs/common/reports/messages.py
@@ -3277,7 +3277,7 @@ class BadClusterStateFormat(ReportItemMessage):
@dataclass(frozen=True)
-class BadClusterState(ReportItemMessage):
+class BadClusterStateData(ReportItemMessage):
"""
crm_mon xml output is invalid despite conforming to the schema
@@ -3285,13 +3285,13 @@ class BadClusterState(ReportItemMessage):
"""
reason: Optional[str] = None
- _code = codes.BAD_CLUSTER_STATE
+ _code = codes.BAD_CLUSTER_STATE_DATA
@property
def message(self) -> str:
return (
"Cannot load cluster status, xml does not describe valid cluster "
- f"status{format_optional(self.reason, template=': {}')}."
+ f"status{format_optional(self.reason, template=': {}')}"
)
@@ -3314,7 +3314,7 @@ class ClusterStatusBundleMemberIdAsImplicit(ReportItemMessage):
return (
"Skipping bundle '{bundle_id}': {resource_word} "
"{bad_ids} {has} the same id as some of the "
- "implicit bundle resources."
+ "implicit bundle resources"
).format(
bundle_id=self.bundle_id,
resource_word=format_plural(self.bad_ids, "resource"),
diff --git a/pcs/lib/pacemaker/status.py b/pcs/lib/pacemaker/status.py
index a86ede55..deb8aa0d 100644
--- a/pcs/lib/pacemaker/status.py
+++ b/pcs/lib/pacemaker/status.py
@@ -2,7 +2,6 @@ from collections import Counter
from typing import (
Optional,
Sequence,
- Union,
cast,
)
@@ -60,11 +59,13 @@ class UnexpectedMemberError(ClusterStatusParsingError):
resource_id: str,
resource_type: str,
member_id: str,
+ member_type: str,
expected_types: list[str],
):
super().__init__(resource_id)
self.resource_type = resource_type
self.member_id = member_id
+ self.member_type = member_type
self.expected_types = expected_types
@@ -106,46 +107,44 @@ def cluster_status_parsing_error_to_report(
) -> reports.ReportItem:
reason = ""
if isinstance(e, EmptyResourceIdError):
- reason = "Resource with empty id."
+ reason = "Resource with an empty id"
elif isinstance(e, EmptyNodeNameError):
reason = (
- f"Resource with id '{e.resource_id}' contains node with empty name."
+ f"Resource '{e.resource_id}' contains a node with an empty name"
)
elif isinstance(e, UnknownPcmkRoleError):
reason = (
- f"Resource with id '{e.resource_id}' contains unknown "
- f"pcmk role '{e.role}'."
+ f"Resource '{e.resource_id}' contains an unknown "
+ f"role '{e.role}'"
)
elif isinstance(e, UnexpectedMemberError):
reason = (
- f"Unexpected resource '{e.member_id}' inside of resource "
- f"'{e.resource_id}' of type '{e.resource_type}'. "
- f"Only resources of type {format_list(e.expected_types, '|')} "
- f"can be in {e.resource_type}."
+ f"Unexpected resource '{e.member_id}' of type '{e.member_type}' "
+ f"inside of resource '{e.resource_id}' of type '{e.resource_type}'."
+ f" Only resources of type {format_list(e.expected_types)} "
+ f"can be in a {e.resource_type}"
)
elif isinstance(e, MixedMembersError):
- reason = (
- f"Primitive and group members mixed in clone '{e.resource_id}'."
- )
+ reason = f"Primitive and group members mixed in clone '{e.resource_id}'"
elif isinstance(e, DifferentMemberIdsError):
- reason = f"Members with different ids in resource '{e.resource_id}'."
+ reason = f"Members with different ids in clone '{e.resource_id}'"
elif isinstance(e, BundleReplicaMissingImplicitResourceError):
reason = (
f"Replica '{e.replica_id}' of bundle '{e.resource_id}' "
- f"is missing implicit {e.implicit_type} resource."
+ f"is missing implicit {e.implicit_type} resource"
)
elif isinstance(e, BundleReplicaInvalidMemberCountError):
reason = (
f"Replica '{e.replica_id}' of bundle '{e.resource_id}' has "
- "invalid number of members."
+ "invalid number of members"
)
elif isinstance(e, BundleDifferentReplicas):
- reason = f"Replicas of bundle '{e.resource_id}' are not the same."
+ reason = f"Replicas of bundle '{e.resource_id}' are not the same"
return reports.ReportItem(
reports.ReportItemSeverity.error(),
- reports.messages.BadClusterState(reason),
+ reports.messages.BadClusterStateData(reason),
)
@@ -160,7 +159,7 @@ def _primitive_to_dto(
target_role = _get_target_role(primitive_el)
node_names = [
- str(node.get("name")) for node in primitive_el.iterfind("node")
+ str(node.attrib["name"]) for node in primitive_el.iterfind("node")
]
if node_names and any(not name for name in node_names):
@@ -168,7 +167,7 @@ def _primitive_to_dto(
return PrimitiveStatusDto(
resource_id,
- str(primitive_el.get("resource_agent")),
+ str(primitive_el.attrib["resource_agent"]),
role,
target_role,
is_true(primitive_el.get("active", "false")),
@@ -179,7 +178,7 @@ def _primitive_to_dto(
is_true(primitive_el.get("failed", "false")),
is_true(primitive_el.get("managed", "false")),
is_true(primitive_el.get("failure_ignored", "false")),
- [str(node.get("name")) for node in primitive_el.iterfind("node")],
+ node_names,
primitive_el.get("pending"),
primitive_el.get("locked_to"),
)
@@ -197,7 +196,11 @@ def _group_to_dto(
member_list.append(_primitive_to_dto(member, remove_clone_suffix))
else:
raise UnexpectedMemberError(
- group_id, "group", str(member.get("id")), ["primitive"]
+ group_id,
+ "group",
+ str(member.attrib["id"]),
+ member.tag,
+ ["primitive"],
)
return GroupStatusDto(
@@ -228,29 +231,28 @@ def _clone_to_dto(
group_list.append(_group_to_dto(member, is_unique))
else:
raise UnexpectedMemberError(
- clone_id, "clone", str(member.get("id")), ["primitive", "group"]
+ clone_id,
+ "clone",
+ str(member.attrib["id"]),
+ member.tag,
+ ["primitive", "group"],
)
if primitive_list and group_list:
raise MixedMembersError(clone_id)
- instance_list: Union[list[PrimitiveStatusDto], list[GroupStatusDto]]
if primitive_list:
if len(set(res.resource_id for res in primitive_list)) > 1:
raise DifferentMemberIdsError(clone_id)
- instance_list = primitive_list
- else:
+ if group_list:
group_ids = set(group.resource_id for group in group_list)
children_ids = set(
tuple(child.resource_id for child in group.members)
for group in group_list
)
-
if len(group_ids) > 1 or len(children_ids) > 1:
raise DifferentMemberIdsError(clone_id)
- instance_list = group_list
-
return CloneStatusDto(
clone_id,
is_true(clone_el.get("multi_state", "false")),
@@ -262,7 +264,7 @@ def _clone_to_dto(
is_true(clone_el.get("failed", "false")),
is_true(clone_el.get("failure_ignored", "false")),
target_role,
- instance_list,
+ primitive_list or group_list,
)
@@ -270,7 +272,7 @@ def _bundle_to_dto(
bundle_el: _Element, _remove_clone_suffix: bool = False
) -> BundleStatusDto:
bundle_id = _get_resource_id(bundle_el)
- bundle_type = str(bundle_el.get("type"))
+ bundle_type = str(bundle_el.attrib["type"])
replica_list = [
_replica_to_dto(replica, bundle_id, bundle_type)
@@ -283,7 +285,7 @@ def _bundle_to_dto(
return BundleStatusDto(
bundle_id,
bundle_type,
- str(bundle_el.get("image")),
+ str(bundle_el.attrib["image"]),
is_true(bundle_el.get("unique", "false")),
is_true(bundle_el.get("maintenance", "false")),
bundle_el.get("description"),
@@ -302,17 +304,18 @@ class ClusterStatusParser:
}
def __init__(self, status: _Element):
- self.status = status
- self.warnings: reports.ReportItemList = []
+ """
+ status -- xml element from crm_mon xml, validated using the appropriate
+ rng schema
+ """
+ self._status = status
+ self._warnings: reports.ReportItemList = []
def status_xml_to_dto(self) -> ResourcesStatusDto:
"""
Return dto containing status of configured resources in the cluster
-
- status -- status xml document from crm_mon, validated using
- the appropriate rng schema
"""
- resource_list = cast(list[_Element], self.status.xpath("resources/*"))
+ resource_list = cast(list[_Element], self._status.xpath("resources/*"))
resource_dto_list = []
for resource in resource_list:
@@ -328,7 +331,7 @@ class ClusterStatusParser:
# the implicitly created resource.
# We only skip such bundles while still providing status of the
# other resources.
- self.warnings.append(
+ self._warnings.append(
reports.ReportItem.warning(
reports.messages.ClusterStatusBundleMemberIdAsImplicit(
e.bundle_id, e.bad_ids
@@ -339,11 +342,11 @@ class ClusterStatusParser:
return ResourcesStatusDto(resource_dto_list)
def get_warnings(self) -> reports.ReportItemList:
- return self.warnings
+ return self._warnings
def _get_resource_id(resource: _Element) -> str:
- resource_id = resource.get("id")
+ resource_id = resource.attrib["id"]
if not resource_id:
raise EmptyResourceIdError()
return str(resource_id)
@@ -374,7 +377,7 @@ def _remove_clone_suffix(resource_id: str) -> str:
def _replica_to_dto(
replica_el: _Element, bundle_id: str, bundle_type: str
) -> BundleReplicaStatusDto:
- replica_id = str(replica_el.get("id"))
+ replica_id = str(replica_el.attrib["id"])
resource_list = [
_primitive_to_dto(resource)
diff --git a/pcs_test/tier0/common/reports/test_messages.py b/pcs_test/tier0/common/reports/test_messages.py
index 48eb730c..0ca95920 100644
--- a/pcs_test/tier0/common/reports/test_messages.py
+++ b/pcs_test/tier0/common/reports/test_messages.py
@@ -2195,23 +2195,23 @@ class BadClusterStateFormat(NameBuildTest):
)
-class BadClusterState(NameBuildTest):
+class BadClusterStateData(NameBuildTest):
def test_no_reason(self):
self.assert_message_from_report(
(
"Cannot load cluster status, xml does not describe "
- "valid cluster status."
+ "valid cluster status"
),
- reports.BadClusterState(),
+ reports.BadClusterStateData(),
)
def test_reason(self):
self.assert_message_from_report(
(
"Cannot load cluster status, xml does not describe "
- "valid cluster status: sample reason."
+ "valid cluster status: sample reason"
),
- reports.BadClusterState("sample reason"),
+ reports.BadClusterStateData("sample reason"),
)
@@ -5843,7 +5843,7 @@ class ClusterStatusBundleMemberIdAsImplicit(NameBuildTest):
self.assert_message_from_report(
(
"Skipping bundle 'resource-bundle': resource 'resource' has "
- "the same id as some of the implicit bundle resources."
+ "the same id as some of the implicit bundle resources"
),
reports.ClusterStatusBundleMemberIdAsImplicit(
"resource-bundle", ["resource"]
@@ -5855,7 +5855,7 @@ class ClusterStatusBundleMemberIdAsImplicit(NameBuildTest):
(
"Skipping bundle 'resource-bundle': resources 'resource-0', "
"'resource-1' have the same id as some of the implicit bundle "
- "resources."
+ "resources"
),
reports.ClusterStatusBundleMemberIdAsImplicit(
"resource-bundle", ["resource-0", "resource-1"]
diff --git a/pcs_test/tier0/lib/commands/test_status.py b/pcs_test/tier0/lib/commands/test_status.py
index 3b6b7665..b12e9531 100644
--- a/pcs_test/tier0/lib/commands/test_status.py
+++ b/pcs_test/tier0/lib/commands/test_status.py
@@ -1342,8 +1342,8 @@ class ResourcesStatus(TestCase):
lambda: status.resources_status(self.env_assist.get_env()),
[
fixture.error(
- report_codes.BAD_CLUSTER_STATE,
- reason="Resource with id 'R7' contains unknown pcmk role 'NotPcmkRole'.",
+ report_codes.BAD_CLUSTER_STATE_DATA,
+ reason="Resource 'R7' contains an unknown role 'NotPcmkRole'",
),
],
False,
diff --git a/pcs_test/tier0/lib/pacemaker/test_status.py b/pcs_test/tier0/lib/pacemaker/test_status.py
index 778e97a6..ced1a47e 100644
--- a/pcs_test/tier0/lib/pacemaker/test_status.py
+++ b/pcs_test/tier0/lib/pacemaker/test_status.py
@@ -12,6 +12,7 @@ from pcs.common import reports
from pcs.common.const import (
PCMK_ROLE_STARTED,
PCMK_ROLES,
+ PCMK_STATUS_ROLE_PROMOTED,
PCMK_STATUS_ROLE_STARTED,
PCMK_STATUS_ROLE_STOPPED,
PCMK_STATUS_ROLE_UNPROMOTED,
@@ -334,8 +335,8 @@ class TestParsingErrorToReport(TestCase):
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
- reason="Resource with empty id.",
+ reports.codes.BAD_CLUSTER_STATE_DATA,
+ reason="Resource with an empty id",
),
)
@@ -346,8 +347,8 @@ class TestParsingErrorToReport(TestCase):
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
- reason="Resource with id 'resource' contains node with empty name.",
+ reports.codes.BAD_CLUSTER_STATE_DATA,
+ reason="Resource 'resource' contains a node with an empty name",
),
)
@@ -358,25 +359,25 @@ class TestParsingErrorToReport(TestCase):
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
- reason="Resource with id 'resource' contains unknown pcmk role 'NotPcmkRole'.",
+ reports.codes.BAD_CLUSTER_STATE_DATA,
+ reason="Resource 'resource' contains an unknown role 'NotPcmkRole'",
),
)
def test_unexpected_member_group(self):
report = status.cluster_status_parsing_error_to_report(
status.UnexpectedMemberError(
- "resource", "group", "member", ["primitive"]
+ "resource", "group", "member", "bundle", ["primitive"]
)
)
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
+ reports.codes.BAD_CLUSTER_STATE_DATA,
reason=(
- "Unexpected resource 'member' inside of resource "
- "'resource' of type 'group'. Only resources of type "
- "'primitive' can be in group."
+ "Unexpected resource 'member' of type 'bundle' inside of "
+ "resource 'resource' of type 'group'. Only resources of "
+ "type 'primitive' can be in a group"
),
),
)
@@ -384,17 +385,17 @@ class TestParsingErrorToReport(TestCase):
def test_unexpected_member_clone(self):
report = status.cluster_status_parsing_error_to_report(
status.UnexpectedMemberError(
- "resource", "clone", "member", ["primitive", "group"]
+ "resource", "clone", "member", "bundle", ["primitive", "group"]
)
)
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
+ reports.codes.BAD_CLUSTER_STATE_DATA,
reason=(
- "Unexpected resource 'member' inside of resource "
- "'resource' of type 'clone'. Only resources of type "
- "'group'|'primitive' can be in clone."
+ "Unexpected resource 'member' of type 'bundle' inside of "
+ "resource 'resource' of type 'clone'. Only resources of "
+ "type 'group', 'primitive' can be in a clone"
),
),
)
@@ -406,8 +407,8 @@ class TestParsingErrorToReport(TestCase):
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
- reason="Primitive and group members mixed in clone 'resource'.",
+ reports.codes.BAD_CLUSTER_STATE_DATA,
+ reason="Primitive and group members mixed in clone 'resource'",
),
)
@@ -418,8 +419,8 @@ class TestParsingErrorToReport(TestCase):
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
- reason="Members with different ids in resource 'resource'.",
+ reports.codes.BAD_CLUSTER_STATE_DATA,
+ reason="Members with different ids in clone 'resource'",
),
)
@@ -432,8 +433,8 @@ class TestParsingErrorToReport(TestCase):
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
- reason="Replica '0' of bundle 'resource' is missing implicit container resource.",
+ reports.codes.BAD_CLUSTER_STATE_DATA,
+ reason="Replica '0' of bundle 'resource' is missing implicit container resource",
),
)
@@ -444,8 +445,8 @@ class TestParsingErrorToReport(TestCase):
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
- reason="Replica '0' of bundle 'resource' has invalid number of members.",
+ reports.codes.BAD_CLUSTER_STATE_DATA,
+ reason="Replica '0' of bundle 'resource' has invalid number of members",
),
)
@@ -456,8 +457,8 @@ class TestParsingErrorToReport(TestCase):
assert_report_item_equal(
report,
fixture.error(
- reports.codes.BAD_CLUSTER_STATE,
- reason="Replicas of bundle 'resource' are not the same.",
+ reports.codes.BAD_CLUSTER_STATE_DATA,
+ reason="Replicas of bundle 'resource' are not the same",
),
)
@@ -549,6 +550,7 @@ class TestPrimitiveStatusToDto(TestCase):
with self.assertRaises(status.UnknownPcmkRoleError) as cm:
status._primitive_to_dto(primitive_xml)
self.assertEqual(cm.exception.resource_id, "resource")
+ self.assertEqual(cm.exception.role, "NotPcmkRole")
def test_target_role(self):
for role in PCMK_ROLES:
@@ -573,6 +575,7 @@ class TestPrimitiveStatusToDto(TestCase):
with self.assertRaises(status.UnknownPcmkRoleError) as cm:
status._primitive_to_dto(primitive_xml)
self.assertEqual(cm.exception.resource_id, "resource")
+ self.assertEqual(cm.exception.role, value)
class TestGroupStatusToDto(TestCase):
@@ -695,7 +698,11 @@ class TestGroupStatusToDto(TestCase):
with self.assertRaises(status.UnexpectedMemberError) as cm:
status._group_to_dto(group_xml)
self.assertEqual(cm.exception.resource_id, "outer-group")
+ self.assertEqual(cm.exception.resource_type, "group")
self.assertEqual(cm.exception.member_id, resource_id)
+ self.assertEqual(
+ cm.exception.member_type, resource_id.split("-")[1]
+ )
self.assertEqual(cm.exception.expected_types, ["primitive"])
def test_remove_clone_suffix(self):
@@ -796,7 +803,7 @@ class TestCloneStatusToDto(TestCase):
fixture_clone_xml(
multi_state=True,
instances=[
- fixture_primitive_xml(role=PCMK_STATUS_ROLE_UNPROMOTED),
+ fixture_primitive_xml(role=PCMK_STATUS_ROLE_PROMOTED),
fixture_primitive_xml(
role=PCMK_STATUS_ROLE_UNPROMOTED, node_names=["node2"]
),
@@ -810,7 +817,7 @@ class TestCloneStatusToDto(TestCase):
fixture_clone_dto(
multi_state=True,
instances=[
- fixture_primitive_dto(role=PCMK_STATUS_ROLE_UNPROMOTED),
+ fixture_primitive_dto(role=PCMK_STATUS_ROLE_PROMOTED),
fixture_primitive_dto(
role=PCMK_STATUS_ROLE_UNPROMOTED, node_names=["node2"]
),
@@ -1003,9 +1010,12 @@ class TestCloneStatusToDto(TestCase):
with self.assertRaises(status.UnexpectedMemberError) as cm:
status._clone_to_dto(clone_xml)
-
self.assertEqual(cm.exception.resource_id, "outer-clone")
+ self.assertEqual(cm.exception.resource_type, "clone")
self.assertEqual(cm.exception.member_id, resource_id)
+ self.assertEqual(
+ cm.exception.member_type, resource_id.split("-")[1]
+ )
self.assertEqual(
cm.exception.expected_types, ["primitive", "group"]
)
--
2.25.1
From c32249a39ef262e3f2106eb8ca01b6efb8e74707 Mon Sep 17 00:00:00 2001
From: Peter Romancik <promanci@redhat.com>
Date: Thu, 1 Feb 2024 17:45:20 +0100
Subject: [PATCH 2/2] store clone instance id in resource status dtos
---
pcs/common/status_dto.py | 2 ++
pcs/lib/pacemaker/status.py | 19 +++++++----
pcs_test/tier0/lib/commands/test_status.py | 3 ++
pcs_test/tier0/lib/pacemaker/test_status.py | 36 ++++++++++++++++-----
4 files changed, 46 insertions(+), 14 deletions(-)
diff --git a/pcs/common/status_dto.py b/pcs/common/status_dto.py
index dcc94eca..240ff930 100644
--- a/pcs/common/status_dto.py
+++ b/pcs/common/status_dto.py
@@ -16,6 +16,7 @@ from pcs.common.interface.dto import DataTransferObject
class PrimitiveStatusDto(DataTransferObject):
# pylint: disable=too-many-instance-attributes
resource_id: str
+ clone_instance_id: Optional[str]
resource_agent: str
role: PcmkStatusRoleType
target_role: Optional[PcmkRoleType]
@@ -35,6 +36,7 @@ class PrimitiveStatusDto(DataTransferObject):
@dataclass(frozen=True)
class GroupStatusDto(DataTransferObject):
resource_id: str
+ clone_instance_id: Optional[str]
maintenance: bool
description: Optional[str]
managed: bool
diff --git a/pcs/lib/pacemaker/status.py b/pcs/lib/pacemaker/status.py
index deb8aa0d..6b37d6cb 100644
--- a/pcs/lib/pacemaker/status.py
+++ b/pcs/lib/pacemaker/status.py
@@ -152,8 +152,9 @@ def _primitive_to_dto(
primitive_el: _Element, remove_clone_suffix: bool = False
) -> PrimitiveStatusDto:
resource_id = _get_resource_id(primitive_el)
+ clone_suffix = None
if remove_clone_suffix:
- resource_id = _remove_clone_suffix(resource_id)
+ resource_id, clone_suffix = _remove_clone_suffix(resource_id)
role = _get_role(primitive_el)
target_role = _get_target_role(primitive_el)
@@ -167,6 +168,7 @@ def _primitive_to_dto(
return PrimitiveStatusDto(
resource_id,
+ clone_suffix,
str(primitive_el.attrib["resource_agent"]),
role,
target_role,
@@ -187,8 +189,11 @@ def _primitive_to_dto(
def _group_to_dto(
group_el: _Element, remove_clone_suffix: bool = False
) -> GroupStatusDto:
- # clone suffix is added even when the clone is non unique
- group_id = _remove_clone_suffix(_get_resource_id(group_el))
+ # clone instance id present even when the clone is non unique
+ group_id, clone_instance_id = _remove_clone_suffix(
+ _get_resource_id(group_el)
+ )
+
member_list = []
for member in group_el:
@@ -205,6 +210,7 @@ def _group_to_dto(
return GroupStatusDto(
group_id,
+ clone_instance_id,
is_true(group_el.get("maintenance", "false")),
group_el.get("description"),
is_true(group_el.get("managed", "false")),
@@ -368,10 +374,11 @@ def _get_target_role(resource: _Element) -> Optional[PcmkRoleType]:
return PcmkRoleType(target_role)
-def _remove_clone_suffix(resource_id: str) -> str:
+def _remove_clone_suffix(resource_id: str) -> tuple[str, Optional[str]]:
if ":" in resource_id:
- return resource_id.rsplit(":", 1)[0]
- return resource_id
+ resource_id, clone_suffix = resource_id.rsplit(":", 1)
+ return resource_id, clone_suffix
+ return resource_id, None
def _replica_to_dto(
diff --git a/pcs_test/tier0/lib/commands/test_status.py b/pcs_test/tier0/lib/commands/test_status.py
index b12e9531..c7c808a3 100644
--- a/pcs_test/tier0/lib/commands/test_status.py
+++ b/pcs_test/tier0/lib/commands/test_status.py
@@ -1280,6 +1280,7 @@ def _fixture_primitive_resource_dto(
) -> PrimitiveStatusDto:
return PrimitiveStatusDto(
resource_id=resource_id,
+ clone_instance_id=None,
resource_agent=resource_agent,
role=PCMK_STATUS_ROLE_STOPPED,
target_role=target_role,
@@ -1448,6 +1449,7 @@ class ResourcesStatus(TestCase):
),
GroupStatusDto(
resource_id="G2",
+ clone_instance_id=None,
maintenance=False,
description=None,
managed=True,
@@ -1475,6 +1477,7 @@ class ResourcesStatus(TestCase):
instances=[
GroupStatusDto(
resource_id="G1",
+ clone_instance_id="0",
maintenance=False,
description=None,
managed=True,
diff --git a/pcs_test/tier0/lib/pacemaker/test_status.py b/pcs_test/tier0/lib/pacemaker/test_status.py
index ced1a47e..a852d45b 100644
--- a/pcs_test/tier0/lib/pacemaker/test_status.py
+++ b/pcs_test/tier0/lib/pacemaker/test_status.py
@@ -85,6 +85,7 @@ def fixture_primitive_xml(
def fixture_primitive_dto(
resource_id: str = "resource",
+ clone_instance_id: Optional[str] = None,
resource_agent: str = "ocf:heartbeat:Dummy",
role: PcmkStatusRoleType = PCMK_STATUS_ROLE_STARTED,
target_role: Optional[str] = None,
@@ -94,6 +95,7 @@ def fixture_primitive_dto(
) -> PrimitiveStatusDto:
return PrimitiveStatusDto(
resource_id,
+ clone_instance_id,
resource_agent,
role,
target_role,
@@ -136,11 +138,13 @@ def fixture_group_xml(
def fixture_group_dto(
resource_id: str = "resource-group",
+ clone_instance_id: Optional[str] = None,
description: Optional[str] = None,
members: Sequence[PrimitiveStatusDto] = (),
) -> GroupStatusDto:
return GroupStatusDto(
resource_id,
+ clone_instance_id,
maintenance=False,
description=description,
managed=True,
@@ -506,7 +510,7 @@ class TestPrimitiveStatusToDto(TestCase):
result = status._primitive_to_dto(primitive_xml, True)
- self.assertEqual(result, fixture_primitive_dto())
+ self.assertEqual(result, fixture_primitive_dto(clone_instance_id="0"))
def test_running_on_multiple_nodes(self):
primitive_xml = etree.fromstring(
@@ -716,7 +720,10 @@ class TestGroupStatusToDto(TestCase):
result = status._group_to_dto(group_xml, True)
self.assertEqual(
result,
- fixture_group_dto(members=[fixture_primitive_dto()]),
+ fixture_group_dto(
+ clone_instance_id="0",
+ members=[fixture_primitive_dto(clone_instance_id="0")],
+ ),
)
@@ -792,8 +799,10 @@ class TestCloneStatusToDto(TestCase):
fixture_clone_dto(
unique=True,
instances=[
- fixture_primitive_dto(),
- fixture_primitive_dto(node_names=["node2"]),
+ fixture_primitive_dto(clone_instance_id="0"),
+ fixture_primitive_dto(
+ clone_instance_id="1", node_names=["node2"]
+ ),
],
),
)
@@ -886,9 +895,12 @@ class TestCloneStatusToDto(TestCase):
result,
fixture_clone_dto(
instances=[
- fixture_group_dto(members=[fixture_primitive_dto()]),
fixture_group_dto(
- members=[fixture_primitive_dto(node_names=["node2"])]
+ clone_instance_id="0", members=[fixture_primitive_dto()]
+ ),
+ fixture_group_dto(
+ clone_instance_id="1",
+ members=[fixture_primitive_dto(node_names=["node2"])],
),
],
),
@@ -923,9 +935,17 @@ class TestCloneStatusToDto(TestCase):
fixture_clone_dto(
unique=True,
instances=[
- fixture_group_dto(members=[fixture_primitive_dto()]),
fixture_group_dto(
- members=[fixture_primitive_dto(node_names=["node2"])]
+ clone_instance_id="0",
+ members=[fixture_primitive_dto(clone_instance_id="0")],
+ ),
+ fixture_group_dto(
+ clone_instance_id="1",
+ members=[
+ fixture_primitive_dto(
+ clone_instance_id="1", node_names=["node2"]
+ )
+ ],
),
],
),
--
2.25.1