1
0
mirror of https://github.com/veggiedefender/torrent-client.git synced 2025-11-06 09:29:16 +02:00
Files
torrent-client/message/message.go

151 lines
3.5 KiB
Go
Raw Normal View History

2019-12-22 14:57:10 -05:00
package message
2019-12-22 12:51:59 -05:00
import (
2019-12-26 21:53:11 -05:00
"bufio"
2019-12-22 12:51:59 -05:00
"encoding/binary"
"errors"
2019-12-22 14:45:16 -05:00
"fmt"
2019-12-22 12:51:59 -05:00
"io"
)
2019-12-22 14:45:16 -05:00
type messageID uint8
2019-12-22 12:51:59 -05:00
// Message ID
const (
2019-12-22 14:45:16 -05:00
MsgChoke messageID = 0
MsgUnchoke messageID = 1
MsgInterested messageID = 2
MsgNotInterested messageID = 3
MsgHave messageID = 4
MsgBitfield messageID = 5
MsgRequest messageID = 6
MsgPiece messageID = 7
MsgCancel messageID = 8
MsgPort messageID = 9
2019-12-22 12:51:59 -05:00
)
// Message stores ID and payload of a message
2019-12-22 12:51:59 -05:00
type Message struct {
2019-12-22 14:45:16 -05:00
ID messageID
2019-12-22 12:51:59 -05:00
Payload []byte
}
// A Bitfield represents the pieces that a peer has
type Bitfield []byte
// FormatRequest formats the ID and payload for a request message
func FormatRequest(index, begin, length int) *Message {
payload := make([]byte, 12)
binary.BigEndian.PutUint32(payload[0:4], uint32(index))
binary.BigEndian.PutUint32(payload[4:8], uint32(begin))
binary.BigEndian.PutUint32(payload[8:12], uint32(length))
return &Message{
ID: MsgRequest,
Payload: payload,
}
}
// ParsePiece parses a piece message and copies its payload into a buffer
func ParsePiece(index int, buf []byte, msg *Message) (int, error) {
if msg.ID != MsgPiece {
return 0, fmt.Errorf("Expected piece (ID %d), got ID %d", MsgPiece, msg.ID)
}
if len(msg.Payload) < 8 {
return 0, errors.New("Payload too short")
}
parsedIndex := int(binary.BigEndian.Uint32(msg.Payload[0:4]))
if parsedIndex != index {
return 0, fmt.Errorf("Expected index %d, got %d", index, parsedIndex)
}
begin := int(binary.BigEndian.Uint32(msg.Payload[4:8]))
if begin >= len(buf) {
return 0, fmt.Errorf("Begin offset too high. %d >= %d", begin, len(buf))
}
data := msg.Payload[8:]
if begin+len(data) > len(buf) {
return 0, fmt.Errorf("Data too long [%d] for offset %d with length %d", len(data), begin, len(buf))
}
copy(buf[begin:], data)
return len(data), nil
}
2019-12-22 12:51:59 -05:00
// Serialize serializes a message into a buffer of the form
// <length prefix><message ID><payload>
// Interprets `nil` as a keep-alive message
2019-12-22 12:51:59 -05:00
func (m *Message) Serialize() []byte {
if m == nil {
return make([]byte, 4)
}
length := uint32(len(m.Payload) + 1) // +1 for id
buf := make([]byte, 4+length)
binary.BigEndian.PutUint32(buf[0:4], length)
buf[4] = byte(m.ID)
copy(buf[5:], m.Payload)
return buf
}
2019-12-24 10:39:22 -05:00
// Read parses a message from a stream. Returns `nil` on keep-alive message
2019-12-26 21:53:11 -05:00
func Read(r *bufio.Reader) (*Message, error) {
2019-12-22 12:51:59 -05:00
lengthBuf := make([]byte, 4)
_, err := io.ReadFull(r, lengthBuf)
if err != nil {
return nil, err
}
length := binary.BigEndian.Uint32(lengthBuf)
2019-12-22 14:45:16 -05:00
// keep-alive message
2019-12-22 12:51:59 -05:00
if length == 0 {
return nil, nil
}
messageBuf := make([]byte, length)
_, err = io.ReadFull(r, messageBuf)
if err != nil {
return nil, err
}
m := Message{
2019-12-22 14:45:16 -05:00
ID: messageID(messageBuf[0]),
2019-12-22 12:51:59 -05:00
Payload: messageBuf[1:],
}
return &m, nil
}
2019-12-22 14:45:16 -05:00
func (m *Message) String() string {
if m == nil {
return "KeepAlive"
}
switch m.ID {
case MsgChoke:
2019-12-26 11:21:14 -05:00
return "Choke"
2019-12-22 14:45:16 -05:00
case MsgUnchoke:
2019-12-26 11:21:14 -05:00
return "Unchoke"
2019-12-22 14:45:16 -05:00
case MsgInterested:
2019-12-26 11:21:14 -05:00
return "Interested"
2019-12-22 14:45:16 -05:00
case MsgNotInterested:
2019-12-26 11:21:14 -05:00
return "NotInterested"
2019-12-22 14:45:16 -05:00
case MsgHave:
2019-12-26 11:21:14 -05:00
return "Have"
2019-12-22 14:45:16 -05:00
case MsgBitfield:
2019-12-26 11:21:14 -05:00
return "Bitfield"
2019-12-22 14:45:16 -05:00
case MsgRequest:
2019-12-26 11:21:14 -05:00
return "Request"
2019-12-22 14:45:16 -05:00
case MsgPiece:
2019-12-26 11:21:14 -05:00
return "Piece"
2019-12-22 14:45:16 -05:00
case MsgCancel:
2019-12-26 11:21:14 -05:00
return "Cancel"
2019-12-22 14:45:16 -05:00
case MsgPort:
2019-12-26 11:21:14 -05:00
return "Port"
2019-12-22 14:45:16 -05:00
default:
2019-12-26 11:21:14 -05:00
return fmt.Sprintf("Unknown#%d", m.ID)
2019-12-22 14:45:16 -05:00
}
}
// HasPiece tells if a bitfield has a particular index set
2019-12-26 11:21:14 -05:00
func (bf Bitfield) HasPiece(index int) bool {
byteIndex := index / 8
offset := index % 8
2019-12-26 11:21:14 -05:00
return bf[byteIndex]>>(7-offset)&1 != 0
}