Home | History | Annotate | Download | only in onetimeinitializer
      1 /*
      2  * Copyright (C) 2012 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.gallery3d.onetimeinitializer;
     18 
     19 import android.content.Context;
     20 import android.content.SharedPreferences;
     21 import android.os.Build;
     22 import android.os.Environment;
     23 import android.preference.PreferenceManager;
     24 import android.util.Log;
     25 
     26 import com.android.gallery3d.app.GalleryApp;
     27 import com.android.gallery3d.common.ApiHelper;
     28 import com.android.gallery3d.data.DataManager;
     29 import com.android.gallery3d.data.LocalAlbum;
     30 import com.android.gallery3d.data.MediaSet;
     31 import com.android.gallery3d.data.Path;
     32 import com.android.gallery3d.gadget.WidgetDatabaseHelper;
     33 import com.android.gallery3d.gadget.WidgetDatabaseHelper.Entry;
     34 import com.android.gallery3d.util.GalleryUtils;
     35 
     36 import java.io.File;
     37 import java.util.HashMap;
     38 import java.util.List;
     39 
     40 /**
     41  * This one-timer migrates local-album gallery app widgets from old paths from prior releases
     42  * to updated paths in the current build version. This migration is needed because of
     43  * bucket ID (i.e., directory hash) change in JB and JB MR1 (The external storage path has changed
     44  * from /mnt/sdcard in pre-JB releases, to /storage/sdcard0 in JB, then again
     45  * to /external/storage/sdcard/0 in JB MR1).
     46  */
     47 public class GalleryWidgetMigrator {
     48     private static final String TAG = "GalleryWidgetMigrator";
     49     private static final String PRE_JB_EXT_PATH = "/mnt/sdcard";
     50     private static final String JB_EXT_PATH = "/storage/sdcard0";
     51     private static final String NEW_EXT_PATH =
     52             Environment.getExternalStorageDirectory().getAbsolutePath();
     53     private static final int RELATIVE_PATH_START = NEW_EXT_PATH.length();
     54     private static final String KEY_EXT_PATH = "external_storage_path";
     55 
     56     /**
     57      * Migrates local-album gallery widgets from prior releases to current release
     58      * due to bucket ID (i.e., directory hash) change.
     59      */
     60     public static void migrateGalleryWidgets(Context context) {
     61         SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
     62         // Migration is only needed when external storage path has changed
     63         String extPath = prefs.getString(KEY_EXT_PATH, null);
     64         boolean isDone = NEW_EXT_PATH.equals(extPath);
     65         if (isDone) return;
     66 
     67         try {
     68             migrateGalleryWidgetsInternal(context);
     69             prefs.edit().putString(KEY_EXT_PATH, NEW_EXT_PATH).commit();
     70         } catch (Throwable t) {
     71             // exception may be thrown if external storage is not available(?)
     72             Log.w(TAG, "migrateGalleryWidgets", t);
     73         }
     74     }
     75 
     76     private static void migrateGalleryWidgetsInternal(Context context) {
     77         GalleryApp galleryApp = (GalleryApp) context.getApplicationContext();
     78         DataManager manager = galleryApp.getDataManager();
     79         WidgetDatabaseHelper dbHelper = new WidgetDatabaseHelper(context);
     80 
     81         // only need to migrate local-album entries of type TYPE_ALBUM
     82         List<Entry> entries = dbHelper.getEntries(WidgetDatabaseHelper.TYPE_ALBUM);
     83         if (entries == null) return;
     84 
     85         // Check each entry's relativePath. If exists, update bucket id using relative
     86         // path combined with external storage path. Otherwise, iterate through old external
     87         // storage paths to find the relative path that matches the old bucket id, and then update
     88         // bucket id and relative path
     89         HashMap<Integer, Entry> localEntries = new HashMap<Integer, Entry>(entries.size());
     90         for (Entry entry : entries) {
     91             Path path = Path.fromString(entry.albumPath);
     92             MediaSet mediaSet = (MediaSet) manager.getMediaObject(path);
     93             if (mediaSet instanceof LocalAlbum) {
     94                 if (entry.relativePath != null && entry.relativePath.length() > 0) {
     95                     // update entry using relative path + external storage path
     96                     updateEntryUsingRelativePath(entry, dbHelper);
     97                 } else {
     98                     int bucketId = Integer.parseInt(path.getSuffix());
     99                     localEntries.put(bucketId, entry);
    100                 }
    101             }
    102         }
    103         if (!localEntries.isEmpty()) migrateLocalEntries(context, localEntries, dbHelper);
    104     }
    105 
    106     private static void migrateLocalEntries(Context context,
    107             HashMap<Integer, Entry> entries, WidgetDatabaseHelper dbHelper) {
    108         SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
    109         String oldExtPath = prefs.getString(KEY_EXT_PATH, null);
    110         if (oldExtPath != null) {
    111             migrateLocalEntries(entries, dbHelper, oldExtPath);
    112             return;
    113         }
    114         // If old external storage path is unknown, it could be either Pre-JB or JB version
    115         // we need to try both.
    116         migrateLocalEntries(entries, dbHelper, PRE_JB_EXT_PATH);
    117         if (!entries.isEmpty() &&
    118                 Build.VERSION.SDK_INT > ApiHelper.VERSION_CODES.JELLY_BEAN) {
    119             migrateLocalEntries(entries, dbHelper, JB_EXT_PATH);
    120         }
    121     }
    122 
    123     private static void migrateLocalEntries(HashMap<Integer, Entry> entries,
    124              WidgetDatabaseHelper dbHelper, String oldExtPath) {
    125         File root = Environment.getExternalStorageDirectory();
    126         // check the DCIM directory first; this should take care of 99% use cases
    127         updatePath(new File(root, "DCIM"), entries, dbHelper, oldExtPath);
    128         // check other directories if DCIM doesn't cut it
    129         if (!entries.isEmpty()) updatePath(root, entries, dbHelper, oldExtPath);
    130     }
    131     private static void updatePath(File root, HashMap<Integer, Entry> entries,
    132             WidgetDatabaseHelper dbHelper, String oldExtStorage) {
    133         File[] files = root.listFiles();
    134         if (files != null) {
    135             for (File file : files) {
    136                 if (file.isDirectory() && !entries.isEmpty()) {
    137                     String path = file.getAbsolutePath();
    138                     String oldPath = oldExtStorage + path.substring(RELATIVE_PATH_START);
    139                     int oldBucketId = GalleryUtils.getBucketId(oldPath);
    140                     Entry entry = entries.remove(oldBucketId);
    141                     if (entry != null) {
    142                         int newBucketId = GalleryUtils.getBucketId(path);
    143                         String newAlbumPath = Path.fromString(entry.albumPath)
    144                                 .getParent()
    145                                 .getChild(newBucketId)
    146                                 .toString();
    147                         Log.d(TAG, "migrate from " + entry.albumPath + " to " + newAlbumPath);
    148                         entry.albumPath = newAlbumPath;
    149                         // update entry's relative path
    150                         entry.relativePath = path.substring(RELATIVE_PATH_START);
    151                         dbHelper.updateEntry(entry);
    152                     }
    153                     updatePath(file, entries, dbHelper, oldExtStorage); // recursion
    154                 }
    155             }
    156         }
    157     }
    158 
    159     private static void updateEntryUsingRelativePath(Entry entry, WidgetDatabaseHelper dbHelper) {
    160         String newPath = NEW_EXT_PATH + entry.relativePath;
    161         int newBucketId = GalleryUtils.getBucketId(newPath);
    162         String newAlbumPath = Path.fromString(entry.albumPath)
    163                 .getParent()
    164                 .getChild(newBucketId)
    165                 .toString();
    166         entry.albumPath = newAlbumPath;
    167         dbHelper.updateEntry(entry);
    168     }
    169 }
    170