mirror of
https://github.com/classilla/tenfourfox.git
synced 2024-09-28 20:56:36 +00:00
408 lines
13 KiB
JavaScript
408 lines
13 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
const Cc = Components.classes;
|
|
const Ci = Components.interfaces;
|
|
const Cr = Components.results;
|
|
const Cu = Components.utils;
|
|
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
|
|
/* Constants for password prompt telemetry.
|
|
* Mirrored in nsLoginManagerPrompter.js */
|
|
const PROMPT_DISPLAYED = 0;
|
|
|
|
const PROMPT_ADD = 1;
|
|
const PROMPT_NOTNOW = 2;
|
|
const PROMPT_NEVER = 3;
|
|
|
|
const PROMPT_UPDATE = 1;
|
|
|
|
/* ==================== LoginManagerPrompter ==================== */
|
|
/*
|
|
* LoginManagerPrompter
|
|
*
|
|
* Implements interfaces for prompting the user to enter/save/change auth info.
|
|
*
|
|
* nsILoginManagerPrompter: Used by Login Manager for saving/changing logins
|
|
* found in HTML forms.
|
|
*/
|
|
function LoginManagerPrompter() {
|
|
}
|
|
|
|
LoginManagerPrompter.prototype = {
|
|
classID : Components.ID("97d12931-abe2-11df-94e2-0800200c9a66"),
|
|
QueryInterface : XPCOMUtils.generateQI([Ci.nsILoginManagerPrompter]),
|
|
|
|
_factory : null,
|
|
_window : null,
|
|
_debug : false, // mirrors signon.debug
|
|
|
|
__pwmgr : null, // Password Manager service
|
|
get _pwmgr() {
|
|
if (!this.__pwmgr)
|
|
this.__pwmgr = Cc["@mozilla.org/login-manager;1"].
|
|
getService(Ci.nsILoginManager);
|
|
return this.__pwmgr;
|
|
},
|
|
|
|
__promptService : null, // Prompt service for user interaction
|
|
get _promptService() {
|
|
if (!this.__promptService)
|
|
this.__promptService = Cc["@mozilla.org/embedcomp/prompt-service;1"].
|
|
getService(Ci.nsIPromptService2);
|
|
return this.__promptService;
|
|
},
|
|
|
|
__strBundle : null, // String bundle for L10N
|
|
get _strBundle() {
|
|
if (!this.__strBundle) {
|
|
var bunService = Cc["@mozilla.org/intl/stringbundle;1"].
|
|
getService(Ci.nsIStringBundleService);
|
|
this.__strBundle = {
|
|
pwmgr : bunService.createBundle("chrome://passwordmgr/locale/passwordmgr.properties"),
|
|
brand : bunService.createBundle("chrome://branding/locale/brand.properties")
|
|
};
|
|
|
|
if (!this.__strBundle)
|
|
throw "String bundle for Login Manager not present!";
|
|
}
|
|
|
|
return this.__strBundle;
|
|
},
|
|
|
|
__ellipsis : null,
|
|
get _ellipsis() {
|
|
if (!this.__ellipsis) {
|
|
this.__ellipsis = "\u2026";
|
|
try {
|
|
this.__ellipsis = Services.prefs.getComplexValue(
|
|
"intl.ellipsis", Ci.nsIPrefLocalizedString).data;
|
|
} catch (e) { }
|
|
}
|
|
return this.__ellipsis;
|
|
},
|
|
|
|
/*
|
|
* log
|
|
*
|
|
* Internal function for logging debug messages to the Error Console window.
|
|
*/
|
|
log : function (message) {
|
|
if (!this._debug)
|
|
return;
|
|
|
|
dump("Pwmgr Prompter: " + message + "\n");
|
|
Services.console.logStringMessage("Pwmgr Prompter: " + message);
|
|
},
|
|
|
|
/* ---------- nsILoginManagerPrompter prompts ---------- */
|
|
|
|
/*
|
|
* init
|
|
*
|
|
*/
|
|
init : function (aWindow, aFactory) {
|
|
this._window = aWindow;
|
|
this._factory = aFactory || null;
|
|
|
|
var prefBranch = Services.prefs.getBranch("signon.");
|
|
this._debug = prefBranch.getBoolPref("debug");
|
|
this.log("===== initialized =====");
|
|
},
|
|
|
|
setE10sData : function (aBrowser, aOpener) {
|
|
throw new Error("This should be filled in when Android is multiprocess");
|
|
},
|
|
|
|
/*
|
|
* promptToSavePassword
|
|
*
|
|
*/
|
|
promptToSavePassword : function (aLogin) {
|
|
this._showSaveLoginNotification(aLogin);
|
|
Services.telemetry.getHistogramById("PWMGR_PROMPT_REMEMBER_ACTION").add(PROMPT_DISPLAYED);
|
|
},
|
|
|
|
/*
|
|
* _showLoginNotification
|
|
*
|
|
* Displays a notification doorhanger.
|
|
* @param aBody
|
|
* String message to be displayed in the doorhanger
|
|
* @param aButtons
|
|
* Buttons to display with the doorhanger
|
|
* @param aUsername
|
|
* Username string used in creating a doorhanger action
|
|
* @param aPassword
|
|
* Password string used in creating a doorhanger action
|
|
*/
|
|
_showLoginNotification : function (aBody, aButtons, aUsername, aPassword) {
|
|
let notifyWin = this._window.top;
|
|
let chromeWin = this._getChromeWindow(notifyWin).wrappedJSObject;
|
|
let browser = chromeWin.BrowserApp.getBrowserForWindow(notifyWin);
|
|
let tabID = chromeWin.BrowserApp.getTabForBrowser(browser).id;
|
|
|
|
let actionText = {
|
|
text: aUsername,
|
|
type: "EDIT",
|
|
bundle: { username: aUsername,
|
|
password: aPassword }
|
|
};
|
|
|
|
// The page we're going to hasn't loaded yet, so we want to persist
|
|
// across the first location change.
|
|
|
|
// Sites like Gmail perform a funky redirect dance before you end up
|
|
// at the post-authentication page. I don't see a good way to
|
|
// heuristically determine when to ignore such location changes, so
|
|
// we'll try ignoring location changes based on a time interval.
|
|
let options = {
|
|
persistWhileVisible: true,
|
|
timeout: Date.now() + 10000,
|
|
actionText: actionText
|
|
}
|
|
|
|
var nativeWindow = this._getNativeWindow();
|
|
if (nativeWindow)
|
|
nativeWindow.doorhanger.show(aBody, "password", aButtons, tabID, options, "LOGIN");
|
|
},
|
|
|
|
/*
|
|
* _showSaveLoginNotification
|
|
*
|
|
* Displays a notification doorhanger (rather than a popup), to allow the user to
|
|
* save the specified login. This allows the user to see the results of
|
|
* their login, and only save a login which they know worked.
|
|
*
|
|
*/
|
|
_showSaveLoginNotification : function (aLogin) {
|
|
let brandShortName = this._strBundle.brand.GetStringFromName("brandShortName");
|
|
let notificationText = this._getLocalizedString("saveLogin", [brandShortName]);
|
|
|
|
let username = aLogin.username ? this._sanitizeUsername(aLogin.username) : "";
|
|
|
|
// The callbacks in |buttons| have a closure to access the variables
|
|
// in scope here; set one to |this._pwmgr| so we can get back to pwmgr
|
|
// without a getService() call.
|
|
var pwmgr = this._pwmgr;
|
|
let promptHistogram = Services.telemetry.getHistogramById("PWMGR_PROMPT_REMEMBER_ACTION");
|
|
|
|
var buttons = [
|
|
{
|
|
label: this._getLocalizedString("neverButton"),
|
|
callback: function() {
|
|
promptHistogram.add(PROMPT_NEVER);
|
|
pwmgr.setLoginSavingEnabled(aLogin.hostname, false);
|
|
}
|
|
},
|
|
{
|
|
label: this._getLocalizedString("rememberButton"),
|
|
callback: function(checked, response) {
|
|
if (response) {
|
|
aLogin.username = response["username"] || aLogin.username;
|
|
aLogin.password = response["password"] || aLogin.password;
|
|
}
|
|
pwmgr.addLogin(aLogin);
|
|
promptHistogram.add(PROMPT_ADD);
|
|
},
|
|
positive: true
|
|
}
|
|
];
|
|
|
|
this._showLoginNotification(notificationText, buttons, aLogin.username, aLogin.password);
|
|
},
|
|
|
|
/*
|
|
* promptToChangePassword
|
|
*
|
|
* Called when we think we detect a password change for an existing
|
|
* login, when the form being submitted contains multiple password
|
|
* fields.
|
|
*
|
|
*/
|
|
promptToChangePassword : function (aOldLogin, aNewLogin) {
|
|
this._showChangeLoginNotification(aOldLogin, aNewLogin.password);
|
|
Services.telemetry.getHistogramById("PWMGR_PROMPT_UPDATE_ACTION").add(PROMPT_DISPLAYED);
|
|
},
|
|
|
|
/*
|
|
* _showChangeLoginNotification
|
|
*
|
|
* Shows the Change Password notification doorhanger.
|
|
*
|
|
*/
|
|
_showChangeLoginNotification : function (aOldLogin, aNewPassword) {
|
|
var notificationText;
|
|
if (aOldLogin.username) {
|
|
let displayUser = this._sanitizeUsername(aOldLogin.username);
|
|
notificationText = this._getLocalizedString("updatePassword", [displayUser]);
|
|
} else {
|
|
notificationText = this._getLocalizedString("updatePasswordNoUser");
|
|
}
|
|
|
|
// The callbacks in |buttons| have a closure to access the variables
|
|
// in scope here; set one to |this._pwmgr| so we can get back to pwmgr
|
|
// without a getService() call.
|
|
var self = this;
|
|
let promptHistogram = Services.telemetry.getHistogramById("PWMGR_PROMPT_UPDATE_ACTION");
|
|
|
|
var buttons = [
|
|
{
|
|
label: this._getLocalizedString("dontUpdateButton"),
|
|
callback: function() {
|
|
promptHistogram.add(PROMPT_NOTNOW);
|
|
// do nothing
|
|
}
|
|
},
|
|
{
|
|
label: this._getLocalizedString("updateButton"),
|
|
callback: function(checked, response) {
|
|
let password = response ? response["password"] : aNewPassword;
|
|
self._updateLogin(aOldLogin, password);
|
|
|
|
promptHistogram.add(PROMPT_UPDATE);
|
|
},
|
|
positive: true
|
|
}
|
|
];
|
|
|
|
this._showLoginNotification(notificationText, buttons, aOldLogin.username, aNewPassword);
|
|
},
|
|
|
|
/*
|
|
* promptToChangePasswordWithUsernames
|
|
*
|
|
* Called when we detect a password change in a form submission, but we
|
|
* don't know which existing login (username) it's for. Asks the user
|
|
* to select a username and confirm the password change.
|
|
*
|
|
* Note: The caller doesn't know the username for aNewLogin, so this
|
|
* function fills in .username and .usernameField with the values
|
|
* from the login selected by the user.
|
|
*
|
|
* Note; XPCOM stupidity: |count| is just |logins.length|.
|
|
*/
|
|
promptToChangePasswordWithUsernames : function (logins, count, aNewLogin) {
|
|
const buttonFlags = Ci.nsIPrompt.STD_YES_NO_BUTTONS;
|
|
|
|
var usernames = logins.map(l => l.username);
|
|
var dialogText = this._getLocalizedString("userSelectText");
|
|
var dialogTitle = this._getLocalizedString("passwordChangeTitle");
|
|
var selectedIndex = { value: null };
|
|
|
|
// If user selects ok, outparam.value is set to the index
|
|
// of the selected username.
|
|
var ok = this._promptService.select(null,
|
|
dialogTitle, dialogText,
|
|
usernames.length, usernames,
|
|
selectedIndex);
|
|
if (ok) {
|
|
// Now that we know which login to use, modify its password.
|
|
var selectedLogin = logins[selectedIndex.value];
|
|
this.log("Updating password for user " + selectedLogin.username);
|
|
this._updateLogin(selectedLogin, aNewLogin.password);
|
|
}
|
|
},
|
|
|
|
/* ---------- Internal Methods ---------- */
|
|
|
|
/*
|
|
* _updateLogin
|
|
*/
|
|
_updateLogin : function (login, newPassword) {
|
|
var now = Date.now();
|
|
var propBag = Cc["@mozilla.org/hash-property-bag;1"].
|
|
createInstance(Ci.nsIWritablePropertyBag);
|
|
if (newPassword) {
|
|
propBag.setProperty("password", newPassword);
|
|
// Explicitly set the password change time here (even though it would
|
|
// be changed automatically), to ensure that it's exactly the same
|
|
// value as timeLastUsed.
|
|
propBag.setProperty("timePasswordChanged", now);
|
|
}
|
|
propBag.setProperty("timeLastUsed", now);
|
|
propBag.setProperty("timesUsedIncrement", 1);
|
|
this._pwmgr.modifyLogin(login, propBag);
|
|
},
|
|
|
|
/*
|
|
* _getChromeWindow
|
|
*
|
|
* Given a content DOM window, returns the chrome window it's in.
|
|
*/
|
|
_getChromeWindow: function (aWindow) {
|
|
var chromeWin = aWindow.QueryInterface(Ci.nsIInterfaceRequestor)
|
|
.getInterface(Ci.nsIWebNavigation)
|
|
.QueryInterface(Ci.nsIDocShell)
|
|
.chromeEventHandler.ownerDocument.defaultView;
|
|
return chromeWin;
|
|
},
|
|
|
|
/*
|
|
* _getNativeWindow
|
|
*
|
|
* Returns the NativeWindow to this prompter, or null if there isn't
|
|
* a NativeWindow available (w/ error sent to logcat).
|
|
*/
|
|
_getNativeWindow : function () {
|
|
let nativeWindow = null;
|
|
try {
|
|
let notifyWin = this._window.top;
|
|
let chromeWin = this._getChromeWindow(notifyWin).wrappedJSObject;
|
|
if (chromeWin.NativeWindow) {
|
|
nativeWindow = chromeWin.NativeWindow;
|
|
} else {
|
|
Cu.reportError("NativeWindow not available on window");
|
|
}
|
|
|
|
} catch (e) {
|
|
// If any errors happen, just assume no native window helper.
|
|
Cu.reportError("No NativeWindow available: " + e);
|
|
}
|
|
return nativeWindow;
|
|
},
|
|
|
|
/*
|
|
* _getLocalizedString
|
|
*
|
|
* Can be called as:
|
|
* _getLocalizedString("key1");
|
|
* _getLocalizedString("key2", ["arg1"]);
|
|
* _getLocalizedString("key3", ["arg1", "arg2"]);
|
|
* (etc)
|
|
*
|
|
* Returns the localized string for the specified key,
|
|
* formatted if required.
|
|
*
|
|
*/
|
|
_getLocalizedString : function (key, formatArgs) {
|
|
if (formatArgs)
|
|
return this._strBundle.pwmgr.formatStringFromName(
|
|
key, formatArgs, formatArgs.length);
|
|
else
|
|
return this._strBundle.pwmgr.GetStringFromName(key);
|
|
},
|
|
|
|
/*
|
|
* _sanitizeUsername
|
|
*
|
|
* Sanitizes the specified username, by stripping quotes and truncating if
|
|
* it's too long. This helps prevent an evil site from messing with the
|
|
* "save password?" prompt too much.
|
|
*/
|
|
_sanitizeUsername : function (username) {
|
|
if (username.length > 30) {
|
|
username = username.substring(0, 30);
|
|
username += this._ellipsis;
|
|
}
|
|
return username.replace(/['"]/g, "");
|
|
},
|
|
}; // end of LoginManagerPrompter implementation
|
|
|
|
|
|
var component = [LoginManagerPrompter];
|
|
this.NSGetFactory = XPCOMUtils.generateNSGetFactory(component);
|