123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144 |
- package buf
- import (
- "io"
- "net"
- "os"
- "syscall"
- "time"
- "github.com/xtls/xray-core/features/stats"
- "github.com/xtls/xray-core/transport/internet/stat"
- )
- // Reader extends io.Reader with MultiBuffer.
- type Reader interface {
- // ReadMultiBuffer reads content from underlying reader, and put it into a MultiBuffer.
- ReadMultiBuffer() (MultiBuffer, error)
- }
- // ErrReadTimeout is an error that happens with IO timeout.
- var ErrReadTimeout = newError("IO timeout")
- // TimeoutReader is a reader that returns error if Read() operation takes longer than the given timeout.
- type TimeoutReader interface {
- ReadMultiBufferTimeout(time.Duration) (MultiBuffer, error)
- }
- // Writer extends io.Writer with MultiBuffer.
- type Writer interface {
- // WriteMultiBuffer writes a MultiBuffer into underlying writer.
- WriteMultiBuffer(MultiBuffer) error
- }
- // WriteAllBytes ensures all bytes are written into the given writer.
- func WriteAllBytes(writer io.Writer, payload []byte, c stats.Counter) error {
- wc := 0
- defer func() {
- if c != nil {
- c.Add(int64(wc))
- }
- }()
- for len(payload) > 0 {
- n, err := writer.Write(payload)
- wc += n
- if err != nil {
- return err
- }
- payload = payload[n:]
- }
- return nil
- }
- func isPacketReader(reader io.Reader) bool {
- _, ok := reader.(net.PacketConn)
- return ok
- }
- // NewReader creates a new Reader.
- // The Reader instance doesn't take the ownership of reader.
- func NewReader(reader io.Reader) Reader {
- if mr, ok := reader.(Reader); ok {
- return mr
- }
- if isPacketReader(reader) {
- return &PacketReader{
- Reader: reader,
- }
- }
- _, isFile := reader.(*os.File)
- if !isFile && useReadv {
- if sc, ok := reader.(syscall.Conn); ok {
- rawConn, err := sc.SyscallConn()
- if err != nil {
- newError("failed to get sysconn").Base(err).WriteToLog()
- } else {
- var counter stats.Counter
- if statConn, ok := reader.(*stat.CounterConnection); ok {
- reader = statConn.Connection
- counter = statConn.ReadCounter
- }
- return NewReadVReader(reader, rawConn, counter)
- }
- }
- }
- return &SingleReader{
- Reader: reader,
- }
- }
- // NewPacketReader creates a new PacketReader based on the given reader.
- func NewPacketReader(reader io.Reader) Reader {
- if mr, ok := reader.(Reader); ok {
- return mr
- }
- return &PacketReader{
- Reader: reader,
- }
- }
- func isPacketWriter(writer io.Writer) bool {
- if _, ok := writer.(net.PacketConn); ok {
- return true
- }
- // If the writer doesn't implement syscall.Conn, it is probably not a TCP connection.
- if _, ok := writer.(syscall.Conn); !ok {
- return true
- }
- return false
- }
- // NewWriter creates a new Writer.
- func NewWriter(writer io.Writer) Writer {
- if mw, ok := writer.(Writer); ok {
- return mw
- }
- iConn := writer
- if statConn, ok := writer.(*stat.CounterConnection); ok {
- iConn = statConn.Connection
- }
- if isPacketWriter(iConn) {
- return &SequentialWriter{
- Writer: writer,
- }
- }
- var counter stats.Counter
- if statConn, ok := writer.(*stat.CounterConnection); ok {
- counter = statConn.WriteCounter
- }
- return &BufferToBytesWriter{
- Writer: iConn,
- counter: counter,
- }
- }
|