Implementation of Control4's Simple Device Discovery Protocol (SDDP)
Project description
sddp-discovery-protocol: Protocol library for Control4's Simple Device Discovery Protocol (SDDP)
A command-line tool and API for servers and clients of Control4's Simple Device Discovery Protocol (SDDP).
Table of contents
- Introduction
- Installation
- Usage
- Known issues and limitations
- Getting help
- Contributing
- License
- Authors and history
Introduction
Simple Device Discovery Protocol (SDDP) is a simple multicast discovery protocol implemented by many "smart home" devices to allow a controlling agent to easily discover and connect to devices on a local subnet.
SDDP was created by Control4, a provider of whole-home automation systems. While it is quite similar to UPnP's standard Simple Service Discovery Protocol (SSDP--note the similar but different spelling), and it serves a virtually identical purpose, SDDP is not a standard protocol and it is not publicly documented. The protocol as implemented by this package was inferred through observation of network traffic
Servers
Controllable smart devices implement the server side of the SDDP protocol. They listen on a well-known UDP multicast address 239.255.255.250:1902, and do the following:
-
Periodically (typically every 20 minutes) send an SDDP NOTIFY packet to the multicast address advertising their presence. An example of such a packet is:
NOTIFY ALIVE SDDP/1.0\r\n From: "192.168.4.237:1902"\r\n Host: "JVC_PROJECTOR-E0DADC152802"\r\n Max-Age: 1800\r\n Type: "JVCKENWOOD:Projector"\r\n Primary-Proxy: "projector"\r\n Proxies: "projector"\r\n Manufacturer: "JVCKENWOOD"\r\n Model: "DLA-RS3100_NZ8"\r\n Driver: "projector_JVCKENWOOD_DLA-RS3100_NZ8.c4i"\r\n
Note that, as with all SDDP packets, there is a statement line, followed by some header lines, each separated with "\r\n". Formatting of the headers is with the same rules as HTTP headers. All string header values are enclosed in double quotes. This package assumes that json.loads() will correctly decode all header values.
-
Receive SDDP NOTIFY packets sent to the multicast address from other devices. Most devices will probably discard these unless they are interested in seeing other devices come and go
-
Receive and respond to SDDP SEARCH packets sent to the multicast address by clients that are performing discovery. An example of such a search packet is:
SEARCH * SDDP/1.0\r\n Host: "192.168.4.237:24378"\r\n
The device will respond to a valid SEARCH request packet with a SEARCH response packet sent from their own unicast address on port 1902 to the unicast address from which the request came. The response looks very much like the NOTIFY packet, but the statement line is different; e.g.,:
SDDP/1.0 200 OK\r\n From: "192.168.4.237:1902"\r\n Host: "JVC_PROJECTOR-E0DADC152802"\r\n Max-Age: 1800\r\n Type: "JVCKENWOOD:Projector"\r\n Primary-Proxy: "projector"\r\n Proxies: "projector"\r\n Manufacturer: "JVCKENWOOD"\r\n Model: "DLA-RS3100_NZ8"\r\n Driver: "projector_JVCKENWOOD_DLA-RS3100_NZ8.c4i"\r\n
Clients
Apps that wish to discover devices on the local subnet implement the client side of SDDP. They send from and receive on an arbitrary dynamic UDP port on their unicast address.
To perform discovery, an SDDP client sends an SDDP SEARCH packet from its unicast address and port to to the SDDP multicast address 239.255.255.250:1902; e.g.:
SEARCH * SDDP/1.0\r\n
Host: "192.168.4.237:24378"\r\n
where the 'Host' header is set to the client's unicast address and port.
Each SDDP server device on the network will immediately respond to the SEARCH packet with a SEARCH response packet sent directly to the client's unicast address and port; e.g.,:
SDDP/1.0 200 OK\r\n
From: "192.168.4.237:1902"\r\n
Host: "JVC_PROJECTOR-E0DADC152802"\r\n
Max-Age: 1800\r\n
Type: "JVCKENWOOD:Projector"\r\n
Primary-Proxy: "projector"\r\n
Proxies: "projector"\r\n
Manufacturer: "JVCKENWOOD"\r\n
Model: "DLA-RS3100_NZ8"\r\n
Driver: "projector_JVCKENWOOD_DLA-RS3100_NZ8.c4i"\r\n
The client will then collect the responses from all server devices and use them for discovery.
The client waits for an arbitrary period of time (by default, in this package, 3 seconds) before assuming that all devices have responded.
Package
Python package sddp-discovery-protocol
provides a command-line tool as well as a runtime API for implementing the client and server sides of SDDP.
Some key features of sddp-discovery-protocol:
- Cross-platform (Linux, MacOS, Windows)
- Fully async model with coroutine pattern (no synchronous callbacks).
- Both client and server supports multi-homed operation (multiple network adapters)
- Server customizes 'From' header for each
- Server supports periodic multicast NOTIFY advertisement and responds to SEARCH queries
- Server supports collection of NOTIFY advertisements from other SDDP servers on the subnet
- Client waits for a configurable time for SEARCH responses
- 'sddp' command-line tool allow performing searches as well as running a standalone server
Installation
Prerequisites
Python: Python 3.8+ is required. See your OS documentation for instructions.
From PyPi
The current released version of sddp-discovery-protocol
can be installed with
pip3 install sddp-discovery-protocol
From GitHub
Poetry is required; it can be installed with:
curl -sSL https://install.python-poetry.org | python3 -
Clone the repository and install sddp-discovery-protocol into a private virtualenv with:
cd <parent-folder>
git clone https://github.com/sammck/sddp-discovery-protocol.git
cd sddp-discovery-protocol
poetry install
You can then launch a bash shell with the virtualenv activated using:
poetry shell
Usage
Command Line
There is a single command tool sddp
that is installed with the package.
Getting the package version
$ sddp version
0.1.0
$
Discovering devices on the local subnet
$ sddp search --help
usage: sddp search [-h] [--pattern PATTERN] [--wait-time WAIT_TIME] [-b BIND_ADDRESSES] [--include-error-responses]
Search for SDDP devices
options:
-h, --help show this help message and exit
--pattern PATTERN The device pattern to search for. Default: "*"
--wait-time WAIT_TIME
The amount of time to wait for responses, in seconds. Default: 3.0
-b BIND_ADDRESSES, --bind BIND_ADDRESSES
The local unicast IP address to bind to. May be repeated. Default: all local non-loopback unicast addresses.
--include-error-responses
Include error responses in the output. Default: False
$ sddp search
[
{
"headers": {
"Driver": "repeater_ip_lutron_radiora2.c4i",
"From": "192.168.4.201:1902",
"Host": "lutron-032e345c",
"Manufacturer": "Lutron",
"Max-Age": 1800,
"Model": "RadioRA2 Main Repeater",
"Primary-Proxy": "repeater_ip_lutron_radiora2",
"Proxies": "repeater_ip_lutron_radiora2",
"Type": "lutron:repeater_ip_lutron_radiora2"
},
"local_addr": "192.168.4.238:59060",
"monotonic_time": 200338.359314361,
"sddp_version": "1.0",
"src_addr": "192.168.4.64:54838",
"status": "OK",
"status_code": 200,
"utc_time": "2023-07-24T00:13:30.734784"
},
{
"headers": {
"Driver": "sonos.c4z",
"From": "192.168.4.183:1902",
"Host": "Sonos-347E5CD9C83A",
"Manufacturer": "Sonos",
"Max-Age": 1800,
"Model": "Zoneplayer",
"Primary-Proxy": "media_service",
"Proxies": "media_service,amplifier",
"Type": "sonos:Zoneplayer"
},
"local_addr": "192.168.4.238:59060",
"monotonic_time": 200338.474668528,
"sddp_version": "1.0",
"src_addr": "192.168.4.183:55292",
"status": "OK",
"status_code": 200,
"utc_time": "2023-07-24T00:13:30.850138"
}
]
If you know the expected value of one or more response headers and are looking for a specific response, you can speed up the search in the successful case by applying a header filter and limiting the responses to 1; in this case the search will terminate as soon as the relevant response is received:
$ sddp search --max-responses 1 -F Type=JVCKENWOOD:Projector
{
"headers": {
"Driver": "projector_JVCKENWOOD_DLA-RS3100_NZ8.c4i",
"From": "192.168.4.237:1902",
"Host": "JVC_PROJECTOR-E0DADC152802",
"Manufacturer": "JVCKENWOOD",
"Max-Age": 1800,
"Model": "DLA-RS3100_NZ8",
"Primary-Proxy": "projector",
"Proxies": "projector",
"Type": "JVCKENWOOD:Projector"
},
"local_addr": "192.168.4.198:58941",
"monotonic_time": 0.093766125,
"sddp_version": "1.0",
"src_addr": "192.168.4.237:1902",
"status": "OK",
"status_code": 200,
"utc_time": "2023-07-24T22:23:54.175783"
}
Running an SDDP server
$ sddp server --help
usage: sddp server [-h] [--advertise-interval ADVERTISE_INTERVAL] [-H HEADERS] [-b BIND_ADDRESSES]
Run an SDDP server
options:
-h, --help show this help message and exit
--advertise-interval ADVERTISE_INTERVAL
The interval at which to send device advertisements,
in seconds. Default: 2/3 of Max-Age header, or
1200 seconds (20 minutes)
-H HEADERS, --header HEADERS
A <name>=<value> header to include in the device
advertisement. May be repeated.
-b BIND_ADDRESSES, --bind BIND_ADDRESSES
The local unicast IP address to bind to.
May be repeated. Default: all local non-loopback
unicast addresses.
$ sddp --log-level=debug server
...
API
Discovering devices on the local subnet
import logging
import asyncio
import sddp_discovery_protocol as sddp
#logging.basicConfig(level=logging.DEBUG)
async def amain():
# all parameters to SddpClient are optional; they allow you to set the IP addresses to bind to, etc.
async with sddp.SddpClient() as client:
# Entering the client.search() context manager sends the search multicast request and reliably collects responses.
# Parameters are optional; they allow you to set search filters, max wait time, max returned responses, etc.
async with client.search() as search_request:
# search_request.iter_responses() is an async generator that yields SddpResponseInfo objects
# as they come in until the max wait time has elapsed or the max number of responses has been received.
async for response_info in search_request.iter_responses():
print(response_info.datagram)
# It is possible to exit the loop early here if you found what you're looking for
loop = asyncio.new_event_loop()
try:
asyncio.set_event_loop(loop)
loop.run_until_complete(amain())
finally:
loop.close()
Running an SDDP server
import logging
import asyncio
import sddp_discovery_protocol as sddp
logging.basicConfig(level=logging.DEBUG)
device_headers = {
"Type": "Acme:TestDevice",
"Primary-Proxy": "test-device",
"Proxies": "test-device",
"Manufacturer": "Acme",
"Model": "TestDevPlus",
"Driver": "test-device_Acme_TestDevPlus.c4i",
}
async def amain():
# The SddpServerContext manager starts the server listening on the multicast port, sending out advertisements,
# and responding to search requests. When the context manager exits, the server will be stopped.
async with sddp.SddpServer(device_headers=device_headers) as server:
# This will wait forever unless another task stops the server
await server.wait_for_done()
loop = asyncio.new_event_loop()
try:
asyncio.set_event_loop(loop)
loop.run_until_complete(amain())
finally:
loop.close()
Known issues and limitations
- SDDP is a proprietary protocol defined by Control4 with responders implemented by its smart-home business partners. The protocol is not standard and is not publicly documented. This package is a best effort to provide an open implementation of the protocol based on limited observation of network traffic. There is no way to know if the implementation is complete or totally accurate.
Getting help
Please report any problems/issues here.
Contributing
Pull requests welcome.
License
sddp-discovery-protocol is distributed under the terms of the MIT License. The license applies to this file and other files in the GitHub repository hosting this file.
Authors and history
The author of sddp-discovery-protocol is Sam McKelvie.
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file sddp_discovery_protocol-1.5.0.tar.gz
.
File metadata
- Download URL: sddp_discovery_protocol-1.5.0.tar.gz
- Upload date:
- Size: 31.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.5.1 CPython/3.10.11 Linux/5.15.0-1041-azure
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 31478f7a8860e81643badf338d4f14471380462049f7b777868d759b83a38600 |
|
MD5 | 1938093a98b45fe4e4b94c21385613af |
|
BLAKE2b-256 | 4960ef92aa2a78aa21577f32933f47ee326fe72b0649192ff0648d0c69057772 |
File details
Details for the file sddp_discovery_protocol-1.5.0-py3-none-any.whl
.
File metadata
- Download URL: sddp_discovery_protocol-1.5.0-py3-none-any.whl
- Upload date:
- Size: 33.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.5.1 CPython/3.10.11 Linux/5.15.0-1041-azure
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | e0e6376f1f1ca6db1634fe6ca6b3b37dee4e95c7b91bd83ecd35e607daf1a899 |
|
MD5 | cf936a3d19e999a62ad7c7c08b41985c |
|
BLAKE2b-256 | be35577c4c5fc9d3e9206d0edf2bb3e85aeb8b8afced6dc41fd61c9774b17d53 |