Android API 30: как установить пакеты как самообновление приложения

Я нашел много тем по этой теме, но они старше 5 лет +, и используемые механизмы в настоящее время больше не работают, поэтому я решил повторно открыть этот вопрос, чтобы узнать, какие требования для начала успешного продолжения самостоятельного обновления в приложении с помощью последний (Android 30) api.

Мои приготовления к этому:

Используемая ОС: Android API 30Android Studio: последняя версия Репозиторий: Nexus 3

Я создал простой макет MainActivity как макет, который содержит RelativLayout с одной простой кнопкой, которая выполняет onClick asyncTask (проверьте логику обновления). Эта задача выполняет простой запрос на отдых в моем nexus 3, чтобы получить последнюю версию моего приложения. Через несколько секунд нажмите ту же кнопку (просто установите нового слушателя после завершения первого щелчка), установка продолжится со следующим кодом:

      public void installUpdate(String url,String fileName) {

    String destination = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS) + "/";
    destination += fileName;
    updateDestination = destination;
    final Uri uri_current = Uri.parse("file://" + destination);

    File file = new File(destination);
    if (file.exists()) {
        file.delete();
    }

    DownloadManager.Request request = new DownloadManager.Request(Uri.parse(url));
    request.addRequestHeader("Authorization", "Basic simplebase64tokenherefornexus");
    request.setDescription("App Update");
    request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, fileName);
    request.setTitle(fileName);
    request.setDestinationUri(uri_current);

    final DownloadManager manager = (DownloadManager) getSystemService(Context.DOWNLOAD_SERVICE);
    final long downloadId = manager.enqueue(request);


    BroadcastReceiver onComplete = new BroadcastReceiver() {
        public void onReceive(Context ctxt, Intent intent) {

            if (!getPackageManager().canRequestPackageInstalls()) {
                startActivityForResult(new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES).setData(Uri.parse(String.format("package:%s", getPackageName()))), 1234);
            } else {
                Uri uri = FileProvider.getUriForFile(MainActivity.this, "com.example.test.fileprovider", new File(file.getAbsolutePath(), fileName));
                Intent install = new Intent(Intent.ACTION_INSTALL_PACKAGE);
                intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                install.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
                install.setDataAndType(uri, manager.getMimeTypeForDownloadedFile(downloadId));
                startActivity(install);
                unregisterReceiver(this);
            }
            finish();
        }
    };
    registerReceiver(onComplete, new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE));
}

Все работает нормально, также корректно вызывается и активируется «установка из неизвестного источника» для моего приложения. Оба файла (установленный apk как выпуск) и apk обновления как выпуск содержат один и тот же ключ подписи v1 + v2. Загруженный apk не поврежден, проверил это с помощью установки вручную в качестве обновления, и он отлично работает, защита Google Play также отключена.

Думаю, я тоже установил правильные разрешения, в моем манифесте я использовал следующие разрешения:

      <manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.test">
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />
<application
    android:allowBackup="true"
    android:testOnly="false"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:networkSecurityConfig="@xml/network_security_config"
    android:roundIcon="@mipmap/ic_launcher"
    android:supportsRtl="true"
    android:theme="@style/AppTheme">
    <uses-library android:name="org.apache.http.legacy" android:required="false"/>
    <provider
        android:name="androidx.core.content.FileProvider"
        android:authorities="com.example.test.fileprovider"
        android:exported="false"
        android:grantUriPermissions="true">
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            android:resource="@xml/provider_path" />
    </provider> ...

мой provider_path:

      <?xml version="1.0" encoding="utf-8"?>
<paths>
    <external-path
        name="external"
        path="." />
    <external-files-path
        name="external_files"
        path="." />
    <files-path
        name="files"
        path="." />
</paths>

Если я попробую его с текущей конфигурацией, я все равно получаю сообщение «При синтаксическом анализе пакета возникла проблема».

Я что-то пропустил или сделал не так, или у кого-нибудь есть лучший практический пример, чтобы это работало?

привет из Берлина :)

1 ответ

для тех, кто сталкивается с той же проблемой:

обнаружил мою проблему, ошибка заключалась в том, что я установил разрешение на установку с неправильным намерением @ трансляция xD

Другие вопросы по тегам