2

我刚开始在 Mac 上使用 Python 中的Watchdog 库,并且正在做一些基本的测试以确保一切都像我预期的那样工作。不幸的是,它们不是——我似乎只能获得包含注册事件的文件的文件夹的路径,而不是文件本身的路径。

下面是一个简单的测试程序(对 Watchdog 提供的示例稍作修改),用于在注册事件时打印出事件类型、路径和时间。

import time
from watchdog.observers import Observer
from watchdog.events import LoggingEventHandler
from watchdog.events import FileSystemEventHandler

class TestEventHandler(FileSystemEventHandler):

def on_any_event(self, event):
    print("event noticed: " + event.event_type + 
                 " on file " + event.src_path + " at " + time.asctime())

if __name__ == "__main__":
    event_handler = TestEventHandler()
    observer = Observer()
    observer.schedule(event_handler, path='~/test', recursive=True)
    observer.start()
    try:
        while True:
            time.sleep(1)
    except KeyboardInterrupt:
        observer.stop()
    observer.join()

src_path 变量应该包含发生事件的文件的路径。

但是,在我的测试中,当我修改文件时, src_path 只打印包含该文件的文件夹的路径,而不是文件本身的路径。例如,当我修改文件moon.txt夹中的文件europa时,程序会打印以下输出:

event noticed: modified on file ~/test/europa at Mon Jul  8 15:32:07 2013

为了获得修改后文件的完整路径,我需要更改什么?

4

2 回答 2

3

问题解决了。事实证明,FSEvents在 OS X 中,仅返回文件修改事件的目录,让您自己扫描目录以找出修改了哪个文件。这在 Watchdog 文档中没有提到,尽管在文档中很容易找到FSEvents

为了获取文件的完整路径,我添加了以下代码片段(受此 StackOverflow 线程的启发)以在目录中查找最近修改的文件,以便在 event.src_path 返回目录时使用。

if(event.is_directory):
    files_in_dir = [event.src_path+"/"+f for f in os.listdir(event.src_path)]
    mod_file_path = max(files_in_dir, key=os.path.getmtime)

mod_file_path包含修改文件的完整路径。

于 2013-07-11T06:37:56.900 回答
1

感谢 ekl 提供您的解决方案。我只是偶然发现了同样的问题。但是,我曾经使用 PatternMatchingEventHandler,这需要对您的解决方案进行一些小改动:

  • FileSystemEventHandler 的子类
  • 创建一个pattern存储模式匹配的属性。这不像原来的那样灵活PatternMatchingEventHandler,但应该可以满足大多数需求,如果你想扩展它,无论如何你都会得到这个想法。

这是您必须放入FileSystemEventHandler子类中的代码:

def __init__(self, pattern='*'):
    super(MidiEventHandler, self).__init__()
    self.pattern = pattern


def on_modified(self, event):
    super(MidiEventHandler, self).on_modified(event)

    if event.is_directory:
        files_in_dir = [event.src_path+"/"+f for f in os.listdir(event.src_path)]
        if len(files_in_dir) > 0:
            modifiedFilename = max(files_in_dir, key=os.path.getmtime)
        else:
            return
    else:
        modifiedFilename = event.src_path

    if fnmatch.fnmatch(os.path.basename(modifiedFilename), self.pattern):
        print "Modified MIDI file: %s" % modifiedFilename

max()我更改的另一件事是在文件列表上运行之前检查目录是否为空。max()不适用于空列表。

于 2013-09-14T01:26:05.167 回答