diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/Data/Torrent.hs | 2 | ||||
-rw-r--r-- | src/Network/Address.hs | 86 | ||||
-rw-r--r-- | src/Network/BitTorrent/DHT/ContactInfo.hs | 43 |
3 files changed, 55 insertions, 76 deletions
diff --git a/src/Data/Torrent.hs b/src/Data/Torrent.hs index b3bf0ec7..73422a04 100644 --- a/src/Data/Torrent.hs +++ b/src/Data/Torrent.hs | |||
@@ -239,10 +239,12 @@ instance BEncode InfoHash where | |||
239 | fromBEncode be = InfoHash <$> fromBEncode be | 239 | fromBEncode be = InfoHash <$> fromBEncode be |
240 | #endif | 240 | #endif |
241 | 241 | ||
242 | #if 0 | ||
242 | instance TableKey KMessageOf InfoHash where | 243 | instance TableKey KMessageOf InfoHash where |
243 | toNodeId = either (error msg) id . S.decode . S.encode | 244 | toNodeId = either (error msg) id . S.decode . S.encode |
244 | where -- TODO unsafe coerse? | 245 | where -- TODO unsafe coerse? |
245 | msg = "tableKey: impossible" | 246 | msg = "tableKey: impossible" |
247 | #endif | ||
246 | 248 | ||
247 | 249 | ||
248 | -- | Convert to\/from raw bytestring. (no encoding) | 250 | -- | Convert to\/from raw bytestring. (no encoding) |
diff --git a/src/Network/Address.hs b/src/Network/Address.hs index d34b0431..9ecd89a3 100644 --- a/src/Network/Address.hs +++ b/src/Network/Address.hs | |||
@@ -71,7 +71,6 @@ module Network.Address | |||
71 | , NodeInfo (..) | 71 | , NodeInfo (..) |
72 | , mapAddress | 72 | , mapAddress |
73 | , traverseAddress | 73 | , traverseAddress |
74 | , rank | ||
75 | 74 | ||
76 | -- * Fingerprint | 75 | -- * Fingerprint |
77 | -- $fingerprint | 76 | -- $fingerprint |
@@ -83,7 +82,7 @@ module Network.Address | |||
83 | -- * Utils | 82 | -- * Utils |
84 | , libUserAgent | 83 | , libUserAgent |
85 | , sockAddrPort | 84 | , sockAddrPort |
86 | , withPort | 85 | , setPort |
87 | , getBindAddress | 86 | , getBindAddress |
88 | ) where | 87 | ) where |
89 | 88 | ||
@@ -148,17 +147,11 @@ sockAddrPort (SockAddrInet6 p _ _ _) = Just p | |||
148 | sockAddrPort _ = Nothing | 147 | sockAddrPort _ = Nothing |
149 | {-# INLINE sockAddrPort #-} | 148 | {-# INLINE sockAddrPort #-} |
150 | 149 | ||
151 | withPort :: SockAddr -> PortNumber -> SockAddr | ||
152 | withPort (SockAddrInet _ ip ) port = SockAddrInet port ip | ||
153 | withPort (SockAddrInet6 _ flow ip scope) port = SockAddrInet6 port flow ip scope | ||
154 | withPort addr _ = addr | ||
155 | {-# INLINE withPort #-} | ||
156 | |||
157 | instance Address a => Address (NodeAddr a) where | 150 | instance Address a => Address (NodeAddr a) where |
158 | toSockAddr NodeAddr {..} = setPort nodePort $ toSockAddr nodeHost | 151 | toSockAddr NodeAddr {..} = setPort nodePort $ toSockAddr nodeHost |
159 | fromSockAddr sa = NodeAddr <$> fromSockAddr sa <*> sockAddrPort sa | 152 | fromSockAddr sa = NodeAddr <$> fromSockAddr sa <*> sockAddrPort sa |
160 | 153 | ||
161 | instance Address a => Address (PeerAddr a) where | 154 | instance Address PeerAddr where |
162 | toSockAddr PeerAddr {..} = setPort peerPort $ toSockAddr peerHost | 155 | toSockAddr PeerAddr {..} = setPort peerPort $ toSockAddr peerHost |
163 | fromSockAddr sa = PeerAddr Nothing <$> fromSockAddr sa <*> sockAddrPort sa | 156 | fromSockAddr sa = PeerAddr Nothing <$> fromSockAddr sa <*> sockAddrPort sa |
164 | 157 | ||
@@ -412,23 +405,18 @@ instance BEncode IPv6 where | |||
412 | {-# INLINE fromBEncode #-} | 405 | {-# INLINE fromBEncode #-} |
413 | #endif | 406 | #endif |
414 | 407 | ||
415 | {----------------------------------------------------------------------- | ||
416 | -- Peer addr | ||
417 | -----------------------------------------------------------------------} | ||
418 | -- TODO check semantic of ord and eq instances | ||
419 | |||
420 | -- | Peer address info normally extracted from peer list or peer | 408 | -- | Peer address info normally extracted from peer list or peer |
421 | -- compact list encoding. | 409 | -- compact list encoding. |
422 | data PeerAddr a = PeerAddr | 410 | data PeerAddr = PeerAddr |
423 | { peerId :: !(Maybe PeerId) | 411 | { peerId :: !(Maybe PeerId) |
424 | 412 | ||
425 | -- | This is usually 'IPv4', 'IPv6', 'IP' or unresolved | 413 | -- | This is usually 'IPv4', 'IPv6', 'IP' or unresolved |
426 | -- 'HostName'. | 414 | -- 'HostName'. |
427 | , peerHost :: !a | 415 | , peerHost :: !IP |
428 | 416 | ||
429 | -- | The port the peer listenning for incoming P2P sessions. | 417 | -- | The port the peer listenning for incoming P2P sessions. |
430 | , peerPort :: {-# UNPACK #-} !PortNumber | 418 | , peerPort :: {-# UNPACK #-} !PortNumber |
431 | } deriving (Show, Eq, Ord, Typeable, Functor) | 419 | } deriving (Show, Eq, Ord, Typeable) |
432 | 420 | ||
433 | #ifdef VERSION_bencoding | 421 | #ifdef VERSION_bencoding |
434 | peer_ip_key, peer_id_key, peer_port_key :: BKey | 422 | peer_ip_key, peer_id_key, peer_port_key :: BKey |
@@ -437,7 +425,7 @@ peer_id_key = "peer id" | |||
437 | peer_port_key = "port" | 425 | peer_port_key = "port" |
438 | 426 | ||
439 | -- | The tracker's 'announce response' compatible encoding. | 427 | -- | The tracker's 'announce response' compatible encoding. |
440 | instance (Typeable a, BEncode a) => BEncode (PeerAddr a) where | 428 | instance BEncode PeerAddr where |
441 | toBEncode PeerAddr {..} = toDict $ | 429 | toBEncode PeerAddr {..} = toDict $ |
442 | peer_ip_key .=! peerHost | 430 | peer_ip_key .=! peerHost |
443 | .: peer_id_key .=? peerId | 431 | .: peer_id_key .=? peerId |
@@ -457,37 +445,39 @@ instance (Typeable a, BEncode a) => BEncode (PeerAddr a) where | |||
457 | -- | 445 | -- |
458 | -- For more info see: <http://www.bittorrent.org/beps/bep_0023.html> | 446 | -- For more info see: <http://www.bittorrent.org/beps/bep_0023.html> |
459 | -- | 447 | -- |
460 | -- TODO: test byte order | 448 | -- WARNING: Input must be exactly 6 or 18 bytes so that we can identify IP version. |
461 | instance (Serialize a) => Serialize (PeerAddr a) where | 449 | -- |
450 | instance Serialize PeerAddr where | ||
462 | put PeerAddr {..} = put peerHost >> put peerPort | 451 | put PeerAddr {..} = put peerHost >> put peerPort |
463 | get = PeerAddr Nothing <$> get <*> get | 452 | get = do |
453 | cnt <- remaining | ||
454 | PeerAddr Nothing <$> isolate (cnt - 2) get <*> get | ||
464 | 455 | ||
465 | -- | @127.0.0.1:6881@ | 456 | -- | @127.0.0.1:6881@ |
466 | instance Default (PeerAddr IPv4) where | 457 | instance Default PeerAddr where |
467 | def = "127.0.0.1:6881" | 458 | def = "127.0.0.1:6881" |
468 | 459 | ||
469 | -- | @127.0.0.1:6881@ | ||
470 | instance Default (PeerAddr IP) where | ||
471 | def = IPv4 <$> def | ||
472 | |||
473 | -- | Example: | 460 | -- | Example: |
474 | -- | 461 | -- |
475 | -- @peerPort \"127.0.0.1:6881\" == 6881@ | 462 | -- @peerPort \"127.0.0.1:6881\" == 6881@ |
476 | -- | 463 | -- |
477 | instance IsString (PeerAddr IPv4) where | 464 | instance IsString PeerAddr where |
478 | fromString str | 465 | fromString str |
479 | | [hostAddrStr, portStr] <- splitWhen (== ':') str | 466 | | [hostAddrStr, portStr] <- splitWhen (== ':') str |
480 | , Just hostAddr <- readMaybe hostAddrStr | 467 | , Just hostAddr <- readMaybe hostAddrStr |
481 | , Just portNum <- toEnum <$> readMaybe portStr | 468 | , Just portNum <- toEnum <$> readMaybe portStr |
482 | = PeerAddr Nothing hostAddr portNum | 469 | = PeerAddr Nothing (IPv4 hostAddr) portNum |
483 | | otherwise = error $ "fromString: unable to parse (PeerAddr IPv4): " ++ str | 470 | | [((ip,port),"")] <- readsIPv6_port str = |
471 | PeerAddr Nothing (IPv6 ip) port | ||
472 | | otherwise = error $ "fromString: unable to parse IP: " ++ str | ||
484 | 473 | ||
485 | instance Read (PeerAddr IPv4) where | 474 | instance Read PeerAddr where |
486 | readsPrec i = RP.readP_to_S $ do | 475 | readsPrec i = RP.readP_to_S $ do |
487 | ipv4 <- RP.readS_to_P (readsPrec i) | 476 | ip <- IPv4 <$> ( RP.readS_to_P (readsPrec i) ) |
477 | <|> IPv6 <$> ( RP.char '[' *> (RP.readS_to_P reads) <* RP.char ']' ) | ||
488 | _ <- RP.char ':' | 478 | _ <- RP.char ':' |
489 | port <- toEnum <$> RP.readS_to_P (readsPrec i) | 479 | port <- toEnum <$> RP.readS_to_P (readsPrec i) |
490 | return $ PeerAddr Nothing ipv4 port | 480 | return $ PeerAddr Nothing ip port |
491 | 481 | ||
492 | readsIPv6_port :: String -> [((IPv6, PortNumber), String)] | 482 | readsIPv6_port :: String -> [((IPv6, PortNumber), String)] |
493 | readsIPv6_port = RP.readP_to_S $ do | 483 | readsIPv6_port = RP.readP_to_S $ do |
@@ -496,27 +486,16 @@ readsIPv6_port = RP.readP_to_S $ do | |||
496 | port <- toEnum <$> read <$> (RP.many1 $ RP.satisfy isDigit) <* RP.eof | 486 | port <- toEnum <$> read <$> (RP.many1 $ RP.satisfy isDigit) <* RP.eof |
497 | return (ip,port) | 487 | return (ip,port) |
498 | 488 | ||
499 | instance IsString (PeerAddr IPv6) where | ||
500 | fromString str | ||
501 | | [((ip,port),"")] <- readsIPv6_port str = | ||
502 | PeerAddr Nothing ip port | ||
503 | | otherwise = error $ "fromString: unable to parse (PeerAddr IPv6): " ++ str | ||
504 | |||
505 | instance IsString (PeerAddr IP) where | ||
506 | fromString str | ||
507 | | '[' `L.elem` str = IPv6 <$> fromString str | ||
508 | | otherwise = IPv4 <$> fromString str | ||
509 | 489 | ||
510 | -- | fingerprint + "at" + dotted.host.inet.addr:port | 490 | -- | fingerprint + "at" + dotted.host.inet.addr:port |
511 | -- TODO: instances for IPv6, HostName | 491 | instance Pretty PeerAddr where |
512 | instance Pretty a => Pretty (PeerAddr a) where | ||
513 | pPrint PeerAddr {..} | 492 | pPrint PeerAddr {..} |
514 | | Just pid <- peerId = pPrint (fingerprint pid) <+> "at" <+> paddr | 493 | | Just pid <- peerId = pPrint (fingerprint pid) <+> "at" <+> paddr |
515 | | otherwise = paddr | 494 | | otherwise = paddr |
516 | where | 495 | where |
517 | paddr = pPrint peerHost <> ":" <> text (show peerPort) | 496 | paddr = pPrint peerHost <> ":" <> text (show peerPort) |
518 | 497 | ||
519 | instance Hashable a => Hashable (PeerAddr a) where | 498 | instance Hashable PeerAddr where |
520 | hashWithSalt s PeerAddr {..} = | 499 | hashWithSalt s PeerAddr {..} = |
521 | s `hashWithSalt` peerId `hashWithSalt` peerHost `hashWithSalt` peerPort | 500 | s `hashWithSalt` peerId `hashWithSalt` peerHost `hashWithSalt` peerPort |
522 | 501 | ||
@@ -524,10 +503,7 @@ instance Hashable a => Hashable (PeerAddr a) where | |||
524 | defaultPorts :: [PortNumber] | 503 | defaultPorts :: [PortNumber] |
525 | defaultPorts = [6881..6889] | 504 | defaultPorts = [6881..6889] |
526 | 505 | ||
527 | _resolvePeerAddr :: (IPAddress i) => PeerAddr HostName -> PeerAddr i | 506 | _peerSockAddr :: PeerAddr -> (Family, SockAddr) |
528 | _resolvePeerAddr = undefined | ||
529 | |||
530 | _peerSockAddr :: PeerAddr IP -> (Family, SockAddr) | ||
531 | _peerSockAddr PeerAddr {..} = | 507 | _peerSockAddr PeerAddr {..} = |
532 | case peerHost of | 508 | case peerHost of |
533 | IPv4 ipv4 -> | 509 | IPv4 ipv4 -> |
@@ -535,11 +511,11 @@ _peerSockAddr PeerAddr {..} = | |||
535 | IPv6 ipv6 -> | 511 | IPv6 ipv6 -> |
536 | (AF_INET6, SockAddrInet6 peerPort 0 (toHostAddress6 ipv6) 0) | 512 | (AF_INET6, SockAddrInet6 peerPort 0 (toHostAddress6 ipv6) 0) |
537 | 513 | ||
538 | peerSockAddr :: PeerAddr IP -> SockAddr | 514 | peerSockAddr :: PeerAddr -> SockAddr |
539 | peerSockAddr = snd . _peerSockAddr | 515 | peerSockAddr = snd . _peerSockAddr |
540 | 516 | ||
541 | -- | Create a socket connected to the address specified in a peerAddr | 517 | -- | Create a socket connected to the address specified in a peerAddr |
542 | peerSocket :: SocketType -> PeerAddr IP -> IO Socket | 518 | peerSocket :: SocketType -> PeerAddr -> IO Socket |
543 | peerSocket socketType pa = do | 519 | peerSocket socketType pa = do |
544 | let (family, addr) = _peerSockAddr pa | 520 | let (family, addr) = _peerSockAddr pa |
545 | sock <- socket family socketType defaultProtocol | 521 | sock <- socket family socketType defaultProtocol |
@@ -607,7 +583,7 @@ instance BEncode a => BEncode (NodeAddr a) where | |||
607 | {-# INLINE fromBEncode #-} | 583 | {-# INLINE fromBEncode #-} |
608 | #endif | 584 | #endif |
609 | 585 | ||
610 | fromPeerAddr :: PeerAddr a -> NodeAddr a | 586 | fromPeerAddr :: PeerAddr -> NodeAddr IP |
611 | fromPeerAddr PeerAddr {..} = NodeAddr | 587 | fromPeerAddr PeerAddr {..} = NodeAddr |
612 | { nodeHost = peerHost | 588 | { nodeHost = peerHost |
613 | , nodePort = peerPort | 589 | , nodePort = peerPort |
@@ -615,12 +591,6 @@ fromPeerAddr PeerAddr {..} = NodeAddr | |||
615 | 591 | ||
616 | ------------------------------------------------------------------------ | 592 | ------------------------------------------------------------------------ |
617 | 593 | ||
618 | -- | Order by closeness: nearest nodes first. | ||
619 | rank :: ( Ord (NodeId dht) | ||
620 | , Bits (NodeId dht) | ||
621 | ) => (x -> NodeId dht) -> NodeId dht -> [x] -> [x] | ||
622 | rank f nid = L.sortBy (comparing (RPC.distance nid . f)) | ||
623 | |||
624 | {----------------------------------------------------------------------- | 594 | {----------------------------------------------------------------------- |
625 | -- Fingerprint | 595 | -- Fingerprint |
626 | -----------------------------------------------------------------------} | 596 | -----------------------------------------------------------------------} |
diff --git a/src/Network/BitTorrent/DHT/ContactInfo.hs b/src/Network/BitTorrent/DHT/ContactInfo.hs index 3db3d5a8..172306a1 100644 --- a/src/Network/BitTorrent/DHT/ContactInfo.hs +++ b/src/Network/BitTorrent/DHT/ContactInfo.hs | |||
@@ -1,6 +1,7 @@ | |||
1 | {-# LANGUAGE BangPatterns #-} | 1 | {-# LANGUAGE BangPatterns #-} |
2 | module Network.BitTorrent.DHT.ContactInfo | 2 | module Network.BitTorrent.DHT.ContactInfo |
3 | ( PeerStore | 3 | ( PeerStore |
4 | , PeerAddr(..) | ||
4 | , Network.BitTorrent.DHT.ContactInfo.lookup | 5 | , Network.BitTorrent.DHT.ContactInfo.lookup |
5 | , Network.BitTorrent.DHT.ContactInfo.freshPeers | 6 | , Network.BitTorrent.DHT.ContactInfo.freshPeers |
6 | , Network.BitTorrent.DHT.ContactInfo.insertPeer | 7 | , Network.BitTorrent.DHT.ContactInfo.insertPeer |
@@ -36,17 +37,17 @@ import Network.Address | |||
36 | -- -- PeerSet | 37 | -- -- PeerSet |
37 | -- -----------------------------------------------------------------------} | 38 | -- -----------------------------------------------------------------------} |
38 | -- | 39 | -- |
39 | -- type PeerSet a = [(PeerAddr a, NodeInfo a, Timestamp)] | 40 | -- type PeerSet a = [(PeerAddr, NodeInfo a, Timestamp)] |
40 | -- | 41 | -- |
41 | -- -- compare PSQueue vs Ordered list | 42 | -- -- compare PSQueue vs Ordered list |
42 | -- | 43 | -- |
43 | -- takeNewest :: PeerSet a -> [PeerAddr a] | 44 | -- takeNewest :: PeerSet a -> [PeerAddr] |
44 | -- takeNewest = undefined | 45 | -- takeNewest = undefined |
45 | -- | 46 | -- |
46 | -- dropOld :: Timestamp -> PeerSet a -> PeerSet a | 47 | -- dropOld :: Timestamp -> PeerSet a -> PeerSet a |
47 | -- dropOld = undefined | 48 | -- dropOld = undefined |
48 | -- | 49 | -- |
49 | -- insert :: PeerAddr a -> Timestamp -> PeerSet a -> PeerSet a | 50 | -- insert :: PeerAddr -> Timestamp -> PeerSet a -> PeerSet a |
50 | -- insert = undefined | 51 | -- insert = undefined |
51 | -- | 52 | -- |
52 | -- type Mask = Int | 53 | -- type Mask = Int |
@@ -116,16 +117,19 @@ import Network.Address | |||
116 | 117 | ||
117 | -- | Storage used to keep track a set of known peers in client, | 118 | -- | Storage used to keep track a set of known peers in client, |
118 | -- tracker or DHT sessions. | 119 | -- tracker or DHT sessions. |
119 | newtype PeerStore ip = PeerStore (HashMap InfoHash (SwarmData ip)) | 120 | newtype PeerStore = PeerStore (HashMap InfoHash SwarmData) |
120 | 121 | ||
121 | type Timestamp = POSIXTime | 122 | type Timestamp = POSIXTime |
122 | 123 | ||
123 | data SwarmData ip = SwarmData | 124 | data SwarmData = SwarmData |
124 | { peers :: !(PSQ (PeerAddr ip) Timestamp) | 125 | { peers :: !(PSQ PeerAddr Timestamp) |
125 | , name :: !(Maybe ByteString) | 126 | , name :: !(Maybe ByteString) |
126 | } | 127 | } |
127 | 128 | ||
128 | 129 | -- | This wrapper will serialize an ip address with a '4' or '6' prefix byte | |
130 | -- to indicate whether it is IPv4 or IPv6. | ||
131 | -- | ||
132 | -- Note: it does not serialize port numbers. | ||
129 | newtype SerializeAddress a = SerializeAddress { unserializeAddress :: a } | 133 | newtype SerializeAddress a = SerializeAddress { unserializeAddress :: a } |
130 | 134 | ||
131 | instance Address a => Serialize (SerializeAddress a) where | 135 | instance Address a => Serialize (SerializeAddress a) where |
@@ -145,7 +149,8 @@ instance Address a => Serialize (SerializeAddress a) where | |||
145 | | otherwise = return $ error "cannot serialize non-IP SerializeAddress" | 149 | | otherwise = return $ error "cannot serialize non-IP SerializeAddress" |
146 | 150 | ||
147 | 151 | ||
148 | instance (Ord ip, Address ip) => Serialize (SwarmData ip) where | 152 | {- XXX: What happened to the ports? |
153 | instance Serialize SwarmData where | ||
149 | get = flip SwarmData <$> get | 154 | get = flip SwarmData <$> get |
150 | <*> ( PSQ.fromList . L.map parseAddr <$> get ) | 155 | <*> ( PSQ.fromList . L.map parseAddr <$> get ) |
151 | where | 156 | where |
@@ -157,17 +162,17 @@ instance (Ord ip, Address ip) => Serialize (SwarmData ip) where | |||
157 | put $ L.map (\(addr :-> _) -> (SerializeAddress <$> addr)) | 162 | put $ L.map (\(addr :-> _) -> (SerializeAddress <$> addr)) |
158 | -- XXX: should we serialize the timestamp? | 163 | -- XXX: should we serialize the timestamp? |
159 | $ PSQ.toList peers | 164 | $ PSQ.toList peers |
165 | -} | ||
160 | 166 | ||
161 | 167 | knownSwarms :: PeerStore -> [ (InfoHash, Int, Maybe ByteString) ] | |
162 | knownSwarms :: PeerStore ip -> [ (InfoHash, Int, Maybe ByteString) ] | ||
163 | knownSwarms (PeerStore m) = L.map (\(ih,SwarmData q n) -> (ih,PSQ.size q,n)) $ HM.toList m | 168 | knownSwarms (PeerStore m) = L.map (\(ih,SwarmData q n) -> (ih,PSQ.size q,n)) $ HM.toList m |
164 | 169 | ||
165 | swarmSingleton :: Ord ip => PeerAddr ip -> SwarmData ip | 170 | swarmSingleton :: PeerAddr -> SwarmData |
166 | swarmSingleton a = SwarmData | 171 | swarmSingleton a = SwarmData |
167 | { peers = PSQ.singleton a 0 | 172 | { peers = PSQ.singleton a 0 |
168 | , name = Nothing } | 173 | , name = Nothing } |
169 | 174 | ||
170 | swarmInsert :: Ord ip => SwarmData ip -> SwarmData ip -> SwarmData ip | 175 | swarmInsert :: SwarmData -> SwarmData -> SwarmData |
171 | swarmInsert old new = SwarmData | 176 | swarmInsert old new = SwarmData |
172 | { peers = L.foldl' (\q (a :-> t) -> PSQ.insertWith (\p _ -> p) a t q) (peers old) (PSQ.toList $ peers new) | 177 | { peers = L.foldl' (\q (a :-> t) -> PSQ.insertWith (\p _ -> p) a t q) (peers old) (PSQ.toList $ peers new) |
173 | , name = name new <|> name old -- TODO: decodeUtf8' check | 178 | , name = name new <|> name old -- TODO: decodeUtf8' check |
@@ -176,12 +181,12 @@ swarmInsert old new = SwarmData | |||
176 | isSwarmOccupied SwarmData{..} = not $ PSQ.null peers | 181 | isSwarmOccupied SwarmData{..} = not $ PSQ.null peers |
177 | 182 | ||
178 | -- | Empty store. | 183 | -- | Empty store. |
179 | instance Default (PeerStore a) where | 184 | instance Default (PeerStore) where |
180 | def = PeerStore HM.empty | 185 | def = PeerStore HM.empty |
181 | {-# INLINE def #-} | 186 | {-# INLINE def #-} |
182 | 187 | ||
183 | -- | Monoid under union operation. | 188 | -- | Monoid under union operation. |
184 | instance Ord a => Monoid (PeerStore a) where | 189 | instance Monoid PeerStore where |
185 | mempty = def | 190 | mempty = def |
186 | {-# INLINE mempty #-} | 191 | {-# INLINE mempty #-} |
187 | 192 | ||
@@ -189,20 +194,22 @@ instance Ord a => Monoid (PeerStore a) where | |||
189 | PeerStore (HM.unionWith swarmInsert a b) | 194 | PeerStore (HM.unionWith swarmInsert a b) |
190 | {-# INLINE mappend #-} | 195 | {-# INLINE mappend #-} |
191 | 196 | ||
197 | {- | ||
192 | -- | Can be used to store peers between invocations of the client | 198 | -- | Can be used to store peers between invocations of the client |
193 | -- software. | 199 | -- software. |
194 | instance (Ord a, Address a) => Serialize (PeerStore a) where | 200 | instance Serialize PeerStore where |
195 | get = PeerStore . HM.fromList <$> get | 201 | get = PeerStore . HM.fromList <$> get |
196 | put (PeerStore m) = put (L.filter (isSwarmOccupied . snd) $ HM.toList m) | 202 | put (PeerStore m) = put (L.filter (isSwarmOccupied . snd) $ HM.toList m) |
203 | -} | ||
197 | 204 | ||
198 | -- | Returns all peers associated with a given info hash. | 205 | -- | Returns all peers associated with a given info hash. |
199 | lookup :: Ord a => InfoHash -> PeerStore a -> [PeerAddr a] | 206 | lookup :: InfoHash -> PeerStore -> [PeerAddr] |
200 | lookup ih (PeerStore m) = maybe [] (PSQ.keys . peers) $ HM.lookup ih m | 207 | lookup ih (PeerStore m) = maybe [] (PSQ.keys . peers) $ HM.lookup ih m |
201 | 208 | ||
202 | batchSize = 64 | 209 | batchSize = 64 |
203 | 210 | ||
204 | -- | Used in 'get_peers' DHT queries. | 211 | -- | Used in 'get_peers' DHT queries. |
205 | freshPeers :: Ord a => InfoHash -> Timestamp -> PeerStore a -> ([PeerAddr a], PeerStore a) | 212 | freshPeers :: InfoHash -> Timestamp -> PeerStore -> ([PeerAddr], PeerStore) |
206 | freshPeers ih tm (PeerStore m) = fromMaybe ([],PeerStore m) $ do | 213 | freshPeers ih tm (PeerStore m) = fromMaybe ([],PeerStore m) $ do |
207 | swarm <- HM.lookup ih m | 214 | swarm <- HM.lookup ih m |
208 | let ps0 = take batchSize $ unfoldr (incomp minView) (peers swarm) | 215 | let ps0 = take batchSize $ unfoldr (incomp minView) (peers swarm) |
@@ -219,7 +226,7 @@ incomp !f !x = do | |||
219 | pure $! ( (result,x'), x' ) | 226 | pure $! ( (result,x'), x' ) |
220 | 227 | ||
221 | -- | Used in 'announce_peer' DHT queries. | 228 | -- | Used in 'announce_peer' DHT queries. |
222 | insertPeer :: Ord a => InfoHash -> Maybe ByteString -> PeerAddr a -> PeerStore a -> PeerStore a | 229 | insertPeer :: InfoHash -> Maybe ByteString -> PeerAddr -> PeerStore -> PeerStore |
223 | insertPeer !ih !name !a !(PeerStore m) = seq a' $ PeerStore (HM.insertWith swarmInsert ih a' m) | 230 | insertPeer !ih !name !a !(PeerStore m) = seq a' $ PeerStore (HM.insertWith swarmInsert ih a' m) |
224 | where | 231 | where |
225 | a' = SwarmData { peers = PSQ.singleton a 0 | 232 | a' = SwarmData { peers = PSQ.singleton a 0 |