electron/atom/browser/api/atom_api_protocol.cc

428 lines
15 KiB
C++
Raw Normal View History

// Copyright (c) 2013 GitHub, Inc.
2014-04-25 09:49:37 +00:00
// Use of this source code is governed by the MIT license that can be
2013-08-24 07:26:10 +00:00
// found in the LICENSE file.
2014-03-16 00:30:26 +00:00
#include "atom/browser/api/atom_api_protocol.h"
2013-08-24 07:26:10 +00:00
#include "atom/browser/atom_browser_client.h"
2014-03-16 00:30:26 +00:00
#include "atom/browser/atom_browser_context.h"
2015-06-18 08:59:03 +00:00
#include "atom/browser/atom_browser_main_parts.h"
#include "atom/browser/api/atom_api_session.h"
2014-03-16 00:30:26 +00:00
#include "atom/browser/net/adapter_request_job.h"
#include "atom/browser/net/atom_url_request_job_factory.h"
2014-04-21 08:33:32 +00:00
#include "atom/common/native_mate_converters/file_path_converter.h"
2015-05-29 15:54:00 +00:00
#include "atom/common/native_mate_converters/gurl_converter.h"
2013-08-24 08:38:19 +00:00
#include "content/public/browser/browser_thread.h"
#include "native_mate/callback.h"
2014-04-21 08:33:32 +00:00
#include "native_mate/dictionary.h"
#include "net/url_request/url_request_context.h"
#include "atom/common/node_includes.h"
2013-08-24 07:26:10 +00:00
2014-04-21 08:33:32 +00:00
using content::BrowserThread;
2013-08-24 07:26:10 +00:00
2014-04-21 08:33:32 +00:00
namespace mate {
2014-04-21 08:33:32 +00:00
template<>
struct Converter<const net::URLRequest*> {
2015-05-22 11:11:22 +00:00
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
2015-07-07 14:43:13 +00:00
const net::URLRequest* val) {
return mate::ObjectTemplateBuilder(isolate)
.SetValue("method", val->method())
.SetValue("url", val->url().spec())
.SetValue("referrer", val->referrer())
.Build()->NewInstance();
2014-04-21 08:33:32 +00:00
}
};
template<>
struct Converter<net::URLRequestContextGetter*> {
static bool FromV8(v8::Isolate* isolate, v8::Local<v8::Value> val,
net::URLRequestContextGetter** out) {
2015-07-16 20:30:43 +00:00
if (val->IsNull()) {
*out = nullptr;
return true;
}
atom::api::Session* session;
if (!Converter<atom::api::Session*>::FromV8(isolate, val, &session))
return false;
2015-07-16 20:30:43 +00:00
*out = session->browser_context()->GetRequestContext();
return true;
}
};
2014-04-21 08:33:32 +00:00
} // namespace mate
2014-04-21 08:33:32 +00:00
namespace atom {
2014-04-21 08:33:32 +00:00
namespace api {
2014-04-21 08:33:32 +00:00
namespace {
2014-04-21 08:33:32 +00:00
typedef net::URLRequestJobFactory::ProtocolHandler ProtocolHandler;
2015-05-22 14:15:13 +00:00
scoped_refptr<base::RefCountedBytes> BufferToRefCountedBytes(
v8::Local<v8::Value> buf) {
scoped_refptr<base::RefCountedBytes> data(new base::RefCountedBytes);
auto start = reinterpret_cast<const unsigned char*>(node::Buffer::Data(buf));
data->data().assign(start, start + node::Buffer::Length(buf));
return data;
}
class CustomProtocolRequestJob : public AdapterRequestJob {
public:
2014-04-21 08:33:32 +00:00
CustomProtocolRequestJob(Protocol* registry,
ProtocolHandler* protocol_handler,
net::URLRequest* request,
net::NetworkDelegate* network_delegate)
2014-04-21 08:33:32 +00:00
: AdapterRequestJob(protocol_handler, request, network_delegate),
registry_(registry) {
}
2015-07-22 16:24:12 +00:00
void GetJobTypeInUI(const Protocol::JsProtocolHandler& callback) {
2015-07-05 17:53:07 +00:00
DCHECK_CURRENTLY_ON(BrowserThread::UI);
v8::Locker locker(registry_->isolate());
v8::HandleScope handle_scope(registry_->isolate());
// Call the JS handler.
2015-07-07 14:43:13 +00:00
v8::Local<v8::Value> result = callback.Run(request());
// Determine the type of the job we are going to create.
if (result->IsString()) {
2014-04-21 08:33:32 +00:00
std::string data = mate::V8ToString(result);
BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
base::Bind(&AdapterRequestJob::CreateStringJobAndStart,
2014-04-21 08:33:32 +00:00
GetWeakPtr(), "text/plain", "UTF-8", data));
return;
} else if (result->IsObject()) {
2015-05-22 11:11:22 +00:00
v8::Local<v8::Object> obj = result->ToObject();
mate::Dictionary dict(registry_->isolate(), obj);
2014-04-21 08:33:32 +00:00
std::string name = mate::V8ToString(obj->GetConstructorName());
if (name == "RequestStringJob") {
2014-04-21 08:33:32 +00:00
std::string mime_type, charset, data;
dict.Get("mimeType", &mime_type);
dict.Get("charset", &charset);
dict.Get("data", &data);
BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
base::Bind(&AdapterRequestJob::CreateStringJobAndStart,
2014-04-21 08:33:32 +00:00
GetWeakPtr(), mime_type, charset, data));
return;
2015-03-16 12:53:45 +00:00
} else if (name == "RequestBufferJob") {
std::string mime_type, encoding;
2015-05-22 11:11:22 +00:00
v8::Local<v8::Value> buffer;
2015-03-16 12:53:45 +00:00
dict.Get("mimeType", &mime_type);
dict.Get("encoding", &encoding);
dict.Get("data", &buffer);
BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
base::Bind(&AdapterRequestJob::CreateBufferJobAndStart,
2015-05-22 14:15:13 +00:00
GetWeakPtr(), mime_type, encoding,
BufferToRefCountedBytes(buffer)));
2015-03-16 12:53:45 +00:00
return;
} else if (name == "RequestFileJob") {
2014-04-21 08:33:32 +00:00
base::FilePath path;
dict.Get("path", &path);
2014-04-21 08:33:32 +00:00
BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
base::Bind(&AdapterRequestJob::CreateFileJobAndStart,
2014-04-21 08:33:32 +00:00
GetWeakPtr(), path));
return;
} else if (name == "RequestErrorJob") {
2015-06-17 01:31:33 +00:00
int error = net::ERR_NOT_IMPLEMENTED;
dict.Get("error", &error);
BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
base::Bind(&AdapterRequestJob::CreateErrorJobAndStart,
GetWeakPtr(), error));
return;
2015-05-29 15:54:00 +00:00
} else if (name == "RequestHttpJob") {
GURL url;
2015-06-17 03:20:09 +00:00
std::string method, referrer;
2015-07-16 20:30:43 +00:00
net::URLRequestContextGetter* getter =
registry_->browser_context()->GetRequestContext();
2015-05-29 15:54:00 +00:00
dict.Get("url", &url);
dict.Get("method", &method);
2015-06-17 03:20:09 +00:00
dict.Get("referrer", &referrer);
dict.Get("session", &getter);
2015-05-29 15:54:00 +00:00
BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
2015-06-18 09:18:11 +00:00
base::Bind(&AdapterRequestJob::CreateHttpJobAndStart, GetWeakPtr(),
base::Unretained(getter), url, method, referrer));
2015-05-29 15:54:00 +00:00
return;
}
}
// Try the default protocol handler if we have.
2013-09-03 08:50:10 +00:00
if (default_protocol_handler()) {
2014-04-21 08:33:32 +00:00
BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
base::Bind(&AdapterRequestJob::CreateJobFromProtocolHandlerAndStart,
GetWeakPtr()));
2013-09-03 08:50:10 +00:00
return;
}
// Fallback to the not implemented error.
2014-04-21 08:33:32 +00:00
BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
base::Bind(&AdapterRequestJob::CreateErrorJobAndStart,
2014-04-21 08:33:32 +00:00
GetWeakPtr(), net::ERR_NOT_IMPLEMENTED));
}
2014-04-21 08:33:32 +00:00
2015-07-22 16:24:12 +00:00
// AdapterRequestJob:
void GetJobType() override {
BrowserThread::PostTask(BrowserThread::UI, FROM_HERE,
base::Bind(&CustomProtocolRequestJob::GetJobTypeInUI,
base::Unretained(this),
registry_->GetProtocolHandler(request()->url().scheme())));
}
2014-04-21 08:33:32 +00:00
private:
Protocol* registry_; // Weak, the Protocol class is expected to live forever.
};
// Always return the same CustomProtocolRequestJob for all requests, because
// the content API needs the ProtocolHandler to return a job immediately, and
// getting the real job from the JS requires asynchronous calls, so we have
// to create an adapter job first.
// Users can also pass an extra ProtocolHandler as the fallback one when
// registered handler doesn't want to deal with the request.
class CustomProtocolHandler : public ProtocolHandler {
public:
2014-04-21 08:33:32 +00:00
CustomProtocolHandler(api::Protocol* registry,
ProtocolHandler* protocol_handler = NULL)
: registry_(registry), protocol_handler_(protocol_handler) {
}
2015-01-10 01:24:36 +00:00
net::URLRequestJob* MaybeCreateJob(
net::URLRequest* request,
2015-01-10 01:24:36 +00:00
net::NetworkDelegate* network_delegate) const override {
2014-04-21 08:33:32 +00:00
return new CustomProtocolRequestJob(registry_, protocol_handler_.get(),
request, network_delegate);
}
2013-09-03 08:50:10 +00:00
ProtocolHandler* ReleaseDefaultProtocolHandler() {
return protocol_handler_.release();
}
ProtocolHandler* original_handler() { return protocol_handler_.get(); }
private:
2014-04-21 08:33:32 +00:00
Protocol* registry_; // Weak, the Protocol class is expected to live forever.
scoped_ptr<ProtocolHandler> protocol_handler_;
DISALLOW_COPY_AND_ASSIGN(CustomProtocolHandler);
};
2015-07-09 09:18:45 +00:00
std::string ConvertErrorCode(int error_code) {
switch (error_code) {
case Protocol::ERR_SCHEME_REGISTERED:
return "The Scheme is already registered";
case Protocol::ERR_SCHEME_UNREGISTERED:
return "The Scheme has not been registered";
case Protocol::ERR_SCHEME_INTERCEPTED:
return "There is no protocol handler to intercept";
case Protocol::ERR_SCHEME_UNINTERCEPTED:
return "The protocol is not intercepted";
case Protocol::ERR_NO_SCHEME:
return "The Scheme does not exist.";
case Protocol::ERR_SCHEME:
return "Cannot intercept custom protocols";
default:
NOTREACHED();
return std::string();
}
}
} // namespace
2015-06-18 08:59:03 +00:00
Protocol::Protocol(AtomBrowserContext* browser_context)
2015-07-16 20:30:43 +00:00
: browser_context_(browser_context),
job_factory_(browser_context->job_factory()) {
CHECK(job_factory_);
2014-04-21 08:33:32 +00:00
}
2014-04-21 08:33:32 +00:00
Protocol::JsProtocolHandler Protocol::GetProtocolHandler(
const std::string& scheme) {
return protocol_handlers_[scheme];
}
2013-08-24 08:38:19 +00:00
2015-07-09 09:18:45 +00:00
void Protocol::OnIOActionCompleted(const JsCompletionCallback& callback,
int error) {
2015-07-05 17:53:07 +00:00
DCHECK_CURRENTLY_ON(BrowserThread::UI);
v8::Locker locker(isolate());
v8::HandleScope handle_scope(isolate());
2015-07-09 09:18:45 +00:00
if (error) {
2015-07-05 17:53:07 +00:00
callback.Run(v8::Exception::Error(
2015-07-09 09:18:45 +00:00
mate::StringToV8(isolate(), ConvertErrorCode(error))));
2015-07-05 17:53:07 +00:00
return;
}
2015-07-09 09:18:45 +00:00
callback.Run(v8::Null(isolate()));
2015-07-05 17:53:07 +00:00
}
2014-04-21 08:33:32 +00:00
mate::ObjectTemplateBuilder Protocol::GetObjectTemplateBuilder(
v8::Isolate* isolate) {
return mate::ObjectTemplateBuilder(isolate)
.SetMethod("registerStandardSchemes", &Protocol::RegisterStandardSchemes)
.SetMethod("isHandledProtocol", &Protocol::IsHandledProtocol)
2015-07-09 09:18:45 +00:00
.SetMethod("_registerProtocol", &Protocol::RegisterProtocol)
.SetMethod("_unregisterProtocol", &Protocol::UnregisterProtocol)
2015-07-07 14:43:13 +00:00
.SetMethod("_interceptProtocol", &Protocol::InterceptProtocol)
.SetMethod("_uninterceptProtocol", &Protocol::UninterceptProtocol);
2013-08-24 07:26:10 +00:00
}
void Protocol::RegisterStandardSchemes(
const std::vector<std::string>& schemes) {
atom::AtomBrowserClient::SetCustomSchemes(schemes);
}
2015-07-05 17:53:07 +00:00
void Protocol::IsHandledProtocol(const std::string& scheme,
2015-07-07 14:43:13 +00:00
const net::CompletionCallback& callback) {
2015-07-05 17:53:07 +00:00
BrowserThread::PostTaskAndReplyWithResult(BrowserThread::IO, FROM_HERE,
base::Bind(&AtomURLRequestJobFactory::IsHandledProtocol,
base::Unretained(job_factory_), scheme),
callback);
2014-04-21 08:33:32 +00:00
}
2015-07-09 09:18:45 +00:00
void Protocol::RegisterProtocol(v8::Isolate* isolate,
const std::string& scheme,
const JsProtocolHandler& handler,
const JsCompletionCallback& callback) {
BrowserThread::PostTaskAndReplyWithResult(BrowserThread::IO, FROM_HERE,
base::Bind(&Protocol::RegisterProtocolInIO,
base::Unretained(this), scheme, handler),
base::Bind(&Protocol::OnIOActionCompleted,
base::Unretained(this), callback));
}
void Protocol::UnregisterProtocol(v8::Isolate* isolate,
const std::string& scheme,
const JsCompletionCallback& callback) {
BrowserThread::PostTaskAndReplyWithResult(BrowserThread::IO, FROM_HERE,
base::Bind(&Protocol::UnregisterProtocolInIO,
base::Unretained(this), scheme),
base::Bind(&Protocol::OnIOActionCompleted,
base::Unretained(this), callback));
}
void Protocol::InterceptProtocol(v8::Isolate* isolate,
const std::string& scheme,
2015-07-07 14:43:13 +00:00
const JsProtocolHandler& handler,
const JsCompletionCallback& callback) {
2015-07-05 17:53:07 +00:00
BrowserThread::PostTaskAndReplyWithResult(BrowserThread::IO, FROM_HERE,
2015-07-09 09:18:45 +00:00
base::Bind(&Protocol::InterceptProtocolInIO,
base::Unretained(this), scheme, handler),
base::Bind(&Protocol::OnIOActionCompleted,
base::Unretained(this), callback));
2013-08-30 02:15:15 +00:00
}
void Protocol::UninterceptProtocol(v8::Isolate* isolate,
2015-07-07 14:43:13 +00:00
const std::string& scheme,
const JsCompletionCallback& callback) {
2015-07-09 09:18:45 +00:00
BrowserThread::PostTaskAndReplyWithResult(BrowserThread::IO, FROM_HERE,
2015-07-07 14:43:13 +00:00
base::Bind(&Protocol::UninterceptProtocolInIO,
base::Unretained(this), scheme),
2015-07-09 09:18:45 +00:00
base::Bind(&Protocol::OnIOActionCompleted,
base::Unretained(this), callback));
2013-08-30 02:15:15 +00:00
}
2015-07-09 09:18:45 +00:00
int Protocol::RegisterProtocolInIO(const std::string& scheme,
const JsProtocolHandler& handler) {
2015-07-05 17:53:07 +00:00
DCHECK_CURRENTLY_ON(BrowserThread::IO);
2014-04-21 08:33:32 +00:00
2015-07-09 09:18:45 +00:00
if (ContainsKey(protocol_handlers_, scheme) ||
job_factory_->IsHandledProtocol(scheme)) {
return ERR_SCHEME_REGISTERED;
}
protocol_handlers_[scheme] = handler;
2014-04-21 08:33:32 +00:00
job_factory_->SetProtocolHandler(scheme, new CustomProtocolHandler(this));
2015-07-09 09:18:45 +00:00
return OK;
2013-08-24 08:38:19 +00:00
}
2015-07-09 09:18:45 +00:00
int Protocol::UnregisterProtocolInIO(const std::string& scheme) {
2015-07-05 17:53:07 +00:00
DCHECK_CURRENTLY_ON(BrowserThread::IO);
2014-04-21 08:33:32 +00:00
2015-07-09 09:18:45 +00:00
ProtocolHandlersMap::iterator it(protocol_handlers_.find(scheme));
if (it == protocol_handlers_.end()) {
return ERR_SCHEME_UNREGISTERED;
}
protocol_handlers_.erase(it);
2014-04-21 08:33:32 +00:00
job_factory_->SetProtocolHandler(scheme, NULL);
2015-07-09 09:18:45 +00:00
return OK;
2013-08-24 08:38:19 +00:00
}
2015-07-09 09:18:45 +00:00
int Protocol::InterceptProtocolInIO(const std::string& scheme,
const JsProtocolHandler& handler) {
2015-07-05 17:53:07 +00:00
DCHECK_CURRENTLY_ON(BrowserThread::IO);
2014-04-21 08:33:32 +00:00
2015-07-09 09:18:45 +00:00
if (!job_factory_->HasProtocolHandler(scheme))
return ERR_NO_SCHEME;
if (ContainsKey(protocol_handlers_, scheme))
return ERR_SCHEME;
protocol_handlers_[scheme] = handler;
2014-04-21 08:33:32 +00:00
ProtocolHandler* original_handler = job_factory_->GetProtocolHandler(scheme);
2015-07-07 14:43:13 +00:00
if (original_handler == nullptr) {
2015-07-09 09:18:45 +00:00
return ERR_SCHEME_INTERCEPTED;
}
2014-04-21 08:33:32 +00:00
job_factory_->ReplaceProtocol(
scheme, new CustomProtocolHandler(this, original_handler));
2015-07-09 09:18:45 +00:00
return OK;
2013-08-30 02:15:15 +00:00
}
2015-07-09 09:18:45 +00:00
int Protocol::UninterceptProtocolInIO(const std::string& scheme) {
2015-07-05 17:53:07 +00:00
DCHECK_CURRENTLY_ON(BrowserThread::IO);
2013-09-03 08:50:10 +00:00
2015-07-09 09:18:45 +00:00
ProtocolHandlersMap::iterator it(protocol_handlers_.find(scheme));
if (it == protocol_handlers_.end())
return ERR_SCHEME_UNREGISTERED;
protocol_handlers_.erase(it);
2013-09-03 08:50:10 +00:00
CustomProtocolHandler* handler = static_cast<CustomProtocolHandler*>(
2014-04-21 08:33:32 +00:00
job_factory_->GetProtocolHandler(scheme));
2015-07-07 14:43:13 +00:00
if (handler->original_handler() == nullptr) {
2015-07-09 09:18:45 +00:00
return ERR_SCHEME_UNINTERCEPTED;
}
2013-09-03 08:50:10 +00:00
2014-04-21 08:33:32 +00:00
// Reset the protocol handler to the orignal one and delete current protocol
// handler.
2013-09-03 08:50:10 +00:00
ProtocolHandler* original_handler = handler->ReleaseDefaultProtocolHandler();
2014-04-21 08:33:32 +00:00
delete job_factory_->ReplaceProtocol(scheme, original_handler);
2015-07-09 09:18:45 +00:00
return OK;
2013-08-30 02:15:15 +00:00
}
2013-08-24 07:26:10 +00:00
// static
2015-06-18 08:59:03 +00:00
mate::Handle<Protocol> Protocol::Create(
v8::Isolate* isolate, AtomBrowserContext* browser_context) {
2015-06-23 02:18:43 +00:00
return mate::CreateHandle(isolate, new Protocol(browser_context));
2013-08-24 07:26:10 +00:00
}
} // namespace api
} // namespace atom
2014-04-21 08:33:32 +00:00
namespace {
2015-05-22 11:11:22 +00:00
void Initialize(v8::Local<v8::Object> exports, v8::Local<v8::Value> unused,
v8::Local<v8::Context> context, void* priv) {
v8::Isolate* isolate = context->GetIsolate();
2014-04-21 08:33:32 +00:00
mate::Dictionary dict(isolate, exports);
2015-06-18 08:59:03 +00:00
auto browser_context = static_cast<atom::AtomBrowserContext*>(
atom::AtomBrowserMainParts::Get()->browser_context());
dict.Set("protocol", atom::api::Protocol::Create(isolate, browser_context));
2014-04-21 08:33:32 +00:00
}
} // namespace
NODE_MODULE_CONTEXT_AWARE_BUILTIN(atom_browser_protocol, Initialize)