forked from cbrand/micropython-mdns
-
Notifications
You must be signed in to change notification settings - Fork 1
/
client.py
238 lines (199 loc) · 8.78 KB
/
client.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
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
import gc
import socket
import time
from collections import namedtuple
from select import select
import uasyncio
from mdns_client.constants import CLASS_IN, LOCAL_MDNS_SUFFIX, MAX_PACKET_SIZE, MDNS_ADDR, MDNS_PORT, TYPE_A
from mdns_client.parser import parse_packet
from mdns_client.structs import DNSQuestion, DNSQuestionWrapper, DNSRecord, DNSResponse
from mdns_client.util import a_record_rdata_to_string, dotted_ip_to_bytes, set_after_timeout
class Callback(namedtuple("Callback", ["id", "callback", "remove_if", "timeout", "created_ticks"])):
id: int
callback: "Callable[[DNSResponse], Awaitable[None]]"
remove_if: "Optional[Callable[[DNSResponse], Awaitable[bool]]]"
timeout: "Optional[float]"
created_ticks: int
@property
def timedout(self) -> bool:
if self.timeout is None:
return False
return self.created_ticks + int(self.timeout * 1000) < time.ticks_ms()
class Client:
def __init__(self, local_addr: str, debug: bool = False):
self.socket: "Optional[socket.socket]" = None
self.local_addr = local_addr
self.debug = debug
self.print_packets = debug
self.stopped = True
self.callbacks: "Dict[int, Callback]" = {}
self.callback_fd_count: int = 0
self.mdns_timeout = 2.0
def add_callback(
self,
callback: "Callable[[DNSResponse], Awaitable[None]]",
remove_if: "Optional[Callable[[DNSResponse], Awaitable[bool]]]" = None,
timeout: "Optional[int]" = None,
) -> Callback:
callback_config = Callback(
id=self.callback_fd_count,
callback=callback,
remove_if=remove_if,
timeout=timeout,
created_ticks=time.ticks_ms(),
)
self.callback_fd_count += 1
self.dprint("Adding callback with id {}".format(callback_config.id))
self.callbacks[callback_config.id] = callback_config
if self.stopped:
self.dprint("Added consumer on stopped mdns client. Starting it now.")
self.stopped = False
loop = uasyncio.get_event_loop()
loop.create_task(self.start())
return callback_config
def _make_socket(self) -> socket.socket:
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
member_info = dotted_ip_to_bytes(MDNS_ADDR) + dotted_ip_to_bytes(self.local_addr)
sock.setsockopt(socket.IPPROTO_IP, socket.IP_ADD_MEMBERSHIP, member_info)
sock.setblocking(False)
return sock
async def start(self) -> None:
self.stopped = False
self._init_socket()
await self.consume()
def _init_socket(self) -> None:
self._close_socket()
self.socket = self._make_socket()
self.socket.bind(("", MDNS_PORT))
def stop(self) -> None:
self.stopped = True
self._close_socket()
def _close_socket(self) -> None:
if self.socket is not None:
self.socket.close()
self.socket = None
async def consume(self) -> None:
while not self.stopped:
await self.process_waiting_data()
await uasyncio.sleep_ms(100)
async def process_waiting_data(self) -> None:
while not self.stopped:
readers, _, _ = select([self.socket], [], [], 0)
if not readers:
break
try:
buffer, addr = self.socket.recvfrom(MAX_PACKET_SIZE)
except MemoryError:
# This seems to happen here without SPIRAM sometimes.
self.dprint(
"Issue processing network data due to insufficient memory. "
"Rebooting the socket to free up cache buffer."
)
self._init_socket()
continue
if addr[0] == self.local_addr:
continue
try:
await self.process_packet(buffer)
except Exception as e:
self.dprint("Issue processing packet: {}".format(e))
finally:
gc.collect()
async def process_packet(self, buffer: bytes) -> None:
parsed_packet = parse_packet(buffer)
if len(self.callbacks) == 0:
if self.print_packets:
print(parsed_packet)
else:
loop = uasyncio.get_event_loop()
for callback in self.callbacks.values():
loop.create_task(callback.callback(parsed_packet))
if callback.timedout:
self.remove_if_present(callback)
elif callback.remove_if is not None:
loop.create_task(self.remove_if_check(callback, parsed_packet))
async def remove_if_check(self, callback: Callback, message: DNSResponse) -> None:
if await callback.remove_if(message):
return self.remove_if_present(callback)
def remove_if_present(self, callback: Callback) -> None:
self.remove_id(callback.id)
def remove_id(self, callback_id: int) -> bool:
deleted = False
if callback_id in self.callbacks:
self.dprint("Removing callback with id {}".format(callback_id))
del self.callbacks[callback_id]
deleted = True
if len(self.callbacks) == 0 and not self.print_packets:
self.dprint("Stopping consumption pipeline as no listeners exist")
self.stop()
return deleted
async def send_question(self, *questions: DNSQuestion) -> None:
question_wrapper = DNSQuestionWrapper(questions=questions)
self._send_bytes(question_wrapper.to_bytes())
async def send_response(self, response: DNSResponse) -> None:
self._send_bytes(response.to_bytes())
def _send_bytes(self, payload: bytes) -> None:
self._init_socket_if_not_done()
try:
self.socket.sendto(payload, (MDNS_ADDR, MDNS_PORT))
except OSError:
# This sendto function sometimes returns an OSError with EBADF
# as a payload. To avoid a failure here, reiinitialize the socket
# and try again once.
self._close_socket()
self._init_socket()
self.socket.sendto(payload, (MDNS_ADDR, MDNS_PORT))
def _init_socket_if_not_done(self) -> None:
if self.socket is None:
self._init_socket()
async def getaddrinfo(
self,
host: "Union[str, bytes, bytearray]",
port: "Union[str, int, None]",
family: int = 0,
type: int = 0,
proto: int = 0,
flags: int = 0,
) -> "List[Tuple[int, int, int, str, Tuple[str, int]]]":
hostcheck = host
while hostcheck.endswith("."):
hostcheck = hostcheck[:-1]
if hostcheck.endswith(LOCAL_MDNS_SUFFIX) and family in (0, socket.AF_INET):
host, resolved_ip = await self.mdns_getaddr(host)
return [(socket.AF_INET, type or socket.SOCK_STREAM, proto, host, (resolved_ip, port))]
else:
self.dprint("Resolving dns request host {} and port {}".format(host, port))
return socket.getaddrinfo(host, port, family, type, proto, flags)
async def mdns_getaddr(self, host: str) -> Tuple[str, str]:
host = host.lower()
self.dprint("Resolving mdns request host {}".format(host))
response = self.scan_for_response(TYPE_A, host, self.mdns_timeout)
await self.send_question(DNSQuestion(host, TYPE_A, CLASS_IN))
record = await response
if record is None:
# The original socket implementation returns -202 on the ESP32 as an error code
raise OSError(-202)
return record.name, a_record_rdata_to_string(record.rdata)
async def scan_for_response(self, expected_type: int, name: str, timeout: float = 1.5) -> "Optional[DNSRecord]":
def matching_record(dns_response: DNSResponse) -> "Optional[DNSRecord]":
for record in dns_response.records:
if record.record_type == expected_type and record.name == name:
return record
result = {"data": None, "event": uasyncio.Event()}
async def scan_response(dns_response: DNSResponse) -> None:
record = matching_record(dns_response)
if record is None:
return None
result["data"] = record
result["event"].set()
loop = uasyncio.get_event_loop()
loop.create_task(set_after_timeout(result["event"], timeout))
async def is_match(dns_response: DNSResponse) -> bool:
return matching_record(dns_response) is not None
self.add_callback(scan_response, is_match, timeout)
await result["event"].wait()
return result["data"]
def dprint(self, message: str) -> None:
if self.debug:
print("MDNS: {}".format(message))