• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "android_webview/native/state_serializer.h"
6 
7 #include <string>
8 
9 #include "base/memory/scoped_vector.h"
10 #include "base/pickle.h"
11 #include "base/time/time.h"
12 #include "content/public/browser/child_process_security_policy.h"
13 #include "content/public/browser/navigation_controller.h"
14 #include "content/public/browser/navigation_entry.h"
15 #include "content/public/browser/render_process_host.h"
16 #include "content/public/browser/web_contents.h"
17 #include "content/public/common/page_state.h"
18 
19 // Reasons for not re-using TabNavigation under chrome/ as of 20121116:
20 // * Android WebView has different requirements for fields to store since
21 //   we are the only ones using values like BaseURLForDataURL.
22 // * TabNavigation does unnecessary copying of data, which in Android
23 //   WebView case, is undesired since save/restore is called in Android
24 //   very frequently.
25 // * TabNavigation is tightly integrated with the rest of chrome session
26 //   restore and sync code, and has other purpose in addition to serializing
27 //   NavigationEntry.
28 
29 using std::string;
30 
31 namespace android_webview {
32 
33 namespace {
34 
35 // Sanity check value that we are restoring from a valid pickle.
36 // This can potentially used as an actual serialization version number in the
37 // future if we ever decide to support restoring from older versions.
38 const uint32 AW_STATE_VERSION = 20130814;
39 
40 }  // namespace
41 
WriteToPickle(const content::WebContents & web_contents,Pickle * pickle)42 bool WriteToPickle(const content::WebContents& web_contents,
43                    Pickle* pickle) {
44   DCHECK(pickle);
45 
46   if (!internal::WriteHeaderToPickle(pickle))
47     return false;
48 
49   const content::NavigationController& controller =
50       web_contents.GetController();
51   const int entry_count = controller.GetEntryCount();
52   const int selected_entry = controller.GetCurrentEntryIndex();
53   DCHECK(entry_count >= 0);
54   DCHECK(selected_entry >= -1);  // -1 is valid
55   DCHECK(selected_entry < entry_count);
56 
57   if (!pickle->WriteInt(entry_count))
58     return false;
59 
60   if (!pickle->WriteInt(selected_entry))
61     return false;
62 
63   for (int i = 0; i < entry_count; ++i) {
64     if (!internal::WriteNavigationEntryToPickle(*controller.GetEntryAtIndex(i),
65                                                 pickle))
66       return false;
67   }
68 
69   // Please update AW_STATE_VERSION if serialization format is changed.
70 
71   return true;
72 }
73 
RestoreFromPickle(PickleIterator * iterator,content::WebContents * web_contents)74 bool RestoreFromPickle(PickleIterator* iterator,
75                        content::WebContents* web_contents) {
76   DCHECK(iterator);
77   DCHECK(web_contents);
78 
79   if (!internal::RestoreHeaderFromPickle(iterator))
80     return false;
81 
82   int entry_count = -1;
83   int selected_entry = -2;  // -1 is a valid value
84 
85   if (!iterator->ReadInt(&entry_count))
86     return false;
87 
88   if (!iterator->ReadInt(&selected_entry))
89     return false;
90 
91   if (entry_count < 0)
92     return false;
93   if (selected_entry < -1)
94     return false;
95   if (selected_entry >= entry_count)
96     return false;
97 
98   ScopedVector<content::NavigationEntry> restored_entries;
99   for (int i = 0; i < entry_count; ++i) {
100     restored_entries.push_back(content::NavigationEntry::Create());
101     if (!internal::RestoreNavigationEntryFromPickle(iterator,
102                                                     restored_entries[i]))
103       return false;
104 
105     restored_entries[i]->SetPageID(i);
106   }
107 
108   // |web_contents| takes ownership of these entries after this call.
109   content::NavigationController& controller = web_contents->GetController();
110   controller.Restore(
111       selected_entry,
112       content::NavigationController::RESTORE_LAST_SESSION_EXITED_CLEANLY,
113       &restored_entries.get());
114   DCHECK_EQ(0u, restored_entries.size());
115 
116   if (controller.GetActiveEntry()) {
117     // Set up the file access rights for the selected navigation entry.
118     // TODO(joth): This is duplicated from chrome/.../session_restore.cc and
119     // should be shared e.g. in  NavigationController. http://crbug.com/68222
120     const int id = web_contents->GetRenderProcessHost()->GetID();
121     const content::PageState& page_state =
122         controller.GetActiveEntry()->GetPageState();
123     const std::vector<base::FilePath>& file_paths =
124         page_state.GetReferencedFiles();
125     for (std::vector<base::FilePath>::const_iterator file = file_paths.begin();
126          file != file_paths.end(); ++file) {
127       content::ChildProcessSecurityPolicy::GetInstance()->GrantReadFile(id,
128                                                                         *file);
129     }
130   }
131 
132   controller.LoadIfNecessary();
133 
134   return true;
135 }
136 
137 namespace internal {
138 
WriteHeaderToPickle(Pickle * pickle)139 bool WriteHeaderToPickle(Pickle* pickle) {
140   return pickle->WriteUInt32(AW_STATE_VERSION);
141 }
142 
RestoreHeaderFromPickle(PickleIterator * iterator)143 bool RestoreHeaderFromPickle(PickleIterator* iterator) {
144   uint32 state_version = -1;
145   if (!iterator->ReadUInt32(&state_version))
146     return false;
147 
148   if (AW_STATE_VERSION != state_version)
149     return false;
150 
151   return true;
152 }
153 
WriteNavigationEntryToPickle(const content::NavigationEntry & entry,Pickle * pickle)154 bool WriteNavigationEntryToPickle(const content::NavigationEntry& entry,
155                                   Pickle* pickle) {
156   if (!pickle->WriteString(entry.GetURL().spec()))
157     return false;
158 
159   if (!pickle->WriteString(entry.GetVirtualURL().spec()))
160     return false;
161 
162   const content::Referrer& referrer = entry.GetReferrer();
163   if (!pickle->WriteString(referrer.url.spec()))
164     return false;
165   if (!pickle->WriteInt(static_cast<int>(referrer.policy)))
166     return false;
167 
168   if (!pickle->WriteString16(entry.GetTitle()))
169     return false;
170 
171   if (!pickle->WriteString(entry.GetPageState().ToEncodedData()))
172     return false;
173 
174   if (!pickle->WriteBool(static_cast<int>(entry.GetHasPostData())))
175     return false;
176 
177   if (!pickle->WriteString(entry.GetOriginalRequestURL().spec()))
178     return false;
179 
180   if (!pickle->WriteString(entry.GetBaseURLForDataURL().spec()))
181     return false;
182 
183   if (!pickle->WriteBool(static_cast<int>(entry.GetIsOverridingUserAgent())))
184     return false;
185 
186   if (!pickle->WriteInt64(entry.GetTimestamp().ToInternalValue()))
187     return false;
188 
189   if (!pickle->WriteInt(entry.GetHttpStatusCode()))
190     return false;
191 
192   // Please update AW_STATE_VERSION if serialization format is changed.
193 
194   return true;
195 }
196 
RestoreNavigationEntryFromPickle(PickleIterator * iterator,content::NavigationEntry * entry)197 bool RestoreNavigationEntryFromPickle(PickleIterator* iterator,
198                                       content::NavigationEntry* entry) {
199   {
200     string url;
201     if (!iterator->ReadString(&url))
202       return false;
203     entry->SetURL(GURL(url));
204   }
205 
206   {
207     string virtual_url;
208     if (!iterator->ReadString(&virtual_url))
209       return false;
210     entry->SetVirtualURL(GURL(virtual_url));
211   }
212 
213   {
214     content::Referrer referrer;
215     string referrer_url;
216     int policy;
217 
218     if (!iterator->ReadString(&referrer_url))
219       return false;
220     if (!iterator->ReadInt(&policy))
221       return false;
222 
223     referrer.url = GURL(referrer_url);
224     referrer.policy = static_cast<blink::WebReferrerPolicy>(policy);
225     entry->SetReferrer(referrer);
226   }
227 
228   {
229     string16 title;
230     if (!iterator->ReadString16(&title))
231       return false;
232     entry->SetTitle(title);
233   }
234 
235   {
236     string content_state;
237     if (!iterator->ReadString(&content_state))
238       return false;
239     entry->SetPageState(
240         content::PageState::CreateFromEncodedData(content_state));
241   }
242 
243   {
244     bool has_post_data;
245     if (!iterator->ReadBool(&has_post_data))
246       return false;
247     entry->SetHasPostData(has_post_data);
248   }
249 
250   {
251     string original_request_url;
252     if (!iterator->ReadString(&original_request_url))
253       return false;
254     entry->SetOriginalRequestURL(GURL(original_request_url));
255   }
256 
257   {
258     string base_url_for_data_url;
259     if (!iterator->ReadString(&base_url_for_data_url))
260       return false;
261     entry->SetBaseURLForDataURL(GURL(base_url_for_data_url));
262   }
263 
264   {
265     bool is_overriding_user_agent;
266     if (!iterator->ReadBool(&is_overriding_user_agent))
267       return false;
268     entry->SetIsOverridingUserAgent(is_overriding_user_agent);
269   }
270 
271   {
272     int64 timestamp;
273     if (!iterator->ReadInt64(&timestamp))
274       return false;
275     entry->SetTimestamp(base::Time::FromInternalValue(timestamp));
276   }
277 
278   {
279     int http_status_code;
280     if (!iterator->ReadInt(&http_status_code))
281       return false;
282     entry->SetHttpStatusCode(http_status_code);
283   }
284 
285   return true;
286 }
287 
288 }  // namespace internal
289 
290 }  // namespace android_webview
291