123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312 |
- // Copyright (c) 2015 GitHub, Inc.
- // Use of this source code is governed by the MIT license that can be
- // found in the LICENSE file.
- #include "atom/browser/api/atom_api_cookies.h"
- #include "atom/browser/atom_browser_context.h"
- #include "atom/common/native_mate_converters/callback.h"
- #include "atom/common/native_mate_converters/gurl_converter.h"
- #include "atom/common/native_mate_converters/value_converter.h"
- #include "base/time/time.h"
- #include "base/values.h"
- #include "content/public/browser/browser_context.h"
- #include "content/public/browser/browser_thread.h"
- #include "native_mate/dictionary.h"
- #include "native_mate/object_template_builder.h"
- #include "net/cookies/cookie_monster.h"
- #include "net/cookies/cookie_store.h"
- #include "net/cookies/cookie_util.h"
- #include "net/url_request/url_request_context.h"
- #include "net/url_request/url_request_context_getter.h"
- using content::BrowserThread;
- namespace mate {
- template <>
- struct Converter<atom::api::Cookies::Error> {
- static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
- atom::api::Cookies::Error val) {
- if (val == atom::api::Cookies::SUCCESS)
- return v8::Null(isolate);
- else
- return v8::Exception::Error(StringToV8(isolate, "Setting cookie failed"));
- }
- };
- template <>
- struct Converter<net::CanonicalCookie> {
- static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
- const net::CanonicalCookie& val) {
- mate::Dictionary dict(isolate, v8::Object::New(isolate));
- dict.Set("name", val.Name());
- dict.Set("value", val.Value());
- dict.Set("domain", val.Domain());
- dict.Set("hostOnly", net::cookie_util::DomainIsHostOnly(val.Domain()));
- dict.Set("path", val.Path());
- dict.Set("secure", val.IsSecure());
- dict.Set("httpOnly", val.IsHttpOnly());
- dict.Set("session", !val.IsPersistent());
- if (val.IsPersistent())
- dict.Set("expirationDate", val.ExpiryDate().ToDoubleT());
- return dict.GetHandle();
- }
- };
- template <>
- struct Converter<net::CookieStore::ChangeCause> {
- static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
- const net::CookieStore::ChangeCause& val) {
- switch (val) {
- case net::CookieStore::ChangeCause::INSERTED:
- case net::CookieStore::ChangeCause::EXPLICIT:
- case net::CookieStore::ChangeCause::EXPLICIT_DELETE_BETWEEN:
- case net::CookieStore::ChangeCause::EXPLICIT_DELETE_PREDICATE:
- case net::CookieStore::ChangeCause::EXPLICIT_DELETE_SINGLE:
- case net::CookieStore::ChangeCause::EXPLICIT_DELETE_CANONICAL:
- return mate::StringToV8(isolate, "explicit");
- case net::CookieStore::ChangeCause::OVERWRITE:
- return mate::StringToV8(isolate, "overwrite");
- case net::CookieStore::ChangeCause::EXPIRED:
- return mate::StringToV8(isolate, "expired");
- case net::CookieStore::ChangeCause::EVICTED:
- return mate::StringToV8(isolate, "evicted");
- case net::CookieStore::ChangeCause::EXPIRED_OVERWRITE:
- return mate::StringToV8(isolate, "expired-overwrite");
- default:
- return mate::StringToV8(isolate, "unknown");
- }
- }
- };
- } // namespace mate
- namespace atom {
- namespace api {
- namespace {
- // Returns whether |domain| matches |filter|.
- bool MatchesDomain(std::string filter, const std::string& domain) {
- // Add a leading '.' character to the filter domain if it doesn't exist.
- if (net::cookie_util::DomainIsHostOnly(filter))
- filter.insert(0, ".");
- std::string sub_domain(domain);
- // Strip any leading '.' character from the input cookie domain.
- if (!net::cookie_util::DomainIsHostOnly(sub_domain))
- sub_domain = sub_domain.substr(1);
- // Now check whether the domain argument is a subdomain of the filter domain.
- for (sub_domain.insert(0, "."); sub_domain.length() >= filter.length();) {
- if (sub_domain == filter)
- return true;
- const size_t next_dot = sub_domain.find('.', 1); // Skip over leading dot.
- sub_domain.erase(0, next_dot);
- }
- return false;
- }
- // Returns whether |cookie| matches |filter|.
- bool MatchesCookie(const base::DictionaryValue* filter,
- const net::CanonicalCookie& cookie) {
- std::string str;
- bool b;
- if (filter->GetString("name", &str) && str != cookie.Name())
- return false;
- if (filter->GetString("path", &str) && str != cookie.Path())
- return false;
- if (filter->GetString("domain", &str) && !MatchesDomain(str, cookie.Domain()))
- return false;
- if (filter->GetBoolean("secure", &b) && b != cookie.IsSecure())
- return false;
- if (filter->GetBoolean("session", &b) && b != !cookie.IsPersistent())
- return false;
- return true;
- }
- // Helper to returns the CookieStore.
- inline net::CookieStore* GetCookieStore(
- scoped_refptr<net::URLRequestContextGetter> getter) {
- return getter->GetURLRequestContext()->cookie_store();
- }
- // Run |callback| on UI thread.
- void RunCallbackInUI(const base::Closure& callback) {
- BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, callback);
- }
- // Remove cookies from |list| not matching |filter|, and pass it to |callback|.
- void FilterCookies(std::unique_ptr<base::DictionaryValue> filter,
- const Cookies::GetCallback& callback,
- const net::CookieList& list) {
- net::CookieList result;
- for (const auto& cookie : list) {
- if (MatchesCookie(filter.get(), cookie))
- result.push_back(cookie);
- }
- RunCallbackInUI(base::Bind(callback, Cookies::SUCCESS, result));
- }
- // Receives cookies matching |filter| in IO thread.
- void GetCookiesOnIO(scoped_refptr<net::URLRequestContextGetter> getter,
- std::unique_ptr<base::DictionaryValue> filter,
- const Cookies::GetCallback& callback) {
- std::string url;
- filter->GetString("url", &url);
- auto filtered_callback =
- base::Bind(FilterCookies, base::Passed(&filter), callback);
- // Empty url will match all url cookies.
- if (url.empty())
- GetCookieStore(getter)->GetAllCookiesAsync(filtered_callback);
- else
- GetCookieStore(getter)->GetAllCookiesForURLAsync(GURL(url),
- filtered_callback);
- }
- // Removes cookie with |url| and |name| in IO thread.
- void RemoveCookieOnIOThread(scoped_refptr<net::URLRequestContextGetter> getter,
- const GURL& url,
- const std::string& name,
- const base::Closure& callback) {
- GetCookieStore(getter)->DeleteCookieAsync(
- url, name, base::BindOnce(RunCallbackInUI, callback));
- }
- // Callback of SetCookie.
- void OnSetCookie(const Cookies::SetCallback& callback, bool success) {
- RunCallbackInUI(
- base::Bind(callback, success ? Cookies::SUCCESS : Cookies::FAILED));
- }
- // Flushes cookie store in IO thread.
- void FlushCookieStoreOnIOThread(
- scoped_refptr<net::URLRequestContextGetter> getter,
- const base::Closure& callback) {
- GetCookieStore(getter)->FlushStore(base::BindOnce(RunCallbackInUI, callback));
- }
- // Sets cookie with |details| in IO thread.
- void SetCookieOnIO(scoped_refptr<net::URLRequestContextGetter> getter,
- std::unique_ptr<base::DictionaryValue> details,
- const Cookies::SetCallback& callback) {
- std::string url, name, value, domain, path;
- bool secure = false;
- bool http_only = false;
- double creation_date;
- double expiration_date;
- double last_access_date;
- details->GetString("url", &url);
- details->GetString("name", &name);
- details->GetString("value", &value);
- details->GetString("domain", &domain);
- details->GetString("path", &path);
- details->GetBoolean("secure", &secure);
- details->GetBoolean("httpOnly", &http_only);
- base::Time creation_time;
- if (details->GetDouble("creationDate", &creation_date)) {
- creation_time = (creation_date == 0)
- ? base::Time::UnixEpoch()
- : base::Time::FromDoubleT(creation_date);
- }
- base::Time expiration_time;
- if (details->GetDouble("expirationDate", &expiration_date)) {
- expiration_time = (expiration_date == 0)
- ? base::Time::UnixEpoch()
- : base::Time::FromDoubleT(expiration_date);
- }
- base::Time last_access_time;
- if (details->GetDouble("lastAccessDate", &last_access_date)) {
- last_access_time = (last_access_date == 0)
- ? base::Time::UnixEpoch()
- : base::Time::FromDoubleT(last_access_date);
- }
- GetCookieStore(getter)->SetCookieWithDetailsAsync(
- GURL(url), name, value, domain, path, creation_time, expiration_time,
- last_access_time, secure, http_only, net::CookieSameSite::DEFAULT_MODE,
- net::COOKIE_PRIORITY_DEFAULT, base::BindOnce(OnSetCookie, callback));
- }
- } // namespace
- Cookies::Cookies(v8::Isolate* isolate, AtomBrowserContext* browser_context)
- : browser_context_(browser_context) {
- Init(isolate);
- auto subscription = browser_context->RegisterCookieChangeCallback(
- base::Bind(&Cookies::OnCookieChanged, base::Unretained(this)));
- browser_context->set_cookie_change_subscription(std::move(subscription));
- }
- Cookies::~Cookies() {}
- void Cookies::Get(const base::DictionaryValue& filter,
- const GetCallback& callback) {
- std::unique_ptr<base::DictionaryValue> copied(filter.CreateDeepCopy());
- auto* getter = browser_context_->GetRequestContext();
- content::BrowserThread::PostTask(
- BrowserThread::IO, FROM_HERE,
- base::BindOnce(GetCookiesOnIO, base::RetainedRef(getter),
- std::move(copied), callback));
- }
- void Cookies::Remove(const GURL& url,
- const std::string& name,
- const base::Closure& callback) {
- auto* getter = browser_context_->GetRequestContext();
- content::BrowserThread::PostTask(
- BrowserThread::IO, FROM_HERE,
- base::BindOnce(RemoveCookieOnIOThread, base::RetainedRef(getter), url,
- name, callback));
- }
- void Cookies::Set(const base::DictionaryValue& details,
- const SetCallback& callback) {
- std::unique_ptr<base::DictionaryValue> copied(details.CreateDeepCopy());
- auto* getter = browser_context_->GetRequestContext();
- content::BrowserThread::PostTask(
- BrowserThread::IO, FROM_HERE,
- base::BindOnce(SetCookieOnIO, base::RetainedRef(getter),
- std::move(copied), callback));
- }
- void Cookies::FlushStore(const base::Closure& callback) {
- auto* getter = browser_context_->GetRequestContext();
- content::BrowserThread::PostTask(
- BrowserThread::IO, FROM_HERE,
- base::BindOnce(FlushCookieStoreOnIOThread, base::RetainedRef(getter),
- callback));
- }
- void Cookies::OnCookieChanged(const CookieDetails* details) {
- Emit("changed", *(details->cookie), details->cause, details->removed);
- }
- // static
- mate::Handle<Cookies> Cookies::Create(v8::Isolate* isolate,
- AtomBrowserContext* browser_context) {
- return mate::CreateHandle(isolate, new Cookies(isolate, browser_context));
- }
- // static
- void Cookies::BuildPrototype(v8::Isolate* isolate,
- v8::Local<v8::FunctionTemplate> prototype) {
- prototype->SetClassName(mate::StringToV8(isolate, "Cookies"));
- mate::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
- .SetMethod("get", &Cookies::Get)
- .SetMethod("remove", &Cookies::Remove)
- .SetMethod("set", &Cookies::Set)
- .SetMethod("flushStore", &Cookies::FlushStore);
- }
- } // namespace api
- } // namespace atom
|