From 3495e2c89f660da59906169b588c3a2bfb63aa02 Mon Sep 17 00:00:00 2001 From: Gil Bregman Date: Wed, 1 Nov 2023 16:12:27 +0200 Subject: [PATCH] Block the usage of the same serial number for two subsystems. Fixes #148 Signed-off-by: Gil Bregman --- control/grpc.py | 80 ++++++++++++++++++++++++++++++++++------------- tests/test_cli.py | 6 ++++ 2 files changed, 65 insertions(+), 21 deletions(-) diff --git a/control/grpc.py b/control/grpc.py index a4cd29d7..782089dc 100644 --- a/control/grpc.py +++ b/control/grpc.py @@ -168,17 +168,17 @@ def get_bdev_namespaces(self, bdev_name) -> list: ns_list = [] local_state_dict = self.gateway_state.local.get_state() for key, val in local_state_dict.items(): - if key.startswith(self.gateway_state.local.NAMESPACE_PREFIX): - try: - req = json_format.Parse(val, pb2.add_namespace_req(), ignore_unknown_fields = True) - ns_bdev_name = req.bdev_name - if ns_bdev_name == bdev_name: - nsid = req.nsid - nqn = req.subsystem_nqn - ns_list.insert(0, {"nqn" : nqn, "nsid" : nsid}) - except Exception as ex: - self.logger.error(f"Got exception trying to get bdev {bdev_name} namespaces: {ex}") - pass + if not key.startswith(self.gateway_state.local.NAMESPACE_PREFIX): + continue + try: + ns = json.loads(val) + if ns["bdev_name"] == bdev_name: + nsid = ns["nsid"] + nqn = ns["subsystem_nqn"] + ns_list.insert(0, {"nqn" : nqn, "nsid" : nsid}) + except Exception as ex: + self.logger.error(f"Got exception trying to get bdev {bdev_name} namespaces: {ex}") + pass return ns_list @@ -249,6 +249,23 @@ def delete_bdev(self, request, context=None): def is_discovery_nqn(self, nqn) -> bool: return nqn == DiscoveryService.DISCOVERY_NQN + def serial_number_already_used(self, context, serial) -> str: + if not context: + return None + state = self.gateway_state.local.get_state() + for key, val in state.items(): + if not key.startswith(self.gateway_state.local.SUBSYSTEM_PREFIX): + continue + try: + subsys = json.loads(val) + sn = subsys["serial_number"] + if serial == sn: + return subsys["subsystem_nqn"] + except Exception: + self.logger.warning("Got exception while parsing {val}: {ex}") + continue + return None + def create_subsystem_safe(self, request, context=None): """Creates a subsystem.""" @@ -264,7 +281,24 @@ def create_subsystem_safe(self, request, context=None): random.seed() randser = random.randint(2, 99999999999999) request.serial_number = f"SPDK{randser}" + self.logger.info(f"No serial number specified, will use {request.serial_number}") + try: + subsys_using_serial = self.serial_number_already_used(context, request.serial_number) + if subsys_using_serial: + self.logger.error(f"Serial number {request.serial_number} already used by subsystem {subsys_using_serial}") + req = {"subsystem_nqn": request.subsystem_nqn, + "serial_number": request.serial_number, + "max_namespaces": request.max_namespaces, + "ana_reporting": request.ana_reporting, + "enable_ha": request.enable_ha, + "method": "nvmf_create_subsystem", "req_id": 0} + ret = {"code": -errno.EEXIST, "message": f"Serial number {request.serial_number} already used by subsystem {subsys_using_serial}"} + msg = "\n".join(["request:", "%s" % json.dumps(req, indent=2), + "Got JSON-RPC error response", + "response:", + json.dumps(ret, indent=2)]) + raise Exception(msg) ret = rpc_nvmf.nvmf_create_subsystem( self.spdk_rpc_client, nqn=request.subsystem_nqn, @@ -614,20 +648,24 @@ def create_listener_safe(self, request, context=None): return pb2.req_status() state = self.gateway_state.local.get_state() - req = None - subsys = state.get(GatewayState.build_subsystem_key(request.nqn)) - if subsys: - self.logger.debug(f"value of sub-system: {subsys}") + enable_ha = False + subsys_str = state.get(GatewayState.build_subsystem_key(request.nqn)) + if subsys_str: + self.logger.debug(f"value of sub-system: {subsys_str}") try: - req = json_format.Parse(subsys, pb2.create_subsystem_req()) - self.logger.info(f"enable_ha: {req.enable_ha}") - except Exception: - self.logger.error(f"Got exception trying to parse subsystem: {ex}") + subsys_dict = json.loads(subsys_str) + try: + enable_ha = subsys_dict["enable_ha"] + except KeyError: + enable_ha = False + self.logger.info(f"enable_ha: {enable_ha}") + except Exception as ex: + self.logger.error(f"Got exception trying to parse subsystem {request.nqn}: {ex}") pass else: - self.logger.info(f"No sub-system for {request.nqn}") + self.logger.info(f"No subsystem for {request.nqn}") - if req and req.enable_ha: + if enable_ha: for x in range (MAX_ANA_GROUPS): try: ret = rpc_nvmf.nvmf_subsystem_listener_set_ana_state( diff --git a/tests/test_cli.py b/tests/test_cli.py index 4d54c926..45169b72 100644 --- a/tests/test_cli.py +++ b/tests/test_cli.py @@ -63,10 +63,13 @@ def test_create_bdev_ipv6(self, caplog, gateway): def test_create_subsystem(self, caplog, gateway): cli(["create_subsystem", "-n", subsystem]) assert "Failed to create" not in caplog.text + assert "ana reporting: False" in caplog.text cli(["get_subsystems"]) assert serial not in caplog.text + caplog.clear() cli(["create_subsystem", "-n", subsystem2, "-s", serial]) assert "Failed to create" not in caplog.text + assert "ana reporting: False" in caplog.text cli(["get_subsystems"]) assert serial in caplog.text @@ -148,8 +151,10 @@ def test_create_bdev_ana_ipv6(self, caplog, gateway): def test_create_subsystem_ana(self, caplog, gateway): + caplog.clear() cli(["create_subsystem", "-n", subsystem, "-a", "-t"]) assert "Failed to create" not in caplog.text + assert "ana reporting: True" in caplog.text cli(["get_subsystems"]) assert serial not in caplog.text @@ -161,6 +166,7 @@ def test_add_namespace_ana(self, caplog, gateway): def test_create_listener_ana(self, caplog, listener, gateway): cli(["create_listener", "-n", subsystem] + listener) assert "Failed to create" not in caplog.text + assert "enable_ha: True" in caplog.text class TestDeleteAna: