gecko/services/sync/modules/engines/forms.js

302 lines
9.3 KiB
JavaScript

/* ***** 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 Bookmarks Sync.
*
* The Initial Developer of the Original Code is Mozilla.
* Portions created by the Initial Developer are Copyright (C) 2008
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Anant Narayanan <anant@kix.in>
*
* 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 ***** */
const EXPORTED_SYMBOLS = ['FormEngine'];
const Cc = Components.classes;
const Ci = Components.interfaces;
const Cu = Components.utils;
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
Cu.import("resource://weave/log4moz.js");
Cu.import("resource://weave/util.js");
Cu.import("resource://weave/engines.js");
Cu.import("resource://weave/stores.js");
Cu.import("resource://weave/trackers.js");
Cu.import("resource://weave/base_records/collection.js");
Cu.import("resource://weave/type_records/forms.js");
function FormEngine() {
this._init();
}
FormEngine.prototype = {
__proto__: SyncEngine.prototype,
name: "forms",
_displayName: "Forms",
description: "Take advantage of form-fill convenience on all your devices",
logName: "Forms",
_storeObj: FormStore,
_trackerObj: FormTracker,
_recordObj: FormRec,
_syncStartup: function FormEngine__syncStartup() {
this._store.cacheFormItems();
SyncEngine.prototype._syncStartup.call(this);
},
/* Wipe cache when sync finishes */
_syncFinish: function FormEngine__syncFinish(error) {
this._store.clearFormCache();
SyncEngine.prototype._syncFinish.call(this);
},
_findDupe: function _findDupe(item) {
// Search through the items to find a matching name/value
for (let [guid, {name, value}] in Iterator(this._store._formItems))
if (name == item.name && value == item.value)
return guid;
}
};
function FormStore() {
this._init();
}
FormStore.prototype = {
__proto__: Store.prototype,
name: "forms",
_logName: "FormStore",
_formItems: null,
get _formDB() {
let file = Cc["@mozilla.org/file/directory_service;1"].
getService(Ci.nsIProperties).
get("ProfD", Ci.nsIFile);
file.append("formhistory.sqlite");
let stor = Cc["@mozilla.org/storage/service;1"].
getService(Ci.mozIStorageService);
let formDB = stor.openDatabase(file);
this.__defineGetter__("_formDB", function() formDB);
return formDB;
},
get _formHistory() {
let formHistory = Cc["@mozilla.org/satchel/form-history;1"].
getService(Ci.nsIFormHistory2);
this.__defineGetter__("_formHistory", function() formHistory);
return formHistory;
},
get _formStatement() {
// This is essentially:
// SELECT * FROM moz_formhistory ORDER BY 1.0 * (lastUsed - minLast) /
// (maxLast - minLast) * timesUsed / minTimes DESC LIMIT 200
let stmnt = this._formDB.createStatement(
"SELECT * FROM moz_formhistory ORDER BY 1.0 * (lastUsed - \
(SELECT lastUsed FROM moz_formhistory ORDER BY lastUsed ASC LIMIT 1)) / \
((SELECT lastUsed FROM moz_formhistory ORDER BY lastUsed DESC LIMIT 1) - \
(SELECT lastUsed FROM moz_formhistory ORDER BY lastUsed ASC LIMIT 1)) * \
timesUsed / (SELECT timesUsed FROM moz_formhistory ORDER BY timesUsed DESC LIMIT 1) \
DESC LIMIT 200"
);
this.__defineGetter__("_formStatement", function() stmnt);
return stmnt;
},
cacheFormItems: function FormStore_cacheFormItems() {
this._log.trace("Caching all form items");
this._formItems = this.getAllIDs();
},
clearFormCache: function FormStore_clearFormCache() {
this._log.trace("Clearing form cache");
this._formItems = null;
},
getAllIDs: function FormStore_getAllIDs() {
let items = {};
let stmnt = this._formStatement;
while (stmnt.executeStep()) {
let nam = stmnt.getUTF8String(1);
let val = stmnt.getUTF8String(2);
let key = Utils.sha1(nam + val);
items[key] = { name: nam, value: val };
}
stmnt.reset();
return items;
},
changeItemID: function FormStore_changeItemID(oldID, newID) {
this._log.warn("FormStore IDs are data-dependent, cannot change!");
},
itemExists: function FormStore_itemExists(id) {
return (id in this._formItems);
},
createRecord: function FormStore_createRecord(guid, cryptoMetaURL) {
let record = new FormRec();
record.id = guid;
if (guid in this._formItems) {
let item = this._formItems[guid];
record.encryption = cryptoMetaURL;
record.name = item.name;
record.value = item.value;
} else {
record.deleted = true;
}
return record;
},
create: function FormStore_create(record) {
this._log.debug("Adding form record for " + record.name);
this._formHistory.addEntry(record.name, record.value);
},
remove: function FormStore_remove(record) {
this._log.trace("Removing form record: " + record.id);
if (record.id in this._formItems) {
let item = this._formItems[record.id];
this._formHistory.removeEntry(item.name, item.value);
return;
}
this._log.trace("Invalid GUID found, ignoring remove request.");
},
update: function FormStore_update(record) {
this._log.warn("Ignoring form record update request!");
},
wipe: function FormStore_wipe() {
this._formHistory.removeAllEntries();
}
};
function FormTracker() {
this._init();
}
FormTracker.prototype = {
__proto__: Tracker.prototype,
name: "forms",
_logName: "FormTracker",
file: "form",
QueryInterface: XPCOMUtils.generateQI([Ci.nsIFormSubmitObserver]),
__observerService: null,
get _observerService() {
if (!this.__observerService)
this.__observerService = Cc["@mozilla.org/observer-service;1"].
getService(Ci.nsIObserverService);
return this.__observerService;
},
_init: function FormTracker__init() {
this.__proto__.__proto__._init.call(this);
this._log.trace("FormTracker initializing!");
this._observerService.addObserver(this, "earlyformsubmit", false);
},
/* 10 points per form element */
notify: function FormTracker_notify(formElement, aWindow, actionURI) {
if (this.ignoreAll)
return;
this._log.trace("Form submission notification for " + actionURI.spec);
// XXX Bug 487541 Copy the logic from nsFormHistory::Notify to avoid
// divergent logic, which can lead to security issues, until there's a
// better way to get satchel's results like with a notification.
// Determine if a dom node has the autocomplete attribute set to "off"
let completeOff = function(domNode) {
let autocomplete = domNode.getAttribute("autocomplete");
return autocomplete && autocomplete.search(/^off$/i) == 0;
}
if (completeOff(formElement)) {
this._log.trace("Form autocomplete set to off");
return;
}
/* Get number of elements in form, add points and changedIDs */
let len = formElement.length;
let elements = formElement.elements;
for (let i = 0; i < len; i++) {
let el = elements.item(i);
// Grab the name for debugging, but check if empty when satchel would
let name = el.name;
if (name === "")
name = el.id;
if (!(el instanceof Ci.nsIDOMHTMLInputElement)) {
this._log.trace(name + " is not a DOMHTMLInputElement: " + el);
continue;
}
if (el.type.search(/^text$/i) != 0) {
this._log.trace(name + "'s type is not 'text': " + el.type);
continue;
}
if (completeOff(el)) {
this._log.trace(name + "'s autocomplete set to off");
continue;
}
if (el.value === "") {
this._log.trace(name + "'s value is empty");
continue;
}
if (el.value == el.defaultValue) {
this._log.trace(name + "'s value is the default");
continue;
}
if (name === "") {
this._log.trace("Text input element has no name or id");
continue;
}
this._log.trace("Logging form element: " + name + " :: " + el.value);
this.addChangedID(Utils.sha1(name + el.value));
this._score += 10;
}
}
};