1 /*
2  * Copyright (C) 2016 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 package com.android.server.pm.shortcutmanagertest;
17 
18 import static com.android.compatibility.common.util.SystemUtil.callWithShellPermissionIdentity;
19 
20 import static junit.framework.Assert.assertEquals;
21 import static junit.framework.Assert.assertFalse;
22 import static junit.framework.Assert.assertNotNull;
23 import static junit.framework.Assert.assertNull;
24 import static junit.framework.Assert.assertTrue;
25 import static junit.framework.Assert.fail;
26 
27 import static org.junit.Assert.assertNotEquals;
28 import static org.mockito.Matchers.any;
29 import static org.mockito.Matchers.anyList;
30 import static org.mockito.Matchers.anyString;
31 import static org.mockito.Matchers.eq;
32 import static org.mockito.Mockito.atLeastOnce;
33 import static org.mockito.Mockito.mock;
34 import static org.mockito.Mockito.reset;
35 import static org.mockito.Mockito.times;
36 import static org.mockito.Mockito.verify;
37 
38 import android.app.Instrumentation;
39 import android.app.role.RoleManager;
40 import android.content.ComponentName;
41 import android.content.Context;
42 import android.content.LocusId;
43 import android.content.pm.LauncherApps;
44 import android.content.pm.LauncherApps.Callback;
45 import android.content.pm.ShortcutInfo;
46 import android.graphics.Bitmap;
47 import android.graphics.BitmapFactory;
48 import android.os.BaseBundle;
49 import android.os.Bundle;
50 import android.os.Handler;
51 import android.os.Looper;
52 import android.os.Parcel;
53 import android.os.ParcelFileDescriptor;
54 import android.os.PersistableBundle;
55 import android.os.UserHandle;
56 import android.os.UserManager;
57 import android.test.MoreAsserts;
58 import android.util.Log;
59 
60 import junit.framework.Assert;
61 
62 import org.hamcrest.BaseMatcher;
63 import org.hamcrest.Description;
64 import org.hamcrest.Matcher;
65 import org.json.JSONException;
66 import org.json.JSONObject;
67 import org.mockito.ArgumentCaptor;
68 import org.mockito.ArgumentMatchers;
69 import org.mockito.hamcrest.MockitoHamcrest;
70 
71 import java.io.BufferedReader;
72 import java.io.File;
73 import java.io.FileNotFoundException;
74 import java.io.FileReader;
75 import java.io.IOException;
76 import java.util.ArrayList;
77 import java.util.Arrays;
78 import java.util.Collection;
79 import java.util.Collections;
80 import java.util.Comparator;
81 import java.util.LinkedHashSet;
82 import java.util.List;
83 import java.util.Set;
84 import java.util.SortedSet;
85 import java.util.TreeSet;
86 import java.util.concurrent.CountDownLatch;
87 import java.util.function.BooleanSupplier;
88 import java.util.function.Consumer;
89 import java.util.function.Function;
90 import java.util.function.Predicate;
91 
92 /**
93  * Common utility methods for ShortcutManager tests.  This is used by both CTS and the unit tests.
94  * Because it's used by CTS too, it can only access the public APIs.
95  */
96 public class ShortcutManagerTestUtils {
97     private static final String TAG = "ShortcutManagerUtils";
98 
99     private static final boolean ENABLE_DUMPSYS = true; // DO NOT SUBMIT WITH true
100 
101     private static final int STANDARD_TIMEOUT_SEC = 5;
102 
103     private static final String[] EMPTY_STRINGS = new String[0];
104 
ShortcutManagerTestUtils()105     private ShortcutManagerTestUtils() {
106     }
107 
readAll(File file)108     public static List<String> readAll(File file) throws FileNotFoundException {
109         return readAll(ParcelFileDescriptor.open(
110                 file.getAbsoluteFile(), ParcelFileDescriptor.MODE_READ_ONLY));
111     }
112 
readAll(ParcelFileDescriptor pfd)113     public static List<String> readAll(ParcelFileDescriptor pfd) {
114         try {
115             try {
116                 final ArrayList<String> ret = new ArrayList<>();
117                 try (BufferedReader r = new BufferedReader(
118                         new FileReader(pfd.getFileDescriptor()))) {
119                     String line;
120                     while ((line = r.readLine()) != null) {
121                         ret.add(line);
122                     }
123                     r.readLine();
124                 }
125                 return ret;
126             } finally {
127                 pfd.close();
128             }
129         } catch (IOException e) {
130             throw new RuntimeException(e);
131         }
132     }
133 
concatResult(List<String> result)134     public static String concatResult(List<String> result) {
135         final StringBuilder sb = new StringBuilder();
136         for (String s : result) {
137             sb.append(s);
138             sb.append("\n");
139         }
140         return sb.toString();
141     }
142 
extractShortcutIds(List<String> result)143     public static List<String> extractShortcutIds(List<String> result) {
144         final String prefix = "ShortcutInfo {id=";
145         final String postfix = ", ";
146 
147         List<String> ids = new ArrayList<>();
148         for (String line : result) {
149             if (line.contains(prefix)) {
150                 ids.add(line.substring(
151                         line.indexOf(prefix) + prefix.length(), line.indexOf(postfix)));
152             }
153         }
154         return ids;
155     }
156 
resultContains(List<String> result, String expected)157     public static boolean resultContains(List<String> result, String expected) {
158         for (String line : result) {
159             if (line.contains(expected)) {
160                 return true;
161             }
162         }
163         return false;
164     }
165 
assertSuccess(List<String> result)166     public static List<String> assertSuccess(List<String> result) {
167         if (!resultContains(result, "Success")) {
168             fail("Command failed.  Result was:\n" + concatResult(result));
169         }
170         return result;
171     }
172 
assertContains(List<String> result, String expected)173     public static List<String> assertContains(List<String> result, String expected) {
174         if (!resultContains(result, expected)) {
175             fail("Didn't contain expected string=" + expected
176                     + "\nActual:\n" + concatResult(result));
177         }
178         return result;
179     }
180 
assertHaveIds(List<String> result, String... expectedIds)181     public static List<String> assertHaveIds(List<String> result, String... expectedIds) {
182         assertSuccess(result);
183 
184         final SortedSet<String> expected = new TreeSet<>(list(expectedIds));
185         final SortedSet<String> actual = new TreeSet<>(extractShortcutIds(result));
186         assertEquals(expected, actual);
187 
188         return result;
189     }
190 
runCommand(Instrumentation instrumentation, String command)191     public static List<String> runCommand(Instrumentation instrumentation, String command) {
192         return runCommand(instrumentation, command, null);
193     }
runCommand(Instrumentation instrumentation, String command, Predicate<List<String>> resultAsserter)194     public static List<String> runCommand(Instrumentation instrumentation, String command,
195             Predicate<List<String>> resultAsserter) {
196         Log.d(TAG, "Running command: " + command);
197         final List<String> result;
198         try {
199             result = readAll(
200                     instrumentation.getUiAutomation().executeShellCommand(command));
201         } catch (Exception e) {
202             throw new RuntimeException(e);
203         }
204         if (resultAsserter != null && !resultAsserter.test(result)) {
205             fail("Command '" + command + "' failed, output was:\n" + concatResult(result));
206         }
207         return result;
208     }
209 
runCommandForNoOutput(Instrumentation instrumentation, String command)210     public static void runCommandForNoOutput(Instrumentation instrumentation, String command) {
211         runCommand(instrumentation, command, result -> result.size() == 0);
212     }
213 
runShortcutCommand(Instrumentation instrumentation, String command, Predicate<List<String>> resultAsserter)214     public static List<String> runShortcutCommand(Instrumentation instrumentation, String command,
215             Predicate<List<String>> resultAsserter) {
216         return runCommand(instrumentation, "cmd shortcut " + command, resultAsserter);
217     }
218 
runShortcutCommandForSuccess(Instrumentation instrumentation, String command)219     public static List<String> runShortcutCommandForSuccess(Instrumentation instrumentation,
220             String command) {
221         return runShortcutCommand(instrumentation, command, result -> result.contains("Success"));
222     }
223 
getParentUser(Context context)224     private static UserHandle getParentUser(Context context) {
225         final UserHandle user = context.getUser();
226         final UserManager userManager = context.getSystemService(UserManager.class);
227         if (!userManager.isManagedProfile(user.getIdentifier())) {
228             return user;
229         }
230 
231         final List<UserHandle> profiles = userManager.getUserProfiles();
232         for (UserHandle handle : profiles) {
233             if (!userManager.isManagedProfile(handle.getIdentifier())) {
234                 return handle;
235             }
236         }
237         return null;
238     }
239 
getDefaultLauncher(Instrumentation instrumentation)240     public static String getDefaultLauncher(Instrumentation instrumentation) throws Exception {
241         final Context context = instrumentation.getContext();
242         final RoleManager roleManager = context.getSystemService(RoleManager.class);
243         final UserHandle user = getParentUser(context);
244         List<String> roleHolders = callWithShellPermissionIdentity(
245                 () -> roleManager.getRoleHoldersAsUser(RoleManager.ROLE_HOME, user));
246         int size = roleHolders.size();
247         if (size == 1) {
248             return roleHolders.get(0);
249         }
250 
251         if (size > 1) {
252             fail("Too many launchers for user " + user.getIdentifier() + " using role "
253                     + RoleManager.ROLE_HOME + ": " + roleHolders);
254         } else {
255             fail("No default launcher for user " + user.getIdentifier() + " using role "
256                     + RoleManager.ROLE_HOME);
257         }
258         return null;
259     }
260 
setDefaultLauncher(Instrumentation instrumentation, String packageName)261     public static void setDefaultLauncher(Instrumentation instrumentation, String packageName) {
262         runCommandForNoOutput(instrumentation, "cmd role add-role-holder --user "
263                 + instrumentation.getContext().getUserId() + " " + RoleManager.ROLE_HOME + " "
264                 + packageName + " 0");
265         waitUntil("Failed to get shortcut access",
266                 () -> hasShortcutAccess(instrumentation, packageName), 60);
267     }
268 
setDefaultLauncher(Instrumentation instrumentation, Context packageContext)269     public static void setDefaultLauncher(Instrumentation instrumentation, Context packageContext) {
270         setDefaultLauncher(instrumentation, packageContext.getPackageName());
271     }
272 
hasShortcutAccess(Instrumentation instrumentation, String packageName)273     public static boolean hasShortcutAccess(Instrumentation instrumentation, String packageName) {
274         final List<String> result = runShortcutCommandForSuccess(instrumentation,
275                 "has-shortcut-access --user " + instrumentation.getContext().getUserId()
276                         + " " + packageName);
277         for (String s : result) {
278             if (s.startsWith("true")) {
279                 return true;
280             } else if (s.startsWith("false")) {
281                 return false;
282             }
283         }
284         fail("Failed to check shortcut access");
285         return false;
286     }
287 
overrideConfig(Instrumentation instrumentation, String config)288     public static void overrideConfig(Instrumentation instrumentation, String config) {
289         runShortcutCommandForSuccess(instrumentation, "override-config " + config);
290     }
291 
resetConfig(Instrumentation instrumentation)292     public static void resetConfig(Instrumentation instrumentation) {
293         runShortcutCommandForSuccess(instrumentation, "reset-config");
294     }
295 
resetThrottling(Instrumentation instrumentation)296     public static void resetThrottling(Instrumentation instrumentation) {
297         runShortcutCommandForSuccess(instrumentation, "reset-throttling");
298     }
299 
resetAllThrottling(Instrumentation instrumentation)300     public static void resetAllThrottling(Instrumentation instrumentation) {
301         runShortcutCommandForSuccess(instrumentation, "reset-all-throttling");
302     }
303 
clearShortcuts(Instrumentation instrumentation, int userId, String packageName)304     public static void clearShortcuts(Instrumentation instrumentation, int userId,
305             String packageName) {
306         runShortcutCommandForSuccess(instrumentation, "clear-shortcuts "
307                 + " --user " + userId + " " + packageName);
308     }
309 
anyContains(List<String> result, String expected)310     public static void anyContains(List<String> result, String expected) {
311         for (String l : result) {
312             if (l.contains(expected)) {
313                 return;
314             }
315         }
316         fail("Result didn't contain '" + expected + "': was\n" + result);
317     }
318 
enableComponent(Instrumentation instrumentation, ComponentName cn, boolean enable)319     public static void enableComponent(Instrumentation instrumentation, ComponentName cn,
320             boolean enable) {
321 
322         final String word = (enable ? "enable" : "disable");
323         runCommand(instrumentation,
324                 "pm " + word + " " + cn.flattenToString()
325                 , result ->concatResult(result).contains(word));
326     }
327 
appOps(Instrumentation instrumentation, String packageName, String op, String mode)328     public static void appOps(Instrumentation instrumentation, String packageName,
329             String op, String mode) {
330         runCommand(instrumentation, "appops set " + packageName + " " + op + " " + mode);
331     }
332 
dumpsysShortcut(Instrumentation instrumentation)333     public static void dumpsysShortcut(Instrumentation instrumentation) {
334         if (!ENABLE_DUMPSYS) {
335             return;
336         }
337         Log.e(TAG, "Dumpsys shortcut");
338         for (String s : runCommand(instrumentation, "dumpsys shortcut")) {
339             Log.e(TAG, s);
340         }
341     }
342 
getCheckinDump(Instrumentation instrumentation)343     public static JSONObject getCheckinDump(Instrumentation instrumentation) throws JSONException {
344         return new JSONObject(concatResult(runCommand(instrumentation, "dumpsys shortcut -c")));
345     }
346 
isLowRamDevice(Instrumentation instrumentation)347     public static boolean isLowRamDevice(Instrumentation instrumentation) throws JSONException {
348         return getCheckinDump(instrumentation).getBoolean("lowRam");
349     }
350 
getIconSize(Instrumentation instrumentation)351     public static int getIconSize(Instrumentation instrumentation) throws JSONException {
352         return getCheckinDump(instrumentation).getInt("iconSize");
353     }
354 
makeBundle(Object... keysAndValues)355     public static Bundle makeBundle(Object... keysAndValues) {
356         assertTrue((keysAndValues.length % 2) == 0);
357 
358         if (keysAndValues.length == 0) {
359             return null;
360         }
361         final Bundle ret = new Bundle();
362 
363         for (int i = keysAndValues.length - 2; i >= 0; i -= 2) {
364             final String key = keysAndValues[i].toString();
365             final Object value = keysAndValues[i + 1];
366 
367             if (value == null) {
368                 ret.putString(key, null);
369             } else if (value instanceof Integer) {
370                 ret.putInt(key, (Integer) value);
371             } else if (value instanceof String) {
372                 ret.putString(key, (String) value);
373             } else if (value instanceof Bundle) {
374                 ret.putBundle(key, (Bundle) value);
375             } else {
376                 fail("Type not supported yet: " + value.getClass().getName());
377             }
378         }
379         return ret;
380     }
381 
makePersistableBundle(Object... keysAndValues)382     public static PersistableBundle makePersistableBundle(Object... keysAndValues) {
383         assertTrue((keysAndValues.length % 2) == 0);
384 
385         if (keysAndValues.length == 0) {
386             return null;
387         }
388         final PersistableBundle ret = new PersistableBundle();
389 
390         for (int i = keysAndValues.length - 2; i >= 0; i -= 2) {
391             final String key = keysAndValues[i].toString();
392             final Object value = keysAndValues[i + 1];
393 
394             if (value == null) {
395                 ret.putString(key, null);
396             } else if (value instanceof Integer) {
397                 ret.putInt(key, (Integer) value);
398             } else if (value instanceof String) {
399                 ret.putString(key, (String) value);
400             } else if (value instanceof PersistableBundle) {
401                 ret.putPersistableBundle(key, (PersistableBundle) value);
402             } else {
403                 fail("Type not supported yet: " + value.getClass().getName());
404             }
405         }
406         return ret;
407     }
408 
array(T... array)409     public static <T> T[] array(T... array) {
410         return array;
411     }
412 
list(T... array)413     public static <T> List<T> list(T... array) {
414         return Arrays.asList(array);
415     }
416 
hashSet(Set<T> in)417     public static <T> Set<T> hashSet(Set<T> in) {
418         return new LinkedHashSet<>(in);
419     }
420 
set(T... values)421     public static <T> Set<T> set(T... values) {
422         return set(v -> v, values);
423     }
424 
set(Function<V, T> converter, V... values)425     public static <T, V> Set<T> set(Function<V, T> converter, V... values) {
426         return set(converter, Arrays.asList(values));
427     }
428 
set(Function<V, T> converter, List<V> values)429     public static <T, V> Set<T> set(Function<V, T> converter, List<V> values) {
430         final LinkedHashSet<T> ret = new LinkedHashSet<>();
431         for (V v : values) {
432             ret.add(converter.apply(v));
433         }
434         return ret;
435     }
436 
locusId(String id)437     public static LocusId locusId(String id) {
438         return new LocusId(id);
439     }
440 
resetAll(Collection<?> mocks)441     public static void resetAll(Collection<?> mocks) {
442         for (Object o : mocks) {
443             reset(o);
444         }
445     }
446 
assertEmpty(T collection)447     public static <T extends Collection<?>> T assertEmpty(T collection) {
448         if (collection == null) {
449             return collection; // okay.
450         }
451         assertEquals(0, collection.size());
452         return collection;
453     }
454 
filter(List<ShortcutInfo> list, Predicate<ShortcutInfo> p)455     public static List<ShortcutInfo> filter(List<ShortcutInfo> list, Predicate<ShortcutInfo> p) {
456         final ArrayList<ShortcutInfo> ret = new ArrayList<>(list);
457         ret.removeIf(si -> !p.test(si));
458         return ret;
459     }
460 
filterByActivity(List<ShortcutInfo> list, ComponentName activity)461     public static List<ShortcutInfo> filterByActivity(List<ShortcutInfo> list,
462             ComponentName activity) {
463         return filter(list, si ->
464                 (si.getActivity().equals(activity)
465                         && (si.isDeclaredInManifest() || si.isDynamic())));
466     }
467 
changedSince(List<ShortcutInfo> list, long time)468     public static List<ShortcutInfo> changedSince(List<ShortcutInfo> list, long time) {
469         return filter(list, si -> si.getLastChangedTimestamp() >= time);
470     }
471 
472     @FunctionalInterface
473     public interface ExceptionRunnable {
run()474         void run() throws Exception;
475     }
476 
assertExpectException(Class<? extends Throwable> expectedExceptionType, String expectedExceptionMessageRegex, ExceptionRunnable r)477     public static void assertExpectException(Class<? extends Throwable> expectedExceptionType,
478             String expectedExceptionMessageRegex, ExceptionRunnable r) {
479         assertExpectException("", expectedExceptionType, expectedExceptionMessageRegex, r);
480     }
481 
assertCannotUpdateImmutable(Runnable r)482     public static void assertCannotUpdateImmutable(Runnable r) {
483         assertExpectException(
484                 IllegalArgumentException.class, "may not be manipulated via APIs", r::run);
485     }
486 
assertDynamicShortcutCountExceeded(Runnable r)487     public static void assertDynamicShortcutCountExceeded(Runnable r) {
488         assertExpectException(IllegalArgumentException.class,
489                 "Max number of dynamic shortcuts exceeded", r::run);
490     }
491 
assertExpectException(String message, Class<? extends Throwable> expectedExceptionType, String expectedExceptionMessageRegex, ExceptionRunnable r)492     public static void assertExpectException(String message,
493             Class<? extends Throwable> expectedExceptionType,
494             String expectedExceptionMessageRegex, ExceptionRunnable r) {
495         try {
496             r.run();
497         } catch (Throwable e) {
498             Assert.assertTrue(
499                     "Expected exception type was " + expectedExceptionType.getName()
500                             + " but caught " + e + " (message=" + message + ")",
501                     expectedExceptionType.isAssignableFrom(e.getClass()));
502             if (expectedExceptionMessageRegex != null) {
503                 MoreAsserts.assertContainsRegex(expectedExceptionMessageRegex, e.getMessage());
504             }
505             return; // Pass
506         }
507         Assert.fail("Expected exception type " + expectedExceptionType.getName()
508                 + " was not thrown");
509     }
510 
assertShortcutIds(List<ShortcutInfo> actualShortcuts, String... expectedIds)511     public static List<ShortcutInfo> assertShortcutIds(List<ShortcutInfo> actualShortcuts,
512             String... expectedIds) {
513         final SortedSet<String> expected = new TreeSet<>(list(expectedIds));
514         final SortedSet<String> actual = new TreeSet<>();
515         for (ShortcutInfo s : actualShortcuts) {
516             actual.add(s.getId());
517         }
518 
519         // Compare the sets.
520         assertEquals(expected, actual);
521         return actualShortcuts;
522     }
523 
assertShortcutIdsOrdered(List<ShortcutInfo> actualShortcuts, String... expectedIds)524     public static List<ShortcutInfo> assertShortcutIdsOrdered(List<ShortcutInfo> actualShortcuts,
525             String... expectedIds) {
526         final ArrayList<String> expected = new ArrayList<>(list(expectedIds));
527         final ArrayList<String> actual = new ArrayList<>();
528         for (ShortcutInfo s : actualShortcuts) {
529             actual.add(s.getId());
530         }
531         assertEquals(expected, actual);
532         return actualShortcuts;
533     }
534 
assertAllHaveIntents( List<ShortcutInfo> actualShortcuts)535     public static List<ShortcutInfo> assertAllHaveIntents(
536             List<ShortcutInfo> actualShortcuts) {
537         for (ShortcutInfo s : actualShortcuts) {
538             assertNotNull("ID " + s.getId(), s.getIntent());
539         }
540         return actualShortcuts;
541     }
542 
assertAllNotHaveIntents( List<ShortcutInfo> actualShortcuts)543     public static List<ShortcutInfo> assertAllNotHaveIntents(
544             List<ShortcutInfo> actualShortcuts) {
545         for (ShortcutInfo s : actualShortcuts) {
546             assertNull("ID " + s.getId(), s.getIntent());
547         }
548         return actualShortcuts;
549     }
550 
assertAllHaveTitle( List<ShortcutInfo> actualShortcuts)551     public static List<ShortcutInfo> assertAllHaveTitle(
552             List<ShortcutInfo> actualShortcuts) {
553         for (ShortcutInfo s : actualShortcuts) {
554             assertNotNull("ID " + s.getId(), s.getShortLabel());
555         }
556         return actualShortcuts;
557     }
558 
assertAllNotHaveTitle( List<ShortcutInfo> actualShortcuts)559     public static List<ShortcutInfo> assertAllNotHaveTitle(
560             List<ShortcutInfo> actualShortcuts) {
561         for (ShortcutInfo s : actualShortcuts) {
562             assertNull("ID " + s.getId(), s.getShortLabel());
563         }
564         return actualShortcuts;
565     }
566 
assertAllKeyFieldsOnly( List<ShortcutInfo> actualShortcuts)567     public static List<ShortcutInfo> assertAllKeyFieldsOnly(
568             List<ShortcutInfo> actualShortcuts) {
569         for (ShortcutInfo s : actualShortcuts) {
570             assertTrue("ID " + s.getId(), s.hasKeyFieldsOnly());
571         }
572         return actualShortcuts;
573     }
574 
assertAllNotKeyFieldsOnly( List<ShortcutInfo> actualShortcuts)575     public static List<ShortcutInfo> assertAllNotKeyFieldsOnly(
576             List<ShortcutInfo> actualShortcuts) {
577         for (ShortcutInfo s : actualShortcuts) {
578             assertFalse("ID " + s.getId(), s.hasKeyFieldsOnly());
579         }
580         return actualShortcuts;
581     }
582 
assertAllDynamic(List<ShortcutInfo> actualShortcuts)583     public static List<ShortcutInfo> assertAllDynamic(List<ShortcutInfo> actualShortcuts) {
584         for (ShortcutInfo s : actualShortcuts) {
585             assertTrue("ID " + s.getId(), s.isDynamic());
586         }
587         return actualShortcuts;
588     }
589 
assertAllPinned(List<ShortcutInfo> actualShortcuts)590     public static List<ShortcutInfo> assertAllPinned(List<ShortcutInfo> actualShortcuts) {
591         for (ShortcutInfo s : actualShortcuts) {
592             assertTrue("ID " + s.getId(), s.isPinned());
593         }
594         return actualShortcuts;
595     }
596 
assertAllDynamicOrPinned( List<ShortcutInfo> actualShortcuts)597     public static List<ShortcutInfo> assertAllDynamicOrPinned(
598             List<ShortcutInfo> actualShortcuts) {
599         for (ShortcutInfo s : actualShortcuts) {
600             assertTrue("ID " + s.getId(), s.isDynamic() || s.isPinned());
601         }
602         return actualShortcuts;
603     }
604 
assertAllManifest( List<ShortcutInfo> actualShortcuts)605     public static List<ShortcutInfo> assertAllManifest(
606             List<ShortcutInfo> actualShortcuts) {
607         for (ShortcutInfo s : actualShortcuts) {
608             assertTrue("ID " + s.getId(), s.isDeclaredInManifest());
609         }
610         return actualShortcuts;
611     }
612 
assertAllNotManifest( List<ShortcutInfo> actualShortcuts)613     public static List<ShortcutInfo> assertAllNotManifest(
614             List<ShortcutInfo> actualShortcuts) {
615         for (ShortcutInfo s : actualShortcuts) {
616             assertFalse("ID " + s.getId(), s.isDeclaredInManifest());
617         }
618         return actualShortcuts;
619     }
620 
assertAllDisabled( List<ShortcutInfo> actualShortcuts)621     public static List<ShortcutInfo> assertAllDisabled(
622             List<ShortcutInfo> actualShortcuts) {
623         for (ShortcutInfo s : actualShortcuts) {
624             assertTrue("ID " + s.getId(), !s.isEnabled());
625         }
626         return actualShortcuts;
627     }
628 
assertAllEnabled( List<ShortcutInfo> actualShortcuts)629     public static List<ShortcutInfo> assertAllEnabled(
630             List<ShortcutInfo> actualShortcuts) {
631         for (ShortcutInfo s : actualShortcuts) {
632             assertTrue("ID " + s.getId(), s.isEnabled());
633         }
634         return actualShortcuts;
635     }
636 
assertAllImmutable( List<ShortcutInfo> actualShortcuts)637     public static List<ShortcutInfo> assertAllImmutable(
638             List<ShortcutInfo> actualShortcuts) {
639         for (ShortcutInfo s : actualShortcuts) {
640             assertTrue("ID " + s.getId(), s.isImmutable());
641         }
642         return actualShortcuts;
643     }
644 
assertDynamicOnly(ShortcutInfo si)645     public static void assertDynamicOnly(ShortcutInfo si) {
646         assertTrue(si.isDynamic());
647         assertFalse(si.isPinned());
648     }
649 
assertPinnedOnly(ShortcutInfo si)650     public static void assertPinnedOnly(ShortcutInfo si) {
651         assertFalse(si.isDynamic());
652         assertFalse(si.isDeclaredInManifest());
653         assertTrue(si.isPinned());
654     }
655 
assertDynamicAndPinned(ShortcutInfo si)656     public static void assertDynamicAndPinned(ShortcutInfo si) {
657         assertTrue(si.isDynamic());
658         assertTrue(si.isPinned());
659     }
660 
assertBitmapSize(int expectedWidth, int expectedHeight, Bitmap bitmap)661     public static void assertBitmapSize(int expectedWidth, int expectedHeight, Bitmap bitmap) {
662         assertEquals("width", expectedWidth, bitmap.getWidth());
663         assertEquals("height", expectedHeight, bitmap.getHeight());
664     }
665 
assertAllUnique(Collection<T> list)666     public static <T> void assertAllUnique(Collection<T> list) {
667         final Set<Object> set = new LinkedHashSet<>();
668         for (T item : list) {
669             if (set.contains(item)) {
670                 fail("Duplicate item found: " + item + " (in the list: " + list + ")");
671             }
672             set.add(item);
673         }
674     }
675 
findShortcut(List<ShortcutInfo> list, String id)676     public static ShortcutInfo findShortcut(List<ShortcutInfo> list, String id) {
677         for (ShortcutInfo si : list) {
678             if (si.getId().equals(id)) {
679                 return si;
680             }
681         }
682         fail("Shortcut " + id + " not found in the list");
683         return null;
684     }
685 
pfdToBitmap(ParcelFileDescriptor pfd)686     public static Bitmap pfdToBitmap(ParcelFileDescriptor pfd) {
687         assertNotNull(pfd);
688         try {
689             try {
690                 return BitmapFactory.decodeFileDescriptor(pfd.getFileDescriptor());
691             } finally {
692                 pfd.close();
693             }
694         } catch (IOException e) {
695             throw new RuntimeException(e);
696         }
697     }
698 
assertBundleEmpty(BaseBundle b)699     public static void assertBundleEmpty(BaseBundle b) {
700         assertTrue(b == null || b.size() == 0);
701     }
702 
assertCallbackNotReceived(LauncherApps.Callback mock)703     public static void assertCallbackNotReceived(LauncherApps.Callback mock) {
704         verify(mock, times(0)).onShortcutsChanged(anyString(), anyList(),
705                 any(UserHandle.class));
706     }
707 
assertCallbackReceived(LauncherApps.Callback mock, UserHandle user, String packageName, String... ids)708     public static void assertCallbackReceived(LauncherApps.Callback mock,
709             UserHandle user, String packageName, String... ids) {
710         verify(mock).onShortcutsChanged(eq(packageName), checkShortcutIds(ids),
711                 eq(user));
712     }
713 
checkAssertSuccess(Runnable r)714     public static boolean checkAssertSuccess(Runnable r) {
715         try {
716             r.run();
717             return true;
718         } catch (AssertionError e) {
719             return false;
720         }
721     }
722 
checkArgument(Predicate<T> checker, String description, List<T> matchedCaptor)723     public static <T> T checkArgument(Predicate<T> checker, String description,
724             List<T> matchedCaptor) {
725         final Matcher<T> m = new BaseMatcher<T>() {
726             @Override
727             public boolean matches(Object item) {
728                 if (item == null) {
729                     return false;
730                 }
731                 final T value = (T) item;
732                 if (!checker.test(value)) {
733                     return false;
734                 }
735 
736                 if (matchedCaptor != null) {
737                     matchedCaptor.add(value);
738                 }
739                 return true;
740             }
741 
742             @Override
743             public void describeTo(Description d) {
744                 d.appendText(description);
745             }
746         };
747         return MockitoHamcrest.argThat(m);
748     }
749 
checkShortcutIds(String... ids)750     public static List<ShortcutInfo> checkShortcutIds(String... ids) {
751         return checkArgument((List<ShortcutInfo> list) -> {
752             final Set<String> actualSet = set(si -> si.getId(), list);
753             return actualSet.equals(set(ids));
754 
755         }, "Shortcut IDs=[" + Arrays.toString(ids) + "]", null);
756     }
757 
parceled(ShortcutInfo si)758     public static ShortcutInfo parceled(ShortcutInfo si) {
759         Parcel p = Parcel.obtain();
760         p.writeParcelable(si, 0);
761         p.setDataPosition(0);
762         ShortcutInfo si2 = p.readParcelable(ShortcutManagerTestUtils.class.getClassLoader());
763         p.recycle();
764         return si2;
765     }
766 
cloneShortcutList(List<ShortcutInfo> list)767     public static List<ShortcutInfo> cloneShortcutList(List<ShortcutInfo> list) {
768         if (list == null) {
769             return null;
770         }
771         final List<ShortcutInfo> ret = new ArrayList<>(list.size());
772         for (ShortcutInfo si : list) {
773             ret.add(parceled(si));
774         }
775 
776         return ret;
777     }
778 
779     private static final Comparator<ShortcutInfo> sRankComparator =
780             (ShortcutInfo a, ShortcutInfo b) -> Integer.compare(a.getRank(), b.getRank());
781 
sortedByRank(List<ShortcutInfo> shortcuts)782     public static List<ShortcutInfo> sortedByRank(List<ShortcutInfo> shortcuts) {
783         final ArrayList<ShortcutInfo> ret = new ArrayList<>(shortcuts);
784         Collections.sort(ret, sRankComparator);
785         return ret;
786     }
787 
waitUntil(String message, BooleanSupplier condition)788     public static void waitUntil(String message, BooleanSupplier condition) {
789         waitUntil(message, condition, STANDARD_TIMEOUT_SEC);
790     }
791 
waitUntil(String message, BooleanSupplier condition, int timeoutSeconds)792     public static void waitUntil(String message, BooleanSupplier condition, int timeoutSeconds) {
793         final long timeout = System.currentTimeMillis() + (timeoutSeconds * 1000L);
794         while (System.currentTimeMillis() < timeout) {
795             if (condition.getAsBoolean()) {
796                 return;
797             }
798             try {
799                 Thread.sleep(100);
800             } catch (InterruptedException e) {
801                 throw new RuntimeException(e);
802             }
803         }
804         fail("Timed out for: " + message);
805     }
806 
anyOrNull(Class<T> clazz)807     public static final <T> T anyOrNull(Class<T> clazz) {
808         return ArgumentMatchers.argThat(value -> true);
809     }
810 
anyStringOrNull()811     public static final String anyStringOrNull() {
812         return ArgumentMatchers.argThat(value -> true);
813     }
814 
assertWith(List<ShortcutInfo> list)815     public static ShortcutListAsserter assertWith(List<ShortcutInfo> list) {
816         return new ShortcutListAsserter(list);
817     }
818 
assertWith(ShortcutInfo... list)819     public static ShortcutListAsserter assertWith(ShortcutInfo... list) {
820         return assertWith(list(list));
821     }
822 
823     /**
824      * New style assertion that allows chained calls.
825      */
826     public static class ShortcutListAsserter {
827         private final ShortcutListAsserter mOriginal;
828         private final List<ShortcutInfo> mList;
829 
ShortcutListAsserter(List<ShortcutInfo> list)830         ShortcutListAsserter(List<ShortcutInfo> list) {
831             this(null, list);
832         }
833 
ShortcutListAsserter(ShortcutListAsserter original, List<ShortcutInfo> list)834         private ShortcutListAsserter(ShortcutListAsserter original, List<ShortcutInfo> list) {
835             mOriginal = (original == null) ? this : original;
836             mList = (list == null) ? new ArrayList<>(0) : new ArrayList<>(list);
837         }
838 
revertToOriginalList()839         public ShortcutListAsserter revertToOriginalList() {
840             return mOriginal;
841         }
842 
selectDynamic()843         public ShortcutListAsserter selectDynamic() {
844             return new ShortcutListAsserter(this,
845                     filter(mList, ShortcutInfo::isDynamic));
846         }
847 
selectManifest()848         public ShortcutListAsserter selectManifest() {
849             return new ShortcutListAsserter(this,
850                     filter(mList, ShortcutInfo::isDeclaredInManifest));
851         }
852 
selectPinned()853         public ShortcutListAsserter selectPinned() {
854             return new ShortcutListAsserter(this,
855                     filter(mList, ShortcutInfo::isPinned));
856         }
857 
selectFloating()858         public ShortcutListAsserter selectFloating() {
859             return new ShortcutListAsserter(this,
860                     filter(mList, (si -> si.isPinned()
861                             && !(si.isDynamic() || si.isDeclaredInManifest()))));
862         }
863 
selectByActivity(ComponentName activity)864         public ShortcutListAsserter selectByActivity(ComponentName activity) {
865             return new ShortcutListAsserter(this,
866                     ShortcutManagerTestUtils.filterByActivity(mList, activity));
867         }
868 
selectByChangedSince(long time)869         public ShortcutListAsserter selectByChangedSince(long time) {
870             return new ShortcutListAsserter(this,
871                     ShortcutManagerTestUtils.changedSince(mList, time));
872         }
873 
selectByIds(String... ids)874         public ShortcutListAsserter selectByIds(String... ids) {
875             final Set<String> idSet = set(ids);
876             final ArrayList<ShortcutInfo> selected = new ArrayList<>();
877             for (ShortcutInfo si : mList) {
878                 if (idSet.contains(si.getId())) {
879                     selected.add(si);
880                     idSet.remove(si.getId());
881                 }
882             }
883             if (idSet.size() > 0) {
884                 fail("Shortcuts not found for IDs=" + idSet);
885             }
886 
887             return new ShortcutListAsserter(this, selected);
888         }
889 
toSortByRank()890         public ShortcutListAsserter toSortByRank() {
891             return new ShortcutListAsserter(this,
892                     ShortcutManagerTestUtils.sortedByRank(mList));
893         }
894 
call(Consumer<List<ShortcutInfo>> c)895         public ShortcutListAsserter call(Consumer<List<ShortcutInfo>> c) {
896             c.accept(mList);
897             return this;
898         }
899 
haveIds(String... expectedIds)900         public ShortcutListAsserter haveIds(String... expectedIds) {
901             assertShortcutIds(mList, expectedIds);
902             return this;
903         }
904 
haveIdsOrdered(String... expectedIds)905         public ShortcutListAsserter haveIdsOrdered(String... expectedIds) {
906             assertShortcutIdsOrdered(mList, expectedIds);
907             return this;
908         }
909 
haveSequentialRanks()910         private ShortcutListAsserter haveSequentialRanks() {
911             for (int i = 0; i < mList.size(); i++) {
912                 final ShortcutInfo si = mList.get(i);
913                 assertEquals("Rank not sequential: id=" + si.getId(), i, si.getRank());
914             }
915             return this;
916         }
917 
haveRanksInOrder(String... expectedIds)918         public ShortcutListAsserter haveRanksInOrder(String... expectedIds) {
919             toSortByRank()
920                     .haveSequentialRanks()
921                     .haveIdsOrdered(expectedIds);
922             return this;
923         }
924 
isEmpty()925         public ShortcutListAsserter isEmpty() {
926             assertEquals(0, mList.size());
927             return this;
928         }
929 
areAllDynamic()930         public ShortcutListAsserter areAllDynamic() {
931             forAllShortcuts(s -> assertTrue("id=" + s.getId(), s.isDynamic()));
932             return this;
933         }
934 
areAllNotDynamic()935         public ShortcutListAsserter areAllNotDynamic() {
936             forAllShortcuts(s -> assertFalse("id=" + s.getId(), s.isDynamic()));
937             return this;
938         }
939 
areAllPinned()940         public ShortcutListAsserter areAllPinned() {
941             forAllShortcuts(s -> assertTrue("id=" + s.getId(), s.isPinned()));
942             return this;
943         }
944 
areAllNotPinned()945         public ShortcutListAsserter areAllNotPinned() {
946             forAllShortcuts(s -> assertFalse("id=" + s.getId(), s.isPinned()));
947             return this;
948         }
949 
areAllManifest()950         public ShortcutListAsserter areAllManifest() {
951             forAllShortcuts(s -> assertTrue("id=" + s.getId(), s.isDeclaredInManifest()));
952             return this;
953         }
954 
areAllNotManifest()955         public ShortcutListAsserter areAllNotManifest() {
956             forAllShortcuts(s -> assertFalse("id=" + s.getId(), s.isDeclaredInManifest()));
957             return this;
958         }
959 
areAllImmutable()960         public ShortcutListAsserter areAllImmutable() {
961             forAllShortcuts(s -> assertTrue("id=" + s.getId(), s.isImmutable()));
962             return this;
963         }
964 
areAllMutable()965         public ShortcutListAsserter areAllMutable() {
966             forAllShortcuts(s -> assertFalse("id=" + s.getId(), s.isImmutable()));
967             return this;
968         }
969 
areAllEnabled()970         public ShortcutListAsserter areAllEnabled() {
971             forAllShortcuts(s -> assertTrue("id=" + s.getId(), s.isEnabled()));
972             areAllWithDisabledReason(ShortcutInfo.DISABLED_REASON_NOT_DISABLED);
973             return this;
974         }
975 
areAllDisabled()976         public ShortcutListAsserter areAllDisabled() {
977             forAllShortcuts(s -> assertFalse("id=" + s.getId(), s.isEnabled()));
978             forAllShortcuts(s -> assertNotEquals("id=" + s.getId(),
979                     ShortcutInfo.DISABLED_REASON_NOT_DISABLED, s.getDisabledReason()));
980             return this;
981         }
982 
areAllFloating()983         public ShortcutListAsserter areAllFloating() {
984             forAllShortcuts(s -> assertTrue("id=" + s.getId(),
985                     s.isPinned() && !s.isDeclaredInManifest() && !s.isDynamic()));
986             return this;
987         }
988 
areAllNotFloating()989         public ShortcutListAsserter areAllNotFloating() {
990             forAllShortcuts(s -> assertTrue("id=" + s.getId(),
991                     !(s.isPinned() && !s.isDeclaredInManifest() && !s.isDynamic())));
992             return this;
993         }
994 
areAllOrphan()995         public ShortcutListAsserter areAllOrphan() {
996             forAllShortcuts(s -> assertTrue("id=" + s.getId(),
997                     !s.isPinned() && !s.isDeclaredInManifest() && !s.isDynamic()));
998             return this;
999         }
1000 
areAllNotOrphan()1001         public ShortcutListAsserter areAllNotOrphan() {
1002             forAllShortcuts(s -> assertTrue("id=" + s.getId(),
1003                     s.isPinned() || s.isDeclaredInManifest() || s.isDynamic()));
1004             return this;
1005         }
1006 
areAllVisibleToPublisher()1007         public ShortcutListAsserter areAllVisibleToPublisher() {
1008             forAllShortcuts(s -> assertTrue("id=" + s.getId(), s.isVisibleToPublisher()));
1009             return this;
1010         }
1011 
areAllNotVisibleToPublisher()1012         public ShortcutListAsserter areAllNotVisibleToPublisher() {
1013             forAllShortcuts(s -> assertFalse("id=" + s.getId(), s.isVisibleToPublisher()));
1014             return this;
1015         }
1016 
areAllWithKeyFieldsOnly()1017         public ShortcutListAsserter areAllWithKeyFieldsOnly() {
1018             forAllShortcuts(s -> assertTrue("id=" + s.getId(), s.hasKeyFieldsOnly()));
1019             return this;
1020         }
1021 
areAllNotWithKeyFieldsOnly()1022         public ShortcutListAsserter areAllNotWithKeyFieldsOnly() {
1023             forAllShortcuts(s -> assertFalse("id=" + s.getId(), s.hasKeyFieldsOnly()));
1024             return this;
1025         }
1026 
areAllWithActivity(ComponentName activity)1027         public ShortcutListAsserter areAllWithActivity(ComponentName activity) {
1028             forAllShortcuts(s -> assertEquals("id=" + s.getId(), activity, s.getActivity()));
1029             return this;
1030         }
1031 
areAllWithNoActivity()1032         public ShortcutListAsserter areAllWithNoActivity() {
1033             forAllShortcuts(s -> assertNull("id=" + s.getId(), s.getActivity()));
1034             return this;
1035         }
1036 
areAllWithIntent()1037         public ShortcutListAsserter areAllWithIntent() {
1038             forAllShortcuts(s -> assertNotNull("id=" + s.getId(), s.getIntent()));
1039             return this;
1040         }
1041 
areAllWithNoIntent()1042         public ShortcutListAsserter areAllWithNoIntent() {
1043             forAllShortcuts(s -> assertNull("id=" + s.getId(), s.getIntent()));
1044             return this;
1045         }
1046 
areAllWithDisabledReason(int disabledReason)1047         public ShortcutListAsserter areAllWithDisabledReason(int disabledReason) {
1048             forAllShortcuts(s -> assertEquals("id=" + s.getId(),
1049                     disabledReason, s.getDisabledReason()));
1050             if (disabledReason >= ShortcutInfo.DISABLED_REASON_VERSION_LOWER) {
1051                 areAllNotVisibleToPublisher();
1052             } else {
1053                 areAllVisibleToPublisher();
1054             }
1055             return this;
1056         }
1057 
forAllShortcuts(Consumer<ShortcutInfo> sa)1058         public ShortcutListAsserter forAllShortcuts(Consumer<ShortcutInfo> sa) {
1059             boolean found = false;
1060             for (int i = 0; i < mList.size(); i++) {
1061                 final ShortcutInfo si = mList.get(i);
1062                 found = true;
1063                 sa.accept(si);
1064             }
1065             assertTrue("No shortcuts found.", found);
1066             return this;
1067         }
1068 
forShortcut(Predicate<ShortcutInfo> p, Consumer<ShortcutInfo> sa)1069         public ShortcutListAsserter forShortcut(Predicate<ShortcutInfo> p,
1070                 Consumer<ShortcutInfo> sa) {
1071             boolean found = false;
1072             for (int i = 0; i < mList.size(); i++) {
1073                 final ShortcutInfo si = mList.get(i);
1074                 if (p.test(si)) {
1075                     found = true;
1076                     try {
1077                         sa.accept(si);
1078                     } catch (Throwable e) {
1079                         throw new AssertionError("Assertion failed for shortcut " + si.getId(), e);
1080                     }
1081                 }
1082             }
1083             assertTrue("Shortcut with the given condition not found.", found);
1084             return this;
1085         }
1086 
forShortcutWithId(String id, Consumer<ShortcutInfo> sa)1087         public ShortcutListAsserter forShortcutWithId(String id, Consumer<ShortcutInfo> sa) {
1088             forShortcut(si -> si.getId().equals(id), sa);
1089 
1090             return this;
1091         }
1092     }
1093 
assertBundlesEqual(BaseBundle b1, BaseBundle b2)1094     public static void assertBundlesEqual(BaseBundle b1, BaseBundle b2) {
1095         if (b1 == null && b2 == null) {
1096             return; // pass
1097         }
1098         assertNotNull("b1 is null but b2 is not", b1);
1099         assertNotNull("b2 is null but b1 is not", b2);
1100 
1101         // HashSet makes the error message readable.
1102         assertEquals(set(b1.keySet()), set(b2.keySet()));
1103 
1104         for (String key : b1.keySet()) {
1105             final Object v1 = b1.get(key);
1106             final Object v2 = b2.get(key);
1107             if (v1 == null) {
1108                 if (v2 == null) {
1109                     return;
1110                 }
1111             }
1112             if (v1.equals(v2)) {
1113                 return;
1114             }
1115 
1116             assertTrue("Only either value is null: key=" + key
1117                     + " b1=" + b1 + " b2=" + b2, v1 != null && v2 != null);
1118             assertEquals("Class mismatch: key=" + key, v1.getClass(), v2.getClass());
1119 
1120             if (v1 instanceof BaseBundle) {
1121                 assertBundlesEqual((BaseBundle) v1, (BaseBundle) v2);
1122 
1123             } else if (v1 instanceof boolean[]) {
1124                 assertTrue(Arrays.equals((boolean[]) v1, (boolean[]) v2));
1125 
1126             } else if (v1 instanceof int[]) {
1127                 MoreAsserts.assertEquals((int[]) v1, (int[]) v2);
1128 
1129             } else if (v1 instanceof double[]) {
1130                 MoreAsserts.assertEquals((double[]) v1, (double[]) v2);
1131 
1132             } else if (v1 instanceof String[]) {
1133                 MoreAsserts.assertEquals((String[]) v1, (String[]) v2);
1134 
1135             } else if (v1 instanceof Double) {
1136                 if (((Double) v1).isNaN()) {
1137                     assertTrue(((Double) v2).isNaN());
1138                 } else {
1139                     assertEquals(v1, v2);
1140                 }
1141 
1142             } else {
1143                 assertEquals(v1, v2);
1144             }
1145         }
1146     }
1147 
waitOnMainThread()1148     public static void waitOnMainThread() throws InterruptedException {
1149         final CountDownLatch latch = new CountDownLatch(1);
1150 
1151         new Handler(Looper.getMainLooper()).post(() -> latch.countDown());
1152 
1153         latch.await();
1154     }
1155 
1156     public static class LauncherCallbackAsserter {
1157         private final LauncherApps.Callback mCallback = mock(LauncherApps.Callback.class);
1158 
getMockCallback()1159         private Callback getMockCallback() {
1160             return mCallback;
1161         }
1162 
assertNoCallbackCalled()1163         public LauncherCallbackAsserter assertNoCallbackCalled() {
1164             verify(mCallback, times(0)).onShortcutsChanged(
1165                     anyString(),
1166                     any(List.class),
1167                     any(UserHandle.class));
1168             return this;
1169         }
1170 
assertNoCallbackCalledForPackage( String publisherPackageName)1171         public LauncherCallbackAsserter assertNoCallbackCalledForPackage(
1172                 String publisherPackageName) {
1173             verify(mCallback, times(0)).onShortcutsChanged(
1174                     eq(publisherPackageName),
1175                     any(List.class),
1176                     any(UserHandle.class));
1177             return this;
1178         }
1179 
assertNoCallbackCalledForPackageAndUser( String publisherPackageName, UserHandle publisherUserHandle)1180         public LauncherCallbackAsserter assertNoCallbackCalledForPackageAndUser(
1181                 String publisherPackageName, UserHandle publisherUserHandle) {
1182             verify(mCallback, times(0)).onShortcutsChanged(
1183                     eq(publisherPackageName),
1184                     any(List.class),
1185                     eq(publisherUserHandle));
1186             return this;
1187         }
1188 
assertCallbackCalledForPackageAndUser( String publisherPackageName, UserHandle publisherUserHandle)1189         public ShortcutListAsserter assertCallbackCalledForPackageAndUser(
1190                 String publisherPackageName, UserHandle publisherUserHandle) {
1191             final ArgumentCaptor<List> shortcuts = ArgumentCaptor.forClass(List.class);
1192             verify(mCallback, atLeastOnce()).onShortcutsChanged(
1193                     eq(publisherPackageName),
1194                     shortcuts.capture(),
1195                     eq(publisherUserHandle));
1196             return new ShortcutListAsserter(shortcuts.getValue());
1197         }
1198     }
1199 
assertForLauncherCallback( LauncherApps launcherApps, Runnable body)1200     public static LauncherCallbackAsserter assertForLauncherCallback(
1201             LauncherApps launcherApps, Runnable body) throws InterruptedException {
1202         final LauncherCallbackAsserter asserter = new LauncherCallbackAsserter();
1203         launcherApps.registerCallback(asserter.getMockCallback(),
1204                 new Handler(Looper.getMainLooper()));
1205 
1206         body.run();
1207 
1208         waitOnMainThread();
1209 
1210         // TODO unregister doesn't work well during unit tests.  Figure out and fix it.
1211         // launcherApps.unregisterCallback(asserter.getMockCallback());
1212 
1213         return asserter;
1214     }
1215 
assertForLauncherCallbackNoThrow( LauncherApps launcherApps, Runnable body)1216     public static LauncherCallbackAsserter assertForLauncherCallbackNoThrow(
1217             LauncherApps launcherApps, Runnable body) {
1218         try {
1219             return assertForLauncherCallback(launcherApps, body);
1220         } catch (InterruptedException e) {
1221             fail("Caught InterruptedException");
1222             return null; // Never happens.
1223         }
1224     }
1225 
retryUntil(BooleanSupplier checker, String message)1226     public static void retryUntil(BooleanSupplier checker, String message) {
1227         retryUntil(checker, message, 30);
1228     }
1229 
retryUntil(BooleanSupplier checker, String message, long timeoutSeconds)1230     public static void retryUntil(BooleanSupplier checker, String message, long timeoutSeconds) {
1231         final long timeOut = System.currentTimeMillis() + timeoutSeconds * 1000;
1232         while (!checker.getAsBoolean()) {
1233             if (System.currentTimeMillis() > timeOut) {
1234                 break;
1235             }
1236             try {
1237                 Thread.sleep(200);
1238             } catch (InterruptedException ignore) {
1239             }
1240         }
1241         assertTrue(message, checker.getAsBoolean());
1242     }
1243 }
1244