keepassxc/src/cli/Merge.cpp

138 lines
4.3 KiB
C++
Raw Normal View History

2017-01-06 16:25:26 -05:00
/*
* Copyright (C) 2010 Felix Geyer <debfx@fobos.de>
*
* 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/>.
*/
2017-02-02 17:30:54 -05:00
#include <cstdlib>
2017-01-30 19:18:35 -05:00
#include "Merge.h"
2017-01-06 16:25:26 -05:00
#include <QCommandLineParser>
2017-01-06 16:25:26 -05:00
#include <QCoreApplication>
#include <QFile>
#include <QSaveFile>
2017-01-06 20:24:50 -05:00
#include <QStringList>
#include <QTextStream>
2017-01-06 16:25:26 -05:00
#include "core/Database.h"
#include "format/KeePass2Reader.h"
#include "format/KeePass2Writer.h"
#include "keys/CompositeKey.h"
2017-01-06 20:24:50 -05:00
2017-01-30 19:18:35 -05:00
int Merge::execute(int argc, char** argv)
2017-01-06 16:25:26 -05:00
{
2017-01-06 16:25:26 -05:00
QCoreApplication app(argc, argv);
2017-02-16 14:26:51 -05:00
QTextStream out(stdout);
2017-01-06 16:25:26 -05:00
QCommandLineParser parser;
2017-02-15 21:05:40 -05:00
parser.setApplicationDescription(QCoreApplication::translate("main", "Merge two databases."));
2017-03-12 13:34:56 -04:00
parser.addPositionalArgument("database1", QCoreApplication::translate("main", "Path of the database to merge into."));
parser.addPositionalArgument("database2", QCoreApplication::translate("main", "Path of the database to merge from."));
QCommandLineOption samePasswordOption(QStringList() << "s" << "same-password",
2017-03-12 13:34:56 -04:00
QCoreApplication::translate("main", "Use the same password for both database files."));
parser.addOption(samePasswordOption);
parser.process(app);
const QStringList args = parser.positionalArguments();
if (args.size() != 2) {
parser.showHelp();
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
2017-02-16 14:26:51 -05:00
out << "Insert the database password\n> ";
out.flush();
2017-01-07 15:14:54 -05:00
static QTextStream inputTextStream(stdin, QIODevice::ReadOnly);
2017-01-08 22:46:30 -05:00
QString line1 = inputTextStream.readLine();
2017-01-14 13:25:30 -05:00
CompositeKey key1 = CompositeKey::readFromLine(line1);
2017-01-08 22:46:30 -05:00
CompositeKey key2;
if (parser.isSet("same-password")) {
key2 = *key1.clone();
2017-01-08 22:46:30 -05:00
}
else {
QString line2 = inputTextStream.readLine();
2017-01-14 13:25:30 -05:00
key2 = CompositeKey::readFromLine(line2);
2017-01-06 16:25:26 -05:00
}
QString databaseFilename1 = args.at(0);
2017-01-07 15:14:54 -05:00
QFile dbFile1(databaseFilename1);
2017-01-06 16:25:26 -05:00
if (!dbFile1.exists()) {
2017-01-07 15:14:54 -05:00
qCritical("File %s does not exist.", qPrintable(databaseFilename1));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
if (!dbFile1.open(QIODevice::ReadOnly)) {
2017-01-07 15:14:54 -05:00
qCritical("Unable to open file %s.", qPrintable(databaseFilename1));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
KeePass2Reader reader1;
2017-01-08 22:46:30 -05:00
Database* db1 = reader1.readDatabase(&dbFile1, key1);
2017-01-06 16:25:26 -05:00
if (reader1.hasError()) {
qCritical("Error while parsing the database:\n%s\n", qPrintable(reader1.errorString()));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
QString databaseFilename2 = args.at(1);
2017-01-07 15:14:54 -05:00
QFile dbFile2(databaseFilename2);
2017-01-06 16:25:26 -05:00
if (!dbFile2.exists()) {
2017-01-07 15:14:54 -05:00
qCritical("File %s does not exist.", qPrintable(databaseFilename2));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
if (!dbFile2.open(QIODevice::ReadOnly)) {
2017-01-07 15:14:54 -05:00
qCritical("Unable to open file %s.", qPrintable(databaseFilename2));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
KeePass2Reader reader2;
2017-01-08 22:46:30 -05:00
Database* db2 = reader2.readDatabase(&dbFile2, key2);
2017-01-06 16:25:26 -05:00
2017-01-08 22:46:30 -05:00
if (reader2.hasError()) {
qCritical("Error while parsing the database:\n%s\n", qPrintable(reader2.errorString()));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
db1->merge(db2);
2017-01-07 15:14:54 -05:00
QSaveFile saveFile(databaseFilename1);
2017-01-06 16:25:26 -05:00
if (!saveFile.open(QIODevice::WriteOnly)) {
2017-01-07 15:14:54 -05:00
qCritical("Unable to open file %s for writing.", qPrintable(databaseFilename1));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
KeePass2Writer writer;
writer.writeDatabase(&saveFile, db1);
if (writer.hasError()) {
qCritical("Error while updating the database:\n%s\n", qPrintable(writer.errorString()));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
if (!saveFile.commit()) {
qCritical("Error while updating the database:\n%s\n", qPrintable(writer.errorString()));
2017-02-02 17:30:54 -05:00
return EXIT_FAILURE;
2017-01-06 16:25:26 -05:00
}
2017-02-16 14:26:51 -05:00
out << "Successfully merged the database files.\n";
2017-02-02 17:30:54 -05:00
return EXIT_SUCCESS;
2017-01-06 16:25:26 -05:00
}