1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-01-18 07:17:32 -05:00
v2fly/common/buf/multi_buffer.go

249 lines
5.1 KiB
Go
Raw Normal View History

2017-04-15 15:07:23 -04:00
package buf
2017-11-03 20:33:28 -04:00
import (
"io"
"net"
2017-11-07 05:40:12 -05:00
"v2ray.com/core/common"
2017-11-03 20:33:28 -04:00
"v2ray.com/core/common/errors"
2018-04-08 17:22:55 -04:00
"v2ray.com/core/common/serial"
2017-11-03 20:33:28 -04:00
)
2017-04-15 15:07:23 -04:00
2017-11-07 10:53:02 -05:00
// ReadAllToMultiBuffer reads all content from the reader into a MultiBuffer, until EOF.
2017-11-03 20:33:28 -04:00
func ReadAllToMultiBuffer(reader io.Reader) (MultiBuffer, error) {
2017-11-08 18:55:28 -05:00
mb := NewMultiBufferCap(128)
2017-11-03 20:33:28 -04:00
2018-04-13 07:54:36 -04:00
if _, err := mb.ReadFrom(reader); err != nil {
mb.Release()
return nil, err
2017-11-03 20:33:28 -04:00
}
2018-04-13 07:54:36 -04:00
return mb, nil
2017-11-03 20:33:28 -04:00
}
2017-11-07 05:40:12 -05:00
// ReadAllToBytes reads all content from the reader into a byte array, until EOF.
2017-11-03 20:33:28 -04:00
func ReadAllToBytes(reader io.Reader) ([]byte, error) {
mb, err := ReadAllToMultiBuffer(reader)
if err != nil {
return nil, err
}
if mb.Len() == 0 {
return nil, nil
}
2017-11-03 20:33:28 -04:00
b := make([]byte, mb.Len())
2017-11-07 05:40:12 -05:00
common.Must2(mb.Read(b))
2017-11-03 20:33:28 -04:00
mb.Release()
return b, nil
}
2017-04-23 13:16:56 -04:00
// MultiBuffer is a list of Buffers. The order of Buffer matters.
2017-04-15 15:07:23 -04:00
type MultiBuffer []*Buffer
2017-11-08 18:55:28 -05:00
// NewMultiBufferCap creates a new MultiBuffer instance.
2018-04-02 14:00:50 -04:00
func NewMultiBufferCap(capacity int32) MultiBuffer {
2017-11-08 18:55:28 -05:00
return MultiBuffer(make([]*Buffer, 0, capacity))
2017-04-15 15:07:23 -04:00
}
2017-04-23 13:16:56 -04:00
// NewMultiBufferValue wraps a list of Buffers into MultiBuffer.
2017-04-15 15:07:23 -04:00
func NewMultiBufferValue(b ...*Buffer) MultiBuffer {
return MultiBuffer(b)
}
2017-11-07 09:02:21 -05:00
// Append appends buffer to the end of this MultiBuffer
2017-04-23 13:16:56 -04:00
func (mb *MultiBuffer) Append(buf *Buffer) {
2018-03-29 15:40:23 -04:00
if buf != nil {
*mb = append(*mb, buf)
}
2017-04-15 15:07:23 -04:00
}
2017-11-07 09:02:21 -05:00
// AppendMulti appends a MultiBuffer to the end of this one.
2017-04-23 13:16:56 -04:00
func (mb *MultiBuffer) AppendMulti(buf MultiBuffer) {
*mb = append(*mb, buf...)
2017-04-15 15:07:23 -04:00
}
2017-12-13 09:55:39 -05:00
// Copy copied the beginning part of the MultiBuffer into the given byte array.
2017-04-24 19:56:08 -04:00
func (mb MultiBuffer) Copy(b []byte) int {
total := 0
for _, bb := range mb {
nBytes := copy(b[total:], bb.Bytes())
total += nBytes
2018-04-02 14:00:50 -04:00
if int32(nBytes) < bb.Len() {
2017-04-24 19:56:08 -04:00
break
}
}
return total
}
2018-04-13 07:54:36 -04:00
// ReadFrom implements io.ReaderFrom.
func (mb *MultiBuffer) ReadFrom(reader io.Reader) (int64, error) {
totalBytes := int64(0)
for {
b := New()
2018-07-30 16:45:06 -04:00
err := b.Reset(ReadFullFrom(reader, Size))
2018-04-13 07:54:36 -04:00
if b.IsEmpty() {
b.Release()
} else {
mb.Append(b)
}
totalBytes += int64(b.Len())
if err != nil {
2018-07-30 16:45:06 -04:00
if errors.Cause(err) == io.EOF || errors.Cause(err) == io.ErrUnexpectedEOF {
2018-04-13 07:54:36 -04:00
return totalBytes, nil
}
return totalBytes, err
}
}
}
2017-11-07 09:02:21 -05:00
// Read implements io.Reader.
2017-04-15 15:07:23 -04:00
func (mb *MultiBuffer) Read(b []byte) (int, error) {
2018-07-29 07:19:03 -04:00
if mb.IsEmpty() {
2018-04-08 17:22:55 -04:00
return 0, io.EOF
}
2017-04-15 15:07:23 -04:00
endIndex := len(*mb)
totalBytes := 0
for i, bb := range *mb {
2017-04-19 15:27:21 -04:00
nBytes, _ := bb.Read(b)
2017-04-15 15:07:23 -04:00
totalBytes += nBytes
b = b[nBytes:]
if bb.IsEmpty() {
bb.Release()
2017-11-07 10:28:39 -05:00
(*mb)[i] = nil
2017-04-15 15:07:23 -04:00
} else {
endIndex = i
break
}
}
*mb = (*mb)[endIndex:]
return totalBytes, nil
2018-04-13 07:54:36 -04:00
}
// WriteTo implements io.WriterTo.
func (mb *MultiBuffer) WriteTo(writer io.Writer) (int64, error) {
defer mb.Release()
totalBytes := int64(0)
for _, b := range *mb {
nBytes, err := writer.Write(b.Bytes())
totalBytes += int64(nBytes)
if err != nil {
return totalBytes, err
}
}
return totalBytes, nil
2017-04-15 15:07:23 -04:00
}
2017-11-07 09:02:21 -05:00
// Write implements io.Writer.
2018-04-08 17:22:55 -04:00
func (mb *MultiBuffer) Write(b []byte) (int, error) {
totalBytes := len(b)
2017-05-01 18:28:16 -04:00
n := len(*mb)
if n > 0 && !(*mb)[n-1].IsFull() {
nBytes, _ := (*mb)[n-1].Write(b)
b = b[nBytes:]
}
for len(b) > 0 {
bb := New()
nBytes, _ := bb.Write(b)
b = b[nBytes:]
mb.Append(bb)
}
2018-04-08 17:22:55 -04:00
return totalBytes, nil
2017-05-01 18:28:16 -04:00
}
2018-04-16 18:31:10 -04:00
// WriteMultiBuffer implements Writer.
func (mb *MultiBuffer) WriteMultiBuffer(b MultiBuffer) error {
*mb = append(*mb, b...)
2018-07-30 16:45:06 -04:00
for i := range b {
b[i] = nil
}
2018-04-16 18:31:10 -04:00
return nil
}
2017-04-23 13:16:56 -04:00
// Len returns the total number of bytes in the MultiBuffer.
2018-04-18 05:45:40 -04:00
func (mb *MultiBuffer) Len() int32 {
if mb == nil {
return 0
}
2018-04-02 14:00:50 -04:00
size := int32(0)
2018-04-18 05:45:40 -04:00
for _, b := range *mb {
2017-04-15 15:07:23 -04:00
size += b.Len()
}
return size
}
2017-04-23 13:16:56 -04:00
// IsEmpty return true if the MultiBuffer has no content.
2017-04-15 15:07:23 -04:00
func (mb MultiBuffer) IsEmpty() bool {
for _, b := range mb {
if !b.IsEmpty() {
return false
}
}
return true
}
2017-04-23 13:16:56 -04:00
// Release releases all Buffers in the MultiBuffer.
2017-11-07 05:40:12 -05:00
func (mb *MultiBuffer) Release() {
for i, b := range *mb {
2017-04-15 15:07:23 -04:00
b.Release()
2017-11-07 05:40:12 -05:00
(*mb)[i] = nil
2017-04-15 15:07:23 -04:00
}
2017-11-08 18:55:28 -05:00
*mb = nil
2017-04-15 15:07:23 -04:00
}
2017-04-16 16:30:29 -04:00
2018-04-08 17:22:55 -04:00
func (mb MultiBuffer) String() string {
v := make([]interface{}, len(mb))
for i, b := range mb {
v[i] = b
}
return serial.Concat(v...)
}
2017-04-23 13:16:56 -04:00
// ToNetBuffers converts this MultiBuffer to net.Buffers. The return net.Buffers points to the same content of the MultiBuffer.
2017-04-16 16:30:29 -04:00
func (mb MultiBuffer) ToNetBuffers() net.Buffers {
bs := make([][]byte, len(mb))
for i, b := range mb {
bs[i] = b.Bytes()
}
return bs
}
2017-12-13 09:55:39 -05:00
// SliceBySize splits the beginning of this MultiBuffer into another one, for at most size bytes.
2018-04-01 18:44:47 -04:00
func (mb *MultiBuffer) SliceBySize(size int32) MultiBuffer {
2017-11-08 18:55:28 -05:00
slice := NewMultiBufferCap(10)
2018-04-02 14:00:50 -04:00
sliceSize := int32(0)
endIndex := len(*mb)
for i, b := range *mb {
2018-04-02 14:00:50 -04:00
if b.Len()+sliceSize > size {
endIndex = i
break
}
2017-04-19 15:27:21 -04:00
sliceSize += b.Len()
slice.Append(b)
2017-11-07 05:40:12 -05:00
(*mb)[i] = nil
}
*mb = (*mb)[endIndex:]
if endIndex == 0 && len(*mb) > 0 {
2018-04-02 16:01:55 -04:00
b := NewSize(size)
common.Must(b.Reset(ReadFullFrom((*mb)[0], size)))
return NewMultiBufferValue(b)
}
return slice
}
2017-05-01 18:28:16 -04:00
2017-11-08 12:14:44 -05:00
// SplitFirst splits out the first Buffer in this MultiBuffer.
2017-05-01 18:28:16 -04:00
func (mb *MultiBuffer) SplitFirst() *Buffer {
if len(*mb) == 0 {
return nil
}
b := (*mb)[0]
2017-11-07 05:40:12 -05:00
(*mb)[0] = nil
2017-05-01 18:28:16 -04:00
*mb = (*mb)[1:]
return b
}