blob: 1ed2f799941c59940ea7a1854924b57e711bae98 [file] [log] [blame]
// Copyright (c) 2017, 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:test/test.dart';
import 'package:stream_transform/stream_transform.dart';
void main() {
var streamTypes = {
'single subscription': () => new StreamController(),
'broadcast': () => new StreamController.broadcast()
};
for (var streamType in streamTypes.keys) {
group('takeUntil on Stream type [$streamType]', () {
StreamController values;
List emittedValues;
bool valuesCanceled;
bool isDone;
List errors;
Stream transformed;
StreamSubscription subscription;
Completer closeTrigger;
setUp(() {
valuesCanceled = false;
values = streamTypes[streamType]()
..onCancel = () {
valuesCanceled = true;
};
emittedValues = [];
errors = [];
isDone = false;
closeTrigger = new Completer();
transformed = values.stream.transform(takeUntil(closeTrigger.future));
subscription = transformed
.listen(emittedValues.add, onError: errors.add, onDone: () {
isDone = true;
});
});
test('forwards cancellation', () async {
await subscription.cancel();
expect(valuesCanceled, true);
});
test('lets values through before trigger', () async {
values.add(1);
values.add(2);
await new Future(() {});
expect(emittedValues, [1, 2]);
});
test('forwards errors', () async {
values.addError('error');
await new Future(() {});
expect(errors, ['error']);
});
test('sends done if original strem ends', () async {
await values.close();
expect(isDone, true);
});
test('sends done when trigger fires', () async {
closeTrigger.complete();
await new Future(() {});
expect(isDone, true);
});
test('cancels value subscription when trigger fires', () async {
closeTrigger.complete();
await new Future(() {});
expect(valuesCanceled, true);
});
if (streamType == 'broadcast') {
test('multiple listeners all get values', () async {
var otherValues = [];
transformed.listen(otherValues.add);
values.add(1);
values.add(2);
await new Future(() {});
expect(emittedValues, [1, 2]);
expect(otherValues, [1, 2]);
});
test('multiple listeners get done when trigger fires', () async {
var otherDone = false;
transformed.listen(null, onDone: () => otherDone = true);
closeTrigger.complete();
await new Future(() {});
expect(otherDone, true);
expect(isDone, true);
});
test('multiple listeners get done when values end', () async {
var otherDone = false;
transformed.listen(null, onDone: () => otherDone = true);
await values.close();
expect(otherDone, true);
expect(isDone, true);
});
test('can cancel and relisten before trigger fires', () async {
values.add(1);
await new Future(() {});
await subscription.cancel();
values.add(2);
await new Future(() {});
subscription = transformed.listen(emittedValues.add);
values.add(3);
await new Future(() {});
expect(emittedValues, [1, 3]);
});
}
});
}
}