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
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

proxy.go 4.7KB

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