forked from cenkalti/rain
-
Notifications
You must be signed in to change notification settings - Fork 0
/
infodownloader.go
85 lines (74 loc) 路 2.09 KB
/
infodownloader.go
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
package infodownloader
import (
"fmt"
"github.com/cenkalti/rain/internal/peer"
"github.com/cenkalti/rain/internal/peerprotocol"
)
const blockSize = 16 * 1024
// InfoDownloader downloads all blocks of a piece from a peer.
type InfoDownloader struct {
Peer *peer.Peer
Bytes []byte
blocks []block
requested map[uint32]struct{}
nextBlockIndex uint32
}
type block struct {
size uint32
}
func New(pe *peer.Peer) *InfoDownloader {
d := &InfoDownloader{
Peer: pe,
Bytes: make([]byte, pe.ExtensionHandshake.MetadataSize),
requested: make(map[uint32]struct{}),
}
d.blocks = d.createBlocks()
return d
}
func (d *InfoDownloader) GotBlock(index uint32, data []byte) error {
if _, ok := d.requested[index]; !ok {
return fmt.Errorf("peer sent unrequested index for metadata message: %q", index)
}
b := &d.blocks[index]
if uint32(len(data)) != b.size {
return fmt.Errorf("peer sent invalid size for metadata message: %q", len(data))
}
delete(d.requested, index)
begin := index * blockSize
end := begin + b.size
copy(d.Bytes[begin:end], data)
return nil
}
func (d *InfoDownloader) createBlocks() []block {
numBlocks := d.Peer.ExtensionHandshake.MetadataSize / blockSize
mod := d.Peer.ExtensionHandshake.MetadataSize % blockSize
if mod != 0 {
numBlocks++
}
blocks := make([]block, numBlocks)
for i := range blocks {
blocks[i] = block{
size: blockSize,
}
}
if mod != 0 && len(blocks) > 0 {
blocks[len(blocks)-1].size = mod
}
return blocks
}
func (d *InfoDownloader) RequestBlocks(queueLength int) {
for ; d.nextBlockIndex < uint32(len(d.blocks)) && len(d.requested) < queueLength; d.nextBlockIndex++ {
msg := peerprotocol.ExtensionMessage{
ExtendedMessageID: d.Peer.ExtensionHandshake.M[peerprotocol.ExtensionKeyMetadata],
Payload: peerprotocol.ExtensionMetadataMessage{
Type: peerprotocol.ExtensionMetadataMessageTypeRequest,
Piece: d.nextBlockIndex,
},
}
d.Peer.SendMessage(msg)
d.requested[d.nextBlockIndex] = struct{}{}
}
}
func (d *InfoDownloader) Done() bool {
return d.nextBlockIndex == uint32(len(d.blocks)) && len(d.requested) == 0
}