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.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. package wrappers
  2. import (
  3. "bytes"
  4. "crypto/aes"
  5. "crypto/cipher"
  6. "io"
  7. "github.com/juju/errors"
  8. )
  9. type BlockCipherReadWriteCloser struct {
  10. buf *bytes.Buffer
  11. conn io.ReadWriteCloser
  12. encryptor cipher.BlockMode
  13. decryptor cipher.BlockMode
  14. }
  15. func (c *BlockCipherReadWriteCloser) 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. return c.flush(p)
  27. }
  28. func (c *BlockCipherReadWriteCloser) Write(p []byte) (int, error) {
  29. if len(p)%aes.BlockSize > 0 {
  30. return 0, errors.Errorf("Incorrect block size %d", len(p))
  31. }
  32. buf := getBuffer()
  33. defer putBuffer(buf)
  34. buf.Grow(len(p))
  35. buf.Write(p)
  36. encrypted := buf.Bytes()
  37. c.encryptor.CryptBlocks(encrypted, p)
  38. return c.conn.Write(encrypted)
  39. }
  40. func (c *BlockCipherReadWriteCloser) Close() error {
  41. defer putBuffer(c.buf)
  42. return c.conn.Close()
  43. }
  44. func (c *BlockCipherReadWriteCloser) flush(p []byte) (int, error) {
  45. sizeToRead := len(p)
  46. if c.buf.Len() < sizeToRead {
  47. sizeToRead = c.buf.Len()
  48. }
  49. sizeToRead = aes.BlockSize * (sizeToRead / aes.BlockSize)
  50. c.decryptor.CryptBlocks(p, c.buf.Bytes()[:sizeToRead])
  51. if sizeToRead == c.buf.Len() {
  52. c.buf.Reset()
  53. } else {
  54. leftover := c.buf.Bytes()[sizeToRead:]
  55. putBuffer(c.buf)
  56. c.buf = getBuffer()
  57. c.buf.Write(leftover)
  58. }
  59. return sizeToRead, nil
  60. }
  61. func NewBlockCipherRWC(conn io.ReadWriteCloser, encryptor, decryptor cipher.BlockMode) io.ReadWriteCloser {
  62. return &BlockCipherReadWriteCloser{
  63. buf: getBuffer(),
  64. conn: conn,
  65. encryptor: encryptor,
  66. decryptor: decryptor,
  67. }
  68. }