1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.content.pm;
18 
19 import android.Manifest;
20 import android.annotation.CurrentTimeMillisLong;
21 import android.annotation.IntDef;
22 import android.annotation.NonNull;
23 import android.annotation.Nullable;
24 import android.annotation.RequiresPermission;
25 import android.annotation.SdkConstant;
26 import android.annotation.SdkConstant.SdkConstantType;
27 import android.annotation.SystemApi;
28 import android.annotation.TestApi;
29 import android.annotation.UnsupportedAppUsage;
30 import android.app.ActivityManager;
31 import android.app.AppGlobals;
32 import android.content.Intent;
33 import android.content.IntentSender;
34 import android.content.pm.PackageManager.DeleteFlags;
35 import android.content.pm.PackageManager.InstallReason;
36 import android.graphics.Bitmap;
37 import android.net.Uri;
38 import android.os.Build;
39 import android.os.FileBridge;
40 import android.os.Handler;
41 import android.os.HandlerExecutor;
42 import android.os.Parcel;
43 import android.os.ParcelFileDescriptor;
44 import android.os.Parcelable;
45 import android.os.ParcelableException;
46 import android.os.RemoteException;
47 import android.os.SystemProperties;
48 import android.os.UserHandle;
49 import android.system.ErrnoException;
50 import android.system.Os;
51 import android.util.ArraySet;
52 import android.util.ExceptionUtils;
53 
54 import com.android.internal.util.IndentingPrintWriter;
55 import com.android.internal.util.Preconditions;
56 import com.android.internal.util.function.pooled.PooledLambda;
57 
58 import java.io.Closeable;
59 import java.io.IOException;
60 import java.io.InputStream;
61 import java.io.OutputStream;
62 import java.lang.annotation.Retention;
63 import java.lang.annotation.RetentionPolicy;
64 import java.security.MessageDigest;
65 import java.util.ArrayList;
66 import java.util.Collections;
67 import java.util.Iterator;
68 import java.util.List;
69 import java.util.Set;
70 import java.util.concurrent.Executor;
71 
72 /**
73  * Offers the ability to install, upgrade, and remove applications on the
74  * device. This includes support for apps packaged either as a single
75  * "monolithic" APK, or apps packaged as multiple "split" APKs.
76  * <p>
77  * An app is delivered for installation through a
78  * {@link PackageInstaller.Session}, which any app can create. Once the session
79  * is created, the installer can stream one or more APKs into place until it
80  * decides to either commit or destroy the session. Committing may require user
81  * intervention to complete the installation, unless the caller falls into one of the
82  * following categories, in which case the installation will complete automatically.
83  * <ul>
84  * <li>the device owner
85  * <li>the affiliated profile owner
86  * </ul>
87  * <p>
88  * Sessions can install brand new apps, upgrade existing apps, or add new splits
89  * into an existing app.
90  * <p>
91  * Apps packaged as multiple split APKs always consist of a single "base" APK
92  * (with a {@code null} split name) and zero or more "split" APKs (with unique
93  * split names). Any subset of these APKs can be installed together, as long as
94  * the following constraints are met:
95  * <ul>
96  * <li>All APKs must have the exact same package name, version code, and signing
97  * certificates.
98  * <li>All APKs must have unique split names.
99  * <li>All installations must contain a single base APK.
100  * </ul>
101  * <p>
102  * The ApiDemos project contains examples of using this API:
103  * <code>ApiDemos/src/com/example/android/apis/content/InstallApk*.java</code>.
104  * <p>
105  * On Android Q or above, an app installed notification will be posted
106  * by system after a new app is installed.
107  * To customize installer's notification icon, you should declare the following in the manifest
108  * &lt;application> as follows: </p>
109  * <pre>
110  * &lt;meta-data android:name="com.android.packageinstaller.notification.smallIcon"
111  * android:resource="@drawable/installer_notification_icon"/>
112  * </pre>
113  * <pre>
114  * &lt;meta-data android:name="com.android.packageinstaller.notification.color"
115  * android:resource="@color/installer_notification_color"/>
116  * </pre>
117  */
118 public class PackageInstaller {
119     private static final String TAG = "PackageInstaller";
120 
121     /** {@hide} */
122     public static final boolean ENABLE_REVOCABLE_FD =
123             SystemProperties.getBoolean("fw.revocable_fd", false);
124 
125     /**
126      * Activity Action: Show details about a particular install session. This
127      * may surface actions such as pause, resume, or cancel.
128      * <p>
129      * This should always be scoped to the installer package that owns the
130      * session. Clients should use {@link SessionInfo#createDetailsIntent()} to
131      * build this intent correctly.
132      * <p>
133      * In some cases, a matching Activity may not exist, so ensure you safeguard
134      * against this.
135      * <p>
136      * The session to show details for is defined in {@link #EXTRA_SESSION_ID}.
137      */
138     @SdkConstant(SdkConstantType.ACTIVITY_INTENT_ACTION)
139     public static final String ACTION_SESSION_DETAILS = "android.content.pm.action.SESSION_DETAILS";
140 
141     /**
142      * Broadcast Action: Explicit broadcast sent to the last known default launcher when a session
143      * for a new install is committed. For managed profile, this is sent to the default launcher
144      * of the primary profile.
145      * <p>
146      * The associated session is defined in {@link #EXTRA_SESSION} and the user for which this
147      * session was created in {@link Intent#EXTRA_USER}.
148      */
149     @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION)
150     public static final String ACTION_SESSION_COMMITTED =
151             "android.content.pm.action.SESSION_COMMITTED";
152 
153     /**
154      * Broadcast Action: Send information about a staged install session when its state is updated.
155      * <p>
156      * The associated session information is defined in {@link #EXTRA_SESSION}.
157      */
158     @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION)
159     public static final String ACTION_SESSION_UPDATED =
160             "android.content.pm.action.SESSION_UPDATED";
161 
162     /** {@hide} */
163     public static final String ACTION_CONFIRM_INSTALL = "android.content.pm.action.CONFIRM_INSTALL";
164 
165     /**
166      * An integer session ID that an operation is working with.
167      *
168      * @see Intent#getIntExtra(String, int)
169      */
170     public static final String EXTRA_SESSION_ID = "android.content.pm.extra.SESSION_ID";
171 
172     /**
173      * {@link SessionInfo} that an operation is working with.
174      *
175      * @see Intent#getParcelableExtra(String)
176      */
177     public static final String EXTRA_SESSION = "android.content.pm.extra.SESSION";
178 
179     /**
180      * Package name that an operation is working with.
181      *
182      * @see Intent#getStringExtra(String)
183      */
184     public static final String EXTRA_PACKAGE_NAME = "android.content.pm.extra.PACKAGE_NAME";
185 
186     /**
187      * Current status of an operation. Will be one of
188      * {@link #STATUS_PENDING_USER_ACTION}, {@link #STATUS_SUCCESS},
189      * {@link #STATUS_FAILURE}, {@link #STATUS_FAILURE_ABORTED},
190      * {@link #STATUS_FAILURE_BLOCKED}, {@link #STATUS_FAILURE_CONFLICT},
191      * {@link #STATUS_FAILURE_INCOMPATIBLE}, {@link #STATUS_FAILURE_INVALID}, or
192      * {@link #STATUS_FAILURE_STORAGE}.
193      * <p>
194      * More information about a status may be available through additional
195      * extras; see the individual status documentation for details.
196      *
197      * @see Intent#getIntExtra(String, int)
198      */
199     public static final String EXTRA_STATUS = "android.content.pm.extra.STATUS";
200 
201     /**
202      * Detailed string representation of the status, including raw details that
203      * are useful for debugging.
204      *
205      * @see Intent#getStringExtra(String)
206      */
207     public static final String EXTRA_STATUS_MESSAGE = "android.content.pm.extra.STATUS_MESSAGE";
208 
209     /**
210      * Another package name relevant to a status. This is typically the package
211      * responsible for causing an operation failure.
212      *
213      * @see Intent#getStringExtra(String)
214      */
215     public static final String
216             EXTRA_OTHER_PACKAGE_NAME = "android.content.pm.extra.OTHER_PACKAGE_NAME";
217 
218     /**
219      * Storage path relevant to a status.
220      *
221      * @see Intent#getStringExtra(String)
222      */
223     public static final String EXTRA_STORAGE_PATH = "android.content.pm.extra.STORAGE_PATH";
224 
225     /** {@hide} */
226     @Deprecated
227     public static final String EXTRA_PACKAGE_NAMES = "android.content.pm.extra.PACKAGE_NAMES";
228 
229     /** {@hide} */
230     public static final String EXTRA_LEGACY_STATUS = "android.content.pm.extra.LEGACY_STATUS";
231     /** {@hide} */
232     public static final String EXTRA_LEGACY_BUNDLE = "android.content.pm.extra.LEGACY_BUNDLE";
233     /** {@hide} */
234     public static final String EXTRA_CALLBACK = "android.content.pm.extra.CALLBACK";
235 
236     /**
237      * User action is currently required to proceed. You can launch the intent
238      * activity described by {@link Intent#EXTRA_INTENT} to involve the user and
239      * continue.
240      * <p>
241      * You may choose to immediately launch the intent if the user is actively
242      * using your app. Otherwise, you should use a notification to guide the
243      * user back into your app before launching.
244      *
245      * @see Intent#getParcelableExtra(String)
246      */
247     public static final int STATUS_PENDING_USER_ACTION = -1;
248 
249     /**
250      * The operation succeeded.
251      */
252     public static final int STATUS_SUCCESS = 0;
253 
254     /**
255      * The operation failed in a generic way. The system will always try to
256      * provide a more specific failure reason, but in some rare cases this may
257      * be delivered.
258      *
259      * @see #EXTRA_STATUS_MESSAGE
260      */
261     public static final int STATUS_FAILURE = 1;
262 
263     /**
264      * The operation failed because it was blocked. For example, a device policy
265      * may be blocking the operation, a package verifier may have blocked the
266      * operation, or the app may be required for core system operation.
267      * <p>
268      * The result may also contain {@link #EXTRA_OTHER_PACKAGE_NAME} with the
269      * specific package blocking the install.
270      *
271      * @see #EXTRA_STATUS_MESSAGE
272      * @see #EXTRA_OTHER_PACKAGE_NAME
273      */
274     public static final int STATUS_FAILURE_BLOCKED = 2;
275 
276     /**
277      * The operation failed because it was actively aborted. For example, the
278      * user actively declined requested permissions, or the session was
279      * abandoned.
280      *
281      * @see #EXTRA_STATUS_MESSAGE
282      */
283     public static final int STATUS_FAILURE_ABORTED = 3;
284 
285     /**
286      * The operation failed because one or more of the APKs was invalid. For
287      * example, they might be malformed, corrupt, incorrectly signed,
288      * mismatched, etc.
289      *
290      * @see #EXTRA_STATUS_MESSAGE
291      */
292     public static final int STATUS_FAILURE_INVALID = 4;
293 
294     /**
295      * The operation failed because it conflicts (or is inconsistent with) with
296      * another package already installed on the device. For example, an existing
297      * permission, incompatible certificates, etc. The user may be able to
298      * uninstall another app to fix the issue.
299      * <p>
300      * The result may also contain {@link #EXTRA_OTHER_PACKAGE_NAME} with the
301      * specific package identified as the cause of the conflict.
302      *
303      * @see #EXTRA_STATUS_MESSAGE
304      * @see #EXTRA_OTHER_PACKAGE_NAME
305      */
306     public static final int STATUS_FAILURE_CONFLICT = 5;
307 
308     /**
309      * The operation failed because of storage issues. For example, the device
310      * may be running low on space, or external media may be unavailable. The
311      * user may be able to help free space or insert different external media.
312      * <p>
313      * The result may also contain {@link #EXTRA_STORAGE_PATH} with the path to
314      * the storage device that caused the failure.
315      *
316      * @see #EXTRA_STATUS_MESSAGE
317      * @see #EXTRA_STORAGE_PATH
318      */
319     public static final int STATUS_FAILURE_STORAGE = 6;
320 
321     /**
322      * The operation failed because it is fundamentally incompatible with this
323      * device. For example, the app may require a hardware feature that doesn't
324      * exist, it may be missing native code for the ABIs supported by the
325      * device, or it requires a newer SDK version, etc.
326      *
327      * @see #EXTRA_STATUS_MESSAGE
328      */
329     public static final int STATUS_FAILURE_INCOMPATIBLE = 7;
330 
331     private final IPackageInstaller mInstaller;
332     private final int mUserId;
333     private final String mInstallerPackageName;
334 
335     private final ArrayList<SessionCallbackDelegate> mDelegates = new ArrayList<>();
336 
337     /** {@hide} */
PackageInstaller(IPackageInstaller installer, String installerPackageName, int userId)338     public PackageInstaller(IPackageInstaller installer,
339             String installerPackageName, int userId) {
340         mInstaller = installer;
341         mInstallerPackageName = installerPackageName;
342         mUserId = userId;
343     }
344 
345     /**
346      * Create a new session using the given parameters, returning a unique ID
347      * that represents the session. Once created, the session can be opened
348      * multiple times across multiple device boots.
349      * <p>
350      * The system may automatically destroy sessions that have not been
351      * finalized (either committed or abandoned) within a reasonable period of
352      * time, typically on the order of a day.
353      *
354      * @throws IOException if parameters were unsatisfiable, such as lack of
355      *             disk space or unavailable media.
356      * @throws SecurityException when installation services are unavailable,
357      *             such as when called from a restricted user.
358      * @throws IllegalArgumentException when {@link SessionParams} is invalid.
359      * @return positive, non-zero unique ID that represents the created session.
360      *         This ID remains consistent across device reboots until the
361      *         session is finalized. IDs are not reused during a given boot.
362      */
createSession(@onNull SessionParams params)363     public int createSession(@NonNull SessionParams params) throws IOException {
364         try {
365             final String installerPackage;
366             if (params.installerPackageName == null) {
367                 installerPackage = mInstallerPackageName;
368             } else {
369                 installerPackage = params.installerPackageName;
370             }
371 
372             return mInstaller.createSession(params, installerPackage, mUserId);
373         } catch (RuntimeException e) {
374             ExceptionUtils.maybeUnwrapIOException(e);
375             throw e;
376         } catch (RemoteException e) {
377             throw e.rethrowFromSystemServer();
378         }
379     }
380 
381     /**
382      * Open an existing session to actively perform work. To succeed, the caller
383      * must be the owner of the install session.
384      *
385      * @throws IOException if parameters were unsatisfiable, such as lack of
386      *             disk space or unavailable media.
387      * @throws SecurityException when the caller does not own the session, or
388      *             the session is invalid.
389      */
openSession(int sessionId)390     public @NonNull Session openSession(int sessionId) throws IOException {
391         try {
392             try {
393                 return new Session(mInstaller.openSession(sessionId));
394             } catch (RemoteException e) {
395                 throw e.rethrowFromSystemServer();
396             }
397         } catch (RuntimeException e) {
398             ExceptionUtils.maybeUnwrapIOException(e);
399             throw e;
400         }
401     }
402 
403     /**
404      * Update the icon representing the app being installed in a specific
405      * session. This should be roughly
406      * {@link ActivityManager#getLauncherLargeIconSize()} in both dimensions.
407      *
408      * @throws SecurityException when the caller does not own the session, or
409      *             the session is invalid.
410      */
updateSessionAppIcon(int sessionId, @Nullable Bitmap appIcon)411     public void updateSessionAppIcon(int sessionId, @Nullable Bitmap appIcon) {
412         try {
413             mInstaller.updateSessionAppIcon(sessionId, appIcon);
414         } catch (RemoteException e) {
415             throw e.rethrowFromSystemServer();
416         }
417     }
418 
419     /**
420      * Update the label representing the app being installed in a specific
421      * session.
422      *
423      * @throws SecurityException when the caller does not own the session, or
424      *             the session is invalid.
425      */
updateSessionAppLabel(int sessionId, @Nullable CharSequence appLabel)426     public void updateSessionAppLabel(int sessionId, @Nullable CharSequence appLabel) {
427         try {
428             final String val = (appLabel != null) ? appLabel.toString() : null;
429             mInstaller.updateSessionAppLabel(sessionId, val);
430         } catch (RemoteException e) {
431             throw e.rethrowFromSystemServer();
432         }
433     }
434 
435     /**
436      * Completely abandon the given session, destroying all staged data and
437      * rendering it invalid. Abandoned sessions will be reported to
438      * {@link SessionCallback} listeners as failures. This is equivalent to
439      * opening the session and calling {@link Session#abandon()}.
440      *
441      * @throws SecurityException when the caller does not own the session, or
442      *             the session is invalid.
443      */
abandonSession(int sessionId)444     public void abandonSession(int sessionId) {
445         try {
446             mInstaller.abandonSession(sessionId);
447         } catch (RemoteException e) {
448             throw e.rethrowFromSystemServer();
449         }
450     }
451 
452     /**
453      * Return details for a specific session. No special permissions are
454      * required to retrieve these details.
455      *
456      * @return details for the requested session, or {@code null} if the session
457      *         does not exist.
458      */
getSessionInfo(int sessionId)459     public @Nullable SessionInfo getSessionInfo(int sessionId) {
460         try {
461             return mInstaller.getSessionInfo(sessionId);
462         } catch (RemoteException e) {
463             throw e.rethrowFromSystemServer();
464         }
465     }
466 
467     /**
468      * Return list of all known install sessions, regardless of the installer.
469      */
getAllSessions()470     public @NonNull List<SessionInfo> getAllSessions() {
471         try {
472             return mInstaller.getAllSessions(mUserId).getList();
473         } catch (RemoteException e) {
474             throw e.rethrowFromSystemServer();
475         }
476     }
477 
478     /**
479      * Return list of all known install sessions owned by the calling app.
480      */
getMySessions()481     public @NonNull List<SessionInfo> getMySessions() {
482         try {
483             return mInstaller.getMySessions(mInstallerPackageName, mUserId).getList();
484         } catch (RemoteException e) {
485             throw e.rethrowFromSystemServer();
486         }
487     }
488 
489     /**
490      * Return list of all staged install sessions.
491      */
getStagedSessions()492     public @NonNull List<SessionInfo> getStagedSessions() {
493         try {
494             // TODO: limit this to the mUserId?
495             return mInstaller.getStagedSessions().getList();
496         } catch (RemoteException e) {
497             throw e.rethrowFromSystemServer();
498         }
499     }
500 
501     /**
502      * Returns an active staged session, or {@code null} if there is none.
503      *
504      * <p>Staged session is active iff:
505      * <ul>
506      *     <li>It is committed, i.e. {@link SessionInfo#isCommitted()} is {@code true}, and
507      *     <li>it is not applied, i.e. {@link SessionInfo#isStagedSessionApplied()} is {@code
508      *     false}, and
509      *     <li>it is not failed, i.e. {@link SessionInfo#isStagedSessionFailed()} is {@code false}.
510      * </ul>
511      *
512      * <p>In case of a multi-apk session, reasoning above is applied to the parent session, since
513      * that is the one that should been {@link Session#commit committed}.
514      */
getActiveStagedSession()515     public @Nullable SessionInfo getActiveStagedSession() {
516         final List<SessionInfo> stagedSessions = getStagedSessions();
517         for (SessionInfo s : stagedSessions) {
518             if (s.isStagedSessionApplied() || s.isStagedSessionFailed()) {
519                 // Finalized session.
520                 continue;
521             }
522             if (s.getParentSessionId() != SessionInfo.INVALID_ID) {
523                 // Child session.
524                 continue;
525             }
526             if (s.isCommitted()) {
527                 return s;
528             }
529         }
530         return null;
531     }
532 
533     /**
534      * Uninstall the given package, removing it completely from the device. This
535      * method is available to:
536      * <ul>
537      * <li>the current "installer of record" for the package
538      * <li>the device owner
539      * <li>the affiliated profile owner
540      * </ul>
541      *
542      * @param packageName The package to uninstall.
543      * @param statusReceiver Where to deliver the result.
544      *
545      * @see android.app.admin.DevicePolicyManager
546      */
547     @RequiresPermission(anyOf = {
548             Manifest.permission.DELETE_PACKAGES,
549             Manifest.permission.REQUEST_DELETE_PACKAGES})
uninstall(@onNull String packageName, @NonNull IntentSender statusReceiver)550     public void uninstall(@NonNull String packageName, @NonNull IntentSender statusReceiver) {
551         uninstall(packageName, 0 /*flags*/, statusReceiver);
552     }
553 
554     /**
555      * Uninstall the given package, removing it completely from the device. This
556      * method is only available to the current "installer of record" for the
557      * package.
558      *
559      * @param packageName The package to uninstall.
560      * @param flags Flags for uninstall.
561      * @param statusReceiver Where to deliver the result.
562      *
563      * @hide
564      */
uninstall(@onNull String packageName, @DeleteFlags int flags, @NonNull IntentSender statusReceiver)565     public void uninstall(@NonNull String packageName, @DeleteFlags int flags,
566             @NonNull IntentSender statusReceiver) {
567         uninstall(new VersionedPackage(packageName, PackageManager.VERSION_CODE_HIGHEST),
568                 flags, statusReceiver);
569     }
570 
571     /**
572      * Uninstall the given package with a specific version code, removing it
573      * completely from the device. If the version code of the package
574      * does not match the one passed in the versioned package argument this
575      * method is a no-op. Use {@link PackageManager#VERSION_CODE_HIGHEST} to
576      * uninstall the latest version of the package.
577      * <p>
578      * This method is available to:
579      * <ul>
580      * <li>the current "installer of record" for the package
581      * <li>the device owner
582      * <li>the affiliated profile owner
583      * </ul>
584      *
585      * @param versionedPackage The versioned package to uninstall.
586      * @param statusReceiver Where to deliver the result.
587      *
588      * @see android.app.admin.DevicePolicyManager
589      */
590     @RequiresPermission(anyOf = {
591             Manifest.permission.DELETE_PACKAGES,
592             Manifest.permission.REQUEST_DELETE_PACKAGES})
uninstall(@onNull VersionedPackage versionedPackage, @NonNull IntentSender statusReceiver)593     public void uninstall(@NonNull VersionedPackage versionedPackage,
594             @NonNull IntentSender statusReceiver) {
595         uninstall(versionedPackage, 0 /*flags*/, statusReceiver);
596     }
597 
598     /**
599      * Uninstall the given package with a specific version code, removing it
600      * completely from the device. This method is only available to the current
601      * "installer of record" for the package. If the version code of the package
602      * does not match the one passed in the versioned package argument this
603      * method is a no-op. Use {@link PackageManager#VERSION_CODE_HIGHEST} to
604      * uninstall the latest version of the package.
605      *
606      * @param versionedPackage The versioned package to uninstall.
607      * @param flags Flags for uninstall.
608      * @param statusReceiver Where to deliver the result.
609      *
610      * @hide
611      */
612     @RequiresPermission(anyOf = {
613             Manifest.permission.DELETE_PACKAGES,
614             Manifest.permission.REQUEST_DELETE_PACKAGES})
uninstall(@onNull VersionedPackage versionedPackage, @DeleteFlags int flags, @NonNull IntentSender statusReceiver)615     public void uninstall(@NonNull VersionedPackage versionedPackage, @DeleteFlags int flags,
616             @NonNull IntentSender statusReceiver) {
617         Preconditions.checkNotNull(versionedPackage, "versionedPackage cannot be null");
618         try {
619             mInstaller.uninstall(versionedPackage, mInstallerPackageName,
620                     flags, statusReceiver, mUserId);
621         } catch (RemoteException e) {
622             throw e.rethrowFromSystemServer();
623         }
624     }
625 
626     /**
627      * Install the given package, which already exists on the device, for the user for which this
628      * installer was created.
629      *
630      * <p>This will
631      * {@link PackageInstaller.SessionParams#setWhitelistedRestrictedPermissions(Set) whitelist
632      * all restricted permissions}.
633      *
634      * @param packageName The package to install.
635      * @param installReason Reason for install.
636      * @param statusReceiver Where to deliver the result.
637      */
638     @RequiresPermission(allOf = {
639             Manifest.permission.INSTALL_PACKAGES,
640             Manifest.permission.INSTALL_EXISTING_PACKAGES})
installExistingPackage(@onNull String packageName, @InstallReason int installReason, @Nullable IntentSender statusReceiver)641     public void installExistingPackage(@NonNull String packageName,
642             @InstallReason int installReason,
643             @Nullable IntentSender statusReceiver) {
644         Preconditions.checkNotNull(packageName, "packageName cannot be null");
645         try {
646             mInstaller.installExistingPackage(packageName,
647                     PackageManager.INSTALL_ALL_WHITELIST_RESTRICTED_PERMISSIONS, installReason,
648                     statusReceiver, mUserId, null);
649         } catch (RemoteException e) {
650             throw e.rethrowFromSystemServer();
651         }
652     }
653 
654 
655     /** {@hide} */
656     @SystemApi
657     @RequiresPermission(android.Manifest.permission.INSTALL_PACKAGES)
setPermissionsResult(int sessionId, boolean accepted)658     public void setPermissionsResult(int sessionId, boolean accepted) {
659         try {
660             mInstaller.setPermissionsResult(sessionId, accepted);
661         } catch (RemoteException e) {
662             throw e.rethrowFromSystemServer();
663         }
664     }
665 
666     /**
667      * Events for observing session lifecycle.
668      * <p>
669      * A typical session lifecycle looks like this:
670      * <ul>
671      * <li>An installer creates a session to indicate pending app delivery. All
672      * install details are available at this point.
673      * <li>The installer opens the session to deliver APK data. Note that a
674      * session may be opened and closed multiple times as network connectivity
675      * changes. The installer may deliver periodic progress updates.
676      * <li>The installer commits or abandons the session, resulting in the
677      * session being finished.
678      * </ul>
679      */
680     public static abstract class SessionCallback {
681         /**
682          * New session has been created. Details about the session can be
683          * obtained from {@link PackageInstaller#getSessionInfo(int)}.
684          */
onCreated(int sessionId)685         public abstract void onCreated(int sessionId);
686 
687         /**
688          * Badging details for an existing session has changed. For example, the
689          * app icon or label has been updated.
690          */
onBadgingChanged(int sessionId)691         public abstract void onBadgingChanged(int sessionId);
692 
693         /**
694          * Active state for session has been changed.
695          * <p>
696          * A session is considered active whenever there is ongoing forward
697          * progress being made, such as the installer holding an open
698          * {@link Session} instance while streaming data into place, or the
699          * system optimizing code as the result of
700          * {@link Session#commit(IntentSender)}.
701          * <p>
702          * If the installer closes the {@link Session} without committing, the
703          * session is considered inactive until the installer opens the session
704          * again.
705          */
onActiveChanged(int sessionId, boolean active)706         public abstract void onActiveChanged(int sessionId, boolean active);
707 
708         /**
709          * Progress for given session has been updated.
710          * <p>
711          * Note that this progress may not directly correspond to the value
712          * reported by
713          * {@link PackageInstaller.Session#setStagingProgress(float)}, as the
714          * system may carve out a portion of the overall progress to represent
715          * its own internal installation work.
716          */
onProgressChanged(int sessionId, float progress)717         public abstract void onProgressChanged(int sessionId, float progress);
718 
719         /**
720          * Session has completely finished, either with success or failure.
721          */
onFinished(int sessionId, boolean success)722         public abstract void onFinished(int sessionId, boolean success);
723     }
724 
725     /** {@hide} */
726     static class SessionCallbackDelegate extends IPackageInstallerCallback.Stub {
727         private static final int MSG_SESSION_CREATED = 1;
728         private static final int MSG_SESSION_BADGING_CHANGED = 2;
729         private static final int MSG_SESSION_ACTIVE_CHANGED = 3;
730         private static final int MSG_SESSION_PROGRESS_CHANGED = 4;
731         private static final int MSG_SESSION_FINISHED = 5;
732 
733         final SessionCallback mCallback;
734         final Executor mExecutor;
735 
SessionCallbackDelegate(SessionCallback callback, Executor executor)736         SessionCallbackDelegate(SessionCallback callback, Executor executor) {
737             mCallback = callback;
738             mExecutor = executor;
739         }
740 
741         @Override
onSessionCreated(int sessionId)742         public void onSessionCreated(int sessionId) {
743             mExecutor.execute(PooledLambda.obtainRunnable(SessionCallback::onCreated, mCallback,
744                     sessionId).recycleOnUse());
745         }
746 
747         @Override
onSessionBadgingChanged(int sessionId)748         public void onSessionBadgingChanged(int sessionId) {
749             mExecutor.execute(PooledLambda.obtainRunnable(SessionCallback::onBadgingChanged,
750                     mCallback, sessionId).recycleOnUse());
751         }
752 
753         @Override
onSessionActiveChanged(int sessionId, boolean active)754         public void onSessionActiveChanged(int sessionId, boolean active) {
755             mExecutor.execute(PooledLambda.obtainRunnable(SessionCallback::onActiveChanged,
756                     mCallback, sessionId, active).recycleOnUse());
757         }
758 
759         @Override
onSessionProgressChanged(int sessionId, float progress)760         public void onSessionProgressChanged(int sessionId, float progress) {
761             mExecutor.execute(PooledLambda.obtainRunnable(SessionCallback::onProgressChanged,
762                     mCallback, sessionId, progress).recycleOnUse());
763         }
764 
765         @Override
onSessionFinished(int sessionId, boolean success)766         public void onSessionFinished(int sessionId, boolean success) {
767             mExecutor.execute(PooledLambda.obtainRunnable(SessionCallback::onFinished,
768                     mCallback, sessionId, success).recycleOnUse());
769         }
770     }
771 
772     /** {@hide} */
773     @Deprecated
addSessionCallback(@onNull SessionCallback callback)774     public void addSessionCallback(@NonNull SessionCallback callback) {
775         registerSessionCallback(callback);
776     }
777 
778     /**
779      * Register to watch for session lifecycle events. No special permissions
780      * are required to watch for these events.
781      */
registerSessionCallback(@onNull SessionCallback callback)782     public void registerSessionCallback(@NonNull SessionCallback callback) {
783         registerSessionCallback(callback, new Handler());
784     }
785 
786     /** {@hide} */
787     @Deprecated
addSessionCallback(@onNull SessionCallback callback, @NonNull Handler handler)788     public void addSessionCallback(@NonNull SessionCallback callback, @NonNull Handler handler) {
789         registerSessionCallback(callback, handler);
790     }
791 
792     /**
793      * Register to watch for session lifecycle events. No special permissions
794      * are required to watch for these events.
795      *
796      * @param handler to dispatch callback events through, otherwise uses
797      *            calling thread.
798      */
registerSessionCallback(@onNull SessionCallback callback, @NonNull Handler handler)799     public void registerSessionCallback(@NonNull SessionCallback callback, @NonNull Handler handler) {
800         synchronized (mDelegates) {
801             final SessionCallbackDelegate delegate = new SessionCallbackDelegate(callback,
802                     new HandlerExecutor(handler));
803             try {
804                 mInstaller.registerCallback(delegate, mUserId);
805             } catch (RemoteException e) {
806                 throw e.rethrowFromSystemServer();
807             }
808             mDelegates.add(delegate);
809         }
810     }
811 
812     /** {@hide} */
813     @Deprecated
removeSessionCallback(@onNull SessionCallback callback)814     public void removeSessionCallback(@NonNull SessionCallback callback) {
815         unregisterSessionCallback(callback);
816     }
817 
818     /**
819      * Unregister a previously registered callback.
820      */
unregisterSessionCallback(@onNull SessionCallback callback)821     public void unregisterSessionCallback(@NonNull SessionCallback callback) {
822         synchronized (mDelegates) {
823             for (Iterator<SessionCallbackDelegate> i = mDelegates.iterator(); i.hasNext();) {
824                 final SessionCallbackDelegate delegate = i.next();
825                 if (delegate.mCallback == callback) {
826                     try {
827                         mInstaller.unregisterCallback(delegate);
828                     } catch (RemoteException e) {
829                         throw e.rethrowFromSystemServer();
830                     }
831                     i.remove();
832                 }
833             }
834         }
835     }
836 
837     /**
838      * An installation that is being actively staged. For an install to succeed,
839      * all existing and new packages must have identical package names, version
840      * codes, and signing certificates.
841      * <p>
842      * A session may contain any number of split packages. If the application
843      * does not yet exist, this session must include a base package.
844      * <p>
845      * If an APK included in this session is already defined by the existing
846      * installation (for example, the same split name), the APK in this session
847      * will replace the existing APK.
848      * <p>
849      * In such a case that multiple packages need to be committed simultaneously,
850      * multiple sessions can be referenced by a single multi-package session.
851      * This session is created with no package name and calling
852      * {@link SessionParams#setMultiPackage()}. The individual session IDs can be
853      * added with {@link #addChildSessionId(int)} and commit of the multi-package
854      * session will result in all child sessions being committed atomically.
855      */
856     public static class Session implements Closeable {
857         /** {@hide} */
858         protected final IPackageInstallerSession mSession;
859 
860         /** {@hide} */
Session(IPackageInstallerSession session)861         public Session(IPackageInstallerSession session) {
862             mSession = session;
863         }
864 
865         /** {@hide} */
866         @Deprecated
setProgress(float progress)867         public void setProgress(float progress) {
868             setStagingProgress(progress);
869         }
870 
871         /**
872          * Set current progress of staging this session. Valid values are
873          * anywhere between 0 and 1.
874          * <p>
875          * Note that this progress may not directly correspond to the value
876          * reported by {@link SessionCallback#onProgressChanged(int, float)}, as
877          * the system may carve out a portion of the overall progress to
878          * represent its own internal installation work.
879          */
setStagingProgress(float progress)880         public void setStagingProgress(float progress) {
881             try {
882                 mSession.setClientProgress(progress);
883             } catch (RemoteException e) {
884                 throw e.rethrowFromSystemServer();
885             }
886         }
887 
888         /** {@hide} */
889         @UnsupportedAppUsage
addProgress(float progress)890         public void addProgress(float progress) {
891             try {
892                 mSession.addClientProgress(progress);
893             } catch (RemoteException e) {
894                 throw e.rethrowFromSystemServer();
895             }
896         }
897 
898         /**
899          * Open a stream to write an APK file into the session.
900          * <p>
901          * The returned stream will start writing data at the requested offset
902          * in the underlying file, which can be used to resume a partially
903          * written file. If a valid file length is specified, the system will
904          * preallocate the underlying disk space to optimize placement on disk.
905          * It's strongly recommended to provide a valid file length when known.
906          * <p>
907          * You can write data into the returned stream, optionally call
908          * {@link #fsync(OutputStream)} as needed to ensure bytes have been
909          * persisted to disk, and then close when finished. All streams must be
910          * closed before calling {@link #commit(IntentSender)}.
911          *
912          * @param name arbitrary, unique name of your choosing to identify the
913          *            APK being written. You can open a file again for
914          *            additional writes (such as after a reboot) by using the
915          *            same name. This name is only meaningful within the context
916          *            of a single install session.
917          * @param offsetBytes offset into the file to begin writing at, or 0 to
918          *            start at the beginning of the file.
919          * @param lengthBytes total size of the file being written, used to
920          *            preallocate the underlying disk space, or -1 if unknown.
921          *            The system may clear various caches as needed to allocate
922          *            this space.
923          * @throws IOException if trouble opening the file for writing, such as
924          *             lack of disk space or unavailable media.
925          * @throws SecurityException if called after the session has been
926          *             sealed or abandoned
927          */
openWrite(@onNull String name, long offsetBytes, long lengthBytes)928         public @NonNull OutputStream openWrite(@NonNull String name, long offsetBytes,
929                 long lengthBytes) throws IOException {
930             try {
931                 if (ENABLE_REVOCABLE_FD) {
932                     return new ParcelFileDescriptor.AutoCloseOutputStream(
933                             mSession.openWrite(name, offsetBytes, lengthBytes));
934                 } else {
935                     final ParcelFileDescriptor clientSocket = mSession.openWrite(name,
936                             offsetBytes, lengthBytes);
937                     return new FileBridge.FileBridgeOutputStream(clientSocket);
938                 }
939             } catch (RuntimeException e) {
940                 ExceptionUtils.maybeUnwrapIOException(e);
941                 throw e;
942             } catch (RemoteException e) {
943                 throw e.rethrowFromSystemServer();
944             }
945         }
946 
947         /** {@hide} */
write(@onNull String name, long offsetBytes, long lengthBytes, @NonNull ParcelFileDescriptor fd)948         public void write(@NonNull String name, long offsetBytes, long lengthBytes,
949                 @NonNull ParcelFileDescriptor fd) throws IOException {
950             try {
951                 mSession.write(name, offsetBytes, lengthBytes, fd);
952             } catch (RuntimeException e) {
953                 ExceptionUtils.maybeUnwrapIOException(e);
954                 throw e;
955             } catch (RemoteException e) {
956                 throw e.rethrowFromSystemServer();
957             }
958         }
959 
960         /**
961          * Ensure that any outstanding data for given stream has been committed
962          * to disk. This is only valid for streams returned from
963          * {@link #openWrite(String, long, long)}.
964          */
fsync(@onNull OutputStream out)965         public void fsync(@NonNull OutputStream out) throws IOException {
966             if (ENABLE_REVOCABLE_FD) {
967                 if (out instanceof ParcelFileDescriptor.AutoCloseOutputStream) {
968                     try {
969                         Os.fsync(((ParcelFileDescriptor.AutoCloseOutputStream) out).getFD());
970                     } catch (ErrnoException e) {
971                         throw e.rethrowAsIOException();
972                     }
973                 } else {
974                     throw new IllegalArgumentException("Unrecognized stream");
975                 }
976             } else {
977                 if (out instanceof FileBridge.FileBridgeOutputStream) {
978                     ((FileBridge.FileBridgeOutputStream) out).fsync();
979                 } else {
980                     throw new IllegalArgumentException("Unrecognized stream");
981                 }
982             }
983         }
984 
985         /**
986          * Return all APK names contained in this session.
987          * <p>
988          * This returns all names which have been previously written through
989          * {@link #openWrite(String, long, long)} as part of this session.
990          *
991          * @throws SecurityException if called after the session has been
992          *             committed or abandoned.
993          */
getNames()994         public @NonNull String[] getNames() throws IOException {
995             try {
996                 return mSession.getNames();
997             } catch (RuntimeException e) {
998                 ExceptionUtils.maybeUnwrapIOException(e);
999                 throw e;
1000             } catch (RemoteException e) {
1001                 throw e.rethrowFromSystemServer();
1002             }
1003         }
1004 
1005         /**
1006          * Open a stream to read an APK file from the session.
1007          * <p>
1008          * This is only valid for names which have been previously written
1009          * through {@link #openWrite(String, long, long)} as part of this
1010          * session. For example, this stream may be used to calculate a
1011          * {@link MessageDigest} of a written APK before committing.
1012          *
1013          * @throws SecurityException if called after the session has been
1014          *             committed or abandoned.
1015          */
openRead(@onNull String name)1016         public @NonNull InputStream openRead(@NonNull String name) throws IOException {
1017             try {
1018                 final ParcelFileDescriptor pfd = mSession.openRead(name);
1019                 return new ParcelFileDescriptor.AutoCloseInputStream(pfd);
1020             } catch (RuntimeException e) {
1021                 ExceptionUtils.maybeUnwrapIOException(e);
1022                 throw e;
1023             } catch (RemoteException e) {
1024                 throw e.rethrowFromSystemServer();
1025             }
1026         }
1027 
1028         /**
1029          * Removes a split.
1030          * <p>
1031          * Split removals occur prior to adding new APKs. If upgrading a feature
1032          * split, it is not expected nor desirable to remove the split prior to
1033          * upgrading.
1034          * <p>
1035          * When split removal is bundled with new APKs, the packageName must be
1036          * identical.
1037          */
removeSplit(@onNull String splitName)1038         public void removeSplit(@NonNull String splitName) throws IOException {
1039             try {
1040                 mSession.removeSplit(splitName);
1041             } catch (RuntimeException e) {
1042                 ExceptionUtils.maybeUnwrapIOException(e);
1043                 throw e;
1044             } catch (RemoteException e) {
1045                 throw e.rethrowFromSystemServer();
1046             }
1047         }
1048 
1049         /**
1050          * Attempt to commit everything staged in this session. This may require
1051          * user intervention, and so it may not happen immediately. The final
1052          * result of the commit will be reported through the given callback.
1053          * <p>
1054          * Once this method is called, the session is sealed and no additional
1055          * mutations may be performed on the session. If the device reboots
1056          * before the session has been finalized, you may commit the session again.
1057          * <p>
1058          * If the installer is the device owner or the affiliated profile owner, there will be no
1059          * user intervention.
1060          *
1061          * @param statusReceiver Called when the state of the session changes. Intents
1062          *                       sent to this receiver contain {@link #EXTRA_STATUS}. Refer to the
1063          *                       individual status codes on how to handle them.
1064          *
1065          * @throws SecurityException if streams opened through
1066          *             {@link #openWrite(String, long, long)} are still open.
1067          *
1068          * @see android.app.admin.DevicePolicyManager
1069          */
commit(@onNull IntentSender statusReceiver)1070         public void commit(@NonNull IntentSender statusReceiver) {
1071             try {
1072                 mSession.commit(statusReceiver, false);
1073             } catch (RemoteException e) {
1074                 throw e.rethrowFromSystemServer();
1075             }
1076         }
1077 
1078         /**
1079          * Attempt to commit a session that has been {@link #transfer(String) transferred}.
1080          *
1081          * <p>If the device reboots before the session has been finalized, you may commit the
1082          * session again.
1083          *
1084          * <p>The caller of this method is responsible to ensure the safety of the session. As the
1085          * session was created by another - usually less trusted - app, it is paramount that before
1086          * committing <u>all</u> public and system {@link SessionInfo properties of the session}
1087          * and <u>all</u> {@link #openRead(String) APKs} are verified by the caller. It might happen
1088          * that new properties are added to the session with a new API revision. In this case the
1089          * callers need to be updated.
1090          *
1091          * @param statusReceiver Called when the state of the session changes. Intents
1092          *                       sent to this receiver contain {@link #EXTRA_STATUS}. Refer to the
1093          *                       individual status codes on how to handle them.
1094          *
1095          * @hide
1096          */
1097         @SystemApi
1098         @RequiresPermission(android.Manifest.permission.INSTALL_PACKAGES)
commitTransferred(@onNull IntentSender statusReceiver)1099         public void commitTransferred(@NonNull IntentSender statusReceiver) {
1100             try {
1101                 mSession.commit(statusReceiver, true);
1102             } catch (RemoteException e) {
1103                 throw e.rethrowFromSystemServer();
1104             }
1105         }
1106 
1107         /**
1108          * Transfer the session to a new owner.
1109          * <p>
1110          * Only sessions that update the installing app can be transferred.
1111          * <p>
1112          * After the transfer to a package with a different uid all method calls on the session
1113          * will cause {@link SecurityException}s.
1114          * <p>
1115          * Once this method is called, the session is sealed and no additional mutations beside
1116          * committing it may be performed on the session.
1117          *
1118          * @param packageName The package of the new owner. Needs to hold the INSTALL_PACKAGES
1119          *                    permission.
1120          *
1121          * @throws PackageManager.NameNotFoundException if the new owner could not be found.
1122          * @throws SecurityException if called after the session has been committed or abandoned.
1123          * @throws SecurityException if the session does not update the original installer
1124          * @throws SecurityException if streams opened through
1125          *                           {@link #openWrite(String, long, long) are still open.
1126          */
transfer(@onNull String packageName)1127         public void transfer(@NonNull String packageName)
1128                 throws PackageManager.NameNotFoundException {
1129             Preconditions.checkNotNull(packageName);
1130 
1131             try {
1132                 mSession.transfer(packageName);
1133             } catch (ParcelableException e) {
1134                 e.maybeRethrow(PackageManager.NameNotFoundException.class);
1135                 throw new RuntimeException(e);
1136             } catch (RemoteException e) {
1137                 throw e.rethrowFromSystemServer();
1138             }
1139         }
1140 
1141         /**
1142          * Release this session object. You can open the session again if it
1143          * hasn't been finalized.
1144          */
1145         @Override
close()1146         public void close() {
1147             try {
1148                 mSession.close();
1149             } catch (RemoteException e) {
1150                 throw e.rethrowFromSystemServer();
1151             }
1152         }
1153 
1154         /**
1155          * Completely abandon this session, destroying all staged data and
1156          * rendering it invalid. Abandoned sessions will be reported to
1157          * {@link SessionCallback} listeners as failures. This is equivalent to
1158          * opening the session and calling {@link Session#abandon()}.
1159          */
abandon()1160         public void abandon() {
1161             try {
1162                 mSession.abandon();
1163             } catch (RemoteException e) {
1164                 throw e.rethrowFromSystemServer();
1165             }
1166         }
1167 
1168         /**
1169          * @return {@code true} if this session will commit more than one package when it is
1170          * committed.
1171          */
isMultiPackage()1172         public boolean isMultiPackage() {
1173             try {
1174                 return mSession.isMultiPackage();
1175             } catch (RemoteException e) {
1176                 throw e.rethrowFromSystemServer();
1177             }
1178         }
1179 
1180         /**
1181          * @return {@code true} if this session will be staged and applied at next reboot.
1182          */
isStaged()1183         public boolean isStaged() {
1184             try {
1185                 return mSession.isStaged();
1186             } catch (RemoteException e) {
1187                 throw e.rethrowFromSystemServer();
1188             }
1189         }
1190 
1191         /**
1192          * @return the session ID of the multi-package session that this belongs to or
1193          * {@link SessionInfo#INVALID_ID} if it does not belong to a multi-package session.
1194          */
getParentSessionId()1195         public int getParentSessionId() {
1196             try {
1197                 return mSession.getParentSessionId();
1198             } catch (RemoteException e) {
1199                 throw e.rethrowFromSystemServer();
1200             }
1201         }
1202 
1203         /**
1204          * @return the set of session IDs that will be committed atomically when this session is
1205          * committed if this is a multi-package session or null if none exist.
1206          */
1207         @NonNull
getChildSessionIds()1208         public int[] getChildSessionIds() {
1209             try {
1210                 return mSession.getChildSessionIds();
1211             } catch (RemoteException e) {
1212                 throw e.rethrowFromSystemServer();
1213             }
1214         }
1215 
1216         /**
1217          * Adds a session ID to the set of sessions that will be committed atomically
1218          * when this session is committed.
1219          *
1220          * <p>If the parent is staged or has rollback enabled, all children must have
1221          * the same properties.
1222          *
1223          * @param sessionId the session ID to add to this multi-package session.
1224          */
addChildSessionId(int sessionId)1225         public void addChildSessionId(int sessionId) {
1226             try {
1227                 mSession.addChildSessionId(sessionId);
1228             } catch (RemoteException e) {
1229                 e.rethrowFromSystemServer();
1230             }
1231         }
1232 
1233         /**
1234          * Removes a session ID from the set of sessions that will be committed
1235          * atomically when this session is committed.
1236          *
1237          * @param sessionId the session ID to remove from this multi-package session.
1238          */
removeChildSessionId(int sessionId)1239         public void removeChildSessionId(int sessionId) {
1240             try {
1241                 mSession.removeChildSessionId(sessionId);
1242             } catch (RemoteException e) {
1243                 e.rethrowFromSystemServer();
1244             }
1245         }
1246     }
1247 
1248     /**
1249      * Parameters for creating a new {@link PackageInstaller.Session}.
1250      */
1251     public static class SessionParams implements Parcelable {
1252 
1253         /** {@hide} */
1254         public static final int MODE_INVALID = -1;
1255 
1256         /**
1257          * Mode for an install session whose staged APKs should fully replace any
1258          * existing APKs for the target app.
1259          */
1260         public static final int MODE_FULL_INSTALL = 1;
1261 
1262         /**
1263          * Mode for an install session that should inherit any existing APKs for the
1264          * target app, unless they have been explicitly overridden (based on split
1265          * name) by the session. For example, this can be used to add one or more
1266          * split APKs to an existing installation.
1267          * <p>
1268          * If there are no existing APKs for the target app, this behaves like
1269          * {@link #MODE_FULL_INSTALL}.
1270          */
1271         public static final int MODE_INHERIT_EXISTING = 2;
1272 
1273         /**
1274          * Special constant to refer to all restricted permissions.
1275          */
1276         public static final @NonNull Set<String> RESTRICTED_PERMISSIONS_ALL = new ArraySet<>();
1277 
1278         /** {@hide} */
1279         public static final int UID_UNKNOWN = -1;
1280 
1281         /** {@hide} */
1282         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1283         public int mode = MODE_INVALID;
1284         /** {@hide} */
1285         @UnsupportedAppUsage
1286         public int installFlags = PackageManager.INSTALL_ALL_WHITELIST_RESTRICTED_PERMISSIONS;
1287         /** {@hide} */
1288         public int installLocation = PackageInfo.INSTALL_LOCATION_INTERNAL_ONLY;
1289         /** {@hide} */
1290         public @InstallReason int installReason = PackageManager.INSTALL_REASON_UNKNOWN;
1291         /** {@hide} */
1292         @UnsupportedAppUsage
1293         public long sizeBytes = -1;
1294         /** {@hide} */
1295         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1296         public String appPackageName;
1297         /** {@hide} */
1298         @UnsupportedAppUsage
1299         public Bitmap appIcon;
1300         /** {@hide} */
1301         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1302         public String appLabel;
1303         /** {@hide} */
1304         public long appIconLastModified = -1;
1305         /** {@hide} */
1306         public Uri originatingUri;
1307         /** {@hide} */
1308         @UnsupportedAppUsage
1309         public int originatingUid = UID_UNKNOWN;
1310         /** {@hide} */
1311         public Uri referrerUri;
1312         /** {@hide} */
1313         public String abiOverride;
1314         /** {@hide} */
1315         public String volumeUuid;
1316         /** {@hide} */
1317         public String[] grantedRuntimePermissions;
1318         /** {@hide} */
1319         public List<String> whitelistedRestrictedPermissions;
1320         /** {@hide} */
1321         public String installerPackageName;
1322         /** {@hide} */
1323         public boolean isMultiPackage;
1324         /** {@hide} */
1325         public boolean isStaged;
1326         /** {@hide} */
1327         public long requiredInstalledVersionCode = PackageManager.VERSION_CODE_HIGHEST;
1328 
1329         /**
1330          * Construct parameters for a new package install session.
1331          *
1332          * @param mode one of {@link #MODE_FULL_INSTALL} or
1333          *            {@link #MODE_INHERIT_EXISTING} describing how the session
1334          *            should interact with an existing app.
1335          */
SessionParams(int mode)1336         public SessionParams(int mode) {
1337             this.mode = mode;
1338         }
1339 
1340         /** {@hide} */
SessionParams(Parcel source)1341         public SessionParams(Parcel source) {
1342             mode = source.readInt();
1343             installFlags = source.readInt();
1344             installLocation = source.readInt();
1345             installReason = source.readInt();
1346             sizeBytes = source.readLong();
1347             appPackageName = source.readString();
1348             appIcon = source.readParcelable(null);
1349             appLabel = source.readString();
1350             originatingUri = source.readParcelable(null);
1351             originatingUid = source.readInt();
1352             referrerUri = source.readParcelable(null);
1353             abiOverride = source.readString();
1354             volumeUuid = source.readString();
1355             grantedRuntimePermissions = source.readStringArray();
1356             whitelistedRestrictedPermissions = source.createStringArrayList();
1357             installerPackageName = source.readString();
1358             isMultiPackage = source.readBoolean();
1359             isStaged = source.readBoolean();
1360             requiredInstalledVersionCode = source.readLong();
1361         }
1362 
1363         /** {@hide} */
copy()1364         public SessionParams copy() {
1365             SessionParams ret = new SessionParams(mode);
1366             ret.installFlags = installFlags;
1367             ret.installLocation = installLocation;
1368             ret.installReason = installReason;
1369             ret.sizeBytes = sizeBytes;
1370             ret.appPackageName = appPackageName;
1371             ret.appIcon = appIcon;  // not a copy.
1372             ret.appLabel = appLabel;
1373             ret.originatingUri = originatingUri;  // not a copy, but immutable.
1374             ret.originatingUid = originatingUid;
1375             ret.referrerUri = referrerUri;  // not a copy, but immutable.
1376             ret.abiOverride = abiOverride;
1377             ret.volumeUuid = volumeUuid;
1378             ret.grantedRuntimePermissions = grantedRuntimePermissions;
1379             ret.whitelistedRestrictedPermissions = whitelistedRestrictedPermissions;
1380             ret.installerPackageName = installerPackageName;
1381             ret.isMultiPackage = isMultiPackage;
1382             ret.isStaged = isStaged;
1383             ret.requiredInstalledVersionCode = requiredInstalledVersionCode;
1384             return ret;
1385         }
1386 
1387         /**
1388          * Check if there are hidden options set.
1389          *
1390          * <p>Hidden options are those options that cannot be verified via public or system-api
1391          * methods on {@link SessionInfo}.
1392          *
1393          * @return {@code true} if any hidden option is set.
1394          *
1395          * @hide
1396          */
areHiddenOptionsSet()1397         public boolean areHiddenOptionsSet() {
1398             return (installFlags & (PackageManager.INSTALL_REQUEST_DOWNGRADE
1399                     | PackageManager.INSTALL_ALLOW_DOWNGRADE
1400                     | PackageManager.INSTALL_DONT_KILL_APP
1401                     | PackageManager.INSTALL_INSTANT_APP
1402                     | PackageManager.INSTALL_FULL_APP
1403                     | PackageManager.INSTALL_VIRTUAL_PRELOAD
1404                     | PackageManager.INSTALL_ALLOCATE_AGGRESSIVE)) != installFlags
1405                     || abiOverride != null || volumeUuid != null;
1406         }
1407 
1408         /**
1409          * Provide value of {@link PackageInfo#installLocation}, which may be used
1410          * to determine where the app will be staged. Defaults to
1411          * {@link PackageInfo#INSTALL_LOCATION_INTERNAL_ONLY}.
1412          */
setInstallLocation(int installLocation)1413         public void setInstallLocation(int installLocation) {
1414             this.installLocation = installLocation;
1415         }
1416 
1417         /**
1418          * Optionally indicate the total size (in bytes) of all APKs that will be
1419          * delivered in this session. The system may use this to ensure enough disk
1420          * space exists before proceeding, or to estimate container size for
1421          * installations living on external storage.
1422          *
1423          * @see PackageInfo#INSTALL_LOCATION_AUTO
1424          * @see PackageInfo#INSTALL_LOCATION_PREFER_EXTERNAL
1425          */
setSize(long sizeBytes)1426         public void setSize(long sizeBytes) {
1427             this.sizeBytes = sizeBytes;
1428         }
1429 
1430         /**
1431          * Optionally set the package name of the app being installed. It's strongly
1432          * recommended that you provide this value when known, so that observers can
1433          * communicate installing apps to users.
1434          * <p>
1435          * If the APKs staged in the session aren't consistent with this package
1436          * name, the install will fail. Regardless of this value, all APKs in the
1437          * app must have the same package name.
1438          */
setAppPackageName(@ullable String appPackageName)1439         public void setAppPackageName(@Nullable String appPackageName) {
1440             this.appPackageName = appPackageName;
1441         }
1442 
1443         /**
1444          * Optionally set an icon representing the app being installed. This should
1445          * be roughly {@link ActivityManager#getLauncherLargeIconSize()} in both
1446          * dimensions.
1447          */
setAppIcon(@ullable Bitmap appIcon)1448         public void setAppIcon(@Nullable Bitmap appIcon) {
1449             this.appIcon = appIcon;
1450         }
1451 
1452         /**
1453          * Optionally set a label representing the app being installed.
1454          */
setAppLabel(@ullable CharSequence appLabel)1455         public void setAppLabel(@Nullable CharSequence appLabel) {
1456             this.appLabel = (appLabel != null) ? appLabel.toString() : null;
1457         }
1458 
1459         /**
1460          * Optionally set the URI where this package was downloaded from. This is
1461          * informational and may be used as a signal for anti-malware purposes.
1462          *
1463          * @see Intent#EXTRA_ORIGINATING_URI
1464          */
setOriginatingUri(@ullable Uri originatingUri)1465         public void setOriginatingUri(@Nullable Uri originatingUri) {
1466             this.originatingUri = originatingUri;
1467         }
1468 
1469         /**
1470          * Sets the UID that initiated the package installation. This is informational
1471          * and may be used as a signal for anti-malware purposes.
1472          */
setOriginatingUid(int originatingUid)1473         public void setOriginatingUid(int originatingUid) {
1474             this.originatingUid = originatingUid;
1475         }
1476 
1477         /**
1478          * Optionally set the URI that referred you to install this package. This is
1479          * informational and may be used as a signal for anti-malware purposes.
1480          *
1481          * @see Intent#EXTRA_REFERRER
1482          */
setReferrerUri(@ullable Uri referrerUri)1483         public void setReferrerUri(@Nullable Uri referrerUri) {
1484             this.referrerUri = referrerUri;
1485         }
1486 
1487         /**
1488          * Sets which runtime permissions to be granted to the package at installation.
1489          *
1490          * @param permissions The permissions to grant or null to grant all runtime
1491          *     permissions.
1492          *
1493          * @hide
1494          */
1495         @TestApi
1496         @SystemApi
1497         @RequiresPermission(android.Manifest.permission.INSTALL_GRANT_RUNTIME_PERMISSIONS)
setGrantedRuntimePermissions(String[] permissions)1498         public void setGrantedRuntimePermissions(String[] permissions) {
1499             installFlags |= PackageManager.INSTALL_GRANT_RUNTIME_PERMISSIONS;
1500             this.grantedRuntimePermissions = permissions;
1501         }
1502 
1503         /**
1504          * Sets which restricted permissions to be whitelisted for the app. Whitelisting
1505          * is not granting the permissions, rather it allows the app to hold permissions
1506          * which are otherwise restricted. Whitelisting a non restricted permission has
1507          * no effect.
1508          *
1509          * <p> Permissions can be hard restricted which means that the app cannot hold
1510          * them or soft restricted where the app can hold the permission but in a weaker
1511          * form. Whether a permission is {@link PermissionInfo#FLAG_HARD_RESTRICTED hard
1512          * restricted} or {@link PermissionInfo#FLAG_SOFT_RESTRICTED soft restricted}
1513          * depends on the permission declaration. Whitelisting a hard restricted permission
1514          * allows the app to hold that permission and whitelisting a soft restricted
1515          * permission allows the app to hold the permission in its full, unrestricted form.
1516          *
1517          * <p> Permissions can also be immutably restricted which means that the whitelist
1518          * state of the permission can be determined only at install time and cannot be
1519          * changed on updated or at a later point via the package manager APIs.
1520          *
1521          * <p>Initially, all restricted permissions are whitelisted but you can change
1522          * which ones are whitelisted by calling this method or the corresponding ones
1523          * on the {@link PackageManager}.
1524          *
1525          * @see PackageManager#addWhitelistedRestrictedPermission(String, String, int)
1526          * @see PackageManager#removeWhitelistedRestrictedPermission(String, String, int)
1527          */
setWhitelistedRestrictedPermissions(@ullable Set<String> permissions)1528         public void setWhitelistedRestrictedPermissions(@Nullable Set<String> permissions) {
1529             if (permissions == RESTRICTED_PERMISSIONS_ALL) {
1530                 installFlags |= PackageManager.INSTALL_ALL_WHITELIST_RESTRICTED_PERMISSIONS;
1531                 whitelistedRestrictedPermissions = null;
1532             } else {
1533                 installFlags &= ~PackageManager.INSTALL_ALL_WHITELIST_RESTRICTED_PERMISSIONS;
1534                 whitelistedRestrictedPermissions = (permissions != null)
1535                         ? new ArrayList<>(permissions) : null;
1536             }
1537         }
1538 
1539         /**
1540          * Request that rollbacks be enabled or disabled for the given upgrade.
1541          *
1542          * <p>If the parent session is staged or has rollback enabled, all children sessions
1543          * must have the same properties.
1544          *
1545          * @param enable set to {@code true} to enable, {@code false} to disable
1546          * @hide
1547          */
1548         @SystemApi @TestApi
setEnableRollback(boolean enable)1549         public void setEnableRollback(boolean enable) {
1550             if (enable) {
1551                 installFlags |= PackageManager.INSTALL_ENABLE_ROLLBACK;
1552             } else {
1553                 installFlags &= ~PackageManager.INSTALL_ENABLE_ROLLBACK;
1554             }
1555         }
1556 
1557         /**
1558          * @deprecated use {@link #setRequestDowngrade(boolean)}.
1559          * {@hide}
1560          */
1561         @SystemApi
1562         @Deprecated
setAllowDowngrade(boolean allowDowngrade)1563         public void setAllowDowngrade(boolean allowDowngrade) {
1564             setRequestDowngrade(allowDowngrade);
1565         }
1566 
1567         /** {@hide} */
1568         @SystemApi
setRequestDowngrade(boolean requestDowngrade)1569         public void setRequestDowngrade(boolean requestDowngrade) {
1570             if (requestDowngrade) {
1571                 installFlags |= PackageManager.INSTALL_REQUEST_DOWNGRADE;
1572             } else {
1573                 installFlags &= ~PackageManager.INSTALL_REQUEST_DOWNGRADE;
1574             }
1575         }
1576 
1577         /**
1578          * Require the given version of the package be installed.
1579          * The install will only be allowed if the existing version code of
1580          * the package installed on the device matches the given version code.
1581          * Use {@link * PackageManager#VERSION_CODE_HIGHEST} to allow
1582          * installation regardless of the currently installed package version.
1583          *
1584          * @hide
1585          */
setRequiredInstalledVersionCode(long versionCode)1586         public void setRequiredInstalledVersionCode(long versionCode) {
1587             requiredInstalledVersionCode = versionCode;
1588         }
1589 
1590         /** {@hide} */
setInstallFlagsForcePermissionPrompt()1591         public void setInstallFlagsForcePermissionPrompt() {
1592             installFlags |= PackageManager.INSTALL_FORCE_PERMISSION_PROMPT;
1593         }
1594 
1595         /** {@hide} */
1596         @SystemApi
setDontKillApp(boolean dontKillApp)1597         public void setDontKillApp(boolean dontKillApp) {
1598             if (dontKillApp) {
1599                 installFlags |= PackageManager.INSTALL_DONT_KILL_APP;
1600             } else {
1601                 installFlags &= ~PackageManager.INSTALL_DONT_KILL_APP;
1602             }
1603         }
1604 
1605         /** {@hide} */
1606         @SystemApi
setInstallAsInstantApp(boolean isInstantApp)1607         public void setInstallAsInstantApp(boolean isInstantApp) {
1608             if (isInstantApp) {
1609                 installFlags |= PackageManager.INSTALL_INSTANT_APP;
1610                 installFlags &= ~PackageManager.INSTALL_FULL_APP;
1611             } else {
1612                 installFlags &= ~PackageManager.INSTALL_INSTANT_APP;
1613                 installFlags |= PackageManager.INSTALL_FULL_APP;
1614             }
1615         }
1616 
1617         /**
1618          * Sets the install as a virtual preload. Will only have effect when called
1619          * by the verifier.
1620          * {@hide}
1621          */
1622         @SystemApi
setInstallAsVirtualPreload()1623         public void setInstallAsVirtualPreload() {
1624             installFlags |= PackageManager.INSTALL_VIRTUAL_PRELOAD;
1625         }
1626 
1627         /**
1628          * Set the reason for installing this package.
1629          * <p>
1630          * The install reason should be a pre-defined integer. The behavior is
1631          * undefined if other values are used.
1632          *
1633          * @see PackageManager#INSTALL_REASON_UNKNOWN
1634          * @see PackageManager#INSTALL_REASON_POLICY
1635          * @see PackageManager#INSTALL_REASON_DEVICE_RESTORE
1636          * @see PackageManager#INSTALL_REASON_DEVICE_SETUP
1637          * @see PackageManager#INSTALL_REASON_USER
1638          */
setInstallReason(@nstallReason int installReason)1639         public void setInstallReason(@InstallReason int installReason) {
1640             this.installReason = installReason;
1641         }
1642 
1643         /** {@hide} */
1644         @SystemApi
1645         @RequiresPermission(android.Manifest.permission.ALLOCATE_AGGRESSIVE)
setAllocateAggressive(boolean allocateAggressive)1646         public void setAllocateAggressive(boolean allocateAggressive) {
1647             if (allocateAggressive) {
1648                 installFlags |= PackageManager.INSTALL_ALLOCATE_AGGRESSIVE;
1649             } else {
1650                 installFlags &= ~PackageManager.INSTALL_ALLOCATE_AGGRESSIVE;
1651             }
1652         }
1653 
1654         /**
1655          * Set the installer package for the app.
1656          *
1657          * By default this is the app that created the {@link PackageInstaller} object.
1658          *
1659          * @param installerPackageName name of the installer package
1660          * {@hide}
1661          */
setInstallerPackageName(String installerPackageName)1662         public void setInstallerPackageName(String installerPackageName) {
1663             this.installerPackageName = installerPackageName;
1664         }
1665 
1666         /**
1667          * Set this session to be the parent of a multi-package install.
1668          *
1669          * A multi-package install session contains no APKs and only references other install
1670          * sessions via ID. When a multi-package session is committed, all of its children
1671          * are committed to the system in an atomic manner. If any children fail to install,
1672          * all of them do, including the multi-package session.
1673          */
setMultiPackage()1674         public void setMultiPackage() {
1675             this.isMultiPackage = true;
1676         }
1677 
1678         /**
1679          * Set this session to be staged to be installed at reboot.
1680          *
1681          * Staged sessions are scheduled to be installed at next reboot. Staged sessions can also be
1682          * multi-package. In that case, if any of the children sessions fail to install at reboot,
1683          * all the other children sessions are aborted as well.
1684          *
1685          * <p>If the parent session is staged or has rollback enabled, all children sessions
1686          * must have the same properties.
1687          *
1688          * {@hide}
1689          */
1690         @SystemApi @TestApi
1691         @RequiresPermission(Manifest.permission.INSTALL_PACKAGES)
setStaged()1692         public void setStaged() {
1693             this.isStaged = true;
1694         }
1695 
1696         /**
1697          * Set this session to be installing an APEX package.
1698          *
1699          * {@hide}
1700          */
1701         @SystemApi @TestApi
1702         @RequiresPermission(Manifest.permission.INSTALL_PACKAGES)
setInstallAsApex()1703         public void setInstallAsApex() {
1704             installFlags |= PackageManager.INSTALL_APEX;
1705         }
1706 
1707         /** @hide */
getEnableRollback()1708         public boolean getEnableRollback() {
1709             return (installFlags & PackageManager.INSTALL_ENABLE_ROLLBACK) != 0;
1710         }
1711 
1712         /** {@hide} */
dump(IndentingPrintWriter pw)1713         public void dump(IndentingPrintWriter pw) {
1714             pw.printPair("mode", mode);
1715             pw.printHexPair("installFlags", installFlags);
1716             pw.printPair("installLocation", installLocation);
1717             pw.printPair("sizeBytes", sizeBytes);
1718             pw.printPair("appPackageName", appPackageName);
1719             pw.printPair("appIcon", (appIcon != null));
1720             pw.printPair("appLabel", appLabel);
1721             pw.printPair("originatingUri", originatingUri);
1722             pw.printPair("originatingUid", originatingUid);
1723             pw.printPair("referrerUri", referrerUri);
1724             pw.printPair("abiOverride", abiOverride);
1725             pw.printPair("volumeUuid", volumeUuid);
1726             pw.printPair("grantedRuntimePermissions", grantedRuntimePermissions);
1727             pw.printPair("whitelistedRestrictedPermissions", whitelistedRestrictedPermissions);
1728             pw.printPair("installerPackageName", installerPackageName);
1729             pw.printPair("isMultiPackage", isMultiPackage);
1730             pw.printPair("isStaged", isStaged);
1731             pw.printPair("requiredInstalledVersionCode", requiredInstalledVersionCode);
1732             pw.println();
1733         }
1734 
1735         @Override
describeContents()1736         public int describeContents() {
1737             return 0;
1738         }
1739 
1740         @Override
writeToParcel(Parcel dest, int flags)1741         public void writeToParcel(Parcel dest, int flags) {
1742             dest.writeInt(mode);
1743             dest.writeInt(installFlags);
1744             dest.writeInt(installLocation);
1745             dest.writeInt(installReason);
1746             dest.writeLong(sizeBytes);
1747             dest.writeString(appPackageName);
1748             dest.writeParcelable(appIcon, flags);
1749             dest.writeString(appLabel);
1750             dest.writeParcelable(originatingUri, flags);
1751             dest.writeInt(originatingUid);
1752             dest.writeParcelable(referrerUri, flags);
1753             dest.writeString(abiOverride);
1754             dest.writeString(volumeUuid);
1755             dest.writeStringArray(grantedRuntimePermissions);
1756             dest.writeStringList(whitelistedRestrictedPermissions);
1757             dest.writeString(installerPackageName);
1758             dest.writeBoolean(isMultiPackage);
1759             dest.writeBoolean(isStaged);
1760             dest.writeLong(requiredInstalledVersionCode);
1761         }
1762 
1763         public static final Parcelable.Creator<SessionParams>
1764                 CREATOR = new Parcelable.Creator<SessionParams>() {
1765                     @Override
1766                     public SessionParams createFromParcel(Parcel p) {
1767                         return new SessionParams(p);
1768                     }
1769 
1770                     @Override
1771                     public SessionParams[] newArray(int size) {
1772                         return new SessionParams[size];
1773                     }
1774                 };
1775     }
1776 
1777     /**
1778      * Details for an active install session.
1779      */
1780     public static class SessionInfo implements Parcelable {
1781 
1782         /**
1783          * A session ID that does not exist or is invalid.
1784          */
1785         public static final int INVALID_ID = -1;
1786         /** {@hide} */
1787         private static final int[] NO_SESSIONS = {};
1788 
1789         /** @hide */
1790         @IntDef(prefix = { "STAGED_SESSION_" }, value = {
1791                 STAGED_SESSION_NO_ERROR,
1792                 STAGED_SESSION_VERIFICATION_FAILED,
1793                 STAGED_SESSION_ACTIVATION_FAILED,
1794                 STAGED_SESSION_UNKNOWN})
1795         @Retention(RetentionPolicy.SOURCE)
1796         public @interface StagedSessionErrorCode{}
1797         /**
1798          * Constant indicating that no error occurred during the preparation or the activation of
1799          * this staged session.
1800          */
1801         public static final int STAGED_SESSION_NO_ERROR = 0;
1802 
1803         /**
1804          * Constant indicating that an error occurred during the verification phase (pre-reboot) of
1805          * this staged session.
1806          */
1807         public static final int STAGED_SESSION_VERIFICATION_FAILED = 1;
1808 
1809         /**
1810          * Constant indicating that an error occurred during the activation phase (post-reboot) of
1811          * this staged session.
1812          */
1813         public static final int STAGED_SESSION_ACTIVATION_FAILED = 2;
1814 
1815         /**
1816          * Constant indicating that an unknown error occurred while processing this staged session.
1817          */
1818         public static final int STAGED_SESSION_UNKNOWN = 3;
1819 
1820         /** {@hide} */
1821         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1822         public int sessionId;
1823         /** {@hide} */
1824         public int userId;
1825         /** {@hide} */
1826         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1827         public String installerPackageName;
1828         /** {@hide} */
1829         @UnsupportedAppUsage
1830         public String resolvedBaseCodePath;
1831         /** {@hide} */
1832         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1833         public float progress;
1834         /** {@hide} */
1835         @UnsupportedAppUsage
1836         public boolean sealed;
1837         /** {@hide} */
1838         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1839         public boolean active;
1840 
1841         /** {@hide} */
1842         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1843         public int mode;
1844         /** {@hide} */
1845         public @InstallReason int installReason;
1846         /** {@hide} */
1847         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1848         public long sizeBytes;
1849         /** {@hide} */
1850         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1851         public String appPackageName;
1852         /** {@hide} */
1853         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1854         public Bitmap appIcon;
1855         /** {@hide} */
1856         @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1857         public CharSequence appLabel;
1858 
1859         /** {@hide} */
1860         public int installLocation;
1861         /** {@hide} */
1862         public Uri originatingUri;
1863         /** {@hide} */
1864         public int originatingUid;
1865         /** {@hide} */
1866         public Uri referrerUri;
1867         /** {@hide} */
1868         public String[] grantedRuntimePermissions;
1869         /** {@hide}*/
1870         public List<String> whitelistedRestrictedPermissions;
1871         /** {@hide} */
1872         public int installFlags;
1873         /** {@hide} */
1874         public boolean isMultiPackage;
1875         /** {@hide} */
1876         public boolean isStaged;
1877         /** {@hide} */
1878         public int parentSessionId = INVALID_ID;
1879         /** {@hide} */
1880         public int[] childSessionIds = NO_SESSIONS;
1881 
1882         /** {@hide} */
1883         public boolean isStagedSessionApplied;
1884         /** {@hide} */
1885         public boolean isStagedSessionReady;
1886         /** {@hide} */
1887         public boolean isStagedSessionFailed;
1888         private int mStagedSessionErrorCode;
1889         private String mStagedSessionErrorMessage;
1890 
1891         /** {@hide} */
1892         public boolean isCommitted;
1893 
1894         /** {@hide} */
1895         public long updatedMillis;
1896 
1897         /** {@hide} */
1898         @UnsupportedAppUsage
SessionInfo()1899         public SessionInfo() {
1900         }
1901 
1902         /** {@hide} */
SessionInfo(Parcel source)1903         public SessionInfo(Parcel source) {
1904             sessionId = source.readInt();
1905             userId = source.readInt();
1906             installerPackageName = source.readString();
1907             resolvedBaseCodePath = source.readString();
1908             progress = source.readFloat();
1909             sealed = source.readInt() != 0;
1910             active = source.readInt() != 0;
1911 
1912             mode = source.readInt();
1913             installReason = source.readInt();
1914             sizeBytes = source.readLong();
1915             appPackageName = source.readString();
1916             appIcon = source.readParcelable(null);
1917             appLabel = source.readString();
1918 
1919             installLocation = source.readInt();
1920             originatingUri = source.readParcelable(null);
1921             originatingUid = source.readInt();
1922             referrerUri = source.readParcelable(null);
1923             grantedRuntimePermissions = source.readStringArray();
1924             whitelistedRestrictedPermissions = source.createStringArrayList();
1925 
1926             installFlags = source.readInt();
1927             isMultiPackage = source.readBoolean();
1928             isStaged = source.readBoolean();
1929             parentSessionId = source.readInt();
1930             childSessionIds = source.createIntArray();
1931             if (childSessionIds == null) {
1932                 childSessionIds = NO_SESSIONS;
1933             }
1934             isStagedSessionApplied = source.readBoolean();
1935             isStagedSessionReady = source.readBoolean();
1936             isStagedSessionFailed = source.readBoolean();
1937             mStagedSessionErrorCode = source.readInt();
1938             mStagedSessionErrorMessage = source.readString();
1939             isCommitted = source.readBoolean();
1940         }
1941 
1942         /**
1943          * Return the ID for this session.
1944          */
getSessionId()1945         public int getSessionId() {
1946             return sessionId;
1947         }
1948 
1949         /**
1950          * Return the user associated with this session.
1951          */
getUser()1952         public @NonNull UserHandle getUser() {
1953             return new UserHandle(userId);
1954         }
1955 
1956         /**
1957          * Return the package name of the app that owns this session.
1958          */
getInstallerPackageName()1959         public @Nullable String getInstallerPackageName() {
1960             return installerPackageName;
1961         }
1962 
1963         /**
1964          * Return current overall progress of this session, between 0 and 1.
1965          * <p>
1966          * Note that this progress may not directly correspond to the value
1967          * reported by
1968          * {@link PackageInstaller.Session#setStagingProgress(float)}, as the
1969          * system may carve out a portion of the overall progress to represent
1970          * its own internal installation work.
1971          */
getProgress()1972         public float getProgress() {
1973             return progress;
1974         }
1975 
1976         /**
1977          * Return if this session is currently active.
1978          * <p>
1979          * A session is considered active whenever there is ongoing forward
1980          * progress being made, such as the installer holding an open
1981          * {@link Session} instance while streaming data into place, or the
1982          * system optimizing code as the result of
1983          * {@link Session#commit(IntentSender)}.
1984          * <p>
1985          * If the installer closes the {@link Session} without committing, the
1986          * session is considered inactive until the installer opens the session
1987          * again.
1988          */
isActive()1989         public boolean isActive() {
1990             return active;
1991         }
1992 
1993         /**
1994          * Return if this session is sealed.
1995          * <p>
1996          * Once sealed, no further changes may be made to the session. A session
1997          * is sealed the moment {@link Session#commit(IntentSender)} is called.
1998          */
isSealed()1999         public boolean isSealed() {
2000             return sealed;
2001         }
2002 
2003         /**
2004          * Return the reason for installing this package.
2005          *
2006          * @return The install reason.
2007          */
getInstallReason()2008         public @InstallReason int getInstallReason() {
2009             return installReason;
2010         }
2011 
2012         /** {@hide} */
2013         @Deprecated
isOpen()2014         public boolean isOpen() {
2015             return isActive();
2016         }
2017 
2018         /**
2019          * Return the package name this session is working with. May be {@code null}
2020          * if unknown.
2021          */
getAppPackageName()2022         public @Nullable String getAppPackageName() {
2023             return appPackageName;
2024         }
2025 
2026         /**
2027          * Return an icon representing the app being installed. May be {@code null}
2028          * if unavailable.
2029          */
getAppIcon()2030         public @Nullable Bitmap getAppIcon() {
2031             if (appIcon == null) {
2032                 // Icon may have been omitted for calls that return bulk session
2033                 // lists, so try fetching the specific icon.
2034                 try {
2035                     final SessionInfo info = AppGlobals.getPackageManager().getPackageInstaller()
2036                             .getSessionInfo(sessionId);
2037                     appIcon = (info != null) ? info.appIcon : null;
2038                 } catch (RemoteException e) {
2039                     throw e.rethrowFromSystemServer();
2040                 }
2041             }
2042             return appIcon;
2043         }
2044 
2045         /**
2046          * Return a label representing the app being installed. May be {@code null}
2047          * if unavailable.
2048          */
getAppLabel()2049         public @Nullable CharSequence getAppLabel() {
2050             return appLabel;
2051         }
2052 
2053         /**
2054          * Return an Intent that can be started to view details about this install
2055          * session. This may surface actions such as pause, resume, or cancel.
2056          * <p>
2057          * In some cases, a matching Activity may not exist, so ensure you safeguard
2058          * against this.
2059          *
2060          * @see PackageInstaller#ACTION_SESSION_DETAILS
2061          */
createDetailsIntent()2062         public @Nullable Intent createDetailsIntent() {
2063             final Intent intent = new Intent(PackageInstaller.ACTION_SESSION_DETAILS);
2064             intent.putExtra(PackageInstaller.EXTRA_SESSION_ID, sessionId);
2065             intent.setPackage(installerPackageName);
2066             intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
2067             return intent;
2068         }
2069 
2070         /**
2071          * Get the mode of the session as set in the constructor of the {@link SessionParams}.
2072          *
2073          * @return One of {@link SessionParams#MODE_FULL_INSTALL}
2074          *         or {@link SessionParams#MODE_INHERIT_EXISTING}
2075          */
getMode()2076         public int getMode() {
2077             return mode;
2078         }
2079 
2080         /**
2081          * Get the value set in {@link SessionParams#setInstallLocation(int)}.
2082          */
getInstallLocation()2083         public int getInstallLocation() {
2084             return installLocation;
2085         }
2086 
2087         /**
2088          * Get the value as set in {@link SessionParams#setSize(long)}.
2089          *
2090          * <p>The value is a hint and does not have to match the actual size.
2091          */
getSize()2092         public long getSize() {
2093             return sizeBytes;
2094         }
2095 
2096         /**
2097          * Get the value set in {@link SessionParams#setOriginatingUri(Uri)}.
2098          */
getOriginatingUri()2099         public @Nullable Uri getOriginatingUri() {
2100             return originatingUri;
2101         }
2102 
2103         /**
2104          * Get the value set in {@link SessionParams#setOriginatingUid(int)}.
2105          */
getOriginatingUid()2106         public int getOriginatingUid() {
2107             return originatingUid;
2108         }
2109 
2110         /**
2111          * Get the value set in {@link SessionParams#setReferrerUri(Uri)}
2112          */
getReferrerUri()2113         public @Nullable Uri getReferrerUri() {
2114             return referrerUri;
2115         }
2116 
2117         /**
2118          * Get the value set in {@link SessionParams#setGrantedRuntimePermissions(String[])}.
2119          *
2120          * @hide
2121          */
2122         @SystemApi
getGrantedRuntimePermissions()2123         public @Nullable String[] getGrantedRuntimePermissions() {
2124             return grantedRuntimePermissions;
2125         }
2126 
2127         /**
2128          * Get the value set in {@link SessionParams#setWhitelistedRestrictedPermissions(Set)}.
2129          * Note that if all permissions are whitelisted this method returns {@link
2130          * SessionParams#RESTRICTED_PERMISSIONS_ALL}.
2131          *
2132          * @hide
2133          */
2134         @TestApi
2135         @SystemApi
getWhitelistedRestrictedPermissions()2136         public @NonNull Set<String> getWhitelistedRestrictedPermissions() {
2137             if ((installFlags & PackageManager.INSTALL_ALL_WHITELIST_RESTRICTED_PERMISSIONS) != 0) {
2138                 return SessionParams.RESTRICTED_PERMISSIONS_ALL;
2139             }
2140             if (whitelistedRestrictedPermissions != null) {
2141                 return new ArraySet<>(whitelistedRestrictedPermissions);
2142             }
2143             return Collections.emptySet();
2144         }
2145 
2146         /**
2147          * Get the value set in {@link SessionParams#setAllowDowngrade(boolean)}.
2148          *
2149          * @deprecated use {@link #getRequestDowngrade()}.
2150          * @hide
2151          */
2152         @SystemApi
2153         @Deprecated
getAllowDowngrade()2154         public boolean getAllowDowngrade() {
2155             return getRequestDowngrade();
2156         }
2157 
2158         /**
2159          * Get the value set in {@link SessionParams#setRequestDowngrade(boolean)}.
2160          *
2161          * @hide
2162          */
2163         @SystemApi
getRequestDowngrade()2164         public boolean getRequestDowngrade() {
2165             return (installFlags & PackageManager.INSTALL_REQUEST_DOWNGRADE) != 0;
2166         }
2167 
2168         /**
2169          * Get the value set in {@link SessionParams#setDontKillApp(boolean)}.
2170          *
2171          * @hide
2172          */
2173         @SystemApi
getDontKillApp()2174         public boolean getDontKillApp() {
2175             return (installFlags & PackageManager.INSTALL_DONT_KILL_APP) != 0;
2176         }
2177 
2178         /**
2179          * If {@link SessionParams#setInstallAsInstantApp(boolean)} was called with {@code true},
2180          * return true. If it was called with {@code false} or if it was not called return false.
2181          *
2182          * @hide
2183          *
2184          * @see #getInstallAsFullApp
2185          */
2186         @SystemApi
getInstallAsInstantApp(boolean isInstantApp)2187         public boolean getInstallAsInstantApp(boolean isInstantApp) {
2188             return (installFlags & PackageManager.INSTALL_INSTANT_APP) != 0;
2189         }
2190 
2191         /**
2192          * If {@link SessionParams#setInstallAsInstantApp(boolean)} was called with {@code false},
2193          * return true. If it was called with {@code true} or if it was not called return false.
2194          *
2195          * @hide
2196          *
2197          * @see #getInstallAsInstantApp
2198          */
2199         @SystemApi
getInstallAsFullApp(boolean isInstantApp)2200         public boolean getInstallAsFullApp(boolean isInstantApp) {
2201             return (installFlags & PackageManager.INSTALL_FULL_APP) != 0;
2202         }
2203 
2204         /**
2205          * Get if {@link SessionParams#setInstallAsVirtualPreload()} was called.
2206          *
2207          * @hide
2208          */
2209         @SystemApi
getInstallAsVirtualPreload()2210         public boolean getInstallAsVirtualPreload() {
2211             return (installFlags & PackageManager.INSTALL_VIRTUAL_PRELOAD) != 0;
2212         }
2213 
2214         /**
2215          * Return whether rollback is enabled or disabled for the given upgrade.
2216          *
2217          * @hide
2218          */
2219         @SystemApi
getEnableRollback()2220         public boolean getEnableRollback() {
2221             return (installFlags & PackageManager.INSTALL_ENABLE_ROLLBACK) != 0;
2222         }
2223 
2224         /**
2225          * Get the value set in {@link SessionParams#setAllocateAggressive(boolean)}.
2226          *
2227          * @hide
2228          */
2229         @SystemApi
getAllocateAggressive()2230         public boolean getAllocateAggressive() {
2231             return (installFlags & PackageManager.INSTALL_ALLOCATE_AGGRESSIVE) != 0;
2232         }
2233 
2234 
2235         /** {@hide} */
2236         @Deprecated
getDetailsIntent()2237         public @Nullable Intent getDetailsIntent() {
2238             return createDetailsIntent();
2239         }
2240 
2241         /**
2242          * Returns true if this session is a multi-package session containing references to other
2243          * sessions.
2244          */
isMultiPackage()2245         public boolean isMultiPackage() {
2246             return isMultiPackage;
2247         }
2248 
2249         /**
2250          * Returns true if this session is a staged session which will be applied at next reboot.
2251          */
isStaged()2252         public boolean isStaged() {
2253             return isStaged;
2254         }
2255 
2256         /**
2257          * Returns the parent multi-package session ID if this session belongs to one,
2258          * {@link #INVALID_ID} otherwise.
2259          */
getParentSessionId()2260         public int getParentSessionId() {
2261             return parentSessionId;
2262         }
2263 
2264         /**
2265          * Returns the set of session IDs that will be committed when this session is commited if
2266          * this session is a multi-package session.
2267          */
2268         @NonNull
getChildSessionIds()2269         public int[] getChildSessionIds() {
2270             return childSessionIds;
2271         }
2272 
checkSessionIsStaged()2273         private void checkSessionIsStaged() {
2274             if (!isStaged) {
2275                 throw new IllegalStateException("Session is not marked as staged.");
2276             }
2277         }
2278 
2279         /**
2280          * Whether the staged session has been applied successfully, meaning that all of its
2281          * packages have been activated and no further action is required.
2282          * Only meaningful if {@code isStaged} is true.
2283          */
isStagedSessionApplied()2284         public boolean isStagedSessionApplied() {
2285             checkSessionIsStaged();
2286             return isStagedSessionApplied;
2287         }
2288 
2289         /**
2290          * Whether the staged session is ready to be applied at next reboot. Only meaningful if
2291          * {@code isStaged} is true.
2292          */
isStagedSessionReady()2293         public boolean isStagedSessionReady() {
2294             checkSessionIsStaged();
2295             return isStagedSessionReady;
2296         }
2297 
2298         /**
2299          * Whether something went wrong and the staged session is declared as failed, meaning that
2300          * it will be ignored at next reboot. Only meaningful if {@code isStaged} is true.
2301          */
isStagedSessionFailed()2302         public boolean isStagedSessionFailed() {
2303             checkSessionIsStaged();
2304             return isStagedSessionFailed;
2305         }
2306 
2307         /**
2308          * If something went wrong with a staged session, clients can check this error code to
2309          * understand which kind of failure happened. Only meaningful if {@code isStaged} is true.
2310          */
getStagedSessionErrorCode()2311         public @StagedSessionErrorCode int getStagedSessionErrorCode() {
2312             checkSessionIsStaged();
2313             return mStagedSessionErrorCode;
2314         }
2315 
2316         /**
2317          * Text description of the error code returned by {@code getStagedSessionErrorCode}, or
2318          * empty string if no error was encountered.
2319          */
getStagedSessionErrorMessage()2320         public @NonNull String getStagedSessionErrorMessage() {
2321             checkSessionIsStaged();
2322             return mStagedSessionErrorMessage;
2323         }
2324 
2325         /** {@hide} */
setStagedSessionErrorCode(@tagedSessionErrorCode int errorCode, String errorMessage)2326         public void setStagedSessionErrorCode(@StagedSessionErrorCode int errorCode,
2327                                               String errorMessage) {
2328             mStagedSessionErrorCode = errorCode;
2329             mStagedSessionErrorMessage = errorMessage;
2330         }
2331 
2332         /**
2333          * Returns {@code true} if {@link Session#commit(IntentSender)}} was called for this
2334          * session.
2335          */
isCommitted()2336         public boolean isCommitted() {
2337             return isCommitted;
2338         }
2339 
2340         /**
2341          * The timestamp of the last update that occurred to the session, including changing of
2342          * states in case of staged sessions.
2343          */
2344         @CurrentTimeMillisLong
getUpdatedMillis()2345         public long getUpdatedMillis() {
2346             return updatedMillis;
2347         }
2348 
2349         @Override
describeContents()2350         public int describeContents() {
2351             return 0;
2352         }
2353 
2354         @Override
writeToParcel(Parcel dest, int flags)2355         public void writeToParcel(Parcel dest, int flags) {
2356             dest.writeInt(sessionId);
2357             dest.writeInt(userId);
2358             dest.writeString(installerPackageName);
2359             dest.writeString(resolvedBaseCodePath);
2360             dest.writeFloat(progress);
2361             dest.writeInt(sealed ? 1 : 0);
2362             dest.writeInt(active ? 1 : 0);
2363 
2364             dest.writeInt(mode);
2365             dest.writeInt(installReason);
2366             dest.writeLong(sizeBytes);
2367             dest.writeString(appPackageName);
2368             dest.writeParcelable(appIcon, flags);
2369             dest.writeString(appLabel != null ? appLabel.toString() : null);
2370 
2371             dest.writeInt(installLocation);
2372             dest.writeParcelable(originatingUri, flags);
2373             dest.writeInt(originatingUid);
2374             dest.writeParcelable(referrerUri, flags);
2375             dest.writeStringArray(grantedRuntimePermissions);
2376             dest.writeStringList(whitelistedRestrictedPermissions);
2377             dest.writeInt(installFlags);
2378             dest.writeBoolean(isMultiPackage);
2379             dest.writeBoolean(isStaged);
2380             dest.writeInt(parentSessionId);
2381             dest.writeIntArray(childSessionIds);
2382             dest.writeBoolean(isStagedSessionApplied);
2383             dest.writeBoolean(isStagedSessionReady);
2384             dest.writeBoolean(isStagedSessionFailed);
2385             dest.writeInt(mStagedSessionErrorCode);
2386             dest.writeString(mStagedSessionErrorMessage);
2387             dest.writeBoolean(isCommitted);
2388         }
2389 
2390         public static final Parcelable.Creator<SessionInfo>
2391                 CREATOR = new Parcelable.Creator<SessionInfo>() {
2392                     @Override
2393                     public SessionInfo createFromParcel(Parcel p) {
2394                         return new SessionInfo(p);
2395                     }
2396 
2397                     @Override
2398                     public SessionInfo[] newArray(int size) {
2399                         return new SessionInfo[size];
2400                     }
2401                 };
2402     }
2403 }
2404