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