mirror of
https://github.com/AlexxIT/go2rtc.git
synced 2025-10-05 16:26:50 +08:00
Rewrite magic source
This commit is contained in:
@@ -40,7 +40,7 @@ func ADTSToCodec(b []byte) *core.Codec {
|
|||||||
//_ = rd.ReadBits(16) // CRC check
|
//_ = rd.ReadBits(16) // CRC check
|
||||||
|
|
||||||
// 3. Encode RTP config
|
// 3. Encode RTP config
|
||||||
wr := bits.NewWriter()
|
wr := bits.NewWriter(nil)
|
||||||
wr.WriteBits8(objType, 5)
|
wr.WriteBits8(objType, 5)
|
||||||
wr.WriteBits8(sampleRateIdx, 4)
|
wr.WriteBits8(sampleRateIdx, 4)
|
||||||
wr.WriteBits16(channels, 4)
|
wr.WriteBits16(channels, 4)
|
||||||
|
@@ -47,7 +47,10 @@ type Producer interface {
|
|||||||
// GetTrack - return Receiver, that can only produce rtp.Packet(s)
|
// GetTrack - return Receiver, that can only produce rtp.Packet(s)
|
||||||
GetTrack(media *Media, codec *Codec) (*Receiver, error)
|
GetTrack(media *Media, codec *Codec) (*Receiver, error)
|
||||||
|
|
||||||
|
// Deprecated: rename to Run()
|
||||||
Start() error
|
Start() error
|
||||||
|
|
||||||
|
// Deprecated: rename to Close()
|
||||||
Stop() error
|
Stop() error
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -59,6 +62,7 @@ type Consumer interface {
|
|||||||
|
|
||||||
AddTrack(media *Media, codec *Codec, track *Receiver) error
|
AddTrack(media *Media, codec *Codec, track *Receiver) error
|
||||||
|
|
||||||
|
// Deprecated: rename to Close()
|
||||||
Stop() error
|
Stop() error
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -102,3 +106,60 @@ const (
|
|||||||
UnsupportedCodec = "unsupported codec"
|
UnsupportedCodec = "unsupported codec"
|
||||||
WrongMediaDirection = "wrong media direction"
|
WrongMediaDirection = "wrong media direction"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type SuperProducer struct {
|
||||||
|
Type string `json:"type,omitempty"`
|
||||||
|
URL string `json:"url,omitempty"`
|
||||||
|
Medias []*Media `json:"medias,omitempty"`
|
||||||
|
Receivers []*Receiver `json:"receivers,omitempty"`
|
||||||
|
Recv int `json:"recv,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SuperProducer) GetMedias() []*Media {
|
||||||
|
return s.Medias
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SuperProducer) GetTrack(media *Media, codec *Codec) (*Receiver, error) {
|
||||||
|
for _, receiver := range s.Receivers {
|
||||||
|
if receiver.Codec == codec {
|
||||||
|
return receiver, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
receiver := NewReceiver(media, codec)
|
||||||
|
s.Receivers = append(s.Receivers, receiver)
|
||||||
|
return receiver, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SuperProducer) Close() error {
|
||||||
|
for _, receiver := range s.Receivers {
|
||||||
|
receiver.Close()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type SuperConsumer struct {
|
||||||
|
Type string `json:"type,omitempty"`
|
||||||
|
URL string `json:"url,omitempty"`
|
||||||
|
Medias []*Media `json:"medias,omitempty"`
|
||||||
|
Senders []*Sender `json:"receivers,omitempty"`
|
||||||
|
Send int `json:"recv,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SuperConsumer) GetMedias() []*Media {
|
||||||
|
return s.Medias
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SuperConsumer) AddTrack(media *Media, codec *Codec, track *Receiver) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
//func (b *SuperConsumer) WriteTo(w io.Writer) (n int64, err error) {
|
||||||
|
// return 0, nil
|
||||||
|
//}
|
||||||
|
|
||||||
|
func (s *SuperConsumer) Close() error {
|
||||||
|
for _, sender := range s.Senders {
|
||||||
|
sender.Close()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
@@ -12,13 +12,13 @@ const (
|
|||||||
BufferDrainAndClear = -1
|
BufferDrainAndClear = -1
|
||||||
)
|
)
|
||||||
|
|
||||||
// ReadSeeker support buffering and Seek over buffer
|
// ReadBuffer support buffering and Seek over buffer
|
||||||
// positive BufferSize will enable buffering mode
|
// positive BufferSize will enable buffering mode
|
||||||
// Seek to negative offset will clear buffer
|
// Seek to negative offset will clear buffer
|
||||||
// Seek with a positive BufferSize will continue buffering after the last read from the buffer
|
// Seek with a positive BufferSize will continue buffering after the last read from the buffer
|
||||||
// Seek with a negative BufferSize will clear buffer after the last read from the buffer
|
// Seek with a negative BufferSize will clear buffer after the last read from the buffer
|
||||||
// Read more than BufferSize will raise error
|
// Read more than BufferSize will raise error
|
||||||
type ReadSeeker struct {
|
type ReadBuffer struct {
|
||||||
io.Reader
|
io.Reader
|
||||||
|
|
||||||
BufferSize int
|
BufferSize int
|
||||||
@@ -27,14 +27,14 @@ type ReadSeeker struct {
|
|||||||
pos int
|
pos int
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewReadSeeker(rd io.Reader) *ReadSeeker {
|
func NewReadBuffer(rd io.Reader) *ReadBuffer {
|
||||||
if rs, ok := rd.(*ReadSeeker); ok {
|
if rs, ok := rd.(*ReadBuffer); ok {
|
||||||
return rs
|
return rs
|
||||||
}
|
}
|
||||||
return &ReadSeeker{Reader: rd}
|
return &ReadBuffer{Reader: rd}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ReadSeeker) Read(p []byte) (n int, err error) {
|
func (r *ReadBuffer) Read(p []byte) (n int, err error) {
|
||||||
// with zero buffer - read as usual
|
// with zero buffer - read as usual
|
||||||
if r.BufferSize == BufferDisable {
|
if r.BufferSize == BufferDisable {
|
||||||
return r.Reader.Read(p)
|
return r.Reader.Read(p)
|
||||||
@@ -65,7 +65,14 @@ func (r *ReadSeeker) Read(p []byte) (n int, err error) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ReadSeeker) Seek(offset int64, whence int) (int64, error) {
|
func (r *ReadBuffer) Close() error {
|
||||||
|
if closer, ok := r.Reader.(io.Closer); ok {
|
||||||
|
return closer.Close()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ReadBuffer) Seek(offset int64, whence int) (int64, error) {
|
||||||
var pos int
|
var pos int
|
||||||
switch whence {
|
switch whence {
|
||||||
case io.SeekStart:
|
case io.SeekStart:
|
||||||
@@ -89,17 +96,17 @@ func (r *ReadSeeker) Seek(offset int64, whence int) (int64, error) {
|
|||||||
return int64(r.pos), nil
|
return int64(r.pos), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ReadSeeker) Peek(n int) ([]byte, error) {
|
func (r *ReadBuffer) Peek(n int) ([]byte, error) {
|
||||||
r.BufferSize = n
|
r.BufferSize = n
|
||||||
b := make([]byte, n)
|
b := make([]byte, n)
|
||||||
if _, err := io.ReadAtLeast(r, b, n); err != nil {
|
if _, err := io.ReadAtLeast(r, b, n); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
r.Rewind()
|
r.Reset()
|
||||||
return b, nil
|
return b, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *ReadSeeker) Rewind() {
|
func (r *ReadBuffer) Reset() {
|
||||||
r.BufferSize = BufferDrainAndClear
|
r.BufferSize = BufferDrainAndClear
|
||||||
r.pos = 0
|
r.pos = 0
|
||||||
}
|
}
|
@@ -12,7 +12,7 @@ func TestReadSeeker(t *testing.T) {
|
|||||||
b := []byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
|
b := []byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
|
||||||
buf := bytes.NewReader(b)
|
buf := bytes.NewReader(b)
|
||||||
|
|
||||||
rd := NewReadSeeker(buf)
|
rd := NewReadBuffer(buf)
|
||||||
rd.BufferSize = ProbeSize
|
rd.BufferSize = ProbeSize
|
||||||
|
|
||||||
// 1. Read to buffer
|
// 1. Read to buffer
|
@@ -18,7 +18,7 @@ const Signature = "FLV"
|
|||||||
type Client struct {
|
type Client struct {
|
||||||
URL string
|
URL string
|
||||||
|
|
||||||
rd *core.ReadSeeker
|
rd *core.ReadBuffer
|
||||||
|
|
||||||
medias []*core.Media
|
medias []*core.Media
|
||||||
receivers []*core.Receiver
|
receivers []*core.Receiver
|
||||||
@@ -30,7 +30,7 @@ type Client struct {
|
|||||||
|
|
||||||
func Open(rd io.Reader) (*Client, error) {
|
func Open(rd io.Reader) (*Client, error) {
|
||||||
client := &Client{
|
client := &Client{
|
||||||
rd: core.NewReadSeeker(rd),
|
rd: core.NewReadBuffer(rd),
|
||||||
}
|
}
|
||||||
if err := client.describe(); err != nil {
|
if err := client.describe(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@@ -53,7 +53,7 @@ func (c *Client) describe() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
c.rd.BufferSize = core.ProbeSize
|
c.rd.BufferSize = core.ProbeSize
|
||||||
defer c.rd.Rewind()
|
defer c.rd.Reset()
|
||||||
|
|
||||||
// Normal software sends:
|
// Normal software sends:
|
||||||
// 1. Video/audio flag in header
|
// 1. Video/audio flag in header
|
||||||
|
@@ -70,8 +70,10 @@ func EncodeToAVCC(b []byte, safeAppend bool) []byte {
|
|||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func DecodeAVCC(b []byte) []byte {
|
func DecodeAVCC(b []byte, safeClone bool) []byte {
|
||||||
b = bytes.Clone(b)
|
if safeClone {
|
||||||
|
b = bytes.Clone(b)
|
||||||
|
}
|
||||||
for i := 0; i < len(b); {
|
for i := 0; i < len(b); {
|
||||||
size := int(binary.BigEndian.Uint32(b[i:]))
|
size := int(binary.BigEndian.Uint32(b[i:]))
|
||||||
b[i] = 0
|
b[i] = 0
|
||||||
|
@@ -2,7 +2,6 @@ package bitstream
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
"errors"
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
@@ -13,17 +12,13 @@ import (
|
|||||||
"github.com/pion/rtp"
|
"github.com/pion/rtp"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Client struct {
|
type Producer struct {
|
||||||
rd *core.ReadSeeker
|
core.SuperProducer
|
||||||
|
rd *core.ReadBuffer
|
||||||
media *core.Media
|
|
||||||
receiver *core.Receiver
|
|
||||||
|
|
||||||
recv int
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func Open(r io.Reader) (*Client, error) {
|
func Open(r io.Reader) (*Producer, error) {
|
||||||
rd := core.NewReadSeeker(r)
|
rd := core.NewReadBuffer(r)
|
||||||
|
|
||||||
buf, err := rd.Peek(256)
|
buf, err := rd.Peek(256)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -43,30 +38,19 @@ func Open(r io.Reader) (*Client, error) {
|
|||||||
return nil, errors.New("bitstream: unsupported header: " + hex.EncodeToString(buf[:8]))
|
return nil, errors.New("bitstream: unsupported header: " + hex.EncodeToString(buf[:8]))
|
||||||
}
|
}
|
||||||
|
|
||||||
client := &Client{
|
prod := &Producer{rd: rd}
|
||||||
rd: rd,
|
prod.Type = "Bitstream producer"
|
||||||
media: &core.Media{
|
prod.Medias = []*core.Media{
|
||||||
|
{
|
||||||
Kind: core.KindVideo,
|
Kind: core.KindVideo,
|
||||||
Direction: core.DirectionRecvonly,
|
Direction: core.DirectionRecvonly,
|
||||||
Codecs: []*core.Codec{codec},
|
Codecs: []*core.Codec{codec},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
return prod, nil
|
||||||
return client, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) GetMedias() []*core.Media {
|
func (c *Producer) Start() error {
|
||||||
return []*core.Media{c.media}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Client) GetTrack(media *core.Media, codec *core.Codec) (*core.Receiver, error) {
|
|
||||||
if c.receiver == nil {
|
|
||||||
c.receiver = core.NewReceiver(media, codec)
|
|
||||||
}
|
|
||||||
return c.receiver, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Client) Start() error {
|
|
||||||
var buf []byte
|
var buf []byte
|
||||||
|
|
||||||
b := make([]byte, core.BufferSize)
|
b := make([]byte, core.BufferSize)
|
||||||
@@ -76,7 +60,7 @@ func (c *Client) Start() error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
c.recv += n
|
c.Recv += n
|
||||||
|
|
||||||
buf = append(buf, b[:n]...)
|
buf = append(buf, b[:n]...)
|
||||||
|
|
||||||
@@ -89,7 +73,7 @@ func (c *Client) Start() error {
|
|||||||
Header: rtp.Header{Timestamp: core.Now90000()},
|
Header: rtp.Header{Timestamp: core.Now90000()},
|
||||||
Payload: annexb.EncodeToAVCC(buf[:i], true),
|
Payload: annexb.EncodeToAVCC(buf[:i], true),
|
||||||
}
|
}
|
||||||
c.receiver.WriteRTP(pkt)
|
c.Receivers[0].WriteRTP(pkt)
|
||||||
|
|
||||||
//log.Printf("[AVC] %v, len: %d", h264.Types(pkt.Payload), len(pkt.Payload))
|
//log.Printf("[AVC] %v, len: %d", h264.Types(pkt.Payload), len(pkt.Payload))
|
||||||
|
|
||||||
@@ -97,22 +81,7 @@ func (c *Client) Start() error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) Stop() error {
|
func (c *Producer) Stop() error {
|
||||||
if c.receiver != nil {
|
_ = c.SuperProducer.Close()
|
||||||
c.receiver.Close()
|
return c.rd.Close()
|
||||||
}
|
|
||||||
if closer, ok := c.rd.Reader.(io.Closer); ok {
|
|
||||||
return closer.Close()
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Client) MarshalJSON() ([]byte, error) {
|
|
||||||
info := &core.Info{
|
|
||||||
Type: "Bitstream active producer",
|
|
||||||
Medias: []*core.Media{c.media},
|
|
||||||
Receivers: []*core.Receiver{c.receiver},
|
|
||||||
Recv: c.recv,
|
|
||||||
}
|
|
||||||
return json.Marshal(info)
|
|
||||||
}
|
}
|
@@ -43,7 +43,7 @@ func (k *Keyframe) AddTrack(media *core.Media, _ *core.Codec, track *core.Receiv
|
|||||||
if !h264.IsKeyframe(packet.Payload) {
|
if !h264.IsKeyframe(packet.Payload) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
b := annexb.DecodeAVCC(packet.Payload)
|
b := annexb.DecodeAVCC(packet.Payload, true)
|
||||||
k.Fire(b)
|
k.Fire(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -2,26 +2,22 @@ package mjpeg
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/json"
|
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
"github.com/AlexxIT/go2rtc/pkg/core"
|
"github.com/AlexxIT/go2rtc/pkg/core"
|
||||||
"github.com/pion/rtp"
|
"github.com/pion/rtp"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Client struct {
|
type Producer struct {
|
||||||
rd *core.ReadSeeker
|
core.SuperProducer
|
||||||
|
rd *core.ReadBuffer
|
||||||
media *core.Media
|
|
||||||
receiver *core.Receiver
|
|
||||||
|
|
||||||
recv int
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewClient(rd io.Reader) *Client {
|
func Open(rd io.Reader) (*Producer, error) {
|
||||||
return &Client{
|
prod := &Producer{rd: core.NewReadBuffer(rd)}
|
||||||
rd: core.NewReadSeeker(rd),
|
prod.Type = "MJPEG producer"
|
||||||
media: &core.Media{
|
prod.Medias = []*core.Media{
|
||||||
|
{
|
||||||
Kind: core.KindVideo,
|
Kind: core.KindVideo,
|
||||||
Direction: core.DirectionRecvonly,
|
Direction: core.DirectionRecvonly,
|
||||||
Codecs: []*core.Codec{
|
Codecs: []*core.Codec{
|
||||||
@@ -33,20 +29,10 @@ func NewClient(rd io.Reader) *Client {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
return prod, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) GetMedias() []*core.Media {
|
func (c *Producer) Start() error {
|
||||||
return []*core.Media{c.media}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Client) GetTrack(media *core.Media, codec *core.Codec) (*core.Receiver, error) {
|
|
||||||
if c.receiver == nil {
|
|
||||||
c.receiver = core.NewReceiver(media, codec)
|
|
||||||
}
|
|
||||||
return c.receiver, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Client) Start() error {
|
|
||||||
var buf []byte // total bufer
|
var buf []byte // total bufer
|
||||||
b := make([]byte, core.BufferSize) // reading buffer
|
b := make([]byte, core.BufferSize) // reading buffer
|
||||||
|
|
||||||
@@ -59,7 +45,7 @@ func (c *Client) Start() error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
c.recv += n
|
c.Recv += n
|
||||||
|
|
||||||
buf = append(buf, b[:n]...)
|
buf = append(buf, b[:n]...)
|
||||||
|
|
||||||
@@ -77,7 +63,7 @@ func (c *Client) Start() error {
|
|||||||
Header: rtp.Header{Timestamp: core.Now90000()},
|
Header: rtp.Header{Timestamp: core.Now90000()},
|
||||||
Payload: buf[:i],
|
Payload: buf[:i],
|
||||||
}
|
}
|
||||||
c.receiver.WriteRTP(pkt)
|
c.Receivers[0].WriteRTP(pkt)
|
||||||
|
|
||||||
//log.Printf("[mjpeg] ts=%d size=%d", pkt.Header.Timestamp, len(pkt.Payload))
|
//log.Printf("[mjpeg] ts=%d size=%d", pkt.Header.Timestamp, len(pkt.Payload))
|
||||||
|
|
||||||
@@ -85,22 +71,7 @@ func (c *Client) Start() error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) Stop() error {
|
func (c *Producer) Stop() error {
|
||||||
if c.receiver != nil {
|
_ = c.SuperProducer.Close()
|
||||||
c.receiver.Close()
|
return c.rd.Close()
|
||||||
}
|
|
||||||
if closer, ok := c.rd.Reader.(io.Closer); ok {
|
|
||||||
return closer.Close()
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Client) MarshalJSON() ([]byte, error) {
|
|
||||||
info := &core.Info{
|
|
||||||
Type: "MJPEG active producer",
|
|
||||||
Medias: []*core.Media{c.media},
|
|
||||||
Receivers: []*core.Receiver{c.receiver},
|
|
||||||
Recv: c.recv,
|
|
||||||
}
|
|
||||||
return json.Marshal(info)
|
|
||||||
}
|
}
|
@@ -15,7 +15,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func Open(r io.Reader) (core.Producer, error) {
|
func Open(r io.Reader) (core.Producer, error) {
|
||||||
rd := core.NewReadSeeker(r)
|
rd := core.NewReadBuffer(r)
|
||||||
|
|
||||||
b, err := rd.Peek(4)
|
b, err := rd.Peek(4)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -27,7 +27,7 @@ func Open(r io.Reader) (core.Producer, error) {
|
|||||||
return bitstream.Open(rd)
|
return bitstream.Open(rd)
|
||||||
|
|
||||||
case bytes.HasPrefix(b, []byte{0xFF, 0xD8}):
|
case bytes.HasPrefix(b, []byte{0xFF, 0xD8}):
|
||||||
return mjpeg.NewClient(rd), nil
|
return mjpeg.Open(rd)
|
||||||
|
|
||||||
case bytes.HasPrefix(b, []byte(flv.Signature)):
|
case bytes.HasPrefix(b, []byte(flv.Signature)):
|
||||||
return flv.Open(rd)
|
return flv.Open(rd)
|
@@ -14,7 +14,7 @@ import (
|
|||||||
type Client struct {
|
type Client struct {
|
||||||
URL string
|
URL string
|
||||||
|
|
||||||
rd *core.ReadSeeker
|
rd *core.ReadBuffer
|
||||||
|
|
||||||
medias []*core.Media
|
medias []*core.Media
|
||||||
receivers []*core.Receiver
|
receivers []*core.Receiver
|
||||||
@@ -23,7 +23,7 @@ type Client struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func Open(rd io.Reader) (*Client, error) {
|
func Open(rd io.Reader) (*Client, error) {
|
||||||
client := &Client{rd: core.NewReadSeeker(rd)}
|
client := &Client{rd: core.NewReadBuffer(rd)}
|
||||||
if err := client.describe(); err != nil {
|
if err := client.describe(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -32,7 +32,7 @@ func Open(rd io.Reader) (*Client, error) {
|
|||||||
|
|
||||||
func (c *Client) describe() error {
|
func (c *Client) describe() error {
|
||||||
c.rd.BufferSize = core.ProbeSize
|
c.rd.BufferSize = core.ProbeSize
|
||||||
defer c.rd.Rewind()
|
defer c.rd.Reset()
|
||||||
|
|
||||||
rd := NewReader()
|
rd := NewReader()
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user