ota.go 2.7 KB

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