blob: df4157a623197ed41a633c7a66c7aec908f17ca4 [file] [log] [blame]
// Copyright (c) 2017, 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:async';
import 'package:matcher/matcher.dart';
import '../backend/invoker.dart';
import 'expect.dart';
/// A matcher that does asynchronous computation.
///
/// Rather than implementing [matches], subclasses implement [matchAsync].
/// [AsyncMatcher.matches] ensures that the test doesn't complete until the
/// returned future completes, and [expect] returns a future that completes when
/// the returned future completes so that tests can wait for it.
abstract class AsyncMatcher extends Matcher {
const AsyncMatcher();
/// Returns `null` if this matches [item], or a [String] description of the
/// failure if it doesn't match.
///
/// This can return a [Future] or a synchronous value. If it returns a
/// [Future], neither [expect] nor the test will complete until that [Future]
/// completes.
///
/// If this returns a [String] synchronously, [expect] will synchronously
/// throw a [TestFailure] and [matches] will synchronusly return `false`.
/*FutureOr<String>*/ matchAsync(item);
bool matches(item, Map matchState) {
var result = matchAsync(item);
expect(
result,
anyOf([
equals(null),
new isInstanceOf<Future>(),
new isInstanceOf<String>()
]),
reason: "matchAsync() may only return a String, a Future, or null.");
if (result is Future) {
Invoker.current.addOutstandingCallback();
result.then((realResult) {
if (realResult != null) fail(formatFailure(this, item, realResult));
Invoker.current.removeOutstandingCallback();
});
} else if (result is String) {
matchState[this] = result;
return false;
}
return true;
}
Description describeMismatch(
item, Description description, Map matchState, bool verbose) =>
new StringDescription(matchState[this]);
}