1 /* 2 * Copyright (C) 2013 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.documentsui; 18 19 import static com.android.documentsui.DocumentsActivity.TAG; 20 21 import android.content.ContentProviderClient; 22 import android.content.ContentResolver; 23 import android.content.Context; 24 import android.content.Intent; 25 import android.content.pm.ApplicationInfo; 26 import android.content.pm.PackageManager; 27 import android.content.pm.ProviderInfo; 28 import android.content.pm.ResolveInfo; 29 import android.database.ContentObserver; 30 import android.database.Cursor; 31 import android.net.Uri; 32 import android.os.AsyncTask; 33 import android.os.Handler; 34 import android.os.SystemClock; 35 import android.provider.DocumentsContract; 36 import android.provider.DocumentsContract.Root; 37 import android.util.Log; 38 39 import com.android.documentsui.DocumentsActivity.State; 40 import com.android.documentsui.model.RootInfo; 41 import com.android.internal.annotations.GuardedBy; 42 import com.android.internal.annotations.VisibleForTesting; 43 import com.google.android.collect.Lists; 44 import com.google.android.collect.Sets; 45 import com.google.common.collect.ArrayListMultimap; 46 import com.google.common.collect.Multimap; 47 48 import libcore.io.IoUtils; 49 50 import java.util.Collection; 51 import java.util.HashSet; 52 import java.util.List; 53 import java.util.Objects; 54 import java.util.concurrent.CountDownLatch; 55 import java.util.concurrent.TimeUnit; 56 57 /** 58 * Cache of known storage backends and their roots. 59 */ 60 public class RootsCache { 61 private static final boolean LOGD = false; 62 63 public static final Uri sNotificationUri = Uri.parse( 64 "content://com.android.documentsui.roots/"); 65 66 private final Context mContext; 67 private final ContentObserver mObserver; 68 69 private final RootInfo mRecentsRoot = new RootInfo(); 70 71 private final Object mLock = new Object(); 72 private final CountDownLatch mFirstLoad = new CountDownLatch(1); 73 74 @GuardedBy("mLock") 75 private Multimap<String, RootInfo> mRoots = ArrayListMultimap.create(); 76 @GuardedBy("mLock") 77 private HashSet<String> mStoppedAuthorities = Sets.newHashSet(); 78 79 @GuardedBy("mObservedAuthorities") 80 private final HashSet<String> mObservedAuthorities = Sets.newHashSet(); 81 RootsCache(Context context)82 public RootsCache(Context context) { 83 mContext = context; 84 mObserver = new RootsChangedObserver(); 85 } 86 87 private class RootsChangedObserver extends ContentObserver { RootsChangedObserver()88 public RootsChangedObserver() { 89 super(new Handler()); 90 } 91 92 @Override onChange(boolean selfChange, Uri uri)93 public void onChange(boolean selfChange, Uri uri) { 94 if (LOGD) Log.d(TAG, "Updating roots due to change at " + uri); 95 updateAuthorityAsync(uri.getAuthority()); 96 } 97 } 98 99 /** 100 * Gather roots from all known storage providers. 101 */ updateAsync()102 public void updateAsync() { 103 // Special root for recents 104 mRecentsRoot.authority = null; 105 mRecentsRoot.rootId = null; 106 mRecentsRoot.derivedIcon = R.drawable.ic_root_recent; 107 mRecentsRoot.flags = Root.FLAG_LOCAL_ONLY | Root.FLAG_SUPPORTS_CREATE 108 | Root.FLAG_SUPPORTS_IS_CHILD; 109 mRecentsRoot.title = mContext.getString(R.string.root_recent); 110 mRecentsRoot.availableBytes = -1; 111 112 new UpdateTask().executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); 113 } 114 115 /** 116 * Gather roots from storage providers belonging to given package name. 117 */ updatePackageAsync(String packageName)118 public void updatePackageAsync(String packageName) { 119 new UpdateTask(packageName).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); 120 } 121 122 /** 123 * Gather roots from storage providers belonging to given authority. 124 */ updateAuthorityAsync(String authority)125 public void updateAuthorityAsync(String authority) { 126 final ProviderInfo info = mContext.getPackageManager().resolveContentProvider(authority, 0); 127 if (info != null) { 128 updatePackageAsync(info.packageName); 129 } 130 } 131 waitForFirstLoad()132 private void waitForFirstLoad() { 133 boolean success = false; 134 try { 135 success = mFirstLoad.await(15, TimeUnit.SECONDS); 136 } catch (InterruptedException e) { 137 } 138 if (!success) { 139 Log.w(TAG, "Timeout waiting for first update"); 140 } 141 } 142 143 /** 144 * Load roots from authorities that are in stopped state. Normal 145 * {@link UpdateTask} passes ignore stopped applications. 146 */ loadStoppedAuthorities()147 private void loadStoppedAuthorities() { 148 final ContentResolver resolver = mContext.getContentResolver(); 149 synchronized (mLock) { 150 for (String authority : mStoppedAuthorities) { 151 if (LOGD) Log.d(TAG, "Loading stopped authority " + authority); 152 mRoots.putAll(authority, loadRootsForAuthority(resolver, authority)); 153 } 154 mStoppedAuthorities.clear(); 155 } 156 } 157 158 private class UpdateTask extends AsyncTask<Void, Void, Void> { 159 private final String mFilterPackage; 160 161 private final Multimap<String, RootInfo> mTaskRoots = ArrayListMultimap.create(); 162 private final HashSet<String> mTaskStoppedAuthorities = Sets.newHashSet(); 163 164 /** 165 * Update all roots. 166 */ UpdateTask()167 public UpdateTask() { 168 this(null); 169 } 170 171 /** 172 * Only update roots belonging to given package name. Other roots will 173 * be copied from cached {@link #mRoots} values. 174 */ UpdateTask(String filterPackage)175 public UpdateTask(String filterPackage) { 176 mFilterPackage = filterPackage; 177 } 178 179 @Override doInBackground(Void... params)180 protected Void doInBackground(Void... params) { 181 final long start = SystemClock.elapsedRealtime(); 182 183 if (mFilterPackage != null) { 184 // Need at least first load, since we're going to be using 185 // previously cached values for non-matching packages. 186 waitForFirstLoad(); 187 } 188 189 mTaskRoots.put(mRecentsRoot.authority, mRecentsRoot); 190 191 final ContentResolver resolver = mContext.getContentResolver(); 192 final PackageManager pm = mContext.getPackageManager(); 193 194 // Pick up provider with action string 195 final Intent intent = new Intent(DocumentsContract.PROVIDER_INTERFACE); 196 final List<ResolveInfo> providers = pm.queryIntentContentProviders(intent, 0); 197 for (ResolveInfo info : providers) { 198 handleDocumentsProvider(info.providerInfo); 199 } 200 201 final long delta = SystemClock.elapsedRealtime() - start; 202 Log.d(TAG, "Update found " + mTaskRoots.size() + " roots in " + delta + "ms"); 203 synchronized (mLock) { 204 mRoots = mTaskRoots; 205 mStoppedAuthorities = mTaskStoppedAuthorities; 206 } 207 mFirstLoad.countDown(); 208 resolver.notifyChange(sNotificationUri, null, false); 209 return null; 210 } 211 handleDocumentsProvider(ProviderInfo info)212 private void handleDocumentsProvider(ProviderInfo info) { 213 // Ignore stopped packages for now; we might query them 214 // later during UI interaction. 215 if ((info.applicationInfo.flags & ApplicationInfo.FLAG_STOPPED) != 0) { 216 if (LOGD) Log.d(TAG, "Ignoring stopped authority " + info.authority); 217 mTaskStoppedAuthorities.add(info.authority); 218 return; 219 } 220 221 // Try using cached roots if filtering 222 boolean cacheHit = false; 223 if (mFilterPackage != null && !mFilterPackage.equals(info.packageName)) { 224 synchronized (mLock) { 225 if (mTaskRoots.putAll(info.authority, mRoots.get(info.authority))) { 226 if (LOGD) Log.d(TAG, "Used cached roots for " + info.authority); 227 cacheHit = true; 228 } 229 } 230 } 231 232 // Cache miss, or loading everything 233 if (!cacheHit) { 234 mTaskRoots.putAll(info.authority, 235 loadRootsForAuthority(mContext.getContentResolver(), info.authority)); 236 } 237 } 238 } 239 240 /** 241 * Bring up requested provider and query for all active roots. 242 */ loadRootsForAuthority(ContentResolver resolver, String authority)243 private Collection<RootInfo> loadRootsForAuthority(ContentResolver resolver, String authority) { 244 if (LOGD) Log.d(TAG, "Loading roots for " + authority); 245 246 synchronized (mObservedAuthorities) { 247 if (mObservedAuthorities.add(authority)) { 248 // Watch for any future updates 249 final Uri rootsUri = DocumentsContract.buildRootsUri(authority); 250 mContext.getContentResolver().registerContentObserver(rootsUri, true, mObserver); 251 } 252 } 253 254 final List<RootInfo> roots = Lists.newArrayList(); 255 final Uri rootsUri = DocumentsContract.buildRootsUri(authority); 256 257 ContentProviderClient client = null; 258 Cursor cursor = null; 259 try { 260 client = DocumentsApplication.acquireUnstableProviderOrThrow(resolver, authority); 261 cursor = client.query(rootsUri, null, null, null, null); 262 while (cursor.moveToNext()) { 263 final RootInfo root = RootInfo.fromRootsCursor(authority, cursor); 264 roots.add(root); 265 } 266 } catch (Exception e) { 267 Log.w(TAG, "Failed to load some roots from " + authority + ": " + e); 268 } finally { 269 IoUtils.closeQuietly(cursor); 270 ContentProviderClient.releaseQuietly(client); 271 } 272 return roots; 273 } 274 275 /** 276 * Return the requested {@link RootInfo}, but only loading the roots for the 277 * requested authority. This is useful when we want to load fast without 278 * waiting for all the other roots to come back. 279 */ getRootOneshot(String authority, String rootId)280 public RootInfo getRootOneshot(String authority, String rootId) { 281 synchronized (mLock) { 282 RootInfo root = getRootLocked(authority, rootId); 283 if (root == null) { 284 mRoots.putAll( 285 authority, loadRootsForAuthority(mContext.getContentResolver(), authority)); 286 root = getRootLocked(authority, rootId); 287 } 288 return root; 289 } 290 } 291 getRootBlocking(String authority, String rootId)292 public RootInfo getRootBlocking(String authority, String rootId) { 293 waitForFirstLoad(); 294 loadStoppedAuthorities(); 295 synchronized (mLock) { 296 return getRootLocked(authority, rootId); 297 } 298 } 299 getRootLocked(String authority, String rootId)300 private RootInfo getRootLocked(String authority, String rootId) { 301 for (RootInfo root : mRoots.get(authority)) { 302 if (Objects.equals(root.rootId, rootId)) { 303 return root; 304 } 305 } 306 return null; 307 } 308 isIconUniqueBlocking(RootInfo root)309 public boolean isIconUniqueBlocking(RootInfo root) { 310 waitForFirstLoad(); 311 loadStoppedAuthorities(); 312 synchronized (mLock) { 313 final int rootIcon = root.derivedIcon != 0 ? root.derivedIcon : root.icon; 314 for (RootInfo test : mRoots.get(root.authority)) { 315 if (Objects.equals(test.rootId, root.rootId)) { 316 continue; 317 } 318 final int testIcon = test.derivedIcon != 0 ? test.derivedIcon : test.icon; 319 if (testIcon == rootIcon) { 320 return false; 321 } 322 } 323 return true; 324 } 325 } 326 getRecentsRoot()327 public RootInfo getRecentsRoot() { 328 return mRecentsRoot; 329 } 330 isRecentsRoot(RootInfo root)331 public boolean isRecentsRoot(RootInfo root) { 332 return mRecentsRoot == root; 333 } 334 getRootsBlocking()335 public Collection<RootInfo> getRootsBlocking() { 336 waitForFirstLoad(); 337 loadStoppedAuthorities(); 338 synchronized (mLock) { 339 return mRoots.values(); 340 } 341 } 342 getMatchingRootsBlocking(State state)343 public Collection<RootInfo> getMatchingRootsBlocking(State state) { 344 waitForFirstLoad(); 345 loadStoppedAuthorities(); 346 synchronized (mLock) { 347 return getMatchingRoots(mRoots.values(), state); 348 } 349 } 350 351 @VisibleForTesting getMatchingRoots(Collection<RootInfo> roots, State state)352 static List<RootInfo> getMatchingRoots(Collection<RootInfo> roots, State state) { 353 final List<RootInfo> matching = Lists.newArrayList(); 354 for (RootInfo root : roots) { 355 final boolean supportsCreate = (root.flags & Root.FLAG_SUPPORTS_CREATE) != 0; 356 final boolean supportsIsChild = (root.flags & Root.FLAG_SUPPORTS_IS_CHILD) != 0; 357 final boolean advanced = (root.flags & Root.FLAG_ADVANCED) != 0; 358 final boolean localOnly = (root.flags & Root.FLAG_LOCAL_ONLY) != 0; 359 final boolean empty = (root.flags & Root.FLAG_EMPTY) != 0; 360 361 // Exclude read-only devices when creating 362 if (state.action == State.ACTION_CREATE && !supportsCreate) continue; 363 // Exclude roots that don't support directory picking 364 if (state.action == State.ACTION_OPEN_TREE && !supportsIsChild) continue; 365 // Exclude advanced devices when not requested 366 if (!state.showAdvanced && advanced) continue; 367 // Exclude non-local devices when local only 368 if (state.localOnly && !localOnly) continue; 369 // Only show empty roots when creating 370 if (state.action != State.ACTION_CREATE && empty) continue; 371 372 // Only include roots that serve requested content 373 final boolean overlap = 374 MimePredicate.mimeMatches(root.derivedMimeTypes, state.acceptMimes) || 375 MimePredicate.mimeMatches(state.acceptMimes, root.derivedMimeTypes); 376 if (!overlap) { 377 continue; 378 } 379 380 matching.add(root); 381 } 382 return matching; 383 } 384 } 385