blob: 63772175f1f91be9aec99d1ad872301c8ea0a24c [file] [log] [blame]
// Copyright 2017 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 SERVICES_NETWORK_RESTRICTED_COOKIE_MANAGER_H_
#define SERVICES_NETWORK_RESTRICTED_COOKIE_MANAGER_H_
#include <string>
#include "base/component_export.h"
#include "base/containers/linked_list.h"
#include "base/macros.h"
#include "base/memory/weak_ptr.h"
#include "base/sequence_checker.h"
#include "base/threading/sequenced_task_runner_handle.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_store.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"
namespace net {
class CookieStore;
} // namespace net
namespace network {
class CookieSettings;
// RestrictedCookieManager implementation.
//
// Instances of this class must be created and used on the sequence that hosts
// the CookieStore passed to the constructor.
class COMPONENT_EXPORT(NETWORK_SERVICE) RestrictedCookieManager
: public mojom::RestrictedCookieManager {
public:
// All the pointers passed to the constructor are expected to point to
// objects that will outlive `this`.
//
// `origin` represents the domain for which the RestrictedCookieManager can
// access cookies. It could either be a frame origin when `role` is
// RestrictedCookieManagerRole::SCRIPT (a script scoped to a particular
// document's frame)), or a request origin when `role` is
// RestrictedCookieManagerRole::NETWORK (a network request).
//
// `isolation_info` must be fully populated, its `frame_origin` field should
// not be used for cookie access decisions, but should be the same as `origin`
// if the `role` is mojom::RestrictedCookieManagerRole::SCRIPT.
RestrictedCookieManager(
mojom::RestrictedCookieManagerRole role,
net::CookieStore* cookie_store,
const CookieSettings* cookie_settings,
const url::Origin& origin,
const net::IsolationInfo& isolation_info,
mojo::PendingRemote<mojom::CookieAccessObserver> cookie_observer);
~RestrictedCookieManager() override;
void OverrideSiteForCookiesForTesting(
const net::SiteForCookies& new_site_for_cookies) {
site_for_cookies_ = new_site_for_cookies;
}
void OverrideOriginForTesting(const url::Origin& new_origin) {
origin_ = new_origin;
}
void OverrideTopFrameOriginForTesting(
const url::Origin& new_top_frame_origin) {
top_frame_origin_ = new_top_frame_origin;
}
void OverrideIsolationInfoForTesting(
const net::IsolationInfo& new_isolation_info) {
site_for_cookies_ = new_isolation_info.site_for_cookies();
top_frame_origin_ = new_isolation_info.top_frame_origin().value();
isolation_info_ = 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,
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,
SetCanonicalCookieCallback callback) override;
void AddChangeListener(
const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
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,
const std::string& cookie,
SetCookieFromStringCallback callback) override;
void GetCookiesString(const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
GetCookiesStringCallback callback) override;
void CookiesEnabledFor(const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
CookiesEnabledForCallback callback) override;
private:
// The state associated with a CookieChangeListener.
class Listener;
// Feeds a net::CookieList to a GetAllForUrl() callback.
void CookieListToGetAllForUrlCallback(
const GURL& url,
const net::SiteForCookies& site_for_cookies,
const url::Origin& top_frame_origin,
const net::CookieOptions& net_options,
mojom::CookieManagerGetOptionsPtr options,
GetAllForUrlCallback callback,
const net::CookieAccessResultList& cookie_list,
const net::CookieAccessResultList& excluded_cookies);
// Reports the result of setting the cookie to |network_context_client_|, and
// invokes the user callback.
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);
// Called when the Mojo pipe associated with a listener is closed.
void RemoveChangeListener(Listener* listener);
// Ensures that this instance may access the cookies for a given URL.
//
// Returns true if the access should be allowed, or false if it should be
// blocked.
//
// |cookie_being_set| should be non-nullptr if setting a cookie, and should be
// nullptr otherwise (getting cookies, subscribing to cookie changes).
//
// If the access would not be allowed, this helper calls
// mojo::ReportBadMessage(), which closes the pipe.
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 mojom::RestrictedCookieManagerRole role_;
net::CookieStore* const cookie_store_;
const CookieSettings* const cookie_settings_;
// TODO(https://crbug/1166215): Consolidate these three fields since
// `isolation_info_` holds copy of those values.
url::Origin origin_;
net::SiteForCookies site_for_cookies_;
url::Origin top_frame_origin_;
net::IsolationInfo isolation_info_;
mojo::Remote<mojom::CookieAccessObserver> cookie_observer_;
base::LinkedList<Listener> listeners_;
SEQUENCE_CHECKER(sequence_checker_);
base::WeakPtrFactory<RestrictedCookieManager> weak_ptr_factory_{this};
DISALLOW_COPY_AND_ASSIGN(RestrictedCookieManager);
};
} // namespace network
#endif // SERVICES_NETWORK_RESTRICTED_COOKIE_MANAGER_H_