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
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

blockcipherrwc.go 1.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. package wrappers
  2. import (
  3. "crypto/aes"
  4. "crypto/cipher"
  5. "net"
  6. "github.com/juju/errors"
  7. )
  8. type BlockCipherReadWriteCloserWithAddr struct {
  9. BufferedReader
  10. conn ReadWriteCloserWithAddr
  11. encryptor cipher.BlockMode
  12. decryptor cipher.BlockMode
  13. }
  14. func (c *BlockCipherReadWriteCloserWithAddr) Read(p []byte) (int, error) {
  15. return c.BufferedRead(p, func() error {
  16. bufferLength := c.Buffer.Len()
  17. for bufferLength%aes.BlockSize != 0 || bufferLength == 0 {
  18. n, err := c.conn.Read(p)
  19. if err != nil {
  20. return errors.Annotate(err, "Cannot read from socket")
  21. }
  22. c.Buffer.Write(p[:n])
  23. bufferLength = c.Buffer.Len()
  24. }
  25. c.decryptor.CryptBlocks(c.Buffer.Bytes(), c.Buffer.Bytes())
  26. return nil
  27. })
  28. }
  29. func (c *BlockCipherReadWriteCloserWithAddr) Write(p []byte) (int, error) {
  30. if len(p)%aes.BlockSize > 0 {
  31. return 0, errors.Errorf("Incorrect block size %d", len(p))
  32. }
  33. encrypted := make([]byte, len(p))
  34. c.encryptor.CryptBlocks(encrypted, p)
  35. return c.conn.Write(encrypted)
  36. }
  37. func (c *BlockCipherReadWriteCloserWithAddr) Close() error {
  38. return c.conn.Close()
  39. }
  40. func (c *BlockCipherReadWriteCloserWithAddr) LocalAddr() *net.TCPAddr {
  41. return c.conn.LocalAddr()
  42. }
  43. func (c *BlockCipherReadWriteCloserWithAddr) RemoteAddr() *net.TCPAddr {
  44. return c.conn.RemoteAddr()
  45. }
  46. func NewBlockCipherRWC(conn ReadWriteCloserWithAddr, encryptor, decryptor cipher.BlockMode) ReadWriteCloserWithAddr {
  47. return &BlockCipherReadWriteCloserWithAddr{
  48. BufferedReader: NewBufferedReader(),
  49. conn: conn,
  50. encryptor: encryptor,
  51. decryptor: decryptor,
  52. }
  53. }