Home | History | Annotate | Download | only in deletionhelper
      1 /*
      2  * Copyright (C) 2016 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.storagemanager.deletionhelper;
     18 
     19 import android.content.Context;
     20 import android.graphics.Bitmap;
     21 import android.media.ThumbnailUtils;
     22 import android.os.SystemProperties;
     23 import android.provider.MediaStore;
     24 import android.support.annotation.VisibleForTesting;
     25 import android.text.format.DateUtils;
     26 
     27 import com.android.storagemanager.utils.AsyncLoader;
     28 import com.android.storagemanager.utils.IconProvider;
     29 
     30 import java.io.File;
     31 import java.util.ArrayList;
     32 import java.util.HashMap;
     33 
     34 /**
     35  * FetchDownloadsLoader is an asynchronous task which returns files in the Downloads
     36  * directory which have not been modified in longer than 90 days.
     37  */
     38 public class FetchDownloadsLoader extends AsyncLoader<FetchDownloadsLoader.DownloadsResult> {
     39     private static final String DEBUG_FILE_AGE_OVERRIDE = "debug.asm.file_age_limit";
     40     private static final int MINIMUM_AGE_DAYS = 0;
     41     private File mDirectory;
     42 
     43     /**
     44      * Sets up a FetchDownloadsLoader in any directory.
     45      *
     46      * @param directory The directory to look into.
     47      */
     48     public FetchDownloadsLoader(Context context, File directory) {
     49         super(context);
     50         mDirectory = directory;
     51     }
     52 
     53     @Override
     54     protected void onDiscardResult(DownloadsResult result) {
     55     }
     56 
     57     @Override
     58     public DownloadsResult loadInBackground() {
     59         return collectFiles(mDirectory);
     60     }
     61 
     62     @VisibleForTesting
     63     static DownloadsResult collectFiles(File dir) {
     64         return collectFiles(dir, new DownloadsResult());
     65     }
     66 
     67     private static DownloadsResult collectFiles(File dir, DownloadsResult result) {
     68         int minimumAgeDays = SystemProperties.getInt(DEBUG_FILE_AGE_OVERRIDE, MINIMUM_AGE_DAYS);
     69         final long last_modified_threshold = System.currentTimeMillis() -
     70                 minimumAgeDays * DateUtils.DAY_IN_MILLIS;
     71         File downloadFiles[] = dir.listFiles();
     72         if (downloadFiles != null && downloadFiles.length > 0) {
     73             for (File currentFile : downloadFiles) {
     74                 if (currentFile.isDirectory()) {
     75                     collectFiles(currentFile, result);
     76                 } else {
     77                     // Skip files that have been modified too recently.
     78                     if (last_modified_threshold < currentFile.lastModified()) {
     79                         continue;
     80                     }
     81 
     82                     if (currentFile.lastModified() < result.youngestLastModified) {
     83                         result.youngestLastModified = currentFile.lastModified();
     84                     }
     85                     result.files.add(currentFile);
     86                     result.totalSize += currentFile.length();
     87 
     88                     if (IconProvider.isImageType(currentFile)) {
     89                         Bitmap thumbnail =
     90                                 ThumbnailUtils.createImageThumbnail(
     91                                         currentFile.getAbsolutePath(),
     92                                         MediaStore.Images.Thumbnails.MINI_KIND);
     93                         result.thumbnails.put(currentFile, thumbnail);
     94                     }
     95                 }
     96             }
     97         }
     98 
     99         return result;
    100     }
    101 
    102     /**
    103      * The DownloadsResult is the result of a {@link FetchDownloadsLoader} with the files
    104      * and the amount of space they use.
    105      */
    106     public static class DownloadsResult {
    107         public long totalSize;
    108         public long youngestLastModified;
    109         public ArrayList<File> files;
    110         public HashMap<File, Bitmap> thumbnails;
    111 
    112         public DownloadsResult() {
    113             this(0, Long.MAX_VALUE, new ArrayList<File>(), new HashMap<>());
    114         }
    115 
    116         public DownloadsResult(
    117                 long totalSize,
    118                 long youngestLastModified,
    119                 ArrayList<File> files,
    120                 HashMap<File, Bitmap> thumbnails) {
    121             this.totalSize = totalSize;
    122             this.youngestLastModified = youngestLastModified;
    123             this.files = files;
    124             this.thumbnails = thumbnails;
    125         }
    126     }
    127 }