能够Network.WebSockets
从 snaplet 内部使用模块会很好,但我不知道如何实际做到这一点。
使用它的runWebSocketsSnap :: MonadSnap m => ServerApp -> m ()
功能Network.WebSockets.Snap
很容易在我的应用程序中包含一个简单的无状态 websocket 服务器:
routes :: [(ByteString, Handler App App ())]
routes = [ ("/ws", runWebSocketsSnap wsApp) ]
wsApp :: PendingConnection -> IO () -- this is the ServerApp type
wsApp pending = do
conn <- acceptRequest pending
forever $ do
msg <- receiveData conn
sendTextData conn ("Echo " `mappend` msg :: Text)
但我的目标是维护 webscket 服务器的状态(例如,已连接客户端的列表,如http://jaspervdj.be/websockets/example.html)。或者,访问 snaplet 的酸性状态存储会很好。
我的第一个想法是liftIO
将 websocket 操作放入Handler App App
monad,并编写一个像这样的应用程序:
wsApp :: PendingConnection -> Handler App App ()
wsApp pending = do
conn <- liftIO $ acceptRequest pending
forever $ do
msg <- liftIO $ receiveData conn
update (SetLastMsg msg)
liftIO $ sendTextData conn ("Stored msg in datastore.")
但是没有一个版本runWebSocketsSnap
采用上述形式的应用程序,我不知道如何修改现有的应用程序(hackage 上的源代码)。在我看来,需要一个替代方案来代替forkIO
在 monad 中采取行动Handler App App
,但我对 Haskell 的理解,尤其是 Snap 中的并发性到此结束......