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.go 4.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. package cli
  2. import (
  3. "fmt"
  4. "net"
  5. "os"
  6. "github.com/9seconds/mtg/v2/antireplay"
  7. "github.com/9seconds/mtg/v2/events"
  8. "github.com/9seconds/mtg/v2/internal/utils"
  9. "github.com/9seconds/mtg/v2/ipblocklist"
  10. "github.com/9seconds/mtg/v2/logger"
  11. "github.com/9seconds/mtg/v2/mtglib"
  12. "github.com/9seconds/mtg/v2/stats"
  13. "github.com/rs/zerolog"
  14. )
  15. type Proxy struct {
  16. base
  17. }
  18. func (c *Proxy) Run(cli *CLI, version string) error {
  19. if err := c.ReadConfig(version); err != nil {
  20. return fmt.Errorf("cannot init config: %w", err)
  21. }
  22. return c.Execute()
  23. }
  24. func (c *Proxy) Execute() error {
  25. zerolog.TimeFieldFormat = zerolog.TimeFormatUnixMs
  26. zerolog.TimestampFieldName = "timestamp"
  27. zerolog.LevelFieldName = "level"
  28. if c.Config.Debug {
  29. zerolog.SetGlobalLevel(zerolog.DebugLevel)
  30. } else {
  31. zerolog.SetGlobalLevel(zerolog.WarnLevel)
  32. }
  33. ctx := utils.RootContext()
  34. opts := mtglib.ProxyOpts{
  35. Logger: logger.NewZeroLogger(zerolog.New(os.Stdout).With().Timestamp().Logger()),
  36. Network: c.Network,
  37. AntiReplayCache: antireplay.NewNoop(),
  38. IPBlocklist: ipblocklist.NewNoop(),
  39. EventStream: events.NewNoopStream(),
  40. Secret: c.Config.Secret,
  41. BufferSize: c.Config.TCPBuffer.Value(mtglib.DefaultBufferSize),
  42. DomainFrontingPort: c.Config.DomainFrontingPort.Value(mtglib.DefaultDomainFrontingPort),
  43. IdleTimeout: c.Config.Network.Timeout.Idle.Value(mtglib.DefaultIdleTimeout),
  44. PreferIP: c.Config.PreferIP.Value(mtglib.DefaultPreferIP),
  45. }
  46. opts.Logger.BindStr("configuration", c.Config.String()).Debug("configuration")
  47. c.setupAntiReplayCache(&opts)
  48. if err := c.setupIPBlocklist(&opts); err != nil {
  49. return fmt.Errorf("cannot setup ipblocklist: %w", err)
  50. }
  51. if err := c.setupEventStream(&opts); err != nil {
  52. return fmt.Errorf("cannot setup event stream: %w", err)
  53. }
  54. proxy, err := mtglib.NewProxy(opts)
  55. if err != nil {
  56. return fmt.Errorf("cannot create a proxy: %w", err)
  57. }
  58. listener, err := net.Listen("tcp", c.Config.BindTo.String())
  59. if err != nil {
  60. return fmt.Errorf("cannot start proxy: %w", err)
  61. }
  62. go proxy.Serve(listener) // nolint: errcheck
  63. <-ctx.Done()
  64. listener.Close()
  65. proxy.Shutdown()
  66. return nil
  67. }
  68. func (c *Proxy) setupAntiReplayCache(opts *mtglib.ProxyOpts) {
  69. if !c.Config.Defense.AntiReplay.Enabled {
  70. return
  71. }
  72. opts.AntiReplayCache = antireplay.NewStableBloomFilter(
  73. c.Config.Defense.AntiReplay.MaxSize.Value(antireplay.DefaultStableBloomFilterMaxSize),
  74. c.Config.Defense.AntiReplay.ErrorRate.Value(antireplay.DefaultStableBloomFilterErrorRate),
  75. )
  76. }
  77. func (c *Proxy) setupIPBlocklist(opts *mtglib.ProxyOpts) error {
  78. if !c.Config.Defense.Blocklist.Enabled {
  79. return nil
  80. }
  81. remoteURLs := []string{}
  82. localFiles := []string{}
  83. for _, v := range c.Config.Defense.Blocklist.URLs {
  84. if v.IsRemote() {
  85. remoteURLs = append(remoteURLs, v.String())
  86. } else {
  87. localFiles = append(localFiles, v.String())
  88. }
  89. }
  90. firehol, err := ipblocklist.NewFirehol(opts.Logger.Named("ipblockist"),
  91. c.Network,
  92. c.Config.Defense.Blocklist.DownloadConcurrency,
  93. remoteURLs,
  94. localFiles)
  95. if err != nil {
  96. return err // nolint: wrapcheck
  97. }
  98. go firehol.Run(c.Config.Defense.Blocklist.UpdateEach.Value(ipblocklist.DefaultFireholUpdateEach))
  99. opts.IPBlocklist = firehol
  100. return nil
  101. }
  102. func (c *Proxy) setupEventStream(opts *mtglib.ProxyOpts) error {
  103. factories := make([]events.ObserverFactory, 0, 2)
  104. if c.Config.Stats.StatsD.Enabled {
  105. statsdFactory, err := stats.NewStatsd(
  106. c.Config.Stats.StatsD.Address.String(),
  107. opts.Logger.Named("statsd"),
  108. c.Config.Stats.StatsD.MetricPrefix.Value(stats.DefaultStatsdMetricPrefix),
  109. c.Config.Stats.StatsD.TagFormat.Value(stats.DefaultStatsdTagFormat))
  110. if err != nil {
  111. return fmt.Errorf("cannot build statsd observer: %w", err)
  112. }
  113. factories = append(factories, statsdFactory.Make)
  114. }
  115. if c.Config.Stats.Prometheus.Enabled {
  116. prometheus := stats.NewPrometheus(
  117. c.Config.Stats.Prometheus.MetricPrefix.Value(stats.DefaultMetricPrefix),
  118. c.Config.Stats.Prometheus.HTTPPath.Value("/"),
  119. )
  120. listener, err := net.Listen("tcp", c.Config.Stats.Prometheus.BindTo.String())
  121. if err != nil {
  122. return fmt.Errorf("cannot start a listener for prometheus: %w", err)
  123. }
  124. go prometheus.Serve(listener) // nolint: errcheck
  125. factories = append(factories, prometheus.Make)
  126. }
  127. if len(factories) > 0 {
  128. opts.EventStream = events.NewEventStream(factories)
  129. }
  130. return nil
  131. }