123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310 |
- package buf
- import (
- "io"
- "github.com/xtls/xray-core/common"
- "github.com/xtls/xray-core/common/errors"
- "github.com/xtls/xray-core/common/serial"
- )
- // ReadAllToBytes reads all content from the reader into a byte array, until EOF.
- func ReadAllToBytes(reader io.Reader) ([]byte, error) {
- mb, err := ReadFrom(reader)
- if err != nil {
- return nil, err
- }
- if mb.Len() == 0 {
- return nil, nil
- }
- b := make([]byte, mb.Len())
- mb, _ = SplitBytes(mb, b)
- ReleaseMulti(mb)
- return b, nil
- }
- // MultiBuffer is a list of Buffers. The order of Buffer matters.
- type MultiBuffer []*Buffer
- // MergeMulti merges content from src to dest, and returns the new address of dest and src
- func MergeMulti(dest MultiBuffer, src MultiBuffer) (MultiBuffer, MultiBuffer) {
- dest = append(dest, src...)
- for idx := range src {
- src[idx] = nil
- }
- return dest, src[:0]
- }
- // MergeBytes merges the given bytes into MultiBuffer and return the new address of the merged MultiBuffer.
- func MergeBytes(dest MultiBuffer, src []byte) MultiBuffer {
- n := len(dest)
- if n > 0 && !(dest)[n-1].IsFull() {
- nBytes, _ := (dest)[n-1].Write(src)
- src = src[nBytes:]
- }
- for len(src) > 0 {
- b := New()
- nBytes, _ := b.Write(src)
- src = src[nBytes:]
- dest = append(dest, b)
- }
- return dest
- }
- // ReleaseMulti releases all content of the MultiBuffer, and returns an empty MultiBuffer.
- func ReleaseMulti(mb MultiBuffer) MultiBuffer {
- for i := range mb {
- mb[i].Release()
- mb[i] = nil
- }
- return mb[:0]
- }
- // Copy copied the beginning part of the MultiBuffer into the given byte array.
- func (mb MultiBuffer) Copy(b []byte) int {
- total := 0
- for _, bb := range mb {
- nBytes := copy(b[total:], bb.Bytes())
- total += nBytes
- if int32(nBytes) < bb.Len() {
- break
- }
- }
- return total
- }
- // ReadFrom reads all content from reader until EOF.
- func ReadFrom(reader io.Reader) (MultiBuffer, error) {
- mb := make(MultiBuffer, 0, 16)
- for {
- b := New()
- _, err := b.ReadFullFrom(reader, Size)
- if b.IsEmpty() {
- b.Release()
- } else {
- mb = append(mb, b)
- }
- if err != nil {
- if errors.Cause(err) == io.EOF || errors.Cause(err) == io.ErrUnexpectedEOF {
- return mb, nil
- }
- return mb, err
- }
- }
- }
- // SplitBytes splits the given amount of bytes from the beginning of the MultiBuffer.
- // It returns the new address of MultiBuffer leftover, and number of bytes written into the input byte slice.
- func SplitBytes(mb MultiBuffer, b []byte) (MultiBuffer, int) {
- totalBytes := 0
- endIndex := -1
- for i := range mb {
- pBuffer := mb[i]
- nBytes, _ := pBuffer.Read(b)
- totalBytes += nBytes
- b = b[nBytes:]
- if !pBuffer.IsEmpty() {
- endIndex = i
- break
- }
- pBuffer.Release()
- mb[i] = nil
- }
- if endIndex == -1 {
- mb = mb[:0]
- } else {
- mb = mb[endIndex:]
- }
- return mb, totalBytes
- }
- // SplitFirstBytes splits the first buffer from MultiBuffer, and then copy its content into the given slice.
- func SplitFirstBytes(mb MultiBuffer, p []byte) (MultiBuffer, int) {
- mb, b := SplitFirst(mb)
- if b == nil {
- return mb, 0
- }
- n := copy(p, b.Bytes())
- b.Release()
- return mb, n
- }
- // Compact returns another MultiBuffer by merging all content of the given one together.
- func Compact(mb MultiBuffer) MultiBuffer {
- if len(mb) == 0 {
- return mb
- }
- mb2 := make(MultiBuffer, 0, len(mb))
- last := mb[0]
- for i := 1; i < len(mb); i++ {
- curr := mb[i]
- if last.Len()+curr.Len() > Size {
- mb2 = append(mb2, last)
- last = curr
- } else {
- common.Must2(last.ReadFrom(curr))
- curr.Release()
- }
- }
- mb2 = append(mb2, last)
- return mb2
- }
- // SplitFirst splits the first Buffer from the beginning of the MultiBuffer.
- func SplitFirst(mb MultiBuffer) (MultiBuffer, *Buffer) {
- if len(mb) == 0 {
- return mb, nil
- }
- b := mb[0]
- mb[0] = nil
- mb = mb[1:]
- return mb, b
- }
- // SplitSize splits the beginning of the MultiBuffer into another one, for at most size bytes.
- func SplitSize(mb MultiBuffer, size int32) (MultiBuffer, MultiBuffer) {
- if len(mb) == 0 {
- return mb, nil
- }
- if mb[0].Len() > size {
- b := New()
- copy(b.Extend(size), mb[0].BytesTo(size))
- mb[0].Advance(size)
- return mb, MultiBuffer{b}
- }
- totalBytes := int32(0)
- var r MultiBuffer
- endIndex := -1
- for i := range mb {
- if totalBytes+mb[i].Len() > size {
- endIndex = i
- break
- }
- totalBytes += mb[i].Len()
- r = append(r, mb[i])
- mb[i] = nil
- }
- if endIndex == -1 {
- // To reuse mb array
- mb = mb[:0]
- } else {
- mb = mb[endIndex:]
- }
- return mb, r
- }
- // SplitMulti splits the beginning of the MultiBuffer into first one, the index i and after into second one
- func SplitMulti(mb MultiBuffer, i int) (MultiBuffer, MultiBuffer) {
- mb2 := make(MultiBuffer, 0, len(mb))
- if i < len(mb) && i >= 0 {
- mb2 = append(mb2, mb[i:]...)
- for j := i; j < len(mb); j++ {
- mb[j] = nil
- }
- mb = mb[:i]
- }
- return mb, mb2
- }
- // WriteMultiBuffer writes all buffers from the MultiBuffer to the Writer one by one, and return error if any, with leftover MultiBuffer.
- func WriteMultiBuffer(writer io.Writer, mb MultiBuffer) (MultiBuffer, error) {
- for {
- mb2, b := SplitFirst(mb)
- mb = mb2
- if b == nil {
- break
- }
- _, err := writer.Write(b.Bytes())
- b.Release()
- if err != nil {
- return mb, err
- }
- }
- return nil, nil
- }
- // Len returns the total number of bytes in the MultiBuffer.
- func (mb MultiBuffer) Len() int32 {
- if mb == nil {
- return 0
- }
- size := int32(0)
- for _, b := range mb {
- size += b.Len()
- }
- return size
- }
- // IsEmpty returns true if the MultiBuffer has no content.
- func (mb MultiBuffer) IsEmpty() bool {
- for _, b := range mb {
- if !b.IsEmpty() {
- return false
- }
- }
- return true
- }
- // String returns the content of the MultiBuffer in string.
- func (mb MultiBuffer) String() string {
- v := make([]interface{}, len(mb))
- for i, b := range mb {
- v[i] = b
- }
- return serial.Concat(v...)
- }
- // MultiBufferContainer is a ReadWriteCloser wrapper over MultiBuffer.
- type MultiBufferContainer struct {
- MultiBuffer
- }
- // Read implements io.Reader.
- func (c *MultiBufferContainer) Read(b []byte) (int, error) {
- if c.MultiBuffer.IsEmpty() {
- return 0, io.EOF
- }
- mb, nBytes := SplitBytes(c.MultiBuffer, b)
- c.MultiBuffer = mb
- return nBytes, nil
- }
- // ReadMultiBuffer implements Reader.
- func (c *MultiBufferContainer) ReadMultiBuffer() (MultiBuffer, error) {
- mb := c.MultiBuffer
- c.MultiBuffer = nil
- return mb, nil
- }
- // Write implements io.Writer.
- func (c *MultiBufferContainer) Write(b []byte) (int, error) {
- c.MultiBuffer = MergeBytes(c.MultiBuffer, b)
- return len(b), nil
- }
- // WriteMultiBuffer implements Writer.
- func (c *MultiBufferContainer) WriteMultiBuffer(b MultiBuffer) error {
- mb, _ := MergeMulti(c.MultiBuffer, b)
- c.MultiBuffer = mb
- return nil
- }
- // Close implements io.Closer.
- func (c *MultiBufferContainer) Close() error {
- c.MultiBuffer = ReleaseMulti(c.MultiBuffer)
- return nil
- }
|