electron/atom/browser/api/atom_api_cookies.cc

243 lines
8.3 KiB
C++
Raw Normal View History

2015-06-14 08:19:53 +00:00
// 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/common/native_mate_converters/callback.h"
#include "atom/common/native_mate_converters/gurl_converter.h"
2015-06-14 08:19:53 +00:00
#include "atom/common/native_mate_converters/value_converter.h"
2015-06-15 13:18:40 +00:00
#include "base/time/time.h"
#include "base/values.h"
#include "content/public/browser/browser_context.h"
2015-06-14 08:19:53 +00:00
#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"
2015-06-14 08:19:53 +00:00
using content::BrowserThread;
2015-12-12 07:33:51 +00:00
namespace mate {
2015-06-14 08:19:53 +00:00
2015-12-12 07:33:51 +00:00
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, "failed"));
2015-06-14 08:19:53 +00:00
}
2015-12-12 07:33:51 +00:00
};
2015-06-15 07:33:09 +00:00
2015-12-12 07:33:51 +00:00
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())
2015-12-12 07:33:51 +00:00
dict.Set("expirationDate", val.ExpiryDate().ToDoubleT());
return dict.GetHandle();
2015-06-15 07:33:09 +00:00
}
2015-12-12 07:33:51 +00:00
};
2015-06-15 07:33:09 +00:00
2015-12-12 07:33:51 +00:00
} // namespace mate
2015-06-15 13:18:40 +00:00
2015-12-12 07:33:51 +00:00
namespace atom {
2015-06-15 13:18:40 +00:00
2015-12-12 07:33:51 +00:00
namespace api {
2015-06-15 13:18:40 +00:00
2015-12-12 07:33:51 +00:00
namespace {
2015-06-14 08:19:53 +00:00
2015-12-12 07:33:51 +00:00
// Returns whether |domain| matches |filter|.
bool MatchesDomain(std::string filter, const std::string& domain) {
2015-06-14 08:19:53 +00:00
// Add a leading '.' character to the filter domain if it doesn't exist.
2015-12-12 07:33:51 +00:00
if (net::cookie_util::DomainIsHostOnly(filter))
filter.insert(0, ".");
2015-06-14 08:19:53 +00:00
2015-12-12 07:33:51 +00:00
std::string sub_domain(domain);
2015-06-14 08:19:53 +00:00
// 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.
2015-12-12 07:33:51 +00:00
for (sub_domain.insert(0, "."); sub_domain.length() >= filter.length();) {
if (sub_domain == filter)
2015-06-14 08:19:53 +00:00
return true;
const size_t next_dot = sub_domain.find('.', 1); // Skip over leading dot.
sub_domain.erase(0, next_dot);
}
return false;
}
2015-12-12 07:33:51 +00:00
// Returns whether |cookie| matches |filter|.
2015-06-14 08:19:53 +00:00
bool MatchesCookie(const base::DictionaryValue* filter,
2015-06-20 02:41:40 +00:00
const net::CanonicalCookie& cookie) {
2015-12-12 07:33:51 +00:00
std::string str;
bool b;
if (filter->GetString("name", &str) && str != cookie.Name())
2015-06-14 08:19:53 +00:00
return false;
2015-12-12 07:33:51 +00:00
if (filter->GetString("path", &str) && str != cookie.Path())
2015-06-14 08:19:53 +00:00
return false;
2015-12-12 07:33:51 +00:00
if (filter->GetString("domain", &str) && !MatchesDomain(str, cookie.Domain()))
2015-06-14 08:19:53 +00:00
return false;
2015-12-12 07:33:51 +00:00
if (filter->GetBoolean("secure", &b) && b != cookie.IsSecure())
2015-06-14 08:19:53 +00:00
return false;
2015-12-12 07:33:51 +00:00
if (filter->GetBoolean("session", &b) && b != !cookie.IsPersistent())
2015-06-14 08:19:53 +00:00
return false;
return true;
}
2015-12-12 07:33:51 +00:00
// Helper to returns the CookieStore.
inline net::CookieStore* GetCookieStore(
scoped_refptr<net::URLRequestContextGetter> getter) {
return getter->GetURLRequestContext()->cookie_store();
2015-06-14 08:19:53 +00:00
}
2015-12-12 07:33:51 +00:00
// Run |callback| on UI thread.
void RunCallbackInUI(const base::Closure& callback) {
BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, callback);
2015-06-14 08:19:53 +00:00
}
2015-12-12 07:33:51 +00:00
// Remove cookies from |list| not matching |filter|, and pass it to |callback|.
void FilterCookies(scoped_ptr<base::DictionaryValue> filter,
const Cookies::GetCallback& callback,
const net::CookieList& list) {
2015-06-14 08:19:53 +00:00
net::CookieList result;
2015-12-12 07:33:51 +00:00
for (const auto& cookie : list) {
if (MatchesCookie(filter.get(), cookie))
2015-06-14 08:19:53 +00:00
result.push_back(cookie);
}
2015-12-12 07:33:51 +00:00
RunCallbackInUI(base::Bind(callback, Cookies::SUCCESS, result));
2015-06-15 07:33:09 +00:00
}
2015-12-12 07:33:51 +00:00
// Receives cookies matching |filter| in IO thread.
void GetCookiesOnIO(scoped_refptr<net::URLRequestContextGetter> getter,
scoped_ptr<base::DictionaryValue> filter,
const Cookies::GetCallback& callback) {
2015-06-15 13:18:40 +00:00
std::string url;
2015-12-12 07:33:51 +00:00
filter->GetString("url", &url);
2015-06-15 13:18:40 +00:00
2015-12-12 07:33:51 +00:00
auto filtered_callback =
base::Bind(FilterCookies, base::Passed(&filter), callback);
2015-06-15 13:18:40 +00:00
2015-12-12 07:33:51 +00:00
net::CookieMonster* monster = GetCookieStore(getter)->GetCookieMonster();
// Empty url will match all url cookies.
if (url.empty())
monster->GetAllCookiesAsync(filtered_callback);
else
monster->GetAllCookiesForURLAsync(GURL(url), filtered_callback);
}
2015-06-15 13:18:40 +00:00
2015-12-12 07:33:51 +00:00
// 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::Bind(RunCallbackInUI, callback));
2015-06-15 13:18:40 +00:00
}
2015-12-12 07:33:51 +00:00
// Callback of SetCookie.
void OnSetCookie(const Cookies::SetCallback& callback, bool success) {
RunCallbackInUI(
base::Bind(callback, success ? Cookies::SUCCESS : Cookies::FAILED));
}
2015-06-15 13:18:40 +00:00
2015-12-12 07:33:51 +00:00
// Sets cookie with |details| in IO thread.
void SetCookieOnIO(scoped_refptr<net::URLRequestContextGetter> getter,
scoped_ptr<base::DictionaryValue> details,
const Cookies::SetCallback& callback) {
std::string url, name, value, domain, path;
2015-06-15 13:18:40 +00:00
bool secure = false;
bool http_only = false;
double expiration_date;
2015-12-12 07:33:51 +00:00
details->GetString("url", &url);
2015-06-15 13:18:40 +00:00
details->GetString("name", &name);
details->GetString("value", &value);
details->GetString("domain", &domain);
details->GetString("path", &path);
details->GetBoolean("secure", &secure);
2015-12-12 07:33:51 +00:00
details->GetBoolean("httpOnly", &http_only);
2015-06-15 13:18:40 +00:00
base::Time expiration_time;
if (details->GetDouble("expirationDate", &expiration_date)) {
expiration_time = (expiration_date == 0) ?
base::Time::UnixEpoch() :
base::Time::FromDoubleT(expiration_date);
}
2015-12-12 07:33:51 +00:00
GetCookieStore(getter)->GetCookieMonster()->SetCookieWithDetailsAsync(
GURL(url), name, value, domain, path, expiration_time, secure, http_only,
2016-03-08 14:28:53 +00:00
false, false, false, net::COOKIE_PRIORITY_DEFAULT,
base::Bind(OnSetCookie, callback));
2015-12-12 07:33:51 +00:00
}
} // namespace
2016-04-25 01:17:54 +00:00
Cookies::Cookies(v8::Isolate* isolate,
content::BrowserContext* browser_context)
: request_context_getter_(browser_context->GetRequestContext()) {
Init(isolate);
2015-12-12 07:33:51 +00:00
}
Cookies::~Cookies() {
}
void Cookies::Get(const base::DictionaryValue& filter,
const GetCallback& callback) {
scoped_ptr<base::DictionaryValue> copied(filter.CreateDeepCopy());
auto getter = make_scoped_refptr(request_context_getter_);
content::BrowserThread::PostTask(
BrowserThread::IO, FROM_HERE,
base::Bind(GetCookiesOnIO, getter, Passed(&copied), callback));
2015-06-15 13:18:40 +00:00
}
2015-12-12 07:33:51 +00:00
void Cookies::Remove(const GURL& url, const std::string& name,
const base::Closure& callback) {
auto getter = make_scoped_refptr(request_context_getter_);
content::BrowserThread::PostTask(
BrowserThread::IO, FROM_HERE,
base::Bind(RemoveCookieOnIOThread, getter, url, name, callback));
2015-06-15 13:18:40 +00:00
}
2015-12-12 07:33:51 +00:00
void Cookies::Set(const base::DictionaryValue& details,
const SetCallback& callback) {
scoped_ptr<base::DictionaryValue> copied(details.CreateDeepCopy());
auto getter = make_scoped_refptr(request_context_getter_);
content::BrowserThread::PostTask(
BrowserThread::IO, FROM_HERE,
base::Bind(SetCookieOnIO, getter, Passed(&copied), callback));
}
2015-06-14 08:19:53 +00:00
// static
2015-06-20 02:41:40 +00:00
mate::Handle<Cookies> Cookies::Create(
v8::Isolate* isolate,
content::BrowserContext* browser_context) {
2016-04-25 01:17:54 +00:00
return mate::CreateHandle(isolate, new Cookies(isolate, browser_context));
2015-06-14 08:19:53 +00:00
}
// static
void Cookies::BuildPrototype(v8::Isolate* isolate,
v8::Local<v8::ObjectTemplate> prototype) {
mate::ObjectTemplateBuilder(isolate, prototype)
.SetMethod("get", &Cookies::Get)
.SetMethod("remove", &Cookies::Remove)
.SetMethod("set", &Cookies::Set);
}
2015-06-14 08:19:53 +00:00
} // namespace api
2015-08-10 07:36:47 +00:00
} // namespace atom