blob: 1f01530557a1541410c6a635a9c9cf9f23ade8a5 [file] [log] [blame]
// Copyright (c) 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "android_webview/native/aw_quota_manager_bridge_impl.h"
#include <set>
#include "android_webview/browser/aw_browser_context.h"
#include "android_webview/browser/aw_content_browser_client.h"
#include "base/android/jni_array.h"
#include "base/android/jni_string.h"
#include "base/synchronization/waitable_event.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/storage_partition.h"
#include "content/public/common/content_client.h"
#include "jni/AwQuotaManagerBridge_jni.h"
#include "storage/browser/quota/quota_manager.h"
#include "storage/common/quota/quota_types.h"
#include "url/gurl.h"
using base::android::AttachCurrentThread;
using content::BrowserThread;
using content::StoragePartition;
using storage::QuotaClient;
using storage::QuotaManager;
namespace android_webview {
namespace {
// This object lives on UI and IO threads. Care need to be taken to make sure
// there are no concurrent accesses to instance variables. Also this object
// is refcounted in the various callbacks, and is destroyed when all callbacks
// are destroyed at the end of DoneOnUIThread.
class GetOriginsTask : public base::RefCountedThreadSafe<GetOriginsTask> {
public:
GetOriginsTask(
const AwQuotaManagerBridgeImpl::GetOriginsCallback& callback,
QuotaManager* quota_manager);
void Run();
private:
friend class base::RefCountedThreadSafe<GetOriginsTask>;
~GetOriginsTask();
void OnOriginsObtained(const std::set<GURL>& origins,
storage::StorageType type);
void OnUsageAndQuotaObtained(const GURL& origin,
storage::QuotaStatusCode status_code,
int64 usage,
int64 quota);
void CheckDone();
void DoneOnUIThread();
AwQuotaManagerBridgeImpl::GetOriginsCallback ui_callback_;
scoped_refptr<QuotaManager> quota_manager_;
std::vector<std::string> origin_;
std::vector<int64> usage_;
std::vector<int64> quota_;
size_t num_callbacks_to_wait_;
size_t num_callbacks_received_;
DISALLOW_COPY_AND_ASSIGN(GetOriginsTask);
};
GetOriginsTask::GetOriginsTask(
const AwQuotaManagerBridgeImpl::GetOriginsCallback& callback,
QuotaManager* quota_manager)
: ui_callback_(callback),
quota_manager_(quota_manager) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
}
GetOriginsTask::~GetOriginsTask() {}
void GetOriginsTask::Run() {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
BrowserThread::PostTask(
BrowserThread::IO,
FROM_HERE,
base::Bind(&QuotaManager::GetOriginsModifiedSince,
quota_manager_,
storage::kStorageTypeTemporary,
base::Time() /* Since beginning of time. */,
base::Bind(&GetOriginsTask::OnOriginsObtained, this)));
}
void GetOriginsTask::OnOriginsObtained(const std::set<GURL>& origins,
storage::StorageType type) {
DCHECK_CURRENTLY_ON(BrowserThread::IO);
num_callbacks_to_wait_ = origins.size();
num_callbacks_received_ = 0u;
for (std::set<GURL>::const_iterator origin = origins.begin();
origin != origins.end();
++origin) {
quota_manager_->GetUsageAndQuota(
*origin,
type,
base::Bind(&GetOriginsTask::OnUsageAndQuotaObtained, this, *origin));
}
CheckDone();
}
void GetOriginsTask::OnUsageAndQuotaObtained(
const GURL& origin,
storage::QuotaStatusCode status_code,
int64 usage,
int64 quota) {
DCHECK_CURRENTLY_ON(BrowserThread::IO);
if (status_code == storage::kQuotaStatusOk) {
origin_.push_back(origin.spec());
usage_.push_back(usage);
quota_.push_back(quota);
}
++num_callbacks_received_;
CheckDone();
}
void GetOriginsTask::CheckDone() {
DCHECK_CURRENTLY_ON(BrowserThread::IO);
if (num_callbacks_received_ == num_callbacks_to_wait_) {
BrowserThread::PostTask(
BrowserThread::UI,
FROM_HERE,
base::Bind(&GetOriginsTask::DoneOnUIThread, this));
} else if (num_callbacks_received_ > num_callbacks_to_wait_) {
NOTREACHED();
}
}
// This method is to avoid copying the 3 vector arguments into a bound callback.
void GetOriginsTask::DoneOnUIThread() {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
ui_callback_.Run(origin_, usage_, quota_);
}
void RunOnUIThread(const base::Closure& task) {
if (BrowserThread::CurrentlyOn(BrowserThread::UI)) {
task.Run();
} else {
BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, task);
}
}
} // namespace
// static
jlong GetDefaultNativeAwQuotaManagerBridge(JNIEnv* env, jclass clazz) {
AwBrowserContext* browser_context =
AwContentBrowserClient::GetAwBrowserContext();
AwQuotaManagerBridgeImpl* bridge = static_cast<AwQuotaManagerBridgeImpl*>(
browser_context->GetQuotaManagerBridge());
DCHECK(bridge);
return reinterpret_cast<intptr_t>(bridge);
}
// static
scoped_refptr<AwQuotaManagerBridge> AwQuotaManagerBridgeImpl::Create(
AwBrowserContext* browser_context) {
return new AwQuotaManagerBridgeImpl(browser_context);
}
AwQuotaManagerBridgeImpl::AwQuotaManagerBridgeImpl(
AwBrowserContext* browser_context)
: browser_context_(browser_context),
weak_factory_(this) {
}
AwQuotaManagerBridgeImpl::~AwQuotaManagerBridgeImpl() {}
void AwQuotaManagerBridgeImpl::Init(JNIEnv* env, jobject object) {
java_ref_ = JavaObjectWeakGlobalRef(env, object);
}
StoragePartition* AwQuotaManagerBridgeImpl::GetStoragePartition() const {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
// AndroidWebview does not use per-site storage partitions.
StoragePartition* storage_partition =
content::BrowserContext::GetDefaultStoragePartition(browser_context_);
DCHECK(storage_partition);
return storage_partition;
}
QuotaManager* AwQuotaManagerBridgeImpl::GetQuotaManager() const {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
QuotaManager* quota_manager = GetStoragePartition()->GetQuotaManager();
DCHECK(quota_manager);
return quota_manager;
}
void AwQuotaManagerBridgeImpl::DeleteAllData(JNIEnv* env, jobject object) {
RunOnUIThread(base::Bind(&AwQuotaManagerBridgeImpl::DeleteAllDataOnUiThread,
this));
}
void AwQuotaManagerBridgeImpl::DeleteAllDataOnUiThread() {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
GetStoragePartition()->ClearData(
// Clear all web storage data except cookies.
StoragePartition::REMOVE_DATA_MASK_APPCACHE |
StoragePartition::REMOVE_DATA_MASK_FILE_SYSTEMS |
StoragePartition::REMOVE_DATA_MASK_INDEXEDDB |
StoragePartition::REMOVE_DATA_MASK_LOCAL_STORAGE |
StoragePartition::REMOVE_DATA_MASK_WEBSQL,
StoragePartition::QUOTA_MANAGED_STORAGE_MASK_TEMPORARY,
GURL(), StoragePartition::OriginMatcherFunction(),
base::Time(), base::Time::Max(), base::Bind(&base::DoNothing));
}
void AwQuotaManagerBridgeImpl::DeleteOrigin(
JNIEnv* env, jobject object, jstring origin) {
base::string16 origin_string(
base::android::ConvertJavaStringToUTF16(env, origin));
RunOnUIThread(base::Bind(&AwQuotaManagerBridgeImpl::DeleteOriginOnUiThread,
this,
origin_string));
}
void AwQuotaManagerBridgeImpl::DeleteOriginOnUiThread(
const base::string16& origin) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
StoragePartition* storage_partition = GetStoragePartition();
storage_partition->ClearDataForOrigin(
// All (temporary) QuotaClient types.
StoragePartition::REMOVE_DATA_MASK_APPCACHE |
StoragePartition::REMOVE_DATA_MASK_FILE_SYSTEMS |
StoragePartition::REMOVE_DATA_MASK_INDEXEDDB |
StoragePartition::REMOVE_DATA_MASK_WEBSQL,
StoragePartition::QUOTA_MANAGED_STORAGE_MASK_TEMPORARY,
GURL(origin),
storage_partition->GetURLRequestContext(),
base::Bind(&base::DoNothing));
}
void AwQuotaManagerBridgeImpl::GetOrigins(
JNIEnv* env, jobject object, jint callback_id) {
RunOnUIThread(base::Bind(&AwQuotaManagerBridgeImpl::GetOriginsOnUiThread,
this,
callback_id));
}
void AwQuotaManagerBridgeImpl::GetOriginsOnUiThread(jint callback_id) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
const GetOriginsCallback ui_callback = base::Bind(
&AwQuotaManagerBridgeImpl::GetOriginsCallbackImpl,
weak_factory_.GetWeakPtr(),
callback_id);
(new GetOriginsTask(ui_callback, GetQuotaManager()))->Run();
}
void AwQuotaManagerBridgeImpl::GetOriginsCallbackImpl(
int jcallback_id,
const std::vector<std::string>& origin,
const std::vector<int64>& usage,
const std::vector<int64>& quota) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jobject> obj = java_ref_.get(env);
if (obj.is_null())
return;
Java_AwQuotaManagerBridge_onGetOriginsCallback(
env,
obj.obj(),
jcallback_id,
base::android::ToJavaArrayOfStrings(env, origin).obj(),
base::android::ToJavaLongArray(env, usage).obj(),
base::android::ToJavaLongArray(env, quota).obj());
}
namespace {
void OnUsageAndQuotaObtained(
const AwQuotaManagerBridgeImpl::QuotaUsageCallback& ui_callback,
storage::QuotaStatusCode status_code,
int64 usage,
int64 quota) {
DCHECK_CURRENTLY_ON(BrowserThread::IO);
if (status_code != storage::kQuotaStatusOk) {
usage = 0;
quota = 0;
}
BrowserThread::PostTask(
BrowserThread::UI,
FROM_HERE,
base::Bind(ui_callback, usage, quota));
}
} // namespace
void AwQuotaManagerBridgeImpl::GetUsageAndQuotaForOrigin(
JNIEnv* env, jobject object,
jstring origin,
jint callback_id,
bool is_quota) {
base::string16 origin_string(
base::android::ConvertJavaStringToUTF16(env, origin));
RunOnUIThread(base::Bind(
&AwQuotaManagerBridgeImpl::GetUsageAndQuotaForOriginOnUiThread,
this,
origin_string,
callback_id,
is_quota));
}
void AwQuotaManagerBridgeImpl::GetUsageAndQuotaForOriginOnUiThread(
const base::string16& origin,
jint callback_id,
bool is_quota) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
const QuotaUsageCallback ui_callback = base::Bind(
&AwQuotaManagerBridgeImpl::QuotaUsageCallbackImpl,
weak_factory_.GetWeakPtr(),
callback_id,
is_quota);
BrowserThread::PostTask(
BrowserThread::IO,
FROM_HERE,
base::Bind(&QuotaManager::GetUsageAndQuota,
GetQuotaManager(),
GURL(origin),
storage::kStorageTypeTemporary,
base::Bind(&OnUsageAndQuotaObtained, ui_callback)));
}
void AwQuotaManagerBridgeImpl::QuotaUsageCallbackImpl(
int jcallback_id, bool is_quota, int64 usage, int64 quota) {
DCHECK_CURRENTLY_ON(BrowserThread::UI);
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jobject> obj = java_ref_.get(env);
if (obj.is_null())
return;
Java_AwQuotaManagerBridge_onGetUsageAndQuotaForOriginCallback(
env, obj.obj(), jcallback_id, is_quota, usage, quota);
}
bool RegisterAwQuotaManagerBridge(JNIEnv* env) {
return RegisterNativesImpl(env);
}
} // namespace android_webview