Skip to content

Commit

Permalink
Merge branch 'master' into contains-all-matching
Browse files Browse the repository at this point in the history
  • Loading branch information
natebosch authored Oct 2, 2024
2 parents 643884b + d872cf8 commit c394c9b
Show file tree
Hide file tree
Showing 4 changed files with 178 additions and 74 deletions.
5 changes: 4 additions & 1 deletion pkgs/checks/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
## 0.3.1-wip

- Update min SDK constraint to 3.4.0.
- Directly compare keys across actual and expected `Map` instances when
checking deep collection equality and all the keys can be directly compared
for equality. This maintains the path into a nested collection for typical
cases of checking for equality against a purely value collection.
- Always wrap Condition descriptions in angle brackets.
- Add `containsMatchingInOrder` and `containsEqualInOrder` to replace the
combined functionality in `containsInOrder`.
Expand Down
184 changes: 122 additions & 62 deletions pkgs/checks/lib/src/collection_equality.dart
Original file line number Diff line number Diff line change
Expand Up @@ -12,10 +12,10 @@ import '../context.dart';
/// {@template deep_collection_equals}
/// Elements, keys, or values within [expected] which are a collections are
/// deeply compared for equality with a collection in the same position within
/// [actual]. Elements which are collection types are note compared with the
/// [actual]. Elements which are collection types are not compared with the
/// native identity based equality or custom equality operator overrides.
///
/// Elements, keys, or values within [expected] which are `Condition` callbacks
/// Elements, keys, or values within [expected] which are [Condition] callbacks
/// are run against the value in the same position within [actual].
/// Condition callbacks must take a `Subject<Object?>` or `Subject<dynamic>` and
/// may not use a more specific generic.
Expand All @@ -24,14 +24,25 @@ import '../context.dart';
/// Note also that the argument type `Subject<Object?>` cannot be inferred and
/// must be explicit in the function definition.
///
/// Elements, keys, or values within [expected] which are any other type are
/// compared using `operator ==` equality.
/// Elements and values within [expected] which are any other type are compared
/// using `operator ==` equality.
///
/// Comparing sets or maps will have a runtime which is polynomial on the the
/// size of those collections. Does not use [Set.contains] or [Map.containsKey],
/// there will not be runtime benefits from hashing. Custom collection behavior
/// is ignored. For example, it is not possible to distinguish between a `Set`
/// and a `Set.identity`.
/// Deep equality checks for [Map] instances depend on the structure of the
/// expected map.
/// If all keys of the expectation are non-collection and non-condition values
/// the keys are compared through the map instances with [Map.containsKey].
///
/// If any key in the expectation is a `Condition` or a collection type the map
/// will be compared as a [Set] of entries where both the key and value must
/// match.
///
/// Comparing sets or maps with complex keys has a runtime which is polynomial
/// on the the size of those collections.
/// These comparisons do not use [Set.contains] or [Map.containsKey],
/// and there will not be runtime benefits from hashing.
/// Custom collection behavior of `contains` is ignored.
/// For example, it is not possible to distinguish between a `Set` and a
/// `Set.identity`.
///
/// Collections may be nested to a maximum depth of 1000. Recursive collections
/// are not allowed.
Expand Down Expand Up @@ -70,7 +81,7 @@ Iterable<String>? _deepCollectionEquals(
} else {
currentExpected as Map;
rejectionWhich = _findMapDifference(
currentActual, currentExpected, path, currentDepth);
currentActual, currentExpected, path, queue, currentDepth);
}
if (rejectionWhich != null) return rejectionWhich;
}
Expand Down Expand Up @@ -100,32 +111,38 @@ List<String>? _findIterableDifference(Object? actual,
'expected an iterable with at least ${index + 1} element(s)'
];
}
var actualValue = actualIterator.current;
var expectedValue = expectedIterator.current;
if (expectedValue is Iterable || expectedValue is Map) {
if (depth + 1 > _maxDepth) throw _ExceededDepthError();
queue.addLast(
_Search(path.append(index), actualValue, expectedValue, depth + 1));
} else if (expectedValue is Condition) {
final failure = softCheck(actualValue, expectedValue);
if (failure != null) {
final which = failure.rejection.which;
return [
'has an element ${path.append(index)}that:',
...indent(failure.detail.actual.skip(1)),
...indent(prefixFirst('Actual: ', failure.rejection.actual),
failure.detail.depth + 1),
if (which != null)
...indent(prefixFirst('which ', which), failure.detail.depth + 1)
];
}
} else {
if (actualValue != expectedValue) {
return [
...prefixFirst('${path.append(index)}is ', literal(actualValue)),
...prefixFirst('which does not equal ', literal(expectedValue))
];
}
final difference = _compareValue(actualIterator.current,
expectedIterator.current, path, index, queue, depth);
if (difference != null) return difference;
}
return null;
}

