OLD | NEW |
---|---|
(Empty) | |
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 "android_webview/native/aw_quota_manager_bridge_impl.h" | |
6 | |
7 #include <set> | |
8 | |
9 #include "android_webview/browser/aw_browser_context.h" | |
10 #include "android_webview/browser/aw_content_browser_client.h" | |
11 #include "base/android/jni_array.h" | |
12 #include "base/android/jni_string.h" | |
13 #include "content/public/browser/browser_thread.h" | |
14 #include "content/public/browser/storage_partition.h" | |
15 #include "content/public/common/content_client.h" | |
16 #include "googleurl/src/gurl.h" | |
17 #include "jni/AwQuotaManagerBridge_jni.h" | |
18 #include "webkit/quota/quota_manager.h" | |
19 #include "webkit/quota/quota_types.h" | |
20 | |
21 using base::android::AttachCurrentThread; | |
22 using content::BrowserThread; | |
23 using content::StoragePartition; | |
24 using quota::QuotaClient; | |
25 using quota::QuotaManager; | |
26 | |
27 namespace android_webview { | |
28 | |
29 namespace { | |
30 | |
31 class GetOriginsTask : public base::RefCountedThreadSafe<GetOriginsTask> { | |
32 public: | |
33 GetOriginsTask( | |
34 const AwQuotaManagerBridgeImpl::GetOriginsCallback& callback, | |
35 QuotaManager* quota_manager); | |
36 | |
37 void Run(); | |
38 | |
39 private: | |
40 friend class base::RefCountedThreadSafe<GetOriginsTask>; | |
41 ~GetOriginsTask(); | |
42 | |
43 void OnOriginsObtained(const std::set<GURL>& origins, | |
44 quota::StorageType type); | |
45 | |
46 void OnUsageAndQuotaObtained(const GURL& origin, | |
47 quota::QuotaStatusCode status_code, | |
48 int64 usage, | |
49 int64 quota); | |
50 | |
51 void CheckDone(); | |
52 void DoneOnUIThread(); | |
53 | |
54 AwQuotaManagerBridgeImpl::GetOriginsCallback ui_callback_; | |
55 scoped_refptr<QuotaManager> quota_manager_; | |
56 | |
57 std::vector<std::string> origin_; | |
58 std::vector<int64> usage_; | |
59 std::vector<int64> quota_; | |
60 | |
61 size_t num_callbacks_to_wait_; | |
62 size_t num_callbacks_received_; | |
63 | |
64 DISALLOW_COPY_AND_ASSIGN(GetOriginsTask); | |
65 }; | |
66 | |
67 GetOriginsTask::GetOriginsTask( | |
68 const AwQuotaManagerBridgeImpl::GetOriginsCallback& callback, | |
69 QuotaManager* quota_manager) | |
70 : ui_callback_(callback), | |
71 quota_manager_(quota_manager) { | |
72 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
73 } | |
74 | |
75 GetOriginsTask::~GetOriginsTask() {} | |
76 | |
77 void GetOriginsTask::Run() { | |
78 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
79 BrowserThread::PostTask( | |
80 BrowserThread::IO, | |
81 FROM_HERE, | |
82 base::Bind(&QuotaManager::GetOriginsModifiedSince, | |
83 quota_manager_, | |
84 quota::kStorageTypeTemporary, | |
85 base::Time() /* Since beginning of time. */, | |
86 base::Bind(&GetOriginsTask::OnOriginsObtained, this))); | |
87 } | |
88 | |
89 void GetOriginsTask::OnOriginsObtained( | |
90 const std::set<GURL>& origins, quota::StorageType type) { | |
91 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
92 num_callbacks_to_wait_ = origins.size(); | |
93 num_callbacks_received_ = 0u; | |
94 | |
95 for (std::set<GURL>::const_iterator origin = origins.begin(); | |
96 origin != origins.end(); | |
97 ++origin) { | |
98 quota_manager_->GetUsageAndQuota( | |
99 *origin, | |
100 type, | |
101 base::Bind(&GetOriginsTask::OnUsageAndQuotaObtained, this, *origin)); | |
102 } | |
103 | |
104 CheckDone(); | |
105 } | |
106 | |
107 void GetOriginsTask::OnUsageAndQuotaObtained(const GURL& origin, | |
108 quota::QuotaStatusCode status_code, | |
109 int64 usage, | |
110 int64 quota) { | |
111 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
112 if (status_code == quota::kQuotaStatusOk) { | |
113 origin_.push_back(origin.spec()); | |
114 usage_.push_back(usage); | |
115 quota_.push_back(quota); | |
116 } | |
117 | |
118 ++num_callbacks_received_; | |
119 CheckDone(); | |
120 } | |
121 | |
122 void GetOriginsTask::CheckDone() { | |
123 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
124 if (num_callbacks_received_ == num_callbacks_to_wait_) { | |
125 BrowserThread::PostTask( | |
126 BrowserThread::UI, | |
127 FROM_HERE, | |
128 base::Bind(&GetOriginsTask::DoneOnUIThread, this)); | |
129 } else if (num_callbacks_received_ > num_callbacks_to_wait_) { | |
130 NOTREACHED(); | |
131 } | |
132 } | |
133 | |
134 void GetOriginsTask::DoneOnUIThread() { | |
135 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
136 ui_callback_.Run(origin_, usage_, quota_); | |
137 } | |
138 | |
139 } // namespace | |
140 | |
141 | |
142 // static | |
143 jint GetDefaultNativeAwQuotaManagerBridge(JNIEnv* env, jclass clazz) { | |
144 content::ContentBrowserClient* browser_client = | |
145 content::GetContentClient()->browser(); | |
146 DCHECK(browser_client); | |
147 | |
148 AwContentBrowserClient* aw_browser_client = | |
149 AwContentBrowserClient::FromContentBrowserClient(browser_client); | |
150 AwBrowserContext* browser_context = aw_browser_client->GetAwBrowserContext(); | |
151 DCHECK(browser_context); | |
152 | |
153 AwQuotaManagerBridgeImpl* bridge = static_cast<AwQuotaManagerBridgeImpl*>( | |
154 browser_context->GetQuotaManagerBridge()); | |
155 DCHECK(bridge); | |
156 return reinterpret_cast<jint>(bridge); | |
157 } | |
158 | |
159 AwQuotaManagerBridgeImpl::AwQuotaManagerBridgeImpl( | |
160 AwBrowserContext* browser_context) | |
161 : weak_factory_(ALLOW_THIS_IN_INITIALIZER_LIST(this)), | |
162 browser_context_(browser_context) { | |
163 } | |
164 | |
165 void AwQuotaManagerBridgeImpl::Init(JNIEnv* env, jobject object) { | |
166 java_ref_ = JavaObjectWeakGlobalRef(env, object); | |
167 } | |
168 | |
169 QuotaManager* AwQuotaManagerBridgeImpl::GetQuotaManager() const { | |
170 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
171 | |
172 // AndroidWebview does not use per-site storage partitions. | |
173 // TODO(boliu): Get @ajwong to review this before submit. | |
174 StoragePartition* storage_partition = | |
175 content::BrowserContext::GetStoragePartitionForSite( | |
176 browser_context_, GURL()); | |
177 DCHECK(storage_partition); | |
178 | |
179 QuotaManager* quota_manager = storage_partition->GetQuotaManager(); | |
180 DCHECK(quota_manager); | |
181 return quota_manager; | |
182 } | |
183 | |
184 namespace { | |
185 | |
186 void IgnoreStatus(quota::QuotaStatusCode status_code) {} | |
187 | |
188 // Not using QuotaClient::kAllClientsMask because | |
189 // QuotaManager::DeleteOriginData would ignore the StrageType. | |
190 int GetAllClientMask() { | |
191 return QuotaClient::kFileSystem | | |
192 QuotaClient::kDatabase | | |
193 QuotaClient::kAppcache | | |
194 QuotaClient::kIndexedDatabase; | |
195 } | |
196 | |
197 void DeleteAllOriginData( | |
198 QuotaManager* quota_manager, | |
199 const std::set<GURL>& origins, | |
200 quota::StorageType type) { | |
201 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
202 // TODO(boliu): Maybe not spam the db thread? | |
203 for (std::set<GURL>::const_iterator origin = origins.begin(); | |
204 origin != origins.end(); | |
205 ++origin) { | |
206 quota_manager->DeleteOriginData(*origin, | |
207 type, | |
208 GetAllClientMask(), | |
209 base::Bind(&IgnoreStatus)); | |
210 } | |
211 } | |
212 | |
213 } // namespace | |
214 | |
215 void AwQuotaManagerBridgeImpl::DeleteAllData(JNIEnv* env, jobject object) { | |
216 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
217 QuotaManager* quota_manager = GetQuotaManager(); | |
218 BrowserThread::PostTask( | |
219 BrowserThread::IO, | |
220 FROM_HERE, | |
221 base::Bind(&QuotaManager::GetOriginsModifiedSince, | |
222 quota_manager, | |
223 quota::kStorageTypeTemporary, | |
224 base::Time() /* Since beginning of time. */, | |
225 base::Bind(&DeleteAllOriginData, | |
226 make_scoped_refptr(quota_manager)))); | |
boliu
2013/02/20 20:14:16
With out make_scoped_refptr here, it would fail to
mkosiba (inactive)
2013/02/21 11:38:30
ah.. it seems like it's fine for class methods (so
| |
227 } | |
228 | |
229 void AwQuotaManagerBridgeImpl::DeleteOrigin( | |
230 JNIEnv* env, jobject object, jstring origin) { | |
231 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
232 BrowserThread::PostTask( | |
233 BrowserThread::IO, | |
234 FROM_HERE, | |
235 base::Bind(&QuotaManager::DeleteOriginData, | |
236 GetQuotaManager(), | |
237 GURL(base::android::ConvertJavaStringToUTF16(env, origin)), | |
238 quota::kStorageTypeTemporary, | |
239 GetAllClientMask(), | |
240 base::Bind(&IgnoreStatus))); | |
241 } | |
242 | |
243 void AwQuotaManagerBridgeImpl::GetOrigins( | |
244 JNIEnv* env, jobject object, jint callback_id) { | |
245 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
246 | |
247 const GetOriginsCallback ui_callback = base::Bind( | |
248 &AwQuotaManagerBridgeImpl::GetOriginsCallbackImpl, | |
249 weak_factory_.GetWeakPtr(), | |
250 callback_id); | |
251 | |
252 (new GetOriginsTask(ui_callback, GetQuotaManager()))->Run(); | |
253 } | |
254 | |
255 void AwQuotaManagerBridgeImpl::GetOriginsCallbackImpl( | |
256 int jcallback_id, | |
257 const std::vector<std::string>& origin, | |
258 const std::vector<int64>& usage, | |
259 const std::vector<int64>& quota) { | |
260 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
261 JNIEnv* env = AttachCurrentThread(); | |
262 ScopedJavaLocalRef<jobject> obj = java_ref_.get(env); | |
263 if (obj.is_null()) | |
264 return; | |
265 | |
266 Java_AwQuotaManagerBridge_onGetOriginsCallback( | |
267 env, | |
268 obj.obj(), | |
269 jcallback_id, | |
270 base::android::ToJavaArrayOfStrings(env, origin).obj(), | |
271 base::android::ToJavaLongArray(env, usage.begin(), usage.size()).obj(), | |
272 base::android::ToJavaLongArray(env, quota.begin(), quota.size()).obj()); | |
273 } | |
274 | |
275 namespace { | |
276 | |
277 void RunGetUsageAndQuotaCallback( | |
278 const AwQuotaManagerBridgeImpl::QuotaUsageCallback& ui_callback, | |
279 int64 usage, | |
280 int64 quota) { | |
281 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
282 ui_callback.Run(usage, quota); | |
283 } | |
284 | |
285 void OnUsageAndQuotaObtained( | |
286 const AwQuotaManagerBridgeImpl::QuotaUsageCallback& ui_callback, | |
287 quota::QuotaStatusCode status_code, | |
288 int64 usage, | |
289 int64 quota) { | |
290 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
291 if (status_code != quota::kQuotaStatusOk) { | |
292 // Indicate error. TODO(boliu): Check this does not mean infinite. | |
293 usage = -1; | |
294 quota = -1; | |
295 } | |
296 BrowserThread::PostTask( | |
297 BrowserThread::UI, | |
298 FROM_HERE, | |
299 base::Bind(&RunGetUsageAndQuotaCallback, | |
300 ui_callback, | |
301 usage, | |
302 quota)); | |
303 } | |
304 | |
305 } // namespace | |
306 | |
307 void AwQuotaManagerBridgeImpl::GetUsageAndQuotaForOrigin( | |
308 JNIEnv* env, jobject object, | |
309 jstring origin, | |
310 jint callback_id, | |
311 bool is_quota) { | |
312 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
313 const QuotaUsageCallback ui_callback = base::Bind( | |
314 &AwQuotaManagerBridgeImpl::QuotaUsageCallbackImpl, | |
315 weak_factory_.GetWeakPtr(), | |
316 callback_id, | |
317 is_quota); | |
318 | |
319 BrowserThread::PostTask( | |
320 BrowserThread::IO, | |
321 FROM_HERE, | |
322 base::Bind(&QuotaManager::GetUsageAndQuota, | |
323 GetQuotaManager(), | |
324 GURL(base::android::ConvertJavaStringToUTF16(env, origin)), | |
325 quota::kStorageTypeTemporary, | |
326 base::Bind(&OnUsageAndQuotaObtained, ui_callback))); | |
327 } | |
328 | |
329 void AwQuotaManagerBridgeImpl::QuotaUsageCallbackImpl( | |
330 int jcallback_id, bool is_quota, int64 usage, int64 quota) { | |
331 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
332 JNIEnv* env = AttachCurrentThread(); | |
333 ScopedJavaLocalRef<jobject> obj = java_ref_.get(env); | |
334 if (obj.is_null()) | |
335 return; | |
336 | |
337 Java_AwQuotaManagerBridge_onGetUsageAndQuotaForOriginCallback( | |
338 env, obj.obj(), jcallback_id, is_quota, usage, quota); | |
339 } | |
340 | |
341 bool RegisterAwQuotaManagerBridge(JNIEnv* env) { | |
342 return RegisterNativesImpl(env) >= 0; | |
343 } | |
344 | |
345 } // namespace android_webview | |
OLD | NEW |