| 
									
										
										
										
											2021-02-08 20:40:58 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2020, Itamar S. <itamar8910@gmail.com> | 
					
						
							|  |  |  |  * Copyright (c) 2021, the SerenityOS developers. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2021-02-08 20:40:58 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "FileWatcher.h"
 | 
					
						
							|  |  |  | #include <AK/Debug.h>
 | 
					
						
							|  |  |  | #include <AK/Function.h>
 | 
					
						
							|  |  |  | #include <AK/LexicalPath.h>
 | 
					
						
							|  |  |  | #include <AK/Noncopyable.h>
 | 
					
						
							|  |  |  | #include <AK/NonnullRefPtr.h>
 | 
					
						
							|  |  |  | #include <AK/RefCounted.h>
 | 
					
						
							|  |  |  | #include <AK/Result.h>
 | 
					
						
							|  |  |  | #include <AK/String.h>
 | 
					
						
							|  |  |  | #include <Kernel/API/InodeWatcherEvent.h>
 | 
					
						
							|  |  |  | #include <LibCore/DirIterator.h>
 | 
					
						
							|  |  |  | #include <LibCore/Notifier.h>
 | 
					
						
							|  |  |  | #include <fcntl.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <sys/stat.h>
 | 
					
						
							| 
									
										
										
										
											2021-03-12 17:29:37 +01:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2021-02-08 20:40:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Core { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Only supported in serenity mode because we use `watch_file`
 | 
					
						
							|  |  |  | #ifdef __serenity__
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static String get_child_path_from_inode_index(const String& path, unsigned child_inode_index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DirIterator iterator(path, Core::DirIterator::SkipDots); | 
					
						
							|  |  |  |     if (iterator.has_error()) { | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (iterator.has_next()) { | 
					
						
							| 
									
										
										
										
											2021-04-21 22:16:32 +02:00
										 |  |  |         auto child_full_path = iterator.next_full_path(); | 
					
						
							| 
									
										
										
										
											2021-02-08 20:40:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-21 22:16:32 +02:00
										 |  |  |         struct stat st = {}; | 
					
						
							|  |  |  |         if (lstat(child_full_path.characters(), &st) < 0) { | 
					
						
							| 
									
										
										
										
											2021-02-08 20:40:58 +01:00
										 |  |  |             return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (st.st_ino == child_inode_index) { | 
					
						
							|  |  |  |             return child_full_path; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | BlockingFileWatcher::BlockingFileWatcher(const String& path) | 
					
						
							|  |  |  |     : m_path(path) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_watcher_fd = watch_file(path.characters(), path.length()); | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |     VERIFY(m_watcher_fd != -1); | 
					
						
							| 
									
										
										
										
											2021-02-08 20:40:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | BlockingFileWatcher::~BlockingFileWatcher() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     close(m_watcher_fd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Optional<FileWatcherEvent> BlockingFileWatcher::wait_for_event() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     InodeWatcherEvent event {}; | 
					
						
							|  |  |  |     int rc = read(m_watcher_fd, &event, sizeof(event)); | 
					
						
							|  |  |  |     if (rc <= 0) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FileWatcherEvent result; | 
					
						
							|  |  |  |     if (event.type == InodeWatcherEvent::Type::ChildAdded) | 
					
						
							|  |  |  |         result.type = FileWatcherEvent::Type::ChildAdded; | 
					
						
							|  |  |  |     else if (event.type == InodeWatcherEvent::Type::ChildRemoved) | 
					
						
							|  |  |  |         result.type = FileWatcherEvent::Type::ChildRemoved; | 
					
						
							|  |  |  |     else if (event.type == InodeWatcherEvent::Type::Modified) | 
					
						
							|  |  |  |         result.type = FileWatcherEvent::Type::Modified; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (result.type == FileWatcherEvent::Type::ChildAdded || result.type == FileWatcherEvent::Type::ChildRemoved) { | 
					
						
							|  |  |  |         auto child_path = get_child_path_from_inode_index(m_path, event.inode_index); | 
					
						
							|  |  |  |         if (!LexicalPath(child_path).is_valid()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         result.child_path = child_path; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Result<NonnullRefPtr<FileWatcher>, String> FileWatcher::watch(const String& path) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto watch_fd = watch_file(path.characters(), path.length()); | 
					
						
							|  |  |  |     if (watch_fd < 0) { | 
					
						
							|  |  |  |         return String::formatted("Could not watch file '{}' : {}", path.characters(), strerror(errno)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fcntl(watch_fd, F_SETFD, FD_CLOEXEC); | 
					
						
							|  |  |  |     if (watch_fd < 0) { | 
					
						
							|  |  |  |         return String::formatted("Could not watch file '{}' : {}", path.characters(), strerror(errno)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dbgln_if(FILE_WATCHER_DEBUG, "Started watcher for file '{}'", path.characters()); | 
					
						
							|  |  |  |     auto notifier = Notifier::construct(watch_fd, Notifier::Event::Read); | 
					
						
							| 
									
										
										
										
											2021-04-23 16:46:57 +02:00
										 |  |  |     return adopt_ref(*new FileWatcher(move(notifier), move(path))); | 
					
						
							| 
									
										
										
										
											2021-02-08 20:40:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FileWatcher::FileWatcher(NonnullRefPtr<Notifier> notifier, const String& path) | 
					
						
							|  |  |  |     : m_notifier(move(notifier)) | 
					
						
							|  |  |  |     , m_path(path) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_notifier->on_ready_to_read = [this] { | 
					
						
							|  |  |  |         InodeWatcherEvent event {}; | 
					
						
							|  |  |  |         int rc = read(m_notifier->fd(), &event, sizeof(event)); | 
					
						
							|  |  |  |         if (rc <= 0) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         FileWatcherEvent result; | 
					
						
							|  |  |  |         if (event.type == InodeWatcherEvent::Type::ChildAdded) { | 
					
						
							|  |  |  |             result.type = FileWatcherEvent::Type::ChildAdded; | 
					
						
							|  |  |  |         } else if (event.type == InodeWatcherEvent::Type::ChildRemoved) { | 
					
						
							|  |  |  |             result.type = FileWatcherEvent::Type::ChildRemoved; | 
					
						
							|  |  |  |         } else if (event.type == InodeWatcherEvent::Type::Modified) { | 
					
						
							|  |  |  |             result.type = FileWatcherEvent::Type::Modified; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             warnln("Unknown event type {} returned by the watch_file descriptor for {}", (unsigned)event.type, m_path.characters()); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (result.type == FileWatcherEvent::Type::ChildAdded || result.type == FileWatcherEvent::Type::ChildRemoved) { | 
					
						
							|  |  |  |             auto child_path = get_child_path_from_inode_index(m_path, event.inode_index); | 
					
						
							|  |  |  |             if (!LexicalPath(child_path).is_valid()) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             result.child_path = child_path; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         on_change(result); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FileWatcher::~FileWatcher() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_notifier->on_ready_to_read = nullptr; | 
					
						
							|  |  |  |     close(m_notifier->fd()); | 
					
						
							|  |  |  |     dbgln_if(FILE_WATCHER_DEBUG, "Ended watcher for file '{}'", m_path.characters()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |