#ifndef SERVICES_NETWORK_RESTRICTED_COOKIE_MANAGER_H_
#define SERVICES_NETWORK_RESTRICTED_COOKIE_MANAGER_H_
#include <set>
#include <string>
#include <tuple>
#include "base/component_export.h"
#include "base/containers/linked_list.h"
#include "base/functional/callback_forward.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/raw_ref.h"
#include "base/memory/weak_ptr.h"
#include "base/sequence_checker.h"
#include "base/timer/timer.h"
#include "mojo/public/cpp/bindings/receiver.h"
#include "mojo/public/cpp/bindings/remote.h"
#include "net/base/isolation_info.h"
#include "net/cookies/canonical_cookie.h"
#include "net/cookies/cookie_change_dispatcher.h"
#include "net/cookies/cookie_inclusion_status.h"
#include "net/cookies/cookie_partition_key_collection.h"
#include "net/cookies/cookie_setting_override.h"
#include "net/cookies/cookie_store.h"
#include "net/first_party_sets/first_party_set_metadata.h"
#include "services/network/public/mojom/cookie_access_observer.mojom.h"
#include "services/network/public/mojom/restricted_cookie_manager.mojom.h"
#include "url/gurl.h"
#include "url/origin.h"
#if BUILDFLAG(IS_OHOS)
#include "cookie_shared_memory_holder.h"
#endif
namespace net {
class CookieStore;
class SiteForCookies;
}
namespace network {
struct CookieWithAccessResultComparer {
bool operator()(
const net::CookieWithAccessResult& cookie_with_access_result1,
const net::CookieWithAccessResult& cookie_with_access_result2) const;
};
using CookieAccesses =
std::set<net::CookieWithAccessResult, CookieWithAccessResultComparer>;
using CookieAccessesByURLAndSite =
std::map<std::pair<GURL, net::SiteForCookies>,
std::unique_ptr<CookieAccesses>>;
class CookieSettings;
class COMPONENT_EXPORT(NETWORK_SERVICE) RestrictedCookieManager
: public mojom::RestrictedCookieManager {
public:
class UmaMetricsUpdater {
public:
UmaMetricsUpdater();
virtual ~UmaMetricsUpdater();
virtual void OnGetCookiesString() = 0;
};
RestrictedCookieManager(
mojom::RestrictedCookieManagerRole role,
net::CookieStore* cookie_store,
const CookieSettings& cookie_settings,
const url::Origin& origin,
const net::IsolationInfo& isolation_info,
const net::CookieSettingOverrides& cookie_setting_overrides,
mojo::PendingRemote<mojom::CookieAccessObserver> cookie_observer,
net::FirstPartySetMetadata first_party_set_metadata,
UmaMetricsUpdater* metrics_updater = nullptr);
RestrictedCookieManager(const RestrictedCookieManager&) = delete;
RestrictedCookieManager& operator=(const RestrictedCookieManager&) = delete;
~RestrictedCookieManager() override;
void OverrideOriginForTesting(const url::Origin& new_origin) {
origin_ = new_origin;
}
void OverrideIsolationInfoForTesting(
const net::IsolationInfo& new_isolation_info);
const CookieSettings& cookie_settings() const { return *cookie_settings_; }
void GetAllForUrl(const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
mojom::CookieManagerGetOptionsPtr options,
GetAllForUrlCallback callback) override;
void SetCanonicalCookie(const net::CanonicalCookie& cookie,
const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
net::CookieInclusionStatus status,
SetCanonicalCookieCallback callback) override;
#if BUILDFLAG(IS_OHOS)
void RegisterCookieChangeObserver(
const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
mojo::ScopedSharedBufferHandle buffer,
RegisterCookieChangeObserverCallback callback) override;
#endif
void AddChangeListener(
const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
mojo::PendingRemote<mojom::CookieChangeListener> listener,
AddChangeListenerCallback callback) override;
void SetCookieFromString(const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
const std::string& cookie,
SetCookieFromStringCallback callback) override;
void GetCookiesString(const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
GetCookiesStringCallback callback) override;
void GetCookiesStringAndExpiryDate(const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
GetCookiesStringAndExpiryDateCallback callback) override;
void CookiesEnabledFor(const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
CookiesEnabledForCallback callback) override;
void InstallReceiver(
mojo::PendingReceiver<mojom::RestrictedCookieManager> pending_receiver,
base::OnceClosure on_disconnect_callback);
static void ComputeFirstPartySetMetadata(
const url::Origin& origin,
const net::CookieStore* cookie_store,
const net::IsolationInfo& isolation_info,
base::OnceCallback<void(net::FirstPartySetMetadata)> callback);
net::CookieSettingOverrides GetCookieSettingOverrides(
bool has_storage_access) const;
private:
class Listener;
bool IsPartitionedCookiesEnabled() const;
void CookieListToGetAllForUrlCallback(
const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
bool has_storage_access,
const net::CookieOptions& net_options,
mojom::CookieManagerGetOptionsPtr options,
GetAllForUrlCallback callback,
const net::CookieAccessResultList& cookie_list,
const net::CookieAccessResultList& excluded_cookies);
void SetCanonicalCookieResult(const GURL& url,
const net::SiteForCookies& site_for_cookies,
const net::CanonicalCookie& cookie,
const net::CookieOptions& net_options,
SetCanonicalCookieCallback user_callback,
net::CookieAccessResult access_result);
void RemoveChangeListener(Listener* listener);
bool ValidateAccessToCookiesAt(
const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
const net::CanonicalCookie* cookie_being_set = nullptr);
const net::SiteForCookies& BoundSiteForCookies() const {
return isolation_info_.site_for_cookies();
}
const url::Origin& BoundTopFrameOrigin() const {
return isolation_info_.top_frame_origin().value();
}
CookieAccesses* GetCookieAccessesForURLAndSite(
const GURL& url,
const net::SiteForCookies& site_for_cookies);
bool SkipAccessNotificationForCookieItem(
CookieAccesses* cookie_accesses,
const net::CookieWithAccessResult& cookie_item);
void OnGotFirstPartySetMetadataForTesting(
base::OnceClosure done_closure,
net::FirstPartySetMetadata first_party_set_metadata);
void OnCookiesAccessed(network::mojom::CookieAccessDetailsPtr details);
void CallCookiesAccessed();
const mojom::RestrictedCookieManagerRole role_;
const raw_ptr<net::CookieStore> cookie_store_;
const raw_ref<const CookieSettings> cookie_settings_;
const net::CookieSettingOverrides cookie_setting_overrides_;
url::Origin origin_;
net::IsolationInfo isolation_info_;
mojo::Remote<mojom::CookieAccessObserver> cookie_observer_;
base::LinkedList<Listener> listeners_;
#if BUILDFLAG(IS_OHOS)
base::LinkedList<CookieSharedMemoryHolder> shm_holders_;
#endif
SEQUENCE_CHECKER(sequence_checker_);
net::FirstPartySetMetadata first_party_set_metadata_;
absl::optional<net::CookiePartitionKey> cookie_partition_key_;
net::CookiePartitionKeyCollection cookie_partition_key_collection_;
CookieAccessesByURLAndSite recent_cookie_accesses_;
bool same_party_attribute_enabled_;
mojo::Receiver<mojom::RestrictedCookieManager> receiver_;
const raw_ptr<UmaMetricsUpdater> metrics_updater_;
std::vector<network::mojom::CookieAccessDetailsPtr> cookie_access_details_;
base::RetainingOneShotTimer cookies_access_timer_;
base::WeakPtrFactory<RestrictedCookieManager> weak_ptr_factory_{this};
};
}
#endif