From 4aad57a9ad66126678c9e45ab4ed3e5a5cf8dec5 Mon Sep 17 00:00:00 2001 From: Victoria Litvinova Date: Wed, 19 Jun 2024 13:23:15 -0700 Subject: [PATCH 1/4] code for paging + testing --- azext_iot/operations/dps.py | 26 ++++++-- .../test_iot_device_registration_group_int.py | 9 +++ .../test_iot_dps_enrollment_group_unit.py | 63 ++++++++++++++++++- 3 files changed, 90 insertions(+), 8 deletions(-) diff --git a/azext_iot/operations/dps.py b/azext_iot/operations/dps.py index 845d6ce41..78a1021e5 100644 --- a/azext_iot/operations/dps.py +++ b/azext_iot/operations/dps.py @@ -819,7 +819,13 @@ def _get_dps_connection_string( def iot_dps_registration_list( - cmd, enrollment_id, dps_name=None, resource_group_name=None, login=None, auth_type_dataplane=None, + cmd, + enrollment_id, + dps_name=None, + resource_group_name=None, + top=None, + login=None, + auth_type_dataplane=None, ): discovery = DPSDiscovery(cmd) target = discovery.get_target( @@ -828,13 +834,23 @@ def iot_dps_registration_list( login=login, auth_type=auth_type_dataplane, ) + registrations = [] try: resolver = SdkResolver(target=target) sdk = resolver.get_sdk(SdkType.dps_sdk) - - return sdk.device_registration_state.query( - enrollment_id, raw=True - ).response.json() + cont_token = None + while True: + result = sdk.device_registration_state.query( + enrollment_id, x_ms_continuation=cont_token, raw=True + ) + registrations.extend(result.response.json()) + cont_token = result.headers.get("x-ms-continuation") + if top and len(registrations) > top: + registrations = registrations[:top] + cont_token = None + if not cont_token: + break + return registrations except ProvisioningServiceErrorDetailsException as e: handle_service_exception(e) diff --git a/azext_iot/tests/dps/device_registration/test_iot_device_registration_group_int.py b/azext_iot/tests/dps/device_registration/test_iot_device_registration_group_int.py index 1c7a2c206..a2a2389e4 100644 --- a/azext_iot/tests/dps/device_registration/test_iot_device_registration_group_int.py +++ b/azext_iot/tests/dps/device_registration/test_iot_device_registration_group_int.py @@ -195,6 +195,15 @@ def test_dps_device_registration_symmetrickey_lifecycle(provisioned_iot_dps_modu assert registration["status"] == "assigned" # Check for both registration from service side + random_registration = cli.invoke( + set_cmd_auth_type( + f"iot dps enrollment-group registration list --dps-name {dps_name} -g {dps_rg} --group-id {group_id} --top 1", + auth_type=auth_phase, + cstring=dps_cstring + ), + ).as_json() + assert len(random_registration) == 1 + service_side_registrations = cli.invoke( set_cmd_auth_type( f"iot dps enrollment-group registration list --dps-name {dps_name} -g {dps_rg} --group-id {group_id}", diff --git a/azext_iot/tests/dps/enrollment_group/test_iot_dps_enrollment_group_unit.py b/azext_iot/tests/dps/enrollment_group/test_iot_dps_enrollment_group_unit.py index 0377ca370..98f031692 100644 --- a/azext_iot/tests/dps/enrollment_group/test_iot_dps_enrollment_group_unit.py +++ b/azext_iot/tests/dps/enrollment_group/test_iot_dps_enrollment_group_unit.py @@ -721,25 +721,82 @@ def serviceclient(self, mocked_response, fixture_gdcs, fixture_dps_sas, patch_ce mocked_response.add( method=responses.POST, url="https://{}/registrations/{}/query?".format(mock_dps_target['entity'], enrollment_id), - body=json.dumps([generate_registration_state_show()]), + body=json.dumps([ + generate_registration_state_show(), + generate_registration_state_show(), + generate_registration_state_show(), + generate_registration_state_show() + ]), status=request.param, content_type="application/json", match_querystring=False ) yield mocked_response - def test_registration_list(self, serviceclient, fixture_cmd): - subject.iot_dps_registration_list( + @pytest.mark.parametrize("top", [None, 3]) + def test_registration_list(self, serviceclient, fixture_cmd, top): + result = subject.iot_dps_registration_list( cmd=fixture_cmd, dps_name=mock_dps_target['entity'], enrollment_id=enrollment_id, resource_group_name=resource_group, + top=top ) request = serviceclient.calls[0].request url = request.url method = request.method assert "{}/registrations/{}/query?".format(mock_dps_target['entity'], enrollment_id) in url assert method == 'POST' + if top: + assert len(result) == top + + @pytest.fixture(params=[200]) + def pagingserviceclient( + self, mocked_response, fixture_gdcs, fixture_dps_sas, patch_certificate_open, request + ): + mocked_response.assert_all_requests_are_fired = False + mocked_response.add( + method=responses.POST, + url="https://{}/registrations/{}/query?".format(mock_dps_target['entity'], enrollment_id), + body=json.dumps([ + generate_registration_state_show(), + generate_registration_state_show(), + generate_registration_state_show(), + generate_registration_state_show() + ]), + headers={"x-ms-continuation": "continuation_token123"}, + status=request.param, + content_type="application/json", + match_querystring=False + ) + mocked_response.add( + method=responses.POST, + url="https://{}/registrations/{}/query?".format(mock_dps_target['entity'], enrollment_id), + body=json.dumps([ + generate_registration_state_show(), + generate_registration_state_show(), + generate_registration_state_show(), + ]), + status=request.param, + content_type="application/json", + match_querystring=False + ) + yield mocked_response + + @pytest.mark.parametrize("top", [None, 3, 6]) + def test_registration_list_paging(self, pagingserviceclient, fixture_cmd, top): + result = subject.iot_dps_registration_list( + cmd=fixture_cmd, + dps_name=mock_dps_target['entity'], + enrollment_id=enrollment_id, + resource_group_name=resource_group, + top=top + ) + if top: + assert len(result) == top + assert len(pagingserviceclient.calls) == (1 + top // 4) + else: + assert len(pagingserviceclient.calls) == 2 def test_registration_list_error(self, fixture_cmd): with pytest.raises(CLIError): From ac3cadfce4f0491b0c25793425906e09528701a0 Mon Sep 17 00:00:00 2001 From: Victoria Litvinova Date: Wed, 19 Jun 2024 14:43:51 -0700 Subject: [PATCH 2/4] there is already a query command... --- azext_iot/operations/dps.py | 15 +-------------- 1 file changed, 1 insertion(+), 14 deletions(-) diff --git a/azext_iot/operations/dps.py b/azext_iot/operations/dps.py index 78a1021e5..1a555743f 100644 --- a/azext_iot/operations/dps.py +++ b/azext_iot/operations/dps.py @@ -834,23 +834,10 @@ def iot_dps_registration_list( login=login, auth_type=auth_type_dataplane, ) - registrations = [] try: resolver = SdkResolver(target=target) sdk = resolver.get_sdk(SdkType.dps_sdk) - cont_token = None - while True: - result = sdk.device_registration_state.query( - enrollment_id, x_ms_continuation=cont_token, raw=True - ) - registrations.extend(result.response.json()) - cont_token = result.headers.get("x-ms-continuation") - if top and len(registrations) > top: - registrations = registrations[:top] - cont_token = None - if not cont_token: - break - return registrations + return _execute_query([enrollment_id], sdk.device_registration_state.query, top) except ProvisioningServiceErrorDetailsException as e: handle_service_exception(e) From fc1958619116120156bcee335fefb10f3e287b18 Mon Sep 17 00:00:00 2001 From: Victoria Litvinova Date: Tue, 25 Jun 2024 11:30:08 -0700 Subject: [PATCH 3/4] pylint --- azext_iot/operations/dps.py | 12 +++++------- 1 file changed, 5 insertions(+), 7 deletions(-) diff --git a/azext_iot/operations/dps.py b/azext_iot/operations/dps.py index 1a555743f..7055867e9 100644 --- a/azext_iot/operations/dps.py +++ b/azext_iot/operations/dps.py @@ -816,15 +816,13 @@ def _get_dps_connection_string( # DPS Registration - - def iot_dps_registration_list( - cmd, - enrollment_id, - dps_name=None, - resource_group_name=None, + cmd, + enrollment_id, + dps_name=None, + resource_group_name=None, top=None, - login=None, + login=None, auth_type_dataplane=None, ): discovery = DPSDiscovery(cmd) From ae8f72ec9f13df1c1700d01f9ab3ddb9e88d3b64 Mon Sep 17 00:00:00 2001 From: Victoria Litvinova Date: Tue, 25 Jun 2024 15:04:05 -0700 Subject: [PATCH 4/4] history and version --- HISTORY.rst | 8 ++++++++ azext_iot/constants.py | 2 +- 2 files changed, 9 insertions(+), 1 deletion(-) diff --git a/HISTORY.rst b/HISTORY.rst index ad77715da..1f8235bd5 100644 --- a/HISTORY.rst +++ b/HISTORY.rst @@ -4,6 +4,14 @@ Release History =============== +0.24.1 ++++++++++++++++ + +**DPS updates** + +* Fix for `az iot dps enrollement-group registration list` to support paging. + + 0.24.0 +++++++++++++++ diff --git a/azext_iot/constants.py b/azext_iot/constants.py index 226021a5a..1bf275449 100644 --- a/azext_iot/constants.py +++ b/azext_iot/constants.py @@ -7,7 +7,7 @@ import os -VERSION = "0.24.0" +VERSION = "0.24.1" EXTENSION_NAME = "azure-iot" EXTENSION_ROOT = os.path.dirname(os.path.abspath(__file__)) EXTENSION_CONFIG_ROOT_KEY = "iotext"