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_response.h"
6
7 #include "base/bind.h"
8 #include "base/bind_helpers.h"
9 #include "base/compiler_specific.h"
10 #include "base/logging.h"
11 #include "base/message_loop/message_loop.h"
12 #include "base/pickle.h"
13 #include "base/strings/string_util.h"
14 #include "net/base/completion_callback.h"
15 #include "net/base/io_buffer.h"
16 #include "net/base/net_errors.h"
17 #include "webkit/browser/appcache/appcache_storage.h"
18
19 namespace appcache {
20
21 namespace {
22
23 // Disk cache entry data indices.
24 enum {
25 kResponseInfoIndex,
26 kResponseContentIndex
27 };
28
29 // An IOBuffer that wraps a pickle's data. Ownership of the
30 // pickle is transfered to the WrappedPickleIOBuffer object.
31 class WrappedPickleIOBuffer : public net::WrappedIOBuffer {
32 public:
WrappedPickleIOBuffer(const Pickle * pickle)33 explicit WrappedPickleIOBuffer(const Pickle* pickle) :
34 net::WrappedIOBuffer(reinterpret_cast<const char*>(pickle->data())),
35 pickle_(pickle) {
36 DCHECK(pickle->data());
37 }
38
39 private:
~WrappedPickleIOBuffer()40 virtual ~WrappedPickleIOBuffer() {}
41
42 scoped_ptr<const Pickle> pickle_;
43 };
44
45 } // anon namespace
46
47
48 // AppCacheResponseInfo ----------------------------------------------
49
AppCacheResponseInfo(AppCacheStorage * storage,const GURL & manifest_url,int64 response_id,net::HttpResponseInfo * http_info,int64 response_data_size)50 AppCacheResponseInfo::AppCacheResponseInfo(
51 AppCacheStorage* storage, const GURL& manifest_url,
52 int64 response_id, net::HttpResponseInfo* http_info,
53 int64 response_data_size)
54 : manifest_url_(manifest_url), response_id_(response_id),
55 http_response_info_(http_info), response_data_size_(response_data_size),
56 storage_(storage) {
57 DCHECK(http_info);
58 DCHECK(response_id != kNoResponseId);
59 storage_->working_set()->AddResponseInfo(this);
60 }
61
~AppCacheResponseInfo()62 AppCacheResponseInfo::~AppCacheResponseInfo() {
63 storage_->working_set()->RemoveResponseInfo(this);
64 }
65
66 // HttpResponseInfoIOBuffer ------------------------------------------
67
HttpResponseInfoIOBuffer()68 HttpResponseInfoIOBuffer::HttpResponseInfoIOBuffer()
69 : response_data_size(kUnkownResponseDataSize) {}
70
HttpResponseInfoIOBuffer(net::HttpResponseInfo * info)71 HttpResponseInfoIOBuffer::HttpResponseInfoIOBuffer(net::HttpResponseInfo* info)
72 : http_info(info), response_data_size(kUnkownResponseDataSize) {}
73
~HttpResponseInfoIOBuffer()74 HttpResponseInfoIOBuffer::~HttpResponseInfoIOBuffer() {}
75
76 // AppCacheResponseIO ----------------------------------------------
77
AppCacheResponseIO(int64 response_id,int64 group_id,AppCacheDiskCacheInterface * disk_cache)78 AppCacheResponseIO::AppCacheResponseIO(
79 int64 response_id, int64 group_id, AppCacheDiskCacheInterface* disk_cache)
80 : response_id_(response_id),
81 group_id_(group_id),
82 disk_cache_(disk_cache),
83 entry_(NULL),
84 buffer_len_(0),
85 weak_factory_(this) {
86 }
87
~AppCacheResponseIO()88 AppCacheResponseIO::~AppCacheResponseIO() {
89 if (entry_)
90 entry_->Close();
91 }
92
ScheduleIOCompletionCallback(int result)93 void AppCacheResponseIO::ScheduleIOCompletionCallback(int result) {
94 base::MessageLoop::current()->PostTask(
95 FROM_HERE, base::Bind(&AppCacheResponseIO::OnIOComplete,
96 weak_factory_.GetWeakPtr(), result));
97 }
98
InvokeUserCompletionCallback(int result)99 void AppCacheResponseIO::InvokeUserCompletionCallback(int result) {
100 // Clear the user callback and buffers prior to invoking the callback
101 // so the caller can schedule additional operations in the callback.
102 buffer_ = NULL;
103 info_buffer_ = NULL;
104 net::CompletionCallback cb = callback_;
105 callback_.Reset();
106 cb.Run(result);
107 }
108
ReadRaw(int index,int offset,net::IOBuffer * buf,int buf_len)109 void AppCacheResponseIO::ReadRaw(int index, int offset,
110 net::IOBuffer* buf, int buf_len) {
111 DCHECK(entry_);
112 int rv = entry_->Read(
113 index, offset, buf, buf_len,
114 base::Bind(&AppCacheResponseIO::OnRawIOComplete,
115 weak_factory_.GetWeakPtr()));
116 if (rv != net::ERR_IO_PENDING)
117 ScheduleIOCompletionCallback(rv);
118 }
119
WriteRaw(int index,int offset,net::IOBuffer * buf,int buf_len)120 void AppCacheResponseIO::WriteRaw(int index, int offset,
121 net::IOBuffer* buf, int buf_len) {
122 DCHECK(entry_);
123 int rv = entry_->Write(
124 index, offset, buf, buf_len,
125 base::Bind(&AppCacheResponseIO::OnRawIOComplete,
126 weak_factory_.GetWeakPtr()));
127 if (rv != net::ERR_IO_PENDING)
128 ScheduleIOCompletionCallback(rv);
129 }
130
OnRawIOComplete(int result)131 void AppCacheResponseIO::OnRawIOComplete(int result) {
132 DCHECK_NE(net::ERR_IO_PENDING, result);
133 OnIOComplete(result);
134 }
135
136
137 // AppCacheResponseReader ----------------------------------------------
138
AppCacheResponseReader(int64 response_id,int64 group_id,AppCacheDiskCacheInterface * disk_cache)139 AppCacheResponseReader::AppCacheResponseReader(
140 int64 response_id, int64 group_id, AppCacheDiskCacheInterface* disk_cache)
141 : AppCacheResponseIO(response_id, group_id, disk_cache),
142 range_offset_(0),
143 range_length_(kint32max),
144 read_position_(0),
145 weak_factory_(this) {
146 }
147
~AppCacheResponseReader()148 AppCacheResponseReader::~AppCacheResponseReader() {
149 }
150
ReadInfo(HttpResponseInfoIOBuffer * info_buf,const net::CompletionCallback & callback)151 void AppCacheResponseReader::ReadInfo(HttpResponseInfoIOBuffer* info_buf,
152 const net::CompletionCallback& callback) {
153 DCHECK(!callback.is_null());
154 DCHECK(!IsReadPending());
155 DCHECK(info_buf);
156 DCHECK(!info_buf->http_info.get());
157 DCHECK(!buffer_.get());
158 DCHECK(!info_buffer_.get());
159
160 info_buffer_ = info_buf;
161 callback_ = callback; // cleared on completion
162 OpenEntryIfNeededAndContinue();
163 }
164
ContinueReadInfo()165 void AppCacheResponseReader::ContinueReadInfo() {
166 if (!entry_) {
167 ScheduleIOCompletionCallback(net::ERR_CACHE_MISS);
168 return;
169 }
170
171 int size = entry_->GetSize(kResponseInfoIndex);
172 if (size <= 0) {
173 ScheduleIOCompletionCallback(net::ERR_CACHE_MISS);
174 return;
175 }
176
177 buffer_ = new net::IOBuffer(size);
178 ReadRaw(kResponseInfoIndex, 0, buffer_.get(), size);
179 }
180
ReadData(net::IOBuffer * buf,int buf_len,const net::CompletionCallback & callback)181 void AppCacheResponseReader::ReadData(net::IOBuffer* buf, int buf_len,
182 const net::CompletionCallback& callback) {
183 DCHECK(!callback.is_null());
184 DCHECK(!IsReadPending());
185 DCHECK(buf);
186 DCHECK(buf_len >= 0);
187 DCHECK(!buffer_.get());
188 DCHECK(!info_buffer_.get());
189
190 buffer_ = buf;
191 buffer_len_ = buf_len;
192 callback_ = callback; // cleared on completion
193 OpenEntryIfNeededAndContinue();
194 }
195
ContinueReadData()196 void AppCacheResponseReader::ContinueReadData() {
197 if (!entry_) {
198 ScheduleIOCompletionCallback(net::ERR_CACHE_MISS);
199 return;
200 }
201
202 if (read_position_ + buffer_len_ > range_length_) {
203 // TODO(michaeln): What about integer overflows?
204 DCHECK(range_length_ >= read_position_);
205 buffer_len_ = range_length_ - read_position_;
206 }
207 ReadRaw(kResponseContentIndex,
208 range_offset_ + read_position_,
209 buffer_.get(),
210 buffer_len_);
211 }
212
SetReadRange(int offset,int length)213 void AppCacheResponseReader::SetReadRange(int offset, int length) {
214 DCHECK(!IsReadPending() && !read_position_);
215 range_offset_ = offset;
216 range_length_ = length;
217 }
218
OnIOComplete(int result)219 void AppCacheResponseReader::OnIOComplete(int result) {
220 if (result >= 0) {
221 if (info_buffer_.get()) {
222 // Deserialize the http info structure, ensuring we got headers.
223 Pickle pickle(buffer_->data(), result);
224 scoped_ptr<net::HttpResponseInfo> info(new net::HttpResponseInfo);
225 bool response_truncated = false;
226 if (!info->InitFromPickle(pickle, &response_truncated) ||
227 !info->headers.get()) {
228 InvokeUserCompletionCallback(net::ERR_FAILED);
229 return;
230 }
231 DCHECK(!response_truncated);
232 info_buffer_->http_info.reset(info.release());
233
234 // Also return the size of the response body
235 DCHECK(entry_);
236 info_buffer_->response_data_size =
237 entry_->GetSize(kResponseContentIndex);
238 } else {
239 read_position_ += result;
240 }
241 }
242 InvokeUserCompletionCallback(result);
243 }
244
OpenEntryIfNeededAndContinue()245 void AppCacheResponseReader::OpenEntryIfNeededAndContinue() {
246 int rv;
247 AppCacheDiskCacheInterface::Entry** entry_ptr = NULL;
248 if (entry_) {
249 rv = net::OK;
250 } else if (!disk_cache_) {
251 rv = net::ERR_FAILED;
252 } else {
253 entry_ptr = new AppCacheDiskCacheInterface::Entry*;
254 open_callback_ =
255 base::Bind(&AppCacheResponseReader::OnOpenEntryComplete,
256 weak_factory_.GetWeakPtr(), base::Owned(entry_ptr));
257 rv = disk_cache_->OpenEntry(response_id_, entry_ptr, open_callback_);
258 }
259
260 if (rv != net::ERR_IO_PENDING)
261 OnOpenEntryComplete(entry_ptr, rv);
262 }
263
OnOpenEntryComplete(AppCacheDiskCacheInterface::Entry ** entry,int rv)264 void AppCacheResponseReader::OnOpenEntryComplete(
265 AppCacheDiskCacheInterface::Entry** entry, int rv) {
266 DCHECK(info_buffer_.get() || buffer_.get());
267
268 if (!open_callback_.is_null()) {
269 if (rv == net::OK) {
270 DCHECK(entry);
271 entry_ = *entry;
272 }
273 open_callback_.Reset();
274 }
275
276 if (info_buffer_.get())
277 ContinueReadInfo();
278 else
279 ContinueReadData();
280 }
281
282 // AppCacheResponseWriter ----------------------------------------------
283
AppCacheResponseWriter(int64 response_id,int64 group_id,AppCacheDiskCacheInterface * disk_cache)284 AppCacheResponseWriter::AppCacheResponseWriter(
285 int64 response_id, int64 group_id, AppCacheDiskCacheInterface* disk_cache)
286 : AppCacheResponseIO(response_id, group_id, disk_cache),
287 info_size_(0),
288 write_position_(0),
289 write_amount_(0),
290 creation_phase_(INITIAL_ATTEMPT),
291 weak_factory_(this) {
292 }
293
~AppCacheResponseWriter()294 AppCacheResponseWriter::~AppCacheResponseWriter() {
295 }
296
WriteInfo(HttpResponseInfoIOBuffer * info_buf,const net::CompletionCallback & callback)297 void AppCacheResponseWriter::WriteInfo(
298 HttpResponseInfoIOBuffer* info_buf,
299 const net::CompletionCallback& callback) {
300 DCHECK(!callback.is_null());
301 DCHECK(!IsWritePending());
302 DCHECK(info_buf);
303 DCHECK(info_buf->http_info.get());
304 DCHECK(!buffer_.get());
305 DCHECK(!info_buffer_.get());
306 DCHECK(info_buf->http_info->headers.get());
307
308 info_buffer_ = info_buf;
309 callback_ = callback; // cleared on completion
310 CreateEntryIfNeededAndContinue();
311 }
312
ContinueWriteInfo()313 void AppCacheResponseWriter::ContinueWriteInfo() {
314 if (!entry_) {
315 ScheduleIOCompletionCallback(net::ERR_FAILED);
316 return;
317 }
318
319 const bool kSkipTransientHeaders = true;
320 const bool kTruncated = false;
321 Pickle* pickle = new Pickle;
322 info_buffer_->http_info->Persist(pickle, kSkipTransientHeaders, kTruncated);
323 write_amount_ = static_cast<int>(pickle->size());
324 buffer_ = new WrappedPickleIOBuffer(pickle); // takes ownership of pickle
325 WriteRaw(kResponseInfoIndex, 0, buffer_.get(), write_amount_);
326 }
327
WriteData(net::IOBuffer * buf,int buf_len,const net::CompletionCallback & callback)328 void AppCacheResponseWriter::WriteData(
329 net::IOBuffer* buf, int buf_len, const net::CompletionCallback& callback) {
330 DCHECK(!callback.is_null());
331 DCHECK(!IsWritePending());
332 DCHECK(buf);
333 DCHECK(buf_len >= 0);
334 DCHECK(!buffer_.get());
335 DCHECK(!info_buffer_.get());
336
337 buffer_ = buf;
338 write_amount_ = buf_len;
339 callback_ = callback; // cleared on completion
340 CreateEntryIfNeededAndContinue();
341 }
342
ContinueWriteData()343 void AppCacheResponseWriter::ContinueWriteData() {
344 if (!entry_) {
345 ScheduleIOCompletionCallback(net::ERR_FAILED);
346 return;
347 }
348 WriteRaw(
349 kResponseContentIndex, write_position_, buffer_.get(), write_amount_);
350 }
351
OnIOComplete(int result)352 void AppCacheResponseWriter::OnIOComplete(int result) {
353 if (result >= 0) {
354 DCHECK(write_amount_ == result);
355 if (!info_buffer_.get())
356 write_position_ += result;
357 else
358 info_size_ = result;
359 }
360 InvokeUserCompletionCallback(result);
361 }
362
CreateEntryIfNeededAndContinue()363 void AppCacheResponseWriter::CreateEntryIfNeededAndContinue() {
364 int rv;
365 AppCacheDiskCacheInterface::Entry** entry_ptr = NULL;
366 if (entry_) {
367 creation_phase_ = NO_ATTEMPT;
368 rv = net::OK;
369 } else if (!disk_cache_) {
370 creation_phase_ = NO_ATTEMPT;
371 rv = net::ERR_FAILED;
372 } else {
373 creation_phase_ = INITIAL_ATTEMPT;
374 entry_ptr = new AppCacheDiskCacheInterface::Entry*;
375 create_callback_ =
376 base::Bind(&AppCacheResponseWriter::OnCreateEntryComplete,
377 weak_factory_.GetWeakPtr(), base::Owned(entry_ptr));
378 rv = disk_cache_->CreateEntry(response_id_, entry_ptr, create_callback_);
379 }
380 if (rv != net::ERR_IO_PENDING)
381 OnCreateEntryComplete(entry_ptr, rv);
382 }
383
OnCreateEntryComplete(AppCacheDiskCacheInterface::Entry ** entry,int rv)384 void AppCacheResponseWriter::OnCreateEntryComplete(
385 AppCacheDiskCacheInterface::Entry** entry, int rv) {
386 DCHECK(info_buffer_.get() || buffer_.get());
387
388 if (creation_phase_ == INITIAL_ATTEMPT) {
389 if (rv != net::OK) {
390 // We may try to overwrite existing entries.
391 creation_phase_ = DOOM_EXISTING;
392 rv = disk_cache_->DoomEntry(response_id_, create_callback_);
393 if (rv != net::ERR_IO_PENDING)
394 OnCreateEntryComplete(NULL, rv);
395 return;
396 }
397 } else if (creation_phase_ == DOOM_EXISTING) {
398 creation_phase_ = SECOND_ATTEMPT;
399 AppCacheDiskCacheInterface::Entry** entry_ptr =
400 new AppCacheDiskCacheInterface::Entry*;
401 create_callback_ =
402 base::Bind(&AppCacheResponseWriter::OnCreateEntryComplete,
403 weak_factory_.GetWeakPtr(), base::Owned(entry_ptr));
404 rv = disk_cache_->CreateEntry(response_id_, entry_ptr, create_callback_);
405 if (rv != net::ERR_IO_PENDING)
406 OnCreateEntryComplete(entry_ptr, rv);
407 return;
408 }
409
410 if (!create_callback_.is_null()) {
411 if (rv == net::OK)
412 entry_ = *entry;
413
414 create_callback_.Reset();
415 }
416
417 if (info_buffer_.get())
418 ContinueWriteInfo();
419 else
420 ContinueWriteData();
421 }
422
423 } // namespace appcache
424