move gotd fork into repo. (#111)
- update to latest telegram layer - remove some references to fields in tg.Entities that don't exist in the schema - originally added here: https://github.com/beeper/td/commit/820929062a2ba0104397bc01235ab58a9cff780e - referenced here - https://github.com/mautrix/telegramgo/commit/124f0967ed195b5a380c9bd02e170ada9710dde3 - https://github.com/mautrix/telegramgo/commit/4205047aab2e0639217148b5d125bfaab668bd8e
This commit is contained in:
@@ -0,0 +1,135 @@
|
||||
package tgtest
|
||||
|
||||
import (
|
||||
"context"
|
||||
"crypto/rsa"
|
||||
"io"
|
||||
"net"
|
||||
"time"
|
||||
|
||||
"github.com/go-faster/errors"
|
||||
"go.uber.org/zap"
|
||||
"nhooyr.io/websocket"
|
||||
|
||||
"go.mau.fi/mautrix-telegram/pkg/gotd/clock"
|
||||
"go.mau.fi/mautrix-telegram/pkg/gotd/crypto"
|
||||
"go.mau.fi/mautrix-telegram/pkg/gotd/exchange"
|
||||
"go.mau.fi/mautrix-telegram/pkg/gotd/mtproto"
|
||||
"go.mau.fi/mautrix-telegram/pkg/gotd/tdsync"
|
||||
"go.mau.fi/mautrix-telegram/pkg/gotd/tmap"
|
||||
"go.mau.fi/mautrix-telegram/pkg/gotd/transport"
|
||||
)
|
||||
|
||||
// Server is a MTProto server structure.
|
||||
type Server struct {
|
||||
// DC ID of this server.
|
||||
dcID int
|
||||
// Key pair of this server.
|
||||
key exchange.PrivateKey // immutable
|
||||
|
||||
// Codec constructor. May be nil.
|
||||
codec func() transport.Codec // immutable,nilable
|
||||
// Server-side message cipher.
|
||||
cipher crypto.Cipher // immutable
|
||||
// Clock to use in key exchange and message ID generation.
|
||||
clock clock.Clock // immutable
|
||||
// MessageID generator
|
||||
msgID mtproto.MessageIDSource // immutable
|
||||
|
||||
readTimeout time.Duration
|
||||
writeTimeout time.Duration
|
||||
|
||||
// RPC handler.
|
||||
handler Handler // immutable
|
||||
|
||||
// users stores session info.
|
||||
users *users
|
||||
|
||||
// type map for logging.
|
||||
types *tmap.Map // immutable
|
||||
log *zap.Logger // immutable
|
||||
}
|
||||
|
||||
// NewPrivateKey creates new private key from RSA private key.
|
||||
func NewPrivateKey(k *rsa.PrivateKey) exchange.PrivateKey {
|
||||
return exchange.PrivateKey{
|
||||
RSA: k,
|
||||
}
|
||||
}
|
||||
|
||||
// NewServer creates new Server.
|
||||
func NewServer(key exchange.PrivateKey, handler Handler, opts ServerOptions) *Server {
|
||||
opts.setDefaults()
|
||||
|
||||
s := &Server{
|
||||
dcID: opts.DC,
|
||||
key: key,
|
||||
codec: opts.Codec,
|
||||
cipher: crypto.NewServerCipher(opts.Random),
|
||||
clock: opts.Clock,
|
||||
msgID: opts.MessageID,
|
||||
readTimeout: opts.ReadTimeout,
|
||||
writeTimeout: opts.WriteTimeout,
|
||||
handler: handler,
|
||||
users: newUsers(),
|
||||
types: opts.Types,
|
||||
log: opts.Logger,
|
||||
}
|
||||
return s
|
||||
}
|
||||
|
||||
// Key returns public key of this server.
|
||||
func (s *Server) Key() exchange.PublicKey {
|
||||
return s.key.Public()
|
||||
}
|
||||
|
||||
// Serve runs server loop using given listener.
|
||||
func (s *Server) Serve(ctx context.Context, l transport.Listener) error {
|
||||
return s.serve(ctx, l)
|
||||
}
|
||||
|
||||
func (s *Server) serve(ctx context.Context, l transport.Listener) error {
|
||||
s.log.Info("Serving")
|
||||
defer func() {
|
||||
s.log.Info("Stopping")
|
||||
}()
|
||||
|
||||
grp := tdsync.NewCancellableGroup(ctx)
|
||||
grp.Go(func(context.Context) error {
|
||||
for {
|
||||
conn, err := l.Accept()
|
||||
if err != nil {
|
||||
if errors.Is(err, net.ErrClosed) {
|
||||
return nil
|
||||
}
|
||||
return errors.Wrap(err, "accept")
|
||||
}
|
||||
|
||||
grp.Go(func(ctx context.Context) error {
|
||||
if err := s.serveConn(ctx, conn); err != nil {
|
||||
// Client disconnected.
|
||||
var syscallErr *net.OpError
|
||||
switch {
|
||||
case errors.Is(err, io.EOF):
|
||||
return nil
|
||||
case errors.As(err, &syscallErr) &&
|
||||
(syscallErr.Op == "write" || syscallErr.Op == "read"):
|
||||
return nil
|
||||
}
|
||||
// TODO(tdakkota): emulate errors too?
|
||||
if code := websocket.CloseStatus(err); code >= 0 {
|
||||
return nil
|
||||
}
|
||||
|
||||
s.log.Info("Serving handler error", zap.Error(err))
|
||||
}
|
||||
return nil
|
||||
})
|
||||
}
|
||||
})
|
||||
grp.Go(func(ctx context.Context) error {
|
||||
<-ctx.Done()
|
||||
return l.Close()
|
||||
})
|
||||
return grp.Wait()
|
||||
}
|
||||
Reference in New Issue
Block a user