• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 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 "webkit/browser/appcache/appcache_quota_client.h"
6 
7 #include <algorithm>
8 #include <map>
9 #include <set>
10 
11 #include "base/bind.h"
12 #include "base/bind_helpers.h"
13 #include "webkit/browser/appcache/appcache_service.h"
14 
15 using quota::QuotaClient;
16 
17 namespace {
NetErrorCodeToQuotaStatus(int code)18 quota::QuotaStatusCode NetErrorCodeToQuotaStatus(int code) {
19   if (code == net::OK)
20     return quota::kQuotaStatusOk;
21   else if (code == net::ERR_ABORTED)
22     return quota::kQuotaErrorAbort;
23   else
24     return quota::kQuotaStatusUnknown;
25 }
26 
RunFront(appcache::AppCacheQuotaClient::RequestQueue * queue)27 void RunFront(appcache::AppCacheQuotaClient::RequestQueue* queue) {
28   base::Closure request = queue->front();
29   queue->pop_front();
30   request.Run();
31 }
32 }  // namespace
33 
34 namespace appcache {
35 
AppCacheQuotaClient(AppCacheService * service)36 AppCacheQuotaClient::AppCacheQuotaClient(AppCacheService* service)
37     : service_(service),
38       appcache_is_ready_(false),
39       quota_manager_is_destroyed_(false) {
40 }
41 
~AppCacheQuotaClient()42 AppCacheQuotaClient::~AppCacheQuotaClient() {
43   DCHECK(pending_batch_requests_.empty());
44   DCHECK(pending_serial_requests_.empty());
45   DCHECK(current_delete_request_callback_.is_null());
46 }
47 
id() const48 QuotaClient::ID AppCacheQuotaClient::id() const {
49   return kAppcache;
50 }
51 
OnQuotaManagerDestroyed()52 void AppCacheQuotaClient::OnQuotaManagerDestroyed() {
53   DeletePendingRequests();
54   if (!current_delete_request_callback_.is_null()) {
55     current_delete_request_callback_.Reset();
56     GetServiceDeleteCallback()->Cancel();
57   }
58 
59   quota_manager_is_destroyed_ = true;
60   if (!service_)
61     delete this;
62 }
63 
GetOriginUsage(const GURL & origin,quota::StorageType type,const GetUsageCallback & callback)64 void AppCacheQuotaClient::GetOriginUsage(
65     const GURL& origin,
66     quota::StorageType type,
67     const GetUsageCallback& callback) {
68   DCHECK(!callback.is_null());
69   DCHECK(!quota_manager_is_destroyed_);
70 
71   if (!service_) {
72     callback.Run(0);
73     return;
74   }
75 
76   if (!appcache_is_ready_) {
77     pending_batch_requests_.push_back(
78         base::Bind(&AppCacheQuotaClient::GetOriginUsage,
79                    base::Unretained(this), origin, type, callback));
80     return;
81   }
82 
83   if (type != quota::kStorageTypeTemporary) {
84     callback.Run(0);
85     return;
86   }
87 
88   const AppCacheStorage::UsageMap* map = GetUsageMap();
89   AppCacheStorage::UsageMap::const_iterator found = map->find(origin);
90   if (found == map->end()) {
91     callback.Run(0);
92     return;
93   }
94   callback.Run(found->second);
95 }
96 
GetOriginsForType(quota::StorageType type,const GetOriginsCallback & callback)97 void AppCacheQuotaClient::GetOriginsForType(
98     quota::StorageType type,
99     const GetOriginsCallback& callback) {
100   GetOriginsHelper(type, std::string(), callback);
101 }
102 
GetOriginsForHost(quota::StorageType type,const std::string & host,const GetOriginsCallback & callback)103 void AppCacheQuotaClient::GetOriginsForHost(
104     quota::StorageType type,
105     const std::string& host,
106     const GetOriginsCallback& callback) {
107   DCHECK(!callback.is_null());
108   if (host.empty()) {
109     callback.Run(std::set<GURL>());
110     return;
111   }
112   GetOriginsHelper(type, host, callback);
113 }
114 
DeleteOriginData(const GURL & origin,quota::StorageType type,const DeletionCallback & callback)115 void AppCacheQuotaClient::DeleteOriginData(const GURL& origin,
116                                            quota::StorageType type,
117                                            const DeletionCallback& callback) {
118   DCHECK(!quota_manager_is_destroyed_);
119 
120   if (!service_) {
121     callback.Run(quota::kQuotaErrorAbort);
122     return;
123   }
124 
125   if (!appcache_is_ready_ || !current_delete_request_callback_.is_null()) {
126     pending_serial_requests_.push_back(
127         base::Bind(&AppCacheQuotaClient::DeleteOriginData,
128                    base::Unretained(this), origin, type, callback));
129     return;
130   }
131 
132   current_delete_request_callback_ = callback;
133   if (type != quota::kStorageTypeTemporary) {
134     DidDeleteAppCachesForOrigin(net::OK);
135     return;
136   }
137 
138   service_->DeleteAppCachesForOrigin(
139       origin, GetServiceDeleteCallback()->callback());
140 }
141 
DoesSupport(quota::StorageType type) const142 bool AppCacheQuotaClient::DoesSupport(quota::StorageType type) const {
143   return type == quota::kStorageTypeTemporary;
144 }
145 
DidDeleteAppCachesForOrigin(int rv)146 void AppCacheQuotaClient::DidDeleteAppCachesForOrigin(int rv) {
147   DCHECK(service_);
148   if (quota_manager_is_destroyed_)
149     return;
150 
151   // Finish the request by calling our callers callback.
152   current_delete_request_callback_.Run(NetErrorCodeToQuotaStatus(rv));
153   current_delete_request_callback_.Reset();
154   if (pending_serial_requests_.empty())
155     return;
156 
157   // Start the next in the queue.
158   RunFront(&pending_serial_requests_);
159 }
160 
GetOriginsHelper(quota::StorageType type,const std::string & opt_host,const GetOriginsCallback & callback)161 void AppCacheQuotaClient::GetOriginsHelper(
162     quota::StorageType type,
163     const std::string& opt_host,
164     const GetOriginsCallback& callback) {
165   DCHECK(!callback.is_null());
166   DCHECK(!quota_manager_is_destroyed_);
167 
168   if (!service_) {
169     callback.Run(std::set<GURL>());
170     return;
171   }
172 
173   if (!appcache_is_ready_) {
174     pending_batch_requests_.push_back(
175         base::Bind(&AppCacheQuotaClient::GetOriginsHelper,
176                    base::Unretained(this), type, opt_host, callback));
177     return;
178   }
179 
180   if (type != quota::kStorageTypeTemporary) {
181     callback.Run(std::set<GURL>());
182     return;
183   }
184 
185   const AppCacheStorage::UsageMap* map = GetUsageMap();
186   std::set<GURL> origins;
187   for (AppCacheStorage::UsageMap::const_iterator iter = map->begin();
188        iter != map->end(); ++iter) {
189     if (opt_host.empty() || iter->first.host() == opt_host)
190       origins.insert(iter->first);
191   }
192   callback.Run(origins);
193 }
194 
ProcessPendingRequests()195 void AppCacheQuotaClient::ProcessPendingRequests() {
196   DCHECK(appcache_is_ready_);
197   while (!pending_batch_requests_.empty())
198     RunFront(&pending_batch_requests_);
199 
200   if (!pending_serial_requests_.empty())
201     RunFront(&pending_serial_requests_);
202 }
203 
DeletePendingRequests()204 void AppCacheQuotaClient::DeletePendingRequests() {
205   pending_batch_requests_.clear();
206   pending_serial_requests_.clear();
207 }
208 
GetUsageMap()209 const AppCacheStorage::UsageMap* AppCacheQuotaClient::GetUsageMap() {
210   DCHECK(service_);
211   return service_->storage()->usage_map();
212 }
213 
214 net::CancelableCompletionCallback*
GetServiceDeleteCallback()215 AppCacheQuotaClient::GetServiceDeleteCallback() {
216   // Lazily created due to CancelableCompletionCallback's threading
217   // restrictions, there is no way to detach from the thread created on.
218   if (!service_delete_callback_) {
219     service_delete_callback_.reset(
220         new net::CancelableCompletionCallback(
221             base::Bind(&AppCacheQuotaClient::DidDeleteAppCachesForOrigin,
222                        base::Unretained(this))));
223   }
224   return service_delete_callback_.get();
225 }
226 
NotifyAppCacheReady()227 void AppCacheQuotaClient::NotifyAppCacheReady() {
228   // Can reoccur during reinitialization.
229   if (!appcache_is_ready_) {
230     appcache_is_ready_ = true;
231     ProcessPendingRequests();
232   }
233 }
234 
NotifyAppCacheDestroyed()235 void AppCacheQuotaClient::NotifyAppCacheDestroyed() {
236   service_ = NULL;
237   while (!pending_batch_requests_.empty())
238     RunFront(&pending_batch_requests_);
239 
240   while (!pending_serial_requests_.empty())
241     RunFront(&pending_serial_requests_);
242 
243   if (!current_delete_request_callback_.is_null()) {
244     current_delete_request_callback_.Run(quota::kQuotaErrorAbort);
245     current_delete_request_callback_.Reset();
246     GetServiceDeleteCallback()->Cancel();
247   }
248 
249   if (quota_manager_is_destroyed_)
250     delete this;
251 }
252 
253 }  // namespace appcache
254