| /* |
| * Copyright (C) 2006 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.unit_tests; |
| |
| import com.android.internal.content.PackageHelper; |
| |
| import android.os.storage.IMountService.Stub; |
| |
| import android.net.Uri; |
| import android.os.FileUtils; |
| |
| import java.io.File; |
| import java.io.FileInputStream; |
| import java.io.FileNotFoundException; |
| import java.io.FileOutputStream; |
| import java.io.IOException; |
| import java.io.InputStream; |
| |
| import android.app.PendingIntent; |
| import android.content.BroadcastReceiver; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.IntentFilter; |
| import android.content.pm.ApplicationInfo; |
| import android.content.pm.IPackageDataObserver; |
| import android.content.pm.IPackageInstallObserver; |
| import android.content.pm.IPackageDeleteObserver; |
| import android.content.pm.IPackageMoveObserver; |
| import android.content.pm.PackageInfo; |
| import android.content.pm.PackageManager; |
| import android.content.pm.PackageParser; |
| import android.content.pm.PackageStats; |
| import android.content.pm.IPackageManager; |
| import android.content.pm.PackageManager.NameNotFoundException; |
| import android.content.res.Resources; |
| import android.content.res.Resources.NotFoundException; |
| import android.test.AndroidTestCase; |
| import android.test.suitebuilder.annotation.LargeTest; |
| import android.test.suitebuilder.annotation.MediumTest; |
| import android.test.suitebuilder.annotation.SmallTest; |
| import android.test.suitebuilder.annotation.Suppress; |
| import android.util.DisplayMetrics; |
| import android.util.Log; |
| import android.os.Environment; |
| import android.os.Handler; |
| import android.os.IBinder; |
| import android.os.storage.IMountService; |
| import android.os.storage.IMountServiceListener; |
| import android.os.storage.StorageEventListener; |
| import android.os.storage.StorageManager; |
| import android.os.storage.StorageResultCode; |
| import android.os.RemoteException; |
| import android.os.ServiceManager; |
| import android.os.StatFs; |
| import android.provider.Settings; |
| import android.provider.Settings.SettingNotFoundException; |
| |
| public class PackageManagerTests extends AndroidTestCase { |
| private static final boolean localLOGV = true; |
| public static final String TAG="PackageManagerTests"; |
| public final long MAX_WAIT_TIME=120*1000; |
| public final long WAIT_TIME_INCR=20*1000; |
| private static final String SECURE_CONTAINERS_PREFIX = "/mnt/asec"; |
| private static final int APP_INSTALL_AUTO = PackageHelper.APP_INSTALL_AUTO; |
| private static final int APP_INSTALL_DEVICE = PackageHelper.APP_INSTALL_INTERNAL; |
| private static final int APP_INSTALL_SDCARD = PackageHelper.APP_INSTALL_EXTERNAL; |
| |
| void failStr(String errMsg) { |
| Log.w(TAG, "errMsg="+errMsg); |
| fail(errMsg); |
| } |
| void failStr(Exception e) { |
| Log.w(TAG, "e.getMessage="+e.getMessage()); |
| Log.w(TAG, "e="+e); |
| } |
| |
| @Override |
| protected void setUp() throws Exception { |
| super.setUp(); |
| } |
| |
| private class PackageInstallObserver extends IPackageInstallObserver.Stub { |
| public int returnCode; |
| private boolean doneFlag = false; |
| |
| public void packageInstalled(String packageName, int returnCode) { |
| synchronized(this) { |
| this.returnCode = returnCode; |
| doneFlag = true; |
| notifyAll(); |
| } |
| } |
| |
| public boolean isDone() { |
| return doneFlag; |
| } |
| } |
| |
| abstract class GenericReceiver extends BroadcastReceiver { |
| private boolean doneFlag = false; |
| boolean received = false; |
| Intent intent; |
| IntentFilter filter; |
| abstract boolean notifyNow(Intent intent); |
| @Override |
| public void onReceive(Context context, Intent intent) { |
| if (notifyNow(intent)) { |
| synchronized (this) { |
| received = true; |
| doneFlag = true; |
| this.intent = intent; |
| notifyAll(); |
| } |
| } |
| } |
| |
| public boolean isDone() { |
| return doneFlag; |
| } |
| |
| public void setFilter(IntentFilter filter) { |
| this.filter = filter; |
| } |
| } |
| |
| class InstallReceiver extends GenericReceiver { |
| String pkgName; |
| |
| InstallReceiver(String pkgName) { |
| this.pkgName = pkgName; |
| IntentFilter filter = new IntentFilter(Intent.ACTION_PACKAGE_ADDED); |
| filter.addDataScheme("package"); |
| super.setFilter(filter); |
| } |
| |
| public boolean notifyNow(Intent intent) { |
| String action = intent.getAction(); |
| if (!Intent.ACTION_PACKAGE_ADDED.equals(action)) { |
| return false; |
| } |
| Uri data = intent.getData(); |
| String installedPkg = data.getEncodedSchemeSpecificPart(); |
| if (pkgName.equals(installedPkg)) { |
| return true; |
| } |
| return false; |
| } |
| } |
| |
| PackageManager getPm() { |
| return mContext.getPackageManager(); |
| } |
| |
| public boolean invokeInstallPackage(Uri packageURI, int flags, |
| final String pkgName, GenericReceiver receiver) throws Exception { |
| PackageInstallObserver observer = new PackageInstallObserver(); |
| final boolean received = false; |
| mContext.registerReceiver(receiver, receiver.filter); |
| final boolean DEBUG = true; |
| try { |
| // Wait on observer |
| synchronized(observer) { |
| synchronized (receiver) { |
| getPm().installPackage(packageURI, observer, flags, null); |
| long waitTime = 0; |
| while((!observer.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| observer.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!observer.isDone()) { |
| throw new Exception("Timed out waiting for packageInstalled callback"); |
| } |
| if (observer.returnCode != PackageManager.INSTALL_SUCCEEDED) { |
| Log.i(TAG, "Failed to install with error code = " + observer.returnCode); |
| return false; |
| } |
| // Verify we received the broadcast |
| waitTime = 0; |
| while((!receiver.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| receiver.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!receiver.isDone()) { |
| throw new Exception("Timed out waiting for PACKAGE_ADDED notification"); |
| } |
| return receiver.received; |
| } |
| } |
| } finally { |
| mContext.unregisterReceiver(receiver); |
| } |
| } |
| |
| public boolean invokeInstallPackageFail(Uri packageURI, int flags, |
| final String pkgName, int result) throws Exception { |
| PackageInstallObserver observer = new PackageInstallObserver(); |
| try { |
| // Wait on observer |
| synchronized(observer) { |
| getPm().installPackage(packageURI, observer, flags, null); |
| long waitTime = 0; |
| while((!observer.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| observer.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!observer.isDone()) { |
| throw new Exception("Timed out waiting for packageInstalled callback"); |
| } |
| return (observer.returnCode == result); |
| } |
| } finally { |
| } |
| } |
| |
| Uri getInstallablePackage(int fileResId, File outFile) { |
| Resources res = mContext.getResources(); |
| InputStream is = null; |
| try { |
| is = res.openRawResource(fileResId); |
| } catch (NotFoundException e) { |
| failStr("Failed to load resource with id: " + fileResId); |
| } |
| FileUtils.setPermissions(outFile.getPath(), |
| FileUtils.S_IRWXU | FileUtils.S_IRWXG | FileUtils.S_IRWXO, |
| -1, -1); |
| assertTrue(FileUtils.copyToFile(is, outFile)); |
| FileUtils.setPermissions(outFile.getPath(), |
| FileUtils.S_IRWXU | FileUtils.S_IRWXG | FileUtils.S_IRWXO, |
| -1, -1); |
| return Uri.fromFile(outFile); |
| } |
| |
| private PackageParser.Package parsePackage(Uri packageURI) { |
| final String archiveFilePath = packageURI.getPath(); |
| PackageParser packageParser = new PackageParser(archiveFilePath); |
| File sourceFile = new File(archiveFilePath); |
| DisplayMetrics metrics = new DisplayMetrics(); |
| metrics.setToDefaults(); |
| PackageParser.Package pkg = packageParser.parsePackage(sourceFile, archiveFilePath, metrics, 0); |
| packageParser = null; |
| return pkg; |
| } |
| private boolean getInstallLoc(int flags, int expInstallLocation) { |
| // Flags explicitly over ride everything else. |
| if ((flags & PackageManager.INSTALL_FORWARD_LOCK) != 0 ) { |
| return false; |
| } else if ((flags & PackageManager.INSTALL_EXTERNAL) != 0 ) { |
| return true; |
| } |
| // Manifest option takes precedence next |
| if (expInstallLocation == PackageInfo.INSTALL_LOCATION_PREFER_EXTERNAL) { |
| return true; |
| } |
| if (expInstallLocation == PackageInfo.INSTALL_LOCATION_INTERNAL_ONLY) { |
| return false; |
| } |
| // TODO Out of memory checks here. |
| boolean checkSd = false; |
| int setLoc = 0; |
| try { |
| setLoc = Settings.System.getInt(mContext.getContentResolver(), Settings.System.SET_INSTALL_LOCATION); |
| } catch (SettingNotFoundException e) { |
| failStr(e); |
| } |
| if (setLoc == 1) { |
| int userPref = APP_INSTALL_AUTO; |
| try { |
| userPref = Settings.System.getInt(mContext.getContentResolver(), Settings.System.DEFAULT_INSTALL_LOCATION); |
| } catch (SettingNotFoundException e) { |
| failStr(e); |
| } |
| if (userPref == APP_INSTALL_DEVICE) { |
| checkSd = false; |
| } else if (userPref == APP_INSTALL_SDCARD) { |
| checkSd = true; |
| } else if (userPref == APP_INSTALL_AUTO) { |
| // Might be determined dynamically. TODO fix this |
| checkSd = false; |
| } |
| } |
| return checkSd; |
| } |
| private void assertInstall(PackageParser.Package pkg, int flags, int expInstallLocation) { |
| try { |
| String pkgName = pkg.packageName; |
| ApplicationInfo info = getPm().getApplicationInfo(pkgName, 0); |
| assertNotNull(info); |
| assertEquals(pkgName, info.packageName); |
| File dataDir = Environment.getDataDirectory(); |
| String appInstallPath = new File(dataDir, "app").getPath(); |
| String drmInstallPath = new File(dataDir, "app-private").getPath(); |
| File srcDir = new File(info.sourceDir); |
| String srcPath = srcDir.getParent(); |
| File publicSrcDir = new File(info.publicSourceDir); |
| String publicSrcPath = publicSrcDir.getParent(); |
| |
| if ((flags & PackageManager.INSTALL_FORWARD_LOCK) != 0) { |
| assertTrue((info.flags & ApplicationInfo.FLAG_FORWARD_LOCK) != 0); |
| assertEquals(srcPath, drmInstallPath); |
| assertEquals(publicSrcPath, appInstallPath); |
| } else { |
| assertFalse((info.flags & ApplicationInfo.FLAG_FORWARD_LOCK) != 0); |
| if (!getInstallLoc(flags, expInstallLocation)) { |
| assertEquals(srcPath, appInstallPath); |
| assertEquals(publicSrcPath, appInstallPath); |
| assertFalse((info.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0); |
| } else { |
| assertTrue((info.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0); |
| assertTrue(srcPath.startsWith(SECURE_CONTAINERS_PREFIX)); |
| assertTrue(publicSrcPath.startsWith(SECURE_CONTAINERS_PREFIX)); |
| } |
| } |
| } catch (NameNotFoundException e) { |
| failStr("failed with exception : " + e); |
| } |
| } |
| private void assertNotInstalled(String pkgName) { |
| try { |
| ApplicationInfo info = getPm().getApplicationInfo(pkgName, 0); |
| fail(pkgName + " shouldnt be installed"); |
| } catch (NameNotFoundException e) { |
| } |
| } |
| |
| class InstallParams { |
| String outFileName; |
| Uri packageURI; |
| PackageParser.Package pkg; |
| InstallParams(PackageParser.Package pkg, String outFileName, Uri packageURI) { |
| this.outFileName = outFileName; |
| this.packageURI = packageURI; |
| this.pkg = pkg; |
| } |
| } |
| |
| private InstallParams sampleInstallFromRawResource(int flags, boolean cleanUp) { |
| return installFromRawResource("install.apk", R.raw.install, flags, cleanUp, |
| false, -1, PackageInfo.INSTALL_LOCATION_AUTO); |
| } |
| |
| public void clearSecureContainersForPkg(String pkgName) { |
| IMountService ms = getMs(); |
| try { |
| String list[] = ms.getSecureContainerList(); |
| if (list != null) { |
| for (String cid : list) { |
| boolean delete = false; |
| // STOPSHIP issues with rename should be fixed. |
| if (cid.contains(pkgName) || |
| cid.contains("smdltmp")) { |
| Log.i(TAG, "Destroying container " + cid); |
| ms.destroySecureContainer(cid, true); |
| } |
| } |
| } |
| } catch (RemoteException e) {} |
| } |
| |
| /* |
| * Utility function that reads a apk bundled as a raw resource |
| * copies it into own data directory and invokes |
| * PackageManager api to install it. |
| */ |
| private InstallParams installFromRawResource(String outFileName, |
| int rawResId, int flags, boolean cleanUp, boolean fail, int result, |
| int expInstallLocation) { |
| File filesDir = mContext.getFilesDir(); |
| File outFile = new File(filesDir, outFileName); |
| Uri packageURI = getInstallablePackage(rawResId, outFile); |
| PackageParser.Package pkg = parsePackage(packageURI); |
| assertNotNull(pkg); |
| InstallParams ip = null; |
| // Make sure the package doesn't exist |
| getPm().deletePackage(pkg.packageName, null, 0); |
| // Clean up the containers as well |
| clearSecureContainersForPkg(pkg.packageName); |
| try { |
| try { |
| if (fail) { |
| assertTrue(invokeInstallPackageFail(packageURI, flags, |
| pkg.packageName, result)); |
| assertNotInstalled(pkg.packageName); |
| } else { |
| InstallReceiver receiver = new InstallReceiver(pkg.packageName); |
| assertTrue(invokeInstallPackage(packageURI, flags, |
| pkg.packageName, receiver)); |
| // Verify installed information |
| assertInstall(pkg, flags, expInstallLocation); |
| ip = new InstallParams(pkg, outFileName, packageURI); |
| } |
| } catch (Exception e) { |
| failStr("Failed with exception : " + e); |
| } |
| return ip; |
| } finally { |
| if (cleanUp) { |
| cleanUpInstall(ip); |
| } |
| } |
| } |
| |
| @MediumTest |
| public void testInstallNormalInternal() { |
| sampleInstallFromRawResource(0, true); |
| } |
| |
| @MediumTest |
| public void testInstallFwdLockedInternal() { |
| sampleInstallFromRawResource(PackageManager.INSTALL_FORWARD_LOCK, true); |
| } |
| |
| @MediumTest |
| public void testInstallSdcard() { |
| sampleInstallFromRawResource(PackageManager.INSTALL_EXTERNAL, true); |
| } |
| |
| /* ------------------------- Test replacing packages --------------*/ |
| class ReplaceReceiver extends GenericReceiver { |
| String pkgName; |
| final static int INVALID = -1; |
| final static int REMOVED = 1; |
| final static int ADDED = 2; |
| final static int REPLACED = 3; |
| int removed = INVALID; |
| // for updated system apps only |
| boolean update = false; |
| |
| ReplaceReceiver(String pkgName) { |
| this.pkgName = pkgName; |
| filter = new IntentFilter(Intent.ACTION_PACKAGE_REMOVED); |
| filter.addAction(Intent.ACTION_PACKAGE_ADDED); |
| if (update) { |
| filter.addAction(Intent.ACTION_PACKAGE_REPLACED); |
| } |
| filter.addDataScheme("package"); |
| super.setFilter(filter); |
| } |
| |
| public boolean notifyNow(Intent intent) { |
| String action = intent.getAction(); |
| Uri data = intent.getData(); |
| String installedPkg = data.getEncodedSchemeSpecificPart(); |
| if (pkgName == null || !pkgName.equals(installedPkg)) { |
| return false; |
| } |
| if (Intent.ACTION_PACKAGE_REMOVED.equals(action)) { |
| removed = REMOVED; |
| } else if (Intent.ACTION_PACKAGE_ADDED.equals(action)) { |
| if (removed != REMOVED) { |
| return false; |
| } |
| boolean replacing = intent.getBooleanExtra(Intent.EXTRA_REPLACING, false); |
| if (!replacing) { |
| return false; |
| } |
| removed = ADDED; |
| if (!update) { |
| return true; |
| } |
| } else if (Intent.ACTION_PACKAGE_REPLACED.equals(action)) { |
| if (removed != ADDED) { |
| return false; |
| } |
| removed = REPLACED; |
| return true; |
| } |
| return false; |
| } |
| } |
| |
| /* |
| * Utility function that reads a apk bundled as a raw resource |
| * copies it into own data directory and invokes |
| * PackageManager api to install first and then replace it |
| * again. |
| */ |
| public void replaceFromRawResource(int flags) { |
| InstallParams ip = sampleInstallFromRawResource(flags, false); |
| boolean replace = ((flags & PackageManager.INSTALL_REPLACE_EXISTING) != 0); |
| Log.i(TAG, "replace=" + replace); |
| GenericReceiver receiver; |
| if (replace) { |
| receiver = new ReplaceReceiver(ip.pkg.packageName); |
| Log.i(TAG, "Creating replaceReceiver"); |
| } else { |
| receiver = new InstallReceiver(ip.pkg.packageName); |
| } |
| try { |
| try { |
| assertEquals(invokeInstallPackage(ip.packageURI, flags, |
| ip.pkg.packageName, receiver), replace); |
| if (replace) { |
| assertInstall(ip.pkg, flags, ip.pkg.installLocation); |
| } |
| } catch (Exception e) { |
| failStr("Failed with exception : " + e); |
| } |
| } finally { |
| cleanUpInstall(ip); |
| } |
| } |
| |
| @MediumTest |
| public void testReplaceFailNormalInternal() { |
| replaceFromRawResource(0); |
| } |
| |
| @MediumTest |
| public void testReplaceFailFwdLockedInternal() { |
| replaceFromRawResource(PackageManager.INSTALL_FORWARD_LOCK); |
| } |
| |
| @MediumTest |
| public void testReplaceFailSdcard() { |
| replaceFromRawResource(PackageManager.INSTALL_EXTERNAL); |
| } |
| |
| @MediumTest |
| public void testReplaceNormalInternal() { |
| replaceFromRawResource(PackageManager.INSTALL_REPLACE_EXISTING); |
| } |
| |
| @MediumTest |
| public void testReplaceFwdLockedInternal() { |
| replaceFromRawResource(PackageManager.INSTALL_REPLACE_EXISTING | |
| PackageManager.INSTALL_FORWARD_LOCK); |
| } |
| |
| @MediumTest |
| public void testReplaceSdcard() { |
| replaceFromRawResource(PackageManager.INSTALL_REPLACE_EXISTING | |
| PackageManager.INSTALL_EXTERNAL); |
| } |
| |
| /* -------------- Delete tests ---*/ |
| class DeleteObserver extends IPackageDeleteObserver.Stub { |
| |
| public boolean succeeded; |
| private boolean doneFlag = false; |
| |
| public boolean isDone() { |
| return doneFlag; |
| } |
| |
| public void packageDeleted(boolean succeeded) throws RemoteException { |
| synchronized(this) { |
| this.succeeded = succeeded; |
| doneFlag = true; |
| notifyAll(); |
| } |
| } |
| } |
| |
| class DeleteReceiver extends GenericReceiver { |
| String pkgName; |
| |
| DeleteReceiver(String pkgName) { |
| this.pkgName = pkgName; |
| IntentFilter filter = new IntentFilter(Intent.ACTION_PACKAGE_REMOVED); |
| filter.addDataScheme("package"); |
| super.setFilter(filter); |
| } |
| |
| public boolean notifyNow(Intent intent) { |
| String action = intent.getAction(); |
| if (!Intent.ACTION_PACKAGE_REMOVED.equals(action)) { |
| return false; |
| } |
| Uri data = intent.getData(); |
| String installedPkg = data.getEncodedSchemeSpecificPart(); |
| if (pkgName.equals(installedPkg)) { |
| return true; |
| } |
| return false; |
| } |
| } |
| |
| public boolean invokeDeletePackage(Uri packageURI, int flags, |
| final String pkgName, GenericReceiver receiver) throws Exception { |
| DeleteObserver observer = new DeleteObserver(); |
| final boolean received = false; |
| mContext.registerReceiver(receiver, receiver.filter); |
| try { |
| // Wait on observer |
| synchronized(observer) { |
| synchronized (receiver) { |
| getPm().deletePackage(pkgName, observer, flags); |
| long waitTime = 0; |
| while((!observer.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| observer.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!observer.isDone()) { |
| throw new Exception("Timed out waiting for packageInstalled callback"); |
| } |
| // Verify we received the broadcast |
| waitTime = 0; |
| while((!receiver.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| receiver.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!receiver.isDone()) { |
| throw new Exception("Timed out waiting for PACKAGE_ADDED notification"); |
| } |
| return receiver.received; |
| } |
| } |
| } finally { |
| mContext.unregisterReceiver(receiver); |
| } |
| } |
| |
| public void deleteFromRawResource(int iFlags, int dFlags) { |
| InstallParams ip = sampleInstallFromRawResource(iFlags, false); |
| boolean retainData = ((dFlags & PackageManager.DONT_DELETE_DATA) != 0); |
| GenericReceiver receiver = new DeleteReceiver(ip.pkg.packageName); |
| DeleteObserver observer = new DeleteObserver(); |
| try { |
| assertTrue(invokeDeletePackage(ip.packageURI, dFlags, |
| ip.pkg.packageName, receiver)); |
| ApplicationInfo info = null; |
| Log.i(TAG, "okay4"); |
| try { |
| info = getPm().getApplicationInfo(ip.pkg.packageName, |
| PackageManager.GET_UNINSTALLED_PACKAGES); |
| } catch (NameNotFoundException e) { |
| info = null; |
| } |
| if (retainData) { |
| assertNotNull(info); |
| assertEquals(info.packageName, ip.pkg.packageName); |
| File file = new File(info.dataDir); |
| assertTrue(file.exists()); |
| } else { |
| assertNull(info); |
| } |
| } catch (Exception e) { |
| failStr(e); |
| } finally { |
| cleanUpInstall(ip); |
| } |
| } |
| |
| @MediumTest |
| public void testDeleteNormalInternal() { |
| deleteFromRawResource(0, 0); |
| } |
| |
| @MediumTest |
| public void testDeleteFwdLockedInternal() { |
| deleteFromRawResource(PackageManager.INSTALL_FORWARD_LOCK, 0); |
| } |
| |
| @MediumTest |
| public void testDeleteSdcard() { |
| deleteFromRawResource(PackageManager.INSTALL_EXTERNAL, 0); |
| } |
| |
| @MediumTest |
| public void testDeleteNormalInternalRetainData() { |
| deleteFromRawResource(0, PackageManager.DONT_DELETE_DATA); |
| } |
| |
| @MediumTest |
| public void testDeleteFwdLockedInternalRetainData() { |
| deleteFromRawResource(PackageManager.INSTALL_FORWARD_LOCK, PackageManager.DONT_DELETE_DATA); |
| } |
| |
| @MediumTest |
| public void testDeleteSdcardRetainData() { |
| deleteFromRawResource(PackageManager.INSTALL_EXTERNAL, PackageManager.DONT_DELETE_DATA); |
| } |
| |
| /* sdcard mount/unmount tests ******/ |
| |
| class SdMountReceiver extends GenericReceiver { |
| String pkgNames[]; |
| boolean status = true; |
| |
| SdMountReceiver(String[] pkgNames) { |
| this.pkgNames = pkgNames; |
| IntentFilter filter = new IntentFilter(Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE); |
| super.setFilter(filter); |
| } |
| |
| public boolean notifyNow(Intent intent) { |
| Log.i(TAG, "okay 1"); |
| String action = intent.getAction(); |
| if (!Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE.equals(action)) { |
| return false; |
| } |
| String rpkgList[] = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST); |
| for (String pkg : pkgNames) { |
| boolean found = false; |
| for (String rpkg : rpkgList) { |
| if (rpkg.equals(pkg)) { |
| found = true; |
| break; |
| } |
| } |
| if (!found) { |
| status = false; |
| return true; |
| } |
| } |
| return true; |
| } |
| } |
| |
| class SdUnMountReceiver extends GenericReceiver { |
| String pkgNames[]; |
| boolean status = true; |
| |
| SdUnMountReceiver(String[] pkgNames) { |
| this.pkgNames = pkgNames; |
| IntentFilter filter = new IntentFilter(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE); |
| super.setFilter(filter); |
| } |
| |
| public boolean notifyNow(Intent intent) { |
| String action = intent.getAction(); |
| if (!Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE.equals(action)) { |
| return false; |
| } |
| String rpkgList[] = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST); |
| for (String pkg : pkgNames) { |
| boolean found = false; |
| for (String rpkg : rpkgList) { |
| if (rpkg.equals(pkg)) { |
| found = true; |
| break; |
| } |
| } |
| if (!found) { |
| status = false; |
| return true; |
| } |
| } |
| return true; |
| } |
| } |
| |
| IMountService getMs() { |
| IBinder service = ServiceManager.getService("mount"); |
| if (service != null) { |
| return IMountService.Stub.asInterface(service); |
| } else { |
| Log.e(TAG, "Can't get mount service"); |
| } |
| return null; |
| } |
| |
| boolean getMediaState() { |
| try { |
| String mPath = Environment.getExternalStorageDirectory().toString(); |
| String state = getMs().getVolumeState(mPath); |
| return Environment.MEDIA_MOUNTED.equals(state); |
| } catch (RemoteException e) { |
| return false; |
| } |
| } |
| |
| boolean mountMedia() { |
| if (getMediaState()) { |
| return true; |
| } |
| try { |
| String mPath = Environment.getExternalStorageDirectory().toString(); |
| int ret = getMs().mountVolume(mPath); |
| return ret == StorageResultCode.OperationSucceeded; |
| } catch (RemoteException e) { |
| return false; |
| } |
| } |
| |
| class StorageListener extends StorageEventListener { |
| String oldState; |
| String newState; |
| String path; |
| private boolean doneFlag = false; |
| @Override |
| public void onStorageStateChanged(String path, String oldState, String newState) { |
| if (localLOGV) Log.i(TAG, "Storage state changed from " + oldState + " to " + newState); |
| synchronized (this) { |
| this.oldState = oldState; |
| this.newState = newState; |
| this.path = path; |
| doneFlag = true; |
| notifyAll(); |
| } |
| } |
| |
| public boolean isDone() { |
| return doneFlag; |
| } |
| } |
| |
| private boolean unmountMedia() { |
| if (!getMediaState()) { |
| return true; |
| } |
| String path = Environment.getExternalStorageDirectory().toString(); |
| StorageListener observer = new StorageListener(); |
| StorageManager sm = (StorageManager) mContext.getSystemService(Context.STORAGE_SERVICE); |
| sm.registerListener(observer); |
| try { |
| // Wait on observer |
| synchronized(observer) { |
| getMs().unmountVolume(path, false); |
| long waitTime = 0; |
| while((!observer.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| observer.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!observer.isDone()) { |
| throw new Exception("Timed out waiting for packageInstalled callback"); |
| } |
| return true; |
| } |
| } catch (Exception e) { |
| return false; |
| } finally { |
| sm.unregisterListener(observer); |
| } |
| } |
| |
| private boolean mountFromRawResource() { |
| // Install pkg on sdcard |
| InstallParams ip = sampleInstallFromRawResource(PackageManager.INSTALL_EXTERNAL, false); |
| if (localLOGV) Log.i(TAG, "Installed pkg on sdcard"); |
| boolean origState = getMediaState(); |
| boolean registeredReceiver = false; |
| SdMountReceiver receiver = new SdMountReceiver(new String[]{ip.pkg.packageName}); |
| try { |
| if (localLOGV) Log.i(TAG, "Unmounting media"); |
| // Unmount media |
| assertTrue(unmountMedia()); |
| if (localLOGV) Log.i(TAG, "Unmounted media"); |
| // Register receiver here |
| PackageManager pm = getPm(); |
| mContext.registerReceiver(receiver, receiver.filter); |
| registeredReceiver = true; |
| |
| // Wait on receiver |
| synchronized (receiver) { |
| if (localLOGV) Log.i(TAG, "Mounting media"); |
| // Mount media again |
| assertTrue(mountMedia()); |
| if (localLOGV) Log.i(TAG, "Mounted media"); |
| if (localLOGV) Log.i(TAG, "Waiting for notification"); |
| long waitTime = 0; |
| // Verify we received the broadcast |
| waitTime = 0; |
| while((!receiver.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| receiver.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!receiver.isDone()) { |
| failStr("Timed out waiting for EXTERNAL_APPLICATIONS notification"); |
| } |
| return receiver.received; |
| } |
| } catch (InterruptedException e) { |
| failStr(e); |
| return false; |
| } finally { |
| if (registeredReceiver) mContext.unregisterReceiver(receiver); |
| // Restore original media state |
| if (origState) { |
| mountMedia(); |
| } else { |
| unmountMedia(); |
| } |
| if (localLOGV) Log.i(TAG, "Cleaning up install"); |
| cleanUpInstall(ip); |
| } |
| } |
| |
| /* |
| * Install package on sdcard. Unmount and then mount the media. |
| * (Use PackageManagerService private api for now) |
| * Make sure the installed package is available. |
| * STOPSHIP will uncomment when MountService api's to mount/unmount |
| * are made asynchronous. |
| */ |
| public void xxxtestMountSdNormalInternal() { |
| assertTrue(mountFromRawResource()); |
| } |
| |
| void cleanUpInstall(InstallParams ip) { |
| if (ip == null) { |
| return; |
| } |
| Runtime.getRuntime().gc(); |
| Log.i(TAG, "Deleting package : " + ip.pkg.packageName); |
| getPm().deletePackage(ip.pkg.packageName, null, 0); |
| File outFile = new File(ip.outFileName); |
| if (outFile != null && outFile.exists()) { |
| outFile.delete(); |
| } |
| } |
| |
| public void testManifestInstallLocationInternal() { |
| installFromRawResource("install.apk", R.raw.install_loc_internal, |
| 0, true, false, -1, PackageInfo.INSTALL_LOCATION_INTERNAL_ONLY); |
| } |
| |
| public void testManifestInstallLocationSdcard() { |
| installFromRawResource("install.apk", R.raw.install_loc_sdcard, |
| 0, true, false, -1, PackageInfo.INSTALL_LOCATION_PREFER_EXTERNAL); |
| } |
| |
| public void testManifestInstallLocationAuto() { |
| installFromRawResource("install.apk", R.raw.install_loc_auto, |
| 0, true, false, -1, PackageInfo.INSTALL_LOCATION_AUTO); |
| } |
| |
| public void testManifestInstallLocationUnspecified() { |
| installFromRawResource("install.apk", R.raw.install_loc_unspecified, |
| 0, true, false, -1, PackageInfo.INSTALL_LOCATION_AUTO); |
| } |
| |
| public void testManifestInstallLocationFwdLockedFlagSdcard() { |
| installFromRawResource("install.apk", R.raw.install_loc_unspecified, |
| PackageManager.INSTALL_FORWARD_LOCK | |
| PackageManager.INSTALL_EXTERNAL, true, true, |
| PackageManager.INSTALL_FAILED_INVALID_INSTALL_LOCATION, |
| PackageInfo.INSTALL_LOCATION_AUTO); |
| } |
| |
| public void testManifestInstallLocationFwdLockedSdcard() { |
| installFromRawResource("install.apk", R.raw.install_loc_sdcard, |
| PackageManager.INSTALL_FORWARD_LOCK, true, false, |
| -1, |
| PackageInfo.INSTALL_LOCATION_PREFER_EXTERNAL); |
| } |
| |
| private void replaceManifestLocation(int iFlags, int rFlags) { |
| InstallParams ip = sampleInstallFromRawResource(iFlags, false); |
| GenericReceiver receiver = new ReplaceReceiver(ip.pkg.packageName); |
| int replaceFlags = rFlags | PackageManager.INSTALL_REPLACE_EXISTING; |
| try { |
| assertEquals(invokeInstallPackage(ip.packageURI, replaceFlags, |
| ip.pkg.packageName, receiver), true); |
| assertInstall(ip.pkg, replaceFlags, ip.pkg.installLocation); |
| } catch (Exception e) { |
| failStr("Failed with exception : " + e); |
| } finally { |
| cleanUpInstall(ip); |
| } |
| } |
| |
| public void testReplaceFlagInternalSdcard() { |
| replaceManifestLocation(0, PackageManager.INSTALL_EXTERNAL); |
| } |
| |
| public void testReplaceFlagSdcardInternal() { |
| replaceManifestLocation(PackageManager.INSTALL_EXTERNAL, 0); |
| } |
| |
| public void testManifestInstallLocationReplaceInternalSdcard() { |
| int iFlags = 0; |
| int iApk = R.raw.install_loc_unspecified; |
| int rFlags = 0; |
| int rApk = R.raw.install_loc_sdcard; |
| InstallParams ip = installFromRawResource("install.apk", iApk, |
| iFlags, false, |
| false, -1, PackageInfo.INSTALL_LOCATION_AUTO); |
| GenericReceiver receiver = new ReplaceReceiver(ip.pkg.packageName); |
| int replaceFlags = rFlags | PackageManager.INSTALL_REPLACE_EXISTING; |
| try { |
| InstallParams rp = installFromRawResource("install.apk", rApk, |
| rFlags, false, |
| false, -1, PackageInfo.INSTALL_LOCATION_PREFER_EXTERNAL); |
| assertInstall(rp.pkg, replaceFlags, rp.pkg.installLocation); |
| } catch (Exception e) { |
| failStr("Failed with exception : " + e); |
| } finally { |
| cleanUpInstall(ip); |
| } |
| } |
| |
| public void testManifestInstallLocationReplaceSdcardInternal() { |
| int iFlags = 0; |
| int iApk = R.raw.install_loc_sdcard; |
| int rFlags = 0; |
| int rApk = R.raw.install_loc_unspecified; |
| InstallParams ip = installFromRawResource("install.apk", iApk, |
| iFlags, false, |
| false, -1, PackageInfo.INSTALL_LOCATION_PREFER_EXTERNAL); |
| GenericReceiver receiver = new ReplaceReceiver(ip.pkg.packageName); |
| int replaceFlags = rFlags | PackageManager.INSTALL_REPLACE_EXISTING; |
| try { |
| InstallParams rp = installFromRawResource("install.apk", rApk, |
| rFlags, false, |
| false, -1, PackageInfo.INSTALL_LOCATION_PREFER_EXTERNAL); |
| assertInstall(rp.pkg, replaceFlags, ip.pkg.installLocation); |
| } catch (Exception e) { |
| failStr("Failed with exception : " + e); |
| } finally { |
| cleanUpInstall(ip); |
| } |
| } |
| |
| public void xxxtestClearAllSecureContainers() { |
| IMountService ms = getMs(); |
| try { |
| String list[] = ms.getSecureContainerList(); |
| if (list != null) { |
| for (String cid : list) { |
| Log.i(TAG, "Destroying container " + cid); |
| ms.destroySecureContainer(cid, false); |
| } |
| } |
| } catch (RemoteException e) {} |
| } |
| |
| class MoveReceiver extends GenericReceiver { |
| String pkgName; |
| final static int INVALID = -1; |
| final static int REMOVED = 1; |
| final static int ADDED = 2; |
| int removed = INVALID; |
| |
| MoveReceiver(String pkgName) { |
| this.pkgName = pkgName; |
| filter = new IntentFilter(Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE); |
| filter.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE); |
| super.setFilter(filter); |
| } |
| |
| public boolean notifyNow(Intent intent) { |
| String action = intent.getAction(); |
| Log.i(TAG, "MoveReceiver::" + action); |
| if (Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE.equals(action)) { |
| String[] list = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST); |
| if (list != null) { |
| for (String pkg : list) { |
| if (pkg.equals(pkgName)) { |
| removed = REMOVED; |
| break; |
| } |
| } |
| } |
| removed = REMOVED; |
| } else if (Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE.equals(action)) { |
| if (removed != REMOVED) { |
| return false; |
| } |
| String[] list = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST); |
| if (list != null) { |
| for (String pkg : list) { |
| if (pkg.equals(pkgName)) { |
| removed = ADDED; |
| return true; |
| } |
| } |
| } |
| } |
| return false; |
| } |
| } |
| |
| private class PackageMoveObserver extends IPackageMoveObserver.Stub { |
| public int returnCode; |
| private boolean doneFlag = false; |
| |
| public void packageMoved(String packageName, int returnCode) { |
| synchronized(this) { |
| this.returnCode = returnCode; |
| doneFlag = true; |
| notifyAll(); |
| } |
| } |
| |
| public boolean isDone() { |
| return doneFlag; |
| } |
| } |
| |
| public boolean invokeMovePackage(String pkgName, int flags, |
| GenericReceiver receiver) throws Exception { |
| PackageMoveObserver observer = new PackageMoveObserver(); |
| final boolean received = false; |
| mContext.registerReceiver(receiver, receiver.filter); |
| try { |
| // Wait on observer |
| synchronized(observer) { |
| synchronized (receiver) { |
| getPm().movePackage(pkgName, observer, flags); |
| long waitTime = 0; |
| while((!observer.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| observer.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!observer.isDone()) { |
| throw new Exception("Timed out waiting for pkgmove callback"); |
| } |
| if (observer.returnCode != PackageManager.MOVE_SUCCEEDED) { |
| return false; |
| } |
| // Verify we received the broadcast |
| waitTime = 0; |
| while((!receiver.isDone()) && (waitTime < MAX_WAIT_TIME) ) { |
| receiver.wait(WAIT_TIME_INCR); |
| waitTime += WAIT_TIME_INCR; |
| } |
| if(!receiver.isDone()) { |
| throw new Exception("Timed out waiting for MOVE notifications"); |
| } |
| return receiver.received; |
| } |
| } |
| } finally { |
| mContext.unregisterReceiver(receiver); |
| } |
| } |
| |
| private int getInstallLoc() { |
| boolean userSetting = false; |
| int origDefaultLoc = PackageInfo.INSTALL_LOCATION_AUTO; |
| try { |
| userSetting = Settings.System.getInt(mContext.getContentResolver(), Settings.System.SET_INSTALL_LOCATION) != 0; |
| origDefaultLoc = Settings.System.getInt(mContext.getContentResolver(), Settings.System.DEFAULT_INSTALL_LOCATION); |
| } catch (SettingNotFoundException e1) { |
| } |
| return origDefaultLoc; |
| } |
| |
| private void setInstallLoc(int loc) { |
| Settings.System.putInt(mContext.getContentResolver(), |
| Settings.System.DEFAULT_INSTALL_LOCATION, loc); |
| } |
| /* |
| * Utility function that reads a apk bundled as a raw resource |
| * copies it into own data directory and invokes |
| * PackageManager api to install first and then replace it |
| * again. |
| */ |
| public void moveFromRawResource(int installFlags, int moveFlags, |
| int expRetCode) { |
| int origDefaultLoc = getInstallLoc(); |
| setInstallLoc(PackageInfo.INSTALL_LOCATION_AUTO); |
| // Install first |
| InstallParams ip = sampleInstallFromRawResource(installFlags, false); |
| ApplicationInfo oldAppInfo = null; |
| try { |
| oldAppInfo = getPm().getApplicationInfo(ip.pkg.packageName, 0); |
| } catch (NameNotFoundException e) { |
| failStr("Pkg hasnt been installed correctly"); |
| } |
| |
| // Create receiver based on expRetCode |
| MoveReceiver receiver = new MoveReceiver(ip.pkg.packageName); |
| try { |
| boolean retCode = invokeMovePackage(ip.pkg.packageName, moveFlags, |
| receiver); |
| if (expRetCode == PackageManager.MOVE_SUCCEEDED) { |
| assertTrue(retCode); |
| ApplicationInfo info = getPm().getApplicationInfo(ip.pkg.packageName, 0); |
| assertNotNull(info); |
| if ((moveFlags & PackageManager.MOVE_INTERNAL) != 0) { |
| assertTrue((info.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) == 0); |
| } else if ((moveFlags & PackageManager.MOVE_EXTERNAL_MEDIA) != 0){ |
| assertTrue((info.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0); |
| } |
| } else { |
| assertFalse(retCode); |
| ApplicationInfo info = getPm().getApplicationInfo(ip.pkg.packageName, 0); |
| assertNotNull(info); |
| assertEquals(oldAppInfo.flags, info.flags); |
| } |
| } catch (Exception e) { |
| failStr("Failed with exception : " + e); |
| } finally { |
| cleanUpInstall(ip); |
| // Restore default install location |
| setInstallLoc(origDefaultLoc); |
| } |
| } |
| |
| public void testMoveAppInternalToExternal() { |
| moveFromRawResource(0, PackageManager.MOVE_EXTERNAL_MEDIA, |
| PackageManager.MOVE_SUCCEEDED); |
| } |
| |
| public void testMoveAppInternalToInternal() { |
| moveFromRawResource(0, PackageManager.MOVE_INTERNAL, |
| PackageManager.MOVE_FAILED_INVALID_LOCATION); |
| } |
| |
| public void testMoveAppExternalToExternal() { |
| moveFromRawResource(PackageManager.INSTALL_EXTERNAL, PackageManager.MOVE_EXTERNAL_MEDIA, |
| PackageManager.MOVE_FAILED_INVALID_LOCATION); |
| } |
| public void testMoveAppExternalToInternal() { |
| moveFromRawResource(PackageManager.INSTALL_EXTERNAL, PackageManager.MOVE_INTERNAL, |
| PackageManager.MOVE_SUCCEEDED); |
| } |
| /* |
| * TODO's |
| * check version numbers for upgrades |
| * check permissions of installed packages |
| * how to do tests on updated system apps? |
| * verify updates to system apps cannot be installed on the sdcard. |
| */ |
| } |