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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2017-01-11 21:00:11 -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 "crypto/Crypto.h"
|
|
|
|
#include "format/KeePass2Reader.h"
|
|
|
|
#include "format/KeePass2Writer.h"
|
|
|
|
#include "keys/CompositeKey.h"
|
|
|
|
#include "keys/FileKey.h"
|
|
|
|
#include "keys/PasswordKey.h"
|
|
|
|
|
2017-01-11 21:00:11 -05:00
|
|
|
/*
|
|
|
|
* Read a key from a line of input.
|
|
|
|
* If the line references a valid file
|
|
|
|
* path, the key is loaded from file.
|
|
|
|
*/
|
|
|
|
CompositeKey readKeyFromLine(QString line)
|
|
|
|
{
|
|
|
|
|
|
|
|
CompositeKey key;
|
|
|
|
if (QFile::exists(line)) {
|
|
|
|
FileKey fileKey;
|
|
|
|
fileKey.load(line);
|
|
|
|
key.addKey(fileKey);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
PasswordKey password;
|
|
|
|
password.setPassword(line);
|
|
|
|
key.addKey(password);
|
|
|
|
}
|
|
|
|
return key;
|
|
|
|
|
|
|
|
}
|
2017-01-06 20:24:50 -05:00
|
|
|
|
2017-01-06 16:25:26 -05:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
2017-01-11 21:00:11 -05:00
|
|
|
|
2017-01-06 16:25:26 -05:00
|
|
|
QCoreApplication app(argc, argv);
|
|
|
|
|
2017-01-11 21:00:11 -05:00
|
|
|
QCommandLineParser parser;
|
|
|
|
parser.setApplicationDescription(QCoreApplication::translate("main", "Merge 2 KeePassXC database files."));
|
|
|
|
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",
|
|
|
|
QCoreApplication::translate("main", "use the same password for both database files."));
|
|
|
|
|
|
|
|
parser.addHelpOption();
|
|
|
|
parser.addOption(samePasswordOption);
|
|
|
|
parser.process(app);
|
|
|
|
|
|
|
|
const QStringList args = parser.positionalArguments();
|
|
|
|
if (args.size() != 2) {
|
|
|
|
parser.showHelp();
|
2017-01-06 16:25:26 -05:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Crypto::init()) {
|
|
|
|
qFatal("Fatal error while testing the cryptographic functions:\n%s", qPrintable(Crypto::errorString()));
|
|
|
|
}
|
|
|
|
|
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-11 21:00:11 -05:00
|
|
|
CompositeKey key1 = readKeyFromLine(line1);
|
2017-01-08 22:46:30 -05:00
|
|
|
|
|
|
|
CompositeKey key2;
|
2017-01-11 21:00:11 -05:00
|
|
|
if (parser.isSet("same-password")) {
|
|
|
|
key2 = *key1.clone();
|
2017-01-08 22:46:30 -05:00
|
|
|
}
|
|
|
|
else {
|
2017-01-11 21:00:11 -05:00
|
|
|
QString line2 = inputTextStream.readLine();
|
|
|
|
key2 = readKeyFromLine(line2);
|
2017-01-06 16:25:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-01-11 21:00:11 -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-01-06 16:25:26 -05:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (!dbFile1.open(QIODevice::ReadOnly)) {
|
2017-01-07 15:14:54 -05:00
|
|
|
qCritical("Unable to open file %s.", qPrintable(databaseFilename1));
|
2017-01-06 16:25:26 -05:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
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()));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-01-11 21:00:11 -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-01-06 16:25:26 -05:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (!dbFile2.open(QIODevice::ReadOnly)) {
|
2017-01-07 15:14:54 -05:00
|
|
|
qCritical("Unable to open file %s.", qPrintable(databaseFilename2));
|
2017-01-06 16:25:26 -05:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
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-01-06 16:25:26 -05:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
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-01-06 16:25:26 -05:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
KeePass2Writer writer;
|
|
|
|
writer.writeDatabase(&saveFile, db1);
|
|
|
|
|
|
|
|
if (writer.hasError()) {
|
|
|
|
qCritical("Error while updating the database:\n%s\n", qPrintable(writer.errorString()));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!saveFile.commit()) {
|
|
|
|
qCritical("Error while updating the database:\n%s\n", qPrintable(writer.errorString()));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-01-06 20:24:50 -05:00
|
|
|
qDebug("Successfully merged the database files.\n");
|
2017-01-06 16:25:26 -05:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
}
|