7#include "fileindexscheduler.h"
10#include "firstrunindexer.h"
11#include "newfileindexer.h"
12#include "modifiedfileindexer.h"
13#include "xattrindexer.h"
14#include "filecontentindexer.h"
15#include "unindexedfileindexer.h"
16#include "indexcleaner.h"
18#include "fileindexerconfig.h"
22#include <QDBusConnection>
31 , m_contentIndexer(nullptr)
32 , m_indexerState(Startup)
33 , m_checkUnindexedFiles(false)
34 , m_checkStaleIndexEntries(false)
35 , m_isGoingIdle(false)
36 , m_isSuspended(false)
37 , m_isFirstRun(firstRun)
43 m_threadPool.setMaxThreadCount(1);
45 connect(&m_powerMonitor, &PowerStateMonitor::powerManagementStatusChanged,
46 this, &FileIndexScheduler::powerManagementStatusChanged);
48 if (m_powerMonitor.isOnBattery()) {
49 m_indexerState = LowPowerIdle;
52 m_contentIndexer =
new FileContentIndexer(m_config->maxUncomittedFiles(), &m_provider, m_timeEstimator,
this);
53 m_contentIndexer->setAutoDelete(
false);
54 connect(m_contentIndexer, &FileContentIndexer::done,
this,
55 &FileIndexScheduler::runnerFinished);
56 connect(m_contentIndexer, &FileContentIndexer::committedBatch, [
this](uint time, uint batchSize) {
57 this->m_timeEstimator.handleNewBatchTime(time, batchSize);
64FileIndexScheduler::~FileIndexScheduler()
66 m_contentIndexer->quit();
67 m_threadPool.waitForDone(0);
70void FileIndexScheduler::startupFinished() {
75void FileIndexScheduler::scheduleIndexing()
77 if (!isIndexerIdle()) {
80 m_isGoingIdle =
false;
83 if (m_indexerState != Suspended) {
84 m_indexerState = Suspended;
85 Q_EMIT stateChanged(m_indexerState);
96 auto runnable =
new FirstRunIndexer(m_db, m_config, m_config->includeFolders());
97 connect(runnable, &FirstRunIndexer::done,
this, &FileIndexScheduler::runnerFinished);
99 m_threadPool.start(runnable);
100 m_indexerState = FirstRun;
101 Q_EMIT stateChanged(m_indexerState);
105 if (!m_newFiles.isEmpty()) {
106 auto runnable =
new NewFileIndexer(m_db, m_config, m_newFiles);
107 connect(runnable, &NewFileIndexer::done,
this, &FileIndexScheduler::runnerFinished);
109 m_threadPool.start(runnable);
111 m_indexerState = NewFiles;
112 Q_EMIT stateChanged(m_indexerState);
116 if (!m_modifiedFiles.isEmpty()) {
117 auto runnable =
new ModifiedFileIndexer(m_db, m_config, m_modifiedFiles);
118 connect(runnable, &ModifiedFileIndexer::done,
this, &FileIndexScheduler::runnerFinished);
120 m_threadPool.start(runnable);
121 m_modifiedFiles.clear();
122 m_indexerState = ModifiedFiles;
123 Q_EMIT stateChanged(m_indexerState);
127 if (!m_xattrFiles.isEmpty()) {
128 auto runnable =
new XAttrIndexer(m_db, m_config, m_xattrFiles);
129 connect(runnable, &XAttrIndexer::done,
this, &FileIndexScheduler::runnerFinished);
131 m_threadPool.start(runnable);
132 m_xattrFiles.clear();
133 m_indexerState = XAttrFiles;
134 Q_EMIT stateChanged(m_indexerState);
139 if (m_powerMonitor.isOnBattery()) {
140 if (m_indexerState != LowPowerIdle) {
141 m_indexerState = LowPowerIdle;
142 Q_EMIT stateChanged(m_indexerState);
148 if (m_indexerState != Startup) {
149 m_indexerState = Startup;
150 Q_EMIT stateChanged(m_indexerState);
157 if (m_checkStaleIndexEntries) {
158 auto runnable =
new IndexCleaner(m_db, m_config);
159 connect(runnable, &IndexCleaner::done,
this, &FileIndexScheduler::runnerFinished);
161 m_threadPool.start(runnable);
162 m_checkStaleIndexEntries =
false;
163 m_indexerState = StaleIndexEntriesClean;
164 Q_EMIT stateChanged(m_indexerState);
168 if (
auto remainingCount = m_provider.size(); remainingCount > 0) {
169 m_timeEstimator.setProgress(remainingCount);
170 m_threadPool.start(m_contentIndexer);
171 m_indexerState = ContentIndexing;
172 Q_EMIT stateChanged(m_indexerState);
176 if (m_checkUnindexedFiles) {
177 auto runnable =
new UnindexedFileIndexer(m_db, m_config);
178 connect(runnable, &UnindexedFileIndexer::done,
this, &FileIndexScheduler::runnerFinished);
180 m_threadPool.start(runnable);
181 m_checkUnindexedFiles =
false;
182 m_indexerState = UnindexedFileCheck;
183 Q_EMIT stateChanged(m_indexerState);
187 if (m_indexerState != Idle) {
188 m_indexerState = Idle;
189 Q_EMIT stateChanged(m_indexerState);
193static void removeStartsWith(QStringList& list,
const QString& dir)
196 [&dir](
const QString& file) {
197 return file.startsWith(dir);
205 [config](
const QString& file) {
206 return !config->shouldBeIndexed(file);
211void FileIndexScheduler::updateConfig()
215 if (m_indexerState == ContentIndexing) {
216 m_contentIndexer->quit();
218 removeShouldNotIndex(m_newFiles, m_config);
219 removeShouldNotIndex(m_modifiedFiles, m_config);
220 removeShouldNotIndex(m_xattrFiles, m_config);
221 m_checkStaleIndexEntries =
true;
222 m_checkUnindexedFiles =
true;
226void FileIndexScheduler::handleFileRemoved(
const QString& file)
228 if (!file.
endsWith(QLatin1Char(
'/'))) {
229 m_newFiles.removeOne(file);
230 m_modifiedFiles.removeOne(file);
231 m_xattrFiles.removeOne(file);
234 removeStartsWith(m_newFiles, file);
235 removeStartsWith(m_modifiedFiles, file);
236 removeStartsWith(m_xattrFiles, file);
240void FileIndexScheduler::powerManagementStatusChanged(
bool isOnBattery)
242 qCDebug(BALOO) <<
"Power state changed - onBattery:" << isOnBattery;
243 if (isOnBattery && m_indexerState == ContentIndexing) {
244 qCDebug(BALOO) <<
"On battery, stopping content indexer";
245 m_contentIndexer->quit();
251void FileIndexScheduler::setSuspend(
bool suspend)
253 m_isSuspended = suspend;
255 qCDebug(BALOO) <<
"Suspending";
256 if (m_indexerState == ContentIndexing) {
257 m_contentIndexer->quit();
262 qCDebug(BALOO) <<
"Resuming";
268uint FileIndexScheduler::getRemainingTime()
270 if (m_indexerState != ContentIndexing) {
273 return m_timeEstimator.calculateTimeLeft();
276void FileIndexScheduler::scheduleCheckUnindexedFiles()
278 m_checkUnindexedFiles =
true;
281void FileIndexScheduler::checkUnindexedFiles()
283 m_checkUnindexedFiles =
true;
287void FileIndexScheduler::scheduleCheckStaleIndexEntries()
289 m_checkStaleIndexEntries =
true;
292void FileIndexScheduler::checkStaleIndexEntries()
294 m_checkStaleIndexEntries =
true;
298uint FileIndexScheduler::getBatchSize()
300 return m_config->maxUncomittedFiles();
303#include "moc_fileindexscheduler.cpp"
Active config class which emits signals if the config was changed, for example if the KCM saved the c...
Implements storage for docIds without any associated data Instantiated for:
KIOCORE_EXPORT QStringList list(const QString &fileClass)
bool registerObject(const QString &path, QObject *object, RegisterOptions options)
QDBusConnection sessionBus()
iterator erase(const_iterator begin, const_iterator end)
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
bool endsWith(QChar c, Qt::CaseSensitivity cs) const const
QFuture< ArgsType< Signal > > connect(Sender *sender, Signal signal)