您的位置:首页 > 移动开发

Android6.0关于预置三方app卸载(一) copy到data/app下

2018-03-12 15:25 507 查看
我们知道每个手机会预装一些三方的应用,但我们需要用户把这些应用删除的权限。这可怎么操作呢,我们在编译的时候可以把应用放在data/app下,但是这样的话,手机一恢复出厂设置预留应用就没有了。这篇博客我们就来解决这个问题。

一、安装预置应用方法

这里提供一个思路,把预留应用放在system/third-app下,然后在第一次开机的时候把这些应用copy到data/app下,当然要在PKMS扫描data/app之前,这样第一次开机的时候就能安装上这些应用了。因为应用在data/app下也能删除。因为应用在system/third-app中也有,而恢复出厂设置的时候system的目录不会清空。在恢复出厂设置后,还会把这些应用copy到data/app下(恢复出厂设置开机等于第一次开机),下面我们看下如何实现。

先看下apk的编译时放在了third_app下,

include $(CLEAR_VARS)
LOCAL_MODULE_PATH := $(TARGET_OUT)/third_app
LOCAL_MODULE := MobileMusic
LOCAL_MODULE_TAGS := optional
LOCAL_MODULE_CLASS := APPS
LOCAL_SRC_FILES := $(LOCAL_MODULE).apk
LOCAL_CERTIFICATE := PRESIGNED
include $(BUILD_PREBUILT)


功能如下,就是把system/third-app下面的apk复制到data/app, 但是在哪里调用这个函数呢?

//install the third apps when system is first boot
private void installThirdApps(){
//the source directory not exists
File storeDir = new File("/system/third_app");
if(!storeDir.exists()){
Log.e(TAG,"/system/third_app is not exist");
return;
}

//get the apk files in /system/third_app
String apkFilesNames[] = storeDir.list();
if(apkFilesNames == null){
Log.e(TAG,"apk file name is null");
return;
}

//copy the apk files to /data/app
boolean installSucc = false;
for(int i = 0; i < apkFilesNames.length; i++){
//Uri srcFileUri = Uri.parse(storeDir+"/"+apkFilesNames[i]);
File srcFile = new File("/system/third_app",apkFilesNames[i]);
Log.e(TAG,"srcFile="+srcFile);
File destFile = new File("/data/app",apkFilesNames[i]);
Log.e(TAG,"destFile="+destFile.toString());
boolean installResult = copyThirdApps(srcFile,destFile);
if(!installResult){
Log.d(TAG,"install failed");
return;
}
}
}

/**
* File copy function.
* It will be used when installThirdApps
* @param srcFile  just like '/system/third_app/***.apk'
* @param destDir  just like '/data/app/***.apk'
* @return
*/
private boolean copyThirdApps(File srcFile, File destDir) {
//do some check actions
if (srcFile == null || destDir == null || !srcFile.exists()) {
Log.e(TAG, "invalid arguments for movePreinstallApkFile()");
Log.e(TAG, "move " + srcFile + " to " + destDir + " failed");
return false;
}

//create new file
try{
destDir.createNewFile();
}catch(Exception e){
Log.e(TAG, "create file faild! due to:" + e);
return false;
}

//set permission
try{
Runtime.getRuntime().exec("chmod 644 "+destDir.getAbsolutePath());
}catch(Exception e){
Log.e(TAG, "chmod file faild! due to:"+e);
}

//do copy
try{
boolean ret = FileUtils.copyFile(srcFile,destDir);
if(!ret){
Log.e(TAG,"copy file faild!");
return false;
}
}catch(Exception e){
Log.e(TAG, "copy file faild! due to:"+e);
}

return true && destDir.exists();
}


在PKMS的构造函数,开始处理非系统应用的时候,但是一定要在扫描data/app之前,这样才能后面扫描到data/app这些复制进去的app,才会第一次开机安装成功。

