Blame SOURCES/add-missing-file-test_stonith_update_scsi_devices.py.patch

ce1222
From e3f9823283517bafa8d309fb6148539e0e8ecdb2 Mon Sep 17 00:00:00 2001
ce1222
From: Miroslav Lisik <mlisik@redhat.com>
ce1222
Date: Fri, 10 Sep 2021 11:40:03 +0200
ce1222
Subject: [PATCH] add missing file test_stonith_update_scsi_devices.py
ce1222
ce1222
---
ce1222
 .../test_stonith_update_scsi_devices.py       | 1153 +++++++++++++++++
ce1222
 1 file changed, 1153 insertions(+)
ce1222
 create mode 100644 pcs_test/tier0/lib/commands/test_stonith_update_scsi_devices.py
ce1222
ce1222
diff --git a/pcs_test/tier0/lib/commands/test_stonith_update_scsi_devices.py b/pcs_test/tier0/lib/commands/test_stonith_update_scsi_devices.py
ce1222
new file mode 100644
ce1222
index 0000000..3bc5132
ce1222
--- /dev/null
ce1222
+++ b/pcs_test/tier0/lib/commands/test_stonith_update_scsi_devices.py
ce1222
@@ -0,0 +1,1153 @@
ce1222
+import json
ce1222
+from unittest import mock, TestCase
ce1222
+
ce1222
+
ce1222
+from pcs_test.tools import fixture
ce1222
+from pcs_test.tools.command_env import get_env_tools
ce1222
+from pcs_test.tools.misc import get_test_resource as rc
ce1222
+
ce1222
+from pcs import settings
ce1222
+from pcs.lib.commands import stonith
ce1222
+from pcs.common import (
ce1222
+    communication,
ce1222
+    reports,
ce1222
+)
ce1222
+from pcs.common.interface import dto
ce1222
+from pcs.common.tools import timeout_to_seconds
ce1222
+
ce1222
+from .cluster.common import (
ce1222
+    corosync_conf_fixture,
ce1222
+    get_two_node,
ce1222
+    node_fixture,
ce1222
+)
ce1222
+
ce1222
+SCSI_STONITH_ID = "scsi-fence-device"
ce1222
+SCSI_NODE = "node1"
ce1222
+_DIGEST = "0" * 31
ce1222
+DEFAULT_DIGEST = _DIGEST + "0"
ce1222
+ALL_DIGEST = _DIGEST + "1"
ce1222
+NONPRIVATE_DIGEST = _DIGEST + "2"
ce1222
+NONRELOADABLE_DIGEST = _DIGEST + "3"
ce1222
+DEVICES_1 = ("/dev/sda",)
ce1222
+DEVICES_2 = ("/dev/sda", "/dev/sdb")
ce1222
+DEVICES_3 = ("/dev/sda", "/dev/sdb", "/dev/sdc")
ce1222
+
ce1222
+DEFAULT_MONITOR = ("monitor", "60s", None, None)
ce1222
+DEFAULT_OPS = (DEFAULT_MONITOR,)
ce1222
+DEFAULT_LRM_START_OPS = (("0", DEFAULT_DIGEST, None, None),)
ce1222
+DEFAULT_LRM_MONITOR_OPS = (("60000", DEFAULT_DIGEST, None, None),)
ce1222
+DEFAULT_LRM_START_OPS_UPDATED = (("0", ALL_DIGEST, None, None),)
ce1222
+DEFAULT_LRM_MONITOR_OPS_UPDATED = (("60000", ALL_DIGEST, None, None),)
ce1222
+
ce1222
+
ce1222
+def _fixture_ops(resource_id, ops):
ce1222
+    return "\n".join(
ce1222
+        [
ce1222
+            (
ce1222
+                '
ce1222
+                ' interval="{interval}" {timeout} name="{name}"/>'
ce1222
+            ).format(
ce1222
+                resource_id=resource_id,
ce1222
+                name=name,
ce1222
+                _interval=_interval if _interval else interval,
ce1222
+                interval=interval,
ce1222
+                timeout=f'timeout="{timeout}"' if timeout else "",
ce1222
+            )
ce1222
+            for name, interval, timeout, _interval in ops
ce1222
+        ]
ce1222
+    )
ce1222
+
ce1222
+
ce1222
+def _fixture_devices_nvpair(resource_id, devices):
ce1222
+    if devices is None:
ce1222
+        return ""
ce1222
+    return (
ce1222
+        '
ce1222
+        ' value="{devices}"/>'
ce1222
+    ).format(resource_id=resource_id, devices=",".join(sorted(devices)))
ce1222
+
ce1222
+
ce1222
+def fixture_scsi(
ce1222
+    stonith_id=SCSI_STONITH_ID, devices=DEVICES_1, resource_ops=DEFAULT_OPS
ce1222
+):
ce1222
+    return """
ce1222
+        <resources>
ce1222
+            <primitive class="stonith" id="{stonith_id}" type="fence_scsi">
ce1222
+                <instance_attributes id="{stonith_id}-instance_attributes">
ce1222
+                    {devices}
ce1222
+                    <nvpair id="{stonith_id}-instance_attributes-pcmk_host_check" name="pcmk_host_check" value="static-list"/>
ce1222
+                    <nvpair id="{stonith_id}-instance_attributes-pcmk_host_list" name="pcmk_host_list" value="node1 node2 node3"/>
ce1222
+                    <nvpair id="{stonith_id}-instance_attributes-pcmk_reboot_action" name="pcmk_reboot_action" value="off"/>
ce1222
+                </instance_attributes>
ce1222
+                <meta_attributes id="{stonith_id}-meta_attributes">
ce1222
+                    <nvpair id="{stonith_id}-meta_attributes-provides" name="provides" value="unfencing"/>
ce1222
+                </meta_attributes>
ce1222
+                <operations>
ce1222
+                    {operations}
ce1222
+                </operations>
ce1222
+            </primitive>
ce1222
+            <primitive class="ocf" id="dummy" provider="pacemaker" type="Dummy"/>
ce1222
+        </resources>
ce1222
+    """.format(
ce1222
+        stonith_id=stonith_id,
ce1222
+        devices=_fixture_devices_nvpair(stonith_id, devices),
ce1222
+        operations=_fixture_ops(stonith_id, resource_ops),
ce1222
+    )
ce1222
+
ce1222
+
ce1222
+def _fixture_lrm_rsc_ops(op_type, resource_id, lrm_ops):
ce1222
+    return [
ce1222
+        (
ce1222
+            '
ce1222
+            'interval="{ms}" {_all} {secure} {restart}/>'
ce1222
+        ).format(
ce1222
+            op_type_id="last" if op_type == "start" else op_type,
ce1222
+            op_type=op_type,
ce1222
+            resource_id=resource_id,
ce1222
+            ms=ms,
ce1222
+            _all=f'op-digest="{_all}"' if _all else "",
ce1222
+            secure=f'op-secure-digest="{secure}"' if secure else "",
ce1222
+            restart=f'op-restart-digest="{restart}"' if restart else "",
ce1222
+        )
ce1222
+        for ms, _all, secure, restart in lrm_ops
ce1222
+    ]
ce1222
+
ce1222
+
ce1222
+def _fixture_lrm_rsc_monitor_ops(resource_id, lrm_monitor_ops):
ce1222
+    return _fixture_lrm_rsc_ops("monitor", resource_id, lrm_monitor_ops)
ce1222
+
ce1222
+
ce1222
+def _fixture_lrm_rsc_start_ops(resource_id, lrm_start_ops):
ce1222
+    return _fixture_lrm_rsc_ops("start", resource_id, lrm_start_ops)
ce1222
+
ce1222
+
ce1222
+def _fixture_status_lrm_ops_base(
ce1222
+    resource_id,
ce1222
+    lrm_ops,
ce1222
+):
ce1222
+    return f"""
ce1222
+        <status>
ce1222
+            <node_state id="1" uname="node1">
ce1222
+                <lrm id="1">
ce1222
+                    <lrm_resources>
ce1222
+                        <lrm_resource id="{resource_id}" type="fence_scsi" class="stonith">
ce1222
+                            {lrm_ops}
ce1222
+                        </lrm_resource>
ce1222
+                    </lrm_resources>
ce1222
+                </lrm>
ce1222
+            </node_state>
ce1222
+        </status>
ce1222
+    """
ce1222
+
ce1222
+
ce1222
+def _fixture_status_lrm_ops(
ce1222
+    resource_id,
ce1222
+    lrm_start_ops=DEFAULT_LRM_START_OPS,
ce1222
+    lrm_monitor_ops=DEFAULT_LRM_MONITOR_OPS,
ce1222
+):
ce1222
+    return _fixture_status_lrm_ops_base(
ce1222
+        resource_id,
ce1222
+        "\n".join(
ce1222
+            _fixture_lrm_rsc_start_ops(resource_id, lrm_start_ops)
ce1222
+            + _fixture_lrm_rsc_monitor_ops(resource_id, lrm_monitor_ops)
ce1222
+        ),
ce1222
+    )
ce1222
+
ce1222
+
ce1222
+def fixture_digests_xml(resource_id, node_name, devices=""):
ce1222
+    return f"""
ce1222
+        <pacemaker-result api-version="2.9" request="crm_resource --digests --resource {resource_id} --node {node_name} --output-as xml devices={devices}">
ce1222
+            <digests resource="{resource_id}" node="{node_name}" task="stop" interval="0ms">
ce1222
+                <digest type="all" hash="{ALL_DIGEST}">
ce1222
+                    <parameters devices="{devices}" pcmk_host_check="static-list" pcmk_host_list="node1 node2 node3" pcmk_reboot_action="off"/>
ce1222
+                </digest>
ce1222
+                <digest type="nonprivate" hash="{NONPRIVATE_DIGEST}">
ce1222
+                    <parameters devices="{devices}"/>
ce1222
+                </digest>
ce1222
+            </digests>
ce1222
+            <status code="0" message="OK"/>
ce1222
+        </pacemaker-result>
ce1222
+    """
ce1222
+
ce1222
+
ce1222
+FIXTURE_CRM_MON_RES_RUNNING_1 = f""" <resources> <resource id="{SCSI_STONITH_ID}" resource_agent="stonith:fence_scsi" role="Started" nodes_running_on="1">
ce1222
+            <node name="{SCSI_NODE}" id="1" cached="true"/>
ce1222
+        </resource>
ce1222
+    </resources>
ce1222
+"""
ce1222
+
ce1222
+FIXTURE_CRM_MON_RES_RUNNING_2 = f"""
ce1222
+    <resources>
ce1222
+        <resource id="{SCSI_STONITH_ID}" resource_agent="stonith:fence_scsi" role="Started" nodes_running_on="1">
ce1222
+            <node name="node1" id="1" cached="true"/>
ce1222
+            <node name="node2" id="2" cached="true"/>
ce1222
+        </resource>
ce1222
+    </resources>
ce1222
+"""
ce1222
+FIXTURE_CRM_MON_NODES = """
ce1222
+    <nodes>
ce1222
+        <node name="node1" id="1" is_dc="true" resources_running="1"/>
ce1222
+        <node name="node2" id="2"/>
ce1222
+        <node name="node3" id="3"/>
ce1222
+    </nodes>
ce1222
+"""
ce1222
+
ce1222
+FIXTURE_CRM_MON_RES_STOPPED = f"""
ce1222
+    <resource id="{SCSI_STONITH_ID}" resource_agent="stonith:fence_scsi" role="Stopped" nodes_running_on="0"/>
ce1222
+"""
ce1222
+
ce1222
+
ce1222
+@mock.patch.object(
ce1222
+    settings,
ce1222
+    "pacemaker_api_result_schema",
ce1222
+    rc("pcmk_api_rng/api-result.rng"),
ce1222
+)
ce1222
+class UpdateScsiDevices(TestCase):
ce1222
+    def setUp(self):
ce1222
+        self.env_assist, self.config = get_env_tools(self)
ce1222
+
ce1222
+        self.existing_nodes = ["node1", "node2", "node3"]
ce1222
+        self.existing_corosync_nodes = [
ce1222
+            node_fixture(node, node_id)
ce1222
+            for node_id, node in enumerate(self.existing_nodes, 1)
ce1222
+        ]
ce1222
+        self.config.env.set_known_nodes(self.existing_nodes)
ce1222
+
ce1222
+    def assert_command_success(
ce1222
+        self,
ce1222
+        devices_before=DEVICES_1,
ce1222
+        devices_updated=DEVICES_2,
ce1222
+        resource_ops=DEFAULT_OPS,
ce1222
+        lrm_monitor_ops=DEFAULT_LRM_MONITOR_OPS,
ce1222
+        lrm_start_ops=DEFAULT_LRM_START_OPS,
ce1222
+        lrm_monitor_ops_updated=DEFAULT_LRM_MONITOR_OPS_UPDATED,
ce1222
+        lrm_start_ops_updated=DEFAULT_LRM_START_OPS_UPDATED,
ce1222
+    ):
ce1222
+        # pylint: disable=too-many-locals
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(
ce1222
+            resources=fixture_scsi(
ce1222
+                devices=devices_before, resource_ops=resource_ops
ce1222
+            ),
ce1222
+            status=_fixture_status_lrm_ops(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                lrm_start_ops=lrm_start_ops,
ce1222
+                lrm_monitor_ops=lrm_monitor_ops,
ce1222
+            ),
ce1222
+        )
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_1, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        devices_opt = "devices={}".format(",".join(devices_updated))
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="start.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID, SCSI_NODE, devices=",".join(devices_updated)
ce1222
+            ),
ce1222
+            args=[devices_opt],
ce1222
+        )
ce1222
+
ce1222
+        for num, op in enumerate(resource_ops, 1):
ce1222
+            name, interval, timeout, _ = op
ce1222
+            if name != "monitor":
ce1222
+                continue
ce1222
+            args = [devices_opt]
ce1222
+            args.append(
ce1222
+                "CRM_meta_interval={}".format(
ce1222
+                    1000 * timeout_to_seconds(interval)
ce1222
+                )
ce1222
+            )
ce1222
+            if timeout:
ce1222
+                args.append(
ce1222
+                    "CRM_meta_timeout={}".format(
ce1222
+                        1000 * timeout_to_seconds(timeout)
ce1222
+                    )
ce1222
+                )
ce1222
+            self.config.runner.pcmk.resource_digests(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                SCSI_NODE,
ce1222
+                name=f"{name}-{num}.op.digests",
ce1222
+                stdout=fixture_digests_xml(
ce1222
+                    SCSI_STONITH_ID,
ce1222
+                    SCSI_NODE,
ce1222
+                    devices=",".join(devices_updated),
ce1222
+                ),
ce1222
+                args=args,
ce1222
+            )
ce1222
+        self.config.corosync_conf.load_content(
ce1222
+            corosync_conf_fixture(
ce1222
+                self.existing_corosync_nodes,
ce1222
+                get_two_node(len(self.existing_corosync_nodes)),
ce1222
+            )
ce1222
+        )
ce1222
+        self.config.http.corosync.get_corosync_online_targets(
ce1222
+            node_labels=self.existing_nodes
ce1222
+        )
ce1222
+        self.config.http.scsi.unfence_node(
ce1222
+            devices_updated, node_labels=self.existing_nodes
ce1222
+        )
ce1222
+        self.config.env.push_cib(
ce1222
+            resources=fixture_scsi(
ce1222
+                devices=devices_updated, resource_ops=resource_ops
ce1222
+            ),
ce1222
+            status=_fixture_status_lrm_ops(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                lrm_start_ops=lrm_start_ops_updated,
ce1222
+                lrm_monitor_ops=lrm_monitor_ops_updated,
ce1222
+            ),
ce1222
+        )
ce1222
+        stonith.update_scsi_devices(
ce1222
+            self.env_assist.get_env(), SCSI_STONITH_ID, devices_updated
ce1222
+        )
ce1222
+        self.env_assist.assert_reports([])
ce1222
+
ce1222
+    def test_update_1_to_1_devices(self):
ce1222
+        self.assert_command_success(
ce1222
+            devices_before=DEVICES_1, devices_updated=DEVICES_1
ce1222
+        )
ce1222
+
ce1222
+    def test_update_2_to_2_devices(self):
ce1222
+        self.assert_command_success(
ce1222
+            devices_before=DEVICES_1, devices_updated=DEVICES_1
ce1222
+        )
ce1222
+
ce1222
+    def test_update_1_to_2_devices(self):
ce1222
+        self.assert_command_success()
ce1222
+
ce1222
+    def test_update_1_to_3_devices(self):
ce1222
+        self.assert_command_success(
ce1222
+            devices_before=DEVICES_1, devices_updated=DEVICES_3
ce1222
+        )
ce1222
+
ce1222
+    def test_update_3_to_1_devices(self):
ce1222
+        self.assert_command_success(
ce1222
+            devices_before=DEVICES_3, devices_updated=DEVICES_1
ce1222
+        )
ce1222
+
ce1222
+    def test_update_3_to_2_devices(self):
ce1222
+        self.assert_command_success(
ce1222
+            devices_before=DEVICES_3, devices_updated=DEVICES_2
ce1222
+        )
ce1222
+
ce1222
+    def test_default_monitor(self):
ce1222
+        self.assert_command_success()
ce1222
+
ce1222
+    def test_no_monitor_ops(self):
ce1222
+        self.assert_command_success(
ce1222
+            resource_ops=(), lrm_monitor_ops=(), lrm_monitor_ops_updated=()
ce1222
+        )
ce1222
+
ce1222
+    def test_1_monitor_with_timeout(self):
ce1222
+        self.assert_command_success(
ce1222
+            resource_ops=(("monitor", "30s", "10s", None),),
ce1222
+            lrm_monitor_ops=(("30000", DEFAULT_DIGEST, None, None),),
ce1222
+            lrm_monitor_ops_updated=(("30000", ALL_DIGEST, None, None),),
ce1222
+        )
ce1222
+
ce1222
+    def test_2_monitor_ops_with_timeouts(self):
ce1222
+        self.assert_command_success(
ce1222
+            resource_ops=(
ce1222
+                ("monitor", "30s", "10s", None),
ce1222
+                ("monitor", "40s", "20s", None),
ce1222
+            ),
ce1222
+            lrm_monitor_ops=(
ce1222
+                ("30000", DEFAULT_DIGEST, None, None),
ce1222
+                ("40000", DEFAULT_DIGEST, None, None),
ce1222
+            ),
ce1222
+            lrm_monitor_ops_updated=(
ce1222
+                ("30000", ALL_DIGEST, None, None),
ce1222
+                ("40000", ALL_DIGEST, None, None),
ce1222
+            ),
ce1222
+        )
ce1222
+
ce1222
+    def test_2_monitor_ops_with_one_timeout(self):
ce1222
+        self.assert_command_success(
ce1222
+            resource_ops=(
ce1222
+                ("monitor", "30s", "10s", None),
ce1222
+                ("monitor", "60s", None, None),
ce1222
+            ),
ce1222
+            lrm_monitor_ops=(
ce1222
+                ("30000", DEFAULT_DIGEST, None, None),
ce1222
+                ("60000", DEFAULT_DIGEST, None, None),
ce1222
+            ),
ce1222
+            lrm_monitor_ops_updated=(
ce1222
+                ("30000", ALL_DIGEST, None, None),
ce1222
+                ("60000", ALL_DIGEST, None, None),
ce1222
+            ),
ce1222
+        )
ce1222
+
ce1222
+    def test_various_start_ops_one_lrm_start_op(self):
ce1222
+        self.assert_command_success(
ce1222
+            resource_ops=(
ce1222
+                ("monitor", "60s", None, None),
ce1222
+                ("start", "0s", "40s", None),
ce1222
+                ("start", "0s", "30s", "1"),
ce1222
+                ("start", "10s", "5s", None),
ce1222
+                ("start", "20s", None, None),
ce1222
+            ),
ce1222
+        )
ce1222
+
ce1222
+    def test_1_nonrecurring_start_op_with_timeout(self):
ce1222
+        self.assert_command_success(
ce1222
+            resource_ops=(
ce1222
+                ("monitor", "60s", None, None),
ce1222
+                ("start", "0s", "40s", None),
ce1222
+            ),
ce1222
+        )
ce1222
+
ce1222
+
ce1222
+@mock.patch.object(
ce1222
+    settings,
ce1222
+    "pacemaker_api_result_schema",
ce1222
+    rc("pcmk_api_rng/api-result.rng"),
ce1222
+)
ce1222
+class TestUpdateScsiDevicesFailures(TestCase):
ce1222
+    # pylint: disable=too-many-public-methods
ce1222
+    def setUp(self):
ce1222
+        self.env_assist, self.config = get_env_tools(self)
ce1222
+
ce1222
+        self.existing_nodes = ["node1", "node2", "node3"]
ce1222
+        self.existing_corosync_nodes = [
ce1222
+            node_fixture(node, node_id)
ce1222
+            for node_id, node in enumerate(self.existing_nodes, 1)
ce1222
+        ]
ce1222
+        self.config.env.set_known_nodes(self.existing_nodes)
ce1222
+
ce1222
+    def test_pcmk_doesnt_support_digests(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported(
ce1222
+            is_supported=False
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, ()
ce1222
+            ),
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_OF_SCSI_DEVICES_NOT_SUPPORTED,
ce1222
+                )
ce1222
+            ],
ce1222
+            expected_in_processor=False,
ce1222
+        )
ce1222
+
ce1222
+    def test_devices_cannot_be_empty(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(resources=fixture_scsi())
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, ()
ce1222
+            )
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.INVALID_OPTION_VALUE,
ce1222
+                    option_name="devices",
ce1222
+                    option_value="",
ce1222
+                    allowed_values=None,
ce1222
+                    cannot_be_empty=True,
ce1222
+                    forbidden_characters=None,
ce1222
+                )
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_nonexistant_id(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(resources=fixture_scsi())
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), "non-existent-id", DEVICES_2
ce1222
+            )
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.ID_NOT_FOUND,
ce1222
+                    id="non-existent-id",
ce1222
+                    expected_types=["primitive"],
ce1222
+                    context_type="cib",
ce1222
+                    context_id="",
ce1222
+                )
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_not_a_resource_id(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(resources=fixture_scsi())
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(),
ce1222
+                f"{SCSI_STONITH_ID}-instance_attributes-devices",
ce1222
+                DEVICES_2,
ce1222
+            )
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.ID_BELONGS_TO_UNEXPECTED_TYPE,
ce1222
+                    id=f"{SCSI_STONITH_ID}-instance_attributes-devices",
ce1222
+                    expected_types=["primitive"],
ce1222
+                    current_type="nvpair",
ce1222
+                )
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_not_supported_resource_type(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(resources=fixture_scsi())
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), "dummy", DEVICES_2
ce1222
+            )
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNSUPPORTED_AGENT,
ce1222
+                    resource_id="dummy",
ce1222
+                    resource_type="Dummy",
ce1222
+                    supported_stonith_types=["fence_scsi"],
ce1222
+                )
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_devices_option_missing(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(resources=fixture_scsi(devices=None))
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            )
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason=(
ce1222
+                        "no devices option configured for stonith device "
ce1222
+                        f"'{SCSI_STONITH_ID}'"
ce1222
+                    ),
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_OTHER,
ce1222
+                )
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_devices_option_empty(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(resources=fixture_scsi(devices=""))
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            )
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason=(
ce1222
+                        "no devices option configured for stonith device "
ce1222
+                        f"'{SCSI_STONITH_ID}'"
ce1222
+                    ),
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_OTHER,
ce1222
+                )
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_stonith_resource_is_not_running(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(resources=fixture_scsi())
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_STOPPED, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason=f"resource '{SCSI_STONITH_ID}' is not running on any node",
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_NOT_RUNNING,
ce1222
+                )
ce1222
+            ],
ce1222
+            expected_in_processor=False,
ce1222
+        )
ce1222
+
ce1222
+    def test_stonith_resource_is_running_on_more_than_one_node(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(resources=fixture_scsi())
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_2, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason=(
ce1222
+                        f"resource '{SCSI_STONITH_ID}' is running on more than "
ce1222
+                        "1 node"
ce1222
+                    ),
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_OTHER,
ce1222
+                )
ce1222
+            ],
ce1222
+            expected_in_processor=False,
ce1222
+        )
ce1222
+
ce1222
+    def test_lrm_op_missing_digest_attributes(self):
ce1222
+        devices = ",".join(DEVICES_2)
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(
ce1222
+            resources=fixture_scsi(),
ce1222
+            status=_fixture_status_lrm_ops_base(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                f'<lrm_rsc_op id="{SCSI_STONITH_ID}_last" operation="start"/>',
ce1222
+            ),
ce1222
+        )
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_1, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="start.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                SCSI_NODE,
ce1222
+                devices=devices,
ce1222
+            ),
ce1222
+            args=[f"devices={devices}"],
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason="no digests attributes in lrm_rsc_op element",
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_OTHER,
ce1222
+                )
ce1222
+            ],
ce1222
+            expected_in_processor=False,
ce1222
+        )
ce1222
+
ce1222
+    def test_crm_resource_digests_missing(self):
ce1222
+        devices = ",".join(DEVICES_2)
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(
ce1222
+            resources=fixture_scsi(),
ce1222
+            status=_fixture_status_lrm_ops_base(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                (
ce1222
+                    f'
ce1222
+                    'operation="start" op-restart-digest="somedigest" />'
ce1222
+                ),
ce1222
+            ),
ce1222
+        )
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_1, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="start.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                SCSI_NODE,
ce1222
+                devices=devices,
ce1222
+            ),
ce1222
+            args=[f"devices={devices}"],
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason=(
ce1222
+                        "necessary digest for 'op-restart-digest' attribute is "
ce1222
+                        "missing"
ce1222
+                    ),
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_OTHER,
ce1222
+                )
ce1222
+            ],
ce1222
+            expected_in_processor=False,
ce1222
+        )
ce1222
+
ce1222
+    def test_no_lrm_start_op(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(
ce1222
+            resources=fixture_scsi(),
ce1222
+            status=_fixture_status_lrm_ops(SCSI_STONITH_ID, lrm_start_ops=()),
ce1222
+        )
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_1, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason=(
ce1222
+                        "lrm_rsc_op element for start operation was not found"
ce1222
+                    ),
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_OTHER,
ce1222
+                )
ce1222
+            ],
ce1222
+            expected_in_processor=False,
ce1222
+        )
ce1222
+
ce1222
+    def test_monitor_ops_and_lrm_monitor_ops_do_not_match(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(
ce1222
+            resources=fixture_scsi(
ce1222
+                resource_ops=(
ce1222
+                    ("monitor", "30s", "10s", None),
ce1222
+                    ("monitor", "30s", "20s", "31"),
ce1222
+                    ("monitor", "60s", None, None),
ce1222
+                )
ce1222
+            ),
ce1222
+            status=_fixture_status_lrm_ops(SCSI_STONITH_ID),
ce1222
+        )
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_1, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="start.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID, SCSI_NODE, devices=",".join(DEVICES_2)
ce1222
+            ),
ce1222
+            args=["devices={}".format(",".join(DEVICES_2))],
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason=(
ce1222
+                        "number of lrm_rsc_op and op elements for monitor "
ce1222
+                        "operation differs"
ce1222
+                    ),
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_OTHER,
ce1222
+                )
ce1222
+            ],
ce1222
+            expected_in_processor=False,
ce1222
+        )
ce1222
+
ce1222
+    def test_lrm_monitor_ops_not_found(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(
ce1222
+            resources=fixture_scsi(
ce1222
+                resource_ops=(("monitor", "30s", None, None),)
ce1222
+            ),
ce1222
+            status=_fixture_status_lrm_ops(SCSI_STONITH_ID),
ce1222
+        )
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_1, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="start.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID, SCSI_NODE, devices=",".join(DEVICES_2)
ce1222
+            ),
ce1222
+            args=["devices={}".format(",".join(DEVICES_2))],
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM,
ce1222
+                    reason=(
ce1222
+                        "monitor lrm_rsc_op element for resource "
ce1222
+                        f"'{SCSI_STONITH_ID}', node '{SCSI_NODE}' and interval "
ce1222
+                        "'30000' not found"
ce1222
+                    ),
ce1222
+                    reason_type=reports.const.STONITH_RESTARTLESS_UPDATE_UNABLE_TO_PERFORM_REASON_OTHER,
ce1222
+                )
ce1222
+            ],
ce1222
+            expected_in_processor=False,
ce1222
+        )
ce1222
+
ce1222
+    def test_node_missing_name_and_missing_auth_token(self):
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(
ce1222
+            resources=fixture_scsi(),
ce1222
+            status=_fixture_status_lrm_ops(SCSI_STONITH_ID),
ce1222
+        )
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_1, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="start.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID, SCSI_NODE, devices=",".join(DEVICES_2)
ce1222
+            ),
ce1222
+            args=["devices={}".format(",".join(DEVICES_2))],
ce1222
+        )
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="monitor.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID, SCSI_NODE, devices=",".join(DEVICES_2)
ce1222
+            ),
ce1222
+            args=[
ce1222
+                "devices={}".format(",".join(DEVICES_2)),
ce1222
+                "CRM_meta_interval=60000",
ce1222
+            ],
ce1222
+        )
ce1222
+        self.config.corosync_conf.load_content(
ce1222
+            corosync_conf_fixture(
ce1222
+                self.existing_corosync_nodes
ce1222
+                + [[("ring0_addr", "custom_node"), ("nodeid", "5")]],
ce1222
+            )
ce1222
+        )
ce1222
+        self.config.env.set_known_nodes(self.existing_nodes[:-1])
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.COROSYNC_CONFIG_MISSING_NAMES_OF_NODES,
ce1222
+                    fatal=True,
ce1222
+                ),
ce1222
+                fixture.error(
ce1222
+                    reports.codes.HOST_NOT_FOUND,
ce1222
+                    host_list=[self.existing_nodes[-1]],
ce1222
+                ),
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def _unfence_failure_common_calls(self):
ce1222
+        devices = ",".join(DEVICES_2)
ce1222
+        self.config.runner.pcmk.is_resource_digests_supported()
ce1222
+        self.config.runner.cib.load(
ce1222
+            resources=fixture_scsi(),
ce1222
+            status=_fixture_status_lrm_ops(SCSI_STONITH_ID),
ce1222
+        )
ce1222
+        self.config.runner.pcmk.load_state(
ce1222
+            resources=FIXTURE_CRM_MON_RES_RUNNING_1, nodes=FIXTURE_CRM_MON_NODES
ce1222
+        )
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="start.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                SCSI_NODE,
ce1222
+                devices=devices,
ce1222
+            ),
ce1222
+            args=[f"devices={devices}"],
ce1222
+        )
ce1222
+        self.config.runner.pcmk.resource_digests(
ce1222
+            SCSI_STONITH_ID,
ce1222
+            SCSI_NODE,
ce1222
+            name="monitor.op.digests",
ce1222
+            stdout=fixture_digests_xml(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                SCSI_NODE,
ce1222
+                devices=devices,
ce1222
+            ),
ce1222
+            args=[
ce1222
+                f"devices={devices}",
ce1222
+                "CRM_meta_interval=60000",
ce1222
+            ],
ce1222
+        )
ce1222
+        self.config.corosync_conf.load_content(
ce1222
+            corosync_conf_fixture(self.existing_corosync_nodes)
ce1222
+        )
ce1222
+
ce1222
+    def test_unfence_failure_unable_to_connect(self):
ce1222
+        self._unfence_failure_common_calls()
ce1222
+        self.config.http.corosync.get_corosync_online_targets(
ce1222
+            node_labels=self.existing_nodes
ce1222
+        )
ce1222
+        self.config.http.scsi.unfence_node(
ce1222
+            DEVICES_2,
ce1222
+            communication_list=[
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[0],
ce1222
+                    raw_data=json.dumps(
ce1222
+                        dict(devices=DEVICES_2, node=self.existing_nodes[0])
ce1222
+                    ),
ce1222
+                    was_connected=False,
ce1222
+                    error_msg="errA",
ce1222
+                ),
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[1],
ce1222
+                    raw_data=json.dumps(
ce1222
+                        dict(devices=DEVICES_2, node=self.existing_nodes[1])
ce1222
+                    ),
ce1222
+                    output=json.dumps(
ce1222
+                        dto.to_dict(
ce1222
+                            communication.dto.InternalCommunicationResultDto(
ce1222
+                                status=communication.const.COM_STATUS_ERROR,
ce1222
+                                status_msg="error",
ce1222
+                                report_list=[
ce1222
+                                    reports.ReportItem.error(
ce1222
+                                        reports.messages.StonithUnfencingFailed(
ce1222
+                                            "errB"
ce1222
+                                        )
ce1222
+                                    ).to_dto()
ce1222
+                                ],
ce1222
+                                data=None,
ce1222
+                            )
ce1222
+                        )
ce1222
+                    ),
ce1222
+                ),
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[2],
ce1222
+                    raw_data=json.dumps(
ce1222
+                        dict(devices=DEVICES_2, node=self.existing_nodes[2])
ce1222
+                    ),
ce1222
+                ),
ce1222
+            ],
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.NODE_COMMUNICATION_ERROR_UNABLE_TO_CONNECT,
ce1222
+                    node=self.existing_nodes[0],
ce1222
+                    command="api/v1/scsi-unfence-node/v1",
ce1222
+                    reason="errA",
ce1222
+                ),
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_UNFENCING_FAILED,
ce1222
+                    reason="errB",
ce1222
+                    context=reports.dto.ReportItemContextDto(
ce1222
+                        node=self.existing_nodes[1],
ce1222
+                    ),
ce1222
+                ),
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_unfence_failure_agent_script_failed(self):
ce1222
+        self._unfence_failure_common_calls()
ce1222
+        self.config.http.corosync.get_corosync_online_targets(
ce1222
+            node_labels=self.existing_nodes
ce1222
+        )
ce1222
+        self.config.http.scsi.unfence_node(
ce1222
+            DEVICES_2,
ce1222
+            communication_list=[
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[0],
ce1222
+                    raw_data=json.dumps(
ce1222
+                        dict(devices=DEVICES_2, node=self.existing_nodes[0])
ce1222
+                    ),
ce1222
+                ),
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[1],
ce1222
+                    raw_data=json.dumps(
ce1222
+                        dict(devices=DEVICES_2, node=self.existing_nodes[1])
ce1222
+                    ),
ce1222
+                    output=json.dumps(
ce1222
+                        dto.to_dict(
ce1222
+                            communication.dto.InternalCommunicationResultDto(
ce1222
+                                status=communication.const.COM_STATUS_ERROR,
ce1222
+                                status_msg="error",
ce1222
+                                report_list=[
ce1222
+                                    reports.ReportItem.error(
ce1222
+                                        reports.messages.StonithUnfencingFailed(
ce1222
+                                            "errB"
ce1222
+                                        )
ce1222
+                                    ).to_dto()
ce1222
+                                ],
ce1222
+                                data=None,
ce1222
+                            )
ce1222
+                        )
ce1222
+                    ),
ce1222
+                ),
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[2],
ce1222
+                    raw_data=json.dumps(
ce1222
+                        dict(devices=DEVICES_2, node=self.existing_nodes[2])
ce1222
+                    ),
ce1222
+                ),
ce1222
+            ],
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.STONITH_UNFENCING_FAILED,
ce1222
+                    reason="errB",
ce1222
+                    context=reports.dto.ReportItemContextDto(
ce1222
+                        node=self.existing_nodes[1],
ce1222
+                    ),
ce1222
+                ),
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_corosync_targets_unable_to_connect(self):
ce1222
+        self._unfence_failure_common_calls()
ce1222
+        self.config.http.corosync.get_corosync_online_targets(
ce1222
+            communication_list=[
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[0],
ce1222
+                    output='{"corosync":true}',
ce1222
+                ),
ce1222
+            ]
ce1222
+            + [
ce1222
+                dict(
ce1222
+                    label=node,
ce1222
+                    was_connected=False,
ce1222
+                    errno=7,
ce1222
+                    error_msg="an error",
ce1222
+                )
ce1222
+                for node in self.existing_nodes[1:]
ce1222
+            ]
ce1222
+        )
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(), SCSI_STONITH_ID, DEVICES_2
ce1222
+            ),
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.NODE_COMMUNICATION_ERROR_UNABLE_TO_CONNECT,
ce1222
+                    force_code=reports.codes.SKIP_OFFLINE_NODES,
ce1222
+                    node=node,
ce1222
+                    command="remote/status",
ce1222
+                    reason="an error",
ce1222
+                )
ce1222
+                for node in self.existing_nodes[1:]
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_corosync_targets_skip_offline_unfence_node_running_corosync(
ce1222
+        self,
ce1222
+    ):
ce1222
+        self._unfence_failure_common_calls()
ce1222
+        self.config.http.corosync.get_corosync_online_targets(
ce1222
+            communication_list=[
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[0],
ce1222
+                    output='{"corosync":true}',
ce1222
+                ),
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[1],
ce1222
+                    output='{"corosync":false}',
ce1222
+                ),
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[2],
ce1222
+                    was_connected=False,
ce1222
+                    errno=7,
ce1222
+                    error_msg="an error",
ce1222
+                ),
ce1222
+            ]
ce1222
+        )
ce1222
+        self.config.http.scsi.unfence_node(
ce1222
+            DEVICES_2,
ce1222
+            communication_list=[
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[0],
ce1222
+                    raw_data=json.dumps(
ce1222
+                        dict(devices=DEVICES_2, node=self.existing_nodes[0])
ce1222
+                    ),
ce1222
+                ),
ce1222
+            ],
ce1222
+        )
ce1222
+        self.config.env.push_cib(
ce1222
+            resources=fixture_scsi(devices=DEVICES_2),
ce1222
+            status=_fixture_status_lrm_ops(
ce1222
+                SCSI_STONITH_ID,
ce1222
+                lrm_start_ops=DEFAULT_LRM_START_OPS_UPDATED,
ce1222
+                lrm_monitor_ops=DEFAULT_LRM_MONITOR_OPS_UPDATED,
ce1222
+            ),
ce1222
+        )
ce1222
+        stonith.update_scsi_devices(
ce1222
+            self.env_assist.get_env(),
ce1222
+            SCSI_STONITH_ID,
ce1222
+            DEVICES_2,
ce1222
+            force_flags=[reports.codes.SKIP_OFFLINE_NODES],
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.warn(
ce1222
+                    reports.codes.NODE_COMMUNICATION_ERROR_UNABLE_TO_CONNECT,
ce1222
+                    node=self.existing_nodes[2],
ce1222
+                    command="remote/status",
ce1222
+                    reason="an error",
ce1222
+                ),
ce1222
+            ]
ce1222
+        )
ce1222
+
ce1222
+    def test_corosync_targets_unable_to_perform_unfencing_operation(
ce1222
+        self,
ce1222
+    ):
ce1222
+        self._unfence_failure_common_calls()
ce1222
+        self.config.http.corosync.get_corosync_online_targets(
ce1222
+            communication_list=[
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[0],
ce1222
+                    was_connected=False,
ce1222
+                    errno=7,
ce1222
+                    error_msg="an error",
ce1222
+                ),
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[1],
ce1222
+                    was_connected=False,
ce1222
+                    errno=7,
ce1222
+                    error_msg="an error",
ce1222
+                ),
ce1222
+                dict(
ce1222
+                    label=self.existing_nodes[2],
ce1222
+                    output='{"corosync":false}',
ce1222
+                ),
ce1222
+            ]
ce1222
+        )
ce1222
+        self.config.http.scsi.unfence_node(DEVICES_2, communication_list=[])
ce1222
+        self.env_assist.assert_raise_library_error(
ce1222
+            lambda: stonith.update_scsi_devices(
ce1222
+                self.env_assist.get_env(),
ce1222
+                SCSI_STONITH_ID,
ce1222
+                DEVICES_2,
ce1222
+                force_flags=[reports.codes.SKIP_OFFLINE_NODES],
ce1222
+            ),
ce1222
+        )
ce1222
+        self.env_assist.assert_reports(
ce1222
+            [
ce1222
+                fixture.warn(
ce1222
+                    reports.codes.NODE_COMMUNICATION_ERROR_UNABLE_TO_CONNECT,
ce1222
+                    node=node,
ce1222
+                    command="remote/status",
ce1222
+                    reason="an error",
ce1222
+                )
ce1222
+                for node in self.existing_nodes[0:2]
ce1222
+            ]
ce1222
+            + [
ce1222
+                fixture.error(
ce1222
+                    reports.codes.UNABLE_TO_PERFORM_OPERATION_ON_ANY_NODE,
ce1222
+                ),
ce1222
+            ]
ce1222
+        )
ce1222
-- 
ce1222
2.31.1
ce1222