45
45
HAP_REPR_STATUS ,
46
46
HAP_REPR_TTL ,
47
47
HAP_REPR_VALUE ,
48
+ HAP_REPR_WRITE_RESPONSE ,
48
49
STANDALONE_AID ,
49
50
)
50
51
from pyhap .encoder import AccessoryEncoder
71
72
def _wrap_char_setter (char , value , client_addr ):
72
73
"""Process an characteristic setter callback trapping and logging all exceptions."""
73
74
try :
74
- char .client_update_value (value , client_addr )
75
+ result = char .client_update_value (value , client_addr )
75
76
except Exception : # pylint: disable=broad-except
76
77
logger .exception (
77
78
"%s: Error while setting characteristic %s to %s" ,
78
79
client_addr ,
79
80
char .display_name ,
80
81
value ,
81
82
)
82
- return HAP_SERVER_STATUS .SERVICE_COMMUNICATION_FAILURE
83
- return HAP_SERVER_STATUS .SUCCESS
83
+ return HAP_SERVER_STATUS .SERVICE_COMMUNICATION_FAILURE , None
84
+ return HAP_SERVER_STATUS .SUCCESS , result
84
85
85
86
86
87
def _wrap_acc_setter (acc , updates_by_service , client_addr ):
@@ -615,7 +616,7 @@ def async_update_advertisement(self):
615
616
self .mdns_service_info = AccessoryMDNSServiceInfo (
616
617
self .accessory , self .state , self .zeroconf_server
617
618
)
618
- asyncio . ensure_future (
619
+ util . async_create_background_task (
619
620
self .advertiser .async_update_service (self .mdns_service_info )
620
621
)
621
622
@@ -627,7 +628,7 @@ def async_persist(self):
627
628
"""
628
629
loop = asyncio .get_event_loop ()
629
630
logger .debug ("Scheduling write of accessory state to disk" )
630
- asyncio . ensure_future (loop .run_in_executor (None , self .persist ))
631
+ util . async_create_background_task (loop .run_in_executor (None , self .persist ))
631
632
632
633
def persist (self ):
633
634
"""Saves the state of the accessory.
@@ -851,6 +852,7 @@ def set_characteristics(self, chars_query, client_addr):
851
852
"iid": 2,
852
853
"value": False, # Value to set
853
854
"ev": True # (Un)subscribe for events from this characteristics.
855
+ "r": True # Request write response
854
856
}]
855
857
}
856
858
@@ -859,7 +861,9 @@ def set_characteristics(self, chars_query, client_addr):
859
861
# TODO: Add support for chars that do no support notifications.
860
862
updates = {}
861
863
setter_results = {}
864
+ setter_responses = {}
862
865
had_error = False
866
+ had_write_response = False
863
867
expired = False
864
868
865
869
if HAP_REPR_PID in chars_query :
@@ -872,6 +876,10 @@ def set_characteristics(self, chars_query, client_addr):
872
876
aid , iid = cq [HAP_REPR_AID ], cq [HAP_REPR_IID ]
873
877
setter_results .setdefault (aid , {})
874
878
879
+ if HAP_REPR_WRITE_RESPONSE in cq :
880
+ setter_responses .setdefault (aid , {})
881
+ had_write_response = True
882
+
875
883
if expired :
876
884
setter_results [aid ][iid ] = HAP_SERVER_STATUS .INVALID_VALUE_IN_REQUEST
877
885
had_error = True
@@ -904,11 +912,21 @@ def set_characteristics(self, chars_query, client_addr):
904
912
# Characteristic level setter callbacks
905
913
char = acc .get_characteristic (aid , iid )
906
914
907
- set_result = _wrap_char_setter (char , value , client_addr )
915
+ set_result , set_result_value = _wrap_char_setter (char , value , client_addr )
908
916
if set_result != HAP_SERVER_STATUS .SUCCESS :
909
917
had_error = True
918
+
910
919
setter_results [aid ][iid ] = set_result
911
920
921
+ if set_result_value is not None :
922
+ if setter_responses .get (aid , None ) is None :
923
+ logger .warning (
924
+ "Returning write response '%s' when it wasn't requested for %s %s" ,
925
+ set_result_value , aid , iid
926
+ )
927
+ had_write_response = True
928
+ setter_responses .setdefault (aid , {})[iid ] = set_result_value
929
+
912
930
if not char .service or (
913
931
not acc .setter_callback and not char .service .setter_callback
914
932
):
@@ -934,7 +952,7 @@ def set_characteristics(self, chars_query, client_addr):
934
952
for char in chars :
935
953
setter_results [aid ][char_to_iid [char ]] = set_result
936
954
937
- if not had_error :
955
+ if not had_error and not had_write_response :
938
956
return None
939
957
940
958
return {
@@ -943,6 +961,11 @@ def set_characteristics(self, chars_query, client_addr):
943
961
HAP_REPR_AID : aid ,
944
962
HAP_REPR_IID : iid ,
945
963
HAP_REPR_STATUS : status ,
964
+ ** (
965
+ {HAP_REPR_VALUE : setter_responses [aid ][iid ]}
966
+ if setter_responses .get (aid , {}).get (iid , None ) is not None
967
+ else {}
968
+ )
946
969
}
947
970
for aid , iid_status in setter_results .items ()
948
971
for iid , status in iid_status .items ()
0 commit comments