зеркало из https://github.com/mozilla/pjs.git
Bug 522545 - some tabs not loaded, are stuck in zombie mode. r=zeniko
This commit is contained in:
Родитель
34589b3eb1
Коммит
4485ee8013
|
@ -1195,13 +1195,14 @@ SessionStoreService.prototype = {
|
|||
tabData.entries[0] = { url: browser.currentURI.spec };
|
||||
tabData.index = 1;
|
||||
}
|
||||
else if (browser.currentURI.spec == "about:blank" &&
|
||||
browser.userTypedValue) {
|
||||
// This can happen if the user opens a lot of tabs simultaneously and we
|
||||
// try to save state before all of them are properly loaded. If we crash
|
||||
// then we get a bunch of about:blank tabs which isn't what we want.
|
||||
tabData.entries[0] = { url: browser.userTypedValue };
|
||||
tabData.index = 1;
|
||||
|
||||
// If there is a userTypedValue set, then either the user has typed something
|
||||
// in the URL bar, or a new tab was opened with a URI to load. userTypedClear
|
||||
// is used to indicate whether the tab was in some sort of loading state with
|
||||
// userTypedValue.
|
||||
if (browser.userTypedValue) {
|
||||
tabData.userTypedValue = browser.userTypedValue;
|
||||
tabData.userTypedClear = browser.userTypedClear;
|
||||
}
|
||||
|
||||
var disallow = [];
|
||||
|
@ -2127,6 +2128,14 @@ SessionStoreService.prototype = {
|
|||
browser.addEventListener("load", browser.__SS_restore, true);
|
||||
}
|
||||
|
||||
// Handle userTypedValue. Setting userTypedValue seems to update gURLbar
|
||||
// as needed. Calling loadURI will cancel form filling in restoreDocument_proxy
|
||||
if (tabData.userTypedValue) {
|
||||
browser.userTypedValue = tabData.userTypedValue;
|
||||
if (tabData.userTypedClear)
|
||||
browser.loadURI(tabData.userTypedValue, null, null, true);
|
||||
}
|
||||
|
||||
aWindow.setTimeout(function(){ _this.restoreHistory(aWindow, aTabs, aTabData, aIdMap); }, 0);
|
||||
},
|
||||
|
||||
|
|
|
@ -108,6 +108,7 @@ _BROWSER_TEST_FILES = \
|
|||
browser_493467.js \
|
||||
browser_495495.js \
|
||||
browser_514751.js \
|
||||
browser_522545.js \
|
||||
browser_526613.js \
|
||||
$(NULL)
|
||||
|
||||
|
|
|
@ -0,0 +1,301 @@
|
|||
/* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* The contents of this file are subject to the Mozilla Public License Version
|
||||
* 1.1 (the "License"); you may not use this file except in compliance with
|
||||
* the License. You may obtain a copy of the License at
|
||||
* http://www.mozilla.org/MPL/
|
||||
*
|
||||
* Software distributed under the License is distributed on an "AS IS" basis,
|
||||
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||||
* for the specific language governing rights and limitations under the
|
||||
* License.
|
||||
*
|
||||
* The Original Code is sessionstore test code.
|
||||
*
|
||||
* The Initial Developer of the Original Code is
|
||||
* Mozilla Corporation.
|
||||
* Portions created by the Initial Developer are Copyright (C) 2008
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Paul O’Shannessy <paul@oshannessy.com>
|
||||
*
|
||||
* Alternatively, the contents of this file may be used under the terms of
|
||||
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
||||
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
||||
* in which case the provisions of the GPL or the LGPL are applicable instead
|
||||
* of those above. If you wish to allow use of your version of this file only
|
||||
* under the terms of either the GPL or the LGPL, and not to allow others to
|
||||
* use your version of this file under the terms of the MPL, indicate your
|
||||
* decision by deleting the provisions above and replace them with the notice
|
||||
* and other provisions required by the GPL or the LGPL. If you do not delete
|
||||
* the provisions above, a recipient may use your version of this file under
|
||||
* the terms of any one of the MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
function test() {
|
||||
/** Test for Bug 522545 **/
|
||||
|
||||
waitForExplicitFinish();
|
||||
|
||||
let ss = Cc["@mozilla.org/browser/sessionstore;1"].
|
||||
getService(Ci.nsISessionStore);
|
||||
let os = Cc["@mozilla.org/observer-service;1"].
|
||||
getService(Ci.nsIObserverService);
|
||||
|
||||
function waitForBrowserState(aState, aSetStateCallback) {
|
||||
let observer = {
|
||||
observe: function(aSubject, aTopic, aData) {
|
||||
os.removeObserver(this, "sessionstore-browser-state-restored");
|
||||
executeSoon(aSetStateCallback);
|
||||
}
|
||||
};
|
||||
os.addObserver(observer, "sessionstore-browser-state-restored", false);
|
||||
ss.setBrowserState(JSON.stringify(aState));
|
||||
}
|
||||
|
||||
// This tests the following use case:
|
||||
// User opens a new tab which gets focus. The user types something into the
|
||||
// address bar, then crashes or quits.
|
||||
function test_newTabFocused() {
|
||||
let state = {
|
||||
windows: [{
|
||||
tabs: [
|
||||
{ entries: [{ url: "about:mozilla" }] },
|
||||
{ entries: [], userTypedValue: "example.com", userTypedClear: 0 }
|
||||
],
|
||||
selected: 2
|
||||
}]
|
||||
};
|
||||
|
||||
waitForBrowserState(state, function() {
|
||||
let browser = gBrowser.selectedBrowser;
|
||||
is(browser.currentURI.spec, "about:blank",
|
||||
"No history entries still sets currentURI to about:blank");
|
||||
is(browser.userTypedValue, "example.com",
|
||||
"userTypedValue was correctly restored");
|
||||
is(browser.userTypedClear, 0,
|
||||
"userTypeClear restored as expected");
|
||||
is(gURLBar.value, "example.com",
|
||||
"Address bar's value correctly restored");
|
||||
// Change tabs to make sure address bar value gets updated
|
||||
gBrowser.selectedTab = gBrowser.tabContainer.getItemAtIndex(0);
|
||||
is(gURLBar.value, "about:mozilla",
|
||||
"Address bar's value correctly updated");
|
||||
runNextTest();
|
||||
});
|
||||
}
|
||||
|
||||
// This tests the following use case:
|
||||
// User opens a new tab which gets focus. The user types something into the
|
||||
// address bar, switches back to the first tab, then crashes or quits.
|
||||
function test_newTabNotFocused() {
|
||||
let state = {
|
||||
windows: [{
|
||||
tabs: [
|
||||
{ entries: [{ url: "about:mozilla" }] },
|
||||
{ entries: [], userTypedValue: "example.org", userTypedClear: 0 }
|
||||
],
|
||||
selected: 1
|
||||
}]
|
||||
};
|
||||
|
||||
waitForBrowserState(state, function() {
|
||||
let browser = gBrowser.getBrowserAtIndex(1);
|
||||
is(browser.currentURI.spec, "about:blank",
|
||||
"No history entries still sets currentURI to about:blank");
|
||||
is(browser.userTypedValue, "example.org",
|
||||
"userTypedValue was correctly restored");
|
||||
is(browser.userTypedClear, 0,
|
||||
"userTypeClear restored as expected");
|
||||
is(gURLBar.value, "about:mozilla",
|
||||
"Address bar's value correctly restored");
|
||||
// Change tabs to make sure address bar value gets updated
|
||||
gBrowser.selectedTab = gBrowser.tabContainer.getItemAtIndex(1);
|
||||
is(gURLBar.value, "example.org",
|
||||
"Address bar's value correctly updated");
|
||||
runNextTest();
|
||||
});
|
||||
}
|
||||
|
||||
// This tests the following use case:
|
||||
// User is in a tab with session history, then types something in the
|
||||
// address bar, then crashes or quits.
|
||||
function test_existingSHEnd_noClear() {
|
||||
let state = {
|
||||
windows: [{
|
||||
tabs: [{
|
||||
entries: [{ url: "about:mozilla" }, { url: "about:config" }],
|
||||
index: 2,
|
||||
userTypedValue: "example.com",
|
||||
userTypedClear: 0
|
||||
}]
|
||||
}]
|
||||
};
|
||||
|
||||
waitForBrowserState(state, function() {
|
||||
let browser = gBrowser.selectedBrowser;
|
||||
is(browser.currentURI.spec, "about:config",
|
||||
"browser.currentURI set to current entry in SH");
|
||||
is(browser.userTypedValue, "example.com",
|
||||
"userTypedValue was correctly restored");
|
||||
is(browser.userTypedClear, 0,
|
||||
"userTypeClear restored as expected");
|
||||
is(gURLBar.value, "example.com",
|
||||
"Address bar's value correctly restored to userTypedValue");
|
||||
runNextTest();
|
||||
});
|
||||
}
|
||||
|
||||
// This tests the following use case:
|
||||
// User is in a tab with session history, presses back at some point, then
|
||||
// types something in the address bar, then crashes or quits.
|
||||
function test_existingSHMiddle_noClear() {
|
||||
let state = {
|
||||
windows: [{
|
||||
tabs: [{
|
||||
entries: [{ url: "about:mozilla" }, { url: "about:config" }],
|
||||
index: 1,
|
||||
userTypedValue: "example.org",
|
||||
userTypedClear: 0
|
||||
}]
|
||||
}]
|
||||
};
|
||||
|
||||
waitForBrowserState(state, function() {
|
||||
let browser = gBrowser.selectedBrowser;
|
||||
is(browser.currentURI.spec, "about:mozilla",
|
||||
"browser.currentURI set to current entry in SH");
|
||||
is(browser.userTypedValue, "example.org",
|
||||
"userTypedValue was correctly restored");
|
||||
is(browser.userTypedClear, 0,
|
||||
"userTypeClear restored as expected");
|
||||
is(gURLBar.value, "example.org",
|
||||
"Address bar's value correctly restored to userTypedValue");
|
||||
runNextTest();
|
||||
});
|
||||
}
|
||||
|
||||
// This test simulates lots of tabs opening at once and then quitting/crashing.
|
||||
function test_getBrowserState_lotsOfTabsOpening() {
|
||||
let uris = [];
|
||||
for (let i = 0; i < 25; i++)
|
||||
uris.push("http://example.com/" + i);
|
||||
|
||||
// We're listening for the first non-"about:blank" load event, which should
|
||||
// indicate one of the tabs has loaded and the others haven't. So one should
|
||||
// be in a non-userTypedValue case, while others should still have
|
||||
// userTypedValue and userTypedClear set.
|
||||
gBrowser.addEventListener("load", function(aEvent) {
|
||||
if (gBrowser.currentURI.spec == "about:blank")
|
||||
return;
|
||||
gBrowser.removeEventListener("load", arguments.callee, true);
|
||||
|
||||
let state = JSON.parse(ss.getBrowserState());
|
||||
|
||||
let hasSH = state.windows[0].tabs.some(function(aTab) {
|
||||
return !("userTypedValue" in aTab) && aTab.entries[0].url;
|
||||
});
|
||||
let hasUTV = state.windows[0].tabs.some(function(aTab) {
|
||||
return aTab.userTypedValue && aTab.userTypedClear && !aTab.entries.length;
|
||||
});
|
||||
|
||||
ok(hasSH, "At least one tab has it's entry in SH");
|
||||
ok(hasUTV, "At least one tab has a userTypedValue with userTypedClear with no loaded URL");
|
||||
|
||||
runNextTest();
|
||||
|
||||
}, true);
|
||||
gBrowser.loadTabs(uris);
|
||||
}
|
||||
|
||||
// This simulates setting a userTypedValue and ensures that just typing in the
|
||||
// URL bar doesn't set userTypedClear as well.
|
||||
function test_getBrowserState_userTypedValue() {
|
||||
let state = {
|
||||
windows: [{
|
||||
tabs: [{ entries: [] }]
|
||||
}]
|
||||
};
|
||||
|
||||
waitForBrowserState(state, function() {
|
||||
let browser = gBrowser.selectedBrowser;
|
||||
// Make sure this tab isn't loading and state is clear before we test.
|
||||
is(browser.userTypedValue, null, "userTypedValue is empty to start");
|
||||
is(browser.userTypedClear, 0, "userTypedClear is 0 to start");
|
||||
|
||||
gURLBar.value = "mozilla.org";
|
||||
let event = document.createEvent("Events");
|
||||
event.initEvent("input", true, false);
|
||||
gURLBar.dispatchEvent(event);
|
||||
|
||||
is(browser.userTypedValue, "mozilla.org",
|
||||
"userTypedValue was set when changing gURLBar.value");
|
||||
is(browser.userTypedClear, 0,
|
||||
"userTypedClear was not changed when changing gURLBar.value");
|
||||
|
||||
// Now make sure ss gets these values too
|
||||
let newState = JSON.parse(ss.getBrowserState());
|
||||
is(newState.windows[0].tabs[0].userTypedValue, "mozilla.org",
|
||||
"sessionstore got correct userTypedValue");
|
||||
is(newState.windows[0].tabs[0].userTypedClear, 0,
|
||||
"sessionstore got correct userTypedClear");
|
||||
runNextTest();
|
||||
});
|
||||
}
|
||||
|
||||
// test_getBrowserState_lotsOfTabsOpening tested userTypedClear in a few cases,
|
||||
// but not necessarily any that had legitimate URIs in the state of loading
|
||||
// (eg, "http://example.com"), so this test will cover that case.
|
||||
function test_userTypedClearLoadURI() {
|
||||
let state = {
|
||||
windows: [{
|
||||
tabs: [
|
||||
{ entries: [], userTypedValue: "http://example.com", userTypedClear: 2 }
|
||||
]
|
||||
}]
|
||||
};
|
||||
|
||||
// Set state here and listen for load event because waitForBrowserState
|
||||
// doesn't guarantee all the tabs have loaded, so the test could continue
|
||||
// before we're in a testable state. This is important here because of the
|
||||
// distinction between "http://example.com" and "http://example.com/".
|
||||
ss.setBrowserState(JSON.stringify(state));
|
||||
gBrowser.addEventListener("load", function(aEvent) {
|
||||
if (gBrowser.currentURI.spec == "about:blank")
|
||||
return;
|
||||
gBrowser.removeEventListener("load", arguments.callee, true);
|
||||
|
||||
let browser = gBrowser.selectedBrowser;
|
||||
is(browser.currentURI.spec, "http://example.com/",
|
||||
"userTypedClear=2 caused userTypedValue to be loaded");
|
||||
is(browser.userTypedValue, null,
|
||||
"userTypedValue was null after loading a URI");
|
||||
is(browser.userTypedClear, 0,
|
||||
"userTypeClear reset to 0");
|
||||
is(gURLBar.value, "http://example.com/",
|
||||
"Address bar's value set after loading URI");
|
||||
runNextTest();
|
||||
}, true);
|
||||
}
|
||||
|
||||
|
||||
let tests = [test_newTabFocused, test_newTabNotFocused,
|
||||
test_existingSHEnd_noClear, test_existingSHMiddle_noClear,
|
||||
test_getBrowserState_lotsOfTabsOpening,
|
||||
test_getBrowserState_userTypedValue, test_userTypedClearLoadURI];
|
||||
let originalState = ss.getBrowserState();
|
||||
function runNextTest() {
|
||||
if (tests.length) {
|
||||
tests.shift().call();
|
||||
} else {
|
||||
ss.setBrowserState(originalState);
|
||||
executeSoon(finish);
|
||||
}
|
||||
}
|
||||
|
||||
// Run the tests!
|
||||
runNextTest();
|
||||
}
|
Загрузка…
Ссылка в новой задаче