2020-03-26 17:34:32 +00:00
|
|
|
// Copyright (c) 2020 Slack Technologies, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include <memory>
|
2020-04-01 07:38:40 +00:00
|
|
|
#include <utility>
|
2020-03-26 17:34:32 +00:00
|
|
|
|
|
|
|
#include "shell/browser/electron_browser_context.h"
|
2020-04-01 07:38:40 +00:00
|
|
|
#include "shell/browser/net/asar/asar_url_loader.h"
|
2020-03-26 17:34:32 +00:00
|
|
|
#include "shell/browser/protocol_registry.h"
|
|
|
|
|
|
|
|
namespace electron {
|
|
|
|
|
2020-04-01 07:38:40 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Provide support for accessing asar archives in file:// protocol.
|
|
|
|
class AsarURLLoaderFactory : public network::mojom::URLLoaderFactory {
|
|
|
|
public:
|
|
|
|
AsarURLLoaderFactory() {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
// network::mojom::URLLoaderFactory:
|
|
|
|
void CreateLoaderAndStart(
|
|
|
|
mojo::PendingReceiver<network::mojom::URLLoader> loader,
|
|
|
|
int32_t routing_id,
|
|
|
|
int32_t request_id,
|
|
|
|
uint32_t options,
|
|
|
|
const network::ResourceRequest& request,
|
|
|
|
mojo::PendingRemote<network::mojom::URLLoaderClient> client,
|
|
|
|
const net::MutableNetworkTrafficAnnotationTag& traffic_annotation)
|
|
|
|
override {
|
|
|
|
asar::CreateAsarURLLoader(request, std::move(loader), std::move(client),
|
|
|
|
new net::HttpResponseHeaders(""));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Clone(
|
|
|
|
mojo::PendingReceiver<network::mojom::URLLoaderFactory> loader) override {
|
|
|
|
receivers_.Add(this, std::move(loader));
|
|
|
|
}
|
|
|
|
|
|
|
|
mojo::ReceiverSet<network::mojom::URLLoaderFactory> receivers_;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2020-03-26 17:34:32 +00:00
|
|
|
// static
|
|
|
|
ProtocolRegistry* ProtocolRegistry::FromBrowserContext(
|
|
|
|
content::BrowserContext* context) {
|
|
|
|
return static_cast<ElectronBrowserContext*>(context)->protocol_registry();
|
|
|
|
}
|
|
|
|
|
|
|
|
ProtocolRegistry::ProtocolRegistry() {}
|
|
|
|
|
|
|
|
ProtocolRegistry::~ProtocolRegistry() = default;
|
|
|
|
|
|
|
|
void ProtocolRegistry::RegisterURLLoaderFactories(
|
2020-04-01 07:38:40 +00:00
|
|
|
URLLoaderFactoryType type,
|
2020-03-26 17:34:32 +00:00
|
|
|
content::ContentBrowserClient::NonNetworkURLLoaderFactoryMap* factories) {
|
2020-04-01 07:38:40 +00:00
|
|
|
// Override the default FileURLLoaderFactory to support asar archives.
|
|
|
|
if (type == URLLoaderFactoryType::kNavigation) {
|
|
|
|
// Always allow navigating to file:// URLs.
|
|
|
|
//
|
|
|
|
// Note that Chromium calls |emplace| to create the default file factory
|
|
|
|
// after this call, so it won't override our asar factory.
|
|
|
|
DCHECK(!base::Contains(*factories, url::kFileScheme));
|
|
|
|
factories->emplace(url::kFileScheme,
|
|
|
|
std::make_unique<AsarURLLoaderFactory>());
|
|
|
|
} else if (type == URLLoaderFactoryType::kDocumentSubResource) {
|
|
|
|
// Only support requesting file:// subresource URLs when Chromium does so,
|
|
|
|
// it is usually supported under file:// or about:blank documents.
|
|
|
|
auto file_factory = factories->find(url::kFileScheme);
|
|
|
|
if (file_factory != factories->end())
|
|
|
|
file_factory->second = std::make_unique<AsarURLLoaderFactory>();
|
|
|
|
}
|
|
|
|
|
2020-03-26 17:34:32 +00:00
|
|
|
for (const auto& it : handlers_) {
|
|
|
|
factories->emplace(it.first, std::make_unique<ElectronURLLoaderFactory>(
|
|
|
|
it.second.first, it.second.second));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProtocolRegistry::RegisterProtocol(ProtocolType type,
|
|
|
|
const std::string& scheme,
|
|
|
|
const ProtocolHandler& handler) {
|
|
|
|
return base::TryEmplace(handlers_, scheme, type, handler).second;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProtocolRegistry::UnregisterProtocol(const std::string& scheme) {
|
|
|
|
return handlers_.erase(scheme) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProtocolRegistry::IsProtocolRegistered(const std::string& scheme) {
|
|
|
|
return base::Contains(handlers_, scheme);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProtocolRegistry::InterceptProtocol(ProtocolType type,
|
|
|
|
const std::string& scheme,
|
|
|
|
const ProtocolHandler& handler) {
|
|
|
|
return base::TryEmplace(intercept_handlers_, scheme, type, handler).second;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProtocolRegistry::UninterceptProtocol(const std::string& scheme) {
|
|
|
|
return intercept_handlers_.erase(scheme) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProtocolRegistry::IsProtocolIntercepted(const std::string& scheme) {
|
|
|
|
return base::Contains(intercept_handlers_, scheme);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace electron
|