mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-04-04 13:07:38 +03:00
CLI: Replace locate command with search
* Introduce search CLI command to replace locate command. Search can provide the same functionality but in a more fine-grained fashion * Replace use of Group::locate in code: Use EntrySearcher in clip cli command best-match option. This removes the matching against group hierarchy of an entry which is kind of nonsense as clip expects exactly one match. Matching against groups can be done using search command. * Remove obsolete Group::locate method
This commit is contained in:
parent
ec81d2bc3f
commit
e8f2c9d126
12 changed files with 103 additions and 165 deletions
|
@ -90,10 +90,6 @@ It provides the ability to query and modify the entries of a KeePass database, d
|
||||||
If both the key file and password are empty, no database will be created.
|
If both the key file and password are empty, no database will be created.
|
||||||
The new database will be in kdbx 4 format.
|
The new database will be in kdbx 4 format.
|
||||||
|
|
||||||
|
|
||||||
*locate* [_options_] <__database__> <__term__>::
|
|
||||||
Locates all the entries that match a specific search term in a database.
|
|
||||||
|
|
||||||
*ls* [_options_] <__database__> [_group_]::
|
*ls* [_options_] <__database__> [_group_]::
|
||||||
Lists the contents of a group in a database.
|
Lists the contents of a group in a database.
|
||||||
If no group is specified, it will default to the root group.
|
If no group is specified, it will default to the root group.
|
||||||
|
@ -127,6 +123,9 @@ It provides the ability to query and modify the entries of a KeePass database, d
|
||||||
If the database has a recycle bin, the group will be moved there.
|
If the database has a recycle bin, the group will be moved there.
|
||||||
If the group is already in the recycle bin, it will be removed permanently.
|
If the group is already in the recycle bin, it will be removed permanently.
|
||||||
|
|
||||||
|
*search* [_options_] <__database__> <__term__>::
|
||||||
|
Searches all entries that match a specific search term in a database.
|
||||||
|
|
||||||
*show* [_options_] <__database__> <__entry__>::
|
*show* [_options_] <__database__> <__entry__>::
|
||||||
Shows the title, username, password, URL and notes of a database entry.
|
Shows the title, username, password, URL and notes of a database entry.
|
||||||
Can also show the current TOTP.
|
Can also show the current TOTP.
|
||||||
|
@ -221,7 +220,7 @@ The same password generation options as documented for the generate command can
|
||||||
Will report an error if no TOTP is configured for the entry.
|
Will report an error if no TOTP is configured for the entry.
|
||||||
|
|
||||||
*-b*, *--best*::
|
*-b*, *--best*::
|
||||||
Try to find and copy to clipboard a unique entry matching the input (similar to *-locate*)
|
Try to find and copy to clipboard a unique entry matching the input
|
||||||
If a unique matching entry is found it will be copied to the clipboard.
|
If a unique matching entry is found it will be copied to the clipboard.
|
||||||
If multiple entries are found they will be listed to refine the search. (no clip performed)
|
If multiple entries are found they will be listed to refine the search. (no clip performed)
|
||||||
|
|
||||||
|
|
|
@ -32,12 +32,12 @@ set(cli_SOURCES
|
||||||
Import.cpp
|
Import.cpp
|
||||||
Info.cpp
|
Info.cpp
|
||||||
List.cpp
|
List.cpp
|
||||||
Locate.cpp
|
|
||||||
Merge.cpp
|
Merge.cpp
|
||||||
Move.cpp
|
Move.cpp
|
||||||
Open.cpp
|
Open.cpp
|
||||||
Remove.cpp
|
Remove.cpp
|
||||||
RemoveGroup.cpp
|
RemoveGroup.cpp
|
||||||
|
Search.cpp
|
||||||
Show.cpp)
|
Show.cpp)
|
||||||
|
|
||||||
add_library(cli STATIC ${cli_SOURCES})
|
add_library(cli STATIC ${cli_SOURCES})
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
#include "Clip.h"
|
#include "Clip.h"
|
||||||
|
|
||||||
#include "Utils.h"
|
#include "Utils.h"
|
||||||
|
#include "core/EntrySearcher.h"
|
||||||
#include "core/Group.h"
|
#include "core/Group.h"
|
||||||
#include "core/Tools.h"
|
#include "core/Tools.h"
|
||||||
|
|
||||||
|
@ -78,16 +79,18 @@ int Clip::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<
|
||||||
|
|
||||||
QString entryPath;
|
QString entryPath;
|
||||||
if (parser->isSet(Clip::BestMatchOption)) {
|
if (parser->isSet(Clip::BestMatchOption)) {
|
||||||
QStringList results = database->rootGroup()->locate(args.at(1));
|
EntrySearcher searcher;
|
||||||
|
const auto& searchTerm = args.at(1);
|
||||||
|
const auto results = searcher.search(QString("title:%1").arg(searchTerm), database->rootGroup(), true);
|
||||||
if (results.count() > 1) {
|
if (results.count() > 1) {
|
||||||
err << QObject::tr("Multiple entries matching:") << endl;
|
err << QObject::tr("Multiple entries matching:") << endl;
|
||||||
for (const QString& result : asConst(results)) {
|
for (const Entry* result : results) {
|
||||||
err << result << endl;
|
err << result->path().prepend('/') << endl;
|
||||||
}
|
}
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
} else {
|
} else {
|
||||||
entryPath = (results.isEmpty()) ? args.at(1) : results[0];
|
entryPath = (results.isEmpty()) ? searchTerm : results[0]->path().prepend('/');
|
||||||
out << QObject::tr("Used matching entry: %1").arg(entryPath) << endl;
|
out << QObject::tr("Using matching entry: %1").arg(entryPath) << endl;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
entryPath = args.at(1);
|
entryPath = args.at(1);
|
||||||
|
|
|
@ -31,12 +31,12 @@
|
||||||
#include "Import.h"
|
#include "Import.h"
|
||||||
#include "Info.h"
|
#include "Info.h"
|
||||||
#include "List.h"
|
#include "List.h"
|
||||||
#include "Locate.h"
|
|
||||||
#include "Merge.h"
|
#include "Merge.h"
|
||||||
#include "Move.h"
|
#include "Move.h"
|
||||||
#include "Open.h"
|
#include "Open.h"
|
||||||
#include "Remove.h"
|
#include "Remove.h"
|
||||||
#include "RemoveGroup.h"
|
#include "RemoveGroup.h"
|
||||||
|
#include "Search.h"
|
||||||
#include "Show.h"
|
#include "Show.h"
|
||||||
#include "Utils.h"
|
#include "Utils.h"
|
||||||
|
|
||||||
|
@ -173,7 +173,6 @@ namespace Commands
|
||||||
s_commands.insert(QStringLiteral("estimate"), QSharedPointer<Command>(new Estimate()));
|
s_commands.insert(QStringLiteral("estimate"), QSharedPointer<Command>(new Estimate()));
|
||||||
s_commands.insert(QStringLiteral("generate"), QSharedPointer<Command>(new Generate()));
|
s_commands.insert(QStringLiteral("generate"), QSharedPointer<Command>(new Generate()));
|
||||||
s_commands.insert(QStringLiteral("help"), QSharedPointer<Command>(new Help()));
|
s_commands.insert(QStringLiteral("help"), QSharedPointer<Command>(new Help()));
|
||||||
s_commands.insert(QStringLiteral("locate"), QSharedPointer<Command>(new Locate()));
|
|
||||||
s_commands.insert(QStringLiteral("ls"), QSharedPointer<Command>(new List()));
|
s_commands.insert(QStringLiteral("ls"), QSharedPointer<Command>(new List()));
|
||||||
s_commands.insert(QStringLiteral("merge"), QSharedPointer<Command>(new Merge()));
|
s_commands.insert(QStringLiteral("merge"), QSharedPointer<Command>(new Merge()));
|
||||||
s_commands.insert(QStringLiteral("mkdir"), QSharedPointer<Command>(new AddGroup()));
|
s_commands.insert(QStringLiteral("mkdir"), QSharedPointer<Command>(new AddGroup()));
|
||||||
|
@ -181,6 +180,7 @@ namespace Commands
|
||||||
s_commands.insert(QStringLiteral("open"), QSharedPointer<Command>(new Open()));
|
s_commands.insert(QStringLiteral("open"), QSharedPointer<Command>(new Open()));
|
||||||
s_commands.insert(QStringLiteral("rm"), QSharedPointer<Command>(new Remove()));
|
s_commands.insert(QStringLiteral("rm"), QSharedPointer<Command>(new Remove()));
|
||||||
s_commands.insert(QStringLiteral("rmdir"), QSharedPointer<Command>(new RemoveGroup()));
|
s_commands.insert(QStringLiteral("rmdir"), QSharedPointer<Command>(new RemoveGroup()));
|
||||||
|
s_commands.insert(QStringLiteral("search"), QSharedPointer<Command>(new Search()));
|
||||||
s_commands.insert(QStringLiteral("show"), QSharedPointer<Command>(new Show()));
|
s_commands.insert(QStringLiteral("show"), QSharedPointer<Command>(new Show()));
|
||||||
|
|
||||||
if (interactive) {
|
if (interactive) {
|
||||||
|
|
|
@ -15,36 +15,37 @@
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "Locate.h"
|
#include "Search.h"
|
||||||
|
|
||||||
#include <QCommandLineParser>
|
#include <QCommandLineParser>
|
||||||
|
|
||||||
#include "Utils.h"
|
#include "Utils.h"
|
||||||
|
#include "core/EntrySearcher.h"
|
||||||
#include "core/Group.h"
|
#include "core/Group.h"
|
||||||
|
|
||||||
Locate::Locate()
|
Search::Search()
|
||||||
{
|
{
|
||||||
name = QString("locate");
|
name = QString("search");
|
||||||
description = QObject::tr("Find entries quickly.");
|
description = QObject::tr("Find entries quickly.");
|
||||||
positionalArguments.append({QString("term"), QObject::tr("Search term."), QString("")});
|
positionalArguments.append({QString("term"), QObject::tr("Search term."), QString("")});
|
||||||
}
|
}
|
||||||
|
|
||||||
int Locate::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<QCommandLineParser> parser)
|
int Search::executeWithDatabase(QSharedPointer<Database> database, QSharedPointer<QCommandLineParser> parser)
|
||||||
{
|
{
|
||||||
auto& out = Utils::STDOUT;
|
auto& out = Utils::STDOUT;
|
||||||
auto& err = Utils::STDERR;
|
auto& err = Utils::STDERR;
|
||||||
|
|
||||||
const QStringList args = parser->positionalArguments();
|
const QStringList args = parser->positionalArguments();
|
||||||
const QString& searchTerm = args.at(1);
|
|
||||||
|
|
||||||
QStringList results = database->rootGroup()->locate(searchTerm);
|
EntrySearcher searcher;
|
||||||
|
auto results = searcher.search(args.at(1), database->rootGroup(), true);
|
||||||
if (results.isEmpty()) {
|
if (results.isEmpty()) {
|
||||||
err << "No results for that search term." << endl;
|
err << "No results for that search term." << endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const QString& result : asConst(results)) {
|
for (const Entry* result : asConst(results)) {
|
||||||
out << result << endl;
|
out << result->path().prepend('/') << endl;
|
||||||
}
|
}
|
||||||
return EXIT_SUCCESS;
|
return EXIT_SUCCESS;
|
||||||
}
|
}
|
|
@ -15,17 +15,17 @@
|
||||||
* 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_LOCATE_H
|
#ifndef KEEPASSXC_SEARCH_H
|
||||||
#define KEEPASSXC_LOCATE_H
|
#define KEEPASSXC_SEARCH_H
|
||||||
|
|
||||||
#include "DatabaseCommand.h"
|
#include "DatabaseCommand.h"
|
||||||
|
|
||||||
class Locate : public DatabaseCommand
|
class Search : public DatabaseCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Locate();
|
Search();
|
||||||
|
|
||||||
int executeWithDatabase(QSharedPointer<Database> db, QSharedPointer<QCommandLineParser> parser) override;
|
int executeWithDatabase(QSharedPointer<Database> db, QSharedPointer<QCommandLineParser> parser) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // KEEPASSXC_LOCATE_H
|
#endif // KEEPASSXC_SEARCH_H
|
|
@ -1079,30 +1079,6 @@ bool Group::resolveAutoTypeEnabled() const
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QStringList Group::locate(const QString& locateTerm, const QString& currentPath) const
|
|
||||||
{
|
|
||||||
// TODO: Replace with EntrySearcher
|
|
||||||
QStringList response;
|
|
||||||
if (locateTerm.isEmpty()) {
|
|
||||||
return response;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const Entry* entry : asConst(m_entries)) {
|
|
||||||
QString entryPath = currentPath + entry->title();
|
|
||||||
if (entryPath.contains(locateTerm, Qt::CaseInsensitive)) {
|
|
||||||
response << entryPath;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const Group* group : asConst(m_children)) {
|
|
||||||
for (const QString& path : group->locate(locateTerm, currentPath + group->name() + QString("/"))) {
|
|
||||||
response << path;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return response;
|
|
||||||
}
|
|
||||||
|
|
||||||
Entry* Group::addEntryWithPath(const QString& entryPath)
|
Entry* Group::addEntryWithPath(const QString& entryPath)
|
||||||
{
|
{
|
||||||
if (entryPath.isEmpty() || findEntryByPath(entryPath)) {
|
if (entryPath.isEmpty() || findEntryByPath(entryPath)) {
|
||||||
|
|
|
@ -114,7 +114,6 @@ public:
|
||||||
Entry* findEntryBySearchTerm(const QString& term, EntryReferenceType referenceType);
|
Entry* findEntryBySearchTerm(const QString& term, EntryReferenceType referenceType);
|
||||||
Group* findGroupByUuid(const QUuid& uuid);
|
Group* findGroupByUuid(const QUuid& uuid);
|
||||||
Group* findGroupByPath(const QString& groupPath);
|
Group* findGroupByPath(const QString& groupPath);
|
||||||
QStringList locate(const QString& locateTerm, const QString& currentPath = {"/"}) const;
|
|
||||||
Entry* addEntryWithPath(const QString& entryPath);
|
Entry* addEntryWithPath(const QString& entryPath);
|
||||||
void setUuid(const QUuid& uuid);
|
void setUuid(const QUuid& uuid);
|
||||||
void setName(const QString& name);
|
void setName(const QString& name);
|
||||||
|
|
|
@ -41,12 +41,12 @@
|
||||||
#include "cli/Import.h"
|
#include "cli/Import.h"
|
||||||
#include "cli/Info.h"
|
#include "cli/Info.h"
|
||||||
#include "cli/List.h"
|
#include "cli/List.h"
|
||||||
#include "cli/Locate.h"
|
|
||||||
#include "cli/Merge.h"
|
#include "cli/Merge.h"
|
||||||
#include "cli/Move.h"
|
#include "cli/Move.h"
|
||||||
#include "cli/Open.h"
|
#include "cli/Open.h"
|
||||||
#include "cli/Remove.h"
|
#include "cli/Remove.h"
|
||||||
#include "cli/RemoveGroup.h"
|
#include "cli/RemoveGroup.h"
|
||||||
|
#include "cli/Search.h"
|
||||||
#include "cli/Show.h"
|
#include "cli/Show.h"
|
||||||
#include "cli/Utils.h"
|
#include "cli/Utils.h"
|
||||||
|
|
||||||
|
@ -226,7 +226,6 @@ void TestCli::testBatchCommands()
|
||||||
QVERIFY(Commands::getCommand("generate"));
|
QVERIFY(Commands::getCommand("generate"));
|
||||||
QVERIFY(Commands::getCommand("help"));
|
QVERIFY(Commands::getCommand("help"));
|
||||||
QVERIFY(Commands::getCommand("import"));
|
QVERIFY(Commands::getCommand("import"));
|
||||||
QVERIFY(Commands::getCommand("locate"));
|
|
||||||
QVERIFY(Commands::getCommand("ls"));
|
QVERIFY(Commands::getCommand("ls"));
|
||||||
QVERIFY(Commands::getCommand("merge"));
|
QVERIFY(Commands::getCommand("merge"));
|
||||||
QVERIFY(Commands::getCommand("mkdir"));
|
QVERIFY(Commands::getCommand("mkdir"));
|
||||||
|
@ -235,6 +234,7 @@ void TestCli::testBatchCommands()
|
||||||
QVERIFY(Commands::getCommand("rm"));
|
QVERIFY(Commands::getCommand("rm"));
|
||||||
QVERIFY(Commands::getCommand("rmdir"));
|
QVERIFY(Commands::getCommand("rmdir"));
|
||||||
QVERIFY(Commands::getCommand("show"));
|
QVERIFY(Commands::getCommand("show"));
|
||||||
|
QVERIFY(Commands::getCommand("search"));
|
||||||
QVERIFY(!Commands::getCommand("doesnotexist"));
|
QVERIFY(!Commands::getCommand("doesnotexist"));
|
||||||
QCOMPARE(Commands::getCommands().size(), 22);
|
QCOMPARE(Commands::getCommands().size(), 22);
|
||||||
}
|
}
|
||||||
|
@ -254,7 +254,6 @@ void TestCli::testInteractiveCommands()
|
||||||
QVERIFY(Commands::getCommand("exit"));
|
QVERIFY(Commands::getCommand("exit"));
|
||||||
QVERIFY(Commands::getCommand("generate"));
|
QVERIFY(Commands::getCommand("generate"));
|
||||||
QVERIFY(Commands::getCommand("help"));
|
QVERIFY(Commands::getCommand("help"));
|
||||||
QVERIFY(Commands::getCommand("locate"));
|
|
||||||
QVERIFY(Commands::getCommand("ls"));
|
QVERIFY(Commands::getCommand("ls"));
|
||||||
QVERIFY(Commands::getCommand("merge"));
|
QVERIFY(Commands::getCommand("merge"));
|
||||||
QVERIFY(Commands::getCommand("mkdir"));
|
QVERIFY(Commands::getCommand("mkdir"));
|
||||||
|
@ -264,6 +263,7 @@ void TestCli::testInteractiveCommands()
|
||||||
QVERIFY(Commands::getCommand("rm"));
|
QVERIFY(Commands::getCommand("rm"));
|
||||||
QVERIFY(Commands::getCommand("rmdir"));
|
QVERIFY(Commands::getCommand("rmdir"));
|
||||||
QVERIFY(Commands::getCommand("show"));
|
QVERIFY(Commands::getCommand("show"));
|
||||||
|
QVERIFY(Commands::getCommand("search"));
|
||||||
QVERIFY(!Commands::getCommand("doesnotexist"));
|
QVERIFY(!Commands::getCommand("doesnotexist"));
|
||||||
QCOMPARE(Commands::getCommands().size(), 22);
|
QCOMPARE(Commands::getCommands().size(), 22);
|
||||||
}
|
}
|
||||||
|
@ -1274,55 +1274,6 @@ void TestCli::testList()
|
||||||
QCOMPARE(m_stdout->readAll(), QByteArray());
|
QCOMPARE(m_stdout->readAll(), QByteArray());
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestCli::testLocate()
|
|
||||||
{
|
|
||||||
Locate locateCmd;
|
|
||||||
QVERIFY(!locateCmd.name.isEmpty());
|
|
||||||
QVERIFY(locateCmd.getDescriptionLine().contains(locateCmd.name));
|
|
||||||
|
|
||||||
setInput("a");
|
|
||||||
execCmd(locateCmd, {"locate", m_dbFile->fileName(), "Sample"});
|
|
||||||
m_stderr->readLine(); // Skip password prompt
|
|
||||||
QCOMPARE(m_stderr->readAll(), QByteArray());
|
|
||||||
QCOMPARE(m_stdout->readAll(), QByteArray("/Sample Entry\n"));
|
|
||||||
|
|
||||||
// Quiet option
|
|
||||||
setInput("a");
|
|
||||||
execCmd(locateCmd, {"locate", m_dbFile->fileName(), "-q", "Sample"});
|
|
||||||
QCOMPARE(m_stderr->readAll(), QByteArray());
|
|
||||||
QCOMPARE(m_stdout->readAll(), QByteArray("/Sample Entry\n"));
|
|
||||||
|
|
||||||
setInput("a");
|
|
||||||
execCmd(locateCmd, {"locate", m_dbFile->fileName(), "Does Not Exist"});
|
|
||||||
m_stderr->readLine(); // skip password prompt
|
|
||||||
QCOMPARE(m_stderr->readAll(), QByteArray("No results for that search term.\n"));
|
|
||||||
QCOMPARE(m_stdout->readAll(), QByteArray());
|
|
||||||
|
|
||||||
// write a modified database
|
|
||||||
auto db = readDatabase();
|
|
||||||
QVERIFY(db);
|
|
||||||
auto* group = db->rootGroup()->findGroupByPath("/General/");
|
|
||||||
QVERIFY(group);
|
|
||||||
auto* entry = new Entry();
|
|
||||||
entry->setUuid(QUuid::createUuid());
|
|
||||||
entry->setTitle("New Entry");
|
|
||||||
group->addEntry(entry);
|
|
||||||
|
|
||||||
TemporaryFile tmpFile;
|
|
||||||
tmpFile.open();
|
|
||||||
tmpFile.close();
|
|
||||||
db->saveAs(tmpFile.fileName());
|
|
||||||
|
|
||||||
setInput("a");
|
|
||||||
execCmd(locateCmd, {"locate", tmpFile.fileName(), "New"});
|
|
||||||
QCOMPARE(m_stdout->readAll(), QByteArray("/General/New Entry\n"));
|
|
||||||
|
|
||||||
setInput("a");
|
|
||||||
execCmd(locateCmd, {"locate", tmpFile.fileName(), "Entry"});
|
|
||||||
QCOMPARE(m_stdout->readAll(),
|
|
||||||
QByteArray("/Sample Entry\n/General/New Entry\n/Homebanking/Subgroup/Subgroup Entry\n"));
|
|
||||||
}
|
|
||||||
|
|
||||||
void TestCli::testMerge()
|
void TestCli::testMerge()
|
||||||
{
|
{
|
||||||
Merge mergeCmd;
|
Merge mergeCmd;
|
||||||
|
@ -1694,6 +1645,76 @@ void TestCli::testRemoveQuiet()
|
||||||
QVERIFY(!db->rootGroup()->findEntryByPath(QString("/%1/Sample Entry").arg(Group::tr("Recycle Bin"))));
|
QVERIFY(!db->rootGroup()->findEntryByPath(QString("/%1/Sample Entry").arg(Group::tr("Recycle Bin"))));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TestCli::testSearch()
|
||||||
|
{
|
||||||
|
Search searchCmd;
|
||||||
|
QVERIFY(!searchCmd.name.isEmpty());
|
||||||
|
QVERIFY(searchCmd.getDescriptionLine().contains(searchCmd.name));
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", m_dbFile->fileName(), "Sample"});
|
||||||
|
m_stderr->readLine(); // Skip password prompt
|
||||||
|
QCOMPARE(m_stderr->readAll(), QByteArray());
|
||||||
|
QCOMPARE(m_stdout->readAll(), QByteArray("/Sample Entry\n"));
|
||||||
|
|
||||||
|
// Quiet option
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", m_dbFile->fileName(), "-q", "Sample"});
|
||||||
|
QCOMPARE(m_stderr->readAll(), QByteArray());
|
||||||
|
QCOMPARE(m_stdout->readAll(), QByteArray("/Sample Entry\n"));
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", m_dbFile->fileName(), "Does Not Exist"});
|
||||||
|
m_stderr->readLine(); // skip password prompt
|
||||||
|
QCOMPARE(m_stderr->readAll(), QByteArray("No results for that search term.\n"));
|
||||||
|
QCOMPARE(m_stdout->readAll(), QByteArray());
|
||||||
|
|
||||||
|
// write a modified database
|
||||||
|
auto db = readDatabase();
|
||||||
|
QVERIFY(db);
|
||||||
|
auto* group = db->rootGroup()->findGroupByPath("/General/");
|
||||||
|
QVERIFY(group);
|
||||||
|
auto* entry = new Entry();
|
||||||
|
entry->setUuid(QUuid::createUuid());
|
||||||
|
entry->setTitle("New Entry");
|
||||||
|
group->addEntry(entry);
|
||||||
|
|
||||||
|
TemporaryFile tmpFile;
|
||||||
|
tmpFile.open();
|
||||||
|
tmpFile.close();
|
||||||
|
db->saveAs(tmpFile.fileName());
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", tmpFile.fileName(), "title:New"});
|
||||||
|
QCOMPARE(m_stdout->readAll(), QByteArray("/General/New Entry\n"));
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", tmpFile.fileName(), "title:Entry"});
|
||||||
|
QCOMPARE(m_stdout->readAll(),
|
||||||
|
QByteArray("/Sample Entry\n/General/New Entry\n/Homebanking/Subgroup/Subgroup Entry\n"));
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", tmpFile.fileName(), "group:General"});
|
||||||
|
QCOMPARE(m_stdout->readAll(), QByteArray("/General/New Entry\n"));
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", tmpFile.fileName(), "group:NewDatabase"});
|
||||||
|
QCOMPARE(m_stdout->readAll(), QByteArray("/Sample Entry\n"));
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", tmpFile.fileName(), "group:/NewDatabase"});
|
||||||
|
QCOMPARE(m_stdout->readAll(),
|
||||||
|
QByteArray("/Sample Entry\n/General/New Entry\n/Homebanking/Subgroup/Subgroup Entry\n"));
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", tmpFile.fileName(), "url:bank"});
|
||||||
|
QCOMPARE(m_stdout->readAll(), QByteArray("/Homebanking/Subgroup/Subgroup Entry\n"));
|
||||||
|
|
||||||
|
setInput("a");
|
||||||
|
execCmd(searchCmd, {"search", tmpFile.fileName(), "u:User Name"});
|
||||||
|
QCOMPARE(m_stdout->readAll(), QByteArray("/Sample Entry\n/Homebanking/Subgroup/Subgroup Entry\n"));
|
||||||
|
}
|
||||||
|
|
||||||
void TestCli::testShow()
|
void TestCli::testShow()
|
||||||
{
|
{
|
||||||
Show showCmd;
|
Show showCmd;
|
||||||
|
|
|
@ -65,7 +65,6 @@ private slots:
|
||||||
void testHelp();
|
void testHelp();
|
||||||
void testInteractiveCommands();
|
void testInteractiveCommands();
|
||||||
void testList();
|
void testList();
|
||||||
void testLocate();
|
|
||||||
void testMerge();
|
void testMerge();
|
||||||
void testMergeWithKeys();
|
void testMergeWithKeys();
|
||||||
void testMove();
|
void testMove();
|
||||||
|
@ -73,6 +72,7 @@ private slots:
|
||||||
void testRemove();
|
void testRemove();
|
||||||
void testRemoveGroup();
|
void testRemoveGroup();
|
||||||
void testRemoveQuiet();
|
void testRemoveQuiet();
|
||||||
|
void testSearch();
|
||||||
void testShow();
|
void testShow();
|
||||||
void testInvalidDbFiles();
|
void testInvalidDbFiles();
|
||||||
void testYubiKeyOption();
|
void testYubiKeyOption();
|
||||||
|
|
|
@ -690,66 +690,6 @@ void TestGroup::testPrint()
|
||||||
QVERIFY(output.contains(QString("subgroup/entry3\n")));
|
QVERIFY(output.contains(QString("subgroup/entry3\n")));
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestGroup::testLocate()
|
|
||||||
{
|
|
||||||
Database* db = new Database();
|
|
||||||
|
|
||||||
Entry* entry1 = new Entry();
|
|
||||||
entry1->setTitle("entry1");
|
|
||||||
entry1->setGroup(db->rootGroup());
|
|
||||||
|
|
||||||
Entry* entry2 = new Entry();
|
|
||||||
entry2->setTitle("entry2");
|
|
||||||
entry2->setGroup(db->rootGroup());
|
|
||||||
|
|
||||||
Group* group1 = new Group();
|
|
||||||
group1->setName("group1");
|
|
||||||
group1->setParent(db->rootGroup());
|
|
||||||
|
|
||||||
Group* group2 = new Group();
|
|
||||||
group2->setName("group2");
|
|
||||||
group2->setParent(group1);
|
|
||||||
|
|
||||||
Entry* entry3 = new Entry();
|
|
||||||
entry3->setTitle("entry3");
|
|
||||||
entry3->setGroup(group1);
|
|
||||||
|
|
||||||
Entry* entry43 = new Entry();
|
|
||||||
entry43->setTitle("entry43");
|
|
||||||
entry43->setGroup(group1);
|
|
||||||
|
|
||||||
Entry* google = new Entry();
|
|
||||||
google->setTitle("Google");
|
|
||||||
google->setGroup(group2);
|
|
||||||
|
|
||||||
QStringList results = db->rootGroup()->locate("entry");
|
|
||||||
QVERIFY(results.size() == 4);
|
|
||||||
QVERIFY(results.contains("/group1/entry43"));
|
|
||||||
|
|
||||||
results = db->rootGroup()->locate("entry1");
|
|
||||||
QVERIFY(results.size() == 1);
|
|
||||||
QVERIFY(results.contains("/entry1"));
|
|
||||||
|
|
||||||
results = db->rootGroup()->locate("Entry1");
|
|
||||||
QVERIFY(results.size() == 1);
|
|
||||||
QVERIFY(results.contains("/entry1"));
|
|
||||||
|
|
||||||
results = db->rootGroup()->locate("invalid");
|
|
||||||
QVERIFY(results.isEmpty());
|
|
||||||
|
|
||||||
results = db->rootGroup()->locate("google");
|
|
||||||
QVERIFY(results.size() == 1);
|
|
||||||
QVERIFY(results.contains("/group1/group2/Google"));
|
|
||||||
|
|
||||||
results = db->rootGroup()->locate("group1");
|
|
||||||
QVERIFY(results.size() == 3);
|
|
||||||
QVERIFY(results.contains("/group1/entry3"));
|
|
||||||
QVERIFY(results.contains("/group1/entry43"));
|
|
||||||
QVERIFY(results.contains("/group1/group2/Google"));
|
|
||||||
|
|
||||||
delete db;
|
|
||||||
}
|
|
||||||
|
|
||||||
void TestGroup::testAddEntryWithPath()
|
void TestGroup::testAddEntryWithPath()
|
||||||
{
|
{
|
||||||
Database* db = new Database();
|
Database* db = new Database();
|
||||||
|
|
|
@ -39,7 +39,6 @@ private slots:
|
||||||
void testFindEntry();
|
void testFindEntry();
|
||||||
void testFindGroupByPath();
|
void testFindGroupByPath();
|
||||||
void testPrint();
|
void testPrint();
|
||||||
void testLocate();
|
|
||||||
void testAddEntryWithPath();
|
void testAddEntryWithPath();
|
||||||
void testIsRecycled();
|
void testIsRecycled();
|
||||||
void testCopyDataFrom();
|
void testCopyDataFrom();
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue