blob: 9e8d338cefeb68263d48463d3f94ee2f7606876b [file] [log] [blame]
// 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.
import "dart:collection";
import "dart:typed_data";
import "package:expect/expect.dart";
void main() {
// Growable lists. Initial length 0.
testConcurrentModification(<int>[].toList());
testConcurrentModification(new List.filled(0, 0, growable: true));
testConcurrentModification([]);
testConcurrentModification(new List.from(const []));
testConcurrentModification(new MyList([]));
testConcurrentModification(new MyList<int>([]).toList());
testConcurrentModification(new Uint8List(0).toList());
testConcurrentModification(new Int8List(0).toList());
testConcurrentModification(new Uint16List(0).toList());
testConcurrentModification(new Int16List(0).toList());
testConcurrentModification(new Uint32List(0).toList());
testConcurrentModification(new Int32List(0).toList());
testConcurrentAddSelf([]);
testConcurrentAddSelf([1, 2, 3]);
}
void testConcurrentModification(List<int> list) {
// add, removeLast.
list.clear();
list.addAll([1, 2, 3, 2, 7, 9, 9, 7, 2, 3, 2, 1]);
// Operations that change the length cause ConcurrentModificationError.
void testModification(action()) {
testIterator(int when) {
list.clear();
list.addAll([0, 1, 2, 3]);
Expect.throws(() {
for (var element in list) {
if (element == when) action();
}
}, (e) => e is ConcurrentModificationError);
}
testForEach(int when) {
list.clear();
list.addAll([0, 1, 2, 3]);
Expect.throws(() {
list.forEach((var element) {
if (element == when) action();
});
}, (e) => e is ConcurrentModificationError);
}
// Test the change at different points of the iteration.
testIterator(0);
testIterator(1);
testIterator(3);
testForEach(0);
testForEach(1);
testForEach(3);
}
testModification(() => list.add(5));
testModification(() => list.addAll([5, 6]));
testModification(() => list.removeLast());
for (int i = 0; i < 4; i++) {
testModification(() => list.remove(i));
testModification(() => list.removeAt(i));
testModification(() => list.removeWhere((x) => x == i));
testModification(() => list.retainWhere((x) => x != i));
testModification(() => list.insert(i, 5));
testModification(() => list.insertAll(i, [5, 6]));
testModification(() => list.removeRange(i, i + 1));
testModification(() => list.replaceRange(i, i + 1, [5, 6]));
}
}
testConcurrentAddSelf(List list) {
Expect.throws(() {
list.addAll(list);
}, (e) => e is ConcurrentModificationError, "testConcurrentAddSelf($list)");
}
class MyList<E> extends ListBase<E> {
// TODO(42496): Use a nullable list because insert() is implemented in terms
// of length=. Change this back to `E` and remove the `as E` below when that
// issue is fixed.
List<E?> _source;
MyList(this._source);
int get length => _source.length;
void set length(int length) {
_source.length = length;
}
void add(E element) {
_source.add(element);
}
E operator [](int index) => _source[index] as E;
void operator []=(int index, E value) {
_source[index] = value;
}
}