blob: a40001d89d7c8a4364cf6c4a10787e91055083f8 [file] [log] [blame]
// Copyright (c) 2019, 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 Dynamic Semantics
/// The new dynamic semantics are a superset of the original behavior. To avoid
/// redundancy and handle nested uses, the semantics are expressed in terms of a
/// separate procedure below:
/// . . .
/// Sets
/// 1. Create a fresh instance [collection] of a class that implements [Set<E>].
///
/// 2. For each [element] in the set literal:
/// i. Evaluate [element] using the procedure below.
///
/// 3. The result of the literal expression is [collection].
/// . . .
/// To evaluate a collection [element]:
/// This procedure handles elements in both list and map literals because the
/// only difference is how a base expression element or entry element is handled.
/// The control flow parts are the same so are unified here.
/// . . .
/// 3. Else, if [element] is an asynchronous [await for-in] element:
///
/// i. Evaluate the stream expression to a value [stream]. It is a dynamic
/// error if [stream] is not an instance of a class that implements
/// [Stream].
/// ii. Create a new [Future], [streamDone].
/// iii. Evaluate [await streamDone].
/// iv. Listen to [stream]. On each data event [event] the stream sends:
/// a. If the [for-in] element declares a variable, create a fresh
/// [variable] for it. Otherwise, use the existing [variable] it
/// refers to.
/// b. Bind [event] to [variable].
/// c. Evaluate the body element using this procedure in the scope of
/// [variable]. If this raises an exception, complete [streamDone]
/// with it as an error.
/// v. If the [for-in] element declares a variable, discard it.
/// vi. If [stream] raises an exception, complete [streamDone] with it as an
/// error. Otherwise, when all events in the stream are in the stream are
/// processed, complete [streamDone] with [null].
/// . . .
/// @description Checks that if [element] of set literal is a an asynchronous
/// [await for-in] element, this element is evaluated using the specified
/// procedure.
/// @author ngl@unipro.ru
import "dart:async";
import "../../Utils/expect.dart";
main() async {
var l1 = [1, 2];
var l2 = [3, 4, 5];
var set1exp = {1, 2, 4, 6};
var s1 = new Stream.fromIterable(l1);
var s2 = new Stream.fromIterable(l2);
var set1 = <int>{await for (var v in s1) v,
await for (var v in s2) if (v.isOdd) (v + 1)};
Expect.isTrue(set1 is Set<int>);
Expect.runtimeIsType<Set<int>>(set1);
Expect.setEquals(set1exp, set1);
}