gecko-dev/toolkit/components/passwordmgr/test/unit/head.js
Matthew Noorenberghe d8f66a2592 Bug 667233 - Support scheme upgrades for searching logins. r=dolske
MozReview-Commit-ID: GAw0HIBgOFw
2016-05-29 17:15:52 -07:00

176 lines
6.2 KiB
JavaScript

/**
* Provides infrastructure for automated login components tests.
*/
"use strict";
////////////////////////////////////////////////////////////////////////////////
//// Globals
let { classes: Cc, interfaces: Ci, utils: Cu, results: Cr } = Components;
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
Cu.import("resource://gre/modules/Services.jsm");
Cu.import("resource://gre/modules/LoginRecipes.jsm");
Cu.import("resource://gre/modules/LoginHelper.jsm");
XPCOMUtils.defineLazyModuleGetter(this, "DownloadPaths",
"resource://gre/modules/DownloadPaths.jsm");
XPCOMUtils.defineLazyModuleGetter(this, "FileUtils",
"resource://gre/modules/FileUtils.jsm");
XPCOMUtils.defineLazyModuleGetter(this, "OS",
"resource://gre/modules/osfile.jsm");
XPCOMUtils.defineLazyModuleGetter(this, "Promise",
"resource://gre/modules/Promise.jsm");
const LoginInfo =
Components.Constructor("@mozilla.org/login-manager/loginInfo;1",
"nsILoginInfo", "init");
// Import LoginTestUtils.jsm as LoginTestUtils.
XPCOMUtils.defineLazyModuleGetter(this, "LoginTestUtils",
"resource://testing-common/LoginTestUtils.jsm");
LoginTestUtils.Assert = Assert;
const TestData = LoginTestUtils.testData;
const newPropertyBag = LoginHelper.newPropertyBag;
/**
* All the tests are implemented with add_task, this starts them automatically.
*/
function run_test()
{
do_get_profile();
run_next_test();
}
////////////////////////////////////////////////////////////////////////////////
//// Global helpers
// Some of these functions are already implemented in other parts of the source
// tree, see bug 946708 about sharing more code.
// While the previous test file should have deleted all the temporary files it
// used, on Windows these might still be pending deletion on the physical file
// system. Thus, start from a new base number every time, to make a collision
// with a file that is still pending deletion highly unlikely.
let gFileCounter = Math.floor(Math.random() * 1000000);
/**
* Returns a reference to a temporary file, that is guaranteed not to exist, and
* to have never been created before.
*
* @param aLeafName
* Suggested leaf name for the file to be created.
*
* @return nsIFile pointing to a non-existent file in a temporary directory.
*
* @note It is not enough to delete the file if it exists, or to delete the file
* after calling nsIFile.createUnique, because on Windows the delete
* operation in the file system may still be pending, preventing a new
* file with the same name to be created.
*/
function getTempFile(aLeafName)
{
// Prepend a serial number to the extension in the suggested leaf name.
let [base, ext] = DownloadPaths.splitBaseNameAndExtension(aLeafName);
let leafName = base + "-" + gFileCounter + ext;
gFileCounter++;
// Get a file reference under the temporary directory for this test file.
let file = FileUtils.getFile("TmpD", [leafName]);
do_check_false(file.exists());
do_register_cleanup(function () {
if (file.exists()) {
file.remove(false);
}
});
return file;
}
////////////////////////////////////////////////////////////////////////////////
const RecipeHelpers = {
initNewParent() {
return (new LoginRecipesParent({ defaults: null })).initializationPromise;
},
};
const MockDocument = {
/**
* Create a document for the given URL containing the given HTML with the ownerDocument of all <form>s having a mocked location.
*/
createTestDocument(aDocumentURL, aContent = "<form>", aType = "text/html") {
let parser = Cc["@mozilla.org/xmlextras/domparser;1"].
createInstance(Ci.nsIDOMParser);
parser.init();
let parsedDoc = parser.parseFromString(aContent, aType);
for (let element of parsedDoc.forms) {
this.mockOwnerDocumentProperty(element, parsedDoc, aDocumentURL);
}
return parsedDoc;
},
mockOwnerDocumentProperty(aElement, aDoc, aURL) {
// Mock the document.location object so we can unit test without a frame. We use a proxy
// instead of just assigning to the property since it's not configurable or writable.
let document = new Proxy(aDoc, {
get(target, property, receiver) {
// document.location is normally null when a document is outside of a "browsing context".
// See https://html.spec.whatwg.org/#the-location-interface
if (property == "location") {
return new URL(aURL);
}
return target[property];
},
});
// Assign element.ownerDocument to the proxy so document.location works.
Object.defineProperty(aElement, "ownerDocument", {
value: document,
});
},
};
//// Initialization functions common to all tests
add_task(function* test_common_initialize()
{
// Before initializing the service for the first time, we should copy the key
// file required to decrypt the logins contained in the SQLite databases used
// by migration tests. This file is not required for the other tests.
yield OS.File.copy(do_get_file("data/key3.db").path,
OS.Path.join(OS.Constants.Path.profileDir, "key3.db"));
// Ensure that the service and the storage module are initialized.
yield Services.logins.initializationPromise;
// Ensure that every test file starts with an empty database.
LoginTestUtils.clearData();
// Clean up after every test.
do_register_cleanup(() => LoginTestUtils.clearData());
});
/**
* Compare two FormLike to see if they represent the same information. Elements
* are compared using their @id attribute.
*/
function formLikeEqual(a, b) {
Assert.strictEqual(Object.keys(a).length, Object.keys(b).length,
"Check the formLikes have the same number of properties");
for (let propName of Object.keys(a)) {
if (propName == "elements") {
Assert.strictEqual(a.elements.length, b.elements.length, "Check element count");
for (let i = 0; i < a.elements.length; i++) {
Assert.strictEqual(a.elements[i].id, b.elements[i].id, "Check element " + i + " id");
}
continue;
}
Assert.strictEqual(a[propName], b[propName], "Compare formLike " + propName + " property");
}
}