1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2024-07-01 11:35:23 +00:00
v2fly/common/buf/multi_buffer.go

216 lines
4.3 KiB
Go
Raw Normal View History

2017-04-15 19:07:23 +00:00
package buf
2017-11-04 00:33:28 +00:00
import (
"io"
2017-11-07 10:40:12 +00:00
"v2ray.com/core/common"
2017-11-04 00:33:28 +00:00
"v2ray.com/core/common/errors"
2018-04-08 21:22:55 +00:00
"v2ray.com/core/common/serial"
2017-11-04 00:33:28 +00:00
)
2017-04-15 19:07:23 +00:00
2017-11-07 10:40:12 +00:00
// ReadAllToBytes reads all content from the reader into a byte array, until EOF.
2017-11-04 00:33:28 +00:00
func ReadAllToBytes(reader io.Reader) ([]byte, error) {
2018-11-18 18:36:36 +00:00
mb, err := ReadFrom(reader)
2017-11-04 00:33:28 +00:00
if err != nil {
return nil, err
}
if mb.Len() == 0 {
return nil, nil
}
2017-11-04 00:33:28 +00:00
b := make([]byte, mb.Len())
mb, _ = SplitBytes(mb, b)
2018-11-17 21:45:07 +00:00
ReleaseMulti(mb)
2017-11-04 00:33:28 +00:00
return b, nil
}
2017-04-23 17:16:56 +00:00
// MultiBuffer is a list of Buffers. The order of Buffer matters.
2017-04-15 19:07:23 +00:00
type MultiBuffer []*Buffer
2018-11-17 08:12:20 +00:00
// 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
2018-03-29 19:40:23 +00:00
}
2018-11-17 08:12:20 +00:00
return dest, src[:0]
2017-04-15 19:07:23 +00:00
}
2018-11-18 18:36:36 +00:00
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
}
2018-11-17 21:45:07 +00:00
// ReleaseMulti release 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]
}
2017-12-13 14:55:39 +00:00
// Copy copied the beginning part of the MultiBuffer into the given byte array.
2017-04-24 23:56:08 +00: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 18:00:50 +00:00
if int32(nBytes) < bb.Len() {
2017-04-24 23:56:08 +00:00
break
}
}
return total
}
2018-11-18 18:36:36 +00:00
// ReadFrom reads all content from reader until EOF.
func ReadFrom(reader io.Reader) (MultiBuffer, error) {
mb := make(MultiBuffer, 0, 16)
2018-04-13 11:54:36 +00:00
for {
b := New()
2018-11-02 14:01:33 +00:00
_, err := b.ReadFullFrom(reader, Size)
2018-04-13 11:54:36 +00:00
if b.IsEmpty() {
b.Release()
} else {
2018-11-18 18:36:36 +00:00
mb = append(mb, b)
2018-04-13 11:54:36 +00:00
}
if err != nil {
2018-07-30 20:45:06 +00:00
if errors.Cause(err) == io.EOF || errors.Cause(err) == io.ErrUnexpectedEOF {
2018-11-18 18:36:36 +00:00
return mb, nil
2018-04-13 11:54:36 +00:00
}
2018-11-18 18:36:36 +00:00
return mb, err
2018-04-13 11:54:36 +00:00
}
}
}
func SplitBytes(mb MultiBuffer, b []byte) (MultiBuffer, int) {
2017-04-15 19:07:23 +00:00
totalBytes := 0
2018-11-18 18:36:36 +00:00
for len(mb) > 0 {
bb := mb[0]
2017-04-19 19:27:21 +00:00
nBytes, _ := bb.Read(b)
2017-04-15 19:07:23 +00:00
totalBytes += nBytes
b = b[nBytes:]
2018-11-18 18:36:36 +00:00
if !bb.IsEmpty() {
2017-04-15 19:07:23 +00:00
break
}
2018-11-18 18:36:36 +00:00
bb.Release()
mb = mb[1:]
2017-05-01 22:28:16 +00:00
}
2018-04-08 21:22:55 +00:00
return mb, totalBytes
2018-04-16 22:31:10 +00:00
}
2017-04-23 17:16:56 +00:00
// Len returns the total number of bytes in the MultiBuffer.
2018-11-16 10:29:16 +00:00
func (mb MultiBuffer) Len() int32 {
2018-04-18 09:45:40 +00:00
if mb == nil {
return 0
}
2018-04-02 18:00:50 +00:00
size := int32(0)
2018-11-16 10:29:16 +00:00
for _, b := range mb {
2017-04-15 19:07:23 +00:00
size += b.Len()
}
return size
}
2017-04-23 17:16:56 +00:00
// IsEmpty return true if the MultiBuffer has no content.
2017-04-15 19:07:23 +00:00
func (mb MultiBuffer) IsEmpty() bool {
for _, b := range mb {
if !b.IsEmpty() {
return false
}
}
return true
}
2018-11-17 21:45:07 +00:00
// String returns the content of the MultiBuffer in string.
2018-04-08 21:22:55 +00:00
func (mb MultiBuffer) String() string {
v := make([]interface{}, len(mb))
for i, b := range mb {
v[i] = b
}
return serial.Concat(v...)
}
2017-12-13 14:55:39 +00:00
// SliceBySize splits the beginning of this MultiBuffer into another one, for at most size bytes.
2018-04-01 22:44:47 +00:00
func (mb *MultiBuffer) SliceBySize(size int32) MultiBuffer {
2018-11-16 10:29:16 +00:00
slice := make(MultiBuffer, 0, 10)
2018-04-02 18:00:50 +00:00
sliceSize := int32(0)
endIndex := len(*mb)
for i, b := range *mb {
2018-04-02 18:00:50 +00:00
if b.Len()+sliceSize > size {
endIndex = i
break
}
2017-04-19 19:27:21 +00:00
sliceSize += b.Len()
2018-11-17 08:12:20 +00:00
slice = append(slice, b)
2017-11-07 10:40:12 +00:00
(*mb)[i] = nil
}
*mb = (*mb)[endIndex:]
if endIndex == 0 && len(*mb) > 0 {
2018-08-16 10:05:33 +00:00
b := New()
2018-11-02 14:01:33 +00:00
common.Must2(b.ReadFullFrom((*mb)[0], size))
2018-11-16 10:08:12 +00:00
return MultiBuffer{b}
}
return slice
}
2017-05-01 22:28:16 +00:00
2017-11-08 17:14:44 +00:00
// SplitFirst splits out the first Buffer in this MultiBuffer.
2017-05-01 22:28:16 +00:00
func (mb *MultiBuffer) SplitFirst() *Buffer {
if len(*mb) == 0 {
return nil
}
b := (*mb)[0]
2017-11-07 10:40:12 +00:00
(*mb)[0] = nil
2017-05-01 22:28:16 +00:00
*mb = (*mb)[1:]
return b
}
2018-11-18 18:36:36 +00:00
type MultiBufferContainer struct {
MultiBuffer
}
func (c *MultiBufferContainer) Read(b []byte) (int, error) {
if c.MultiBuffer.IsEmpty() {
return 0, io.EOF
}
mb, nBytes := SplitBytes(c.MultiBuffer, b)
2018-11-18 18:36:36 +00:00
c.MultiBuffer = mb
return nBytes, nil
2018-11-18 18:36:36 +00:00
}
func (c *MultiBufferContainer) ReadMultiBuffer() (MultiBuffer, error) {
mb := c.MultiBuffer
c.MultiBuffer = nil
return mb, nil
}
func (c *MultiBufferContainer) Write(b []byte) (int, error) {
c.MultiBuffer = MergeBytes(c.MultiBuffer, b)
return len(b), nil
}
func (c *MultiBufferContainer) WriteMultiBuffer(b MultiBuffer) error {
mb, _ := MergeMulti(c.MultiBuffer, b)
c.MultiBuffer = mb
return nil
}
func (c *MultiBufferContainer) Close() error {
c.MultiBuffer = ReleaseMulti(c.MultiBuffer)
return nil
}