1372 lines
40 KiB
C++
1372 lines
40 KiB
C++
// Copyright (c) 2008-2009 The Chromium Embedded Framework Authors. All rights
|
|
// reserved. Use of this source code is governed by a BSD-style license that
|
|
// can be found in the LICENSE file.
|
|
|
|
#include <limits>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "libcef/common/cef_messages.h"
|
|
#include "libcef/common/net/http_header_utils.h"
|
|
#include "libcef/common/net/upload_data.h"
|
|
#include "libcef/common/request_impl.h"
|
|
#include "libcef/browser/navigate_params.h"
|
|
|
|
#include "base/command_line.h"
|
|
#include "base/logging.h"
|
|
#include "base/strings/string_util.h"
|
|
#include "base/strings/sys_string_conversions.h"
|
|
#include "base/strings/utf_string_conversions.h"
|
|
#include "components/navigation_interception/navigation_params.h"
|
|
#include "content/public/browser/browser_thread.h"
|
|
#include "content/public/browser/resource_request_info.h"
|
|
#include "content/public/common/content_switches.h"
|
|
#include "content/public/common/resource_type.h"
|
|
#include "net/base/elements_upload_data_stream.h"
|
|
#include "net/base/load_flags.h"
|
|
#include "net/base/upload_data_stream.h"
|
|
#include "net/base/upload_element_reader.h"
|
|
#include "net/base/upload_bytes_element_reader.h"
|
|
#include "net/base/upload_file_element_reader.h"
|
|
#include "net/http/http_request_headers.h"
|
|
#include "net/http/http_util.h"
|
|
#include "net/url_request/url_fetcher.h"
|
|
#include "net/url_request/url_request.h"
|
|
#include "third_party/WebKit/public/platform/WebCachePolicy.h"
|
|
#include "third_party/WebKit/public/platform/WebHTTPHeaderVisitor.h"
|
|
#include "third_party/WebKit/public/platform/WebString.h"
|
|
#include "third_party/WebKit/public/platform/WebURL.h"
|
|
#include "third_party/WebKit/public/platform/WebURLError.h"
|
|
#include "third_party/WebKit/public/platform/WebURLRequest.h"
|
|
#include "third_party/WebKit/public/web/WebSecurityPolicy.h"
|
|
|
|
namespace {
|
|
|
|
const char kReferrerLowerCase[] = "referer";
|
|
const char kContentTypeLowerCase[] = "content-type";
|
|
const char kApplicationFormURLEncoded[] = "application/x-www-form-urlencoded";
|
|
|
|
// A subclass of net::UploadBytesElementReader that keeps the associated
|
|
// UploadElement alive until the request completes.
|
|
class BytesElementReader : public net::UploadBytesElementReader {
|
|
public:
|
|
explicit BytesElementReader(std::unique_ptr<net::UploadElement> element)
|
|
: net::UploadBytesElementReader(element->bytes(),
|
|
element->bytes_length()),
|
|
element_(std::move(element)) {
|
|
DCHECK_EQ(net::UploadElement::TYPE_BYTES, element_->type());
|
|
}
|
|
|
|
private:
|
|
std::unique_ptr<net::UploadElement> element_;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(BytesElementReader);
|
|
};
|
|
|
|
base::TaskRunner* GetFileTaskRunner() {
|
|
return content::BrowserThread::GetTaskRunnerForThread(
|
|
content::BrowserThread::FILE).get();
|
|
}
|
|
|
|
// A subclass of net::UploadFileElementReader that keeps the associated
|
|
// UploadElement alive until the request completes.
|
|
class FileElementReader : public net::UploadFileElementReader {
|
|
public:
|
|
explicit FileElementReader(std::unique_ptr<net::UploadElement> element)
|
|
: net::UploadFileElementReader(
|
|
GetFileTaskRunner(),
|
|
element->file_path(),
|
|
element->file_range_offset(),
|
|
element->file_range_length(),
|
|
element->expected_file_modification_time()),
|
|
element_(std::move(element)) {
|
|
DCHECK_EQ(net::UploadElement::TYPE_FILE, element_->type());
|
|
}
|
|
|
|
private:
|
|
std::unique_ptr<net::UploadElement> element_;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(FileElementReader);
|
|
};
|
|
|
|
// GetURLRequestReferrerPolicy() and GetURLRequestReferrer() are based on
|
|
// SetReferrerForRequest() from
|
|
// content/browser/loader/resource_dispatcher_host_impl.cc
|
|
|
|
net::URLRequest::ReferrerPolicy GetURLRequestReferrerPolicy(
|
|
cef_referrer_policy_t policy) {
|
|
base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
|
|
net::URLRequest::ReferrerPolicy net_referrer_policy =
|
|
net::URLRequest::CLEAR_REFERRER_ON_TRANSITION_FROM_SECURE_TO_INSECURE;
|
|
switch (static_cast<blink::WebReferrerPolicy>(policy)) {
|
|
case blink::WebReferrerPolicyAlways:
|
|
case blink::WebReferrerPolicyNever:
|
|
case blink::WebReferrerPolicyOrigin:
|
|
net_referrer_policy = net::URLRequest::NEVER_CLEAR_REFERRER;
|
|
break;
|
|
case blink::WebReferrerPolicyNoReferrerWhenDowngrade:
|
|
net_referrer_policy =
|
|
net::URLRequest::CLEAR_REFERRER_ON_TRANSITION_FROM_SECURE_TO_INSECURE;
|
|
break;
|
|
case blink::WebReferrerPolicyOriginWhenCrossOrigin:
|
|
net_referrer_policy =
|
|
net::URLRequest::ORIGIN_ONLY_ON_TRANSITION_CROSS_ORIGIN;
|
|
break;
|
|
case blink::WebReferrerPolicyDefault:
|
|
default:
|
|
net_referrer_policy =
|
|
command_line->HasSwitch(switches::kReducedReferrerGranularity)
|
|
? net::URLRequest::
|
|
REDUCE_REFERRER_GRANULARITY_ON_TRANSITION_CROSS_ORIGIN
|
|
: net::URLRequest::
|
|
CLEAR_REFERRER_ON_TRANSITION_FROM_SECURE_TO_INSECURE;
|
|
break;
|
|
}
|
|
return net_referrer_policy;
|
|
}
|
|
|
|
std::string GetURLRequestReferrer(const CefString& referrer_url) {
|
|
base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
|
|
if (!GURL(referrer_url.ToString()).is_valid() ||
|
|
command_line->HasSwitch(switches::kNoReferrers)) {
|
|
return std::string();
|
|
}
|
|
|
|
return referrer_url;
|
|
}
|
|
|
|
blink::WebString FilePathStringToWebString(
|
|
const base::FilePath::StringType& str) {
|
|
#if defined(OS_POSIX)
|
|
return base::WideToUTF16(base::SysNativeMBToWide(str));
|
|
#elif defined(OS_WIN)
|
|
return base::WideToUTF16(str);
|
|
#endif
|
|
}
|
|
|
|
// Read |headers| into |map|.
|
|
void GetHeaderMap(const net::HttpRequestHeaders& headers,
|
|
CefRequest::HeaderMap& map) {
|
|
map.clear();
|
|
|
|
if (headers.IsEmpty())
|
|
return;
|
|
|
|
net::HttpRequestHeaders::Iterator it(headers);
|
|
while (it.GetNext()) {
|
|
const std::string& name = it.name();
|
|
|
|
// Do not include Referer in the header map.
|
|
if (!base::LowerCaseEqualsASCII(name, kReferrerLowerCase))
|
|
map.insert(std::make_pair(name, it.value()));
|
|
};
|
|
}
|
|
|
|
// Read |request| into |map|. If a Referer value is specified populate
|
|
// |referrer|.
|
|
void GetHeaderMap(const blink::WebURLRequest& request,
|
|
CefRequest::HeaderMap& map,
|
|
CefString& referrer) {
|
|
map.clear();
|
|
|
|
class HeaderVisitor : public blink::WebHTTPHeaderVisitor {
|
|
public:
|
|
HeaderVisitor(CefRequest::HeaderMap* map, CefString* referrer)
|
|
: map_(map),
|
|
referrer_(referrer) {
|
|
}
|
|
|
|
void visitHeader(const blink::WebString& name,
|
|
const blink::WebString& value) override {
|
|
const base::string16& nameStr = name;
|
|
const base::string16& valueStr = value;
|
|
if (base::LowerCaseEqualsASCII(nameStr, kReferrerLowerCase))
|
|
*referrer_ = valueStr;
|
|
else
|
|
map_->insert(std::make_pair(nameStr, valueStr));
|
|
}
|
|
|
|
private:
|
|
CefRequest::HeaderMap* map_;
|
|
CefString* referrer_;
|
|
};
|
|
|
|
HeaderVisitor visitor(&map, &referrer);
|
|
request.visitHTTPHeaderFields(&visitor);
|
|
}
|
|
|
|
// Read |source| into |map|.
|
|
void GetHeaderMap(const CefRequest::HeaderMap& source,
|
|
CefRequest::HeaderMap& map) {
|
|
map.clear();
|
|
|
|
CefRequest::HeaderMap::const_iterator it = source.begin();
|
|
for (; it != source.end(); ++it) {
|
|
const CefString& name = it->first;
|
|
|
|
// Do not include Referer in the header map.
|
|
if (!base::LowerCaseEqualsASCII(name.ToString(), kReferrerLowerCase))
|
|
map.insert(std::make_pair(name, it->second));
|
|
}
|
|
}
|
|
|
|
// Read |map| into |request|.
|
|
void SetHeaderMap(const CefRequest::HeaderMap& map,
|
|
blink::WebURLRequest& request) {
|
|
CefRequest::HeaderMap::const_iterator it = map.begin();
|
|
for (; it != map.end(); ++it) {
|
|
request.setHTTPHeaderField(base::string16(it->first),
|
|
base::string16(it->second));
|
|
}
|
|
}
|
|
|
|
// Type used in UploadDataStream.
|
|
typedef std::vector<std::unique_ptr<net::UploadElementReader>> UploadElementReaders;
|
|
|
|
} // namespace
|
|
|
|
|
|
#define CHECK_READONLY_RETURN(val) \
|
|
if (read_only_) { \
|
|
NOTREACHED() << "object is read only"; \
|
|
return val; \
|
|
}
|
|
|
|
#define CHECK_READONLY_RETURN_VOID() \
|
|
if (read_only_) { \
|
|
NOTREACHED() << "object is read only"; \
|
|
return; \
|
|
}
|
|
|
|
#define SETBOOLFLAG(obj, flags, method, FLAG) \
|
|
obj.method((flags & (FLAG)) == (FLAG))
|
|
|
|
|
|
// CefRequest -----------------------------------------------------------------
|
|
|
|
// static
|
|
CefRefPtr<CefRequest> CefRequest::Create() {
|
|
CefRefPtr<CefRequest> request(new CefRequestImpl());
|
|
return request;
|
|
}
|
|
|
|
|
|
// CefRequestImpl -------------------------------------------------------------
|
|
|
|
CefRequestImpl::CefRequestImpl()
|
|
: read_only_(false),
|
|
track_changes_(false) {
|
|
base::AutoLock lock_scope(lock_);
|
|
Reset();
|
|
}
|
|
|
|
bool CefRequestImpl::IsReadOnly() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return read_only_;
|
|
}
|
|
|
|
CefString CefRequestImpl::GetURL() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return url_;
|
|
}
|
|
|
|
void CefRequestImpl::SetURL(const CefString& url) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
if (url_ != url) {
|
|
url_ = url;
|
|
Changed(kChangedUrl);
|
|
}
|
|
}
|
|
|
|
CefString CefRequestImpl::GetMethod() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return method_;
|
|
}
|
|
|
|
void CefRequestImpl::SetMethod(const CefString& method) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
if (method_ != method) {
|
|
method_ = method;
|
|
Changed(kChangedMethod);
|
|
}
|
|
}
|
|
|
|
void CefRequestImpl::SetReferrer(const CefString& referrer_url,
|
|
ReferrerPolicy policy) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
|
|
// Call GetAsReferrer here for consistency since the same logic will later be
|
|
// applied by URLRequest::SetReferrer().
|
|
const GURL& gurl = GURL(referrer_url.ToString()).GetAsReferrer();
|
|
if (referrer_url_ != gurl.spec() || referrer_policy_ != policy) {
|
|
referrer_url_ = gurl.spec();
|
|
referrer_policy_ = policy;
|
|
Changed(kChangedReferrer);
|
|
}
|
|
}
|
|
|
|
CefString CefRequestImpl::GetReferrerURL() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return referrer_url_;
|
|
}
|
|
|
|
CefRequestImpl::ReferrerPolicy CefRequestImpl::GetReferrerPolicy() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return referrer_policy_;
|
|
}
|
|
|
|
CefRefPtr<CefPostData> CefRequestImpl::GetPostData() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return postdata_;
|
|
}
|
|
|
|
void CefRequestImpl::SetPostData(CefRefPtr<CefPostData> postData) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
postdata_ = postData;
|
|
Changed(kChangedPostData);
|
|
}
|
|
|
|
void CefRequestImpl::GetHeaderMap(HeaderMap& headerMap) {
|
|
base::AutoLock lock_scope(lock_);
|
|
headerMap = headermap_;
|
|
}
|
|
|
|
void CefRequestImpl::SetHeaderMap(const HeaderMap& headerMap) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
::GetHeaderMap(headerMap, headermap_);
|
|
Changed(kChangedHeaderMap);
|
|
}
|
|
|
|
void CefRequestImpl::Set(const CefString& url,
|
|
const CefString& method,
|
|
CefRefPtr<CefPostData> postData,
|
|
const HeaderMap& headerMap) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
if (url_ != url) {
|
|
url_ = url;
|
|
Changed(kChangedUrl);
|
|
}
|
|
if (method_ != method) {
|
|
method_ = method;
|
|
Changed(kChangedMethod);
|
|
}
|
|
postdata_ = postData;
|
|
::GetHeaderMap(headerMap, headermap_);
|
|
Changed(kChangedPostData | kChangedHeaderMap);
|
|
}
|
|
|
|
int CefRequestImpl::GetFlags() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return flags_;
|
|
}
|
|
|
|
void CefRequestImpl::SetFlags(int flags) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
if (flags_ != flags) {
|
|
flags_ = flags;
|
|
Changed(kChangedFlags);
|
|
}
|
|
}
|
|
|
|
CefString CefRequestImpl::GetFirstPartyForCookies() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return first_party_for_cookies_;
|
|
}
|
|
|
|
void CefRequestImpl::SetFirstPartyForCookies(const CefString& url) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
if (first_party_for_cookies_ != url) {
|
|
first_party_for_cookies_ = url;
|
|
Changed(kChangedFirstPartyForCookies);
|
|
}
|
|
}
|
|
|
|
CefRequestImpl::ResourceType CefRequestImpl::GetResourceType() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return resource_type_;
|
|
}
|
|
|
|
CefRequestImpl::TransitionType CefRequestImpl::GetTransitionType() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return transition_type_;
|
|
}
|
|
|
|
uint64 CefRequestImpl::GetIdentifier() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return identifier_;
|
|
}
|
|
|
|
void CefRequestImpl::Set(net::URLRequest* request) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
|
|
Reset();
|
|
|
|
url_ = request->url().spec();
|
|
method_ = request->method();
|
|
identifier_ = request->identifier();
|
|
|
|
// URLRequest::SetReferrer ensures that we do not send username and password
|
|
// fields in the referrer.
|
|
GURL referrer(request->referrer());
|
|
|
|
// Our consumer should have made sure that this is a safe referrer. See for
|
|
// instance WebCore::FrameLoader::HideReferrer.
|
|
if (referrer.is_valid()) {
|
|
referrer_url_ = referrer.spec();
|
|
switch (request->referrer_policy()) {
|
|
case net::URLRequest::
|
|
CLEAR_REFERRER_ON_TRANSITION_FROM_SECURE_TO_INSECURE:
|
|
referrer_policy_ = REFERRER_POLICY_NO_REFERRER_WHEN_DOWNGRADE;
|
|
break;
|
|
case net::URLRequest::
|
|
REDUCE_REFERRER_GRANULARITY_ON_TRANSITION_CROSS_ORIGIN:
|
|
referrer_policy_ = REFERRER_POLICY_ORIGIN_WHEN_CROSS_ORIGIN;
|
|
break;
|
|
case net::URLRequest::ORIGIN_ONLY_ON_TRANSITION_CROSS_ORIGIN:
|
|
referrer_policy_ = REFERRER_POLICY_ORIGIN_WHEN_CROSS_ORIGIN;
|
|
break;
|
|
case net::URLRequest::NEVER_CLEAR_REFERRER:
|
|
referrer_policy_ = REFERRER_POLICY_ALWAYS;
|
|
break;
|
|
case net::URLRequest::ORIGIN:
|
|
referrer_policy_ = REFERRER_POLICY_ORIGIN;
|
|
break;
|
|
case net::URLRequest::NO_REFERRER:
|
|
referrer_policy_ = REFERRER_POLICY_NEVER;
|
|
break;
|
|
case net::URLRequest::MAX_REFERRER_POLICY:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Transfer request headers.
|
|
::GetHeaderMap(request->extra_request_headers(), headermap_);
|
|
|
|
// Transfer post data, if any.
|
|
const net::UploadDataStream* data = request->get_upload();
|
|
if (data) {
|
|
postdata_ = CefPostData::Create();
|
|
static_cast<CefPostDataImpl*>(postdata_.get())->Set(*data);
|
|
}
|
|
|
|
first_party_for_cookies_ = request->first_party_for_cookies().spec();
|
|
|
|
const content::ResourceRequestInfo* info =
|
|
content::ResourceRequestInfo::ForRequest(request);
|
|
if (info) {
|
|
resource_type_ =
|
|
static_cast<cef_resource_type_t>(info->GetResourceType());
|
|
transition_type_ =
|
|
static_cast<cef_transition_type_t>(info->GetPageTransition());
|
|
}
|
|
}
|
|
|
|
void CefRequestImpl::Get(net::URLRequest* request, bool changed_only) const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
if (ShouldSet(kChangedMethod, changed_only))
|
|
request->set_method(method_);
|
|
|
|
if (ShouldSet(kChangedReferrer, changed_only)) {
|
|
request->SetReferrer(GetURLRequestReferrer(referrer_url_));
|
|
request->set_referrer_policy(GetURLRequestReferrerPolicy(referrer_policy_));
|
|
}
|
|
|
|
if (ShouldSet(kChangedHeaderMap, changed_only)) {
|
|
net::HttpRequestHeaders headers;
|
|
headers.AddHeadersFromString(HttpHeaderUtils::GenerateHeaders(headermap_));
|
|
request->SetExtraRequestHeaders(headers);
|
|
}
|
|
|
|
if (ShouldSet(kChangedPostData, changed_only)) {
|
|
if (postdata_.get()) {
|
|
request->set_upload(base::WrapUnique(
|
|
static_cast<CefPostDataImpl*>(postdata_.get())->Get()));
|
|
} else if (request->get_upload()) {
|
|
request->set_upload(std::unique_ptr<net::UploadDataStream>());
|
|
}
|
|
}
|
|
|
|
if (!first_party_for_cookies_.empty() &&
|
|
ShouldSet(kChangedFirstPartyForCookies, changed_only)) {
|
|
request->set_first_party_for_cookies(
|
|
GURL(std::string(first_party_for_cookies_)));
|
|
}
|
|
}
|
|
|
|
void CefRequestImpl::Set(
|
|
const navigation_interception::NavigationParams& params,
|
|
bool is_main_frame) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
|
|
Reset();
|
|
|
|
url_ = params.url().spec();
|
|
method_ = params.is_post() ? "POST" : "GET";
|
|
|
|
const content::Referrer& sanitized_referrer =
|
|
content::Referrer::SanitizeForRequest(params.url(), params.referrer());
|
|
referrer_url_ = sanitized_referrer.url.spec();
|
|
referrer_policy_ =
|
|
static_cast<cef_referrer_policy_t>(sanitized_referrer.policy);
|
|
|
|
resource_type_ = is_main_frame ? RT_MAIN_FRAME : RT_SUB_FRAME;
|
|
transition_type_ =
|
|
static_cast<cef_transition_type_t>(params.transition_type());
|
|
}
|
|
|
|
void CefRequestImpl::Set(const blink::WebURLRequest& request) {
|
|
DCHECK(!request.isNull());
|
|
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
|
|
Reset();
|
|
|
|
url_ = request.url().string();
|
|
method_ = request.httpMethod();
|
|
|
|
::GetHeaderMap(request, headermap_, referrer_url_);
|
|
referrer_policy_ =
|
|
static_cast<cef_referrer_policy_t>(request.referrerPolicy());
|
|
|
|
const blink::WebHTTPBody& body = request.httpBody();
|
|
if (!body.isNull()) {
|
|
postdata_ = new CefPostDataImpl();
|
|
static_cast<CefPostDataImpl*>(postdata_.get())->Set(body);
|
|
}
|
|
|
|
first_party_for_cookies_ = request.firstPartyForCookies().string();
|
|
|
|
if (request.getCachePolicy() == blink::WebCachePolicy::BypassingCache)
|
|
flags_ |= UR_FLAG_SKIP_CACHE;
|
|
if (request.allowStoredCredentials())
|
|
flags_ |= UR_FLAG_ALLOW_CACHED_CREDENTIALS;
|
|
if (request.reportUploadProgress())
|
|
flags_ |= UR_FLAG_REPORT_UPLOAD_PROGRESS;
|
|
}
|
|
|
|
void CefRequestImpl::Get(blink::WebURLRequest& request,
|
|
int64& upload_data_size) const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
request.setURL(GURL(url_.ToString()));
|
|
request.setHTTPMethod(blink::WebString::fromUTF8(method_.ToString()));
|
|
|
|
if (!referrer_url_.empty()) {
|
|
const blink::WebString& referrer =
|
|
blink::WebSecurityPolicy::generateReferrerHeader(
|
|
static_cast<blink::WebReferrerPolicy>(referrer_policy_),
|
|
GURL(url_.ToString()),
|
|
blink::WebString::fromUTF8(referrer_url_));
|
|
if (!referrer.isEmpty()) {
|
|
request.setHTTPReferrer(
|
|
referrer,
|
|
static_cast<blink::WebReferrerPolicy>(referrer_policy_));
|
|
}
|
|
}
|
|
|
|
if (postdata_.get()) {
|
|
blink::WebHTTPBody body;
|
|
body.initialize();
|
|
static_cast<CefPostDataImpl*>(postdata_.get())->Get(body);
|
|
request.setHTTPBody(body);
|
|
|
|
if (flags_ & UR_FLAG_REPORT_UPLOAD_PROGRESS) {
|
|
// Attempt to determine the upload data size.
|
|
CefPostData::ElementVector elements;
|
|
postdata_->GetElements(elements);
|
|
if (elements.size() == 1 && elements[0]->GetType() == PDE_TYPE_BYTES) {
|
|
CefPostDataElementImpl* impl =
|
|
static_cast<CefPostDataElementImpl*>(elements[0].get());
|
|
upload_data_size = impl->GetBytesCount();
|
|
}
|
|
}
|
|
}
|
|
|
|
::SetHeaderMap(headermap_, request);
|
|
|
|
if (!first_party_for_cookies_.empty())
|
|
request.setFirstPartyForCookies(GURL(first_party_for_cookies_.ToString()));
|
|
|
|
request.setCachePolicy((flags_ & UR_FLAG_SKIP_CACHE) ?
|
|
blink::WebCachePolicy::BypassingCache :
|
|
blink::WebCachePolicy::UseProtocolCachePolicy);
|
|
|
|
SETBOOLFLAG(request, flags_, setAllowStoredCredentials,
|
|
UR_FLAG_ALLOW_CACHED_CREDENTIALS);
|
|
SETBOOLFLAG(request, flags_, setReportUploadProgress,
|
|
UR_FLAG_REPORT_UPLOAD_PROGRESS);
|
|
}
|
|
|
|
// static
|
|
void CefRequestImpl::Get(const CefMsg_LoadRequest_Params& params,
|
|
blink::WebURLRequest& request) {
|
|
request.setURL(params.url);
|
|
if (!params.method.empty())
|
|
request.setHTTPMethod(base::ASCIIToUTF16(params.method));
|
|
|
|
if (params.referrer.is_valid()) {
|
|
const blink::WebString& referrer =
|
|
blink::WebSecurityPolicy::generateReferrerHeader(
|
|
static_cast<blink::WebReferrerPolicy>(params.referrer_policy),
|
|
params.url,
|
|
blink::WebString::fromUTF8(params.referrer.spec()));
|
|
if (!referrer.isEmpty()) {
|
|
request.setHTTPReferrer(
|
|
referrer,
|
|
static_cast<blink::WebReferrerPolicy>(params.referrer_policy));
|
|
}
|
|
}
|
|
|
|
if (!params.headers.empty()) {
|
|
for (net::HttpUtil::HeadersIterator i(params.headers.begin(),
|
|
params.headers.end(), "\n");
|
|
i.GetNext(); ) {
|
|
request.addHTTPHeaderField(blink::WebString::fromUTF8(i.name()),
|
|
blink::WebString::fromUTF8(i.values()));
|
|
}
|
|
}
|
|
|
|
if (params.upload_data.get()) {
|
|
const base::string16& method = request.httpMethod();
|
|
if (method == base::ASCIIToUTF16("GET") ||
|
|
method == base::ASCIIToUTF16("HEAD")) {
|
|
request.setHTTPMethod(base::ASCIIToUTF16("POST"));
|
|
}
|
|
|
|
// The comparison performed by httpHeaderField() is case insensitive.
|
|
if (request.httpHeaderField(base::ASCIIToUTF16(
|
|
net::HttpRequestHeaders::kContentType)).length()== 0) {
|
|
request.setHTTPHeaderField(
|
|
base::ASCIIToUTF16(net::HttpRequestHeaders::kContentType),
|
|
base::ASCIIToUTF16(kApplicationFormURLEncoded));
|
|
}
|
|
|
|
blink::WebHTTPBody body;
|
|
body.initialize();
|
|
|
|
const ScopedVector<net::UploadElement>& elements =
|
|
params.upload_data->elements();
|
|
ScopedVector<net::UploadElement>::const_iterator it =
|
|
elements.begin();
|
|
for (; it != elements.end(); ++it) {
|
|
const net::UploadElement& element = **it;
|
|
if (element.type() == net::UploadElement::TYPE_BYTES) {
|
|
blink::WebData data;
|
|
data.assign(element.bytes(), element.bytes_length());
|
|
body.appendData(data);
|
|
} else if (element.type() == net::UploadElement::TYPE_FILE) {
|
|
body.appendFile(FilePathStringToWebString(element.file_path().value()));
|
|
} else {
|
|
NOTREACHED();
|
|
}
|
|
}
|
|
|
|
request.setHTTPBody(body);
|
|
}
|
|
|
|
if (params.first_party_for_cookies.is_valid())
|
|
request.setFirstPartyForCookies(params.first_party_for_cookies);
|
|
|
|
request.setCachePolicy((params.load_flags & UR_FLAG_SKIP_CACHE) ?
|
|
blink::WebCachePolicy::BypassingCache :
|
|
blink::WebCachePolicy::UseProtocolCachePolicy);
|
|
|
|
SETBOOLFLAG(request, params.load_flags, setAllowStoredCredentials,
|
|
UR_FLAG_ALLOW_CACHED_CREDENTIALS);
|
|
SETBOOLFLAG(request, params.load_flags, setReportUploadProgress,
|
|
UR_FLAG_REPORT_UPLOAD_PROGRESS);
|
|
}
|
|
|
|
void CefRequestImpl::Get(CefNavigateParams& params) const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
params.url = GURL(url_.ToString());
|
|
params.method = method_;
|
|
|
|
// Referrer policy will be applied later in the request pipeline.
|
|
params.referrer.url = GURL(referrer_url_.ToString());
|
|
params.referrer.policy =
|
|
static_cast<blink::WebReferrerPolicy>(referrer_policy_);
|
|
|
|
if (!headermap_.empty())
|
|
params.headers = HttpHeaderUtils::GenerateHeaders(headermap_);
|
|
|
|
if (postdata_) {
|
|
CefPostDataImpl* impl = static_cast<CefPostDataImpl*>(postdata_.get());
|
|
params.upload_data = new net::UploadData();
|
|
impl->Get(*params.upload_data.get());
|
|
}
|
|
|
|
params.first_party_for_cookies = GURL(first_party_for_cookies_.ToString());
|
|
params.load_flags = flags_;
|
|
}
|
|
|
|
void CefRequestImpl::Get(net::URLFetcher& fetcher,
|
|
int64& upload_data_size) const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
if (!referrer_url_.empty()) {
|
|
fetcher.SetReferrer(GetURLRequestReferrer(referrer_url_));
|
|
fetcher.SetReferrerPolicy(GetURLRequestReferrerPolicy(referrer_policy_));
|
|
}
|
|
|
|
CefRequest::HeaderMap headerMap = headermap_;
|
|
|
|
std::string content_type;
|
|
|
|
// Extract the Content-Type header value.
|
|
{
|
|
HeaderMap::iterator it = headerMap.begin();
|
|
for (; it != headerMap.end(); ++it) {
|
|
if (base::LowerCaseEqualsASCII(it->first.ToString(),
|
|
kContentTypeLowerCase)) {
|
|
content_type = it->second;
|
|
headerMap.erase(it);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
fetcher.SetExtraRequestHeaders(
|
|
HttpHeaderUtils::GenerateHeaders(headerMap));
|
|
|
|
if (postdata_.get()) {
|
|
CefPostData::ElementVector elements;
|
|
postdata_->GetElements(elements);
|
|
if (elements.size() == 1) {
|
|
// Default to URL encoding if not specified.
|
|
if (content_type.empty())
|
|
content_type = kApplicationFormURLEncoded;
|
|
|
|
CefPostDataElementImpl* impl =
|
|
static_cast<CefPostDataElementImpl*>(elements[0].get());
|
|
|
|
switch (elements[0]->GetType()) {
|
|
case PDE_TYPE_BYTES: {
|
|
const size_t size = impl->GetBytesCount();
|
|
if (flags_ & UR_FLAG_REPORT_UPLOAD_PROGRESS) {
|
|
// Return the upload data size.
|
|
upload_data_size = size;
|
|
}
|
|
fetcher.SetUploadData(
|
|
content_type,
|
|
std::string(static_cast<char*>(impl->GetBytes()), size));
|
|
break;
|
|
}
|
|
case PDE_TYPE_FILE:
|
|
fetcher.SetUploadFilePath(
|
|
content_type,
|
|
base::FilePath(impl->GetFile()),
|
|
0, std::numeric_limits<uint64_t>::max(),
|
|
GetFileTaskRunner());
|
|
break;
|
|
case PDE_TYPE_EMPTY:
|
|
break;
|
|
}
|
|
} else if (elements.size() > 1) {
|
|
NOTIMPLEMENTED() << " multi-part form data is not supported";
|
|
}
|
|
}
|
|
|
|
if (!first_party_for_cookies_.empty())
|
|
fetcher.SetInitiatorURL(GURL(first_party_for_cookies_.ToString()));
|
|
|
|
if (flags_ & UR_FLAG_NO_RETRY_ON_5XX)
|
|
fetcher.SetAutomaticallyRetryOn5xx(false);
|
|
|
|
int load_flags = 0;
|
|
|
|
if (flags_ & UR_FLAG_SKIP_CACHE)
|
|
load_flags |= net::LOAD_BYPASS_CACHE;
|
|
|
|
if (!(flags_ & UR_FLAG_ALLOW_CACHED_CREDENTIALS)) {
|
|
load_flags |= net::LOAD_DO_NOT_SEND_AUTH_DATA;
|
|
load_flags |= net::LOAD_DO_NOT_SEND_COOKIES;
|
|
load_flags |= net::LOAD_DO_NOT_SAVE_COOKIES;
|
|
}
|
|
|
|
fetcher.SetLoadFlags(load_flags);
|
|
}
|
|
|
|
void CefRequestImpl::SetReadOnly(bool read_only) {
|
|
base::AutoLock lock_scope(lock_);
|
|
if (read_only_ == read_only)
|
|
return;
|
|
|
|
read_only_ = read_only;
|
|
|
|
if (postdata_.get())
|
|
static_cast<CefPostDataImpl*>(postdata_.get())->SetReadOnly(read_only);
|
|
}
|
|
|
|
void CefRequestImpl::SetTrackChanges(bool track_changes) {
|
|
base::AutoLock lock_scope(lock_);
|
|
if (track_changes_ == track_changes)
|
|
return;
|
|
|
|
track_changes_ = track_changes;
|
|
changes_ = kChangedNone;
|
|
|
|
if (postdata_.get()) {
|
|
static_cast<CefPostDataImpl*>(postdata_.get())->
|
|
SetTrackChanges(track_changes);
|
|
}
|
|
}
|
|
|
|
uint8_t CefRequestImpl::GetChanges() const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
uint8_t changes = changes_;
|
|
if (postdata_.get() &&
|
|
static_cast<CefPostDataImpl*>(postdata_.get())->HasChanges()) {
|
|
changes |= kChangedPostData;
|
|
}
|
|
return changes;
|
|
}
|
|
|
|
void CefRequestImpl::Changed(uint8_t changes) {
|
|
lock_.AssertAcquired();
|
|
if (track_changes_)
|
|
changes_ |= changes;
|
|
}
|
|
|
|
bool CefRequestImpl::ShouldSet(uint8_t changes, bool changed_only) const {
|
|
lock_.AssertAcquired();
|
|
|
|
// Always change if changes are not being tracked.
|
|
if (!track_changes_)
|
|
return true;
|
|
|
|
// Always change if changed-only was not requested.
|
|
if (!changed_only)
|
|
return true;
|
|
|
|
// Change if the |changes| bit flag has been set.
|
|
if ((changes_ & changes) == changes)
|
|
return true;
|
|
|
|
if ((changes & kChangedPostData) == kChangedPostData) {
|
|
// Change if the post data object was modified directly.
|
|
if (postdata_.get() &&
|
|
static_cast<CefPostDataImpl*>(postdata_.get())->HasChanges()) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void CefRequestImpl::Reset() {
|
|
lock_.AssertAcquired();
|
|
DCHECK(!read_only_);
|
|
|
|
url_.clear();
|
|
method_ = "GET";
|
|
referrer_url_.clear();
|
|
referrer_policy_ = REFERRER_POLICY_DEFAULT;
|
|
postdata_ = NULL;
|
|
headermap_.clear();
|
|
resource_type_ = RT_SUB_RESOURCE;
|
|
transition_type_ = TT_EXPLICIT;
|
|
identifier_ = 0U;
|
|
flags_ = UR_FLAG_NONE;
|
|
first_party_for_cookies_.clear();
|
|
|
|
changes_ = kChangedNone;
|
|
}
|
|
|
|
// CefPostData ----------------------------------------------------------------
|
|
|
|
// static
|
|
CefRefPtr<CefPostData> CefPostData::Create() {
|
|
CefRefPtr<CefPostData> postdata(new CefPostDataImpl());
|
|
return postdata;
|
|
}
|
|
|
|
|
|
// CefPostDataImpl ------------------------------------------------------------
|
|
|
|
CefPostDataImpl::CefPostDataImpl()
|
|
: read_only_(false),
|
|
has_excluded_elements_(false),
|
|
track_changes_(false),
|
|
has_changes_(false) {
|
|
}
|
|
|
|
bool CefPostDataImpl::IsReadOnly() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return read_only_;
|
|
}
|
|
|
|
bool CefPostDataImpl::HasExcludedElements() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return has_excluded_elements_;
|
|
}
|
|
|
|
size_t CefPostDataImpl::GetElementCount() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return elements_.size();
|
|
}
|
|
|
|
void CefPostDataImpl::GetElements(ElementVector& elements) {
|
|
base::AutoLock lock_scope(lock_);
|
|
elements = elements_;
|
|
}
|
|
|
|
bool CefPostDataImpl::RemoveElement(CefRefPtr<CefPostDataElement> element) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN(false);
|
|
|
|
ElementVector::iterator it = elements_.begin();
|
|
for (; it != elements_.end(); ++it) {
|
|
if (it->get() == element.get()) {
|
|
elements_.erase(it);
|
|
Changed();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool CefPostDataImpl::AddElement(CefRefPtr<CefPostDataElement> element) {
|
|
bool found = false;
|
|
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN(false);
|
|
|
|
// check that the element isn't already in the list before adding
|
|
ElementVector::const_iterator it = elements_.begin();
|
|
for (; it != elements_.end(); ++it) {
|
|
if (it->get() == element.get()) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found) {
|
|
elements_.push_back(element);
|
|
Changed();
|
|
}
|
|
|
|
return !found;
|
|
}
|
|
|
|
void CefPostDataImpl::RemoveElements() {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
elements_.clear();
|
|
Changed();
|
|
}
|
|
|
|
void CefPostDataImpl::Set(const net::UploadData& data) {
|
|
{
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
}
|
|
|
|
CefRefPtr<CefPostDataElement> postelem;
|
|
|
|
const ScopedVector<net::UploadElement>& elements = data.elements();
|
|
ScopedVector<net::UploadElement>::const_iterator it = elements.begin();
|
|
for (; it != elements.end(); ++it) {
|
|
postelem = CefPostDataElement::Create();
|
|
static_cast<CefPostDataElementImpl*>(postelem.get())->Set(**it);
|
|
AddElement(postelem);
|
|
}
|
|
}
|
|
|
|
void CefPostDataImpl::Set(const net::UploadDataStream& data_stream) {
|
|
{
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
}
|
|
|
|
CefRefPtr<CefPostDataElement> postelem;
|
|
|
|
const UploadElementReaders* elements = data_stream.GetElementReaders();
|
|
if (elements) {
|
|
UploadElementReaders::const_iterator it = elements->begin();
|
|
for (; it != elements->end(); ++it) {
|
|
postelem = CefPostDataElement::Create();
|
|
static_cast<CefPostDataElementImpl*>(postelem.get())->Set(**it);
|
|
if (postelem->GetType() != PDE_TYPE_EMPTY)
|
|
AddElement(postelem);
|
|
else if (!has_excluded_elements_)
|
|
has_excluded_elements_ = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
void CefPostDataImpl::Get(net::UploadData& data) const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
ScopedVector<net::UploadElement> data_elements;
|
|
ElementVector::const_iterator it = elements_.begin();
|
|
for (; it != elements_.end(); ++it) {
|
|
net::UploadElement* element = new net::UploadElement();
|
|
static_cast<CefPostDataElementImpl*>(it->get())->Get(*element);
|
|
data_elements.push_back(element);
|
|
}
|
|
data.swap_elements(&data_elements);
|
|
}
|
|
|
|
net::UploadDataStream* CefPostDataImpl::Get() const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
UploadElementReaders element_readers;
|
|
ElementVector::const_iterator it = elements_.begin();
|
|
for (; it != elements_.end(); ++it) {
|
|
element_readers.push_back(base::WrapUnique(
|
|
static_cast<CefPostDataElementImpl*>(it->get())->Get()));
|
|
}
|
|
|
|
return new net::ElementsUploadDataStream(std::move(element_readers), 0);
|
|
}
|
|
|
|
void CefPostDataImpl::Set(const blink::WebHTTPBody& data) {
|
|
{
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
}
|
|
|
|
CefRefPtr<CefPostDataElement> postelem;
|
|
blink::WebHTTPBody::Element element;
|
|
size_t size = data.elementCount();
|
|
for (size_t i = 0; i < size; ++i) {
|
|
if (data.elementAt(i, element)) {
|
|
postelem = CefPostDataElement::Create();
|
|
static_cast<CefPostDataElementImpl*>(postelem.get())->Set(element);
|
|
AddElement(postelem);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CefPostDataImpl::Get(blink::WebHTTPBody& data) const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
blink::WebHTTPBody::Element element;
|
|
ElementVector::const_iterator it = elements_.begin();
|
|
for (; it != elements_.end(); ++it) {
|
|
static_cast<CefPostDataElementImpl*>(it->get())->Get(element);
|
|
if (element.type == blink::WebHTTPBody::Element::TypeData) {
|
|
data.appendData(element.data);
|
|
} else if (element.type == blink::WebHTTPBody::Element::TypeFile) {
|
|
data.appendFile(element.filePath);
|
|
} else {
|
|
NOTREACHED();
|
|
}
|
|
}
|
|
}
|
|
|
|
void CefPostDataImpl::SetReadOnly(bool read_only) {
|
|
base::AutoLock lock_scope(lock_);
|
|
if (read_only_ == read_only)
|
|
return;
|
|
|
|
read_only_ = read_only;
|
|
|
|
ElementVector::const_iterator it = elements_.begin();
|
|
for (; it != elements_.end(); ++it) {
|
|
static_cast<CefPostDataElementImpl*>(it->get())->SetReadOnly(read_only);
|
|
}
|
|
}
|
|
|
|
void CefPostDataImpl::SetTrackChanges(bool track_changes) {
|
|
base::AutoLock lock_scope(lock_);
|
|
if (track_changes_ == track_changes)
|
|
return;
|
|
|
|
track_changes_ = track_changes;
|
|
has_changes_ = false;
|
|
|
|
ElementVector::const_iterator it = elements_.begin();
|
|
for (; it != elements_.end(); ++it) {
|
|
static_cast<CefPostDataElementImpl*>(it->get())->
|
|
SetTrackChanges(track_changes);
|
|
}
|
|
}
|
|
|
|
bool CefPostDataImpl::HasChanges() const {
|
|
base::AutoLock lock_scope(lock_);
|
|
if (has_changes_)
|
|
return true;
|
|
|
|
ElementVector::const_iterator it = elements_.begin();
|
|
for (; it != elements_.end(); ++it) {
|
|
if (static_cast<CefPostDataElementImpl*>(it->get())->HasChanges())
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void CefPostDataImpl::Changed() {
|
|
lock_.AssertAcquired();
|
|
if (track_changes_ && !has_changes_)
|
|
has_changes_ = true;
|
|
}
|
|
|
|
|
|
// CefPostDataElement ---------------------------------------------------------
|
|
|
|
// static
|
|
CefRefPtr<CefPostDataElement> CefPostDataElement::Create() {
|
|
CefRefPtr<CefPostDataElement> element(new CefPostDataElementImpl());
|
|
return element;
|
|
}
|
|
|
|
|
|
// CefPostDataElementImpl -----------------------------------------------------
|
|
|
|
CefPostDataElementImpl::CefPostDataElementImpl()
|
|
: type_(PDE_TYPE_EMPTY),
|
|
read_only_(false),
|
|
track_changes_(false),
|
|
has_changes_(false) {
|
|
memset(&data_, 0, sizeof(data_));
|
|
}
|
|
|
|
CefPostDataElementImpl::~CefPostDataElementImpl() {
|
|
Cleanup();
|
|
}
|
|
|
|
bool CefPostDataElementImpl::IsReadOnly() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return read_only_;
|
|
}
|
|
|
|
void CefPostDataElementImpl::SetToEmpty() {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
|
|
Cleanup();
|
|
Changed();
|
|
}
|
|
|
|
void CefPostDataElementImpl::SetToFile(const CefString& fileName) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
|
|
// Clear any data currently in the element
|
|
Cleanup();
|
|
|
|
// Assign the new data
|
|
type_ = PDE_TYPE_FILE;
|
|
cef_string_copy(fileName.c_str(), fileName.length(), &data_.filename);
|
|
|
|
Changed();
|
|
}
|
|
|
|
void CefPostDataElementImpl::SetToBytes(size_t size, const void* bytes) {
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
|
|
// Clear any data currently in the element
|
|
Cleanup();
|
|
|
|
// Assign the new data
|
|
void* data = malloc(size);
|
|
DCHECK(data != NULL);
|
|
if (data == NULL)
|
|
return;
|
|
|
|
memcpy(data, bytes, size);
|
|
|
|
type_ = PDE_TYPE_BYTES;
|
|
data_.bytes.bytes = data;
|
|
data_.bytes.size = size;
|
|
|
|
Changed();
|
|
}
|
|
|
|
CefPostDataElement::Type CefPostDataElementImpl::GetType() {
|
|
base::AutoLock lock_scope(lock_);
|
|
return type_;
|
|
}
|
|
|
|
CefString CefPostDataElementImpl::GetFile() {
|
|
base::AutoLock lock_scope(lock_);
|
|
DCHECK(type_ == PDE_TYPE_FILE);
|
|
CefString filename;
|
|
if (type_ == PDE_TYPE_FILE)
|
|
filename.FromString(data_.filename.str, data_.filename.length, false);
|
|
return filename;
|
|
}
|
|
|
|
size_t CefPostDataElementImpl::GetBytesCount() {
|
|
base::AutoLock lock_scope(lock_);
|
|
DCHECK(type_ == PDE_TYPE_BYTES);
|
|
size_t size = 0;
|
|
if (type_ == PDE_TYPE_BYTES)
|
|
size = data_.bytes.size;
|
|
return size;
|
|
}
|
|
|
|
size_t CefPostDataElementImpl::GetBytes(size_t size, void* bytes) {
|
|
base::AutoLock lock_scope(lock_);
|
|
DCHECK(type_ == PDE_TYPE_BYTES);
|
|
size_t rv = 0;
|
|
if (type_ == PDE_TYPE_BYTES) {
|
|
rv = (size < data_.bytes.size ? size : data_.bytes.size);
|
|
memcpy(bytes, data_.bytes.bytes, rv);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
void CefPostDataElementImpl::Set(const net::UploadElement& element) {
|
|
{
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
}
|
|
|
|
if (element.type() == net::UploadElement::TYPE_BYTES) {
|
|
SetToBytes(element.bytes_length(), element.bytes());
|
|
} else if (element.type() == net::UploadElement::TYPE_FILE) {
|
|
SetToFile(element.file_path().value());
|
|
} else {
|
|
NOTREACHED();
|
|
}
|
|
}
|
|
|
|
void CefPostDataElementImpl::Set(
|
|
const net::UploadElementReader& element_reader) {
|
|
{
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
}
|
|
|
|
const net::UploadBytesElementReader* bytes_reader =
|
|
element_reader.AsBytesReader();
|
|
if (bytes_reader) {
|
|
SetToBytes(bytes_reader->length(), bytes_reader->bytes());
|
|
return;
|
|
}
|
|
|
|
const net::UploadFileElementReader* file_reader =
|
|
element_reader.AsFileReader();
|
|
if (file_reader) {
|
|
SetToFile(file_reader->path().value());
|
|
return;
|
|
}
|
|
|
|
// Chunked uploads cannot currently be represented.
|
|
SetToEmpty();
|
|
}
|
|
|
|
void CefPostDataElementImpl::Get(net::UploadElement& element) const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
if (type_ == PDE_TYPE_BYTES) {
|
|
element.SetToBytes(static_cast<char*>(data_.bytes.bytes), data_.bytes.size);
|
|
} else if (type_ == PDE_TYPE_FILE) {
|
|
base::FilePath path = base::FilePath(CefString(&data_.filename));
|
|
element.SetToFilePath(path);
|
|
} else {
|
|
NOTREACHED();
|
|
}
|
|
}
|
|
|
|
net::UploadElementReader* CefPostDataElementImpl::Get() const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
if (type_ == PDE_TYPE_BYTES) {
|
|
net::UploadElement* element = new net::UploadElement();
|
|
element->SetToBytes(static_cast<char*>(data_.bytes.bytes),
|
|
data_.bytes.size);
|
|
return new BytesElementReader(base::WrapUnique(element));
|
|
} else if (type_ == PDE_TYPE_FILE) {
|
|
net::UploadElement* element = new net::UploadElement();
|
|
base::FilePath path = base::FilePath(CefString(&data_.filename));
|
|
element->SetToFilePath(path);
|
|
return new FileElementReader(base::WrapUnique(element));
|
|
} else {
|
|
NOTREACHED();
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
void CefPostDataElementImpl::Set(const blink::WebHTTPBody::Element& element) {
|
|
{
|
|
base::AutoLock lock_scope(lock_);
|
|
CHECK_READONLY_RETURN_VOID();
|
|
}
|
|
|
|
if (element.type == blink::WebHTTPBody::Element::TypeData) {
|
|
SetToBytes(element.data.size(),
|
|
static_cast<const void*>(element.data.data()));
|
|
} else if (element.type == blink::WebHTTPBody::Element::TypeFile) {
|
|
SetToFile(base::string16(element.filePath));
|
|
} else {
|
|
NOTREACHED();
|
|
}
|
|
}
|
|
|
|
void CefPostDataElementImpl::Get(blink::WebHTTPBody::Element& element) const {
|
|
base::AutoLock lock_scope(lock_);
|
|
|
|
if (type_ == PDE_TYPE_BYTES) {
|
|
element.type = blink::WebHTTPBody::Element::TypeData;
|
|
element.data.assign(
|
|
static_cast<char*>(data_.bytes.bytes), data_.bytes.size);
|
|
} else if (type_ == PDE_TYPE_FILE) {
|
|
element.type = blink::WebHTTPBody::Element::TypeFile;
|
|
element.filePath.assign(base::string16(CefString(&data_.filename)));
|
|
} else {
|
|
NOTREACHED();
|
|
}
|
|
}
|
|
|
|
void CefPostDataElementImpl::SetReadOnly(bool read_only) {
|
|
base::AutoLock lock_scope(lock_);
|
|
if (read_only_ == read_only)
|
|
return;
|
|
|
|
read_only_ = read_only;
|
|
}
|
|
|
|
void CefPostDataElementImpl::SetTrackChanges(bool track_changes) {
|
|
base::AutoLock lock_scope(lock_);
|
|
if (track_changes_ == track_changes)
|
|
return;
|
|
|
|
track_changes_ = track_changes;
|
|
has_changes_ = false;
|
|
}
|
|
|
|
bool CefPostDataElementImpl::HasChanges() const {
|
|
base::AutoLock lock_scope(lock_);
|
|
return has_changes_;
|
|
}
|
|
|
|
void CefPostDataElementImpl::Changed() {
|
|
lock_.AssertAcquired();
|
|
if (track_changes_ && !has_changes_)
|
|
has_changes_ = true;
|
|
}
|
|
|
|
void CefPostDataElementImpl::Cleanup() {
|
|
if (type_ == PDE_TYPE_EMPTY)
|
|
return;
|
|
|
|
if (type_ == PDE_TYPE_BYTES)
|
|
free(data_.bytes.bytes);
|
|
else if (type_ == PDE_TYPE_FILE)
|
|
cef_string_clear(&data_.filename);
|
|
type_ = PDE_TYPE_EMPTY;
|
|
memset(&data_, 0, sizeof(data_));
|
|
}
|