/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.managedprovisioning.task; import static android.app.admin.DevicePolicyManager.ACTION_PROVISION_MANAGED_DEVICE; import static android.content.pm.PackageManager.INSTALL_ALLOW_TEST; import static android.content.pm.PackageManager.INSTALL_REPLACE_EXISTING; import static com.android.managedprovisioning.task.InstallPackageTask.ERROR_INSTALLATION_FAILED; import static com.android.managedprovisioning.task.InstallPackageTask.ERROR_PACKAGE_INVALID; import static org.mockito.ArgumentMatchers.anyLong; import static org.mockito.Matchers.any; import static org.mockito.Matchers.anyInt; import static org.mockito.Matchers.anyString; import static org.mockito.Matchers.eq; import static org.mockito.Mockito.never; import static org.mockito.Mockito.timeout; import static org.mockito.Mockito.verify; import static org.mockito.Mockito.verifyNoMoreInteractions; import static org.mockito.Mockito.when; import android.app.admin.DevicePolicyManager; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.IntentSender; import android.content.pm.PackageInstaller; import android.content.pm.PackageManager; import android.os.Process; import android.os.UserHandle; import android.test.AndroidTestCase; import android.test.suitebuilder.annotation.SmallTest; import com.android.managedprovisioning.model.ProvisioningParams; import org.mockito.ArgumentCaptor; import org.mockito.Mock; import org.mockito.MockitoAnnotations; import org.mockito.stubbing.Answer; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStream; import java.util.Arrays; public class InstallPackageTaskTest extends AndroidTestCase { private static final String TEST_PACKAGE_NAME = "com.android.test"; private static final String OTHER_PACKAGE_NAME = "com.android.other"; private static final ProvisioningParams TEST_PARAMS = new ProvisioningParams.Builder() .setDeviceAdminPackageName(TEST_PACKAGE_NAME) .setProvisioningAction(ACTION_PROVISION_MANAGED_DEVICE) .build(); private static final int TEST_USER_ID = 123; private static final byte[] APK_CONTENT = new byte[]{'t', 'e', 's', 't'}; private static final long TIMEOUT = 10000; private static int sSessionId = 0; @Mock private Context mMockContext; @Mock private PackageManager mPackageManager; @Mock private PackageInstaller mPackageInstaller; @Mock private PackageInstaller.Session mSession; @Mock private OutputStream mSessionWriteStream; @Mock private DevicePolicyManager mDpm; @Mock private AbstractProvisioningTask.Callback mCallback; @Mock private DownloadPackageTask mDownloadPackageTask; private InstallPackageTask mTask; private String mTestPackageLocation; @Override protected void setUp() throws Exception { super.setUp(); // this is necessary for mockito to work System.setProperty("dexmaker.dexcache", getContext().getCacheDir().toString()); MockitoAnnotations.initMocks(this); when(mMockContext.getPackageManager()).thenReturn(mPackageManager); when(mMockContext.getPackageName()).thenReturn(getContext().getPackageName()); when(mPackageManager.getPackageInstaller()).thenReturn(mPackageInstaller); when(mPackageInstaller.createSession(any(PackageInstaller.SessionParams.class))).thenAnswer( (Answer) invocation -> sSessionId++); when(mPackageInstaller.openSession(anyInt())).thenReturn(mSession); when(mSession.openWrite(anyString(), anyLong(), anyLong())).thenReturn(mSessionWriteStream); when(mMockContext.registerReceiver(any(BroadcastReceiver.class), any(IntentFilter.class))).thenAnswer( (Answer) invocation -> (Intent) getContext().registerReceiver( invocation.getArgument(0), invocation.getArgument(1))); when(mMockContext.getSystemServiceName(eq(DevicePolicyManager.class))) .thenReturn(Context.DEVICE_POLICY_SERVICE); when(mMockContext.getSystemService(eq(Context.DEVICE_POLICY_SERVICE))).thenReturn(mDpm); when(mMockContext.getUser()).thenReturn(Process.myUserHandle()); when(mMockContext.getUserId()).thenReturn(UserHandle.myUserId()); mTestPackageLocation = File.createTempFile("test", "apk").getPath(); try (FileOutputStream out = new FileOutputStream(mTestPackageLocation)) { out.write(APK_CONTENT); } mTask = new InstallPackageTask(mDownloadPackageTask, mMockContext, TEST_PARAMS, mCallback); } @SmallTest public void testNoDownloadLocation() { // GIVEN no package was downloaded when(mDownloadPackageTask.getDownloadedPackageLocation()).thenReturn(null); // WHEN running the InstallPackageTask without specifying an install location mTask.run(TEST_USER_ID); // THEN no package is installed, but we get a success callback verify(mPackageManager, never()).getPackageInstaller(); verify(mCallback).onSuccess(mTask); verifyNoMoreInteractions(mCallback); } @SmallTest public void testSuccess() throws Exception { // GIVEN a package was downloaded to TEST_LOCATION when(mDownloadPackageTask.getDownloadedPackageLocation()).thenReturn(mTestPackageLocation); // WHEN running the InstallPackageTask specifying an install location mTask.run(TEST_USER_ID); // THEN package installed is invoked with an install observer IntentSender observer = verifyPackageInstalled(INSTALL_REPLACE_EXISTING); // WHEN the package installed callback is invoked with success Intent fillIn = new Intent(); fillIn.putExtra(PackageInstaller.EXTRA_PACKAGE_NAME, TEST_PACKAGE_NAME); fillIn.putExtra(PackageInstaller.EXTRA_STATUS, PackageInstaller.STATUS_SUCCESS); observer.sendIntent(getContext(), 0, fillIn, null, null); // THEN we receive a success callback verify(mCallback, timeout(TIMEOUT)).onSuccess(mTask); verifyNoMoreInteractions(mCallback); } @SmallTest public void testSuccess_allowTestOnly() throws Exception { // GIVEN a package was downloaded to TEST_LOCATION when(mDownloadPackageTask.getDownloadedPackageLocation()).thenReturn(mTestPackageLocation); // WHEN package to be installed is the current device owner. when(mDpm.isDeviceOwnerApp(eq(TEST_PACKAGE_NAME))).thenReturn(true); // WHEN running the InstallPackageTask specifying an install location mTask.run(TEST_USER_ID); // THEN package installed is invoked with an install observer IntentSender observer = verifyPackageInstalled( INSTALL_REPLACE_EXISTING | INSTALL_ALLOW_TEST); // WHEN the package installed callback is invoked with success Intent fillIn = new Intent(); fillIn.putExtra(PackageInstaller.EXTRA_PACKAGE_NAME, TEST_PACKAGE_NAME); fillIn.putExtra(PackageInstaller.EXTRA_STATUS, PackageInstaller.STATUS_SUCCESS); observer.sendIntent(getContext(), 0, fillIn, null, null); // THEN we receive a success callback verify(mCallback, timeout(TIMEOUT)).onSuccess(mTask); verifyNoMoreInteractions(mCallback); } @SmallTest public void testInstallFailedVersionDowngrade() throws Exception { // GIVEN a package was downloaded to TEST_LOCATION when(mDownloadPackageTask.getDownloadedPackageLocation()).thenReturn(mTestPackageLocation); // WHEN running the InstallPackageTask with a package already at a higher version mTask.run(TEST_USER_ID); // THEN package installed is invoked with an install observer IntentSender observer = verifyPackageInstalled(INSTALL_REPLACE_EXISTING); // WHEN the package installed callback is invoked with version downgrade error Intent fillIn = new Intent(); fillIn.putExtra(PackageInstaller.EXTRA_PACKAGE_NAME, TEST_PACKAGE_NAME); fillIn.putExtra(PackageInstaller.EXTRA_STATUS, PackageInstaller.STATUS_FAILURE); fillIn.putExtra(PackageInstaller.EXTRA_LEGACY_STATUS, PackageManager.INSTALL_FAILED_VERSION_DOWNGRADE); observer.sendIntent(getContext(), 0, fillIn, null, null); // THEN we get a success callback, because an existing version of the DPC is present verify(mCallback, timeout(TIMEOUT)).onSuccess(mTask); verifyNoMoreInteractions(mCallback); } @SmallTest public void testInstallFailedOtherError() throws Exception { // GIVEN a package was downloaded to TEST_LOCATION when(mDownloadPackageTask.getDownloadedPackageLocation()).thenReturn(mTestPackageLocation); // WHEN running the InstallPackageTask with a package already at a higher version mTask.run(TEST_USER_ID); // THEN package installed is invoked with an install observer IntentSender observer = verifyPackageInstalled(INSTALL_REPLACE_EXISTING); // WHEN the package installed callback is invoked with version invalid apk error Intent fillIn = new Intent(); fillIn.putExtra(PackageInstaller.EXTRA_PACKAGE_NAME, TEST_PACKAGE_NAME); fillIn.putExtra(PackageInstaller.EXTRA_STATUS, PackageInstaller.STATUS_FAILURE); fillIn.putExtra(PackageInstaller.EXTRA_LEGACY_STATUS, PackageManager.INSTALL_FAILED_INVALID_APK); observer.sendIntent(getContext(), 0, fillIn, null, null); // THEN we get a success callback, because an existing version of the DPC is present verify(mCallback, timeout(TIMEOUT)).onError(mTask, ERROR_INSTALLATION_FAILED); verifyNoMoreInteractions(mCallback); } @SmallTest public void testDifferentPackageName() throws Exception { // GIVEN a package was downloaded to TEST_LOCATION when(mDownloadPackageTask.getDownloadedPackageLocation()).thenReturn(mTestPackageLocation); // WHEN running the InstallPackageTask with a package already at a higher version mTask.run(TEST_USER_ID); // THEN package installed is invoked with an install observer IntentSender observer = verifyPackageInstalled(INSTALL_REPLACE_EXISTING); // WHEN the package installed callback is invoked with the wrong package Intent fillIn = new Intent(); fillIn.putExtra(PackageInstaller.EXTRA_PACKAGE_NAME, OTHER_PACKAGE_NAME); observer.sendIntent(getContext(), 0, fillIn, null, null); // THEN we get a success callback, because the wrong package name verify(mCallback, timeout(TIMEOUT)).onError(mTask, ERROR_PACKAGE_INVALID); verifyNoMoreInteractions(mCallback); } private IntentSender verifyPackageInstalled(int installFlags) throws IOException { ArgumentCaptor paramsCaptor = ArgumentCaptor.forClass(PackageInstaller.SessionParams.class); ArgumentCaptor fileContentCaptor = ArgumentCaptor.forClass(byte[].class); verify(mPackageInstaller).createSession(paramsCaptor.capture()); assertEquals(installFlags, paramsCaptor.getValue().installFlags); verify(mSessionWriteStream).write(fileContentCaptor.capture(), eq(0), eq(APK_CONTENT.length)); assertTrue(Arrays.equals(APK_CONTENT, Arrays.copyOf(fileContentCaptor.getValue(), APK_CONTENT.length))); ArgumentCaptor intentSenderCaptor = ArgumentCaptor.forClass(IntentSender.class); // THEN package installation was started and we will receive a status callback verify(mSession).commit(intentSenderCaptor.capture()); return intentSenderCaptor.getValue(); } }