Baloo

filecontentindexer.cpp
1/*
2 SPDX-FileCopyrightText: 2015 Vishesh Handa <vhanda@kde.org>
3
4 SPDX-License-Identifier: LGPL-2.1-or-later
5*/
6
7#include "filecontentindexer.h"
8#include "baloodebug.h"
9#include "config.h"
10#include "extractorprocess.h"
11#include "filecontentindexerprovider.h"
12#include "timeestimator.h"
13
14#include <QEventLoop>
15#include <QElapsedTimer>
16#include <QDBusConnection>
17
18using namespace Baloo;
19
20namespace {
21 // TODO KF6 -- remove/combine with started/finished DBus signal
22 void sendChangedSignal(const QStringList& updatedFiles)
23 {
24 auto message = QDBusMessage::createSignal(QStringLiteral("/files"),
25 QStringLiteral("org.kde"),
26 QStringLiteral("changed"));
27 message.setArguments({updatedFiles});
29 }
30}
31
32FileContentIndexer::FileContentIndexer(uint batchSize, //
33 FileContentIndexerProvider *provider,
34 TimeEstimator &timeEstimator,
35 QObject *parent)
36 : QObject(parent)
37 , m_batchSize(batchSize)
38 , m_provider(provider)
39 , m_stop(0)
40 , m_timeEstimator(timeEstimator)
41 , m_extractorPath(QStringLiteral(KDE_INSTALL_FULL_LIBEXECDIR_KF "/baloo_file_extractor"))
42{
43 Q_ASSERT(provider);
44
46 m_monitorWatcher.setConnection(bus);
47 m_monitorWatcher.setWatchMode(QDBusServiceWatcher::WatchForUnregistration);
48 connect(&m_monitorWatcher, &QDBusServiceWatcher::serviceUnregistered, this,
49 &FileContentIndexer::monitorClosed);
50
51 bus.registerObject(QStringLiteral("/fileindexer"),
53}
54
55void FileContentIndexer::run()
56{
57 ExtractorProcess process{m_extractorPath};
58 connect(&process, &ExtractorProcess::startedIndexingFile, this, &FileContentIndexer::slotStartedIndexingFile);
59 connect(&process, &ExtractorProcess::finishedIndexingFile, this, &FileContentIndexer::slotFinishedIndexingFile);
60 m_stop.storeRelaxed(false);
61 auto batchSize = m_batchSize;
62 uint finishedCount = 0;
63 uint totalCount = m_provider->size();
64
65 while (true) {
66 //
67 // WARNING: This will go mad, if the Extractor does not commit after N=m_batchSize files
68 // cause then we will keep fetching the same N files again and again.
69 //
70 QVector<quint64> idList = m_provider->fetch(batchSize);
71 if (idList.isEmpty() || m_stop.loadRelaxed()) {
72 break;
73 }
74 QStringList updatedFiles;
75 updatedFiles.reserve(idList.size());
76
77 QEventLoop loop;
78 connect(&process, &ExtractorProcess::done, &loop, &QEventLoop::quit);
79
80 bool hadErrors = false;
81 connect(&process, &ExtractorProcess::failed, &loop, [&hadErrors, &loop]() { hadErrors = true; loop.quit(); });
82
83 auto onFileFinished = [&updatedFiles, &finishedCount, &totalCount, this](const QString &filePath, bool updated) {
84 finishedCount++;
85 m_timeEstimator.setProgress(totalCount - finishedCount);
86 if (updated) {
87 updatedFiles.append(filePath);
88 }
89 };
90 connect(&process, &ExtractorProcess::finishedIndexingFile, &loop, onFileFinished, Qt::DirectConnection);
91
92 QElapsedTimer timer;
93 timer.start();
94
95 uint batchStartCount = finishedCount;
96 process.index(idList);
97 loop.exec();
98 batchSize = idList.size();
99
100 if (hadErrors && !m_stop.loadRelaxed()) {
101 if (batchSize == 1) {
102 auto failedId = idList.first();
103 auto ok = m_provider->markFailed(failedId);
104 if (!ok) {
105 qCCritical(BALOO) << "Not able to commit to DB, DB likely is in a bad state. Exiting";
106 exit(1);
107 }
108 batchSize = m_batchSize;
109 } else {
110 batchSize /= 2;
111 }
112 // reset to old value - nothing committed
113 finishedCount = batchStartCount;
114 m_timeEstimator.setProgress(totalCount - finishedCount);
115 process.start();
116 } else {
117 // Notify some metadata may have changed after a batch has been finished and committed
118 sendChangedSignal(updatedFiles);
119
120 // Update remaining time estimate
121 auto elapsed = timer.elapsed();
123 [this, elapsed, batchSize] { committedBatch(elapsed, batchSize); },
125 }
126 }
127 QMetaObject::invokeMethod(this, &FileContentIndexer::done, Qt::QueuedConnection);
128}
129
130void FileContentIndexer::slotStartedIndexingFile(const QString& filePath)
131{
132 m_currentFile = filePath;
133 if (!m_registeredMonitors.isEmpty()) {
134 Q_EMIT startedIndexingFile(filePath);
135 }
136}
137
138void FileContentIndexer::slotFinishedIndexingFile(const QString &filePath)
139{
140 m_currentFile = QString();
141 if (!m_registeredMonitors.isEmpty()) {
142 Q_EMIT finishedIndexingFile(filePath);
143 }
144}
145
146void FileContentIndexer::registerMonitor(const QDBusMessage& message)
147{
148 if (!m_registeredMonitors.contains(message.service())) {
149 m_registeredMonitors << message.service();
150 m_monitorWatcher.addWatchedService(message.service());
151 }
152}
153
154void FileContentIndexer::unregisterMonitor(const QDBusMessage& message)
155{
156 m_registeredMonitors.removeAll(message.service());
157 m_monitorWatcher.removeWatchedService(message.service());
158}
159
160void FileContentIndexer::monitorClosed(const QString& service)
161{
162 m_registeredMonitors.removeAll(service);
163 m_monitorWatcher.removeWatchedService(service);
164}
165
166#include "moc_filecontentindexer.cpp"
Implements storage for docIds without any associated data Instantiated for:
Definition coding.cpp:11
bool registerObject(const QString &path, QObject *object, RegisterOptions options)
bool send(const QDBusMessage &message) const const
QDBusConnection sessionBus()
QDBusMessage createSignal(const QString &path, const QString &interface, const QString &name)
QString service() const const
void serviceUnregistered(const QString &serviceName)
qint64 elapsed() const const
int exec(ProcessEventsFlags flags)
void quit()
void append(QList< T > &&value)
T & first()
bool isEmpty() const const
void reserve(qsizetype size)
qsizetype size() const const
bool invokeMethod(QObject *context, Functor &&function, FunctorReturnType *ret)
Q_EMITQ_EMIT
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
DirectConnection
QFuture< ArgsType< Signal > > connect(Sender *sender, Signal signal)
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 12:03:41 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.