Bug 894793 - Many open message tabs are gone after a crash or restart of Thunderbird. r=irving

This commit is contained in:
alta88 2014-03-03 13:58:43 -07:00
Родитель 9559a28968
Коммит 146d8cd3c1
2 изменённых файлов: 74 добавлений и 41 удалений

Просмотреть файл

@ -14,7 +14,9 @@ const Cc = Components.classes;
const Ci = Components.interfaces;
const Cu = Components.utils;
Cu.import("resource:///modules/IOUtils.js");
Cu.import("resource://gre/modules/AsyncShutdown.jsm");
Cu.import("resource://gre/modules/NetUtil.jsm");
Cu.import("resource://gre/modules/osfile.jsm");
Cu.import("resource://gre/modules/Services.jsm");
/**
@ -54,6 +56,21 @@ var sessionStoreManager =
*/
_shutdownStateSaved: false,
/**
* Cache the session file async writeAtomic Promise for AsyncShutdown.
*/
_promise: null,
/**
* Gets the nsIFile used for session storage.
*/
get sessionFile()
{
let sessionFile = Services.dirsvc.get("ProfD", Ci.nsIFile);
sessionFile.append("session.json");
return sessionFile;
},
/**
* This is called on startup, and when a new 3 pane window is opened after
* the last 3 pane window was closed (e.g., on the mac, closing the last
@ -79,24 +96,40 @@ var sessionStoreManager =
*/
_loadSessionFile: function ssm_loadSessionFile()
{
let sessionFile = this.sessionFile;
if (sessionFile.exists()) {
// get string containing session state
let data = IOUtils.loadFileToString(sessionFile);
if (!this.sessionFile.exists())
return;
// delete the file in case there is something crash-inducing about
// the restoration process
sessionFile.remove(false);
// clear the current state so that subsequent writes won't think
// the state hasn't changed.
this._currentStateString = null;
// Read the session state data from file, synchronously.
let inStream = Cc["@mozilla.org/network/file-input-stream;1"]
.createInstance(Ci.nsIFileInputStream);
inStream.init(this.sessionFile, -1, 0, 0);
let data = NetUtil.readInputStreamToString(inStream,
inStream.available(),
{ charset: "UTF-8" });
inStream.close();
if (data) {
try {
// parse the session state into JS objects
this._initialState = JSON.parse(data);
} catch (ex) {}
}
// Clear the current state so that subsequent writes won't think
// the state hasn't changed.
this._currentStateString = null;
try {
// Parse the session state into a JSON object.
this._initialState = JSON.parse(data);
}
catch (ex) {
Cu.reportError("sessionStoreManager: error in session state data, " + ex);
}
if (!data || !this._initialState) {
// If the file exists but there is a data read or parse fail, save the
// bad file.
let errorFile = "session_error_" +
(new Date().toISOString()).replace(/\D/g, "") + ".json";
let errorFilePath = OS.Path.join(OS.Constants.Path.profileDir, errorFile);
OS.File.move(this.sessionFile.path, errorFilePath)
.then(null, error => Cu.reportError("sessionStoreManager: failed to rename " +
this.sessionFile.path + " to " +
errorFilePath + ": " + error));
}
},
@ -125,21 +158,18 @@ var sessionStoreManager =
{
let data = JSON.stringify(aStateObj);
// write to disk only if state changed since last write
// Write async to disk only if state changed since last write.
if (data == this._currentStateString)
return;
// XXX ideally, we shouldn't be writing to disk on the UI thread,
// but the session file should be small so it might not be too big a
// problem.
let foStream = Cc["@mozilla.org/network/safe-file-output-stream;1"]
.createInstance(Ci.nsIFileOutputStream);
foStream.init(this.sessionFile, -1, -1, 0);
foStream.write(data, data.length);
foStream.QueryInterface(Ci.nsISafeOutputStream).finish();
foStream.close();
this._currentStateString = data;
this._promise = OS.File
.writeAtomic(this.sessionFile.path, data,
{ tmpPath: this.sessionFile.path + ".tmp",
flush: true })
.then(() => sessionStoreManager._currentStateString = data,
error => Cu.reportError("sessionStoreManager: error " +
"storing session state data, " +
error))
},
/**
@ -294,15 +324,10 @@ var sessionStoreManager =
this._sessionAutoSaveTimerIntervalMS,
Ci.nsITimer.TYPE_REPEATING_SLACK);
}
},
/**
* Gets the file used for session storage.
*/
get sessionFile()
{
let sessionFile = Services.dirsvc.get("ProfD", Ci.nsIFile);
sessionFile.append("session.json");
return sessionFile;
}
};
AsyncShutdown.profileBeforeChange.addBlocker(
"sessionStoreManager: session.json",
() => { return sessionStoreManager._promise; }
);

Просмотреть файл

@ -30,6 +30,9 @@ var windowHelper;
var folderA, folderB;
// With async file writes, use a delay larger than the session autosave timer.
const asyncFileWriteDelayMS = 300;
/* ........ Helper Functions ................*/
/**
@ -50,7 +53,8 @@ function readFile() {
}
function waitForFileRefresh() {
controller.sleep(sessionStoreManager._sessionAutoSaveTimerIntervalMS);
controller.sleep(sessionStoreManager._sessionAutoSaveTimerIntervalMS +
asyncFileWriteDelayMS);
jumlib.assert(sessionStoreManager.sessionFile.exists(),
"file should exist");
}
@ -401,7 +405,11 @@ function test_bad_session_file_simple() {
jumlib.assert(!sessionStoreManager._initialState,
"saved state is bad so state object should be null");
// the bad session file should have also been deleted
// Wait for bad file async rename to finish.
controller.sleep(sessionStoreManager._sessionAutoSaveTimerIntervalMS +
asyncFileWriteDelayMS);
// The bad session file should now not exist.
jumlib.assert(!sessionStoreManager.sessionFile.exists(),
"file should not exist");
}