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