Highly-opinionated (ex-bullshit-free) MTPROTO proxy for Telegram. If you use v1.0 or upgrade broke you proxy, please read the chapter Version 2
Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

mtproto_proxy.go 3.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. package wrappers
  2. import (
  3. "bytes"
  4. "net"
  5. "github.com/juju/errors"
  6. "github.com/9seconds/mtg/mtproto"
  7. "github.com/9seconds/mtg/mtproto/rpc"
  8. )
  9. type MTProtoProxy struct {
  10. conn WrapPacketReadWriteCloser
  11. req *rpc.ProxyRequest
  12. readCounter uint32
  13. writeCounter uint32
  14. }
  15. func (m *MTProtoProxy) Read() ([]byte, error) {
  16. m.LogDebug("Read packet",
  17. "counter", m.readCounter,
  18. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  19. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  20. )
  21. packet, err := m.conn.Read()
  22. if err != nil {
  23. return nil, errors.Annotate(err, "Cannot read packet")
  24. }
  25. m.LogDebug("Read packet length",
  26. "counter", m.readCounter,
  27. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  28. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  29. "length", len(packet),
  30. )
  31. if len(packet) < 4 {
  32. return nil, errors.Annotate(err, "Incorrect packet length")
  33. }
  34. tag, packet := packet[:4], packet[4:]
  35. m.LogDebug("Read RPC tag",
  36. "counter", m.readCounter,
  37. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  38. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  39. "tag", tag,
  40. )
  41. m.readCounter++
  42. switch {
  43. case bytes.Equal(tag, rpc.TagProxyAns):
  44. return m.readProxyAns(packet)
  45. case bytes.Equal(tag, rpc.TagSimpleAck):
  46. return m.readSimpleAck(packet)
  47. case bytes.Equal(tag, rpc.TagCloseExt):
  48. return m.readCloseExt(packet)
  49. }
  50. return nil, errors.Errorf("Unknown RPC answer %v", tag)
  51. }
  52. func (m *MTProtoProxy) readProxyAns(data []byte) ([]byte, error) {
  53. if len(data) < 12 {
  54. return nil, errors.Errorf("Incorrect data of proxy answer: %d", len(data))
  55. }
  56. return data[12:], nil
  57. }
  58. func (m *MTProtoProxy) readSimpleAck(data []byte) ([]byte, error) {
  59. if len(data) != 12 {
  60. return nil, errors.Errorf("Incorrect data of simple ack: %d", len(data))
  61. }
  62. return data[8:12], nil // 0:8 - connection id
  63. }
  64. func (m *MTProtoProxy) readCloseExt(data []byte) ([]byte, error) {
  65. return nil, errors.New("Connection has been closed remotely by RPC call")
  66. }
  67. func (m *MTProtoProxy) Write(p []byte) (int, error) {
  68. m.LogDebug("Write packet",
  69. "length", len(p),
  70. "counter", m.writeCounter,
  71. "simple_ack", m.req.Options.ReadHacks.SimpleAck,
  72. "quick_ack", m.req.Options.ReadHacks.QuickAck,
  73. )
  74. m.writeCounter++
  75. if _, err := m.conn.Write(p); err != nil {
  76. return 0, err
  77. }
  78. return len(p), nil
  79. }
  80. func (m *MTProtoProxy) LogDebug(msg string, data ...interface{}) {
  81. data = append(data, []interface{}{"type", "proxy"})
  82. m.conn.LogDebug(msg, data...)
  83. }
  84. func (m *MTProtoProxy) LogInfo(msg string, data ...interface{}) {
  85. data = append(data, []interface{}{"type", "proxy"})
  86. m.conn.LogInfo(msg, data...)
  87. }
  88. func (m *MTProtoProxy) LogWarn(msg string, data ...interface{}) {
  89. data = append(data, []interface{}{"type", "proxy"})
  90. m.conn.LogWarn(msg, data...)
  91. }
  92. func (m *MTProtoProxy) LogError(msg string, data ...interface{}) {
  93. data = append(data, []interface{}{"type", "proxy"})
  94. m.conn.LogError(msg, data...)
  95. }
  96. func (m *MTProtoProxy) LocalAddr() *net.TCPAddr {
  97. return m.conn.LocalAddr()
  98. }
  99. func (m *MTProtoProxy) RemoteAddr() *net.TCPAddr {
  100. return m.conn.RemoteAddr()
  101. }
  102. func (m *MTProtoProxy) Close() error {
  103. return m.conn.Close()
  104. }
  105. func NewMTProtoProxy(conn WrapPacketReadWriteCloser, connOpts *mtproto.ConnectionOpts, adTag []byte) (WrapPacketReadWriteCloser, error) {
  106. req, err := rpc.NewProxyRequest(connOpts.ClientAddr, conn.LocalAddr(), connOpts, adTag)
  107. if err != nil {
  108. return nil, errors.Annotate(err, "Cannot create new RPC proxy request")
  109. }
  110. return &MTProtoProxy{
  111. conn: conn,
  112. req: req,
  113. }, nil
  114. }