• 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 #ifndef WEBKIT_BROWSER_APPCACHE_APPCACHE_RESPONSE_H_
6 #define WEBKIT_BROWSER_APPCACHE_APPCACHE_RESPONSE_H_
7 
8 #include "base/compiler_specific.h"
9 #include "base/memory/ref_counted.h"
10 #include "base/memory/scoped_ptr.h"
11 #include "base/memory/weak_ptr.h"
12 #include "net/base/completion_callback.h"
13 #include "net/http/http_response_info.h"
14 #include "url/gurl.h"
15 #include "webkit/browser/webkit_storage_browser_export.h"
16 #include "webkit/common/appcache/appcache_interfaces.h"
17 
18 namespace net {
19 class IOBuffer;
20 }
21 
22 namespace appcache {
23 
24 class AppCacheStorage;
25 
26 static const int kUnkownResponseDataSize = -1;
27 
28 // Response info for a particular response id. Instances are tracked in
29 // the working set.
30 class WEBKIT_STORAGE_BROWSER_EXPORT AppCacheResponseInfo
31     : public base::RefCounted<AppCacheResponseInfo> {
32  public:
33   // AppCacheResponseInfo takes ownership of the http_info.
34   AppCacheResponseInfo(AppCacheStorage* storage, const GURL& manifest_url,
35                        int64 response_id, net::HttpResponseInfo* http_info,
36                        int64 response_data_size);
37 
manifest_url()38   const GURL& manifest_url() const { return manifest_url_; }
response_id()39   int64 response_id() const { return response_id_; }
http_response_info()40   const net::HttpResponseInfo* http_response_info() const {
41     return http_response_info_.get();
42   }
response_data_size()43   int64 response_data_size() const { return response_data_size_; }
44 
45  private:
46   friend class base::RefCounted<AppCacheResponseInfo>;
47   virtual ~AppCacheResponseInfo();
48 
49   const GURL manifest_url_;
50   const int64 response_id_;
51   const scoped_ptr<net::HttpResponseInfo> http_response_info_;
52   const int64 response_data_size_;
53   AppCacheStorage* storage_;
54 };
55 
56 // A refcounted wrapper for HttpResponseInfo so we can apply the
57 // refcounting semantics used with IOBuffer with these structures too.
58 struct WEBKIT_STORAGE_BROWSER_EXPORT HttpResponseInfoIOBuffer
59     : public base::RefCountedThreadSafe<HttpResponseInfoIOBuffer> {
60   scoped_ptr<net::HttpResponseInfo> http_info;
61   int response_data_size;
62 
63   HttpResponseInfoIOBuffer();
64   explicit HttpResponseInfoIOBuffer(net::HttpResponseInfo* info);
65 
66  private:
67   friend class base::RefCountedThreadSafe<HttpResponseInfoIOBuffer>;
68   virtual ~HttpResponseInfoIOBuffer();
69 };
70 
71 // Low level storage API used by the response reader and writer.
72 class WEBKIT_STORAGE_BROWSER_EXPORT AppCacheDiskCacheInterface {
73  public:
74   class Entry {
75    public:
76     virtual int Read(int index, int64 offset, net::IOBuffer* buf, int buf_len,
77                      const net::CompletionCallback& callback) = 0;
78     virtual int Write(int index, int64 offset, net::IOBuffer* buf, int buf_len,
79                       const net::CompletionCallback& callback) = 0;
80     virtual int64 GetSize(int index) = 0;
81     virtual void Close() = 0;
82    protected:
~Entry()83     virtual ~Entry() {}
84   };
85 
86   virtual int CreateEntry(int64 key, Entry** entry,
87                           const net::CompletionCallback& callback) = 0;
88   virtual int OpenEntry(int64 key, Entry** entry,
89                         const net::CompletionCallback& callback) = 0;
90   virtual int DoomEntry(int64 key, const net::CompletionCallback& callback) = 0;
91 
92  protected:
93   friend class base::RefCounted<AppCacheDiskCacheInterface>;
~AppCacheDiskCacheInterface()94   virtual ~AppCacheDiskCacheInterface() {}
95 };
96 
97 // Common base class for response reader and writer.
98 class WEBKIT_STORAGE_BROWSER_EXPORT AppCacheResponseIO {
99  public:
100   virtual ~AppCacheResponseIO();
response_id()101   int64 response_id() const { return response_id_; }
102 
103  protected:
104   AppCacheResponseIO(int64 response_id,
105                      int64 group_id,
106                      AppCacheDiskCacheInterface* disk_cache);
107 
108   virtual void OnIOComplete(int result) = 0;
109 
IsIOPending()110   bool IsIOPending() { return !callback_.is_null(); }
111   void ScheduleIOCompletionCallback(int result);
112   void InvokeUserCompletionCallback(int result);
113   void ReadRaw(int index, int offset, net::IOBuffer* buf, int buf_len);
114   void WriteRaw(int index, int offset, net::IOBuffer* buf, int buf_len);
115 
116   const int64 response_id_;
117   const int64 group_id_;
118   AppCacheDiskCacheInterface* disk_cache_;
119   AppCacheDiskCacheInterface::Entry* entry_;
120   scoped_refptr<HttpResponseInfoIOBuffer> info_buffer_;
121   scoped_refptr<net::IOBuffer> buffer_;
122   int buffer_len_;
123   net::CompletionCallback callback_;
124   base::WeakPtrFactory<AppCacheResponseIO> weak_factory_;
125 
126  private:
127   void OnRawIOComplete(int result);
128 };
129 
130 // Reads existing response data from storage. If the object is deleted
131 // and there is a read in progress, the implementation will return
132 // immediately but will take care of any side effect of cancelling the
133 // operation.  In other words, instances are safe to delete at will.
134 class WEBKIT_STORAGE_BROWSER_EXPORT AppCacheResponseReader
135     : public AppCacheResponseIO {
136  public:
137   virtual ~AppCacheResponseReader();
138 
139   // Reads http info from storage. Always returns the result of the read
140   // asynchronously through the 'callback'. Returns the number of bytes read
141   // or a net:: error code. Guaranteed to not perform partial reads of
142   // the info data. The reader acquires a reference to the 'info_buf' until
143   // completion at which time the callback is invoked with either a negative
144   // error code or the number of bytes read. The 'info_buf' argument should
145   // contain a NULL http_info when ReadInfo is called. The 'callback' is a
146   // required parameter.
147   // Should only be called where there is no Read operation in progress.
148   // (virtual for testing)
149   virtual void ReadInfo(HttpResponseInfoIOBuffer* info_buf,
150                         const net::CompletionCallback& callback);
151 
152   // Reads data from storage. Always returns the result of the read
153   // asynchronously through the 'callback'. Returns the number of bytes read
154   // or a net:: error code. EOF is indicated with a return value of zero.
155   // The reader acquires a reference to the provided 'buf' until completion
156   // at which time the callback is invoked with either a negative error code
157   // or the number of bytes read. The 'callback' is a required parameter.
158   // Should only be called where there is no Read operation in progress.
159   // (virtual for testing)
160   virtual void ReadData(net::IOBuffer* buf, int buf_len,
161                         const net::CompletionCallback& callback);
162 
163   // Returns true if there is a read operation, for data or info, pending.
IsReadPending()164   bool IsReadPending() { return IsIOPending(); }
165 
166   // Used to support range requests. If not called, the reader will
167   // read the entire response body. If called, this must be called prior
168   // to the first call to the ReadData method.
169   void SetReadRange(int offset, int length);
170 
171  protected:
172   friend class AppCacheStorageImpl;
173   friend class MockAppCacheStorage;
174 
175   // Should only be constructed by the storage class.
176   AppCacheResponseReader(int64 response_id,
177                          int64 group_id,
178                          AppCacheDiskCacheInterface* disk_cache);
179 
180   virtual void OnIOComplete(int result) OVERRIDE;
181   void ContinueReadInfo();
182   void ContinueReadData();
183   void OpenEntryIfNeededAndContinue();
184   void OnOpenEntryComplete(AppCacheDiskCacheInterface::Entry** entry, int rv);
185 
186   int range_offset_;
187   int range_length_;
188   int read_position_;
189   net::CompletionCallback open_callback_;
190   base::WeakPtrFactory<AppCacheResponseReader> weak_factory_;
191 };
192 
193 // Writes new response data to storage. If the object is deleted
194 // and there is a write in progress, the implementation will return
195 // immediately but will take care of any side effect of cancelling the
196 // operation. In other words, instances are safe to delete at will.
197 class WEBKIT_STORAGE_BROWSER_EXPORT AppCacheResponseWriter
198     : public AppCacheResponseIO {
199  public:
200   virtual ~AppCacheResponseWriter();
201 
202   // Writes the http info to storage. Always returns the result of the write
203   // asynchronously through the 'callback'. Returns the number of bytes written
204   // or a net:: error code. The writer acquires a reference to the 'info_buf'
205   // until completion at which time the callback is invoked with either a
206   // negative error code or the number of bytes written. The 'callback' is a
207   // required parameter. The contents of 'info_buf' are not modified.
208   // Should only be called where there is no Write operation in progress.
209   void WriteInfo(HttpResponseInfoIOBuffer* info_buf,
210                  const net::CompletionCallback& callback);
211 
212   // Writes data to storage. Always returns the result of the write
213   // asynchronously through the 'callback'. Returns the number of bytes written
214   // or a net:: error code. Guaranteed to not perform partial writes.
215   // The writer acquires a reference to the provided 'buf' until completion at
216   // which time the callback is invoked with either a negative error code or
217   // the number of bytes written. The 'callback' is a required parameter.
218   // The contents of 'buf' are not modified.
219   // Should only be called where there is no Write operation in progress.
220   void WriteData(net::IOBuffer* buf, int buf_len,
221                  const net::CompletionCallback& callback);
222 
223   // Returns true if there is a write pending.
IsWritePending()224   bool IsWritePending() { return IsIOPending(); }
225 
226   // Returns the amount written, info and data.
amount_written()227   int64 amount_written() { return info_size_ + write_position_; }
228 
229  private:
230   friend class AppCacheStorageImpl;
231   friend class MockAppCacheStorage;
232 
233   enum CreationPhase {
234     NO_ATTEMPT,
235     INITIAL_ATTEMPT,
236     DOOM_EXISTING,
237     SECOND_ATTEMPT
238   };
239 
240   // Should only be constructed by the storage class.
241   AppCacheResponseWriter(int64 response_id,
242                          int64 group_id,
243                          AppCacheDiskCacheInterface* disk_cache);
244 
245   virtual void OnIOComplete(int result) OVERRIDE;
246   void ContinueWriteInfo();
247   void ContinueWriteData();
248   void CreateEntryIfNeededAndContinue();
249   void OnCreateEntryComplete(AppCacheDiskCacheInterface::Entry** entry, int rv);
250 
251   int info_size_;
252   int write_position_;
253   int write_amount_;
254   CreationPhase creation_phase_;
255   net::CompletionCallback create_callback_;
256   base::WeakPtrFactory<AppCacheResponseWriter> weak_factory_;
257 };
258 
259 }  // namespace appcache
260 
261 #endif  // WEBKIT_BROWSER_APPCACHE_APPCACHE_RESPONSE_H_
262