2019-04-23 21:39:21 +00:00
|
|
|
// Copyright (c) 2019 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_protocol_ns.h"
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
#include "atom/browser/atom_browser_context.h"
|
2019-04-29 02:37:45 +00:00
|
|
|
#include "atom/common/native_mate_converters/net_converter.h"
|
|
|
|
#include "atom/common/native_mate_converters/once_callback.h"
|
2019-04-23 21:39:21 +00:00
|
|
|
#include "atom/common/promise_util.h"
|
|
|
|
#include "base/stl_util.h"
|
|
|
|
|
|
|
|
namespace atom {
|
|
|
|
namespace api {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Convert error code to string.
|
|
|
|
std::string ErrorCodeToString(ProtocolError error) {
|
|
|
|
switch (error) {
|
|
|
|
case PROTOCOL_REGISTERED:
|
|
|
|
return "The scheme has been registered";
|
|
|
|
case PROTOCOL_NOT_REGISTERED:
|
|
|
|
return "The scheme has not been registered";
|
|
|
|
case PROTOCOL_INTERCEPTED:
|
|
|
|
return "The scheme has been intercepted";
|
|
|
|
case PROTOCOL_NOT_INTERCEPTED:
|
|
|
|
return "The scheme has not been intercepted";
|
|
|
|
default:
|
|
|
|
return "Unexpected error";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Noop() {}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
ProtocolNS::ProtocolNS(v8::Isolate* isolate,
|
|
|
|
AtomBrowserContext* browser_context) {
|
|
|
|
Init(isolate);
|
|
|
|
AttachAsUserData(browser_context);
|
|
|
|
}
|
|
|
|
|
|
|
|
ProtocolNS::~ProtocolNS() = default;
|
|
|
|
|
|
|
|
void ProtocolNS::RegisterURLLoaderFactories(
|
|
|
|
content::ContentBrowserClient::NonNetworkURLLoaderFactoryMap* factories) {
|
2019-04-29 02:37:45 +00:00
|
|
|
for (const auto& it : handlers_) {
|
|
|
|
factories->emplace(it.first, std::make_unique<AtomURLLoaderFactory>(
|
|
|
|
it.second.first, it.second.second));
|
|
|
|
}
|
2019-04-23 21:39:21 +00:00
|
|
|
}
|
|
|
|
|
2019-04-29 02:37:45 +00:00
|
|
|
ProtocolError ProtocolNS::RegisterProtocol(ProtocolType type,
|
|
|
|
const std::string& scheme,
|
|
|
|
const ProtocolHandler& handler) {
|
2019-04-23 21:39:21 +00:00
|
|
|
ProtocolError error = PROTOCOL_OK;
|
|
|
|
if (!base::ContainsKey(handlers_, scheme))
|
2019-04-29 02:37:45 +00:00
|
|
|
handlers_[scheme] = std::make_pair(type, handler);
|
2019-04-23 21:39:21 +00:00
|
|
|
else
|
|
|
|
error = PROTOCOL_REGISTERED;
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProtocolNS::UnregisterProtocol(const std::string& scheme,
|
|
|
|
mate::Arguments* args) {
|
|
|
|
ProtocolError error = PROTOCOL_OK;
|
|
|
|
if (base::ContainsKey(handlers_, scheme))
|
|
|
|
handlers_.erase(scheme);
|
|
|
|
else
|
|
|
|
error = PROTOCOL_NOT_REGISTERED;
|
|
|
|
HandleOptionalCallback(args, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProtocolNS::IsProtocolRegistered(const std::string& scheme) {
|
|
|
|
return base::ContainsKey(handlers_, scheme);
|
|
|
|
}
|
|
|
|
|
|
|
|
v8::Local<v8::Promise> ProtocolNS::IsProtocolHandled(
|
|
|
|
const std::string& scheme) {
|
|
|
|
util::Promise promise(isolate());
|
|
|
|
promise.Resolve(IsProtocolRegistered(scheme));
|
|
|
|
return promise.GetHandle();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProtocolNS::HandleOptionalCallback(mate::Arguments* args,
|
|
|
|
ProtocolError error) {
|
|
|
|
CompletionCallback callback;
|
|
|
|
if (args->GetNext(&callback)) {
|
|
|
|
if (error == PROTOCOL_OK)
|
|
|
|
callback.Run(v8::Null(args->isolate()));
|
|
|
|
else
|
|
|
|
callback.Run(v8::Exception::Error(
|
|
|
|
mate::StringToV8(isolate(), ErrorCodeToString(error))));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
mate::Handle<ProtocolNS> ProtocolNS::Create(
|
|
|
|
v8::Isolate* isolate,
|
|
|
|
AtomBrowserContext* browser_context) {
|
|
|
|
return mate::CreateHandle(isolate, new ProtocolNS(isolate, browser_context));
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
void ProtocolNS::BuildPrototype(v8::Isolate* isolate,
|
|
|
|
v8::Local<v8::FunctionTemplate> prototype) {
|
|
|
|
prototype->SetClassName(mate::StringToV8(isolate, "Protocol"));
|
|
|
|
mate::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
|
2019-04-29 02:37:45 +00:00
|
|
|
.SetMethod("registerStringProtocol",
|
|
|
|
&ProtocolNS::RegisterProtocolFor<ProtocolType::kString>)
|
|
|
|
.SetMethod("registerBufferProtocol",
|
|
|
|
&ProtocolNS::RegisterProtocolFor<ProtocolType::kBuffer>)
|
|
|
|
.SetMethod("registerFileProtocol",
|
|
|
|
&ProtocolNS::RegisterProtocolFor<ProtocolType::kFile>)
|
|
|
|
.SetMethod("registerHttpProtocol",
|
|
|
|
&ProtocolNS::RegisterProtocolFor<ProtocolType::kHttp>)
|
|
|
|
.SetMethod("registerStreamProtocol",
|
|
|
|
&ProtocolNS::RegisterProtocolFor<ProtocolType::kStream>)
|
2019-04-23 21:39:21 +00:00
|
|
|
.SetMethod("unregisterProtocol", &ProtocolNS::UnregisterProtocol)
|
|
|
|
.SetMethod("isProtocolRegistered", &ProtocolNS::IsProtocolRegistered)
|
|
|
|
.SetMethod("isProtocolHandled", &ProtocolNS::IsProtocolHandled)
|
|
|
|
.SetMethod("interceptStringProtocol", &Noop)
|
|
|
|
.SetMethod("interceptBufferProtocol", &Noop)
|
|
|
|
.SetMethod("interceptFileProtocol", &Noop)
|
|
|
|
.SetMethod("interceptHttpProtocol", &Noop)
|
|
|
|
.SetMethod("interceptStreamProtocol", &Noop)
|
|
|
|
.SetMethod("uninterceptProtocol", &Noop);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace api
|
|
|
|
} // namespace atom
|