-
Notifications
You must be signed in to change notification settings - Fork 52
/
Copy pathaccessory.py
179 lines (142 loc) · 6.56 KB
/
accessory.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
import functools
import logging
from pyhap.accessory import Accessory
from pyhap.const import CATEGORY_DOOR_LOCK
from service import Service
log = logging.getLogger()
# Lock class performs no logic, forwarding requests to Service class
class Lock(Accessory):
category = CATEGORY_DOOR_LOCK
def __init__(self, *args, service: Service, lock_state_at_startup=1, **kwargs):
super().__init__(*args, **kwargs)
self._last_client_public_keys = None
self._lock_target_state = lock_state_at_startup
self._lock_current_state = lock_state_at_startup
self.service = service
self.service.on_endpoint_authenticated = self.on_endpoint_authenticated
self.add_lock_service()
self.add_nfc_access_service()
self.add_unpair_hook()
def on_endpoint_authenticated(self, endpoint):
self._lock_target_state = 0 if self._lock_current_state else 1
log.info(
f"Toggling lock state due to endpoint authentication event {self._lock_target_state} -> {self._lock_current_state} {endpoint}"
)
self.lock_target_state.set_value(self._lock_target_state, should_notify=True)
self._lock_current_state = self._lock_target_state
self.lock_current_state.set_value(self._lock_current_state, should_notify=True)
def add_unpair_hook(self):
unpair = self.driver.unpair
@functools.wraps(unpair)
def patched_unpair(client_uuid):
unpair(client_uuid)
self.on_unpair(client_uuid)
self.driver.unpair = patched_unpair
def add_preload_service(self, service, chars=None, unique_id=None):
"""Create a service with the given name and add it to this acc."""
if isinstance(service, str):
service = self.driver.loader.get_service(service)
if unique_id is not None:
service.unique_id = unique_id
if chars:
chars = chars if isinstance(chars, list) else [chars]
for char_name in chars:
if isinstance(char_name, str):
char = self.driver.loader.get_char(char_name)
service.add_characteristic(char)
else:
service.add_characteristic(char_name)
self.add_service(service)
return service
def add_info_service(self):
serv_info = self.driver.loader.get_service("AccessoryInformation")
serv_info.configure_char("Name", value=self.display_name)
serv_info.configure_char("SerialNumber", value="default")
serv_info.add_characteristic(self.driver.loader.get_char("HardwareFinish"))
serv_info.configure_char(
"HardwareFinish", getter_callback=self.get_hardware_finish
)
self.add_service(serv_info)
def add_lock_service(self):
self.service_lock_mechanism = self.add_preload_service("LockMechanism")
self.lock_current_state = self.service_lock_mechanism.configure_char(
"LockCurrentState", getter_callback=self.get_lock_current_state, value=0
)
self.lock_target_state = self.service_lock_mechanism.configure_char(
"LockTargetState",
getter_callback=self.get_lock_target_state,
setter_callback=self.set_lock_target_state,
value=0,
)
self.service_lock_management = self.add_preload_service("LockManagement")
self.lock_control_point = self.service_lock_management.configure_char(
"LockControlPoint",
setter_callback=self.set_lock_control_point,
)
self.lock_version = self.service_lock_management.configure_char(
"Version",
getter_callback=self.get_lock_version,
)
def add_nfc_access_service(self):
self.service_nfc = self.add_preload_service("NFCAccess")
self.char_nfc_access_supported_configuration = self.service_nfc.configure_char(
"NFCAccessSupportedConfiguration",
getter_callback=self.get_nfc_access_supported_configuration,
)
self.char_nfc_access_control_point = self.service_nfc.configure_char(
"NFCAccessControlPoint",
getter_callback=self.get_nfc_access_control_point,
setter_callback=self.set_nfc_access_control_point,
)
self.configuration_state = self.service_nfc.configure_char(
"ConfigurationState", getter_callback=self.get_configuration_state
)
def _update_hap_pairings(self):
client_public_keys = set(self.clients.values())
if self._last_client_public_keys == client_public_keys:
return
self._last_client_public_keys = client_public_keys
self.service.update_hap_pairings(client_public_keys)
def get_lock_current_state(self):
log.info("get_lock_current_state")
return self._lock_current_state
def get_lock_target_state(self):
log.info("get_lock_target_state")
return self._lock_target_state
def set_lock_target_state(self, value):
log.info(f"set_lock_target_state {value}")
self._lock_target_state = self._lock_current_state = value
self.lock_current_state.set_value(self._lock_current_state, should_notify=True)
return self._lock_target_state
def get_lock_version(self):
log.info("get_lock_version")
return ""
def set_lock_control_point(self, value):
log.info(f"set_lock_control_point: {value}")
# All methods down here are forwarded to Service
def get_hardware_finish(self):
self._update_hap_pairings()
log.info("get_hardware_finish")
return self.service.get_hardware_finish()
def get_nfc_access_supported_configuration(self):
self._update_hap_pairings()
log.info("get_nfc_access_supported_configuration")
return self.service.get_nfc_access_supported_configuration()
def get_nfc_access_control_point(self):
self._update_hap_pairings()
log.info("get_nfc_access_control_point")
return self.service.get_nfc_access_control_point()
def set_nfc_access_control_point(self, value):
self._update_hap_pairings()
log.info(f"set_nfc_access_control_point {value}")
return self.service.set_nfc_access_control_point(value)
def get_configuration_state(self):
self._update_hap_pairings()
log.info("get_configuration_state")
return self.service.get_configuration_state()
@property
def clients(self):
return self.driver.state.paired_clients
def on_unpair(self, client_id):
log.info(f"on_unpair {client_id}")
self._update_hap_pairings()