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.
5 #include "content/browser/loader/certificate_resource_handler.h"
7 #include "base/strings/string_util.h"
8 #include "content/browser/loader/resource_request_info_impl.h"
9 #include "content/public/browser/content_browser_client.h"
10 #include "content/public/common/resource_response.h"
11 #include "net/base/io_buffer.h"
12 #include "net/base/mime_sniffer.h"
13 #include "net/base/mime_util.h"
14 #include "net/http/http_response_headers.h"
15 #include "net/url_request/redirect_info.h"
16 #include "net/url_request/url_request.h"
17 #include "net/url_request/url_request_status.h"
21 CertificateResourceHandler::CertificateResourceHandler(
22 net::URLRequest
* request
)
23 : ResourceHandler(request
),
26 resource_buffer_(NULL
),
27 cert_type_(net::CERTIFICATE_MIME_TYPE_UNKNOWN
) {
30 CertificateResourceHandler::~CertificateResourceHandler() {
33 bool CertificateResourceHandler::OnUploadProgress(uint64 position
,
38 bool CertificateResourceHandler::OnRequestRedirected(
39 const net::RedirectInfo
& redirect_info
,
40 ResourceResponse
* resp
,
45 bool CertificateResourceHandler::OnResponseStarted(ResourceResponse
* resp
,
47 cert_type_
= net::GetCertificateMimeTypeForMimeType(resp
->head
.mime_type
);
48 return cert_type_
!= net::CERTIFICATE_MIME_TYPE_UNKNOWN
;
51 bool CertificateResourceHandler::OnWillStart(const GURL
& url
, bool* defer
) {
55 bool CertificateResourceHandler::OnBeforeNetworkStart(const GURL
& url
,
60 bool CertificateResourceHandler::OnWillRead(scoped_refptr
<net::IOBuffer
>* buf
,
63 static const int kReadBufSize
= 32768;
65 // TODO(gauravsh): Should we use 'min_size' here?
66 DCHECK(buf
&& buf_size
);
67 if (!read_buffer_
.get()) {
68 read_buffer_
= new net::IOBuffer(kReadBufSize
);
70 *buf
= read_buffer_
.get();
71 *buf_size
= kReadBufSize
;
76 bool CertificateResourceHandler::OnReadCompleted(int bytes_read
, bool* defer
) {
80 // We have more data to read.
81 DCHECK(read_buffer_
.get());
82 content_length_
+= bytes_read
;
84 // Release the ownership of the buffer, and store a reference
85 // to it. A new one will be allocated in OnWillRead().
86 scoped_refptr
<net::IOBuffer
> buffer
;
87 read_buffer_
.swap(buffer
);
88 // TODO(gauravsh): Should this be handled by a separate thread?
89 buffer_
.push_back(std::make_pair(buffer
, bytes_read
));
94 void CertificateResourceHandler::OnResponseCompleted(
95 const net::URLRequestStatus
& urs
,
96 const std::string
& sec_info
,
98 if (urs
.status() != net::URLRequestStatus::SUCCESS
)
103 const void* content_bytes
= NULL
;
104 if (resource_buffer_
.get())
105 content_bytes
= resource_buffer_
->data();
107 // Note that it's up to the browser to verify that the certificate
108 // data is well-formed.
109 const ResourceRequestInfo
* info
= GetRequestInfo();
110 GetContentClient()->browser()->AddCertificate(
111 cert_type_
, content_bytes
, content_length_
,
112 info
->GetChildID(), info
->GetRenderFrameID());
115 void CertificateResourceHandler::AssembleResource() {
116 // 0-length IOBuffers are not allowed.
117 if (content_length_
== 0) {
118 resource_buffer_
= NULL
;
122 // Create the new buffer.
123 resource_buffer_
= new net::IOBuffer(content_length_
);
125 // Copy the data into it.
126 size_t bytes_copied
= 0;
127 for (size_t i
= 0; i
< buffer_
.size(); ++i
) {
128 net::IOBuffer
* data
= buffer_
[i
].first
.get();
129 size_t data_len
= buffer_
[i
].second
;
130 DCHECK(data
!= NULL
);
131 DCHECK_LE(bytes_copied
+ data_len
, content_length_
);
132 memcpy(resource_buffer_
->data() + bytes_copied
, data
->data(), data_len
);
133 bytes_copied
+= data_len
;
135 DCHECK_EQ(content_length_
, bytes_copied
);
138 void CertificateResourceHandler::OnDataDownloaded(int bytes_downloaded
) {
142 } // namespace content