blob: fd1c23ddb2ae03432c126d1f399618fab3054c6b [file] [log] [blame]
/*
* Copyright (C) 2008 Apple Inc. All Rights Reserved.
* Copyright (C) 2009, 2011 Google Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
#include "third_party/blink/renderer/modules/filesystem/worker_global_scope_file_system.h"
#include <memory>
#include "third_party/blink/public/mojom/filesystem/file_system.mojom-blink.h"
#include "third_party/blink/renderer/core/fileapi/file_error.h"
#include "third_party/blink/renderer/core/workers/worker_global_scope.h"
#include "third_party/blink/renderer/modules/filesystem/async_callback_helper.h"
#include "third_party/blink/renderer/modules/filesystem/directory_entry_sync.h"
#include "third_party/blink/renderer/modules/filesystem/dom_file_system.h"
#include "third_party/blink/renderer/modules/filesystem/entry.h"
#include "third_party/blink/renderer/modules/filesystem/file_entry_sync.h"
#include "third_party/blink/renderer/modules/filesystem/file_system_callbacks.h"
#include "third_party/blink/renderer/modules/filesystem/local_file_system.h"
#include "third_party/blink/renderer/modules/filesystem/sync_callback_helper.h"
#include "third_party/blink/renderer/platform/bindings/exception_state.h"
#include "third_party/blink/renderer/platform/instrumentation/use_counter.h"
#include "third_party/blink/renderer/platform/weborigin/security_origin.h"
namespace blink {
void WorkerGlobalScopeFileSystem::webkitRequestFileSystem(
WorkerGlobalScope& worker,
int type,
int64_t size,
V8FileSystemCallback* success_callback,
V8ErrorCallback* error_callback) {
ExecutionContext* secure_context = worker.GetExecutionContext();
auto error_callback_wrapper =
AsyncCallbackHelper::ErrorCallback(error_callback);
if (!secure_context->GetSecurityOrigin()->CanAccessFileSystem()) {
DOMFileSystem::ReportError(&worker, std::move(error_callback_wrapper),
base::File::FILE_ERROR_SECURITY);
return;
} else if (secure_context->GetSecurityOrigin()->IsLocal()) {
UseCounter::Count(secure_context, WebFeature::kFileAccessedFileSystem);
}
mojom::blink::FileSystemType file_system_type =
static_cast<mojom::blink::FileSystemType>(type);
if (!DOMFileSystemBase::IsValidType(file_system_type)) {
DOMFileSystem::ReportError(&worker, std::move(error_callback_wrapper),
base::File::FILE_ERROR_INVALID_OPERATION);
return;
}
auto success_callback_wrapper =
AsyncCallbackHelper::SuccessCallback<DOMFileSystem>(success_callback);
LocalFileSystem::From(worker)->RequestFileSystem(
&worker, file_system_type, size,
std::make_unique<FileSystemCallbacks>(std::move(success_callback_wrapper),
std::move(error_callback_wrapper),
&worker, file_system_type),
LocalFileSystem::kAsynchronous);
}
DOMFileSystemSync* WorkerGlobalScopeFileSystem::webkitRequestFileSystemSync(
WorkerGlobalScope& worker,
int type,
int64_t size,
ExceptionState& exception_state) {
ExecutionContext* secure_context = worker.GetExecutionContext();
if (!secure_context->GetSecurityOrigin()->CanAccessFileSystem()) {
exception_state.ThrowSecurityError(file_error::kSecurityErrorMessage);
return nullptr;
} else if (secure_context->GetSecurityOrigin()->IsLocal()) {
UseCounter::Count(secure_context, WebFeature::kFileAccessedFileSystem);
}
mojom::blink::FileSystemType file_system_type =
static_cast<mojom::blink::FileSystemType>(type);
if (!DOMFileSystemBase::IsValidType(file_system_type)) {
exception_state.ThrowDOMException(
DOMExceptionCode::kInvalidModificationError,
"the type must be kTemporary or kPersistent.");
return nullptr;
}
auto* sync_helper = MakeGarbageCollected<FileSystemCallbacksSyncHelper>();
auto success_callback_wrapper =
WTF::Bind(&FileSystemCallbacksSyncHelper::OnSuccess,
WrapPersistentIfNeeded(sync_helper));
auto error_callback_wrapper =
WTF::Bind(&FileSystemCallbacksSyncHelper::OnError,
WrapPersistentIfNeeded(sync_helper));
auto callbacks = std::make_unique<FileSystemCallbacks>(
std::move(success_callback_wrapper), std::move(error_callback_wrapper),
&worker, file_system_type);
LocalFileSystem::From(worker)->RequestFileSystem(
&worker, file_system_type, size, std::move(callbacks),
LocalFileSystem::kSynchronous);
DOMFileSystem* file_system = sync_helper->GetResultOrThrow(exception_state);
return file_system ? MakeGarbageCollected<DOMFileSystemSync>(file_system)
: nullptr;
}
void WorkerGlobalScopeFileSystem::webkitResolveLocalFileSystemURL(
WorkerGlobalScope& worker,
const String& url,
V8EntryCallback* success_callback,
V8ErrorCallback* error_callback) {
KURL completed_url = worker.CompleteURL(url);
ExecutionContext* secure_context = worker.GetExecutionContext();
auto error_callback_wrapper =
AsyncCallbackHelper::ErrorCallback(error_callback);
if (!secure_context->GetSecurityOrigin()->CanAccessFileSystem() ||
!secure_context->GetSecurityOrigin()->CanRequest(completed_url)) {
DOMFileSystem::ReportError(&worker, std::move(error_callback_wrapper),
base::File::FILE_ERROR_SECURITY);
return;
} else if (secure_context->GetSecurityOrigin()->IsLocal()) {
UseCounter::Count(secure_context, WebFeature::kFileAccessedFileSystem);
}
if (!completed_url.IsValid()) {
DOMFileSystem::ReportError(&worker, std::move(error_callback_wrapper),
base::File::FILE_ERROR_INVALID_URL);
return;
}
auto success_callback_wrapper =
AsyncCallbackHelper::SuccessCallback<Entry>(success_callback);
LocalFileSystem::From(worker)->ResolveURL(
&worker, completed_url,
std::make_unique<ResolveURICallbacks>(std::move(success_callback_wrapper),
std::move(error_callback_wrapper),
&worker),
LocalFileSystem::kAsynchronous);
}
EntrySync* WorkerGlobalScopeFileSystem::webkitResolveLocalFileSystemSyncURL(
WorkerGlobalScope& worker,
const String& url,
ExceptionState& exception_state) {
KURL completed_url = worker.CompleteURL(url);
ExecutionContext* secure_context = worker.GetExecutionContext();
if (!secure_context->GetSecurityOrigin()->CanAccessFileSystem() ||
!secure_context->GetSecurityOrigin()->CanRequest(completed_url)) {
exception_state.ThrowSecurityError(file_error::kSecurityErrorMessage);
return nullptr;
} else if (secure_context->GetSecurityOrigin()->IsLocal()) {
UseCounter::Count(secure_context, WebFeature::kFileAccessedFileSystem);
}
if (!completed_url.IsValid()) {
exception_state.ThrowDOMException(DOMExceptionCode::kEncodingError,
"the URL '" + url + "' is invalid.");
return nullptr;
}
auto* sync_helper = MakeGarbageCollected<EntryCallbacksSyncHelper>();
auto success_callback_wrapper =
WTF::Bind(&EntryCallbacksSyncHelper::OnSuccess,
WrapPersistentIfNeeded(sync_helper));
auto error_callback_wrapper = WTF::Bind(&EntryCallbacksSyncHelper::OnError,
WrapPersistentIfNeeded(sync_helper));
std::unique_ptr<ResolveURICallbacks> callbacks =
std::make_unique<ResolveURICallbacks>(std::move(success_callback_wrapper),
std::move(error_callback_wrapper),
&worker);
LocalFileSystem::From(worker)->ResolveURL(&worker, completed_url,
std::move(callbacks),
LocalFileSystem::kSynchronous);
Entry* entry = sync_helper->GetResultOrThrow(exception_state);
return entry ? EntrySync::Create(entry) : nullptr;
}
static_assert(static_cast<int>(WorkerGlobalScopeFileSystem::kTemporary) ==
static_cast<int>(mojom::blink::FileSystemType::kTemporary),
"WorkerGlobalScopeFileSystem::kTemporary should match "
"FileSystemTypeTemporary");
static_assert(static_cast<int>(WorkerGlobalScopeFileSystem::kPersistent) ==
static_cast<int>(mojom::blink::FileSystemType::kPersistent),
"WorkerGlobalScopeFileSystem::kPersistent should match "
"FileSystemTypePersistent");
} // namespace blink