You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
lal/pkg/logic/group.go

316 lines
9.1 KiB
Go

// Copyright 2019, Chef. All rights reserved.
// https://github.com/q191201771/lal
//
// Use of this source code is governed by a MIT-style license
// that can be found in the License file.
//
// Author: Chef (191201771@qq.com)
package logic
import (
"sync"
"github.com/q191201771/lal/pkg/httpflv"
"github.com/q191201771/lal/pkg/rtmp"
log "github.com/q191201771/naza/pkg/nazalog"
"github.com/q191201771/naza/pkg/unique"
)
type Group struct {
UniqueKey string
appName string
streamName string
exitChan chan struct{}
mutex sync.Mutex
pubSession *rtmp.ServerSession
pullSession *rtmp.PullSession
rtmpSubSessionSet map[*rtmp.ServerSession]struct{}
httpflvSubSessionSet map[*httpflv.SubSession]struct{}
// rtmp chunk格式
metadata []byte
avcKeySeqHeader []byte
aacSeqHeader []byte
// httpflv tag格式
// TODO chef: 如果没有开启httpflv监听可以不做格式转换节约CPU资源
metadataTag *httpflv.Tag
avcKeySeqHeaderTag *httpflv.Tag
aacSeqHeaderTag *httpflv.Tag
}
var _ rtmp.PubSessionObserver = &Group{}
func NewGroup(appName string, streamName string) *Group {
uk := unique.GenUniqueKey("GROUP")
log.Infof("lifecycle new group. [%s] appName=%s, streamName=%s", uk, appName, streamName)
return &Group{
UniqueKey: uk,
appName: appName,
streamName: streamName,
exitChan: make(chan struct{}, 1),
rtmpSubSessionSet: make(map[*rtmp.ServerSession]struct{}),
httpflvSubSessionSet: make(map[*httpflv.SubSession]struct{}),
}
}
func (group *Group) RunLoop() {
<-group.exitChan
}
func (group *Group) Dispose(err error) {
log.Infof("lifecycle dispose group. [%s]", group.UniqueKey)
group.exitChan <- struct{}{}
group.mutex.Lock()
defer group.mutex.Unlock()
if group.pubSession != nil {
group.pubSession.Dispose()
}
for session := range group.rtmpSubSessionSet {
session.Dispose()
}
for session := range group.httpflvSubSessionSet {
session.Dispose()
}
}
func (group *Group) AddRTMPPubSession(session *rtmp.ServerSession) bool {
log.Debugf("add PubSession into group. [%s] [%s]", group.UniqueKey, session.UniqueKey)
group.mutex.Lock()
5 years ago
if group.pubSession != nil {
log.Errorf("PubSession already exist in group. [%s] old=%s, new=%s", group.UniqueKey, group.pubSession.UniqueKey, session.UniqueKey)
return false
5 years ago
}
group.pubSession = session
group.mutex.Unlock()
session.SetPubSessionObserver(group)
return true
}
func (group *Group) DelRTMPPubSession(session *rtmp.ServerSession) {
log.Debugf("del PubSession from group. [%s] [%s]", group.UniqueKey, session.UniqueKey)
group.mutex.Lock()
defer group.mutex.Unlock()
group.pubSession = nil
group.metadata = nil
group.avcKeySeqHeader = nil
group.aacSeqHeader = nil
group.metadataTag = nil
group.avcKeySeqHeaderTag = nil
group.aacSeqHeaderTag = nil
}
func (group *Group) AddRTMPSubSession(session *rtmp.ServerSession) {
log.Debugf("add SubSession into group. [%s] [%s]", group.UniqueKey, session.UniqueKey)
group.mutex.Lock()
defer group.mutex.Unlock()
group.rtmpSubSessionSet[session] = struct{}{}
// TODO chef: 多长没有拉流session存在的功能
//group.turnToEmptyTick = 0
}
func (group *Group) DelRTMPSubSession(session *rtmp.ServerSession) {
log.Debugf("del SubSession from group. [%s] [%s]", group.UniqueKey, session.UniqueKey)
group.mutex.Lock()
defer group.mutex.Unlock()
delete(group.rtmpSubSessionSet, session)
}
func (group *Group) AddHTTPFLVSubSession(session *httpflv.SubSession) {
log.Debugf("add httpflv SubSession into group. [%s] [%s]", group.UniqueKey, session.UniqueKey)
session.WriteHTTPResponseHeader()
session.WriteFLVHeader()
group.mutex.Lock()
defer group.mutex.Unlock()
group.httpflvSubSessionSet[session] = struct{}{}
}
func (group *Group) DelHTTPFLVSubSession(session *httpflv.SubSession) {
log.Debugf("del httpflv SubSession from group. [%s] [%s]", group.UniqueKey, session.UniqueKey)
group.mutex.Lock()
defer group.mutex.Unlock()
delete(group.httpflvSubSessionSet, session)
}
func (group *Group) IsTotalEmpty() bool {
group.mutex.Lock()
defer group.mutex.Unlock()
return group.pubSession == nil && len(group.rtmpSubSessionSet) == 0 && len(group.httpflvSubSessionSet) == 0
}
func (group *Group) IsInExist() bool {
group.mutex.Lock()
defer group.mutex.Unlock()
return group.pubSession != nil
}
// PubSession or PullSession
func (group *Group) ReadRTMPAVMsgCB(header rtmp.Header, timestampAbs uint32, message []byte) {
group.mutex.Lock()
defer group.mutex.Unlock()
group.broadcastRTMP(header, timestampAbs, message)
}
func (group *Group) broadcastRTMP(header rtmp.Header, timestampAbs uint32, message []byte) {
5 years ago
//log.Infof("%+v", header)
var (
currHeader rtmp.Header
absChunks []byte
currTag *httpflv.Tag
)
// # 1. 设置好 rtmp 头部信息
currHeader.MsgLen = uint32(len(message))
currHeader.Timestamp = timestampAbs
currHeader.MsgTypeID = header.MsgTypeID
currHeader.MsgStreamID = rtmp.MSID1
switch header.MsgTypeID {
case rtmp.TypeidDataMessageAMF0:
currHeader.CSID = rtmp.CSIDAMF
//prevHeader = nil
case rtmp.TypeidAudio:
currHeader.CSID = rtmp.CSIDAudio
//prevHeader = group.prevAudioHeader
case rtmp.TypeidVideo:
currHeader.CSID = rtmp.CSIDVideo
//prevHeader = group.prevVideoHeader
}
// # 2. 广播。遍历所有rtmp sub session决定是否转发
for session := range group.rtmpSubSessionSet {
// ## 2.1. 一个message广播给多个sub session时只做一次chunk切割
if absChunks == nil {
absChunks = rtmp.Message2Chunks(message, &currHeader)
}
// ## 2.2. 如果是新的sub session发送已缓存的信息
if session.IsFresh {
// 发送缓存的头部信息
if group.metadata != nil {
session.AsyncWrite(group.metadata)
}
if group.avcKeySeqHeader != nil {
session.AsyncWrite(group.avcKeySeqHeader)
}
if group.aacSeqHeader != nil {
session.AsyncWrite(group.aacSeqHeader)
}
session.IsFresh = false
}
// ## 2.3. 判断当前包的类型以及sub session的状态决定是否发送并更新sub session的状态
switch header.MsgTypeID {
case rtmp.TypeidDataMessageAMF0:
session.AsyncWrite(absChunks)
case rtmp.TypeidAudio:
session.AsyncWrite(absChunks)
case rtmp.TypeidVideo:
if session.WaitKeyNalu {
if message[0] == 0x17 && message[1] == 0x0 {
session.AsyncWrite(absChunks)
}
if message[0] == 0x17 && message[1] == 0x1 {
session.AsyncWrite(absChunks)
session.WaitKeyNalu = false
}
} else {
session.AsyncWrite(absChunks)
}
}
}
// # 3. 广播。遍历所有httpflv sub session决定是否转发
for session := range group.httpflvSubSessionSet {
// ## 3.1. 将当前 message 转换成 tag 格式
if currTag == nil {
currTag = Trans.RTMPMsg2FLVTag(header, timestampAbs, message)
}
// ## 3.2. 如果是新的sub session发送已缓存的信息
if session.IsFresh {
// 发送缓存的头部信息
if group.metadataTag != nil {
log.Debugf("send cache metadata. [%s]", session.UniqueKey)
session.WriteTag(group.metadataTag)
}
if group.avcKeySeqHeaderTag != nil {
session.WriteTag(group.avcKeySeqHeaderTag)
}
if group.aacSeqHeaderTag != nil {
session.WriteTag(group.aacSeqHeaderTag)
}
session.IsFresh = false
}
// ## 3.3. 判断当前包的类型以及sub session的状态决定是否发送并更新sub session的状态
switch header.MsgTypeID {
case rtmp.TypeidDataMessageAMF0:
session.WriteTag(currTag)
case rtmp.TypeidAudio:
session.WriteTag(currTag)
case rtmp.TypeidVideo:
if session.WaitKeyNalu {
if message[0] == 0x17 && message[1] == 0x0 {
session.WriteTag(currTag)
}
if message[0] == 0x17 && message[1] == 0x1 {
session.WriteTag(currTag)
session.WaitKeyNalu = false
}
} else {
session.WriteTag(currTag)
}
}
}
// # 4. 缓存 rtmp 以及 httpflv 的 metadata 和 avc key seq header 和 aac seq header
// 由于可能没有订阅者,所以可能需要重新打包
switch header.MsgTypeID {
case rtmp.TypeidDataMessageAMF0:
if absChunks == nil {
absChunks = rtmp.Message2Chunks(message, &currHeader)
}
if currTag == nil {
currTag = Trans.RTMPMsg2FLVTag(header, timestampAbs, message)
}
group.metadata = absChunks
group.metadataTag = currTag
log.Debugf("cache metadata. [%s]", group.UniqueKey)
case rtmp.TypeidVideo:
// TODO chef: magic number
if message[0] == 0x17 && message[1] == 0x0 {
if absChunks == nil {
absChunks = rtmp.Message2Chunks(message, &currHeader)
}
if currTag == nil {
currTag = Trans.RTMPMsg2FLVTag(header, timestampAbs, message)
}
group.avcKeySeqHeader = absChunks
group.avcKeySeqHeaderTag = currTag
log.Debugf("cache avc key seq header. [%s]", group.UniqueKey)
}
case rtmp.TypeidAudio:
if (message[0]>>4) == 0x0a && message[1] == 0x0 {
if absChunks == nil {
absChunks = rtmp.Message2Chunks(message, &currHeader)
}
if currTag == nil {
currTag = Trans.RTMPMsg2FLVTag(header, timestampAbs, message)
}
group.aacSeqHeader = absChunks
group.aacSeqHeaderTag = currTag
log.Debugf("cache aac seq header. [%s]", group.UniqueKey)
}
}
}