Skip to content

Commit

Permalink
Merge pull request #1902 from digitallyinduced/updated-websockets
Browse files Browse the repository at this point in the history
Use built-in ping pong mechanism of the latest version of websockets
  • Loading branch information
mpscholten authored Feb 3, 2024
2 parents df77216 + 2a85c32 commit 839cfee
Show file tree
Hide file tree
Showing 2 changed files with 36 additions and 34 deletions.
37 changes: 3 additions & 34 deletions IHP/WebSocket.hs
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ where

import IHP.Prelude
import qualified Network.WebSockets as Websocket
import Network.WebSockets.Connection.PingPong (withPingPong, defaultPingPongOptions)
import IHP.ApplicationContext
import IHP.Controller.RequestContext
import qualified Data.UUID as UUID
Expand Down Expand Up @@ -60,20 +61,11 @@ class WSApp state where
connectionOptions = WebSocket.defaultConnectionOptions

startWSApp :: forall state. (WSApp state, ?applicationContext :: ApplicationContext, ?requestContext :: RequestContext, ?context :: ControllerContext, ?modelContext :: ModelContext) => Websocket.Connection -> IO ()
startWSApp connection' = do
startWSApp connection = do
state <- newIORef (initialState @state)
lastPongAt <- getCurrentTime >>= newIORef


let connection = installPongHandler lastPongAt connection'
let ?state = state
let ?connection = connection
let pingHandler = do
seconds <- secondsSinceLastPong lastPongAt
when (seconds > pingWaitTime * 2) (throwIO PongTimeout)
onPing @state

result <- Exception.try ((WebSocket.withPingThread connection pingWaitTime pingHandler (run @state)) `Exception.finally` onClose @state)
result <- Exception.try ((withPingPong defaultPingPongOptions connection (\connection -> let ?connection = connection in run @state)) `Exception.finally` (let ?connection = connection in onClose @state))
case result of
Left (e@Exception.SomeException{}) ->
case Exception.fromException e of
Expand Down Expand Up @@ -117,26 +109,3 @@ instance Websocket.WebSocketsData UUID where
fromLazyByteString byteString = UUID.fromLazyASCIIBytes byteString |> Maybe.fromJust
toLazyByteString = UUID.toLazyASCIIBytes

data PongTimeout
= PongTimeout
deriving (Show)

instance Exception PongTimeout

pingWaitTime :: Int
pingWaitTime = 30

installPongHandler :: IORef UTCTime -> WebSocket.Connection -> WebSocket.Connection
installPongHandler lastPongAt connection =
connection { WebSocket.connectionOptions = connection.connectionOptions { WebSocket.connectionOnPong = connectionOnPong lastPongAt } }

connectionOnPong :: IORef UTCTime -> IO ()
connectionOnPong lastPongAt = do
now <- getCurrentTime
writeIORef lastPongAt now

secondsSinceLastPong :: IORef UTCTime -> IO Int
secondsSinceLastPong lastPongAt = do
now <- getCurrentTime
last <- readIORef lastPongAt
pure $ ceiling $ nominalDiffTimeToSeconds $ diffUTCTime now last
33 changes: 33 additions & 0 deletions NixSupport/haskell-packages/websockets.nix
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
{ mkDerivation, async, attoparsec, base, base64-bytestring, binary
, bytestring, case-insensitive, containers, criterion, entropy
, HUnit, lib, network, QuickCheck, random, SHA, streaming-commons
, test-framework, test-framework-hunit, test-framework-quickcheck2
, text
}:
mkDerivation {
pname = "websockets";
version = "0.13.0.0";
sha256 = "1da95b71akggyikbxdmja3gcaqrz8sp6ri5jrsyavc2ickvi9y4s";
isLibrary = true;
isExecutable = true;
libraryHaskellDepends = [
async attoparsec base base64-bytestring binary bytestring
case-insensitive containers entropy network random SHA
streaming-commons text
];
testHaskellDepends = [
async attoparsec base base64-bytestring binary bytestring
case-insensitive containers entropy HUnit network QuickCheck random
SHA streaming-commons test-framework test-framework-hunit
test-framework-quickcheck2 text
];
benchmarkHaskellDepends = [
async attoparsec base base64-bytestring binary bytestring
case-insensitive containers criterion entropy network random SHA
text
];
doCheck = false;
homepage = "http://jaspervdj.be/websockets";
description = "A sensible and clean way to write WebSocket-capable servers in Haskell";
license = lib.licenses.bsd3;
}

0 comments on commit 839cfee

Please sign in to comment.