Merge pull request #613 from grote/backup-binder

Use BackupRequester to request backup in chunks
This commit is contained in:
Torsten Grote 2024-02-22 13:10:23 -03:00 committed by GitHub
commit 8c27814407
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
8 changed files with 200 additions and 60 deletions

View file

@ -119,9 +119,21 @@ internal interface LargeBackupTestBase : LargeTestBase {
coEvery { coEvery {
spyKVBackup.finishBackup() spyKVBackup.finishBackup()
} answers { } answers {
val oldMap = HashMap<String, String>()
// @pm@ and android can get backed up multiple times (if we need more than one request)
// so we need to keep the data it backed up before
if (backupResult.kv.containsKey(packageName)) {
backupResult.kv[packageName]?.forEach { (key, value) ->
// if a key existing in new data, we use its value from new data, don't override
if (!data.containsKey(key)) oldMap[key] = value
}
}
backupResult.kv[packageName!!] = data backupResult.kv[packageName!!] = data
.mapValues { entry -> entry.value.sha256() } .mapValues { entry -> entry.value.sha256() }
.toMutableMap() .toMutableMap()
.apply {
putAll(oldMap)
}
packageName = null packageName = null
data = mutableMapOf() data = mutableMapOf()

View file

@ -39,6 +39,7 @@ import com.stevesoltys.seedvault.restore.install.isInstalled
import com.stevesoltys.seedvault.settings.SettingsManager import com.stevesoltys.seedvault.settings.SettingsManager
import com.stevesoltys.seedvault.storage.StorageRestoreService import com.stevesoltys.seedvault.storage.StorageRestoreService
import com.stevesoltys.seedvault.transport.TRANSPORT_ID import com.stevesoltys.seedvault.transport.TRANSPORT_ID
import com.stevesoltys.seedvault.transport.backup.NUM_PACKAGES_PER_TRANSACTION
import com.stevesoltys.seedvault.transport.restore.RestoreCoordinator import com.stevesoltys.seedvault.transport.restore.RestoreCoordinator
import com.stevesoltys.seedvault.ui.AppBackupState import com.stevesoltys.seedvault.ui.AppBackupState
import com.stevesoltys.seedvault.ui.AppBackupState.FAILED import com.stevesoltys.seedvault.ui.AppBackupState.FAILED
@ -70,7 +71,7 @@ import java.util.LinkedList
private val TAG = RestoreViewModel::class.java.simpleName private val TAG = RestoreViewModel::class.java.simpleName
internal const val PACKAGES_PER_CHUNK = 100 internal const val PACKAGES_PER_CHUNK = NUM_PACKAGES_PER_TRANSACTION
internal class RestoreViewModel( internal class RestoreViewModel(
app: Application, app: Application,

View file

@ -1,19 +1,16 @@
package com.stevesoltys.seedvault.transport package com.stevesoltys.seedvault.transport
import android.app.Service import android.app.Service
import android.app.backup.BackupManager
import android.app.backup.IBackupManager import android.app.backup.IBackupManager
import android.content.Context import android.content.Context
import android.content.Intent import android.content.Intent
import android.os.IBinder import android.os.IBinder
import android.os.RemoteException
import android.util.Log import android.util.Log
import androidx.annotation.WorkerThread import androidx.annotation.WorkerThread
import com.stevesoltys.seedvault.BackupMonitor
import com.stevesoltys.seedvault.crypto.KeyManager import com.stevesoltys.seedvault.crypto.KeyManager
import com.stevesoltys.seedvault.transport.backup.BackupRequester
import com.stevesoltys.seedvault.transport.backup.PackageService import com.stevesoltys.seedvault.transport.backup.PackageService
import com.stevesoltys.seedvault.ui.notification.BackupNotificationManager import com.stevesoltys.seedvault.ui.notification.BackupNotificationManager
import com.stevesoltys.seedvault.ui.notification.NotificationBackupObserver
import org.koin.core.component.KoinComponent import org.koin.core.component.KoinComponent
import org.koin.core.component.inject import org.koin.core.component.inject
import org.koin.core.context.GlobalContext.get import org.koin.core.context.GlobalContext.get
@ -70,25 +67,10 @@ fun requestBackup(context: Context): Boolean {
val backupManager: IBackupManager = get().get() val backupManager: IBackupManager = get().get()
return if (backupManager.isBackupEnabled) { return if (backupManager.isBackupEnabled) {
val packageService: PackageService = get().get() val packageService: PackageService = get().get()
val packages = packageService.eligiblePackages
val appTotals = packageService.expectedAppTotals
val result = try {
Log.d(TAG, "Backup is enabled, request backup...") Log.d(TAG, "Backup is enabled, request backup...")
val observer = NotificationBackupObserver(context, packages.size, appTotals) val backupRequester = BackupRequester(context, backupManager, packageService)
backupManager.requestBackup(packages, observer, BackupMonitor(), 0) return backupRequester.requestBackup()
} catch (e: RemoteException) {
Log.e(TAG, "Error during backup: ", e)
val nm: BackupNotificationManager = get().get()
nm.onBackupError()
}
if (result == BackupManager.SUCCESS) {
Log.i(TAG, "Backup request succeeded ")
true
} else {
Log.e(TAG, "Backup request failed: $result")
false
}
} else { } else {
Log.i(TAG, "Backup is not enabled") Log.i(TAG, "Backup is not enabled")
true // this counts as success true // this counts as success

View file

@ -158,7 +158,8 @@ internal class BackupCoordinator(
*/ */
suspend fun getBackupQuota(packageName: String, isFullBackup: Boolean): Long { suspend fun getBackupQuota(packageName: String, isFullBackup: Boolean): Long {
if (packageName != MAGIC_PACKAGE_MANAGER) { if (packageName != MAGIC_PACKAGE_MANAGER) {
// try to back up APK here as later methods are sometimes not called called // try to back up APK here as later methods are sometimes not called
// TODO move this into BackupWorker
backUpApk(context.packageManager.getPackageInfo(packageName, GET_SIGNING_CERTIFICATES)) backUpApk(context.packageManager.getPackageInfo(packageName, GET_SIGNING_CERTIFICATES))
} }
@ -379,6 +380,7 @@ internal class BackupCoordinator(
} }
} }
// hook in here to back up APKs of apps that are otherwise not allowed for backup // hook in here to back up APKs of apps that are otherwise not allowed for backup
// TODO move this into BackupWorker
if (isPmBackup && settingsManager.canDoBackupNow()) { if (isPmBackup && settingsManager.canDoBackupNow()) {
try { try {
backUpApksOfNotBackedUpPackages() backUpApksOfNotBackedUpPackages()
@ -424,10 +426,12 @@ internal class BackupCoordinator(
val packageName = packageInfo.packageName val packageName = packageInfo.packageName
try { try {
nm.onOptOutAppBackup(packageName, i + 1, notBackedUpPackages.size) nm.onOptOutAppBackup(packageName, i + 1, notBackedUpPackages.size)
val packageState = val packageState = if (packageInfo.isStopped()) WAS_STOPPED else NOT_ALLOWED
if (packageInfo.isStopped()) WAS_STOPPED else NOT_ALLOWED
val wasBackedUp = backUpApk(packageInfo, packageState) val wasBackedUp = backUpApk(packageInfo, packageState)
if (!wasBackedUp) { if (wasBackedUp) {
Log.d(TAG, "Was backed up: $packageName")
} else {
Log.d(TAG, "Not backed up: $packageName - ${packageState.name}")
val packageMetadata = val packageMetadata =
metadataManager.getPackageMetadata(packageName) metadataManager.getPackageMetadata(packageName)
val oldPackageState = packageMetadata?.state val oldPackageState = packageMetadata?.state

View file

@ -0,0 +1,108 @@
/*
* SPDX-FileCopyrightText: 2024 The Calyx Institute
* SPDX-License-Identifier: Apache-2.0
*/
package com.stevesoltys.seedvault.transport.backup
import android.app.backup.BackupManager
import android.app.backup.IBackupManager
import android.content.Context
import android.os.RemoteException
import android.util.Log
import androidx.annotation.WorkerThread
import com.stevesoltys.seedvault.BackupMonitor
import com.stevesoltys.seedvault.ui.notification.BackupNotificationManager
import com.stevesoltys.seedvault.ui.notification.NotificationBackupObserver
import org.koin.core.component.KoinComponent
import org.koin.core.context.GlobalContext
private val TAG = BackupRequester::class.java.simpleName
internal const val NUM_PACKAGES_PER_TRANSACTION = 100
/**
* Used for requesting a backup of all installed packages,
* in chunks if there are more than [NUM_PACKAGES_PER_TRANSACTION].
*
* Can only be used once for one backup.
* Make a new instance for subsequent backups.
*/
@WorkerThread
internal class BackupRequester(
context: Context,
private val backupManager: IBackupManager,
val packageService: PackageService,
) : KoinComponent {
private val packages = packageService.eligiblePackages
private val observer = NotificationBackupObserver(
context = context,
backupRequester = this,
requestedPackages = packages.size,
appTotals = packageService.expectedAppTotals,
)
private val monitor = BackupMonitor()
/**
* The current package index.
*
* Used for splitting the packages into chunks.
*/
private var packageIndex: Int = 0
/**
* Request the backup to happen. Should be called short after constructing this object.
*/
fun requestBackup(): Boolean {
if (packageIndex != 0) error("requestBackup() called more than once!")
return request(getNextChunk())
}
/**
* Backs up the next chunk of packages.
*
* @return true, if backup for all packages was already requested and false,
* if there are more packages that we just have requested backup for.
*/
fun requestNext(): Boolean {
if (packageIndex <= 0) error("requestBackup() must be called first!")
// Backup next chunk if there are more packages to back up.
return if (packageIndex < packages.size) {
request(getNextChunk())
false
} else {
true
}
}
private fun request(chunk: Array<String>): Boolean {
Log.i(TAG, "${chunk.toList()}")
val result = try {
backupManager.requestBackup(chunk, observer, monitor, 0)
} catch (e: RemoteException) {
Log.e(TAG, "Error during backup: ", e)
val nm: BackupNotificationManager = GlobalContext.get().get()
nm.onBackupError()
}
return if (result == BackupManager.SUCCESS) {
Log.i(TAG, "Backup request succeeded")
true
} else {
Log.e(TAG, "Backup request failed: $result")
false
}
}
private fun getNextChunk(): Array<String> {
val nextChunkIndex =
(packageIndex + NUM_PACKAGES_PER_TRANSACTION).coerceAtMost(packages.size)
val packageChunk = packages.subList(packageIndex, nextChunkIndex).toTypedArray()
val numBackingUp = packageIndex + packageChunk.size
Log.i(TAG, "Requesting backup for $numBackingUp/${packages.size} packages...")
packageIndex += packageChunk.size
return packageChunk
}
}

View file

@ -38,7 +38,7 @@ internal class PackageService(
private val packageManager: PackageManager = context.packageManager private val packageManager: PackageManager = context.packageManager
private val myUserId = UserHandle.myUserId() private val myUserId = UserHandle.myUserId()
val eligiblePackages: Array<String> val eligiblePackages: List<String>
@WorkerThread @WorkerThread
@Throws(RemoteException::class) @Throws(RemoteException::class)
get() { get() {
@ -70,11 +70,12 @@ internal class PackageService(
val packageArray = eligibleApps.toMutableList() val packageArray = eligibleApps.toMutableList()
packageArray.add(MAGIC_PACKAGE_MANAGER) packageArray.add(MAGIC_PACKAGE_MANAGER)
return packageArray.toTypedArray() return packageArray
} }
/** /**
* A list of packages that will not be backed up. * A list of packages that will not be backed up,
* because they are currently force-stopped for example.
*/ */
val notBackedUpPackages: List<PackageInfo> val notBackedUpPackages: List<PackageInfo>
@WorkerThread @WorkerThread
@ -127,16 +128,16 @@ internal class PackageService(
@WorkerThread @WorkerThread
get() { get() {
var appsTotal = 0 var appsTotal = 0
var appsOptOut = 0 var appsNotIncluded = 0
packageManager.getInstalledPackages(GET_INSTRUMENTATION).forEach { packageInfo -> packageManager.getInstalledPackages(GET_INSTRUMENTATION).forEach { packageInfo ->
if (packageInfo.isUserVisible(context)) { if (packageInfo.isUserVisible(context)) {
appsTotal++ appsTotal++
if (packageInfo.doesNotGetBackedUp()) { if (packageInfo.doesNotGetBackedUp()) {
appsOptOut++ appsNotIncluded++
} }
} }
} }
return ExpectedAppTotals(appsTotal, appsOptOut) return ExpectedAppTotals(appsTotal, appsNotIncluded)
} }
fun getVersionName(packageName: String): String? = try { fun getVersionName(packageName: String): String? = try {
@ -201,6 +202,7 @@ internal class PackageService(
*/ */
private fun PackageInfo.doesNotGetBackedUp(): Boolean { private fun PackageInfo.doesNotGetBackedUp(): Boolean {
if (packageName == MAGIC_PACKAGE_MANAGER || applicationInfo == null) return true if (packageName == MAGIC_PACKAGE_MANAGER || applicationInfo == null) return true
if (packageName == plugin.providerPackageName) return true
return !allowsBackup() || isStopped() return !allowsBackup() || isStopped()
} }
} }
@ -211,9 +213,11 @@ internal data class ExpectedAppTotals(
*/ */
val appsTotal: Int, val appsTotal: Int,
/** /**
* The number of non-system apps that has opted-out of backup. * The number of non-system apps that do not get backed up.
* These are included here, because we'll at least back up their APKs,
* so at least the app itself does get restored.
*/ */
val appsOptOut: Int, val appsNotGettingBackedUp: Int,
) )
internal fun PackageInfo.isUserVisible(context: Context): Boolean { internal fun PackageInfo.isUserVisible(context: Context): Boolean {

View file

@ -27,6 +27,7 @@ import com.stevesoltys.seedvault.restore.REQUEST_CODE_UNINSTALL
import com.stevesoltys.seedvault.settings.ACTION_APP_STATUS_LIST import com.stevesoltys.seedvault.settings.ACTION_APP_STATUS_LIST
import com.stevesoltys.seedvault.settings.SettingsActivity import com.stevesoltys.seedvault.settings.SettingsActivity
import com.stevesoltys.seedvault.transport.backup.ExpectedAppTotals import com.stevesoltys.seedvault.transport.backup.ExpectedAppTotals
import kotlin.math.min
private const val CHANNEL_ID_OBSERVER = "NotificationBackupObserver" private const val CHANNEL_ID_OBSERVER = "NotificationBackupObserver"
private const val CHANNEL_ID_SUCCESS = "NotificationBackupSuccess" private const val CHANNEL_ID_SUCCESS = "NotificationBackupSuccess"
@ -53,6 +54,11 @@ internal class BackupNotificationManager(private val context: Context) {
private var expectedOptOutApps: Int? = null private var expectedOptOutApps: Int? = null
private var expectedAppTotals: ExpectedAppTotals? = null private var expectedAppTotals: ExpectedAppTotals? = null
/**
* Used as a (temporary) hack to fix progress reporting when fake d2d is enabled.
*/
private var optOutAppsDone = false
private fun getObserverChannel(): NotificationChannel { private fun getObserverChannel(): NotificationChannel {
val title = context.getString(R.string.notification_channel_title) val title = context.getString(R.string.notification_channel_title)
return NotificationChannel(CHANNEL_ID_OBSERVER, title, IMPORTANCE_LOW).apply { return NotificationChannel(CHANNEL_ID_OBSERVER, title, IMPORTANCE_LOW).apply {
@ -87,23 +93,34 @@ internal class BackupNotificationManager(private val context: Context) {
updateBackupNotification( updateBackupNotification(
infoText = "", // This passes quickly, no need to show something here infoText = "", // This passes quickly, no need to show something here
transferred = 0, transferred = 0,
expected = expectedPackages expected = appTotals.appsTotal
) )
expectedApps = expectedPackages expectedApps = expectedPackages
expectedOptOutApps = appTotals.appsOptOut expectedOptOutApps = appTotals.appsNotGettingBackedUp
expectedAppTotals = appTotals expectedAppTotals = appTotals
optOutAppsDone = false
Log.i(TAG, "onBackupStarted $expectedApps + $expectedOptOutApps = ${appTotals.appsTotal}")
} }
/** /**
* This should get called before [onBackupUpdate]. * This should get called before [onBackupUpdate].
* In case of d2d backups, this actually gets called some time after
* some apps were already backed up, so [onBackupUpdate] was called several times.
*/ */
fun onOptOutAppBackup(packageName: String, transferred: Int, expected: Int) { fun onOptOutAppBackup(packageName: String, transferred: Int, expected: Int) {
val text = "Opt-out APK for $packageName" if (optOutAppsDone) return
val text = "APK for $packageName"
if (expectedApps == null) { if (expectedApps == null) {
updateBackgroundBackupNotification(text) updateBackgroundBackupNotification(text)
} else { } else {
updateBackupNotification(text, transferred, expected + (expectedApps ?: 0)) updateBackupNotification(text, transferred, expected + (expectedApps ?: 0))
if (expectedOptOutApps != null && expectedOptOutApps != expected) {
Log.w(TAG, "Number of packages not getting backed up mismatch: " +
"$expectedOptOutApps != $expected")
}
expectedOptOutApps = expected expectedOptOutApps = expected
if (transferred == expected) optOutAppsDone = true
} }
} }
@ -116,7 +133,7 @@ internal class BackupNotificationManager(private val context: Context) {
val addend = expectedOptOutApps ?: 0 val addend = expectedOptOutApps ?: 0
updateBackupNotification( updateBackupNotification(
infoText = app, infoText = app,
transferred = transferred + addend, transferred = min(transferred + addend, expected + addend),
expected = expected + addend expected = expected + addend
) )
} }
@ -167,13 +184,17 @@ internal class BackupNotificationManager(private val context: Context) {
// //
// This won't bring back the expected finish notification in this case, // This won't bring back the expected finish notification in this case,
// but at least we don't leave stuck notifications laying around. // but at least we don't leave stuck notifications laying around.
nm.activeNotifications.forEach { notification -> // FIXME the service gets destroyed for each chunk when requesting backup in chunks
// only consider ongoing notifications in our ID space (storage backup uses > 1000) // This leads to the cancellation of an ongoing backup notification.
if (notification.isOngoing && notification.id < 1000) { // So for now, we'll remove automatic notification clean-up
Log.w(TAG, "Needed to clean up notification with ID ${notification.id}") // and find out if it is still necessary. If not, this comment can be removed.
nm.cancel(notification.id) // nm.activeNotifications.forEach { notification ->
} // // only consider ongoing notifications in our ID space (storage backup uses > 1000)
} // if (notification.isOngoing && notification.id < 1000) {
// Log.w(TAG, "Needed to clean up notification with ID ${notification.id}")
// nm.cancel(notification.id)
// }
// }
} }
fun onBackupFinished(success: Boolean, numBackedUp: Int?, size: Long) { fun onBackupFinished(success: Boolean, numBackedUp: Int?, size: Long) {

View file

@ -10,6 +10,7 @@ import android.util.Log.isLoggable
import com.stevesoltys.seedvault.MAGIC_PACKAGE_MANAGER import com.stevesoltys.seedvault.MAGIC_PACKAGE_MANAGER
import com.stevesoltys.seedvault.R import com.stevesoltys.seedvault.R
import com.stevesoltys.seedvault.metadata.MetadataManager import com.stevesoltys.seedvault.metadata.MetadataManager
import com.stevesoltys.seedvault.transport.backup.BackupRequester
import com.stevesoltys.seedvault.transport.backup.ExpectedAppTotals import com.stevesoltys.seedvault.transport.backup.ExpectedAppTotals
import org.koin.core.component.KoinComponent import org.koin.core.component.KoinComponent
import org.koin.core.component.inject import org.koin.core.component.inject
@ -18,7 +19,8 @@ private val TAG = NotificationBackupObserver::class.java.simpleName
internal class NotificationBackupObserver( internal class NotificationBackupObserver(
private val context: Context, private val context: Context,
private val expectedPackages: Int, private val backupRequester: BackupRequester,
private val requestedPackages: Int,
appTotals: ExpectedAppTotals, appTotals: ExpectedAppTotals,
) : IBackupObserver.Stub(), KoinComponent { ) : IBackupObserver.Stub(), KoinComponent {
@ -30,7 +32,7 @@ internal class NotificationBackupObserver(
init { init {
// Inform the notification manager that a backup has started // Inform the notification manager that a backup has started
// and inform about the expected numbers, so it can compute a total. // and inform about the expected numbers, so it can compute a total.
nm.onBackupStarted(expectedPackages, appTotals) nm.onBackupStarted(requestedPackages, appTotals)
} }
/** /**
@ -73,25 +75,31 @@ internal class NotificationBackupObserver(
* as a whole failed. * as a whole failed.
*/ */
override fun backupFinished(status: Int) { override fun backupFinished(status: Int) {
if (backupRequester.requestNext()) {
if (isLoggable(TAG, INFO)) { if (isLoggable(TAG, INFO)) {
Log.i(TAG, "Backup finished $numPackages/$expectedPackages. Status: $status") Log.i(TAG, "Backup finished $numPackages/$requestedPackages. Status: $status")
} }
val success = status == 0 val success = status == 0
val numBackedUp = if (success) metadataManager.getPackagesNumBackedUp() else null val numBackedUp = if (success) metadataManager.getPackagesNumBackedUp() else null
val size = if (success) metadataManager.getPackagesBackupSize() else 0L val size = if (success) metadataManager.getPackagesBackupSize() else 0L
nm.onBackupFinished(success, numBackedUp, size) nm.onBackupFinished(success, numBackedUp, size)
} }
}
private fun showProgressNotification(packageName: String?) { private fun showProgressNotification(packageName: String?) {
if (packageName == null || currentPackage == packageName) return if (packageName == null || currentPackage == packageName) return
if (isLoggable(TAG, INFO)) { if (isLoggable(TAG, INFO)) Log.i(
"Showing progress notification for $currentPackage $numPackages/$expectedPackages".let { TAG, "Showing progress notification for " +
Log.i(TAG, it) "$currentPackage $numPackages/$requestedPackages"
} )
}
currentPackage = packageName currentPackage = packageName
val app = getAppName(packageName) val appName = getAppName(packageName)
val app = if (appName != packageName) {
"${getAppName(packageName)} ($packageName)"
} else {
packageName
}
numPackages += 1 numPackages += 1
nm.onBackupUpdate(app, numPackages) nm.onBackupUpdate(app, numPackages)
} }