protocol_handler_registry.cc revision cedac228d2dd51db4b79ea1e72c7f249408ee061
15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/custom_handlers/protocol_handler_registry.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <utility>
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/bind.h"
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/command_line.h"
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/logging.h"
122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/prefs/pref_service.h"
137dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch#include "chrome/browser/chrome_notification_types.h"
145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/custom_handlers/register_protocol_handler_infobar_delegate.h"
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/net/chrome_url_request_context.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/profiles/profile_io_data.h"
175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/chrome_switches.h"
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/custom_handlers/protocol_handler.h"
195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/pref_names.h"
20cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)#include "components/pref_registry/pref_registry_syncable.h"
215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/public/browser/child_process_security_policy.h"
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "grit/generated_resources.h"
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/base/network_delegate.h"
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/url_request/url_request_redirect_job.h"
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ui/base/l10n/l10n_util.h"
265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using content::BrowserThread;
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using content::ChildProcessSecurityPolicy;
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace {
315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const ProtocolHandler& LookupHandler(
335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandlerRegistry::ProtocolHandlerMap& handler_map,
345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) {
355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerRegistry::ProtocolHandlerMap::const_iterator p =
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      handler_map.find(scheme);
375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p != handler_map.end())
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return p->second;
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return ProtocolHandler::EmptyProtocolHandler();
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// If true default protocol handlers will be removed if the OS level
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// registration for a protocol is no longer Chrome.
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ShouldRemoveHandlersNotInOS() {
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_LINUX)
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // We don't do this on Linux as the OS registration there is not reliable,
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // and Chrome OS doesn't have any notion of OS registration.
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // TODO(benwells): When Linux support is more reliable remove this
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // difference (http://crbug.com/88255).
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#else
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return ShellIntegration::CanSetAsDefaultProtocolClient() !=
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      ShellIntegration::SET_DEFAULT_NOT_ALLOWED;
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace
605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// IOThreadDelegate ------------------------------------------------------------
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// IOThreadDelegate is an IO thread specific object. Access to the class should
642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// all be done via the IO thread. The registry living on the UI thread makes
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// a best effort to update the IO object after local updates are completed.
662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)class ProtocolHandlerRegistry::IOThreadDelegate
672a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    : public base::RefCountedThreadSafe<
682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          ProtocolHandlerRegistry::IOThreadDelegate> {
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Creates a new instance. If |enabled| is true the registry is considered
725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // enabled on the IO thread.
732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  explicit IOThreadDelegate(bool enabled);
745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Returns true if the protocol has a default protocol handler.
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Should be called only from the IO thread.
775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool IsHandledProtocol(const std::string& scheme) const;
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Clears the default for the provided protocol.
805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Should be called only from the IO thread.
815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void ClearDefault(const std::string& scheme);
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Makes this ProtocolHandler the default handler for its protocol.
845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Should be called only from the IO thread.
855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void SetDefault(const ProtocolHandler& handler);
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Creates a URL request job for the given request if there is a matching
885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // protocol handler, returns NULL otherwise.
895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  net::URLRequestJob* MaybeCreateJob(
905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      net::URLRequest* request, net::NetworkDelegate* network_delegate) const;
915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Indicate that the registry has been enabled in the IO thread's
935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // copy of the data.
945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void Enable() { enabled_ = true; }
955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Indicate that the registry has been disabled in the IO thread's copy of
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // the data.
985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void Disable() { enabled_ = false; }
995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) private:
1012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  friend class base::RefCountedThreadSafe<IOThreadDelegate>;
1022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual ~IOThreadDelegate();
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Copy of protocol handlers use only on the IO thread.
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerRegistry::ProtocolHandlerMap default_handlers_;
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Is the registry enabled on the IO thread.
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enabled_;
1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DISALLOW_COPY_AND_ASSIGN(IOThreadDelegate);
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)ProtocolHandlerRegistry::IOThreadDelegate::IOThreadDelegate(bool)
1142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    : enabled_(true) {}
1152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)ProtocolHandlerRegistry::IOThreadDelegate::~IOThreadDelegate() {}
1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool ProtocolHandlerRegistry::IOThreadDelegate::IsHandledProtocol(
1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
1195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
1205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return enabled_ && !LookupHandler(default_handlers_, scheme).IsEmpty();
1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ProtocolHandlerRegistry::IOThreadDelegate::ClearDefault(
1242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& scheme) {
1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  default_handlers_.erase(scheme);
1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ProtocolHandlerRegistry::IOThreadDelegate::SetDefault(
1302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const ProtocolHandler& handler) {
1315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ClearDefault(handler.protocol());
1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  default_handlers_.insert(std::make_pair(handler.protocol(), handler));
1345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Create a new job for the supplied |URLRequest| if a default handler
1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// is registered and the associated handler is able to interpret
1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// the url from |request|.
1392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)net::URLRequestJob* ProtocolHandlerRegistry::IOThreadDelegate::MaybeCreateJob(
1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    net::URLRequest* request, net::NetworkDelegate* network_delegate) const {
1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandler handler = LookupHandler(default_handlers_,
1445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                          request->url().scheme());
1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (handler.IsEmpty())
1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  GURL translated_url(handler.TranslateUrl(request->url()));
1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!translated_url.is_valid())
1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return new net::URLRequestRedirectJob(
1532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      request, network_delegate, translated_url,
154a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      net::URLRequestRedirectJob::REDIRECT_307_TEMPORARY_REDIRECT,
155a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      "Protocol Handler Registry");
1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// JobInterceptorFactory -------------------------------------------------------
1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Instances of JobInterceptorFactory are produced for ownership by the IO
1615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// thread where it handler URL requests. We should never hold
1625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// any pointers on this class, only produce them in response to
1632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// requests via |ProtocolHandlerRegistry::CreateJobInterceptorFactory|.
1642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)ProtocolHandlerRegistry::JobInterceptorFactory::JobInterceptorFactory(
1652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    IOThreadDelegate* io_thread_delegate)
1662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    : io_thread_delegate_(io_thread_delegate) {
167868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  DCHECK(io_thread_delegate_.get());
1682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DetachFromThread();
1692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)ProtocolHandlerRegistry::JobInterceptorFactory::~JobInterceptorFactory() {
1725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ProtocolHandlerRegistry::JobInterceptorFactory::Chain(
1752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    scoped_ptr<net::URLRequestJobFactory> job_factory) {
1762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  job_factory_ = job_factory.Pass();
1775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)net::URLRequestJob*
1802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)ProtocolHandlerRegistry::JobInterceptorFactory::
1812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)MaybeCreateJobWithProtocolHandler(
1822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& scheme,
1832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    net::URLRequest* request,
1842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    net::NetworkDelegate* network_delegate) const {
1855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
1862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  net::URLRequestJob* job = io_thread_delegate_->MaybeCreateJob(
1872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      request, network_delegate);
1882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (job)
1892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return job;
1902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return job_factory_->MaybeCreateJobWithProtocolHandler(
1912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      scheme, request, network_delegate);
1925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool ProtocolHandlerRegistry::JobInterceptorFactory::IsHandledProtocol(
1952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& scheme) const {
1965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
1972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return io_thread_delegate_->IsHandledProtocol(scheme) ||
1982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      job_factory_->IsHandledProtocol(scheme);
1992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
2005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool ProtocolHandlerRegistry::JobInterceptorFactory::IsHandledURL(
2022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const GURL& url) const {
2032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
2042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return (url.is_valid() &&
2052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      io_thread_delegate_->IsHandledProtocol(url.scheme())) ||
2062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      job_factory_->IsHandledURL(url);
2075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
209b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles)bool ProtocolHandlerRegistry::JobInterceptorFactory::IsSafeRedirectTarget(
210b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles)    const GURL& location) const {
211b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
212b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles)  return job_factory_->IsSafeRedirectTarget(location);
213b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles)}
214b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles)
2155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// DefaultClientObserver ------------------------------------------------------
2165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::DefaultClientObserver::DefaultClientObserver(
2185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ProtocolHandlerRegistry* registry)
2195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    : worker_(NULL),
2205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      registry_(registry) {
2215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(registry_);
2225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::DefaultClientObserver::~DefaultClientObserver() {
2255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (worker_)
2265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    worker_->ObserverDestroyed();
2275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DefaultClientObserverList::iterator iter = std::find(
2295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      registry_->default_client_observers_.begin(),
2305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      registry_->default_client_observers_.end(), this);
2315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  registry_->default_client_observers_.erase(iter);
2325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::DefaultClientObserver::SetDefaultWebClientUIState(
2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ShellIntegration::DefaultWebClientUIState state) {
2365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (worker_) {
2375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (ShouldRemoveHandlersNotInOS() &&
2385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        (state == ShellIntegration::STATE_NOT_DEFAULT)) {
2395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      registry_->ClearDefault(worker_->protocol());
2405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
2415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  } else {
2425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    NOTREACHED();
2435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::DefaultClientObserver::
2475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    IsInteractiveSetDefaultPermitted() {
2485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
2495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::DefaultClientObserver::SetWorker(
2525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ShellIntegration::DefaultProtocolClientWorker* worker) {
2535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  worker_ = worker;
2545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::DefaultClientObserver::IsOwnedByWorker() {
2575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
2585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Delegate --------------------------------------------------------------------
2615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::Delegate::~Delegate() {}
2635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Delegate::RegisterExternalHandler(
2655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& protocol) {
2665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ChildProcessSecurityPolicy* policy =
2675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ChildProcessSecurityPolicy::GetInstance();
2685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!policy->IsWebSafeScheme(protocol)) {
2695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    policy->RegisterWebSafeScheme(protocol);
2705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Delegate::DeregisterExternalHandler(
2745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& protocol) {
2755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::Delegate::IsExternalHandlerRegistered(
2785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& protocol) {
2795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // NOTE(koz): This function is safe to call from any thread, despite living
2805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // in ProfileIOData.
2815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return ProfileIOData::IsHandledProtocol(protocol);
2825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ShellIntegration::DefaultProtocolClientWorker*
2855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::Delegate::CreateShellWorker(
2865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ShellIntegration::DefaultWebClientObserver* observer,
2875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& protocol) {
2885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return new ShellIntegration::DefaultProtocolClientWorker(observer, protocol);
2895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::DefaultClientObserver*
2925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::Delegate::CreateShellObserver(
2935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ProtocolHandlerRegistry* registry) {
2945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return new DefaultClientObserver(registry);
2955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Delegate::RegisterWithOSAsDefaultClient(
2985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& protocol, ProtocolHandlerRegistry* registry) {
2995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DefaultClientObserver* observer = CreateShellObserver(registry);
3005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // The worker pointer is reference counted. While it is running the
3015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // message loops of the FILE and UI thread will hold references to it
3025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // and it will be automatically freed once all its tasks have finished.
3035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  scoped_refptr<ShellIntegration::DefaultProtocolClientWorker> worker;
3045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  worker = CreateShellWorker(observer, protocol);
305868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  observer->SetWorker(worker.get());
3065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  registry->default_client_observers_.push_back(observer);
3075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  worker->StartSetAsDefault();
3085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// ProtocolHandlerRegistry -----------------------------------------------------
3115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::ProtocolHandlerRegistry(Profile* profile,
3135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Delegate* delegate)
3145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    : profile_(profile),
3155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      delegate_(delegate),
3165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      enabled_(true),
3175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      is_loading_(false),
3185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      is_loaded_(false),
3192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      io_thread_delegate_(new IOThreadDelegate(enabled_)){
3205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::SilentlyHandleRegisterHandlerRequest(
3235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
3245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (handler.IsEmpty() || !CanSchemeBeOverridden(handler.protocol()))
3255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;
3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!enabled() || IsRegistered(handler) || HasIgnoredEquivalent(handler))
3285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;
3295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (AttemptReplace(handler))
3315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;
3325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
3345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::OnAcceptRegisterProtocolHandler(
3375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
3385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
3395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  RegisterProtocolHandler(handler);
3405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  SetDefault(handler);
3415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
3425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
3435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::OnDenyRegisterProtocolHandler(
3465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
3475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
3485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  RegisterProtocolHandler(handler);
3495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
3505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
3515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::OnIgnoreRegisterProtocolHandler(
3545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
3555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
3565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  IgnoreProtocolHandler(handler);
3575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
3585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
3595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::AttemptReplace(const ProtocolHandler& handler) {
3625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
3635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandler old_default = GetHandlerFor(handler.protocol());
3645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool make_new_handler_default = handler.IsSameOrigin(old_default);
3655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList to_replace(GetReplacedHandlers(handler));
3665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (to_replace.empty())
3675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
3685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (ProtocolHandlerList::iterator p = to_replace.begin();
3695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       p != to_replace.end(); ++p) {
3705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    RemoveHandler(*p);
3715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (make_new_handler_default) {
3735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    OnAcceptRegisterProtocolHandler(handler);
3745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  } else {
3755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    InsertHandler(handler);
3765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    NotifyChanged();
3775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
3795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::ProtocolHandlerList
3825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetReplacedHandlers(
3835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) const {
3845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList replaced_handlers;
3855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(handler.protocol());
3865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!handlers)
3875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return replaced_handlers;
3885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (ProtocolHandlerList::const_iterator p = handlers->begin();
3895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       p != handlers->end(); p++) {
3905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (handler.IsSameOrigin(*p)) {
3915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      replaced_handlers.push_back(*p);
3925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
3935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return replaced_handlers;
3955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::ClearDefault(const std::string& scheme) {
3985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
3995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  default_handlers_.erase(scheme);
4015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BrowserThread::PostTask(
4025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::IO,
4035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
4042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&IOThreadDelegate::ClearDefault, io_thread_delegate_, scheme));
4055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
4065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
4075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::IsDefault(
4105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) const {
4115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
4125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return GetHandlerFor(handler.protocol()) == handler;
4135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::InstallDefaultsForChromeOS() {
4165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_CHROMEOS)
4175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Only chromeos has default protocol handlers at this point.
4185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddPredefinedHandler(
4195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      ProtocolHandler::CreateProtocolHandler(
4205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          "mailto",
421cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          GURL(l10n_util::GetStringUTF8(IDS_GOOGLE_MAILTO_HANDLER_URL))));
4225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddPredefinedHandler(
4235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      ProtocolHandler::CreateProtocolHandler(
4245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          "webcal",
425cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)          GURL(l10n_util::GetStringUTF8(IDS_GOOGLE_WEBCAL_HANDLER_URL))));
4265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#else
4275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();  // this method should only ever be called in chromeos.
4285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
4295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::InitProtocolSettings() {
4325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
4335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Any further default additions to the table will get rejected from now on.
4355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  is_loaded_ = true;
4365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  is_loading_ = true;
4375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  PrefService* prefs = profile_->GetPrefs();
4395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (prefs->HasPrefPath(prefs::kCustomHandlersEnabled)) {
4405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (prefs->GetBoolean(prefs::kCustomHandlersEnabled)) {
4415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      Enable();
4425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    } else {
4435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      Disable();
4445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
4455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  std::vector<const base::DictionaryValue*> registered_handlers =
4475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GetHandlersFromPref(prefs::kRegisteredProtocolHandlers);
4485d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (std::vector<const base::DictionaryValue*>::const_iterator p =
4495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       registered_handlers.begin();
4505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       p != registered_handlers.end(); ++p) {
4515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ProtocolHandler handler = ProtocolHandler::CreateProtocolHandler(*p);
4525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    RegisterProtocolHandler(handler);
4535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    bool is_default = false;
4545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if ((*p)->GetBoolean("default", &is_default) && is_default) {
4555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      SetDefault(handler);
4565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
4575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4585d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  std::vector<const base::DictionaryValue*> ignored_handlers =
4595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    GetHandlersFromPref(prefs::kIgnoredProtocolHandlers);
4605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (std::vector<const base::DictionaryValue*>::const_iterator p =
4615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       ignored_handlers.begin();
4625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       p != ignored_handlers.end(); ++p) {
4635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    IgnoreProtocolHandler(ProtocolHandler::CreateProtocolHandler(*p));
4645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  is_loading_ = false;
4665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // For each default protocol handler, check that we are still registered
4685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // with the OS as the default application.
4695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (ShouldRemoveHandlersNotInOS()) {
4705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    for (ProtocolHandlerMap::const_iterator p = default_handlers_.begin();
4715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)         p != default_handlers_.end(); ++p) {
4725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      ProtocolHandler handler = p->second;
4735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      DefaultClientObserver* observer = delegate_->CreateShellObserver(this);
4745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      scoped_refptr<ShellIntegration::DefaultProtocolClientWorker> worker;
4755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      worker = delegate_->CreateShellWorker(observer, handler.protocol());
476868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)      observer->SetWorker(worker.get());
4775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      default_client_observers_.push_back(observer);
4785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      worker->StartCheckIsDefault();
4795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
4805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int ProtocolHandlerRegistry::GetHandlerIndex(const std::string& scheme) const {
4845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
4855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandler& handler = GetHandlerFor(scheme);
4865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (handler.IsEmpty())
4875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return -1;
4885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(scheme);
4895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!handlers)
4905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return -1;
4915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::const_iterator p;
4935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int i;
4945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (i = 0, p = handlers->begin(); p != handlers->end(); ++p, ++i) {
4955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (*p == handler)
4965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return i;
4975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return -1;
4995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::ProtocolHandlerList
5025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetHandlersFor(
5035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
5045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::const_iterator p = protocol_handlers_.find(scheme);
5065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p == protocol_handlers_.end()) {
5075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return ProtocolHandlerList();
5085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return p->second;
5105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::ProtocolHandlerList
5135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetIgnoredHandlers() {
5145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return ignored_protocol_handlers_;
5155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::GetRegisteredProtocols(
5185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    std::vector<std::string>* output) const {
5195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::const_iterator p;
5215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (p = protocol_handlers_.begin(); p != protocol_handlers_.end(); ++p) {
5225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!p->second.empty())
5235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      output->push_back(p->first);
5245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::CanSchemeBeOverridden(
5285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
5295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(scheme);
5315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // If we already have a handler for this scheme, we can add more.
5325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (handlers != NULL && !handlers->empty())
5335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;
5345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Don't override a scheme if it already has an external handler.
5355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return !delegate_->IsExternalHandlerRegistered(scheme);
5365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::IsRegistered(
5395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) const {
5405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(handler.protocol());
5425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!handlers) {
5435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
5445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return std::find(handlers->begin(), handlers->end(), handler) !=
5465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      handlers->end();
5475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::IsIgnored(const ProtocolHandler& handler) const {
5505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::const_iterator i;
5525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (i = ignored_protocol_handlers_.begin();
5535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       i != ignored_protocol_handlers_.end(); ++i) {
5545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (*i == handler) {
5555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return true;
5565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
5575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
5595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::HasRegisteredEquivalent(
5625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) const {
5635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(handler.protocol());
5655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!handlers) {
5665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
5675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::const_iterator i;
5695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (i = handlers->begin(); i != handlers->end(); ++i) {
5705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (handler.IsEquivalent(*i)) {
5715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return true;
5725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
5735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
5755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::HasIgnoredEquivalent(
5785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) const {
5795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::const_iterator i;
5815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (i = ignored_protocol_handlers_.begin();
5825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       i != ignored_protocol_handlers_.end(); ++i) {
5835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (handler.IsEquivalent(*i)) {
5845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return true;
5855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
5865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
5885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::RemoveIgnoredHandler(
5915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
5925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool should_notify = false;
5945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::iterator p = std::find(
5955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      ignored_protocol_handlers_.begin(), ignored_protocol_handlers_.end(),
5965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      handler);
5975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p != ignored_protocol_handlers_.end()) {
5985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ignored_protocol_handlers_.erase(p);
5995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Save();
6005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    should_notify = true;
6015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (should_notify)
6035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    NotifyChanged();
6045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::IsHandledProtocol(
6075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
6085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return enabled_ && !GetHandlerFor(scheme).IsEmpty();
6105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::RemoveHandler(
6135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
6145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList& handlers = protocol_handlers_[handler.protocol()];
6165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::iterator p =
6175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      std::find(handlers.begin(), handlers.end(), handler);
6185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p != handlers.end()) {
6195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    handlers.erase(p);
6205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMap::iterator q = default_handlers_.find(handler.protocol());
6225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (q != default_handlers_.end() && q->second == handler) {
6235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Make the new top handler in the list the default.
6245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!handlers.empty()) {
6255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // NOTE We pass a copy because SetDefault() modifies handlers.
6265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      SetDefault(ProtocolHandler(handlers[0]));
6275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    } else {
6285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::PostTask(
6295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          BrowserThread::IO, FROM_HERE,
6302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          base::Bind(&IOThreadDelegate::ClearDefault, io_thread_delegate_,
6312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                     q->second.protocol()));
6325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      default_handlers_.erase(q);
6345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
6355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!IsHandledProtocol(handler.protocol())) {
6385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delegate_->DeregisterExternalHandler(handler.protocol());
6395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
6415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
6425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::RemoveDefaultHandler(const std::string& scheme) {
6455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandler current_default = GetHandlerFor(scheme);
6475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!current_default.IsEmpty())
6485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    RemoveHandler(current_default);
6495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const ProtocolHandler& ProtocolHandlerRegistry::GetHandlerFor(
6525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
6535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return LookupHandler(default_handlers_, scheme);
6555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Enable() {
6585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enabled_) {
6605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
6615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  enabled_ = true;
6635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BrowserThread::PostTask(
6645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::IO,
6655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
6662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&IOThreadDelegate::Enable, io_thread_delegate_));
6675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMap::const_iterator p;
6695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (p = default_handlers_.begin(); p != default_handlers_.end(); ++p) {
6705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delegate_->RegisterExternalHandler(p->first);
6715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
6735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
6745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Disable() {
6775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!enabled_) {
6795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
6805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  enabled_ = false;
6825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BrowserThread::PostTask(
6835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::IO,
6845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
6852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&IOThreadDelegate::Disable, io_thread_delegate_));
6865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMap::const_iterator p;
6885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (p = default_handlers_.begin(); p != default_handlers_.end(); ++p) {
6895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delegate_->DeregisterExternalHandler(p->first);
6905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
6925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
6935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Shutdown() {
6965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  delegate_.reset(NULL);
6985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // We free these now in case there are any outstanding workers running. If
6995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // we didn't free them they could respond to workers and try to update the
7005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // protocol handler registry after it was deleted.
7015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Observers remove themselves from this list when they are deleted; so
7025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // we delete the last item until none are left in the list.
7035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while (!default_client_observers_.empty()) {
7045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delete default_client_observers_.back();
7055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static
7097dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochvoid ProtocolHandlerRegistry::RegisterProfilePrefs(
710c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    user_prefs::PrefRegistrySyncable* registry) {
7112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  registry->RegisterListPref(prefs::kRegisteredProtocolHandlers,
712c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                             user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
7132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  registry->RegisterListPref(prefs::kIgnoredProtocolHandlers,
714c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                             user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
715c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  registry->RegisterBooleanPref(
716c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      prefs::kCustomHandlersEnabled,
717c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      true,
718c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
7195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::~ProtocolHandlerRegistry() {
7225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(default_client_observers_.empty());
7245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::PromoteHandler(const ProtocolHandler& handler) {
7275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(IsRegistered(handler));
7295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::iterator p =
7305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      protocol_handlers_.find(handler.protocol());
7315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList& list = p->second;
7325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  list.erase(std::find(list.begin(), list.end(), handler));
7335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  list.insert(list.begin(), handler);
7345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Save() {
7375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (is_loading_) {
7395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
7405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7415d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::Value> registered_protocol_handlers(
7425d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      EncodeRegisteredHandlers());
7435d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::Value> ignored_protocol_handlers(EncodeIgnoredHandlers());
7445d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::Value> enabled(base::Value::CreateBooleanValue(enabled_));
7455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  profile_->GetPrefs()->Set(prefs::kRegisteredProtocolHandlers,
7465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      *registered_protocol_handlers);
7475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  profile_->GetPrefs()->Set(prefs::kIgnoredProtocolHandlers,
7485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      *ignored_protocol_handlers);
7495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  profile_->GetPrefs()->Set(prefs::kCustomHandlersEnabled, *enabled);
7505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const ProtocolHandlerRegistry::ProtocolHandlerList*
7535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetHandlerList(
7545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
7555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::const_iterator p = protocol_handlers_.find(scheme);
7575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p == protocol_handlers_.end()) {
7585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
7595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return &p->second;
7615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::SetDefault(const ProtocolHandler& handler) {
7645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMap::const_iterator p = default_handlers_.find(
7665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      handler.protocol());
7675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // If we're not loading, and we are setting a default for a new protocol,
7685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // register with the OS.
7695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!is_loading_ && p == default_handlers_.end())
7705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      delegate_->RegisterWithOSAsDefaultClient(handler.protocol(), this);
7715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  default_handlers_.erase(handler.protocol());
7725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  default_handlers_.insert(std::make_pair(handler.protocol(), handler));
7735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  PromoteHandler(handler);
7745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BrowserThread::PostTask(
7755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::IO,
7765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
7772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&IOThreadDelegate::SetDefault, io_thread_delegate_, handler));
7785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::InsertHandler(const ProtocolHandler& handler) {
7815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::iterator p =
7835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      protocol_handlers_.find(handler.protocol());
7845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p != protocol_handlers_.end()) {
7865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    p->second.push_back(handler);
7875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
7885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList new_list;
7915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  new_list.push_back(handler);
7925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  protocol_handlers_[handler.protocol()] = new_list;
7935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7955d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)base::Value* ProtocolHandlerRegistry::EncodeRegisteredHandlers() {
7965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7975d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* protocol_handlers = new base::ListValue();
7985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (ProtocolHandlerMultiMap::iterator i = protocol_handlers_.begin();
7995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       i != protocol_handlers_.end(); ++i) {
8005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    for (ProtocolHandlerList::iterator j = i->second.begin();
8015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)         j != i->second.end(); ++j) {
8025d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      base::DictionaryValue* encoded = j->Encode();
8035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      if (IsDefault(*j)) {
8045d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)        encoded->Set("default", base::Value::CreateBooleanValue(true));
8055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      }
8065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      protocol_handlers->Append(encoded);
8075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
8085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return protocol_handlers;
8105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8125d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)base::Value* ProtocolHandlerRegistry::EncodeIgnoredHandlers() {
8135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8145d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* handlers = new base::ListValue();
8155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (ProtocolHandlerList::iterator i = ignored_protocol_handlers_.begin();
8165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       i != ignored_protocol_handlers_.end(); ++i) {
8175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    handlers->Append(i->Encode());
8185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return handlers;
8205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::NotifyChanged() {
8235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  content::NotificationService::current()->Notify(
8255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      chrome::NOTIFICATION_PROTOCOL_HANDLER_REGISTRY_CHANGED,
8265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      content::Source<Profile>(profile_),
8275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      content::NotificationService::NoDetails());
8285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::RegisterProtocolHandler(
8315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
8325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(CanSchemeBeOverridden(handler.protocol()));
8345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(!handler.IsEmpty());
8355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (IsRegistered(handler)) {
8365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
8375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enabled_ && !delegate_->IsExternalHandlerRegistered(handler.protocol()))
8395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delegate_->RegisterExternalHandler(handler.protocol());
8405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  InsertHandler(handler);
8415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8435d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)std::vector<const base::DictionaryValue*>
8445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetHandlersFromPref(const char* pref_name) const {
8455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  std::vector<const base::DictionaryValue*> result;
8475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  PrefService* prefs = profile_->GetPrefs();
8485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!prefs->HasPrefPath(pref_name)) {
8495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return result;
8505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  const base::ListValue* handlers = prefs->GetList(pref_name);
8535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (handlers) {
8545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    for (size_t i = 0; i < handlers->GetSize(); ++i) {
8555d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      const base::DictionaryValue* dict;
8565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      if (!handlers->GetDictionary(i, &dict))
8575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        continue;
8585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      if (ProtocolHandler::IsValidDict(dict)) {
8595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        result.push_back(dict);
8605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      }
8615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
8625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return result;
8645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::IgnoreProtocolHandler(
8675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
8685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ignored_protocol_handlers_.push_back(handler);
8705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::AddPredefinedHandler(
8735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
8745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(!is_loaded_);  // Must be called prior InitProtocolSettings.
8755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  RegisterProtocolHandler(handler);
8765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  SetDefault(handler);
8775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)scoped_ptr<ProtocolHandlerRegistry::JobInterceptorFactory>
8802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)ProtocolHandlerRegistry::CreateJobInterceptorFactory() {
8815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // this is always created on the UI thread (in profile_io's
8835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // InitializeOnUIThread. Any method calls must be done
8845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // on the IO thread (this is checked).
885868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  return scoped_ptr<JobInterceptorFactory>(
886868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)      new JobInterceptorFactory(io_thread_delegate_.get()));
8875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
888