ota.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. package shadowsocks
  2. import (
  3. "bytes"
  4. "crypto/hmac"
  5. "crypto/sha1"
  6. "io"
  7. "v2ray.com/core/common"
  8. "v2ray.com/core/common/buf"
  9. "v2ray.com/core/common/bytespool"
  10. "v2ray.com/core/common/serial"
  11. )
  12. const (
  13. // AuthSize is the number of extra bytes for Shadowsocks OTA.
  14. AuthSize = 10
  15. )
  16. type KeyGenerator func() []byte
  17. type Authenticator struct {
  18. key KeyGenerator
  19. }
  20. func NewAuthenticator(keygen KeyGenerator) *Authenticator {
  21. return &Authenticator{
  22. key: keygen,
  23. }
  24. }
  25. func (v *Authenticator) Authenticate(data []byte) buf.Supplier {
  26. hasher := hmac.New(sha1.New, v.key())
  27. common.Must2(hasher.Write(data))
  28. res := hasher.Sum(nil)
  29. return func(b []byte) (int, error) {
  30. return copy(b, res[:AuthSize]), nil
  31. }
  32. }
  33. func HeaderKeyGenerator(key []byte, iv []byte) func() []byte {
  34. return func() []byte {
  35. newKey := make([]byte, 0, len(key)+len(iv))
  36. newKey = append(newKey, iv...)
  37. newKey = append(newKey, key...)
  38. return newKey
  39. }
  40. }
  41. func ChunkKeyGenerator(iv []byte) func() []byte {
  42. chunkID := 0
  43. return func() []byte {
  44. newKey := make([]byte, 0, len(iv)+4)
  45. newKey = append(newKey, iv...)
  46. newKey = serial.IntToBytes(chunkID, newKey)
  47. chunkID++
  48. return newKey
  49. }
  50. }
  51. type ChunkReader struct {
  52. reader io.Reader
  53. auth *Authenticator
  54. }
  55. func NewChunkReader(reader io.Reader, auth *Authenticator) *ChunkReader {
  56. return &ChunkReader{
  57. reader: reader,
  58. auth: auth,
  59. }
  60. }
  61. func (v *ChunkReader) ReadMultiBuffer() (buf.MultiBuffer, error) {
  62. size, err := serial.ReadUint16(v.reader)
  63. if err != nil {
  64. return nil, newError("failed to read size")
  65. }
  66. size += AuthSize
  67. buffer := bytespool.Alloc(int32(size))
  68. defer bytespool.Free(buffer)
  69. if _, err := io.ReadFull(v.reader, buffer[:size]); err != nil {
  70. return nil, err
  71. }
  72. authBytes := buffer[:AuthSize]
  73. payload := buffer[AuthSize:size]
  74. actualAuthBytes := make([]byte, AuthSize)
  75. v.auth.Authenticate(payload)(actualAuthBytes)
  76. if !bytes.Equal(authBytes, actualAuthBytes) {
  77. return nil, newError("invalid auth")
  78. }
  79. var mb buf.MultiBuffer
  80. common.Must2(mb.Write(payload))
  81. return mb, nil
  82. }
  83. type ChunkWriter struct {
  84. writer io.Writer
  85. auth *Authenticator
  86. buffer []byte
  87. }
  88. func NewChunkWriter(writer io.Writer, auth *Authenticator) *ChunkWriter {
  89. return &ChunkWriter{
  90. writer: writer,
  91. auth: auth,
  92. buffer: make([]byte, 32*1024),
  93. }
  94. }
  95. // WriteMultiBuffer implements buf.Writer.
  96. func (w *ChunkWriter) WriteMultiBuffer(mb buf.MultiBuffer) error {
  97. defer mb.Release()
  98. for {
  99. payloadLen, _ := mb.Read(w.buffer[2+AuthSize:])
  100. serial.Uint16ToBytes(uint16(payloadLen), w.buffer[:0])
  101. w.auth.Authenticate(w.buffer[2+AuthSize : 2+AuthSize+payloadLen])(w.buffer[2:])
  102. if err := buf.WriteAllBytes(w.writer, w.buffer[:2+AuthSize+payloadLen]); err != nil {
  103. return err
  104. }
  105. if mb.IsEmpty() {
  106. break
  107. }
  108. }
  109. return nil
  110. }