You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

131 lines
3.7 KiB

2 years ago
package webrtc
import (
"fmt"
"github.com/pion/rtcp"
. "github.com/pion/webrtc/v3"
. "m7s.live/engine/v4"
"m7s.live/engine/v4/codec"
"m7s.live/engine/v4/track"
"m7s.live/engine/v4/util"
"m7s.live/plugin/webrtc/v4/extend"
"net"
"strings"
"time"
)
type WebRTCSubscriberPro struct {
Subscriber
WebRTCIO
videoTrack *TrackLocalStaticRTP
audioTrack *TrackLocalStaticRTP
isH265 bool
}
func (suber *WebRTCSubscriberPro) OnEvent(event any) {
switch v := event.(type) {
case *track.Video:
if v.CodecID == codec.CodecID_H264 {
suber.isH265 = false
pli := "42001f"
//pli = fmt.Sprintf("%x", v.GetDecoderConfiguration().Raw[0][1:4])
if !strings.Contains(suber.SDP, pli) {
pli = reg_level.FindAllStringSubmatch(suber.SDP, -1)[0][1]
}
suber.videoTrack, _ = NewTrackLocalStaticRTP(RTPCodecCapability{MimeType: MimeTypeH264, SDPFmtpLine: "level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=" + pli}, "video", "m7s")
rtpSender, _ := suber.PeerConnection.AddTrack(suber.videoTrack)
go func() {
rtcpBuf := make([]byte, 1500)
for {
if n, _, rtcpErr := rtpSender.Read(rtcpBuf); rtcpErr != nil {
return
} else {
if p, err := rtcp.Unmarshal(rtcpBuf[:n]); err == nil {
for _, pp := range p {
switch pp.(type) {
case *rtcp.PictureLossIndication:
// fmt.Println("PictureLossIndication")
}
}
}
}
}
}()
suber.Subscriber.AddTrack(v) //接受这个track
}
start := time.Now().UnixMilli()
var rtcDc *DataChannel
if v.CodecID == codec.CodecID_H265 {
suber.isH265 = true
nInSendH265Track := 0
suber.PeerConnection.OnDataChannel(func(dc *DataChannel) {
rtcDc = dc
rtcDc.OnOpen(func() {
annexB := v.GetAnnexB()
var h265frame []byte
for _, p := range annexB { //拼接消息头
h265frame = AddBufs(h265frame, p)
}
va := v.IDRing.Value
va.AUList.Range(func(au *util.BLL) bool {
packets := au.ToBuffers()
for _, packet := range packets {
h265frame = AddBufs(h265frame, Add3ZoneOne(packet))
}
return true
})
SendH265FrameData(rtcDc, h265frame, va.Timestamp.UnixMilli()-start)
})
rtcDc.OnMessage(func(msg DataChannelMessage) {
msg_ := string(msg.Data)
fmt.Println(msg_)
})
rtcDc.OnClose(func() {
nInSendH265Track--
})
})
go extend.PlayFullAnnexB(v, suber.IO, func(frame net.Buffers) error {
var h265frame []byte
for _, packet := range frame {
if len(h265frame) == 0 {
h265frame = packet
} else {
h265frame = AddBufs(h265frame, packet)
}
}
timestamp := time.Now().UnixMilli()
SendH265FrameData(rtcDc, h265frame, timestamp-start)
return nil
})
}
case *track.Audio:
audioMimeType := MimeTypePCMA
if v.CodecID == codec.CodecID_PCMU {
audioMimeType = MimeTypePCMU
}
if v.CodecID == codec.CodecID_PCMA || v.CodecID == codec.CodecID_PCMU {
suber.audioTrack, _ = NewTrackLocalStaticRTP(RTPCodecCapability{MimeType: audioMimeType}, "audio", "m7s")
suber.PeerConnection.AddTrack(suber.audioTrack)
suber.Subscriber.AddTrack(v) //接受这个track
}
case VideoRTP:
suber.videoTrack.WriteRTP(&v.Packet)
case AudioRTP:
suber.audioTrack.WriteRTP(&v.Packet)
case ISubscriber:
suber.OnConnectionStateChange(func(pcs PeerConnectionState) {
suber.Info("Connection State has changed:" + pcs.String())
switch pcs {
case PeerConnectionStateConnected:
suber.Info("Connection State has changed:")
go suber.PlayRTP()
case PeerConnectionStateDisconnected, PeerConnectionStateFailed:
suber.Stop()
suber.PeerConnection.Close()
}
})
default:
suber.Subscriber.OnEvent(event)
}
}