Eliminated some dead code. In other cases, exported a currently unused function, since it was a logical part of the API. Of course this improves the API documentation. It may also sometimes let ghc optimize code better, since it can know a function is internal to a module. 364 modules still to go, according to git grep -E 'module [A-Za-z.]+ where'
		
			
				
	
	
		
			165 lines
		
	
	
	
		
			4.9 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
	
		
			4.9 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
{- generic directory watching interface
 | 
						|
 -
 | 
						|
 - Uses inotify, or kqueue, or fsevents, or win32-notify to watch a directory
 | 
						|
 - (and subdirectories) for changes, and runs hooks for different
 | 
						|
 - sorts of events as they occur.
 | 
						|
 -
 | 
						|
 - Copyright 2012-2013 Joey Hess <id@joeyh.name>
 | 
						|
 -
 | 
						|
 - License: BSD-2-clause
 | 
						|
 -}
 | 
						|
 | 
						|
{-# LANGUAGE CPP #-}
 | 
						|
 | 
						|
module Utility.DirWatcher (
 | 
						|
	canWatch,
 | 
						|
	eventsCoalesce,
 | 
						|
	closingTracked,
 | 
						|
	modifyTracked,
 | 
						|
	DirWatcherHandle,
 | 
						|
	watchDir,
 | 
						|
	stopWatchDir,
 | 
						|
) where
 | 
						|
 | 
						|
import Utility.DirWatcher.Types
 | 
						|
 | 
						|
#if WITH_INOTIFY
 | 
						|
import qualified Utility.DirWatcher.INotify as INotify
 | 
						|
import qualified System.INotify as INotify
 | 
						|
#endif
 | 
						|
#if WITH_KQUEUE
 | 
						|
import qualified Utility.DirWatcher.Kqueue as Kqueue
 | 
						|
import Control.Concurrent
 | 
						|
#endif
 | 
						|
#if WITH_FSEVENTS
 | 
						|
import qualified Utility.DirWatcher.FSEvents as FSEvents
 | 
						|
import qualified System.OSX.FSEvents as FSEvents
 | 
						|
#endif
 | 
						|
#if WITH_WIN32NOTIFY
 | 
						|
import qualified Utility.DirWatcher.Win32Notify as Win32Notify
 | 
						|
import qualified System.Win32.Notify as Win32Notify
 | 
						|
#endif
 | 
						|
 | 
						|
type Pruner = FilePath -> Bool
 | 
						|
 | 
						|
canWatch :: Bool
 | 
						|
#if (WITH_INOTIFY || WITH_KQUEUE || WITH_FSEVENTS || WITH_WIN32NOTIFY)
 | 
						|
canWatch = True
 | 
						|
#else
 | 
						|
#if defined linux_HOST_OS
 | 
						|
#warning "Building without inotify support"
 | 
						|
#endif
 | 
						|
canWatch = False
 | 
						|
#endif
 | 
						|
 | 
						|
{- With inotify, discrete events will be received when making multiple changes
 | 
						|
 - to the same filename. For example, adding it, deleting it, and adding it
 | 
						|
 - again will be three events.
 | 
						|
 - 
 | 
						|
 - OTOH, with kqueue, often only one event is received, indicating the most
 | 
						|
 - recent state of the file. -}
 | 
						|
eventsCoalesce :: Bool
 | 
						|
#if (WITH_INOTIFY || WITH_WIN32NOTIFY)
 | 
						|
eventsCoalesce = False
 | 
						|
#else
 | 
						|
#if (WITH_KQUEUE || WITH_FSEVENTS)
 | 
						|
eventsCoalesce = True
 | 
						|
#else
 | 
						|
eventsCoalesce = error "eventsCoalesce not defined"
 | 
						|
#endif
 | 
						|
#endif
 | 
						|
 | 
						|
{- With inotify, file closing is tracked to some extent, so an add event
 | 
						|
 - will always be received for a file once its writer closes it, and
 | 
						|
 - (typically) not before. This may mean multiple add events for the same file.
 | 
						|
 -
 | 
						|
 - OTOH, with kqueue, add events will often be received while a file is
 | 
						|
 - still being written to, and then no add event will be received once the
 | 
						|
 - writer closes it.
 | 
						|
 - 
 | 
						|
 - fsevents sometimes behaves similarly, but has sometimes been 
 | 
						|
 - seen to behave like kqueue. -}
 | 
						|
closingTracked :: Bool
 | 
						|
#if (WITH_INOTIFY || WITH_WIN32NOTIFY)
 | 
						|
closingTracked = True
 | 
						|
#else
 | 
						|
#if (WITH_KQUEUE || WITH_FSEVENTS)
 | 
						|
closingTracked = False
 | 
						|
#else
 | 
						|
closingTracked = error "closingTracked not defined"
 | 
						|
#endif
 | 
						|
#endif
 | 
						|
 | 
						|
{- With inotify, modifications to existing files can be tracked.
 | 
						|
 - Kqueue does not support this.
 | 
						|
 - Fsevents generates events when an existing file is reopened and rewritten,
 | 
						|
 - but not necessarily when it's opened once and modified repeatedly. -}
 | 
						|
modifyTracked :: Bool
 | 
						|
#if (WITH_INOTIFY || WITH_FSEVENTS || WITH_WIN32NOTIFY)
 | 
						|
modifyTracked = True
 | 
						|
#else
 | 
						|
#if WITH_KQUEUE
 | 
						|
modifyTracked = False
 | 
						|
#else
 | 
						|
modifyTracked = error "modifyTracked not defined"
 | 
						|
#endif
 | 
						|
#endif
 | 
						|
 | 
						|
{- Starts a watcher thread. The runstartup action is passed a scanner action
 | 
						|
 - to run, that will return once the initial directory scan is complete.
 | 
						|
 - Once runstartup returns, the watcher thread continues running,
 | 
						|
 - and processing events. Returns a DirWatcherHandle that can be used
 | 
						|
 - to shutdown later. -}
 | 
						|
#if WITH_INOTIFY
 | 
						|
type DirWatcherHandle = INotify.INotify
 | 
						|
watchDir :: FilePath -> Pruner -> Bool -> WatchHooks -> (IO () -> IO ()) -> IO DirWatcherHandle
 | 
						|
watchDir dir prune scanevents hooks runstartup = do
 | 
						|
	i <- INotify.initINotify
 | 
						|
	runstartup $ INotify.watchDir i dir prune scanevents hooks
 | 
						|
	return i
 | 
						|
#else
 | 
						|
#if WITH_KQUEUE
 | 
						|
type DirWatcherHandle = ThreadId
 | 
						|
watchDir :: FilePath -> Pruner -> Bool -> WatchHooks -> (IO Kqueue.Kqueue -> IO Kqueue.Kqueue) -> IO DirWatcherHandle
 | 
						|
watchDir dir prune _scanevents hooks runstartup = do
 | 
						|
	kq <- runstartup $ Kqueue.initKqueue dir prune
 | 
						|
	forkIO $ Kqueue.runHooks kq hooks
 | 
						|
#else
 | 
						|
#if WITH_FSEVENTS
 | 
						|
type DirWatcherHandle = FSEvents.EventStream
 | 
						|
watchDir :: FilePath -> Pruner -> Bool -> WatchHooks -> (IO FSEvents.EventStream -> IO FSEvents.EventStream) -> IO DirWatcherHandle
 | 
						|
watchDir dir prune scanevents hooks runstartup =
 | 
						|
	runstartup $ FSEvents.watchDir dir prune scanevents hooks
 | 
						|
#else
 | 
						|
#if WITH_WIN32NOTIFY
 | 
						|
type DirWatcherHandle = Win32Notify.WatchManager
 | 
						|
watchDir :: FilePath -> Pruner -> Bool -> WatchHooks -> (IO Win32Notify.WatchManager -> IO Win32Notify.WatchManager) -> IO DirWatcherHandle
 | 
						|
watchDir dir prune scanevents hooks runstartup =
 | 
						|
	runstartup $ Win32Notify.watchDir dir prune scanevents hooks
 | 
						|
#else
 | 
						|
type DirWatcherHandle = ()
 | 
						|
watchDir :: FilePath -> Pruner -> Bool -> WatchHooks -> (IO () -> IO ()) -> IO DirWatcherHandle
 | 
						|
watchDir = error "watchDir not defined"
 | 
						|
#endif
 | 
						|
#endif
 | 
						|
#endif
 | 
						|
#endif
 | 
						|
 | 
						|
stopWatchDir :: DirWatcherHandle -> IO ()
 | 
						|
#if WITH_INOTIFY
 | 
						|
stopWatchDir = INotify.killINotify
 | 
						|
#else
 | 
						|
#if WITH_KQUEUE
 | 
						|
stopWatchDir = killThread
 | 
						|
#else
 | 
						|
#if WITH_FSEVENTS
 | 
						|
stopWatchDir = FSEvents.eventStreamDestroy
 | 
						|
#else
 | 
						|
#if WITH_WIN32NOTIFY
 | 
						|
stopWatchDir = Win32Notify.killWatchManager
 | 
						|
#else
 | 
						|
stopWatchDir = error "stopWatchDir not defined"
 | 
						|
#endif
 | 
						|
#endif
 | 
						|
#endif
 | 
						|
#endif
 |