我正在尝试弄清楚如何使用Haskell管道实现一个非平凡的协议(通过TCP(。 我认为不平凡的例子:
- 读取一些标头字节,如果它们与预期匹配,则忽略它们并继续;否则,向客户端返回错误。
- 读取指示字段长度的 N 个字节,然后将该字节数读入字节串。
- 在客户端和服务器之间执行来回握手,就像功能协商一样。 协商后,根据协商的内容调用不同的服务器端代码。 (例如协商服务器和客户端商定的协议版本(
- 如果客户端无法足够快地协商协议,则连接超时,并向客户端发送错误
到目前为止,我正在挣扎...任何帮助或指向某些示例代码的指针将不胜感激!
这个问题有些模糊,但如果您正在寻找基于先前解析的结果控制管道中操作的示例,那么网络字符串协议的实现可能就足够了:
#!/usr/bin/env stack
-- stack --resolver lts-10.3 script
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE OverloadedStrings #-}
import Conduit
import Data.ByteString (ByteString)
import Data.Word8 (_colon, _comma, _0, _9, Word8)
import Control.Exception.Safe (throwString)
netstring :: forall m. MonadThrow m => ConduitM ByteString ByteString m ()
netstring = do
len <- takeWhileCE (/= _colon) .| foldMCE addDigit 0
mchar1 <- headCE
case mchar1 of
Just c
| c == _colon -> return ()
| otherwise -> throwString $ "Didn't find a colon: " ++ show c
Nothing -> throwString "Missing colon"
takeCE len
mchar2 <- headCE
case mchar2 of
Just c
| c == _comma -> return ()
| otherwise -> throwString $ "Didn't end with a comma: " ++ show c
Nothing -> throwString "Missing trailing comma"
where
addDigit :: Int -> Word8 -> m Int
addDigit total char
| char < _0 || char > _9 = throwString "Invalid character in len"
addDigit total char = return $! total * 10 + fromIntegral (char - _0)
main :: IO ()
main = do
let bs = "5:hello,6: world,"
res <- runConduit
$ yield bs
.| ((,) <$> (netstring .| foldC) <*> (netstring .| foldC))
print res