addons.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. // +build !confonly
  2. package encoding
  3. import (
  4. "io"
  5. "github.com/golang/protobuf/proto"
  6. "v2ray.com/core/common/buf"
  7. "v2ray.com/core/common/errors"
  8. "v2ray.com/core/common/protocol"
  9. )
  10. // EncodeHeaderAddons Add addons byte to the header
  11. func EncodeHeaderAddons(buffer *buf.Buffer, addons *Addons) error {
  12. if err := buffer.WriteByte(0); err != nil {
  13. return newError("failed to write addons protobuf length").Base(err)
  14. }
  15. return nil
  16. }
  17. func DecodeHeaderAddons(buffer *buf.Buffer, reader io.Reader) (*Addons, error) {
  18. addons := new(Addons)
  19. buffer.Clear()
  20. if _, err := buffer.ReadFullFrom(reader, 1); err != nil {
  21. return nil, newError("failed to read addons protobuf length").Base(err)
  22. }
  23. if length := int32(buffer.Byte(0)); length != 0 {
  24. buffer.Clear()
  25. if _, err := buffer.ReadFullFrom(reader, length); err != nil {
  26. return nil, newError("failed to read addons protobuf value").Base(err)
  27. }
  28. if err := proto.Unmarshal(buffer.Bytes(), addons); err != nil {
  29. return nil, newError("failed to unmarshal addons protobuf value").Base(err)
  30. }
  31. }
  32. return addons, nil
  33. }
  34. // EncodeBodyAddons returns a Writer that auto-encrypt content written by caller.
  35. func EncodeBodyAddons(writer io.Writer, request *protocol.RequestHeader, addons *Addons) buf.Writer {
  36. if request.Command == protocol.RequestCommandUDP {
  37. return NewMultiLengthPacketWriter(writer.(buf.Writer))
  38. }
  39. return buf.NewWriter(writer)
  40. }
  41. // DecodeBodyAddons returns a Reader from which caller can fetch decrypted body.
  42. func DecodeBodyAddons(reader io.Reader, request *protocol.RequestHeader, addons *Addons) buf.Reader {
  43. if request.Command == protocol.RequestCommandUDP {
  44. return NewLengthPacketReader(reader)
  45. }
  46. return buf.NewReader(reader)
  47. }
  48. func NewMultiLengthPacketWriter(writer buf.Writer) *MultiLengthPacketWriter {
  49. return &MultiLengthPacketWriter{
  50. Writer: writer,
  51. }
  52. }
  53. type MultiLengthPacketWriter struct {
  54. buf.Writer
  55. }
  56. func (w *MultiLengthPacketWriter) WriteMultiBuffer(mb buf.MultiBuffer) error {
  57. defer buf.ReleaseMulti(mb)
  58. if len(mb)+1 > 64*1024*1024 {
  59. return errors.New("value too large")
  60. }
  61. sliceSize := len(mb) + 1
  62. mb2Write := make(buf.MultiBuffer, 0, sliceSize)
  63. for _, b := range mb {
  64. length := b.Len()
  65. if length == 0 || length+2 > buf.Size {
  66. continue
  67. }
  68. eb := buf.New()
  69. if err := eb.WriteByte(byte(length >> 8)); err != nil {
  70. eb.Release()
  71. continue
  72. }
  73. if err := eb.WriteByte(byte(length)); err != nil {
  74. eb.Release()
  75. continue
  76. }
  77. if _, err := eb.Write(b.Bytes()); err != nil {
  78. eb.Release()
  79. continue
  80. }
  81. mb2Write = append(mb2Write, eb)
  82. }
  83. if mb2Write.IsEmpty() {
  84. return nil
  85. }
  86. return w.Writer.WriteMultiBuffer(mb2Write)
  87. }
  88. type LengthPacketWriter struct {
  89. io.Writer
  90. cache []byte
  91. }
  92. func (w *LengthPacketWriter) WriteMultiBuffer(mb buf.MultiBuffer) error {
  93. length := mb.Len() // none of mb is nil
  94. if length == 0 {
  95. return nil
  96. }
  97. defer func() {
  98. w.cache = w.cache[:0]
  99. }()
  100. w.cache = append(w.cache, byte(length>>8), byte(length))
  101. for i, b := range mb {
  102. w.cache = append(w.cache, b.Bytes()...)
  103. b.Release()
  104. mb[i] = nil
  105. }
  106. if _, err := w.Write(w.cache); err != nil {
  107. return newError("failed to write a packet").Base(err)
  108. }
  109. return nil
  110. }
  111. func NewLengthPacketReader(reader io.Reader) *LengthPacketReader {
  112. return &LengthPacketReader{
  113. Reader: reader,
  114. cache: make([]byte, 2),
  115. }
  116. }
  117. type LengthPacketReader struct {
  118. io.Reader
  119. cache []byte
  120. }
  121. func (r *LengthPacketReader) ReadMultiBuffer() (buf.MultiBuffer, error) {
  122. if _, err := io.ReadFull(r.Reader, r.cache); err != nil { // maybe EOF
  123. return nil, newError("failed to read packet length").Base(err)
  124. }
  125. length := int32(r.cache[0])<<8 | int32(r.cache[1])
  126. mb := make(buf.MultiBuffer, 0, length/buf.Size+1)
  127. for length > 0 {
  128. size := length
  129. if size > buf.Size {
  130. size = buf.Size
  131. }
  132. length -= size
  133. b := buf.New()
  134. if _, err := b.ReadFullFrom(r.Reader, size); err != nil {
  135. return nil, newError("failed to read packet payload").Base(err)
  136. }
  137. mb = append(mb, b)
  138. }
  139. return mb, nil
  140. }