protocol_handler_registry.cc revision 116680a4aac90f2aa7413d9095a592090648e557
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));
33946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  RegisterProtocolHandler(handler, USER);
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));
34846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  RegisterProtocolHandler(handler, USER);
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));
35646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  IgnoreProtocolHandler(handler, USER);
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)  }
44646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)
44746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  RegisterProtocolHandlersFromPref(prefs::kPolicyRegisteredProtocolHandlers,
44846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                                   POLICY);
44946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  RegisterProtocolHandlersFromPref(prefs::kRegisteredProtocolHandlers, USER);
45046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  IgnoreProtocolHandlersFromPref(prefs::kPolicyIgnoredProtocolHandlers, POLICY);
45146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  IgnoreProtocolHandlersFromPref(prefs::kIgnoredProtocolHandlers, USER);
45246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)
4535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  is_loading_ = false;
4545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // For each default protocol handler, check that we are still registered
4565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // with the OS as the default application.
4575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (ShouldRemoveHandlersNotInOS()) {
4585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    for (ProtocolHandlerMap::const_iterator p = default_handlers_.begin();
4595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)         p != default_handlers_.end(); ++p) {
4605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      ProtocolHandler handler = p->second;
4615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      DefaultClientObserver* observer = delegate_->CreateShellObserver(this);
4625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      scoped_refptr<ShellIntegration::DefaultProtocolClientWorker> worker;
4635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      worker = delegate_->CreateShellWorker(observer, handler.protocol());
464868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)      observer->SetWorker(worker.get());
4655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      default_client_observers_.push_back(observer);
4665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      worker->StartCheckIsDefault();
4675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
4685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int ProtocolHandlerRegistry::GetHandlerIndex(const std::string& scheme) const {
4725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
4735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandler& handler = GetHandlerFor(scheme);
4745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (handler.IsEmpty())
4755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return -1;
4765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(scheme);
4775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!handlers)
4785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return -1;
4795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::const_iterator p;
4815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int i;
4825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (i = 0, p = handlers->begin(); p != handlers->end(); ++p, ++i) {
4835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (*p == handler)
4845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return i;
4855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return -1;
4875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::ProtocolHandlerList
4905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetHandlersFor(
4915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
4925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
4935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::const_iterator p = protocol_handlers_.find(scheme);
4945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p == protocol_handlers_.end()) {
4955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return ProtocolHandlerList();
4965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return p->second;
4985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::ProtocolHandlerList
5015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetIgnoredHandlers() {
5025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return ignored_protocol_handlers_;
5035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::GetRegisteredProtocols(
5065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    std::vector<std::string>* output) const {
5075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::const_iterator p;
5095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (p = protocol_handlers_.begin(); p != protocol_handlers_.end(); ++p) {
5105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!p->second.empty())
5115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      output->push_back(p->first);
5125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::CanSchemeBeOverridden(
5165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
5175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(scheme);
5195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // If we already have a handler for this scheme, we can add more.
5205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (handlers != NULL && !handlers->empty())
5215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;
5225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Don't override a scheme if it already has an external handler.
5235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return !delegate_->IsExternalHandlerRegistered(scheme);
5245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::IsRegistered(
5275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) const {
5285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(handler.protocol());
5305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!handlers) {
5315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
5325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return std::find(handlers->begin(), handlers->end(), handler) !=
5345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      handlers->end();
5355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::IsIgnored(const ProtocolHandler& handler) const {
5385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::const_iterator i;
5405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (i = ignored_protocol_handlers_.begin();
5415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       i != ignored_protocol_handlers_.end(); ++i) {
5425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (*i == handler) {
5435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return true;
5445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
5455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
5475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::HasRegisteredEquivalent(
5505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) const {
5515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const ProtocolHandlerList* handlers = GetHandlerList(handler.protocol());
5535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!handlers) {
5545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
5555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::const_iterator i;
5575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (i = handlers->begin(); i != handlers->end(); ++i) {
5585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (handler.IsEquivalent(*i)) {
5595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return true;
5605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
5615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
5635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::HasIgnoredEquivalent(
5665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) const {
5675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList::const_iterator i;
5695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (i = ignored_protocol_handlers_.begin();
5705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       i != ignored_protocol_handlers_.end(); ++i) {
5715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (handler.IsEquivalent(*i)) {
5725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return true;
5735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
5745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
5765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::RemoveIgnoredHandler(
5795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
5805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool should_notify = false;
58246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (HandlerExists(handler, ignored_protocol_handlers_) &&
58346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      HandlerExists(handler, user_ignored_protocol_handlers_)) {
58446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    EraseHandler(handler, &user_ignored_protocol_handlers_);
5855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Save();
58646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    if (!HandlerExists(handler, policy_ignored_protocol_handlers_)) {
58746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      EraseHandler(handler, &ignored_protocol_handlers_);
58846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      should_notify = true;
58946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    }
5905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (should_notify)
5925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    NotifyChanged();
5935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProtocolHandlerRegistry::IsHandledProtocol(
5965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
5975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
5985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return enabled_ && !GetHandlerFor(scheme).IsEmpty();
5995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::RemoveHandler(
6025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
6035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList& handlers = protocol_handlers_[handler.protocol()];
60546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  bool erase_success = false;
60646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (HandlerExists(handler, handlers) &&
60746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      HandlerExists(handler, &user_protocol_handlers_)) {
60846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    EraseHandler(handler, &user_protocol_handlers_);
60946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    if (!HandlerExists(handler, &policy_protocol_handlers_)) {
61046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      erase_success = true;
61146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      EraseHandler(handler, &protocol_handlers_);
61246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    }
6135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMap::iterator q = default_handlers_.find(handler.protocol());
61546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (erase_success && q != default_handlers_.end() && q->second == handler) {
6165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Make the new top handler in the list the default.
6175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!handlers.empty()) {
6185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // NOTE We pass a copy because SetDefault() modifies handlers.
6195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      SetDefault(ProtocolHandler(handlers[0]));
6205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    } else {
6215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::PostTask(
6225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          BrowserThread::IO, FROM_HERE,
6232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          base::Bind(&IOThreadDelegate::ClearDefault, io_thread_delegate_,
6242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                     q->second.protocol()));
6255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      default_handlers_.erase(q);
6275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
6285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
63046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (erase_success && !IsHandledProtocol(handler.protocol())) {
6315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delegate_->DeregisterExternalHandler(handler.protocol());
6325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
63446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (erase_success)
63546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    NotifyChanged();
6365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::RemoveDefaultHandler(const std::string& scheme) {
6395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandler current_default = GetHandlerFor(scheme);
6415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!current_default.IsEmpty())
6425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    RemoveHandler(current_default);
6435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const ProtocolHandler& ProtocolHandlerRegistry::GetHandlerFor(
6465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
6475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return LookupHandler(default_handlers_, scheme);
6495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Enable() {
6525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enabled_) {
6545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
6555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  enabled_ = true;
6575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BrowserThread::PostTask(
6585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::IO,
6595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
6602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&IOThreadDelegate::Enable, io_thread_delegate_));
6615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMap::const_iterator p;
6635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (p = default_handlers_.begin(); p != default_handlers_.end(); ++p) {
6645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delegate_->RegisterExternalHandler(p->first);
6655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
6675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
6685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Disable() {
6715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!enabled_) {
6735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
6745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  enabled_ = false;
6765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BrowserThread::PostTask(
6775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::IO,
6785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
6792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&IOThreadDelegate::Disable, io_thread_delegate_));
6805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMap::const_iterator p;
6825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (p = default_handlers_.begin(); p != default_handlers_.end(); ++p) {
6835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delegate_->DeregisterExternalHandler(p->first);
6845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Save();
6865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NotifyChanged();
6875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Shutdown() {
6905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
6915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  delegate_.reset(NULL);
6925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // We free these now in case there are any outstanding workers running. If
6935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // we didn't free them they could respond to workers and try to update the
6945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // protocol handler registry after it was deleted.
6955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Observers remove themselves from this list when they are deleted; so
6965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // we delete the last item until none are left in the list.
6975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while (!default_client_observers_.empty()) {
6985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delete default_client_observers_.back();
6995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static
7037dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochvoid ProtocolHandlerRegistry::RegisterProfilePrefs(
704c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    user_prefs::PrefRegistrySyncable* registry) {
7052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  registry->RegisterListPref(prefs::kRegisteredProtocolHandlers,
706c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                             user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
7072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  registry->RegisterListPref(prefs::kIgnoredProtocolHandlers,
708c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                             user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
70946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  registry->RegisterListPref(prefs::kPolicyRegisteredProtocolHandlers,
71046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                             user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
71146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  registry->RegisterListPref(prefs::kPolicyIgnoredProtocolHandlers,
71246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                             user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
713c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  registry->RegisterBooleanPref(
714c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      prefs::kCustomHandlersEnabled,
715c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      true,
716c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
7175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::~ProtocolHandlerRegistry() {
7205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(default_client_observers_.empty());
7225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::PromoteHandler(const ProtocolHandler& handler) {
7255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(IsRegistered(handler));
7275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::iterator p =
7285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      protocol_handlers_.find(handler.protocol());
7295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList& list = p->second;
7305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  list.erase(std::find(list.begin(), list.end(), handler));
7315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  list.insert(list.begin(), handler);
7325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::Save() {
7355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (is_loading_) {
7375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
7385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7395d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::Value> registered_protocol_handlers(
7405d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      EncodeRegisteredHandlers());
7415d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::Value> ignored_protocol_handlers(EncodeIgnoredHandlers());
7425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  profile_->GetPrefs()->Set(prefs::kRegisteredProtocolHandlers,
7435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      *registered_protocol_handlers);
7445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  profile_->GetPrefs()->Set(prefs::kIgnoredProtocolHandlers,
7455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      *ignored_protocol_handlers);
746116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  profile_->GetPrefs()->SetBoolean(prefs::kCustomHandlersEnabled, enabled_);
7475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const ProtocolHandlerRegistry::ProtocolHandlerList*
7505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetHandlerList(
7515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const std::string& scheme) const {
7525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::const_iterator p = protocol_handlers_.find(scheme);
7545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p == protocol_handlers_.end()) {
7555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
7565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return &p->second;
7585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::SetDefault(const ProtocolHandler& handler) {
7615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMap::const_iterator p = default_handlers_.find(
7635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      handler.protocol());
7645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // If we're not loading, and we are setting a default for a new protocol,
7655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // register with the OS.
7665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!is_loading_ && p == default_handlers_.end())
7675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      delegate_->RegisterWithOSAsDefaultClient(handler.protocol(), this);
7685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  default_handlers_.erase(handler.protocol());
7695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  default_handlers_.insert(std::make_pair(handler.protocol(), handler));
7705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  PromoteHandler(handler);
7715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BrowserThread::PostTask(
7725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BrowserThread::IO,
7735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
7742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&IOThreadDelegate::SetDefault, io_thread_delegate_, handler));
7755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::InsertHandler(const ProtocolHandler& handler) {
7785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerMultiMap::iterator p =
7805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      protocol_handlers_.find(handler.protocol());
7815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (p != protocol_handlers_.end()) {
7835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    p->second.push_back(handler);
7845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
7855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ProtocolHandlerList new_list;
7885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  new_list.push_back(handler);
7895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  protocol_handlers_[handler.protocol()] = new_list;
7905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7925d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)base::Value* ProtocolHandlerRegistry::EncodeRegisteredHandlers() {
7935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
7945d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* protocol_handlers = new base::ListValue();
79546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  for (ProtocolHandlerMultiMap::iterator i = user_protocol_handlers_.begin();
79646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)       i != user_protocol_handlers_.end();
79746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)       ++i) {
7985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    for (ProtocolHandlerList::iterator j = i->second.begin();
7995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)         j != i->second.end(); ++j) {
8005d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      base::DictionaryValue* encoded = j->Encode();
8015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      if (IsDefault(*j)) {
802116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch        encoded->Set("default", new base::FundamentalValue(true));
8035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      }
8045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      protocol_handlers->Append(encoded);
8055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
8065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return protocol_handlers;
8085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8105d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)base::Value* ProtocolHandlerRegistry::EncodeIgnoredHandlers() {
8115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8125d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* handlers = new base::ListValue();
81346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  for (ProtocolHandlerList::iterator i =
81446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)           user_ignored_protocol_handlers_.begin();
81546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)       i != user_ignored_protocol_handlers_.end();
81646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)       ++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(
83146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    const ProtocolHandler& handler,
83246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    const HandlerSource source) {
8335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(CanSchemeBeOverridden(handler.protocol()));
8355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(!handler.IsEmpty());
83646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  ProtocolHandlerMultiMap& map =
83746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      (source == POLICY) ? policy_protocol_handlers_ : user_protocol_handlers_;
83846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  ProtocolHandlerList& list = map[handler.protocol()];
83946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (!HandlerExists(handler, list))
84046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    list.push_back(handler);
8415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (IsRegistered(handler)) {
8425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
8435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enabled_ && !delegate_->IsExternalHandlerRegistered(handler.protocol()))
8455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    delegate_->RegisterExternalHandler(handler.protocol());
8465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  InsertHandler(handler);
8475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)std::vector<const base::DictionaryValue*>
8505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProtocolHandlerRegistry::GetHandlersFromPref(const char* pref_name) const {
8515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
8525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  std::vector<const base::DictionaryValue*> result;
8535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  PrefService* prefs = profile_->GetPrefs();
8545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!prefs->HasPrefPath(pref_name)) {
8555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return result;
8565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8585d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  const base::ListValue* handlers = prefs->GetList(pref_name);
8595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (handlers) {
8605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    for (size_t i = 0; i < handlers->GetSize(); ++i) {
8615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      const base::DictionaryValue* dict;
8625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      if (!handlers->GetDictionary(i, &dict))
8635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        continue;
8645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      if (ProtocolHandler::IsValidDict(dict)) {
8655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        result.push_back(dict);
8665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      }
8675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
8685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return result;
8705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
87246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)void ProtocolHandlerRegistry::RegisterProtocolHandlersFromPref(
87346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    const char* pref_name,
87446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    const HandlerSource source) {
87546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  std::vector<const base::DictionaryValue*> registered_handlers =
87646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      GetHandlersFromPref(pref_name);
87746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  for (std::vector<const base::DictionaryValue*>::const_iterator p =
87846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)           registered_handlers.begin();
87946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)       p != registered_handlers.end();
88046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)       ++p) {
88146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    ProtocolHandler handler = ProtocolHandler::CreateProtocolHandler(*p);
88246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    RegisterProtocolHandler(handler, source);
88346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    bool is_default = false;
88446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    if ((*p)->GetBoolean("default", &is_default) && is_default) {
88546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      SetDefault(handler);
88646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    }
88746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  }
88846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)}
88946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)
8905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::IgnoreProtocolHandler(
89146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    const ProtocolHandler& handler,
89246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    const HandlerSource source) {
89346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
89446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  ProtocolHandlerList& list = (source == POLICY)
89546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                                  ? policy_ignored_protocol_handlers_
89646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                                  : user_ignored_protocol_handlers_;
89746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (!HandlerExists(handler, list))
89846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    list.push_back(handler);
89946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (HandlerExists(handler, ignored_protocol_handlers_))
90046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    return;
9015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ignored_protocol_handlers_.push_back(handler);
9025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
9035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
90446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)void ProtocolHandlerRegistry::IgnoreProtocolHandlersFromPref(
90546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    const char* pref_name,
90646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    const HandlerSource source) {
90746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  std::vector<const base::DictionaryValue*> ignored_handlers =
90846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)      GetHandlersFromPref(pref_name);
90946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  for (std::vector<const base::DictionaryValue*>::const_iterator p =
91046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)           ignored_handlers.begin();
91146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)       p != ignored_handlers.end();
91246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)       ++p) {
91346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    IgnoreProtocolHandler(ProtocolHandler::CreateProtocolHandler(*p), source);
91446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  }
91546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)}
91646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)
91746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)bool ProtocolHandlerRegistry::HandlerExists(const ProtocolHandler& handler,
91846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                                            ProtocolHandlerMultiMap* map) {
91946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  return HandlerExists(handler, (*map)[handler.protocol()]);
92046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)}
92146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)
92246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)bool ProtocolHandlerRegistry::HandlerExists(const ProtocolHandler& handler,
92346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                                            const ProtocolHandlerList& list) {
92446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  return std::find(list.begin(), list.end(), handler) != list.end();
92546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)}
92646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)
92746d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)void ProtocolHandlerRegistry::EraseHandler(const ProtocolHandler& handler,
92846d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                                           ProtocolHandlerMultiMap* map) {
92946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  EraseHandler(handler, &(*map)[handler.protocol()]);
93046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)}
93146d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)
93246d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)void ProtocolHandlerRegistry::EraseHandler(const ProtocolHandler& handler,
93346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)                                           ProtocolHandlerList* list) {
93446d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  list->erase(std::find(list->begin(), list->end(), handler));
93546d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)}
93646d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)
9375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProtocolHandlerRegistry::AddPredefinedHandler(
9385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const ProtocolHandler& handler) {
9395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(!is_loaded_);  // Must be called prior InitProtocolSettings.
94046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  RegisterProtocolHandler(handler, USER);
9415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  SetDefault(handler);
9425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
9435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
9442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)scoped_ptr<ProtocolHandlerRegistry::JobInterceptorFactory>
9452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)ProtocolHandlerRegistry::CreateJobInterceptorFactory() {
9465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
9475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // this is always created on the UI thread (in profile_io's
9485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // InitializeOnUIThread. Any method calls must be done
9495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // on the IO thread (this is checked).
950868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  return scoped_ptr<JobInterceptorFactory>(
951868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)      new JobInterceptorFactory(io_thread_delegate_.get()));
9525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
953