| // 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. |
| |
| #import "ios/web/navigation/session_storage_builder.h" |
| |
| #include <memory> |
| |
| #include "base/check_op.h" |
| #include "base/mac/foundation_util.h" |
| #include "ios/web/common/features.h" |
| #import "ios/web/navigation/navigation_item_impl.h" |
| #import "ios/web/navigation/navigation_item_storage_builder.h" |
| #include "ios/web/navigation/navigation_manager_impl.h" |
| #import "ios/web/navigation/wk_navigation_util.h" |
| #import "ios/web/public/session/crw_session_storage.h" |
| #import "ios/web/public/session/serializable_user_data_manager.h" |
| #import "ios/web/public/web_client.h" |
| #import "ios/web/session/session_certificate_policy_cache_impl.h" |
| #include "ios/web/session/session_certificate_policy_cache_storage_builder.h" |
| #import "ios/web/web_state/web_state_impl.h" |
| |
| #if !defined(__has_feature) || !__has_feature(objc_arc) |
| #error "This file requires ARC support." |
| #endif |
| |
| namespace web { |
| |
| CRWSessionStorage* SessionStorageBuilder::BuildStorage( |
| WebStateImpl* web_state) const { |
| DCHECK(web_state); |
| web::NavigationManagerImpl* navigation_manager = |
| web_state->navigation_manager_.get(); |
| DCHECK(navigation_manager); |
| CRWSessionStorage* session_storage = [[CRWSessionStorage alloc] init]; |
| session_storage.hasOpener = web_state->HasOpener(); |
| session_storage.lastCommittedItemIndex = |
| navigation_manager->GetLastCommittedItemIndex(); |
| if (session_storage.lastCommittedItemIndex == -1) { |
| // This can happen when a session is saved during restoration. Instead, |
| // default to GetItemCount() - 1. |
| session_storage.lastCommittedItemIndex = |
| navigation_manager->GetItemCount() - 1; |
| } |
| NSMutableArray* item_storages = [[NSMutableArray alloc] init]; |
| NavigationItemStorageBuilder item_storage_builder; |
| size_t originalIndex = session_storage.lastCommittedItemIndex; |
| // Drop URLs larger than a certain threshold. |
| for (size_t index = 0; |
| index < static_cast<size_t>(navigation_manager->GetItemCount()); |
| ++index) { |
| web::NavigationItemImpl* item = |
| navigation_manager->GetNavigationItemImplAtIndex(index); |
| if (item->ShouldSkipSerialization() || |
| item->GetURL().spec().size() > url::kMaxURLChars) { |
| if (index <= originalIndex) { |
| session_storage.lastCommittedItemIndex--; |
| } |
| continue; |
| } |
| if (base::FeatureList::IsEnabled(features::kReduceSessionSize)) { |
| // Go through the builder who's a friend of web::NavigationItemImpl |
| // and has access to raw fields, so for example url is not |
| // counted twice if virtyual url is empty. |
| if (item_storage_builder.ItemStoredSize(item) > kMaxNavigationItemSize) { |
| if (index <= originalIndex) { |
| session_storage.lastCommittedItemIndex--; |
| } |
| continue; |
| } |
| } |
| |
| [item_storages addObject:item_storage_builder.BuildStorage(item)]; |
| } |
| |
| int loc = 0; |
| int len = 0; |
| session_storage.lastCommittedItemIndex = wk_navigation_util::GetSafeItemRange( |
| session_storage.lastCommittedItemIndex, item_storages.count, &loc, &len); |
| |
| DCHECK_LT(session_storage.lastCommittedItemIndex, |
| static_cast<NSInteger>(len)); |
| session_storage.itemStorages = |
| [item_storages subarrayWithRange:NSMakeRange(loc, len)]; |
| SessionCertificatePolicyCacheStorageBuilder cert_builder; |
| session_storage.certPolicyCacheStorage = cert_builder.BuildStorage( |
| &web_state->GetSessionCertificatePolicyCacheImpl()); |
| web::SerializableUserDataManager* user_data_manager = |
| web::SerializableUserDataManager::FromWebState(web_state); |
| web::GetWebClient()->AddSerializableData(user_data_manager, web_state); |
| [session_storage |
| setSerializableUserData:user_data_manager->CreateSerializableUserData()]; |
| session_storage.userAgentType = |
| web_state->GetUserAgentForSessionRestoration(); |
| |
| return session_storage; |
| } |
| |
| void SessionStorageBuilder::ExtractSessionState( |
| WebStateImpl* web_state, |
| CRWSessionStorage* storage) const { |
| DCHECK(web_state); |
| DCHECK(storage); |
| web_state->created_with_opener_ = storage.hasOpener; |
| NSArray* item_storages = storage.itemStorages; |
| web::ScopedNavigationItemList items(item_storages.count); |
| NavigationItemStorageBuilder item_storage_builder; |
| for (size_t index = 0; index < item_storages.count; ++index) { |
| std::unique_ptr<NavigationItemImpl> item_impl = |
| item_storage_builder.BuildNavigationItemImpl(item_storages[index]); |
| |
| web::NavigationManagerImpl* navigation_manager = |
| web_state->navigation_manager_.get(); |
| navigation_manager->RewriteItemURLIfNecessary(item_impl.get()); |
| items[index] = std::move(item_impl); |
| } |
| web_state->navigation_manager_->Restore(storage.lastCommittedItemIndex, |
| std::move(items)); |
| |
| SessionCertificatePolicyCacheStorageBuilder cert_builder; |
| std::unique_ptr<SessionCertificatePolicyCacheImpl> cert_policy_cache = |
| cert_builder.BuildSessionCertificatePolicyCache( |
| storage.certPolicyCacheStorage, web_state->GetBrowserState()); |
| if (!cert_policy_cache) |
| cert_policy_cache = std::make_unique<SessionCertificatePolicyCacheImpl>( |
| web_state->GetBrowserState()); |
| web_state->certificate_policy_cache_ = std::move(cert_policy_cache); |
| web::SerializableUserDataManager::FromWebState(web_state) |
| ->AddSerializableUserData(storage.userData); |
| UserAgentType user_agent_type = storage.userAgentType; |
| if (user_agent_type == UserAgentType::AUTOMATIC && |
| !features::UseWebClientDefaultUserAgent()) { |
| user_agent_type = UserAgentType::MOBILE; |
| } |
| web_state->SetUserAgent(user_agent_type); |
| } |
| |
| } // namespace web |