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

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586
  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. c.decryptor.CryptBlocks(c.buf.Bytes(), c.buf.Bytes())
  27. return c.flush(p)
  28. }
  29. func (c *BlockCipherReadWriteCloser) 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 *BlockCipherReadWriteCloser) Close() error {
  42. defer putBuffer(c.buf)
  43. return c.conn.Close()
  44. }
  45. func (c *BlockCipherReadWriteCloser) flush(p []byte) (int, error) {
  46. sizeToRead := len(p)
  47. if c.buf.Len() < sizeToRead {
  48. sizeToRead = c.buf.Len()
  49. }
  50. data := c.buf.Bytes()
  51. copy(p, data[:sizeToRead])
  52. if sizeToRead == c.buf.Len() {
  53. c.buf.Reset()
  54. } else {
  55. newBuf := getBuffer()
  56. newBuf.Write(data[sizeToRead:])
  57. putBuffer(c.buf)
  58. c.buf = newBuf
  59. }
  60. return sizeToRead, nil
  61. }
  62. func NewBlockCipherRWC(conn io.ReadWriteCloser, encryptor, decryptor cipher.BlockMode) io.ReadWriteCloser {
  63. return &BlockCipherReadWriteCloser{
  64. buf: getBuffer(),
  65. conn: conn,
  66. encryptor: encryptor,
  67. decryptor: decryptor,
  68. }
  69. }