List<String>? _compareValue(Object? actualValue, Object? expectedValue,
_Path path, Object? pathAppend, Queue<_Search> queue, int depth) {
if (expectedValue is Iterable || expectedValue is Map) {
if (depth + 1 > _maxDepth) throw _ExceededDepthError();
queue.addLast(_Search(
path.append(pathAppend), actualValue, expectedValue, depth + 1));
} else if (expectedValue is Condition) {
final failure = softCheck(actualValue, expectedValue);
if (failure != null) {
final which = failure.rejection.which;
return [
'has an element ${path.append(pathAppend)}that:',
...indent(failure.detail.actual.skip(1)),
...indent(prefixFirst('Actual: ', failure.rejection.actual),
failure.detail.depth + 1),
if (which != null)
...indent(prefixFirst('which ', which), failure.detail.depth + 1)
];
}
} else {
if (actualValue != expectedValue) {
return [
...prefixFirst('${path.append(pathAppend)}is ', literal(actualValue)),
...prefixFirst('which does not equal ', literal(expectedValue))
];
}
}
return null;
Expand Down Expand Up @@ -165,39 +182,82 @@ Iterable<String>? _findSetDifference(
}

Iterable<String>? _findMapDifference(
Object? actual, Map<Object?, Object?> expected, _Path path, int depth) {
Object? actual,
Map<Object?, Object?> expected,
_Path path,
Queue<_Search> queue,
int depth) {
if (actual is! Map) {
return ['${path}is not a Map'];
}
Iterable<String> describeEntry(MapEntry<Object?, Object?> entry) {
final key = literal(entry.key);
final value = literal(entry.value);
return [
...key.take(key.length - 1),
'${key.last}: ${value.first}',
...value.skip(1)
];
if (expected.keys
.any((key) => key is Condition || key is Iterable || key is Map)) {
return _findAmbiguousMapDifference(actual, expected, path, depth);
} else {
return _findUnambiguousMapDifference(actual, expected, path, queue, depth);
}
}

return unorderedCompare(
actual.entries,
expected.entries,
(actual, expected) =>
_elementMatches(actual.key, expected.key, depth) &&
_elementMatches(actual.value, expected.value, depth),
(expectedEntry, _, count) => [
...prefixFirst(
'${path}has no entry to match ', describeEntry(expectedEntry)),
if (count > 1) 'or ${count - 1} other entries',
],
(actualEntry, _, count) => [
...prefixFirst(
'${path}has unexpected entry ', describeEntry(actualEntry)),
if (count > 1) 'and ${count - 1} other unexpected entries',
],
);
Iterable<String> _describeEntry(MapEntry<Object?, Object?> entry) {
final key = literal(entry.key);
final value = literal(entry.value);
return [
...key.take(key.length - 1),
'${key.last}: ${value.first}',
...value.skip(1)
];
}

/// Returns a description of a difference found between [actual] and [expected]
/// when [expected] has only direct key values and there is a 1:1 mapping
/// between an expected value and a checked value in the map.
Iterable<String>? _findUnambiguousMapDifference(
Map<Object?, Object?> actual,
Map<Object?, Object?> expected,
_Path path,
Queue<_Search> queue,
int depth) {
for (final entry in expected.entries) {
assert(entry.key is! Condition);
assert(entry.key is! Iterable);
assert(entry.key is! Map);
if (!actual.containsKey(entry.key)) {
return prefixFirst(
'${path}has no key matching expected entry ', _describeEntry(entry));
}
final difference = _compareValue(
actual[entry.key], entry.value, path, entry.key, queue, depth);
if (difference != null) return difference;
}
for (final entry in actual.entries) {
if (!expected.containsKey(entry.key)) {
return prefixFirst(
'${path}has an unexpected key for entry ', _describeEntry(entry));
}
}
return null;
}

Iterable<String>? _findAmbiguousMapDifference(Map<Object?, Object?> actual,
Map<Object?, Object?> expected, _Path path, int depth) =>
unorderedCompare(
actual.entries,
expected.entries,
(actual, expected) =>
_elementMatches(actual.key, expected.key, depth) &&
_elementMatches(actual.value, expected.value, depth),
(expectedEntry, _, count) => [
...prefixFirst(
'${path}has no entry to match ', _describeEntry(expectedEntry)),
if (count > 1) 'or ${count - 1} other entries',
],
(actualEntry, _, count) => [
...prefixFirst(
'${path}has unexpected entry ', _describeEntry(actualEntry)),
if (count > 1) 'and ${count - 1} other unexpected entries',
],
);

class _Path {
final _Path? parent;
final Object? index;
Expand Down
21 changes: 18 additions & 3 deletions pkgs/checks/test/extensions/collection_equality_test.dart
Original file line number Diff line number Diff line change
Expand Up @@ -129,9 +129,9 @@ void main() {
}, {
'a': (Subject<dynamic> it) => it.isA<String>().startsWith('a')
})).isNotNull().deepEquals([
"has no entry to match 'a': <A value that:",
' is a String',
" starts with 'a'>",
"has an element at ['a'] that:",
" Actual: 'b'",
" which does not start with 'a'",
]);
});

