2021-10-22 12:49:03 -04:00
|
|
|
{-|
|
|
|
|
|
2021-11-17 13:15:36 -05:00
|
|
|
Module : Network.Gemini.Capsule
|
|
|
|
Description : Gemini capsule stuff
|
2021-10-22 12:49:03 -04:00
|
|
|
Copyright : (C) Jonathan Lamothe
|
|
|
|
License : AGPL-3.0-or-later
|
|
|
|
Maintainer : jonathan@jlamothe.net
|
|
|
|
Stability : experimental
|
|
|
|
Portability : POSIX
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU Affero General Public License as
|
|
|
|
published by the Free Software Foundation, either version 3 of the
|
|
|
|
License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful, but
|
|
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Affero General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public
|
|
|
|
License along with this program. If not, see
|
|
|
|
<https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
-}
|
|
|
|
|
2021-11-18 20:00:52 -05:00
|
|
|
{-# LANGUAGE LambdaCase, ScopedTypeVariables #-}
|
2021-10-24 12:47:37 -04:00
|
|
|
|
2021-11-17 13:15:36 -05:00
|
|
|
module Network.Gemini.Capsule (
|
2021-11-18 20:00:52 -05:00
|
|
|
runGemCapsule
|
2021-10-22 14:29:15 -04:00
|
|
|
) where
|
|
|
|
|
2021-11-05 20:56:15 -04:00
|
|
|
import Control.Concurrent (forkIO)
|
|
|
|
import Control.Exception (IOException, try)
|
2021-11-10 15:04:00 -05:00
|
|
|
import Control.Exception.Base (bracket, finally)
|
2021-11-05 20:56:15 -04:00
|
|
|
import Control.Monad (void)
|
2021-11-10 15:04:00 -05:00
|
|
|
import qualified Data.Connection as C
|
2021-11-10 14:06:20 -05:00
|
|
|
import Data.IORef (IORef, newIORef, readIORef, writeIORef)
|
2021-11-05 20:56:15 -04:00
|
|
|
import Data.TLSSetting (makeServerParams)
|
|
|
|
import Data.X509 (Certificate, CertificateChain (..), getSigned, signedObject)
|
2021-11-10 15:04:00 -05:00
|
|
|
import qualified Network.Socket as S
|
2021-11-05 20:56:15 -04:00
|
|
|
import Network.TLS (ServerParams, onClientCertificate, serverHooks)
|
2021-10-28 20:50:20 -04:00
|
|
|
import System.IO.Streams.TCP (bindAndListen)
|
2021-11-13 11:26:15 -05:00
|
|
|
import System.IO.Streams.TLS (accept)
|
2021-10-22 14:29:15 -04:00
|
|
|
|
2021-11-17 13:15:36 -05:00
|
|
|
import Network.Gemini.Capsule.Internal
|
|
|
|
import Network.Gemini.Capsule.Types
|
2021-10-22 14:29:15 -04:00
|
|
|
|
2021-11-17 15:32:39 -05:00
|
|
|
-- | Builds and runs a Gemini capsule
|
|
|
|
runGemCapsule
|
|
|
|
:: GemCapSettings
|
|
|
|
-- ^ The capsule settings
|
2021-10-28 20:50:20 -04:00
|
|
|
-> GemHandler
|
|
|
|
-- ^ The handler
|
2021-11-05 20:56:15 -04:00
|
|
|
-> IO a
|
2021-11-17 15:32:39 -05:00
|
|
|
runGemCapsule settings handler = bracket
|
2021-10-28 20:50:20 -04:00
|
|
|
( bindAndListen
|
2021-11-17 15:32:39 -05:00
|
|
|
(capConnections settings)
|
|
|
|
(fromIntegral $ capPort settings)
|
2021-10-28 20:50:20 -04:00
|
|
|
)
|
2021-11-10 15:04:00 -05:00
|
|
|
S.close
|
2021-11-05 20:56:15 -04:00
|
|
|
( \sock -> do
|
|
|
|
params <- makeServerParams
|
2021-11-17 15:32:39 -05:00
|
|
|
(capCert settings)
|
|
|
|
(capCertChain settings)
|
|
|
|
(capKey settings)
|
2021-11-05 20:56:15 -04:00
|
|
|
listenLoop sock params handler
|
|
|
|
)
|
2021-10-28 20:50:20 -04:00
|
|
|
|
2021-11-10 15:04:00 -05:00
|
|
|
listenLoop :: S.Socket -> ServerParams -> GemHandler -> IO a
|
2021-11-05 20:56:15 -04:00
|
|
|
listenLoop sock params handler = do
|
|
|
|
certRef <- newIORef Nothing
|
|
|
|
let params' = adjustServerParams certRef params
|
|
|
|
try (accept params' sock) >>= \case
|
2021-11-18 20:00:52 -05:00
|
|
|
Left (_::IOException) -> return ()
|
|
|
|
Right conn -> void $ forkIO $ finally
|
2021-11-10 15:04:00 -05:00
|
|
|
(readIORef certRef >>= runConnection conn handler)
|
|
|
|
(C.close conn)
|
2021-11-05 20:56:15 -04:00
|
|
|
listenLoop sock params handler
|
|
|
|
|
|
|
|
adjustServerParams
|
|
|
|
:: IORef (Maybe Certificate)
|
|
|
|
-> ServerParams
|
|
|
|
-> ServerParams
|
|
|
|
adjustServerParams certRef params = let
|
|
|
|
hooks = serverHooks params
|
|
|
|
certHook = onClientCertificate hooks
|
|
|
|
|
2021-11-12 19:52:09 -05:00
|
|
|
certHook' chain = do
|
|
|
|
case chain of
|
|
|
|
CertificateChain [] -> return ()
|
|
|
|
CertificateChain (se:_) -> do
|
|
|
|
let cert = signedObject $ getSigned se
|
|
|
|
writeIORef certRef (Just cert)
|
|
|
|
certHook chain
|
2021-11-05 20:56:15 -04:00
|
|
|
|
|
|
|
hooks' = hooks { onClientCertificate = certHook' }
|
|
|
|
in params { serverHooks = hooks' }
|
|
|
|
|
2021-10-22 12:49:03 -04:00
|
|
|
--jl
|