electron/atom/browser/api/atom_api_url_request.cc

264 lines
6.8 KiB
C++
Raw Normal View History

// Copyright (c) 2013 GitHub, Inc.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#include "atom/browser/api/atom_api_url_request.h"
#include "atom/browser/api/atom_api_session.h"
#include "native_mate/dictionary.h"
#include "atom/browser/net/atom_url_request.h"
#include "atom/common/node_includes.h"
2016-09-21 15:35:03 +00:00
#include "atom/common/native_mate_converters/net_converter.h"
#include "atom/common/native_mate_converters/string16_converter.h"
#include "atom/common/native_mate_converters/callback.h"
namespace {
const char* const kResponse = "response";
const char* const kData = "data";
const char* const kEnd = "end";
}
namespace mate {
template<>
2016-09-21 15:35:03 +00:00
struct Converter<scoped_refptr<const net::HttpResponseHeaders>> {
2016-09-26 12:59:53 +00:00
static v8::Local<v8::Value> ToV8(
v8::Isolate* isolate,
scoped_refptr<const net::HttpResponseHeaders> val) {
mate::Dictionary dict = mate::Dictionary::CreateEmpty(isolate);
if (val) {
size_t iter = 0;
std::string name;
std::string value;
while (val->EnumerateHeaderLines(&iter, &name, &value)) {
dict.Set(name, value);
}
}
return dict.GetHandle();
}
};
template<>
2016-09-21 15:35:03 +00:00
struct Converter<scoped_refptr<const net::IOBufferWithSize>> {
static v8::Local<v8::Value> ToV8(
2016-09-26 12:59:53 +00:00
v8::Isolate* isolate,
scoped_refptr<const net::IOBufferWithSize> buffer) {
return node::Buffer::Copy(isolate,
buffer->data(),
buffer->size()).ToLocalChecked();
}
2016-09-26 12:03:49 +00:00
2016-09-26 12:59:53 +00:00
static bool FromV8(
v8::Isolate* isolate,
v8::Local<v8::Value> val,
scoped_refptr<const net::IOBufferWithSize>* out) {
2016-09-26 12:03:49 +00:00
auto size = node::Buffer::Length(val);
if (size == 0) {
// Support conversoin from empty buffer. A use case is
// a GET request without body.
// Since zero-sized IOBuffer(s) are not supported, we set the
// out pointer to null.
*out = nullptr;
return true;
}
auto data = node::Buffer::Data(val);
if (!data) {
// This is an error as size is positif but data is null.
return false;
}
auto io_buffer = new net::IOBufferWithSize(size);
if (!io_buffer) {
// Assuming allocation failed.
return false;
}
// We do a deep copy. We could have used Buffer's internal memory
// but that is much more complicated to be properly handled.
memcpy(io_buffer->data(), data, size);
*out = io_buffer;
return true;
}
};
}
namespace atom {
namespace api {
2016-09-26 12:59:53 +00:00
URLRequest::URLRequest(v8::Isolate* isolate, v8::Local<v8::Object> wrapper)
: weak_ptr_factory_(this) {
2016-09-26 12:59:53 +00:00
InitWith(isolate, wrapper);
}
URLRequest::~URLRequest() {
}
// static
mate::WrappableBase* URLRequest::New(mate::Arguments* args) {
v8::Local<v8::Object> options;
args->GetNext(&options);
mate::Dictionary dict(args->isolate(), options);
std::string url;
dict.Get("url", &url);
std::string method;
dict.Get("method", &method);
std::string session_name;
dict.Get("session", &session_name);
auto session = Session::FromPartition(args->isolate(), session_name);
auto browser_context = session->browser_context();
auto api_url_request = new URLRequest(args->isolate(), args->GetThis());
auto weak_ptr = api_url_request->weak_ptr_factory_.GetWeakPtr();
2016-09-21 15:35:03 +00:00
auto atom_url_request = AtomURLRequest::Create(
browser_context,
method,
url,
weak_ptr);
2016-09-21 15:35:03 +00:00
api_url_request->atom_request_ = atom_url_request;
return api_url_request;
}
// static
void URLRequest::BuildPrototype(v8::Isolate* isolate,
v8::Local<v8::FunctionTemplate> prototype) {
prototype->SetClassName(mate::StringToV8(isolate, "URLRequest"));
mate::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
// Request API
.MakeDestroyable()
2016-09-26 12:03:49 +00:00
.SetMethod("_writeBuffer", &URLRequest::WriteBuffer)
.SetMethod("abort", &URLRequest::Abort)
2016-09-21 15:35:03 +00:00
.SetMethod("_setHeader", &URLRequest::SetHeader)
.SetMethod("_getHeader", &URLRequest::GetHeader)
.SetMethod("_removaHeader", &URLRequest::RemoveHeader)
// Response APi
2016-09-26 12:59:53 +00:00
.SetProperty("_statusCode", &URLRequest::StatusCode)
.SetProperty("_statusMessage", &URLRequest::StatusMessage)
.SetProperty("_rawResponseHeaders", &URLRequest::RawResponseHeaders)
.SetProperty("_httpVersionMajor", &URLRequest::ResponseHttpVersionMajor)
.SetProperty("_httpVersionMinor", &URLRequest::ResponseHttpVersionMinor);
}
2016-09-26 12:59:53 +00:00
bool URLRequest::WriteBuffer(
scoped_refptr<const net::IOBufferWithSize> buffer,
bool is_last) {
return atom_request_->WriteBuffer(buffer, is_last);
}
void URLRequest::Abort() {
2016-09-21 15:35:03 +00:00
atom_request_->Abort();
}
2016-09-26 12:59:53 +00:00
bool URLRequest::SetHeader(const std::string& name,
const std::string& value) {
2016-09-26 12:03:49 +00:00
if (!net::HttpUtil::IsValidHeaderName(name)) {
return false;
}
if (!net::HttpUtil::IsValidHeaderValue(value)) {
return false;
}
2016-09-21 15:35:03 +00:00
atom_request_->SetHeader(name, value);
2016-09-26 12:03:49 +00:00
return true;
}
2016-09-21 15:35:03 +00:00
std::string URLRequest::GetHeader(const std::string& name) {
return atom_request_->GetHeader(name);
}
2016-09-21 15:35:03 +00:00
void URLRequest::RemoveHeader(const std::string& name) {
atom_request_->RemoveHeader(name);
}
2016-09-21 15:35:03 +00:00
void URLRequest::OnAuthenticationRequired(
2016-09-26 12:59:53 +00:00
scoped_refptr<const net::AuthChallengeInfo> auth_info) {
2016-09-21 15:35:03 +00:00
EmitRequestEvent(
"login",
auth_info.get(),
base::Bind(&AtomURLRequest::PassLoginInformation, atom_request_));
}
void URLRequest::OnResponseStarted() {
EmitRequestEvent("response");
}
2016-09-21 15:35:03 +00:00
void URLRequest::OnResponseData(
2016-09-26 12:59:53 +00:00
scoped_refptr<const net::IOBufferWithSize> buffer) {
if (!buffer || !buffer->data() || !buffer->size()) {
return;
}
EmitResponseEvent("data", buffer);
}
void URLRequest::OnResponseCompleted() {
EmitResponseEvent("end");
2016-09-26 12:03:49 +00:00
unpin();
atom_request_ = nullptr;
}
2016-09-21 15:35:03 +00:00
int URLRequest::StatusCode() const {
if (auto response_headers = atom_request_->GetResponseHeaders()) {
return response_headers->response_code();
}
return -1;
}
2016-09-21 15:35:03 +00:00
std::string URLRequest::StatusMessage() const {
std::string result;
2016-09-21 15:35:03 +00:00
if (auto response_headers = atom_request_->GetResponseHeaders()) {
result = response_headers->GetStatusText();
}
return result;
}
2016-09-26 12:59:53 +00:00
scoped_refptr<const net::HttpResponseHeaders>
URLRequest::RawResponseHeaders() const {
2016-09-21 15:35:03 +00:00
return atom_request_->GetResponseHeaders();
}
2016-09-21 15:35:03 +00:00
uint32_t URLRequest::ResponseHttpVersionMajor() const {
if (auto response_headers = atom_request_->GetResponseHeaders()) {
return response_headers->GetHttpVersion().major_value();
}
return 0;
}
2016-09-21 15:35:03 +00:00
uint32_t URLRequest::ResponseHttpVersionMinor() const {
if (auto response_headers = atom_request_->GetResponseHeaders()) {
return response_headers->GetHttpVersion().minor_value();
}
return 0;
}
void URLRequest::pin() {
if (wrapper_.IsEmpty()) {
wrapper_.Reset(isolate(), GetWrapper());
}
}
void URLRequest::unpin() {
wrapper_.Reset();
}
} // namespace mate
} // namepsace mate