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 "content/browser/net/view_http_cache_job_factory.h"
6
7 #include "base/bind.h"
8 #include "base/bind_helpers.h"
9 #include "base/callback.h"
10 #include "base/compiler_specific.h"
11 #include "base/memory/weak_ptr.h"
12 #include "base/message_loop/message_loop.h"
13 #include "base/strings/string_util.h"
14 #include "content/public/common/url_constants.h"
15 #include "net/base/completion_callback.h"
16 #include "net/base/net_errors.h"
17 #include "net/url_request/url_request.h"
18 #include "net/url_request/url_request_simple_job.h"
19 #include "net/url_request/view_cache_helper.h"
20
21 namespace content {
22 namespace {
23
24 // A job subclass that dumps an HTTP cache entry.
25 class ViewHttpCacheJob : public net::URLRequestJob {
26 public:
ViewHttpCacheJob(net::URLRequest * request,net::NetworkDelegate * network_delegate)27 ViewHttpCacheJob(net::URLRequest* request,
28 net::NetworkDelegate* network_delegate)
29 : net::URLRequestJob(request, network_delegate),
30 core_(new Core),
31 callback_(base::Bind(&ViewHttpCacheJob::OnStartCompleted,
32 base::Unretained(this))),
33 weak_factory_(this) {
34 }
35
36 // net::URLRequestJob implementation.
37 virtual void Start() OVERRIDE;
38 virtual void Kill() OVERRIDE;
GetMimeType(std::string * mime_type) const39 virtual bool GetMimeType(std::string* mime_type) const OVERRIDE{
40 return core_->GetMimeType(mime_type);
41 }
GetCharset(std::string * charset)42 virtual bool GetCharset(std::string* charset) OVERRIDE{
43 return core_->GetCharset(charset);
44 }
ReadRawData(net::IOBuffer * buf,int buf_size,int * bytes_read)45 virtual bool ReadRawData(net::IOBuffer* buf,
46 int buf_size, int *bytes_read) OVERRIDE{
47 return core_->ReadRawData(buf, buf_size, bytes_read);
48 }
49
50 private:
51 class Core : public base::RefCounted<Core> {
52 public:
Core()53 Core()
54 : data_offset_(0),
55 callback_(base::Bind(&Core::OnIOComplete, this)) {
56 }
57
58 int Start(const net::URLRequest& request, const base::Closure& callback);
59
60 // Prevents it from invoking its callback. It will self-delete.
Orphan()61 void Orphan() {
62 user_callback_.Reset();
63 }
64
65 bool GetMimeType(std::string* mime_type) const;
66 bool GetCharset(std::string* charset);
67 bool ReadRawData(net::IOBuffer* buf, int buf_size, int *bytes_read);
68
69 private:
70 friend class base::RefCounted<Core>;
71
~Core()72 ~Core() {}
73
74 // Called when ViewCacheHelper completes the operation.
75 void OnIOComplete(int result);
76
77 std::string data_;
78 int data_offset_;
79 net::ViewCacheHelper cache_helper_;
80 net::CompletionCallback callback_;
81 base::Closure user_callback_;
82
83 DISALLOW_COPY_AND_ASSIGN(Core);
84 };
85
~ViewHttpCacheJob()86 virtual ~ViewHttpCacheJob() {}
87
88 void StartAsync();
89 void OnStartCompleted();
90
91 scoped_refptr<Core> core_;
92 base::Closure callback_;
93
94 base::WeakPtrFactory<ViewHttpCacheJob> weak_factory_;
95
96 DISALLOW_COPY_AND_ASSIGN(ViewHttpCacheJob);
97 };
98
Start()99 void ViewHttpCacheJob::Start() {
100 base::MessageLoop::current()->PostTask(
101 FROM_HERE,
102 base::Bind(&ViewHttpCacheJob::StartAsync, weak_factory_.GetWeakPtr()));
103 }
104
Kill()105 void ViewHttpCacheJob::Kill() {
106 weak_factory_.InvalidateWeakPtrs();
107 if (core_.get()) {
108 core_->Orphan();
109 core_ = NULL;
110 }
111 net::URLRequestJob::Kill();
112 }
113
StartAsync()114 void ViewHttpCacheJob::StartAsync() {
115 DCHECK(request());
116
117 if (!request())
118 return;
119
120 int rv = core_->Start(*request(), callback_);
121 if (rv != net::ERR_IO_PENDING) {
122 DCHECK_EQ(net::OK, rv);
123 OnStartCompleted();
124 }
125 }
126
OnStartCompleted()127 void ViewHttpCacheJob::OnStartCompleted() {
128 NotifyHeadersComplete();
129 }
130
Start(const net::URLRequest & request,const base::Closure & callback)131 int ViewHttpCacheJob::Core::Start(const net::URLRequest& request,
132 const base::Closure& callback) {
133 DCHECK(!callback.is_null());
134 DCHECK(user_callback_.is_null());
135
136 AddRef(); // Released on OnIOComplete().
137 std::string cache_key =
138 request.url().spec().substr(strlen(kChromeUINetworkViewCacheURL));
139
140 int rv;
141 if (cache_key.empty()) {
142 rv = cache_helper_.GetContentsHTML(request.context(),
143 kChromeUINetworkViewCacheURL,
144 &data_, callback_);
145 } else {
146 rv = cache_helper_.GetEntryInfoHTML(cache_key, request.context(),
147 &data_, callback_);
148 }
149
150 if (rv == net::ERR_IO_PENDING)
151 user_callback_ = callback;
152
153 return rv;
154 }
155
GetMimeType(std::string * mime_type) const156 bool ViewHttpCacheJob::Core::GetMimeType(std::string* mime_type) const {
157 mime_type->assign("text/html");
158 return true;
159 }
160
GetCharset(std::string * charset)161 bool ViewHttpCacheJob::Core::GetCharset(std::string* charset) {
162 charset->assign("UTF-8");
163 return true;
164 }
165
ReadRawData(net::IOBuffer * buf,int buf_size,int * bytes_read)166 bool ViewHttpCacheJob::Core::ReadRawData(net::IOBuffer* buf,
167 int buf_size,
168 int* bytes_read) {
169 DCHECK(bytes_read);
170 int remaining = static_cast<int>(data_.size()) - data_offset_;
171 if (buf_size > remaining)
172 buf_size = remaining;
173 memcpy(buf->data(), data_.data() + data_offset_, buf_size);
174 data_offset_ += buf_size;
175 *bytes_read = buf_size;
176 return true;
177 }
178
OnIOComplete(int result)179 void ViewHttpCacheJob::Core::OnIOComplete(int result) {
180 DCHECK_EQ(net::OK, result);
181
182 if (!user_callback_.is_null())
183 user_callback_.Run();
184
185 // We may be holding the last reference to this job. Do not access |this|
186 // after Release().
187 Release(); // Acquired on Start().
188 }
189
190 } // namespace.
191
192 // Static.
IsSupportedURL(const GURL & url)193 bool ViewHttpCacheJobFactory::IsSupportedURL(const GURL& url) {
194 return url.SchemeIs(kChromeUIScheme) &&
195 url.host() == kChromeUINetworkViewCacheHost;
196 }
197
198 // Static.
CreateJobForRequest(net::URLRequest * request,net::NetworkDelegate * network_delegate)199 net::URLRequestJob* ViewHttpCacheJobFactory::CreateJobForRequest(
200 net::URLRequest* request, net::NetworkDelegate* network_delegate) {
201 return new ViewHttpCacheJob(request, network_delegate);
202 }
203
204 } // namespace content
205