You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

145 lines
3.2 KiB

package codec
import (
"encoding/binary"
"io"
"github.com/go-faster/errors"
"github.com/gotd/td/bin"
)
// AbridgedClientStart is starting bytes sent by client in Abridged mode.
//
// Note that server does not respond with it.
var AbridgedClientStart = [1]byte{0xef}
// Abridged is intermediate MTProto transport.
//
// See https://core.telegram.org/mtproto/mtproto-transports#abridged
type Abridged struct{}
var (
_ TaggedCodec = Abridged{}
)
// WriteHeader sends protocol tag.
func (i Abridged) WriteHeader(w io.Writer) error {
if _, err := w.Write(AbridgedClientStart[:]); err != nil {
return errors.Wrap(err, "write abridged header")
}
return nil
}
// ReadHeader reads protocol tag.
func (i Abridged) ReadHeader(r io.Reader) error {
var b [1]byte
if _, err := io.ReadFull(r, b[:]); err != nil {
return errors.Wrap(err, "read abridged header")
}
if b != AbridgedClientStart {
return ErrProtocolHeaderMismatch
}
return nil
}
// ObfuscatedTag returns protocol tag for obfuscation.
func (i Abridged) ObfuscatedTag() (r [4]byte) {
d := AbridgedClientStart[0]
return [4]byte{d, d, d, d}
}
// Write encode to writer message from given buffer.
func (i Abridged) Write(w io.Writer, b *bin.Buffer) error {
if err := checkOutgoingMessage(b); err != nil {
return err
}
if err := checkAlign(b, 4); err != nil {
return err
}
if err := writeAbridged(w, b); err != nil {
return errors.Wrap(err, "write abridged")
}
return nil
}
// Read fills buffer with received message.
func (i Abridged) Read(r io.Reader, b *bin.Buffer) error {
if err := readAbridged(r, b); err != nil {
return errors.Wrap(err, "read abridged")
}
return checkProtocolError(b)
}
func writeAbridged(w io.Writer, b *bin.Buffer) error {
length := b.Len()
// Re-using b.Buf if possible to reduce allocations.
b.Expand(4)
b.Buf = b.Buf[:length]
// Re-using b.Buf if possible to reduce allocations.
inner := bin.Buffer{Buf: b.Buf[length:length]}
encodeLength := b.Len() >> 2
// `0x7f == 127`, literally use one bit to distinguish length byte size.
if encodeLength < 127 {
// Payloads are wrapped in the following envelope:
//
// Length: payload length, divided by four, and encoded as a single byte,
// only if the resulting packet length is a value between 0x01..0x7e.
inner.Put([]byte{byte(encodeLength)})
} else {
// If the packet length divided by four is bigger than or equal to 127 (>= 0x7f),
// the following envelope must be used, instead:
//
var buf [5]byte
// Header: A single byte of value 0x7f
buf[0] = 0x7f
// Length: payload length, divided by four, and encoded as 3 length bytes (little endian)
binary.LittleEndian.PutUint32(buf[1:], uint32(encodeLength))
inner.Put(buf[:4])
}
if _, err := w.Write(inner.Buf); err != nil {
return err
}
if _, err := w.Write(b.Raw()); err != nil {
return err
}
return nil
}
func readAbridged(r io.Reader, b *bin.Buffer) error {
b.ResetN(bin.Word)
_, err := io.ReadFull(r, b.Buf[:1])
if err != nil {
return err
}
if b.Buf[0] >= 127 {
_, err := io.ReadFull(r, b.Buf[0:3])
if err != nil {
return err
}
}
n, err := b.Int()
if err != nil {
return err
}
b.ResetN(n << 2)
if _, err := io.ReadFull(r, b.Buf); err != nil {
return errors.Wrap(err, "read payload")
}
return nil
}