module Hydra.Network.Ouroboros (
withIOManager,
module Hydra.Network,
module Hydra.Network.Ouroboros,
module Hydra.Network.Ouroboros.VersionedProtocol,
) where
import Control.Monad.Class.MonadAsync (wait)
import Hydra.Network.Ouroboros.VersionedProtocol (
HydraNetworkConfig (..),
HydraVersionedProtocolData (..),
hydraVersionedProtocolCodec,
hydraVersionedProtocolDataCodec,
)
import Hydra.Prelude
import Codec.CBOR.Term (Term)
import Codec.CBOR.Term qualified as CBOR
import Control.Concurrent.STM (
TChan,
dupTChan,
newBroadcastTChanIO,
readTChan,
writeTChan,
)
import Control.Exception (IOException)
import Data.Aeson (object, (.=))
import Data.Aeson qualified as Aeson
import Data.Aeson.Types qualified as Aeson
import Data.Map.Strict as Map
import Data.Text qualified as T
import Hydra.Logging (Tracer (..), nullTracer)
import Hydra.Network (
Host (..),
Network (..),
NetworkCallback (..),
NetworkComponent,
PortNumber,
)
import Hydra.Network.Message (
HydraHandshakeRefused (..),
HydraVersionedProtocolNumber (..),
KnownHydraVersions (..),
)
import Hydra.Network.Ouroboros.Client as FireForget (
FireForgetClient (..),
fireForgetClientPeer,
)
import Hydra.Network.Ouroboros.Server as FireForget (
FireForgetServer (..),
fireForgetServerPeer,
)
import Hydra.Network.Ouroboros.Type (
FireForget (..),
Message (..),
codecFireForget,
)
import Network.Mux.Compat (
WithMuxBearer (..),
)
import Network.Socket (
AddrInfo (addrAddress),
NameInfoFlag (..),
SockAddr,
Socket,
defaultHints,
getAddrInfo,
getNameInfo,
getPeerName,
)
import Network.TypedProtocol.Codec (
AnyMessageAndAgency (..),
)
import Network.TypedProtocol.Pipelined ()
import Ouroboros.Network.Driver.Simple (
TraceSendRecv (..),
)
import Ouroboros.Network.ErrorPolicy (
ErrorPolicyTrace,
WithAddr (WithAddr),
nullErrorPolicies,
)
import Ouroboros.Network.IOManager (IOManager, withIOManager)
import Ouroboros.Network.Mux (
MiniProtocol (
MiniProtocol,
miniProtocolLimits,
miniProtocolNum,
miniProtocolRun
),
MiniProtocolCb,
MiniProtocolLimits (..),
MiniProtocolNum (MiniProtocolNum),
MuxMode (..),
OuroborosApplication (..),
OuroborosApplicationWithMinimalCtx,
RunMiniProtocol (..),
mkMiniProtocolCbFromPeer,
)
import Ouroboros.Network.Protocol.Handshake.Codec (codecHandshake, noTimeLimitsHandshake)
import Ouroboros.Network.Protocol.Handshake.Type (Handshake, HandshakeProtocolError (..), Message (..), RefuseReason (..))
import Ouroboros.Network.Protocol.Handshake.Version (acceptableVersion, queryVersion, simpleSingletonVersions)
import Ouroboros.Network.Server.Socket (AcceptedConnectionsLimit (AcceptedConnectionsLimit))
import Ouroboros.Network.Snocket (makeSocketBearer, socketSnocket)
import Ouroboros.Network.Socket (
AcceptConnectionsPolicyTrace,
ConnectionId (..),
HandshakeCallbacks (..),
NetworkConnectTracers (..),
NetworkServerTracers (..),
SomeResponderApplication (..),
connectToNodeSocket,
newNetworkMutableState,
withServerNode,
)
import Ouroboros.Network.Subscription (
IPSubscriptionTarget (IPSubscriptionTarget),
SubscriptionTrace,
WithIPList,
)
import Ouroboros.Network.Subscription qualified as Subscription
import Ouroboros.Network.Subscription.Ip (SubscriptionParams (..), WithIPList (WithIPList))
import Ouroboros.Network.Subscription.Worker (LocalAddresses (LocalAddresses))
withOuroborosNetwork ::
forall inbound outbound.
(ToCBOR outbound, FromCBOR outbound) =>
(ToCBOR inbound, FromCBOR inbound) =>
Tracer IO (WithHost (TraceOuroborosNetwork outbound)) ->
HydraNetworkConfig ->
(HydraHandshakeRefused -> IO ()) ->
NetworkComponent IO inbound outbound ()
withOuroborosNetwork :: forall inbound outbound.
(ToCBOR outbound, FromCBOR outbound, ToCBOR inbound,
FromCBOR inbound) =>
Tracer IO (WithHost (TraceOuroborosNetwork outbound))
-> HydraNetworkConfig
-> (HydraHandshakeRefused -> IO ())
-> NetworkComponent IO inbound outbound ()
withOuroborosNetwork
Tracer IO (WithHost (TraceOuroborosNetwork outbound))
tracer
HydraNetworkConfig{HydraVersionedProtocolNumber
protocolVersion :: HydraVersionedProtocolNumber
$sel:protocolVersion:HydraNetworkConfig :: HydraNetworkConfig -> HydraVersionedProtocolNumber
protocolVersion, Host
localHost :: Host
$sel:localHost:HydraNetworkConfig :: HydraNetworkConfig -> Host
localHost, [Host]
remoteHosts :: [Host]
$sel:remoteHosts:HydraNetworkConfig :: HydraNetworkConfig -> [Host]
remoteHosts}
HydraHandshakeRefused -> IO ()
handshakeCallback
NetworkCallback{inbound -> IO ()
deliver :: inbound -> IO ()
$sel:deliver:NetworkCallback :: forall msg (m :: * -> *). NetworkCallback msg m -> msg -> m ()
deliver}
Network IO outbound -> IO ()
between = do
TChan outbound
bchan <- IO (TChan outbound)
forall a. IO (TChan a)
newBroadcastTChanIO
let newBroadcastChannel :: IO (TChan outbound)
newBroadcastChannel = STM IO (TChan outbound) -> IO (TChan outbound)
forall a. HasCallStack => STM IO a -> IO a
forall (m :: * -> *) a.
(MonadSTM m, HasCallStack) =>
STM m a -> m a
atomically (STM IO (TChan outbound) -> IO (TChan outbound))
-> STM IO (TChan outbound) -> IO (TChan outbound)
forall a b. (a -> b) -> a -> b
$ TChan outbound -> STM (TChan outbound)
forall a. TChan a -> STM (TChan a)
dupTChan TChan outbound
bchan
(IOManager -> IO ()) -> IO ()
WithIOManager
withIOManager ((IOManager -> IO ()) -> IO ()) -> (IOManager -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \IOManager
iomgr -> do
IOManager
-> OuroborosApplicationWithMinimalCtx
'ResponderMode SockAddr LByteString IO Void ()
-> IO ()
-> IO ()
forall a b.
IOManager
-> OuroborosApplicationWithMinimalCtx
'ResponderMode SockAddr LByteString IO a b
-> IO b
-> IO ()
withServerListening IOManager
iomgr OuroborosApplicationWithMinimalCtx
'ResponderMode SockAddr LByteString IO Void ()
forall addr.
OuroborosApplicationWithMinimalCtx
'ResponderMode addr LByteString IO Void ()
hydraServer (IO () -> IO ()) -> IO () -> IO ()
forall a b. (a -> b) -> a -> b
$
IO Void -> IO () -> IO ()
forall a b. IO a -> IO b -> IO ()
forall (m :: * -> *) a b. MonadAsync m => m a -> m b -> m ()
race_ (IOManager
-> IO (TChan outbound)
-> (TChan outbound
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void)
-> IO Void
forall t.
IOManager
-> IO t
-> (t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void)
-> IO Void
connect IOManager
iomgr IO (TChan outbound)
newBroadcastChannel TChan outbound
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void
forall addr.
TChan outbound
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode addr LByteString IO () Void
hydraClient) (IO () -> IO ()) -> IO () -> IO ()
forall a b. (a -> b) -> a -> b
$ do
Network IO outbound -> IO ()
between (Network IO outbound -> IO ()) -> Network IO outbound -> IO ()
forall a b. (a -> b) -> a -> b
$
Network
{ $sel:broadcast:Network :: outbound -> IO ()
broadcast = STM () -> IO ()
STM IO () -> IO ()
forall a. HasCallStack => STM IO a -> IO a
forall (m :: * -> *) a.
(MonadSTM m, HasCallStack) =>
STM m a -> m a
atomically (STM () -> IO ()) -> (outbound -> STM ()) -> outbound -> IO ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TChan outbound -> outbound -> STM ()
forall a. TChan a -> a -> STM ()
writeTChan TChan outbound
bchan
}
where
resolveSockAddr :: Host -> IO SockAddr
resolveSockAddr :: Host -> IO SockAddr
resolveSockAddr Host{Text
hostname :: Text
$sel:hostname:Host :: Host -> Text
hostname, PortNumber
port :: PortNumber
$sel:port:Host :: Host -> PortNumber
port} = do
[AddrInfo]
is <- Maybe AddrInfo
-> Maybe ServiceName -> Maybe ServiceName -> IO [AddrInfo]
getAddrInfo (AddrInfo -> Maybe AddrInfo
forall a. a -> Maybe a
Just AddrInfo
defaultHints) (ServiceName -> Maybe ServiceName
forall a. a -> Maybe a
Just (ServiceName -> Maybe ServiceName)
-> ServiceName -> Maybe ServiceName
forall a b. (a -> b) -> a -> b
$ Text -> ServiceName
forall a. ToString a => a -> ServiceName
toString Text
hostname) (ServiceName -> Maybe ServiceName
forall a. a -> Maybe a
Just (ServiceName -> Maybe ServiceName)
-> ServiceName -> Maybe ServiceName
forall a b. (a -> b) -> a -> b
$ PortNumber -> ServiceName
forall b a. (Show a, IsString b) => a -> b
show PortNumber
port)
case [AddrInfo]
is of
(AddrInfo
info : [AddrInfo]
_) -> SockAddr -> IO SockAddr
forall a. a -> IO a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (SockAddr -> IO SockAddr) -> SockAddr -> IO SockAddr
forall a b. (a -> b) -> a -> b
$ AddrInfo -> SockAddr
addrAddress AddrInfo
info
[AddrInfo]
_ -> Text -> IO SockAddr
forall a t. (HasCallStack, IsText t) => t -> a
error Text
"getAdrrInfo failed.. do proper error handling"
getHost :: SockAddr -> IO Host
getHost :: SockAddr -> IO Host
getHost SockAddr
sockAddr = do
(Maybe ServiceName
mHost, Maybe ServiceName
mPort) <- [NameInfoFlag]
-> Bool
-> Bool
-> SockAddr
-> IO (Maybe ServiceName, Maybe ServiceName)
getNameInfo [NameInfoFlag
NI_NUMERICHOST, NameInfoFlag
NI_NUMERICSERV] Bool
True Bool
True SockAddr
sockAddr
IO Host -> (Host -> IO Host) -> Maybe Host -> IO Host
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (Text -> IO Host
forall a t. (HasCallStack, IsText t) => t -> a
error Text
"getNameInfo failed.. do proper error handling") Host -> IO Host
forall a. a -> IO a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Maybe Host -> IO Host) -> Maybe Host -> IO Host
forall a b. (a -> b) -> a -> b
$ do
Text
host <- ServiceName -> Text
T.pack (ServiceName -> Text) -> Maybe ServiceName -> Maybe Text
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe ServiceName
mHost
PortNumber
port <- ServiceName -> Maybe PortNumber
forall a. Read a => ServiceName -> Maybe a
readMaybe (ServiceName -> Maybe PortNumber)
-> Maybe ServiceName -> Maybe PortNumber
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Maybe ServiceName
mPort
Host -> Maybe Host
forall a. a -> Maybe a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Host -> Maybe Host) -> Host -> Maybe Host
forall a b. (a -> b) -> a -> b
$ Text -> PortNumber -> Host
Host Text
host PortNumber
port
connect ::
IOManager ->
IO t ->
( t ->
OuroborosApplicationWithMinimalCtx
InitiatorMode
SockAddr
LByteString
IO
()
Void
) ->
IO Void
connect :: forall t.
IOManager
-> IO t
-> (t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void)
-> IO Void
connect IOManager
iomgr IO t
newBroadcastChannel t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void
app = do
NetworkMutableState SockAddr
networkState <- IO (NetworkMutableState SockAddr)
forall addr. IO (NetworkMutableState addr)
newNetworkMutableState
SockAddr
localAddr <- Host -> IO SockAddr
resolveSockAddr Host
localHost{port = 0}
[SockAddr]
remoteAddrs <- [Host] -> (Host -> IO SockAddr) -> IO [SockAddr]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
t a -> (a -> m b) -> m (t b)
forM [Host]
remoteHosts Host -> IO SockAddr
resolveSockAddr
let sn :: SocketSnocket
sn = IOManager -> SocketSnocket
socketSnocket IOManager
iomgr
SocketSnocket
-> Tracer IO (WithIPList (SubscriptionTrace SockAddr))
-> Tracer IO (WithAddr SockAddr ErrorPolicyTrace)
-> NetworkMutableState SockAddr
-> IPSubscriptionParams ()
-> (Socket -> IO ())
-> IO Void
forall a.
SocketSnocket
-> Tracer IO (WithIPList (SubscriptionTrace SockAddr))
-> Tracer IO (WithAddr SockAddr ErrorPolicyTrace)
-> NetworkMutableState SockAddr
-> IPSubscriptionParams a
-> (Socket -> IO a)
-> IO Void
Subscription.ipSubscriptionWorker
SocketSnocket
sn
((WithIPList (SubscriptionTrace SockAddr)
-> WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithIPList (SubscriptionTrace SockAddr))
forall a' a. (a' -> a) -> Tracer IO a -> Tracer IO a'
forall (f :: * -> *) a' a.
Contravariant f =>
(a' -> a) -> f a -> f a'
contramap (Host
-> TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound)
forall trace. Host -> trace -> WithHost trace
WithHost Host
localHost (TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound))
-> (WithIPList (SubscriptionTrace SockAddr)
-> TraceOuroborosNetwork outbound)
-> WithIPList (SubscriptionTrace SockAddr)
-> WithHost (TraceOuroborosNetwork outbound)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. WithIPList (SubscriptionTrace SockAddr)
-> TraceOuroborosNetwork outbound
forall {k} (msg :: k).
WithIPList (SubscriptionTrace SockAddr)
-> TraceOuroborosNetwork msg
TraceSubscriptions) Tracer IO (WithHost (TraceOuroborosNetwork outbound))
tracer)
((WithAddr SockAddr ErrorPolicyTrace
-> WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithAddr SockAddr ErrorPolicyTrace)
forall a' a. (a' -> a) -> Tracer IO a -> Tracer IO a'
forall (f :: * -> *) a' a.
Contravariant f =>
(a' -> a) -> f a -> f a'
contramap (Host
-> TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound)
forall trace. Host -> trace -> WithHost trace
WithHost Host
localHost (TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound))
-> (WithAddr SockAddr ErrorPolicyTrace
-> TraceOuroborosNetwork outbound)
-> WithAddr SockAddr ErrorPolicyTrace
-> WithHost (TraceOuroborosNetwork outbound)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. WithAddr SockAddr ErrorPolicyTrace
-> TraceOuroborosNetwork outbound
forall {k} (msg :: k).
WithAddr SockAddr ErrorPolicyTrace -> TraceOuroborosNetwork msg
TraceErrorPolicy) Tracer IO (WithHost (TraceOuroborosNetwork outbound))
tracer)
NetworkMutableState SockAddr
networkState
(SockAddr -> [SockAddr] -> IPSubscriptionParams ()
forall a.
SockAddr -> [SockAddr] -> SubscriptionParams a IPSubscriptionTarget
subscriptionParams SockAddr
localAddr [SockAddr]
remoteAddrs)
( \Socket
sock ->
IOManager
-> IO t
-> (t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void)
-> Socket
-> IO ()
forall t.
IOManager
-> IO t
-> (t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void)
-> Socket
-> IO ()
actualConnect IOManager
iomgr IO t
newBroadcastChannel t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void
app Socket
sock IO ()
-> (HandshakeProtocolError HydraVersionedProtocolNumber -> IO ())
-> IO ()
forall e a. Exception e => IO a -> (e -> IO a) -> IO a
forall (m :: * -> *) e a.
(MonadCatch m, Exception e) =>
m a -> (e -> m a) -> m a
`catch` \HandshakeProtocolError HydraVersionedProtocolNumber
e -> do
Host
host <- SockAddr -> IO Host
getHost (SockAddr -> IO Host) -> IO SockAddr -> IO Host
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< Socket -> IO SockAddr
getPeerName Socket
sock
Host
-> HandshakeProtocolError HydraVersionedProtocolNumber -> IO ()
onHandshakeError Host
host HandshakeProtocolError HydraVersionedProtocolNumber
e
)
onHandshakeError :: Host -> HandshakeProtocolError HydraVersionedProtocolNumber -> IO ()
onHandshakeError :: Host
-> HandshakeProtocolError HydraVersionedProtocolNumber -> IO ()
onHandshakeError Host
remoteHost = \case
HandshakeError (VersionMismatch [HydraVersionedProtocolNumber]
theirVersions [Int]
_) -> do
HydraHandshakeRefused -> IO ()
handshakeCallback
HydraHandshakeRefused
{ $sel:ourVersion:HydraHandshakeRefused :: HydraVersionedProtocolNumber
ourVersion = HydraVersionedProtocolNumber
protocolVersion
, $sel:theirVersions:HydraHandshakeRefused :: KnownHydraVersions
theirVersions = [HydraVersionedProtocolNumber] -> KnownHydraVersions
KnownHydraVersions [HydraVersionedProtocolNumber]
theirVersions
, Host
remoteHost :: Host
$sel:remoteHost:HydraHandshakeRefused :: Host
remoteHost
}
HandshakeProtocolError HydraVersionedProtocolNumber
_ ->
HydraHandshakeRefused -> IO ()
handshakeCallback
HydraHandshakeRefused
{ $sel:ourVersion:HydraHandshakeRefused :: HydraVersionedProtocolNumber
ourVersion = HydraVersionedProtocolNumber
protocolVersion
, $sel:theirVersions:HydraHandshakeRefused :: KnownHydraVersions
theirVersions = KnownHydraVersions
NoKnownHydraVersions
, Host
remoteHost :: Host
$sel:remoteHost:HydraHandshakeRefused :: Host
remoteHost
}
subscriptionParams ::
SockAddr ->
[SockAddr] ->
SubscriptionParams a IPSubscriptionTarget
subscriptionParams :: forall a.
SockAddr -> [SockAddr] -> SubscriptionParams a IPSubscriptionTarget
subscriptionParams SockAddr
localAddr [SockAddr]
remoteAddrs =
SubscriptionParams
{ spLocalAddresses :: LocalAddresses SockAddr
spLocalAddresses = Maybe SockAddr
-> Maybe SockAddr -> Maybe SockAddr -> LocalAddresses SockAddr
forall addr.
Maybe addr -> Maybe addr -> Maybe addr -> LocalAddresses addr
LocalAddresses (SockAddr -> Maybe SockAddr
forall a. a -> Maybe a
Just SockAddr
localAddr) Maybe SockAddr
forall a. Maybe a
Nothing Maybe SockAddr
forall a. Maybe a
Nothing
, spConnectionAttemptDelay :: SockAddr -> Maybe DiffTime
spConnectionAttemptDelay = Maybe DiffTime -> SockAddr -> Maybe DiffTime
forall a b. a -> b -> a
const Maybe DiffTime
forall a. Maybe a
Nothing
, spErrorPolicies :: ErrorPolicies
spErrorPolicies = ErrorPolicies
nullErrorPolicies
, spSubscriptionTarget :: IPSubscriptionTarget
spSubscriptionTarget = [SockAddr] -> Int -> IPSubscriptionTarget
IPSubscriptionTarget [SockAddr]
remoteAddrs ([SockAddr] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [SockAddr]
remoteAddrs)
}
actualConnect ::
IOManager ->
IO t ->
(t -> OuroborosApplicationWithMinimalCtx 'InitiatorMode SockAddr LByteString IO () Void) ->
Socket ->
IO ()
actualConnect :: forall t.
IOManager
-> IO t
-> (t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void)
-> Socket
-> IO ()
actualConnect IOManager
iomgr IO t
newBroadcastChannel t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void
app Socket
sn = do
t
chan <- IO t
newBroadcastChannel
IOManager
-> Codec
(Handshake HydraVersionedProtocolNumber Term)
DeserialiseFailure
IO
LByteString
-> ProtocolTimeLimits (Handshake HydraVersionedProtocolNumber Term)
-> VersionDataCodec
Term HydraVersionedProtocolNumber HydraVersionedProtocolData
-> NetworkConnectTracers SockAddr HydraVersionedProtocolNumber
-> HandshakeCallbacks HydraVersionedProtocolData
-> Versions
HydraVersionedProtocolNumber
HydraVersionedProtocolData
(OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void)
-> Socket
-> IO ()
forall (appType :: MuxMode) vNumber vData a b.
(Ord vNumber, Typeable vNumber, Show vNumber,
HasInitiator appType ~ 'True) =>
IOManager
-> Codec (Handshake vNumber Term) DeserialiseFailure IO LByteString
-> ProtocolTimeLimits (Handshake vNumber Term)
-> VersionDataCodec Term vNumber vData
-> NetworkConnectTracers SockAddr vNumber
-> HandshakeCallbacks vData
-> Versions
vNumber
vData
(OuroborosApplicationWithMinimalCtx
appType SockAddr LByteString IO a b)
-> Socket
-> IO ()
connectToNodeSocket
IOManager
iomgr
(CodecCBORTerm (ServiceName, Maybe Int) HydraVersionedProtocolNumber
-> Codec
(Handshake HydraVersionedProtocolNumber Term)
DeserialiseFailure
IO
LByteString
forall vNumber (m :: * -> *) failure.
(MonadST m, Ord vNumber, Show failure) =>
CodecCBORTerm (failure, Maybe Int) vNumber
-> Codec (Handshake vNumber Term) DeserialiseFailure m LByteString
codecHandshake CodecCBORTerm (ServiceName, Maybe Int) HydraVersionedProtocolNumber
hydraVersionedProtocolCodec)
ProtocolTimeLimits (Handshake HydraVersionedProtocolNumber Term)
forall {k} (vNumber :: k).
ProtocolTimeLimits (Handshake vNumber Term)
noTimeLimitsHandshake
VersionDataCodec
Term HydraVersionedProtocolNumber HydraVersionedProtocolData
hydraVersionedProtocolDataCodec
NetworkConnectTracers SockAddr HydraVersionedProtocolNumber
networkConnectTracers
((HydraVersionedProtocolData
-> HydraVersionedProtocolData -> Accept HydraVersionedProtocolData)
-> (HydraVersionedProtocolData -> Bool)
-> HandshakeCallbacks HydraVersionedProtocolData
forall vData.
(vData -> vData -> Accept vData)
-> (vData -> Bool) -> HandshakeCallbacks vData
HandshakeCallbacks HydraVersionedProtocolData
-> HydraVersionedProtocolData -> Accept HydraVersionedProtocolData
forall v. Acceptable v => v -> v -> Accept v
acceptableVersion HydraVersionedProtocolData -> Bool
forall v. Queryable v => v -> Bool
queryVersion)
(HydraVersionedProtocolNumber
-> HydraVersionedProtocolData
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void
-> Versions
HydraVersionedProtocolNumber
HydraVersionedProtocolData
(OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void)
forall vNum vData r. vNum -> vData -> r -> Versions vNum vData r
simpleSingletonVersions HydraVersionedProtocolNumber
protocolVersion HydraVersionedProtocolData
MkHydraVersionedProtocolData (t
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode SockAddr LByteString IO () Void
app t
chan))
Socket
sn
where
networkConnectTracers :: NetworkConnectTracers SockAddr HydraVersionedProtocolNumber
networkConnectTracers :: NetworkConnectTracers SockAddr HydraVersionedProtocolNumber
networkConnectTracers =
NetworkConnectTracers
{ nctMuxTracer :: Tracer IO (WithMuxBearer (ConnectionId SockAddr) MuxTrace)
nctMuxTracer = Tracer IO (WithMuxBearer (ConnectionId SockAddr) MuxTrace)
forall (m :: * -> *) a. Applicative m => Tracer m a
nullTracer
, nctHandshakeTracer :: Tracer
IO
(WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term)))
nctHandshakeTracer = (WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithHost (TraceOuroborosNetwork outbound))
-> Tracer
IO
(WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term)))
forall a' a. (a' -> a) -> Tracer IO a -> Tracer IO a'
forall (f :: * -> *) a' a.
Contravariant f =>
(a' -> a) -> f a -> f a'
contramap (Host
-> TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound)
forall trace. Host -> trace -> WithHost trace
WithHost Host
localHost (TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound))
-> (WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> TraceOuroborosNetwork outbound)
-> WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> WithHost (TraceOuroborosNetwork outbound)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> TraceOuroborosNetwork outbound
forall {k} (msg :: k).
WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> TraceOuroborosNetwork msg
TraceHandshake) Tracer IO (WithHost (TraceOuroborosNetwork outbound))
tracer
}
withServerListening ::
IOManager ->
OuroborosApplicationWithMinimalCtx 'ResponderMode SockAddr LByteString IO a b ->
IO b ->
IO ()
withServerListening :: forall a b.
IOManager
-> OuroborosApplicationWithMinimalCtx
'ResponderMode SockAddr LByteString IO a b
-> IO b
-> IO ()
withServerListening IOManager
iomgr OuroborosApplicationWithMinimalCtx
'ResponderMode SockAddr LByteString IO a b
app IO b
continuation = do
NetworkMutableState SockAddr
networkState <- IO (NetworkMutableState SockAddr)
forall addr. IO (NetworkMutableState addr)
newNetworkMutableState
SockAddr
localAddr <- Host -> IO SockAddr
resolveSockAddr Host
localHost
(IOException -> IO ()) -> IO () -> IO ()
forall e a. Exception e => (e -> IO a) -> IO a -> IO a
forall (m :: * -> *) e a.
(MonadCatch m, Exception e) =>
(e -> m a) -> m a -> m a
handle IOException -> IO ()
onIOException
(IO () -> IO ()) -> IO () -> IO ()
forall a b. (a -> b) -> a -> b
$ SocketSnocket
-> MakeBearer IO Socket
-> (Socket -> SockAddr -> IO ())
-> NetworkServerTracers SockAddr HydraVersionedProtocolNumber
-> NetworkMutableState SockAddr
-> AcceptedConnectionsLimit
-> SockAddr
-> Codec
(Handshake HydraVersionedProtocolNumber Term)
DeserialiseFailure
IO
LByteString
-> ProtocolTimeLimits (Handshake HydraVersionedProtocolNumber Term)
-> VersionDataCodec
Term HydraVersionedProtocolNumber HydraVersionedProtocolData
-> HandshakeCallbacks HydraVersionedProtocolData
-> Versions
HydraVersionedProtocolNumber
HydraVersionedProtocolData
(SomeResponderApplication SockAddr LByteString IO b)
-> ErrorPolicies
-> (SockAddr -> Async Void -> IO ())
-> IO ()
forall vNumber vData t fd addr b.
(Ord vNumber, Typeable vNumber, Show vNumber, Ord addr) =>
Snocket IO fd addr
-> MakeBearer IO fd
-> (fd -> addr -> IO ())
-> NetworkServerTracers addr vNumber
-> NetworkMutableState addr
-> AcceptedConnectionsLimit
-> addr
-> Codec (Handshake vNumber Term) DeserialiseFailure IO LByteString
-> ProtocolTimeLimits (Handshake vNumber Term)
-> VersionDataCodec Term vNumber vData
-> HandshakeCallbacks vData
-> Versions
vNumber vData (SomeResponderApplication addr LByteString IO b)
-> ErrorPolicies
-> (addr -> Async Void -> IO t)
-> IO t
withServerNode
(IOManager -> SocketSnocket
socketSnocket IOManager
iomgr)
MakeBearer IO Socket
makeSocketBearer
Socket -> SockAddr -> IO ()
forall a b. a -> b -> IO ()
notConfigureSocket
NetworkServerTracers SockAddr HydraVersionedProtocolNumber
networkServerTracers
NetworkMutableState SockAddr
networkState
(Word32 -> Word32 -> DiffTime -> AcceptedConnectionsLimit
AcceptedConnectionsLimit Word32
forall a. Bounded a => a
maxBound Word32
forall a. Bounded a => a
maxBound DiffTime
0)
SockAddr
localAddr
(CodecCBORTerm (ServiceName, Maybe Int) HydraVersionedProtocolNumber
-> Codec
(Handshake HydraVersionedProtocolNumber Term)
DeserialiseFailure
IO
LByteString
forall vNumber (m :: * -> *) failure.
(MonadST m, Ord vNumber, Show failure) =>
CodecCBORTerm (failure, Maybe Int) vNumber
-> Codec (Handshake vNumber Term) DeserialiseFailure m LByteString
codecHandshake CodecCBORTerm (ServiceName, Maybe Int) HydraVersionedProtocolNumber
hydraVersionedProtocolCodec)
ProtocolTimeLimits (Handshake HydraVersionedProtocolNumber Term)
forall {k} (vNumber :: k).
ProtocolTimeLimits (Handshake vNumber Term)
noTimeLimitsHandshake
VersionDataCodec
Term HydraVersionedProtocolNumber HydraVersionedProtocolData
hydraVersionedProtocolDataCodec
((HydraVersionedProtocolData
-> HydraVersionedProtocolData -> Accept HydraVersionedProtocolData)
-> (HydraVersionedProtocolData -> Bool)
-> HandshakeCallbacks HydraVersionedProtocolData
forall vData.
(vData -> vData -> Accept vData)
-> (vData -> Bool) -> HandshakeCallbacks vData
HandshakeCallbacks HydraVersionedProtocolData
-> HydraVersionedProtocolData -> Accept HydraVersionedProtocolData
forall v. Acceptable v => v -> v -> Accept v
acceptableVersion HydraVersionedProtocolData -> Bool
forall v. Queryable v => v -> Bool
queryVersion)
(HydraVersionedProtocolNumber
-> HydraVersionedProtocolData
-> SomeResponderApplication SockAddr LByteString IO b
-> Versions
HydraVersionedProtocolNumber
HydraVersionedProtocolData
(SomeResponderApplication SockAddr LByteString IO b)
forall vNum vData r. vNum -> vData -> r -> Versions vNum vData r
simpleSingletonVersions HydraVersionedProtocolNumber
protocolVersion HydraVersionedProtocolData
MkHydraVersionedProtocolData (OuroborosApplicationWithMinimalCtx
'ResponderMode SockAddr LByteString IO a b
-> SomeResponderApplication SockAddr LByteString IO b
forall (appType :: MuxMode) addr bytes (m :: * -> *) a b.
(HasResponder appType ~ 'True) =>
OuroborosApplicationWithMinimalCtx appType addr bytes m a b
-> SomeResponderApplication addr bytes m b
SomeResponderApplication OuroborosApplicationWithMinimalCtx
'ResponderMode SockAddr LByteString IO a b
app))
ErrorPolicies
nullErrorPolicies
((SockAddr -> Async Void -> IO ()) -> IO ())
-> (SockAddr -> Async Void -> IO ()) -> IO ()
forall a b. (a -> b) -> a -> b
$ \SockAddr
_addr Async Void
serverAsync -> do
IO Void -> IO b -> IO ()
forall a b. IO a -> IO b -> IO ()
forall (m :: * -> *) a b. MonadAsync m => m a -> m b -> m ()
race_ (Async IO Void -> IO Void
forall a. Async IO a -> IO a
forall (m :: * -> *) a. MonadAsync m => Async m a -> m a
wait Async Void
Async IO Void
serverAsync) IO b
continuation
where
notConfigureSocket :: a -> b -> IO ()
notConfigureSocket :: forall a b. a -> b -> IO ()
notConfigureSocket a
_ b
_ = () -> IO ()
forall a. a -> IO a
forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
networkServerTracers :: NetworkServerTracers SockAddr HydraVersionedProtocolNumber
networkServerTracers :: NetworkServerTracers SockAddr HydraVersionedProtocolNumber
networkServerTracers =
NetworkServerTracers
{ nstMuxTracer :: Tracer IO (WithMuxBearer (ConnectionId SockAddr) MuxTrace)
nstMuxTracer = Tracer IO (WithMuxBearer (ConnectionId SockAddr) MuxTrace)
forall (m :: * -> *) a. Applicative m => Tracer m a
nullTracer
, nstHandshakeTracer :: Tracer
IO
(WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term)))
nstHandshakeTracer = (WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithHost (TraceOuroborosNetwork outbound))
-> Tracer
IO
(WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term)))
forall a' a. (a' -> a) -> Tracer IO a -> Tracer IO a'
forall (f :: * -> *) a' a.
Contravariant f =>
(a' -> a) -> f a -> f a'
contramap (Host
-> TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound)
forall trace. Host -> trace -> WithHost trace
WithHost Host
localHost (TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound))
-> (WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> TraceOuroborosNetwork outbound)
-> WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> WithHost (TraceOuroborosNetwork outbound)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> TraceOuroborosNetwork outbound
forall {k} (msg :: k).
WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> TraceOuroborosNetwork msg
TraceHandshake) Tracer IO (WithHost (TraceOuroborosNetwork outbound))
tracer
, nstErrorPolicyTracer :: Tracer IO (WithAddr SockAddr ErrorPolicyTrace)
nstErrorPolicyTracer = (WithAddr SockAddr ErrorPolicyTrace
-> WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithAddr SockAddr ErrorPolicyTrace)
forall a' a. (a' -> a) -> Tracer IO a -> Tracer IO a'
forall (f :: * -> *) a' a.
Contravariant f =>
(a' -> a) -> f a -> f a'
contramap (Host
-> TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound)
forall trace. Host -> trace -> WithHost trace
WithHost Host
localHost (TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound))
-> (WithAddr SockAddr ErrorPolicyTrace
-> TraceOuroborosNetwork outbound)
-> WithAddr SockAddr ErrorPolicyTrace
-> WithHost (TraceOuroborosNetwork outbound)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. WithAddr SockAddr ErrorPolicyTrace
-> TraceOuroborosNetwork outbound
forall {k} (msg :: k).
WithAddr SockAddr ErrorPolicyTrace -> TraceOuroborosNetwork msg
TraceErrorPolicy) Tracer IO (WithHost (TraceOuroborosNetwork outbound))
tracer
, nstAcceptPolicyTracer :: Tracer IO AcceptConnectionsPolicyTrace
nstAcceptPolicyTracer = (AcceptConnectionsPolicyTrace
-> WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO (WithHost (TraceOuroborosNetwork outbound))
-> Tracer IO AcceptConnectionsPolicyTrace
forall a' a. (a' -> a) -> Tracer IO a -> Tracer IO a'
forall (f :: * -> *) a' a.
Contravariant f =>
(a' -> a) -> f a -> f a'
contramap (Host
-> TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound)
forall trace. Host -> trace -> WithHost trace
WithHost Host
localHost (TraceOuroborosNetwork outbound
-> WithHost (TraceOuroborosNetwork outbound))
-> (AcceptConnectionsPolicyTrace -> TraceOuroborosNetwork outbound)
-> AcceptConnectionsPolicyTrace
-> WithHost (TraceOuroborosNetwork outbound)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. AcceptConnectionsPolicyTrace -> TraceOuroborosNetwork outbound
forall {k} (msg :: k).
AcceptConnectionsPolicyTrace -> TraceOuroborosNetwork msg
TraceAcceptPolicy) Tracer IO (WithHost (TraceOuroborosNetwork outbound))
tracer
}
onIOException :: IOException -> IO ()
onIOException :: IOException -> IO ()
onIOException IOException
ioException =
NetworkServerListenException -> IO ()
forall e a. Exception e => e -> IO a
forall (m :: * -> *) e a. (MonadThrow m, Exception e) => e -> m a
throwIO (NetworkServerListenException -> IO ())
-> NetworkServerListenException -> IO ()
forall a b. (a -> b) -> a -> b
$
NetworkServerListenException
{ IOException
ioException :: IOException
$sel:ioException:NetworkServerListenException :: IOException
ioException
, Host
localHost :: Host
$sel:localHost:NetworkServerListenException :: Host
localHost
}
hydraClient ::
TChan outbound ->
OuroborosApplicationWithMinimalCtx 'InitiatorMode addr LByteString IO () Void
hydraClient :: forall addr.
TChan outbound
-> OuroborosApplicationWithMinimalCtx
'InitiatorMode addr LByteString IO () Void
hydraClient TChan outbound
chan =
[MiniProtocol
'InitiatorMode
(MinimalInitiatorContext addr)
(ResponderContext addr)
LByteString
IO
()
Void]
-> OuroborosApplication
'InitiatorMode
(MinimalInitiatorContext addr)
(ResponderContext addr)
LByteString
IO
()
Void
forall (mode :: MuxMode) initiatorCtx responderCtx bytes
(m :: * -> *) a b.
[MiniProtocol mode initiatorCtx responderCtx bytes m a b]
-> OuroborosApplication mode initiatorCtx responderCtx bytes m a b
OuroborosApplication
[ MiniProtocol
{ miniProtocolNum :: MiniProtocolNum
miniProtocolNum = Word16 -> MiniProtocolNum
MiniProtocolNum Word16
42
, miniProtocolLimits :: MiniProtocolLimits
miniProtocolLimits = MiniProtocolLimits
maximumMiniProtocolLimits
, miniProtocolRun :: RunMiniProtocol
'InitiatorMode
(MinimalInitiatorContext addr)
(ResponderContext addr)
LByteString
IO
()
Void
miniProtocolRun = MiniProtocolCb (MinimalInitiatorContext addr) LByteString IO ()
-> RunMiniProtocol
'InitiatorMode
(MinimalInitiatorContext addr)
(ResponderContext addr)
LByteString
IO
()
Void
forall initiatorCtx bytes (m :: * -> *) a responderCtx.
MiniProtocolCb initiatorCtx bytes m a
-> RunMiniProtocol
'InitiatorMode initiatorCtx responderCtx bytes m a Void
InitiatorProtocolOnly MiniProtocolCb (MinimalInitiatorContext addr) LByteString IO ()
forall ctx. MiniProtocolCb ctx LByteString IO ()
initiator
}
]
where
initiator :: MiniProtocolCb ctx LByteString IO ()
initiator :: forall ctx. MiniProtocolCb ctx LByteString IO ()
initiator =
(ctx
-> (Tracer IO (TraceSendRecv (FireForget outbound)),
Codec (FireForget outbound) DeserialiseFailure IO LByteString,
Peer (FireForget outbound) 'AsClient 'StIdle IO ()))
-> MiniProtocolCb ctx LByteString IO ()
forall (pr :: PeerRole) ps (st :: ps) failure bytes ctx
(m :: * -> *) a.
(MonadThrow m, Show failure,
forall (st' :: ps). Show (ClientHasAgency st'),
forall (st' :: ps). Show (ServerHasAgency st'), ShowProxy ps) =>
(ctx
-> (Tracer m (TraceSendRecv ps), Codec ps failure m bytes,
Peer ps pr st m a))
-> MiniProtocolCb ctx bytes m a
mkMiniProtocolCbFromPeer
( (Tracer IO (TraceSendRecv (FireForget outbound)),
Codec (FireForget outbound) DeserialiseFailure IO LByteString,
Peer (FireForget outbound) 'AsClient 'StIdle IO ())
-> ctx
-> (Tracer IO (TraceSendRecv (FireForget outbound)),
Codec (FireForget outbound) DeserialiseFailure IO LByteString,
Peer (FireForget outbound) 'AsClient 'StIdle IO ())
forall a b. a -> b -> a
const
(Tracer IO (TraceSendRecv (FireForget outbound))
forall (m :: * -> *) a. Applicative m => Tracer m a
nullTracer, Codec (FireForget outbound) DeserialiseFailure IO LByteString
forall a (m :: * -> *).
(MonadST m, ToCBOR a, FromCBOR a) =>
Codec (FireForget a) DeserialiseFailure m LByteString
codecFireForget, FireForgetClient outbound IO ()
-> Peer (FireForget outbound) 'AsClient 'StIdle IO ()
forall (m :: * -> *) msg a.
Monad m =>
FireForgetClient msg m a
-> Peer (FireForget msg) 'AsClient 'StIdle m a
fireForgetClientPeer (FireForgetClient outbound IO ()
-> Peer (FireForget outbound) 'AsClient 'StIdle IO ())
-> FireForgetClient outbound IO ()
-> Peer (FireForget outbound) 'AsClient 'StIdle IO ()
forall a b. (a -> b) -> a -> b
$ TChan outbound -> FireForgetClient outbound IO ()
client TChan outbound
chan)
)
hydraServer ::
OuroborosApplicationWithMinimalCtx 'ResponderMode addr LByteString IO Void ()
hydraServer :: forall addr.
OuroborosApplicationWithMinimalCtx
'ResponderMode addr LByteString IO Void ()
hydraServer =
[MiniProtocol
'ResponderMode
(MinimalInitiatorContext addr)
(ResponderContext addr)
LByteString
IO
Void
()]
-> OuroborosApplication
'ResponderMode
(MinimalInitiatorContext addr)
(ResponderContext addr)
LByteString
IO
Void
()
forall (mode :: MuxMode) initiatorCtx responderCtx bytes
(m :: * -> *) a b.
[MiniProtocol mode initiatorCtx responderCtx bytes m a b]
-> OuroborosApplication mode initiatorCtx responderCtx bytes m a b
OuroborosApplication
[ MiniProtocol
{ miniProtocolNum :: MiniProtocolNum
miniProtocolNum = Word16 -> MiniProtocolNum
MiniProtocolNum Word16
42
, miniProtocolLimits :: MiniProtocolLimits
miniProtocolLimits = MiniProtocolLimits
maximumMiniProtocolLimits
, miniProtocolRun :: RunMiniProtocol
'ResponderMode
(MinimalInitiatorContext addr)
(ResponderContext addr)
LByteString
IO
Void
()
miniProtocolRun = MiniProtocolCb (ResponderContext addr) LByteString IO ()
-> RunMiniProtocol
'ResponderMode
(MinimalInitiatorContext addr)
(ResponderContext addr)
LByteString
IO
Void
()
forall responderCtx bytes (m :: * -> *) b initiatorCtx.
MiniProtocolCb responderCtx bytes m b
-> RunMiniProtocol
'ResponderMode initiatorCtx responderCtx bytes m Void b
ResponderProtocolOnly MiniProtocolCb (ResponderContext addr) LByteString IO ()
forall ctx. MiniProtocolCb ctx LByteString IO ()
responder
}
]
where
responder :: MiniProtocolCb ctx LByteString IO ()
responder :: forall ctx. MiniProtocolCb ctx LByteString IO ()
responder = (ctx
-> (Tracer IO (TraceSendRecv (FireForget inbound)),
Codec (FireForget inbound) DeserialiseFailure IO LByteString,
Peer (FireForget inbound) 'AsServer 'StIdle IO ()))
-> MiniProtocolCb ctx LByteString IO ()
forall (pr :: PeerRole) ps (st :: ps) failure bytes ctx
(m :: * -> *) a.
(MonadThrow m, Show failure,
forall (st' :: ps). Show (ClientHasAgency st'),
forall (st' :: ps). Show (ServerHasAgency st'), ShowProxy ps) =>
(ctx
-> (Tracer m (TraceSendRecv ps), Codec ps failure m bytes,
Peer ps pr st m a))
-> MiniProtocolCb ctx bytes m a
mkMiniProtocolCbFromPeer ((Tracer IO (TraceSendRecv (FireForget inbound)),
Codec (FireForget inbound) DeserialiseFailure IO LByteString,
Peer (FireForget inbound) 'AsServer 'StIdle IO ())
-> ctx
-> (Tracer IO (TraceSendRecv (FireForget inbound)),
Codec (FireForget inbound) DeserialiseFailure IO LByteString,
Peer (FireForget inbound) 'AsServer 'StIdle IO ())
forall a b. a -> b -> a
const (Tracer IO (TraceSendRecv (FireForget inbound))
forall (m :: * -> *) a. Applicative m => Tracer m a
nullTracer, Codec (FireForget inbound) DeserialiseFailure IO LByteString
forall a (m :: * -> *).
(MonadST m, ToCBOR a, FromCBOR a) =>
Codec (FireForget a) DeserialiseFailure m LByteString
codecFireForget, FireForgetServer inbound IO ()
-> Peer (FireForget inbound) 'AsServer 'StIdle IO ()
forall (m :: * -> *) msg a.
Monad m =>
FireForgetServer msg m a
-> Peer (FireForget msg) 'AsServer 'StIdle m a
fireForgetServerPeer FireForgetServer inbound IO ()
server))
maximumMiniProtocolLimits :: MiniProtocolLimits
maximumMiniProtocolLimits :: MiniProtocolLimits
maximumMiniProtocolLimits =
MiniProtocolLimits{maximumIngressQueue :: Int
maximumIngressQueue = Int
forall a. Bounded a => a
maxBound}
client ::
TChan outbound ->
FireForgetClient outbound IO ()
client :: TChan outbound -> FireForgetClient outbound IO ()
client TChan outbound
chan =
IO (FireForgetClient outbound IO ())
-> FireForgetClient outbound IO ()
forall (m :: * -> *) msg a.
m (FireForgetClient msg m a) -> FireForgetClient msg m a
Idle (IO (FireForgetClient outbound IO ())
-> FireForgetClient outbound IO ())
-> IO (FireForgetClient outbound IO ())
-> FireForgetClient outbound IO ()
forall a b. (a -> b) -> a -> b
$ do
STM IO outbound -> IO outbound
forall a. HasCallStack => STM IO a -> IO a
forall (m :: * -> *) a.
(MonadSTM m, HasCallStack) =>
STM m a -> m a
atomically (TChan outbound -> STM outbound
forall a. TChan a -> STM a
readTChan TChan outbound
chan) IO outbound
-> (outbound -> FireForgetClient outbound IO ())
-> IO (FireForgetClient outbound IO ())
forall (f :: * -> *) a b. Functor f => f a -> (a -> b) -> f b
<&> \outbound
msg ->
outbound
-> IO (FireForgetClient outbound IO ())
-> FireForgetClient outbound IO ()
forall msg (m :: * -> *) a.
msg -> m (FireForgetClient msg m a) -> FireForgetClient msg m a
SendMsg outbound
msg (FireForgetClient outbound IO ()
-> IO (FireForgetClient outbound IO ())
forall a. a -> IO a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (FireForgetClient outbound IO ()
-> IO (FireForgetClient outbound IO ()))
-> FireForgetClient outbound IO ()
-> IO (FireForgetClient outbound IO ())
forall a b. (a -> b) -> a -> b
$ TChan outbound -> FireForgetClient outbound IO ()
client TChan outbound
chan)
server :: FireForgetServer inbound IO ()
server :: FireForgetServer inbound IO ()
server =
FireForgetServer
{ $sel:recvMsg:FireForgetServer :: inbound -> IO (FireForgetServer inbound IO ())
recvMsg = \inbound
msg -> inbound -> IO ()
deliver inbound
msg IO ()
-> FireForgetServer inbound IO ()
-> IO (FireForgetServer inbound IO ())
forall (f :: * -> *) a b. Functor f => f a -> b -> f b
$> FireForgetServer inbound IO ()
server
, $sel:recvMsgDone:FireForgetServer :: IO ()
recvMsgDone = () -> IO ()
forall a. a -> IO a
forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
}
data NetworkServerListenException = NetworkServerListenException
{ NetworkServerListenException -> IOException
ioException :: IOException
, NetworkServerListenException -> Host
localHost :: Host
}
deriving stock (Int -> NetworkServerListenException -> ShowS
[NetworkServerListenException] -> ShowS
NetworkServerListenException -> ServiceName
(Int -> NetworkServerListenException -> ShowS)
-> (NetworkServerListenException -> ServiceName)
-> ([NetworkServerListenException] -> ShowS)
-> Show NetworkServerListenException
forall a.
(Int -> a -> ShowS)
-> (a -> ServiceName) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> NetworkServerListenException -> ShowS
showsPrec :: Int -> NetworkServerListenException -> ShowS
$cshow :: NetworkServerListenException -> ServiceName
show :: NetworkServerListenException -> ServiceName
$cshowList :: [NetworkServerListenException] -> ShowS
showList :: [NetworkServerListenException] -> ShowS
Show)
instance Exception NetworkServerListenException
data WithHost trace = WithHost Host trace
deriving stock (Int -> WithHost trace -> ShowS
[WithHost trace] -> ShowS
WithHost trace -> ServiceName
(Int -> WithHost trace -> ShowS)
-> (WithHost trace -> ServiceName)
-> ([WithHost trace] -> ShowS)
-> Show (WithHost trace)
forall trace. Show trace => Int -> WithHost trace -> ShowS
forall trace. Show trace => [WithHost trace] -> ShowS
forall trace. Show trace => WithHost trace -> ServiceName
forall a.
(Int -> a -> ShowS)
-> (a -> ServiceName) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: forall trace. Show trace => Int -> WithHost trace -> ShowS
showsPrec :: Int -> WithHost trace -> ShowS
$cshow :: forall trace. Show trace => WithHost trace -> ServiceName
show :: WithHost trace -> ServiceName
$cshowList :: forall trace. Show trace => [WithHost trace] -> ShowS
showList :: [WithHost trace] -> ShowS
Show)
instance ToJSON trace => ToJSON (WithHost trace) where
toJSON :: WithHost trace -> Value
toJSON (WithHost Host
h trace
tr) =
[Pair] -> Value
object
[ Key
"host" Key -> Host -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= Host
h
, Key
"data" Key -> trace -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= trace
tr
]
data TraceOuroborosNetwork msg
= TraceSubscriptions (WithIPList (SubscriptionTrace SockAddr))
| TraceErrorPolicy (WithAddr SockAddr ErrorPolicyTrace)
| TraceAcceptPolicy AcceptConnectionsPolicyTrace
| TraceHandshake (WithMuxBearer (ConnectionId SockAddr) (TraceSendRecv (Handshake HydraVersionedProtocolNumber CBOR.Term)))
| TraceSendRecv (TraceSendRecv (FireForget msg))
instance ToJSON msg => ToJSON (TraceOuroborosNetwork msg) where
toJSON :: TraceOuroborosNetwork msg -> Value
toJSON = \case
TraceSubscriptions WithIPList (SubscriptionTrace SockAddr)
withIpList ->
Text -> [Pair] -> Value
tagged Text
"TraceSubscriptions" [Key
"subscriptions" Key -> Value -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= WithIPList (SubscriptionTrace SockAddr) -> Value
encodeWithIPList WithIPList (SubscriptionTrace SockAddr)
withIpList]
TraceErrorPolicy WithAddr SockAddr ErrorPolicyTrace
withAddr ->
Text -> [Pair] -> Value
tagged Text
"TraceErrorPolicy" [Key
"errors" Key -> Value -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= WithAddr SockAddr ErrorPolicyTrace -> Value
encodeWithAddr WithAddr SockAddr ErrorPolicyTrace
withAddr]
TraceAcceptPolicy AcceptConnectionsPolicyTrace
accept ->
Text -> [Pair] -> Value
tagged Text
"TraceAcceptPolicy" [Key
"accept" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= forall b a. (Show a, IsString b) => a -> b
show @Text AcceptConnectionsPolicyTrace
accept]
TraceHandshake WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
handshake ->
Text -> [Pair] -> Value
tagged Text
"TraceHandshake" [Key
"handshake" Key -> [Pair] -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> [Pair]
encodeTraceSendRecvHandshake WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
handshake]
TraceSendRecv TraceSendRecv (FireForget msg)
sndRcv ->
Text -> [Pair] -> Value
tagged Text
"TraceSendRecv" [Key
"trace" Key -> [Pair] -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= TraceSendRecv (FireForget msg) -> [Pair]
forall msg. ToJSON msg => TraceSendRecv (FireForget msg) -> [Pair]
encodeTraceSendRecvFireForget TraceSendRecv (FireForget msg)
sndRcv]
tagged :: Text -> [Aeson.Pair] -> Aeson.Value
tagged :: Text -> [Pair] -> Value
tagged Text
tag [Pair]
pairs = [Pair] -> Value
object ((Key
"tag" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= Text
tag) Pair -> [Pair] -> [Pair]
forall a. a -> [a] -> [a]
: [Pair]
pairs)
encodeWithIPList :: WithIPList (SubscriptionTrace SockAddr) -> Aeson.Value
encodeWithIPList :: WithIPList (SubscriptionTrace SockAddr) -> Value
encodeWithIPList (WithIPList LocalAddresses SockAddr
src [SockAddr]
dsts SubscriptionTrace SockAddr
ev) =
Text -> [Pair] -> Value
tagged
Text
"WithIPList"
[ Key
"src" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= forall b a. (Show a, IsString b) => a -> b
show @Text LocalAddresses SockAddr
src
, Key
"dsts" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= forall b a. (Show a, IsString b) => a -> b
show @Text [SockAddr]
dsts
, Key
"event" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= forall b a. (Show a, IsString b) => a -> b
show @Text SubscriptionTrace SockAddr
ev
]
encodeWithAddr :: WithAddr SockAddr ErrorPolicyTrace -> Aeson.Value
encodeWithAddr :: WithAddr SockAddr ErrorPolicyTrace -> Value
encodeWithAddr (WithAddr SockAddr
addr ErrorPolicyTrace
ev) =
Text -> [Pair] -> Value
tagged
Text
"WithAddr"
[ Key
"addr" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= forall b a. (Show a, IsString b) => a -> b
show @Text SockAddr
addr
, Key
"event" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= forall b a. (Show a, IsString b) => a -> b
show @Text ErrorPolicyTrace
ev
]
encodeTraceSendRecvHandshake ::
WithMuxBearer (ConnectionId SockAddr) (TraceSendRecv (Handshake HydraVersionedProtocolNumber CBOR.Term)) ->
[Aeson.Pair]
encodeTraceSendRecvHandshake :: WithMuxBearer
(ConnectionId SockAddr)
(TraceSendRecv (Handshake HydraVersionedProtocolNumber Term))
-> [Pair]
encodeTraceSendRecvHandshake = \case
WithMuxBearer ConnectionId SockAddr
peerId (TraceSendMsg (AnyMessageAndAgency PeerHasAgency pr st
agency Message (Handshake HydraVersionedProtocolNumber Term) st st'
msg)) ->
[ Key
"event" Key -> ServiceName -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ServiceName
"send" :: String)
, Key
"agency" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (PeerHasAgency pr st -> Text
forall b a. (Show a, IsString b) => a -> b
show PeerHasAgency pr st
agency :: Text)
, Key
"peer" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ConnectionId SockAddr -> Text
forall b a. (Show a, IsString b) => a -> b
show ConnectionId SockAddr
peerId :: Text)
]
[Pair] -> [Pair] -> [Pair]
forall a. [a] -> [a] -> [a]
++ Message (Handshake HydraVersionedProtocolNumber Term) st st'
-> [Pair]
forall (from :: Handshake HydraVersionedProtocolNumber Term)
(to :: Handshake HydraVersionedProtocolNumber Term).
Message (Handshake HydraVersionedProtocolNumber Term) from to
-> [Pair]
encodeMsg Message (Handshake HydraVersionedProtocolNumber Term) st st'
msg
WithMuxBearer ConnectionId SockAddr
peerId (TraceRecvMsg (AnyMessageAndAgency PeerHasAgency pr st
agency Message (Handshake HydraVersionedProtocolNumber Term) st st'
msg)) ->
[ Key
"event" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (Text
"receive" :: Text)
, Key
"agency" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (PeerHasAgency pr st -> Text
forall b a. (Show a, IsString b) => a -> b
show PeerHasAgency pr st
agency :: Text)
, Key
"peer" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ConnectionId SockAddr -> Text
forall b a. (Show a, IsString b) => a -> b
show ConnectionId SockAddr
peerId :: Text)
]
[Pair] -> [Pair] -> [Pair]
forall a. [a] -> [a] -> [a]
++ Message (Handshake HydraVersionedProtocolNumber Term) st st'
-> [Pair]
forall (from :: Handshake HydraVersionedProtocolNumber Term)
(to :: Handshake HydraVersionedProtocolNumber Term).
Message (Handshake HydraVersionedProtocolNumber Term) from to
-> [Pair]
encodeMsg Message (Handshake HydraVersionedProtocolNumber Term) st st'
msg
where
encodeMsg ::
Message (Handshake HydraVersionedProtocolNumber Term) from to ->
[Aeson.Pair]
encodeMsg :: forall (from :: Handshake HydraVersionedProtocolNumber Term)
(to :: Handshake HydraVersionedProtocolNumber Term).
Message (Handshake HydraVersionedProtocolNumber Term) from to
-> [Pair]
encodeMsg = \case
MsgProposeVersions Map vNumber1 vParams1
versions ->
[ Key
"tag" Key -> ServiceName -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ServiceName
"ProposeVersions" :: String)
, Key
"versions" Key -> [Text] -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (vNumber1 -> Text
forall b a. (Show a, IsString b) => a -> b
show (vNumber1 -> Text) -> [vNumber1] -> [Text]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Map vNumber1 vParams1 -> [vNumber1]
forall k a. Map k a -> [k]
Map.keys Map vNumber1 vParams1
versions :: [Text])
]
MsgAcceptVersion vNumber1
v vParams1
_ ->
[ Key
"tag" Key -> ServiceName -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ServiceName
"AcceptVersion" :: String)
, Key
"version" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (vNumber1 -> Text
forall b a. (Show a, IsString b) => a -> b
show vNumber1
v :: Text)
]
MsgRefuse RefuseReason vNumber1
reason ->
[ Key
"tag" Key -> ServiceName -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ServiceName
"RefuseVersions" :: String)
, Key
"reason" Key -> Value -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= RefuseReason vNumber1 -> Value
forall vNumber. RefuseReason vNumber -> Value
encodeRefuseReason RefuseReason vNumber1
reason
]
MsgReplyVersions Map vNumber1 vParams1
versions ->
[ Key
"tag" Key -> ServiceName -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ServiceName
"ReplyVersions" :: String)
, Key
"versions" Key -> [Text] -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (vNumber1 -> Text
forall b a. (Show a, IsString b) => a -> b
show (vNumber1 -> Text) -> [vNumber1] -> [Text]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Map vNumber1 vParams1 -> [vNumber1]
forall k a. Map k a -> [k]
Map.keys Map vNumber1 vParams1
versions :: [Text])
]
MsgQueryReply Map vNumber1 vParams1
versions ->
[ Key
"tag" Key -> ServiceName -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ServiceName
"MsgQueryReply" :: String)
, Key
"versions" Key -> [Text] -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (vNumber1 -> Text
forall b a. (Show a, IsString b) => a -> b
show (vNumber1 -> Text) -> [vNumber1] -> [Text]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Map vNumber1 vParams1 -> [vNumber1]
forall k a. Map k a -> [k]
Map.keys Map vNumber1 vParams1
versions :: [Text])
]
encodeRefuseReason ::
RefuseReason vNumber ->
Aeson.Value
encodeRefuseReason :: forall vNumber. RefuseReason vNumber -> Value
encodeRefuseReason = \case
VersionMismatch{} -> Text -> Value
Aeson.String Text
"VersionMismatchOrUnknown"
HandshakeDecodeError{} -> Text -> Value
Aeson.String Text
"HandshakeDecodeError"
Refused{} -> Text -> Value
Aeson.String Text
"ServerRejected"
encodeTraceSendRecvFireForget ::
forall msg.
ToJSON msg =>
TraceSendRecv (FireForget msg) ->
[Aeson.Pair]
encodeTraceSendRecvFireForget :: forall msg. ToJSON msg => TraceSendRecv (FireForget msg) -> [Pair]
encodeTraceSendRecvFireForget = \case
TraceSendMsg (AnyMessageAndAgency PeerHasAgency pr st
agency Message (FireForget msg) st st'
msg) ->
[ Key
"event" Key -> ServiceName -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (ServiceName
"send" :: String)
, Key
"agency" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (PeerHasAgency pr st -> Text
forall b a. (Show a, IsString b) => a -> b
show PeerHasAgency pr st
agency :: Text)
]
[Pair] -> [Pair] -> [Pair]
forall a. [a] -> [a] -> [a]
++ Message (FireForget msg) st st' -> [Pair]
forall (from :: FireForget msg) (to :: FireForget msg).
Message (FireForget msg) from to -> [Pair]
encodeMsg Message (FireForget msg) st st'
msg
TraceRecvMsg (AnyMessageAndAgency PeerHasAgency pr st
agency Message (FireForget msg) st st'
msg) ->
[ Key
"event" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (Text
"receive" :: Text)
, Key
"agency" Key -> Text -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= (PeerHasAgency pr st -> Text
forall b a. (Show a, IsString b) => a -> b
show PeerHasAgency pr st
agency :: Text)
]
[Pair] -> [Pair] -> [Pair]
forall a. [a] -> [a] -> [a]
++ Message (FireForget msg) st st' -> [Pair]
forall (from :: FireForget msg) (to :: FireForget msg).
Message (FireForget msg) from to -> [Pair]
encodeMsg Message (FireForget msg) st st'
msg
where
encodeMsg ::
Message (FireForget msg) from to ->
[Aeson.Pair]
encodeMsg :: forall (from :: FireForget msg) (to :: FireForget msg).
Message (FireForget msg) from to -> [Pair]
encodeMsg = \case
MsgSend msg1
msg ->
[ Key
"send" Key -> msg1 -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= msg1
msg
]
Message (FireForget msg) from to
R:MessageFireForgetfromto (*) msg from to
MsgDone ->
[ Key
"done" Key -> () -> Pair
forall v. ToJSON v => Key -> v -> Pair
forall e kv v. (KeyValue e kv, ToJSON v) => Key -> v -> kv
.= ()
]