1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201
|
{-# LANGUAGE OverloadedStrings #-}
module Network.DNS.IO (
-- * Receiving DNS messages
receive
, receiveFrom
, receiveVC
-- * Sending pre-encoded messages
, send
, sendTo
, sendVC
, sendAll
-- ** Encoding queries for transmission
, encodeQuestion
, encodeVC
-- ** Creating query response messages
, responseA
, responseAAAA
) where
import qualified Control.Exception as E
import qualified Data.ByteString as B
import qualified Data.ByteString.Builder as BB
import qualified Data.ByteString.Char8 as BS
import qualified Data.ByteString.Lazy.Char8 as LBS
import Data.IP (IPv4, IPv6)
import Time.System (timeCurrent)
import Time.Types (Elapsed(..), Seconds(..))
import Network.Socket (Socket, SockAddr)
import Network.Socket.ByteString (recv, recvFrom)
import qualified Network.Socket.ByteString as Socket
import System.IO.Error
import Network.DNS.Decode (decodeAt)
import Network.DNS.Encode (encode)
import Network.DNS.Imports
import Network.DNS.Types.Internal
----------------------------------------------------------------
-- | Receive and decode a single 'DNSMessage' from a UDP 'Socket', throwing away
-- the client address. Messages longer than 'maxUdpSize' are silently
-- truncated, but this should not occur in practice, since we cap the advertised
-- EDNS UDP buffer size limit at the same value. A 'DNSError' is raised if I/O
-- or message decoding fails.
--
receive :: Socket -> IO DNSMessage
receive sock = do
let bufsiz = fromIntegral maxUdpSize
bs <- recv sock bufsiz `E.catch` \e -> E.throwIO $ NetworkFailure e
Elapsed (Seconds now) <- timeCurrent
case decodeAt now bs of
Left e -> E.throwIO e
Right msg -> return msg
-- | Receive and decode a single 'DNSMessage' from a UDP 'Socket'. Messages
-- longer than 'maxUdpSize' are silently truncated, but this should not occur
-- in practice, since we cap the advertised EDNS UDP buffer size limit at the
-- same value. A 'DNSError' is raised if I/O or message decoding fails.
--
receiveFrom :: Socket -> IO (DNSMessage, SockAddr)
receiveFrom sock = do
let bufsiz = fromIntegral maxUdpSize
(bs, client) <- recvFrom sock bufsiz `E.catch` \e -> E.throwIO $ NetworkFailure e
Elapsed (Seconds now) <- timeCurrent
case decodeAt now bs of
Left e -> E.throwIO e
Right msg -> return (msg, client)
-- | Receive and decode a single 'DNSMesage' from a virtual-circuit (TCP). It
-- is up to the caller to implement any desired timeout. An 'DNSError' is
-- raised if I/O or message decoding fails.
--
receiveVC :: Socket -> IO DNSMessage
receiveVC sock = do
len <- toLen <$> recvDNS sock 2
bs <- recvDNS sock len
Elapsed (Seconds now) <- timeCurrent
case decodeAt now bs of
Left e -> E.throwIO e
Right msg -> return msg
where
toLen bs = case B.unpack bs of
[hi, lo] -> 256 * (fromIntegral hi) + (fromIntegral lo)
_ -> 0 -- never reached
recvDNS :: Socket -> Int -> IO ByteString
recvDNS sock len = recv1 `E.catch` \e -> E.throwIO $ NetworkFailure e
where
recv1 = do
bs1 <- recvCore len
if BS.length bs1 == len then
return bs1
else do
loop bs1
loop bs0 = do
let left = len - BS.length bs0
bs1 <- recvCore left
let bs = bs0 `BS.append` bs1
if BS.length bs == len then
return bs
else
loop bs
eofE = mkIOError eofErrorType "connection terminated" Nothing Nothing
recvCore len0 = do
bs <- recv sock len0
if bs == "" then
E.throwIO eofE
else
return bs
----------------------------------------------------------------
-- | Send an encoded 'DNSMessage' datagram over UDP. The message length is
-- implicit in the size of the UDP datagram. With TCP you must use 'sendVC',
-- because TCP does not have message boundaries, and each message needs to be
-- prepended with an explicit length. The socket must be explicitly connected
-- to the destination nameserver.
--
send :: Socket -> ByteString -> IO ()
send = (void .). Socket.send
{-# INLINE send #-}
-- | Send an encoded 'DNSMessage' datagram over UDP to a given address. The
-- message length is implicit in the size of the UDP datagram. With TCP you
-- must use 'sendVC', because TCP does not have message boundaries, and each
-- message needs to be prepended with an explicit length.
--
sendTo :: Socket -> ByteString -> SockAddr -> IO ()
sendTo sock str addr = Socket.sendTo sock str addr >> return ()
{-# INLINE sendTo #-}
-- | Send a single encoded 'DNSMessage' over TCP. An explicit length is
-- prepended to the encoded buffer before transmission. If you want to
-- send a batch of multiple encoded messages back-to-back over a single
-- TCP connection, and then loop to collect the results, use 'encodeVC'
-- to prefix each message with a length, and then use 'sendAll' to send
-- a concatenated batch of the resulting encapsulated messages.
--
sendVC :: Socket -> ByteString -> IO ()
sendVC = (. encodeVC). sendAll
{-# INLINE sendVC #-}
-- | Send one or more encoded 'DNSMessage' buffers over TCP, each allready
-- encapsulated with an explicit length prefix (perhaps via 'encodeVC') and
-- then concatenated into a single buffer. DO NOT use 'sendAll' with UDP.
--
sendAll :: Socket -> BS.ByteString -> IO ()
sendAll = Socket.sendAll
{-# INLINE sendAll #-}
-- | The encoded 'DNSMessage' has the specified request ID. The default values
-- of the RD, AD, CD and DO flag bits, as well as various EDNS features, can be
-- adjusted via the 'QueryControls' parameter.
--
-- The caller is responsible for generating the ID via a securely seeded
-- CSPRNG.
--
encodeQuestion :: Identifier -- ^ Crypto random request id
-> Question -- ^ Query name and type
-> QueryControls -- ^ Query flag and EDNS overrides
-> ByteString
encodeQuestion idt q ctls = encode $ makeQuery idt q ctls
-- | Encapsulate an encoded 'DNSMessage' buffer for transmission over a TCP
-- virtual circuit. With TCP the buffer needs to start with an explicit
-- length (the length is implicit with UDP).
--
encodeVC :: ByteString -> ByteString
encodeVC legacyQuery =
let len = LBS.toStrict . BB.toLazyByteString $ BB.int16BE $ fromIntegral $ BS.length legacyQuery
in len <> legacyQuery
{-# INLINE encodeVC #-}
----------------------------------------------------------------
-- | Compose a response with a single IPv4 RRset. If the query
-- had an EDNS pseudo-header, a suitable EDNS pseudo-header must
-- be added to the response message, or else a 'FormatErr' response
-- must be sent. The response TTL defaults to 300 seconds, and
-- should be updated (to the same value across all the RRs) if some
-- other TTL value is more appropriate.
--
responseA :: Identifier -> Question -> [IPv4] -> DNSMessage
responseA idt q ips = makeResponse idt q as
where
dom = qname q
as = ResourceRecord dom A classIN 300 . RD_A <$> ips
-- | Compose a response with a single IPv6 RRset. If the query
-- had an EDNS pseudo-header, a suitable EDNS pseudo-header must
-- be added to the response message, or else a 'FormatErr' response
-- must be sent. The response TTL defaults to 300 seconds, and
-- should be updated (to the same value across all the RRs) if some
-- other TTL value is more appropriate.
--
responseAAAA :: Identifier -> Question -> [IPv6] -> DNSMessage
responseAAAA idt q ips = makeResponse idt q as
where
dom = qname q
as = ResourceRecord dom AAAA classIN 300 . RD_AAAA <$> ips
|