Skip to content

Commit d28580e

Browse files
committed
Remove EOF errors from receive functions
It is generally accepted that EOF errors in `Network` were a mistake. Using exceptions for control flow is an anti-pattern in the Haskell community. While we are in the process of introducing epoch breaking changes it makes sense to attempt to remove these historic accidents. There should really be no perceived change for users of `Network.ByteString(.Lazy).recv`, however `recvFrom` does change behavior. There are two key differences in `recvFrom`: 1. It is now returning a `(0, address)` on an EOF. 2. It will peek the socket's address and `getPeerName` on an EOF. This has some implications for performance, and such may be controversial. The second point poses a question. Does the return type of `SocketAddress sa => IO (Int, sa)` make sense for `recvBufFrom`?
1 parent ad33b8c commit d28580e

File tree

2 files changed

+6
-21
lines changed

2 files changed

+6
-21
lines changed

Network/Socket/Buffer.hs

+5-15
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ module Network.Socket.Buffer (
1111

1212
import qualified Control.Exception as E
1313
import Foreign.Marshal.Alloc (alloca)
14-
import GHC.IO.Exception (IOErrorType(EOF, InvalidArgument))
14+
import GHC.IO.Exception (IOErrorType(InvalidArgument))
1515
import System.IO.Error (mkIOError, ioeSetErrorString)
1616

1717
#if defined(mingw32_HOST_OS)
@@ -97,13 +97,9 @@ recvBufFrom s ptr nbytes
9797
flags = 0
9898
len <- throwSocketErrorWaitRead s "Network.Socket.recvBufFrom" $
9999
c_recvfrom fd ptr cnbytes flags ptr_sa ptr_len
100-
let len' = fromIntegral len
101-
if len' == 0
102-
then ioError (mkEOFError "Network.Socket.recvFrom")
103-
else do
104-
sockaddr <- peekSocketAddress ptr_sa
105-
`E.catch` \(E.SomeException _) -> getPeerName s
106-
return (len', sockaddr)
100+
sockaddr <- peekSocketAddress ptr_sa
101+
`E.catch` \(E.SomeException _) -> getPeerName s
102+
return (fromIntegral len, sockaddr)
107103

108104
-- | Receive data from the socket. The socket must be in a connected
109105
-- state. This function may return fewer bytes than specified. If the
@@ -133,19 +129,13 @@ recvBuf s ptr nbytes
133129
len <- throwSocketErrorWaitRead s "Network.Socket.recvBuf" $
134130
c_recv fd (castPtr ptr) (fromIntegral nbytes) 0{-flags-}
135131
#endif
136-
let len' = fromIntegral len
137-
if len' == 0
138-
then ioError (mkEOFError "Network.Socket.recvBuf")
139-
else return len'
132+
return $ fromIntegral len
140133

141134
mkInvalidRecvArgError :: String -> IOError
142135
mkInvalidRecvArgError loc = ioeSetErrorString (mkIOError
143136
InvalidArgument
144137
loc Nothing Nothing) "non-positive length"
145138

146-
mkEOFError :: String -> IOError
147-
mkEOFError loc = ioeSetErrorString (mkIOError EOF loc Nothing Nothing) "end of file"
148-
149139
#if !defined(mingw32_HOST_OS)
150140
foreign import ccall unsafe "send"
151141
c_send :: CInt -> Ptr a -> CSize -> CInt -> IO CInt

Network/Socket/ByteString/IO.hsc

+1-6
Original file line numberDiff line numberDiff line change
@@ -43,14 +43,12 @@ module Network.Socket.ByteString.IO
4343
) where
4444

4545
import Control.Concurrent (threadWaitWrite, rtsSupportsBoundThreads)
46-
import Control.Exception as E (catch, throwIO)
4746
import Data.ByteString (ByteString)
4847
import qualified Data.ByteString as B
4948
import Data.ByteString.Internal (createAndTrim)
5049
import Data.ByteString.Unsafe (unsafeUseAsCStringLen)
5150
import Foreign.Marshal.Alloc (allocaBytes)
5251
import Foreign.Ptr (castPtr)
53-
import System.IO.Error (isEOFError)
5452

5553
import Network.Socket.Buffer
5654
import Network.Socket.ByteString.Internal
@@ -220,10 +218,7 @@ recv :: Socket -- ^ Connected socket
220218
-> IO ByteString -- ^ Data received
221219
recv s nbytes
222220
| nbytes < 0 = ioError (mkInvalidRecvArgError "Network.Socket.ByteString.recv")
223-
| otherwise = createAndTrim nbytes $ \ptr ->
224-
E.catch
225-
(recvBuf s ptr nbytes)
226-
(\e -> if isEOFError e then return 0 else throwIO e)
221+
| otherwise = createAndTrim nbytes $ \ptr -> recvBuf s ptr nbytes
227222

228223
-- | Receive data from the socket. The socket need not be in a
229224
-- connected state. Returns @(bytes, address)@ where @bytes@ is a

0 commit comments

Comments
 (0)