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.

relay.go 1.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. package relay
  2. import (
  3. "context"
  4. "io"
  5. "sync"
  6. "time"
  7. )
  8. type Relay struct {
  9. ctx context.Context
  10. ctxCancel context.CancelFunc
  11. logger Logger
  12. eastBuffer []byte
  13. westBuffer []byte
  14. tickChannel chan struct{}
  15. errorChannel chan error
  16. tickTimeout time.Duration
  17. }
  18. func (r *Relay) Process(eastConn, westConn io.ReadWriteCloser) error {
  19. eastConn = conn{
  20. ReadWriteCloser: eastConn,
  21. relay: r,
  22. }
  23. westConn = conn{
  24. ReadWriteCloser: westConn,
  25. relay: r,
  26. }
  27. defer func() {
  28. r.ctxCancel()
  29. eastConn.Close()
  30. westConn.Close()
  31. }()
  32. go r.runObserver()
  33. wg := &sync.WaitGroup{}
  34. wg.Add(2) // nolint: gomnd
  35. go r.transmit(eastConn, westConn, r.westBuffer, "west", wg)
  36. r.transmit(westConn, eastConn, r.eastBuffer, "east", wg)
  37. wg.Wait()
  38. return <-r.errorChannel
  39. }
  40. func (r *Relay) transmit(src io.ReadCloser, dst io.WriteCloser,
  41. buffer []byte, direction string, wg *sync.WaitGroup) {
  42. defer func() {
  43. wg.Done()
  44. src.Close()
  45. dst.Close()
  46. }()
  47. if _, err := io.CopyBuffer(dst, src, buffer); err != nil {
  48. r.logger.Printf("error '%v' happened on direction %s", err, direction)
  49. select {
  50. case <-r.ctx.Done():
  51. case r.errorChannel <- err:
  52. default:
  53. }
  54. }
  55. }
  56. func (r *Relay) runObserver() {
  57. ticker := time.NewTicker(time.Second)
  58. defer func() {
  59. ticker.Stop()
  60. select {
  61. case <-ticker.C:
  62. default:
  63. }
  64. }()
  65. lastTickAt := time.Now()
  66. ctx := r.ctx
  67. for {
  68. select {
  69. case <-ctx.Done():
  70. return
  71. case <-r.tickChannel:
  72. lastTickAt = time.Now()
  73. case <-ticker.C:
  74. if time.Since(lastTickAt) > r.tickTimeout {
  75. r.logger.Printf("exit due to a timeout")
  76. r.ctxCancel()
  77. return
  78. }
  79. }
  80. }
  81. }