| // Copyright (c) 2012 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 "android_webview/browser/state_serializer.h" |
| |
| #include <string> |
| |
| #include "base/pickle.h" |
| #include "base/time/time.h" |
| #include "content/public/browser/navigation_controller.h" |
| #include "content/public/browser/navigation_entry.h" |
| #include "content/public/browser/render_process_host.h" |
| #include "content/public/browser/restore_type.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/common/page_state.h" |
| |
| // Reasons for not re-using TabNavigation under chrome/ as of 20121116: |
| // * Android WebView has different requirements for fields to store since |
| // we are the only ones using values like BaseURLForDataURL. |
| // * TabNavigation does unnecessary copying of data, which in Android |
| // WebView case, is undesired since save/restore is called in Android |
| // very frequently. |
| // * TabNavigation is tightly integrated with the rest of chrome session |
| // restore and sync code, and has other purpose in addition to serializing |
| // NavigationEntry. |
| |
| using std::string; |
| |
| namespace android_webview { |
| |
| namespace { |
| |
| const uint32_t AW_STATE_VERSION = internal::AW_STATE_VERSION_DATA_URL; |
| |
| } // namespace |
| |
| void WriteToPickle(content::WebContents& web_contents, base::Pickle* pickle) { |
| DCHECK(pickle); |
| |
| internal::WriteHeaderToPickle(pickle); |
| |
| const content::NavigationController& controller = |
| web_contents.GetController(); |
| const int entry_count = controller.GetEntryCount(); |
| const int selected_entry = controller.GetCurrentEntryIndex(); |
| DCHECK_GE(entry_count, 0); |
| DCHECK_GE(selected_entry, -1); // -1 is valid |
| DCHECK_LT(selected_entry, entry_count); |
| |
| pickle->WriteInt(entry_count); |
| pickle->WriteInt(selected_entry); |
| for (int i = 0; i < entry_count; ++i) { |
| internal::WriteNavigationEntryToPickle(*controller.GetEntryAtIndex(i), |
| pickle); |
| } |
| |
| // Please update AW_STATE_VERSION and IsSupportedVersion() if serialization |
| // format is changed. |
| // Make sure the serialization format is updated in a backwards compatible |
| // way. |
| } |
| |
| bool RestoreFromPickle(base::PickleIterator* iterator, |
| content::WebContents* web_contents) { |
| DCHECK(iterator); |
| DCHECK(web_contents); |
| |
| uint32_t state_version = internal::RestoreHeaderFromPickle(iterator); |
| if (!state_version) |
| return false; |
| |
| int entry_count = -1; |
| int selected_entry = -2; // -1 is a valid value |
| |
| if (!iterator->ReadInt(&entry_count)) |
| return false; |
| |
| if (!iterator->ReadInt(&selected_entry)) |
| return false; |
| |
| if (entry_count < 0) |
| return false; |
| if (selected_entry < -1) |
| return false; |
| if (selected_entry >= entry_count) |
| return false; |
| |
| std::vector<std::unique_ptr<content::NavigationEntry>> entries; |
| entries.reserve(entry_count); |
| for (int i = 0; i < entry_count; ++i) { |
| entries.push_back(content::NavigationEntry::Create()); |
| if (!internal::RestoreNavigationEntryFromPickle(state_version, iterator, |
| entries[i].get())) |
| return false; |
| } |
| |
| // |web_contents| takes ownership of these entries after this call. |
| content::NavigationController& controller = web_contents->GetController(); |
| controller.Restore(selected_entry, |
| content::RestoreType::LAST_SESSION_EXITED_CLEANLY, |
| &entries); |
| DCHECK_EQ(0u, entries.size()); |
| controller.LoadIfNecessary(); |
| |
| return true; |
| } |
| |
| namespace internal { |
| |
| void WriteHeaderToPickle(base::Pickle* pickle) { |
| WriteHeaderToPickle(AW_STATE_VERSION, pickle); |
| } |
| |
| void WriteHeaderToPickle(uint32_t state_version, base::Pickle* pickle) { |
| pickle->WriteUInt32(state_version); |
| } |
| |
| uint32_t RestoreHeaderFromPickle(base::PickleIterator* iterator) { |
| uint32_t state_version = -1; |
| if (!iterator->ReadUInt32(&state_version)) |
| return 0; |
| |
| if (IsSupportedVersion(state_version)) { |
| return state_version; |
| } |
| |
| return 0; |
| } |
| |
| bool IsSupportedVersion(uint32_t state_version) { |
| return state_version == internal::AW_STATE_VERSION_INITIAL || |
| state_version == internal::AW_STATE_VERSION_DATA_URL; |
| } |
| |
| void WriteNavigationEntryToPickle(const content::NavigationEntry& entry, |
| base::Pickle* pickle) { |
| WriteNavigationEntryToPickle(AW_STATE_VERSION, entry, pickle); |
| } |
| |
| void WriteNavigationEntryToPickle(uint32_t state_version, |
| const content::NavigationEntry& entry, |
| base::Pickle* pickle) { |
| DCHECK(IsSupportedVersion(state_version)); |
| pickle->WriteString(entry.GetURL().spec()); |
| pickle->WriteString(entry.GetVirtualURL().spec()); |
| |
| const content::Referrer& referrer = entry.GetReferrer(); |
| pickle->WriteString(referrer.url.spec()); |
| pickle->WriteInt(static_cast<int>(referrer.policy)); |
| |
| pickle->WriteString16(entry.GetTitle()); |
| pickle->WriteString(entry.GetPageState().ToEncodedData()); |
| pickle->WriteBool(static_cast<int>(entry.GetHasPostData())); |
| pickle->WriteString(entry.GetOriginalRequestURL().spec()); |
| pickle->WriteString(entry.GetBaseURLForDataURL().spec()); |
| |
| if (state_version >= internal::AW_STATE_VERSION_DATA_URL) { |
| const char* data = nullptr; |
| size_t size = 0; |
| const scoped_refptr<const base::RefCountedString>& s = |
| entry.GetDataURLAsString(); |
| if (s) { |
| data = s->front_as<char>(); |
| size = s->size(); |
| } |
| // Even when |entry.GetDataForDataURL()| is null we still need to write a |
| // zero-length entry to ensure the fields all line up when read back in. |
| pickle->WriteData(data, size); |
| } |
| |
| pickle->WriteBool(static_cast<int>(entry.GetIsOverridingUserAgent())); |
| pickle->WriteInt64(entry.GetTimestamp().ToInternalValue()); |
| pickle->WriteInt(entry.GetHttpStatusCode()); |
| |
| // Please update AW_STATE_VERSION and IsSupportedVersion() if serialization |
| // format is changed. |
| // Make sure the serialization format is updated in a backwards compatible |
| // way. |
| } |
| |
| bool RestoreNavigationEntryFromPickle(base::PickleIterator* iterator, |
| content::NavigationEntry* entry) { |
| return RestoreNavigationEntryFromPickle(AW_STATE_VERSION, iterator, entry); |
| } |
| |
| bool RestoreNavigationEntryFromPickle(uint32_t state_version, |
| base::PickleIterator* iterator, |
| content::NavigationEntry* entry) { |
| DCHECK(IsSupportedVersion(state_version)); |
| { |
| string url; |
| if (!iterator->ReadString(&url)) |
| return false; |
| entry->SetURL(GURL(url)); |
| } |
| |
| { |
| string virtual_url; |
| if (!iterator->ReadString(&virtual_url)) |
| return false; |
| entry->SetVirtualURL(GURL(virtual_url)); |
| } |
| |
| { |
| content::Referrer referrer; |
| string referrer_url; |
| int policy; |
| |
| if (!iterator->ReadString(&referrer_url)) |
| return false; |
| if (!iterator->ReadInt(&policy)) |
| return false; |
| |
| referrer.url = GURL(referrer_url); |
| referrer.policy = static_cast<network::mojom::ReferrerPolicy>(policy); |
| entry->SetReferrer(referrer); |
| } |
| |
| { |
| base::string16 title; |
| if (!iterator->ReadString16(&title)) |
| return false; |
| entry->SetTitle(title); |
| } |
| |
| { |
| string content_state; |
| if (!iterator->ReadString(&content_state)) |
| return false; |
| entry->SetPageState( |
| content::PageState::CreateFromEncodedData(content_state)); |
| } |
| |
| { |
| bool has_post_data; |
| if (!iterator->ReadBool(&has_post_data)) |
| return false; |
| entry->SetHasPostData(has_post_data); |
| } |
| |
| { |
| string original_request_url; |
| if (!iterator->ReadString(&original_request_url)) |
| return false; |
| entry->SetOriginalRequestURL(GURL(original_request_url)); |
| } |
| |
| { |
| string base_url_for_data_url; |
| if (!iterator->ReadString(&base_url_for_data_url)) |
| return false; |
| entry->SetBaseURLForDataURL(GURL(base_url_for_data_url)); |
| } |
| |
| if (state_version >= internal::AW_STATE_VERSION_DATA_URL) { |
| const char* data; |
| int size; |
| if (!iterator->ReadData(&data, &size)) |
| return false; |
| if (size > 0) { |
| scoped_refptr<base::RefCountedString> ref = new base::RefCountedString(); |
| ref->data().assign(data, size); |
| entry->SetDataURLAsString(ref); |
| } |
| } |
| |
| { |
| bool is_overriding_user_agent; |
| if (!iterator->ReadBool(&is_overriding_user_agent)) |
| return false; |
| entry->SetIsOverridingUserAgent(is_overriding_user_agent); |
| } |
| |
| { |
| int64_t timestamp; |
| if (!iterator->ReadInt64(×tamp)) |
| return false; |
| entry->SetTimestamp(base::Time::FromInternalValue(timestamp)); |
| } |
| |
| { |
| int http_status_code; |
| if (!iterator->ReadInt(&http_status_code)) |
| return false; |
| entry->SetHttpStatusCode(http_status_code); |
| } |
| |
| return true; |
| } |
| |
| } // namespace internal |
| |
| } // namespace android_webview |