/* * Copyright (C) 2013 Felix Geyer * Copyright (C) 2017 KeePassXC Team * * 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 . */ #include "PasswordGenerator.h" #include "crypto/Random.h" const char* PasswordGenerator::DefaultAdditionalChars = ""; const char* PasswordGenerator::DefaultExcludedChars = ""; PasswordGenerator::PasswordGenerator() : m_length(0) , m_classes(nullptr) , m_flags(nullptr) , m_additional(PasswordGenerator::DefaultAdditionalChars) , m_excluded(PasswordGenerator::DefaultExcludedChars) { } void PasswordGenerator::setLength(int length) { if (length <= 0) { m_length = DefaultLength; return; } m_length = length; } void PasswordGenerator::setCharClasses(const CharClasses& classes) { if (classes == 0) { m_classes = DefaultCharset; return; } m_classes = classes; } void PasswordGenerator::setFlags(const GeneratorFlags& flags) { m_flags = flags; } void PasswordGenerator::setAdditionalChars(const QString& chars) { m_additional = chars; } void PasswordGenerator::setExcludedChars(const QString& chars) { m_excluded = chars; } QString PasswordGenerator::generatePassword() const { Q_ASSERT(isValid()); const QVector groups = passwordGroups(); QVector passwordChars; for (const PasswordGroup& group : groups) { for (QChar ch : group) { passwordChars.append(ch); } } QString password; if (m_flags & CharFromEveryGroup) { for (const auto& group : groups) { int pos = randomGen()->randomUInt(static_cast(group.size())); password.append(group[pos]); } for (int i = groups.size(); i < m_length; i++) { int pos = randomGen()->randomUInt(static_cast(passwordChars.size())); password.append(passwordChars[pos]); } // shuffle chars for (int i = (password.size() - 1); i >= 1; i--) { int j = randomGen()->randomUInt(static_cast(i + 1)); QChar tmp = password[i]; password[i] = password[j]; password[j] = tmp; } } else { for (int i = 0; i < m_length; i++) { int pos = randomGen()->randomUInt(static_cast(passwordChars.size())); password.append(passwordChars[pos]); } } return password; } bool PasswordGenerator::isValid() const { if (m_classes == 0 && m_additional.isEmpty()) { return false; } else if (m_length == 0) { return false; } if ((m_flags & CharFromEveryGroup) && (m_length < numCharClasses())) { return false; } return !passwordGroups().isEmpty(); } QVector PasswordGenerator::passwordGroups() const { QVector passwordGroups; if (m_classes & LowerLetters) { PasswordGroup group; for (int i = 97; i <= (97 + 25); i++) { if ((m_flags & ExcludeLookAlike) && (i == 108)) { // "l" continue; } group.append(i); } passwordGroups.append(group); } if (m_classes & UpperLetters) { PasswordGroup group; for (int i = 65; i <= (65 + 25); i++) { if ((m_flags & ExcludeLookAlike) && (i == 73 || i == 79)) { // "I" and "O" continue; } group.append(i); } passwordGroups.append(group); } if (m_classes & Numbers) { PasswordGroup group; for (int i = 48; i < (48 + 10); i++) { if ((m_flags & ExcludeLookAlike) && (i == 48 || i == 49)) { // "0" and "1" continue; } group.append(i); } passwordGroups.append(group); } if (m_classes & Braces) { PasswordGroup group; // ()[]{} group.append(40); group.append(41); group.append(91); group.append(93); group.append(123); group.append(125); passwordGroups.append(group); } if (m_classes & Punctuation) { PasswordGroup group; // .,:; 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); // "|" } 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); passwordGroups.append(group); } if (m_classes & Logograms) { PasswordGroup group; // #$%& for (int i = 35; i <= 38; i++) { group.append(i); } // @^`~ group.append(64); group.append(94); group.append(96); group.append(126); passwordGroups.append(group); } if (m_classes & EASCII) { PasswordGroup group; // [U+0080, U+009F] are C1 control characters, // U+00A0 is non-breaking space for (int i = 161; i <= 172; i++) { group.append(i); } // U+00AD is soft hyphen (format character) for (int i = 174; i <= 255; i++) { if ((m_flags & ExcludeLookAlike) && (i == 249)) { // "﹒" continue; } group.append(i); } passwordGroups.append(group); } if (!m_additional.isEmpty()) { PasswordGroup group; for (auto ch : m_additional) { group.append(ch); } passwordGroups.append(group); } // 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); } } if (!group.isEmpty()) { passwordGroups.replace(i, group); ++i; } else { passwordGroups.remove(i); } } return passwordGroups; } int PasswordGenerator::numCharClasses() const { int numClasses = 0; if (m_classes & LowerLetters) { numClasses++; } if (m_classes & UpperLetters) { numClasses++; } if (m_classes & Numbers) { numClasses++; } 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) { numClasses++; } if (m_classes & EASCII) { numClasses++; } return numClasses; }