PackageMonitor.java revision d0d7503fd3e941113094532f28986f49b11b5fdb
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
17package com.android.internal.content;
18
19import android.app.Activity;
20import android.content.Context;
21import android.content.Intent;
22import android.content.IntentFilter;
23import android.net.Uri;
24import android.os.Handler;
25import android.os.HandlerThread;
26import android.os.Looper;
27
28import java.util.HashSet;
29
30/**
31 * Helper class for monitoring the state of packages: adding, removing,
32 * updating, and disappearing and reappearing on the SD card.
33 */
34public abstract class PackageMonitor extends android.content.BroadcastReceiver {
35    static final IntentFilter sPackageFilt = new IntentFilter();
36    static final IntentFilter sNonDataFilt = new IntentFilter();
37    static final IntentFilter sExternalFilt = new IntentFilter();
38
39    static final Object sLock = new Object();
40    static HandlerThread sBackgroundThread;
41    static Handler sBackgroundHandler;
42
43    static {
44        sPackageFilt.addAction(Intent.ACTION_PACKAGE_ADDED);
45        sPackageFilt.addAction(Intent.ACTION_PACKAGE_REMOVED);
46        sPackageFilt.addAction(Intent.ACTION_PACKAGE_CHANGED);
47        sPackageFilt.addAction(Intent.ACTION_QUERY_PACKAGE_RESTART);
48        sPackageFilt.addAction(Intent.ACTION_PACKAGE_RESTARTED);
49        sPackageFilt.addAction(Intent.ACTION_UID_REMOVED);
50        sPackageFilt.addDataScheme("package");
51        sNonDataFilt.addAction(Intent.ACTION_UID_REMOVED);
52        sExternalFilt.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE);
53        sExternalFilt.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE);
54    }
55
56    final HashSet<String> mUpdatingPackages = new HashSet<String>();
57
58    Context mRegisteredContext;
59    Handler mRegisteredHandler;
60    String[] mDisappearingPackages;
61    String[] mAppearingPackages;
62    String[] mModifiedPackages;
63    int mChangeType;
64    boolean mSomePackagesChanged;
65
66    String[] mTempArray = new String[1];
67
68    public void register(Context context, Looper thread, boolean externalStorage) {
69        if (mRegisteredContext != null) {
70            throw new IllegalStateException("Already registered");
71        }
72        mRegisteredContext = context;
73        if (thread == null) {
74            synchronized (sLock) {
75                if (sBackgroundThread == null) {
76                    sBackgroundThread = new HandlerThread("PackageMonitor",
77                            android.os.Process.THREAD_PRIORITY_BACKGROUND);
78                    sBackgroundThread.start();
79                    sBackgroundHandler = new Handler(sBackgroundThread.getLooper());
80                }
81                mRegisteredHandler = sBackgroundHandler;
82            }
83        } else {
84            mRegisteredHandler = new Handler(thread);
85        }
86        context.registerReceiver(this, sPackageFilt, null, mRegisteredHandler);
87        context.registerReceiver(this, sNonDataFilt, null, mRegisteredHandler);
88        if (externalStorage) {
89            context.registerReceiver(this, sExternalFilt, null, mRegisteredHandler);
90        }
91    }
92
93    public Handler getRegisteredHandler() {
94        return mRegisteredHandler;
95    }
96
97    public void unregister() {
98        if (mRegisteredContext == null) {
99            throw new IllegalStateException("Not registered");
100        }
101        mRegisteredContext.unregisterReceiver(this);
102        mRegisteredContext = null;
103    }
104
105    //not yet implemented
106    boolean isPackageUpdating(String packageName) {
107        synchronized (mUpdatingPackages) {
108            return mUpdatingPackages.contains(packageName);
109        }
110    }
111
112    public void onBeginPackageChanges() {
113    }
114
115    public void onPackageAdded(String packageName, int uid) {
116    }
117
118    public void onPackageRemoved(String packageName, int uid) {
119    }
120
121    public void onPackageUpdateStarted(String packageName, int uid) {
122    }
123
124    public void onPackageUpdateFinished(String packageName, int uid) {
125    }
126
127    public void onPackageChanged(String packageName, int uid, String[] components) {
128    }
129
130    public boolean onHandleForceStop(Intent intent, String[] packages, int uid, boolean doit) {
131        return false;
132    }
133
134    public void onUidRemoved(int uid) {
135    }
136
137    public void onPackagesAvailable(String[] packages) {
138    }
139
140    public void onPackagesUnavailable(String[] packages) {
141    }
142
143    public static final int PACKAGE_UNCHANGED = 0;
144    public static final int PACKAGE_UPDATING = 1;
145    public static final int PACKAGE_TEMPORARY_CHANGE = 2;
146    public static final int PACKAGE_PERMANENT_CHANGE = 3;
147
148    public void onPackageDisappeared(String packageName, int reason) {
149    }
150
151    public void onPackageAppeared(String packageName, int reason) {
152    }
153
154    public void onPackageModified(String packageName) {
155    }
156
157    public boolean didSomePackagesChange() {
158        return mSomePackagesChanged;
159    }
160
161    public int isPackageAppearing(String packageName) {
162        if (mAppearingPackages != null) {
163            for (int i=mAppearingPackages.length-1; i>=0; i--) {
164                if (packageName.equals(mAppearingPackages[i])) {
165                    return mChangeType;
166                }
167            }
168        }
169        return PACKAGE_UNCHANGED;
170    }
171
172    public boolean anyPackagesAppearing() {
173        return mAppearingPackages != null;
174    }
175
176    public int isPackageDisappearing(String packageName) {
177        if (mDisappearingPackages != null) {
178            for (int i=mDisappearingPackages.length-1; i>=0; i--) {
179                if (packageName.equals(mDisappearingPackages[i])) {
180                    return mChangeType;
181                }
182            }
183        }
184        return PACKAGE_UNCHANGED;
185    }
186
187    public boolean anyPackagesDisappearing() {
188        return mDisappearingPackages != null;
189    }
190
191    public boolean isPackageModified(String packageName) {
192        if (mModifiedPackages != null) {
193            for (int i=mModifiedPackages.length-1; i>=0; i--) {
194                if (packageName.equals(mModifiedPackages[i])) {
195                    return true;
196                }
197            }
198        }
199        return false;
200    }
201
202    public void onSomePackagesChanged() {
203    }
204
205    public void onFinishPackageChanges() {
206    }
207
208    String getPackageName(Intent intent) {
209        Uri uri = intent.getData();
210        String pkg = uri != null ? uri.getSchemeSpecificPart() : null;
211        return pkg;
212    }
213
214    @Override
215    public void onReceive(Context context, Intent intent) {
216        onBeginPackageChanges();
217
218        mDisappearingPackages = mAppearingPackages = null;
219        mSomePackagesChanged = false;
220
221        String action = intent.getAction();
222        if (Intent.ACTION_PACKAGE_ADDED.equals(action)) {
223            String pkg = getPackageName(intent);
224            int uid = intent.getIntExtra(Intent.EXTRA_UID, 0);
225            // We consider something to have changed regardless of whether
226            // this is just an update, because the update is now finished
227            // and the contents of the package may have changed.
228            mSomePackagesChanged = true;
229            if (pkg != null) {
230                mAppearingPackages = mTempArray;
231                mTempArray[0] = pkg;
232                if (intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)) {
233                    mModifiedPackages = mTempArray;
234                    mChangeType = PACKAGE_UPDATING;
235                    onPackageUpdateFinished(pkg, uid);
236                    onPackageModified(pkg);
237                } else {
238                    mChangeType = PACKAGE_PERMANENT_CHANGE;
239                    onPackageAdded(pkg, uid);
240                }
241                onPackageAppeared(pkg, mChangeType);
242                if (mChangeType == PACKAGE_UPDATING) {
243                    synchronized (mUpdatingPackages) {
244                        mUpdatingPackages.remove(pkg);
245                    }
246                }
247            }
248        } else if (Intent.ACTION_PACKAGE_REMOVED.equals(action)) {
249            String pkg = getPackageName(intent);
250            int uid = intent.getIntExtra(Intent.EXTRA_UID, 0);
251            if (pkg != null) {
252                mDisappearingPackages = mTempArray;
253                mTempArray[0] = pkg;
254                if (intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)) {
255                    mChangeType = PACKAGE_UPDATING;
256                    synchronized (mUpdatingPackages) {
257                        //not used for now
258                        //mUpdatingPackages.add(pkg);
259                    }
260                    onPackageUpdateStarted(pkg, uid);
261                } else {
262                    mChangeType = PACKAGE_PERMANENT_CHANGE;
263                    // We only consider something to have changed if this is
264                    // not a replace; for a replace, we just need to consider
265                    // it when it is re-added.
266                    mSomePackagesChanged = true;
267                    onPackageRemoved(pkg, uid);
268                }
269                onPackageDisappeared(pkg, mChangeType);
270            }
271        } else if (Intent.ACTION_PACKAGE_CHANGED.equals(action)) {
272            String pkg = getPackageName(intent);
273            int uid = intent.getIntExtra(Intent.EXTRA_UID, 0);
274            String[] components = intent.getStringArrayExtra(
275                    Intent.EXTRA_CHANGED_COMPONENT_NAME_LIST);
276            if (pkg != null) {
277                mModifiedPackages = mTempArray;
278                mTempArray[0] = pkg;
279                onPackageChanged(pkg, uid, components);
280                // XXX Don't want this to always cause mSomePackagesChanged,
281                // since it can happen a fair amount.
282                onPackageModified(pkg);
283            }
284        } else if (Intent.ACTION_QUERY_PACKAGE_RESTART.equals(action)) {
285            mDisappearingPackages = intent.getStringArrayExtra(Intent.EXTRA_PACKAGES);
286            mChangeType = PACKAGE_TEMPORARY_CHANGE;
287            boolean canRestart = onHandleForceStop(intent,
288                    mDisappearingPackages,
289                    intent.getIntExtra(Intent.EXTRA_UID, 0), false);
290            if (canRestart) setResultCode(Activity.RESULT_OK);
291        } else if (Intent.ACTION_PACKAGE_RESTARTED.equals(action)) {
292            mDisappearingPackages = new String[] {getPackageName(intent)};
293            mChangeType = PACKAGE_TEMPORARY_CHANGE;
294            onHandleForceStop(intent, mDisappearingPackages,
295                    intent.getIntExtra(Intent.EXTRA_UID, 0), true);
296        } else if (Intent.ACTION_UID_REMOVED.equals(action)) {
297            onUidRemoved(intent.getIntExtra(Intent.EXTRA_UID, 0));
298        } else if (Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE.equals(action)) {
299            String[] pkgList = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST);
300            mAppearingPackages = pkgList;
301            mChangeType = PACKAGE_TEMPORARY_CHANGE;
302            mSomePackagesChanged = true;
303            if (pkgList != null) {
304                onPackagesAvailable(pkgList);
305                for (int i=0; i<pkgList.length; i++) {
306                    onPackageAppeared(pkgList[i], PACKAGE_TEMPORARY_CHANGE);
307                }
308            }
309        } else if (Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE.equals(action)) {
310            String[] pkgList = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST);
311            mDisappearingPackages = pkgList;
312            mChangeType = PACKAGE_TEMPORARY_CHANGE;
313            mSomePackagesChanged = true;
314            if (pkgList != null) {
315                onPackagesUnavailable(pkgList);
316                for (int i=0; i<pkgList.length; i++) {
317                    onPackageDisappeared(pkgList[i], PACKAGE_TEMPORARY_CHANGE);
318                }
319            }
320        }
321
322        if (mSomePackagesChanged) {
323            onSomePackagesChanged();
324        }
325
326        onFinishPackageChanges();
327    }
328}
329