1
1
"""Module for the Accessory classes."""
2
2
import itertools
3
3
import logging
4
+ from typing import TYPE_CHECKING , Any , Callable , Dict , Iterable , List , Optional
4
5
from uuid import UUID
5
6
6
- from pyhap import SUPPORT_QR_CODE , util
7
- from pyhap .const import (
7
+ from . import SUPPORT_QR_CODE , util
8
+ from .const import (
8
9
CATEGORY_BRIDGE ,
9
10
CATEGORY_OTHER ,
10
11
HAP_PROTOCOL_VERSION ,
14
15
HAP_REPR_VALUE ,
15
16
STANDALONE_AID ,
16
17
)
17
- from pyhap .iid_manager import IIDManager
18
- from pyhap .service import Service
18
+ from .iid_manager import IIDManager
19
+ from .service import Service
19
20
20
21
if SUPPORT_QR_CODE :
21
22
import base36
22
23
from pyqrcode import QRCode
23
24
24
25
26
+ if TYPE_CHECKING :
27
+ from .accessory_driver import AccessoryDriver
28
+ from .characteristic import Characteristic
29
+
30
+
25
31
HAP_PROTOCOL_INFORMATION_SERVICE_UUID = UUID ("000000A2-0000-1000-8000-0026BB765291" )
26
32
27
33
logger = logging .getLogger (__name__ )
@@ -35,7 +41,13 @@ class Accessory:
35
41
36
42
category = CATEGORY_OTHER
37
43
38
- def __init__ (self , driver , display_name , aid = None , iid_manager = None ):
44
+ def __init__ (
45
+ self ,
46
+ driver : "AccessoryDriver" ,
47
+ display_name : Optional [str ],
48
+ aid : Optional [int ] = None ,
49
+ iid_manager : Optional [IIDManager ] = None ,
50
+ ) -> None :
39
51
"""Initialise with the given properties.
40
52
41
53
:param display_name: Name to be displayed in the Home app.
@@ -47,24 +59,24 @@ def __init__(self, driver, display_name, aid=None, iid_manager=None):
47
59
will assign the standalone AID to this `Accessory`.
48
60
:type aid: int
49
61
"""
50
- self .aid = aid
51
- self .display_name = display_name
62
+ self .aid : Optional [ int ] = aid
63
+ self .display_name : Optional [ str ] = display_name
52
64
self .driver = driver
53
- self .services = []
65
+ self .services : List [ Service ] = []
54
66
self .iid_manager = iid_manager or IIDManager ()
55
- self .setter_callback = None
67
+ self .setter_callback : Optional [ Callable [[ Any ], None ]] = None
56
68
57
69
self .add_info_service ()
58
70
if aid == STANDALONE_AID :
59
71
self .add_protocol_version_service ()
60
72
61
- def __repr__ (self ):
73
+ def __repr__ (self ) -> str :
62
74
"""Return the representation of the accessory."""
63
75
services = [s .display_name for s in self .services ]
64
76
return f"<accessory display_name='{ self .display_name } ' services={ services } >"
65
77
66
78
@property
67
- def available (self ):
79
+ def available (self ) -> bool :
68
80
"""Accessory is available.
69
81
70
82
If available is False, get_characteristics will return
@@ -75,7 +87,7 @@ def available(self):
75
87
"""
76
88
return True
77
89
78
- def add_info_service (self ):
90
+ def add_info_service (self ) -> None :
79
91
"""Helper method to add the required `AccessoryInformation` service.
80
92
81
93
Called in `__init__` to be sure that it is the first service added.
@@ -116,7 +128,12 @@ def set_info_service(
116
128
self .display_name ,
117
129
)
118
130
119
- def add_preload_service (self , service , chars = None , unique_id = None ):
131
+ def add_preload_service (
132
+ self ,
133
+ service : Service ,
134
+ chars : Optional [Iterable ["Characteristic" ]] = None ,
135
+ unique_id : Optional [str ] = None ,
136
+ ) -> Service :
120
137
"""Create a service with the given name and add it to this acc."""
121
138
service = self .driver .loader .get_service (service )
122
139
if unique_id is not None :
@@ -129,12 +146,12 @@ def add_preload_service(self, service, chars=None, unique_id=None):
129
146
self .add_service (service )
130
147
return service
131
148
132
- def set_primary_service (self , primary_service ) :
149
+ def set_primary_service (self , primary_service : Service ) -> None :
133
150
"""Set the primary service of the acc."""
134
151
for service in self .services :
135
152
service .is_primary_service = service .type_id == primary_service .type_id
136
153
137
- def add_service (self , * servs ) :
154
+ def add_service (self , * servs : Service ) -> None :
138
155
"""Add the given services to this Accessory.
139
156
140
157
This also assigns unique IIDS to the services and their Characteristics.
@@ -153,7 +170,7 @@ def add_service(self, *servs):
153
170
c .broker = self
154
171
self .iid_manager .assign (c )
155
172
156
- def get_service (self , name ) :
173
+ def get_service (self , name : str ) -> Optional [ Service ] :
157
174
"""Return a Service with the given name.
158
175
159
176
A single Service is returned even if more than one Service with the same name
@@ -168,7 +185,7 @@ def get_service(self, name):
168
185
"""
169
186
return next ((s for s in self .services if s .display_name == name ), None )
170
187
171
- def xhm_uri (self ):
188
+ def xhm_uri (self ) -> str :
172
189
"""Generates the X-HM:// uri (Setup Code URI)
173
190
174
191
:rtype: str
@@ -195,7 +212,7 @@ def xhm_uri(self):
195
212
196
213
return "X-HM://" + encoded_payload + self .driver .state .setup_id
197
214
198
- def get_characteristic (self , aid , iid ) :
215
+ def get_characteristic (self , aid : int , iid : int ) -> Optional [ "Characteristic" ] :
199
216
"""Get the characteristic for the given IID.
200
217
201
218
The AID is used to verify if the search is in the correct accessory.
@@ -205,7 +222,7 @@ def get_characteristic(self, aid, iid):
205
222
206
223
return self .iid_manager .get_obj (iid )
207
224
208
- def to_HAP (self ) :
225
+ def to_HAP (self , include_value : bool = True ) -> Dict [ str , Any ] :
209
226
"""A HAP representation of this Accessory.
210
227
211
228
:return: A HAP representation of this accessory. For example:
@@ -224,7 +241,7 @@ def to_HAP(self):
224
241
"""
225
242
return {
226
243
HAP_REPR_AID : self .aid ,
227
- HAP_REPR_SERVICES : [s .to_HAP () for s in self .services ],
244
+ HAP_REPR_SERVICES : [s .to_HAP (include_value = include_value ) for s in self .services ],
228
245
}
229
246
230
247
def setup_message (self ):
@@ -325,13 +342,18 @@ class Bridge(Accessory):
325
342
326
343
category = CATEGORY_BRIDGE
327
344
328
- def __init__ (self , driver , display_name , iid_manager = None ):
345
+ def __init__ (
346
+ self ,
347
+ driver : "AccessoryDriver" ,
348
+ display_name : Optional [str ],
349
+ iid_manager : Optional [IIDManager ] = None ,
350
+ ) -> None :
329
351
super ().__init__ (
330
352
driver , display_name , aid = STANDALONE_AID , iid_manager = iid_manager
331
353
)
332
354
self .accessories = {} # aid: acc
333
355
334
- def add_accessory (self , acc ) :
356
+ def add_accessory (self , acc : "Accessory" ) -> None :
335
357
"""Add the given ``Accessory`` to this ``Bridge``.
336
358
337
359
Every ``Accessory`` in a ``Bridge`` must have an AID and this AID must be
@@ -364,14 +386,14 @@ def add_accessory(self, acc):
364
386
365
387
self .accessories [acc .aid ] = acc
366
388
367
- def to_HAP (self ) :
389
+ def to_HAP (self , include_value : bool = True ) -> List [ Dict [ str , Any ]] :
368
390
"""Returns a HAP representation of itself and all contained accessories.
369
391
370
392
.. seealso:: Accessory.to_HAP
371
393
"""
372
- return [acc .to_HAP () for acc in (super (), * self .accessories .values ())]
394
+ return [acc .to_HAP (include_value = include_value ) for acc in (super (), * self .accessories .values ())]
373
395
374
- def get_characteristic (self , aid , iid ) :
396
+ def get_characteristic (self , aid : int , iid : int ) -> Optional [ "Characteristic" ] :
375
397
""".. seealso:: Accessory.to_HAP"""
376
398
if self .aid == aid :
377
399
return self .iid_manager .get_obj (iid )
@@ -382,17 +404,17 @@ def get_characteristic(self, aid, iid):
382
404
383
405
return acc .get_characteristic (aid , iid )
384
406
385
- async def run (self ):
407
+ async def run (self ) -> None :
386
408
"""Schedule tasks for each of the accessories' run method."""
387
409
for acc in self .accessories .values ():
388
410
self .driver .async_add_job (acc .run )
389
411
390
- async def stop (self ):
412
+ async def stop (self ) -> None :
391
413
"""Calls stop() on all contained accessories."""
392
414
await self .driver .async_add_job (super ().stop )
393
415
for acc in self .accessories .values ():
394
416
await self .driver .async_add_job (acc .stop )
395
417
396
418
397
- def get_topic (aid , iid ) :
419
+ def get_topic (aid : int , iid : int ) -> str :
398
420
return str (aid ) + "." + str (iid )
0 commit comments