2016-02-14 18:10:21 +09:00
|
|
|
{- This file is part of Vervis.
|
|
|
|
-
|
2019-01-15 07:03:49 +09:00
|
|
|
- Written in 2016, 2018, 2019 by fr33domlover <fr33domlover@riseup.net>.
|
2016-02-14 18:10:21 +09:00
|
|
|
-
|
|
|
|
- ♡ Copying is an act of love. Please copy, reuse and share.
|
|
|
|
-
|
|
|
|
- The author(s) have dedicated all copyright and related and neighboring
|
|
|
|
- rights to this software to the public domain worldwide. This software is
|
|
|
|
- distributed without any warranty.
|
|
|
|
-
|
|
|
|
- You should have received a copy of the CC0 Public Domain Dedication along
|
|
|
|
- with this software. If not, see
|
|
|
|
- <http://creativecommons.org/publicdomain/zero/1.0/>.
|
|
|
|
-}
|
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
{-# OPTIONS_GHC -fno-warn-orphans #-}
|
2016-02-14 18:10:21 +09:00
|
|
|
|
2016-02-23 17:45:03 +09:00
|
|
|
module Vervis.Application
|
2016-02-13 12:35:30 +09:00
|
|
|
( getApplicationDev
|
|
|
|
, appMain
|
|
|
|
, develMain
|
|
|
|
, makeFoundation
|
|
|
|
, makeLogWare
|
|
|
|
-- * for DevelMain
|
|
|
|
, getApplicationRepl
|
|
|
|
, shutdownApp
|
|
|
|
-- * for GHCI
|
|
|
|
, handler
|
|
|
|
, db
|
2016-02-14 18:10:21 +09:00
|
|
|
)
|
|
|
|
where
|
2016-02-13 12:35:30 +09:00
|
|
|
|
2018-04-01 04:22:37 +09:00
|
|
|
import Control.Monad.Logger (liftLoc, runLoggingT, logInfo, logError)
|
2016-02-13 12:35:30 +09:00
|
|
|
import Database.Persist.Postgresql (createPostgresqlPool, pgConnStr,
|
|
|
|
pgPoolSize, runSqlPool)
|
2018-05-26 19:27:05 +09:00
|
|
|
import Graphics.SVGFonts.Fonts (lin2)
|
|
|
|
import Graphics.SVGFonts.ReadFont (loadFont)
|
2016-02-23 17:45:03 +09:00
|
|
|
import Vervis.Import
|
2016-02-13 12:35:30 +09:00
|
|
|
import Language.Haskell.TH.Syntax (qLocation)
|
|
|
|
import Network.Wai (Middleware)
|
|
|
|
import Network.Wai.Handler.Warp (Settings, defaultSettings,
|
|
|
|
defaultShouldDisplayException,
|
|
|
|
runSettings, setHost,
|
|
|
|
setOnException, setPort, getPort)
|
|
|
|
import Network.Wai.Middleware.RequestLogger (Destination (Logger),
|
|
|
|
IPAddrSource (..),
|
|
|
|
OutputFormat (..), destination,
|
|
|
|
mkRequestLogger, outputFormat)
|
|
|
|
import System.Log.FastLogger (defaultBufSize, newStdoutLoggerSet,
|
|
|
|
toLogStr)
|
2016-03-10 07:27:25 +09:00
|
|
|
import Yesod.Default.Main (LogFunc)
|
2018-03-04 06:33:59 +09:00
|
|
|
import Yesod.Mail.Send (runMailer)
|
2016-02-13 12:35:30 +09:00
|
|
|
|
2018-04-05 09:03:27 +09:00
|
|
|
import qualified Data.Text as T (unpack)
|
|
|
|
|
2019-01-15 07:03:49 +09:00
|
|
|
import Control.Concurrent.Local (forkCheck)
|
|
|
|
|
|
|
|
import Vervis.ActorKey (generateActorKey, actorKeyRotator)
|
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
-- Import all relevant handler modules here.
|
|
|
|
-- Don't forget to add new modules to your cabal file!
|
2016-02-23 17:45:03 +09:00
|
|
|
import Vervis.Handler.Common
|
2016-04-21 09:32:22 +09:00
|
|
|
import Vervis.Handler.Git
|
2016-05-25 06:48:21 +09:00
|
|
|
import Vervis.Handler.Group
|
2016-02-23 17:45:03 +09:00
|
|
|
import Vervis.Handler.Home
|
2016-03-07 09:42:06 +09:00
|
|
|
import Vervis.Handler.Key
|
2016-02-23 17:45:03 +09:00
|
|
|
import Vervis.Handler.Person
|
|
|
|
import Vervis.Handler.Project
|
2016-02-27 14:41:36 +09:00
|
|
|
import Vervis.Handler.Repo
|
2016-05-29 22:17:55 +09:00
|
|
|
import Vervis.Handler.Role
|
2016-05-25 06:48:21 +09:00
|
|
|
import Vervis.Handler.Sharer
|
2016-05-01 07:32:22 +09:00
|
|
|
import Vervis.Handler.Ticket
|
2016-06-04 15:57:54 +09:00
|
|
|
import Vervis.Handler.Wiki
|
2016-08-08 20:05:19 +09:00
|
|
|
import Vervis.Handler.Workflow
|
2016-02-13 12:35:30 +09:00
|
|
|
|
2016-09-01 01:51:02 +09:00
|
|
|
import Vervis.Migration (migrateDB)
|
2016-03-06 20:58:48 +09:00
|
|
|
import Vervis.Ssh (runSsh)
|
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
-- This line actually creates our YesodDispatch instance. It is the second half
|
|
|
|
-- of the call to mkYesodData which occurs in Foundation.hs. Please see the
|
|
|
|
-- comments there for more details.
|
|
|
|
mkYesodDispatch "App" resourcesApp
|
|
|
|
|
2016-03-10 07:27:25 +09:00
|
|
|
loggingFunction :: App -> LogFunc
|
|
|
|
loggingFunction app = messageLoggerSource app (appLogger app)
|
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
-- | This function allocates resources (such as a database connection pool),
|
|
|
|
-- performs initialization and returns a foundation datatype value. This is also
|
|
|
|
-- the place to put your migrate statements to have automatic database
|
|
|
|
-- migrations handled by Yesod.
|
|
|
|
makeFoundation :: AppSettings -> IO App
|
|
|
|
makeFoundation appSettings = do
|
|
|
|
-- Some basic initializations: HTTP connection manager, logger, and static
|
|
|
|
-- subsite.
|
2019-01-14 11:30:39 +09:00
|
|
|
--appHttpManager <- newManager tlsManagerSettings
|
2016-02-13 12:35:30 +09:00
|
|
|
appLogger <- newStdoutLoggerSet defaultBufSize >>= makeYesodLogger
|
|
|
|
appStatic <-
|
|
|
|
(if appMutableStatic appSettings then staticDevel else static)
|
|
|
|
(appStaticDir appSettings)
|
|
|
|
|
2018-02-25 18:28:55 +09:00
|
|
|
appMailQueue <-
|
2018-03-04 06:33:59 +09:00
|
|
|
case appMail appSettings of
|
2018-02-25 18:28:55 +09:00
|
|
|
Nothing -> return Nothing
|
|
|
|
Just _ -> Just <$> newChan
|
|
|
|
|
2018-05-26 19:27:05 +09:00
|
|
|
appSvgFont <-
|
|
|
|
if appLoadFontFromLibData appSettings
|
2018-12-05 12:41:19 +09:00
|
|
|
then lin2
|
2018-05-26 19:27:05 +09:00
|
|
|
else loadFont "data/LinLibertineCut.svg"
|
|
|
|
|
2019-01-15 07:08:44 +09:00
|
|
|
appActorKey <- newTVarIO =<< generateActorKey
|
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
-- We need a log function to create a connection pool. We need a connection
|
|
|
|
-- pool to create our foundation. And we need our foundation to get a
|
|
|
|
-- logging function. To get out of this loop, we initially create a
|
|
|
|
-- temporary foundation without a real connection pool, get a log function
|
|
|
|
-- from there, and then create the real foundation.
|
|
|
|
let mkFoundation appConnPool = App {..}
|
|
|
|
-- The App {..} syntax is an example of record wild cards. For more
|
|
|
|
-- information, see:
|
|
|
|
-- https://ocharles.org.uk/blog/posts/2014-12-04-record-wildcards.html
|
|
|
|
tempFoundation = mkFoundation $ error "connPool forced in tempFoundation"
|
2016-03-10 07:27:25 +09:00
|
|
|
logFunc = loggingFunction tempFoundation
|
2016-02-13 12:35:30 +09:00
|
|
|
|
|
|
|
-- Create the database connection pool
|
|
|
|
pool <- flip runLoggingT logFunc $ createPostgresqlPool
|
|
|
|
(pgConnStr $ appDatabaseConf appSettings)
|
|
|
|
(pgPoolSize $ appDatabaseConf appSettings)
|
|
|
|
|
|
|
|
-- Perform database migration using our application's logging settings.
|
2016-09-01 01:51:02 +09:00
|
|
|
--runLoggingT (runSqlPool (runMigration migrateAll) pool) logFunc
|
2018-04-01 04:22:37 +09:00
|
|
|
flip runLoggingT logFunc $
|
|
|
|
flip runSqlPool pool $ do
|
|
|
|
r <- migrateDB
|
|
|
|
case r of
|
|
|
|
Left err -> do
|
2018-04-05 09:03:27 +09:00
|
|
|
let msg = "DB migration failed: " <> err
|
2018-04-01 04:22:37 +09:00
|
|
|
$logError msg
|
2018-04-05 09:03:27 +09:00
|
|
|
error $ T.unpack msg
|
2018-04-01 04:22:37 +09:00
|
|
|
Right (_from, _to) -> $logInfo "DB migration success"
|
2016-02-13 12:35:30 +09:00
|
|
|
|
|
|
|
-- Return the foundation
|
|
|
|
return $ mkFoundation pool
|
|
|
|
|
|
|
|
-- | Convert our foundation to a WAI Application by calling @toWaiAppPlain@ and
|
|
|
|
-- applying some additional middlewares.
|
|
|
|
makeApplication :: App -> IO Application
|
|
|
|
makeApplication foundation = do
|
|
|
|
logWare <- makeLogWare foundation
|
|
|
|
-- Create the WAI application and apply middlewares
|
|
|
|
appPlain <- toWaiAppPlain foundation
|
|
|
|
return $ logWare $ defaultMiddlewaresNoLogging appPlain
|
|
|
|
|
|
|
|
makeLogWare :: App -> IO Middleware
|
|
|
|
makeLogWare foundation =
|
|
|
|
mkRequestLogger def
|
|
|
|
{ outputFormat =
|
|
|
|
if appDetailedRequestLogging $ appSettings foundation
|
|
|
|
then Detailed True
|
|
|
|
else Apache
|
|
|
|
(if appIpFromHeader $ appSettings foundation
|
|
|
|
then FromFallback
|
|
|
|
else FromSocket)
|
|
|
|
, destination = Logger $ loggerSet $ appLogger foundation
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
-- | Warp settings for the given foundation value.
|
|
|
|
warpSettings :: App -> Settings
|
|
|
|
warpSettings foundation =
|
|
|
|
setPort (appPort $ appSettings foundation)
|
|
|
|
$ setHost (appHost $ appSettings foundation)
|
|
|
|
$ setOnException (\_req e ->
|
2016-03-10 07:27:25 +09:00
|
|
|
when (defaultShouldDisplayException e) $ loggingFunction
|
2016-02-13 12:35:30 +09:00
|
|
|
foundation
|
|
|
|
$(qLocation >>= liftLoc)
|
|
|
|
"yesod"
|
|
|
|
LevelError
|
|
|
|
(toLogStr $ "Exception from Warp: " ++ show e))
|
|
|
|
defaultSettings
|
|
|
|
|
|
|
|
-- | For yesod devel, return the Warp settings and WAI Application.
|
|
|
|
getApplicationDev :: IO (Settings, Application)
|
|
|
|
getApplicationDev = do
|
|
|
|
settings <- getAppSettings
|
|
|
|
foundation <- makeFoundation settings
|
|
|
|
wsettings <- getDevSettings $ warpSettings foundation
|
|
|
|
app <- makeApplication foundation
|
|
|
|
return (wsettings, app)
|
|
|
|
|
|
|
|
getAppSettings :: IO AppSettings
|
|
|
|
getAppSettings = loadAppSettings [configSettingsYml] [] useEnv
|
|
|
|
|
|
|
|
-- | main function for use by yesod devel
|
|
|
|
develMain :: IO ()
|
|
|
|
develMain = develMainHelper getApplicationDev
|
|
|
|
|
2019-01-15 07:08:44 +09:00
|
|
|
actorKeyPeriodicRotator :: App -> IO ()
|
|
|
|
actorKeyPeriodicRotator app =
|
|
|
|
actorKeyRotator (appActorKeyRotation $ appSettings app) (appActorKey app)
|
|
|
|
|
2016-03-10 07:27:25 +09:00
|
|
|
sshServer :: App -> IO ()
|
|
|
|
sshServer foundation =
|
|
|
|
runSsh
|
|
|
|
(appSettings foundation)
|
|
|
|
(appConnPool foundation)
|
|
|
|
(loggingFunction foundation)
|
|
|
|
|
2018-02-25 18:28:55 +09:00
|
|
|
mailer :: App -> IO ()
|
|
|
|
mailer foundation =
|
2018-03-04 06:33:59 +09:00
|
|
|
case (appMail $ appSettings foundation, appMailQueue foundation) of
|
2018-02-25 18:28:55 +09:00
|
|
|
(Nothing , Nothing) -> return ()
|
|
|
|
(Nothing , Just _) -> error "Mail queue unnecessarily created"
|
|
|
|
(Just _ , Nothing) -> error "Mail queue wasn't created"
|
2018-03-04 06:33:59 +09:00
|
|
|
(Just mail, Just queue) ->
|
2018-02-25 18:28:55 +09:00
|
|
|
runMailer
|
2018-03-04 06:33:59 +09:00
|
|
|
mail
|
|
|
|
-- (appConnPool foundation)
|
2018-02-25 18:28:55 +09:00
|
|
|
(loggingFunction foundation)
|
|
|
|
(readChan queue)
|
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
-- | The @main@ function for an executable running this site.
|
|
|
|
appMain :: IO ()
|
|
|
|
appMain = do
|
|
|
|
-- Get the settings from all relevant sources
|
|
|
|
settings <- loadAppSettingsArgs
|
2016-03-06 20:58:48 +09:00
|
|
|
-- Fall back to compile-time values, set to [] to require values at
|
|
|
|
-- runtime
|
2016-02-13 12:35:30 +09:00
|
|
|
[configSettingsYmlValue]
|
|
|
|
|
2016-03-06 20:58:48 +09:00
|
|
|
-- Allow environment variables to override
|
2016-02-13 12:35:30 +09:00
|
|
|
useEnv
|
|
|
|
|
|
|
|
-- Generate the foundation from the settings
|
|
|
|
foundation <- makeFoundation settings
|
|
|
|
|
|
|
|
-- Generate a WAI Application from the foundation
|
|
|
|
app <- makeApplication foundation
|
|
|
|
|
2019-01-15 07:08:44 +09:00
|
|
|
-- Run actor signature key periodic generation thread
|
|
|
|
forkCheck $ actorKeyPeriodicRotator foundation
|
|
|
|
|
2018-02-25 18:28:55 +09:00
|
|
|
-- Run SSH server
|
2019-01-15 07:03:49 +09:00
|
|
|
forkCheck $ sshServer foundation
|
2016-03-06 20:58:48 +09:00
|
|
|
|
2018-02-25 18:28:55 +09:00
|
|
|
-- Run mailer if SMTP is enabled
|
2019-01-15 07:03:49 +09:00
|
|
|
forkCheck $ mailer foundation
|
2018-02-25 18:28:55 +09:00
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
-- Run the application with Warp
|
|
|
|
runSettings (warpSettings foundation) app
|
|
|
|
|
|
|
|
|
|
|
|
--------------------------------------------------------------
|
|
|
|
-- Functions for DevelMain.hs (a way to run the app from GHCi)
|
|
|
|
--------------------------------------------------------------
|
|
|
|
getApplicationRepl :: IO (Int, App, Application)
|
|
|
|
getApplicationRepl = do
|
|
|
|
settings <- getAppSettings
|
|
|
|
foundation <- makeFoundation settings
|
|
|
|
wsettings <- getDevSettings $ warpSettings foundation
|
|
|
|
app1 <- makeApplication foundation
|
|
|
|
return (getPort wsettings, foundation, app1)
|
|
|
|
|
|
|
|
shutdownApp :: App -> IO ()
|
|
|
|
shutdownApp _ = return ()
|
|
|
|
|
|
|
|
|
|
|
|
---------------------------------------------
|
|
|
|
-- Functions for use in development with GHCi
|
|
|
|
---------------------------------------------
|
|
|
|
|
|
|
|
-- | Run a handler
|
|
|
|
handler :: Handler a -> IO a
|
|
|
|
handler h = getAppSettings >>= makeFoundation >>= flip unsafeHandler h
|
|
|
|
|
|
|
|
-- | Run DB queries
|
|
|
|
db :: ReaderT SqlBackend (HandlerT App IO) a -> IO a
|
|
|
|
db = handler . runDB
|