Rewrite JPEG snapshot consumer
This commit is contained in:
+51
-38
@@ -1,6 +1,8 @@
|
||||
package magic
|
||||
|
||||
import (
|
||||
"io"
|
||||
|
||||
"github.com/AlexxIT/go2rtc/pkg/core"
|
||||
"github.com/AlexxIT/go2rtc/pkg/h264"
|
||||
"github.com/AlexxIT/go2rtc/pkg/h264/annexb"
|
||||
@@ -10,83 +12,94 @@ import (
|
||||
)
|
||||
|
||||
type Keyframe struct {
|
||||
core.Listener
|
||||
|
||||
UserAgent string
|
||||
RemoteAddr string
|
||||
|
||||
medias []*core.Media
|
||||
sender *core.Sender
|
||||
core.SuperConsumer
|
||||
wr *core.WriteBuffer
|
||||
}
|
||||
|
||||
func (k *Keyframe) GetMedias() []*core.Media {
|
||||
if k.medias == nil {
|
||||
k.medias = append(k.medias, &core.Media{
|
||||
Kind: core.KindVideo,
|
||||
Direction: core.DirectionSendonly,
|
||||
Codecs: []*core.Codec{
|
||||
{Name: core.CodecH264},
|
||||
{Name: core.CodecH265},
|
||||
{Name: core.CodecJPEG},
|
||||
func NewKeyframe() *Keyframe {
|
||||
return &Keyframe{
|
||||
core.SuperConsumer{
|
||||
Medias: []*core.Media{
|
||||
{
|
||||
Kind: core.KindVideo,
|
||||
Direction: core.DirectionSendonly,
|
||||
Codecs: []*core.Codec{
|
||||
{Name: core.CodecH264},
|
||||
{Name: core.CodecH265},
|
||||
{Name: core.CodecJPEG},
|
||||
},
|
||||
},
|
||||
},
|
||||
})
|
||||
},
|
||||
core.NewWriteBuffer(nil),
|
||||
}
|
||||
return k.medias
|
||||
}
|
||||
|
||||
func (k *Keyframe) AddTrack(media *core.Media, _ *core.Codec, track *core.Receiver) error {
|
||||
var handler core.HandlerFunc
|
||||
sender := core.NewSender(media, track.Codec)
|
||||
|
||||
switch track.Codec.Name {
|
||||
case core.CodecH264:
|
||||
handler = func(packet *rtp.Packet) {
|
||||
sender.Handler = func(packet *rtp.Packet) {
|
||||
if !h264.IsKeyframe(packet.Payload) {
|
||||
return
|
||||
}
|
||||
b := annexb.DecodeAVCC(packet.Payload, true)
|
||||
k.Fire(b)
|
||||
if n, err := k.wr.Write(b); err == nil {
|
||||
k.Send += n
|
||||
}
|
||||
}
|
||||
|
||||
if track.Codec.IsRTP() {
|
||||
handler = h264.RTPDepay(track.Codec, handler)
|
||||
sender.Handler = h264.RTPDepay(track.Codec, sender.Handler)
|
||||
}
|
||||
|
||||
case core.CodecH265:
|
||||
handler = func(packet *rtp.Packet) {
|
||||
sender.Handler = func(packet *rtp.Packet) {
|
||||
if !h265.IsKeyframe(packet.Payload) {
|
||||
return
|
||||
}
|
||||
k.Fire(packet.Payload)
|
||||
b := annexb.DecodeAVCC(packet.Payload, true)
|
||||
if n, err := k.wr.Write(b); err == nil {
|
||||
k.Send += n
|
||||
}
|
||||
}
|
||||
|
||||
if track.Codec.IsRTP() {
|
||||
handler = h265.RTPDepay(track.Codec, handler)
|
||||
sender.Handler = h264.RTPDepay(track.Codec, sender.Handler)
|
||||
} else {
|
||||
sender.Handler = h264.RepairAVCC(track.Codec, sender.Handler)
|
||||
}
|
||||
|
||||
case core.CodecJPEG:
|
||||
handler = func(packet *rtp.Packet) {
|
||||
k.Fire(packet.Payload)
|
||||
sender.Handler = func(packet *rtp.Packet) {
|
||||
if n, err := k.wr.Write(packet.Payload); err == nil {
|
||||
k.Send += n
|
||||
}
|
||||
}
|
||||
|
||||
if track.Codec.IsRTP() {
|
||||
handler = mjpeg.RTPDepay(handler)
|
||||
sender.Handler = mjpeg.RTPDepay(sender.Handler)
|
||||
}
|
||||
}
|
||||
|
||||
k.sender = core.NewSender(media, track.Codec)
|
||||
k.sender.Handler = handler
|
||||
k.sender.HandleRTP(track)
|
||||
sender.HandleRTP(track)
|
||||
k.Senders = append(k.Senders, sender)
|
||||
return nil
|
||||
}
|
||||
|
||||
func (k *Keyframe) CodecName() string {
|
||||
if k.sender != nil {
|
||||
return k.sender.Codec.Name
|
||||
if len(k.Senders) != 1 {
|
||||
return ""
|
||||
}
|
||||
return ""
|
||||
return k.Senders[0].Codec.Name
|
||||
}
|
||||
|
||||
func (k *Keyframe) WriteTo(wr io.Writer) (int64, error) {
|
||||
return k.wr.WriteTo(wr)
|
||||
}
|
||||
|
||||
func (k *Keyframe) Stop() error {
|
||||
if k.sender != nil {
|
||||
k.sender.Close()
|
||||
}
|
||||
return nil
|
||||
_ = k.SuperConsumer.Close()
|
||||
return k.wr.Close()
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user