Home | History | Annotate | Download | only in storage_monitor
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "chrome/browser/storage_monitor/media_storage_util.h"
      6 
      7 #include <vector>
      8 
      9 #include "base/callback.h"
     10 #include "base/file_util.h"
     11 #include "base/logging.h"
     12 #include "base/metrics/histogram.h"
     13 #include "base/strings/string_util.h"
     14 #include "base/strings/utf_string_conversions.h"
     15 #include "chrome/browser/storage_monitor/removable_device_constants.h"
     16 #include "chrome/browser/storage_monitor/storage_monitor.h"
     17 #include "content/public/browser/browser_thread.h"
     18 
     19 using content::BrowserThread;
     20 
     21 namespace {
     22 
     23 // MediaDeviceNotification.DeviceInfo histogram values.
     24 enum DeviceInfoHistogramBuckets {
     25   MASS_STORAGE_DEVICE_NAME_AND_UUID_AVAILABLE,
     26   MASS_STORAGE_DEVICE_UUID_MISSING,
     27   MASS_STORAGE_DEVICE_NAME_MISSING,
     28   MASS_STORAGE_DEVICE_NAME_AND_UUID_MISSING,
     29   MTP_STORAGE_DEVICE_NAME_AND_UUID_AVAILABLE,
     30   MTP_STORAGE_DEVICE_UUID_MISSING,
     31   MTP_STORAGE_DEVICE_NAME_MISSING,
     32   MTP_STORAGE_DEVICE_NAME_AND_UUID_MISSING,
     33   DEVICE_INFO_BUCKET_BOUNDARY
     34 };
     35 
     36 #if !defined(OS_WIN)
     37 const char kRootPath[] = "/";
     38 #endif
     39 
     40 typedef std::vector<StorageInfo> StorageInfoList;
     41 
     42 base::FilePath::StringType FindRemovableStorageLocationById(
     43     const std::string& device_id) {
     44   StorageInfoList devices =
     45       StorageMonitor::GetInstance()->GetAllAvailableStorages();
     46   for (StorageInfoList::const_iterator it = devices.begin();
     47        it != devices.end(); ++it) {
     48     if (it->device_id() == device_id
     49         && StorageInfo::IsRemovableDevice(device_id))
     50       return it->location();
     51   }
     52   return base::FilePath::StringType();
     53 }
     54 
     55 void FilterAttachedDevicesOnFileThread(MediaStorageUtil::DeviceIdSet* devices) {
     56   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
     57   MediaStorageUtil::DeviceIdSet missing_devices;
     58 
     59   for (MediaStorageUtil::DeviceIdSet::const_iterator it = devices->begin();
     60        it != devices->end();
     61        ++it) {
     62     StorageInfo::Type type;
     63     std::string unique_id;
     64     if (!StorageInfo::CrackDeviceId(*it, &type, &unique_id)) {
     65       missing_devices.insert(*it);
     66       continue;
     67     }
     68 
     69     if (type == StorageInfo::FIXED_MASS_STORAGE ||
     70         type == StorageInfo::ITUNES ||
     71         type == StorageInfo::IPHOTO ||
     72         type == StorageInfo::PICASA) {
     73       if (!base::PathExists(base::FilePath::FromUTF8Unsafe(unique_id)))
     74         missing_devices.insert(*it);
     75       continue;
     76     }
     77 
     78     if (!MediaStorageUtil::IsRemovableStorageAttached(*it))
     79       missing_devices.insert(*it);
     80   }
     81 
     82   for (MediaStorageUtil::DeviceIdSet::const_iterator it =
     83            missing_devices.begin();
     84        it != missing_devices.end();
     85        ++it) {
     86     devices->erase(*it);
     87   }
     88 }
     89 
     90 }  // namespace
     91 
     92 // static
     93 bool MediaStorageUtil::HasDcim(const base::FilePath& mount_point) {
     94   DCHECK(!content::BrowserThread::CurrentlyOn(content::BrowserThread::UI));
     95 
     96   base::FilePath::StringType dcim_dir(kDCIMDirectoryName);
     97   if (!base::DirectoryExists(mount_point.Append(dcim_dir))) {
     98     // Check for lowercase 'dcim' as well.
     99     base::FilePath dcim_path_lower(
    100         mount_point.Append(StringToLowerASCII(dcim_dir)));
    101     if (!base::DirectoryExists(dcim_path_lower))
    102       return false;
    103   }
    104   return true;
    105 }
    106 
    107 // static
    108 bool MediaStorageUtil::CanCreateFileSystem(const std::string& device_id,
    109                                            const base::FilePath& path) {
    110   StorageInfo::Type type;
    111   if (!StorageInfo::CrackDeviceId(device_id, &type, NULL))
    112     return false;
    113 
    114   if (type == StorageInfo::MAC_IMAGE_CAPTURE)
    115     return true;
    116 
    117   return path.IsAbsolute() && !path.ReferencesParent();
    118 }
    119 
    120 // static
    121 void MediaStorageUtil::FilterAttachedDevices(DeviceIdSet* devices,
    122                                              const base::Closure& done) {
    123   if (BrowserThread::CurrentlyOn(BrowserThread::FILE)) {
    124     FilterAttachedDevicesOnFileThread(devices);
    125     done.Run();
    126     return;
    127   }
    128   BrowserThread::PostTaskAndReply(BrowserThread::FILE,
    129                                   FROM_HERE,
    130                                   base::Bind(&FilterAttachedDevicesOnFileThread,
    131                                              devices),
    132                                   done);
    133 }
    134 
    135 // TODO(kmadhusu) Write unit tests for GetDeviceInfoFromPath().
    136 bool MediaStorageUtil::GetDeviceInfoFromPath(const base::FilePath& path,
    137                                              StorageInfo* device_info,
    138                                              base::FilePath* relative_path) {
    139   DCHECK(device_info);
    140   DCHECK(relative_path);
    141 
    142   if (!path.IsAbsolute())
    143     return false;
    144 
    145   StorageInfo info;
    146   StorageMonitor* monitor = StorageMonitor::GetInstance();
    147   bool found_device = monitor->GetStorageInfoForPath(path, &info);
    148 
    149   if (found_device && StorageInfo::IsRemovableDevice(info.device_id())) {
    150     base::FilePath sub_folder_path;
    151     base::FilePath device_path(info.location());
    152     if (path != device_path) {
    153       bool success = device_path.AppendRelativePath(path, &sub_folder_path);
    154       DCHECK(success);
    155     }
    156 
    157     *device_info = info;
    158     *relative_path = sub_folder_path;
    159     return true;
    160   }
    161 
    162   // On Posix systems, there's one root so any absolute path could be valid.
    163   // TODO(gbillock): Delete this stanza? Posix systems should have the root
    164   // volume information. If not, we should move the below into the
    165   // right GetStorageInfoForPath implementations.
    166 #if !defined(OS_POSIX)
    167   if (!found_device)
    168     return false;
    169 #endif
    170 
    171   // Handle non-removable devices. Note: this is just overwriting
    172   // good values from StorageMonitor.
    173   // TODO(gbillock): Make sure return values from that class are definitive,
    174   // and don't do this here.
    175   info.set_device_id(
    176       StorageInfo::MakeDeviceId(StorageInfo::FIXED_MASS_STORAGE,
    177                                 path.AsUTF8Unsafe()));
    178   *device_info = info;
    179   *relative_path = base::FilePath();
    180   return true;
    181 }
    182 
    183 // static
    184 base::FilePath MediaStorageUtil::FindDevicePathById(
    185     const std::string& device_id) {
    186   StorageInfo::Type type;
    187   std::string unique_id;
    188   if (!StorageInfo::CrackDeviceId(device_id, &type, &unique_id))
    189     return base::FilePath();
    190 
    191   if (type == StorageInfo::FIXED_MASS_STORAGE ||
    192       type == StorageInfo::ITUNES ||
    193       type == StorageInfo::IPHOTO ||
    194       type == StorageInfo::PICASA) {
    195     // For this type, the unique_id is the path.
    196     return base::FilePath::FromUTF8Unsafe(unique_id);
    197   }
    198 
    199   // For ImageCapture, the synthetic filesystem will be rooted at a fake
    200   // top-level directory which is the device_id.
    201   if (type == StorageInfo::MAC_IMAGE_CAPTURE) {
    202 #if !defined(OS_WIN)
    203     return base::FilePath(kRootPath + device_id);
    204 #endif
    205   }
    206 
    207   DCHECK(type == StorageInfo::MTP_OR_PTP ||
    208          type == StorageInfo::REMOVABLE_MASS_STORAGE_WITH_DCIM ||
    209          type == StorageInfo::REMOVABLE_MASS_STORAGE_NO_DCIM);
    210   return base::FilePath(FindRemovableStorageLocationById(device_id));
    211 }
    212 
    213 // static
    214 void MediaStorageUtil::RecordDeviceInfoHistogram(
    215     bool mass_storage,
    216     const std::string& device_uuid,
    217     const base::string16& device_label) {
    218   unsigned int event_number = 0;
    219   if (!mass_storage)
    220     event_number = 4;
    221 
    222   if (device_label.empty())
    223     event_number += 2;
    224 
    225   if (device_uuid.empty())
    226     event_number += 1;
    227   enum DeviceInfoHistogramBuckets event =
    228       static_cast<enum DeviceInfoHistogramBuckets>(event_number);
    229   if (event >= DEVICE_INFO_BUCKET_BOUNDARY) {
    230     NOTREACHED();
    231     return;
    232   }
    233   UMA_HISTOGRAM_ENUMERATION("MediaDeviceNotifications.DeviceInfo", event,
    234                             DEVICE_INFO_BUCKET_BOUNDARY);
    235 }
    236 
    237 bool MediaStorageUtil::IsRemovableStorageAttached(const std::string& id) {
    238   StorageInfoList devices =
    239       StorageMonitor::GetInstance()->GetAllAvailableStorages();
    240   for (StorageInfoList::const_iterator it = devices.begin();
    241        it != devices.end(); ++it) {
    242     if (StorageInfo::IsRemovableDevice(id) && it->device_id() == id)
    243       return true;
    244   }
    245   return false;
    246 }
    247