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 android.content.pm.cts.shortcuthost; 17 18 import static org.junit.Assert.assertNotNull; 19 import static org.junit.Assert.assertNull; 20 import static org.junit.Assert.assertTrue; 21 import static org.junit.Assert.fail; 22 23 import com.android.compatibility.common.tradefed.build.CompatibilityBuildHelper; 24 import com.android.ddmlib.testrunner.RemoteAndroidTestRunner; 25 import com.android.ddmlib.testrunner.TestResult.TestStatus; 26 import com.android.tradefed.device.DeviceNotAvailableException; 27 import com.android.tradefed.log.LogUtil.CLog; 28 import com.android.tradefed.result.CollectingTestListener; 29 import com.android.tradefed.result.TestDescription; 30 import com.android.tradefed.result.TestResult; 31 import com.android.tradefed.result.TestRunResult; 32 import com.android.tradefed.testtype.junit4.BaseHostJUnit4Test; 33 import com.android.tradefed.testtype.junit4.DeviceTestRunOptions; 34 35 import org.junit.After; 36 import org.junit.Before; 37 38 import java.io.FileNotFoundException; 39 import java.util.ArrayList; 40 import java.util.Collections; 41 import java.util.Map; 42 import java.util.regex.MatchResult; 43 import java.util.regex.Matcher; 44 import java.util.regex.Pattern; 45 46 import javax.annotation.Nullable; 47 48 abstract public class BaseShortcutManagerHostTest extends BaseHostJUnit4Test { 49 protected static final boolean DUMPSYS_IN_TEARDOWN = false; // DO NOT SUBMIT WITH TRUE 50 51 protected static final boolean NO_UNINSTALL_IN_TEARDOWN = false; // DO NOT SUBMIT WITH TRUE 52 53 private static final String RUNNER = "androidx.test.runner.AndroidJUnitRunner"; 54 55 protected boolean mIsMultiuserSupported; 56 protected boolean mIsManagedUserSupported; 57 58 private int mInitialUserId; 59 private ArrayList<Integer> mOriginalUsers; 60 61 @Before setUp()62 public void setUp() throws Exception { 63 assertNotNull(getBuild()); // ensure build has been set before test is run. 64 65 mIsMultiuserSupported = getDevice().isMultiUserSupported(); 66 if (!mIsMultiuserSupported) { 67 CLog.w("Multi user not supporeted"); 68 } 69 mIsManagedUserSupported = getDevice().hasFeature("android.software.managed_users"); 70 if (!mIsManagedUserSupported) { 71 CLog.w("Managed users not supporeted"); 72 } 73 74 if (mIsMultiuserSupported) { 75 mInitialUserId = getDevice().getCurrentUser(); 76 mOriginalUsers = new ArrayList<>(getDevice().listUsers()); 77 } 78 } 79 80 @After tearDown()81 public void tearDown() throws Exception { 82 removeTestUsers(); 83 } 84 dumpsys(String label)85 protected void dumpsys(String label) throws DeviceNotAvailableException { 86 CLog.w("dumpsys shortcuts #" + label); 87 88 CLog.w(getDevice().executeShellCommand("dumpsys shortcut")); 89 } 90 executeShellCommandWithLog(String command)91 protected String executeShellCommandWithLog(String command) throws DeviceNotAvailableException { 92 CLog.i("Executing command: " + command); 93 final String output = getDevice().executeShellCommand(command); 94 CLog.i(output); 95 return output; 96 } 97 clearShortcuts(String packageName, int userId)98 protected void clearShortcuts(String packageName, int userId) throws Exception { 99 assertContainsRegex("Success", 100 getDevice().executeShellCommand("cmd shortcut clear-shortcuts --user " + userId 101 + " " + packageName)); 102 } 103 installAppAsUser(String appFileName, int userId)104 protected void installAppAsUser(String appFileName, int userId) throws FileNotFoundException, 105 DeviceNotAvailableException { 106 CLog.i("Installing app " + appFileName + " for user " + userId); 107 CompatibilityBuildHelper buildHelper = new CompatibilityBuildHelper(getBuild()); 108 String result = getDevice().installPackageForUser( 109 buildHelper.getTestFile(appFileName), true, true, userId, "-t"); 110 assertNull("Failed to install " + appFileName + " for user " + userId + ": " + result, 111 result); 112 } 113 getPrimaryUserId()114 protected int getPrimaryUserId() throws DeviceNotAvailableException { 115 return getDevice().getPrimaryUserId(); 116 } 117 118 /** Returns true if the specified tests passed. Tests are run as given user. */ runDeviceTestsAsUser( String pkgName, @Nullable String testClassName, int userId)119 protected void runDeviceTestsAsUser( 120 String pkgName, @Nullable String testClassName, int userId) 121 throws DeviceNotAvailableException { 122 runDeviceTestsAsUser(pkgName, testClassName, null /*testMethodName*/, userId); 123 } 124 125 /** Returns true if the specified tests passed. Tests are run as given user. */ runDeviceTestsAsUser( String pkgName, @Nullable String testClassName, String testMethodName, int userId)126 protected void runDeviceTestsAsUser( 127 String pkgName, @Nullable String testClassName, String testMethodName, int userId) 128 throws DeviceNotAvailableException { 129 130 final DeviceTestRunOptions opts = new DeviceTestRunOptions(pkgName); 131 if (testClassName != null) { 132 if (testClassName.startsWith(".")) { 133 testClassName = pkgName + testClassName; 134 } 135 opts.setTestClassName(testClassName); 136 } 137 if (testMethodName != null) { 138 opts.setTestMethodName(testMethodName); 139 } 140 opts.setUserId(userId); 141 142 runDeviceTests(opts); 143 } 144 removeTestUsers()145 private void removeTestUsers() throws Exception { 146 if (!mIsMultiuserSupported) { 147 return; 148 } 149 getDevice().switchUser(mInitialUserId); 150 for (int userId : getDevice().listUsers()) { 151 if (!mOriginalUsers.contains(userId)) { 152 getDevice().removeUser(userId); 153 } 154 } 155 } 156 getOrCreateSecondaryUser()157 protected int getOrCreateSecondaryUser() throws Exception { 158 if (getDevice().isUserSecondary(mInitialUserId)) { 159 return mInitialUserId; 160 } 161 for (int userId : getDevice().listUsers()) { 162 if (getDevice().isUserSecondary(userId)) { 163 return userId; 164 } 165 } 166 return createUser(); 167 } 168 createUser()169 protected int createUser() throws Exception{ 170 return getDevice().createUser("TestUser_" + System.currentTimeMillis()); 171 } 172 createProfile(int parentUserId)173 protected int createProfile(int parentUserId) throws Exception{ 174 final String command = "pm create-user --profileOf " + parentUserId 175 + " --managed TestUser_" + System.currentTimeMillis(); 176 CLog.d("Starting command: " + command); 177 final String output = getDevice().executeShellCommand(command); 178 CLog.d("Output for command " + command + ": " + output); 179 180 if (output.startsWith("Success")) { 181 try { 182 return Integer.parseInt(output.substring(output.lastIndexOf(" ")).trim()); 183 } catch (NumberFormatException e) { 184 CLog.e("Failed to parse result: %s", output); 185 } 186 } else { 187 CLog.e("Failed to create user: %s", output); 188 } 189 throw new IllegalStateException(); 190 } 191 192 /** 193 * Variant of {@link #assertContainsRegex(String,String,String)} using a 194 * generic message. 195 */ assertContainsRegex( String expectedRegex, String actual)196 public MatchResult assertContainsRegex( 197 String expectedRegex, String actual) { 198 return assertContainsRegex(null, expectedRegex, actual); 199 } 200 201 /** 202 * Asserts that {@code expectedRegex} matches any substring of {@code actual} 203 * and fails with {@code message} if it does not. The Matcher is returned in 204 * case the test needs access to any captured groups. Note that you can also 205 * use this for a literal string, by wrapping your expected string in 206 * {@link Pattern#quote}. 207 */ assertContainsRegex( String message, String expectedRegex, String actual)208 public MatchResult assertContainsRegex( 209 String message, String expectedRegex, String actual) { 210 if (actual == null) { 211 failNotContains(message, expectedRegex, actual); 212 } 213 Matcher matcher = getMatcher(expectedRegex, actual); 214 if (!matcher.find()) { 215 failNotContains(message, expectedRegex, actual); 216 } 217 return matcher; 218 } 219 220 /** 221 * Asserts that {@code expectedRegex} does not exactly match {@code actual}, 222 * and fails with {@code message} if it does. Note that you can also use 223 * this for a literal string, by wrapping your expected string in 224 * {@link Pattern#quote}. 225 */ assertNotMatchesRegex( String message, String expectedRegex, String actual)226 public void assertNotMatchesRegex( 227 String message, String expectedRegex, String actual) { 228 Matcher matcher = getMatcher(expectedRegex, actual); 229 if (matcher.matches()) { 230 failMatch(message, expectedRegex, actual); 231 } 232 } 233 getMatcher(String expectedRegex, String actual)234 private Matcher getMatcher(String expectedRegex, String actual) { 235 Pattern pattern = Pattern.compile(expectedRegex); 236 return pattern.matcher(actual); 237 } 238 failMatch( String message, String expectedRegex, String actual)239 private void failMatch( 240 String message, String expectedRegex, String actual) { 241 failWithMessage(message, "expected not to match regex:<" + expectedRegex 242 + "> but was:<" + actual + '>'); 243 } 244 failWithMessage(String userMessage, String ourMessage)245 private void failWithMessage(String userMessage, String ourMessage) { 246 fail((userMessage == null) 247 ? ourMessage 248 : userMessage + ' ' + ourMessage); 249 } 250 failNotContains( String message, String expectedRegex, String actual)251 private void failNotContains( 252 String message, String expectedRegex, String actual) { 253 String actualDesc = (actual == null) ? "null" : ('<' + actual + '>'); 254 failWithMessage(message, "expected to contain regex:<" + expectedRegex 255 + "> but was:" + actualDesc); 256 } 257 waitForBroadcastIdle()258 protected void waitForBroadcastIdle() throws Exception { 259 runCommand("am wait-for-broadcast-idle"); 260 } 261 runCommand(String command)262 private String runCommand(String command) throws Exception { 263 return runCommand(command, "", true); 264 } 265 runCommand(String command, String expectedOutputPattern)266 private String runCommand(String command, String expectedOutputPattern) throws Exception { 267 return runCommand(command, expectedOutputPattern, true); 268 } 269 runCommandAndNotMatch(String command, String expectedOutputPattern)270 private String runCommandAndNotMatch(String command, String expectedOutputPattern) 271 throws Exception { 272 return runCommand(command, expectedOutputPattern, false); 273 } 274 runCommand(String command, String expectedOutputPattern, boolean shouldMatch)275 private String runCommand(String command, String expectedOutputPattern, 276 boolean shouldMatch) throws Exception { 277 CLog.d("Executing command: " + command); 278 final String output = getDevice().executeShellCommand(command); 279 280 CLog.d("Output:\n" 281 + "====================\n" 282 + output 283 + "===================="); 284 285 final Pattern pat = Pattern.compile( 286 expectedOutputPattern, Pattern.MULTILINE | Pattern.COMMENTS); 287 if (pat.matcher(output.trim()).find() != shouldMatch) { 288 fail("Output from \"" + command + "\" " 289 + (shouldMatch ? "didn't match" : "unexpectedly matched") 290 + " \"" + expectedOutputPattern + "\""); 291 } 292 return output; 293 } 294 } 295