Skip to content

Commit

Permalink
Adding support for listing a group. (keepassxreboot#652)
Browse files Browse the repository at this point in the history
* Adding support for listing a group.

* added findGroupByPath

* Removing useless asserts.

* Code review.
  • Loading branch information
louib authored Jun 21, 2017
1 parent 9d6cf95 commit 48ea024
Show file tree
Hide file tree
Showing 5 changed files with 133 additions and 6 deletions.
17 changes: 15 additions & 2 deletions src/cli/List.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,9 @@ int List::execute(int argc, char** argv)
QCommandLineParser parser;
parser.setApplicationDescription(QCoreApplication::translate("main", "List database entries."));
parser.addPositionalArgument("database", QCoreApplication::translate("main", "Path of the database."));
parser.addPositionalArgument("group",
QCoreApplication::translate("main", "Path of the group to list. Default is /"),
QString("[group]"));
QCommandLineOption printUuidsOption(
QStringList() << "u"
<< "print-uuids",
Expand All @@ -57,7 +60,7 @@ int List::execute(int argc, char** argv)
parser.process(arguments);

const QStringList args = parser.positionalArguments();
if (args.size() != 1) {
if (args.size() != 1 && args.size() != 2) {
QCoreApplication app(argc, argv);
parser.showHelp();
return EXIT_FAILURE;
Expand All @@ -76,7 +79,17 @@ int List::execute(int argc, char** argv)
return EXIT_FAILURE;
}

out << db->rootGroup()->print(parser.isSet("print-uuids"));
Group* group = db->rootGroup();
if (args.size() == 2) {
QString groupPath = args.at(1);
group = db->rootGroup()->findGroupByPath(groupPath);
if (group == nullptr) {
qCritical("Cannot find group %s.", qPrintable(groupPath));
return EXIT_FAILURE;
}
}

out << group->print(parser.isSet("print-uuids"));
out.flush();
return EXIT_SUCCESS;
}
37 changes: 34 additions & 3 deletions src/core/Group.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -486,7 +486,6 @@ QList<Entry*> Group::entriesRecursive(bool includeHistoryItems) const

Entry* Group::findEntry(QString entryId)
{
Q_ASSERT(!entryId.isEmpty());
Q_ASSERT(!entryId.isNull());

if (Uuid::isUuid(entryId)) {
Expand Down Expand Up @@ -527,12 +526,11 @@ Entry* Group::findEntryByUuid(const Uuid& uuid)
Entry* Group::findEntryByPath(QString entryPath, QString basePath)
{

Q_ASSERT(!entryPath.isEmpty());
Q_ASSERT(!entryPath.isNull());

for (Entry* entry : asConst(m_entries)) {
QString currentEntryPath = basePath + entry->title();
if (entryPath == currentEntryPath) {
if (entryPath == currentEntryPath || entryPath == QString("/" + currentEntryPath)) {
return entry;
}
}
Expand All @@ -547,6 +545,39 @@ Entry* Group::findEntryByPath(QString entryPath, QString basePath)
return nullptr;
}

Group* Group::findGroupByPath(QString groupPath, QString basePath)
{

Q_ASSERT(!groupPath.isNull());

QStringList possiblePaths;
possiblePaths << groupPath;
if (!groupPath.startsWith("/")) {
possiblePaths << QString("/" + groupPath);
}
if (!groupPath.endsWith("/")) {
possiblePaths << QString(groupPath + "/");
}
if (!groupPath.endsWith("/") && !groupPath.endsWith("/")) {
possiblePaths << QString("/" + groupPath + "/");
}

if (possiblePaths.contains(basePath)) {
return this;
}

for (Group* innerGroup : children()) {
QString innerBasePath = basePath + innerGroup->name() + "/";
Group* group = innerGroup->findGroupByPath(groupPath, innerBasePath);
if (group != nullptr) {
return group;
}
}

return nullptr;

}

QString Group::print(bool printUuids, QString baseName, int depth)
{

Expand Down
3 changes: 2 additions & 1 deletion src/core/Group.h
Original file line number Diff line number Diff line change
Expand Up @@ -79,10 +79,11 @@ class Group : public QObject
static const int DefaultIconNumber;
static const int RecycleBinIconNumber;

Group* findChildByName(const QString& name);
Entry* findEntry(QString entryId);
Entry* findEntryByUuid(const Uuid& uuid);
Entry* findEntryByPath(QString entryPath, QString basePath = QString(""));
Group* findChildByName(const QString& name);
Group* findGroupByPath(QString groupPath, QString basePath = QString("/"));
void setUuid(const Uuid& uuid);
void setName(const QString& name);
void setNotes(const QString& notes);
Expand Down
81 changes: 81 additions & 0 deletions tests/TestGroup.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -599,6 +599,15 @@ void TestGroup::testFindEntry()
QVERIFY(entry != nullptr);
QCOMPARE(entry->title(), QString("entry1"));

// We also can find the entry with the leading slash.
entry = db->rootGroup()->findEntry(QString("/entry1"));
QVERIFY(entry != nullptr);
QCOMPARE(entry->title(), QString("entry1"));

// But two slashes should not be accepted.
entry = db->rootGroup()->findEntry(QString("//entry1"));
QVERIFY(entry == nullptr);

entry = db->rootGroup()->findEntry(entry2->uuid().toHex());
QVERIFY(entry != nullptr);
QCOMPARE(entry->title(), QString("entry2"));
Expand All @@ -607,6 +616,14 @@ void TestGroup::testFindEntry()
QVERIFY(entry != nullptr);
QCOMPARE(entry->title(), QString("entry2"));

entry = db->rootGroup()->findEntry(QString("/entry2"));
QVERIFY(entry == nullptr);

// We also can find the entry with the leading slash.
entry = db->rootGroup()->findEntry(QString("/group1/entry2"));
QVERIFY(entry != nullptr);
QCOMPARE(entry->title(), QString("entry2"));

// Should also find the entry only by title.
entry = db->rootGroup()->findEntry(QString("entry2"));
QVERIFY(entry != nullptr);
Expand All @@ -629,6 +646,70 @@ void TestGroup::testFindEntry()
delete db;
}

void TestGroup::testFindGroupByPath()
{
Database* db = new Database();

Group* group1 = new Group();
group1->setName("group1");
group1->setParent(db->rootGroup());

Group* group2 = new Group();
group2->setName("group2");
group2->setParent(group1);

Group* group;

group = db->rootGroup()->findGroupByPath("/");
QVERIFY(group != nullptr);
QCOMPARE(group->uuid(), db->rootGroup()->uuid());

// We also accept it if the leading slash is missing.
group = db->rootGroup()->findGroupByPath("");
QVERIFY(group != nullptr);
QCOMPARE(group->uuid(), db->rootGroup()->uuid());

group = db->rootGroup()->findGroupByPath("/group1/");
QVERIFY(group != nullptr);
QCOMPARE(group->uuid(), group1->uuid());

// We also accept it if the leading slash is missing.
group = db->rootGroup()->findGroupByPath("group1/");
QVERIFY(group != nullptr);
QCOMPARE(group->uuid(), group1->uuid());

// Too many slashes at the end
group = db->rootGroup()->findGroupByPath("group1//");
QVERIFY(group == nullptr);

// Missing a slash at the end.
group = db->rootGroup()->findGroupByPath("/group1");
QVERIFY(group != nullptr);
QCOMPARE(group->uuid(), group1->uuid());

// Too many slashes at the start
group = db->rootGroup()->findGroupByPath("//group1");
QVERIFY(group == nullptr);

group = db->rootGroup()->findGroupByPath("/group1/group2/");
QVERIFY(group != nullptr);
QCOMPARE(group->uuid(), group2->uuid());

// We also accept it if the leading slash is missing.
group = db->rootGroup()->findGroupByPath("group1/group2/");
QVERIFY(group != nullptr);
QCOMPARE(group->uuid(), group2->uuid());

group = db->rootGroup()->findGroupByPath("group1/group2");
QVERIFY(group != nullptr);
QCOMPARE(group->uuid(), group2->uuid());

group = db->rootGroup()->findGroupByPath("invalid");
QVERIFY(group == nullptr);

delete db;
}

void TestGroup::testPrint()
{
Database* db = new Database();
Expand Down
1 change: 1 addition & 0 deletions tests/TestGroup.h
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@ private slots:
void testMergeDatabase();
void testMergeConflictKeepBoth();
void testFindEntry();
void testFindGroupByPath();
void testPrint();

private:
Expand Down

0 comments on commit 48ea024

Please sign in to comment.