-
Benoit Lavenier authoredBenoit Lavenier authored
login-controllers.js 15.48 KiB
angular.module('cesium.login.controllers', ['cesium.services'])
.controller('LoginModalCtrl', LoginModalController)
.controller('AuthCtrl', AuthController)
;
function LoginModalController($scope, $timeout, $q, $ionicPopover, CryptoUtils,
UIUtils, BMA, Modals, csSettings, Device, parameters) {
'ngInject';
parameters = parameters || {};
$scope.computing = false;
$scope.pubkey = null;
$scope.formData = {};
$scope.showPubkey = false;
$scope.showComputePubkeyButton = false;
$scope.autoComputePubkey = false;
$scope.pubkeyPattern = '^(:?{0}|{1})$'.format(BMA.constants.regexp.PUBKEY, BMA.constants.regexp.PUBKEY_WITH_CHECKSUM);
$scope.isAuth = parameters.auth;
$scope.showMethods = angular.isDefined(parameters.showMethods) ? parameters.showMethods : true;
$scope.expectedPubkey = parameters.expectedPubkey;
$scope.scryptParamsValues = _.keys(CryptoUtils.constants.SCRYPT_PARAMS)
.reduce(function(res, key) {
return res.concat({id: key, label: 'LOGIN.SCRYPT.' + key, params: CryptoUtils.constants.SCRYPT_PARAMS[key]});
}, [{id: 'USER', label: 'LOGIN.SCRYPT.USER', params: {}}]);
// modal init
$scope.init = function() {
// Should auto-compute pubkey ?
$scope.autoComputePubkey = ionic.Platform.grade.toLowerCase()==='a' &&
!UIUtils.screen.isSmall();
// Init remember me
$scope.formData.rememberMe = csSettings.data.rememberMe;
// Init keep auth, from idle time
$scope.formData.keepAuthIdle = csSettings.data.keepAuthIdle;
$scope.formData.keepAuth = ($scope.formData.keepAuthIdle == csSettings.constants.KEEP_AUTH_IDLE_SESSION);
// Init method
var method = parameters.method || csSettings.data.login && csSettings.data.login.method || 'SCRYPT_DEFAULT';
var params = csSettings.data.login && csSettings.data.login.params;
if ($scope.isAuth && method == 'PUBKEY') {
method = 'SCRYPT_DEFAULT'; // PUBKEY not enable if auth need
}
$scope.changeMethod(method, params);
};
// modal enter
$scope.enter = function() {
UIUtils.loading.hide();
// Ink effect
UIUtils.ink({selector: '.modal-login .ink'});
};
$scope.$on('modal.shown', $scope.enter);
// modal leave
$scope.leave = function() {
$scope.formData = {};
$scope.computing = false;
$scope.pubkey = null;
$scope.methods = [];
};
$scope.$on('modal.hide', $scope.leave);
// Login form submit
$scope.doLogin = function() {
if(!$scope.form.$valid) {
return;
}
// removeIf(no-device)
Device.keyboard.close();
// endRemoveIf(no-device)
var method = $scope.formData.method;
var keepAuthIdle = $scope.formData.keepAuthIdle;
var promise;
// Scrypt
if (method === 'SCRYPT_DEFAULT' || method === 'SCRYPT_ADVANCED') {
if (!$scope.formData.username || !$scope.formData.password) return;
var scryptPrams = $scope.formData.scrypt && $scope.formData.scrypt.params;
UIUtils.loading.show();
promise = CryptoUtils.scryptKeypair($scope.formData.username, $scope.formData.password, scryptPrams)
.then(function(keypair) {
if (!keypair) return UIUtils.loading.hide(10);
var pubkey = CryptoUtils.util.encode_base58(keypair.signPk);
// Check pubkey
if (parameters.expectedPubkey && parameters.expectedPubkey != pubkey) {
$scope.pubkey = pubkey;
$scope.showPubkey = true;
$scope.pubkeyError = true;
return UIUtils.loading.hide(10);
}
$scope.pubkeyError = false;
return {
pubkey: pubkey,
keypair: keypair,
params: ($scope.formData.scrypt && $scope.formData.scrypt.id != 'SCRYPT_DEFAULT') ? scryptPrams : undefined
};
})
.catch(UIUtils.onError('ERROR.CRYPTO_UNKNOWN_ERROR'));
}
// File
else if (method === 'FILE') {
if (!$scope.formData.file || !$scope.formData.file.valid || !$scope.formData.file.pubkey) return;
// If checkbox keep auth checked: set idle time to session
keepAuthIdle = ($scope.formData.keepAuth && csSettings.constants.KEEP_AUTH_IDLE_SESSION) || keepAuthIdle;
promise =
UIUtils.loading.show()
.then(function() {
return $scope.readKeyFile($scope.formData.file, {
withSecret: ($scope.isAuth || $scope.formData.keepAuth)
});
})
.then(function(keypair) {
if (!keypair) return UIUtils.loading.hide(10);
var pubkey = CryptoUtils.util.encode_base58(keypair.signPk);
// Check pubkey
if (parameters.expectedPubkey && parameters.expectedPubkey != pubkey) {
$scope.formData.file.valid = false;
return UIUtils.loading.hide(10);
}
// TODO: if WIF, force redirection to a transfer modal, using a temporary wallet:
// var wallet = csWallet.instance('WIF');
// return wallet.login(...)
$scope.pubkeyError = false;
return {
pubkey: pubkey,
keypair: keypair
};
})
.catch(UIUtils.onError('ERROR.AUTH_FILE_ERROR'));
}
// Pubkey
else if (method === 'PUBKEY') {
if (!$scope.formData.pubkey) return;
var matches = BMA.regexp.PUBKEY_WITH_CHECKSUM.exec($scope.formData.pubkey);
// Check checksum
if (matches) {
var pubkey = matches[1];
var checksum = matches[2];
var expectedChecksum = CryptoUtils.pkChecksum(pubkey);
if (checksum != expectedChecksum) {
$scope.form.pubkey.$error = {checksum: true};
}
else {
promise = $q.when({
pubkey: pubkey
});
}
}
else {
promise = $q.when({
pubkey: $scope.formData.pubkey.trim()
});
}
}
if (!promise) {
console.warn('[login] unknown method: ', method);
return;
}
return promise.then(function(res) {
if (!res) return;
// Update settings (if need)
var rememberMeChanged = !angular.equals(csSettings.data.rememberMe, $scope.formData.rememberMe);
var keepAuthIdleChanged = !angular.equals(csSettings.data.keepAuthIdle, keepAuthIdle);
var methodChanged = !angular.equals(csSettings.data.login && csSettings.data.login.method, method);
var paramsChanged = !angular.equals(csSettings.data.login && csSettings.data.login.params, res.params);
if (rememberMeChanged || keepAuthIdleChanged || methodChanged || paramsChanged) {
csSettings.data.rememberMe = $scope.formData.rememberMe;
csSettings.data.keepAuthIdle = keepAuthIdle;
csSettings.data.useLocalStorage = csSettings.data.rememberMe ? true : csSettings.data.useLocalStorage;
csSettings.data.login = csSettings.data.login || {};
csSettings.data.login.method = method;
csSettings.data.login.params = res.params;
$timeout(csSettings.store, 500);
}
if (parameters.success) {
parameters.success($scope.formData);
}
// hide loading
if (parameters.silent) {
UIUtils.loading.hide();
}
// Return result then close
return $scope.closeModal(res);
});
};
$scope.onScryptFormChanged = function() {
if ($scope.computing) return; // avoid multiple call
$scope.pubkey = null;
$scope.pubkeyError = false;
$scope.showPubkey = !!$scope.formData.username && !!$scope.formData.password;
if ($scope.autoComputePubkey && $scope.showPubkey) {
$scope.computePubkey();
$scope.showComputePubkeyButton = false;
}
else {
$scope.showComputePubkeyButton = !$scope.autoComputePubkey && $scope.showPubkey;
}
};
$scope.$watch('formData.username + formData.password', $scope.onScryptFormChanged, true);
$scope.computePubkey = function() {
$scope.showComputePubkeyButton = false;
$scope.computing = true;
$scope.pubkey = null;
return $timeout(function() {
var salt = $scope.formData.username;
var pwd = $scope.formData.password;
var scryptPrams = $scope.formData.scrypt && $scope.formData.scrypt.params;
return CryptoUtils.scryptSignPk(salt, pwd, scryptPrams)
.then(function (signPk) {
// If model has changed before the response, then retry
if (salt !== $scope.formData.username || pwd !== $scope.formData.password) {
return $scope.computePubkey();
}
$scope.pubkey = CryptoUtils.util.encode_base58(signPk);
if ($scope.expectedPubkey && $scope.expectedPubkey != $scope.pubkey) {
$scope.pubkeyError = true;
}
$scope.computing = false;
}
)
.catch(function (err) {
UIUtils.onError('ERROR.CRYPTO_UNKNOWN_ERROR')(err);
$scope.computing = false;
$scope.onScryptFormChanged();
});
}, 100);
};
$scope.showJoinModal = function() {
$scope.closeModal();
$timeout(function() {
Modals.showJoin();
}, 300);
};
$scope.showAccountSecurityModal = function() {
$scope.closeModal();
$timeout(function() {
Modals.showAccountSecurity();
}, 300);
};
$scope.showHelpModal = function(parameters) {
return Modals.showHelp(parameters);
};
$scope.changeMethod = function(method, params){
$scope.hideMethodsPopover();
if (method == $scope.formData.method) return; // same method
console.debug("[login] method is: " + method);
$scope.formData.method = method;
if ($scope.form) {
// hide form's fields errors on the form
delete $scope.form.$submitted;
}
// Scrypt (advanced or not)
if (method == 'SCRYPT_DEFAULT' || method == 'SCRYPT_ADVANCED') {
// Search scrypt object
var scrypt;
if (params) {
scrypt = _.find($scope.scryptParamsValues, function(item){
return item.params && angular.equals(item.params, params);
});
if (!scrypt) {
scrypt = _.findWhere($scope.scryptParamsValues, {id: 'USER'}) || {};
scrypt.params = params;
}
}
else {
scrypt = _.findWhere($scope.scryptParamsValues, {id: 'DEFAULT'});
}
$scope.changeScrypt(scrypt);
$scope.autoComputePubkey = $scope.autoComputePubkey && (method == 'SCRYPT_DEFAULT');
}
else {
$scope.formData.username = null;
$scope.formData.password = null;
$scope.formData.pubkey = null;
$scope.formData.computing = false;
}
};
$scope.changeScrypt = function(scrypt) {
// Protect params against changes
$scope.formData.scrypt = angular.copy(scrypt||{});
$scope.onScryptFormChanged();
};
$scope.readKeyFile = function(file, options) {
options = options || {};
options.password = options.password || $scope.formData.file.password || function() {
$scope.formData.file.password = undefined;
return Modals.showPassword({
title: 'ACCOUNT.SECURITY.KEYFILE.PASSWORD_POPUP.TITLE',
subTitle: 'ACCOUNT.SECURITY.KEYFILE.PASSWORD_POPUP.HELP'
})
.then(function (password) {
// Remember password (for validation)
$scope.formData.file.password = password;
return password;
});
};
return CryptoUtils.readKeyFile($scope.formData.file, options)
.catch(function(err) {
if (err && err == 'CANCELLED') return;
if (err && err == 'BAD_PASSWORD') {
return $scope.readKeyFile($scope.formData.file, options); // Loop (ask the password again)
}
throw err;
});
};
$scope.fileChanged = function(event) {
$scope.validatingFile = true;
$scope.formData.file = event && event.target && event.target.files && event.target.files.length && event.target.files[0];
if (!$scope.formData.file) {
$scope.validatingFile = false;
return;
}
$timeout(function() {
console.debug("[login] key file changed: ", $scope.formData.file);
$scope.validatingFile = true;
return $scope.readKeyFile($scope.formData.file, {withSecret: false, password: $scope.formData.file.password})
.then(function(keypair) {
if (!keypair || !keypair.signPk) {
$scope.formData.file.valid = false;
$scope.formData.file.pubkey = undefined;
}
else {
$scope.formData.file.pubkey = CryptoUtils.util.encode_base58(keypair.signPk);
$scope.formData.file.valid = !$scope.expectedPubkey || $scope.expectedPubkey == $scope.formData.file.pubkey;
$scope.validatingFile = false;
}
})
.catch(function(err) {
if (err && err == 'CANCELLED') {
$scope.removeKeyFile();
return;
}
$scope.validatingFile = false;
$scope.formData.file.valid = false;
$scope.formData.file.pubkey = undefined;
UIUtils.onError('ERROR.AUTH_FILE_ERROR')(err);
});
});
};
/**
* On file drop
*/
$scope.onKeyFileDrop = function(file) {
if (!file || !file.fileData) return;
$scope.formData.file = {
name: file.fileData.name,
size: file.fileData.size,
content: file.fileContent
};
$scope.validatingFile = true;
$timeout(function() {
return $scope.readKeyFile($scope.formData.file, {withSecret: false})
.then(function (keypair) {
if (!keypair || !keypair.signPk) {
$scope.formData.file.valid = false;
$scope.formData.file.pubkey = undefined;
}
else {
$scope.formData.file.pubkey = CryptoUtils.util.encode_base58(keypair.signPk);
$scope.formData.file.valid = !$scope.expectedPubkey || $scope.expectedPubkey == $scope.formData.file.pubkey;
$scope.validatingFile = false;
}
})
.catch(function (err) {
$scope.validatingFile = false;
$scope.formData.file.valid = false;
$scope.formData.file.pubkey = undefined;
UIUtils.onError('ERROR.AUTH_FILE_ERROR')(err);
});
});
};
$scope.removeKeyFile = function() {
$scope.formData.file = undefined;
};
/* -- modals -- */
$scope.showWotLookupModal = function(searchText) {
return Modals.showWotLookup({q: searchText})
.then(function(res){
if (res && res.pubkey) {
$scope.formData.pubkey = res.pubkey;
}
});
};
/* -- popover -- */
$scope.showMethodsPopover = function(event) {
if (event.defaultPrevented) return;
if (!$scope.methodsPopover) {
$ionicPopover.fromTemplateUrl('templates/login/popover_methods.html', {
scope: $scope
}).then(function(popover) {
$scope.methodsPopover = popover;
//Cleanup the popover when we're done with it!
$scope.$on('$destroy', function() {
$scope.methodsPopover.remove();
});
$scope.methodsPopover.show(event)
.then(function() {
UIUtils.ink({selector: '.popover-login-methods .item'});
});
});
}
else {
$scope.methodsPopover.show(event);
}
};
$scope.hideMethodsPopover = function() {
if ($scope.methodsPopover) {
$scope.methodsPopover.hide();
}
};
// Default action
$scope.init();
// TODO : for DEV only
/*$timeout(function() {
$scope.formData = {
username: 'benoit.lavenier@e-is.pro',
password: ''
};
//$scope.form = {$valid:true};
}, 900);*/
}
function AuthController($scope, $controller){
// Initialize the super class and extend it.
angular.extend(this, $controller('LoginModalCtrl', {$scope: $scope, parameters: {auth: true}}));
$scope.setForm = function(form) {
$scope.form = form;
};
}