mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-03 11:57:39 +03:00
171 lines
3.4 KiB
Go
171 lines
3.4 KiB
Go
package bufio
|
|
|
|
import (
|
|
"io"
|
|
"os"
|
|
"sync"
|
|
|
|
"github.com/sagernet/sing/common"
|
|
"github.com/sagernet/sing/common/buf"
|
|
N "github.com/sagernet/sing/common/network"
|
|
)
|
|
|
|
type BufferedReader struct {
|
|
upstream N.ExtendedReader
|
|
buffer *buf.Buffer
|
|
}
|
|
|
|
func NewBufferedReader(upstream io.Reader, buffer *buf.Buffer) *BufferedReader {
|
|
return &BufferedReader{
|
|
upstream: NewExtendedReader(upstream),
|
|
buffer: buffer,
|
|
}
|
|
}
|
|
|
|
func (r *BufferedReader) Read(p []byte) (n int, err error) {
|
|
if r.buffer == nil {
|
|
return r.upstream.Read(p)
|
|
}
|
|
if r.buffer.Closed() {
|
|
return 0, os.ErrClosed
|
|
}
|
|
if r.buffer.IsEmpty() {
|
|
r.buffer.Reset()
|
|
err = r.upstream.ReadBuffer(r.buffer)
|
|
if err != nil {
|
|
r.buffer.Release()
|
|
return
|
|
}
|
|
}
|
|
return r.buffer.Read(p)
|
|
}
|
|
|
|
func (r *BufferedReader) ReadBuffer(buffer *buf.Buffer) error {
|
|
if r.buffer == nil {
|
|
return r.upstream.ReadBuffer(buffer)
|
|
}
|
|
if r.buffer.Closed() {
|
|
return os.ErrClosed
|
|
}
|
|
var err error
|
|
if r.buffer.IsEmpty() {
|
|
r.buffer.Reset()
|
|
err = r.upstream.ReadBuffer(r.buffer)
|
|
if err != nil {
|
|
r.buffer.Release()
|
|
return err
|
|
}
|
|
}
|
|
if r.buffer.Len() > buffer.FreeLen() {
|
|
err = common.Error(buffer.ReadFullFrom(r.buffer, buffer.FreeLen()))
|
|
} else {
|
|
err = common.Error(buffer.ReadFullFrom(r.buffer, r.buffer.Len()))
|
|
}
|
|
if err != nil {
|
|
r.buffer.Release()
|
|
}
|
|
return err
|
|
}
|
|
|
|
//TODO: add CopyBuffer
|
|
/*func (r *BufferedReader) WriteTo(w io.Writer) (n int64, err error) {
|
|
if r.buffer == nil {
|
|
return Copy(w, r.upstream)
|
|
}
|
|
if r.buffer.Closed() {
|
|
return 0, os.ErrClosed
|
|
}
|
|
defer r.buffer.Release()
|
|
return CopyExtendedBuffer(NewExtendedWriter(w), NewExtendedReader(r.upstream), r.buffer)
|
|
}*/
|
|
|
|
func (r *BufferedReader) ReadCached() *buf.Buffer {
|
|
buffer := r.buffer
|
|
r.buffer = nil
|
|
return buffer
|
|
}
|
|
|
|
func (r *BufferedReader) Close() error {
|
|
r.buffer.Release()
|
|
r.buffer = nil
|
|
return nil
|
|
}
|
|
|
|
func (r *BufferedReader) Upstream() any {
|
|
return r.upstream
|
|
}
|
|
|
|
func (r *BufferedReader) ReaderReplaceable() bool {
|
|
buffer := r.buffer
|
|
return buffer == nil || buffer.Closed()
|
|
}
|
|
|
|
func (r *BufferedReader) CreateReadWaiter() (ReadWaiter, bool) {
|
|
reader, created := CreateReadWaiter(r.upstream)
|
|
if !created {
|
|
return nil, false
|
|
}
|
|
return &bufferedReadWaiter{r, reader}, true
|
|
}
|
|
|
|
type BufferedWriter struct {
|
|
upstream io.Writer
|
|
buffer *buf.Buffer
|
|
access sync.Mutex
|
|
}
|
|
|
|
func NewBufferedWriter(upstream io.Writer, buffer *buf.Buffer) *BufferedWriter {
|
|
return &BufferedWriter{
|
|
upstream: upstream,
|
|
buffer: buffer,
|
|
}
|
|
}
|
|
|
|
func (w *BufferedWriter) Write(p []byte) (n int, err error) {
|
|
w.access.Lock()
|
|
defer w.access.Unlock()
|
|
if w.buffer == nil {
|
|
return w.upstream.Write(p)
|
|
}
|
|
for {
|
|
var writeN int
|
|
writeN, err = w.buffer.Write(p[n:])
|
|
n += writeN
|
|
if n == len(p) {
|
|
return
|
|
}
|
|
_, err = w.upstream.Write(w.buffer.Bytes())
|
|
if err != nil {
|
|
return
|
|
}
|
|
w.buffer.FullReset()
|
|
}
|
|
}
|
|
|
|
func (w *BufferedWriter) Fallthrough() error {
|
|
w.access.Lock()
|
|
defer w.access.Unlock()
|
|
if w.buffer == nil {
|
|
return nil
|
|
}
|
|
if !w.buffer.IsEmpty() {
|
|
_, err := w.upstream.Write(w.buffer.Bytes())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
w.buffer.Release()
|
|
w.buffer = nil
|
|
return nil
|
|
}
|
|
|
|
/*func (w *BufferedWriter) ReadFrom(r io.Reader) (n int64, err error) {
|
|
if w.buffer == nil {
|
|
return Copy(w.upstream, r)
|
|
}
|
|
return CopyExtendedBuffer(NewExtendedWriter(w), NewExtendedReader(r), w.buffer)
|
|
}*/
|
|
|
|
func (w *BufferedWriter) WriterReplaceable() bool {
|
|
return w.buffer == nil
|
|
}
|