// 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> #include <utility> #include "services/network/public/cpp/self_deleting_url_loader_factory.h" #include "shell/browser/electron_browser_context.h" #include "shell/browser/net/asar/asar_url_loader.h" #include "shell/browser/protocol_registry.h" namespace electron { namespace { // Provide support for accessing asar archives in file:// protocol. class AsarURLLoaderFactory : public network::SelfDeletingURLLoaderFactory { public: static mojo::PendingRemote<network::mojom::URLLoaderFactory> Create() { mojo::PendingRemote<network::mojom::URLLoaderFactory> pending_remote; // The AsarURLLoaderFactory will delete itself when there are no more // receivers - see the SelfDeletingURLLoaderFactory::OnDisconnect method. new AsarURLLoaderFactory(pending_remote.InitWithNewPipeAndPassReceiver()); return pending_remote; } private: AsarURLLoaderFactory( mojo::PendingReceiver<network::mojom::URLLoaderFactory> factory_receiver) : network::SelfDeletingURLLoaderFactory(std::move(factory_receiver)) {} ~AsarURLLoaderFactory() override = default; // 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("")); } }; } // namespace // static ProtocolRegistry* ProtocolRegistry::FromBrowserContext( content::BrowserContext* context) { return static_cast<ElectronBrowserContext*>(context)->protocol_registry(); } ProtocolRegistry::ProtocolRegistry() {} ProtocolRegistry::~ProtocolRegistry() = default; void ProtocolRegistry::RegisterURLLoaderFactories( URLLoaderFactoryType type, content::ContentBrowserClient::NonNetworkURLLoaderFactoryMap* factories) { // 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, AsarURLLoaderFactory::Create()); } 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 = AsarURLLoaderFactory::Create(); } for (const auto& it : handlers_) { factories->emplace(it.first, ElectronURLLoaderFactory::Create( 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