uPnP client library for Python 3.
This library can be used to discover and consume uPnP devices and their services.
It's originally based on Ferry Boender's work and his blog post entitled Exploring UPnP with Python.
Python 3.9+
pip install upnpclient>>> import upnpclient
>>> devices = upnpclient.discover()
>>> devices
[<Device 'OpenWRT router'>,
<Device 'Harmony Hub'>,
<Device 'walternate: root'>]If you already know the URL for a device's description XML, you can create it directly:
>>> d = upnpclient.Device("http://192.168.1.1:5000/rootDesc.xml")Each device exposes different services depending on what it is. Check what's available:
>>> d.services
[<Service service_id='urn:upnp-org:serviceId:Layer3Forwarding1'>,
<Service service_id='urn:upnp-org:serviceId:WANCommonIFC1'>,
<Service service_id='urn:upnp-org:serviceId:WANIPConn1'>]Services are accessed by the last part of their service_id. For example, urn:upnp-org:serviceId:WANIPConn1 becomes d.WANIPConn1:
>>> d.WANIPConn1.actions
[<Action 'GetStatusInfo'>,
<Action 'GetNATRSIPStatus'>,
<Action 'GetExternalIPAddress'>,
<Action 'AddPortMapping'>,
...]If the service name isn't a valid Python attribute, use dictionary-style access:
>>> d["WANIPConn1"]["GetStatusInfo"]()>>> d.WANIPConn1.GetStatusInfo()
{'NewConnectionStatus': 'Connected',
'NewLastConnectionError': 'ERROR_NONE',
'NewUptime': 14851479}
>>> d.WANIPConn1.GetExternalIPAddress()
{'NewExternalIPAddress': '123.123.123.123'}To see what arguments an action expects:
>>> d.WANIPConn1.AddPortMapping.argsdef_in
[('NewRemoteHost',
{'allowed_values': set(), 'datatype': 'string', 'name': 'RemoteHost'}),
('NewExternalPort',
{'allowed_values': set(), 'datatype': 'ui2', 'name': 'ExternalPort'}),
('NewProtocol',
{'allowed_values': {'TCP', 'UDP'},
'datatype': 'string',
'name': 'PortMappingProtocol'}),
('NewInternalPort',
{'allowed_values': set(), 'datatype': 'ui2', 'name': 'InternalPort'}),
('NewInternalClient',
{'allowed_values': set(), 'datatype': 'string', 'name': 'InternalClient'}),
('NewEnabled',
{'allowed_values': set(),
'datatype': 'boolean',
'name': 'PortMappingEnabled'}),
('NewPortMappingDescription',
{'allowed_values': set(),
'datatype': 'string',
'name': 'PortMappingDescription'}),
('NewLeaseDuration',
{'allowed_values': set(),
'datatype': 'ui4',
'name': 'PortMappingLeaseDuration'})]Then call it with those arguments:
>>> d.WANIPConn1.AddPortMapping(
... NewRemoteHost='0.0.0.0',
... NewExternalPort=12345,
... NewProtocol='TCP',
... NewInternalPort=12345,
... NewInternalClient='192.168.1.10',
... NewEnabled='1',
... NewPortMappingDescription='Testing',
... NewLeaseDuration=10000)
{}Similarly, argsdef_out shows what an action returns:
>>> d.WANIPConn1.GetStatusInfo.argsdef_out
[('NewConnectionStatus',
{'allowed_values': {'Connected', 'Disconnected', 'Connecting'},
'datatype': 'string',
'name': 'ConnectionStatus'}),
('NewLastConnectionError',
{'allowed_values': set(),
'datatype': 'string',
'name': 'LastConnectionError'}),
('NewUptime',
{'allowed_values': set(), 'datatype': 'ui4', 'name': 'Uptime'})]You may pass a requests compatible authentication object and/or a dictionary containing headers to use on the HTTP calls to your uPnP device.
These may be set on the Device itself on creation for use with every HTTP
call:
device = upnpclient.Device(
"http://192.168.1.1:5000/rootDesc.xml",
http_auth=('myusername', 'mypassword'),
http_headers={'Some-Required-Header': 'somevalue'},
)Or on a per-call basis:
device.Layer3Forwarding1.GetDefaultConnectionService(
http_auth=('myusername', 'mypassword'),
http_headers={'Some-Required-Header': 'somevalue'},
)If you've set either at Device level, they can be overridden per-call by
setting them to None.