haskellbuildpandocshake-build-system

Haskell Shake with Twitch?


I'm switching (or trying to) from the brilliant tup to haskell shake as my build system..

Only I can't figure out how to get shake to rebuild files on changes.

I could of course use inotify or a wrapper like filewatcher or even watchman.

Since I'm using shake though, I was wondering how to integrate with twitch which shares the do syntax, but otherwise doesn't provide much in way of documentation..

The ultimate goal is to use pandoc for multi format documents.

The only reason tup was inadequate was because it doesn't support targets.


Solution

  • First of all, you should to write your own shake build rules. Then, when some source file will be changed, you should to run your build rules to produce your targets.

    Like this:

    main = defaultMain $ do
      "src/*.md" |> const build
    
    build = shakeArgs shake{shakeFiles="out"} $ do
        want ["out/foo.html", "out/foo.pdf"]
    
        "out/*.html" %> \out -> do
            let src = "src" </> dropDirectory1 out -<.> "md"
            cmd_ "pandoc -o" [out] src
    
        "out/*.pdf" %> \out -> do
            let src = "src" </> dropDirectory1 out -<.> "md"
            cmd_ "pandoc -o" [out] src
    

    When a markdown file in src directory will be changed, then out/foo.html and out/foo.pdf will be updated.

    If you want to optimize work of shake then you can do like this:

    main = defaultMain $ do
      "src/*.md" |> build . dependentTargets
    
    build targets = shakeArgs shake{shakeFiles="out"} $ do
        want targets
        ...
    
    dependentTargets src
        | "*.md" ?== src = ["out/foo.html", "out/foo.pdf"]
        | otherwise = []
    

    The package twitch recommends to use extension OverloadedStrings for compile code like this:

    "src/*.md" |> ...
    

    But this leads to ambiguous code in other parts of the program. For fix that, you can explicitly converting String to Dep like this:

    import Data.String
    
    fromString "src/*.md" |> ...
    

    You can improve this code by redefining the (|>) operator:

    import Data.String
    import Twitch hiding ((|>))
    
    pattern |> callback = addModify callback $ fromString pattern
    
    "src/*.md" |> ...