Expand All @@ -147,6 +147,21 @@ void main() {
]);
});

test('maintains paths through maps when the keys are all values', () {
check(deepCollectionEquals({
'a': [
{'b': 'c'}
]
}, {
'a': [
{'b': 'd'}
]
})).isNotNull().deepEquals([
"at ['a'][<0>]['b'] is 'c'",
"which does not equal 'd'",
]);
});

test('reports recursive lists', () {
var l = <Object>[];
l.add(l);
Expand Down
42 changes: 34 additions & 8 deletions pkgs/test/lib/src/runner/browser/static/run_wasm_chrome.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,15 +7,41 @@
// affect chrome.
(async () => {
// Fetch and compile Wasm binary.
let data = document.getElementById('WasmBootstrapInfo').dataset;
let modulePromise = WebAssembly.compileStreaming(fetch(data.wasmurl));
let data = document.getElementById("WasmBootstrapInfo").dataset;

// Instantiate the Dart module, importing from the global scope.
let dart2wasm = await import('./' + data.jsruntimeurl);
let dartInstance = await dart2wasm.instantiate(modulePromise, {});
let dart2wasmJsRuntime = await import("./" + data.jsruntimeurl);

// Call `main`. If tasks are placed into the event loop (by scheduling tasks
// explicitly or awaiting Futures), these will automatically keep the script
// alive even after `main` returns.
await dart2wasm.invoke(dartInstance);
// Support three versions of dart2wasm:
//
// (1) Versions before 3.6.0-167.0.dev require the user to compile using the
// browser's `WebAssembly` API, the compiled module needs to be instantiated
// using the JS runtime.
//
// (2) Versions starting with 3.6.0-167.0.dev added helpers for compiling and
// instantiating.
//
// (3) Versions starting with 3.6.0-212.0.dev made compilation functions
// return a new type that comes with instantiation and invoke methods.

if (dart2wasmJsRuntime.compileStreaming !== undefined) {
// Version (2) or (3).
let compiledModule = await dart2wasmJsRuntime.compileStreaming(
fetch(data.wasmurl),
);
if (compiledModule.instantiate !== undefined) {
// Version (3).
let instantiatedModule = await compiledModule.instantiate();
instantiatedModule.invokeMain();
} else {
// Version (2).
let dartInstance = await dart2wasmJsRuntime.instantiate(compiledModule, {});
await dart2wasmJsRuntime.invoke(dartInstance);
}
} else {
// Version (1).
let modulePromise = WebAssembly.compileStreaming(fetch(data.wasmurl));
let dartInstance = await dart2wasmJsRuntime.instantiate(modulePromise, {});
await dart2wasmJsRuntime.invoke(dartInstance);
}
})();

0 comments on commit c394c9b

Please sign in to comment.