| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112 | package bufimport (	"io"	"v2ray.com/core/common/errors"	"v2ray.com/core/common/signal")type errorHandler func(error) errortype dataHandler func(MultiBuffer)type copyHandler struct {	onReadError  []errorHandler	onData       []dataHandler	onWriteError []errorHandler}func (h *copyHandler) readFrom(reader Reader) (MultiBuffer, error) {	mb, err := reader.ReadMultiBuffer()	if err != nil {		for _, handler := range h.onReadError {			err = handler(err)		}	}	return mb, err}func (h *copyHandler) writeTo(writer Writer, mb MultiBuffer) error {	err := writer.WriteMultiBuffer(mb)	if err != nil {		for _, handler := range h.onWriteError {			err = handler(err)		}	}	return err}type SizeCounter struct {	Size int64}type CopyOption func(*copyHandler)func IgnoreReaderError() CopyOption {	return func(handler *copyHandler) {		handler.onReadError = append(handler.onReadError, func(err error) error {			return nil		})	}}func IgnoreWriterError() CopyOption {	return func(handler *copyHandler) {		handler.onWriteError = append(handler.onWriteError, func(err error) error {			return nil		})	}}func UpdateActivity(timer signal.ActivityUpdater) CopyOption {	return func(handler *copyHandler) {		handler.onData = append(handler.onData, func(MultiBuffer) {			timer.Update()		})	}}func CountSize(sc *SizeCounter) CopyOption {	return func(handler *copyHandler) {		handler.onData = append(handler.onData, func(b MultiBuffer) {			sc.Size += int64(b.Len())		})	}}func copyInternal(reader Reader, writer Writer, handler *copyHandler) error {	for {		buffer, err := handler.readFrom(reader)		if err != nil {			return err		}		if buffer.IsEmpty() {			buffer.Release()			continue		}		for _, handler := range handler.onData {			handler(buffer)		}		if err := handler.writeTo(writer, buffer); err != nil {			buffer.Release()			return err		}	}}// Copy dumps all payload from reader to writer or stops when an error occurs.// ActivityTimer gets updated as soon as there is a payload.func Copy(reader Reader, writer Writer, options ...CopyOption) error {	handler := new(copyHandler)	for _, option := range options {		option(handler)	}	err := copyInternal(reader, writer, handler)	if err != nil && errors.Cause(err) != io.EOF {		return err	}	return nil}
 |