2016-02-14 18:10:21 +09:00
|
|
|
{- This file is part of Vervis.
|
|
|
|
-
|
2019-01-15 07:08:44 +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
|
|
|
{-# Language CPP #-}
|
2016-02-14 18:10:21 +09:00
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
-- | Settings are centralized, as much as possible, into this file. This
|
|
|
|
-- includes database connection settings, static file locations, etc.
|
|
|
|
-- In addition, you can configure a number of different aspects of Yesod
|
|
|
|
-- by overriding methods in the Yesod typeclass. That instance is
|
|
|
|
-- declared in the Foundation.hs file.
|
2016-02-23 17:45:03 +09:00
|
|
|
module Vervis.Settings where
|
2016-02-13 12:35:30 +09:00
|
|
|
|
2018-05-16 09:02:54 +09:00
|
|
|
import ClassyPrelude.Conduit hiding (throw)
|
2016-02-29 23:04:23 +09:00
|
|
|
import Yesod hiding (Header, parseTime)
|
|
|
|
import Yesod.Static
|
|
|
|
import Data.Default (Default (..))
|
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
import Control.Exception (throw)
|
|
|
|
import Data.Aeson (Result (..), fromJSON, withObject, (.!=),
|
|
|
|
(.:?))
|
|
|
|
import Data.FileEmbed (embedFile)
|
2018-07-01 17:15:23 +09:00
|
|
|
import Data.Time.Interval (TimeInterval)
|
|
|
|
import Data.Time.Interval.Aeson (interval)
|
2016-02-13 12:35:30 +09:00
|
|
|
import Data.Yaml (decodeEither')
|
|
|
|
import Database.Persist.Postgresql (PostgresConf)
|
|
|
|
import Language.Haskell.TH.Syntax (Exp, Name, Q)
|
|
|
|
import Network.Wai.Handler.Warp (HostPreference)
|
|
|
|
import Yesod.Default.Config2 (applyEnvValue, configSettingsYml)
|
|
|
|
import Yesod.Default.Util (WidgetFileSettings, widgetFileNoReload,
|
|
|
|
widgetFileReload)
|
2018-03-04 06:33:59 +09:00
|
|
|
import Yesod.Mail.Send (MailSettings)
|
2018-02-25 18:28:55 +09:00
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
-- | Runtime settings to configure this application. These settings can be
|
|
|
|
-- loaded from various sources: defaults, environment variables, config files,
|
|
|
|
-- theoretically even a database.
|
|
|
|
data AppSettings = AppSettings
|
2016-04-20 00:42:54 +09:00
|
|
|
{ -- | Directory from which to serve static files.
|
|
|
|
appStaticDir :: String
|
|
|
|
-- | Configuration settings for accessing the database.
|
2016-02-13 12:35:30 +09:00
|
|
|
, appDatabaseConf :: PostgresConf
|
2019-02-24 10:35:07 +09:00
|
|
|
-- | Maximal number of remote instance-scope keys to cache in our local
|
|
|
|
-- database per instance.
|
|
|
|
, appMaxInstanceKeys :: Maybe Int
|
|
|
|
-- | Maximal number of keys (personal keys or usage of shared keys) to
|
|
|
|
-- remember cached in our database per remote actor.
|
|
|
|
, appMaxActorKeys :: Maybe Int
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Base for all generated URLs. If @Nothing@, determined from the
|
|
|
|
-- request headers.
|
2016-02-13 12:35:30 +09:00
|
|
|
, appRoot :: Maybe Text
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Host/interface the server should bind to.
|
2016-02-13 12:35:30 +09:00
|
|
|
, appHost :: HostPreference
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Port to listen on
|
2016-02-13 12:35:30 +09:00
|
|
|
, appPort :: Int
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Get the IP address from the header when logging. Useful when sitting
|
|
|
|
-- behind a reverse proxy.
|
2016-02-13 12:35:30 +09:00
|
|
|
, appIpFromHeader :: Bool
|
|
|
|
|
2018-07-01 17:15:23 +09:00
|
|
|
-- | Path of session cookie encryption key file
|
|
|
|
, appClientSessionKeyFile :: FilePath
|
|
|
|
-- | Idle timeout for session cookie expiration
|
|
|
|
, appClientSessionTimeout :: TimeInterval
|
|
|
|
|
2019-01-15 07:08:44 +09:00
|
|
|
-- Maximal accepted difference between current time and Date header
|
|
|
|
, appHttpSigTimeLimit :: TimeInterval
|
|
|
|
|
|
|
|
-- How often to generate a new actor key for making HTTP signatures
|
|
|
|
, appActorKeyRotation :: TimeInterval
|
|
|
|
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Use detailed request logging system
|
2016-02-13 12:35:30 +09:00
|
|
|
, appDetailedRequestLogging :: Bool
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Should all log messages be displayed?
|
2016-02-13 12:35:30 +09:00
|
|
|
, appShouldLogAll :: Bool
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Use the reload version of templates
|
2016-02-13 12:35:30 +09:00
|
|
|
, appReloadTemplates :: Bool
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Assume that files in the static dir may change after compilation
|
2016-02-13 12:35:30 +09:00
|
|
|
, appMutableStatic :: Bool
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Perform no stylesheet/script combining
|
2016-02-13 12:35:30 +09:00
|
|
|
, appSkipCombining :: Bool
|
|
|
|
|
2018-05-26 19:27:05 +09:00
|
|
|
-- | Load SVG font file from the data file path of the @SVGFonts@
|
|
|
|
-- library, instead of the app's production runtime data directory.
|
|
|
|
, appLoadFontFromLibData :: Bool
|
|
|
|
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Path to the directory under which git repos are placed
|
2016-02-27 14:41:36 +09:00
|
|
|
, appRepoDir :: FilePath
|
2018-05-22 05:32:34 +09:00
|
|
|
-- | Number of context lines to display around changes in commit diff
|
|
|
|
, appDiffContextLines :: Int
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Port for the SSH server component to listen on
|
2016-03-05 12:56:25 +09:00
|
|
|
, appSshPort :: Int
|
2016-04-20 00:42:54 +09:00
|
|
|
-- | Path to the server's SSH private key file
|
2016-03-05 12:56:25 +09:00
|
|
|
, appSshKeyFile :: FilePath
|
2016-04-20 01:03:27 +09:00
|
|
|
-- | Whether new user accounts can be created.
|
|
|
|
, appRegister :: Bool
|
2016-07-28 06:46:48 +09:00
|
|
|
-- | The maximal number of user accounts that can be registered.
|
|
|
|
, appAccounts :: Maybe Int
|
2018-03-04 06:33:59 +09:00
|
|
|
-- | SMTP server details for sending email, and other email related
|
|
|
|
-- details. If set to 'Nothing', no email will be sent.
|
|
|
|
, appMail :: Maybe MailSettings
|
2019-01-30 12:12:42 +09:00
|
|
|
|
2019-02-26 23:00:22 +09:00
|
|
|
-- | Signing key file for signing object capabilities sent to remote
|
|
|
|
-- users
|
2019-01-30 12:12:42 +09:00
|
|
|
, appCapabilitySigningKeyFile :: FilePath
|
2019-02-26 23:00:22 +09:00
|
|
|
-- | Salt for encoding and decoding hashids
|
2019-02-09 06:54:22 +09:00
|
|
|
, appHashidsSaltFile :: FilePath
|
2019-02-26 23:00:22 +09:00
|
|
|
-- | What do to when we wish to insert a new 'VerifKey' or
|
|
|
|
-- 'VerifKeySharedUsage' into the database, but we've reached the
|
|
|
|
-- configured storage limit.
|
|
|
|
--
|
|
|
|
-- 'True' means we simply reject HTTP signatures when it happens, which
|
|
|
|
-- means we basically don't support servers that use more keys or custom
|
|
|
|
-- setup other than what Vervis does.
|
|
|
|
--
|
|
|
|
-- 'False' means we do accept HTTP signatures even if we've reached the
|
|
|
|
-- storage limit setting. We simply handle it by remembering only the
|
|
|
|
-- amount of keys the limit allows, and otherwise we have to refetch keys
|
|
|
|
-- over HTTP, which possibly means we have to do more HTTP key fetching,
|
|
|
|
-- and the target server gets a higher load of key fetch GET requests.
|
|
|
|
, appRejectOnMaxKeys :: Bool
|
2016-02-13 12:35:30 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
instance FromJSON AppSettings where
|
2016-04-20 00:47:26 +09:00
|
|
|
parseJSON = withObject "AppSettings" $ \ o -> do
|
2016-02-13 12:35:30 +09:00
|
|
|
let defaultDev =
|
|
|
|
#if DEVELOPMENT
|
|
|
|
True
|
|
|
|
#else
|
|
|
|
False
|
|
|
|
#endif
|
|
|
|
appStaticDir <- o .: "static-dir"
|
|
|
|
appDatabaseConf <- o .: "database"
|
2019-02-24 10:35:07 +09:00
|
|
|
appMaxInstanceKeys <- o .:? "max-instance-keys"
|
|
|
|
appMaxActorKeys <- o .:? "max-actor-keys"
|
2016-02-13 12:35:30 +09:00
|
|
|
appRoot <- o .:? "approot"
|
|
|
|
appHost <- fromString <$> o .: "host"
|
2016-03-05 12:56:25 +09:00
|
|
|
appPort <- o .: "http-port"
|
2016-02-13 12:35:30 +09:00
|
|
|
appIpFromHeader <- o .: "ip-from-header"
|
|
|
|
|
2018-07-01 17:15:23 +09:00
|
|
|
appClientSessionKeyFile <- o .: "client-session-key"
|
|
|
|
appClientSessionTimeout <- interval <$> o .: "client-session-timeout"
|
|
|
|
|
2019-01-15 07:08:44 +09:00
|
|
|
appHttpSigTimeLimit <- interval <$> o .: "request-time-limit"
|
|
|
|
appActorKeyRotation <- interval <$> o .: "actor-key-rotation"
|
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
appDetailedRequestLogging <- o .:? "detailed-logging" .!= defaultDev
|
|
|
|
appShouldLogAll <- o .:? "should-log-all" .!= defaultDev
|
|
|
|
appReloadTemplates <- o .:? "reload-templates" .!= defaultDev
|
|
|
|
appMutableStatic <- o .:? "mutable-static" .!= defaultDev
|
|
|
|
appSkipCombining <- o .:? "skip-combining" .!= defaultDev
|
|
|
|
|
2018-05-26 19:27:05 +09:00
|
|
|
appLoadFontFromLibData <- o .:? "load-font-from-lib-data" .!= defaultDev
|
|
|
|
|
2016-02-27 14:41:36 +09:00
|
|
|
appRepoDir <- o .: "repo-dir"
|
2018-05-22 05:32:34 +09:00
|
|
|
appDiffContextLines <- o .: "diff-context-lines"
|
2016-03-05 12:56:25 +09:00
|
|
|
appSshPort <- o .: "ssh-port"
|
|
|
|
appSshKeyFile <- o .: "ssh-key-file"
|
2016-04-20 01:03:27 +09:00
|
|
|
appRegister <- o .: "registration"
|
2016-07-28 06:46:48 +09:00
|
|
|
appAccounts <- o .: "max-accounts"
|
2018-03-04 06:33:59 +09:00
|
|
|
appMail <- o .:? "mail"
|
2016-02-13 12:35:30 +09:00
|
|
|
|
2019-01-30 12:12:42 +09:00
|
|
|
appCapabilitySigningKeyFile <- o .: "capability-signing-key"
|
2019-02-09 06:54:22 +09:00
|
|
|
appHashidsSaltFile <- o .: "hashids-salt-file"
|
2019-02-26 23:00:22 +09:00
|
|
|
appRejectOnMaxKeys <- o .: "reject-on-max-keys"
|
2019-01-30 12:12:42 +09:00
|
|
|
|
2016-02-13 12:35:30 +09:00
|
|
|
return AppSettings {..}
|
|
|
|
|
|
|
|
-- | Settings for 'widgetFile', such as which template languages to support and
|
|
|
|
-- default Hamlet settings.
|
|
|
|
--
|
|
|
|
-- For more information on modifying behavior, see:
|
|
|
|
--
|
|
|
|
-- https://github.com/yesodweb/yesod/wiki/Overriding-widgetFile
|
|
|
|
widgetFileSettings :: WidgetFileSettings
|
|
|
|
widgetFileSettings = def
|
|
|
|
|
|
|
|
-- | How static files should be combined.
|
|
|
|
combineSettings :: CombineSettings
|
|
|
|
combineSettings = def
|
|
|
|
|
|
|
|
-- The rest of this file contains settings which rarely need changing by a
|
|
|
|
-- user.
|
|
|
|
|
|
|
|
widgetFile :: String -> Q Exp
|
2016-02-14 18:10:21 +09:00
|
|
|
widgetFile =
|
|
|
|
let wf =
|
|
|
|
if appReloadTemplates compileTimeAppSettings
|
2016-02-13 12:35:30 +09:00
|
|
|
then widgetFileReload
|
2016-02-14 18:10:21 +09:00
|
|
|
else widgetFileNoReload
|
|
|
|
in wf widgetFileSettings
|
2016-02-13 12:35:30 +09:00
|
|
|
|
|
|
|
-- | Raw bytes at compile time of @config/settings.yml@
|
|
|
|
configSettingsYmlBS :: ByteString
|
|
|
|
configSettingsYmlBS = $(embedFile configSettingsYml)
|
|
|
|
|
|
|
|
-- | @config/settings.yml@, parsed to a @Value@.
|
|
|
|
configSettingsYmlValue :: Value
|
|
|
|
configSettingsYmlValue = either throw id $ decodeEither' configSettingsYmlBS
|
|
|
|
|
2016-02-14 18:10:21 +09:00
|
|
|
-- | A version of @AppSettings@ parsed at compile time from
|
|
|
|
-- @config/settings.yml@.
|
2016-02-13 12:35:30 +09:00
|
|
|
compileTimeAppSettings :: AppSettings
|
|
|
|
compileTimeAppSettings =
|
|
|
|
case fromJSON $ applyEnvValue False mempty configSettingsYmlValue of
|
2016-02-14 18:10:21 +09:00
|
|
|
Error e -> error e
|
2016-02-13 12:35:30 +09:00
|
|
|
Success settings -> settings
|
|
|
|
|
|
|
|
-- The following two functions can be used to combine multiple CSS or JS files
|
|
|
|
-- at compile time to decrease the number of http requests.
|
|
|
|
-- Sample usage (inside a Widget):
|
|
|
|
--
|
|
|
|
-- > $(combineStylesheets 'StaticR [style1_css, style2_css])
|
|
|
|
|
|
|
|
combineStylesheets :: Name -> [Route Static] -> Q Exp
|
2016-02-14 18:10:21 +09:00
|
|
|
combineStylesheets =
|
|
|
|
combineStylesheets'
|
|
|
|
(appSkipCombining compileTimeAppSettings)
|
|
|
|
combineSettings
|
2016-02-13 12:35:30 +09:00
|
|
|
|
|
|
|
combineScripts :: Name -> [Route Static] -> Q Exp
|
2016-02-14 18:10:21 +09:00
|
|
|
combineScripts =
|
|
|
|
combineScripts'
|
|
|
|
(appSkipCombining compileTimeAppSettings)
|
|
|
|
combineSettings
|