blob: 686f0452f4c9a64e49c98ee3cca871f0c1c5eb1e (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
|
-- | Filesystem polling with an inotify backend. Works only on linux.
--
module Hakyll.Web.Preview.Poll
( previewPoll
) where
import Control.Monad (forM_, when)
import Data.Set (Set)
import qualified Data.Set as S
import System.FilePath (takeDirectory, (</>))
import Data.List (isPrefixOf)
import System.INotify
import Hakyll.Core.Configuration
import Hakyll.Core.ResourceProvider
import Hakyll.Core.Identifier
-- | Calls the given callback when the directory tree changes
--
previewPoll :: HakyllConfiguration -- ^ Configuration
-> Set Resource -- ^ Resources to watch
-> IO () -- ^ Action called when something changes
-> IO () -- ^ Can block forever
previewPoll _ resources callback = do
-- Initialize inotify
inotify <- initINotify
let -- A set of file paths
paths = S.map (toFilePath . unResource) resources
-- A list of directories. Run it through a set so we have every
-- directory only once.
directories = S.toList $ S.map (notEmpty . takeDirectory) paths
-- Problem: we can't add a watcher for "". So we make sure a directory
-- name is not empty
notEmpty "" = "."
notEmpty x = x
-- Execute the callback when path is known
ifResource path =
let path' = if "./" `isPrefixOf` path then drop 2 path else path
in when (path' `S.member` paths) callback
-- Add a watcher for every directory
forM_ directories $ \directory -> do
_ <- addWatch inotify [Modify] directory $ \e -> case e of
(Modified _ (Just p)) -> ifResource $ directory </> p
_ -> return ()
return ()
|