blob: fe1543aa385c24781e0008fccaa1a044251636c7 [file] [log] [blame]
// Copyright 2015 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.
#include "base/android/jni_android.h"
#include "base/android/jni_string.h"
#include "base/bind.h"
#include "base/command_line.h"
#include "base/time/time.h"
#include "chrome/android/public/profiles/jni_headers/CookiesFetcher_jni.h"
#include "chrome/browser/profiles/profile_manager.h"
#include "content/public/browser/browser_context.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/storage_partition.h"
#include "content/public/common/content_switches.h"
#include "net/url_request/url_request_context.h"
#include "services/network/public/mojom/cookie_manager.mojom.h"
using base::android::JavaParamRef;
using base::android::ScopedJavaLocalRef;
namespace {
// Returns the cookie service at the client end of the mojo pipe.
network::mojom::CookieManager* GetCookieServiceClient() {
return content::BrowserContext::GetDefaultStoragePartition(
ProfileManager::GetPrimaryUserProfile()->GetOffTheRecordProfile())
->GetCookieManagerForBrowserProcess();
}
// Passes the fetched |cookies| to the application so that can be saved in a
// file.
void OnCookiesFetchFinished(const net::CookieList& cookies) {
JNIEnv* env = base::android::AttachCurrentThread();
ScopedJavaLocalRef<jobjectArray> joa =
Java_CookiesFetcher_createCookiesArray(env, cookies.size());
int index = 0;
for (auto i = cookies.cbegin(); i != cookies.cend(); ++i) {
std::string domain = i->Domain();
if (domain.length() > 1 && domain[0] == '.')
domain = domain.substr(1);
ScopedJavaLocalRef<jobject> java_cookie = Java_CookiesFetcher_createCookie(
env, base::android::ConvertUTF8ToJavaString(env, i->Name()),
base::android::ConvertUTF8ToJavaString(env, i->Value()),
base::android::ConvertUTF8ToJavaString(env, i->Domain()),
base::android::ConvertUTF8ToJavaString(env, i->Path()),
i->CreationDate().ToDeltaSinceWindowsEpoch().InMicroseconds(),
i->ExpiryDate().ToDeltaSinceWindowsEpoch().InMicroseconds(),
i->LastAccessDate().ToDeltaSinceWindowsEpoch().InMicroseconds(),
i->IsSecure(), i->IsHttpOnly(), static_cast<int>(i->SameSite()),
i->Priority());
env->SetObjectArrayElement(joa.obj(), index++, java_cookie.obj());
}
Java_CookiesFetcher_onCookieFetchFinished(env, joa);
}
} // namespace
// Fetches cookies for the off-the-record session (i.e. incognito mode). It is a
// no-op for the standard session. Typically associated with the #onPause of
// Android's activty lifecycle.
void JNI_CookiesFetcher_PersistCookies(JNIEnv* env) {
if (!ProfileManager::GetPrimaryUserProfile()->HasOffTheRecordProfile()) {
// There is no work to be done. We might consider calling
// the Java callback if needed.
return;
}
GetCookieServiceClient()->GetAllCookies(
base::BindOnce(&OnCookiesFetchFinished));
}
// Creates and sets a canonical cookie for the off-the-record session (i.e.
// incognito mode). It is a no-op for the standard session. Typically associated
// with the #onResume of Android's activty lifecycle.
static void JNI_CookiesFetcher_RestoreCookies(
JNIEnv* env,
const JavaParamRef<jstring>& name,
const JavaParamRef<jstring>& value,
const JavaParamRef<jstring>& domain,
const JavaParamRef<jstring>& path,
jlong creation,
jlong expiration,
jlong last_access,
jboolean secure,
jboolean httponly,
jint same_site,
jint priority) {
if (!ProfileManager::GetPrimaryUserProfile()->HasOffTheRecordProfile()) {
return; // Don't create it. There is nothing to do.
}
std::unique_ptr<net::CanonicalCookie> cookie(
std::make_unique<net::CanonicalCookie>(
base::android::ConvertJavaStringToUTF8(env, name),
base::android::ConvertJavaStringToUTF8(env, value),
base::android::ConvertJavaStringToUTF8(env, domain),
base::android::ConvertJavaStringToUTF8(env, path),
base::Time::FromDeltaSinceWindowsEpoch(
base::TimeDelta::FromMicroseconds(creation)),
base::Time::FromDeltaSinceWindowsEpoch(
base::TimeDelta::FromMicroseconds(expiration)),
base::Time::FromDeltaSinceWindowsEpoch(
base::TimeDelta::FromMicroseconds(last_access)),
secure, httponly, static_cast<net::CookieSameSite>(same_site),
static_cast<net::CookiePriority>(priority)));
// Assume HTTPS - since the cookies are being restored from another store,
// they have already gone through the strict secure check.
//
// Similarly, permit samesite cookies to be imported.
net::CookieOptions options;
options.set_include_httponly();
options.set_same_site_cookie_context(
net::CookieOptions::SameSiteCookieContext::SAME_SITE_STRICT);
GetCookieServiceClient()->SetCanonicalCookie(
*cookie, "https", options,
network::mojom::CookieManager::SetCanonicalCookieCallback());
}