• Home
  • History
  • Annotate
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2010 Google Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30 
31 #include "config.h"
32 #include "modules/filesystem/FileSystemCallbacks.h"
33 
34 #include "core/dom/ExecutionContext.h"
35 #include "core/fileapi/File.h"
36 #include "core/fileapi/FileError.h"
37 #include "core/html/VoidCallback.h"
38 #include "core/inspector/InspectorInstrumentation.h"
39 #include "modules/filesystem/DOMFilePath.h"
40 #include "modules/filesystem/DOMFileSystem.h"
41 #include "modules/filesystem/DOMFileSystemBase.h"
42 #include "modules/filesystem/DirectoryEntry.h"
43 #include "modules/filesystem/DirectoryReader.h"
44 #include "modules/filesystem/Entry.h"
45 #include "modules/filesystem/EntryCallback.h"
46 #include "modules/filesystem/ErrorCallback.h"
47 #include "modules/filesystem/FileCallback.h"
48 #include "modules/filesystem/FileEntry.h"
49 #include "modules/filesystem/FileSystemCallback.h"
50 #include "modules/filesystem/FileWriterBase.h"
51 #include "modules/filesystem/FileWriterBaseCallback.h"
52 #include "modules/filesystem/Metadata.h"
53 #include "modules/filesystem/MetadataCallback.h"
54 #include "platform/FileMetadata.h"
55 #include "public/platform/WebFileWriter.h"
56 
57 namespace blink {
58 
FileSystemCallbacksBase(ErrorCallback * errorCallback,DOMFileSystemBase * fileSystem,ExecutionContext * context)59 FileSystemCallbacksBase::FileSystemCallbacksBase(ErrorCallback* errorCallback, DOMFileSystemBase* fileSystem, ExecutionContext* context)
60     : m_errorCallback(errorCallback)
61     , m_fileSystem(fileSystem)
62     , m_executionContext(context)
63     , m_asyncOperationId(0)
64 {
65     if (m_fileSystem)
66         m_fileSystem->addPendingCallbacks();
67     if (m_executionContext)
68         m_asyncOperationId = InspectorInstrumentation::traceAsyncOperationStarting(m_executionContext.get(), "FileSystem");
69 }
70 
~FileSystemCallbacksBase()71 FileSystemCallbacksBase::~FileSystemCallbacksBase()
72 {
73     if (m_fileSystem)
74         m_fileSystem->removePendingCallbacks();
75     if (m_asyncOperationId && m_executionContext)
76         InspectorInstrumentation::traceAsyncOperationCompleted(m_executionContext.get(), m_asyncOperationId);
77 }
78 
didFail(int code)79 void FileSystemCallbacksBase::didFail(int code)
80 {
81     if (m_errorCallback)
82         handleEventOrScheduleCallback(m_errorCallback.release(), FileError::create(static_cast<FileError::ErrorCode>(code)));
83 }
84 
shouldScheduleCallback() const85 bool FileSystemCallbacksBase::shouldScheduleCallback() const
86 {
87     return !shouldBlockUntilCompletion() && m_executionContext && m_executionContext->activeDOMObjectsAreSuspended();
88 }
89 
90 #if !ENABLE(OILPAN)
91 template <typename CB, typename CBArg>
handleEventOrScheduleCallback(RawPtr<CB> callback,RawPtr<CBArg> arg)92 void FileSystemCallbacksBase::handleEventOrScheduleCallback(RawPtr<CB> callback, RawPtr<CBArg> arg)
93 {
94     handleEventOrScheduleCallback(callback, arg.get());
95 }
96 #endif
97 
98 template <typename CB, typename CBArg>
handleEventOrScheduleCallback(RawPtr<CB> callback,CBArg * arg)99 void FileSystemCallbacksBase::handleEventOrScheduleCallback(RawPtr<CB> callback, CBArg* arg)
100 {
101     ASSERT(callback);
102     InspectorInstrumentationCookie cookie = InspectorInstrumentation::traceAsyncOperationCompletedCallbackStarting(m_executionContext.get(), m_asyncOperationId);
103     if (shouldScheduleCallback())
104         DOMFileSystem::scheduleCallback(m_executionContext.get(), callback.get(), arg);
105     else if (callback)
106         callback->handleEvent(arg);
107     m_executionContext.clear();
108     InspectorInstrumentation::traceAsyncCallbackCompleted(cookie);
109 }
110 
111 template <typename CB, typename CBArg>
handleEventOrScheduleCallback(RawPtr<CB> callback,PassRefPtrWillBeRawPtr<CBArg> arg)112 void FileSystemCallbacksBase::handleEventOrScheduleCallback(RawPtr<CB> callback, PassRefPtrWillBeRawPtr<CBArg> arg)
113 {
114     ASSERT(callback);
115     InspectorInstrumentationCookie cookie = InspectorInstrumentation::traceAsyncOperationCompletedCallbackStarting(m_executionContext.get(), m_asyncOperationId);
116     if (shouldScheduleCallback())
117         DOMFileSystem::scheduleCallback(m_executionContext.get(), callback.get(), arg);
118     else if (callback)
119         callback->handleEvent(arg.get());
120     m_executionContext.clear();
121     InspectorInstrumentation::traceAsyncCallbackCompleted(cookie);
122 }
123 
124 template <typename CB>
handleEventOrScheduleCallback(RawPtr<CB> callback)125 void FileSystemCallbacksBase::handleEventOrScheduleCallback(RawPtr<CB> callback)
126 {
127     ASSERT(callback);
128     InspectorInstrumentationCookie cookie = InspectorInstrumentation::traceAsyncOperationCompletedCallbackStarting(m_executionContext.get(), m_asyncOperationId);
129     if (shouldScheduleCallback())
130         DOMFileSystem::scheduleCallback(m_executionContext.get(), callback.get());
131     else if (callback)
132         callback->handleEvent();
133     m_executionContext.clear();
134     InspectorInstrumentation::traceAsyncCallbackCompleted(cookie);
135 }
136 
137 // EntryCallbacks -------------------------------------------------------------
138 
create(EntryCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,DOMFileSystemBase * fileSystem,const String & expectedPath,bool isDirectory)139 PassOwnPtr<AsyncFileSystemCallbacks> EntryCallbacks::create(EntryCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, DOMFileSystemBase* fileSystem, const String& expectedPath, bool isDirectory)
140 {
141     return adoptPtr(new EntryCallbacks(successCallback, errorCallback, context, fileSystem, expectedPath, isDirectory));
142 }
143 
EntryCallbacks(EntryCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,DOMFileSystemBase * fileSystem,const String & expectedPath,bool isDirectory)144 EntryCallbacks::EntryCallbacks(EntryCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, DOMFileSystemBase* fileSystem, const String& expectedPath, bool isDirectory)
145     : FileSystemCallbacksBase(errorCallback, fileSystem, context)
146     , m_successCallback(successCallback)
147     , m_expectedPath(expectedPath)
148     , m_isDirectory(isDirectory)
149 {
150 }
151 
didSucceed()152 void EntryCallbacks::didSucceed()
153 {
154     if (m_successCallback) {
155         if (m_isDirectory)
156             handleEventOrScheduleCallback(m_successCallback.release(), DirectoryEntry::create(m_fileSystem, m_expectedPath));
157         else
158             handleEventOrScheduleCallback(m_successCallback.release(), FileEntry::create(m_fileSystem, m_expectedPath));
159     }
160 }
161 
162 // EntriesCallbacks -----------------------------------------------------------
163 
create(EntriesCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,DirectoryReaderBase * directoryReader,const String & basePath)164 PassOwnPtr<AsyncFileSystemCallbacks> EntriesCallbacks::create(EntriesCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, DirectoryReaderBase* directoryReader, const String& basePath)
165 {
166     return adoptPtr(new EntriesCallbacks(successCallback, errorCallback, context, directoryReader, basePath));
167 }
168 
EntriesCallbacks(EntriesCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,DirectoryReaderBase * directoryReader,const String & basePath)169 EntriesCallbacks::EntriesCallbacks(EntriesCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, DirectoryReaderBase* directoryReader, const String& basePath)
170     : FileSystemCallbacksBase(errorCallback, directoryReader->filesystem(), context)
171     , m_successCallback(successCallback)
172     , m_directoryReader(directoryReader)
173     , m_basePath(basePath)
174 {
175     ASSERT(m_directoryReader);
176 }
177 
didReadDirectoryEntry(const String & name,bool isDirectory)178 void EntriesCallbacks::didReadDirectoryEntry(const String& name, bool isDirectory)
179 {
180     if (isDirectory)
181         m_entries.append(DirectoryEntry::create(m_directoryReader->filesystem(), DOMFilePath::append(m_basePath, name)));
182     else
183         m_entries.append(FileEntry::create(m_directoryReader->filesystem(), DOMFilePath::append(m_basePath, name)));
184 }
185 
didReadDirectoryEntries(bool hasMore)186 void EntriesCallbacks::didReadDirectoryEntries(bool hasMore)
187 {
188     m_directoryReader->setHasMoreEntries(hasMore);
189     EntryHeapVector entries;
190     entries.swap(m_entries);
191     // FIXME: delay the callback iff shouldScheduleCallback() is true.
192     InspectorInstrumentationCookie cookie = InspectorInstrumentation::traceAsyncCallbackStarting(m_executionContext.get(), m_asyncOperationId);
193     if (m_successCallback)
194         m_successCallback->handleEvent(entries);
195     InspectorInstrumentation::traceAsyncCallbackCompleted(cookie);
196     if (!hasMore)
197         InspectorInstrumentation::traceAsyncOperationCompleted(m_executionContext.get(), m_asyncOperationId);
198 }
199 
200 // FileSystemCallbacks --------------------------------------------------------
201 
create(FileSystemCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,FileSystemType type)202 PassOwnPtr<AsyncFileSystemCallbacks> FileSystemCallbacks::create(FileSystemCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, FileSystemType type)
203 {
204     return adoptPtr(new FileSystemCallbacks(successCallback, errorCallback, context, type));
205 }
206 
FileSystemCallbacks(FileSystemCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,FileSystemType type)207 FileSystemCallbacks::FileSystemCallbacks(FileSystemCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, FileSystemType type)
208     : FileSystemCallbacksBase(errorCallback, nullptr, context)
209     , m_successCallback(successCallback)
210     , m_type(type)
211 {
212 }
213 
didOpenFileSystem(const String & name,const KURL & rootURL)214 void FileSystemCallbacks::didOpenFileSystem(const String& name, const KURL& rootURL)
215 {
216     if (m_successCallback)
217         handleEventOrScheduleCallback(m_successCallback.release(), DOMFileSystem::create(m_executionContext.get(), name, m_type, rootURL));
218 }
219 
220 // ResolveURICallbacks --------------------------------------------------------
221 
create(EntryCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context)222 PassOwnPtr<AsyncFileSystemCallbacks> ResolveURICallbacks::create(EntryCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context)
223 {
224     return adoptPtr(new ResolveURICallbacks(successCallback, errorCallback, context));
225 }
226 
ResolveURICallbacks(EntryCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context)227 ResolveURICallbacks::ResolveURICallbacks(EntryCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context)
228     : FileSystemCallbacksBase(errorCallback, nullptr, context)
229     , m_successCallback(successCallback)
230 {
231 }
232 
didResolveURL(const String & name,const KURL & rootURL,FileSystemType type,const String & filePath,bool isDirectory)233 void ResolveURICallbacks::didResolveURL(const String& name, const KURL& rootURL, FileSystemType type, const String& filePath, bool isDirectory)
234 {
235     DOMFileSystem* filesystem = DOMFileSystem::create(m_executionContext.get(), name, type, rootURL);
236     DirectoryEntry* root = filesystem->root();
237 
238     String absolutePath;
239     if (!DOMFileSystemBase::pathToAbsolutePath(type, root, filePath, absolutePath)) {
240         handleEventOrScheduleCallback(m_errorCallback.release(), FileError::create(FileError::INVALID_MODIFICATION_ERR));
241         return;
242     }
243 
244     if (isDirectory)
245         handleEventOrScheduleCallback(m_successCallback.release(), DirectoryEntry::create(filesystem, absolutePath));
246     else
247         handleEventOrScheduleCallback(m_successCallback.release(), FileEntry::create(filesystem, absolutePath));
248 }
249 
250 // MetadataCallbacks ----------------------------------------------------------
251 
create(MetadataCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,DOMFileSystemBase * fileSystem)252 PassOwnPtr<AsyncFileSystemCallbacks> MetadataCallbacks::create(MetadataCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, DOMFileSystemBase* fileSystem)
253 {
254     return adoptPtr(new MetadataCallbacks(successCallback, errorCallback, context, fileSystem));
255 }
256 
MetadataCallbacks(MetadataCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,DOMFileSystemBase * fileSystem)257 MetadataCallbacks::MetadataCallbacks(MetadataCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, DOMFileSystemBase* fileSystem)
258     : FileSystemCallbacksBase(errorCallback, fileSystem, context)
259     , m_successCallback(successCallback)
260 {
261 }
262 
didReadMetadata(const FileMetadata & metadata)263 void MetadataCallbacks::didReadMetadata(const FileMetadata& metadata)
264 {
265     if (m_successCallback)
266         handleEventOrScheduleCallback(m_successCallback.release(), Metadata::create(metadata));
267 }
268 
269 // FileWriterBaseCallbacks ----------------------------------------------------
270 
create(PassRefPtrWillBeRawPtr<FileWriterBase> fileWriter,FileWriterBaseCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context)271 PassOwnPtr<AsyncFileSystemCallbacks> FileWriterBaseCallbacks::create(PassRefPtrWillBeRawPtr<FileWriterBase> fileWriter, FileWriterBaseCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context)
272 {
273     return adoptPtr(new FileWriterBaseCallbacks(fileWriter, successCallback, errorCallback, context));
274 }
275 
FileWriterBaseCallbacks(PassRefPtrWillBeRawPtr<FileWriterBase> fileWriter,FileWriterBaseCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context)276 FileWriterBaseCallbacks::FileWriterBaseCallbacks(PassRefPtrWillBeRawPtr<FileWriterBase> fileWriter, FileWriterBaseCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context)
277     : FileSystemCallbacksBase(errorCallback, nullptr, context)
278     , m_fileWriter(fileWriter.get())
279     , m_successCallback(successCallback)
280 {
281 }
282 
didCreateFileWriter(PassOwnPtr<WebFileWriter> fileWriter,long long length)283 void FileWriterBaseCallbacks::didCreateFileWriter(PassOwnPtr<WebFileWriter> fileWriter, long long length)
284 {
285     m_fileWriter->initialize(fileWriter, length);
286     if (m_successCallback)
287         handleEventOrScheduleCallback(m_successCallback.release(), m_fileWriter.release());
288 }
289 
290 // SnapshotFileCallback -------------------------------------------------------
291 
create(DOMFileSystemBase * filesystem,const String & name,const KURL & url,FileCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context)292 PassOwnPtr<AsyncFileSystemCallbacks> SnapshotFileCallback::create(DOMFileSystemBase* filesystem, const String& name, const KURL& url, FileCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context)
293 {
294     return adoptPtr(new SnapshotFileCallback(filesystem, name, url, successCallback, errorCallback, context));
295 }
296 
SnapshotFileCallback(DOMFileSystemBase * filesystem,const String & name,const KURL & url,FileCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context)297 SnapshotFileCallback::SnapshotFileCallback(DOMFileSystemBase* filesystem, const String& name, const KURL& url, FileCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context)
298     : FileSystemCallbacksBase(errorCallback, filesystem, context)
299     , m_name(name)
300     , m_url(url)
301     , m_successCallback(successCallback)
302 {
303 }
304 
didCreateSnapshotFile(const FileMetadata & metadata,PassRefPtr<BlobDataHandle> snapshot)305 void SnapshotFileCallback::didCreateSnapshotFile(const FileMetadata& metadata, PassRefPtr<BlobDataHandle> snapshot)
306 {
307     if (!m_successCallback)
308         return;
309 
310     // We can't directly use the snapshot blob data handle because the content type on it hasn't been set.
311     // The |snapshot| param is here to provide a a chain of custody thru thread bridging that is held onto until
312     // *after* we've coined a File with a new handle that has the correct type set on it. This allows the
313     // blob storage system to track when a temp file can and can't be safely deleted.
314 
315     handleEventOrScheduleCallback(m_successCallback.release(), DOMFileSystemBase::createFile(metadata, m_url, m_fileSystem->type(), m_name));
316 }
317 
318 // VoidCallbacks --------------------------------------------------------------
319 
create(VoidCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,DOMFileSystemBase * fileSystem)320 PassOwnPtr<AsyncFileSystemCallbacks> VoidCallbacks::create(VoidCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, DOMFileSystemBase* fileSystem)
321 {
322     return adoptPtr(new VoidCallbacks(successCallback, errorCallback, context, fileSystem));
323 }
324 
VoidCallbacks(VoidCallback * successCallback,ErrorCallback * errorCallback,ExecutionContext * context,DOMFileSystemBase * fileSystem)325 VoidCallbacks::VoidCallbacks(VoidCallback* successCallback, ErrorCallback* errorCallback, ExecutionContext* context, DOMFileSystemBase* fileSystem)
326     : FileSystemCallbacksBase(errorCallback, fileSystem, context)
327     , m_successCallback(successCallback)
328 {
329 }
330 
didSucceed()331 void VoidCallbacks::didSucceed()
332 {
333     if (m_successCallback)
334         handleEventOrScheduleCallback(m_successCallback.release());
335 }
336 
337 } // namespace blink
338