electron/atom/browser/net/url_request_fetch_job.cc

199 lines
6 KiB
C++
Raw Normal View History

2015-05-29 15:54:00 +00:00
// Copyright (c) 2015 GitHub, Inc.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#include "atom/browser/net/url_request_fetch_job.h"
2015-06-17 01:31:33 +00:00
#include <algorithm>
2015-06-17 03:34:47 +00:00
#include <string>
2015-05-29 15:54:00 +00:00
#include "base/strings/string_util.h"
#include "base/thread_task_runner_handle.h"
2015-05-29 15:54:00 +00:00
#include "net/base/io_buffer.h"
#include "net/base/net_errors.h"
2015-06-17 01:31:33 +00:00
#include "net/http/http_response_headers.h"
#include "net/url_request/url_fetcher.h"
#include "net/url_request/url_fetcher_response_writer.h"
#include "net/url_request/url_request_context_builder.h"
2015-05-29 15:54:00 +00:00
#include "net/url_request/url_request_status.h"
namespace atom {
2015-06-17 01:31:33 +00:00
namespace {
// Convert string to RequestType.
net::URLFetcher::RequestType GetRequestType(const std::string& raw) {
std::string method = StringToUpperASCII(raw);
if (method.empty() || method == "GET")
return net::URLFetcher::GET;
else if (method == "POST")
return net::URLFetcher::POST;
else if (method == "HEAD")
return net::URLFetcher::HEAD;
else if (method == "DELETE")
return net::URLFetcher::DELETE_REQUEST;
else if (method == "PUT")
return net::URLFetcher::PUT;
else if (method == "PATCH")
return net::URLFetcher::PATCH;
else // Use "GET" as fallback.
return net::URLFetcher::GET;
}
2015-06-17 01:31:33 +00:00
// Pipe the response writer back to URLRequestFetchJob.
class ResponsePiper : public net::URLFetcherResponseWriter {
public:
explicit ResponsePiper(URLRequestFetchJob* job)
: first_write_(true), job_(job) {}
// net::URLFetcherResponseWriter:
int Initialize(const net::CompletionCallback& callback) override {
return net::OK;
}
int Write(net::IOBuffer* buffer,
int num_bytes,
const net::CompletionCallback& callback) override {
if (first_write_) {
// The URLFetcherResponseWriter doesn't have an event when headers have
// been read, so we have to emulate by hooking to first write event.
job_->HeadersCompleted();
first_write_ = false;
}
2015-06-17 02:19:58 +00:00
return job_->DataAvailable(buffer, num_bytes);
2015-06-17 01:31:33 +00:00
}
int Finish(const net::CompletionCallback& callback) override {
return net::OK;
}
private:
bool first_write_;
URLRequestFetchJob* job_;
DISALLOW_COPY_AND_ASSIGN(ResponsePiper);
};
} // namespace
2015-05-29 15:54:00 +00:00
URLRequestFetchJob::URLRequestFetchJob(
scoped_refptr<net::URLRequestContextGetter> request_context_getter,
2015-05-29 15:54:00 +00:00
net::URLRequest* request,
net::NetworkDelegate* network_delegate,
const GURL& url,
2015-06-17 03:20:09 +00:00
const std::string& method,
const std::string& referrer)
2015-05-29 15:54:00 +00:00
: net::URLRequestJob(request, network_delegate),
pending_buffer_size_(0) {
// Use |request|'s method if |method| is not specified.
net::URLFetcher::RequestType request_type;
if (method.empty())
request_type = GetRequestType(request->method());
else
request_type = GetRequestType(method);
fetcher_.reset(net::URLFetcher::Create(url, request_type, this));
// Use request context if provided else create one.
if (request_context_getter)
fetcher_->SetRequestContext(request_context_getter.get());
else
fetcher_->SetRequestContext(GetRequestContext());
fetcher_->SaveResponseWithWriter(make_scoped_ptr(new ResponsePiper(this)));
2015-06-17 03:20:09 +00:00
// Use |request|'s referrer if |referrer| is not specified.
2015-06-17 03:34:47 +00:00
if (referrer.empty()) {
2015-06-17 03:20:09 +00:00
fetcher_->SetReferrer(request->referrer());
2015-06-17 03:34:47 +00:00
} else {
2015-06-17 03:20:09 +00:00
fetcher_->SetReferrer(referrer);
2015-06-17 03:34:47 +00:00
}
2015-06-17 03:30:31 +00:00
// Use |request|'s headers.
fetcher_->SetExtraRequestHeaders(request->extra_request_headers().ToString());
}
2015-06-17 01:31:33 +00:00
net::URLRequestContextGetter* URLRequestFetchJob::GetRequestContext() {
if (!url_request_context_getter_.get()) {
auto task_runner = base::ThreadTaskRunnerHandle::Get();
net::URLRequestContextBuilder builder;
builder.set_proxy_service(net::ProxyService::CreateDirect());
url_request_context_getter_ =
new net::TrivialURLRequestContextGetter(builder.Build(), task_runner);
}
return url_request_context_getter_.get();
}
2015-06-17 01:31:33 +00:00
void URLRequestFetchJob::HeadersCompleted() {
response_info_.reset(new net::HttpResponseInfo);
response_info_->headers = fetcher_->GetResponseHeaders();
NotifyHeadersComplete();
}
2015-05-29 15:54:00 +00:00
2015-06-17 02:19:58 +00:00
int URLRequestFetchJob::DataAvailable(net::IOBuffer* buffer, int num_bytes) {
// Clear the IO_PENDING status.
2015-06-17 01:31:33 +00:00
SetStatus(net::URLRequestStatus());
2015-06-17 02:19:58 +00:00
// Do nothing if pending_buffer_ is empty, i.e. there's no ReadRawData()
// operation waiting for IO completion.
if (!pending_buffer_.get())
2015-06-17 03:34:47 +00:00
return net::ERR_IO_PENDING;
2015-06-17 02:19:58 +00:00
// pending_buffer_ is set to the IOBuffer instance provided to ReadRawData()
// by URLRequestJob.
int bytes_read = std::min(num_bytes, pending_buffer_size_);
memcpy(pending_buffer_->data(), buffer->data(), bytes_read);
2015-06-17 03:04:15 +00:00
// Clear the buffers before notifying the read is complete, so that it is
// safe for the observer to read.
pending_buffer_ = nullptr;
pending_buffer_size_ = 0;
2015-06-17 02:19:58 +00:00
NotifyReadComplete(bytes_read);
return bytes_read;
2015-06-17 01:31:33 +00:00
}
2015-05-29 15:54:00 +00:00
void URLRequestFetchJob::Start() {
2015-06-16 09:08:53 +00:00
fetcher_->Start();
2015-05-29 15:54:00 +00:00
}
void URLRequestFetchJob::Kill() {
URLRequestJob::Kill();
2015-06-17 01:31:33 +00:00
fetcher_.reset();
2015-05-29 15:54:00 +00:00
}
bool URLRequestFetchJob::ReadRawData(net::IOBuffer* dest,
int dest_size,
int* bytes_read) {
2015-06-17 02:19:58 +00:00
pending_buffer_ = dest;
pending_buffer_size_ = dest_size;
SetStatus(net::URLRequestStatus(net::URLRequestStatus::IO_PENDING, 0));
return false;
2015-05-29 15:54:00 +00:00
}
2015-06-17 01:31:33 +00:00
bool URLRequestFetchJob::GetMimeType(std::string* mime_type) const {
if (!response_info_)
return false;
2015-05-29 15:54:00 +00:00
2015-06-17 01:31:33 +00:00
return response_info_->headers->GetMimeType(mime_type);
2015-05-29 15:54:00 +00:00
}
2015-06-17 01:31:33 +00:00
void URLRequestFetchJob::GetResponseInfo(net::HttpResponseInfo* info) {
if (response_info_)
*info = *response_info_;
2015-05-29 15:54:00 +00:00
}
2015-06-17 01:31:33 +00:00
int URLRequestFetchJob::GetResponseCode() const {
if (!response_info_)
return -1;
return response_info_->headers->response_code();
2015-05-29 15:54:00 +00:00
}
2015-06-17 01:31:33 +00:00
void URLRequestFetchJob::OnURLFetchComplete(const net::URLFetcher* source) {
2015-06-17 03:04:15 +00:00
pending_buffer_ = nullptr;
pending_buffer_size_ = 0;
2015-06-17 02:19:58 +00:00
NotifyDone(fetcher_->GetStatus());
if (fetcher_->GetStatus().is_success())
NotifyReadComplete(0);
2015-05-29 15:54:00 +00:00
}
} // namespace atom