main.hs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. {-# LANGUAGE CPP #-}
  2. {-# LANGUAGE RecordWildCards #-}
  3. import Control.Monad (unless)
  4. import Data.Monoid
  5. import Data.Version (showVersion)
  6. import Options.Applicative
  7. import System.Environment (getEnvironment)
  8. import System.Exit (ExitCode (ExitSuccess), exitWith, exitFailure)
  9. import System.FilePath (splitSearchPath)
  10. import System.Process (rawSystem)
  11. import AddHandler (addHandler)
  12. import Devel (DevelOpts (..), devel, DevelTermOpt(..))
  13. import Keter (keter)
  14. import Options (injectDefaults)
  15. import qualified Paths_yesod_bin
  16. import System.IO (hPutStrLn, stderr)
  17. import HsFile (mkHsFile)
  18. #ifndef WINDOWS
  19. import Build (touch)
  20. touch' :: IO ()
  21. touch' = touch
  22. windowsWarning :: String
  23. windowsWarning = ""
  24. #else
  25. touch' :: IO ()
  26. touch' = return ()
  27. windowsWarning :: String
  28. windowsWarning = " (does not work on Windows)"
  29. #endif
  30. data CabalPgm = Cabal | CabalDev deriving (Show, Eq)
  31. data Options = Options
  32. { optCabalPgm :: CabalPgm
  33. , optVerbose :: Bool
  34. , optCommand :: Command
  35. }
  36. deriving (Show, Eq)
  37. data Command = Init [String]
  38. | HsFiles
  39. | Configure
  40. | Build { buildExtraArgs :: [String] }
  41. | Touch
  42. | Devel { _develDisableApi :: Bool
  43. , _develSuccessHook :: Maybe String
  44. , _develFailHook :: Maybe String
  45. , _develRescan :: Int
  46. , _develBuildDir :: Maybe String
  47. , develIgnore :: [String]
  48. , develExtraArgs :: [String]
  49. , _develPort :: Int
  50. , _develTlsPort :: Int
  51. , _proxyTimeout :: Int
  52. , _noReverseProxy :: Bool
  53. , _interruptOnly :: Bool
  54. }
  55. | Test
  56. | AddHandler
  57. { addHandlerRoute :: Maybe String
  58. , addHandlerPattern :: Maybe String
  59. , addHandlerMethods :: [String]
  60. }
  61. | Keter
  62. { _keterNoRebuild :: Bool
  63. , _keterNoCopyTo :: Bool
  64. , _keterBuildArgs :: [String]
  65. }
  66. | Version
  67. deriving (Show, Eq)
  68. cabalCommand :: Options -> String
  69. cabalCommand mopt
  70. | optCabalPgm mopt == CabalDev = "cabal-dev"
  71. | otherwise = "cabal"
  72. main :: IO ()
  73. main = do
  74. o <- execParser =<< injectDefaults "yesod"
  75. [ ("yesod.devel.extracabalarg" , \o args -> o { optCommand =
  76. case optCommand o of
  77. d@Devel{} -> d { develExtraArgs = args }
  78. c -> c
  79. })
  80. , ("yesod.devel.ignore" , \o args -> o { optCommand =
  81. case optCommand o of
  82. d@Devel{} -> d { develIgnore = args }
  83. c -> c
  84. })
  85. , ("yesod.build.extracabalarg" , \o args -> o { optCommand =
  86. case optCommand o of
  87. b@Build{} -> b { buildExtraArgs = args }
  88. c -> c
  89. })
  90. ] optParser'
  91. let cabal = rawSystem' (cabalCommand o)
  92. case optCommand o of
  93. Init _ -> initErrorMsg
  94. HsFiles -> mkHsFile
  95. Configure -> cabal ["configure"]
  96. Build es -> touch' >> cabal ("build":es)
  97. Touch -> touch'
  98. Keter{..} -> keter (cabalCommand o) _keterNoRebuild _keterNoCopyTo _keterBuildArgs
  99. Version -> putStrLn ("yesod-bin version: " ++ showVersion Paths_yesod_bin.version)
  100. AddHandler{..} -> addHandler addHandlerRoute addHandlerPattern addHandlerMethods
  101. Test -> cabalTest cabal
  102. Devel{..} ->do
  103. (configOpts, menv) <- handleGhcPackagePath
  104. let develOpts = DevelOpts
  105. { isCabalDev = optCabalPgm o == CabalDev
  106. , forceCabal = _develDisableApi
  107. , verbose = optVerbose o
  108. , eventTimeout = _develRescan
  109. , successHook = _develSuccessHook
  110. , failHook = _develFailHook
  111. , buildDir = _develBuildDir
  112. , develPort = _develPort
  113. , develTlsPort = _develTlsPort
  114. , proxyTimeout = _proxyTimeout
  115. , useReverseProxy = not _noReverseProxy
  116. , terminateWith = if _interruptOnly then TerminateOnlyInterrupt else TerminateOnEnter
  117. , develConfigOpts = configOpts
  118. , develEnv = menv
  119. }
  120. devel develOpts develExtraArgs
  121. where
  122. cabalTest cabal = do
  123. env <- getEnvironment
  124. case lookup "STACK_EXE" env of
  125. Nothing -> do
  126. touch'
  127. _ <- cabal ["configure", "--enable-tests", "-flibrary-only"]
  128. _ <- cabal ["build"]
  129. cabal ["test"]
  130. Just _ -> do
  131. hPutStrLn stderr "'yesod test' is no longer needed with Stack"
  132. hPutStrLn stderr "Instead, please just run 'stack test'"
  133. exitFailure
  134. initErrorMsg = do
  135. mapM_ putStrLn
  136. [ "The init command has been removed."
  137. , "Please use 'stack new <project name> <template>' instead where the"
  138. , "available templates can be found by running 'stack templates'. For"
  139. , "a Yesod based application you should probably choose one of the"
  140. , "pre-canned Yesod templates."
  141. ]
  142. exitFailure
  143. handleGhcPackagePath :: IO ([String], Maybe [(String, String)])
  144. handleGhcPackagePath = do
  145. env <- getEnvironment
  146. case lookup "GHC_PACKAGE_PATH" env of
  147. Nothing -> return ([], Nothing)
  148. Just gpp -> do
  149. let opts = "--package-db=clear"
  150. : "--package-db=global"
  151. : map ("--package-db=" ++)
  152. (drop 1 $ reverse $ splitSearchPath gpp)
  153. return (opts, Just $ filter (\(x, _) -> x /= "GHC_PACKAGE_PATH") env)
  154. optParser' :: ParserInfo Options
  155. optParser' = info (helper <*> optParser) ( fullDesc <> header "Yesod Web Framework command line utility" )
  156. optParser :: Parser Options
  157. optParser = Options
  158. <$> flag Cabal CabalDev ( long "dev" <> short 'd' <> help "use cabal-dev" )
  159. <*> switch ( long "verbose" <> short 'v' <> help "More verbose output" )
  160. <*> subparser ( command "init" (info initOptions
  161. (progDesc "Command no longer available, please use 'stack new'"))
  162. <> command "hsfiles" (info (pure HsFiles)
  163. (progDesc "Create a hsfiles file for the current folder"))
  164. <> command "configure" (info (pure Configure)
  165. (progDesc "Configure a project for building"))
  166. <> command "build" (info (Build <$> extraCabalArgs)
  167. (progDesc $ "Build project (performs TH dependency analysis)" ++ windowsWarning))
  168. <> command "touch" (info (pure Touch)
  169. (progDesc $ "Touch any files with altered TH dependencies but do not build" ++ windowsWarning))
  170. <> command "devel" (info develOptions
  171. (progDesc "Run project with the devel server"))
  172. <> command "test" (info (pure Test)
  173. (progDesc "Build and run the integration tests"))
  174. <> command "add-handler" (info addHandlerOptions
  175. (progDesc ("Add a new handler and module to the project."
  176. ++ " Interactively asks for input if you do not specify arguments.")))
  177. <> command "keter" (info keterOptions
  178. (progDesc "Build a keter bundle"))
  179. <> command "version" (info (pure Version)
  180. (progDesc "Print the version of Yesod"))
  181. )
  182. initOptions :: Parser Command
  183. initOptions = Init <$> many (argument str mempty)
  184. keterOptions :: Parser Command
  185. keterOptions = Keter
  186. <$> switch ( long "nobuild" <> short 'n' <> help "Skip rebuilding" )
  187. <*> switch ( long "nocopyto" <> help "Ignore copy-to directive in keter config file" )
  188. <*> optStrToList ( long "build-args" <> help "Build arguments" )
  189. where
  190. optStrToList m = option (words <$> str) $ value [] <> m
  191. defaultRescan :: Int
  192. defaultRescan = 10
  193. develOptions :: Parser Command
  194. develOptions = Devel <$> switch ( long "disable-api" <> short 'd'
  195. <> help "Disable fast GHC API rebuilding")
  196. <*> optStr ( long "success-hook" <> short 's' <> metavar "COMMAND"
  197. <> help "Run COMMAND after rebuild succeeds")
  198. <*> optStr ( long "failure-hook" <> short 'f' <> metavar "COMMAND"
  199. <> help "Run COMMAND when rebuild fails")
  200. <*> option auto ( long "event-timeout" <> short 't' <> value defaultRescan <> metavar "N"
  201. <> help ("Force rescan of files every N seconds (default "
  202. ++ show defaultRescan
  203. ++ ", use -1 to rely on FSNotify alone)") )
  204. <*> optStr ( long "builddir" <> short 'b'
  205. <> help "Set custom cabal build directory, default `dist'")
  206. <*> many ( strOption ( long "ignore" <> short 'i' <> metavar "DIR"
  207. <> help "ignore file changes in DIR" )
  208. )
  209. <*> extraCabalArgs
  210. <*> option auto ( long "port" <> short 'p' <> value 3000 <> metavar "N"
  211. <> help "Devel server listening port" )
  212. <*> option auto ( long "tls-port" <> short 'q' <> value 3443 <> metavar "N"
  213. <> help "Devel server listening port (tls)" )
  214. <*> option auto ( long "proxy-timeout" <> short 'x' <> value 0 <> metavar "N"
  215. <> help "Devel server timeout before returning 'not ready' message (in seconds, 0 for none)" )
  216. <*> switch ( long "disable-reverse-proxy" <> short 'n'
  217. <> help "Disable reverse proxy" )
  218. <*> switch ( long "interrupt-only" <> short 'c'
  219. <> help "Disable exiting when enter is pressed")
  220. extraCabalArgs :: Parser [String]
  221. extraCabalArgs = many (strOption ( long "extra-cabal-arg" <> short 'e' <> metavar "ARG"
  222. <> help "pass extra argument ARG to cabal")
  223. )
  224. addHandlerOptions :: Parser Command
  225. addHandlerOptions = AddHandler
  226. <$> optStr ( long "route" <> short 'r' <> metavar "ROUTE"
  227. <> help "Name of route (without trailing R). Required.")
  228. <*> optStr ( long "pattern" <> short 'p' <> metavar "PATTERN"
  229. <> help "Route pattern (ex: /entry/#EntryId). Defaults to \"\".")
  230. <*> many (strOption ( long "method" <> short 'm' <> metavar "METHOD"
  231. <> help "Takes one method. Use this multiple times to add multiple methods. Defaults to none.")
  232. )
  233. -- | Optional @String@ argument
  234. optStr :: Mod OptionFields (Maybe String) -> Parser (Maybe String)
  235. optStr m = option (Just <$> str) $ value Nothing <> m
  236. -- | Like @rawSystem@, but exits if it receives a non-success result.
  237. rawSystem' :: String -> [String] -> IO ()
  238. rawSystem' x y = do
  239. res <- rawSystem x y
  240. unless (res == ExitSuccess) $ exitWith res