From ab6b6f36a0429f44e060f547e2cadf912c64d268 Mon Sep 17 00:00:00 2001
From: Adrian Martin <22200464+AdriandMartin@users.noreply.github.com>
Date: Sun, 23 Feb 2025 14:43:06 +0100
Subject: [PATCH] Feature: HTML export from CLI tool (#11590)
This commit introduces support for exporting a KeePassXC database in
HTML format via the CLI tool. The key changes include:
- Refactoring HtmlExporter:
- Moved HtmlExporter to the format directory and made its API
compatible with CsvExporter.
- Since the original HtmlExporter had a direct dependency on the
gui/Icons functions and indirect dependencies on the
gui/DatabaseIcons class, only the non-GUI parts were moved to
format/HtmlExporter.
- All icon-related functionality was encapsulated in a new child
class, gui/HtmlGuiExporter.
- The gui/HtmlGuiExporter retains the original functionality of the
HtmlExporter class.
- The format/HtmlExporter now generates HTML export without icons.
Adding icon support to format/HtmlExporter would require moving
icon management logic to the core, which could have broader
implications.
- CLI integration:
- Updated cli/Export to use format/HtmlExporter.
- GUI Integration:
- Updated gui/export/ExportDialog to use gui/HtmlGuiExporter.
- Build System Updates:
- Updated CMakeLists.txt to build HtmlExporter as part of core_SOURCES
and HtmlGuiExporter as part of gui_SOURCES.
- Testing:
- Updated TestCli to automatically verify the output of the HTML
export.
Signed-off-by: AdriandMartin
---
share/translations/keepassxc_en.ts | 8 +-
src/CMakeLists.txt | 3 +-
src/cli/Export.cpp | 8 +-
src/{gui => format}/HtmlExporter.cpp | 134 ++++++++++++++++-----------
src/{gui => format}/HtmlExporter.h | 23 +++--
src/gui/HtmlGuiExporter.cpp | 48 ++++++++++
src/gui/HtmlGuiExporter.h | 30 ++++++
src/gui/export/ExportDialog.cpp | 4 +-
tests/TestCli.cpp | 12 +++
9 files changed, 197 insertions(+), 73 deletions(-)
rename src/{gui => format}/HtmlExporter.cpp (82%)
rename src/{gui => format}/HtmlExporter.h (73%)
create mode 100644 src/gui/HtmlGuiExporter.cpp
create mode 100644 src/gui/HtmlGuiExporter.h
diff --git a/share/translations/keepassxc_en.ts b/share/translations/keepassxc_en.ts
index b007ea73e..51db21d4b 100644
--- a/share/translations/keepassxc_en.ts
+++ b/share/translations/keepassxc_en.ts
@@ -8080,10 +8080,6 @@ Do you want to overwrite it?
Exit interactive mode.
-
- Format to use when exporting. Available choices are 'xml' or 'csv'. Defaults to 'xml'.
-
- Exports the content of a database to standard output in the specified format.
@@ -9228,6 +9224,10 @@ This option is deprecated, use --set-key-file instead.
Passkey
+
+ Format to use when exporting. Available choices are 'xml', 'csv' or 'html'. Defaults to 'xml'.
+
+ start minimized to the system tray
diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt
index f6a20e107..66d074c9d 100644
--- a/src/CMakeLists.txt
+++ b/src/CMakeLists.txt
@@ -71,6 +71,7 @@ set(core_SOURCES
format/BitwardenReader.cpp
format/CsvExporter.cpp
format/CsvParser.cpp
+ format/HtmlExporter.cpp
format/KeePass1Reader.cpp
format/KeePass2.cpp
format/KeePass2RandomStream.cpp
@@ -129,7 +130,7 @@ set(gui_SOURCES
gui/FileDialog.cpp
gui/Font.cpp
gui/GuiTools.cpp
- gui/HtmlExporter.cpp
+ gui/HtmlGuiExporter.cpp
gui/IconModels.cpp
gui/KMessageWidget.cpp
gui/MainWindow.cpp
diff --git a/src/cli/Export.cpp b/src/cli/Export.cpp
index a7f454776..36b38b1de 100644
--- a/src/cli/Export.cpp
+++ b/src/cli/Export.cpp
@@ -21,13 +21,14 @@
#include "Utils.h"
#include "core/Global.h"
#include "format/CsvExporter.h"
+#include "format/HtmlExporter.h"
#include
const QCommandLineOption Export::FormatOption = QCommandLineOption(
QStringList() << "f" << "format",
- QObject::tr("Format to use when exporting. Available choices are 'xml' or 'csv'. Defaults to 'xml'."),
- QStringLiteral("xml|csv"));
+ QObject::tr("Format to use when exporting. Available choices are 'xml', 'csv' or 'html'. Defaults to 'xml'."),
+ QStringLiteral("xml|csv|html"));
Export::Export()
{
@@ -53,6 +54,9 @@ int Export::executeWithDatabase(QSharedPointer database, QSharedPointe
} else if (format.startsWith(QStringLiteral("csv"), Qt::CaseInsensitive)) {
CsvExporter csvExporter;
out << csvExporter.exportDatabase(database);
+ } else if (format.startsWith(QStringLiteral("html"), Qt::CaseInsensitive)) {
+ HtmlExporter htmlExporter;
+ out << htmlExporter.exportDatabase(database);
} else {
err << QObject::tr("Unsupported format %1").arg(format) << Qt::endl;
return EXIT_FAILURE;
diff --git a/src/gui/HtmlExporter.cpp b/src/format/HtmlExporter.cpp
similarity index 82%
rename from src/gui/HtmlExporter.cpp
rename to src/format/HtmlExporter.cpp
index 0c45259ac..933df4e35 100644
--- a/src/gui/HtmlExporter.cpp
+++ b/src/format/HtmlExporter.cpp
@@ -17,28 +17,13 @@
#include "HtmlExporter.h"
-#include
#include
#include "core/Group.h"
#include "core/Metadata.h"
-#include "gui/Icons.h"
namespace
{
- QString PixmapToHTML(const QPixmap& pixmap)
- {
- if (pixmap.isNull()) {
- return "";
- }
-
- // Based on https://stackoverflow.com/a/6621278
- QByteArray a;
- QBuffer buffer(&a);
- pixmap.save(&buffer, "PNG");
- return QString("";
- }
-
QString formatEntry(const Entry& entry)
{
// Here we collect the table rows with this entry's data fields
@@ -127,15 +112,62 @@ QString HtmlExporter::errorString() const
return m_error;
}
+QString HtmlExporter::groupIconToHtml(const Group* /* group */)
+{
+ return "";
+}
+
+QString HtmlExporter::entryIconToHtml(const Entry* /* entry */)
+{
+ return "";
+}
+
bool HtmlExporter::exportDatabase(QIODevice* device,
const QSharedPointer& db,
bool sorted,
bool ascending)
+{
+ if (device->write(exportHeader(db).toUtf8()) == -1) {
+ m_error = device->errorString();
+ return false;
+ }
+
+ if (db->rootGroup()) {
+ if (device->write(exportGroup(*db->rootGroup(), QString(), sorted, ascending).toUtf8()) == -1) {
+ m_error = device->errorString();
+ return false;
+ }
+ }
+
+ if (device->write(exportFooter().toUtf8()) == -1) {
+ m_error = device->errorString();
+ return false;
+ }
+
+ return true;
+}
+
+QString HtmlExporter::exportDatabase(const QSharedPointer& db, bool sorted, bool ascending)
+{
+ QString response;
+
+ response = exportHeader(db);
+ if (!response.isEmpty()) {
+ if (db->rootGroup()) {
+ response.append(exportGroup(*db->rootGroup(), QString(), sorted, ascending));
+ }
+ response.append(exportFooter());
+ }
+
+ return response;
+}
+
+QString HtmlExporter::exportHeader(const QSharedPointer& db)
{
const auto meta = db->metadata();
if (!meta) {
m_error = "Internal error: metadata is NULL";
- return false;
+ return "";
}
const auto header = QString(""
@@ -171,33 +203,23 @@ bool HtmlExporter::exportDatabase(QIODevice* device,
+ "