Make api::Protocol thread safe

This commit is contained in:
Cheng Zhao 2016-06-15 21:11:42 +09:00
parent 659384383e
commit fe0e17d1c3
4 changed files with 94 additions and 62 deletions

View file

@ -27,7 +27,8 @@ namespace atom {
namespace api { namespace api {
Protocol::Protocol(v8::Isolate* isolate, AtomBrowserContext* browser_context) Protocol::Protocol(v8::Isolate* isolate, AtomBrowserContext* browser_context)
: request_context_getter_(browser_context->GetRequestContext()), : request_context_getter_(static_cast<brightray::URLRequestContextGetter*>(
browser_context->GetRequestContext())),
weak_factory_(this) { weak_factory_(this) {
Init(isolate); Init(isolate);
} }
@ -44,16 +45,20 @@ void Protocol::UnregisterProtocol(
content::BrowserThread::PostTaskAndReplyWithResult( content::BrowserThread::PostTaskAndReplyWithResult(
content::BrowserThread::IO, FROM_HERE, content::BrowserThread::IO, FROM_HERE,
base::Bind(&Protocol::UnregisterProtocolInIO, base::Bind(&Protocol::UnregisterProtocolInIO,
base::Unretained(this), scheme), request_context_getter_, scheme),
base::Bind(&Protocol::OnIOCompleted, base::Bind(&Protocol::OnIOCompleted,
GetWeakPtr(), callback)); GetWeakPtr(), callback));
} }
// static
Protocol::ProtocolError Protocol::UnregisterProtocolInIO( Protocol::ProtocolError Protocol::UnregisterProtocolInIO(
scoped_refptr<brightray::URLRequestContextGetter> request_context_getter,
const std::string& scheme) { const std::string& scheme) {
if (!job_factory()->HasProtocolHandler(scheme)) auto job_factory = static_cast<AtomURLRequestJobFactory*>(
request_context_getter->job_factory());
if (!job_factory->HasProtocolHandler(scheme))
return PROTOCOL_NOT_REGISTERED; return PROTOCOL_NOT_REGISTERED;
job_factory()->SetProtocolHandler(scheme, nullptr); job_factory->SetProtocolHandler(scheme, nullptr);
return PROTOCOL_OK; return PROTOCOL_OK;
} }
@ -62,12 +67,15 @@ void Protocol::IsProtocolHandled(const std::string& scheme,
content::BrowserThread::PostTaskAndReplyWithResult( content::BrowserThread::PostTaskAndReplyWithResult(
content::BrowserThread::IO, FROM_HERE, content::BrowserThread::IO, FROM_HERE,
base::Bind(&Protocol::IsProtocolHandledInIO, base::Bind(&Protocol::IsProtocolHandledInIO,
base::Unretained(this), scheme), request_context_getter_, scheme),
callback); callback);
} }
bool Protocol::IsProtocolHandledInIO(const std::string& scheme) { // static
return job_factory()->IsHandledProtocol(scheme); bool Protocol::IsProtocolHandledInIO(
scoped_refptr<brightray::URLRequestContextGetter> request_context_getter,
const std::string& scheme) {
return request_context_getter->job_factory()->IsHandledProtocol(scheme);
} }
void Protocol::UninterceptProtocol( void Protocol::UninterceptProtocol(
@ -77,18 +85,18 @@ void Protocol::UninterceptProtocol(
content::BrowserThread::PostTaskAndReplyWithResult( content::BrowserThread::PostTaskAndReplyWithResult(
content::BrowserThread::IO, FROM_HERE, content::BrowserThread::IO, FROM_HERE,
base::Bind(&Protocol::UninterceptProtocolInIO, base::Bind(&Protocol::UninterceptProtocolInIO,
base::Unretained(this), scheme), request_context_getter_, scheme),
base::Bind(&Protocol::OnIOCompleted, base::Bind(&Protocol::OnIOCompleted,
GetWeakPtr(), callback)); GetWeakPtr(), callback));
} }
// static
Protocol::ProtocolError Protocol::UninterceptProtocolInIO( Protocol::ProtocolError Protocol::UninterceptProtocolInIO(
scoped_refptr<brightray::URLRequestContextGetter> request_context_getter,
const std::string& scheme) { const std::string& scheme) {
if (!original_protocols_.contains(scheme)) return static_cast<AtomURLRequestJobFactory*>(
return PROTOCOL_NOT_INTERCEPTED; request_context_getter->job_factory())->UninterceptProtocol(scheme) ?
job_factory()->ReplaceProtocol(scheme, PROTOCOL_OK : PROTOCOL_NOT_INTERCEPTED;
original_protocols_.take_and_erase(scheme));
return PROTOCOL_OK;
} }
void Protocol::OnIOCompleted( void Protocol::OnIOCompleted(
@ -119,6 +127,13 @@ std::string Protocol::ErrorCodeToString(ProtocolError error) {
} }
} }
AtomURLRequestJobFactory* Protocol::GetJobFactoryInIO() const {
request_context_getter_->GetURLRequestContext(); // Force init.
return static_cast<AtomURLRequestJobFactory*>(
static_cast<brightray::URLRequestContextGetter*>(
request_context_getter_.get())->job_factory());
}
// static // static
mate::Handle<Protocol> Protocol::Create( mate::Handle<Protocol> Protocol::Create(
v8::Isolate* isolate, AtomBrowserContext* browser_context) { v8::Isolate* isolate, AtomBrowserContext* browser_context) {

View file

@ -13,7 +13,6 @@
#include "atom/browser/atom_browser_context.h" #include "atom/browser/atom_browser_context.h"
#include "atom/browser/net/atom_url_request_job_factory.h" #include "atom/browser/net/atom_url_request_job_factory.h"
#include "base/callback.h" #include "base/callback.h"
#include "base/containers/scoped_ptr_hash_map.h"
#include "base/memory/weak_ptr.h" #include "base/memory/weak_ptr.h"
#include "content/public/browser/browser_thread.h" #include "content/public/browser/browser_thread.h"
#include "native_mate/arguments.h" #include "native_mate/arguments.h"
@ -45,17 +44,6 @@ class Protocol : public mate::TrackableObject<Protocol> {
protected: protected:
Protocol(v8::Isolate* isolate, AtomBrowserContext* browser_context); Protocol(v8::Isolate* isolate, AtomBrowserContext* browser_context);
AtomURLRequestJobFactory* job_factory() const {
request_context_getter_->GetURLRequestContext(); // Force init.
return static_cast<AtomURLRequestJobFactory*>(
static_cast<brightray::URLRequestContextGetter*>(
request_context_getter_.get())->job_factory());
}
base::WeakPtr<Protocol> GetWeakPtr() {
return weak_factory_.GetWeakPtr();
}
private: private:
// Possible errors. // Possible errors.
enum ProtocolError { enum ProtocolError {
@ -86,13 +74,13 @@ class Protocol : public mate::TrackableObject<Protocol> {
net::URLRequest* request, net::URLRequest* request,
net::NetworkDelegate* network_delegate) const override { net::NetworkDelegate* network_delegate) const override {
RequestJob* request_job = new RequestJob(request, network_delegate); RequestJob* request_job = new RequestJob(request, network_delegate);
request_job->SetHandlerInfo(isolate_, request_context_, handler_); request_job->SetHandlerInfo(isolate_, request_context_.get(), handler_);
return request_job; return request_job;
} }
private: private:
v8::Isolate* isolate_; v8::Isolate* isolate_;
net::URLRequestContextGetter* request_context_; scoped_refptr<net::URLRequestContextGetter> request_context_;
Protocol::Handler handler_; Protocol::Handler handler_;
DISALLOW_COPY_AND_ASSIGN(CustomProtocolHandler); DISALLOW_COPY_AND_ASSIGN(CustomProtocolHandler);
@ -111,19 +99,24 @@ class Protocol : public mate::TrackableObject<Protocol> {
content::BrowserThread::PostTaskAndReplyWithResult( content::BrowserThread::PostTaskAndReplyWithResult(
content::BrowserThread::IO, FROM_HERE, content::BrowserThread::IO, FROM_HERE,
base::Bind(&Protocol::RegisterProtocolInIO<RequestJob>, base::Bind(&Protocol::RegisterProtocolInIO<RequestJob>,
base::Unretained(this), scheme, handler), request_context_getter_, isolate(), scheme, handler),
base::Bind(&Protocol::OnIOCompleted, base::Bind(&Protocol::OnIOCompleted,
GetWeakPtr(), callback)); GetWeakPtr(), callback));
} }
template<typename RequestJob> template<typename RequestJob>
ProtocolError RegisterProtocolInIO(const std::string& scheme, static ProtocolError RegisterProtocolInIO(
const Handler& handler) { scoped_refptr<brightray::URLRequestContextGetter> request_context_getter,
if (job_factory()->IsHandledProtocol(scheme)) v8::Isolate* isolate,
const std::string& scheme,
const Handler& handler) {
auto job_factory = static_cast<AtomURLRequestJobFactory*>(
request_context_getter->job_factory());
if (job_factory->IsHandledProtocol(scheme))
return PROTOCOL_REGISTERED; return PROTOCOL_REGISTERED;
std::unique_ptr<CustomProtocolHandler<RequestJob>> protocol_handler( std::unique_ptr<CustomProtocolHandler<RequestJob>> protocol_handler(
new CustomProtocolHandler<RequestJob>( new CustomProtocolHandler<RequestJob>(
isolate(), request_context_getter_.get(), handler)); isolate, request_context_getter.get(), handler));
if (job_factory()->SetProtocolHandler(scheme, std::move(protocol_handler))) if (job_factory->SetProtocolHandler(scheme, std::move(protocol_handler)))
return PROTOCOL_OK; return PROTOCOL_OK;
else else
return PROTOCOL_FAIL; return PROTOCOL_FAIL;
@ -131,12 +124,16 @@ class Protocol : public mate::TrackableObject<Protocol> {
// Unregister the protocol handler that handles |scheme|. // Unregister the protocol handler that handles |scheme|.
void UnregisterProtocol(const std::string& scheme, mate::Arguments* args); void UnregisterProtocol(const std::string& scheme, mate::Arguments* args);
ProtocolError UnregisterProtocolInIO(const std::string& scheme); static ProtocolError UnregisterProtocolInIO(
scoped_refptr<brightray::URLRequestContextGetter> request_context_getter,
const std::string& scheme);
// Whether the protocol has handler registered. // Whether the protocol has handler registered.
void IsProtocolHandled(const std::string& scheme, void IsProtocolHandled(const std::string& scheme,
const BooleanCallback& callback); const BooleanCallback& callback);
bool IsProtocolHandledInIO(const std::string& scheme); static bool IsProtocolHandledInIO(
scoped_refptr<brightray::URLRequestContextGetter> request_context_getter,
const std::string& scheme);
// Replace the protocol handler with a new one. // Replace the protocol handler with a new one.
template<typename RequestJob> template<typename RequestJob>
@ -148,32 +145,36 @@ class Protocol : public mate::TrackableObject<Protocol> {
content::BrowserThread::PostTaskAndReplyWithResult( content::BrowserThread::PostTaskAndReplyWithResult(
content::BrowserThread::IO, FROM_HERE, content::BrowserThread::IO, FROM_HERE,
base::Bind(&Protocol::InterceptProtocolInIO<RequestJob>, base::Bind(&Protocol::InterceptProtocolInIO<RequestJob>,
base::Unretained(this), scheme, handler), request_context_getter_, isolate(), scheme, handler),
base::Bind(&Protocol::OnIOCompleted, base::Bind(&Protocol::OnIOCompleted,
GetWeakPtr(), callback)); GetWeakPtr(), callback));
} }
template<typename RequestJob> template<typename RequestJob>
ProtocolError InterceptProtocolInIO(const std::string& scheme, static ProtocolError InterceptProtocolInIO(
const Handler& handler) { scoped_refptr<brightray::URLRequestContextGetter> request_context_getter,
if (!job_factory()->IsHandledProtocol(scheme)) v8::Isolate* isolate,
const std::string& scheme,
const Handler& handler) {
auto job_factory = static_cast<AtomURLRequestJobFactory*>(
request_context_getter->job_factory());
if (!job_factory->IsHandledProtocol(scheme))
return PROTOCOL_NOT_REGISTERED; return PROTOCOL_NOT_REGISTERED;
// It is possible a protocol is handled but can not be intercepted. // It is possible a protocol is handled but can not be intercepted.
if (!job_factory()->HasProtocolHandler(scheme)) if (!job_factory->HasProtocolHandler(scheme))
return PROTOCOL_FAIL; return PROTOCOL_FAIL;
if (ContainsKey(original_protocols_, scheme))
return PROTOCOL_INTERCEPTED;
std::unique_ptr<CustomProtocolHandler<RequestJob>> protocol_handler( std::unique_ptr<CustomProtocolHandler<RequestJob>> protocol_handler(
new CustomProtocolHandler<RequestJob>( new CustomProtocolHandler<RequestJob>(
isolate(), request_context_getter_.get(), handler)); isolate, request_context_getter.get(), handler));
original_protocols_.set( if (!job_factory->InterceptProtocol(scheme, std::move(protocol_handler)))
scheme, return PROTOCOL_INTERCEPTED;
job_factory()->ReplaceProtocol(scheme, std::move(protocol_handler)));
return PROTOCOL_OK; return PROTOCOL_OK;
} }
// Restore the |scheme| to its original protocol handler. // Restore the |scheme| to its original protocol handler.
void UninterceptProtocol(const std::string& scheme, mate::Arguments* args); void UninterceptProtocol(const std::string& scheme, mate::Arguments* args);
ProtocolError UninterceptProtocolInIO(const std::string& scheme); static ProtocolError UninterceptProtocolInIO(
scoped_refptr<brightray::URLRequestContextGetter> request_context_getter,
const std::string& scheme);
// Convert error code to JS exception and call the callback. // Convert error code to JS exception and call the callback.
void OnIOCompleted(const CompletionCallback& callback, ProtocolError error); void OnIOCompleted(const CompletionCallback& callback, ProtocolError error);
@ -181,14 +182,13 @@ class Protocol : public mate::TrackableObject<Protocol> {
// Convert error code to string. // Convert error code to string.
std::string ErrorCodeToString(ProtocolError error); std::string ErrorCodeToString(ProtocolError error);
scoped_refptr<net::URLRequestContextGetter> request_context_getter_; AtomURLRequestJobFactory* GetJobFactoryInIO() const;
// Map that stores the original protocols of schemes. base::WeakPtr<Protocol> GetWeakPtr() {
using OriginalProtocolsMap = base::ScopedPtrHashMap< return weak_factory_.GetWeakPtr();
std::string, }
std::unique_ptr<net::URLRequestJobFactory::ProtocolHandler>>;
OriginalProtocolsMap original_protocols_;
scoped_refptr<brightray::URLRequestContextGetter> request_context_getter_;
base::WeakPtrFactory<Protocol> weak_factory_; base::WeakPtrFactory<Protocol> weak_factory_;
DISALLOW_COPY_AND_ASSIGN(Protocol); DISALLOW_COPY_AND_ASSIGN(Protocol);

View file

@ -5,6 +5,7 @@
#include "atom/browser/net/atom_url_request_job_factory.h" #include "atom/browser/net/atom_url_request_job_factory.h"
#include "base/memory/ptr_util.h"
#include "base/stl_util.h" #include "base/stl_util.h"
#include "content/public/browser/browser_thread.h" #include "content/public/browser/browser_thread.h"
#include "net/base/load_flags.h" #include "net/base/load_flags.h"
@ -41,14 +42,24 @@ bool AtomURLRequestJobFactory::SetProtocolHandler(
return true; return true;
} }
std::unique_ptr<ProtocolHandler> AtomURLRequestJobFactory::ReplaceProtocol( bool AtomURLRequestJobFactory::InterceptProtocol(
const std::string& scheme, const std::string& scheme,
std::unique_ptr<ProtocolHandler> protocol_handler) { std::unique_ptr<ProtocolHandler> protocol_handler) {
if (!ContainsKey(protocol_handler_map_, scheme)) if (!ContainsKey(protocol_handler_map_, scheme) ||
return nullptr; ContainsKey(original_protocols_, scheme))
return false;
ProtocolHandler* original_protocol_handler = protocol_handler_map_[scheme]; ProtocolHandler* original_protocol_handler = protocol_handler_map_[scheme];
protocol_handler_map_[scheme] = protocol_handler.release(); protocol_handler_map_[scheme] = protocol_handler.release();
return make_scoped_ptr(original_protocol_handler); original_protocols_.set(scheme, base::WrapUnique(original_protocol_handler));
return true;
}
bool AtomURLRequestJobFactory::UninterceptProtocol(const std::string& scheme) {
if (!original_protocols_.contains(scheme))
return false;
protocol_handler_map_[scheme] =
original_protocols_.take_and_erase(scheme).release();
return true;
} }
ProtocolHandler* AtomURLRequestJobFactory::GetProtocolHandler( ProtocolHandler* AtomURLRequestJobFactory::GetProtocolHandler(

View file

@ -7,11 +7,11 @@
#define ATOM_BROWSER_NET_ATOM_URL_REQUEST_JOB_FACTORY_H_ #define ATOM_BROWSER_NET_ATOM_URL_REQUEST_JOB_FACTORY_H_
#include <map> #include <map>
#include <memory>
#include <string> #include <string>
#include <vector> #include <vector>
#include "base/memory/scoped_ptr.h" #include "base/containers/scoped_ptr_hash_map.h"
#include "base/synchronization/lock.h"
#include "net/url_request/url_request_job_factory.h" #include "net/url_request/url_request_job_factory.h"
namespace atom { namespace atom {
@ -27,11 +27,11 @@ class AtomURLRequestJobFactory : public net::URLRequestJobFactory {
bool SetProtocolHandler(const std::string& scheme, bool SetProtocolHandler(const std::string& scheme,
std::unique_ptr<ProtocolHandler> protocol_handler); std::unique_ptr<ProtocolHandler> protocol_handler);
// Intercepts the ProtocolHandler for a scheme. Returns the original protocol // Intercepts the ProtocolHandler for a scheme.
// handler on success, otherwise returns NULL. bool InterceptProtocol(
std::unique_ptr<ProtocolHandler> ReplaceProtocol(
const std::string& scheme, const std::string& scheme,
std::unique_ptr<ProtocolHandler> protocol_handler); std::unique_ptr<ProtocolHandler> protocol_handler);
bool UninterceptProtocol(const std::string& scheme);
// Returns the protocol handler registered with scheme. // Returns the protocol handler registered with scheme.
ProtocolHandler* GetProtocolHandler(const std::string& scheme) const; ProtocolHandler* GetProtocolHandler(const std::string& scheme) const;
@ -60,6 +60,12 @@ class AtomURLRequestJobFactory : public net::URLRequestJobFactory {
ProtocolHandlerMap protocol_handler_map_; ProtocolHandlerMap protocol_handler_map_;
// Map that stores the original protocols of schemes.
using OriginalProtocolsMap = base::ScopedPtrHashMap<
std::string, std::unique_ptr<ProtocolHandler>>;
// Can only be accessed in IO thread.
OriginalProtocolsMap original_protocols_;
DISALLOW_COPY_AND_ASSIGN(AtomURLRequestJobFactory); DISALLOW_COPY_AND_ASSIGN(AtomURLRequestJobFactory);
}; };