1 /*
2  * Copyright (C) 2010 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 com.android.internal.content;
18 
19 import android.annotation.UnsupportedAppUsage;
20 import android.app.Activity;
21 import android.content.Context;
22 import android.content.Intent;
23 import android.content.IntentFilter;
24 import android.net.Uri;
25 import android.os.Bundle;
26 import android.os.Handler;
27 import android.os.Looper;
28 import android.os.UserHandle;
29 import android.util.Slog;
30 import com.android.internal.os.BackgroundThread;
31 import com.android.internal.util.Preconditions;
32 
33 import java.util.HashSet;
34 
35 /**
36  * Helper class for monitoring the state of packages: adding, removing,
37  * updating, and disappearing and reappearing on the SD card.
38  */
39 public abstract class PackageMonitor extends android.content.BroadcastReceiver {
40     static final IntentFilter sPackageFilt = new IntentFilter();
41     static final IntentFilter sNonDataFilt = new IntentFilter();
42     static final IntentFilter sExternalFilt = new IntentFilter();
43 
44     static {
45         sPackageFilt.addAction(Intent.ACTION_PACKAGE_ADDED);
46         sPackageFilt.addAction(Intent.ACTION_PACKAGE_REMOVED);
47         sPackageFilt.addAction(Intent.ACTION_PACKAGE_CHANGED);
48         sPackageFilt.addAction(Intent.ACTION_QUERY_PACKAGE_RESTART);
49         sPackageFilt.addAction(Intent.ACTION_PACKAGE_RESTARTED);
50         sPackageFilt.addAction(Intent.ACTION_PACKAGE_DATA_CLEARED);
51         sPackageFilt.addDataScheme("package");
52         sNonDataFilt.addAction(Intent.ACTION_UID_REMOVED);
53         sNonDataFilt.addAction(Intent.ACTION_USER_STOPPED);
54         sNonDataFilt.addAction(Intent.ACTION_PACKAGES_SUSPENDED);
55         sNonDataFilt.addAction(Intent.ACTION_PACKAGES_UNSUSPENDED);
56         sExternalFilt.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE);
57         sExternalFilt.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE);
58     }
59 
60     final HashSet<String> mUpdatingPackages = new HashSet<String>();
61 
62     Context mRegisteredContext;
63     Handler mRegisteredHandler;
64     String[] mDisappearingPackages;
65     String[] mAppearingPackages;
66     String[] mModifiedPackages;
67     int mChangeType;
68     int mChangeUserId = UserHandle.USER_NULL;
69     boolean mSomePackagesChanged;
70     String[] mModifiedComponents;
71 
72     String[] mTempArray = new String[1];
73 
74     @UnsupportedAppUsage
register(Context context, Looper thread, boolean externalStorage)75     public void register(Context context, Looper thread, boolean externalStorage) {
76         register(context, thread, null, externalStorage);
77     }
78 
79     @UnsupportedAppUsage
register(Context context, Looper thread, UserHandle user, boolean externalStorage)80     public void register(Context context, Looper thread, UserHandle user,
81             boolean externalStorage) {
82         register(context, user, externalStorage,
83                 (thread == null) ? BackgroundThread.getHandler() : new Handler(thread));
84     }
85 
register(Context context, UserHandle user, boolean externalStorage, Handler handler)86     public void register(Context context, UserHandle user,
87         boolean externalStorage, Handler handler) {
88         if (mRegisteredContext != null) {
89             throw new IllegalStateException("Already registered");
90         }
91         mRegisteredContext = context;
92         mRegisteredHandler = Preconditions.checkNotNull(handler);
93         if (user != null) {
94             context.registerReceiverAsUser(this, user, sPackageFilt, null, mRegisteredHandler);
95             context.registerReceiverAsUser(this, user, sNonDataFilt, null, mRegisteredHandler);
96             if (externalStorage) {
97                 context.registerReceiverAsUser(this, user, sExternalFilt, null,
98                         mRegisteredHandler);
99             }
100         } else {
101             context.registerReceiver(this, sPackageFilt, null, mRegisteredHandler);
102             context.registerReceiver(this, sNonDataFilt, null, mRegisteredHandler);
103             if (externalStorage) {
104                 context.registerReceiver(this, sExternalFilt, null, mRegisteredHandler);
105             }
106         }
107     }
108 
getRegisteredHandler()109     public Handler getRegisteredHandler() {
110         return mRegisteredHandler;
111     }
112 
113     @UnsupportedAppUsage
unregister()114     public void unregister() {
115         if (mRegisteredContext == null) {
116             throw new IllegalStateException("Not registered");
117         }
118         mRegisteredContext.unregisterReceiver(this);
119         mRegisteredContext = null;
120     }
121 
122     //not yet implemented
isPackageUpdating(String packageName)123     boolean isPackageUpdating(String packageName) {
124         synchronized (mUpdatingPackages) {
125             return mUpdatingPackages.contains(packageName);
126         }
127     }
128 
onBeginPackageChanges()129     public void onBeginPackageChanges() {
130     }
131 
132     /**
133      * Called when a package is really added (and not replaced).
134      */
onPackageAdded(String packageName, int uid)135     public void onPackageAdded(String packageName, int uid) {
136     }
137 
138     /**
139      * Called when a package is really removed (and not replaced).
140      */
141     @UnsupportedAppUsage
onPackageRemoved(String packageName, int uid)142     public void onPackageRemoved(String packageName, int uid) {
143     }
144 
145     /**
146      * Called when a package is really removed (and not replaced) for
147      * all users on the device.
148      */
onPackageRemovedAllUsers(String packageName, int uid)149     public void onPackageRemovedAllUsers(String packageName, int uid) {
150     }
151 
onPackageUpdateStarted(String packageName, int uid)152     public void onPackageUpdateStarted(String packageName, int uid) {
153     }
154 
onPackageUpdateFinished(String packageName, int uid)155     public void onPackageUpdateFinished(String packageName, int uid) {
156     }
157 
158     /**
159      * Direct reflection of {@link Intent#ACTION_PACKAGE_CHANGED
160      * Intent.ACTION_PACKAGE_CHANGED} being received, informing you of
161      * changes to the enabled/disabled state of components in a package
162      * and/or of the overall package.
163      *
164      * @param packageName The name of the package that is changing.
165      * @param uid The user ID the package runs under.
166      * @param components Any components in the package that are changing.  If
167      * the overall package is changing, this will contain an entry of the
168      * package name itself.
169      * @return Return true to indicate you care about this change, which will
170      * result in {@link #onSomePackagesChanged()} being called later.  If you
171      * return false, no further callbacks will happen about this change.  The
172      * default implementation returns true if this is a change to the entire
173      * package.
174      */
175     @UnsupportedAppUsage
onPackageChanged(String packageName, int uid, String[] components)176     public boolean onPackageChanged(String packageName, int uid, String[] components) {
177         if (components != null) {
178             for (String name : components) {
179                 if (packageName.equals(name)) {
180                     return true;
181                 }
182             }
183         }
184         return false;
185     }
186 
onHandleForceStop(Intent intent, String[] packages, int uid, boolean doit)187     public boolean onHandleForceStop(Intent intent, String[] packages, int uid, boolean doit) {
188         return false;
189     }
190 
onHandleUserStop(Intent intent, int userHandle)191     public void onHandleUserStop(Intent intent, int userHandle) {
192     }
193 
onUidRemoved(int uid)194     public void onUidRemoved(int uid) {
195     }
196 
onPackagesAvailable(String[] packages)197     public void onPackagesAvailable(String[] packages) {
198     }
199 
onPackagesUnavailable(String[] packages)200     public void onPackagesUnavailable(String[] packages) {
201     }
202 
onPackagesSuspended(String[] packages)203     public void onPackagesSuspended(String[] packages) {
204     }
205 
onPackagesSuspended(String[] packages, Bundle launcherExtras)206     public void onPackagesSuspended(String[] packages, Bundle launcherExtras) {
207         onPackagesSuspended(packages);
208     }
209 
onPackagesUnsuspended(String[] packages)210     public void onPackagesUnsuspended(String[] packages) {
211     }
212 
213     public static final int PACKAGE_UNCHANGED = 0;
214     public static final int PACKAGE_UPDATING = 1;
215     public static final int PACKAGE_TEMPORARY_CHANGE = 2;
216     public static final int PACKAGE_PERMANENT_CHANGE = 3;
217 
218     /**
219      * Called when a package disappears for any reason.
220      */
onPackageDisappeared(String packageName, int reason)221     public void onPackageDisappeared(String packageName, int reason) {
222     }
223 
224     /**
225      * Called when a package appears for any reason.
226      */
onPackageAppeared(String packageName, int reason)227     public void onPackageAppeared(String packageName, int reason) {
228     }
229 
230     /**
231      * Called when an existing package is updated or its disabled state changes.
232      */
onPackageModified(String packageName)233     public void onPackageModified(String packageName) {
234     }
235 
didSomePackagesChange()236     public boolean didSomePackagesChange() {
237         return mSomePackagesChanged;
238     }
239 
isPackageAppearing(String packageName)240     public int isPackageAppearing(String packageName) {
241         if (mAppearingPackages != null) {
242             for (int i=mAppearingPackages.length-1; i>=0; i--) {
243                 if (packageName.equals(mAppearingPackages[i])) {
244                     return mChangeType;
245                 }
246             }
247         }
248         return PACKAGE_UNCHANGED;
249     }
250 
anyPackagesAppearing()251     public boolean anyPackagesAppearing() {
252         return mAppearingPackages != null;
253     }
254 
255     @UnsupportedAppUsage
isPackageDisappearing(String packageName)256     public int isPackageDisappearing(String packageName) {
257         if (mDisappearingPackages != null) {
258             for (int i=mDisappearingPackages.length-1; i>=0; i--) {
259                 if (packageName.equals(mDisappearingPackages[i])) {
260                     return mChangeType;
261                 }
262             }
263         }
264         return PACKAGE_UNCHANGED;
265     }
266 
anyPackagesDisappearing()267     public boolean anyPackagesDisappearing() {
268         return mDisappearingPackages != null;
269     }
270 
isReplacing()271     public boolean isReplacing() {
272         return mChangeType == PACKAGE_UPDATING;
273     }
274 
275     @UnsupportedAppUsage
isPackageModified(String packageName)276     public boolean isPackageModified(String packageName) {
277         if (mModifiedPackages != null) {
278             for (int i=mModifiedPackages.length-1; i>=0; i--) {
279                 if (packageName.equals(mModifiedPackages[i])) {
280                     return true;
281                 }
282             }
283         }
284         return false;
285     }
286 
isComponentModified(String className)287     public boolean isComponentModified(String className) {
288         if (className == null || mModifiedComponents == null) {
289             return false;
290         }
291         for (int i = mModifiedComponents.length - 1; i >= 0; i--) {
292             if (className.equals(mModifiedComponents[i])) {
293                 return true;
294             }
295         }
296         return false;
297     }
298 
onSomePackagesChanged()299     public void onSomePackagesChanged() {
300     }
301 
onFinishPackageChanges()302     public void onFinishPackageChanges() {
303     }
304 
onPackageDataCleared(String packageName, int uid)305     public void onPackageDataCleared(String packageName, int uid) {
306     }
307 
getChangingUserId()308     public int getChangingUserId() {
309         return mChangeUserId;
310     }
311 
getPackageName(Intent intent)312     String getPackageName(Intent intent) {
313         Uri uri = intent.getData();
314         String pkg = uri != null ? uri.getSchemeSpecificPart() : null;
315         return pkg;
316     }
317 
318     @Override
onReceive(Context context, Intent intent)319     public void onReceive(Context context, Intent intent) {
320         mChangeUserId = intent.getIntExtra(Intent.EXTRA_USER_HANDLE,
321                 UserHandle.USER_NULL);
322         if (mChangeUserId == UserHandle.USER_NULL) {
323             Slog.w("PackageMonitor", "Intent broadcast does not contain user handle: " + intent);
324             return;
325         }
326         onBeginPackageChanges();
327 
328         mDisappearingPackages = mAppearingPackages = null;
329         mSomePackagesChanged = false;
330         mModifiedComponents = null;
331 
332         String action = intent.getAction();
333         if (Intent.ACTION_PACKAGE_ADDED.equals(action)) {
334             String pkg = getPackageName(intent);
335             int uid = intent.getIntExtra(Intent.EXTRA_UID, 0);
336             // We consider something to have changed regardless of whether
337             // this is just an update, because the update is now finished
338             // and the contents of the package may have changed.
339             mSomePackagesChanged = true;
340             if (pkg != null) {
341                 mAppearingPackages = mTempArray;
342                 mTempArray[0] = pkg;
343                 if (intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)) {
344                     mModifiedPackages = mTempArray;
345                     mChangeType = PACKAGE_UPDATING;
346                     onPackageUpdateFinished(pkg, uid);
347                     onPackageModified(pkg);
348                 } else {
349                     mChangeType = PACKAGE_PERMANENT_CHANGE;
350                     onPackageAdded(pkg, uid);
351                 }
352                 onPackageAppeared(pkg, mChangeType);
353                 if (mChangeType == PACKAGE_UPDATING) {
354                     synchronized (mUpdatingPackages) {
355                         mUpdatingPackages.remove(pkg);
356                     }
357                 }
358             }
359         } else if (Intent.ACTION_PACKAGE_REMOVED.equals(action)) {
360             String pkg = getPackageName(intent);
361             int uid = intent.getIntExtra(Intent.EXTRA_UID, 0);
362             if (pkg != null) {
363                 mDisappearingPackages = mTempArray;
364                 mTempArray[0] = pkg;
365                 if (intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)) {
366                     mChangeType = PACKAGE_UPDATING;
367                     synchronized (mUpdatingPackages) {
368                         //not used for now
369                         //mUpdatingPackages.add(pkg);
370                     }
371                     onPackageUpdateStarted(pkg, uid);
372                 } else {
373                     mChangeType = PACKAGE_PERMANENT_CHANGE;
374                     // We only consider something to have changed if this is
375                     // not a replace; for a replace, we just need to consider
376                     // it when it is re-added.
377                     mSomePackagesChanged = true;
378                     onPackageRemoved(pkg, uid);
379                     if (intent.getBooleanExtra(Intent.EXTRA_REMOVED_FOR_ALL_USERS, false)) {
380                         onPackageRemovedAllUsers(pkg, uid);
381                     }
382                 }
383                 onPackageDisappeared(pkg, mChangeType);
384             }
385         } else if (Intent.ACTION_PACKAGE_CHANGED.equals(action)) {
386             String pkg = getPackageName(intent);
387             int uid = intent.getIntExtra(Intent.EXTRA_UID, 0);
388             mModifiedComponents = intent.getStringArrayExtra(
389                     Intent.EXTRA_CHANGED_COMPONENT_NAME_LIST);
390             if (pkg != null) {
391                 mModifiedPackages = mTempArray;
392                 mTempArray[0] = pkg;
393                 mChangeType = PACKAGE_PERMANENT_CHANGE;
394                 if (onPackageChanged(pkg, uid, mModifiedComponents)) {
395                     mSomePackagesChanged = true;
396                 }
397                 onPackageModified(pkg);
398             }
399         } else if (Intent.ACTION_PACKAGE_DATA_CLEARED.equals(action)) {
400             String pkg = getPackageName(intent);
401             int uid = intent.getIntExtra(Intent.EXTRA_UID, 0);
402             if (pkg != null) {
403                 onPackageDataCleared(pkg, uid);
404             }
405         } else if (Intent.ACTION_QUERY_PACKAGE_RESTART.equals(action)) {
406             mDisappearingPackages = intent.getStringArrayExtra(Intent.EXTRA_PACKAGES);
407             mChangeType = PACKAGE_TEMPORARY_CHANGE;
408             boolean canRestart = onHandleForceStop(intent,
409                     mDisappearingPackages,
410                     intent.getIntExtra(Intent.EXTRA_UID, 0), false);
411             if (canRestart) setResultCode(Activity.RESULT_OK);
412         } else if (Intent.ACTION_PACKAGE_RESTARTED.equals(action)) {
413             mDisappearingPackages = new String[] {getPackageName(intent)};
414             mChangeType = PACKAGE_TEMPORARY_CHANGE;
415             onHandleForceStop(intent, mDisappearingPackages,
416                     intent.getIntExtra(Intent.EXTRA_UID, 0), true);
417         } else if (Intent.ACTION_UID_REMOVED.equals(action)) {
418             onUidRemoved(intent.getIntExtra(Intent.EXTRA_UID, 0));
419         } else if (Intent.ACTION_USER_STOPPED.equals(action)) {
420             if (intent.hasExtra(Intent.EXTRA_USER_HANDLE)) {
421                 onHandleUserStop(intent, intent.getIntExtra(Intent.EXTRA_USER_HANDLE, 0));
422             }
423         } else if (Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE.equals(action)) {
424             String[] pkgList = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST);
425             mAppearingPackages = pkgList;
426             mChangeType = intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)
427                     ? PACKAGE_UPDATING : PACKAGE_TEMPORARY_CHANGE;
428             mSomePackagesChanged = true;
429             if (pkgList != null) {
430                 onPackagesAvailable(pkgList);
431                 for (int i=0; i<pkgList.length; i++) {
432                     onPackageAppeared(pkgList[i], mChangeType);
433                 }
434             }
435         } else if (Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE.equals(action)) {
436             String[] pkgList = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST);
437             mDisappearingPackages = pkgList;
438             mChangeType = intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)
439                     ? PACKAGE_UPDATING : PACKAGE_TEMPORARY_CHANGE;
440             mSomePackagesChanged = true;
441             if (pkgList != null) {
442                 onPackagesUnavailable(pkgList);
443                 for (int i=0; i<pkgList.length; i++) {
444                     onPackageDisappeared(pkgList[i], mChangeType);
445                 }
446             }
447         } else if (Intent.ACTION_PACKAGES_SUSPENDED.equals(action)) {
448             String[] pkgList = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST);
449             final Bundle launcherExtras = intent.getBundleExtra(Intent.EXTRA_LAUNCHER_EXTRAS);
450             mSomePackagesChanged = true;
451             onPackagesSuspended(pkgList, launcherExtras);
452         } else if (Intent.ACTION_PACKAGES_UNSUSPENDED.equals(action)) {
453             String[] pkgList = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST);
454             mSomePackagesChanged = true;
455             onPackagesUnsuspended(pkgList);
456         }
457 
458         if (mSomePackagesChanged) {
459             onSomePackagesChanged();
460         }
461 
462         onFinishPackageChanges();
463         mChangeUserId = UserHandle.USER_NULL;
464     }
465 }
466