| Index: tests/corelib/growable_list_test.dart
|
| diff --git a/tests/corelib/growable_list_test.dart b/tests/corelib/growable_list_test.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..6f1b9a0e86a25c7c6c9aaedefd42a1a9ca0ecabf
|
| --- /dev/null
|
| +++ b/tests/corelib/growable_list_test.dart
|
| @@ -0,0 +1,132 @@
|
| +// Copyright (c) 2011, 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.
|
| +
|
| +// Sanity check on the growing behavior of a growable list.
|
| +
|
| +import "package:expect/expect.dart";
|
| +import "dart:collection" show IterableBase;
|
| +
|
| +// Iterable generating numbers in range [0..count).
|
| +// May perform callback at some point underways.
|
| +class TestIterableBase<E> extends IterableBase<E> {
|
| + final int length;
|
| + final int count;
|
| + // call [callback] if generating callbackIndex.
|
| + final int callbackIndex;
|
| + final Function callback;
|
| + TestIterableBase(this.length, this.count,
|
| + this.callbackIndex, this.callback);
|
| + Iterator<E> get iterator => new CallbackIterator(this);
|
| +}
|
| +
|
| +class TestIterable<E> extends TestIterableBase<E> {
|
| + TestIterable(count, [callbackIndex = -1, callback])
|
| + : super(-1, count, callbackIndex, callback);
|
| + int get length => throw "SHOULD NOT BE CALLED";
|
| +}
|
| +
|
| +// Implement Set for private EfficientLength interface.
|
| +@proxy // Avoid warnings because we don't actually implement Set.
|
| +class EfficientTestIterable<E> extends TestIterableBase<E>
|
| + implements Set<E> {
|
| + EfficientTestIterable(length, count, [callbackIndex = -1, callback])
|
| + : super(length, count, callbackIndex, callback);
|
| +}
|
| +
|
| +class CallbackIterator<E> implements Iterator<E> {
|
| + TestIterableBase _iterable;
|
| + int _current = null;
|
| + int _nextIndex = 0;
|
| + CallbackIterator(this._iterable);
|
| + bool moveNext() {
|
| + if (_nextIndex >= _iterable.count) {
|
| + _current = null;
|
| + return false;
|
| + }
|
| + _current = _nextIndex;
|
| + _nextIndex++;
|
| + if (_current == _iterable.callbackIndex) {
|
| + _iterable.callback();
|
| + }
|
| + return true;
|
| + }
|
| + int get current => _current;
|
| +}
|
| +
|
| +
|
| +void main() {
|
| + // Without EfficientLength interface
|
| + {
|
| + // Change length of list after 20 additions.
|
| + var l = [];
|
| + var ci = new TestIterable<E>(257, 200, () {
|
| + l.add("X");
|
| + });
|
| + Expect.throws(() {
|
| + l.addAll(ci);
|
| + }, (e) => e is ConcurrentModificationError);
|
| + }
|
| +
|
| + {
|
| + // Change length of list after 20 additions.
|
| + var l = [];
|
| + var ci = new TestIterable<E>(257, 200, () {
|
| + l.length = 0;
|
| + });
|
| + Expect.throws(() {
|
| + l.addAll(ci);
|
| + }, (e) => e is ConcurrentModificationError);
|
| + }
|
| +
|
| + // With EfficientLength interface (uses length).
|
| + {
|
| + // Change length of list after 20 additions.
|
| + var l = [];
|
| + var ci = new EfficientTestIterable<E>(257, 257, 20, () {
|
| + l.add("X");
|
| + });
|
| + Expect.throws(() {
|
| + l.addAll(ci);
|
| + }, (e) => e is ConcurrentModificationError);
|
| + }
|
| +
|
| + {
|
| + var l = [];
|
| + var ci = new EfficientTestIterable<E>(257, 257, 20, () {
|
| + l.length = 0;
|
| + });
|
| + Expect.throws(() {
|
| + l.addAll(ci);
|
| + }, (e) => e is ConcurrentModificationError);
|
| + }
|
| +
|
| + {
|
| + // Length 50, only 25 elements.
|
| + var l = [];
|
| + var ci = new EfficientTestIterable<E>(500, 250);
|
| + l.addAll(ci);
|
| + Expect.listEquals(new List.generate(250, (x)=>x), l);
|
| + }
|
| +
|
| + {
|
| + // Length 25, but 50 elements.
|
| + var l = [];
|
| + var ci = new EfficientTestIterable<E>(250, 500);
|
| + l.addAll(ci);
|
| + Expect.listEquals(new List.generate(500, (x)=>x), l);
|
| + }
|
| +
|
| + {
|
| + // Adding to yourself.
|
| + var l = [1];
|
| + Expect.throws(() { l.addAll(l); }, (e) => e is ConcurrentModificationError);
|
| + }
|
| +
|
| + {
|
| + // Adding to yourself.
|
| + var l = [1, 2, 3];
|
| + Expect.throws(() { l.addAll(l); }, (e) => e is ConcurrentModificationError);
|
| + }
|
| +}
|
| +
|
|
|