Как проверить, является ли приложение несистемным в Android? - PullRequest
33 голосов
/ 09 января 2012

Я получаю список ApplicationInfo объектов с packageManager.getInstalledApplications (0) и пытаюсь классифицировать их по тому, являются ли они системным приложением.

Некоторое времяЯ использовал методику, описанную здесь , однако, увидев, что в моем приложении некоторые приложения не были в списке несистемных приложений (например, Facebook , который при наличиипросит систему установить себя на SD-карту).После следующего прочтения фактической документации для ApplicationInfo.FLAG_SYSTEM и понимания того, что она на самом деле не фильтрует системные приложения, я сейчас ищу новый подход.

Я предполагаю, что существует большой разрыв между UID системных и несистемных приложений, которые я могу собрать, чтобы провести это различие, но пока я не нашел ответа.Я также изучил другие флаги, такие как ApplicationInfo.FLAG_EXTERNAL_STORAGE, однако я поддерживаю API 1.5.

У кого-нибудь есть реальное решение для этого (не включая FLAG_SYSTEM)?

Ответы [ 11 ]

25 голосов
/ 09 января 2012
PackageManager pm = mcontext.getPackageManager();
List<PackageInfo> list = pm.getInstalledPackages(0);

for(PackageInfo pi : list) {
    ApplicationInfo ai = pm.getApplicationInfo(pi.packageName, 0);

    System.out.println(">>>>>>packages is<<<<<<<<" + ai.publicSourceDir);

    if ((ai.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
        System.out.println(">>>>>>packages is system package"+pi.packageName);          
    }
}
23 голосов
/ 02 февраля 2013

У меня сложилось впечатление, что все приложения в системном образе являются системными приложениями (и обычно устанавливаются в /system/app).

Если FLAG_SYSTEM установлен только для системных приложений, это будет работать даже для приложений во внешнем хранилище:

boolean isUserApp(ApplicationInfo ai) {
    int mask = ApplicationInfo.FLAG_SYSTEM | ApplicationInfo.FLAG_UPDATED_SYSTEM_APP;
    return (ai.flags & mask) == 0;
}

Альтернативой является использование в вашем телефоне программы командной строки pm.

Синтаксис:

pm list packages [-f] [-d] [-e] [-s] [-3] [-i] [-u] [--user USER_ID] [FILTER]

pm list packages: prints all packages, optionally only
  those whose package name contains the text in FILTER.  Options:
    -f: see their associated file.
    -d: filter to only show disbled packages.
    -e: filter to only show enabled packages.
    -s: filter to only show system packages.
    -3: filter to only show third party packages.
    -i: see the installer for the packages.
    -u: also include uninstalled packages.

Код:

ProcessBuilder builder = new ProcessBuilder("pm", "list", "packages", "-s");
Process process = builder.start();

InputStream in = process.getInputStream();
Scanner scanner = new Scanner(in);
Pattern pattern = Pattern.compile("^package:.+");
int skip = "package:".length();

Set<String> systemApps = new HashSet<String>();
while (scanner.hasNext(pattern)) {
    String pckg = scanner.next().substring(skip);
    systemApps.add(pckg);
}

scanner.close();
process.destroy();

Тогда:

boolean isUserApp(String pckg) {
    return !mSystemApps.contains(pckg);
}
16 голосов
/ 08 июля 2014

Вы можете проверить подпись приложения, которое оно подписало с помощью системы. Как ниже

/**
 * Match signature of application to identify that if it is signed by system
 * or not.
 * 
 * @param packageName
 *            package of application. Can not be blank.
 * @return <code>true</code> if application is signed by system certificate,
 *         otherwise <code>false</code>
 */
public boolean isSystemApp(String packageName) {
    try {
        // Get packageinfo for target application
        PackageInfo targetPkgInfo = mPackageManager.getPackageInfo(
                packageName, PackageManager.GET_SIGNATURES);
        // Get packageinfo for system package
        PackageInfo sys = mPackageManager.getPackageInfo(
                "android", PackageManager.GET_SIGNATURES);
        // Match both packageinfo for there signatures
        return (targetPkgInfo != null && targetPkgInfo.signatures != null && sys.signatures[0]
                .equals(targetPkgInfo.signatures[0]));
    } catch (PackageManager.NameNotFoundException e) {
        return false;
    }
}

Вы можете получить больше кода в моем блоге Как проверить, является ли приложение системным приложением или нет (Подписано подписью)

6 голосов
/ 09 января 2012

Ну, на мой взгляд, это неаккуратное решение (что если / data / app не каталог приложений на всех устройствах?), Но после тщательного поиска я пришел к этомус:

for (ApplicationInfo ai : appInfo) {
    if (ai.sourceDir.startsWith("/data/app/")) {
        //Non-system app
    }
    else {
        //System app
    }
}
5 голосов
/ 27 января 2016

Существует 2 типа несистемных приложений:

  1. Приложения, загруженные из Google Play Store
  2. Предварительно загруженные приложения от производителя устройства

Этот код вернет список всех вышеперечисленных приложений:

ArrayList<ApplicationInfo> mAllApp = 
        mPackageManager.getInstalledApplications(PackageManager.GET_META_DATA);

for(int i = 0; i < mAllApp.size(); i++) {
    if((mAllApp.get(i).flags & ApplicationInfo.FLAG_SYSTEM) == 0) {
         // 1. Applications downloaded from Google Play Store
        mAllApp1.add(mAllApp.get(i));
    }

    if((mAllApp.get(i).flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0) {
        // 2. Applications preloaded in device by manufecturer
        mAllApp1.add(mAllApp.get(i));
    }
}
5 голосов
/ 24 июня 2014

Здесь есть небольшое недоразумение. Для Android понятие «системное приложение» - это то, которое устанавливается на образ системы, оно говорит ничего о том, от какого разработчика оно пришло. Таким образом, если OEM решит предварительно загрузить Facebook в образ системы, это системное приложение, и оно останется таковым, независимо от того, где будут установлены обновления приложения. Они точно не будут установлены на образ системы, поскольку он доступен только для чтения.

Итак, ApplicationInfo.FLAG_SYSTEM верен, но это не тот вопрос, который вы задаете. Я думаю, вы спрашиваете, подписан ли пакет с помощью системного сертификата. Что не обязательно является хорошим показателем чего-либо, это может варьироваться от устройства к устройству, и некоторые удивительные компоненты на Android Android не подписаны сертификатом системы, даже если вы ожидаете, что они будут.

В более новых версиях Android существует новый путь / system / priv-app /, который пытается быть местом установки «реальных» системных приложений. Приложения, которые предварительно загружены в образ системы, затем попадают в / system / app /. См. Приложение AOSP Privileged vs System

5 голосов
/ 20 августа 2013

Если приложение является несистемным приложением, оно должно иметь намерение запуска, с помощью которого оно может быть запущено. Если цель запуска равна нулю, значит, это системное приложение.

Пример системных приложений: "com.android.browser.provider", "com.google.android.voicesearch".

Для указанных выше приложений вы получите NULL при запросе запуска Intent.

PackageManager pm = getPackageManager();
List<ApplicationInfo> packages = pm.getInstalledApplications(PackageManager.GET_META_DATA);
for(ApplicationInfo packageInfo:packages){
    if( pm.getLaunchIntentForPackage(packageInfo.packageName) != null ){
                String currAppName = pm.getApplicationLabel(packageInfo).toString();
               //This app is a non-system app
    }
}
2 голосов
/ 30 декабря 2017

Вот различные возможные способы узнать, является ли приложение системным приложением по имени пакета (использовались некоторые коды из этого поста)

package com.test.util;

import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;

import java.io.IOException;
import java.io.InputStream;
import java.util.HashSet;
import java.util.Scanner;
import java.util.Set;
import java.util.regex.Pattern;

import timber.log.Timber;


public class SystemAppChecker {
    private PackageManager packageManager = null;

    public SystemAppChecker(Context context) {
        packageManager = context.getPackageManager();
    }

    /**
     * Check if system app by 'pm' command-line program
     *
     * @param packageName
     *            package name of application. Cannot be null.
     * @return <code>true</code> if package is a system app.
     */
    public boolean isSystemAppByPM(String packageName) {
        if (packageName == null) {
            throw new IllegalArgumentException("Package name cannot be null");
        }
        ProcessBuilder builder = new ProcessBuilder("pm", "list", "packages", "-s");
        Process process = null;
        try {
            process = builder.start();
        } catch (IOException e) {
            Timber.e(e);
            return false;
        }

        InputStream in = process.getInputStream();
        Scanner scanner = new Scanner(in);
        Pattern pattern = Pattern.compile("^package:.+");
        int skip = "package:".length();

        Set<String> systemApps = new HashSet<String>();
        while (scanner.hasNext(pattern)) {
            String pckg = scanner.next().substring(skip);
            systemApps.add(pckg);
        }

        scanner.close();
        process.destroy();

        if (systemApps.contains(packageName)) {
            return true;
        }
        return false;
    }

    /**
     * Check if application is preloaded.
     *
     * @param packageName
     *            package name of application. Cannot be null.
     * @return <code>true</code> if package is preloaded.
     */
    public boolean isSystemPreloaded(String packageName) {
        if (packageName == null) {
            throw new IllegalArgumentException("Package name cannot be null");
        }
        try {
            ApplicationInfo ai = packageManager.getApplicationInfo(
                    packageName, 0);
            if (ai.sourceDir.startsWith("/system/app/") || ai.sourceDir.startsWith("/system/priv-app/")) {
                return true;
            }
        } catch (NameNotFoundException e) {
            Timber.e(e);
        }
        return false;
    }

    /**
     * Check if the app is system signed or not
     *
     * @param packageName
     *            package of application. Cannot be blank.
     * @return <code>true</code> if application is signed by system certificate,
     *         otherwise <code>false</code>
     */
    public boolean isSystemSigned(String packageName) {
        if (packageName == null) {
            throw new IllegalArgumentException("Package name cannot be null");
        }
        try {
            // Get packageinfo for target application
            PackageInfo targetPkgInfo = packageManager.getPackageInfo(
                    packageName, PackageManager.GET_SIGNATURES);
            // Get packageinfo for system package
            PackageInfo sys = packageManager.getPackageInfo(
                    "android", PackageManager.GET_SIGNATURES);
            // Match both packageinfo for there signatures
            return (targetPkgInfo != null && targetPkgInfo.signatures != null && sys.signatures[0]
                    .equals(targetPkgInfo.signatures[0]));
        } catch (PackageManager.NameNotFoundException e) {
            Timber.e(e);
        }
        return false;
    }

    /**
     * Check if application is installed in the device's system image
     *
     * @param packageName
     *            package name of application. Cannot be null.
     * @return <code>true</code> if package is a system app.
     */
    public boolean isSystemAppByFLAG(String packageName) {
        if (packageName == null) {
            throw new IllegalArgumentException("Package name cannot be null");
        }
        try {
            ApplicationInfo ai = packageManager.getApplicationInfo(
                    packageName, 0);
            // Check if FLAG_SYSTEM or FLAG_UPDATED_SYSTEM_APP are set.
            if (ai != null
                    && (ai.flags & (ApplicationInfo.FLAG_SYSTEM | ApplicationInfo.FLAG_UPDATED_SYSTEM_APP)) != 0) {
                return true;
            }
        } catch (NameNotFoundException e) {
            Timber.e(e);
        }
        return false;
    }
}
1 голос
/ 27 декабря 2016

Если у вас есть APK-файл и вы хотите проверить, является ли это Системное приложение или Пользователь установил простую логику: - Системное приложение Файлы недоступны для записи

private boolean isSystemApkFile(File file){
   return !file.canWrite();
}
1 голос
/ 13 декабря 2013
if (!packageInfo.sourceDir.toLowerCase().startsWith("/system/"))
...