|  | // Copyright 2015 The Chromium Authors | 
|  | // 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/browser/profiles/android/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/cookies/cookie_partition_key.h" | 
|  | #include "net/cookies/cookie_util.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() { | 
|  | // Since restoring Incognito CCT session from cookies is not supported, it is | 
|  | // safe to use the primary OTR profile here. | 
|  | return ProfileManager::GetPrimaryUserProfile() | 
|  | ->GetPrimaryOTRProfile(/*create_if_needed=*/true) | 
|  | ->GetDefaultStoragePartition() | 
|  | ->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 pk; | 
|  | if (!net::CookiePartitionKey::Serialize(i->PartitionKey(), pk)) | 
|  | continue; | 
|  | 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->LastUpdateDate().ToDeltaSinceWindowsEpoch().InMicroseconds(), | 
|  | i->IsSecure(), i->IsHttpOnly(), static_cast<int>(i->SameSite()), | 
|  | i->Priority(), i->IsSameParty(), | 
|  | base::android::ConvertUTF8ToJavaString(env, pk), | 
|  | static_cast<int>(i->SourceScheme()), i->SourcePort()); | 
|  | 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()->HasPrimaryOTRProfile()) { | 
|  | // 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, | 
|  | jlong last_update, | 
|  | jboolean secure, | 
|  | jboolean httponly, | 
|  | jint same_site, | 
|  | jint priority, | 
|  | jboolean same_party, | 
|  | const JavaParamRef<jstring>& partition_key, | 
|  | jint source_scheme, | 
|  | jint source_port) { | 
|  | if (!ProfileManager::GetPrimaryUserProfile()->HasPrimaryOTRProfile()) | 
|  | return;  // Don't create it. There is nothing to do. | 
|  |  | 
|  | std::string domain_str(base::android::ConvertJavaStringToUTF8(env, domain)); | 
|  | std::string path_str(base::android::ConvertJavaStringToUTF8(env, path)); | 
|  |  | 
|  | absl::optional<net::CookiePartitionKey> pk; | 
|  | if (!net::CookiePartitionKey::Deserialize( | 
|  | base::android::ConvertJavaStringToUTF8(env, partition_key), pk)) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | std::unique_ptr<net::CanonicalCookie> cookie = | 
|  | net::CanonicalCookie::FromStorage( | 
|  | base::android::ConvertJavaStringToUTF8(env, name), | 
|  | base::android::ConvertJavaStringToUTF8(env, value), domain_str, | 
|  | path_str, | 
|  | base::Time::FromDeltaSinceWindowsEpoch(base::Microseconds(creation)), | 
|  | base::Time::FromDeltaSinceWindowsEpoch( | 
|  | base::Microseconds(expiration)), | 
|  | base::Time::FromDeltaSinceWindowsEpoch( | 
|  | base::Microseconds(last_access)), | 
|  | base::Time::FromDeltaSinceWindowsEpoch( | 
|  | base::Microseconds(last_update)), | 
|  | secure, httponly, static_cast<net::CookieSameSite>(same_site), | 
|  | static_cast<net::CookiePriority>(priority), same_party, pk, | 
|  | static_cast<net::CookieSourceScheme>(source_scheme), source_port); | 
|  | // FromStorage() uses a less strict version of IsCanonical(), we need to check | 
|  | // the stricter version as well here. This is safe because this function is | 
|  | // only used for incognito cookies which don't survive Chrome updates and | 
|  | // therefore should never be the "older" less strict variety. | 
|  | if (!cookie || !cookie->IsCanonical()) | 
|  | return; | 
|  |  | 
|  | // 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::MakeInclusive()); | 
|  | options.set_do_not_update_access_time(); | 
|  | GetCookieServiceClient()->SetCanonicalCookie( | 
|  | *cookie, | 
|  | net::cookie_util::CookieDomainAndPathToURL( | 
|  | domain_str, path_str, | 
|  | static_cast<net::CookieSourceScheme>(source_scheme)), | 
|  | options, network::mojom::CookieManager::SetCanonicalCookieCallback()); | 
|  | } |