3

我正在尝试学习 python-watchdog,但我有点困惑为什么我设置的作业不止一次运行。所以,这是我的设置:

#handler.py
import os
from watchdog.events import FileSystemEventHandler
from actions import run_something

def getext(filename):
    return os.path.splitext(filename)[-1].lower()

class ChangeHandler(FileSystemEventHandler):

    def on_any_event(self, event):

        if event.is_directory:
            return
        if getext(event.src_path) == '.done':
            run_something()
        else: 
            print "event not directory.. exiting..."
            pass

观察者的设置如下:

#observer.py
import os
import time
from watchdog.observers import Observer
from handler import ChangeHandler

BASEDIR = "/path/to/some/directory/bin"

def main():

    while 1:

        event_handler = ChangeHandler()
        observer = Observer()
        observer.schedule(event_handler, BASEDIR, recursive=True)
        observer.start()
        try:
            while True:
                time.sleep(1)
        except KeyboardInterrupt:
            observer.stop()
        observer.join()

 if __name__ == '__main__':
    main()

最后,像这样的动作:

#actions.py
import os
import subprocess

def run_something():
    output = subprocess.check_output(['./run.sh'])
    print output
    return None

..where./run.sh只是我想在.done找到带有扩展名的文件时运行的 shell 脚本/path/to/some/directory/bin

#run.sh
#!/bin/bash
echo "Job Start: $(date)"
rm -rf /path/to/some/directory/bin/job.done # remove the .done file
echo "Job Done: $(date)"

但是,当我发出 apython observer.py然后执行 a touch job.doneon 时/path/to/some/directory/bin,我看到我的 shell 脚本./run.sh运行了 3 次而不是 1 次。

我很困惑为什么这会运行三次而不是一次(我确实删除了job.done我的 bash 脚本中的文件)

4

1 回答 1

6

要调试看门狗脚本,打印看门狗所看到的事件很有用。一个文件编辑或 CLI 命令,例如touch,可能会导致多个看门狗事件。例如,如果您插入一个打印语句:

class ChangeHandler(FileSystemEventHandler):

    def on_any_event(self, event):
        print(event)

记录每个事件,运行

% touch job.done

生成

2014-12-24 13:11:02 - <FileCreatedEvent: src_path='/home/unutbu/tmp/job.done'>
2014-12-24 13:11:02 - <DirModifiedEvent: src_path='/home/unutbu/tmp'>
2014-12-24 13:11:02 - <FileModifiedEvent: src_path='/home/unutbu/tmp/job.done'>

上面有两个以src_path结尾的事件job.done。因此,

    if getext(event.src_path) == '.done':
        run_something()

运行两次,因为有 aFileCreatedEvent和 a FileModifiedEvent。您最好只监视FileModifiedEvents。

于 2014-12-24T18:13:20.640 回答