blob: fb0fabacab4bf84e08bcceeb88b20f10fb7281eb [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2007 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.test;
18
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080019import junit.framework.Assert;
20
21import java.util.Arrays;
22import java.util.HashMap;
23import java.util.HashSet;
24import java.util.Map;
25import java.util.Set;
Karl Rosaen97eb32e2009-06-29 18:06:40 -070026import java.util.ArrayList;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080027import java.util.regex.MatchResult;
28import java.util.regex.Matcher;
29import java.util.regex.Pattern;
30
31/**
32 * Contains additional assertion methods not found in JUnit.
33 */
34public final class MoreAsserts {
35
36 private MoreAsserts() { }
37
38 /**
39 * Asserts that the class {@code expected} is assignable from the object
40 * {@code actual}. This verifies {@code expected} is a parent class or a
41 * interface that {@code actual} implements.
42 */
43 public static void assertAssignableFrom(Class<?> expected, Object actual) {
44 assertAssignableFrom(expected, actual.getClass());
45 }
46
47 /**
48 * Asserts that class {@code expected} is assignable from the class
49 * {@code actual}. This verifies {@code expected} is a parent class or a
50 * interface that {@code actual} implements.
51 */
52 public static void assertAssignableFrom(Class<?> expected, Class<?> actual) {
53 Assert.assertTrue(
54 "Expected " + expected.getCanonicalName() +
55 " to be assignable from actual class " + actual.getCanonicalName(),
56 expected.isAssignableFrom(actual));
57 }
58
59 /**
60 * Asserts that {@code actual} is not equal {@code unexpected}, according
61 * to both {@code ==} and {@link Object#equals}.
62 */
63 public static void assertNotEqual(
64 String message, Object unexpected, Object actual) {
65 if (equal(unexpected, actual)) {
66 failEqual(message, unexpected);
67 }
68 }
69
70 /**
71 * Variant of {@link #assertNotEqual(String,Object,Object)} using a
72 * generic message.
73 */
74 public static void assertNotEqual(Object unexpected, Object actual) {
75 assertNotEqual(null, unexpected, actual);
76 }
77
78 /**
79 * Asserts that array {@code actual} is the same size and every element equals
80 * those in array {@code expected}. On failure, message indicates specific
81 * element mismatch.
82 */
83 public static void assertEquals(
84 String message, byte[] expected, byte[] actual) {
85 if (expected.length != actual.length) {
86 failWrongLength(message, expected.length, actual.length);
87 }
88 for (int i = 0; i < expected.length; i++) {
89 if (expected[i] != actual[i]) {
90 failWrongElement(message, i, expected[i], actual[i]);
91 }
92 }
93 }
94
95 /**
96 * Asserts that array {@code actual} is the same size and every element equals
97 * those in array {@code expected}. On failure, message indicates specific
98 * element mismatch.
99 */
100 public static void assertEquals(byte[] expected, byte[] actual) {
101 assertEquals(null, expected, actual);
102 }
103
104 /**
105 * Asserts that array {@code actual} is the same size and every element equals
106 * those in array {@code expected}. On failure, message indicates first
107 * specific element mismatch.
108 */
109 public static void assertEquals(
110 String message, int[] expected, int[] actual) {
111 if (expected.length != actual.length) {
112 failWrongLength(message, expected.length, actual.length);
113 }
114 for (int i = 0; i < expected.length; i++) {
115 if (expected[i] != actual[i]) {
116 failWrongElement(message, i, expected[i], actual[i]);
117 }
118 }
119 }
120
121 /**
122 * Asserts that array {@code actual} is the same size and every element equals
123 * those in array {@code expected}. On failure, message indicates first
124 * specific element mismatch.
125 */
126 public static void assertEquals(int[] expected, int[] actual) {
127 assertEquals(null, expected, actual);
128 }
129
130 /**
131 * Asserts that array {@code actual} is the same size and every element equals
132 * those in array {@code expected}. On failure, message indicates first
133 * specific element mismatch.
134 */
135 public static void assertEquals(
136 String message, double[] expected, double[] actual) {
137 if (expected.length != actual.length) {
138 failWrongLength(message, expected.length, actual.length);
139 }
140 for (int i = 0; i < expected.length; i++) {
141 if (expected[i] != actual[i]) {
142 failWrongElement(message, i, expected[i], actual[i]);
143 }
144 }
145 }
146
147 /**
148 * Asserts that array {@code actual} is the same size and every element equals
149 * those in array {@code expected}. On failure, message indicates first
150 * specific element mismatch.
151 */
152 public static void assertEquals(double[] expected, double[] actual) {
153 assertEquals(null, expected, actual);
154 }
155
156 /**
157 * Asserts that array {@code actual} is the same size and every element
158 * is the same as those in array {@code expected}. Note that this uses
159 * {@code equals()} instead of {@code ==} to compare the objects.
Jeff Smitha45746e2012-07-19 14:19:24 -0500160 * {@code null} will be considered equal to {@code null} (unlike SQL).
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800161 * On failure, message indicates first specific element mismatch.
162 */
163 public static void assertEquals(
164 String message, Object[] expected, Object[] actual) {
165 if (expected.length != actual.length) {
166 failWrongLength(message, expected.length, actual.length);
167 }
168 for (int i = 0; i < expected.length; i++) {
169 Object exp = expected[i];
170 Object act = actual[i];
171 // The following borrowed from java.util.equals(Object[], Object[]).
172 if (!((exp==null) ? act==null : exp.equals(act))) {
173 failWrongElement(message, i, exp, act);
174 }
175 }
176 }
177
178 /**
179 * Asserts that array {@code actual} is the same size and every element
180 * is the same as those in array {@code expected}. Note that this uses
181 * {@code ==} instead of {@code equals()} to compare the objects.
182 * On failure, message indicates first specific element mismatch.
183 */
184 public static void assertEquals(Object[] expected, Object[] actual) {
185 assertEquals(null, expected, actual);
186 }
187
188 /** Asserts that two sets contain the same elements. */
189 public static void assertEquals(
190 String message, Set<? extends Object> expected, Set<? extends Object> actual) {
191 Set<Object> onlyInExpected = new HashSet<Object>(expected);
192 onlyInExpected.removeAll(actual);
193 Set<Object> onlyInActual = new HashSet<Object>(actual);
194 onlyInActual.removeAll(expected);
195 if (onlyInExpected.size() != 0 || onlyInActual.size() != 0) {
196 Set<Object> intersection = new HashSet<Object>(expected);
197 intersection.retainAll(actual);
198 failWithMessage(
199 message,
200 "Sets do not match.\nOnly in expected: " + onlyInExpected
201 + "\nOnly in actual: " + onlyInActual
202 + "\nIntersection: " + intersection);
203 }
204 }
205
206 /** Asserts that two sets contain the same elements. */
207 public static void assertEquals(Set<? extends Object> expected, Set<? extends Object> actual) {
208 assertEquals(null, expected, actual);
209 }
210
211 /**
212 * Asserts that {@code expectedRegex} exactly matches {@code actual} and
213 * fails with {@code message} if it does not. The MatchResult is returned
214 * in case the test needs access to any captured groups. Note that you can
215 * also use this for a literal string, by wrapping your expected string in
216 * {@link Pattern#quote}.
217 */
218 public static MatchResult assertMatchesRegex(
219 String message, String expectedRegex, String actual) {
220 if (actual == null) {
221 failNotMatches(message, expectedRegex, actual);
222 }
223 Matcher matcher = getMatcher(expectedRegex, actual);
224 if (!matcher.matches()) {
225 failNotMatches(message, expectedRegex, actual);
226 }
227 return matcher;
228 }
229
230 /**
231 * Variant of {@link #assertMatchesRegex(String,String,String)} using a
232 * generic message.
233 */
234 public static MatchResult assertMatchesRegex(
235 String expectedRegex, String actual) {
236 return assertMatchesRegex(null, expectedRegex, actual);
237 }
238
239 /**
240 * Asserts that {@code expectedRegex} matches any substring of {@code actual}
241 * and fails with {@code message} if it does not. The Matcher is returned in
242 * case the test needs access to any captured groups. Note that you can also
243 * use this for a literal string, by wrapping your expected string in
244 * {@link Pattern#quote}.
245 */
246 public static MatchResult assertContainsRegex(
247 String message, String expectedRegex, String actual) {
248 if (actual == null) {
249 failNotContains(message, expectedRegex, actual);
250 }
251 Matcher matcher = getMatcher(expectedRegex, actual);
252 if (!matcher.find()) {
253 failNotContains(message, expectedRegex, actual);
254 }
255 return matcher;
256 }
257
258 /**
259 * Variant of {@link #assertContainsRegex(String,String,String)} using a
260 * generic message.
261 */
262 public static MatchResult assertContainsRegex(
263 String expectedRegex, String actual) {
264 return assertContainsRegex(null, expectedRegex, actual);
265 }
266
267 /**
268 * Asserts that {@code expectedRegex} does not exactly match {@code actual},
269 * and fails with {@code message} if it does. Note that you can also use
270 * this for a literal string, by wrapping your expected string in
271 * {@link Pattern#quote}.
272 */
273 public static void assertNotMatchesRegex(
274 String message, String expectedRegex, String actual) {
275 Matcher matcher = getMatcher(expectedRegex, actual);
276 if (matcher.matches()) {
277 failMatch(message, expectedRegex, actual);
278 }
279 }
280
281 /**
282 * Variant of {@link #assertNotMatchesRegex(String,String,String)} using a
283 * generic message.
284 */
285 public static void assertNotMatchesRegex(
286 String expectedRegex, String actual) {
287 assertNotMatchesRegex(null, expectedRegex, actual);
288 }
289
290 /**
291 * Asserts that {@code expectedRegex} does not match any substring of
292 * {@code actual}, and fails with {@code message} if it does. Note that you
293 * can also use this for a literal string, by wrapping your expected string
294 * in {@link Pattern#quote}.
295 */
296 public static void assertNotContainsRegex(
297 String message, String expectedRegex, String actual) {
298 Matcher matcher = getMatcher(expectedRegex, actual);
299 if (matcher.find()) {
300 failContains(message, expectedRegex, actual);
301 }
302 }
303
304 /**
305 * Variant of {@link #assertNotContainsRegex(String,String,String)} using a
306 * generic message.
307 */
308 public static void assertNotContainsRegex(
309 String expectedRegex, String actual) {
310 assertNotContainsRegex(null, expectedRegex, actual);
311 }
312
313 /**
314 * Asserts that {@code actual} contains precisely the elements
315 * {@code expected}, and in the same order.
316 */
317 public static void assertContentsInOrder(
318 String message, Iterable<?> actual, Object... expected) {
Karl Rosaen97eb32e2009-06-29 18:06:40 -0700319 ArrayList actualList = new ArrayList();
320 for (Object o : actual) {
321 actualList.add(o);
322 }
323 Assert.assertEquals(message, Arrays.asList(expected), actualList);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800324 }
325
326 /**
327 * Variant of assertContentsInOrder(String, Iterable<?>, Object...)
328 * using a generic message.
329 */
330 public static void assertContentsInOrder(
331 Iterable<?> actual, Object... expected) {
332 assertContentsInOrder((String) null, actual, expected);
333 }
334
335 /**
336 * Asserts that {@code actual} contains precisely the elements
337 * {@code expected}, but in any order.
338 */
339 public static void assertContentsInAnyOrder(String message, Iterable<?> actual,
340 Object... expected) {
341 HashMap<Object, Object> expectedMap = new HashMap<Object, Object>(expected.length);
342 for (Object expectedObj : expected) {
343 expectedMap.put(expectedObj, expectedObj);
344 }
345
346 for (Object actualObj : actual) {
347 if (expectedMap.remove(actualObj) == null) {
348 failWithMessage(message, "Extra object in actual: (" + actualObj.toString() + ")");
349 }
350 }
351
352 if (expectedMap.size() > 0) {
353 failWithMessage(message, "Extra objects in expected.");
354 }
355 }
356
357 /**
358 * Variant of assertContentsInAnyOrder(String, Iterable<?>, Object...)
359 * using a generic message.
360 */
361 public static void assertContentsInAnyOrder(Iterable<?> actual, Object... expected) {
362 assertContentsInAnyOrder((String)null, actual, expected);
363 }
364
365 /**
366 * Asserts that {@code iterable} is empty.
367 */
368 public static void assertEmpty(String message, Iterable<?> iterable) {
369 if (iterable.iterator().hasNext()) {
370 failNotEmpty(message, iterable.toString());
371 }
372 }
373
374 /**
375 * Variant of {@link #assertEmpty(String, Iterable)} using a
376 * generic message.
377 */
378 public static void assertEmpty(Iterable<?> iterable) {
379 assertEmpty(null, iterable);
380 }
381
382 /**
383 * Asserts that {@code map} is empty.
384 */
385 public static void assertEmpty(String message, Map<?,?> map) {
386 if (!map.isEmpty()) {
387 failNotEmpty(message, map.toString());
388 }
389 }
390
391 /**
392 * Variant of {@link #assertEmpty(String, Map)} using a generic
393 * message.
394 */
395 public static void assertEmpty(Map<?,?> map) {
396 assertEmpty(null, map);
397 }
398
399 /**
400 * Asserts that {@code iterable} is not empty.
401 */
402 public static void assertNotEmpty(String message, Iterable<?> iterable) {
403 if (!iterable.iterator().hasNext()) {
404 failEmpty(message);
405 }
406 }
407
408 /**
409 * Variant of assertNotEmpty(String, Iterable<?>)
410 * using a generic message.
411 */
412 public static void assertNotEmpty(Iterable<?> iterable) {
413 assertNotEmpty(null, iterable);
414 }
415
416 /**
417 * Asserts that {@code map} is not empty.
418 */
419 public static void assertNotEmpty(String message, Map<?,?> map) {
420 if (map.isEmpty()) {
421 failEmpty(message);
422 }
423 }
424
425 /**
426 * Variant of {@link #assertNotEmpty(String, Map)} using a generic
427 * message.
428 */
429 public static void assertNotEmpty(Map<?,?> map) {
430 assertNotEmpty(null, map);
431 }
432
433 /**
434 * Utility for testing equals() and hashCode() results at once.
435 * Tests that lhs.equals(rhs) matches expectedResult, as well as
436 * rhs.equals(lhs). Also tests that hashCode() return values are
437 * equal if expectedResult is true. (hashCode() is not tested if
438 * expectedResult is false, as unequal objects can have equal hashCodes.)
439 *
440 * @param lhs An Object for which equals() and hashCode() are to be tested.
441 * @param rhs As lhs.
442 * @param expectedResult True if the objects should compare equal,
443 * false if not.
444 */
445 public static void checkEqualsAndHashCodeMethods(
446 String message, Object lhs, Object rhs, boolean expectedResult) {
447
448 if ((lhs == null) && (rhs == null)) {
449 Assert.assertTrue(
450 "Your check is dubious...why would you expect null != null?",
451 expectedResult);
452 return;
453 }
454
455 if ((lhs == null) || (rhs == null)) {
456 Assert.assertFalse(
457 "Your check is dubious...why would you expect an object "
458 + "to be equal to null?", expectedResult);
459 }
460
461 if (lhs != null) {
462 Assert.assertEquals(message, expectedResult, lhs.equals(rhs));
463 }
464 if (rhs != null) {
465 Assert.assertEquals(message, expectedResult, rhs.equals(lhs));
466 }
467
468 if (expectedResult) {
469 String hashMessage =
470 "hashCode() values for equal objects should be the same";
471 if (message != null) {
472 hashMessage += ": " + message;
473 }
474 Assert.assertTrue(hashMessage, lhs.hashCode() == rhs.hashCode());
475 }
476 }
477
478 /**
479 * Variant of
480 * checkEqualsAndHashCodeMethods(String,Object,Object,boolean...)}
481 * using a generic message.
482 */
483 public static void checkEqualsAndHashCodeMethods(Object lhs, Object rhs,
484 boolean expectedResult) {
485 checkEqualsAndHashCodeMethods((String) null, lhs, rhs, expectedResult);
486 }
487
488 private static Matcher getMatcher(String expectedRegex, String actual) {
489 Pattern pattern = Pattern.compile(expectedRegex);
490 return pattern.matcher(actual);
491 }
492
493 private static void failEqual(String message, Object unexpected) {
494 failWithMessage(message, "expected not to be:<" + unexpected + ">");
495 }
496
497 private static void failWrongLength(
498 String message, int expected, int actual) {
499 failWithMessage(message, "expected array length:<" + expected
500 + "> but was:<" + actual + '>');
501 }
502
503 private static void failWrongElement(
504 String message, int index, Object expected, Object actual) {
505 failWithMessage(message, "expected array element[" + index + "]:<"
506 + expected + "> but was:<" + actual + '>');
507 }
508
509 private static void failNotMatches(
510 String message, String expectedRegex, String actual) {
511 String actualDesc = (actual == null) ? "null" : ('<' + actual + '>');
512 failWithMessage(message, "expected to match regex:<" + expectedRegex
513 + "> but was:" + actualDesc);
514 }
515
516 private static void failNotContains(
517 String message, String expectedRegex, String actual) {
518 String actualDesc = (actual == null) ? "null" : ('<' + actual + '>');
519 failWithMessage(message, "expected to contain regex:<" + expectedRegex
520 + "> but was:" + actualDesc);
521 }
522
523 private static void failMatch(
524 String message, String expectedRegex, String actual) {
525 failWithMessage(message, "expected not to match regex:<" + expectedRegex
526 + "> but was:<" + actual + '>');
527 }
528
529 private static void failContains(
530 String message, String expectedRegex, String actual) {
531 failWithMessage(message, "expected not to contain regex:<" + expectedRegex
532 + "> but was:<" + actual + '>');
533 }
534
535 private static void failNotEmpty(
536 String message, String actual) {
537 failWithMessage(message, "expected to be empty, but contained: <"
538 + actual + ">");
539 }
540
541 private static void failEmpty(String message) {
542 failWithMessage(message, "expected not to be empty, but was");
543 }
544
545 private static void failWithMessage(String userMessage, String ourMessage) {
546 Assert.fail((userMessage == null)
547 ? ourMessage
548 : userMessage + ' ' + ourMessage);
549 }
550
551 private static boolean equal(Object a, Object b) {
552 return a == b || (a != null && a.equals(b));
553 }
554
555}