Files
engine/subscriber.go
2020-08-06 22:31:22 +08:00

109 lines
2.5 KiB
Go

package engine
import (
"context"
"time"
"github.com/Monibuca/engine/v2/avformat"
"github.com/pkg/errors"
)
// SubscriberInfo 订阅者可序列化信息,用于控制台输出
type SubscriberInfo struct {
ID string
TotalDrop int //总丢帧
TotalPacket int
Type string
BufferLength int
Delay uint32
SubscribeTime time.Time
}
// Subscriber 订阅者实体定义
type Subscriber struct {
context.Context
*Stream
SubscriberInfo
OnData func(*avformat.SendPacket) error
Cancel context.CancelFunc
Sign string
OffsetTime uint32
startTime uint32
avformat.SendPacket
}
// IsClosed 检查订阅者是否已经关闭
func (s *Subscriber) IsClosed() bool {
return s.Context != nil && s.Err() != nil
}
// Close 关闭订阅者
func (s *Subscriber) Close() {
if s.Cancel != nil {
s.Cancel()
}
}
//Subscribe 开始订阅
func (s *Subscriber) Subscribe(streamPath string) (err error) {
if !config.EnableWaitStream {
if _, ok := streamCollection.Load(streamPath); !ok {
return errors.Errorf("Stream not found:%s", streamPath)
}
}
GetStream(streamPath).Subscribe(s)
if s.Context == nil {
return errors.Errorf("stream not exist:%s", streamPath)
}
defer s.UnSubscribe(s)
select {
//等待发布者首屏数据,如果发布者尚为发布,则会等待,否则就会往下执行
case <-s.WaitPub:
case <-s.Context.Done():
return s.Err()
}
s.sendAv(s.VideoTag, 0)
packet := s.FirstScreen.Clone()
s.startTime = packet.Timestamp
s.send(packet)
packet.GoNext()
for atsent, dropping, droped := false, false, 0; s.Err() == nil; packet.GoNext() {
s.TotalPacket++
if !dropping {
if packet.Type == avformat.FLV_TAG_TYPE_AUDIO && !atsent {
s.sendAv(s.AudioTag, 0)
atsent = true
}
s.send(packet)
if s.checkDrop(packet) {
dropping = true
droped = 0
}
} else if packet.IsKeyFrame {
//遇到关键帧则退出丢帧
dropping = false
//fmt.Println("drop package ", droped)
s.TotalDrop += droped
s.send(packet)
} else {
droped++
}
}
return s.Err()
}
func (s *Subscriber) sendAv(packet *avformat.AVPacket, t uint32) {
s.AVPacket = packet
s.Timestamp = t
s.OnData(&s.SendPacket)
}
func (s *Subscriber) send(packet *Ring) {
packet.Wait()
s.sendAv(&packet.AVPacket, packet.Timestamp-s.startTime)
}
func (s *Subscriber) checkDrop(packet *Ring) bool {
pIndex := s.AVRing.Index
s.BufferLength = pIndex - packet.Index
s.Delay = s.AVRing.Timestamp - packet.Timestamp
return s.BufferLength > s.AVRing.Size/2
}