ota.go 2.8 KB

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