зеркало из https://github.com/mozilla/gecko-dev.git
Bug 634834 - Throw an error if JSON stringification during history.push/replaceState navigates the page. r=bz, a=blocking
This commit is contained in:
Родитель
bacdd77be9
Коммит
44db5ba090
|
@ -9642,13 +9642,33 @@ nsDocShell::AddState(nsIVariant *aData, const nsAString& aTitle,
|
|||
|
||||
nsresult rv;
|
||||
|
||||
nsCOMPtr<nsIDocument> document = do_GetInterface(GetAsSupports(this));
|
||||
NS_ENSURE_TRUE(document, NS_ERROR_FAILURE);
|
||||
|
||||
// Step 1: Clone aData by getting its JSON representation
|
||||
// Step 1: Clone aData by getting its JSON representation.
|
||||
//
|
||||
// StringifyJSValVariant might cause arbitrary JS to run, and this code
|
||||
// might navigate the page we're on, potentially to a different origin! (bug
|
||||
// 634834) To protect against this, we abort if our principal changes due
|
||||
// to the stringify call.
|
||||
nsString dataStr;
|
||||
rv = StringifyJSValVariant(aData, dataStr);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
{
|
||||
nsCOMPtr<nsIDocument> origDocument =
|
||||
do_GetInterface(GetAsSupports(this));
|
||||
if (!origDocument)
|
||||
return NS_ERROR_DOM_SECURITY_ERR;
|
||||
nsCOMPtr<nsIPrincipal> origPrincipal = origDocument->NodePrincipal();
|
||||
|
||||
rv = StringifyJSValVariant(aData, dataStr);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
nsCOMPtr<nsIDocument> newDocument =
|
||||
do_GetInterface(GetAsSupports(this));
|
||||
if (!newDocument)
|
||||
return NS_ERROR_DOM_SECURITY_ERR;
|
||||
nsCOMPtr<nsIPrincipal> newPrincipal = newDocument->NodePrincipal();
|
||||
|
||||
PRBool principalsEqual = PR_FALSE;
|
||||
origPrincipal->Equals(newPrincipal, &principalsEqual);
|
||||
NS_ENSURE_TRUE(principalsEqual, NS_ERROR_DOM_SECURITY_ERR);
|
||||
}
|
||||
|
||||
// Check that the state object isn't too long.
|
||||
// Default max length: 640k chars.
|
||||
|
@ -9663,6 +9683,9 @@ nsDocShell::AddState(nsIVariant *aData, const nsAString& aTitle,
|
|||
NS_ENSURE_TRUE(dataStr.Length() <= (PRUint32)maxStateObjSize,
|
||||
NS_ERROR_ILLEGAL_VALUE);
|
||||
|
||||
nsCOMPtr<nsIDocument> document = do_GetInterface(GetAsSupports(this));
|
||||
NS_ENSURE_TRUE(document, NS_ERROR_FAILURE);
|
||||
|
||||
// Step 2: Resolve aURL
|
||||
PRBool equalURIs = PR_TRUE;
|
||||
nsCOMPtr<nsIURI> oldURI = mCurrentURI;
|
||||
|
|
|
@ -97,6 +97,8 @@ _TEST_FILES = \
|
|||
test_bug615501_2.html \
|
||||
test_bug615501_3.html \
|
||||
file_bug615501.html \
|
||||
test_bug634834.html \
|
||||
file_bug634834.html \
|
||||
$(NULL)
|
||||
|
||||
ifeq ($(MOZ_WIDGET_TOOLKIT),cocoa)
|
||||
|
|
|
@ -0,0 +1,5 @@
|
|||
<html>
|
||||
<body>
|
||||
Nothing to see here; just an empty page.
|
||||
</body>
|
||||
</html>
|
|
@ -0,0 +1,53 @@
|
|||
<!DOCTYPE HTML>
|
||||
<html>
|
||||
<!--
|
||||
https://bugzilla.mozilla.org/show_bug.cgi?id=634834
|
||||
-->
|
||||
<head>
|
||||
<title>Test for Bug 634834</title>
|
||||
<script type="application/javascript" src="/MochiKit/packed.js"></script>
|
||||
<script type="application/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
|
||||
<script type="application/javascript" src="/tests/SimpleTest/EventUtils.js"></script>
|
||||
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
|
||||
</head>
|
||||
<body>
|
||||
<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=634834">Mozilla Bug 634834</a>
|
||||
|
||||
<iframe id='iframe' src='file_bug634834.html' onload='iframe_loaded()'></iframe>
|
||||
|
||||
<script type='application/javascript;version=1.7'>
|
||||
SimpleTest.waitForExplicitFinish();
|
||||
|
||||
function iframe_loaded() {
|
||||
var loadedAfterPushstate = false;
|
||||
$('iframe').onload = function() {
|
||||
loadedAfterPushstate = true;
|
||||
}
|
||||
|
||||
var obj = { name: 'name' };
|
||||
obj.__defineGetter__('a', function() {
|
||||
$('iframe').contentWindow.location = 'http://example.com';
|
||||
|
||||
// Wait until we've loaded example.com.
|
||||
do {
|
||||
var r = new XMLHttpRequest();
|
||||
r.open("GET", location.href, false);
|
||||
r.overrideMimeType("text/plain");
|
||||
try { r.send(null); }
|
||||
catch (e) {}
|
||||
} while (!loadedAfterPushstate);
|
||||
});
|
||||
|
||||
try {
|
||||
$('iframe').contentWindow.history.pushState(obj, '');
|
||||
ok(false, 'pushState should throw exception.');
|
||||
}
|
||||
catch(e) {
|
||||
ok(true, 'pushState threw an exception.');
|
||||
}
|
||||
SimpleTest.finish();
|
||||
}
|
||||
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
Загрузка…
Ссылка в новой задаче