Upstream version 9.38.198.0
[platform/framework/web/crosswalk.git] / src / third_party / WebKit / Source / core / loader / NavigationScheduler.cpp
1 /*
2  * Copyright (C) 2006, 2007, 2008, 2009, 2010 Apple Inc. All rights reserved.
3  * Copyright (C) 2008 Nokia Corporation and/or its subsidiary(-ies)
4  * Copyright (C) 2008, 2009 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/)
5  * Copyright (C) 2009 Adam Barth. All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  *
11  * 1.  Redistributions of source code must retain the above copyright
12  *     notice, this list of conditions and the following disclaimer.
13  * 2.  Redistributions in binary form must reproduce the above copyright
14  *     notice, this list of conditions and the following disclaimer in the
15  *     documentation and/or other materials provided with the distribution.
16  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
17  *     its contributors may be used to endorse or promote products derived
18  *     from this software without specific prior written permission.
19  *
20  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
21  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
22  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
23  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
24  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
25  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
26  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
27  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
29  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30  */
31
32 #include "config.h"
33 #include "core/loader/NavigationScheduler.h"
34
35 #include "bindings/core/v8/ScriptController.h"
36 #include "core/events/Event.h"
37 #include "core/fetch/ResourceLoaderOptions.h"
38 #include "core/frame/LocalFrame.h"
39 #include "core/frame/csp/ContentSecurityPolicy.h"
40 #include "core/html/HTMLFormElement.h"
41 #include "core/inspector/InspectorInstrumentation.h"
42 #include "core/loader/DocumentLoader.h"
43 #include "core/loader/FormState.h"
44 #include "core/loader/FormSubmission.h"
45 #include "core/loader/FrameLoadRequest.h"
46 #include "core/loader/FrameLoader.h"
47 #include "core/loader/FrameLoaderClient.h"
48 #include "core/loader/FrameLoaderStateMachine.h"
49 #include "core/page/BackForwardClient.h"
50 #include "core/page/Page.h"
51 #include "platform/SharedBuffer.h"
52 #include "platform/UserGestureIndicator.h"
53 #include "wtf/CurrentTime.h"
54
55 namespace blink {
56
57 unsigned NavigationDisablerForBeforeUnload::s_navigationDisableCount = 0;
58
59 class ScheduledNavigation {
60     WTF_MAKE_NONCOPYABLE(ScheduledNavigation); WTF_MAKE_FAST_ALLOCATED;
61 public:
62     ScheduledNavigation(double delay, bool lockBackForwardList, bool isLocationChange)
63         : m_delay(delay)
64         , m_lockBackForwardList(lockBackForwardList)
65         , m_isLocationChange(isLocationChange)
66         , m_wasUserGesture(UserGestureIndicator::processingUserGesture())
67     {
68         if (m_wasUserGesture)
69             m_userGestureToken = UserGestureIndicator::currentToken();
70     }
71     virtual ~ScheduledNavigation() { }
72
73     virtual void fire(LocalFrame*) = 0;
74
75     virtual bool shouldStartTimer(LocalFrame*) { return true; }
76
77     double delay() const { return m_delay; }
78     bool lockBackForwardList() const { return m_lockBackForwardList; }
79     bool isLocationChange() const { return m_isLocationChange; }
80     PassOwnPtr<UserGestureIndicator> createUserGestureIndicator()
81     {
82         if (m_wasUserGesture &&  m_userGestureToken)
83             return adoptPtr(new UserGestureIndicator(m_userGestureToken));
84         return adoptPtr(new UserGestureIndicator(DefinitelyNotProcessingUserGesture));
85     }
86
87 protected:
88     void clearUserGesture() { m_wasUserGesture = false; }
89
90 private:
91     double m_delay;
92     bool m_lockBackForwardList;
93     bool m_isLocationChange;
94     bool m_wasUserGesture;
95     RefPtr<UserGestureToken> m_userGestureToken;
96 };
97
98 class ScheduledURLNavigation : public ScheduledNavigation {
99 protected:
100     ScheduledURLNavigation(double delay, Document* originDocument, const String& url, const Referrer& referrer, bool lockBackForwardList, bool isLocationChange)
101         : ScheduledNavigation(delay, lockBackForwardList, isLocationChange)
102         , m_originDocument(originDocument)
103         , m_url(url)
104         , m_referrer(referrer)
105         , m_shouldCheckMainWorldContentSecurityPolicy(CheckContentSecurityPolicy)
106     {
107         if (ContentSecurityPolicy::shouldBypassMainWorld(originDocument))
108             m_shouldCheckMainWorldContentSecurityPolicy = DoNotCheckContentSecurityPolicy;
109     }
110
111     virtual void fire(LocalFrame* frame) OVERRIDE
112     {
113         OwnPtr<UserGestureIndicator> gestureIndicator = createUserGestureIndicator();
114         FrameLoadRequest request(m_originDocument.get(), ResourceRequest(KURL(ParsedURLString, m_url), m_referrer), "_self", m_shouldCheckMainWorldContentSecurityPolicy);
115         request.setLockBackForwardList(lockBackForwardList());
116         request.setClientRedirect(ClientRedirect);
117         frame->loader().load(request);
118     }
119
120     Document* originDocument() const { return m_originDocument.get(); }
121     String url() const { return m_url; }
122     const Referrer& referrer() const { return m_referrer; }
123
124 private:
125     RefPtrWillBePersistent<Document> m_originDocument;
126     String m_url;
127     Referrer m_referrer;
128     ContentSecurityPolicyCheck m_shouldCheckMainWorldContentSecurityPolicy;
129 };
130
131 class ScheduledRedirect FINAL : public ScheduledURLNavigation {
132 public:
133     ScheduledRedirect(double delay, Document* originDocument, const String& url, bool lockBackForwardList)
134         : ScheduledURLNavigation(delay, originDocument, url, Referrer(), lockBackForwardList, false)
135     {
136         clearUserGesture();
137     }
138
139     virtual bool shouldStartTimer(LocalFrame* frame) OVERRIDE { return frame->loader().allAncestorsAreComplete(); }
140
141     virtual void fire(LocalFrame* frame) OVERRIDE
142     {
143         OwnPtr<UserGestureIndicator> gestureIndicator = createUserGestureIndicator();
144         FrameLoadRequest request(originDocument(), ResourceRequest(KURL(ParsedURLString, url()), referrer()), "_self");
145         request.setLockBackForwardList(lockBackForwardList());
146         if (equalIgnoringFragmentIdentifier(frame->document()->url(), request.resourceRequest().url()))
147             request.resourceRequest().setCachePolicy(ReloadIgnoringCacheData);
148         request.setClientRedirect(ClientRedirect);
149         frame->loader().load(request);
150     }
151 };
152
153 class ScheduledLocationChange FINAL : public ScheduledURLNavigation {
154 public:
155     ScheduledLocationChange(Document* originDocument, const String& url, const Referrer& referrer, bool lockBackForwardList)
156         : ScheduledURLNavigation(0.0, originDocument, url, referrer, lockBackForwardList, true) { }
157 };
158
159 class ScheduledReload FINAL : public ScheduledNavigation {
160 public:
161     ScheduledReload()
162         : ScheduledNavigation(0.0, true, true)
163     {
164     }
165
166     virtual void fire(LocalFrame* frame) OVERRIDE
167     {
168         OwnPtr<UserGestureIndicator> gestureIndicator = createUserGestureIndicator();
169         frame->loader().reload(NormalReload, KURL(), nullAtom, ClientRedirect);
170     }
171 };
172
173 class ScheduledPageBlock FINAL : public ScheduledURLNavigation {
174 public:
175     ScheduledPageBlock(Document* originDocument, const String& url, const Referrer& referrer)
176         : ScheduledURLNavigation(0.0, originDocument, url, referrer, true, true)
177     {
178     }
179
180     virtual void fire(LocalFrame* frame) OVERRIDE
181     {
182         OwnPtr<UserGestureIndicator> gestureIndicator = createUserGestureIndicator();
183         SubstituteData substituteData(SharedBuffer::create(), "text/plain", "UTF-8", KURL(), ForceSynchronousLoad);
184         FrameLoadRequest request(originDocument(), ResourceRequest(KURL(ParsedURLString, url()), referrer(), ReloadIgnoringCacheData), substituteData);
185         request.setLockBackForwardList(true);
186         request.setClientRedirect(ClientRedirect);
187         frame->loader().load(request);
188     }
189 };
190
191 class ScheduledHistoryNavigation FINAL : public ScheduledNavigation {
192 public:
193     explicit ScheduledHistoryNavigation(int historySteps)
194         : ScheduledNavigation(0, false, true)
195         , m_historySteps(historySteps)
196     {
197     }
198
199     virtual void fire(LocalFrame* frame) OVERRIDE
200     {
201         OwnPtr<UserGestureIndicator> gestureIndicator = createUserGestureIndicator();
202         // go(i!=0) from a frame navigates into the history of the frame only,
203         // in both IE and NS (but not in Mozilla). We can't easily do that.
204         frame->page()->deprecatedLocalMainFrame()->loader().client()->navigateBackForward(m_historySteps);
205     }
206
207 private:
208     int m_historySteps;
209 };
210
211 class ScheduledFormSubmission FINAL : public ScheduledNavigation {
212 public:
213     ScheduledFormSubmission(PassRefPtrWillBeRawPtr<FormSubmission> submission, bool lockBackForwardList)
214         : ScheduledNavigation(0, lockBackForwardList, true)
215         , m_submission(submission)
216     {
217         ASSERT(m_submission->state());
218     }
219
220     virtual void fire(LocalFrame* frame) OVERRIDE
221     {
222         OwnPtr<UserGestureIndicator> gestureIndicator = createUserGestureIndicator();
223         FrameLoadRequest frameRequest(m_submission->state()->sourceDocument());
224         m_submission->populateFrameLoadRequest(frameRequest);
225         frameRequest.setLockBackForwardList(lockBackForwardList());
226         frameRequest.setTriggeringEvent(m_submission->event());
227         frameRequest.setFormState(m_submission->state());
228         frame->loader().load(frameRequest);
229     }
230
231 private:
232     RefPtrWillBePersistent<FormSubmission> m_submission;
233 };
234
235 NavigationScheduler::NavigationScheduler(LocalFrame* frame)
236     : m_frame(frame)
237     , m_timer(this, &NavigationScheduler::timerFired)
238 {
239 }
240
241 NavigationScheduler::~NavigationScheduler()
242 {
243 }
244
245 bool NavigationScheduler::locationChangePending()
246 {
247     return m_redirect && m_redirect->isLocationChange();
248 }
249
250 inline bool NavigationScheduler::shouldScheduleNavigation() const
251 {
252     return m_frame->page();
253 }
254
255 inline bool NavigationScheduler::shouldScheduleNavigation(const String& url) const
256 {
257     return shouldScheduleNavigation() && (protocolIsJavaScript(url) || NavigationDisablerForBeforeUnload::isNavigationAllowed());
258 }
259
260 void NavigationScheduler::scheduleRedirect(double delay, const String& url)
261 {
262     if (!shouldScheduleNavigation(url))
263         return;
264     if (delay < 0 || delay > INT_MAX / 1000)
265         return;
266     if (url.isEmpty())
267         return;
268
269     // We want a new back/forward list item if the refresh timeout is > 1 second.
270     if (!m_redirect || delay <= m_redirect->delay())
271         schedule(adoptPtr(new ScheduledRedirect(delay, m_frame->document(), url, delay <= 1)));
272 }
273
274 bool NavigationScheduler::mustLockBackForwardList(LocalFrame* targetFrame)
275 {
276     // Non-user navigation before the page has finished firing onload should not create a new back/forward item.
277     // See https://webkit.org/b/42861 for the original motivation for this.
278     if (!UserGestureIndicator::processingUserGesture() && !targetFrame->document()->loadEventFinished())
279         return true;
280
281     // From the HTML5 spec for location.assign():
282     //  "If the browsing context's session history contains only one Document,
283     //   and that was the about:blank Document created when the browsing context
284     //   was created, then the navigation must be done with replacement enabled."
285     if (!targetFrame->loader().stateMachine()->committedMultipleRealLoads()
286         && equalIgnoringCase(targetFrame->document()->url(), blankURL()))
287         return true;
288
289     // Navigation of a subframe during loading of an ancestor frame does not create a new back/forward item.
290     // The definition of "during load" is any time before all handlers for the load event have been run.
291     // See https://bugs.webkit.org/show_bug.cgi?id=14957 for the original motivation for this.
292     Frame* parentFrame = targetFrame->tree().parent();
293     return parentFrame && parentFrame->isLocalFrame() && !toLocalFrame(parentFrame)->loader().allAncestorsAreComplete();
294 }
295
296 void NavigationScheduler::scheduleLocationChange(Document* originDocument, const String& url, const Referrer& referrer, bool lockBackForwardList)
297 {
298     if (!shouldScheduleNavigation(url))
299         return;
300     if (url.isEmpty())
301         return;
302
303     lockBackForwardList = lockBackForwardList || mustLockBackForwardList(m_frame);
304
305     // If the URL we're going to navigate to is the same as the current one, except for the
306     // fragment part, we don't need to schedule the location change. We'll skip this
307     // optimization for cross-origin navigations to minimize the navigator's ability to
308     // execute timing attacks.
309     if (originDocument->securityOrigin()->canAccess(m_frame->document()->securityOrigin())) {
310         KURL parsedURL(ParsedURLString, url);
311         if (parsedURL.hasFragmentIdentifier() && equalIgnoringFragmentIdentifier(m_frame->document()->url(), parsedURL)) {
312             FrameLoadRequest request(originDocument, ResourceRequest(m_frame->document()->completeURL(url), referrer), "_self");
313             request.setLockBackForwardList(lockBackForwardList);
314             if (lockBackForwardList)
315                 request.setClientRedirect(ClientRedirect);
316             m_frame->loader().load(request);
317             return;
318         }
319     }
320
321     schedule(adoptPtr(new ScheduledLocationChange(originDocument, url, referrer, lockBackForwardList)));
322 }
323
324 void NavigationScheduler::schedulePageBlock(Document* originDocument, const Referrer& referrer)
325 {
326     ASSERT(m_frame->page());
327     const KURL& url = m_frame->document()->url();
328     schedule(adoptPtr(new ScheduledPageBlock(originDocument, url, referrer)));
329 }
330
331 void NavigationScheduler::scheduleFormSubmission(PassRefPtrWillBeRawPtr<FormSubmission> submission)
332 {
333     ASSERT(m_frame->page());
334     schedule(adoptPtr(new ScheduledFormSubmission(submission, mustLockBackForwardList(m_frame))));
335 }
336
337 void NavigationScheduler::scheduleReload()
338 {
339     if (!shouldScheduleNavigation())
340         return;
341     if (m_frame->document()->url().isEmpty())
342         return;
343     schedule(adoptPtr(new ScheduledReload));
344 }
345
346 void NavigationScheduler::scheduleHistoryNavigation(int steps)
347 {
348     if (!shouldScheduleNavigation())
349         return;
350
351     // Invalid history navigations (such as history.forward() during a new load) have the side effect of cancelling any scheduled
352     // redirects. We also avoid the possibility of cancelling the current load by avoiding the scheduled redirection altogether.
353     BackForwardClient& backForward = m_frame->page()->backForward();
354     if (steps > backForward.forwardListCount() || -steps > backForward.backListCount()) {
355         cancel();
356         return;
357     }
358
359     // In all other cases, schedule the history traversal to occur asynchronously.
360     if (steps)
361         schedule(adoptPtr(new ScheduledHistoryNavigation(steps)));
362     else
363         schedule(adoptPtr(new ScheduledReload));
364 }
365
366 void NavigationScheduler::timerFired(Timer<NavigationScheduler>*)
367 {
368     if (!m_frame->page())
369         return;
370     if (m_frame->page()->defersLoading()) {
371         InspectorInstrumentation::frameClearedScheduledNavigation(m_frame);
372         return;
373     }
374
375     RefPtr<LocalFrame> protect(m_frame);
376
377     OwnPtr<ScheduledNavigation> redirect(m_redirect.release());
378     redirect->fire(m_frame);
379     InspectorInstrumentation::frameClearedScheduledNavigation(m_frame);
380 }
381
382 void NavigationScheduler::schedule(PassOwnPtr<ScheduledNavigation> redirect)
383 {
384     ASSERT(m_frame->page());
385
386     // In a back/forward navigation, we sometimes restore history state to iframes, even though the state was generated
387     // dynamically and JS will try to put something different in the iframe. In this case, we will load stale things
388     // and/or confuse the JS when it shortly thereafter tries to schedule a location change. Let the JS have its way.
389     // FIXME: This check seems out of place.
390     if (!m_frame->loader().stateMachine()->committedFirstRealDocumentLoad() && m_frame->loader().provisionalDocumentLoader()) {
391         RefPtr<Frame> protect(m_frame);
392         m_frame->loader().provisionalDocumentLoader()->stopLoading();
393         if (!m_frame->host())
394             return;
395     }
396
397     cancel();
398     m_redirect = redirect;
399     startTimer();
400 }
401
402 void NavigationScheduler::startTimer()
403 {
404     if (!m_redirect)
405         return;
406
407     ASSERT(m_frame->page());
408     if (m_timer.isActive())
409         return;
410     if (!m_redirect->shouldStartTimer(m_frame))
411         return;
412
413     m_timer.startOneShot(m_redirect->delay(), FROM_HERE);
414     InspectorInstrumentation::frameScheduledNavigation(m_frame, m_redirect->delay());
415 }
416
417 void NavigationScheduler::cancel()
418 {
419     if (m_timer.isActive())
420         InspectorInstrumentation::frameClearedScheduledNavigation(m_frame);
421     m_timer.stop();
422     m_redirect.clear();
423 }
424
425 } // namespace blink