Skip to content

Commit

Permalink
Fix #5925 Add file-watch-hook for post-processing
Browse files Browse the repository at this point in the history
  • Loading branch information
mpilgrem committed Jun 3, 2024
1 parent 6db66dc commit 4f6fe19
Show file tree
Hide file tree
Showing 8 changed files with 169 additions and 25 deletions.
3 changes: 3 additions & 0 deletions ChangeLog.md
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,9 @@ Other enhancements:
* Add the `ls globals` command to list all global packages for the version of
GHC specified by the snapshot.
* Add `stack -h` (equivalent to `stack --help`).
* In YAML configuration files, the `file-watch-hook` key is introduced to allow
`--file-watch` post-processing to be customised with a executable or `sh`
shell script.

Bug fixes:

Expand Down
46 changes: 46 additions & 0 deletions doc/yaml_configuration.md
Original file line number Diff line number Diff line change
Expand Up @@ -1091,6 +1091,21 @@ Other paths added by Stack - things like the project's binary directory and the
compiler's binary directory - will take precedence over those specified here
(the automatic paths get prepended).

### file-watch-hook

:octicons-tag-24: UNRELEASED

Specifies the location of an executable or `sh` shell script to be run after
each attempted build with
[`build --file-watch`](build_command.md#-file-watch-flag). An absolute or
relative path can be specified. Changes to the configuration after the
initial `build --file-watch` command are ignored.

If the project-level configuration is provided in the `global-project` directory
in the [Stack root](stack_root.md), a relative path is assumed to be relative to
the current directory. Otherwise, it is assumed to be relative to the directory
of the project-level configuration file.

### ghc-build

[:octicons-tag-24: 1.3.0](https://github.com/commercialhaskell/stack/releases/tag/v1.3.0)
Expand Down Expand Up @@ -2118,3 +2133,34 @@ case $HOOK_GHC_TYPE in
;;
esac
~~~

### `--file-watch` post-processing

:octicons-tag-24: UNRELEASED

On Unix-like operating systems and Windows, Stack's `build --file-watch`
post-processing can be fully customised by specifying an executable or a `sh`
shell script (a 'hook') using the [`file-watch-hook`](#file-watch-hook)
configuration option. On Unix-like operating systems, the script file must be
made executable. A script is run by the `sh` application (which is provided by
MSYS2 on Windows).

The following environment variables are always available to the executable or script:

* `HOOK_FW_RESULT` (Equal to `""` if the build did not fail. Equal to the result
of `displayException e`, if exception `e` thown during the build.)

An example script is:

~~~sh
#!/bin/sh

set -eu

if [ -z "$HOOK_FW_RESULT" ]; then
echo "Success! Waiting for next file change."
else
echo "Build failed with exception:"
echo $HOOK_FW_RESULT
fi
~~~
11 changes: 8 additions & 3 deletions src/Stack/Build.hs
Original file line number Diff line number Diff line change
Expand Up @@ -111,15 +111,20 @@ buildCmd opts = do
prettyThrowIO GHCProfOptionInvalid
local (over globalOptsL modifyGO) $
case opts.fileWatch of
FileWatchPoll -> fileWatchPoll (inner . Just)
FileWatch -> fileWatch (inner . Just)
FileWatchPoll -> withFileWatchHook fileWatchPoll
FileWatch -> withFileWatchHook fileWatch
NoFileWatch -> inner Nothing
where
withFileWatchHook fileWatchAction =
-- This loads the full configuration in order to obtain the file-watch-hook
-- setting. That is likely not the most efficient approach.
withConfig YesReexec $ withEnvConfig NeedTargets opts $
fileWatchAction (inner . Just)
inner ::
Maybe (Set (Path Abs File) -> IO ())
-> RIO Runner ()
inner setLocalFiles = withConfig YesReexec $ withEnvConfig NeedTargets opts $
Stack.Build.build setLocalFiles
Stack.Build.build setLocalFiles
-- Read the build command from the CLI and enable it to run
modifyGO =
case opts.command of
Expand Down
20 changes: 19 additions & 1 deletion src/Stack/Config.hs
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,7 @@ import Path.IO
( XdgDirectory (..), canonicalizePath, doesDirExist
, doesFileExist, ensureDir, forgivingAbsence
, getAppUserDataDir, getCurrentDir, getXdgDir, resolveDir
, resolveDir', resolveFile'
, resolveDir', resolveFile, resolveFile'
)
import RIO.List ( unzip )
import RIO.Process
Expand Down Expand Up @@ -429,6 +429,23 @@ configFromConfigMonoid
-- resolveDirMaybe.
`catchAny`
const (throwIO (NoSuchDirectory userPath))
fileWatchHook <-
case getFirst configMonoid.fileWatchHook of
Nothing -> pure Nothing
Just userPath ->
( case mproject of
-- Not in a project
Nothing -> Just <$> resolveFile' userPath
-- Resolves to the project dir and appends the user path if it is
-- relative
Just (_, configYaml) ->
Just <$> resolveFile (parent configYaml) userPath
)
-- TODO: Either catch specific exceptions or add a
-- parseRelAsAbsFileMaybe utility and use it along with
-- resolveFileMaybe.
`catchAny`
const (throwIO (NoSuchFile userPath))
jobs <-
case getFirst configMonoid.jobs of
Nothing -> liftIO getNumProcessors
Expand Down Expand Up @@ -587,6 +604,7 @@ configFromConfigMonoid
, compilerCheck
, compilerRepository
, localBin
, fileWatchHook
, requireStackVersion
, jobs
, overrideGccPath
Expand Down
98 changes: 77 additions & 21 deletions src/Stack/FileWatch.hs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
{-# LANGUAGE NoImplicitPrelude #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE NoImplicitPrelude #-}
{-# LANGUAGE OverloadedRecordDot #-}
{-# LANGUAGE OverloadedStrings #-}

module Stack.FileWatch
( WatchMode (WatchModePoll)
Expand All @@ -11,25 +12,34 @@ import Control.Concurrent.STM ( check )
import qualified Data.Map.Merge.Strict as Map
import qualified Data.Map.Strict as Map
import qualified Data.Set as Set
import qualified Data.Text as T
import GHC.IO.Exception
( IOErrorType (InvalidArgument), IOException (..) )
import Path ( parent )
import Path ( fileExtension, parent )
import Path.IO ( doesFileExist, executable, getPermissions )
import RIO.Process
( EnvVars, HasProcessContext (..), proc, runProcess
, withModifyEnvVars
)
import System.Permissions ( osIsWindows )
import Stack.Prelude
import Stack.Types.Config ( Config (..), HasConfig (..) )
import Stack.Types.Runner ( HasRunner (..), Runner (..) )
import System.FSNotify
( WatchConfig, WatchMode (..), confWatchMode, defaultConfig
, eventPath, watchDir, withManagerConf
)
import System.IO ( getLine )

fileWatch ::
HasTerm env
=> ((Set (Path Abs File) -> IO ()) -> RIO env ())
(HasConfig env, HasTerm env)
=> ((Set (Path Abs File) -> IO ()) -> RIO Runner ())
-> RIO env ()
fileWatch = fileWatchConf defaultConfig

fileWatchPoll ::
HasTerm env
=> ((Set (Path Abs File) -> IO ()) -> RIO env ())
(HasConfig env, HasTerm env)
=> ((Set (Path Abs File) -> IO ()) -> RIO Runner ())
-> RIO env ()
fileWatchPoll =
fileWatchConf $ defaultConfig { confWatchMode = WatchModePoll 1000000 }
Expand All @@ -39,11 +49,13 @@ fileWatchPoll =
-- The action provided takes a callback that is used to set the files to be
-- watched. When any of those files are changed, we rerun the action again.
fileWatchConf ::
HasTerm env
(HasConfig env, HasTerm env)
=> WatchConfig
-> ((Set (Path Abs File) -> IO ()) -> RIO env ())
-> ((Set (Path Abs File) -> IO ()) -> RIO Runner ())
-> RIO env ()
fileWatchConf cfg inner =
fileWatchConf cfg inner = do
runner <- view runnerL
mHook <- view $ configL . to (.fileWatchHook)
withRunInIO $ \run -> withManagerConf cfg $ \manager -> do
allFiles <- newTVarIO Set.empty
dirtyVar <- newTVarIO True
Expand Down Expand Up @@ -134,7 +146,7 @@ fileWatchConf cfg inner =
dirty <- readTVar dirtyVar
check dirty

eres <- tryAny $ inner setWatched
eres <- tryAny $ runRIO runner (inner setWatched)

-- Clear dirtiness flag after the build to avoid an infinite loop caused
-- by the build itself triggering dirtiness. This could be viewed as a
Expand All @@ -143,19 +155,63 @@ fileWatchConf cfg inner =
-- https://github.com/commercialhaskell/stack/issues/822
atomically $ writeTVar dirtyVar False

case eres of
Left e ->
case fromException e of
Just ExitSuccess ->
prettyInfo $ style Good $ fromString $ displayException e
_ -> case fromException e :: Maybe PrettyException of
Just pe -> prettyError $ pretty pe
_ -> prettyInfo $ style Error $ fromString $ displayException e
_ -> prettyInfo $
style Good (flow "Success! Waiting for next file change.")
let defaultAction = case eres of
Left e ->
case fromException e of
Just ExitSuccess ->
prettyInfo $ style Good $ fromString $ displayException e
_ -> case fromException e :: Maybe PrettyException of
Just pe -> prettyError $ pretty pe
_ -> prettyInfo $ style Error $ fromString $ displayException e
_ -> prettyInfo $
style Good (flow "Success! Waiting for next file change.")

case mHook of
Nothing -> defaultAction
Just hook -> do
hookIsExecutable <- handleIO (\_ -> pure False) $ if osIsWindows
then
-- can't really detect executable on windows, only file extension
doesFileExist hook
else executable <$> getPermissions hook
if hookIsExecutable
then runFileWatchHook eres hook
else do
prettyWarn $
flow "File watch hook not executable. Falling back on default."
defaultAction

prettyInfoL
[ "Type"
, style Shell "help"
, flow "for the available commands. Press enter to force a rebuild."
]

runFileWatchHook ::
(HasProcessContext env, HasTerm env)
=> Either SomeException ()
-> Path Abs File
-> RIO env ()
runFileWatchHook buildResult hook =
withModifyEnvVars insertBuildResultInEnv $ do
let (cmd, args) = if osIsWindows && isShFile
then ("sh", [toFilePath hook])
else (toFilePath hook, [])
menv <- view processContextL
exit <- withProcessContext menv $ proc cmd args runProcess
case exit of
ExitSuccess -> pure ()
ExitFailure i -> do
prettyWarnL
[ flow "File watch hook exited with code:"
, style Error (fromString $ show i) <> "."
]
pure ()
where
insertBuildResultInEnv :: EnvVars -> EnvVars
insertBuildResultInEnv = Map.insert "HOOK_FW_RESULT" $ case buildResult of
Left e -> T.pack $ displayException e
Right _ -> ""
isShFile = case fileExtension hook of
Just ".sh" -> True
_ -> False
3 changes: 3 additions & 0 deletions src/Stack/Types/Config.hs
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,9 @@ data Config = Config
-- ^ Specifies the repository containing the compiler sources
, localBin :: !(Path Abs Dir)
-- ^ Directory we should install executables into
, fileWatchHook :: !(Maybe (Path Abs File))
-- ^ Optional path of executable used to override --file-watch
-- post-processing.
, requireStackVersion :: !VersionRange
-- ^ Require a version of Stack within this range.
, jobs :: !Int
Expand Down
6 changes: 6 additions & 0 deletions src/Stack/Types/Config/Exception.hs
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ data ConfigException
| UnableToExtractArchive Text (Path Abs File)
| BadStackVersionException VersionRange
| NoSuchDirectory FilePath
| NoSuchFile FilePath
| ParseGHCVariantException String
| BadStackRoot (Path Abs Dir)
| Won'tCreateStackRootInDirectoryOwnedByDifferentUser
Expand Down Expand Up @@ -97,6 +98,11 @@ instance Exception ConfigException where
, "No directory could be located matching the supplied path: "
, dir
]
displayException (NoSuchFile file) = concat
[ "Error: [S-xxxx]\n"
, "No file could be located matching the supplied path: "
, file
]
displayException (ParseGHCVariantException v) = concat
[ "Error: [S-3938]\n"
, "Invalid ghc-variant value: "
Expand Down
7 changes: 7 additions & 0 deletions src/Stack/Types/ConfigMonoid.hs
Original file line number Diff line number Diff line change
Expand Up @@ -119,6 +119,8 @@ data ConfigMonoid = ConfigMonoid
-- ^ See: 'configConcurrentTests'
, localBinPath :: !(First FilePath)
-- ^ Used to override the binary installation dir
, fileWatchHook :: !(First FilePath)
-- ^ Path to executable used to override --file-watch post-processing.
, templateParameters :: !(Map Text Text)
-- ^ Template parameters.
, scmInit :: !(First SCM)
Expand Down Expand Up @@ -265,6 +267,7 @@ parseConfigMonoidObject rootDir obj = do
hpackForce <- FirstFalse <$> obj ..:? configMonoidHpackForceName
concurrentTests <- First <$> obj ..:? configMonoidConcurrentTestsName
localBinPath <- First <$> obj ..:? configMonoidLocalBinPathName
fileWatchHook <- First <$> obj ..:? configMonoidFileWatchHookName
templates <- obj ..:? "templates"
(scmInit, templateParameters) <-
case templates of
Expand Down Expand Up @@ -373,6 +376,7 @@ parseConfigMonoidObject rootDir obj = do
, hpackForce
, concurrentTests
, localBinPath
, fileWatchHook
, templateParameters
, scmInit
, ghcOptionsByName
Expand Down Expand Up @@ -495,6 +499,9 @@ configMonoidConcurrentTestsName = "concurrent-tests"
configMonoidLocalBinPathName :: Text
configMonoidLocalBinPathName = "local-bin-path"

configMonoidFileWatchHookName :: Text
configMonoidFileWatchHookName = "file-watch-hook"

configMonoidScmInitName :: Text
configMonoidScmInitName = "scm-init"

Expand Down

0 comments on commit 4f6fe19

Please sign in to comment.