mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-04-04 04:57:39 +03:00
* Deprecated qSort() -> std::sort() * Replace QDateTime::toString(Qt::DefaultLocaleShortDate) with Clock::toString() * Replace QDateTime::toString(Qt::SystemLocaleShortDate) with QLocale::system().toString(..., QLocale::ShortFormat) * Use QDateTime::startOfDay() instead of QDate(QDateTime) Note: QDateTime::startOfDay() is only available in Qt 5.14, we need to guard it * Replace QString::SkipEmptyParts with Qt::SkipEmptyParts Note: Its designated replacement, Qt::SplitBehavior, was only added in Qt 5.14. * Don't call deprecated QFlags(nullptr) constructor * QSet::{toList->values} * Replace QList::toSet, QSet::fromList with Tools::asSet() * QHash::insertMulti -> QMultiHash::insert * QProcess::startDetached: non-deprecated overload * QProcess::{pid->processId} * QPainter::{HighQuality->}Antialiasing * QPalette::{background->window}() * Use Qt::{Background,Foreground}Role * endl -> Qt::endl, flush -> Qt::flush * Make YubiKey::s_interfaceMutex non-recursive * OpenSSHKeyGenDialog: use non-deprecated QComboBox::sizeAdjustPolicy setting
104 lines
3.6 KiB
C++
104 lines
3.6 KiB
C++
/*
|
|
* Copyright (C) 2019 KeePassXC Team <team@keepassxc.org>
|
|
*
|
|
* 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 "Analyze.h"
|
|
|
|
#include "Utils.h"
|
|
#include "core/Global.h"
|
|
#include "core/Group.h"
|
|
#include "core/HibpOffline.h"
|
|
|
|
#include <QCommandLineParser>
|
|
#include <QFile>
|
|
|
|
const QCommandLineOption Analyze::HIBPDatabaseOption = QCommandLineOption(
|
|
{"H", "hibp"},
|
|
QObject::tr("Check if any passwords have been publicly leaked. FILENAME must be the path of a file listing "
|
|
"SHA-1 hashes of leaked passwords in HIBP format, as available from "
|
|
"https://haveibeenpwned.com/Passwords."),
|
|
QObject::tr("FILENAME"));
|
|
|
|
const QCommandLineOption Analyze::OkonOption =
|
|
QCommandLineOption("okon",
|
|
QObject::tr("Path to okon-cli to search a formatted HIBP file"),
|
|
QObject::tr("okon-cli"));
|
|
|
|
Analyze::Analyze()
|
|
{
|
|
name = QString("analyze");
|
|
description = QObject::tr("Analyze passwords for weaknesses and problems.");
|
|
options.append(Analyze::HIBPDatabaseOption);
|
|
options.append(Analyze::OkonOption);
|
|
}
|
|
|
|
int Analyze::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<QCommandLineParser> parser)
|
|
{
|
|
auto& out = Utils::STDOUT;
|
|
auto& err = Utils::STDERR;
|
|
|
|
QList<QPair<const Entry*, int>> findings;
|
|
QString error;
|
|
|
|
auto hibpDatabase = parser->value(Analyze::HIBPDatabaseOption);
|
|
if (!QFile::exists(hibpDatabase) || hibpDatabase.isEmpty()) {
|
|
err << QObject::tr("Cannot find HIBP file: %1").arg(hibpDatabase);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
auto okon = parser->value(Analyze::OkonOption);
|
|
if (!okon.isEmpty()) {
|
|
out << QObject::tr("Evaluating database entries using okon…") << Qt::endl;
|
|
|
|
if (!HibpOffline::okonReport(database, okon, hibpDatabase, findings, &error)) {
|
|
err << error << Qt::endl;
|
|
return EXIT_FAILURE;
|
|
}
|
|
} else {
|
|
QFile hibpFile(hibpDatabase);
|
|
if (!hibpFile.open(QFile::ReadOnly)) {
|
|
err << QObject::tr("Failed to open HIBP file %1: %2").arg(hibpDatabase).arg(hibpFile.errorString())
|
|
<< Qt::endl;
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
out << QObject::tr("Evaluating database entries against HIBP file, this will take a while…") << Qt::endl;
|
|
|
|
if (!HibpOffline::report(database, hibpFile, findings, &error)) {
|
|
err << error << Qt::endl;
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
for (const auto& finding : findings) {
|
|
const auto entry = finding.first;
|
|
auto count = finding.second;
|
|
|
|
QString path = entry->title();
|
|
for (auto g = entry->group(); g && g != g->database()->rootGroup(); g = g->parentGroup()) {
|
|
path.prepend("/").prepend(g->name());
|
|
}
|
|
|
|
if (count > 0) {
|
|
out << QObject::tr("Password for '%1' has been leaked %2 time(s)!", "", count).arg(path).arg(count)
|
|
<< Qt::endl;
|
|
} else {
|
|
out << QObject::tr("Password for '%1' has been leaked!").arg(path) << Qt::endl;
|
|
}
|
|
}
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|