// Copyright 2018 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef NET_COOKIES_COOKIE_MONSTER_CHANGE_DISPATCHER_H_ #define NET_COOKIES_COOKIE_MONSTER_CHANGE_DISPATCHER_H_ #include #include #include #include "base/callback.h" #include "base/callback_list.h" #include "base/compiler_specific.h" #include "base/containers/linked_list.h" #include "base/memory/scoped_refptr.h" #include "base/memory/weak_ptr.h" #include "base/single_thread_task_runner.h" #include "base/threading/thread_checker.h" #include "net/cookies/cookie_change_dispatcher.h" #include "url/gurl.h" namespace net { class CanonicalCookie; // CookieChangeDispatcher implementation used by CookieMonster. class CookieMonsterChangeDispatcher : public CookieChangeDispatcher { public: using CookieChangeCallbackList = base::CallbackList; CookieMonsterChangeDispatcher(); ~CookieMonsterChangeDispatcher() override; // The key in CookieNameMap for a cookie name. static std::string NameKey(std::string name); // The key in CookieDomainName for a cookie domain. static std::string DomainKey(const std::string& domain); // The key in CookieDomainName for a listener URL. static std::string DomainKey(const GURL& url); // net::CookieChangeDispatcher std::unique_ptr AddCallbackForCookie( const GURL& url, const std::string& name, CookieChangeCallback callback) override WARN_UNUSED_RESULT; std::unique_ptr AddCallbackForUrl( const GURL& url, CookieChangeCallback callback) override WARN_UNUSED_RESULT; std::unique_ptr AddCallbackForAllChanges( CookieChangeCallback callback) override WARN_UNUSED_RESULT; // |notify_global_hooks| is true if the function should run the // global hooks in addition to the per-cookie hooks. // // TODO(pwnall): Remove |notify_global_hooks| and fix consumers. void DispatchChange(const CanonicalCookie& cookie, CookieChangeCause cause, bool notify_global_hooks); private: class Subscription : public base::LinkNode, public CookieChangeSubscription { public: Subscription(base::WeakPtr change_dispatcher, std::string domain_key, std::string name_key, GURL url, net::CookieChangeCallback callback); ~Subscription() override; // The lookup key used in the domain subscription map. // // The empty string means no domain filtering. const std::string& domain_key() const { return domain_key_; } // The lookup key used in the name subscription map. // // The empty string means no name filtering. const std::string& name_key() const { return name_key_; } // Dispatches a cookie change notification if the listener is interested. void DispatchChange(const net::CanonicalCookie& cookie, net::CookieChangeCause change_cause); private: base::WeakPtr change_dispatcher_; const std::string domain_key_; // kGlobalDomainKey means no filtering. const std::string name_key_; // kGlobalNameKey means no filtering. const GURL url_; // empty() means no URL-based filtering. net::CookieOptions options_; const net::CookieChangeCallback callback_; void DoDispatchChange(const net::CanonicalCookie& cookie, net::CookieChangeCause change_cause) const; // Used to post DoDispatchChange() calls to this subscription's thread. scoped_refptr task_runner_; THREAD_CHECKER(thread_checker_); // Used to cancel delayed calls to DoDispatchChange() when the subscription // gets destroyed. base::WeakPtrFactory weak_ptr_factory_; DISALLOW_COPY_AND_ASSIGN(Subscription); }; // The last level of the subscription data structures. using SubscriptionList = base::LinkedList; // Buckets subscriptions according to cookie names. // // Map keys are cookie names, as we only support exact name matching. using CookieNameMap = std::map; // Buckets subscriptions according to cookie domains. // // Map keys are the eTLD+1 of cookie domains. Cookies are either host-locked, // or visible to all the subdomain of a given domain. A cookie's scope cannot // exceed eTLD+1, so we stop there. using CookieDomainMap = std::map; void DispatchChangeToDomainKey(const CanonicalCookie& cookie, CookieChangeCause cause, const std::string& domain_key); void DispatchChangeToNameKey(const CanonicalCookie& cookie, CookieChangeCause cause, CookieNameMap& name_map, const std::string& name_key); // Inserts a subscription into the map. // // Called by the AddCallback* methods, after creating the Subscription. void LinkSubscription(Subscription* subscription); // Removes a subscription from the map. // // Called by the Subscription destructor. void UnlinkSubscription(Subscription* subscription); CookieDomainMap cookie_domain_map_; THREAD_CHECKER(thread_checker_); // Vends weak pointers to subscriptions. base::WeakPtrFactory weak_ptr_factory_; DISALLOW_COPY_AND_ASSIGN(CookieMonsterChangeDispatcher); }; } // namespace net #endif // NET_COOKIES_COOKIE_MONSTER_CHANGE_DISPATCHER_H_