Skip to content

Commit e3e4729

Browse files
[multicast_dns] MDnsClient::listen supports onError callback (#8888)
MDnsClient::Listen now supports an optional onError callback function, called in case of a stream error. If omitted any errors on the stream are considered unhandled, and will be passed to the current [Zone]'s error handler. By default unhandled async errors are treated as if they were uncaught top-level errors. This fixes an unhandled exception occuring (tested on Android), when the network is disconnected. Issue: flutter/flutter#165482
1 parent a1c3689 commit e3e4729

File tree

4 files changed

+66
-2
lines changed

4 files changed

+66
-2
lines changed

packages/multicast_dns/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
## 0.3.2+9
2+
3+
* Optional error callback for `MDnsClient::start` to prevent uncatched exceptions.
4+
15
## 0.3.2+8
26

37
* Fixes stack overflows ocurring during the parsing of domain names in MDNS messages.

packages/multicast_dns/lib/multicast_dns.dart

+11-1
Original file line numberDiff line numberDiff line change
@@ -84,13 +84,20 @@ class MDnsClient {
8484
/// for the mDNS query. If not provided, defaults to either `224.0.0.251` or
8585
/// or `FF02::FB`.
8686
///
87+
/// The [onError] function allows to provide a callback function, called in
88+
/// case of a stream error. If omitted any errors on the stream are considered
89+
/// unhandled, and will be passed to the current [Zone]'s error handler. By
90+
/// default unhandled async errors are treated as if they were uncaught top-level
91+
/// errors.
92+
///
8793
/// Subsequent calls to this method are ignored while the mDNS client is in
8894
/// started state.
8995
Future<void> start({
9096
InternetAddress? listenAddress,
9197
NetworkInterfacesFactory? interfacesFactory,
9298
int mDnsPort = mDnsPort,
9399
InternetAddress? mDnsAddress,
100+
Function? onError,
94101
}) async {
95102
listenAddress ??= InternetAddress.anyIPv4;
96103
interfacesFactory ??= allInterfacesFactory;
@@ -152,7 +159,10 @@ class MDnsClient {
152159
// Join multicast on this interface.
153160
incoming.joinMulticast(_mDnsAddress!, interface);
154161
}
155-
incoming.listen((RawSocketEvent event) => _handleIncoming(event, incoming));
162+
incoming.listen(
163+
(RawSocketEvent event) => _handleIncoming(event, incoming),
164+
onError: onError,
165+
);
156166
_started = true;
157167
_starting = false;
158168
}

packages/multicast_dns/pubspec.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ name: multicast_dns
22
description: Dart package for performing mDNS queries (e.g. Bonjour, Avahi).
33
repository: https://github.com/flutter/packages/tree/main/packages/multicast_dns
44
issue_tracker: https://github.com/flutter/flutter/issues?q=is%3Aissue+is%3Aopen+label%3A%22p%3A+multicast_dns%22
5-
version: 0.3.2+8
5+
version: 0.3.2+9
66

77
environment:
88
sdk: ^3.4.0

packages/multicast_dns/test/client_test.dart

+50
Original file line numberDiff line numberDiff line change
@@ -158,6 +158,35 @@ void main() {
158158
});
159159
}
160160
});
161+
162+
test('Calls onError callback in case of socket error', () async {
163+
final FakeRawDatagramSocketThatSendsError datagramSocket =
164+
FakeRawDatagramSocketThatSendsError();
165+
final MDnsClient client = MDnsClient(
166+
rawDatagramSocketFactory: (
167+
dynamic host,
168+
int port, {
169+
bool reuseAddress = true,
170+
bool reusePort = true,
171+
int ttl = 1,
172+
}) async {
173+
return datagramSocket;
174+
},
175+
);
176+
177+
Completer<void> onErrorCalledCompleter = Completer<void>();
178+
await client.start(
179+
mDnsPort: 1234,
180+
interfacesFactory:
181+
(InternetAddressType type) async => <NetworkInterface>[],
182+
onError: (e) {
183+
expect(e, 'Error');
184+
onErrorCalledCompleter.complete();
185+
},
186+
);
187+
188+
await onErrorCalledCompleter.future.timeout(Duration(seconds: 5));
189+
});
161190
}
162191

163192
class FakeRawDatagramSocket extends Fake implements RawDatagramSocket {
@@ -196,6 +225,27 @@ class FakeRawDatagramSocket extends Fake implements RawDatagramSocket {
196225
}
197226
}
198227

228+
class FakeRawDatagramSocketThatSendsError extends Fake
229+
implements RawDatagramSocket {
230+
@override
231+
InternetAddress address = InternetAddress.anyIPv4;
232+
233+
@override
234+
StreamSubscription<RawSocketEvent> listen(
235+
void Function(RawSocketEvent event)? onData, {
236+
Function? onError,
237+
void Function()? onDone,
238+
bool? cancelOnError,
239+
}) {
240+
return Stream<RawSocketEvent>.error('Error').listen(
241+
onData,
242+
onError: onError,
243+
cancelOnError: cancelOnError,
244+
onDone: onDone,
245+
);
246+
}
247+
}
248+
199249
class FakeNetworkInterface implements NetworkInterface {
200250
FakeNetworkInterface(this._name, this._addresses, this._index);
201251

0 commit comments

Comments
 (0)