blob: 849e3049193eaa5febfa5a97f43a032804391fd6 [file] [log] [blame]
// Copyright (c) 2016, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'dart:async';
import 'package:stream_channel/stream_channel.dart';
import 'package:test/test.dart';
void main() {
late StreamController streamController;
late StreamController sinkController;
late StreamChannel channel;
setUp(() {
streamController = StreamController();
sinkController = StreamController();
channel = StreamChannel.withGuarantees(
streamController.stream, sinkController.sink);
});
group('with a broadcast stream', () {
setUp(() {
streamController = StreamController.broadcast();
channel = StreamChannel.withGuarantees(
streamController.stream, sinkController.sink);
});
test('buffers events', () async {
streamController.add(1);
streamController.add(2);
streamController.add(3);
await pumpEventQueue();
expect(channel.stream.toList(), completion(equals([1, 2, 3])));
unawaited(streamController.close());
});
test('only allows a single subscription', () {
channel.stream.listen(null);
expect(() => channel.stream.listen(null), throwsStateError);
});
});
test(
'closing the event sink causes the stream to close before it emits any '
'more events', () {
streamController.add(1);
streamController.add(2);
streamController.add(3);
expect(
channel.stream
.listen(expectAsync1((event) {
if (event == 2) channel.sink.close();
}, count: 2))
.asFuture(),
completes);
});
test('after the stream closes, the sink ignores events', () async {
unawaited(streamController.close());
// Wait for the done event to be delivered.
await channel.stream.toList();
channel.sink.add(1);
channel.sink.add(2);
channel.sink.add(3);
unawaited(channel.sink.close());
// None of our channel.sink additions should make it to the other endpoint.
sinkController.stream.listen(expectAsync1((_) {}, count: 0),
onDone: expectAsync0(() {}, count: 0));
await pumpEventQueue();
});
test("canceling the stream's subscription has no effect on the sink",
() async {
unawaited(channel.stream.listen(null).cancel());
await pumpEventQueue();
channel.sink.add(1);
channel.sink.add(2);
channel.sink.add(3);
unawaited(channel.sink.close());
expect(sinkController.stream.toList(), completion(equals([1, 2, 3])));
});
test("canceling the stream's subscription doesn't stop a done event",
() async {
unawaited(channel.stream.listen(null).cancel());
await pumpEventQueue();
unawaited(streamController.close());
await pumpEventQueue();
channel.sink.add(1);
channel.sink.add(2);
channel.sink.add(3);
unawaited(channel.sink.close());
// The sink should be ignoring events because the stream closed.
sinkController.stream.listen(expectAsync1((_) {}, count: 0),
onDone: expectAsync0(() {}, count: 0));
await pumpEventQueue();
});
test('forwards errors to the other endpoint', () {
channel.sink.addError('error');
expect(sinkController.stream.first, throwsA('error'));
});
test('Sink.done completes once the stream is done', () {
channel.stream.listen(null);
expect(channel.sink.done, completes);
streamController.close();
});
test("events can't be added to an explicitly-closed sink", () {
sinkController.stream.listen(null); // Work around sdk#19095.
expect(channel.sink.close(), completes);
expect(() => channel.sink.add(1), throwsStateError);
expect(() => channel.sink.addError('oh no'), throwsStateError);
expect(() => channel.sink.addStream(Stream.fromIterable([])),
throwsStateError);
});
test("events can't be added while a stream is being added", () {
var controller = StreamController();
channel.sink.addStream(controller.stream);
expect(() => channel.sink.add(1), throwsStateError);
expect(() => channel.sink.addError('oh no'), throwsStateError);
expect(() => channel.sink.addStream(Stream.fromIterable([])),
throwsStateError);
expect(() => channel.sink.close(), throwsStateError);
controller.close();
});
group('with allowSinkErrors: false', () {
setUp(() {
streamController = StreamController();
sinkController = StreamController();
channel = StreamChannel.withGuarantees(
streamController.stream, sinkController.sink,
allowSinkErrors: false);
});
test('forwards errors to Sink.done but not the stream', () {
channel.sink.addError('oh no');
expect(channel.sink.done, throwsA('oh no'));
sinkController.stream
.listen(null, onError: expectAsync1((dynamic _) {}, count: 0));
});
test('adding an error causes the stream to emit a done event', () {
expect(channel.sink.done, throwsA('oh no'));
streamController.add(1);
streamController.add(2);
streamController.add(3);
expect(
channel.stream
.listen(expectAsync1((event) {
if (event == 2) channel.sink.addError('oh no');
}, count: 2))
.asFuture(),
completes);
});
test('adding an error closes the inner sink', () {
channel.sink.addError('oh no');
expect(channel.sink.done, throwsA('oh no'));
expect(sinkController.stream.toList(), completion(isEmpty));
});
test(
'adding an error via via addStream causes the stream to emit a done '
'event', () async {
var canceled = false;
var controller = StreamController(onCancel: () {
canceled = true;
});
// This future shouldn't get the error, because it's sent to [Sink.done].
expect(channel.sink.addStream(controller.stream), completes);
controller.addError('oh no');
expect(channel.sink.done, throwsA('oh no'));
await pumpEventQueue();
expect(canceled, isTrue);
// Even though the sink is closed, this shouldn't throw an error because
// the user didn't explicitly close it.
channel.sink.add(1);
});
});
}