109 lines
3.1 KiB
Go
109 lines
3.1 KiB
Go
package torrent
|
|
|
|
import (
|
|
"testing"
|
|
"time"
|
|
|
|
. "gopkg.in/check.v1"
|
|
|
|
"github.com/bradfitz/iter"
|
|
|
|
"bitbucket.org/anacrolix/go.torrent/internal/pieceordering"
|
|
|
|
"bitbucket.org/anacrolix/go.torrent/peer_protocol"
|
|
)
|
|
|
|
func TestCancelRequestOptimized(t *testing.T) {
|
|
c := &connection{
|
|
PeerMaxRequests: 1,
|
|
PeerPieces: []bool{false, true},
|
|
post: make(chan peer_protocol.Message),
|
|
writeCh: make(chan []byte),
|
|
}
|
|
if len(c.Requests) != 0 {
|
|
t.FailNow()
|
|
}
|
|
// Keepalive timeout of 0 works because I'm just that good.
|
|
go c.writeOptimizer(0 * time.Millisecond)
|
|
c.Request(newRequest(1, 2, 3))
|
|
if len(c.Requests) != 1 {
|
|
t.Fatal("request was not posted")
|
|
}
|
|
// Posting this message should removing the pending Request.
|
|
if !c.Cancel(newRequest(1, 2, 3)) {
|
|
t.Fatal("request was not found")
|
|
}
|
|
// Check that the write optimization has filtered out the Request message.
|
|
for _, b := range []string{
|
|
// The initial request triggers an Interested message.
|
|
"\x00\x00\x00\x01\x02",
|
|
// Let a keep-alive through to verify there were no pending messages.
|
|
"\x00\x00\x00\x00",
|
|
} {
|
|
bb := string(<-c.writeCh)
|
|
if b != bb {
|
|
t.Fatalf("received message %q is not expected: %q", bb, b)
|
|
}
|
|
}
|
|
close(c.post)
|
|
// Drain the write channel until it closes.
|
|
for b := range c.writeCh {
|
|
bs := string(b)
|
|
if bs != "\x00\x00\x00\x00" {
|
|
t.Fatal("got unexpected non-keepalive")
|
|
}
|
|
}
|
|
}
|
|
|
|
func pieceOrderingAsSlice(po *pieceordering.Instance) (ret []int) {
|
|
for e := po.First(); e != nil; e = e.Next() {
|
|
ret = append(ret, e.Piece())
|
|
}
|
|
return
|
|
}
|
|
|
|
func testRequestOrder(expected []int, ro *pieceordering.Instance, t *C) {
|
|
t.Assert(pieceOrderingAsSlice(ro), DeepEquals, expected)
|
|
}
|
|
|
|
type suite struct{}
|
|
|
|
var _ = Suite(suite{})
|
|
|
|
func Test(t *testing.T) { TestingT(t) }
|
|
|
|
// Tests the request ordering based on a connections priorities.
|
|
func (suite) TestPieceRequestOrder(t *C) {
|
|
c := connection{
|
|
pieceRequestOrder: pieceordering.New(),
|
|
piecePriorities: []int{1, 4, 0, 3, 2},
|
|
}
|
|
testRequestOrder(nil, c.pieceRequestOrder, t)
|
|
c.pendPiece(2, piecePriorityNone)
|
|
testRequestOrder(nil, c.pieceRequestOrder, t)
|
|
c.pendPiece(1, piecePriorityNormal)
|
|
c.pendPiece(2, piecePriorityNormal)
|
|
testRequestOrder([]int{2, 1}, c.pieceRequestOrder, t)
|
|
c.pendPiece(0, piecePriorityNormal)
|
|
testRequestOrder([]int{2, 0, 1}, c.pieceRequestOrder, t)
|
|
c.pendPiece(1, piecePriorityReadahead)
|
|
testRequestOrder([]int{1, 2, 0}, c.pieceRequestOrder, t)
|
|
c.pendPiece(4, piecePriorityNow)
|
|
// now(4), r(1), normal(0, 2)
|
|
testRequestOrder([]int{4, 1, 2, 0}, c.pieceRequestOrder, t)
|
|
c.pendPiece(2, piecePriorityReadahead)
|
|
// N(4), R(1, 2), N(0)
|
|
testRequestOrder([]int{4, 2, 1, 0}, c.pieceRequestOrder, t)
|
|
c.pendPiece(1, piecePriorityNow)
|
|
// now(4, 1), readahead(2), normal(0)
|
|
// in the same order, the keys will be: -15+6, -15+12, -5, 1
|
|
// so we test that a very low priority (for this connection), "now"
|
|
// piece has been placed after a readahead piece.
|
|
testRequestOrder([]int{4, 2, 1, 0}, c.pieceRequestOrder, t)
|
|
// Note this intentially sets to None a piece that's not in the order.
|
|
for i := range iter.N(5) {
|
|
c.pendPiece(i, piecePriorityNone)
|
|
}
|
|
testRequestOrder(nil, c.pieceRequestOrder, t)
|
|
}
|