blob: 088ca803a23227ab6d6966ab4ecce97539990bb7 [file] [log] [blame]
// Copyright (c) 2018, 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 Here are some examples of finite unfoldings, and the effect they
/// have on types of expressions:
///
/// class C<X extends C<X>> {
/// X next;
/// C(this.next);
/// }
///
/// class D extends C<D> {
/// D(D next): super(next);
/// }
///
/// main() {
/// D d = new D(new D(null));
/// C<dynamic> c0 = d;
/// C<C<dynamic>> c1 = d;
/// C<C<C<dynamic>>> c2 = d;
///
/// c0.next.unknown(42); // Statically OK, `c0.next` is `dynamic`.
/// c1.next.unknown(43); // Compile-time error.
/// c1.next.next.unknown(44); // Statically OK.
/// c2.next.next.unknown(45); // Compile-time error.
/// c2.next.next.next.unknown(46); // Statically OK.
///
/// // With type `D`, the static analysis is aware of the cyclic
/// // structure of the type, and every level of nesting is handled
/// // safely. But `D` may be less useful because there may be a
/// // similar type `D2`, and this code will only work with `D`.
/// d.next.next.next.next.next.next.next.unknown(46); // Compile-time error.
/// }
///
/// @description Checks that [c2.next.next.next.unknown(46)] line from the test
/// example is statically OK and returns [null] in runtime with nnbd enabled.
/// @author iarkh@unipro.ru
/// @issue #32903
import "../../Utils/expect.dart";
class C<X extends C<X>> {
X? next;
C(this.next);
}
class D extends C<D> {
D(D? next): super(next);
}
main() {
D d = new D(new D(null));
C<C<C<dynamic>>> c2 = d;
Expect.isNull(c2.next?.next?.next?.unknown(46));
}