create_directory_operation.cc revision cedac228d2dd51db4b79ea1e72c7f249408ee061
1// Copyright (c) 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/file_system/create_directory_operation.h"
6
7#include "chrome/browser/chromeos/drive/drive.pb.h"
8#include "chrome/browser/chromeos/drive/file_system/operation_observer.h"
9#include "chrome/browser/chromeos/drive/file_system_util.h"
10#include "chrome/browser/chromeos/drive/resource_metadata.h"
11#include "content/public/browser/browser_thread.h"
12
13using content::BrowserThread;
14
15namespace drive {
16namespace file_system {
17
18namespace {
19
20FileError CreateDirectoryRecursively(
21    internal::ResourceMetadata* metadata,
22    const std::string& parent_local_id,
23    const base::FilePath& relative_file_path,
24    std::set<std::string>* updated_local_ids,
25    std::set<base::FilePath>* changed_directories) {
26  // Split the first component and remaining ones of |relative_file_path|.
27  std::vector<base::FilePath::StringType> components;
28  relative_file_path.GetComponents(&components);
29  DCHECK(!components.empty());
30  base::FilePath title(components[0]);
31  base::FilePath remaining_path;
32  title.AppendRelativePath(relative_file_path, &remaining_path);
33
34  ResourceEntry entry;
35  const base::Time now = base::Time::Now();
36  entry.set_title(title.AsUTF8Unsafe());
37  entry.mutable_file_info()->set_is_directory(true);
38  entry.mutable_file_info()->set_last_modified(now.ToInternalValue());
39  entry.mutable_file_info()->set_last_accessed(now.ToInternalValue());
40  entry.set_parent_local_id(parent_local_id);
41  entry.set_metadata_edit_state(ResourceEntry::DIRTY);
42  entry.set_modification_date(base::Time::Now().ToInternalValue());
43
44  std::string local_id;
45  FileError error = metadata->AddEntry(entry, &local_id);
46  if (error != FILE_ERROR_OK)
47    return error;
48
49  base::FilePath path;
50  error = metadata->GetFilePath(local_id, &path);
51  if (error != FILE_ERROR_OK)
52    return error;
53
54  updated_local_ids->insert(local_id);
55  changed_directories->insert(path.DirName());
56
57  if (remaining_path.empty())  // All directories are created successfully.
58    return FILE_ERROR_OK;
59
60  // Create descendant directories.
61  return CreateDirectoryRecursively(metadata, local_id, remaining_path,
62                                    updated_local_ids, changed_directories);
63}
64
65FileError UpdateLocalState(internal::ResourceMetadata* metadata,
66                           const base::FilePath& directory_path,
67                           bool is_exclusive,
68                           bool is_recursive,
69                           std::set<std::string>* updated_local_ids,
70                           std::set<base::FilePath>* changed_directories) {
71  // Get the existing deepest entry.
72  std::vector<base::FilePath::StringType> components;
73  directory_path.GetComponents(&components);
74
75  if (components.empty() || components[0] != util::kDriveGrandRootDirName)
76    return FILE_ERROR_NOT_FOUND;
77
78  base::FilePath existing_deepest_path(components[0]);
79  std::string local_id = util::kDriveGrandRootLocalId;
80  for (size_t i = 1; i < components.size(); ++i) {
81    std::string child_local_id;
82    FileError error =
83        metadata->GetChildId(local_id, components[i], &child_local_id);
84    if (error == FILE_ERROR_NOT_FOUND)
85      break;
86    if (error != FILE_ERROR_OK)
87      return error;
88    existing_deepest_path = existing_deepest_path.Append(components[i]);
89    local_id = child_local_id;
90  }
91
92  ResourceEntry entry;
93  FileError error = metadata->GetResourceEntryById(local_id, &entry);
94  if (error != FILE_ERROR_OK)
95    return error;
96
97  if (!entry.file_info().is_directory())
98    return FILE_ERROR_NOT_A_DIRECTORY;
99
100  if (directory_path == existing_deepest_path)
101    return is_exclusive ? FILE_ERROR_EXISTS : FILE_ERROR_OK;
102
103  // If it is not recursive creation, the found directory must be the direct
104  // parent of |directory_path| to ensure creating exact one directory.
105  if (!is_recursive && existing_deepest_path != directory_path.DirName())
106    return FILE_ERROR_NOT_FOUND;
107
108  // Create directories under the found directory.
109  base::FilePath remaining_path;
110  existing_deepest_path.AppendRelativePath(directory_path, &remaining_path);
111  return CreateDirectoryRecursively(metadata, entry.local_id(), remaining_path,
112                                    updated_local_ids, changed_directories);
113}
114
115}  // namespace
116
117CreateDirectoryOperation::CreateDirectoryOperation(
118    base::SequencedTaskRunner* blocking_task_runner,
119    OperationObserver* observer,
120    internal::ResourceMetadata* metadata)
121    : blocking_task_runner_(blocking_task_runner),
122      observer_(observer),
123      metadata_(metadata),
124      weak_ptr_factory_(this) {
125  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
126}
127
128CreateDirectoryOperation::~CreateDirectoryOperation() {
129  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
130}
131
132void CreateDirectoryOperation::CreateDirectory(
133    const base::FilePath& directory_path,
134    bool is_exclusive,
135    bool is_recursive,
136    const FileOperationCallback& callback) {
137  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
138  DCHECK(!callback.is_null());
139
140  std::set<std::string>* updated_local_ids = new std::set<std::string>;
141  std::set<base::FilePath>* changed_directories = new std::set<base::FilePath>;
142  base::PostTaskAndReplyWithResult(
143      blocking_task_runner_.get(),
144      FROM_HERE,
145      base::Bind(&UpdateLocalState,
146                 metadata_, directory_path, is_exclusive, is_recursive,
147                 updated_local_ids, changed_directories),
148      base::Bind(&CreateDirectoryOperation::
149                     CreateDirectoryAfterUpdateLocalState,
150                 weak_ptr_factory_.GetWeakPtr(),
151                 callback,
152                 base::Owned(updated_local_ids),
153                 base::Owned(changed_directories)));
154}
155
156void CreateDirectoryOperation::CreateDirectoryAfterUpdateLocalState(
157      const FileOperationCallback& callback,
158      const std::set<std::string>* updated_local_ids,
159      const std::set<base::FilePath>* changed_directories,
160      FileError error) {
161  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
162  DCHECK(!callback.is_null());
163
164  for (std::set<std::string>::const_iterator it = updated_local_ids->begin();
165       it != updated_local_ids->end(); ++it)
166    observer_->OnEntryUpdatedByOperation(*it);
167
168  for (std::set<base::FilePath>::const_iterator it =
169           changed_directories->begin(); it != changed_directories->end(); ++it)
170    observer_->OnDirectoryChangedByOperation(*it);
171
172  callback.Run(error);
173}
174
175}  // namespace file_system
176}  // namespace drive
177