1
0
Fork 0
mirror of https://code.sup39.dev/repos/Wqawg synced 2025-01-10 17:46:45 +09:00
vervis/src/Vervis/Settings.hs

243 lines
10 KiB
Haskell
Raw Normal View History

{- This file is part of Vervis.
-
- Written in 2016, 2018, 2019 by fr33domlover <fr33domlover@riseup.net>.
-
- 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-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.
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)
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)
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
{ -- | 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
-- | 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
-- | Base for all generated URLs. If @Nothing@, determined from the
-- request headers.
2016-02-13 12:35:30 +09:00
, appRoot :: Maybe Text
-- | The instance's host (e.g. \"dev.angeley.es\"), currently used just
-- for display.
, appInstanceHost :: Text
-- | Host/interface the server should bind to.
2016-02-13 12:35:30 +09:00
, appHost :: HostPreference
-- | Port to listen on
2016-02-13 12:35:30 +09:00
, appPort :: Int
-- | 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
-- | Path of session cookie encryption key file
, appClientSessionKeyFile :: FilePath
-- | Idle timeout for session cookie expiration
, appClientSessionTimeout :: TimeInterval
-- 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
-- | Use detailed request logging system
2016-02-13 12:35:30 +09:00
, appDetailedRequestLogging :: Bool
-- | Should all log messages be displayed?
2016-02-13 12:35:30 +09:00
, appShouldLogAll :: Bool
-- | Use the reload version of templates
2016-02-13 12:35:30 +09:00
, appReloadTemplates :: Bool
-- | Assume that files in the static dir may change after compilation
2016-02-13 12:35:30 +09:00
, appMutableStatic :: Bool
-- | Perform no stylesheet/script combining
2016-02-13 12:35:30 +09:00
, appSkipCombining :: Bool
-- | Load SVG font file from the data file path of the @SVGFonts@
-- library, instead of the app's production runtime data directory.
, appLoadFontFromLibData :: Bool
-- | 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
-- | Port for the SSH server component to listen on
2016-03-05 12:56:25 +09:00
, appSshPort :: Int
-- | Path to the server's SSH private key file
2016-03-05 12:56:25 +09:00
, appSshKeyFile :: FilePath
-- | 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
-- | Whether to send verification email to new users. If not, users will
-- be instantly considered verified, without sending email.
, appEmailVerification :: Bool
-- | SMTP server details for sending email, and other email related
-- details. If set to 'Nothing', no email will be sent.
, appMail :: Maybe MailSettings
-- | Signing key file for signing object capabilities sent to remote
-- users
, appCapabilitySigningKeyFile :: FilePath
-- | Salt for encoding and decoding hashids
, appHashidsSaltFile :: FilePath
-- | 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
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"
appMaxInstanceKeys <- o .:? "max-instance-keys"
appMaxActorKeys <- o .:? "max-actor-keys"
2016-02-13 12:35:30 +09:00
appRoot <- o .:? "approot"
appInstanceHost <- o .: "instance-host"
2016-02-13 12:35:30 +09:00
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"
appClientSessionKeyFile <- o .: "client-session-key"
appClientSessionTimeout <- interval <$> o .: "client-session-timeout"
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
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"
appRegister <- o .: "registration"
2016-07-28 06:46:48 +09:00
appAccounts <- o .: "max-accounts"
appEmailVerification <- o .:? "email-verification" .!= not defaultDev
appMail <- o .:? "mail"
2016-02-13 12:35:30 +09:00
appCapabilitySigningKeyFile <- o .: "capability-signing-key"
appHashidsSaltFile <- o .: "hashids-salt-file"
appRejectOnMaxKeys <- o .: "reject-on-max-keys"
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
widgetFile =
let wf =
if appReloadTemplates compileTimeAppSettings
2016-02-13 12:35:30 +09:00
then widgetFileReload
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
-- | 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
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
combineStylesheets =
combineStylesheets'
(appSkipCombining compileTimeAppSettings)
combineSettings
2016-02-13 12:35:30 +09:00
combineScripts :: Name -> [Route Static] -> Q Exp
combineScripts =
combineScripts'
(appSkipCombining compileTimeAppSettings)
combineSettings