blob: d7b98ad7802338e0aa77bc6649f01f77e9343227 [file] [log] [blame]
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -08001/* -*- Mode: C++; c-file-style: "gnu"; indent-tabs-mode:nil -*- */
2/*
3 * Copyright (c) 2012-2013 University of California, Los Angeles
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation;
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17 *
18 * Author: Jared Lindblom <lindblom@cs.ucla.edu>
19 * Alexander Afanasyev <alexander.afanasyev@ucla.edu>
20 * Zhenkai Zhu <zhenkai@cs.ucla.edu>
21 */
22
Alexander Afanasyevf4cde4e2016-12-25 13:42:57 -080023#include "fs-watcher.hpp"
24#include "db-helper.hpp"
25#include "logging.hpp"
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -080026
27#include <boost/bind.hpp>
28
29#include <QDirIterator>
30#include <QRegExp>
31
32using namespace std;
33using namespace boost;
34
35INIT_LOGGER ("FsWatcher");
36
Alexander Afanasyev9ca444e2013-01-25 16:29:35 -080037FsWatcher::FsWatcher (QString dirPath,
38 LocalFile_Change_Callback onChange, LocalFile_Change_Callback onDelete,
39 QObject* parent)
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -080040 : QObject(parent)
41 , m_watcher (new QFileSystemWatcher())
Alexander Afanasyev583449a2013-01-28 17:04:06 -080042 , m_scheduler (new Scheduler ())
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -080043 , m_dirPath (dirPath)
Alexander Afanasyev9ca444e2013-01-25 16:29:35 -080044 , m_onChange (onChange)
45 , m_onDelete (onDelete)
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -080046{
47 _LOG_DEBUG ("Monitor dir: " << m_dirPath.toStdString ());
48 // add main directory to monitor
Alexander Afanasyev69e13172013-01-25 17:16:27 -080049
Zhenkai Zhud1756272013-02-01 17:02:18 -080050 initFileStateDb();
51
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -080052 m_watcher->addPath (m_dirPath);
53
54 // register signals (callback functions)
55 connect (m_watcher, SIGNAL (directoryChanged (QString)), this, SLOT (DidDirectoryChanged (QString)));
56 connect (m_watcher, SIGNAL (fileChanged (QString)), this, SLOT (DidFileChanged (QString)));
57
Alexander Afanasyev583449a2013-01-28 17:04:06 -080058 m_scheduler->start ();
59
Alexander Afanasyev7943c6b2013-01-29 18:43:24 -080060 Scheduler::scheduleOneTimeTask (m_scheduler, 0,
Zhenkai Zhud1756272013-02-01 17:02:18 -080061 bind (&FsWatcher::ScanDirectory_NotifyRemovals_Execute, this, m_dirPath),
Alexander Afanasyev7a647002013-01-30 11:54:52 -080062 "rescan-r-" + m_dirPath.toStdString ()); // only one task will be scheduled per directory
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -080063
Alexander Afanasyev7943c6b2013-01-29 18:43:24 -080064 Scheduler::scheduleOneTimeTask (m_scheduler, 0,
Zhenkai Zhud1756272013-02-01 17:02:18 -080065 bind (&FsWatcher::ScanDirectory_NotifyUpdates_Execute, this, m_dirPath),
Alexander Afanasyev7a647002013-01-30 11:54:52 -080066 "rescan-" +m_dirPath.toStdString ()); // only one task will be scheduled per directory
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -080067}
68
69FsWatcher::~FsWatcher()
70{
Alexander Afanasyev583449a2013-01-28 17:04:06 -080071 m_scheduler->shutdown ();
Zhenkai Zhud1756272013-02-01 17:02:18 -080072 sqlite3_close(m_db);
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -080073}
74
75void
76FsWatcher::DidDirectoryChanged (QString dirPath)
77{
Yingdi Yua264b872013-07-10 18:07:23 -070078 _LOG_DEBUG ("Triggered DirPath(DidDirectoryChanged): " << dirPath.toStdString ());
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -080079
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -080080 filesystem::path absPathTriggeredDir (dirPath.toStdString ());
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -080081 if (!filesystem::exists (filesystem::path (absPathTriggeredDir)))
82 {
83 Scheduler::scheduleOneTimeTask (m_scheduler, 0.5,
Zhenkai Zhud1756272013-02-01 17:02:18 -080084 bind (&FsWatcher::ScanDirectory_NotifyRemovals_Execute, this, dirPath),
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -080085 "r-" + dirPath.toStdString ()); // only one task will be scheduled per directory
86 }
87 else
88 {
89 // m_executor.execute (bind (&FsWatcher::ScanDirectory_NotifyUpdates_Execute, this, dirPath));
90 Scheduler::scheduleOneTimeTask (m_scheduler, 0.5,
Zhenkai Zhud1756272013-02-01 17:02:18 -080091 bind (&FsWatcher::ScanDirectory_NotifyUpdates_Execute, this, dirPath),
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -080092 dirPath.toStdString ()); // only one task will be scheduled per directory
Alexander Afanasyevc80207d2013-01-29 20:07:39 -080093
94 // m_executor.execute (bind (&FsWatcher::ScanDirectory_NotifyUpdates_Execute, this, dirPath));
95 Scheduler::scheduleOneTimeTask (m_scheduler, 300,
Zhenkai Zhud1756272013-02-01 17:02:18 -080096 bind (&FsWatcher::ScanDirectory_NotifyUpdates_Execute, this, dirPath),
Alexander Afanasyevc80207d2013-01-29 20:07:39 -080097 "rescan-"+dirPath.toStdString ()); // only one task will be scheduled per directory
Alexander Afanasyev7a647002013-01-30 11:54:52 -080098
99 Scheduler::scheduleOneTimeTask (m_scheduler, 300,
Zhenkai Zhud1756272013-02-01 17:02:18 -0800100 bind (&FsWatcher::ScanDirectory_NotifyRemovals_Execute, this, m_dirPath),
Alexander Afanasyev7a647002013-01-30 11:54:52 -0800101 "rescan-r-" + m_dirPath.toStdString ()); // only one task will be scheduled per directory
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -0800102 }
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800103}
104
105void
106FsWatcher::DidFileChanged (QString filePath)
107{
Alexander Afanasyev69e13172013-01-25 17:16:27 -0800108 if (!filePath.startsWith (m_dirPath))
109 {
Zhenkai Zhud1756272013-02-01 17:02:18 -0800110 _LOG_ERROR ("Got notification about a file not from the monitored directory: " << filePath.toStdString());
Alexander Afanasyev69e13172013-01-25 17:16:27 -0800111 return;
112 }
Alexander Afanasyeve70a2d82013-02-19 22:49:10 -0800113 QString absFilePath = filePath;
114
Alexander Afanasyev69e13172013-01-25 17:16:27 -0800115 filesystem::path absPathTriggeredFile (filePath.toStdString ());
116 filePath.remove (0, m_dirPath.size ());
117
118 filesystem::path triggeredFile (filePath.toStdString ());
119 if (filesystem::exists (filesystem::path (absPathTriggeredFile)))
120 {
121 _LOG_DEBUG ("Triggered UPDATE of file: " << triggeredFile.relative_path ().generic_string ());
Alexander Afanasyev583449a2013-01-28 17:04:06 -0800122 // m_onChange (triggeredFile.relative_path ());
123
Alexander Afanasyeve70a2d82013-02-19 22:49:10 -0800124 m_watcher->removePath (absFilePath);
125 m_watcher->addPath (absFilePath);
126
Yingdi Yu57f667b2013-07-11 10:37:59 -0700127 Scheduler::scheduleDelayOneTimeTask (m_scheduler, 0.5,
Alexander Afanasyev583449a2013-01-28 17:04:06 -0800128 bind (m_onChange, triggeredFile.relative_path ()),
129 triggeredFile.relative_path ().string());
Yingdi Yu57f667b2013-07-11 10:37:59 -0700130
Alexander Afanasyev69e13172013-01-25 17:16:27 -0800131 }
132 else
133 {
134 _LOG_DEBUG ("Triggered DELETE of file: " << triggeredFile.relative_path ().generic_string ());
Alexander Afanasyev583449a2013-01-28 17:04:06 -0800135 // m_onDelete (triggeredFile.relative_path ());
136
Alexander Afanasyeve70a2d82013-02-19 22:49:10 -0800137 m_watcher->removePath (absFilePath);
138
Zhenkai Zhu50746f62013-02-19 21:20:25 -0800139 deleteFile(triggeredFile.relative_path());
Alexander Afanasyev583449a2013-01-28 17:04:06 -0800140 Scheduler::scheduleOneTimeTask (m_scheduler, 0.5,
141 bind (m_onDelete, triggeredFile.relative_path ()),
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -0800142 "r-" + triggeredFile.relative_path ().string());
Alexander Afanasyev69e13172013-01-25 17:16:27 -0800143 }
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800144}
145
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800146void
Zhenkai Zhud1756272013-02-01 17:02:18 -0800147FsWatcher::ScanDirectory_NotifyUpdates_Execute (QString dirPath)
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800148{
Yingdi Yua264b872013-07-10 18:07:23 -0700149 _LOG_TRACE (" >> ScanDirectory_NotifyUpdates_Execute " << dirPath.toStdString());
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -0800150
Alexander Afanasyev69e13172013-01-25 17:16:27 -0800151 // exclude working only on last component, not the full path; iterating through all directories, even excluded from monitoring
152 QRegExp exclude ("^(\\.|\\.\\.|\\.chronoshare|.*~|.*\\.swp)$");
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800153
154 QDirIterator dirIterator (dirPath,
Alexander Afanasyev69e13172013-01-25 17:16:27 -0800155 QDir::Dirs | QDir::Files | /*QDir::Hidden |*/ QDir::NoSymLinks | QDir::NoDotAndDotDot,
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800156 QDirIterator::Subdirectories); // directory iterator (recursive)
157
158 // iterate through directory recursively
159 while (dirIterator.hasNext ())
160 {
161 dirIterator.next ();
162
163 // Get FileInfo
164 QFileInfo fileInfo = dirIterator.fileInfo ();
165
166 QString name = fileInfo.fileName ();
Alexander Afanasyev4a8781c2013-01-29 17:59:44 -0800167 _LOG_DEBUG ("+++ Scanning: " << name.toStdString ());
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800168
169 if (!exclude.exactMatch (name))
170 {
Alexander Afanasyev69e13172013-01-25 17:16:27 -0800171 _LOG_DEBUG ("Not excluded file/dir: " << fileInfo.absoluteFilePath ().toStdString ());
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800172 QString absFilePath = fileInfo.absoluteFilePath ();
173
174 // _LOG_DEBUG ("Attempt to add path to watcher: " << absFilePath.toStdString ());
Alexander Afanasyevc4942042013-02-19 13:54:25 -0800175 m_watcher->removePath (absFilePath);
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800176 m_watcher->addPath (absFilePath);
177
Alexander Afanasyevf695aed2013-01-30 13:28:42 -0800178 if (fileInfo.isFile ())
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800179 {
Alexander Afanasyevf695aed2013-01-30 13:28:42 -0800180 QString relFile = absFilePath;
181 relFile.remove (0, m_dirPath.size ());
182 filesystem::path aFile (relFile.toStdString ());
183
Zhenkai Zhud1756272013-02-01 17:02:18 -0800184 if (
185 //!m_fileState->LookupFile (aFile.relative_path ().generic_string ()) /* file does not exist there, but exists locally: added */)
Alexander Afanasyev89024ac2013-02-14 09:38:10 -0800186 !fileExists(aFile.relative_path()) /*file does not exist in db, but exists in fs: add */)
Alexander Afanasyevf695aed2013-01-30 13:28:42 -0800187 {
Alexander Afanasyev89024ac2013-02-14 09:38:10 -0800188 addFile(aFile.relative_path());
Alexander Afanasyevf695aed2013-01-30 13:28:42 -0800189 DidFileChanged (absFilePath);
190 }
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800191 }
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800192 }
193 else
194 {
195 // _LOG_DEBUG ("Excluded file/dir: " << fileInfo.filePath ().toStdString ());
196 }
197 }
198}
199
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -0800200
201void
Zhenkai Zhud1756272013-02-01 17:02:18 -0800202FsWatcher::ScanDirectory_NotifyRemovals_Execute (QString dirPath)
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -0800203{
204 _LOG_DEBUG ("Triggered DirPath: " << dirPath.toStdString ());
205
206 filesystem::path absPathTriggeredDir (dirPath.toStdString ());
207 dirPath.remove (0, m_dirPath.size ());
208
209 filesystem::path triggeredDir (dirPath.toStdString ());
210
Zhenkai Zhud1756272013-02-01 17:02:18 -0800211 /*
Alexander Afanasyev506ff222013-01-29 18:12:55 -0800212 FileItemsPtr files = m_fileState->LookupFilesInFolderRecursively (triggeredDir.relative_path ().generic_string ());
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -0800213 for (std::list<FileItem>::iterator file = files->begin (); file != files->end (); file ++)
214 {
215 filesystem::path testFile = filesystem::path (m_dirPath.toStdString ()) / file->filename ();
216 _LOG_DEBUG ("Check file for deletion [" << testFile.generic_string () << "]");
217
218 if (!filesystem::exists (testFile))
219 {
Alexander Afanasyev7a647002013-01-30 11:54:52 -0800220 if (removeIncomplete || file->is_complete ())
221 {
222 _LOG_DEBUG ("Notifying about removed file [" << file->filename () << "]");
223 m_onDelete (file->filename ());
224 }
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -0800225 }
226 }
Zhenkai Zhud1756272013-02-01 17:02:18 -0800227 */
228
229 vector<string> files;
Alexander Afanasyev89024ac2013-02-14 09:38:10 -0800230 getFilesInDir(triggeredDir.relative_path(), files);
Zhenkai Zhud1756272013-02-01 17:02:18 -0800231 for (vector<string>::iterator file = files.begin(); file != files.end(); file++)
232 {
Alexander Afanasyev89024ac2013-02-14 09:38:10 -0800233 filesystem::path targetFile = filesystem::path (m_dirPath.toStdString()) / *file;
Zhenkai Zhud1756272013-02-01 17:02:18 -0800234 if (!filesystem::exists (targetFile))
235 {
Alexander Afanasyev89024ac2013-02-14 09:38:10 -0800236 deleteFile(*file);
237 m_onDelete(*file);
Zhenkai Zhud1756272013-02-01 17:02:18 -0800238 }
239 }
Alexander Afanasyev0a30a0c2013-01-29 17:25:42 -0800240}
241
Zhenkai Zhud1756272013-02-01 17:02:18 -0800242const string INIT_DATABASE = "\
243CREATE TABLE IF NOT EXISTS \n\
244 Files( \n\
245 filename TEXT NOT NULL, \n\
246 PRIMARY KEY (filename) \n\
247); \n\
248CREATE INDEX filename_index ON Files (filename); \n\
249";
250
251void
252FsWatcher::initFileStateDb()
253{
Alexander Afanasyev89024ac2013-02-14 09:38:10 -0800254 filesystem::path dbFolder = filesystem::path (m_dirPath.toStdString()) / ".chronoshare" / "fs_watcher";
Zhenkai Zhud1756272013-02-01 17:02:18 -0800255 filesystem::create_directories(dbFolder);
256
Alexander Afanasyev89024ac2013-02-14 09:38:10 -0800257 int res = sqlite3_open((dbFolder / "filestate.db").string ().c_str (), &m_db);
Zhenkai Zhud1756272013-02-01 17:02:18 -0800258 if (res != SQLITE_OK)
259 {
260 BOOST_THROW_EXCEPTION(Error::Db() << errmsg_info_str("Cannot open database: " + (dbFolder / "filestate.db").string()));
261 }
262
263 char *errmsg = 0;
264 res = sqlite3_exec(m_db, INIT_DATABASE.c_str(), NULL, NULL, &errmsg);
265 if (res != SQLITE_OK && errmsg != 0)
266 {
267 // _LOG_TRACE ("Init \"error\": " << errmsg);
268 cout << "FS-Watcher DB error: " << errmsg << endl;
269 sqlite3_free (errmsg);
270 }
271}
272
273bool
274FsWatcher::fileExists(const filesystem::path &filename)
275{
276 sqlite3_stmt *stmt;
277 sqlite3_prepare_v2(m_db, "SELECT * FROM Files WHERE filename = ?;", -1, &stmt, 0);
278 sqlite3_bind_text(stmt, 1, filename.c_str(), -1, SQLITE_STATIC);
279 bool retval = false;
280 if (sqlite3_step (stmt) == SQLITE_ROW)
281 {
282 retval = true;
283 }
284 sqlite3_finalize(stmt);
285
286 return retval;
287}
288
289void
290FsWatcher::addFile(const filesystem::path &filename)
291{
292 sqlite3_stmt *stmt;
293 sqlite3_prepare_v2(m_db, "INSERT OR IGNORE INTO Files (filename) VALUES (?);", -1, &stmt, 0);
294 sqlite3_bind_text(stmt, 1, filename.c_str(), -1, SQLITE_STATIC);
295 sqlite3_step(stmt);
296 sqlite3_finalize(stmt);
297}
298
299void
300FsWatcher::deleteFile(const filesystem::path &filename)
301{
302 sqlite3_stmt *stmt;
303 sqlite3_prepare_v2(m_db, "DELETE FROM Files WHERE filename = ?;", -1, &stmt, 0);
304 sqlite3_bind_text(stmt, 1, filename.c_str(), -1, SQLITE_STATIC);
305 sqlite3_step(stmt);
306 sqlite3_finalize(stmt);
307}
308
309void
310FsWatcher::getFilesInDir(const filesystem::path &dir, vector<string> &files)
311{
312 sqlite3_stmt *stmt;
313 sqlite3_prepare_v2(m_db, "SELECT * FROM Files WHERE filename LIKE ?;", -1, &stmt, 0);
314
315 string dirStr = dir.string();
316 ostringstream escapedDir;
317 for (string::const_iterator ch = dirStr.begin (); ch != dirStr.end (); ch ++)
318 {
319 if (*ch == '%')
320 escapedDir << "\\%";
321 else
322 escapedDir << *ch;
323 }
324 escapedDir << "/" << "%";
325 string escapedDirStr = escapedDir.str ();
326 sqlite3_bind_text (stmt, 1, escapedDirStr.c_str (), escapedDirStr.size (), SQLITE_STATIC);
327
328 while(sqlite3_step(stmt) == SQLITE_ROW)
329 {
330 string filename(reinterpret_cast<const char *>(sqlite3_column_text (stmt, 0)), sqlite3_column_bytes(stmt, 0));
331 files.push_back(filename);
332 }
333
334 sqlite3_finalize (stmt);
335
336}
Alexander Afanasyev9e5a4702013-01-24 13:15:23 -0800337
338#if WAF
339#include "fs-watcher.moc"
340#include "fs-watcher.cc.moc"
341#endif