Getting Started¶
To obtain an instance of PackageInstaller
or PackageUninstaller
, use the getInstance(Context)
method:
val packageInstaller = PackageInstaller.getInstance(context)
val packageUninstaller = PackageUninstaller.getInstance(context)
var packageInstaller = PackageInstaller.getInstance(context);
var packageUninstaller = PackageUninstaller.getInstance(context);
Simple session launch¶
Launching an install or uninstall session with default parameters and getting its result back is as easy as writing this:
try {
when (val result = packageInstaller.createSession(apkUri).await()) {
Session.State.Succeeded -> println("Success")
is Session.State.Failed -> println(result.failure.message)
}
} catch (cancellationException: CancellationException) {
println("Cancelled")
throw cancellationException
} catch (exception: Exception) {
println(exception)
}
Session launches when await()
is called.
var subscriptions = new DisposableSubscriptionContainer();
var session = packageInstaller.createSession(new InstallParameters.Builder(apkUri).build());
session.addStateListener(subscriptions, new Session.TerminalStateListener<>(session) {
@Override
public void onSuccess(@NonNull UUID sessionId) {
System.out.println("Success");
}
@Override
public void onFailure(@NonNull UUID sessionId, @NonNull InstallFailure failure) {
if (failure instanceof Failure.Exceptional f) {
System.out.println(f.getException());
} else {
System.out.println(failure.getCause().getMessage());
}
}
@Override
public void onCancelled(@NonNull UUID sessionId) {
System.out.println("Cancelled");
}
});
Session launches when TerminalStateListener
is added to it.
It works as long as you don't care about UI lifecycle and unpredictable situations such as process death.
Handling UI lifecycle¶
If you're launching a session inside of a long-living service which is not expected to be killed (such as a foreground service), the previous example is good to go. However, when you are dealing with UI components such as Activities or Fragments, it's good practice to remove attached state listeners when appropriate:
When using ackpine-ktx
artifact and calling Session.await()
, the listener will be automatically detached when parent coroutine scope is cancelled. So if you're calling await()
inside of a viewModelScope
or lifecycleScope
, it should be fine. Note that cancelling await()
also cancels the session, this is done to respect coroutines' structured concurrency.
var subscriptions = new DisposableSubscriptionContainer();
var session = packageInstaller.createSession(...);
session.addStateListener(subscriptions, ...);
// when lifecycle is destroyed
subscriptions.clear();
Handling process death¶
Handling process death is not any different with Ackpine as with any other persisted state handling. You can save a session's ID and then re-retrieve the session from PackageInstaller
:
savedStateHandle[SESSION_ID_KEY] = session.id
// after process restart
val id: UUID? = savedStateHandle[SESSION_ID_KEY]
if (id != null) {
val result = packageInstaller.getSession(id)?.await()
// or anything else you want to do with the session
}
savedStateHandle.set(SESSION_ID_KEY, session.getId());
// after process restart
UUID id = savedStateHandle.get(SESSION_ID_KEY);
if (id != null) {
// using Guava
Futures.addCallback(packageInstaller.getSessionAsync(id), new FutureCallback<>() {
@Override
public void onSuccess(@Nullable ProgressSession<InstallFailure> session) {
if (session != null) {
session.addStateListener(subscriptions, ...);
// or anything else you want to do with the session
}
}
@Override
public void onFailure(@NonNull Throwable t) {
}
}, MoreExecutors.directExecutor());
}
Observing progress¶
Install sessions provide progress updates:
session.progress // Flow<Progress>
.onEach { progress ->
updateProgress(progress.progress, progress.max)
}
.launchIn(coroutineScope)
session.addProgressListener(subscriptions, (sessionId, progress) -> {
updateProgress(progress.getProgress(), progress.getMax());
});
Error handling¶
Error causes are delivered as Failure
objects through state listener or as a return value from await()
. They're sealed hierarchies of typed errors, and you can match on their type. For example:
val failure = failedResult.failure
val error = when (failure) {
is InstallFailure.Aborted -> "Aborted"
is InstallFailure.Blocked -> "Blocked by ${failure.otherPackageName}"
is InstallFailure.Conflict -> "Conflicting with ${failure.otherPackageName}"
is InstallFailure.Exceptional -> failure.exception.message
is InstallFailure.Generic -> "Generic failure"
is InstallFailure.Incompatible -> "Incompatible"
is InstallFailure.Invalid -> "Invalid"
is InstallFailure.Storage -> "Storage path: ${failure.storagePath}"
is InstallFailure.Timeout -> "Timeout"
else -> "Unknown failure"
}
var error = "";
if (failure instanceof InstallFailure.Aborted) {
error = "Aborted";
} else if (failure instanceof InstallFailure.Blocked f) {
error = "Blocked by " + f.getOtherPackageName();
} else if (failure instanceof InstallFailure.Conflict f) {
error = "Conflicting with " + f.getOtherPackageName();
} else if (failure instanceof InstallFailure.Exceptional f) {
error = f.getException().getMessage();
} else if (failure instanceof InstallFailure.Generic) {
error = "Generic failure";
} else if (failure instanceof InstallFailure.Incompatible) {
error = "Incompatible";
} else if (failure instanceof InstallFailure.Invalid) {
error = "Invalid";
} else if (failure instanceof InstallFailure.Storage f) {
error = "Storage path: " + f.getStoragePath();
} else if (failure instanceof InstallFailure.Timeout) {
error = "Timeout";
} else {
error = "Unknown failure";
}
When using await()
, exceptions are never delivered as a Failure.Exceptional
object. Instead, they are thrown.
Every example on this page is using PackageInstaller
, but APIs for PackageUninstaller
are absolutely the same except for progress updates.