CLI: Export database as CSV

* Changed `Extract` to `Export` to support additional formats
* Allow database expot as CSV.  Added a `--format` option to the `Export` command for that, which defaults to xml, so the current behavior is unchanged.
*The `CsvExporter` had to be refactored a bit, but nothing major. It can
now print to a file or return a string.
This commit is contained in:
louib 2019-06-16 21:33:13 -04:00 committed by Jonathan White
parent 547c246e88
commit 77fcde875e
11 changed files with 157 additions and 89 deletions

View File

@ -7,13 +7,15 @@
- Group sorting feature [#3282](https://github.com/keepassxreboot/keepassxc/issues/3282) - Group sorting feature [#3282](https://github.com/keepassxreboot/keepassxc/issues/3282)
- CLI: Add 'flatten' option to the 'ls' command [#3276](https://github.com/keepassxreboot/keepassxc/issues/3276) - CLI: Add 'flatten' option to the 'ls' command [#3276](https://github.com/keepassxreboot/keepassxc/issues/3276)
- CLI: Add password generation options to `Add` and `Edit` commands [#3275](https://github.com/keepassxreboot/keepassxc/issues/3275) - CLI: Add password generation options to `Add` and `Edit` commands [#3275](https://github.com/keepassxreboot/keepassxc/issues/3275)
- CLI: Add CSV export to the 'export' command [#3277]
- Add 'Monospaced font' option to the Notes field [#3321](https://github.com/keepassxreboot/keepassxc/issues/3321) - Add 'Monospaced font' option to the Notes field [#3321](https://github.com/keepassxreboot/keepassxc/issues/3321)
### Changed ### Changed
- 💥💥 CLI: The password length option `-l` for the CLI commands - CLI: The password length option `-l` for the CLI commands
`Add` and `Edit` is now `-L` [#3275](https://github.com/keepassxreboot/keepassxc/issues/3275) `Add` and `Edit` is now `-L` [#3275](https://github.com/keepassxreboot/keepassxc/issues/3275)
- 💥💥 CLI: the `-u` shorthand for the `--upper` password generation option has been renamed `-U` [#3275](https://github.com/keepassxreboot/keepassxc/issues/3275) - CLI: the `-u` shorthand for the `--upper` password generation option has been renamed `-U` [#3275](https://github.com/keepassxreboot/keepassxc/issues/3275)
- CLI: Renamed command `extract` -> `export`. [#3277]
- Rework the Entry Preview panel [#3306](https://github.com/keepassxreboot/keepassxc/issues/3306) - Rework the Entry Preview panel [#3306](https://github.com/keepassxreboot/keepassxc/issues/3306)
- Move notes to General tab on Group Preview Panel [#3336](https://github.com/keepassxreboot/keepassxc/issues/3336) - Move notes to General tab on Group Preview Panel [#3336](https://github.com/keepassxreboot/keepassxc/issues/3336)
- Drop to background when copy feature [#3253](https://github.com/keepassxreboot/keepassxc/issues/3253) - Drop to background when copy feature [#3253](https://github.com/keepassxreboot/keepassxc/issues/3253)

View File

@ -23,7 +23,7 @@ set(cli_SOURCES
Diceware.cpp Diceware.cpp
Edit.cpp Edit.cpp
Estimate.cpp Estimate.cpp
Extract.cpp Export.cpp
Generate.cpp Generate.cpp
List.cpp List.cpp
Locate.cpp Locate.cpp

View File

@ -29,7 +29,7 @@
#include "Diceware.h" #include "Diceware.h"
#include "Edit.h" #include "Edit.h"
#include "Estimate.h" #include "Estimate.h"
#include "Extract.h" #include "Export.h"
#include "Generate.h" #include "Generate.h"
#include "List.h" #include "List.h"
#include "Locate.h" #include "Locate.h"
@ -114,7 +114,7 @@ void populateCommands()
commands.insert(QString("diceware"), new Diceware()); commands.insert(QString("diceware"), new Diceware());
commands.insert(QString("edit"), new Edit()); commands.insert(QString("edit"), new Edit());
commands.insert(QString("estimate"), new Estimate()); commands.insert(QString("estimate"), new Estimate());
commands.insert(QString("extract"), new Extract()); commands.insert(QString("export"), new Export());
commands.insert(QString("generate"), new Generate()); commands.insert(QString("generate"), new Generate());
commands.insert(QString("locate"), new Locate()); commands.insert(QString("locate"), new Locate());
commands.insert(QString("ls"), new List()); commands.insert(QString("ls"), new List());

65
src/cli/Export.cpp Normal file
View File

@ -0,0 +1,65 @@
/*
* 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 "Export.h"
#include "cli/TextStream.h"
#include "cli/Utils.h"
#include "core/Database.h"
#include "format/CsvExporter.h"
const QCommandLineOption Export::FormatOption =
QCommandLineOption(QStringList() << "f"
<< "format",
QObject::tr("Format to use when exporting. Available choices are xml or csv. Defaults to xml."),
QObject::tr("xml|csv"));
Export::Export()
{
name = QString("export");
options.append(Export::FormatOption);
description = QObject::tr("Exports the content of a database to standard output in the specified format.");
}
int Export::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<QCommandLineParser> parser)
{
TextStream outputTextStream(Utils::STDOUT, QIODevice::WriteOnly);
TextStream errorTextStream(Utils::STDERR, QIODevice::WriteOnly);
QString format = parser->value(Export::FormatOption);
if (format.isEmpty() || format == QString("xml")) {
QByteArray xmlData;
QString errorMessage;
if (!database->extract(xmlData, &errorMessage)) {
errorTextStream << QObject::tr("Unable to export database to XML: %1").arg(errorMessage) << endl;
return EXIT_FAILURE;
}
outputTextStream << xmlData.constData() << endl;
} else if (format == QString("csv")) {
CsvExporter csvExporter;
outputTextStream << csvExporter.exportDatabase(database);
} else {
errorTextStream << QObject::tr("Unsupported format %1").arg(format) << endl;
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}

View File

@ -15,17 +15,19 @@
* along with this program. If not, see <http://www.gnu.org/licenses/>. * along with this program. If not, see <http://www.gnu.org/licenses/>.
*/ */
#ifndef KEEPASSXC_EXTRACT_H #ifndef KEEPASSXC_EXPORT_H
#define KEEPASSXC_EXTRACT_H #define KEEPASSXC_EXPORT_H
#include "DatabaseCommand.h" #include "DatabaseCommand.h"
class Extract : public DatabaseCommand class Export : public DatabaseCommand
{ {
public: public:
Extract(); Export();
int executeWithDatabase(QSharedPointer<Database> db, QSharedPointer<QCommandLineParser> parser) override; int executeWithDatabase(QSharedPointer<Database> db, QSharedPointer<QCommandLineParser> parser) override;
static const QCommandLineOption FormatOption;
}; };
#endif // KEEPASSXC_EXTRACT_H #endif // KEEPASSXC_EXPORT_H

View File

@ -1,46 +0,0 @@
/*
* 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 "Extract.h"
#include "cli/TextStream.h"
#include "cli/Utils.h"
#include "core/Database.h"
Extract::Extract()
{
name = QString("extract");
description = QObject::tr("Extract and print the content of a database.");
}
int Extract::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<QCommandLineParser>)
{
TextStream outputTextStream(Utils::STDOUT, QIODevice::WriteOnly);
TextStream errorTextStream(Utils::STDERR, QIODevice::WriteOnly);
QByteArray xmlData;
QString errorMessage;
if (!database->extract(xmlData, &errorMessage)) {
errorTextStream << QObject::tr("Unable to extract database %1").arg(errorMessage) << endl;
return EXIT_FAILURE;
}
outputTextStream << xmlData.constData() << endl;
return EXIT_SUCCESS;
}

View File

@ -36,8 +36,8 @@ The same password generation options as documented for the generate command can
.IP "estimate [options] [password]" .IP "estimate [options] [password]"
Estimates the entropy of a password. The password to estimate can be provided as a positional argument, or using the standard input. Estimates the entropy of a password. The password to estimate can be provided as a positional argument, or using the standard input.
.IP "extract [options] <database>" .IP "export [options] <database>"
Extracts and prints the contents of a database to standard output in XML format. Exports the content of a database to standard output in the specified format (defaults to XML).
.IP "generate [options]" .IP "generate [options]"
Generate a random password. Generate a random password.
@ -164,6 +164,12 @@ otherwise the program will fail. If the wordlist has < 4000 words a warning will
be printed to STDERR. be printed to STDERR.
.SS "Export options"
.IP "-f, --format"
Format to use when exporting. Available choices are xml or csv. Defaults to xml.
.SS "List options" .SS "List options"
.IP "-R, --recursive" .IP "-R, --recursive"

View File

@ -35,21 +35,22 @@ bool CsvExporter::exportDatabase(const QString& filename, const QSharedPointer<c
bool CsvExporter::exportDatabase(QIODevice* device, const QSharedPointer<const Database>& db) bool CsvExporter::exportDatabase(QIODevice* device, const QSharedPointer<const Database>& db)
{ {
QString header; if (device->write(exportHeader().toUtf8()) == -1) {
addColumn(header, "Group");
addColumn(header, "Title");
addColumn(header, "Username");
addColumn(header, "Password");
addColumn(header, "URL");
addColumn(header, "Notes");
header.append("\n");
if (device->write(header.toUtf8()) == -1) {
m_error = device->errorString(); m_error = device->errorString();
return false; return false;
} }
return writeGroup(device, db->rootGroup()); if (device->write(exportGroup(db->rootGroup()).toUtf8()) == -1) {
m_error = device->errorString();
return false;
}
return true;
}
QString CsvExporter::exportDatabase(const QSharedPointer<const Database>& db)
{
return exportHeader() + exportGroup(db->rootGroup());
} }
QString CsvExporter::errorString() const QString CsvExporter::errorString() const
@ -57,8 +58,21 @@ QString CsvExporter::errorString() const
return m_error; return m_error;
} }
bool CsvExporter::writeGroup(QIODevice* device, const Group* group, QString groupPath) QString CsvExporter::exportHeader()
{ {
QString header;
addColumn(header, "Group");
addColumn(header, "Title");
addColumn(header, "Username");
addColumn(header, "Password");
addColumn(header, "URL");
addColumn(header, "Notes");
return header + QString("\n");
}
QString CsvExporter::exportGroup(const Group* group, QString groupPath)
{
QString response;
if (!groupPath.isEmpty()) { if (!groupPath.isEmpty()) {
groupPath.append("/"); groupPath.append("/");
} }
@ -76,21 +90,15 @@ bool CsvExporter::writeGroup(QIODevice* device, const Group* group, QString grou
addColumn(line, entry->notes()); addColumn(line, entry->notes());
line.append("\n"); line.append("\n");
response.append(line);
if (device->write(line.toUtf8()) == -1) {
m_error = device->errorString();
return false;
}
} }
const QList<Group*>& children = group->children(); const QList<Group*>& children = group->children();
for (const Group* child : children) { for (const Group* child : children) {
if (!writeGroup(device, child, groupPath)) { response.append(exportGroup(child, groupPath));
return false;
}
} }
return true; return response;
} }
void CsvExporter::addColumn(QString& str, const QString& column) void CsvExporter::addColumn(QString& str, const QString& column)

View File

@ -31,10 +31,12 @@ class CsvExporter
public: public:
bool exportDatabase(const QString& filename, const QSharedPointer<const Database>& db); bool exportDatabase(const QString& filename, const QSharedPointer<const Database>& db);
bool exportDatabase(QIODevice* device, const QSharedPointer<const Database>& db); bool exportDatabase(QIODevice* device, const QSharedPointer<const Database>& db);
QString exportDatabase(const QSharedPointer<const Database>& db);
QString errorString() const; QString errorString() const;
private: private:
bool writeGroup(QIODevice* device, const Group* group, QString groupPath = QString()); QString exportGroup(const Group* group, QString groupPath = QString());
QString exportHeader();
void addColumn(QString& str, const QString& column); void addColumn(QString& str, const QString& column);
QString m_error; QString m_error;

View File

@ -39,7 +39,7 @@
#include "cli/Diceware.h" #include "cli/Diceware.h"
#include "cli/Edit.h" #include "cli/Edit.h"
#include "cli/Estimate.h" #include "cli/Estimate.h"
#include "cli/Extract.h" #include "cli/Export.h"
#include "cli/Generate.h" #include "cli/Generate.h"
#include "cli/List.h" #include "cli/List.h"
#include "cli/Locate.h" #include "cli/Locate.h"
@ -170,7 +170,7 @@ void TestCli::testCommand()
QVERIFY(Command::getCommand("diceware")); QVERIFY(Command::getCommand("diceware"));
QVERIFY(Command::getCommand("edit")); QVERIFY(Command::getCommand("edit"));
QVERIFY(Command::getCommand("estimate")); QVERIFY(Command::getCommand("estimate"));
QVERIFY(Command::getCommand("extract")); QVERIFY(Command::getCommand("export"));
QVERIFY(Command::getCommand("generate")); QVERIFY(Command::getCommand("generate"));
QVERIFY(Command::getCommand("locate")); QVERIFY(Command::getCommand("locate"));
QVERIFY(Command::getCommand("ls")); QVERIFY(Command::getCommand("ls"));
@ -742,14 +742,14 @@ void TestCli::testEstimate()
} }
} }
void TestCli::testExtract() void TestCli::testExport()
{ {
Extract extractCmd; Export exportCmd;
QVERIFY(!extractCmd.name.isEmpty()); QVERIFY(!exportCmd.name.isEmpty());
QVERIFY(extractCmd.getDescriptionLine().contains(extractCmd.name)); QVERIFY(exportCmd.getDescriptionLine().contains(exportCmd.name));
Utils::Test::setNextPassword("a"); Utils::Test::setNextPassword("a");
extractCmd.execute({"extract", m_dbFile->fileName()}); exportCmd.execute({"export", m_dbFile->fileName()});
m_stdoutFile->seek(0); m_stdoutFile->seek(0);
m_stdoutFile->readLine(); // skip prompt line m_stdoutFile->readLine(); // skip prompt line
@ -768,12 +768,41 @@ void TestCli::testExtract()
// Quiet option // Quiet option
QScopedPointer<Database> dbQuiet(new Database()); QScopedPointer<Database> dbQuiet(new Database());
qint64 pos = m_stdoutFile->pos(); qint64 pos = m_stdoutFile->pos();
qint64 posErr = m_stderrFile->pos();
Utils::Test::setNextPassword("a"); Utils::Test::setNextPassword("a");
extractCmd.execute({"extract", "-q", m_dbFile->fileName()}); exportCmd.execute({"export", "-f", "xml", "-q", m_dbFile->fileName()});
m_stdoutFile->seek(pos); m_stdoutFile->seek(pos);
m_stderrFile->seek(posErr);
reader.readDatabase(m_stdoutFile.data(), dbQuiet.data()); reader.readDatabase(m_stdoutFile.data(), dbQuiet.data());
QVERIFY(!reader.hasError()); QVERIFY(!reader.hasError());
QVERIFY(db.data()); QVERIFY(db.data());
QCOMPARE(m_stderrFile->readAll(), QByteArray(""));
// CSV exporting
pos = m_stdoutFile->pos();
posErr = m_stderrFile->pos();
Utils::Test::setNextPassword("a");
exportCmd.execute({"export", "-f", "csv", m_dbFile->fileName()});
m_stdoutFile->seek(pos);
m_stdoutFile->readLine(); // skip prompt line
m_stderrFile->seek(posErr);
QByteArray csvHeader = m_stdoutFile->readLine();
QCOMPARE(csvHeader, QByteArray("\"Group\",\"Title\",\"Username\",\"Password\",\"URL\",\"Notes\"\n"));
QByteArray csvData = m_stdoutFile->readAll();
QVERIFY(csvData.contains(QByteArray(
"\"NewDatabase\",\"Sample Entry\",\"User Name\",\"Password\",\"http://www.somesite.com/\",\"Notes\"\n")));
QCOMPARE(m_stderrFile->readAll(), QByteArray(""));
// test invalid format
pos = m_stdoutFile->pos();
posErr = m_stderrFile->pos();
Utils::Test::setNextPassword("a");
exportCmd.execute({"export", "-f", "yaml", m_dbFile->fileName()});
m_stdoutFile->seek(pos);
m_stdoutFile->readLine(); // skip prompt line
m_stderrFile->seek(posErr);
QCOMPARE(m_stdoutFile->readLine(), QByteArray(""));
QCOMPARE(m_stderrFile->readLine(), QByteArray("Unsupported format yaml\n"));
} }
void TestCli::testGenerate_data() void TestCli::testGenerate_data()

View File

@ -52,7 +52,7 @@ private slots:
void testEdit(); void testEdit();
void testEstimate_data(); void testEstimate_data();
void testEstimate(); void testEstimate();
void testExtract(); void testExport();
void testGenerate_data(); void testGenerate_data();
void testGenerate(); void testGenerate();
void testKeyFileOption(); void testKeyFileOption();