if (!mOnlyCore) {
EventLog.writeEvent(EventLogTags.BOOT_PROGRESS_PMS_DATA_SCAN_START,
SystemClock.uptimeMillis());
if(isFirstBoot()){//判断第一次开机
Log.i(TAG, "It's first boot, install the third apps");
installThirdApps();//安装三方应用(copy到data/app下)
}
scanDirLI(mAppInstallDir, 0, scanFlags | SCAN_REQUIRE_KNOWN, 0);

scanDirLI(mDrmAppPrivateInstallDir, PackageParser.PARSE_FORWARD_LOCK,
scanFlags | SCAN_REQUIRE_KNOWN, 0);a


二、adb install处理

下面几个问题,我们知道PKMS会在开机的时候扫描data/app,所以一般我们把一个apk push到data/app下的时候,再把手机重启也会安装应用。但是一定要重启。

那adb install又是如何做到安装应用的,我们来看下代码,下面这段代码是pc侧的

else if (!strcmp(argv[0], "install")) {
if (argc < 2) return usage();
return install_app(ttype, serial, argc, argv);
}


我们来看下install_app这个函数,也是把apk文件push到/data/local/tmp下,然后调用了pm命令安装应用。

static int install_app(transport_type transport, const char* serial, int argc,
const char** argv)
{
static const char *const DATA_DEST = "/data/local/tmp/%s";
static const char *const SD_DEST = "/sdcard/tmp/%s";
const char* where = DATA_DEST;
int i;
struct stat sb;

for (i = 1; i < argc; i++) {
if (!strcmp(argv[i], "-s")) {
where = SD_DEST;
}
}

// Find last APK argument.
// All other arguments passed through verbatim.
int last_apk = -1;
for (i = argc - 1; i >= 0; i--) {
const char* file = argv[i];
char* dot = strrchr(file, '.');
if (dot && !strcasecmp(dot, ".apk")) {
if (stat(file, &sb) == -1 || !S_ISREG(sb.st_mode)) {
fprintf(stderr, "Invalid APK file: %s\n", file);
return -1;
}

last_apk = i;
break;
}
}

if (last_apk == -1) {
fprintf(stderr, "Missing APK file\n");
return -1;
}

const char* apk_file = argv[last_apk];
char apk_dest[PATH_MAX];
snprintf(apk_dest, sizeof apk_dest, where, get_basename(apk_file));
int err = do_sync_push(apk_file, apk_dest, 0 /* no show progress */);//把apk文件push到/data/local/tmp/目录下
if (err) {
goto cleanup_apk;
} else {
argv[last_apk] = apk_dest; /* destination name, not source location */
}

err = pm_command(transport, serial, argc, argv);//调用了pm命令

cleanup_apk:
delete_file(transport, serial, apk_dest);
return err;
}


看下pm_command函数,是调用了pm install命令来安装应用。

static int pm_command(transport_type transport, const char* serial,
int argc, const char** argv)
{
std::string cmd = "shell:pm";

while (argc-- > 0) {
cmd += " " + escape_arg(*argv++);
}

return send_shell_command(transport, serial, cmd);
}


我们来看下pm install的处理,调用了runInstall函数

if ("install".equals(op)) {
return runInstall();
}


在runInstall函数会调用PKMS的installPackageAsUser函数安装应用

VerificationParams verificationParams = new VerificationParams(verificationURI,
originatingURI, referrerURI, VerificationParams.NO_UID, null);

mPm.installPackageAsUser(apkFilePath, obs.getBinder(), installFlags,
installerPackageName, verificationParams, abi, userId);


三、注意

上面PKMS中把system/third_app下面的apk放在data/app有点问题,原因是android6.0 编译apk文件,会自动生成一个目录,所以代码需要修改一下。

主要是下面这个函数,system/third_app目录下多了一层目录。

private void installThirdApps(){
//the source directory not exists
File storeDir = new File("/system/third_app");
if(!storeDir.exists()){
Log.e(TAG,"/system/third_app is not exist");
return;
}

//get the apk files in /system/third_app
String apkDirFilesNames[] = storeDir.list();
if(apkDirFilesNames == null){
Log.e(TAG,"apk file name is null");
return;
}

//copy the apk files to /data/app
boolean installSucc = false;
for(int i = 0; i < apkDirFilesNames.length; i++){
File srcFileDir = new File("/system/third_app", apkDirFilesNames[i]);
Log.e(TAG,"srcFile=" + srcFileDir);
String srcFileNames[] = srcFileDir.list();
for(int j = 0; j < srcFileNames.length; j++) {
File srcFile = new File(srcFileDir, srcFileNames[j]);
File destFile = new File("/data/app", srcFileNames[j]);
boolean installResult = copyThirdApps(srcFile, destFile);
if(!installResult){
Log.d(TAG,"install failed");
return;
}
}
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