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
Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

streamcipher.go 1.7KB

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