// Copyright (c) 2019 The Chromium Embedded Framework Authors. All rights // reserved. Use of this source code is governed by a BSD-style license that can // be found in the LICENSE file. #include "libcef/browser/net_service/cookie_manager_impl.h" #include "libcef/common/net_service/net_service_util.h" #include "libcef/common/time_util.h" #include "base/bind.h" #include "base/logging.h" #include "content/public/browser/browser_context.h" #include "content/public/browser/storage_partition.h" #include "services/network/public/mojom/cookie_manager.mojom.h" #include "url/gurl.h" using network::mojom::CookieManager; namespace { // Do not keep a reference to the object returned by this method. CefBrowserContext* GetBrowserContext(const CefBrowserContext::Getter& getter) { CEF_REQUIRE_UIT(); DCHECK(!getter.is_null()); // Will return nullptr if the BrowserContext has been destroyed. return getter.Run(); } // Do not keep a reference to the object returned by this method. CookieManager* GetCookieManager(CefBrowserContext* browser_context) { CEF_REQUIRE_UIT(); return content::BrowserContext::GetDefaultStoragePartition( browser_context->AsBrowserContext()) ->GetCookieManagerForBrowserProcess(); } // Always execute the callback asynchronously. void RunAsyncCompletionOnUIThread(CefRefPtr callback) { if (!callback.get()) return; CEF_POST_TASK(CEF_UIT, base::Bind(&CefCompletionCallback::OnComplete, callback.get())); } // Always execute the callback asynchronously. void SetCookieCallbackImpl(CefRefPtr callback, net::CookieInclusionStatus status) { if (!callback.get()) return; if (!status.IsInclude()) { LOG(WARNING) << "SetCookie failed with reason: " << status.GetDebugString(); } CEF_POST_TASK(CEF_UIT, base::Bind(&CefSetCookieCallback::OnComplete, callback.get(), status.IsInclude())); } // Always execute the callback asynchronously. void DeleteCookiesCallbackImpl(CefRefPtr callback, uint32_t num_deleted) { if (!callback.get()) return; CEF_POST_TASK(CEF_UIT, base::Bind(&CefDeleteCookiesCallback::OnComplete, callback.get(), num_deleted)); } void ExecuteVisitor(CefRefPtr visitor, const CefBrowserContext::Getter& browser_context_getter, const std::vector& cookies) { CEF_REQUIRE_UIT(); auto browser_context = GetBrowserContext(browser_context_getter); if (!browser_context) return; auto cookie_manager = GetCookieManager(browser_context); int total = cookies.size(), count = 0; for (const auto& cc : cookies) { CefCookie cookie; net_service::MakeCefCookie(cc, cookie); bool deleteCookie = false; bool keepLooping = visitor->Visit(cookie, count, total, deleteCookie); if (deleteCookie) { cookie_manager->DeleteCanonicalCookie( cc, CookieManager::DeleteCanonicalCookieCallback()); } if (!keepLooping) break; count++; } } // Always execute the callback asynchronously. void GetAllCookiesCallbackImpl( CefRefPtr visitor, const CefBrowserContext::Getter& browser_context_getter, const net::CookieList& cookies) { CEF_POST_TASK(CEF_UIT, base::Bind(&ExecuteVisitor, visitor, browser_context_getter, cookies)); } void GetCookiesCallbackImpl( CefRefPtr visitor, const CefBrowserContext::Getter& browser_context_getter, const net::CookieAccessResultList& include_cookies, const net::CookieAccessResultList&) { net::CookieList cookies; for (const auto& status : include_cookies) { cookies.push_back(status.cookie); } GetAllCookiesCallbackImpl(visitor, browser_context_getter, cookies); } } // namespace CefCookieManagerImpl::CefCookieManagerImpl() {} void CefCookieManagerImpl::Initialize( CefBrowserContext::Getter browser_context_getter, CefRefPtr callback) { CEF_REQUIRE_UIT(); DCHECK(!browser_context_getter.is_null()); DCHECK(browser_context_getter_.is_null()); browser_context_getter_ = browser_context_getter; RunAsyncCompletionOnUIThread(callback); } void CefCookieManagerImpl::SetSupportedSchemes( const std::vector& schemes, bool include_defaults, CefRefPtr callback) { if (!CEF_CURRENTLY_ON_UIT()) { CEF_POST_TASK(CEF_UIT, base::Bind(&CefCookieManagerImpl::SetSupportedSchemes, this, schemes, include_defaults, callback)); return; } std::vector all_schemes; for (const auto& scheme : schemes) all_schemes.push_back(scheme); if (include_defaults) { // Add default schemes that should always support cookies. // This list should match CookieMonster::kDefaultCookieableSchemes. all_schemes.push_back("http"); all_schemes.push_back("https"); all_schemes.push_back("ws"); all_schemes.push_back("wss"); } auto browser_context = GetBrowserContext(browser_context_getter_); if (!browser_context) return; // This will be forwarded to the CookieMonster that lives in the // NetworkService process when the NetworkContext is created via // AlloyContentBrowserClient::CreateNetworkContext. browser_context->set_cookieable_schemes(base::make_optional(all_schemes)); RunAsyncCompletionOnUIThread(callback); } bool CefCookieManagerImpl::VisitAllCookies( CefRefPtr visitor) { if (!visitor.get()) return false; if (!CEF_CURRENTLY_ON_UIT()) { CEF_POST_TASK( CEF_UIT, base::Bind(base::IgnoreResult(&CefCookieManagerImpl::VisitAllCookies), this, visitor)); return true; } auto browser_context = GetBrowserContext(browser_context_getter_); if (!browser_context) return false; GetCookieManager(browser_context) ->GetAllCookies(base::Bind(&GetAllCookiesCallbackImpl, visitor, browser_context_getter_)); return true; } bool CefCookieManagerImpl::VisitUrlCookies( const CefString& url, bool includeHttpOnly, CefRefPtr visitor) { if (!visitor.get()) return false; GURL gurl = GURL(url.ToString()); if (!gurl.is_valid()) return false; if (!CEF_CURRENTLY_ON_UIT()) { CEF_POST_TASK( CEF_UIT, base::Bind(base::IgnoreResult(&CefCookieManagerImpl::VisitUrlCookies), this, url, includeHttpOnly, visitor)); return true; } net::CookieOptions options; if (includeHttpOnly) options.set_include_httponly(); options.set_same_site_cookie_context( net::CookieOptions::SameSiteCookieContext::MakeInclusive()); auto browser_context = GetBrowserContext(browser_context_getter_); if (!browser_context) return false; GetCookieManager(browser_context) ->GetCookieList(gurl, options, base::Bind(&GetCookiesCallbackImpl, visitor, browser_context_getter_)); return true; } bool CefCookieManagerImpl::SetCookie(const CefString& url, const CefCookie& cookie, CefRefPtr callback) { GURL gurl = GURL(url.ToString()); if (!gurl.is_valid()) return false; if (!CEF_CURRENTLY_ON_UIT()) { CEF_POST_TASK( CEF_UIT, base::Bind(base::IgnoreResult(&CefCookieManagerImpl::SetCookie), this, url, cookie, callback)); return true; } std::string name = CefString(&cookie.name).ToString(); std::string value = CefString(&cookie.value).ToString(); std::string domain = CefString(&cookie.domain).ToString(); std::string path = CefString(&cookie.path).ToString(); base::Time expiration_time; if (cookie.has_expires) cef_time_to_basetime(cookie.expires, expiration_time); net::CookieSameSite same_site = net_service::MakeCookieSameSite(cookie.same_site); net::CookiePriority priority = net_service::MakeCookiePriority(cookie.priority); auto canonical_cookie = net::CanonicalCookie::CreateSanitizedCookie( gurl, name, value, domain, path, base::Time(), // Creation time. expiration_time, base::Time(), // Last access time. cookie.secure ? true : false, cookie.httponly ? true : false, same_site, priority); if (!canonical_cookie) { SetCookieCallbackImpl( callback, net::CookieInclusionStatus( net::CookieInclusionStatus::EXCLUDE_UNKNOWN_ERROR)); return true; } net::CookieOptions options; if (cookie.httponly) options.set_include_httponly(); options.set_same_site_cookie_context( net::CookieOptions::SameSiteCookieContext::MakeInclusive()); auto browser_context = GetBrowserContext(browser_context_getter_); if (!browser_context) return false; GetCookieManager(browser_context) ->SetCanonicalCookie(*canonical_cookie, gurl, options, base::Bind(SetCookieCallbackImpl, callback)); return true; } bool CefCookieManagerImpl::DeleteCookies( const CefString& url, const CefString& cookie_name, CefRefPtr callback) { // Empty URLs are allowed but not invalid URLs. GURL gurl = GURL(url.ToString()); if (!gurl.is_empty() && !gurl.is_valid()) return false; if (!CEF_CURRENTLY_ON_UIT()) { CEF_POST_TASK( CEF_UIT, base::Bind(base::IgnoreResult(&CefCookieManagerImpl::DeleteCookies), this, url, cookie_name, callback)); return true; } network::mojom::CookieDeletionFilterPtr deletion_filter = network::mojom::CookieDeletionFilter::New(); if (gurl.is_empty()) { // Delete all cookies. } else if (cookie_name.empty()) { // Delete all matching host cookies. deletion_filter->host_name = gurl.host(); } else { // Delete all matching host and domain cookies. deletion_filter->url = gurl; deletion_filter->cookie_name = cookie_name; } auto browser_context = GetBrowserContext(browser_context_getter_); if (!browser_context) return false; GetCookieManager(browser_context) ->DeleteCookies(std::move(deletion_filter), base::Bind(DeleteCookiesCallbackImpl, callback)); return true; } bool CefCookieManagerImpl::FlushStore( CefRefPtr callback) { if (!CEF_CURRENTLY_ON_UIT()) { CEF_POST_TASK( CEF_UIT, base::Bind(base::IgnoreResult(&CefCookieManagerImpl::FlushStore), this, callback)); return true; } auto browser_context = GetBrowserContext(browser_context_getter_); if (!browser_context) return false; GetCookieManager(browser_context) ->FlushCookieStore(base::Bind(RunAsyncCompletionOnUIThread, callback)); return true; } // CefCookieManager methods ---------------------------------------------------- // static CefRefPtr CefCookieManager::GetGlobalManager( CefRefPtr callback) { CefRefPtr context = CefRequestContext::GetGlobalContext(); return context ? context->GetCookieManager(callback) : nullptr; }