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
Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

streamcipher.go 1.6KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465
  1. package wrappers
  2. import (
  3. "crypto/cipher"
  4. "net"
  5. "github.com/juju/errors"
  6. "go.uber.org/zap"
  7. )
  8. // StreamCipher is a wrapper which encrypts/decrypts stream with AES-CTR
  9. // (as a part of obfuscated2 protocol).
  10. type StreamCipher struct {
  11. encryptor cipher.Stream
  12. decryptor cipher.Stream
  13. conn StreamReadWriteCloser
  14. logger *zap.SugaredLogger
  15. }
  16. func (s *StreamCipher) Read(p []byte) (int, error) {
  17. n, err := s.conn.Read(p)
  18. if err != nil {
  19. return 0, errors.Annotate(err, "Cannot read stream ciphered data")
  20. }
  21. s.decryptor.XORKeyStream(p, p[:n])
  22. return n, nil
  23. }
  24. func (s *StreamCipher) Write(p []byte) (int, error) {
  25. encrypted := make([]byte, len(p))
  26. s.encryptor.XORKeyStream(encrypted, p)
  27. return s.conn.Write(encrypted)
  28. }
  29. // Logger returns an instance of the logger for this wrapper.
  30. func (s *StreamCipher) Logger() *zap.SugaredLogger {
  31. return s.logger
  32. }
  33. // LocalAddr returns local address of the underlying net.Conn.
  34. func (s *StreamCipher) LocalAddr() *net.TCPAddr {
  35. return s.conn.LocalAddr()
  36. }
  37. // RemoteAddr returns remote address of the underlying net.Conn.
  38. func (s *StreamCipher) RemoteAddr() *net.TCPAddr {
  39. return s.conn.RemoteAddr()
  40. }
  41. // Close closes underlying net.Conn instance.
  42. func (s *StreamCipher) Close() error {
  43. return s.conn.Close()
  44. }
  45. // NewStreamCipher creates new stream cipher wrapper.
  46. func NewStreamCipher(conn StreamReadWriteCloser, encryptor, decryptor cipher.Stream) StreamReadWriteCloser {
  47. return &StreamCipher{
  48. conn: conn,
  49. logger: conn.Logger().Named("stream-cipher"),
  50. encryptor: encryptor,
  51. decryptor: decryptor,
  52. }
  53. }