blob: b0efcdca510e0868348b09d5a78a5610c5fd7dbc [file] [log] [blame]
#!/usr/bin/env python
#
# Copyright 2007 Google Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
"""Monitors a directory tree for changes."""
import logging
import sys
import types
from google.appengine.tools.devappserver2 import inotify_file_watcher
from google.appengine.tools.devappserver2 import mtime_file_watcher
from google.appengine.tools.devappserver2 import win32_file_watcher
class _MultipleFileWatcher(object):
"""A FileWatcher that combines multiple file watchers.
For file watchers that can only watch a single directory, this class can
manage multiple watchers and allows the caller to treat them as a single
unit.
"""
def __init__(self, watchers):
"""Initialize a MultipleFileWatcher instance.
Args:
watchers: a list of watchers to treat as a single watcher.
"""
self._file_watchers = watchers
def start(self):
for watcher in self._file_watchers:
watcher.start()
def quit(self):
for watcher in self._file_watchers:
watcher.quit()
def has_changes(self):
# .has_changes() returns True if there has been any changes since the
# last call to .has_changes() so it must be called for every FileWatcher
# to prevent spurious change notifications on subsequent calls.
return any([watcher.has_changes() for watcher in self._file_watchers])
def _create_watcher(directories, watcher_class):
"""Creates the best watcher based on multiple directory support.
For file watchers that can support multiple directories, directly instantiate
an instance passing in an iterable of directories names. For file watchers
that only support single directories, instantiate one directly if there is
only a single directory to watch or wrap them in a MultipleFileWatcher if
there are multiple directories to watch.
Args:
directories: an iterable of all the directories to watch.
watcher_class: a callable that creates the per-directory FileWatcher
instance. Must be callable with a single item of the type held by
directories.
Returns:
A FileWatcher appropriate for the list of directories.
"""
if watcher_class.SUPPORTS_MULTIPLE_DIRECTORIES:
return watcher_class(directories)
elif len(directories) == 1:
return watcher_class(directories[0])
else:
return _MultipleFileWatcher([watcher_class(d) for d in directories])
def _create_linux_watcher(directories):
"""Create a watcher for Linux.
While we prefer InotifyFileWatcher for Linux, there are only a limited number
of inotify instances available per user (for example, 128 on a Goobuntu 12.04
install). Try to create an InotifyFileWatcher but fall back on
MTimeFileWatcher if the user is out of resources.
Args:
directories: A list representing the paths of the directories to monitor.
Returns:
An InotifyFileWatcher if the user has available resources and an
MtimeFileWatcher if not.
"""
# TODO: develop a way to check if the filesystem supports inotify.
# (for example, NFS does not) and also use MTimeFileWatcher in that case.
try:
return _create_watcher(directories,
inotify_file_watcher.InotifyFileWatcher)
except OSError as e:
logging.warning('Could not create InotifyFileWatcher;'
' falling back to MTimeFileWatcher: %s', e)
return _create_watcher(directories, mtime_file_watcher.MtimeFileWatcher)
def get_file_watcher(directories, use_mtime_file_watcher):
"""Returns an instance that monitors a hierarchy of directories.
Args:
directories: A list representing the paths of the directories to monitor.
use_mtime_file_watcher: A bool containing whether to use mtime polling to
monitor file changes even if other options are available on the current
platform.
Returns:
A FileWatcher appropriate for the current platform. start() must be called
before has_changes().
"""
assert not isinstance(directories, types.StringTypes), 'expected list got str'
if use_mtime_file_watcher:
return _create_watcher(directories, mtime_file_watcher.MtimeFileWatcher)
elif sys.platform.startswith('linux'):
return _create_linux_watcher(directories)
elif sys.platform.startswith('win'):
return _create_watcher(directories, win32_file_watcher.Win32FileWatcher)
else:
return _create_watcher(directories, mtime_file_watcher.MtimeFileWatcher)
# NOTE: The Darwin-specific watcher implementation (found in the deleted file
# fsevents_file_watcher.py) was incorrect - the Mac OS X FSEvents
# implementation does not detect changes in symlinked files or directories. It
# also does not provide file-level change precision before Mac OS 10.7.
#
# It is still possible to provide an efficient implementation by watching all
# symlinked directories and using mtime checking for symlinked files. On any
# change in a directory, it would have to be rescanned to see if a new
# symlinked file or directory was added. It also might be possible to use
# kevents instead of the Carbon API to detect files changes.