Compare commits
7 Commits
5272365e5a
...
a415643923
Author | SHA1 | Date | |
---|---|---|---|
a415643923 | |||
b8c7514142 | |||
35bcd35ec9 | |||
ba81551a2c | |||
d8a205fbe5 | |||
ed4228326e | |||
2bb26bc63c |
@ -15,6 +15,7 @@ DEFINES += QT_DEPRECATED_WARNINGS
|
|||||||
# You can also select to disable deprecated APIs only up to a certain version of Qt.
|
# You can also select to disable deprecated APIs only up to a certain version of Qt.
|
||||||
#DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
|
#DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
|
||||||
SOURCES += \
|
SOURCES += \
|
||||||
|
commandtag.cpp \
|
||||||
main.cpp \
|
main.cpp \
|
||||||
commandadd.cpp \
|
commandadd.cpp \
|
||||||
commanddelete.cpp \
|
commanddelete.cpp \
|
||||||
@ -27,6 +28,7 @@ HEADERS += \
|
|||||||
command.h \
|
command.h \
|
||||||
commandadd.h \
|
commandadd.h \
|
||||||
commanddelete.h \
|
commanddelete.h \
|
||||||
|
commandtag.h \
|
||||||
commandupdate.h \
|
commandupdate.h \
|
||||||
commandsearch.h \
|
commandsearch.h \
|
||||||
commandlist.h
|
commandlist.h
|
||||||
|
@ -2,7 +2,6 @@
|
|||||||
#include <QThread>
|
#include <QThread>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include "command.h"
|
#include "command.h"
|
||||||
#include "looqsgeneralexception.h"
|
|
||||||
|
|
||||||
void Command::execute()
|
void Command::execute()
|
||||||
{
|
{
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
#include <QCommandLineParser>
|
#include <QCommandLineParser>
|
||||||
#include "commandsearch.h"
|
#include "commandsearch.h"
|
||||||
#include "databasefactory.h"
|
|
||||||
#include "logger.h"
|
#include "logger.h"
|
||||||
|
|
||||||
int CommandSearch::handle(QStringList arguments)
|
int CommandSearch::handle(QStringList arguments)
|
||||||
|
65
cli/commandtag.cpp
Normal file
65
cli/commandtag.cpp
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
#include <QCommandLineParser>
|
||||||
|
#include "commandtag.h"
|
||||||
|
#include "logger.h"
|
||||||
|
|
||||||
|
int CommandTag::handle(QStringList arguments)
|
||||||
|
{
|
||||||
|
QCommandLineParser parser;
|
||||||
|
parser.addPositionalArgument("add", "Adds a tag to a file",
|
||||||
|
"add [tag] [paths...]. Adds the tag to the specified paths");
|
||||||
|
parser.addPositionalArgument("remove", "Removes a file associated to tag", "remove [tag] [file]");
|
||||||
|
parser.addPositionalArgument("delete", "Deletes a tag", "delete [tag]");
|
||||||
|
parser.addPositionalArgument("list", "Lists paths associated with a tag, or all tags", "list [tag]");
|
||||||
|
parser.addHelpOption();
|
||||||
|
|
||||||
|
parser.parse(arguments);
|
||||||
|
|
||||||
|
QStringList args = parser.positionalArguments();
|
||||||
|
if(args.length() == 0)
|
||||||
|
{
|
||||||
|
parser.showHelp(EXIT_FAILURE);
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString cmd = args[0];
|
||||||
|
qDebug() << cmd;
|
||||||
|
if(cmd == "add")
|
||||||
|
{
|
||||||
|
if(args.length() < 3)
|
||||||
|
{
|
||||||
|
Logger::error() << "Not enough arguments provided. 'add' requires a tag followed by at least one path"
|
||||||
|
<< Qt::endl;
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
QString tag = args[1];
|
||||||
|
auto paths = args.mid(2).toVector();
|
||||||
|
for(int i = 0; i < paths.size(); i++)
|
||||||
|
{
|
||||||
|
QFileInfo info{paths[i]};
|
||||||
|
if(!info.exists())
|
||||||
|
{
|
||||||
|
Logger::error() << "Can't add tag for file " + info.absoluteFilePath() + " because it does not exist"
|
||||||
|
<< Qt::endl;
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
QString absolutePath = info.absoluteFilePath();
|
||||||
|
if(!this->dbService->fileExistsInDatabase(absolutePath))
|
||||||
|
{
|
||||||
|
Logger::error() << "Only files that have been indexed can be tagged. File not in index: " + absolutePath
|
||||||
|
<< Qt::endl;
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
paths[i] = absolutePath;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool result = this->dbService->addTag(tag, paths);
|
||||||
|
if(!result)
|
||||||
|
{
|
||||||
|
Logger::error() << "Failed to assign tags" << Qt::endl;
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
return EXIT_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
13
cli/commandtag.h
Normal file
13
cli/commandtag.h
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
#ifndef COMMANDTAG_H
|
||||||
|
#define COMMANDTAG_H
|
||||||
|
#include "command.h"
|
||||||
|
|
||||||
|
class CommandTag : public Command
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
using Command::Command;
|
||||||
|
|
||||||
|
int handle(QStringList arguments) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // COMMANDTAG_H
|
@ -21,6 +21,7 @@
|
|||||||
#include "commandupdate.h"
|
#include "commandupdate.h"
|
||||||
#include "commandsearch.h"
|
#include "commandsearch.h"
|
||||||
#include "commandlist.h"
|
#include "commandlist.h"
|
||||||
|
#include "commandtag.h"
|
||||||
#include "databasefactory.h"
|
#include "databasefactory.h"
|
||||||
#include "logger.h"
|
#include "logger.h"
|
||||||
#include "sandboxedprocessor.h"
|
#include "sandboxedprocessor.h"
|
||||||
@ -31,7 +32,7 @@
|
|||||||
void printUsage(QString argv0)
|
void printUsage(QString argv0)
|
||||||
{
|
{
|
||||||
qInfo() << "Usage:" << argv0 << "command";
|
qInfo() << "Usage:" << argv0 << "command";
|
||||||
qInfo() << "Valid commands: add, update, delete, search, list. Each command has a --help option.";
|
qInfo() << "Valid commands: add, update, search, delete, tag, list. Each command has a --help option.";
|
||||||
}
|
}
|
||||||
|
|
||||||
Command *commandFromName(QString name, SqliteDbService &dbService)
|
Command *commandFromName(QString name, SqliteDbService &dbService)
|
||||||
@ -56,6 +57,10 @@ Command *commandFromName(QString name, SqliteDbService &dbService)
|
|||||||
{
|
{
|
||||||
return new CommandList(dbService);
|
return new CommandList(dbService);
|
||||||
}
|
}
|
||||||
|
if(name == "tag")
|
||||||
|
{
|
||||||
|
return new CommandTag(dbService);
|
||||||
|
}
|
||||||
|
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
@ -193,7 +193,7 @@ int main(int argc, char *argv[])
|
|||||||
Logger::error() << error << Qt::endl;
|
Logger::error() << error << Qt::endl;
|
||||||
QMessageBox::critical(nullptr, "Error during upgrade",
|
QMessageBox::critical(nullptr, "Error during upgrade",
|
||||||
error);
|
error);
|
||||||
qApp->quit();
|
exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
|
||||||
);
|
);
|
||||||
|
@ -877,6 +877,8 @@ void MainWindow::handleSearchResults(const QVector<SearchResult> &results)
|
|||||||
makePreviews(1);
|
makePreviews(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ui->tabWidget->setTabEnabled(1, previewDirty);
|
||||||
|
|
||||||
QString statusText = "Results: " + QString::number(results.size()) + " files";
|
QString statusText = "Results: " + QString::number(results.size()) + " files";
|
||||||
statusText += ", previewable: " + QString::number(this->previewCoordinator.previewableCount());
|
statusText += ", previewable: " + QString::number(this->previewCoordinator.previewableCount());
|
||||||
if(hasDeleted)
|
if(hasDeleted)
|
||||||
|
@ -146,10 +146,7 @@ SaveFileResult FileSaver::saveFile(const QFileInfo &fileInfo)
|
|||||||
if(mustFillContent)
|
if(mustFillContent)
|
||||||
{
|
{
|
||||||
auto filetype = this->dbService->queryFileType(fileInfo.absolutePath());
|
auto filetype = this->dbService->queryFileType(fileInfo.absolutePath());
|
||||||
if(filetype)
|
mustFillContent = !filetype.has_value() || filetype.value() == 'c';
|
||||||
{
|
|
||||||
mustFillContent = filetype.value() == 'c';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
6
shared/migrations/5.sql
Normal file
6
shared/migrations/5.sql
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
CREATE TABLE tag(id integer PRIMARY KEY, name varchar(128) UNIQUE);
|
||||||
|
CREATE TABLE filetag(fileid integer, tagid integer);
|
||||||
|
CREATE INDEX filetag_fileid ON filetag(fileid);
|
||||||
|
CREATE INDEX tag_id ON tag(id);
|
||||||
|
CREATE INDEX file_path ON file ( path );
|
||||||
|
UPDATE file SET filetype='c' WHERE filetype='f';
|
@ -4,5 +4,6 @@
|
|||||||
<file>2.sql</file>
|
<file>2.sql</file>
|
||||||
<file>3.sql</file>
|
<file>3.sql</file>
|
||||||
<file>4.sql</file>
|
<file>4.sql</file>
|
||||||
|
<file>5.sql</file>
|
||||||
</qresource>
|
</qresource>
|
||||||
</RCC>
|
</RCC>
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
#include <QFileInfo>
|
#include <QFileInfo>
|
||||||
#include <QDateTime>
|
#include <QDateTime>
|
||||||
#include <QSqlError>
|
#include <QSqlError>
|
||||||
|
#include "looqsgeneralexception.h"
|
||||||
#include "sqlitedbservice.h"
|
#include "sqlitedbservice.h"
|
||||||
#include "filedata.h"
|
#include "filedata.h"
|
||||||
#include "logger.h"
|
#include "logger.h"
|
||||||
@ -15,7 +16,7 @@ QVector<SearchResult> SqliteDbService::search(const LooqsQuery &query)
|
|||||||
|
|
||||||
std::optional<QChar> SqliteDbService::queryFileType(QString absPath)
|
std::optional<QChar> SqliteDbService::queryFileType(QString absPath)
|
||||||
{
|
{
|
||||||
auto query = exec("SELCET filetype FROM file WHERE path = ?", {absPath});
|
auto query = exec("SELECT filetype FROM file WHERE path = ?", {absPath});
|
||||||
if(!query.next())
|
if(!query.next())
|
||||||
{
|
{
|
||||||
return {};
|
return {};
|
||||||
@ -151,7 +152,7 @@ QSqlQuery SqliteDbService::exec(QString querystr, std::initializer_list<QVariant
|
|||||||
}
|
}
|
||||||
if(!query.exec())
|
if(!query.exec())
|
||||||
{
|
{
|
||||||
throw LooqsGeneralException("Error while exec(): " + query.lastError().text());
|
throw LooqsGeneralException("Error while exec(): " + query.lastError().text() + " for query: " + querystr);
|
||||||
}
|
}
|
||||||
return query;
|
return query;
|
||||||
}
|
}
|
||||||
@ -233,3 +234,56 @@ SaveFileResult SqliteDbService::saveFile(QFileInfo fileInfo, QVector<PageData> &
|
|||||||
}
|
}
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool SqliteDbService::addTag(QString tag, QString path)
|
||||||
|
{
|
||||||
|
QVector<QString> paths;
|
||||||
|
paths.append(path);
|
||||||
|
return addTag(tag, paths);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SqliteDbService::addTag(QString tag, const QVector<QString> &paths)
|
||||||
|
{
|
||||||
|
QSqlDatabase db = dbFactory->forCurrentThread();
|
||||||
|
QSqlQuery tagQuery(db);
|
||||||
|
QSqlQuery fileTagQuery(db);
|
||||||
|
|
||||||
|
tagQuery.prepare("INSERT OR IGNORE INTO tag (name) VALUES(?)");
|
||||||
|
tagQuery.addBindValue(tag);
|
||||||
|
|
||||||
|
fileTagQuery.prepare("INSERT INTO filetag(fileid, tagid) VALUES((SELECT id FROM file WHERE path = ?), (SELECT id "
|
||||||
|
"FROM tag WHERE name = ?))");
|
||||||
|
fileTagQuery.bindValue(1, tag);
|
||||||
|
if(!db.transaction())
|
||||||
|
{
|
||||||
|
Logger::error() << "Failed to open transaction to add paths for tag " << tag << " : " << db.lastError()
|
||||||
|
<< Qt::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if(!tagQuery.exec())
|
||||||
|
{
|
||||||
|
db.rollback();
|
||||||
|
Logger::error() << "Failed INSERT query" << tagQuery.lastError() << Qt::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for(const QString &path : paths)
|
||||||
|
{
|
||||||
|
fileTagQuery.bindValue(0, path);
|
||||||
|
if(!fileTagQuery.exec())
|
||||||
|
{
|
||||||
|
db.rollback();
|
||||||
|
Logger::error() << "Failed to add paths to tag" << Qt::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!db.commit())
|
||||||
|
{
|
||||||
|
db.rollback();
|
||||||
|
Logger::error() << "Failed to commit tag insertion transaction" << db.lastError() << Qt::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
@ -28,6 +28,8 @@ class SqliteDbService
|
|||||||
bool fileExistsInDatabase(QString path);
|
bool fileExistsInDatabase(QString path);
|
||||||
bool fileExistsInDatabase(QString path, qint64 mtime);
|
bool fileExistsInDatabase(QString path, qint64 mtime);
|
||||||
bool fileExistsInDatabase(QString path, qint64 mtime, QChar filetype);
|
bool fileExistsInDatabase(QString path, qint64 mtime, QChar filetype);
|
||||||
|
bool addTag(QString tag, QString path);
|
||||||
|
bool addTag(QString tag, const QVector<QString> &paths);
|
||||||
QVector<SearchResult> search(const LooqsQuery &query);
|
QVector<SearchResult> search(const LooqsQuery &query);
|
||||||
|
|
||||||
std::optional<QChar> queryFileType(QString absPath);
|
std::optional<QChar> queryFileType(QString absPath);
|
||||||
|
Loading…
Reference in New Issue
Block a user