package proxy import ( "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/telegram" "github.com/9seconds/mtg/wrappers" ) type Proxy struct { clientInit client.Init tg telegram.Telegram conf *config.Config } 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) defer func() { conn.Close() if err := recover(); err != nil { log.Errorw("Crash of accept handler", "error", err) } }() log.Infow("Client connected", "addr", conn.RemoteAddr()) client, opts, err := p.clientInit(conn, connID, p.conf) if err != nil { log.Errorw("Cannot initialize client connection", "error", err) return } defer client.(io.Closer).Close() server, err := p.getTelegramConn(opts, connID) if err != nil { log.Errorw("Cannot initialize server connection", "error", err) return } defer server.(io.Closer).Close() wait := &sync.WaitGroup{} wait.Add(2) if p.conf.UseMiddleProxy() { clientPacket := client.(wrappers.PacketReadWriteCloser) serverPacket := server.(wrappers.PacketReadWriteCloser) go p.middlePipe(clientPacket, serverPacket, wait, &opts.ReadHacks) go p.middlePipe(serverPacket, clientPacket, wait, &opts.WriteHacks) } else { clientStream := client.(wrappers.StreamReadWriteCloser) serverStream := server.(wrappers.StreamReadWriteCloser) go p.directPipe(clientStream, serverStream, wait) go p.directPipe(serverStream, clientStream, wait) } wait.Wait() log.Infow("Client disconnected", "addr", conn.RemoteAddr()) } func (p *Proxy) getTelegramConn(opts *mtproto.ConnectionOpts, connID string) (wrappers.Wrap, error) { streamConn, err := p.tg.Dial(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 wrappers.PacketWriteCloser, wait *sync.WaitGroup, hacks *mtproto.Hacks) { defer func() { src.Close() dst.Close() 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 wrappers.StreamWriteCloser, wait *sync.WaitGroup) { defer func() { src.Close() dst.Close() wait.Done() }() if _, err := io.Copy(dst, src); err != nil { src.Logger().Warnw("Cannot pump sockets", "error", err) } } 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, } }