blob: 8a7acc884ab2faf64c733fa0199d78cfc9ecb2c8 [file] [log] [blame] [view]
# Working with Python CHIP Controller
The Python CHIP controller is a library that allows to create a Matter fabric
and commission Matter devices with it.
The `chip-repl` is a REPl which sets up a Python CHIP Controller and allows to
explore the Python CHIP Controller API and communicate with devices from the
command line.
<hr>
- [Source files](#source-files)
- [Building Python CHIP Controller](#building-and-installing)
- [Running the CHIP REPL](#running-the-chip-repl)
- [Using Python CHIP Controller REPL for Matter accessory testing](#using-python-chip-controller-repl-for-matter-accessory-testing)
- [Example usage of the Python CHIP Controller REPL](#example-usage-of-the-python-chip-controller-repl)
- [Explore Clusters, Attributes and Commands](#explore-clusters-attributes-and-commands)
<hr>
## Source files
You can find source files of the Python CHIP Controller tool in the
`src/controller/python` directory.
The tool uses the generic CHIP Device Controller library, available in the
`src/controller` directory.
<hr>
## Building and installing
Before you can use the Python controller, you must compile it from the source on
Linux (amd64 / aarch64) or macOS.
> To ensure compatibility, build the Python CHIP controller and the Matter
> device from the same revision of the connectedhomeip repository.
To build and run the Python CHIP controller:
1. Install all necessary packages and prepare the build system. For more
details, see the [Building Matter](BUILDING.md) documentation:
```
sudo apt-get update
sudo apt-get upgrade
sudo apt-get install git gcc g++ python pkg-config libssl-dev libdbus-1-dev libglib2.0-dev libavahi-client-dev ninja-build python3-venv python3-dev python3-pip unzip libgirepository1.0-dev libcairo2-dev bluez
```
If the Python CHIP controller is built on a Raspberry Pi, install additional
packages and reboot the device:
```
sudo apt-get install pi-bluetooth
sudo reboot
```
2. Clone the Project CHIP repository:
```
git clone https://github.com/project-chip/connectedhomeip.git
```
3. Enter the `connectedhomeip` directory:
```
cd connectedhomeip
```
4. Initialize the git submodules:
```
git submodule update --init
```
5. Build and install the Python CHIP controller:
```
scripts/build_python.sh -m platform -i separate
```
> Note: This builds the Python CHIP Controller along with the CHIP REPL as
> Python wheels and installs it into a separate Python virtual environment.
> To get more details about available build configurations, run the
> following command: `scripts/build_python.sh --help`
<hr>
## Running the CHIP REPL
1. Activate the Python virtual environment with the Python CHIP Controller
installed:
```
source out/python_env/bin/activate
```
2. Run the CHIP REPL to explore the API of the Python CHIP controller:
```
chip-repl
```
<hr>
## Using Python CHIP Controller REPL for Matter accessory testing
This section describes how to use Python CHIP controller to test the Matter
accessory. Below steps depend on the application clusters that you implemented
on the device side and may be different for your accessory.
### Step 1: Prepare the Matter accessory.
This tutorial is using the Matter Light Bulb example with the Bluetooth LE
commissioning. However, you can adapt this procedure to other available Matter
examples.
Build and program the device with the Matter accessory firmware by following the
example's documentation.
### Step 2: Enable Bluetooth LE advertising on Matter accessory device.
Some examples are configured to advertise automatically on boot. Other examples
require physical trigger, for example pushing a button. Follow the documentation
of the Matter accessory example to learn how Bluetooth LE advertising is enabled
for the given example.
### Step 3: Discover commissionable Matter accessory device
An uncommissioned accessory device advertises over Bluetooth LE or via mDNS if
already on the network. Run the following command to scan all advertised Matter
devices:
```
devCtrl.DiscoverCommissionableNodes()
```
### Step 4: Set network pairing credentials
You must provide the controller with network credentials that will be further
used in the device commissioning procedure to configure the device with a
network interface, such as Thread or Wi-Fi.
#### Setting Thread network credentials
1. Fetch and store the current Active Operational Dataset from the Thread Border
Router. Depending on if the Thread Border Router is running on Docker or
natively on Raspberry Pi, execute the following commands:
- For Docker:
```
sudo docker exec -it otbr sh -c "sudo ot-ctl dataset active -x"
0e080000000000010000000300001335060004001fffe002084fe76e9a8b5edaf50708fde46f999f0698e20510d47f5027a414ffeebaefa92285cc84fa030f4f70656e5468726561642d653439630102e49c0410b92f8c7fbb4f9f3e08492ee3915fbd2f0c0402a0fff8
Done
```
- For native installation:
```
sudo ot-ctl dataset active -x
0e080000000000010000000300001335060004001fffe002084fe76e9a8b5edaf50708fde46f999f0698e20510d47f5027a414ffeebaefa92285cc84fa030f4f70656e5468726561642d653439630102e49c0410b92f8c7fbb4f9f3e08492ee3915fbd2f0c0402a0fff8
Done
```
Matter specification does not define how the Thread or Wi-Fi credentials are
obtained by Controller. For example, for Thread, instead of fetching
datasets directly from the Thread Border Router, you might also use a
different out-of-band method.
2. Set the previously obtained Active Operational Dataset as a byte array using
the following command:
```
thread_dataset = bytes.fromhex("0e080000000000010000000300001335060004001fffe002084fe76e9a8b5edaf50708fde46f999f0698e20510d47f5027a414ffeebaefa92285cc84fa030f4f70656e5468726561642d653439630102e49c0410b92f8c7fbb4f9f3e08492ee3915fbd2f0c0402a0fff8")
devCtrl.SetThreadOperationalDataset(thread_dataset)
```
#### Setting Wi-Fi network credentials
Assuming your Wi-Fi SSID is _TESTSSID_, and your Wi-Fi password is _P455W4RD_,
set the credentials to the controller by executing the following command:
```
devCtrl.SetWiFiCredentials(<ssid>, <password>)
```
### Step 5: Commission the Matter accessory device over Bluetooth LE
The controller uses a 12-bit value called **discriminator** to discern between
multiple commissionable device advertisements, as well as a 27-bit **setup PIN
code** to authenticate the device. You can find these values in the logging
terminal of the device (for example, UART). For example:
```
I: 254 [DL]Device Configuration:
I: 257 [DL] Serial Number: TEST_SN
I: 260 [DL] Vendor Id: 65521 (0xFFF1)
I: 263 [DL] Product Id: 32768 (0x8000)
I: 267 [DL] Hardware Version: 1
I: 270 [DL] Setup Pin Code: 20202021
I: 273 [DL] Setup Discriminator: 3840 (0xF00)
I: 278 [DL] Manufacturing Date: (not set)
I: 281 [DL] Device Type: 65535 (0xFFFF)
```
Run the following command to establish the secure connection over Bluetooth LE,
with the following assumptions for the Matter accessory device:
- The discriminator of the device is _3840_
- The setup pin code of the device is _20202021_
- The temporary Node ID is _1234_
```
devCtrl.ConnectBLE(3840, 20202021, 1234)
```
You can skip the last parameter, the Node ID, in the command. If you skip it,
the controller will assign it randomly. In that case, note down the Node ID,
because it is required later in the configuration process.
It is also possible to use the QR setup code instead. It typically is shown on
the terminal of the device as well. For example:
```
CHIP:SVR: SetupQRCode: [MT:-24J0AFN00KA0648G00]
```
Use the following command to commission the device with the QR code:
```
devCtrl.CommissionWithCode("MT:-24J0AFN00KA0648G00", 1234)
```
After connecting the device over Bluetooth LE, the controller will go through
the following stages:
- Establishing a secure connection that completes the PASE
(Password-Authenticated Session Establishment) session using SPAKE2+
protocol and results in printing the following log:
```
Secure Session to Device Established
```
- Providing the device with a network interface using ZCL Network
Commissioning cluster commands, and the network pairing credentials set in
the previous step.
- Discovering the IPv6 address of the Matter accessory using the SRP (Service
Registration Protocol) for Thread devices, or the mDNS (Multicast Domain
Name System) protocol for Wi-Fi or Ethernet devices. It results in printing
log that indicates that the node address has been updated. The IPv6 address
of the device is cached in the controller for later usage.
- Closing the Bluetooth LE connection, as the commissioning process is
finished and the Python CHIP controller is now using only the IPv6 traffic
to reach the device.
### Step 6: Control application clusters.
For the light bulb example, execute the following command to toggle the LED
state:
```
await devCtrl.SendCommand(1234, 1, Clusters.OnOff.Commands.Toggle())
```
To change the brightness of the LED, use the following command, with the level
value somewhere between 0 and 255.
```
commandToSend = LevelControl.Commands.MoveToLevel(level=50, transitionTime=Null, optionsMask=0, optionsOverride=0)
await devCtrl.SendCommand(1234, 1, commandToSend)
```
### Step 7: Read basic information out of the accessory.
Every Matter accessory device supports a Basic Information Cluster, which
maintains collection of attributes that a controller can obtain from a device,
such as the vendor name, the product name, or software version. Use
`ReadAttribute()` command to read those values from the device:
```
attributes = [
(0, Clusters.BasicInformation.Attributes.VendorName),
(0, Clusters.BasicInformation.Attributes.ProductName),
(0, Clusters.BasicInformation.Attributes.SoftwareVersion),
]
await devCtrl.ReadAttribute(1234, attributes)
```
> In REPL, you can type `Clusters.BasicInformation.Attributes.` and then use the
> TAB key.
<hr>
## Example usage of the Python CHIP Controller REPL
These section covers a few useful commands of the Python CHIP Controller along
with examples demonstrating how they can be called from the REPL.
The
[CHIP Device Controller API documentation offer](https://project-chip.github.io/connectedhomeip-doc/testing/ChipDeviceCtrlAPI.html#chip-chipdevicectrl)
the full list of available commands.
### `SetThreadOperationalDataset(<thread-dataset>)`
Provides the controller with Thread network credentials that will be used in the
device commissioning procedure to configure the device with a Thread interface.
```
thread_dataset = bytes.fromhex("0e080000000000010000000300001335060004001fffe002084fe76e9a8b5edaf50708fde46f999f0698e20510d47f5027a414ffeebaefa92285cc84fa030f4f70656e5468726561642d653439630102e49c0410b92f8c7fbb4f9f3e08492ee3915fbd2f0c0402a0fff8")
devCtrl.SetThreadOperationalDataset(thread_dataset)
```
### `SetWiFiCredentials(<ssid>: str, <password>: str)`
Provides the controller with Wi-Fi network credentials that will be used in the
device commissioning procedure to configure the device with a Wi-Fi interface.
```
devCtrl.SetWiFiCredentials('TESTSSID', 'P455W4RD')
```
### `CommissionWithCode(<setupPayload>: str, <nodeid>: int, <discoveryType>: DiscoveryType)`
Commission with the given nodeid from the setupPayload. setupPayload may be a QR
or the manual setup code.
```
devCtrl.CommissionWithCode("MT:-24J0AFN00KA0648G00", 1234)
```
### `SendCommand(<nodeid>: int, <endpoint>: int, Clusters.<cluster>.Commands.<command>(<arguments>))`
Send a Matter command to the device. For example:
```python
commandToSend = Clusters.LevelControl.Commands.MoveWithOnOff(moveMode=1, rate=2, optionsMask=0, optionsOverride=0)
await devCtrl.SendCommand(1234, 1, commandToSend)
```
To see available arguments just create a command object without argument:
```
Clusters.LevelControl.Commands.MoveWithOnOff()
```
Shows which arguments are available:
```
MoveWithOnOff(
│ moveMode=0,
│ rate=Null,
│ optionsMask=0,
│ optionsOverride=0
)
```
### `ReadAttribute(<nodeid>: int, [(<endpoint id>: int, Clusters.<cluster>.Attributes.<attribute>)])`
Read the value of an attribute. For example:
```python
await devCtrl.ReadAttribute(1234, [(0, Clusters.BasicInformation.Attributes.VendorName)])
```
### `WriteAttribute(<nodeid>: int, [(<endpointid>: int, Clusters.<cluster>.Attributes.<attribute>(value=<attribute value>))])`
Write a value to an attribute. For example:
```python
await devCtrl.WriteAttribute(1234, [(1, Clusters.UnitTesting.Attributes.Int8u(value=1))])
await devCtrl.WriteAttribute(1234, [(1, Clusters.UnitTesting.Attributes.Boolean(value=True))])
await devCtrl.WriteAttribute(1234, [(1, Clusters.UnitTesting.Attributes.OctetString(value=b'123123\x00'))])
await devCtrl.WriteAttribute(1234, [(1, Clusters.UnitTesting.Attributes.CharString(value='233233'))])
```
### `ReadAttribute(<nodeid>: int, [(<endpoint>: int, Clusters.<cluster>.Attributes.<attribute>)], reportInterval=(<min interval>: int, <max interval>: int))`
Configure Matter attribute reporting settings. For example:
```python
await devCtrl.ReadAttribute(1234, [(1, Clusters.OccupancySensing.Attributes.Occupancy)], reportInterval=(10, 20))
```
To shutdown an existing attribute subscription use the `Shutdown()` function on
the returned subscription object:
```python
sub = await devCtrl.ReadAttribute(1234, [(1, Clusters.OccupancySensing.Attributes.Occupancy)], reportInterval=(10, 20))
sub.Shutdown()
```
## Explore Clusters, Attributes and Commands
In the Python REPL the Clusters and Attributes are classes. The `Clusters`
module contains all clusters. Tab completion can be used to explore available
clusters, attributes and commands.
For example, to get a list of Clusters, type `Clusters.` and hit tab. Continue
to hit tab to cycle through the available Clusters. Pressing return will select
the Cluster.
To explore Attributes, use the same technique but with the Attributes sub-class
of the Clusters class, for example, type `Clusters.(cluster name).Attributes.`
and hit tab.
The same is true for Commands, use the Commands sub-class. type
`Clusters.(cluster name).Commands.` and hit tab.