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 символов.

proxy_request.go 2.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. package wrappers
  2. import (
  3. "bytes"
  4. "io"
  5. "io/ioutil"
  6. "net"
  7. "github.com/juju/errors"
  8. "github.com/9seconds/mtg/mtproto"
  9. "github.com/9seconds/mtg/mtproto/rpc"
  10. "github.com/9seconds/mtg/utils"
  11. "github.com/9seconds/mtg/wrappers"
  12. )
  13. type ProxyRequestReadWriteCloserWithAddr struct {
  14. wrappers.BufferedReader
  15. conn wrappers.ReadWriteCloserWithAddr
  16. req *rpc.ProxyRequest
  17. }
  18. func (p *ProxyRequestReadWriteCloserWithAddr) Read(buf []byte) (int, error) {
  19. return p.BufferedRead(buf, func() error {
  20. ans := make([]byte, 4)
  21. if _, err := io.ReadFull(p.conn, ans); err != nil {
  22. return errors.Annotate(err, "Cannot read RPC tag")
  23. }
  24. switch {
  25. case bytes.Equal(ans, rpc.TagProxyAns):
  26. return p.readProxyAns()
  27. case bytes.Equal(ans, rpc.TagSimpleAck):
  28. return p.readSimpleAck()
  29. case bytes.Equal(ans, rpc.TagCloseExt):
  30. return p.readCloseExt()
  31. }
  32. return errors.Errorf("Unknown RPC answer %v", ans)
  33. })
  34. }
  35. func (p *ProxyRequestReadWriteCloserWithAddr) readCloseExt() error {
  36. return errors.New("Connection has been closed remotely")
  37. }
  38. func (p *ProxyRequestReadWriteCloserWithAddr) readProxyAns() (err error) {
  39. if _, err = io.CopyN(ioutil.Discard, p.conn, 8+4); err != nil {
  40. return errors.Annotate(err, "Cannot skip flags and connid")
  41. }
  42. buf, err := utils.ReadCurrentData(p.conn)
  43. if err != nil {
  44. return errors.Annotate(err, "Cannot read proxy answer")
  45. }
  46. p.Buffer.Write(buf)
  47. return nil
  48. }
  49. func (p *ProxyRequestReadWriteCloserWithAddr) readSimpleAck() error {
  50. if _, err := io.CopyN(ioutil.Discard, p.conn, 8); err != nil {
  51. return errors.Annotate(err, "Cannot skip connid")
  52. }
  53. ackData := make([]byte, 4)
  54. if _, err := io.ReadFull(p.conn, ackData); err != nil {
  55. return errors.Annotate(err, "Cannot read simple ack")
  56. }
  57. p.Buffer.Write(ackData)
  58. return nil
  59. }
  60. func (p *ProxyRequestReadWriteCloserWithAddr) Write(raw []byte) (int, error) {
  61. if _, err := p.conn.Write(p.req.Bytes(raw)); err != nil {
  62. return 0, err
  63. }
  64. return len(raw), nil
  65. }
  66. func (p *ProxyRequestReadWriteCloserWithAddr) Close() error {
  67. return p.conn.Close()
  68. }
  69. func (p *ProxyRequestReadWriteCloserWithAddr) LocalAddr() *net.TCPAddr {
  70. return p.conn.LocalAddr()
  71. }
  72. func (p *ProxyRequestReadWriteCloserWithAddr) RemoteAddr() *net.TCPAddr {
  73. return p.conn.RemoteAddr()
  74. }
  75. func (p *ProxyRequestReadWriteCloserWithAddr) SocketID() string {
  76. return p.conn.SocketID()
  77. }
  78. func NewProxyRequestRWC(conn wrappers.ReadWriteCloserWithAddr, connOpts *mtproto.ConnectionOpts, adTag []byte) (wrappers.ReadWriteCloserWithAddr, error) {
  79. req, err := rpc.NewProxyRequest(connOpts.ClientAddr, conn.LocalAddr(), connOpts, adTag)
  80. if err != nil {
  81. return nil, errors.Annotate(err, "Cannot create new RPC proxy request")
  82. }
  83. return &ProxyRequestReadWriteCloserWithAddr{
  84. BufferedReader: wrappers.NewBufferedReader(),
  85. conn: conn,
  86. req: req,
  87. }, nil
  88. }