2012-07-17 18:54:07 +00:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
|
|
|
|
2013-02-06 19:18:26 +00:00
|
|
|
const Clutter = imports.gi.Clutter;
|
2012-07-17 18:54:07 +00:00
|
|
|
const Gio = imports.gi.Gio;
|
2013-03-18 04:59:56 +00:00
|
|
|
const GLib = imports.gi.GLib;
|
2012-07-17 18:54:07 +00:00
|
|
|
const Lang = imports.lang;
|
2012-08-07 14:49:22 +00:00
|
|
|
const Mainloop = imports.mainloop;
|
2012-07-17 18:54:07 +00:00
|
|
|
const Signals = imports.signals;
|
2013-07-15 21:56:44 +00:00
|
|
|
const St = imports.gi.St;
|
2012-07-17 18:54:07 +00:00
|
|
|
|
|
|
|
const Batch = imports.gdm.batch;
|
|
|
|
const Fprint = imports.gdm.fingerprint;
|
2013-10-10 08:21:47 +00:00
|
|
|
const OVirt = imports.gdm.oVirt;
|
2012-07-17 18:54:07 +00:00
|
|
|
const Main = imports.ui.main;
|
2012-08-03 15:10:45 +00:00
|
|
|
const Params = imports.misc.params;
|
2013-07-18 18:40:10 +00:00
|
|
|
const ShellEntry = imports.ui.shellEntry;
|
2013-06-27 12:54:19 +00:00
|
|
|
const SmartcardManager = imports.misc.smartcardManager;
|
2012-07-17 18:54:07 +00:00
|
|
|
const Tweener = imports.ui.tweener;
|
|
|
|
|
|
|
|
const PASSWORD_SERVICE_NAME = 'gdm-password';
|
|
|
|
const FINGERPRINT_SERVICE_NAME = 'gdm-fingerprint';
|
2013-06-27 12:54:19 +00:00
|
|
|
const SMARTCARD_SERVICE_NAME = 'gdm-smartcard';
|
2013-10-10 08:21:47 +00:00
|
|
|
const OVIRT_SERVICE_NAME = 'gdm-ovirtcred';
|
2012-07-17 18:54:07 +00:00
|
|
|
const FADE_ANIMATION_TIME = 0.16;
|
2013-02-06 19:18:26 +00:00
|
|
|
const CLONE_FADE_ANIMATION_TIME = 0.25;
|
2012-07-17 18:54:07 +00:00
|
|
|
|
|
|
|
const LOGIN_SCREEN_SCHEMA = 'org.gnome.login-screen';
|
2013-07-29 18:18:30 +00:00
|
|
|
const PASSWORD_AUTHENTICATION_KEY = 'enable-password-authentication';
|
2012-07-17 18:54:07 +00:00
|
|
|
const FINGERPRINT_AUTHENTICATION_KEY = 'enable-fingerprint-authentication';
|
2013-06-27 12:54:19 +00:00
|
|
|
const SMARTCARD_AUTHENTICATION_KEY = 'enable-smartcard-authentication';
|
2012-07-17 18:54:07 +00:00
|
|
|
const BANNER_MESSAGE_KEY = 'banner-message-enable';
|
|
|
|
const BANNER_MESSAGE_TEXT_KEY = 'banner-message-text';
|
2012-08-07 14:49:22 +00:00
|
|
|
const ALLOWED_FAILURES_KEY = 'allowed-failures';
|
2012-07-17 18:54:07 +00:00
|
|
|
|
|
|
|
const LOGO_KEY = 'logo';
|
2012-10-30 17:26:30 +00:00
|
|
|
const DISABLE_USER_LIST_KEY = 'disable-user-list';
|
2012-07-17 18:54:07 +00:00
|
|
|
|
2013-03-18 04:59:56 +00:00
|
|
|
// Give user 16ms to read each character of a PAM message
|
|
|
|
const USER_READ_TIME = 16
|
|
|
|
|
2013-08-19 16:00:33 +00:00
|
|
|
const MessageType = {
|
|
|
|
NONE: 0,
|
|
|
|
ERROR: 1,
|
|
|
|
INFO: 2,
|
|
|
|
HINT: 3
|
|
|
|
};
|
|
|
|
|
2012-07-17 18:54:07 +00:00
|
|
|
function fadeInActor(actor) {
|
|
|
|
if (actor.opacity == 255 && actor.visible)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
let hold = new Batch.Hold();
|
|
|
|
actor.show();
|
|
|
|
let [minHeight, naturalHeight] = actor.get_preferred_height(-1);
|
|
|
|
|
|
|
|
actor.opacity = 0;
|
|
|
|
actor.set_height(0);
|
|
|
|
Tweener.addTween(actor,
|
|
|
|
{ opacity: 255,
|
|
|
|
height: naturalHeight,
|
|
|
|
time: FADE_ANIMATION_TIME,
|
|
|
|
transition: 'easeOutQuad',
|
|
|
|
onComplete: function() {
|
|
|
|
this.set_height(-1);
|
|
|
|
hold.release();
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
return hold;
|
|
|
|
}
|
|
|
|
|
|
|
|
function fadeOutActor(actor) {
|
|
|
|
if (!actor.visible || actor.opacity == 0) {
|
|
|
|
actor.opacity = 0;
|
|
|
|
actor.hide();
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
let hold = new Batch.Hold();
|
|
|
|
Tweener.addTween(actor,
|
|
|
|
{ opacity: 0,
|
|
|
|
height: 0,
|
|
|
|
time: FADE_ANIMATION_TIME,
|
|
|
|
transition: 'easeOutQuad',
|
|
|
|
onComplete: function() {
|
|
|
|
this.hide();
|
|
|
|
this.set_height(-1);
|
|
|
|
hold.release();
|
|
|
|
},
|
|
|
|
});
|
|
|
|
return hold;
|
|
|
|
}
|
|
|
|
|
2013-02-06 19:18:26 +00:00
|
|
|
function cloneAndFadeOutActor(actor) {
|
|
|
|
// Immediately hide actor so its sibling can have its space
|
|
|
|
// and position, but leave a non-reactive clone on-screen,
|
|
|
|
// so from the user's point of view it smoothly fades away
|
|
|
|
// and reveals its sibling.
|
|
|
|
actor.hide();
|
|
|
|
|
|
|
|
let clone = new Clutter.Clone({ source: actor,
|
|
|
|
reactive: false });
|
|
|
|
|
|
|
|
Main.uiGroup.add_child(clone);
|
|
|
|
|
|
|
|
let [x, y] = actor.get_transformed_position();
|
|
|
|
clone.set_position(x, y);
|
|
|
|
|
|
|
|
let hold = new Batch.Hold();
|
|
|
|
Tweener.addTween(clone,
|
|
|
|
{ opacity: 0,
|
|
|
|
time: CLONE_FADE_ANIMATION_TIME,
|
|
|
|
transition: 'easeOutQuad',
|
|
|
|
onComplete: function() {
|
|
|
|
clone.destroy();
|
|
|
|
hold.release();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return hold;
|
|
|
|
}
|
|
|
|
|
2014-03-07 21:14:05 +00:00
|
|
|
const VerificationStatus = {
|
|
|
|
NOT_VERIFYING: 0,
|
2014-03-08 00:09:48 +00:00
|
|
|
ASKING_FOR_USERNAME: 1,
|
|
|
|
VERIFYING: 2,
|
|
|
|
VERIFICATION_FAILED: 3,
|
|
|
|
VERIFICATION_SUCCEEDED: 4,
|
2014-03-07 21:14:05 +00:00
|
|
|
};
|
|
|
|
|
2012-07-17 18:54:07 +00:00
|
|
|
const ShellUserVerifier = new Lang.Class({
|
|
|
|
Name: 'ShellUserVerifier',
|
|
|
|
|
2012-08-03 15:10:45 +00:00
|
|
|
_init: function(client, params) {
|
|
|
|
params = Params.parse(params, { reauthenticationOnly: false });
|
|
|
|
this._reauthOnly = params.reauthenticationOnly;
|
|
|
|
|
2012-07-17 18:54:07 +00:00
|
|
|
this._client = client;
|
|
|
|
|
|
|
|
this._settings = new Gio.Settings({ schema: LOGIN_SCREEN_SCHEMA });
|
2013-07-29 18:18:30 +00:00
|
|
|
this._settings.connect('changed',
|
|
|
|
Lang.bind(this, this._updateDefaultService));
|
2012-07-17 18:54:07 +00:00
|
|
|
|
|
|
|
this._fprintManager = new Fprint.FprintManager();
|
2013-06-27 12:54:19 +00:00
|
|
|
this._smartcardManager = SmartcardManager.getSmartcardManager();
|
|
|
|
|
|
|
|
// We check for smartcards right away, since an inserted smartcard
|
|
|
|
// at startup should result in immediately initiating authentication.
|
|
|
|
// This is different than fingeprint readers, where we only check them
|
|
|
|
// after a user has been picked.
|
|
|
|
this._checkForSmartcard();
|
|
|
|
|
|
|
|
this._smartcardManager.connect('smartcard-inserted',
|
|
|
|
Lang.bind(this, this._checkForSmartcard));
|
|
|
|
this._smartcardManager.connect('smartcard-removed',
|
|
|
|
Lang.bind(this, this._checkForSmartcard));
|
|
|
|
|
2013-03-18 04:59:56 +00:00
|
|
|
this._messageQueue = [];
|
|
|
|
this._messageQueueTimeoutId = 0;
|
|
|
|
this.hasPendingMessages = false;
|
2013-07-22 14:59:57 +00:00
|
|
|
this.reauthenticating = false;
|
2012-08-07 14:49:22 +00:00
|
|
|
|
2013-10-10 08:21:47 +00:00
|
|
|
this._oVirtCredentialsManager = OVirt.getOVirtCredentialsManager();
|
2014-03-07 21:10:25 +00:00
|
|
|
this._oVirtCredentialsManager.connect('user-authenticated', Lang.bind(this, this._oVirtUserAuthenticated));
|
2013-10-10 08:21:47 +00:00
|
|
|
if (this._oVirtCredentialsManager.hasToken())
|
|
|
|
this._oVirtUserAuthenticated(this._oVirtCredentialsManager.getToken());
|
|
|
|
|
2014-03-07 21:09:32 +00:00
|
|
|
this._reset();
|
|
|
|
},
|
|
|
|
|
|
|
|
_reset: function() {
|
|
|
|
// Clear previous attempts to authenticate
|
2014-03-07 21:14:05 +00:00
|
|
|
this.verificationStatus = VerificationStatus.NOT_VERIFYING;
|
2014-03-08 00:09:48 +00:00
|
|
|
this._userName = null;
|
2014-03-07 21:09:32 +00:00
|
|
|
this._failCounter = 0;
|
|
|
|
this._updateDefaultService();
|
|
|
|
this.emit('reset');
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
|
2014-03-08 00:09:48 +00:00
|
|
|
begin: function() {
|
|
|
|
if (this._mode == AuthPromptMode.UNLOCK_ONLY) {
|
|
|
|
// The user is constant at the unlock screen, so it will immediately
|
|
|
|
// respond to the request with the username
|
|
|
|
needsUsername = true;
|
|
|
|
} else if (this.serviceIsForeground(GdmUtil.OVIRT_SERVICE_NAME) ||
|
|
|
|
(this.smartcardDetected &&
|
|
|
|
this.serviceIsForeground(GdmUtil.SMARTCARD_SERVICE_NAME))) {
|
|
|
|
// We don't need to know the username if the user preempted the login screen
|
|
|
|
// with a smartcard or with preauthenticated oVirt credentials
|
|
|
|
needsUsername = false;
|
|
|
|
} else {
|
|
|
|
// In all other cases, we should get the username up front.
|
|
|
|
needsUsername = true;
|
|
|
|
}
|
2014-03-07 21:14:05 +00:00
|
|
|
|
2014-03-08 00:09:48 +00:00
|
|
|
if (needsUsername) {
|
|
|
|
this.verificationStatus = VerificationStatus.ASKING_FOR_USERNAME;
|
|
|
|
this.emit('needs-username');
|
|
|
|
} else {
|
|
|
|
this._beginAuthentication();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
gotUserName: function(userName) {
|
2012-07-17 18:54:07 +00:00
|
|
|
this._userName = userName;
|
2014-03-08 00:09:48 +00:00
|
|
|
this._beginAuthentication();
|
|
|
|
},
|
|
|
|
|
|
|
|
_beginAuthentication: function() {
|
|
|
|
this.verificationStatus = VerificationStatus.VERIFYING;
|
|
|
|
this._cancellable = new Gio.Cancellable();
|
2013-07-22 14:59:57 +00:00
|
|
|
this.reauthenticating = false;
|
2012-07-17 18:54:07 +00:00
|
|
|
|
2012-08-26 12:54:02 +00:00
|
|
|
this._checkForFingerprintReader();
|
|
|
|
|
2014-03-08 00:09:48 +00:00
|
|
|
if (this._userName) {
|
2012-07-17 18:54:07 +00:00
|
|
|
// If possible, reauthenticate an already running session,
|
|
|
|
// so any session specific credentials get updated appropriately
|
2014-03-08 00:09:48 +00:00
|
|
|
this._client.open_reauthentication_channel(this._userName, this._cancellable,
|
2012-07-17 18:54:07 +00:00
|
|
|
Lang.bind(this, this._reauthenticationChannelOpened));
|
|
|
|
} else {
|
|
|
|
this._client.get_user_verifier(this._cancellable, Lang.bind(this, this._userVerifierGot));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
clear: function() {
|
2012-08-26 12:54:02 +00:00
|
|
|
if (this._cancellable) {
|
|
|
|
this._cancellable.cancel();
|
|
|
|
this._cancellable = null;
|
|
|
|
}
|
2012-07-17 18:54:07 +00:00
|
|
|
|
|
|
|
if (this._userVerifier) {
|
2014-03-07 23:49:41 +00:00
|
|
|
this._userVerifier.call_cancel_sync(null);
|
2012-07-17 18:54:07 +00:00
|
|
|
this._userVerifier.run_dispose();
|
|
|
|
this._userVerifier = null;
|
|
|
|
}
|
2013-03-18 04:59:56 +00:00
|
|
|
|
|
|
|
this._clearMessageQueue();
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
|
2014-03-07 20:59:21 +00:00
|
|
|
_doAfterPendingMessages: function(func) {
|
|
|
|
if (this.hasPendingMessages) {
|
|
|
|
let signalId = this.connect('no-more-messages', Lang.bind(this, function() {
|
|
|
|
this.disconnect(signalId);
|
|
|
|
func();
|
|
|
|
}));
|
2013-03-18 04:59:56 +00:00
|
|
|
} else {
|
2014-03-07 20:59:21 +00:00
|
|
|
func();
|
2013-03-18 04:59:56 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2014-03-07 20:59:21 +00:00
|
|
|
answerQuery: function(serviceName, answer) {
|
|
|
|
this._doAfterPendingMessages(Lang.bind(this, function() {
|
|
|
|
this._userVerifier.call_answer_query(serviceName, answer, this._cancellable, null);
|
|
|
|
}));
|
|
|
|
},
|
|
|
|
|
2013-03-18 04:59:56 +00:00
|
|
|
_getIntervalForMessage: function(message) {
|
|
|
|
// We probably could be smarter here
|
|
|
|
return message.length * USER_READ_TIME;
|
|
|
|
},
|
|
|
|
|
2014-03-07 22:53:33 +00:00
|
|
|
_finishMessageQueue: function() {
|
2013-03-18 04:59:56 +00:00
|
|
|
if (!this.hasPendingMessages)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this._messageQueue = [];
|
2012-10-29 16:39:00 +00:00
|
|
|
|
2013-03-18 04:59:56 +00:00
|
|
|
this.hasPendingMessages = false;
|
|
|
|
this.emit('no-more-messages');
|
|
|
|
},
|
|
|
|
|
|
|
|
_queueMessageTimeout: function() {
|
|
|
|
if (this._messageQueue.length == 0) {
|
2014-03-07 22:53:33 +00:00
|
|
|
this._finishMessageQueue();
|
2013-03-18 04:59:56 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this._messageQueueTimeoutId != 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
let message = this._messageQueue.shift();
|
2013-08-19 16:00:33 +00:00
|
|
|
|
|
|
|
this.emit('show-message', message.text, message.type);
|
2013-03-18 04:59:56 +00:00
|
|
|
|
|
|
|
this._messageQueueTimeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT,
|
|
|
|
message.interval,
|
|
|
|
Lang.bind(this, function() {
|
|
|
|
this._messageQueueTimeoutId = 0;
|
|
|
|
this._queueMessageTimeout();
|
2013-11-29 00:45:39 +00:00
|
|
|
return GLib.SOURCE_REMOVE;
|
2013-03-18 04:59:56 +00:00
|
|
|
}));
|
|
|
|
},
|
|
|
|
|
2013-08-19 16:00:33 +00:00
|
|
|
_queueMessage: function(message, messageType) {
|
2013-03-18 04:59:56 +00:00
|
|
|
let interval = this._getIntervalForMessage(message);
|
|
|
|
|
|
|
|
this.hasPendingMessages = true;
|
2013-08-19 16:00:33 +00:00
|
|
|
this._messageQueue.push({ text: message, type: messageType, interval: interval });
|
2013-03-18 04:59:56 +00:00
|
|
|
this._queueMessageTimeout();
|
|
|
|
},
|
|
|
|
|
|
|
|
_clearMessageQueue: function() {
|
2014-03-07 22:53:33 +00:00
|
|
|
this._finishMessageQueue();
|
2013-03-18 04:59:56 +00:00
|
|
|
|
|
|
|
if (this._messageQueueTimeoutId != 0) {
|
|
|
|
GLib.source_remove(this._messageQueueTimeoutId);
|
|
|
|
this._messageQueueTimeoutId = 0;
|
|
|
|
}
|
2013-08-19 16:00:33 +00:00
|
|
|
this.emit('show-message', null, MessageType.NONE);
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_checkForFingerprintReader: function() {
|
|
|
|
this._haveFingerprintReader = false;
|
|
|
|
|
2013-07-29 18:18:30 +00:00
|
|
|
if (!this._settings.get_boolean(FINGERPRINT_AUTHENTICATION_KEY)) {
|
|
|
|
this._updateDefaultService();
|
2012-07-17 18:54:07 +00:00
|
|
|
return;
|
2013-07-29 18:18:30 +00:00
|
|
|
}
|
2012-07-17 18:54:07 +00:00
|
|
|
|
|
|
|
this._fprintManager.GetDefaultDeviceRemote(Gio.DBusCallFlags.NONE, this._cancellable, Lang.bind(this,
|
|
|
|
function(device, error) {
|
|
|
|
if (!error && device)
|
|
|
|
this._haveFingerprintReader = true;
|
2013-07-29 18:18:30 +00:00
|
|
|
this._updateDefaultService();
|
2012-07-17 18:54:07 +00:00
|
|
|
}));
|
|
|
|
},
|
|
|
|
|
2014-03-07 20:31:13 +00:00
|
|
|
_oVirtUserAuthenticated: function() {
|
2014-03-07 21:18:40 +00:00
|
|
|
if (this.verificationStatus != GdmUtil.VerificationStatus.VERIFICATION_SUCCEEDED)
|
|
|
|
this._reset();
|
2013-10-10 08:21:47 +00:00
|
|
|
},
|
|
|
|
|
2013-06-27 12:54:19 +00:00
|
|
|
_checkForSmartcard: function() {
|
|
|
|
let smartcardDetected;
|
|
|
|
|
|
|
|
if (!this._settings.get_boolean(SMARTCARD_AUTHENTICATION_KEY))
|
|
|
|
smartcardDetected = false;
|
|
|
|
else if (this.reauthenticating)
|
|
|
|
smartcardDetected = this._smartcardManager.hasInsertedLoginToken();
|
|
|
|
else
|
|
|
|
smartcardDetected = this._smartcardManager.hasInsertedTokens();
|
|
|
|
|
2014-03-07 21:18:40 +00:00
|
|
|
if (this.smartcardDetected == smartcardDetected)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this.smartcardDetected = smartcardDetected;
|
|
|
|
|
|
|
|
// Most of the time we want to reset if the user inserts or removes
|
|
|
|
// a smartcard. Smartcard insertion "preempts" what the user was
|
|
|
|
// doing, and smartcard removal aborts the preemption.
|
|
|
|
// The exceptions are: 1) Don't reset on smartcard insertion if we're already verifying
|
|
|
|
// with a smartcard
|
|
|
|
// 2) Don't reset if we've already succeeded at verification and
|
|
|
|
// the user is getting logged in.
|
|
|
|
if (this._serviceIsDefault(SMARTCARD_SERVICE_NAME) &&
|
|
|
|
this.verificationStatus == VerificationStatus.VERIFYING &&
|
|
|
|
this.smartcardDetected)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (this.verificationStatus != VerificationStatus.VERIFICATION_SUCCEEDED)
|
|
|
|
this._reset();
|
|
|
|
|
|
|
|
this.emit('smartcard-status-changed');
|
2013-06-27 12:54:19 +00:00
|
|
|
},
|
|
|
|
|
2012-09-06 14:40:13 +00:00
|
|
|
_reportInitError: function(where, error) {
|
|
|
|
logError(error, where);
|
|
|
|
|
2013-08-19 16:00:33 +00:00
|
|
|
this._queueMessage(_("Authentication error"), MessageType.ERROR);
|
2012-09-06 14:40:13 +00:00
|
|
|
this._verificationFailed(false);
|
|
|
|
},
|
|
|
|
|
2012-07-17 18:54:07 +00:00
|
|
|
_reauthenticationChannelOpened: function(client, result) {
|
|
|
|
try {
|
|
|
|
this._userVerifier = client.open_reauthentication_channel_finish(result);
|
2012-09-06 14:40:13 +00:00
|
|
|
} catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) {
|
|
|
|
return;
|
|
|
|
} catch(e if e.matches(Gio.DBusError, Gio.DBusError.ACCESS_DENIED) &&
|
|
|
|
!this._reauthOnly) {
|
|
|
|
// Gdm emits org.freedesktop.DBus.Error.AccessDenied when there is
|
|
|
|
// no session to reauthenticate. Fall back to performing verification
|
|
|
|
// from this login session
|
2012-07-17 18:54:07 +00:00
|
|
|
client.get_user_verifier(this._cancellable, Lang.bind(this, this._userVerifierGot));
|
2012-09-06 14:40:13 +00:00
|
|
|
return;
|
|
|
|
} catch(e) {
|
|
|
|
this._reportInitError('Failed to open reauthentication channel', e);
|
|
|
|
return;
|
2012-07-17 18:54:07 +00:00
|
|
|
}
|
2012-09-06 14:40:13 +00:00
|
|
|
|
2013-07-22 14:59:57 +00:00
|
|
|
this.reauthenticating = true;
|
2012-09-06 14:40:13 +00:00
|
|
|
this._connectSignals();
|
|
|
|
this._beginVerification();
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_userVerifierGot: function(client, result) {
|
2012-08-26 12:54:02 +00:00
|
|
|
try {
|
|
|
|
this._userVerifier = client.get_user_verifier_finish(result);
|
2012-09-06 14:40:13 +00:00
|
|
|
} catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) {
|
|
|
|
return;
|
|
|
|
} catch(e) {
|
|
|
|
this._reportInitError('Failed to obtain user verifier', e);
|
2012-08-26 12:54:02 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-07-17 18:54:07 +00:00
|
|
|
this._connectSignals();
|
|
|
|
this._beginVerification();
|
|
|
|
},
|
|
|
|
|
|
|
|
_connectSignals: function() {
|
|
|
|
this._userVerifier.connect('info', Lang.bind(this, this._onInfo));
|
|
|
|
this._userVerifier.connect('problem', Lang.bind(this, this._onProblem));
|
|
|
|
this._userVerifier.connect('info-query', Lang.bind(this, this._onInfoQuery));
|
|
|
|
this._userVerifier.connect('secret-info-query', Lang.bind(this, this._onSecretInfoQuery));
|
|
|
|
this._userVerifier.connect('conversation-stopped', Lang.bind(this, this._onConversationStopped));
|
|
|
|
this._userVerifier.connect('reset', Lang.bind(this, this._onReset));
|
|
|
|
this._userVerifier.connect('verification-complete', Lang.bind(this, this._onVerificationComplete));
|
|
|
|
},
|
|
|
|
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
_getForegroundService: function() {
|
2014-03-07 20:31:13 +00:00
|
|
|
if (this._oVirtCredentialsManager.hasToken())
|
|
|
|
return OVIRT_SERVICE_NAME;
|
|
|
|
if (this.smartcardDetected)
|
|
|
|
return SMARTCARD_SERVICE_NAME;
|
2013-06-27 12:54:19 +00:00
|
|
|
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
return this._defaultService;
|
|
|
|
},
|
|
|
|
|
|
|
|
serviceIsForeground: function(serviceName) {
|
|
|
|
return serviceName == this._getForegroundService();
|
|
|
|
},
|
|
|
|
|
2013-06-27 12:54:19 +00:00
|
|
|
serviceIsDefault: function(serviceName) {
|
|
|
|
return serviceName == this._defaultService;
|
|
|
|
},
|
|
|
|
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
_updateDefaultService: function() {
|
2013-07-29 18:18:30 +00:00
|
|
|
if (this._settings.get_boolean(PASSWORD_AUTHENTICATION_KEY))
|
|
|
|
this._defaultService = PASSWORD_SERVICE_NAME;
|
2013-06-27 12:54:19 +00:00
|
|
|
else if (this.smartcardDetected)
|
|
|
|
this._defaultService = SMARTCARD_SERVICE_NAME;
|
2013-07-29 18:18:30 +00:00
|
|
|
else if (this._haveFingerprintReader)
|
|
|
|
this._defaultService = FINGERPRINT_SERVICE_NAME;
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
},
|
|
|
|
|
2013-08-16 14:29:26 +00:00
|
|
|
_startService: function(serviceName) {
|
2013-08-21 22:05:55 +00:00
|
|
|
if (this._userName) {
|
|
|
|
this._userVerifier.call_begin_verification_for_user(serviceName,
|
|
|
|
this._userName,
|
|
|
|
this._cancellable,
|
|
|
|
Lang.bind(this, function(obj, result) {
|
|
|
|
try {
|
|
|
|
obj.call_begin_verification_for_user_finish(result);
|
|
|
|
} catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) {
|
|
|
|
return;
|
|
|
|
} catch(e) {
|
|
|
|
this._reportInitError('Failed to start verification for user', e);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
} else {
|
|
|
|
this._userVerifier.call_begin_verification(serviceName,
|
|
|
|
this._cancellable,
|
|
|
|
Lang.bind(this, function(obj, result) {
|
|
|
|
try {
|
|
|
|
obj.call_begin_verification_finish(result);
|
|
|
|
} catch(e if e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED)) {
|
|
|
|
return;
|
|
|
|
} catch(e) {
|
|
|
|
this._reportInitError('Failed to start verification', e);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
}
|
2013-08-16 14:29:26 +00:00
|
|
|
},
|
2012-07-17 18:54:07 +00:00
|
|
|
|
2013-08-16 14:29:26 +00:00
|
|
|
_beginVerification: function() {
|
|
|
|
this._startService(this._getForegroundService());
|
|
|
|
|
2013-07-29 18:18:30 +00:00
|
|
|
if (this._userName && this._haveFingerprintReader && !this.serviceIsForeground(FINGERPRINT_SERVICE_NAME))
|
2013-08-16 14:29:26 +00:00
|
|
|
this._startService(FINGERPRINT_SERVICE_NAME);
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_onInfo: function(client, serviceName, info) {
|
2013-07-29 18:23:45 +00:00
|
|
|
if (this.serviceIsForeground(serviceName)) {
|
2013-08-19 16:00:33 +00:00
|
|
|
this._queueMessage(info, MessageType.INFO);
|
2013-07-29 18:23:45 +00:00
|
|
|
} else if (serviceName == FINGERPRINT_SERVICE_NAME &&
|
2012-07-17 18:54:07 +00:00
|
|
|
this._haveFingerprintReader) {
|
2013-07-29 18:23:45 +00:00
|
|
|
// We don't show fingerprint messages directly since it's
|
|
|
|
// not the main auth service. Instead we use the messages
|
|
|
|
// as a cue to display our own message.
|
2012-08-20 00:15:18 +00:00
|
|
|
|
|
|
|
// Translators: this message is shown below the password entry field
|
|
|
|
// to indicate the user can swipe their finger instead
|
2013-08-19 16:00:33 +00:00
|
|
|
this._queueMessage(_("(or swipe finger)"), MessageType.HINT);
|
2012-07-17 18:54:07 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_onProblem: function(client, serviceName, problem) {
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
if (!this.serviceIsForeground(serviceName))
|
2012-07-17 18:54:07 +00:00
|
|
|
return;
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
|
2013-08-19 16:00:33 +00:00
|
|
|
this._queueMessage(problem, MessageType.ERROR);
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_onInfoQuery: function(client, serviceName, question) {
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
if (!this.serviceIsForeground(serviceName))
|
2012-07-17 18:54:07 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
this.emit('ask-question', serviceName, question, '');
|
|
|
|
},
|
|
|
|
|
|
|
|
_onSecretInfoQuery: function(client, serviceName, secretQuestion) {
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
if (!this.serviceIsForeground(serviceName))
|
2012-07-17 18:54:07 +00:00
|
|
|
return;
|
|
|
|
|
2013-10-10 08:21:47 +00:00
|
|
|
if (serviceName == OVIRT_SERVICE_NAME) {
|
|
|
|
// The only question asked by this service is "Token?"
|
|
|
|
this.answerQuery(serviceName, this._oVirtCredentialsManager.getToken());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-07-17 18:54:07 +00:00
|
|
|
this.emit('ask-question', serviceName, secretQuestion, '\u25cf');
|
|
|
|
},
|
|
|
|
|
|
|
|
_onReset: function() {
|
2014-03-07 21:09:32 +00:00
|
|
|
this._reset();
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_onVerificationComplete: function() {
|
2014-03-07 21:14:05 +00:00
|
|
|
this.verificationStatus = VerificationStatus.VERIFICATION_SUCCEEDED;
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
|
2012-09-06 14:40:13 +00:00
|
|
|
_verificationFailed: function(retry) {
|
2012-08-07 14:49:22 +00:00
|
|
|
// For Not Listed / enterprise logins, immediately reset
|
|
|
|
// the dialog
|
|
|
|
// Otherwise, we allow ALLOWED_FAILURES attempts. After that, we
|
|
|
|
// go back to the welcome screen.
|
|
|
|
|
2012-10-29 16:40:55 +00:00
|
|
|
this._failCounter++;
|
2012-09-06 14:40:13 +00:00
|
|
|
let canRetry = retry && this._userName &&
|
|
|
|
this._failCounter < this._settings.get_int(ALLOWED_FAILURES_KEY);
|
|
|
|
|
2014-03-07 21:14:05 +00:00
|
|
|
this.verificationStatus = VerificationStatus.VERIFICATION_FAILED;
|
2014-03-07 23:45:21 +00:00
|
|
|
this.emit('verification-failed');
|
|
|
|
|
2014-03-07 20:59:21 +00:00
|
|
|
this._doAfterPendingMessages(Lang.bind(this, function() {
|
|
|
|
if (canRetry)
|
2014-03-08 00:09:48 +00:00
|
|
|
this._beginAuthentication();
|
2014-03-07 20:59:21 +00:00
|
|
|
else
|
2014-03-07 23:49:41 +00:00
|
|
|
this.clear();
|
2014-03-07 20:59:21 +00:00
|
|
|
}));
|
2012-08-07 14:49:22 +00:00
|
|
|
},
|
|
|
|
|
2012-07-17 18:54:07 +00:00
|
|
|
_onConversationStopped: function(client, serviceName) {
|
2013-10-10 08:21:47 +00:00
|
|
|
// If the login failed with the preauthenticated oVirt credentials
|
|
|
|
// then discard the credentials and revert to default authentication
|
|
|
|
// mechanism.
|
|
|
|
if (this.serviceIsForeground(OVIRT_SERVICE_NAME)) {
|
|
|
|
this._oVirtCredentialsManager.resetToken();
|
|
|
|
this._verificationFailed(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-07-17 18:54:07 +00:00
|
|
|
// if the password service fails, then cancel everything.
|
|
|
|
// But if, e.g., fingerprint fails, still give
|
|
|
|
// password authentication a chance to succeed
|
util: abstract out default auth service in code
Right now, the primary way a user logs in is with
a password. They can also swipe their finger, if their
fingerprint is enrolled, but it's expected the fingerprint
auth service won't ask questions the user has to respond to
by typing. As such, we ignore questions that comes from
anything but the main auth service: gdm-password.
In the future, if a user inserts a smartcard, we'll want
to treat the gdm-smartcard service as the main auth service,
and let any questions from it get to the user.
This commit tries to prepare for that eventuality by storing
the name of the default auth service away in a _defaultService variable
before verification has begun, and then later checking incoming
queries against that service instead of checking against
string 'gdm-password' directly.
Of course, right now, _defaultService is always gdm-password.
https://bugzilla.gnome.org/show_bug.cgi?id=683437
2013-07-28 23:42:26 +00:00
|
|
|
if (this.serviceIsForeground(serviceName)) {
|
2012-09-06 14:40:13 +00:00
|
|
|
this._verificationFailed(true);
|
2012-08-20 01:37:54 +00:00
|
|
|
}
|
2012-07-17 18:54:07 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
Signals.addSignalMethods(ShellUserVerifier.prototype);
|