Home | History | Annotate | Download | only in drive
      1 // Copyright 2013 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/chromeos/drive/fileapi_worker.h"
      6 
      7 #include "base/files/file_path.h"
      8 #include "base/logging.h"
      9 #include "base/task_runner_util.h"
     10 #include "base/threading/sequenced_worker_pool.h"
     11 #include "chrome/browser/chromeos/drive/drive.pb.h"
     12 #include "chrome/browser/chromeos/drive/file_errors.h"
     13 #include "chrome/browser/chromeos/drive/file_system_interface.h"
     14 #include "chrome/browser/chromeos/drive/file_system_util.h"
     15 #include "chrome/browser/chromeos/drive/resource_entry_conversion.h"
     16 #include "content/public/browser/browser_thread.h"
     17 #include "webkit/common/fileapi/directory_entry.h"
     18 
     19 using content::BrowserThread;
     20 
     21 namespace drive {
     22 namespace fileapi_internal {
     23 namespace {
     24 
     25 // The summary of opening mode is:
     26 // - PLATFORM_FILE_OPEN: Open the existing file. Fail if not exists.
     27 // - PLATFORM_FILE_CREATE: Create the file if not exists. Fail if exists.
     28 // - PLATFORM_FILE_OPEN_ALWAYS: Open the existing file. Create a new file
     29 //     if not exists.
     30 // - PLATFORM_FILE_CREATE_ALWAYS: Create a new file if not exists. If exists
     31 //     open it with truncate.
     32 // - PLATFORM_FILE_OPEN_TRUNCATE: Open the existing file with truncate.
     33 //     Fail if not exists.
     34 OpenMode GetOpenMode(int file_flag) {
     35   if (file_flag & (base::PLATFORM_FILE_OPEN |
     36                    base::PLATFORM_FILE_OPEN_TRUNCATED))
     37     return OPEN_FILE;
     38 
     39   if (file_flag & base::PLATFORM_FILE_CREATE)
     40     return CREATE_FILE;
     41 
     42   DCHECK(file_flag & (base::PLATFORM_FILE_OPEN_ALWAYS |
     43                       base::PLATFORM_FILE_CREATE_ALWAYS));
     44   return OPEN_OR_CREATE_FILE;
     45 }
     46 
     47 // Runs |callback| with the PlatformFileError converted from |error|.
     48 void RunStatusCallbackByFileError(const StatusCallback& callback,
     49                                   FileError error) {
     50   callback.Run(FileErrorToPlatformError(error));
     51 }
     52 
     53 // Runs |callback| with arguments converted from |error| and |entry|.
     54 void RunGetFileInfoCallback(const GetFileInfoCallback& callback,
     55                             FileError error,
     56                             scoped_ptr<ResourceEntry> entry) {
     57   if (error != FILE_ERROR_OK) {
     58     callback.Run(FileErrorToPlatformError(error), base::PlatformFileInfo());
     59     return;
     60   }
     61 
     62   DCHECK(entry);
     63   base::PlatformFileInfo file_info;
     64   ConvertResourceEntryToPlatformFileInfo(*entry, &file_info);
     65   callback.Run(base::PLATFORM_FILE_OK, file_info);
     66 }
     67 
     68 // Runs |callback| with arguments converted from |error| and |resource_entries|.
     69 void RunReadDirectoryCallback(
     70     const ReadDirectoryCallback& callback,
     71     FileError error,
     72     scoped_ptr<ResourceEntryVector> resource_entries) {
     73   if (error != FILE_ERROR_OK) {
     74     callback.Run(FileErrorToPlatformError(error),
     75                  std::vector<fileapi::DirectoryEntry>(), false);
     76     return;
     77   }
     78 
     79   DCHECK(resource_entries);
     80 
     81   std::vector<fileapi::DirectoryEntry> entries;
     82   // Convert drive files to File API's directory entry.
     83   entries.reserve(resource_entries->size());
     84   for (size_t i = 0; i < resource_entries->size(); ++i) {
     85     const ResourceEntry& resource_entry = (*resource_entries)[i];
     86     fileapi::DirectoryEntry entry;
     87     entry.name = resource_entry.base_name();
     88 
     89     const PlatformFileInfoProto& file_info = resource_entry.file_info();
     90     entry.is_directory = file_info.is_directory();
     91     entry.size = file_info.size();
     92     entry.last_modified_time =
     93         base::Time::FromInternalValue(file_info.last_modified());
     94     entries.push_back(entry);
     95   }
     96 
     97   callback.Run(base::PLATFORM_FILE_OK, entries, false);
     98 }
     99 
    100 // Runs |callback| with arguments based on |error|, |local_path| and |entry|.
    101 void RunCreateSnapshotFileCallback(const CreateSnapshotFileCallback& callback,
    102                                    FileError error,
    103                                    const base::FilePath& local_path,
    104                                    scoped_ptr<ResourceEntry> entry) {
    105   if (error != FILE_ERROR_OK) {
    106     callback.Run(
    107         FileErrorToPlatformError(error),
    108         base::PlatformFileInfo(), base::FilePath(),
    109         webkit_blob::ScopedFile::ScopeOutPolicy());
    110     return;
    111   }
    112 
    113   DCHECK(entry);
    114 
    115   // When reading file, last modified time specified in file info will be
    116   // compared to the last modified time of the local version of the drive file.
    117   // Since those two values don't generally match (last modification time on the
    118   // drive server vs. last modification time of the local, downloaded file), so
    119   // we have to opt out from this check. We do this by unsetting last_modified
    120   // value in the file info passed to the CreateSnapshot caller.
    121   base::PlatformFileInfo file_info;
    122   ConvertResourceEntryToPlatformFileInfo(*entry, &file_info);
    123   file_info.last_modified = base::Time();
    124 
    125   // If the file is a hosted document, a temporary JSON file is created to
    126   // represent the document. The JSON file is not cached and its lifetime
    127   // is managed by ShareableFileReference.
    128   webkit_blob::ScopedFile::ScopeOutPolicy scope_out_policy =
    129       entry->file_specific_info().is_hosted_document() ?
    130       webkit_blob::ScopedFile::DELETE_ON_SCOPE_OUT :
    131       webkit_blob::ScopedFile::DONT_DELETE_ON_SCOPE_OUT;
    132 
    133   callback.Run(base::PLATFORM_FILE_OK, file_info, local_path, scope_out_policy);
    134 }
    135 
    136 // Runs |callback| with arguments converted from |error| and |local_path|.
    137 void RunCreateWritableSnapshotFileCallback(
    138     const CreateWritableSnapshotFileCallback& callback,
    139     FileError error,
    140     const base::FilePath& local_path,
    141     const base::Closure& close_callback) {
    142   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    143   callback.Run(FileErrorToPlatformError(error), local_path, close_callback);
    144 }
    145 
    146 // Runs |callback| with |error| and |platform_file|.
    147 void RunOpenFileCallback(const OpenFileCallback& callback,
    148                          const base::Closure& close_callback,
    149                          base::PlatformFileError* error,
    150                          base::PlatformFile platform_file) {
    151   callback.Run(*error, platform_file, close_callback);
    152 }
    153 
    154 // Part of OpenFile(). Called after FileSystem::OpenFile().
    155 void OpenFileAfterFileSystemOpenFile(int file_flags,
    156                                      const OpenFileCallback& callback,
    157                                      FileError error,
    158                                      const base::FilePath& local_path,
    159                                      const base::Closure& close_callback) {
    160   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    161 
    162   if (error != FILE_ERROR_OK) {
    163     callback.Run(FileErrorToPlatformError(error),
    164                  base::kInvalidPlatformFileValue,
    165                  base::Closure());
    166     return;
    167   }
    168 
    169   // Here, the file should be at |local_path|, but there may be timing issue.
    170   // Because the file is managed by Drive file system, so, in order to avoid
    171   // unexpected file creation, CREATE, OPEN_ALWAYS and CREATE_ALWAYS are
    172   // translated into OPEN or OPEN_TRUNCATED, here. Keep OPEN and OPEN_TRUNCATED
    173   // as is.
    174   if (file_flags & (base::PLATFORM_FILE_CREATE |
    175                     base::PLATFORM_FILE_OPEN_ALWAYS)) {
    176     file_flags &= ~(base::PLATFORM_FILE_CREATE |
    177                     base::PLATFORM_FILE_OPEN_ALWAYS);
    178     file_flags |= base::PLATFORM_FILE_OPEN;
    179   } else if (file_flags & base::PLATFORM_FILE_CREATE_ALWAYS) {
    180     file_flags &= ~base::PLATFORM_FILE_CREATE_ALWAYS;
    181     file_flags |= base::PLATFORM_FILE_OPEN_TRUNCATED;
    182   }
    183 
    184   // Cache file prepared for modification is available. Open it locally.
    185   base::PlatformFileError* result =
    186       new base::PlatformFileError(base::PLATFORM_FILE_ERROR_FAILED);
    187   bool posted = base::PostTaskAndReplyWithResult(
    188       BrowserThread::GetBlockingPool(), FROM_HERE,
    189       base::Bind(&base::CreatePlatformFile,
    190                  local_path, file_flags, static_cast<bool*>(NULL), result),
    191       base::Bind(&RunOpenFileCallback,
    192                  callback, close_callback, base::Owned(result)));
    193   DCHECK(posted);
    194 }
    195 
    196 }  // namespace
    197 
    198 void RunFileSystemCallback(
    199     const FileSystemGetter& file_system_getter,
    200     const base::Callback<void(FileSystemInterface*)>& callback,
    201     const base::Closure& on_error_callback) {
    202   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    203   FileSystemInterface* file_system = file_system_getter.Run();
    204 
    205   if (!file_system) {
    206     if (!on_error_callback.is_null())
    207       on_error_callback.Run();
    208     return;
    209   }
    210 
    211   callback.Run(file_system);
    212 }
    213 
    214 void GetFileInfo(const base::FilePath& file_path,
    215                  const GetFileInfoCallback& callback,
    216                  FileSystemInterface* file_system) {
    217   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    218   file_system->GetResourceEntry(
    219       file_path,
    220       base::Bind(&RunGetFileInfoCallback, callback));
    221 }
    222 
    223 void Copy(const base::FilePath& src_file_path,
    224           const base::FilePath& dest_file_path,
    225           bool preserve_last_modified,
    226           const StatusCallback& callback,
    227           FileSystemInterface* file_system) {
    228   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    229   file_system->Copy(src_file_path, dest_file_path, preserve_last_modified,
    230                     base::Bind(&RunStatusCallbackByFileError, callback));
    231 }
    232 
    233 void Move(const base::FilePath& src_file_path,
    234           const base::FilePath& dest_file_path,
    235           bool preserve_last_modified,
    236           const StatusCallback& callback,
    237           FileSystemInterface* file_system) {
    238   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    239   file_system->Move(src_file_path, dest_file_path, preserve_last_modified,
    240                     base::Bind(&RunStatusCallbackByFileError, callback));
    241 }
    242 
    243 void CopyInForeignFile(const base::FilePath& src_foreign_file_path,
    244                        const base::FilePath& dest_file_path,
    245                        const StatusCallback& callback,
    246                        FileSystemInterface* file_system) {
    247   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    248   file_system->TransferFileFromLocalToRemote(
    249       src_foreign_file_path, dest_file_path,
    250       base::Bind(&RunStatusCallbackByFileError, callback));
    251 }
    252 
    253 void ReadDirectory(const base::FilePath& file_path,
    254                    const ReadDirectoryCallback& callback,
    255                    FileSystemInterface* file_system) {
    256   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    257   file_system->ReadDirectory(file_path,
    258                              base::Bind(&RunReadDirectoryCallback, callback));
    259 }
    260 
    261 void Remove(const base::FilePath& file_path,
    262             bool is_recursive,
    263             const StatusCallback& callback,
    264             FileSystemInterface* file_system) {
    265   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    266   file_system->Remove(file_path, is_recursive,
    267                       base::Bind(&RunStatusCallbackByFileError, callback));
    268 }
    269 
    270 void CreateDirectory(const base::FilePath& file_path,
    271                      bool is_exclusive,
    272                      bool is_recursive,
    273                      const StatusCallback& callback,
    274                      FileSystemInterface* file_system) {
    275   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    276   file_system->CreateDirectory(
    277       file_path, is_exclusive, is_recursive,
    278       base::Bind(&RunStatusCallbackByFileError, callback));
    279 }
    280 
    281 void CreateFile(const base::FilePath& file_path,
    282                 bool is_exclusive,
    283                 const StatusCallback& callback,
    284                 FileSystemInterface* file_system) {
    285   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    286   file_system->CreateFile(file_path, is_exclusive,
    287                           std::string(),  // no mime type; guess from file_path
    288                           base::Bind(&RunStatusCallbackByFileError, callback));
    289 }
    290 
    291 void Truncate(const base::FilePath& file_path,
    292               int64 length,
    293               const StatusCallback& callback,
    294               FileSystemInterface* file_system) {
    295   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    296   file_system->TruncateFile(
    297       file_path, length,
    298       base::Bind(&RunStatusCallbackByFileError, callback));
    299 }
    300 
    301 void CreateSnapshotFile(const base::FilePath& file_path,
    302                         const CreateSnapshotFileCallback& callback,
    303                         FileSystemInterface* file_system) {
    304   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    305   file_system->GetFile(file_path,
    306                        base::Bind(&RunCreateSnapshotFileCallback, callback));
    307 }
    308 
    309 void CreateWritableSnapshotFile(
    310     const base::FilePath& file_path,
    311     const CreateWritableSnapshotFileCallback& callback,
    312     FileSystemInterface* file_system) {
    313   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    314   file_system->OpenFile(
    315       file_path,
    316       OPEN_FILE,
    317       std::string(),  // no mime type; we never create a new file here.
    318       base::Bind(&RunCreateWritableSnapshotFileCallback, callback));
    319 }
    320 
    321 void OpenFile(const base::FilePath& file_path,
    322               int file_flags,
    323               const OpenFileCallback& callback,
    324               FileSystemInterface* file_system) {
    325   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    326 
    327   // Returns an error if any unsupported flag is found.
    328   if (file_flags & ~(base::PLATFORM_FILE_OPEN |
    329                      base::PLATFORM_FILE_CREATE |
    330                      base::PLATFORM_FILE_OPEN_ALWAYS |
    331                      base::PLATFORM_FILE_CREATE_ALWAYS |
    332                      base::PLATFORM_FILE_OPEN_TRUNCATED |
    333                      base::PLATFORM_FILE_READ |
    334                      base::PLATFORM_FILE_WRITE |
    335                      base::PLATFORM_FILE_WRITE_ATTRIBUTES |
    336                      base::PLATFORM_FILE_APPEND)) {
    337     base::MessageLoopProxy::current()->PostTask(
    338         FROM_HERE,
    339         base::Bind(callback,
    340                    base::PLATFORM_FILE_ERROR_FAILED,
    341                    base::kInvalidPlatformFileValue,
    342                    base::Closure()));
    343     return;
    344   }
    345 
    346   file_system->OpenFile(
    347       file_path, GetOpenMode(file_flags),
    348       std::string(),  // no mime type; guess from file_path
    349       base::Bind(&OpenFileAfterFileSystemOpenFile, file_flags, callback));
    350 }
    351 
    352 void TouchFile(const base::FilePath& file_path,
    353                const base::Time& last_access_time,
    354                const base::Time& last_modified_time,
    355                const StatusCallback& callback,
    356                FileSystemInterface* file_system) {
    357   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    358   file_system->TouchFile(file_path, last_access_time, last_modified_time,
    359                          base::Bind(&RunStatusCallbackByFileError, callback));
    360 
    361 }
    362 
    363 }  // namespace fileapi_internal
    364 }  // namespace drive
    365