| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175 |
- package proxy
-
- import (
- "context"
- "io"
- "net"
- "sync"
-
- "github.com/juju/errors"
- uuid "github.com/satori/go.uuid"
- "go.uber.org/zap"
-
- "github.com/9seconds/mtg/client"
- "github.com/9seconds/mtg/config"
- "github.com/9seconds/mtg/mtproto"
- "github.com/9seconds/mtg/stats"
- "github.com/9seconds/mtg/telegram"
- "github.com/9seconds/mtg/wrappers"
- )
-
- // Proxy is a core of this program.
- type Proxy struct {
- clientInit client.Init
- tg telegram.Telegram
- conf *config.Config
- }
-
- // Serve runs TCP proxy server.
- func (p *Proxy) Serve() error {
- lsock, err := net.Listen("tcp", p.conf.BindAddr())
- if err != nil {
- return errors.Annotate(err, "Cannot create listen socket")
- }
-
- for {
- if conn, err := lsock.Accept(); err != nil {
- zap.S().Errorw("Cannot allocate incoming connection", "error", err)
- } else {
- go p.accept(conn)
- }
- }
- }
-
- func (p *Proxy) accept(conn net.Conn) {
- connID := uuid.NewV4().String()
- log := zap.S().With("connection_id", connID).Named("main")
- ctx, cancel := context.WithCancel(context.Background())
-
- defer func() {
- cancel()
- conn.Close() // nolint: errcheck
-
- if err := recover(); err != nil {
- stats.NewCrash()
- log.Errorw("Crash of accept handler", "error", err)
- }
- }()
-
- log.Infow("Client connected", "addr", conn.RemoteAddr())
-
- clientConn, opts, err := p.clientInit(ctx, cancel, conn, connID, p.conf)
- if err != nil {
- log.Errorw("Cannot initialize client connection", "error", err)
- return
- }
- defer clientConn.(io.Closer).Close() // nolint: errcheck
-
- stats.ClientConnected(opts.ConnectionType, clientConn.RemoteAddr())
- defer stats.ClientDisconnected(opts.ConnectionType, clientConn.RemoteAddr())
-
- serverConn, err := p.getTelegramConn(ctx, cancel, opts, connID)
- if err != nil {
- log.Errorw("Cannot initialize server connection", "error", err)
- return
- }
- defer serverConn.(io.Closer).Close() // nolint: errcheck
-
- go func() {
- <-ctx.Done()
- serverConn.(io.Closer).Close()
- clientConn.(io.Closer).Close()
- }()
-
- wait := &sync.WaitGroup{}
- wait.Add(2)
-
- if p.conf.UseMiddleProxy() {
- clientPacket := clientConn.(wrappers.PacketReadWriteCloser)
- serverPacket := serverConn.(wrappers.PacketReadWriteCloser)
- go p.middlePipe(clientPacket, serverPacket, wait, &opts.ReadHacks)
- go p.middlePipe(serverPacket, clientPacket, wait, &opts.WriteHacks)
- } else {
- clientStream := clientConn.(wrappers.StreamReadWriteCloser)
- serverStream := serverConn.(wrappers.StreamReadWriteCloser)
- go p.directPipe(clientStream, serverStream, wait, p.conf.ReadBufferSize)
- go p.directPipe(serverStream, clientStream, wait, p.conf.WriteBufferSize)
- }
-
- wait.Wait()
-
- log.Infow("Client disconnected", "addr", conn.RemoteAddr())
- }
-
- func (p *Proxy) getTelegramConn(ctx context.Context, cancel context.CancelFunc,
- opts *mtproto.ConnectionOpts, connID string) (wrappers.Wrap, error) {
- streamConn, err := p.tg.Dial(ctx, cancel, connID, opts)
- if err != nil {
- return nil, errors.Annotate(err, "Cannot dial to Telegram")
- }
-
- packetConn, err := p.tg.Init(opts, streamConn)
- if err != nil {
- return nil, errors.Annotate(err, "Cannot handshake telegram")
- }
-
- return packetConn, nil
- }
-
- func (p *Proxy) middlePipe(src wrappers.PacketReadCloser, dst io.WriteCloser,
- wait *sync.WaitGroup, hacks *mtproto.Hacks) {
- defer func() {
- src.Close() // nolint: errcheck
- dst.Close() // nolint: errcheck
- wait.Done()
- }()
-
- for {
- hacks.SimpleAck = false
- hacks.QuickAck = false
-
- packet, err := src.Read()
- if err != nil {
- src.Logger().Warnw("Cannot read packet", "error", err)
- return
- }
- if _, err = dst.Write(packet); err != nil {
- src.Logger().Warnw("Cannot write packet", "error", err)
- return
- }
- }
- }
-
- func (p *Proxy) directPipe(src wrappers.StreamReadCloser, dst io.WriteCloser,
- wait *sync.WaitGroup, bufferSize int) {
- defer func() {
- src.Close() // nolint: errcheck
- dst.Close() // nolint: errcheck
- wait.Done()
- }()
-
- buffer := make([]byte, bufferSize)
- if _, err := io.CopyBuffer(dst, src, buffer); err != nil {
- src.Logger().Warnw("Cannot pump sockets", "error", err)
- }
- }
-
- // NewProxy returns new proxy instance.
- func NewProxy(conf *config.Config) *Proxy {
- var clientInit client.Init
- var tg telegram.Telegram
-
- if conf.UseMiddleProxy() {
- clientInit = client.MiddleInit
- tg = telegram.NewMiddleTelegram(conf)
- } else {
- clientInit = client.DirectInit
- tg = telegram.NewDirectTelegram(conf)
- }
-
- return &Proxy{
- conf: conf,
- clientInit: clientInit,
- tg: tg,
- }
- }
|