2013-03-12 16:54:05 -04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 Felix Geyer <debfx@fobos.de>
|
2017-06-09 17:40:36 -04:00
|
|
|
* Copyright (C) 2017 KeePassXC Team <team@keepassxc.org>
|
2013-03-12 16:54:05 -04:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 2 or (at your option)
|
|
|
|
* version 3 of the License.
|
|
|
|
*
|
|
|
|
* 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "PasswordGenerator.h"
|
|
|
|
|
|
|
|
#include "crypto/Random.h"
|
|
|
|
|
2019-11-16 11:51:56 -05:00
|
|
|
const char* PasswordGenerator::DefaultAdditionalChars = "";
|
2018-06-10 22:37:09 -04:00
|
|
|
const char* PasswordGenerator::DefaultExcludedChars = "";
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
PasswordGenerator::PasswordGenerator()
|
|
|
|
: m_length(0)
|
2018-10-28 10:47:24 -04:00
|
|
|
, m_classes(nullptr)
|
|
|
|
, m_flags(nullptr)
|
2019-11-16 11:51:56 -05:00
|
|
|
, m_additional(PasswordGenerator::DefaultAdditionalChars)
|
2018-06-10 22:37:09 -04:00
|
|
|
, m_excluded(PasswordGenerator::DefaultExcludedChars)
|
2014-01-12 17:33:36 -05:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void PasswordGenerator::setLength(int length)
|
|
|
|
{
|
2018-01-22 07:47:20 -05:00
|
|
|
if (length <= 0) {
|
|
|
|
m_length = DefaultLength;
|
|
|
|
return;
|
|
|
|
}
|
2014-01-12 17:33:36 -05:00
|
|
|
m_length = length;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PasswordGenerator::setCharClasses(const CharClasses& classes)
|
|
|
|
{
|
2018-01-22 07:47:20 -05:00
|
|
|
if (classes == 0) {
|
2018-03-31 16:01:30 -04:00
|
|
|
m_classes = DefaultCharset;
|
2018-01-22 07:47:20 -05:00
|
|
|
return;
|
|
|
|
}
|
2014-01-12 17:33:36 -05:00
|
|
|
m_classes = classes;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PasswordGenerator::setFlags(const GeneratorFlags& flags)
|
|
|
|
{
|
|
|
|
m_flags = flags;
|
|
|
|
}
|
2013-03-12 16:54:05 -04:00
|
|
|
|
2019-11-16 11:51:56 -05:00
|
|
|
void PasswordGenerator::setAdditionalChars(const QString& chars)
|
|
|
|
{
|
|
|
|
m_additional = chars;
|
|
|
|
}
|
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
void PasswordGenerator::setExcludedChars(const QString& chars)
|
|
|
|
{
|
|
|
|
m_excluded = chars;
|
|
|
|
}
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
QString PasswordGenerator::generatePassword() const
|
2013-03-12 16:54:05 -04:00
|
|
|
{
|
2014-01-12 17:33:36 -05:00
|
|
|
Q_ASSERT(isValid());
|
2013-03-12 16:54:05 -04:00
|
|
|
|
2016-09-02 13:51:51 -04:00
|
|
|
const QVector<PasswordGroup> groups = passwordGroups();
|
2013-03-12 16:54:05 -04:00
|
|
|
|
|
|
|
QVector<QChar> passwordChars;
|
2016-09-02 13:51:51 -04:00
|
|
|
for (const PasswordGroup& group : groups) {
|
|
|
|
for (QChar ch : group) {
|
2013-03-12 16:54:05 -04:00
|
|
|
passwordChars.append(ch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QString password;
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
if (m_flags & CharFromEveryGroup) {
|
2018-09-29 13:00:47 -04:00
|
|
|
for (const auto& group : groups) {
|
|
|
|
int pos = randomGen()->randomUInt(static_cast<quint32>(group.size()));
|
2013-03-12 16:54:05 -04:00
|
|
|
|
2018-09-29 13:00:47 -04:00
|
|
|
password.append(group[pos]);
|
2013-03-12 16:54:05 -04:00
|
|
|
}
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
for (int i = groups.size(); i < m_length; i++) {
|
2018-09-29 13:00:47 -04:00
|
|
|
int pos = randomGen()->randomUInt(static_cast<quint32>(passwordChars.size()));
|
2013-03-12 16:54:05 -04:00
|
|
|
|
|
|
|
password.append(passwordChars[pos]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// shuffle chars
|
|
|
|
for (int i = (password.size() - 1); i >= 1; i--) {
|
2018-09-29 13:00:47 -04:00
|
|
|
int j = randomGen()->randomUInt(static_cast<quint32>(i + 1));
|
2013-03-12 16:54:05 -04:00
|
|
|
|
|
|
|
QChar tmp = password[i];
|
|
|
|
password[i] = password[j];
|
|
|
|
password[j] = tmp;
|
|
|
|
}
|
2018-03-31 16:01:30 -04:00
|
|
|
} else {
|
2014-01-12 17:33:36 -05:00
|
|
|
for (int i = 0; i < m_length; i++) {
|
2018-09-29 13:00:47 -04:00
|
|
|
int pos = randomGen()->randomUInt(static_cast<quint32>(passwordChars.size()));
|
2013-03-12 16:54:05 -04:00
|
|
|
|
|
|
|
password.append(passwordChars[pos]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return password;
|
|
|
|
}
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
bool PasswordGenerator::isValid() const
|
2013-03-12 16:54:05 -04:00
|
|
|
{
|
2019-11-16 11:51:56 -05:00
|
|
|
if (m_classes == 0 && m_additional.isEmpty()) {
|
2013-03-12 16:54:05 -04:00
|
|
|
return false;
|
2018-03-31 16:01:30 -04:00
|
|
|
} else if (m_length == 0) {
|
2013-03-12 16:54:05 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
if ((m_flags & CharFromEveryGroup) && (m_length < numCharClasses())) {
|
2013-03-12 16:54:05 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-09-29 13:00:47 -04:00
|
|
|
return !passwordGroups().isEmpty();
|
2013-03-12 16:54:05 -04:00
|
|
|
}
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
QVector<PasswordGroup> PasswordGenerator::passwordGroups() const
|
2013-03-12 16:54:05 -04:00
|
|
|
{
|
|
|
|
QVector<PasswordGroup> passwordGroups;
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
if (m_classes & LowerLetters) {
|
2013-03-12 16:54:05 -04:00
|
|
|
PasswordGroup group;
|
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
for (int i = 97; i <= (97 + 25); i++) {
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
if ((m_flags & ExcludeLookAlike) && (i == 108)) { // "l"
|
2013-03-12 16:54:05 -04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
group.append(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
2014-01-12 17:33:36 -05:00
|
|
|
if (m_classes & UpperLetters) {
|
2013-03-12 16:54:05 -04:00
|
|
|
PasswordGroup group;
|
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
for (int i = 65; i <= (65 + 25); i++) {
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
if ((m_flags & ExcludeLookAlike) && (i == 73 || i == 79)) { // "I" and "O"
|
2013-03-12 16:54:05 -04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
group.append(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
2014-01-12 17:33:36 -05:00
|
|
|
if (m_classes & Numbers) {
|
2013-03-12 16:54:05 -04:00
|
|
|
PasswordGroup group;
|
|
|
|
|
|
|
|
for (int i = 48; i < (48 + 10); i++) {
|
2014-01-12 17:33:36 -05:00
|
|
|
if ((m_flags & ExcludeLookAlike) && (i == 48 || i == 49)) { // "0" and "1"
|
2013-03-12 16:54:05 -04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
group.append(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
2018-06-10 22:37:09 -04:00
|
|
|
if (m_classes & Braces) {
|
2013-03-12 16:54:05 -04:00
|
|
|
PasswordGroup group;
|
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
// ()[]{}
|
|
|
|
group.append(40);
|
|
|
|
group.append(41);
|
|
|
|
group.append(91);
|
|
|
|
group.append(93);
|
|
|
|
group.append(123);
|
|
|
|
group.append(125);
|
2013-03-12 16:54:05 -04:00
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
|
|
|
if (m_classes & Punctuation) {
|
|
|
|
PasswordGroup group;
|
2013-03-12 16:54:05 -04:00
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
// .,:;
|
|
|
|
group.append(44);
|
|
|
|
group.append(46);
|
|
|
|
group.append(58);
|
|
|
|
group.append(59);
|
|
|
|
|
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
|
|
|
if (m_classes & Quotes) {
|
|
|
|
PasswordGroup group;
|
|
|
|
|
|
|
|
// "'
|
|
|
|
group.append(34);
|
|
|
|
group.append(39);
|
|
|
|
|
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
|
|
|
if (m_classes & Dashes) {
|
|
|
|
PasswordGroup group;
|
|
|
|
|
|
|
|
// -/\_|
|
|
|
|
group.append(45);
|
|
|
|
group.append(47);
|
|
|
|
group.append(92);
|
|
|
|
group.append(95);
|
|
|
|
if (!(m_flags & ExcludeLookAlike)) {
|
|
|
|
group.append(124); // "|"
|
2013-03-12 16:54:05 -04:00
|
|
|
}
|
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
|
|
|
if (m_classes & Math) {
|
|
|
|
PasswordGroup group;
|
|
|
|
|
|
|
|
// !*+-<=>?
|
|
|
|
group.append(33);
|
|
|
|
group.append(42);
|
|
|
|
group.append(43);
|
|
|
|
group.append(60);
|
|
|
|
group.append(61);
|
|
|
|
group.append(62);
|
|
|
|
group.append(63);
|
2013-03-12 16:54:05 -04:00
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
|
|
|
if (m_classes & Logograms) {
|
|
|
|
PasswordGroup group;
|
|
|
|
|
|
|
|
// #$%&
|
|
|
|
for (int i = 35; i <= 38; i++) {
|
2013-03-12 16:54:05 -04:00
|
|
|
group.append(i);
|
|
|
|
}
|
2018-06-10 22:37:09 -04:00
|
|
|
// @^`~
|
|
|
|
group.append(64);
|
|
|
|
group.append(94);
|
|
|
|
group.append(96);
|
|
|
|
group.append(126);
|
2013-03-12 16:54:05 -04:00
|
|
|
|
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
2017-02-25 17:41:37 -05:00
|
|
|
if (m_classes & EASCII) {
|
|
|
|
PasswordGroup group;
|
|
|
|
|
2017-04-30 19:18:42 -04:00
|
|
|
// [U+0080, U+009F] are C1 control characters,
|
|
|
|
// U+00A0 is non-breaking space
|
|
|
|
for (int i = 161; i <= 172; i++) {
|
2017-02-25 17:41:37 -05:00
|
|
|
group.append(i);
|
|
|
|
}
|
2017-04-30 19:18:42 -04:00
|
|
|
// U+00AD is soft hyphen (format character)
|
|
|
|
for (int i = 174; i <= 255; i++) {
|
2017-02-25 17:41:37 -05:00
|
|
|
if ((m_flags & ExcludeLookAlike) && (i == 249)) { // "﹒"
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
group.append(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
2019-11-16 11:51:56 -05:00
|
|
|
if (!m_additional.isEmpty()) {
|
|
|
|
PasswordGroup group;
|
|
|
|
|
|
|
|
for (auto ch : m_additional) {
|
|
|
|
group.append(ch);
|
|
|
|
}
|
|
|
|
|
|
|
|
passwordGroups.append(group);
|
|
|
|
}
|
2013-03-12 16:54:05 -04:00
|
|
|
|
2018-06-10 22:37:09 -04:00
|
|
|
// Loop over character groups and remove excluded characters from them;
|
|
|
|
// remove empty groups
|
|
|
|
int i = 0;
|
|
|
|
while (i != passwordGroups.size()) {
|
|
|
|
PasswordGroup group = passwordGroups[i];
|
|
|
|
|
|
|
|
for (QChar ch : m_excluded) {
|
|
|
|
int j = group.indexOf(ch);
|
|
|
|
while (j != -1) {
|
|
|
|
group.remove(j);
|
|
|
|
j = group.indexOf(ch);
|
|
|
|
}
|
|
|
|
}
|
2018-09-29 13:00:47 -04:00
|
|
|
if (!group.isEmpty()) {
|
2018-06-10 22:37:09 -04:00
|
|
|
passwordGroups.replace(i, group);
|
2018-09-29 13:00:47 -04:00
|
|
|
++i;
|
2018-06-10 22:37:09 -04:00
|
|
|
} else {
|
|
|
|
passwordGroups.remove(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-12 16:54:05 -04:00
|
|
|
return passwordGroups;
|
|
|
|
}
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
int PasswordGenerator::numCharClasses() const
|
2013-03-12 16:54:05 -04:00
|
|
|
{
|
|
|
|
int numClasses = 0;
|
|
|
|
|
2014-01-12 17:33:36 -05:00
|
|
|
if (m_classes & LowerLetters) {
|
2013-03-12 16:54:05 -04:00
|
|
|
numClasses++;
|
|
|
|
}
|
2014-01-12 17:33:36 -05:00
|
|
|
if (m_classes & UpperLetters) {
|
2013-03-12 16:54:05 -04:00
|
|
|
numClasses++;
|
|
|
|
}
|
2014-01-12 17:33:36 -05:00
|
|
|
if (m_classes & Numbers) {
|
2013-03-12 16:54:05 -04:00
|
|
|
numClasses++;
|
|
|
|
}
|
2018-06-10 22:37:09 -04:00
|
|
|
if (m_classes & Braces) {
|
|
|
|
numClasses++;
|
|
|
|
}
|
|
|
|
if (m_classes & Punctuation) {
|
|
|
|
numClasses++;
|
|
|
|
}
|
|
|
|
if (m_classes & Quotes) {
|
|
|
|
numClasses++;
|
|
|
|
}
|
|
|
|
if (m_classes & Dashes) {
|
|
|
|
numClasses++;
|
|
|
|
}
|
|
|
|
if (m_classes & Math) {
|
|
|
|
numClasses++;
|
|
|
|
}
|
|
|
|
if (m_classes & Logograms) {
|
2013-03-12 16:54:05 -04:00
|
|
|
numClasses++;
|
|
|
|
}
|
2017-04-28 15:36:43 -04:00
|
|
|
if (m_classes & EASCII) {
|
|
|
|
numClasses++;
|
|
|
|
}
|
2013-03-12 16:54:05 -04:00
|
|
|
|
|
|
|
return numClasses;
|
|
|
|
}
|