Merge pull request #563 from seedvault-app/bugfix/binder-exception-too-many-packages

Fix binder exception when restoring a large number of applications
This commit is contained in:
Torsten Grote 2023-11-27 08:43:08 -03:00 committed by GitHub
commit 392809274c
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -1,6 +1,8 @@
package com.stevesoltys.seedvault.restore package com.stevesoltys.seedvault.restore
import android.app.Application import android.app.Application
import android.app.backup.BackupManager
import android.app.backup.BackupTransport
import android.app.backup.IBackupManager import android.app.backup.IBackupManager
import android.app.backup.IRestoreObserver import android.app.backup.IRestoreObserver
import android.app.backup.IRestoreSession import android.app.backup.IRestoreSession
@ -63,10 +65,13 @@ import org.calyxos.backup.storage.api.StorageBackup
import org.calyxos.backup.storage.restore.RestoreService.Companion.EXTRA_TIMESTAMP_START import org.calyxos.backup.storage.restore.RestoreService.Companion.EXTRA_TIMESTAMP_START
import org.calyxos.backup.storage.restore.RestoreService.Companion.EXTRA_USER_ID import org.calyxos.backup.storage.restore.RestoreService.Companion.EXTRA_USER_ID
import org.calyxos.backup.storage.ui.restore.SnapshotViewModel import org.calyxos.backup.storage.ui.restore.SnapshotViewModel
import java.lang.IllegalStateException
import java.util.LinkedList 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 class RestoreViewModel( internal class RestoreViewModel(
app: Application, app: Application,
settingsManager: SettingsManager, settingsManager: SettingsManager,
@ -137,6 +142,7 @@ internal class RestoreViewModel(
Log.d(TAG, "Ignoring RestoreSet with no last backup time: $token.") Log.d(TAG, "Ignoring RestoreSet with no last backup time: $token.")
null null
} }
else -> RestorableBackup(metadata) else -> RestorableBackup(metadata)
} }
} }
@ -149,7 +155,6 @@ internal class RestoreViewModel(
} }
override fun onRestorableBackupClicked(restorableBackup: RestorableBackup) { override fun onRestorableBackupClicked(restorableBackup: RestorableBackup) {
restoreCoordinator.beforeStartRestore(restorableBackup.backupMetadata)
mChosenRestorableBackup.value = restorableBackup mChosenRestorableBackup.value = restorableBackup
mDisplayFragment.setEvent(RESTORE_APPS) mDisplayFragment.setEvent(RESTORE_APPS)
} }
@ -173,14 +178,17 @@ internal class RestoreViewModel(
internal fun onNextClickedAfterInstallingApps() { internal fun onNextClickedAfterInstallingApps() {
mDisplayFragment.postEvent(RESTORE_BACKUP) mDisplayFragment.postEvent(RESTORE_BACKUP)
val token = mChosenRestorableBackup.value?.token ?: throw AssertionError()
viewModelScope.launch(ioDispatcher) { viewModelScope.launch(ioDispatcher) {
startRestore(token) startRestore()
} }
} }
@WorkerThread @WorkerThread
private fun startRestore(token: Long) { private fun startRestore() {
val token = mChosenRestorableBackup.value?.token
?: throw IllegalStateException("No chosen backup")
Log.d(TAG, "Starting new restore session to restore backup $token") Log.d(TAG, "Starting new restore session to restore backup $token")
// if we had no token before (i.e. restore from setup wizard), // if we had no token before (i.e. restore from setup wizard),
@ -200,21 +208,29 @@ internal class RestoreViewModel(
return return
} }
// we need to retrieve the restore sets before starting the restore val restorableBackup = mChosenRestorableBackup.value
// otherwise restoreAll() won't work as they need the restore sets cached internally val packages = restorableBackup?.packageMetadataMap?.keys?.toList()
val observer = RestoreObserver { observer -> ?: run {
// this lambda gets executed after we got the restore sets Log.e(TAG, "Chosen backup has empty package metadata map")
// now we can start the restore of all available packages
val restoreAllResult = session.restoreAll(token, observer, monitor)
if (restoreAllResult != 0) {
Log.e(TAG, "restoreAll() returned non-zero value")
mRestoreBackupResult.postValue( mRestoreBackupResult.postValue(
RestoreBackupResult(app.getString(R.string.restore_finished_error)) RestoreBackupResult(app.getString(R.string.restore_set_error))
) )
return
} }
}
val observer = RestoreObserver(
restoreCoordinator = restoreCoordinator,
restorableBackup = restorableBackup,
session = session,
packages = packages,
monitor = monitor
)
// We need to retrieve the restore sets before starting the restore.
// Otherwise, restorePackages() won't work as they need the restore sets cached internally.
if (session.getAvailableRestoreSets(observer, monitor) != 0) { if (session.getAvailableRestoreSets(observer, monitor) != 0) {
Log.e(TAG, "getAvailableRestoreSets() returned non-zero value") Log.e(TAG, "getAvailableRestoreSets() returned non-zero value")
mRestoreBackupResult.postValue( mRestoreBackupResult.postValue(
RestoreBackupResult(app.getString(R.string.restore_set_error)) RestoreBackupResult(app.getString(R.string.restore_set_error))
) )
@ -229,6 +245,7 @@ internal class RestoreViewModel(
// check previous package first and change status // check previous package first and change status
updateLatestPackage(list) updateLatestPackage(list)
// add current package // add current package
list.addFirst(AppRestoreResult(packageName, getAppName(app, packageName), IN_PROGRESS)) list.addFirst(AppRestoreResult(packageName, getAppName(app, packageName), IN_PROGRESS))
mRestoreProgress.postValue(list) mRestoreProgress.postValue(list)
@ -294,8 +311,27 @@ internal class RestoreViewModel(
} }
@WorkerThread @WorkerThread
private inner class RestoreObserver(private val startRestore: (RestoreObserver) -> Unit) : private inner class RestoreObserver(
IRestoreObserver.Stub() { private val restoreCoordinator: RestoreCoordinator,
private val restorableBackup: RestorableBackup,
private val session: IRestoreSession,
private val packages: List<String>,
private val monitor: BackupMonitor,
) : IRestoreObserver.Stub() {
/**
* The current package index.
*
* Used for splitting the packages into chunks.
*/
private var packageIndex: Int = 0
/**
* Map of results for each chunk.
*
* The key is the chunk index, the value is the result.
*/
private val chunkResults = mutableMapOf<Int, Int>()
/** /**
* Supply a list of the restore datasets available from the current transport. * Supply a list of the restore datasets available from the current transport.
@ -307,7 +343,33 @@ internal class RestoreViewModel(
* the current device. If no applicable datasets exist, restoreSets will be null. * the current device. If no applicable datasets exist, restoreSets will be null.
*/ */
override fun restoreSetsAvailable(restoreSets: Array<out RestoreSet>?) { override fun restoreSetsAvailable(restoreSets: Array<out RestoreSet>?) {
startRestore(this) // this gets executed after we got the restore sets
// now we can start the restore of all available packages
restoreNextPackages()
}
/**
* Restore the next chunk of packages.
*
* We need to restore in chunks, otherwise [BackupTransport.startRestore] in the
* framework's [PerformUnifiedRestoreTask] may fail due to an oversize Binder
* transaction, causing the entire restoration to fail.
*/
private fun restoreNextPackages() {
// Make sure metadata for selected backup is cached before starting each chunk.
val backupMetadata = restorableBackup.backupMetadata
restoreCoordinator.beforeStartRestore(backupMetadata)
val nextChunkIndex = (packageIndex + PACKAGES_PER_CHUNK).coerceAtMost(packages.size)
val packageChunk = packages.subList(packageIndex, nextChunkIndex).toTypedArray()
packageIndex += packageChunk.size
val token = backupMetadata.token
val result = session.restorePackages(token, this, packageChunk, monitor)
if (result != BackupManager.SUCCESS) {
Log.e(TAG, "restorePackages() returned non-zero value: $result")
}
} }
/** /**
@ -341,14 +403,35 @@ internal class RestoreViewModel(
* as a whole failed. * as a whole failed.
*/ */
override fun restoreFinished(result: Int) { override fun restoreFinished(result: Int) {
val restoreResult = RestoreBackupResult( val chunkIndex = packageIndex / PACKAGES_PER_CHUNK
if (result == 0) null chunkResults[chunkIndex] = result
else app.getString(R.string.restore_finished_error)
) // Restore next chunk if successful and there are more packages to restore.
onRestoreComplete(restoreResult) if (packageIndex < packages.size) {
restoreNextPackages()
return
}
// Restore finished, time to get the result.
onRestoreComplete(getRestoreResult())
closeSession() closeSession()
} }
private fun getRestoreResult(): RestoreBackupResult {
val failedChunks = chunkResults
.filter { it.value != BackupManager.SUCCESS }
.map { "chunk ${it.key} failed with error ${it.value}" }
return if (failedChunks.isNotEmpty()) {
Log.e(TAG, "Restore failed: $failedChunks")
return RestoreBackupResult(
errorMsg = app.getString(R.string.restore_finished_error)
)
} else {
RestoreBackupResult(errorMsg = null)
}
}
} }
@UiThread @UiThread