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
Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

blockcipher.go 2.0KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. package stream
  2. import (
  3. "crypto/aes"
  4. "crypto/cipher"
  5. "fmt"
  6. "net"
  7. "time"
  8. "go.uber.org/zap"
  9. "github.com/9seconds/mtg/conntypes"
  10. "github.com/9seconds/mtg/utils"
  11. )
  12. type wrapperBlockCipher struct {
  13. bufferedReader
  14. parent conntypes.StreamReadWriteCloser
  15. encryptor cipher.BlockMode
  16. decryptor cipher.BlockMode
  17. }
  18. func (w *wrapperBlockCipher) Write(p []byte) (int, error) {
  19. encrypted, err := w.encrypt(p)
  20. if err != nil {
  21. return 0, err
  22. }
  23. return w.parent.Write(encrypted)
  24. }
  25. func (w *wrapperBlockCipher) WriteTimeout(p []byte, timeout time.Duration) (int, error) {
  26. encrypted, err := w.encrypt(p)
  27. if err != nil {
  28. return 0, err
  29. }
  30. return w.parent.WriteTimeout(encrypted, timeout)
  31. }
  32. func (w *wrapperBlockCipher) encrypt(p []byte) ([]byte, error) {
  33. if len(p)%aes.BlockSize > 0 {
  34. return nil, fmt.Errorf("incorrect block size %d", len(p))
  35. }
  36. encrypted := make([]byte, len(p))
  37. w.encryptor.CryptBlocks(encrypted, p)
  38. return encrypted, nil
  39. }
  40. func (w *wrapperBlockCipher) Close() error {
  41. return w.parent.Close()
  42. }
  43. func (w *wrapperBlockCipher) Conn() net.Conn {
  44. return w.parent.Conn()
  45. }
  46. func (w *wrapperBlockCipher) Logger() *zap.SugaredLogger {
  47. return w.parent.Logger().Named("block-cipher")
  48. }
  49. func (w *wrapperBlockCipher) LocalAddr() *net.TCPAddr {
  50. return w.parent.LocalAddr()
  51. }
  52. func (w *wrapperBlockCipher) RemoteAddr() *net.TCPAddr {
  53. return w.parent.RemoteAddr()
  54. }
  55. func newBlockCipher(parent conntypes.StreamReadWriteCloser,
  56. encryptor, decryptor cipher.BlockMode) conntypes.StreamReadWriteCloser {
  57. cipher := &wrapperBlockCipher{
  58. parent: parent,
  59. encryptor: encryptor,
  60. decryptor: decryptor,
  61. }
  62. cipher.readFunc = func() ([]byte, error) {
  63. var currentBuffer []byte
  64. for len(currentBuffer) == 0 || len(currentBuffer)%aes.BlockSize != 0 {
  65. rv, err := utils.ReadFull(cipher.parent)
  66. if err != nil {
  67. return nil, fmt.Errorf("cannot read data: %w", err)
  68. }
  69. currentBuffer = append(currentBuffer, rv...)
  70. }
  71. cipher.decryptor.CryptBlocks(currentBuffer, currentBuffer)
  72. return currentBuffer, nil
  73. }
  74. return cipher
  75. }