mirror of
https://github.com/AlexxIT/go2rtc.git
synced 2025-10-06 00:37:00 +08:00
Rewrite MJPEG consumer
This commit is contained in:
@@ -80,8 +80,6 @@ func handlerKeyframe(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const header = "--frame\r\nContent-Type: image/jpeg\r\nContent-Length: "
|
|
||||||
|
|
||||||
func handlerStream(w http.ResponseWriter, r *http.Request) {
|
func handlerStream(w http.ResponseWriter, r *http.Request) {
|
||||||
if r.Method != "POST" {
|
if r.Method != "POST" {
|
||||||
outputMjpeg(w, r)
|
outputMjpeg(w, r)
|
||||||
@@ -98,26 +96,10 @@ func outputMjpeg(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
flusher := w.(http.Flusher)
|
|
||||||
|
|
||||||
cons := &mjpeg.Consumer{
|
cons := &mjpeg.Consumer{
|
||||||
RemoteAddr: tcp.RemoteAddr(r),
|
RemoteAddr: tcp.RemoteAddr(r),
|
||||||
UserAgent: r.UserAgent(),
|
UserAgent: r.UserAgent(),
|
||||||
}
|
}
|
||||||
cons.Listen(func(msg any) {
|
|
||||||
switch msg := msg.(type) {
|
|
||||||
case []byte:
|
|
||||||
data := []byte(header + strconv.Itoa(len(msg)))
|
|
||||||
data = append(data, '\r', '\n', '\r', '\n')
|
|
||||||
data = append(data, msg...)
|
|
||||||
data = append(data, '\r', '\n')
|
|
||||||
|
|
||||||
// Chrome bug: mjpeg image always shows the second to last image
|
|
||||||
// https://bugs.chromium.org/p/chromium/issues/detail?id=527446
|
|
||||||
_, _ = w.Write(data)
|
|
||||||
flusher.Flush()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
if err := stream.AddConsumer(cons); err != nil {
|
if err := stream.AddConsumer(cons); err != nil {
|
||||||
log.Error().Err(err).Msg("[api.mjpeg] add consumer")
|
log.Error().Err(err).Msg("[api.mjpeg] add consumer")
|
||||||
@@ -130,11 +112,33 @@ func outputMjpeg(w http.ResponseWriter, r *http.Request) {
|
|||||||
h.Set("Connection", "close")
|
h.Set("Connection", "close")
|
||||||
h.Set("Pragma", "no-cache")
|
h.Set("Pragma", "no-cache")
|
||||||
|
|
||||||
<-r.Context().Done()
|
wr := &writer{wr: w, buf: []byte(header)}
|
||||||
|
_, _ = cons.WriteTo(wr)
|
||||||
|
|
||||||
stream.RemoveConsumer(cons)
|
stream.RemoveConsumer(cons)
|
||||||
|
}
|
||||||
|
|
||||||
//log.Trace().Msg("[api.mjpeg] close")
|
const header = "--frame\r\nContent-Type: image/jpeg\r\nContent-Length: "
|
||||||
|
|
||||||
|
type writer struct {
|
||||||
|
wr io.Writer
|
||||||
|
buf []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *writer) Write(p []byte) (n int, err error) {
|
||||||
|
w.buf = w.buf[:len(header)]
|
||||||
|
w.buf = append(w.buf, strconv.Itoa(len(p))...)
|
||||||
|
w.buf = append(w.buf, "\r\n\r\n"...)
|
||||||
|
w.buf = append(w.buf, p...)
|
||||||
|
w.buf = append(w.buf, "\r\n"...)
|
||||||
|
|
||||||
|
// Chrome bug: mjpeg image always shows the second to last image
|
||||||
|
// https://bugs.chromium.org/p/chromium/issues/detail?id=527446
|
||||||
|
if n, err = w.wr.Write(w.buf); err == nil {
|
||||||
|
w.wr.(http.Flusher).Flush()
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func inputMjpeg(w http.ResponseWriter, r *http.Request) {
|
func inputMjpeg(w http.ResponseWriter, r *http.Request) {
|
||||||
@@ -168,11 +172,6 @@ func handlerWS(tr *ws.Transport, _ *ws.Message) error {
|
|||||||
RemoteAddr: tcp.RemoteAddr(tr.Request),
|
RemoteAddr: tcp.RemoteAddr(tr.Request),
|
||||||
UserAgent: tr.Request.UserAgent(),
|
UserAgent: tr.Request.UserAgent(),
|
||||||
}
|
}
|
||||||
cons.Listen(func(msg any) {
|
|
||||||
if data, ok := msg.([]byte); ok {
|
|
||||||
tr.Write(data)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
if err := stream.AddConsumer(cons); err != nil {
|
if err := stream.AddConsumer(cons); err != nil {
|
||||||
log.Error().Err(err).Caller().Send()
|
log.Error().Err(err).Caller().Send()
|
||||||
@@ -181,9 +180,21 @@ func handlerWS(tr *ws.Transport, _ *ws.Message) error {
|
|||||||
|
|
||||||
tr.Write(&ws.Message{Type: "mjpeg"})
|
tr.Write(&ws.Message{Type: "mjpeg"})
|
||||||
|
|
||||||
|
wr := &writer2{tr: tr} // TODO: fixme
|
||||||
|
go cons.WriteTo(wr)
|
||||||
|
|
||||||
tr.OnClose(func() {
|
tr.OnClose(func() {
|
||||||
stream.RemoveConsumer(cons)
|
stream.RemoveConsumer(cons)
|
||||||
})
|
})
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type writer2 struct {
|
||||||
|
tr *ws.Transport
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *writer2) Write(p []byte) (n int, err error) {
|
||||||
|
w.tr.Write(p)
|
||||||
|
return len(p), nil
|
||||||
|
}
|
||||||
|
83
pkg/core/writebuffer.go
Normal file
83
pkg/core/writebuffer.go
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
package core
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io"
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
type WriteBuffer struct {
|
||||||
|
io.Writer
|
||||||
|
err error
|
||||||
|
mu sync.Mutex
|
||||||
|
wg sync.WaitGroup
|
||||||
|
state byte
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWriteBuffer(wr io.Writer) *WriteBuffer {
|
||||||
|
if wr == nil {
|
||||||
|
wr = bytes.NewBuffer(nil)
|
||||||
|
}
|
||||||
|
return &WriteBuffer{Writer: wr}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *WriteBuffer) Write(p []byte) (n int, err error) {
|
||||||
|
w.mu.Lock()
|
||||||
|
if w.err != nil {
|
||||||
|
err = w.err
|
||||||
|
} else if n, err = w.Writer.Write(p); err != nil {
|
||||||
|
w.err = err
|
||||||
|
w.done()
|
||||||
|
}
|
||||||
|
w.mu.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *WriteBuffer) WriteTo(wr io.Writer) (n int64, err error) {
|
||||||
|
w.Reset(wr)
|
||||||
|
w.wg.Wait()
|
||||||
|
return 0, w.err // TODO: fix counter
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *WriteBuffer) Close() error {
|
||||||
|
if closer, ok := w.Writer.(io.Closer); ok {
|
||||||
|
return closer.Close()
|
||||||
|
}
|
||||||
|
w.mu.Lock()
|
||||||
|
w.done()
|
||||||
|
w.mu.Unlock()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *WriteBuffer) Reset(wr io.Writer) {
|
||||||
|
w.mu.Lock()
|
||||||
|
w.add()
|
||||||
|
if buf, ok := wr.(*bytes.Buffer); ok {
|
||||||
|
if _, err := io.Copy(wr, buf); err != nil {
|
||||||
|
w.err = err
|
||||||
|
w.done()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
w.Writer = wr
|
||||||
|
w.mu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
none = iota
|
||||||
|
start
|
||||||
|
end
|
||||||
|
)
|
||||||
|
|
||||||
|
func (w *WriteBuffer) add() {
|
||||||
|
if w.state == none {
|
||||||
|
w.state = start
|
||||||
|
w.wg.Add(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *WriteBuffer) done() {
|
||||||
|
if w.state == start {
|
||||||
|
w.state = end
|
||||||
|
w.wg.Done()
|
||||||
|
}
|
||||||
|
}
|
@@ -2,19 +2,21 @@ package mjpeg
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"io"
|
||||||
|
|
||||||
"github.com/AlexxIT/go2rtc/pkg/core"
|
"github.com/AlexxIT/go2rtc/pkg/core"
|
||||||
"github.com/pion/rtp"
|
"github.com/pion/rtp"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Consumer struct {
|
type Consumer struct {
|
||||||
core.Listener
|
|
||||||
|
|
||||||
UserAgent string
|
UserAgent string
|
||||||
RemoteAddr string
|
RemoteAddr string
|
||||||
|
|
||||||
medias []*core.Media
|
medias []*core.Media
|
||||||
sender *core.Sender
|
sender *core.Sender
|
||||||
|
|
||||||
|
wr *core.WriteBuffer
|
||||||
|
|
||||||
send int
|
send int
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -34,11 +36,16 @@ func (c *Consumer) GetMedias() []*core.Media {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (c *Consumer) AddTrack(media *core.Media, _ *core.Codec, track *core.Receiver) error {
|
func (c *Consumer) AddTrack(media *core.Media, _ *core.Codec, track *core.Receiver) error {
|
||||||
|
if c.wr == nil {
|
||||||
|
c.wr = core.NewWriteBuffer(nil)
|
||||||
|
}
|
||||||
|
|
||||||
if c.sender == nil {
|
if c.sender == nil {
|
||||||
c.sender = core.NewSender(media, track.Codec)
|
c.sender = core.NewSender(media, track.Codec)
|
||||||
c.sender.Handler = func(packet *rtp.Packet) {
|
c.sender.Handler = func(packet *rtp.Packet) {
|
||||||
c.Fire(packet.Payload)
|
if n, err := c.wr.Write(packet.Payload); err == nil {
|
||||||
c.send += len(packet.Payload)
|
c.send += n
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if track.Codec.IsRTP() {
|
if track.Codec.IsRTP() {
|
||||||
@@ -50,10 +57,17 @@ func (c *Consumer) AddTrack(media *core.Media, _ *core.Codec, track *core.Receiv
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Consumer) WriteTo(wr io.Writer) (int64, error) {
|
||||||
|
return c.wr.WriteTo(wr)
|
||||||
|
}
|
||||||
|
|
||||||
func (c *Consumer) Stop() error {
|
func (c *Consumer) Stop() error {
|
||||||
if c.sender != nil {
|
if c.sender != nil {
|
||||||
c.sender.Close()
|
c.sender.Close()
|
||||||
}
|
}
|
||||||
|
if c.wr != nil {
|
||||||
|
_ = c.wr.Close()
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user