| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394 |
- package rpc
-
- import (
- "bytes"
- "crypto/rand"
- "encoding/binary"
- "net"
-
- "github.com/juju/errors"
-
- "github.com/9seconds/mtg/mtproto"
- )
-
- const (
- rpcProxyRequestConnectionIDLength = 8
- rpcProxyRequestIPPortLength = 16 + 4
- )
-
- var (
- rpcProxyRequestTag = []byte{0xee, 0xf1, 0xce, 0x36}
- rpcProxyRequestExtraSize = []byte{0x18, 0x00, 0x00, 0x00}
- rpcProxyRequestProxyTag = []byte{0xae, 0x26, 0x1e, 0xdb}
- )
-
- type RPCProxyRequest struct {
- Flags RPCProxyRequestFlags
- ConnectionID [rpcProxyRequestConnectionIDLength]byte
- RemoteIPPort [rpcProxyRequestIPPortLength]byte
- LocalIPPort [rpcProxyRequestIPPortLength]byte
- ADTag []byte
- Extras Extras
- }
-
- func (r *RPCProxyRequest) Bytes(message []byte) []byte {
- buf := &bytes.Buffer{}
-
- flags := r.Flags
- if r.Extras.QuickAck {
- flags |= RPCProxyRequestFlagsQuickAck
- }
-
- if bytes.HasPrefix(message, rpcProxyRequestFlagsEncryptedPrefix[:]) {
- flags |= RPCProxyRequestFlagsEncrypted
- }
-
- buf.Write(rpcProxyRequestTag)
- buf.Write(flags.Bytes())
- buf.Write(r.ConnectionID[:])
- buf.Write(r.RemoteIPPort[:])
- buf.Write(r.LocalIPPort[:])
- buf.Write(rpcProxyRequestExtraSize)
- buf.Write(rpcProxyRequestProxyTag)
- buf.WriteByte(byte(len(r.ADTag)))
- buf.Write(r.ADTag)
-
- for i := 0; i < (buf.Len() % 4); i++ {
- buf.WriteByte(0x00)
- }
- buf.Write(message)
-
- return buf.Bytes()
- }
-
- func NewRPCProxyRequest(connectionType mtproto.ConnectionType, local, remote *net.TCPAddr, adTag []byte, extras Extras) (*RPCProxyRequest, error) {
- flags := RPCProxyRequestFlagsHasAdTag | RPCProxyRequestFlagsMagic | RPCProxyRequestFlagsExtMode2
-
- switch connectionType {
- case mtproto.ConnectionTypeAbridged:
- flags |= RPCProxyRequestFlagsAbdridged
- case mtproto.ConnectionTypeIntermediate:
- flags |= RPCProxyRequestFlagsIntermediate
- }
-
- request := RPCProxyRequest{
- Flags: flags,
- ADTag: adTag,
- Extras: extras,
- }
-
- if _, err := rand.Read(request.ConnectionID[:]); err != nil {
- return nil, errors.Annotate(err, "Cannot generate connection ID")
- }
-
- port := make([]byte, 4)
- copy(request.LocalIPPort[:], local.IP.To16())
- binary.LittleEndian.PutUint32(port, uint32(local.Port))
- copy(request.LocalIPPort[16:], port)
-
- copy(request.RemoteIPPort[:], remote.IP.To16())
- binary.LittleEndian.PutUint32(port, uint32(remote.Port))
- copy(request.RemoteIPPort[16:], port)
-
- return &request, nil
- }
|