blob: 6c23faa04b521d1dfb40690e1341abe58bbfd907 [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.
/// @assertion Execution of a statement s of the form yield e; proceeds as
/// follows:
/// First, the expression e is evaluated to an object o. If the enclosing
/// function m is marked async* and the stream u associated with m has been
/// paused, then execution of m is suspended until u is resumed or canceled.
/// Next, o is added to the iterable or stream associated with the immediately
/// enclosing function.
/// If the enclosing function m is marked async* and the stream u associated
/// with m has been canceled, then let c be the finally clause (17.11) of the
/// innermost enclosing try-finally statement, if any. If c is defined, let h be
/// the handler induced by c. If h is defined, control is transferred to h. If
/// h is undefined, the immediately enclosing function terminates.
/// The stream associated with an asynchronous generator could be canceled by
/// any code with a reference to that stream at any point where the generator
/// was passivated. Such a cancellation constitutes an irretrievable error for
/// the generator. At this point, the only plausible action for the generator
/// is to clean up after itself via its finally clauses.
/// Otherwise, if the enclosing function m is marked async* (9) then the
/// enclosing function may suspend.
///
/// @description Check that the immediately enclosing function terminates and
/// enclosing 'finally' handlers are executed in innermost-first order, if the
/// stream u associated with m has been canceled.
///
/// @issue #25539
/// @author a.semenov@unipro.ru
import 'dart:async';
import '../../../../Utils/expect.dart';
Stream<int> generator(List<int> log) async* {
bool canceled = false;
int i = 0;
int order = 0;
try {
while (true) {
try {
canceled = true;
try {
yield i++;
canceled = false;
} finally {
if (canceled) {
log.add(order++);
}
}
} finally {
if (canceled) {
log.add(order++);
}
}
}
} finally {
if (canceled) {
log.add(order++);
}
}
}
test() async {
List<int> log = [];
Stream<int> s = generator(log);
late StreamSubscription<int> ss;
ss = s.listen(
(int x) async {
if (x == 5) { // let generator to work some time
await ss.cancel();
Expect.listEquals([0, 1, 2], log);
asyncEnd();
}
}
);
}
main() {
asyncStart();
test();
}