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 символов.

blockcipherrwc.go 1.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. package wrappers
  2. import (
  3. "bytes"
  4. "crypto/aes"
  5. "crypto/cipher"
  6. "net"
  7. "github.com/juju/errors"
  8. )
  9. type BlockCipherReadWriteCloserWithAddr struct {
  10. buf *bytes.Buffer
  11. conn ReadWriteCloserWithAddr
  12. encryptor cipher.BlockMode
  13. decryptor cipher.BlockMode
  14. }
  15. func (c *BlockCipherReadWriteCloserWithAddr) Read(p []byte) (int, error) {
  16. if c.buf.Len() > 0 {
  17. return c.flush(p)
  18. }
  19. for c.buf.Len() == 0 || c.buf.Len()%aes.BlockSize != 0 {
  20. n, err := c.conn.Read(p)
  21. if err != nil {
  22. return 0, errors.Annotate(err, "Cannot read from socket")
  23. }
  24. c.buf.Write(p[:n])
  25. }
  26. c.decryptor.CryptBlocks(c.buf.Bytes(), c.buf.Bytes())
  27. return c.flush(p)
  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. buf := getBuffer()
  34. defer putBuffer(buf)
  35. buf.Grow(len(p))
  36. buf.Write(p)
  37. encrypted := buf.Bytes()
  38. c.encryptor.CryptBlocks(encrypted, p)
  39. return c.conn.Write(encrypted)
  40. }
  41. func (c *BlockCipherReadWriteCloserWithAddr) Close() error {
  42. defer putBuffer(c.buf)
  43. return c.conn.Close()
  44. }
  45. func (c *BlockCipherReadWriteCloserWithAddr) Addr() *net.TCPAddr {
  46. return c.conn.Addr()
  47. }
  48. func (c *BlockCipherReadWriteCloserWithAddr) flush(p []byte) (int, error) {
  49. sizeToRead := len(p)
  50. if c.buf.Len() < sizeToRead {
  51. sizeToRead = c.buf.Len()
  52. }
  53. data := c.buf.Bytes()
  54. copy(p, data[:sizeToRead])
  55. if sizeToRead == c.buf.Len() {
  56. c.buf.Reset()
  57. } else {
  58. newBuf := getBuffer()
  59. newBuf.Write(data[sizeToRead:])
  60. putBuffer(c.buf)
  61. c.buf = newBuf
  62. }
  63. return sizeToRead, nil
  64. }
  65. func NewBlockCipherRWC(conn ReadWriteCloserWithAddr, encryptor, decryptor cipher.BlockMode) ReadWriteCloserWithAddr {
  66. return &BlockCipherReadWriteCloserWithAddr{
  67. buf: getBuffer(),
  68. conn: conn,
  69. encryptor: encryptor,
  70. decryptor: decryptor,
  71. }
  72. }