mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-02-23 07:59:54 -05:00
CLI: Add Import XML command (#3572)
The CLI now contains an "import" command that creates a new database from the specified XML export. The new database is in kdbx 4 format, and does not currently accept a keyfile in database creation. This change is required to create new databases from XML backups. Fixes #2458
This commit is contained in:
parent
82cfedfa43
commit
dbe15d32e5
@ -616,8 +616,7 @@ QList<Entry*> BrowserService::searchEntries(const QString& url, const StringPair
|
|||||||
// Check if database is connected with KeePassXC-Browser
|
// Check if database is connected with KeePassXC-Browser
|
||||||
auto databaseConnected = [&](const QSharedPointer<Database>& db) {
|
auto databaseConnected = [&](const QSharedPointer<Database>& db) {
|
||||||
for (const StringPair& keyPair : keyList) {
|
for (const StringPair& keyPair : keyList) {
|
||||||
QString key =
|
QString key = db->metadata()->customData()->value(QLatin1String(ASSOCIATE_KEY_PREFIX) + keyPair.first);
|
||||||
db->metadata()->customData()->value(QLatin1String(ASSOCIATE_KEY_PREFIX) + keyPair.first);
|
|
||||||
if (!key.isEmpty() && keyPair.second == key) {
|
if (!key.isEmpty() && keyPair.second == key) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -29,6 +29,7 @@ set(cli_SOURCES
|
|||||||
Export.cpp
|
Export.cpp
|
||||||
Generate.cpp
|
Generate.cpp
|
||||||
Help.cpp
|
Help.cpp
|
||||||
|
Import.cpp
|
||||||
List.cpp
|
List.cpp
|
||||||
Locate.cpp
|
Locate.cpp
|
||||||
Merge.cpp
|
Merge.cpp
|
||||||
|
@ -36,6 +36,7 @@
|
|||||||
#include "Export.h"
|
#include "Export.h"
|
||||||
#include "Generate.h"
|
#include "Generate.h"
|
||||||
#include "Help.h"
|
#include "Help.h"
|
||||||
|
#include "Import.h"
|
||||||
#include "List.h"
|
#include "List.h"
|
||||||
#include "Locate.h"
|
#include "Locate.h"
|
||||||
#include "Merge.h"
|
#include "Merge.h"
|
||||||
@ -180,6 +181,7 @@ namespace Commands
|
|||||||
s_commands.insert(QStringLiteral("quit"), QSharedPointer<Command>(new Exit("quit")));
|
s_commands.insert(QStringLiteral("quit"), QSharedPointer<Command>(new Exit("quit")));
|
||||||
} else {
|
} else {
|
||||||
s_commands.insert(QStringLiteral("export"), QSharedPointer<Command>(new Export()));
|
s_commands.insert(QStringLiteral("export"), QSharedPointer<Command>(new Export()));
|
||||||
|
s_commands.insert(QStringLiteral("import"), QSharedPointer<Command>(new Import()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,7 +71,7 @@ int Create::execute(const QStringList& arguments)
|
|||||||
|
|
||||||
auto key = QSharedPointer<CompositeKey>::create();
|
auto key = QSharedPointer<CompositeKey>::create();
|
||||||
|
|
||||||
auto password = getPasswordFromStdin();
|
auto password = Utils::getPasswordFromStdin();
|
||||||
if (!password.isNull()) {
|
if (!password.isNull()) {
|
||||||
key->addKey(password);
|
key->addKey(password);
|
||||||
}
|
}
|
||||||
@ -107,28 +107,6 @@ int Create::execute(const QStringList& arguments)
|
|||||||
return EXIT_SUCCESS;
|
return EXIT_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Read optional password from stdin.
|
|
||||||
*
|
|
||||||
* @return Pointer to the PasswordKey or null if passwordkey is skipped
|
|
||||||
* by user
|
|
||||||
*/
|
|
||||||
QSharedPointer<PasswordKey> Create::getPasswordFromStdin()
|
|
||||||
{
|
|
||||||
QSharedPointer<PasswordKey> passwordKey;
|
|
||||||
QTextStream out(Utils::STDOUT, QIODevice::WriteOnly);
|
|
||||||
|
|
||||||
out << QObject::tr("Insert password to encrypt database (Press enter to leave blank): ");
|
|
||||||
out.flush();
|
|
||||||
QString password = Utils::getPassword();
|
|
||||||
|
|
||||||
if (!password.isEmpty()) {
|
|
||||||
passwordKey = QSharedPointer<PasswordKey>(new PasswordKey(password));
|
|
||||||
}
|
|
||||||
|
|
||||||
return passwordKey;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load a key file from disk. When the path specified does not exist a
|
* Load a key file from disk. When the path specified does not exist a
|
||||||
* new file will be generated. No folders will be generated so the parent
|
* new file will be generated. No folders will be generated so the parent
|
||||||
|
@ -21,7 +21,6 @@
|
|||||||
#include "Command.h"
|
#include "Command.h"
|
||||||
|
|
||||||
#include "keys/FileKey.h"
|
#include "keys/FileKey.h"
|
||||||
#include "keys/PasswordKey.h"
|
|
||||||
|
|
||||||
class Create : public Command
|
class Create : public Command
|
||||||
{
|
{
|
||||||
@ -30,7 +29,6 @@ public:
|
|||||||
int execute(const QStringList& arguments) override;
|
int execute(const QStringList& arguments) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QSharedPointer<PasswordKey> getPasswordFromStdin();
|
|
||||||
bool loadFileKey(const QString& path, QSharedPointer<FileKey>& fileKey);
|
bool loadFileKey(const QString& path, QSharedPointer<FileKey>& fileKey);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
100
src/cli/Import.cpp
Normal file
100
src/cli/Import.cpp
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
/*
|
||||||
|
* 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 <cstdlib>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include <QFileInfo>
|
||||||
|
#include <QString>
|
||||||
|
#include <QTextStream>
|
||||||
|
|
||||||
|
#include "Import.h"
|
||||||
|
|
||||||
|
#include "cli/TextStream.h"
|
||||||
|
#include "cli/Utils.h"
|
||||||
|
#include "core/Database.h"
|
||||||
|
#include "keys/CompositeKey.h"
|
||||||
|
#include "keys/Key.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a database file from an XML export of another database.
|
||||||
|
* A password can be specified to encrypt the database.
|
||||||
|
* If none is specified the function will fail.
|
||||||
|
*
|
||||||
|
* If the database is being saved in a non existant directory, the
|
||||||
|
* function will fail.
|
||||||
|
*
|
||||||
|
* @return EXIT_SUCCESS on success, or EXIT_FAILURE on failure
|
||||||
|
*/
|
||||||
|
Import::Import()
|
||||||
|
{
|
||||||
|
name = QString("import");
|
||||||
|
description = QObject::tr("Import the contents of an XML database.");
|
||||||
|
positionalArguments.append({QString("xml"), QObject::tr("Path of the XML database export."), QString("")});
|
||||||
|
positionalArguments.append({QString("database"), QObject::tr("Path of the new database."), QString("")});
|
||||||
|
}
|
||||||
|
|
||||||
|
int Import::execute(const QStringList& arguments)
|
||||||
|
{
|
||||||
|
QSharedPointer<QCommandLineParser> parser = getCommandLineParser(arguments);
|
||||||
|
if (parser.isNull()) {
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
TextStream outputTextStream(parser->isSet(Command::QuietOption) ? Utils::DEVNULL : Utils::STDOUT,
|
||||||
|
QIODevice::WriteOnly);
|
||||||
|
TextStream errorTextStream(Utils::STDERR, QIODevice::WriteOnly);
|
||||||
|
|
||||||
|
const QStringList args = parser->positionalArguments();
|
||||||
|
const QString xmlExportPath = args.at(0);
|
||||||
|
const QString dbPath = args.at(1);
|
||||||
|
|
||||||
|
if (QFileInfo::exists(dbPath)) {
|
||||||
|
errorTextStream << QObject::tr("File %1 already exists.").arg(dbPath) << endl;
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto key = QSharedPointer<CompositeKey>::create();
|
||||||
|
|
||||||
|
auto password = Utils::getPasswordFromStdin();
|
||||||
|
if (!password.isNull()) {
|
||||||
|
key->addKey(password);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (key->isEmpty()) {
|
||||||
|
errorTextStream << QObject::tr("No key is set. Aborting database creation.") << endl;
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString errorMessage;
|
||||||
|
Database db;
|
||||||
|
db.setKdf(KeePass2::uuidToKdf(KeePass2::KDF_ARGON2));
|
||||||
|
db.setKey(key);
|
||||||
|
|
||||||
|
if (!db.import(xmlExportPath, &errorMessage)) {
|
||||||
|
errorTextStream << QObject::tr("Unable to import XML database export %1").arg(errorMessage) << endl;
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!db.save(dbPath, &errorMessage, true, false)) {
|
||||||
|
errorTextStream << QObject::tr("Failed to save the database: %1.").arg(errorMessage) << endl;
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
outputTextStream << QObject::tr("Successfully imported database.") << endl;
|
||||||
|
return EXIT_SUCCESS;
|
||||||
|
}
|
30
src/cli/Import.h
Normal file
30
src/cli/Import.h
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef KEEPASSXC_IMPORT_H
|
||||||
|
#define KEEPASSXC_IMPORT_H
|
||||||
|
|
||||||
|
#include "Command.h"
|
||||||
|
|
||||||
|
class Import : public Command
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Import();
|
||||||
|
int execute(const QStringList& arguments) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // KEEPASSXC_IMPORT_H
|
@ -127,7 +127,7 @@ namespace Utils
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (isPasswordProtected) {
|
if (isPasswordProtected) {
|
||||||
out << QObject::tr("Insert password to unlock %1: ").arg(databaseFilename) << flush;
|
out << QObject::tr("Enter password to unlock %1: ").arg(databaseFilename) << flush;
|
||||||
QString line = Utils::getPassword(outputDescriptor);
|
QString line = Utils::getPassword(outputDescriptor);
|
||||||
auto passwordKey = QSharedPointer<PasswordKey>::create();
|
auto passwordKey = QSharedPointer<PasswordKey>::create();
|
||||||
passwordKey->setPassword(line);
|
passwordKey->setPassword(line);
|
||||||
@ -217,6 +217,28 @@ namespace Utils
|
|||||||
return line;
|
return line;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read optional password from stdin.
|
||||||
|
*
|
||||||
|
* @return Pointer to the PasswordKey or null if passwordkey is skipped
|
||||||
|
* by user
|
||||||
|
*/
|
||||||
|
QSharedPointer<PasswordKey> getPasswordFromStdin()
|
||||||
|
{
|
||||||
|
QSharedPointer<PasswordKey> passwordKey;
|
||||||
|
QTextStream out(Utils::STDOUT, QIODevice::WriteOnly);
|
||||||
|
|
||||||
|
out << QObject::tr("Enter password to encrypt database (optional): ");
|
||||||
|
out.flush();
|
||||||
|
QString password = Utils::getPassword();
|
||||||
|
|
||||||
|
if (!password.isEmpty()) {
|
||||||
|
passwordKey = QSharedPointer<PasswordKey>(new PasswordKey(password));
|
||||||
|
}
|
||||||
|
|
||||||
|
return passwordKey;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A valid and running event loop is needed to use the global QClipboard,
|
* A valid and running event loop is needed to use the global QClipboard,
|
||||||
* so we need to use this from the CLI.
|
* so we need to use this from the CLI.
|
||||||
|
@ -40,6 +40,7 @@ namespace Utils
|
|||||||
|
|
||||||
void setStdinEcho(bool enable);
|
void setStdinEcho(bool enable);
|
||||||
QString getPassword(FILE* outputDescriptor = STDOUT);
|
QString getPassword(FILE* outputDescriptor = STDOUT);
|
||||||
|
QSharedPointer<PasswordKey> getPasswordFromStdin();
|
||||||
int clipText(const QString& text);
|
int clipText(const QString& text);
|
||||||
QSharedPointer<Database> unlockDatabase(const QString& databaseFilename,
|
QSharedPointer<Database> unlockDatabase(const QString& databaseFilename,
|
||||||
const bool isPasswordProtected = true,
|
const bool isPasswordProtected = true,
|
||||||
|
@ -51,6 +51,9 @@ Generates a random password.
|
|||||||
.IP "help [command]"
|
.IP "help [command]"
|
||||||
Displays a list of available commands, or detailed information about the specified command.
|
Displays a list of available commands, or detailed information about the specified command.
|
||||||
|
|
||||||
|
.IP "import [options] <xml> <database>"
|
||||||
|
Imports the contents of an XML database to the target database.
|
||||||
|
|
||||||
.IP "locate [options] <database> <term>"
|
.IP "locate [options] <database> <term>"
|
||||||
Locates all the entries that match a specific search term in a database.
|
Locates all the entries that match a specific search term in a database.
|
||||||
|
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
#include "core/Group.h"
|
#include "core/Group.h"
|
||||||
#include "core/Merger.h"
|
#include "core/Merger.h"
|
||||||
#include "core/Metadata.h"
|
#include "core/Metadata.h"
|
||||||
|
#include "format/KdbxXmlReader.h"
|
||||||
#include "format/KeePass2Reader.h"
|
#include "format/KeePass2Reader.h"
|
||||||
#include "format/KeePass2Writer.h"
|
#include "format/KeePass2Writer.h"
|
||||||
#include "keys/FileKey.h"
|
#include "keys/FileKey.h"
|
||||||
@ -332,6 +333,24 @@ bool Database::extract(QByteArray& xmlOutput, QString* error)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Database::import(const QString& xmlExportPath, QString* error)
|
||||||
|
{
|
||||||
|
KdbxXmlReader reader(KeePass2::FILE_VERSION_4);
|
||||||
|
QFile file(xmlExportPath);
|
||||||
|
file.open(QIODevice::ReadOnly);
|
||||||
|
|
||||||
|
reader.readDatabase(&file, this);
|
||||||
|
|
||||||
|
if (reader.hasError()) {
|
||||||
|
if (error) {
|
||||||
|
*error = reader.errorString();
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Remove the old backup and replace it with a new one
|
* Remove the old backup and replace it with a new one
|
||||||
* backups are named <filename>.old.<extension>
|
* backups are named <filename>.old.<extension>
|
||||||
|
@ -73,6 +73,7 @@ public:
|
|||||||
bool save(QString* error = nullptr, bool atomic = true, bool backup = false);
|
bool save(QString* error = nullptr, bool atomic = true, bool backup = false);
|
||||||
bool save(const QString& filePath, QString* error = nullptr, bool atomic = true, bool backup = false);
|
bool save(const QString& filePath, QString* error = nullptr, bool atomic = true, bool backup = false);
|
||||||
bool extract(QByteArray&, QString* error = nullptr);
|
bool extract(QByteArray&, QString* error = nullptr);
|
||||||
|
bool import(const QString& xmlExportPath, QString* error = nullptr);
|
||||||
|
|
||||||
bool isInitialized() const;
|
bool isInitialized() const;
|
||||||
void setInitialized(bool initialized);
|
void setInitialized(bool initialized);
|
||||||
|
@ -43,6 +43,7 @@
|
|||||||
#include "cli/Export.h"
|
#include "cli/Export.h"
|
||||||
#include "cli/Generate.h"
|
#include "cli/Generate.h"
|
||||||
#include "cli/Help.h"
|
#include "cli/Help.h"
|
||||||
|
#include "cli/Import.h"
|
||||||
#include "cli/List.h"
|
#include "cli/List.h"
|
||||||
#include "cli/Locate.h"
|
#include "cli/Locate.h"
|
||||||
#include "cli/Merge.h"
|
#include "cli/Merge.h"
|
||||||
@ -101,6 +102,12 @@ void TestCli::initTestCase()
|
|||||||
QVERIFY(sourceDbFileYubiKeyProtected.open(QIODevice::ReadOnly));
|
QVERIFY(sourceDbFileYubiKeyProtected.open(QIODevice::ReadOnly));
|
||||||
QVERIFY(Tools::readAllFromDevice(&sourceDbFileYubiKeyProtected, m_yubiKeyProtectedDbData));
|
QVERIFY(Tools::readAllFromDevice(&sourceDbFileYubiKeyProtected, m_yubiKeyProtectedDbData));
|
||||||
sourceDbFileYubiKeyProtected.close();
|
sourceDbFileYubiKeyProtected.close();
|
||||||
|
|
||||||
|
// Load the NewDatabase.xml file into temporary storage
|
||||||
|
QFile sourceXmlFile(QString(KEEPASSX_TEST_DATA_DIR).append("/NewDatabase.xml"));
|
||||||
|
QVERIFY(sourceXmlFile.open(QIODevice::ReadOnly));
|
||||||
|
QVERIFY(Tools::readAllFromDevice(&sourceXmlFile, m_xmlData));
|
||||||
|
sourceXmlFile.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestCli::init()
|
void TestCli::init()
|
||||||
@ -115,6 +122,11 @@ void TestCli::init()
|
|||||||
m_dbFile2->write(m_dbData2);
|
m_dbFile2->write(m_dbData2);
|
||||||
m_dbFile2->close();
|
m_dbFile2->close();
|
||||||
|
|
||||||
|
m_xmlFile.reset(new TemporaryFile());
|
||||||
|
m_xmlFile->open();
|
||||||
|
m_xmlFile->write(m_xmlData);
|
||||||
|
m_xmlFile->close();
|
||||||
|
|
||||||
m_keyFileProtectedDbFile.reset(new TemporaryFile());
|
m_keyFileProtectedDbFile.reset(new TemporaryFile());
|
||||||
m_keyFileProtectedDbFile->open();
|
m_keyFileProtectedDbFile->open();
|
||||||
m_keyFileProtectedDbFile->write(m_keyFileProtectedDbData);
|
m_keyFileProtectedDbFile->write(m_keyFileProtectedDbData);
|
||||||
@ -191,6 +203,7 @@ void TestCli::testBatchCommands()
|
|||||||
QVERIFY(Commands::getCommand("export"));
|
QVERIFY(Commands::getCommand("export"));
|
||||||
QVERIFY(Commands::getCommand("generate"));
|
QVERIFY(Commands::getCommand("generate"));
|
||||||
QVERIFY(Commands::getCommand("help"));
|
QVERIFY(Commands::getCommand("help"));
|
||||||
|
QVERIFY(Commands::getCommand("import"));
|
||||||
QVERIFY(Commands::getCommand("locate"));
|
QVERIFY(Commands::getCommand("locate"));
|
||||||
QVERIFY(Commands::getCommand("ls"));
|
QVERIFY(Commands::getCommand("ls"));
|
||||||
QVERIFY(Commands::getCommand("merge"));
|
QVERIFY(Commands::getCommand("merge"));
|
||||||
@ -201,7 +214,7 @@ void TestCli::testBatchCommands()
|
|||||||
QVERIFY(Commands::getCommand("rmdir"));
|
QVERIFY(Commands::getCommand("rmdir"));
|
||||||
QVERIFY(Commands::getCommand("show"));
|
QVERIFY(Commands::getCommand("show"));
|
||||||
QVERIFY(!Commands::getCommand("doesnotexist"));
|
QVERIFY(!Commands::getCommand("doesnotexist"));
|
||||||
QCOMPARE(Commands::getCommands().size(), 20);
|
QCOMPARE(Commands::getCommands().size(), 21);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestCli::testInteractiveCommands()
|
void TestCli::testInteractiveCommands()
|
||||||
@ -537,7 +550,7 @@ void TestCli::testCreate()
|
|||||||
m_stdoutFile->reset();
|
m_stdoutFile->reset();
|
||||||
|
|
||||||
QCOMPARE(m_stdoutFile->readLine(),
|
QCOMPARE(m_stdoutFile->readLine(),
|
||||||
QByteArray("Insert password to encrypt database (Press enter to leave blank): \n"));
|
QByteArray("Enter password to encrypt database (optional): \n"));
|
||||||
QCOMPARE(m_stdoutFile->readLine(), QByteArray("Successfully created new database.\n"));
|
QCOMPARE(m_stdoutFile->readLine(), QByteArray("Successfully created new database.\n"));
|
||||||
|
|
||||||
Utils::Test::setNextPassword("a");
|
Utils::Test::setNextPassword("a");
|
||||||
@ -566,7 +579,7 @@ void TestCli::testCreate()
|
|||||||
m_stderrFile->seek(errPos);
|
m_stderrFile->seek(errPos);
|
||||||
|
|
||||||
QCOMPARE(m_stdoutFile->readLine(),
|
QCOMPARE(m_stdoutFile->readLine(),
|
||||||
QByteArray("Insert password to encrypt database (Press enter to leave blank): \n"));
|
QByteArray("Enter password to encrypt database (optional): \n"));
|
||||||
QCOMPARE(m_stdoutFile->readLine(), QByteArray("Successfully created new database.\n"));
|
QCOMPARE(m_stdoutFile->readLine(), QByteArray("Successfully created new database.\n"));
|
||||||
|
|
||||||
Utils::Test::setNextPassword("a");
|
Utils::Test::setNextPassword("a");
|
||||||
@ -584,7 +597,7 @@ void TestCli::testCreate()
|
|||||||
m_stderrFile->seek(errPos);
|
m_stderrFile->seek(errPos);
|
||||||
|
|
||||||
QCOMPARE(m_stdoutFile->readLine(),
|
QCOMPARE(m_stdoutFile->readLine(),
|
||||||
QByteArray("Insert password to encrypt database (Press enter to leave blank): \n"));
|
QByteArray("Enter password to encrypt database (optional): \n"));
|
||||||
QCOMPARE(m_stdoutFile->readLine(), QByteArray("Successfully created new database.\n"));
|
QCOMPARE(m_stdoutFile->readLine(), QByteArray("Successfully created new database.\n"));
|
||||||
|
|
||||||
Utils::Test::setNextPassword("a");
|
Utils::Test::setNextPassword("a");
|
||||||
@ -983,6 +996,59 @@ void TestCli::testGenerate()
|
|||||||
QCOMPARE(m_stderrFile->readLine(), QByteArray("Invalid password length bleuh\n"));
|
QCOMPARE(m_stderrFile->readLine(), QByteArray("Invalid password length bleuh\n"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TestCli::testImport()
|
||||||
|
{
|
||||||
|
Import importCmd;
|
||||||
|
QVERIFY(!importCmd.name.isEmpty());
|
||||||
|
QVERIFY(importCmd.getDescriptionLine().contains(importCmd.name));
|
||||||
|
|
||||||
|
QScopedPointer<QTemporaryDir> testDir(new QTemporaryDir());
|
||||||
|
QString databaseFilename = testDir->path() + "testImport1.kdbx";
|
||||||
|
|
||||||
|
Utils::Test::setNextPassword("a");
|
||||||
|
importCmd.execute({"import", m_xmlFile->fileName(), databaseFilename});
|
||||||
|
|
||||||
|
m_stderrFile->reset();
|
||||||
|
m_stdoutFile->reset();
|
||||||
|
|
||||||
|
QCOMPARE(m_stdoutFile->readLine(), QByteArray("Enter password to encrypt database (optional): \n"));
|
||||||
|
QCOMPARE(m_stdoutFile->readLine(), QByteArray("Successfully imported database.\n"));
|
||||||
|
|
||||||
|
Utils::Test::setNextPassword("a");
|
||||||
|
auto db = QSharedPointer<Database>(Utils::unlockDatabase(databaseFilename, true, "", "", Utils::DEVNULL));
|
||||||
|
QVERIFY(db);
|
||||||
|
auto* entry = db->rootGroup()->findEntryByPath("/Sample Entry 1");
|
||||||
|
QVERIFY(entry);
|
||||||
|
QCOMPARE(entry->username(), QString("User Name"));
|
||||||
|
|
||||||
|
// Should refuse to create the database if it already exists.
|
||||||
|
qint64 pos = m_stdoutFile->pos();
|
||||||
|
qint64 errPos = m_stderrFile->pos();
|
||||||
|
importCmd.execute({"import", m_xmlFile->fileName(), databaseFilename});
|
||||||
|
m_stdoutFile->seek(pos);
|
||||||
|
m_stderrFile->seek(errPos);
|
||||||
|
// Output should be empty when there is an error.
|
||||||
|
QCOMPARE(m_stdoutFile->readAll(), QByteArray(""));
|
||||||
|
QString errorMessage = QString("File " + databaseFilename + " already exists.\n");
|
||||||
|
QCOMPARE(m_stderrFile->readAll(), errorMessage.toUtf8());
|
||||||
|
|
||||||
|
// Quiet option
|
||||||
|
QScopedPointer<QTemporaryDir> testDirQuiet(new QTemporaryDir());
|
||||||
|
QString databaseFilenameQuiet = testDirQuiet->path() + "testImport2.kdbx";
|
||||||
|
|
||||||
|
pos = m_stdoutFile->pos();
|
||||||
|
Utils::Test::setNextPassword("a");
|
||||||
|
importCmd.execute({"import", "-q", m_xmlFile->fileName(), databaseFilenameQuiet});
|
||||||
|
m_stdoutFile->seek(pos);
|
||||||
|
|
||||||
|
QCOMPARE(m_stdoutFile->readAll(),
|
||||||
|
QByteArray("Enter password to encrypt database (optional): \n"));
|
||||||
|
|
||||||
|
Utils::Test::setNextPassword("a");
|
||||||
|
auto dbQuiet = QSharedPointer<Database>(Utils::unlockDatabase(databaseFilenameQuiet, true, "", "", Utils::DEVNULL));
|
||||||
|
QVERIFY(dbQuiet);
|
||||||
|
}
|
||||||
|
|
||||||
void TestCli::testKeyFileOption()
|
void TestCli::testKeyFileOption()
|
||||||
{
|
{
|
||||||
List listCmd;
|
List listCmd;
|
||||||
|
@ -58,6 +58,7 @@ private slots:
|
|||||||
void testExport();
|
void testExport();
|
||||||
void testGenerate_data();
|
void testGenerate_data();
|
||||||
void testGenerate();
|
void testGenerate();
|
||||||
|
void testImport();
|
||||||
void testKeyFileOption();
|
void testKeyFileOption();
|
||||||
void testNoPasswordOption();
|
void testNoPasswordOption();
|
||||||
void testHelp();
|
void testHelp();
|
||||||
@ -77,11 +78,13 @@ private slots:
|
|||||||
private:
|
private:
|
||||||
QByteArray m_dbData;
|
QByteArray m_dbData;
|
||||||
QByteArray m_dbData2;
|
QByteArray m_dbData2;
|
||||||
|
QByteArray m_xmlData;
|
||||||
QByteArray m_yubiKeyProtectedDbData;
|
QByteArray m_yubiKeyProtectedDbData;
|
||||||
QByteArray m_keyFileProtectedDbData;
|
QByteArray m_keyFileProtectedDbData;
|
||||||
QByteArray m_keyFileProtectedNoPasswordDbData;
|
QByteArray m_keyFileProtectedNoPasswordDbData;
|
||||||
QScopedPointer<TemporaryFile> m_dbFile;
|
QScopedPointer<TemporaryFile> m_dbFile;
|
||||||
QScopedPointer<TemporaryFile> m_dbFile2;
|
QScopedPointer<TemporaryFile> m_dbFile2;
|
||||||
|
QScopedPointer<TemporaryFile> m_xmlFile;
|
||||||
QScopedPointer<TemporaryFile> m_keyFileProtectedDbFile;
|
QScopedPointer<TemporaryFile> m_keyFileProtectedDbFile;
|
||||||
QScopedPointer<TemporaryFile> m_keyFileProtectedNoPasswordDbFile;
|
QScopedPointer<TemporaryFile> m_keyFileProtectedNoPasswordDbFile;
|
||||||
QScopedPointer<TemporaryFile> m_yubiKeyProtectedDbFile;
|
QScopedPointer<TemporaryFile> m_yubiKeyProtectedDbFile;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user