/**
* Nextcloud - passman
*
* @copyright Copyright (c) 2016, Sander Brand (brantje@gmail.com)
* @copyright Copyright (c) 2016, Marcos Zuriaga Miguel (wolfi@wolfi.es)
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see .
*
*/
(function () {
'use strict';
/**
* @ngdoc directive
* @name passmanApp.directive:passwordGen
* @description
* # passwordGen
*/
angular.module('passmanApp')
.directive('passwordGen', function ($timeout) {
/* jshint ignore:start */
function Arcfour () {
this.j = this.i = 0, this.S = []
}
function ARC4init (r) {
var t, n, e;
for (t = 0; 256 > t; ++t)this.S[t] = t
for (t = n = 0; 256 > t; ++t)n = n + this.S[t] + r[t % r.length] & 255, e = this.S[t], this.S[t] = this.S[n], this.S[n] = e
this.j = this.i = 0
}
function ARC4next () {
var r;
return this.i = this.i + 1 & 255, this.j = this.j + this.S[this.i] & 255, r = this.S[this.i], this.S[this.i] = this.S[this.j], this.S[this.j] = r, this.S[r + this.S[this.i] & 255]
}
function prng_newstate () {
return new Arcfour
}
function generatePassword (r, t, n, e, o, i, p, g) {
var _, a, s, f, d, h, u, l, c, v, w, y, m;
if (void 0 === r && (r = 8 + get_random(0, 1)), r > 256 && (r = 256, document.getElementById("length").value = 256), i > 256 && (i = 256), void 0 === t && (t = !0), void 0 === n && (n = !0), void 0 === e && (e = !0), void 0 === o && (o = !1), void 0 === i && (i = 0), void 0 === p && (p = !1), void 0 === g && (g = !0), _ = 0, a = 0, s = 0, g && (_ = a = s = 1), f = [], n && _ > 0)for (d = 0; _ > d; d++)f[f.length] = "L"
if (t && a > 0)for (d = 0; a > d; d++)f[f.length] = "U"
if (e && i > 0)for (d = 0; i > d; d++)f[f.length] = "D"
if (o && s > 0)for (d = 0; s > d; d++)f[f.length] = "S"
for (; f.length < r;)f[f.length] = "A"
for (f.sort(function () {
return 2 * get_random(0, 1) - 1
}), h = "", u = "abcdefghjkmnpqrstuvwxyz", p || (u += "ilo"), n && (h += u), l = "ABCDEFGHJKMNPQRSTUVWXYZ", p || (l += "ILO"), t && (h += l), c = "23456789", p || (c += "10"), e && (h += c), v = "!@#$%^&*", o && (h += v), w = "", y = 0; r > y; y++) {
switch (f[y]) {
case"L":
m = u;
break;
case"U":
m = l;
break;
case"D":
m = c;
break;
case"S":
m = v;
break;
case"A":
m = h
}
d = get_random(0, m.length - 1), w += m.charAt(d)
}
return w
}
function rng_seed_int (r) {
rng_pool[rng_pptr++] ^= 255 & r, rng_pool[rng_pptr++] ^= r >> 8 & 255, rng_pool[rng_pptr++] ^= r >> 16 & 255, rng_pool[rng_pptr++] ^= r >> 24 & 255, rng_pptr >= rng_psize && (rng_pptr -= rng_psize)
}
function rng_seed_time () {
rng_seed_int((new Date).getTime())
}
function rng_get_byte () {
if (null == rng_state) {
for (rng_seed_time(), rng_state = prng_newstate(), rng_state.init(rng_pool), rng_pptr = 0; rng_pptr < rng_pool.length; ++rng_pptr)rng_pool[rng_pptr] = 0
rng_pptr = 0
}
return rng_state.next()
}
function rng_get_bytes (r) {
var t;
for (t = 0; t < r.length; ++t)r[t] = rng_get_byte()
}
function SecureRandom () {
}
function get_random (r, t) {
var n, e, o, i = t - r + 1
for (rng_seed_time(), n = [], e = 0; 4 > e; e++)n[e] = 0
for (rng_get_bytes(n), o = 0, e = 0; 4 > e; e++)o *= 256, o += n[e]
return o %= i, o += r
}
function get_random_password (r, t) {
var n;
var pwlen, newpw;
for ("number" != typeof r && (r = 12), "number" != typeof t && (t = 16), r > t && (n = r, r = t, t = n), pwlen = get_random(r, t), newpw = ""; newpw.length < pwlen;)newpw += String.fromCharCode(get_random(32, 127))
return newpw
}
var rng_psize, rng_state, rng_pool, rng_pptr, t, z, crypt_obj, num, buf, i
if (Arcfour.prototype.init = ARC4init, Arcfour.prototype.next = ARC4next, rng_psize = 256, null == rng_pool) {
if (rng_pool = [], rng_pptr = 0, "undefined" != typeof navigator && "Netscape" == navigator.appName && navigator.appVersion < "5" && "undefined" != typeof window && window.crypto)for (z = window.crypto.random(32), t = 0; t < z.length; ++t)rng_pool[rng_pptr++] = 255 & z.charCodeAt(t)
try {
if (crypt_obj = null, "undefined" != typeof window && void 0 !== window.crypto ? crypt_obj = window.crypto : "undefined" != typeof window && void 0 !== window.msCrypto && (crypt_obj = window.msCrypto), void 0 !== crypt_obj && "function" == typeof crypt_obj.getRandomValues && rng_psize > rng_pptr)for (num = Math.floor((rng_psize - rng_pptr) / 2) + 1, buf = new Uint16Array(num), crypt_obj.getRandomValues(buf), i = 0; i < buf.length; i++)t = buf[i], rng_pool[rng_pptr++] = t >>> 8, rng_pool[rng_pptr++] = 255 & t
} catch (e) {
}
for (; rng_psize > rng_pptr;)t = Math.floor(65536 * Math.random()), rng_pool[rng_pptr++] = t >>> 8, rng_pool[rng_pptr++] = 255 & t
rng_pptr = 0, rng_seed_time()
}
SecureRandom.prototype.nextBytes = rng_get_bytes;
/* jshint ignore:end */
return {
scope: {
model: "=ngModel",
length: "@",
placeholder: "@",
settings: '=settings',
callback: '&callback'
},
restrict: "E",
replace: "true",
template: "" +
"
",
link: function (scope) {
scope.callback = scope.callback();
scope.$watch("model", function () {
scope.password = scope.model;
});
scope.passwordVisible = false;
scope.toggleVisibility = function () {
scope.passwordVisible = !scope.passwordVisible;
};
scope.passwordNotNull = false;
scope.$watch("settings", function () {
if (scope.settings) {
if (!scope.password && scope.settings.generateOnCreate) {
scope.generatePasswordStart();
}
}
});
scope.$watch("password", function () {
scope.model = scope.password;
scope.password_repeat = scope.model;
});
//
scope.onSuccess = function (e) {
//@TODO move OC.Notification to a service
OC.Notification.showTemporary('Password copied to clipboard!');
e.clearSelection();
};
scope.onError = function (e) {
OC.Notification.showTemporary('Press Ctrl+C to copy!');
};
scope.progressDivShow = false;
scope.generatePasswordStart = function () {
scope.progressDivShow = true;
scope.progressValue = 0;
scope.progressWidth = {"width": scope.progressValue + "%"};
scope.generatePasswordProgress();
};
scope.generatePasswordProgress = function () {
$timeout(function () {
if (scope.progressValue < 100) {
scope.password = scope._generatePassword(scope.settings);
scope.progressValue += 10;
scope.progressWidth = {"width": scope.progressValue + "%"};
scope.disabled = true;
scope.generatePasswordProgress();
} else {
scope.disabled = false;
if (scope.callback) {
scope.callback(scope.password);
}
}
}, 10);
};
scope._generatePassword = function (settings) {
var _settings = {
'length': 12,
'useUppercase': true,
'useLowercase': true,
'useDigits': true,
'useSpecialChars': true,
'minimumDigitCount': 3,
'avoidAmbiguousCharacters': false,
'requireEveryCharType': true
};
settings = angular.merge(_settings, settings);
/* jshint ignore:start */
var password = generatePassword(settings['length'],
settings.useUppercase,
settings.useLowercase,
settings.useDigits,
settings.useSpecialChars,
settings.minimumDigitCount,
settings.avoidAmbiguousCharacters,
settings.requireEveryCharType);
/* jshint ignore:end */
return password;
};
}
};
});
}());