mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-01 05:43:46 +00:00
Bug 760036 - move SearchService metadata I/O off the main thread, r=enndeakin
--HG-- extra : rebase_source : 1c5f12f3730f4631a706a7b590ce6807f84fa62c
This commit is contained in:
parent
77cd14edf4
commit
28e38bbdf3
@ -8,9 +8,21 @@ const Cr = Components.results;
|
||||
|
||||
Components.utils.import("resource://gre/modules/XPCOMUtils.jsm");
|
||||
Components.utils.import("resource://gre/modules/Services.jsm");
|
||||
Components.utils.import("resource://gre/modules/commonjs/promise/core.js");
|
||||
|
||||
XPCOMUtils.defineLazyModuleGetter(this, "DeferredTask",
|
||||
"resource://gre/modules/DeferredTask.jsm");
|
||||
XPCOMUtils.defineLazyModuleGetter(this, "OS",
|
||||
"resource://gre/modules/osfile.jsm");
|
||||
XPCOMUtils.defineLazyModuleGetter(this, "Task",
|
||||
"resource://gre/modules/Task.jsm");
|
||||
|
||||
// A text encoder to UTF8, used whenever we commit the
|
||||
// engine metadata to disk.
|
||||
XPCOMUtils.defineLazyGetter(this, "gEncoder",
|
||||
function() {
|
||||
return new TextEncoder();
|
||||
});
|
||||
|
||||
const PERMS_FILE = 0644;
|
||||
const PERMS_DIRECTORY = 0755;
|
||||
@ -265,6 +277,71 @@ function FAIL(message, resultCode) {
|
||||
throw Components.Exception(message, resultCode || Cr.NS_ERROR_INVALID_ARG);
|
||||
}
|
||||
|
||||
/**
|
||||
* Utilities for dealing with promises and Task.jsm
|
||||
*/
|
||||
const TaskUtils = {
|
||||
/**
|
||||
* Add logging to a promise.
|
||||
*
|
||||
* @param {Promise} promise
|
||||
* @return {Promise} A promise behaving as |promise|, but with additional
|
||||
* logging in case of uncaught error.
|
||||
*/
|
||||
captureErrors: function captureErrors(promise) {
|
||||
return promise.then(
|
||||
null,
|
||||
function onError(reason) {
|
||||
LOG("Uncaught asynchronous error: " + reason + " at\n" + reason.stack);
|
||||
throw reason;
|
||||
}
|
||||
);
|
||||
},
|
||||
/**
|
||||
* Spawn a new Task from a generator.
|
||||
*
|
||||
* This function behaves as |Task.spawn|, with the exception that it
|
||||
* adds logging in case of uncaught error. For more information, see
|
||||
* the documentation of |Task.jsm|.
|
||||
*
|
||||
* @param {generator} gen Some generator.
|
||||
* @return {Promise} A promise built from |gen|, with the same semantics
|
||||
* as |Task.spawn(gen)|.
|
||||
*/
|
||||
spawn: function spawn(gen) {
|
||||
return this.captureErrors(Task.spawn(gen));
|
||||
},
|
||||
/**
|
||||
* Execute a mozIStorage statement asynchronously, wrapping the
|
||||
* result in a promise.
|
||||
*
|
||||
* @param {mozIStorageStaement} statement A statement to be executed
|
||||
* asynchronously. The semantics are the same as these of |statement.execute|.
|
||||
* @param {function*} onResult A callback, called for each successive result.
|
||||
*
|
||||
* @return {Promise} A promise, resolved successfully if |statement.execute|
|
||||
* succeeds, rejected if it fails.
|
||||
*/
|
||||
executeStatement: function executeStatement(statement, onResult) {
|
||||
let deferred = Promise.defer();
|
||||
onResult = onResult || function() {};
|
||||
statement.executeAsync({
|
||||
handleResult: onResult,
|
||||
handleError: function handleError(aError) {
|
||||
deferred.reject(aError);
|
||||
},
|
||||
handleCompletion: function handleCompletion(aReason) {
|
||||
statement.finalize();
|
||||
// Note that, in case of error, deferred.reject(aError)
|
||||
// has already been called by this point, so the call to
|
||||
// |deferred.resolve| is simply ignored.
|
||||
deferred.resolve(aReason);
|
||||
}
|
||||
});
|
||||
return deferred.promise;
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Ensures an assertion is met before continuing. Should be used to indicate
|
||||
* fatal errors.
|
||||
@ -2420,13 +2497,7 @@ function SearchService() {
|
||||
if (getBoolPref(BROWSER_SEARCH_PREF + "log", false))
|
||||
LOG = DO_LOG;
|
||||
|
||||
/**
|
||||
* If initialization is not complete yet, an array of
|
||||
* |nsIBrowserSearchInitObserver| expecting the result of the end of
|
||||
* initialization.
|
||||
* Once initialization is complete, |null|.
|
||||
*/
|
||||
this._initObservers = [];
|
||||
this._initObservers = Promise.defer();
|
||||
}
|
||||
|
||||
SearchService.prototype = {
|
||||
@ -2442,15 +2513,9 @@ SearchService.prototype = {
|
||||
_ensureInitialized: function SRCH_SVC__ensureInitialized() {
|
||||
if (gInitialized) {
|
||||
if (!Components.isSuccessCode(this._initRV)) {
|
||||
LOG("_ensureInitialized: failure");
|
||||
throw this._initRV;
|
||||
}
|
||||
|
||||
// Ensure that the following calls to |_ensureInitialized| can be inlined
|
||||
// to a noop. Note that we could do this at the end of both |_init| and
|
||||
// |_syncInit|, to save one call to a non-empty |_ensureInitialized|, but
|
||||
// this would complicate code.
|
||||
delete this._ensureInitialized;
|
||||
this._ensureInitialized = function SRCH_SVC__ensureInitializedDone() { };
|
||||
return;
|
||||
}
|
||||
|
||||
@ -2464,6 +2529,7 @@ SearchService.prototype = {
|
||||
//Components.utils.reportError(warning);
|
||||
LOG(warning);
|
||||
|
||||
engineMetadataService.syncInit();
|
||||
this._syncInit();
|
||||
if (!Components.isSuccessCode(this._initRV)) {
|
||||
throw this._initRV;
|
||||
@ -2471,11 +2537,11 @@ SearchService.prototype = {
|
||||
},
|
||||
|
||||
// Synchronous implementation of the initializer.
|
||||
// Used as by |_ensureInitialized| as a fallback if initialization is not
|
||||
// Used by |_ensureInitialized| as a fallback if initialization is not
|
||||
// complete. In this implementation, it is also used by |init|.
|
||||
_syncInit: function SRCH_SVC__syncInit() {
|
||||
try {
|
||||
this._loadEngines();
|
||||
this._syncLoadEngines();
|
||||
} catch (ex) {
|
||||
this._initRV = Cr.NS_ERROR_FAILURE;
|
||||
LOG("_syncInit: failure loading engines: " + ex);
|
||||
@ -2483,16 +2549,8 @@ SearchService.prototype = {
|
||||
this._addObservers();
|
||||
|
||||
gInitialized = true;
|
||||
|
||||
// Notify all of the init observers
|
||||
this._initObservers.forEach(function (observer) {
|
||||
try {
|
||||
observer.onInitComplete(this._initRV);
|
||||
} catch (x) {
|
||||
LOG("nsIBrowserInitObserver failed with error " + x);
|
||||
}
|
||||
}, this);
|
||||
this._initObservers = null;
|
||||
this._initObservers.resolve(this._initRV);
|
||||
LOG("_syncInit: Completed _syncInit");
|
||||
},
|
||||
|
||||
_engines: { },
|
||||
@ -2594,8 +2652,8 @@ SearchService.prototype = {
|
||||
}
|
||||
},
|
||||
|
||||
_loadEngines: function SRCH_SVC__loadEngines() {
|
||||
LOG("_loadEngines: start");
|
||||
_syncLoadEngines: function SRCH_SVC__syncLoadEngines() {
|
||||
LOG("_syncLoadEngines: start");
|
||||
// See if we have a cache file so we don't have to parse a bunch of XML.
|
||||
let cache = {};
|
||||
let cacheEnabled = getBoolPref(BROWSER_SEARCH_PREF + "cache.enabled", true);
|
||||
@ -3164,26 +3222,37 @@ SearchService.prototype = {
|
||||
|
||||
// nsIBrowserSearchService
|
||||
init: function SRCH_SVC_init(observer) {
|
||||
if (gInitialized) {
|
||||
if (observer) {
|
||||
executeSoon(function () {
|
||||
observer.onInitComplete(this._initRV);
|
||||
});
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (observer)
|
||||
this._initObservers.push(observer);
|
||||
|
||||
let self = this;
|
||||
if (!this._initStarted) {
|
||||
executeSoon((function () {
|
||||
// Someone may have since called syncInit via ensureInitialized - if so,
|
||||
// nothing to do here.
|
||||
if (!gInitialized)
|
||||
this._syncInit();
|
||||
}).bind(this));
|
||||
this._initStarted = true;
|
||||
TaskUtils.spawn(function task() {
|
||||
try {
|
||||
yield engineMetadataService.init();
|
||||
if (gInitialized) {
|
||||
// No need to pursue asynchronous initialization,
|
||||
// synchronous fallback had to be called and has finished.
|
||||
return;
|
||||
}
|
||||
// Complete initialization. In the current implementation,
|
||||
// this is done by calling the synchronous initializer.
|
||||
// Future versions might introduce an actually synchronous
|
||||
// implementation.
|
||||
self._syncInit();
|
||||
} catch (ex) {
|
||||
self._initObservers.reject(ex);
|
||||
}
|
||||
});
|
||||
}
|
||||
if (observer) {
|
||||
TaskUtils.captureErrors(this._initObservers.promise.then(
|
||||
function onSuccess() {
|
||||
observer.onInitComplete(self._initRV);
|
||||
},
|
||||
function onError(aReason) {
|
||||
Components.utils.reportError("Internal error while initializing SearchService: " + aReason);
|
||||
observer.onInitComplete(Components.results.NS_ERROR_UNEXPECTED);
|
||||
}
|
||||
));
|
||||
}
|
||||
},
|
||||
|
||||
@ -3568,48 +3637,183 @@ SearchService.prototype = {
|
||||
};
|
||||
|
||||
var engineMetadataService = {
|
||||
_jsonFile: OS.Path.join(OS.Constants.Path.profileDir, "search-metadata.json"),
|
||||
|
||||
/**
|
||||
* @type {nsIFile|null} The file holding the metadata.
|
||||
* Possible values for |_initState|.
|
||||
*
|
||||
* We have two paths to perform initialization: a default asynchronous
|
||||
* path and a fallback synchronous path that can interrupt the async
|
||||
* path. For this reason, initialization is actually something of a
|
||||
* finite state machine, represented with the following states:
|
||||
*
|
||||
* @enum
|
||||
*/
|
||||
get _jsonFile() {
|
||||
delete this._jsonFile;
|
||||
return this._jsonFile = FileUtils.getFile(NS_APP_USER_PROFILE_50_DIR,
|
||||
["search-metadata.json"]);
|
||||
_InitStates: {
|
||||
NOT_STARTED: "NOT_STARTED"
|
||||
/**Initialization has not started*/,
|
||||
JSON_LOADING_ATTEMPTED: "JSON_LOADING_ATTEMPTED"
|
||||
/**JSON file was loaded or does not exist*/,
|
||||
FINISHED_SUCCESS: "FINISHED_SUCCESS"
|
||||
/**Setup complete, with a success*/
|
||||
},
|
||||
|
||||
/**
|
||||
* Lazy getter for the file containing json data.
|
||||
* The latest step completed by initialization. One of |InitStates|
|
||||
*
|
||||
* @type {engineMetadataService._InitStates}
|
||||
*/
|
||||
get _store() {
|
||||
delete this._store;
|
||||
return this._store = this._loadStore();
|
||||
_initState: null,
|
||||
|
||||
// A promise fulfilled once initialization is complete
|
||||
_initializer: null,
|
||||
|
||||
/**
|
||||
* Asynchronous initializer
|
||||
*
|
||||
* Note: In the current implementation, initialization never fails.
|
||||
*/
|
||||
init: function epsInit() {
|
||||
if (!this._initializer) {
|
||||
// Launch asynchronous initialization
|
||||
let initializer = this._initializer = Promise.defer();
|
||||
TaskUtils.spawn((function task_init() {
|
||||
LOG("metadata init: starting");
|
||||
switch(this._initState) {
|
||||
case engineMetadataService._InitStates.NOT_STARTED:
|
||||
// 1. Load json file if it exists
|
||||
try {
|
||||
let contents = yield OS.File.read(this._jsonFile);
|
||||
if (this._initState == engineMetadataService._InitStates.FINISHED_SUCCESS) {
|
||||
// No need to pursue asynchronous initialization,
|
||||
// synchronous fallback was called and has finished.
|
||||
return;
|
||||
}
|
||||
this._store = JSON.parse(new TextDecoder().decode(contents));
|
||||
this._initState = engineMetadataService._InitStates.FINISHED_SUCCESS;
|
||||
} catch (ex) {
|
||||
if (this._initState == engineMetadataService._InitStates.FINISHED_SUCCESS) {
|
||||
// No need to pursue asynchronous initialization,
|
||||
// synchronous fallback was called and has finished.
|
||||
return;
|
||||
}
|
||||
if (ex.becauseNoSuchFile) {
|
||||
// If the file does not exist, we need to continue initialization
|
||||
this._initState = engineMetadataService._InitStates.JSON_LOADING_ATTEMPTED;
|
||||
} else {
|
||||
// Otherwise, we are done
|
||||
LOG("metadata init: could not load JSON file " + ex);
|
||||
this._store = {};
|
||||
this._initState = engineMetadataService._InitStates.FINISHED_SUCCESS;
|
||||
return;
|
||||
}
|
||||
}
|
||||
// Fall through to the next state
|
||||
|
||||
case engineMetadataService._InitStates.JSON_LOADING_ATTEMPTED:
|
||||
// 2. Otherwise, load db
|
||||
try {
|
||||
let store = yield this._asyncMigrateOldDB();
|
||||
if (this._initState == engineMetadataService._InitStates.FINISHED_SUCCESS) {
|
||||
// No need to pursue asynchronous initialization,
|
||||
// synchronous fallback was called and has finished.
|
||||
return;
|
||||
}
|
||||
if (!store) {
|
||||
LOG("metadata init: No store to migrate to disk");
|
||||
this._store = {};
|
||||
} else {
|
||||
// Commit the migrated store to disk immediately
|
||||
LOG("metadata init: Committing the migrated store to disk");
|
||||
this._store = store;
|
||||
this._commit(store);
|
||||
}
|
||||
} catch (ex) {
|
||||
if (this._initState == engineMetadataService._InitStates.FINISHED_SUCCESS) {
|
||||
// No need to pursue asynchronous initialization,
|
||||
// synchronous fallback was called and has finished.
|
||||
return;
|
||||
}
|
||||
LOG("metadata init: Error migrating store, using an empty store: " + ex);
|
||||
this._store = {};
|
||||
}
|
||||
this._initState = engineMetadataService._InitStates.FINISHED_SUCCESS;
|
||||
break;
|
||||
|
||||
default:
|
||||
throw new Error("Internal error: invalid state " + this._initState);
|
||||
}}).bind(this)).then(
|
||||
|
||||
// 3. Inform any observers
|
||||
function onSuccess() {
|
||||
initializer.resolve();
|
||||
},
|
||||
function onError() {
|
||||
initializer.reject();
|
||||
}
|
||||
);
|
||||
}
|
||||
return TaskUtils.captureErrors(this._initializer.promise);
|
||||
},
|
||||
|
||||
// Perform loading the first time |_store| is accessed.
|
||||
_loadStore: function() {
|
||||
let jsonFile = this._jsonFile;
|
||||
if (!jsonFile.exists()) {
|
||||
LOG("loadStore: search-metadata.json does not exist");
|
||||
/**
|
||||
* Synchronous implementation of initializer
|
||||
*
|
||||
* This initializer is able to pick wherever the async initializer
|
||||
* is waiting. The asynchronous initializer is expected to stop
|
||||
* if it detects that the synchronous initializer has completed
|
||||
* initialization.
|
||||
*/
|
||||
syncInit: function epsSyncInit() {
|
||||
LOG("metadata syncInit: starting");
|
||||
switch(this._initState) {
|
||||
case engineMetadataService._InitStates.NOT_STARTED:
|
||||
let jsonFile = new FileUtils.File(this._jsonFile);
|
||||
// 1. Load json file if it exists
|
||||
if (jsonFile.exists()) {
|
||||
try {
|
||||
let uri = Services.io.newFileURI(jsonFile);
|
||||
let stream = Services.io.newChannelFromURI(uri).open();
|
||||
this._store = parseJsonFromStream(stream);
|
||||
} catch (x) {
|
||||
LOG("metadata syncInit: could not load JSON file " + x);
|
||||
this._store = {};
|
||||
}
|
||||
this._initState = this._InitStates.FINISHED_SUCCESS;
|
||||
break;
|
||||
}
|
||||
this._initState = this._InitStates.JSON_LOADING_ATTEMPTED;
|
||||
// Fall through to the next state
|
||||
|
||||
// First check to see whether there's an existing SQLite DB to migrate
|
||||
let store = this._migrateOldDB();
|
||||
if (store) {
|
||||
// Commit the migrated store to disk immediately
|
||||
LOG("Committing the migrated store to disk");
|
||||
this._commit(store);
|
||||
return store;
|
||||
}
|
||||
case engineMetadataService._InitStates.JSON_LOADING_ATTEMPTED:
|
||||
// 2. No json, attempt to migrate from a database
|
||||
try {
|
||||
let store = this._syncMigrateOldDB();
|
||||
if (!store) {
|
||||
LOG("metadata syncInit: No store to migrate to disk");
|
||||
this._store = {};
|
||||
} else {
|
||||
// Commit the migrated store to disk immediately
|
||||
LOG("metadata syncInit: Committing the migrated store to disk");
|
||||
this._store = store;
|
||||
this._commit(store);
|
||||
}
|
||||
} catch (ex) {
|
||||
LOG("metadata syncInit: Error migrating store, using an empty store: " + ex);
|
||||
this._store = {};
|
||||
}
|
||||
this._initState = engineMetadataService._InitStates.FINISHED_SUCCESS;
|
||||
break;
|
||||
|
||||
// Migration failed, or this is a first-run - just use an empty store
|
||||
return {};
|
||||
default:
|
||||
throw new Error("Internal error: invalid state " + this._initState);
|
||||
}
|
||||
|
||||
LOG("loadStore: attempting to load store from JSON file");
|
||||
try {
|
||||
return parseJsonFromStream(NetUtil.newChannel(jsonFile).open());
|
||||
} catch (x) {
|
||||
LOG("loadStore failed to load file: "+x);
|
||||
return {};
|
||||
// 3. Inform any observers
|
||||
if (this._initializer) {
|
||||
this._initializer.resolve();
|
||||
} else {
|
||||
this._initializer = Promise.resolve();
|
||||
}
|
||||
},
|
||||
|
||||
@ -3689,44 +3893,86 @@ var engineMetadataService = {
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
* Migrate search.sqlite
|
||||
*
|
||||
* Notes:
|
||||
* - we do not remove search.sqlite after migration, so as to allow
|
||||
* downgrading and forensics;
|
||||
*/
|
||||
_migrateOldDB: function SRCH_SVC_EMS_migrate() {
|
||||
LOG("SRCH_SVC_EMS_migrate start");
|
||||
let sqliteFile = FileUtils.getFile(NS_APP_USER_PROFILE_50_DIR,
|
||||
["search.sqlite"]);
|
||||
if (!sqliteFile.exists()) {
|
||||
LOG("SRCH_SVC_EMS_migrate search.sqlite does not exist");
|
||||
return null;
|
||||
}
|
||||
let store = {};
|
||||
try {
|
||||
LOG("SRCH_SVC_EMS_migrate Migrating data from SQL");
|
||||
const sqliteDb = Services.storage.openDatabase(sqliteFile);
|
||||
const statement = sqliteDb.createStatement("SELECT * from engine_data");
|
||||
while (statement.executeStep()) {
|
||||
let row = statement.row;
|
||||
let engine = row.engineid;
|
||||
let name = row.name;
|
||||
let value = row.value;
|
||||
if (!store[engine]) {
|
||||
store[engine] = {};
|
||||
}
|
||||
_syncMigrateOldDB: function SRCH_SVC_EMS_migrate() {
|
||||
LOG("SRCH_SVC_EMS_migrate start");
|
||||
let sqliteFile = FileUtils.getFile(NS_APP_USER_PROFILE_50_DIR,
|
||||
["search.sqlite"]);
|
||||
if (!sqliteFile.exists()) {
|
||||
LOG("SRCH_SVC_EMS_migrate search.sqlite does not exist");
|
||||
return null;
|
||||
}
|
||||
let store = {};
|
||||
try {
|
||||
LOG("SRCH_SVC_EMS_migrate Migrating data from SQL");
|
||||
const sqliteDb = Services.storage.openDatabase(sqliteFile);
|
||||
const statement = sqliteDb.createStatement("SELECT * from engine_data");
|
||||
while (statement.executeStep()) {
|
||||
let row = statement.row;
|
||||
let engine = row.engineid;
|
||||
let name = row.name;
|
||||
let value = row.value;
|
||||
if (!store[engine]) {
|
||||
store[engine] = {};
|
||||
}
|
||||
store[engine][name] = value;
|
||||
}
|
||||
statement.finalize();
|
||||
sqliteDb.close();
|
||||
} catch (ex) {
|
||||
LOG("SRCH_SVC_EMS_migrate failed: " + ex);
|
||||
return null;
|
||||
}
|
||||
return store;
|
||||
},
|
||||
} catch (ex) {
|
||||
LOG("SRCH_SVC_EMS_migrate failed: " + ex);
|
||||
return null;
|
||||
}
|
||||
return store;
|
||||
},
|
||||
|
||||
/**
|
||||
* Migrate search.sqlite, asynchronously
|
||||
*
|
||||
* Notes:
|
||||
* - we do not remove search.sqlite after migration, so as to allow
|
||||
* downgrading and forensics;
|
||||
*/
|
||||
_asyncMigrateOldDB: function SRCH_SVC_EMS_asyncMigrate() {
|
||||
LOG("SRCH_SVC_EMS_asyncMigrate start");
|
||||
return TaskUtils.spawn(function task() {
|
||||
let sqliteFile = FileUtils.getFile(NS_APP_USER_PROFILE_50_DIR,
|
||||
["search.sqlite"]);
|
||||
if (!(yield OS.File.exists(sqliteFile.path))) {
|
||||
LOG("SRCH_SVC_EMS_migrate search.sqlite does not exist");
|
||||
throw new Task.Result(); // Bail out
|
||||
}
|
||||
let store = {};
|
||||
LOG("SRCH_SVC_EMS_migrate Migrating data from SQL");
|
||||
const sqliteDb = Services.storage.openDatabase(sqliteFile);
|
||||
const statement = sqliteDb.createStatement("SELECT * from engine_data");
|
||||
try {
|
||||
yield TaskUtils.executeStatement(
|
||||
statement,
|
||||
function onResult(aResultSet) {
|
||||
while (true) {
|
||||
let row = aResultSet.getNextRow();
|
||||
if (!row) {
|
||||
break;
|
||||
}
|
||||
let engine = row.engineid;
|
||||
let name = row.name;
|
||||
let value = row.value;
|
||||
if (!store[engine]) {
|
||||
store[engine] = {};
|
||||
}
|
||||
store[engine][name] = value;
|
||||
}
|
||||
}
|
||||
);
|
||||
} catch(ex) {
|
||||
// If loading the db failed, ignore the db
|
||||
throw new Task.Result(); // Bail out
|
||||
} finally {
|
||||
sqliteDb.asyncClose();
|
||||
}
|
||||
throw new Task.Result(store);
|
||||
});
|
||||
},
|
||||
|
||||
/**
|
||||
* Commit changes to disk, asynchronously.
|
||||
@ -3741,7 +3987,6 @@ var engineMetadataService = {
|
||||
*/
|
||||
_commit: function epsCommit(aStore) {
|
||||
LOG("epsCommit: start");
|
||||
|
||||
let store = aStore || this._store;
|
||||
if (!store) {
|
||||
LOG("epsCommit: nothing to do");
|
||||
@ -3750,39 +3995,32 @@ var engineMetadataService = {
|
||||
|
||||
if (!this._lazyWriter) {
|
||||
LOG("epsCommit: initializing lazy writer");
|
||||
let jsonFile = this._jsonFile;
|
||||
function writeCommit() {
|
||||
LOG("epsWriteCommit: start");
|
||||
let ostream = FileUtils.
|
||||
openSafeFileOutputStream(jsonFile,
|
||||
MODE_WRONLY | MODE_CREATE | MODE_TRUNCATE);
|
||||
|
||||
// Obtain a converter to convert our data to a UTF-8 encoded input stream.
|
||||
let converter = Cc["@mozilla.org/intl/scriptableunicodeconverter"].
|
||||
createInstance(Ci.nsIScriptableUnicodeConverter);
|
||||
converter.charset = "UTF-8";
|
||||
|
||||
let callback = function(result) {
|
||||
if (Components.isSuccessCode(result)) {
|
||||
ostream.close();
|
||||
let data = gEncoder.encode(JSON.stringify(store));
|
||||
let path = engineMetadataService._jsonFile;
|
||||
LOG("epsCommit path " + path);
|
||||
let promise = OS.File.writeAtomic(path, data, { tmpPath: path + ".tmp" });
|
||||
promise = promise.then(
|
||||
function onSuccess() {
|
||||
Services.obs.notifyObservers(null,
|
||||
SEARCH_SERVICE_TOPIC,
|
||||
SEARCH_SERVICE_METADATA_WRITTEN);
|
||||
SEARCH_SERVICE_TOPIC,
|
||||
SEARCH_SERVICE_METADATA_WRITTEN);
|
||||
LOG("epsWriteCommit: done " + result);
|
||||
}
|
||||
LOG("epsWriteCommit: done " + result);
|
||||
};
|
||||
// Asynchronously copy the data to the file.
|
||||
let istream = converter.convertToInputStream(JSON.stringify(store));
|
||||
NetUtil.asyncCopy(istream, ostream, callback);
|
||||
);
|
||||
TaskUtils.captureErrors(promise);
|
||||
}
|
||||
this._lazyWriter = new DeferredTask(writeCommit, LAZY_SERIALIZE_DELAY);
|
||||
}
|
||||
LOG("epsCommit: (re)setting timer");
|
||||
this._lazyWriter.start();
|
||||
},
|
||||
_lazyWriter: null,
|
||||
_lazyWriter: null
|
||||
};
|
||||
|
||||
engineMetadataService._initState = engineMetadataService._InitStates.NOT_STARTED;
|
||||
|
||||
const SEARCH_UPDATE_LOG_PREFIX = "*** Search update: ";
|
||||
|
||||
/**
|
||||
|
Loading…
Reference in New Issue
Block a user