- Get Python 3: https://www.python.org/downloads/
You must fork and clone the repositories below. Follow the videos and instructions found here.
IMPORTANT: When cloning the repositories and environments, ensure they are all siblings to each other. This makes things much easier down the line.
source-directory/
|-- azure-cli/
|-- azure-iot-cli-extension/
|-- .env3/
IMPORTANT: Ensure you keep the Python virtual environment you created above. It is required for development.
After following the videos, ensure you have:
-
Python virtual environment
-
Functional development az cli
You can run this setup in bash
or cmd
environments, this documentation just show the powershell
flavor.
-
Create a directory for your development extensions to live in
mkdir path/to/source/extensions/azure-iot
-
Set
AZURE_EXTENSION_DIR
to the following$env:AZURE_EXTENSION_DIR="path/to/source/extensions"
Similar to the video, have your virtual environment activated then execute the following command
(.env3) azdev setup -c path/to/source/azure-cli
-
Change directories
cd path/to/source/azure-iot-cli-extension
-
Install the extension (should only be needed once)
pip install -U --target path/to/source/extensions/azure-iot .
Run a command that is present in the iot extension space
az iot central app -h
If this works, then you should now be able to make changes to the extension and have them reflected immediately in your az cli.
Tests are organized into folders by resource in azext_iot\tests\
:
- Central
- Digital Twins
- Device Provisioning Service
- IoT Hub
- Product
- Utility
You may need to install the dev_requirements for this
pip install -r path/to/source/dev_requirements
Example unit tests runs:
Hub:
pytest azext_iot/tests/iothub/core/test_iot_ext_unit.py
DPS:
pytest azext_iot/tests/dps/core/test_dps_discovery_unit.py
Unit tests end in "_unit.py" so execute the following command to run all unit tests,
pytest -k "_unit.py"
Execute the following command to run the IoT Hub unit tests:
pytest azext_iot/tests/iothub/ -k "_unit.py"
Integration tests are run against Azure resources and depend on environment variables.
Example int tests runs:
Hub:
pytest azext_iot/tests/iothub/core/test_iothub_storage_int.py
DPS:
pytest azext_iot/tests/dps/core/test_dps_discovery_int.py
Integration tests end in "_int.py" so execute the following command to run all integration tests,
pytest -k "_int.py"
Execute the following command to run the IoT Hub integration tests:
pytest azext_iot/tests/iothub/ -k "_int.py"
To run specific test in any integration test file, such as:
pytest azext_iot/tests/central/test_iot_central_int.py::TestIotCentral::test_central_query_methods_run
The following resources will be needed for the integration tests.
- IoT Hub
- IoT Device Provisioning Service
- Azure Digital Twins instance
- IoT Central App
- Storage Account (with an empty Container)
- Event Grid Topic
- Event Hub Namespace with an Event Hub
- Service Bus Namespace with a Topic
- Azure Data Explorer Cluster with a Database
If specified in the pytest.ini configuration file, those resources will be used. Please ensure that the resources are in a clean, new state (ex: Iot Hub should not have any devices). Otherwise, new resources will be generated during the test startup and deleted during the test teardown.
Note: If you interrupt test execution (for example via ctrl + C), the teardown or clean up processes may not run leaving resources in an indeterminant state.
You can either manually set the environment variables or use the pytest.ini.example
file in the root of the extension repo. To use that file, rename it to pytest.ini
, open it and set the variables as indicated below.
AZURE_TEST_RUN_LIVE=True
azext_iot_testrg=
azext_iot_testhub=
azext_iot_testdps=
azext_iot_testdps_hub=
azext_iot_teststorageaccount=
azext_iot_teststoragecontainer=
azext_iot_central_app_id=
azext_iot_central_scope_id=
azext_iot_central_primarykey=
azext_iot_central_storage_cstring=
azext_iot_central_storage_container=
azext_dt_adx_cluster=
azext_dt_adx_database=
azext_dt_adx_rg=
azext_dt_ep_eventgrid_topic=
azext_dt_ep_servicebus_namespace=
azext_dt_ep_servicebus_policy=
azext_dt_ep_servicebus_topic=
azext_dt_ep_eventhub_namespace=
azext_dt_ep_eventhub_policy=
azext_dt_ep_eventhub_topic=
azext_dt_ep_eventhub_topic_consumer_group=
azext_dt_ep_rg=
azext_dt_region=
To run almost all of the tests, only the azext_iot_testrg
is needed.
To run all tests, azext_iot_testrg
, azext_iot_central_app_id
, and azext_iot_central_primarykey
are needed because the IoT Central Primary Key cannot be currently retrieved through the CLI.
For all resources, if the environmental variable is not provided, a new instance will be created for the test run and deleted at the end of the test run.
Variable name | Tests Used for | Description |
---|---|---|
az_iot_testrg |
All | The resource group that contains the IoT Hub and DPS instances or where all test resources are created. It will be the default resource group if any other resource group variables are not provided. |
azext_iot_teststorageaccount |
Iot Hub and Central Storage Tests | The storage account used for running IoT Hub and Central storage tests. During these tests, your hub will be assigned a System-Assigned AAD identity, and will be granted the role of "Storage Blob Data Contributor" on the storage account you provide. Both the hub's identity and the RBAC role will be removed once the test completes. No role assignments are made for the IoT Central App. |
azext_iot_teststoragecontainer |
Iot Hub Storage Tests | The name of blob container belonging to the azext_iot_teststorageaccount storage account. Defaults to 'devices' when not specified. |
azext_iot_testhub |
Iot Hub Tests | The name of the Iot Hub instance. |
azext_iot_testdps |
Iot DPS Tests | The name of the Iot DPS instance. |
azext_iot_testdps_hub |
Iot DPS Tests | The name of the Iot Hub instance to use for DPS tests. |
azext_iot_central_app_id |
Iot Central Tests | The IoT Central Application Id or name. |
azext_iot_central_scope_id |
Iot Central Tests | The ID scope for the DPS associated with the IoT Central Application. |
azext_iot_central_token |
Iot Central Tests | The api token to use for the IoT Central Application. This is only used to populate --token argument in IoT Central commands. |
azext_iot_central_dns_suffix |
Iot Central Tests | The DNS Suffix to use for the IoT Central Application. This is only used to populate --central-dns-suffix argument in IoT Central commands. |
azext_iot_central_primarykey |
Iot Central Tests | The IoT Central Application Id or name. Cannot be currently retrieved through the CLI. |
azext_iot_central_storage_container |
Iot Central Storage Tests | The name of blob container belonging to the azext_iot_teststorageaccount storage account. Defaults to 'central' when not specified. |
azext_dt_region |
Digital Twin Tests | The region to restrict Digital Twins creation. If not specified, will create the Digital Twins in a supported region. |
azext_dt_adx_cluster |
Digital Twin Data History Tests | The name of the Azure Data Explorer Cluster to use. Azure Data Explorer Cluster creation time can take up to 20 minutes, so having a cluster is recommended. |
azext_dt_adx_database |
Digital Twin Data History Tests | The name of the database in the Azure Data Explorer Cluster to use. |
azext_dt_adx_rg |
Digital Twin Data History Tests | The resource group that contains the Azure Data Explorer Cluster. If not provided, az_iot_testrg will be used as the resource group. |
azext_dt_ep_eventgrid_topic |
Digital Twin Endpoint Tests | The Event Grid Topic to use. |
azext_dt_ep_servicebus_namespace |
Digital Twin Endpoint Tests | The Service Bus Namespace to use. |
azext_dt_ep_servicebus_policy |
Digital Twin Endpoint Tests | The policy for the topic in the Service Bus Namespace to use. |
azext_dt_ep_servicebus_topic |
Digital Twin Endpoint Tests | The topic in the Service Bus Namespace to use. |
azext_dt_ep_eventhub_namespace |
Digital Twin Data History and Endpoint Tests | The Event Hub Namespace to use. |
azext_dt_ep_eventhub_policy |
Digital Twin Endpoint Tests | The policy for the Event Hub instance in the Event Hub Namespace to use. |
azext_dt_ep_eventhub_topic |
Digital Twin Data History and Endpoint Tests | The Event Hub instance in the Event Hub Namespace to use. |
azext_dt_ep_eventhub_topic_consumer_group |
Digital Twin Data History Tests | The Event Hub consumer group name to use. Defaults to "$Default". |
azext_dt_ep_rg |
Digital Twin Data History and Endpoint Tests | The resource group that contains the endpoint (Event Hub, Event Grid, Service Bus) variables. If not provided, az_iot_testrg will be used as the resource group. |
azext_dt_max_models_per_batch |
Digital Twin Model Tests | The maximum number of models per batch to submit to the DT Instance if the total set of models exceed the single page API limit. |
use_tags |
IoT Hub, DPS, and Central Tagging | Flag to enable resource tagging. Please see "Test Resource Tagging" for more details. |
definition_id |
IoT Hub, DPS, and Central Tagging | Definition Id populated by an internal pipeline run. Can be manually set to customize the pipeline name tag. Please see "Test Resource Tagging" for more details. |
job_display_name |
IoT Hub, DPS, and Central Tagging | Job Display Name populated by an internal pipeline run. Can be manually set to customize the pipeline name tag. Please see "Test Resource Tagging" for more details. |
job_id |
IoT Hub, DPS, and Central Tagging | Job Id populated by an internal pipeline run. Can be manually set to customize the pipeline name tag. Please see "Test Resource Tagging" for more details. |
IoT Digital Twins test for creation of larger ontologies require ontology submodules to be cloned.
Run the following command to clone required submodules.
git submodule update --init --recursive
There are 4 more test variables used for tagging the test resources:
use_tags
definition_id
job_display_name
job_id
If use_tags
is set to "True", then the resources created and used in Iot Central, DPS, and Hub tests will be tagged. The tested resources (IoT Hub, DPS, and Central App instances) will have two types of tags:
- number of test method runs (ex:
test_central_device_c2d_purge_success : 1
means that the test "test_central_device_c2d_purge_success" was run once) - a pipeline name or id (ex:
pipeline_id : 00 Test IoT Central Python310 00000000-0000-0000-0000-000000000
generated from internal pipeline runs) which can be manually set withdefinition_id
,job_display_name
andjob_id
.
Other resources created for these tests (ex: Storage Accounts for IoT Hub and Central tests) will have tags showing which test resource is associated (ex: iot_resource : test-app-xxx
shows that the tagged storage account was created to be tested with the IoT Central App test-app-xxx
).
These are mainly used for test pipeline debugging and differentiating what resources were created for what runs.
Execute the following command to run both Unit and Integration tests and output a code coverage report to the console and to a .coverage
file. You can configure code coverage with the .coveragerc
file.
pytest -v . --cov=azext_iot --cov-config .coveragerc
The repo uses the linter in azdev
. For more information, see here: https://github.com/Azure/azure-cli-dev-tools#style-linter-check-and-testing
We use our flake8 and pylint rules. We recommend you set up your IDE as per the VSCode setup below for best compliance.
To manually run pylint with our rules, run this command:
pylint azext_iot/ --rcfile=.pylintrc
To manually run flake8 with our rules, run this command:
flake8 azext_iot/ --statistics --config=setup.cfg
We are also starting to use python black
. To set this up on VSCode, see the following blog post.
https://medium.com/@marcobelo/setting-up-python-black-on-visual-studio-code-5318eba4cd00
-
Install VSCode
-
Install the required extensions
- (ms-python.python is recommended)
-
Set up
settings.json
{ "python.pythonPath": "path/to/source/env3/Scripts/python.exe", "python.venvPath": "path/to/source/", "python.linting.pylintEnabled": true, "python.autoComplete.extraPaths": [ "path/to/source/env3/Lib/site-packages" ], "python.linting.flake8Enabled": true, "python.linting.flake8Args": [ "--config=setup.cfg" ], "files.associations": { "*/.azure-devops/.yml": "azure-pipelines" } }
-
Set up
launch.json
{ "version": "0.2.0", "configurations": [ { "name": "Azure CLI Debug (Integrated Console)", "type": "python", "request": "launch", "pythonPath": "${config:python.pythonPath}", "program": "${workspaceRoot}/../azure-cli/src/azure-cli/azure/cli/__main__.py", "cwd": "${workspaceRoot}", "args": [ "--help" ], "console": "integratedTerminal", "debugOptions": [ "WaitOnAbnormalExit", "WaitOnNormalExit", "RedirectOutput" ], "justMyCode": false } ] }
-
launch.json was derived from this file
-
Note: your "program" path might be different if you did not set up the folder structure as siblings as recommended above
-
Note: when passing args, ensure they are all comma separated.
Correct:
"args": [ "--a", "value", "--b", "value" ],
Incorrect:
"args": [ "--a value --b value" ],
-
-
Set up python black.
-
You should now be able to place breakpoints in VSCode and see execution halt as the code hits them.
https://docs.python.org/3/library/pdb.html
pip install pdbpp
- If you need a breakpoint, put
import pdb; pdb.set_trace()
in your code - Run your command, it should break execution wherever you put the breakpoint.
This project welcomes contributions and suggestions. Most contributions require you to agree to a Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us the rights to use your contribution. For details, visit https://cla.microsoft.com.
When you submit a pull request, a CLA-bot will automatically determine whether you need to provide a CLA and decorate the PR appropriately (e.g., label, comment). Simply follow the instructions provided by the bot. You will only need to do this once across all repos using our CLA.