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
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.

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