blob: 88325149f1894d3fd3d8552d83a5403820675cbb [file] [log] [blame] [edit]
// Copyright (c) 2023, 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.
// Originally from package:expect in the Dart SDK.
/// This library contains an Expect class with static methods that can be used
/// for simple unit-tests.
library expect;
/// Expect is used for tests that do not want to make use of the
/// Dart unit test library - for example, the core language tests.
/// Third parties are discouraged from using this, and should use
/// the expect() function in the unit test library instead for
/// test assertions.
class Expect {
/// Return a slice of a string.
///
/// The slice will contain at least the substring from [start] to the lower of
/// [end] and `start + length`.
/// If the result is no more than `length - 10` characters long,
/// context may be added by extending the range of the slice, by decreasing
/// [start] and increasing [end], up to at most length characters.
/// If the start or end of the slice are not matching the start or end of
/// the string, ellipses are added before or after the slice.
/// Characters other than printable ASCII are escaped.
static String _truncateString(String string, int start, int end, int length) {
if (end - start > length) {
end = start + length;
} else if (end - start < length) {
int overflow = length - (end - start);
if (overflow > 10) overflow = 10;
// Add context.
start = start - ((overflow + 1) ~/ 2);
end = end + (overflow ~/ 2);
if (start < 0) start = 0;
if (end > string.length) end = string.length;
}
final StringBuffer buf = StringBuffer();
if (start > 0) buf.write('...');
_escapeSubstring(buf, string, 0, string.length);
if (end < string.length) buf.write('...');
return buf.toString();
}
/// Return the string with characters that are not printable ASCII characters
/// escaped as either "\xXX" codes or "\uXXXX" codes.
static String _escapeString(String string) {
final StringBuffer buf = StringBuffer();
_escapeSubstring(buf, string, 0, string.length);
return buf.toString();
}
static void _escapeSubstring(
StringBuffer buf,
String string,
int start,
int end,
) {
const hexDigits = '0123456789ABCDEF';
for (int i = start; i < end; i++) {
final int code = string.codeUnitAt(i);
if (0x20 <= code && code < 0x7F) {
if (code == 0x5C) {
buf.write(r'\\');
} else {
buf.writeCharCode(code);
}
} else if (code < 0x100) {
buf.write(r'\x');
buf.write(hexDigits[code >> 4]);
buf.write(hexDigits[code & 15]);
} else {
buf.write(r'\u{');
buf.write(code.toRadixString(16).toUpperCase());
buf.write('}');
}
}
}
/// Find the difference between two strings.
///
/// This finds the first point where two strings differ, and returns
/// a text describing the difference.
///
/// For small strings (length less than 20) nothing is done, and "" is
/// returned. Small strings can be compared visually, but for longer strings
/// only a slice containing the first difference will be shown.
static String _stringDifference(String expected, String actual) {
if (expected.length < 20 && actual.length < 20) return '';
for (int i = 0; i < expected.length && i < actual.length; i++) {
if (expected.codeUnitAt(i) != actual.codeUnitAt(i)) {
final int start = i;
i++;
while (i < expected.length && i < actual.length) {
if (expected.codeUnitAt(i) == actual.codeUnitAt(i)) break;
i++;
}
final int end = i;
final truncExpected = _truncateString(expected, start, end, 20);
final truncActual = _truncateString(actual, start, end, 20);
return 'at index $start: Expected <$truncExpected>, '
'Found: <$truncActual>';
}
}
return '';
}
/// Checks whether the expected and actual values are equal (using `==`).
static void equals(dynamic expected, dynamic actual, [String reason = '']) {
if (expected == actual) return;
final String msg = _getMessage(reason);
if (expected is String && actual is String) {
final String stringDifference = _stringDifference(expected, actual);
if (stringDifference.isNotEmpty) {
fail('Expect.equals($stringDifference$msg) fails.');
}
fail('Expect.equals(expected: <${_escapeString(expected)}>'
', actual: <${_escapeString(actual)}>$msg) fails.');
}
fail('Expect.equals(expected: <$expected>, actual: <$actual>$msg) fails.');
}
/// Checks whether the actual value is a bool and its value is true.
static void isTrue(dynamic actual, [String reason = '']) {
if (_identical(actual, true)) return;
final String msg = _getMessage(reason);
fail('Expect.isTrue($actual$msg) fails.');
}
/// Checks whether the actual value is a bool and its value is false.
static void isFalse(dynamic actual, [String reason = '']) {
if (_identical(actual, false)) return;
final String msg = _getMessage(reason);
fail('Expect.isFalse($actual$msg) fails.');
}
/// Checks whether [actual] is null.
static void isNull(dynamic actual, [String reason = '']) {
if (null == actual) return;
final String msg = _getMessage(reason);
fail('Expect.isNull(actual: <$actual>$msg) fails.');
}
/// Checks whether [actual] is not null.
static void isNotNull(dynamic actual, [String reason = '']) {
if (null != actual) return;
final String msg = _getMessage(reason);
fail('Expect.isNotNull(actual: <$actual>$msg) fails.');
}
static String _getMessage(String reason) =>
(reason.isEmpty) ? '' : ", '$reason'";
static Never fail(String message) {
throw ExpectException(message);
}
}
/// Exception thrown on a failed expectation check.
///
/// Always recognized by [Expect.throws] as an unexpected error.
class ExpectException {
/// Call this to provide a function that associates a test name with this
/// failure.
///
/// Used by async_helper/async_minitest.dart to inject logic to bind the
/// `group()` and `test()` name strings to a test failure.
static void setTestNameCallback(String Function() getName) {
_getTestName = getName;
}
static String Function() _getTestName = _kEmptyString;
final String message;
final String name;
ExpectException(this.message) : name = _getTestName();
@override
String toString() {
if (name != '') return 'In test "$name" $message';
return message;
}
/// Initial value for _getTestName.
static String _kEmptyString() => '';
}
/// Used in [Expect] because [Expect.identical] shadows the real [identical].
bool _identical(a, b) => identical(a, b);