mirror of
				https://github.com/Monibuca/plugin-rtsp.git
				synced 2025-10-31 02:46:47 +08:00 
			
		
		
		
	Compare commits
	
		
			2 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | e6d0489d9c | ||
|   | 68d0d9aa08 | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | |||||||
|  | .vscode | ||||||
|  | node_modules | ||||||
							
								
								
									
										70
									
								
								README.en.md
									
									
									
									
									
								
							
							
						
						
									
										70
									
								
								README.en.md
									
									
									
									
									
								
							| @@ -1,70 +0,0 @@ | |||||||
| _[简体中文](https://github.com/Monibuca/plugin-rtsp) | English_ |  | ||||||
| # RTSP Plugin |  | ||||||
|  |  | ||||||
| The RTSP plugin provides the ability to push and pull the RTSP protocol and also to push and pull the RTSP protocol to remote servers. |  | ||||||
|  |  | ||||||
| ## Plugin address |  | ||||||
|  |  | ||||||
| https://github.com/Monibuca/plugin-rtsp |  | ||||||
|  |  | ||||||
| ## Plugin introduction |  | ||||||
|  |  | ||||||
| ```go |  | ||||||
| import ( |  | ||||||
|     _ "m7s.live/plugin/rtsp/v4" |  | ||||||
| ) |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ## Push and Pull address form |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| rtsp://localhost/live/test |  | ||||||
| ``` |  | ||||||
| - `localhost` is the m7s server domain name or IP address, and the default port `554` can be omitted, otherwise it is required to be written. |  | ||||||
| - `live` represents `appName` |  | ||||||
| - `test` represents `streamName` |  | ||||||
| - `live/test` in m7s will serve as the stream identity. |  | ||||||
|  |  | ||||||
| For example, push stream to m7s through ffmpeg |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| ffmpeg -i [video source] -c:v h264 -c:a aac -f rtsp rtsp://localhost/live/test |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| This will create a stream named `live/test` inside m7s. |  | ||||||
|  |  | ||||||
| If the `live/test` stream already exists in m7s, then you can use the RTSP protocol to play it. |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| ffplay rtsp://localhost/live/test |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ## Configuration |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| rtsp: |  | ||||||
|     publish: # Refer to the global configuration format |  | ||||||
|     subscribe: # Refer to the global configuration format |  | ||||||
|     pull: # Format reference document https://m7s.live/guide/config.html#%E6%8F%92%E4%BB%B6%E9%85%8D%E7%BD%AE |  | ||||||
|     push: # Format reference document https://m7s.live/guide/config.html#%E6%8F%92%E4%BB%B6%E9%85%8D%E7%BD%AE |  | ||||||
|     listenaddr: :554 |  | ||||||
|     udpaddr: :8000 |  | ||||||
|     rtcpaddr: :8001 |  | ||||||
|     readbuffercount: 2048 |  | ||||||
|     writebuffercount: 2048 |  | ||||||
| ``` |  | ||||||
| :::tip Configuration override |  | ||||||
| publish and subscribe, any section not configured will use global configuration. |  | ||||||
| ::: |  | ||||||
|  |  | ||||||
| ## API |  | ||||||
|  |  | ||||||
| ### `rtsp/api/list` |  | ||||||
| Get all RTSP streams |  | ||||||
|  |  | ||||||
| ### `rtsp/api/pull?target=[RTSP address]&streamPath=[Stream identity]&save=[0|1|2]` |  | ||||||
| Pull the RTSP to m7s from a remote server |  | ||||||
| - save meaning: 0, do not save; 1, save to pullonstart; 2, save to pullonsub |  | ||||||
| - The RTSP address needs to be urlencoded to prevent special characters from affecting parsing |  | ||||||
| ### `rtsp/api/push?target=[RTSP address]&streamPath=[Stream identity]` |  | ||||||
| Push local streams to remote servers |  | ||||||
							
								
								
									
										81
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										81
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,68 +1,63 @@ | |||||||
| _[English](https://github.com/Monibuca/plugin-rtsp/blob/v4/README.en.md) | 简体中文_ |  | ||||||
| # RTSP插件 | # RTSP插件 | ||||||
| rtsp插件提供rtsp协议的推拉流能力,以及向远程服务器推拉rtsp协议的能力。 |  | ||||||
| ## 插件地址 | ## 插件地址 | ||||||
|  |  | ||||||
| https://github.com/Monibuca/plugin-rtsp | github.com/Monibuca/plugin-rtsp | ||||||
|  |  | ||||||
| ## 插件引入 | ## 插件引入 | ||||||
| ```go | ```go | ||||||
| import ( | import ( | ||||||
|     _ "m7s.live/plugin/rtsp/v4" |     _ "github.com/Monibuca/plugin-rtsp" | ||||||
| ) | ) | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## 推拉地址形式 | ## 默认插件配置 | ||||||
| ``` |  | ||||||
| rtsp://localhost/live/test |  | ||||||
| ``` |  | ||||||
| - `localhost`是m7s的服务器域名或者IP地址,默认端口`554`可以不写,否则需要写 |  | ||||||
| - `live`代表`appName` |  | ||||||
| - `test`代表`streamName` |  | ||||||
| - m7s中`live/test`将作为`streamPath`为流的唯一标识 |  | ||||||
|  |  | ||||||
|  | ```toml | ||||||
|  | [RTSP] | ||||||
|  | # 端口接收推流 | ||||||
|  | ListenAddr = ":554" | ||||||
|  | Reconnect = true | ||||||
|  | [RTSP.AutoPullList] | ||||||
|  | "live/rtsp1" = "rtsp://admin:admin@192.168.1.212:554/cam/realmonitor?channel=1&subtype=1" | ||||||
|  | "live/rtsp2" = "rtsp://admin:admin@192.168.1.212:554/cam/realmonitor?channel=2&subtype=1" | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | - `ListenAddr`是监听的地址 | ||||||
|  | - `Reconnect` 是否自动重连 | ||||||
|  | - `RTSP.AutoPullList` 可以配置多项,用于自动拉流,key是streamPath,value是远程rtsp地址 | ||||||
|  |  | ||||||
|  | ### 特殊功能 | ||||||
|  |  | ||||||
|  | 当自动拉流列表中当的streamPath为sub/xxx 这种形式的话,在gb28181的分屏显示时会优先采用rtsp流,已实现分屏观看子码流效果 | ||||||
|  | ## 插件功能 | ||||||
|  |  | ||||||
|  | ### 接收RTSP协议的推流 | ||||||
|  |  | ||||||
| 例如通过ffmpeg向m7s进行推流 | 例如通过ffmpeg向m7s进行推流 | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| ffmpeg -i [视频源] -c:v h264 -c:a aac -f rtsp rtsp://localhost/live/test | ffmpeg -i **** rtsp://localhost/live/test | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| 会在m7s内部形成一个名为live/test的流 | 会在m7s内部形成一个名为live/test的流 | ||||||
|  |  | ||||||
|  | ### 从远程拉取rtsp到m7s中 | ||||||
|  |  | ||||||
| 如果m7s中已经存在live/test流的话就可以用rtsp协议进行播放 | 可调用接口 | ||||||
| ```bash | `/api/rtsp/pull?target=[RTSP地址]&streamPath=[流标识]` | ||||||
| ffplay rtsp://localhost/live/test |  | ||||||
|  | ## 使用编程方式拉流 | ||||||
|  | ```go | ||||||
|  | new(RTSPClient).PullStream("live/user1","rtsp://xxx.xxx.xxx.xxx/live/user1")  | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## 配置 | ### 罗列所有的rtsp协议的流 | ||||||
|  |  | ||||||
| ```yaml | 可调用接口 | ||||||
| rtsp: | `/api/rtsp/list` | ||||||
|     publish: # 参考全局配置格式 |  | ||||||
|     subscribe: # 参考全局配置格式 |  | ||||||
|     pull: # 格式参考文档 https://m7s.live/guide/config.html#%E6%8F%92%E4%BB%B6%E9%85%8D%E7%BD%AE |  | ||||||
|     push: # 格式参考文档 https://m7s.live/guide/config.html#%E6%8F%92%E4%BB%B6%E9%85%8D%E7%BD%AE |  | ||||||
|     listenaddr: :554 |  | ||||||
|     udpaddr: :8000 |  | ||||||
|     rtcpaddr: :8001 |  | ||||||
|     readbuffercount: 2048 # 读取缓存队列大小 |  | ||||||
|     writebuffercount: 2048 # 写出缓存队列大小 |  | ||||||
| ``` |  | ||||||
| :::tip 配置覆盖 |  | ||||||
| publish |  | ||||||
| subscribe |  | ||||||
| 两项中未配置部分将使用全局配置 |  | ||||||
| ::: |  | ||||||
| ## API |  | ||||||
|  |  | ||||||
| ### `rtsp/api/list` | ### 从m7s中拉取rtsp协议流 | ||||||
| 获取所有rtsp流 |  | ||||||
|  |  | ||||||
| ### `rtsp/api/pull?target=[RTSP地址]&streamPath=[流标识]&save=[0|1|2]` | 直接通过协议rtsp://xxx.xxx.xxx.xxx/live/user1 即可播放 | ||||||
| 从远程拉取rtsp到m7s中 | > h265 编码拉流尚未实现,敬请期待 | ||||||
| - save含义:0、不保存;1、保存到pullonstart;2、保存到pullonsub |  | ||||||
| - RTSP地址需要进行urlencode 防止其中的特殊字符影响解析 |  | ||||||
| ### `rtsp/api/push?target=[RTSP地址]&streamPath=[流标识]` |  | ||||||
| 将本地的流推送到远端 |  | ||||||
|   | |||||||
							
								
								
									
										308
									
								
								client.go
									
									
									
									
									
								
							
							
						
						
									
										308
									
								
								client.go
									
									
									
									
									
								
							| @@ -1,150 +1,214 @@ | |||||||
| package rtsp | package rtsp | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"errors" | ||||||
| 	"net" | 	"time" | ||||||
|  | 	"unsafe" | ||||||
|  |  | ||||||
| 	"github.com/bluenviron/gortsplib/v4" | 	. "github.com/Monibuca/engine/v3" | ||||||
| 	"github.com/bluenviron/gortsplib/v4/pkg/base" | 	. "github.com/Monibuca/utils/v3" | ||||||
| 	"github.com/bluenviron/gortsplib/v4/pkg/url" | 	"github.com/Monibuca/utils/v3/codec" | ||||||
| 	"go.uber.org/zap" | 	"github.com/aler9/gortsplib" | ||||||
| 	"m7s.live/engine/v4" | 	"github.com/aler9/gortsplib/pkg/aac" | ||||||
|  | 	"github.com/aler9/gortsplib/pkg/base" | ||||||
|  | 	"github.com/pion/rtp" | ||||||
|  | 	"github.com/pion/rtp/codecs" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type RTSPClient struct { | type RTSPClient struct { | ||||||
| 	*gortsplib.Client `json:"-" yaml:"-"` | 	RTSPublisher | ||||||
| 	gortsplib.Transport | 	Transport         gortsplib.Transport | ||||||
| 	DialContext func(ctx context.Context, network, address string) (net.Conn, error) `json:"-" yaml:"-"` | 	*gortsplib.Client `json:"-"` | ||||||
| } |  | ||||||
| type RTSPPuller struct { |  | ||||||
| 	RTSPPublisher |  | ||||||
| 	engine.Puller |  | ||||||
| 	RTSPClient |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (p *RTSPClient) Disconnect() { | // PullStream 从外部拉流 | ||||||
| 	if p.Client != nil { | func (rtsp *RTSPClient) PullStream(streamPath string, rtspUrl string) (err error) { | ||||||
| 		p.Client.Close() | 	rtsp.Stream = &Stream{ | ||||||
|  | 		StreamPath: streamPath, | ||||||
|  | 		Type:       "RTSP Pull", | ||||||
|  | 		ExtraProp:  rtsp, | ||||||
| 	} | 	} | ||||||
| } | 	if result := rtsp.Publish(); result { | ||||||
|  | 		rtsp.URL = rtspUrl | ||||||
| func (p *RTSPPuller) Connect() error { | 		if config.Reconnect { | ||||||
| 	client := &gortsplib.Client{ | 			go func() { | ||||||
| 		DialContext: p.DialContext, | 				for rtsp.pullStream(); rtsp.Err() == nil; rtsp.pullStream() { | ||||||
|  | 					Printf("reconnecting:%s in 5 seconds", rtspUrl) | ||||||
| 		AnyPortEnable: true, | 					if rtsp.Transport == gortsplib.TransportTCP { | ||||||
|  | 						rtsp.Transport = gortsplib.TransportUDP | ||||||
|  | 					} else { | ||||||
|  | 						rtsp.Transport = gortsplib.TransportTCP | ||||||
|  | 					} | ||||||
|  | 					time.Sleep(time.Second * 5) | ||||||
|  | 				} | ||||||
|  | 				if rtsp.IsTimeout { | ||||||
|  | 					rtsp.processFunc = nil | ||||||
|  | 					go rtsp.PullStream(streamPath, rtspUrl) | ||||||
|  | 				} | ||||||
|  | 			}() | ||||||
|  | 		} else { | ||||||
|  | 			go rtsp.pullStream() | ||||||
|  | 		} | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	return errors.New("publish badname") | ||||||
|  | } | ||||||
|  | func (rtsp *RTSPClient) PushStream(streamPath string, rtspUrl string) (err error) { | ||||||
|  | 	if s := FindStream(streamPath); s != nil { | ||||||
|  | 		var tracks gortsplib.Tracks | ||||||
|  | 		var sub RTSPSubscriber | ||||||
|  | 		sub.Type = "RTSP push out" | ||||||
|  | 		sub.vt = s.WaitVideoTrack("h264", "h265") | ||||||
|  | 		sub.at = s.WaitAudioTrack("aac", "pcma", "pcmu") | ||||||
|  | 		ssrc := uintptr(unsafe.Pointer(&sub)) | ||||||
|  | 		var trackIds = 0 | ||||||
|  | 		if sub.vt != nil { | ||||||
|  | 			trackId := trackIds | ||||||
|  | 			var vtrack *gortsplib.Track | ||||||
|  | 			var vpacketer rtp.Packetizer | ||||||
|  | 			switch sub.vt.CodecID { | ||||||
|  | 			case codec.CodecID_H264: | ||||||
|  | 				if vtrack, err = gortsplib.NewTrackH264(96, &gortsplib.TrackConfigH264{ | ||||||
|  | 					SPS: sub.vt.ExtraData.NALUs[0], | ||||||
|  | 					PPS: sub.vt.ExtraData.NALUs[1], | ||||||
|  | 				}); err == nil { | ||||||
|  | 					vpacketer = rtp.NewPacketizer(1200, 96, uint32(ssrc), &codecs.H264Payloader{}, rtp.NewFixedSequencer(1), 90000) | ||||||
|  | 				} else { | ||||||
|  | 					return err | ||||||
|  | 				} | ||||||
|  | 			case codec.CodecID_H265: | ||||||
|  | 				vtrack = NewH265Track(96, sub.vt.ExtraData.NALUs) | ||||||
|  | 				vpacketer = rtp.NewPacketizer(1200, 96, uint32(ssrc), &H265Payloader{}, rtp.NewFixedSequencer(1), 90000) | ||||||
|  | 			} | ||||||
|  | 			var st uint32 | ||||||
|  | 			onVideo := func(ts uint32, pack *VideoPack) { | ||||||
|  | 				for _, nalu := range pack.NALUs { | ||||||
|  | 					for _, pack := range vpacketer.Packetize(nalu, (ts-st)*90) { | ||||||
|  | 						rtp, _ := pack.Marshal() | ||||||
|  | 						rtsp.WritePacketRTP(trackId, rtp) | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				st = ts | ||||||
|  | 			} | ||||||
|  | 			sub.OnVideo = func(ts uint32, pack *VideoPack) { | ||||||
|  | 				if st = ts; st != 0 { | ||||||
|  | 					sub.OnVideo = onVideo | ||||||
|  | 				} | ||||||
|  | 				onVideo(ts, pack) | ||||||
|  | 			} | ||||||
|  | 			tracks = append(tracks, vtrack) | ||||||
|  | 			trackIds++ | ||||||
|  | 		} | ||||||
|  | 		if sub.at != nil { | ||||||
|  | 			var st uint32 | ||||||
|  | 			trackId := trackIds | ||||||
|  | 			switch sub.at.CodecID { | ||||||
|  | 			case codec.CodecID_PCMA, codec.CodecID_PCMU: | ||||||
|  | 				atrack := NewG711Track(97, map[byte]string{7: "pcma", 8: "pcmu"}[sub.at.CodecID]) | ||||||
|  | 				apacketizer := rtp.NewPacketizer(1200, 97, uint32(ssrc), &codecs.G711Payloader{}, rtp.NewFixedSequencer(1), 8000) | ||||||
|  | 				sub.OnAudio = func(ts uint32, pack *AudioPack) { | ||||||
|  | 					for _, pack := range apacketizer.Packetize(pack.Raw, (ts-st)*8) { | ||||||
|  | 						buf, _ := pack.Marshal() | ||||||
|  | 						rtsp.WritePacketRTP(trackId, buf) | ||||||
|  | 					} | ||||||
|  | 					st = ts | ||||||
|  | 				} | ||||||
|  | 				tracks = append(tracks, atrack) | ||||||
|  | 			case codec.CodecID_AAC: | ||||||
|  | 				var mpegConf aac.MPEG4AudioConfig | ||||||
|  | 				mpegConf.Decode(sub.at.ExtraData[2:]) | ||||||
|  | 				conf := &gortsplib.TrackConfigAAC{ | ||||||
|  | 					Type:              int(mpegConf.Type), | ||||||
|  | 					SampleRate:        mpegConf.SampleRate, | ||||||
|  | 					ChannelCount:      mpegConf.ChannelCount, | ||||||
|  | 					AOTSpecificConfig: mpegConf.AOTSpecificConfig, | ||||||
|  | 				} | ||||||
|  | 				if atrack, err := gortsplib.NewTrackAAC(97, conf); err == nil { | ||||||
|  | 					apacketizer := rtp.NewPacketizer(1200, 97, uint32(ssrc), &AACPayloader{}, rtp.NewFixedSequencer(1), uint32(mpegConf.SampleRate)) | ||||||
|  | 					sub.OnAudio = func(ts uint32, pack *AudioPack) { | ||||||
|  | 						for _, pack := range apacketizer.Packetize(pack.Raw, (ts-st)*uint32(mpegConf.SampleRate)/1000) { | ||||||
|  | 							buf, _ := pack.Marshal() | ||||||
|  | 							rtsp.WritePacketRTP(trackId, buf) | ||||||
|  | 						} | ||||||
|  | 						st = ts | ||||||
|  | 					} | ||||||
|  | 					tracks = append(tracks, atrack) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return rtsp.StartPublishing(rtspUrl, tracks) | ||||||
|  | 	} | ||||||
|  | 	return errors.New("stream not exist") | ||||||
|  | } | ||||||
|  | func (client *RTSPClient) pullStream() { | ||||||
|  | 	if client.Err() != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	client.Client = &gortsplib.Client{ | ||||||
|  | 		OnPacketRTP: func(trackID int, payload []byte) { | ||||||
|  | 			// Println("OnPacketRTP", trackID, len(payload)) | ||||||
|  | 			if f := client.processFunc[trackID]; f != nil { | ||||||
|  | 				var clone []byte | ||||||
|  | 				f(append(clone, payload...)) | ||||||
|  | 			} | ||||||
|  | 		}, | ||||||
|  | 		ReadBufferSize: config.ReadBufferSize, | ||||||
|  | 		Transport:      &client.Transport, | ||||||
| 	} | 	} | ||||||
| 	p.Transport = gortsplib.TransportTCP |  | ||||||
| 	client.Transport = &p.Transport |  | ||||||
| 	// parse URL | 	// parse URL | ||||||
| 	u, err := url.Parse(p.RemoteURL) | 	u, err := base.ParseURL(client.URL) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		Printf("ParseURL:%s error:%v", client.URL, err) | ||||||
|  | 		return | ||||||
| 	} | 	} | ||||||
| 	// connect to the server | 	// connect to the server | ||||||
| 	if err = client.Start(u.Scheme, u.Host); err != nil { | 	if err = client.Start(u.Scheme, u.Host); err != nil { | ||||||
| 		return err | 		Printf("connect:%s error:%v", client.URL, err) | ||||||
|  | 		return | ||||||
| 	} | 	} | ||||||
| 	p.Client = client | 	client.OnClose = func() { | ||||||
| 	p.SetIO(p.Client) | 		client.Client.Close() | ||||||
| 	return nil | 	} | ||||||
| } | 	//client.close should be after connected! | ||||||
|  | 	defer client.Client.Close() | ||||||
| func (p *RTSPPuller) Pull() (err error) { |  | ||||||
| 	u, _ := url.Parse(p.RemoteURL) |  | ||||||
| 	var res *base.Response | 	var res *base.Response | ||||||
| 	if res, err = p.Options(u); err != nil { | 	if res, err = client.Options(u); err != nil { | ||||||
| 		p.Error("Options", zap.Error(err)) | 		Printf("option:%s error:%v", client.URL, err) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	p.Debug("Options", zap.Any("res", res)) | 	Println(res) | ||||||
| 	// find published tracks | 	// find published tracks | ||||||
| 	session, res, err := p.Describe(u) | 	tracks, baseURL, res, err := client.Describe(u) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		p.Error("Describe", zap.Error(err)) | 		Printf("Describe:%s error:%v", client.URL, err) | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	p.Debug("Describe", zap.Any("res", res)) |  | ||||||
| 	p.session = session |  | ||||||
| 	err = p.SetTracks() |  | ||||||
| 	if err != nil { |  | ||||||
| 		p.Error("SetTracks", zap.Error(err)) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	if err = p.SetupAll(session.BaseURL, session.Medias); err != nil { |  | ||||||
| 		p.Error("SetupAndPlay", zap.Error(err)) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	p.OnPacketRTPAny(p.OnPacket) |  | ||||||
| 	res, err = p.Play(nil) |  | ||||||
| 	p.Debug("Play", zap.Any("res", res)) |  | ||||||
| 	if err != nil { |  | ||||||
| 		p.Error("Play", zap.Error(err)) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return p.Wait() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type RTSPPusher struct { |  | ||||||
| 	RTSPSubscriber |  | ||||||
| 	engine.Pusher |  | ||||||
| 	RTSPClient |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *RTSPPusher) OnEvent(event any) { |  | ||||||
| 	switch v := event.(type) { |  | ||||||
| 	case engine.VideoRTP: |  | ||||||
| 		p.Client.WritePacketRTP(p.videoTrack, v.Packet) |  | ||||||
| 	case engine.AudioRTP: |  | ||||||
| 		p.Client.WritePacketRTP(p.audioTrack, v.Packet) |  | ||||||
| 	default: |  | ||||||
| 		p.RTSPSubscriber.OnEvent(event) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *RTSPPusher) Connect() error { |  | ||||||
| 	p.Client = &gortsplib.Client{ |  | ||||||
| 		DialContext:    p.DialContext, |  | ||||||
| 		WriteQueueSize: rtspConfig.WriteBufferCount, |  | ||||||
| 	} |  | ||||||
| 	// parse URL |  | ||||||
| 	u, err := url.Parse(p.RemoteURL) |  | ||||||
| 	if err != nil { |  | ||||||
| 		p.Error("url.Parse", zap.Error(err)) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	// connect to the server |  | ||||||
| 	if err = p.Client.Start(u.Scheme, u.Host); err != nil { |  | ||||||
| 		p.Error("Client.Start", zap.Error(err)) |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	p.SetIO(p.Client) |  | ||||||
| 	_, err = p.Client.Options(u) |  | ||||||
| 	return err |  | ||||||
| } |  | ||||||
| func (p *RTSPPusher) Push() (err error) { |  | ||||||
| 	var u *url.URL |  | ||||||
| 	u, err = url.Parse(p.RemoteURL) |  | ||||||
| 	// startTime := time.Now() |  | ||||||
| 	// for len(p.tracks) < 2 { |  | ||||||
| 	// 	if time.Sleep(time.Second); time.Since(startTime) > time.Second*10 { |  | ||||||
| 	// 		return fmt.Errorf("timeout") |  | ||||||
| 	// 	} |  | ||||||
| 	// } |  | ||||||
| 	if _, err = p.Announce(u, p.session); err != nil { |  | ||||||
| 		p.Error("Announce", zap.Error(err)) |  | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	err = p.SetupAll(p.session.BaseURL, p.session.Medias) | 	Println(res) | ||||||
| 	if err != nil { | 	if client.processFunc == nil { | ||||||
| 		p.Error("Setup", zap.Error(err)) | 		client.setTracks(tracks) | ||||||
|  | 	} | ||||||
|  | 	for _, track := range tracks { | ||||||
|  | 		if res, err = client.Setup(true, track, baseURL, 0, 0); err != nil { | ||||||
|  | 			Printf("Setup:%s error:%v", baseURL.String(), err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		Println(res) | ||||||
|  | 	} | ||||||
|  | 	// start reading tracks | ||||||
|  | 	if res, err = client.Play(nil); err != nil { | ||||||
|  | 		Printf("Play:%s error:%v", baseURL.String(), err) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  | 	Println(res) | ||||||
| 	if _, err = p.Record(); err != nil { | 	// wait until a fatal error | ||||||
| 		p.Error("Record", zap.Error(err)) | 	var fatalChan = make(chan error) | ||||||
| 		return | 	go func() { | ||||||
|  | 		fatalChan <- client.Wait() | ||||||
|  | 	}() | ||||||
|  | 	select { | ||||||
|  | 	case err := <-fatalChan: | ||||||
|  | 		Printf("Wait:%s error:%v", baseURL.String(), err) | ||||||
|  | 	case <-client.Done(): | ||||||
|  | 		Printf("client:%s done", client.URL) | ||||||
| 	} | 	} | ||||||
| 	p.PlayRTP() |  | ||||||
| 	return |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										52
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										52
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,49 +1,11 @@ | |||||||
| module m7s.live/plugin/rtsp/v4 | module github.com/Monibuca/plugin-rtsp/v3 | ||||||
|  |  | ||||||
| go 1.18 | go 1.16 | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
|   github.com/bluenviron/gortsplib/v4 v4.2.1 | 	github.com/Monibuca/engine/v3 v3.4.1 | ||||||
| 	github.com/bluenviron/mediacommon v1.5.0 | 	github.com/Monibuca/utils/v3 v3.0.5 | ||||||
| 	github.com/pion/rtp v1.8.2 | 	github.com/aler9/gortsplib v0.0.0-20211212220644-6f374e396529 | ||||||
| 	go.uber.org/zap v1.24.0 | 	github.com/pion/rtp v1.7.4 | ||||||
| 	m7s.live/engine/v4 v4.13.12 | 	github.com/pion/sdp/v3 v3.0.4 | ||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( |  | ||||||
| 	github.com/cnotch/ipchub v1.1.0 // indirect |  | ||||||
| 	github.com/denisbrodbeck/machineid v1.0.1 // indirect |  | ||||||
| 	github.com/go-ole/go-ole v1.2.6 // indirect |  | ||||||
| 	github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect |  | ||||||
| 	github.com/golang/mock v1.6.0 // indirect |  | ||||||
| 	github.com/google/pprof v0.0.0-20210407192527-94a9f03dee38 // indirect |  | ||||||
| 	github.com/google/uuid v1.3.1 // indirect |  | ||||||
| 	github.com/logrusorgru/aurora/v4 v4.0.0 // indirect |  | ||||||
| 	github.com/lufia/plan9stats v0.0.0-20230326075908-cb1d2100619a // indirect |  | ||||||
| 	github.com/mcuadros/go-defaults v1.2.0 // indirect |  | ||||||
| 	github.com/onsi/ginkgo/v2 v2.9.5 // indirect |  | ||||||
| 	github.com/pion/randutil v0.1.0 // indirect |  | ||||||
| 	github.com/pion/rtcp v1.2.10 // indirect |  | ||||||
| 	github.com/pion/sdp/v3 v3.0.6 // indirect |  | ||||||
| 	github.com/pion/webrtc/v3 v3.2.14 // indirect |  | ||||||
| 	github.com/power-devops/perfstat v0.0.0-20221212215047-62379fc7944b // indirect |  | ||||||
| 	github.com/q191201771/naza v0.30.11 // indirect |  | ||||||
| 	github.com/quic-go/qtls-go1-20 v0.3.1 // indirect |  | ||||||
| 	github.com/quic-go/quic-go v0.37.4 // indirect |  | ||||||
| 	github.com/shirou/gopsutil/v3 v3.23.7 // indirect |  | ||||||
| 	github.com/shoenig/go-m1cpu v0.1.6 // indirect |  | ||||||
| 	github.com/tklauser/go-sysconf v0.3.11 // indirect |  | ||||||
| 	github.com/tklauser/numcpus v0.6.0 // indirect |  | ||||||
| 	github.com/yapingcat/gomedia v0.0.0-20230727105416-c491e66c9d2a // indirect |  | ||||||
| 	github.com/yusufpapurcu/wmi v1.2.3 // indirect |  | ||||||
| 	go.uber.org/atomic v1.10.0 // indirect |  | ||||||
| 	go.uber.org/multierr v1.11.0 // indirect |  | ||||||
| 	golang.org/x/crypto v0.14.0 // indirect |  | ||||||
| 	golang.org/x/exp v0.0.0-20221205204356-47842c84f3db // indirect |  | ||||||
| 	golang.org/x/mod v0.10.0 // indirect |  | ||||||
| 	golang.org/x/net v0.17.0 // indirect |  | ||||||
| 	golang.org/x/sync v0.2.0 // indirect |  | ||||||
| 	golang.org/x/sys v0.13.0 // indirect |  | ||||||
| 	golang.org/x/tools v0.9.1 // indirect |  | ||||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect |  | ||||||
| ) |  | ||||||
							
								
								
									
										300
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										300
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,14 +1,14 @@ | |||||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||||
| github.com/benbjohnson/clock v1.1.0 h1:Q92kusRqC1XV2MjkWETPvjJVqKetz1OzxZB7mHJLju8= | github.com/BurntSushi/toml v0.4.1 h1:GaI7EiDXDRfa8VshkTj7Fym7ha+y8/XxIgD2okUIjLw= | ||||||
| github.com/bluenviron/gortsplib/v4 v3.11.0 h1:oE8j9/2+T2JyttEO9SkJKskRerdMaW10zPoI0vEIlz4= | github.com/BurntSushi/toml v0.4.1/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ= | ||||||
| github.com/bluenviron/gortsplib/v4 v3.11.0/go.mod h1:LZewC0KP6zhBceENKiSMNEryqGHljoIMq5W1SVS4vNs= | github.com/Monibuca/engine/v3 v3.4.1 h1:Ap2VbwTkMUkv80NPeUX2sNdV5Vz5nPVoU/6RU51PSAc= | ||||||
| github.com/bluenviron/gortsplib/v4 v4.2.1 h1:LugQr3TIKoj6GjOf470teDP8goWiL8PTrX2OaF+L2Vc= | github.com/Monibuca/engine/v3 v3.4.1/go.mod h1:rgAUey5ziRhlh6WugWyA5fYKyGOvcwhtTMDk4sukE7E= | ||||||
| github.com/bluenviron/gortsplib/v4 v4.2.1/go.mod h1:VOoeI2VxRKh5eEg6Y48DGb/oLxU1i+X0Xzv9z8dvsUQ= | github.com/Monibuca/utils/v3 v3.0.5 h1:w14x0HkWTbF4MmHbINLlOwe4VJNoSOeaQChMk5E/4es= | ||||||
| github.com/bluenviron/mediacommon v1.5.0 h1:lS0YKNo22ZOyCsYcLh3jn3TgUALqYw0f7RVwalC09vI= | github.com/Monibuca/utils/v3 v3.0.5/go.mod h1:RpNS95gapWs6gimwh8Xn2x72FN5tO7Powabj7dTFyvE= | ||||||
| github.com/bluenviron/mediacommon v1.5.0/go.mod h1:Ij/kE1LEucSjryNBVTyPL/gBI0d6/Css3f5PyrM957w= | github.com/aler9/gortsplib v0.0.0-20211212220644-6f374e396529 h1:j2tfs+eUubyZnuwmYWzK+IS681IixfUyD8bivz4sqAw= | ||||||
| github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= | github.com/aler9/gortsplib v0.0.0-20211212220644-6f374e396529/go.mod h1:fyQrQyHo8QvdR/h357tkv1g36VesZlzEPsdAu2VrHHc= | ||||||
| github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= | github.com/asticode/go-astikit v0.20.0/go.mod h1:h4ly7idim1tNhaVkdVBeXQZEE3L0xblP7fCWbgwipF0= | ||||||
| github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= | github.com/asticode/go-astits v1.10.0/go.mod h1:DkOWmBNQpnr9mv24KfZjq4JawCFX1FCqjLVGvO0DygQ= | ||||||
| github.com/cnotch/apirouter v0.0.0-20200731232942-89e243a791f3/go.mod h1:5deJPLON/x/s2dLOQfuKS0lenhOIT4xX0pvtN/OEIuY= | github.com/cnotch/apirouter v0.0.0-20200731232942-89e243a791f3/go.mod h1:5deJPLON/x/s2dLOQfuKS0lenhOIT4xX0pvtN/OEIuY= | ||||||
| github.com/cnotch/ipchub v1.1.0 h1:hH0lh2mU3AZXPiqMwA0pdtqrwo7PFIMRGush9OobMUs= | github.com/cnotch/ipchub v1.1.0 h1:hH0lh2mU3AZXPiqMwA0pdtqrwo7PFIMRGush9OobMUs= | ||||||
| github.com/cnotch/ipchub v1.1.0/go.mod h1:2PbeBs2q2VxxTVCn1eYCDwpAWuVXbq1+N0FU7GimOH4= | github.com/cnotch/ipchub v1.1.0/go.mod h1:2PbeBs2q2VxxTVCn1eYCDwpAWuVXbq1+N0FU7GimOH4= | ||||||
| @@ -17,291 +17,85 @@ github.com/cnotch/queue v0.0.0-20200326024423-6e88bdbf2ad4/go.mod h1:zOssjAlNusO | |||||||
| github.com/cnotch/queue v0.0.0-20201224060551-4191569ce8f6/go.mod h1:zOssjAlNusOxvtaqT+EMA+Iyi8rrtKr4/XfzN1Fgoeg= | github.com/cnotch/queue v0.0.0-20201224060551-4191569ce8f6/go.mod h1:zOssjAlNusOxvtaqT+EMA+Iyi8rrtKr4/XfzN1Fgoeg= | ||||||
| github.com/cnotch/scheduler v0.0.0-20200522024700-1d2da93eefc5/go.mod h1:F4GE3SZkJZ8an1Y0ZCqvSM3jeozNuKzoC67erG1PhIo= | github.com/cnotch/scheduler v0.0.0-20200522024700-1d2da93eefc5/go.mod h1:F4GE3SZkJZ8an1Y0ZCqvSM3jeozNuKzoC67erG1PhIo= | ||||||
| github.com/cnotch/xlog v0.0.0-20201208005456-cfda439cd3a0/go.mod h1:RW9oHsR79ffl3sR3yMGgxYupMn2btzdtJUwoxFPUE5E= | github.com/cnotch/xlog v0.0.0-20201208005456-cfda439cd3a0/go.mod h1:RW9oHsR79ffl3sR3yMGgxYupMn2btzdtJUwoxFPUE5E= | ||||||
|  | github.com/davecgh/go-spew v1.1.0 h1:ZDRjVQ15GmhC3fiQ8ni8+OwkZQO4DARzQgrnXU1Liz8= | ||||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= |  | ||||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= |  | ||||||
| github.com/denisbrodbeck/machineid v1.0.1 h1:geKr9qtkB876mXguW2X6TU4ZynleN6ezuMSRhl4D7AQ= |  | ||||||
| github.com/denisbrodbeck/machineid v1.0.1/go.mod h1:dJUwb7PTidGDeYyUBmXZ2GphQBbjJCrnectwCyxcUSI= |  | ||||||
| github.com/emitter-io/address v1.0.0/go.mod h1:GfZb5+S/o8694B1GMGK2imUYQyn2skszMvGNA5D84Ug= | github.com/emitter-io/address v1.0.0/go.mod h1:GfZb5+S/o8694B1GMGK2imUYQyn2skszMvGNA5D84Ug= | ||||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||||
| github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= | github.com/funny/slab v0.0.0-20180511031532-b1fad5e5d478 h1:Db9StoJ6RZN3YttC0Pm0I4Y5izITRYch3RMbT59BYN0= | ||||||
| github.com/go-logr/logr v1.2.4 h1:g01GSCwiDw2xSZfjJ2/T9M+S6pFdcNtFYsp+Y43HYDQ= | github.com/funny/slab v0.0.0-20180511031532-b1fad5e5d478/go.mod h1:0j1+svBH8ABEIPdUP0AIg4qedsybnXGJBakCEw8cfoo= | ||||||
| github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= | github.com/funny/utest v0.0.0-20161029064919-43870a374500 h1:Z0r1CZnoIWFB/Uiwh1BU5FYmuFe6L5NPi6XWQEmsTRg= | ||||||
| github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= | github.com/funny/utest v0.0.0-20161029064919-43870a374500/go.mod h1:mUn39tBov9jKnTWV1RlOYoNzxdBFHiSzXWdY1FoNGGg= | ||||||
| github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= |  | ||||||
| github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI= |  | ||||||
| github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572/go.mod h1:9Pwr4B2jHnOSGXyyzV8ROjYa2ojvAY6HCGYYfMoC3Ls= |  | ||||||
| github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc= |  | ||||||
| github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= |  | ||||||
| github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||||
| github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= |  | ||||||
| github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= |  | ||||||
| github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= |  | ||||||
| github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= |  | ||||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= |  | ||||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= |  | ||||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= |  | ||||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= |  | ||||||
| github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg= |  | ||||||
| github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= |  | ||||||
| github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= |  | ||||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= |  | ||||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= |  | ||||||
| github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= |  | ||||||
| github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= |  | ||||||
| github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= |  | ||||||
| github.com/google/pprof v0.0.0-20210407192527-94a9f03dee38 h1:yAJXTCF9TqKcTiHJAE8dj7HMvPfh66eeA2JYW7eFpSE= |  | ||||||
| github.com/google/pprof v0.0.0-20210407192527-94a9f03dee38/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= |  | ||||||
| github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= |  | ||||||
| github.com/google/uuid v1.3.1 h1:KjJaJ9iWZ3jOFZIf1Lqf4laDRCasjl0BCmnEGxkdLb4= |  | ||||||
| github.com/google/uuid v1.3.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= |  | ||||||
| github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= | github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= | ||||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||||
| github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= | github.com/icza/bitio v1.0.0 h1:squ/m1SHyFeCA6+6Gyol1AxV9nmPPlJFT8c2vKdj3U8= | ||||||
|  | github.com/icza/bitio v1.0.0/go.mod h1:0jGnlLAx8MKMr9VGnn/4YrvZiprkvBelsVIbA9Jjr9A= | ||||||
|  | github.com/icza/mighty v0.0.0-20180919140131-cfd07d671de6 h1:8UsGZ2rr2ksmEru6lToqnXgA8Mz1DP11X4zSJ159C3k= | ||||||
|  | github.com/icza/mighty v0.0.0-20180919140131-cfd07d671de6/go.mod h1:xQig96I1VNBDIWGCdTt54nHt6EeI639SmHycLYL7FkA= | ||||||
| github.com/kelindar/process v0.0.0-20170730150328-69a29e249ec3/go.mod h1:+lTCLnZFXOkqwD8sLPl6u4erAc0cP8wFegQHfipz7KE= | github.com/kelindar/process v0.0.0-20170730150328-69a29e249ec3/go.mod h1:+lTCLnZFXOkqwD8sLPl6u4erAc0cP8wFegQHfipz7KE= | ||||||
| github.com/kelindar/rate v1.0.0/go.mod h1:AjT4G+hTItNwt30lucEGZIz8y7Uk5zPho6vurIZ+1Es= | github.com/kelindar/rate v1.0.0/go.mod h1:AjT4G+hTItNwt30lucEGZIz8y7Uk5zPho6vurIZ+1Es= | ||||||
| github.com/kelindar/tcp v1.0.0/go.mod h1:JB5hj1cshLU60XrLij2BBxW3JQ4hOye8vqbyvuKb52k= | github.com/kelindar/tcp v1.0.0/go.mod h1:JB5hj1cshLU60XrLij2BBxW3JQ4hOye8vqbyvuKb52k= | ||||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | github.com/logrusorgru/aurora v2.0.3+incompatible h1:tOpm7WcpBTn4fjmVfgpQq0EfczGlG91VSDkswnjF5A8= | ||||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | github.com/logrusorgru/aurora v2.0.3+incompatible/go.mod h1:7rIyQOR62GCctdiQpZ/zOJlFyk6y+94wXzv6RNZgaR4= | ||||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | github.com/mattn/go-colorable v0.1.8 h1:c1ghPdyEDarC70ftn0y+A/Ee++9zz8ljHG1b13eJ0s8= | ||||||
| github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= | ||||||
| github.com/logrusorgru/aurora/v4 v4.0.0 h1:sRjfPpun/63iADiSvGGjgA1cAYegEWMPCJdUpJYn9JA= | github.com/mattn/go-isatty v0.0.12 h1:wuysRhFDzyxgEmMf5xjvJ2M9dZoWAXNNr5LSBS7uHXY= | ||||||
| github.com/logrusorgru/aurora/v4 v4.0.0/go.mod h1:lP0iIa2nrnT/qoFXcOZSrZQpJ1o6n2CUf/hyHi2Q4ZQ= | github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= | ||||||
| github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0/go.mod h1:zJYVVT2jmtg6P3p1VtQj7WsuWi/y4VnjVBn7F8KPB3I= |  | ||||||
| github.com/lufia/plan9stats v0.0.0-20230326075908-cb1d2100619a h1:N9zuLhTvBSRt0gWSiJswwQ2HqDmtX/ZCDJURnKUt1Ik= |  | ||||||
| github.com/lufia/plan9stats v0.0.0-20230326075908-cb1d2100619a/go.mod h1:JKx41uQRwqlTZabZc+kILPrO/3jlKnQ2Z8b7YiVw5cE= |  | ||||||
| github.com/mcuadros/go-defaults v1.2.0 h1:FODb8WSf0uGaY8elWJAkoLL0Ri6AlZ1bFlenk56oZtc= |  | ||||||
| github.com/mcuadros/go-defaults v1.2.0/go.mod h1:WEZtHEVIGYVDqkKSWBdWKUVdRyKlMfulPaGDWIVeCWY= |  | ||||||
| github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs= |  | ||||||
| github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= |  | ||||||
| github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= |  | ||||||
| github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= |  | ||||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||||
| github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||||
| github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= |  | ||||||
| github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= |  | ||||||
| github.com/onsi/ginkgo v1.16.5/go.mod h1:+E8gABHa3K6zRBolWtd+ROzc/U5bkGt0FwiG042wbpU= |  | ||||||
| github.com/onsi/ginkgo/v2 v2.9.5 h1:+6Hr4uxzP4XIUyAkg61dWBw8lb/gc4/X5luuxN/EC+Q= |  | ||||||
| github.com/onsi/ginkgo/v2 v2.9.5/go.mod h1:tvAoo1QUJwNEU2ITftXTpR7R1RbCzoZUOs3RonqW57k= |  | ||||||
| github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= | github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= | ||||||
| github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= |  | ||||||
| github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= |  | ||||||
| github.com/onsi/gomega v1.17.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= |  | ||||||
| github.com/onsi/gomega v1.27.6 h1:ENqfyGeS5AX/rlXDd/ETokDz93u0YufY1Pgxuy/PvWE= |  | ||||||
| github.com/pion/datachannel v1.5.5/go.mod h1:iMz+lECmfdCMqFRhXhcA/219B0SQlbpoR2V118yimL0= |  | ||||||
| github.com/pion/dtls/v2 v2.2.7/go.mod h1:8WiMkebSHFD0T+dIU+UeBaoV7kDhOW5oDCzZ7WZ/F9s= |  | ||||||
| github.com/pion/ice/v2 v2.3.9/go.mod h1:lT3kv5uUIlHfXHU/ZRD7uKD/ufM202+eTa3C/umgGf4= |  | ||||||
| github.com/pion/interceptor v0.1.17/go.mod h1:SY8kpmfVBvrbUzvj2bsXz7OJt5JvmVNZ+4Kjq7FcwrI= |  | ||||||
| github.com/pion/logging v0.2.2/go.mod h1:k0/tDVsRCX2Mb2ZEmTqNa7CWsQPc+YYCB7Q+5pahoms= |  | ||||||
| github.com/pion/mdns v0.0.7/go.mod h1:4iP2UbeFhLI/vWju/bw6ZfwjJzk0z8DNValjGxR/dD8= |  | ||||||
| github.com/pion/randutil v0.1.0 h1:CFG1UdESneORglEsnimhUjf33Rwjubwj6xfiOXBa3mA= | github.com/pion/randutil v0.1.0 h1:CFG1UdESneORglEsnimhUjf33Rwjubwj6xfiOXBa3mA= | ||||||
| github.com/pion/randutil v0.1.0/go.mod h1:XcJrSMMbbMRhASFVOlj/5hQial/Y8oH/HVo7TBZq+j8= | github.com/pion/randutil v0.1.0/go.mod h1:XcJrSMMbbMRhASFVOlj/5hQial/Y8oH/HVo7TBZq+j8= | ||||||
| github.com/pion/rtcp v1.2.10 h1:nkr3uj+8Sp97zyItdN60tE/S6vk4al5CPRR6Gejsdjc= | github.com/pion/rtcp v1.2.4 h1:NT3H5LkUGgaEapvp0HGik+a+CpflRF7KTD7H+o7OWIM= | ||||||
| github.com/pion/rtcp v1.2.10/go.mod h1:ztfEwXZNLGyF1oQDttz/ZKIBaeeg/oWbRYqzBM9TL1I= | github.com/pion/rtcp v1.2.4/go.mod h1:52rMNPWFsjr39z9B9MhnkqhPLoeHTv1aN63o/42bWE0= | ||||||
|  | github.com/pion/rtp v1.6.1/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | ||||||
| github.com/pion/rtp v1.6.2/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | github.com/pion/rtp v1.6.2/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | ||||||
| github.com/pion/rtp v1.7.13/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | github.com/pion/rtp v1.7.4 h1:4dMbjb1SuynU5OpA3kz1zHK+u+eOCQjW3MAeVHf1ODA= | ||||||
| github.com/pion/rtp v1.8.0/go.mod h1:pBGHaFt/yW7bf1jjWAoUjpSNoDnw98KTMg+jWWvziqU= | github.com/pion/rtp v1.7.4/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | ||||||
| github.com/pion/rtp v1.8.2 h1:oKMM0K1/QYQ5b5qH+ikqDSZRipP5mIxPJcgcvw5sH0w= | github.com/pion/sdp/v3 v3.0.2/go.mod h1:bNiSknmJE0HYBprTHXKPQ3+JjacTv5uap92ueJZKsRk= | ||||||
| github.com/pion/rtp v1.8.2/go.mod h1:pBGHaFt/yW7bf1jjWAoUjpSNoDnw98KTMg+jWWvziqU= | github.com/pion/sdp/v3 v3.0.4 h1:2Kf+dgrzJflNCSw3TV5v2VLeI0s/qkzy2r5jlR0wzf8= | ||||||
| github.com/pion/sctp v1.8.5/go.mod h1:SUFFfDpViyKejTAdwD1d/HQsCu+V/40cCs2nZIvC3s0= | github.com/pion/sdp/v3 v3.0.4/go.mod h1:bNiSknmJE0HYBprTHXKPQ3+JjacTv5uap92ueJZKsRk= | ||||||
| github.com/pion/sctp v1.8.7/go.mod h1:g1Ul+ARqZq5JEmoFy87Q/4CePtKnTJ1QCL9dBBdN6AU= |  | ||||||
| github.com/pion/sdp/v3 v3.0.6 h1:WuDLhtuFUUVpTfus9ILC4HRyHsW6TdugjEX/QY9OiUw= |  | ||||||
| github.com/pion/sdp/v3 v3.0.6/go.mod h1:iiFWFpQO8Fy3S5ldclBkpXqmWy02ns78NOKoLLL0YQw= |  | ||||||
| github.com/pion/srtp/v2 v2.0.16/go.mod h1:NCLCV+U+NpxQ+vXhfOETet4OgKioIgrFjZmIM3ldJYE= |  | ||||||
| github.com/pion/stun v0.6.1/go.mod h1:/hO7APkX4hZKu/D0f2lHzNyvdkTGtIy3NDmLR7kSz/8= |  | ||||||
| github.com/pion/transport v0.14.1/go.mod h1:4tGmbk00NeYA3rUa9+n+dzCCoKkcy3YlYb99Jn2fNnI= |  | ||||||
| github.com/pion/transport/v2 v2.0.0/go.mod h1:HS2MEBJTwD+1ZI2eSXSvHJx/HnzQqRy2/LXxt6eVMHc= |  | ||||||
| github.com/pion/transport/v2 v2.1.0/go.mod h1:AdSw4YBZVDkZm8fpoz+fclXyQwANWmZAlDuQdctTThQ= |  | ||||||
| github.com/pion/transport/v2 v2.2.0/go.mod h1:AdSw4YBZVDkZm8fpoz+fclXyQwANWmZAlDuQdctTThQ= |  | ||||||
| github.com/pion/transport/v2 v2.2.1/go.mod h1:cXXWavvCnFF6McHTft3DWS9iic2Mftcz1Aq29pGcU5g= |  | ||||||
| github.com/pion/turn/v2 v2.1.2/go.mod h1:1kjnPkBcex3dhCU2Am+AAmxDcGhLX3WnMfmkNpvSTQU= |  | ||||||
| github.com/pion/webrtc/v3 v3.2.14 h1:GlqnBnnLlcYYA/LOwqLLU1plZYwx0Y/e/57bZ2tzQcU= |  | ||||||
| github.com/pion/webrtc/v3 v3.2.14/go.mod h1:r1mtixc2MH847mmQTPwlEvGge7D18C2T5qp8jI9Lm44= |  | ||||||
| github.com/pixelbender/go-sdp v1.1.0/go.mod h1:6IBlz9+BrUHoFTea7gcp4S54khtOhjCW/nVDLhmZBAs= | github.com/pixelbender/go-sdp v1.1.0/go.mod h1:6IBlz9+BrUHoFTea7gcp4S54khtOhjCW/nVDLhmZBAs= | ||||||
| github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= | github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= | ||||||
|  | github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||||
|  | github.com/pkg/profile v1.4.0/go.mod h1:NWz/XGvpEW1FyYQ7fCx4dqYBLlfTcE+A9FLAkNKqjFE= | ||||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
| github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c/go.mod h1:OmDBASR4679mdNQnz2pUhc2G8CO2JrUAVFDRBDP/hJE= | github.com/q191201771/naza v0.19.1 h1:4KLcxT2CHztO+7miPRtBG3FFgadSQYQw1gPPPKN7rnY= | ||||||
| github.com/power-devops/perfstat v0.0.0-20221212215047-62379fc7944b h1:0LFwY6Q3gMACTjAbMZBjXAqTOzOwFaj2Ld6cjeQ7Rig= | github.com/q191201771/naza v0.19.1/go.mod h1:5LeGupZZFtYP1g/S203n9vXoUNVdlRnPIfM6rExjqt0= | ||||||
| github.com/power-devops/perfstat v0.0.0-20221212215047-62379fc7944b/go.mod h1:OmDBASR4679mdNQnz2pUhc2G8CO2JrUAVFDRBDP/hJE= |  | ||||||
| github.com/q191201771/naza v0.30.11 h1:az4PkAAFMA2szF6eoVJXrnProgLHV41n3IRuQSaO6mQ= |  | ||||||
| github.com/q191201771/naza v0.30.11/go.mod h1:n+dpJjQSh90PxBwxBNuifOwQttywvSIN5TkWSSYCeBk= |  | ||||||
| github.com/quic-go/qtls-go1-20 v0.3.1 h1:O4BLOM3hwfVF3AcktIylQXyl7Yi2iBNVy5QsV+ySxbg= |  | ||||||
| github.com/quic-go/qtls-go1-20 v0.3.1/go.mod h1:X9Nh97ZL80Z+bX/gUXMbipO6OxdiDi58b/fMC9mAL+k= |  | ||||||
| github.com/quic-go/quic-go v0.37.4 h1:ke8B73yMCWGq9MfrCCAw0Uzdm7GaViC3i39dsIdDlH4= |  | ||||||
| github.com/quic-go/quic-go v0.37.4/go.mod h1:YsbH1r4mSHPJcLF4k4zruUkLBqctEMBDR6VPvcYjIsU= |  | ||||||
| github.com/sclevine/agouti v3.0.0+incompatible/go.mod h1:b4WX9W9L1sfQKXeJf1mUTLZKJ48R1S7H23Ji7oFO5Bw= |  | ||||||
| github.com/shirou/gopsutil/v3 v3.23.7 h1:C+fHO8hfIppoJ1WdsVm1RoI0RwXoNdfTK7yWXV0wVj4= |  | ||||||
| github.com/shirou/gopsutil/v3 v3.23.7/go.mod h1:c4gnmoRC0hQuaLqvxnx1//VXQ0Ms/X9UnJF8pddY5z4= |  | ||||||
| github.com/shoenig/go-m1cpu v0.1.6 h1:nxdKQNcEB6vzgA2E2bvzKIYRuNj7XNJ4S/aRSwKzFtM= |  | ||||||
| github.com/shoenig/go-m1cpu v0.1.6/go.mod h1:1JJMcUBvfNwpq05QDQVAnx3gUHr9IYF7GNg9SUEw2VQ= |  | ||||||
| github.com/shoenig/test v0.6.4 h1:kVTaSd7WLz5WZ2IaoM0RSzRsUD+m8wRR+5qvntpn4LU= |  | ||||||
| github.com/shoenig/test v0.6.4/go.mod h1:byHiCGXqrVaflBLAMq/srcZIHynQPQgeyvkvXnjqq0k= |  | ||||||
| github.com/sqs/goreturns v0.0.0-20181028201513-538ac6014518/go.mod h1:CKI4AZ4XmGV240rTHfO0hfE83S6/a3/Q1siZJ/vXf7A= | github.com/sqs/goreturns v0.0.0-20181028201513-538ac6014518/go.mod h1:CKI4AZ4XmGV240rTHfO0hfE83S6/a3/Q1siZJ/vXf7A= | ||||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
| github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= | github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||||
| github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= |  | ||||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||||
|  | github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= | ||||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||||
| github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= |  | ||||||
| github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= |  | ||||||
| github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= |  | ||||||
| github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= |  | ||||||
| github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= |  | ||||||
| github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk= |  | ||||||
| github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= |  | ||||||
| github.com/tklauser/go-sysconf v0.3.11 h1:89WgdJhk5SNwJfu+GKyYveZ4IaJ7xAkecBo+KdJV0CM= |  | ||||||
| github.com/tklauser/go-sysconf v0.3.11/go.mod h1:GqXfhXY3kiPa0nAXPDIQIWzJbMCB7AmcWpGR8lSZfqI= |  | ||||||
| github.com/tklauser/numcpus v0.6.0 h1:kebhY2Qt+3U6RNK7UqpYNA+tJ23IBEGKkB7JQBfDYms= |  | ||||||
| github.com/tklauser/numcpus v0.6.0/go.mod h1:FEZLMke0lhOUG6w2JadTzp0a+Nl8PF/GFkQ5UVIcaL4= |  | ||||||
| github.com/yapingcat/gomedia v0.0.0-20230727105416-c491e66c9d2a h1:x60q0A7QmoUTzixNz7zVTdEA9JC0oYqm8S51PdbTWgs= |  | ||||||
| github.com/yapingcat/gomedia v0.0.0-20230727105416-c491e66c9d2a/go.mod h1:WSZ59bidJOO40JSJmLqlkBJrjZCtjbKKkygEMfzY/kc= |  | ||||||
| github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= |  | ||||||
| github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= |  | ||||||
| github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= |  | ||||||
| github.com/yusufpapurcu/wmi v1.2.3 h1:E1ctvB7uKFMOJw3fdOW32DwGE9I7t++CRUEMKvFoFiw= |  | ||||||
| github.com/yusufpapurcu/wmi v1.2.3/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= |  | ||||||
| go.uber.org/atomic v1.10.0 h1:9qC72Qh0+3MqyJbAn8YU5xVq1frD8bn3JtD2oXtafVQ= |  | ||||||
| go.uber.org/atomic v1.10.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0= |  | ||||||
| go.uber.org/goleak v1.1.11 h1:wy28qYRKZgnJTxGxvye5/wgWr1EKjmUDGYox5mGlRlI= |  | ||||||
| go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= |  | ||||||
| go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= |  | ||||||
| go.uber.org/zap v1.24.0 h1:FiJd5l1UOLj0wCgbSE0rwwXHzEdAZS6hiiSnxJN/D60= |  | ||||||
| go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg= |  | ||||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= |  | ||||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= |  | ||||||
| golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= | golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= | ||||||
| golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= |  | ||||||
| golang.org/x/crypto v0.8.0/go.mod h1:mRqEX+O9/h5TFCrQhkgjo2yKi0yYA+9ecGkdQoHrywE= |  | ||||||
| golang.org/x/crypto v0.10.0/go.mod h1:o4eNf7Ede1fv+hwOwZsTHl9EsPFO6q6ZvYR8vYfY45I= |  | ||||||
| golang.org/x/crypto v0.11.0/go.mod h1:xgJhtzW8F9jGdVFWZESrid1U1bjeNy4zgy5cRr/CIio= |  | ||||||
| golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc= |  | ||||||
| golang.org/x/crypto v0.14.0/go.mod h1:MVFd36DqK4CsrnJYDkBA3VC4m2GkXAM0PvzMCn4JQf4= |  | ||||||
| golang.org/x/exp v0.0.0-20221205204356-47842c84f3db h1:D/cFflL63o2KSLJIwjlcIt8PR064j/xsmdEJL/YvY/o= |  | ||||||
| golang.org/x/exp v0.0.0-20221205204356-47842c84f3db/go.mod h1:CxIveKay+FTh1D0yPZemJVgC/95VzuuOLq5Qi4xnoYc= |  | ||||||
| golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= |  | ||||||
| golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= |  | ||||||
| golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= |  | ||||||
| golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= |  | ||||||
| golang.org/x/mod v0.10.0 h1:lFO9qtOdlre5W1jxS3r/4szv2/6iXxScdzjoBMXNhYk= |  | ||||||
| golang.org/x/mod v0.10.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= |  | ||||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | golang.org/x/net v0.0.0-20210610132358-84b48f89b13b h1:k+E048sYJHyVnsr1GDrRZWQ32D2C7lWs9JRc0bel53A= | ||||||
| golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= | golang.org/x/net v0.0.0-20210610132358-84b48f89b13b/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= |  | ||||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= |  | ||||||
| golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= |  | ||||||
| golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT/9KSuxbyM7479/AVlXFRxuMCk= |  | ||||||
| golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= |  | ||||||
| golang.org/x/net v0.1.0/go.mod h1:Cx3nUiGt4eDBEyega/BKRp+/AlGL8hYe7U9odMt2Cco= |  | ||||||
| golang.org/x/net v0.5.0/go.mod h1:DivGGAXEgPSlEBzxGzZI+ZLohi+xUj054jfeKui00ws= |  | ||||||
| golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= |  | ||||||
| golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc= |  | ||||||
| golang.org/x/net v0.9.0/go.mod h1:d48xBJpPfHeWQsugry2m+kC02ZBRGRgulfHnEXEuWns= |  | ||||||
| golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= |  | ||||||
| golang.org/x/net v0.11.0/go.mod h1:2L/ixqYpgIVXmeoSA/4Lu7BzTG4KIyPIryS4IsOd1oQ= |  | ||||||
| golang.org/x/net v0.12.0/go.mod h1:zEVYFnQC7m/vmpQFELhcD1EWkZlX69l4oqgmer6hfKA= |  | ||||||
| golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM= |  | ||||||
| golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE= |  | ||||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20201207232520-09787c993a3a h1:DcqTD9SDLc+1P/r1EmRBwnVsrOwW+kk2vWf9n+1sGhs= | ||||||
| golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= |  | ||||||
| golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= |  | ||||||
| golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= |  | ||||||
| golang.org/x/sync v0.2.0 h1:PUR+T4wwASmuSTYdKjYHI5TD22Wy5ogLU5qZCOLxBrI= |  | ||||||
| golang.org/x/sync v0.2.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= |  | ||||||
| golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= |  | ||||||
| golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= |  | ||||||
| golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= |  | ||||||
| golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= |  | ||||||
| golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= |  | ||||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= |  | ||||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20210423082822-04245dca01da h1:b3NXsE2LusjYGGjL5bxEVZZORm/YEFFrWFjR8eFrw/c= | ||||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= |  | ||||||
| golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= |  | ||||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.4.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.9.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.10.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE= |  | ||||||
| golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= | golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= | ||||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||||
| golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= |  | ||||||
| golang.org/x/term v0.1.0/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= |  | ||||||
| golang.org/x/term v0.4.0/go.mod h1:9P2UbLfCdcvo3p/nzKvsmas4TnlujnuoV9hGgYzW1lQ= |  | ||||||
| golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= |  | ||||||
| golang.org/x/term v0.6.0/go.mod h1:m6U89DPEgQRMq3DNkDClhWw02AUbt2daBVO4cn4Hv9U= |  | ||||||
| golang.org/x/term v0.7.0/go.mod h1:P32HKFT3hSsZrRxla30E9HqToFYAQPCMs/zFMBUFqPY= |  | ||||||
| golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= |  | ||||||
| golang.org/x/term v0.9.0/go.mod h1:M6DEAAIenWoTxdKrOltXcmDY3rSplQUkrvaDU5FcQyo= |  | ||||||
| golang.org/x/term v0.10.0/go.mod h1:lpqdcUyK/oCiQxvxVrppt5ggO2KCZ5QblwqPnfZ6d5o= |  | ||||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= |  | ||||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= |  | ||||||
| golang.org/x/text v0.4.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= |  | ||||||
| golang.org/x/text v0.6.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= |  | ||||||
| golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= |  | ||||||
| golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= |  | ||||||
| golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= |  | ||||||
| golang.org/x/text v0.10.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= |  | ||||||
| golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= |  | ||||||
| golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k= |  | ||||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= |  | ||||||
| golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= |  | ||||||
| golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= |  | ||||||
| golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= |  | ||||||
| golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= |  | ||||||
| golang.org/x/tools v0.9.1 h1:8WMNJAz3zrtPmnYC7ISf5dEn3MT0gY7jBJfw27yrrLo= |  | ||||||
| golang.org/x/tools v0.9.1/go.mod h1:owI94Op576fPu3cIGQeHs3joujW/2Oc6MtlxbF5dfNc= |  | ||||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= |  | ||||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= |  | ||||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= |  | ||||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= |  | ||||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= |  | ||||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= |  | ||||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= |  | ||||||
| google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= |  | ||||||
| google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= |  | ||||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= |  | ||||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= |  | ||||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= |  | ||||||
| google.golang.org/protobuf v1.30.0 h1:kPPoIgf3TsEvrm0PFe15JQ+570QVxYzEvvHqChK+cng= |  | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= |  | ||||||
| gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f h1:BLraFXnmrev5lT+xlilqcH8XK9/i0At2xKjWk4p6zsU= |  | ||||||
| gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= |  | ||||||
| gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | ||||||
| gopkg.in/natefinch/lumberjack.v2 v2.0.0/go.mod h1:l0ndWWf7gzL7RNwBG7wST/UCcT4T24xpD6X8LsfU/+k= | gopkg.in/natefinch/lumberjack.v2 v2.0.0/go.mod h1:l0ndWWf7gzL7RNwBG7wST/UCcT4T24xpD6X8LsfU/+k= | ||||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | ||||||
| gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||||
| gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= | ||||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= |  | ||||||
| gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= |  | ||||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= |  | ||||||
| gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= |  | ||||||
| m7s.live/engine/v4 v4.13.12 h1:GYdIPlvwkS57DbZGvQxgMJIKL+hyYZpOv/8qNjGbY6E= |  | ||||||
| m7s.live/engine/v4 v4.13.12/go.mod h1:cRR/WOZbPSAQfYxIHuCkj1YMg+C54CYlFpOJ88q+OG4= |  | ||||||
|   | |||||||
							
								
								
									
										176
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										176
									
								
								main.go
									
									
									
									
									
								
							| @@ -1,101 +1,113 @@ | |||||||
| package rtsp | package rtsp | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"strconv" | 	"time" | ||||||
| 	"sync" |  | ||||||
|  |  | ||||||
| 	"github.com/bluenviron/gortsplib/v4" | 	. "github.com/Monibuca/engine/v3" | ||||||
| 	"go.uber.org/zap" | 	. "github.com/Monibuca/utils/v3" | ||||||
| 	. "m7s.live/engine/v4" | 	"github.com/aler9/gortsplib" | ||||||
| 	"m7s.live/engine/v4/config" |  | ||||||
| 	"m7s.live/engine/v4/util" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type RTSPConfig struct { | var config = struct { | ||||||
| 	config.Publish | 	ListenAddr     string | ||||||
| 	config.Subscribe | 	UDPAddr        string | ||||||
| 	config.Pull | 	RTCPAddr       string | ||||||
| 	config.Push | 	Timeout        int | ||||||
| 	ListenAddr       string `default:":554"` | 	Reconnect      bool | ||||||
| 	UDPAddr          string `default:":8000"` | 	AutoPullList   map[string]string | ||||||
| 	RTCPAddr         string `default:":8001"` | 	AutoPushList   map[string]string | ||||||
| 	WriteBufferCount int    `default:"2048"` | 	ReadBufferSize int | ||||||
| 	sync.Map | }{":554", ":8000", ":8001", 0, false, nil, nil, 2048} | ||||||
| 	server *gortsplib.Server |  | ||||||
|  | var pconfig = PluginConfig{ | ||||||
|  | 	Name:   "RTSP", | ||||||
|  | 	Config: &config, | ||||||
| } | } | ||||||
|  |  | ||||||
| func (conf *RTSPConfig) OnEvent(event any) { | func init() { | ||||||
| 	switch v := event.(type) { | 	pconfig.Install(runPlugin) | ||||||
| 	case FirstConfig: | } | ||||||
| 		conf.server = &gortsplib.Server{ |  | ||||||
| 			Handler:           conf, | func getRtspList() (info []*Stream) { | ||||||
| 			RTSPAddress:       conf.ListenAddr, | 	for _, s := range Streams.ToList() { | ||||||
| 			UDPRTPAddress:     conf.UDPAddr, | 		switch s.ExtraProp.(type) { | ||||||
| 			UDPRTCPAddress:    conf.RTCPAddr, | 		case *RTSPublisher: | ||||||
| 			MulticastIPRange:  "224.1.0.0/16", | 			info = append(info, s) | ||||||
| 			MulticastRTPPort:  8002, | 		case *RTSPClient: | ||||||
| 			MulticastRTCPPort: 8003, | 			info = append(info, s) | ||||||
| 			WriteQueueSize:    conf.WriteBufferCount, |  | ||||||
| 		} |  | ||||||
| 		if err := conf.server.Start(); err != nil { |  | ||||||
| 			RTSPPlugin.Error("server start", zap.Error(err)) |  | ||||||
| 			RTSPPlugin.Disabled = true |  | ||||||
| 		} |  | ||||||
| 		for streamPath, url := range conf.PullOnStart { |  | ||||||
| 			if err := RTSPPlugin.Pull(streamPath, url, new(RTSPPuller), 0); err != nil { |  | ||||||
| 				RTSPPlugin.Error("pull", zap.String("streamPath", streamPath), zap.String("url", url), zap.Error(err)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	case SEpublish: |  | ||||||
| 		if url, ok := conf.PushList[v.Target.Path]; ok { |  | ||||||
| 			if err := RTSPPlugin.Push(v.Target.Path, url, new(RTSPPusher), false); err != nil { |  | ||||||
| 				RTSPPlugin.Error("push", zap.String("streamPath", v.Target.Path), zap.String("url", url), zap.Error(err)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	case InvitePublish: //按需拉流 |  | ||||||
| 		if remoteURL := conf.CheckPullOnSub(v.Target); remoteURL != "" { |  | ||||||
| 			if err := RTSPPlugin.Pull(v.Target, remoteURL, new(RTSPPuller), 0); err != nil { |  | ||||||
| 				RTSPPlugin.Error("pull", zap.String("streamPath", v.Target), zap.String("url", remoteURL), zap.Error(err)) |  | ||||||
| 			} |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } |  | ||||||
|  |  | ||||||
| var rtspConfig = &RTSPConfig{} |  | ||||||
| var RTSPPlugin = InstallPlugin(rtspConfig) |  | ||||||
|  |  | ||||||
| func filterStreams() (ss []*Stream) { |  | ||||||
| 	Streams.Range(func(key string, s *Stream) { |  | ||||||
| 		switch s.Publisher.(type) { |  | ||||||
| 		case *RTSPPublisher, *RTSPPuller: |  | ||||||
| 			ss = append(ss, s) |  | ||||||
| 		} |  | ||||||
| 	}) |  | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
|  | func runPlugin() { | ||||||
|  | 	http.HandleFunc("/api/rtsp/list", func(w http.ResponseWriter, r *http.Request) { | ||||||
|  | 		CORS(w, r) | ||||||
|  | 		if r.URL.Query().Get("json") != "" { | ||||||
|  | 			if jsonData, err := json.Marshal(getRtspList()); err == nil { | ||||||
|  | 				w.Write(jsonData) | ||||||
|  | 			} else { | ||||||
|  | 				w.WriteHeader(500) | ||||||
|  | 			} | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		sse := NewSSE(w, r.Context()) | ||||||
|  | 		var err error | ||||||
|  | 		for tick := time.NewTicker(time.Second); err == nil; <-tick.C { | ||||||
|  | 			err = sse.WriteJSON(getRtspList()) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | 	http.HandleFunc("/api/rtsp/pull", func(w http.ResponseWriter, r *http.Request) { | ||||||
|  | 		CORS(w, r) | ||||||
|  | 		targetURL := r.URL.Query().Get("target") | ||||||
|  | 		streamPath := r.URL.Query().Get("streamPath") | ||||||
|  | 		save := r.URL.Query().Get("save") | ||||||
|  | 		if err := (&RTSPClient{Transport: gortsplib.TransportTCP}).PullStream(streamPath, targetURL); err == nil { | ||||||
|  | 			if save == "1" { | ||||||
|  | 				if config.AutoPullList == nil { | ||||||
|  | 					config.AutoPullList = make(map[string]string) | ||||||
|  | 				} | ||||||
|  | 				config.AutoPullList[streamPath] = targetURL | ||||||
|  | 				if err := pconfig.Save(); err != nil { | ||||||
|  | 					Println(err) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			w.Write([]byte(`{"code":0}`)) | ||||||
|  | 		} else { | ||||||
|  | 			w.Write([]byte(fmt.Sprintf(`{"code":1,"msg":"%s"}`, err.Error()))) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | 	for streamPath, url := range config.AutoPullList { | ||||||
|  | 		if err := (&RTSPClient{Transport: gortsplib.TransportTCP}).PullStream(streamPath, url); err != nil { | ||||||
|  | 			Println(err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| func (*RTSPConfig) API_list(w http.ResponseWriter, r *http.Request) { | 	go AddHook(HOOK_PUBLISH, func(s *Stream) { | ||||||
| 	util.ReturnFetchValue(filterStreams, w, r) | 		for streamPath, url := range config.AutoPushList { | ||||||
| } | 			if s.StreamPath == streamPath { | ||||||
|  | 				(&RTSPClient{}).PushStream(streamPath, url) | ||||||
| func (*RTSPConfig) API_Pull(rw http.ResponseWriter, r *http.Request) { | 			} | ||||||
| 	query := r.URL.Query() | 		} | ||||||
| 	save, _ := strconv.Atoi(query.Get("save")) | 	}) | ||||||
| 	err := RTSPPlugin.Pull(query.Get("streamPath"), query.Get("target"), new(RTSPPuller), save) | 	if config.ListenAddr != "" { | ||||||
| 	if err != nil { | 		go log.Fatal(ListenRtsp(config.ListenAddr)) | ||||||
| 		util.ReturnError(util.APIErrorQueryParse, err.Error(), rw, r) |  | ||||||
| 	} else { |  | ||||||
| 		util.ReturnOK(rw, r) |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (*RTSPConfig) API_Push(rw http.ResponseWriter, r *http.Request) { | func ListenRtsp(addr string) error { | ||||||
| 	query := r.URL.Query() | 	defer log.Println("rtsp server start!") | ||||||
| 	err := RTSPPlugin.Push(query.Get("streamPath"), query.Get("target"), new(RTSPPusher), query.Has("save")) | 	s := &gortsplib.Server{ | ||||||
| 	if err != nil { | 		Handler:           &RTSPServer{}, | ||||||
| 		util.ReturnError(util.APIErrorQueryParse, err.Error(), rw, r) | 		RTSPAddress:       addr, | ||||||
| 	} else { | 		UDPRTPAddress:     config.UDPAddr, | ||||||
| 		util.ReturnOK(rw, r) | 		UDPRTCPAddress:    config.RTCPAddr, | ||||||
|  | 		MulticastIPRange:  "224.1.0.0/16", | ||||||
|  | 		MulticastRTPPort:  8002, | ||||||
|  | 		MulticastRTCPPort: 8003, | ||||||
| 	} | 	} | ||||||
|  | 	return s.StartAndWait() | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								payloader.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								payloader.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | package rtsp | ||||||
|  |  | ||||||
|  | // AACayloader payloads AAC packets | ||||||
|  | type AACPayloader struct{} | ||||||
|  |  | ||||||
|  | // Payload fragments an AAC packet across one or more byte arrays | ||||||
|  | func (p *AACPayloader) Payload(mtu uint16, payload []byte) [][]byte { | ||||||
|  | 	var out [][]byte | ||||||
|  | 	o := make([]byte, len(payload)+4) | ||||||
|  | 	//AU_HEADER_LENGTH,因为单位是bit, 除以8就是auHeader的字节长度;又因为单个auheader字节长度2字节,所以再除以2就是auheader的个数。 | ||||||
|  | 	o[0] = 0x00 //高位 | ||||||
|  | 	o[1] = 0x10 //低位 | ||||||
|  | 	//AU_HEADER | ||||||
|  | 	o[2] = (byte)((len(payload) & 0x1fe0) >> 5) //高位 | ||||||
|  | 	o[3] = (byte)((len(payload) & 0x1f) << 3)   //低位 | ||||||
|  | 	copy(o[4:], payload) | ||||||
|  | 	return append(out, o) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type H265Payloader struct{} | ||||||
|  |  | ||||||
|  | func (p *H265Payloader) Payload(mtu uint16, payload []byte) [][]byte { | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										204
									
								
								publisher.go
									
									
									
									
									
								
							
							
						
						
									
										204
									
								
								publisher.go
									
									
									
									
									
								
							| @@ -1,119 +1,121 @@ | |||||||
| package rtsp | package rtsp | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"encoding/base64" | ||||||
|  | 	"encoding/hex" | ||||||
|  | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
|  |  | ||||||
| 	"github.com/bluenviron/gortsplib/v4/pkg/description" | 	. "github.com/Monibuca/engine/v3" | ||||||
| 	"github.com/bluenviron/gortsplib/v4/pkg/format" | 	. "github.com/Monibuca/utils/v3" | ||||||
| 	"github.com/bluenviron/mediacommon/pkg/codecs/mpeg4audio" | 	"github.com/aler9/gortsplib" | ||||||
| 	"github.com/pion/rtp" |  | ||||||
| 	"go.uber.org/zap" |  | ||||||
| 	. "m7s.live/engine/v4" |  | ||||||
| 	"m7s.live/engine/v4/common" |  | ||||||
| 	. "m7s.live/engine/v4/track" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type RTSPPublisher struct { | type RTSPublisher struct { | ||||||
| 	Publisher | 	*Stream     `json:"-"` | ||||||
| 	Tracks map[*description.Media]common.AVTrack `json:"-" yaml:"-"` | 	stream      *gortsplib.ServerStream | ||||||
| 	RTSPIO | 	processFunc []func([]byte) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (p *RTSPPublisher) SetTracks() error { | func (p *RTSPublisher) setTracks(tracks gortsplib.Tracks) { | ||||||
| 	p.Tracks = make(map[*description.Media]common.AVTrack, len(p.session.Medias)) | 	if p.processFunc != nil { | ||||||
| 	defer func() { | 		p.processFunc = p.processFunc[:len(tracks)] | ||||||
| 		for _, track := range p.Tracks { | 		return | ||||||
| 			p.Info("set track", zap.String("name", track.GetName())) | 	} else { | ||||||
|  | 		p.processFunc = make([]func([]byte), len(tracks)) | ||||||
|  | 	} | ||||||
|  | 	for i, track := range tracks { | ||||||
|  | 		v, ok := track.Media.Attribute("rtpmap") | ||||||
|  | 		if !ok { | ||||||
|  | 			continue | ||||||
| 		} | 		} | ||||||
| 	}() |  | ||||||
| 	for _, track := range p.session.Medias { | 		fmtp := make(map[string]string) | ||||||
| 		for _, forma := range track.Formats { | 		if v, ok := track.Media.Attribute("fmtp"); ok { | ||||||
| 			switch f := forma.(type) { | 			if tmp := strings.SplitN(v, " ", 2); len(tmp) == 2 { | ||||||
| 			case *format.H264: | 				for _, kv := range strings.Split(tmp[1], ";") { | ||||||
| 				vt := p.VideoTrack | 					kv = strings.Trim(kv, " ") | ||||||
| 				if vt == nil { |  | ||||||
| 					vt = NewH264(p.Stream, f.PayloadType()) | 					if len(kv) == 0 { | ||||||
| 					p.VideoTrack = vt | 						continue | ||||||
| 				} |  | ||||||
| 				p.Tracks[track] = p.VideoTrack |  | ||||||
| 				if len(f.SPS) > 0 { |  | ||||||
| 					vt.WriteSliceBytes(f.SPS) |  | ||||||
| 				} |  | ||||||
| 				if len(f.PPS) > 0 { |  | ||||||
| 					vt.WriteSliceBytes(f.PPS) |  | ||||||
| 				} |  | ||||||
| 			case *format.H265: |  | ||||||
| 				vt := p.VideoTrack |  | ||||||
| 				if vt == nil { |  | ||||||
| 					vt = NewH265(p.Stream, f.PayloadType()) |  | ||||||
| 					p.VideoTrack = vt |  | ||||||
| 				} |  | ||||||
| 				p.Tracks[track] = p.VideoTrack |  | ||||||
| 				if len(f.VPS) > 0 { |  | ||||||
| 					vt.WriteSliceBytes(f.VPS) |  | ||||||
| 				} |  | ||||||
| 				if len(f.SPS) > 0 { |  | ||||||
| 					vt.WriteSliceBytes(f.SPS) |  | ||||||
| 				} |  | ||||||
| 				if len(f.PPS) > 0 { |  | ||||||
| 					vt.WriteSliceBytes(f.PPS) |  | ||||||
| 				} |  | ||||||
| 			case *format.MPEG4Audio: |  | ||||||
| 				at := p.AudioTrack |  | ||||||
| 				if at == nil { |  | ||||||
| 					at := NewAAC(p.Stream, f.PayloadType(), uint32(f.Config.SampleRate)) |  | ||||||
| 					at.IndexDeltaLength = f.IndexDeltaLength |  | ||||||
| 					at.IndexLength = f.IndexLength |  | ||||||
| 					at.SizeLength = f.SizeLength |  | ||||||
| 					if f.Config.Type == mpeg4audio.ObjectTypeAACLC { |  | ||||||
| 						at.Mode = 1 |  | ||||||
| 					} | 					} | ||||||
| 					at.Channels = uint8(f.Config.ChannelCount) | 					tmp := strings.SplitN(kv, "=", 2) | ||||||
| 					asc, _ := f.Config.Marshal() | 					if len(tmp) == 2 { | ||||||
| 					// 复用AVCC写入逻辑,解析出AAC的配置信息 | 						fmtp[strings.TrimSpace(tmp[0])] = tmp[1] | ||||||
| 					at.WriteSequenceHead(append([]byte{0xAF, 0x00}, asc...)) |  | ||||||
| 					p.AudioTrack = at |  | ||||||
| 				} |  | ||||||
| 				p.Tracks[track] = p.AudioTrack |  | ||||||
| 			case *format.G711: |  | ||||||
| 				at := p.AudioTrack |  | ||||||
| 				if at == nil { |  | ||||||
| 					at := NewG711(p.Stream, !f.MULaw, f.PayloadType(), uint32(f.ClockRate())) |  | ||||||
| 					p.AudioTrack = at |  | ||||||
| 				} |  | ||||||
| 				p.Tracks[track] = p.AudioTrack |  | ||||||
| 			default: |  | ||||||
| 				rtpMap := strings.ToLower(forma.RTPMap()) |  | ||||||
| 				if strings.Contains(rtpMap, "pcm") { |  | ||||||
| 					isMulaw := false |  | ||||||
| 					if strings.Contains(rtpMap, "pcmu") { |  | ||||||
| 						isMulaw = true |  | ||||||
| 					} | 					} | ||||||
| 					at := p.AudioTrack |  | ||||||
| 					if at == nil { |  | ||||||
| 						at := NewG711(p.Stream, !isMulaw, f.PayloadType(), uint32(f.ClockRate())) |  | ||||||
| 						p.AudioTrack = at |  | ||||||
| 					} |  | ||||||
| 					p.Tracks[track] = p.AudioTrack |  | ||||||
| 				} else { |  | ||||||
| 					p.Warn("unknown format", zap.Any("format", f.FMTP())) |  | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} |  | ||||||
| 	if p.VideoTrack == nil { |  | ||||||
| 		p.Config.PubVideo = false |  | ||||||
| 		p.Info("no video track") |  | ||||||
| 	} |  | ||||||
| 	if p.AudioTrack == nil { |  | ||||||
| 		p.Config.PubAudio = false |  | ||||||
| 		p.Info("no audio track") |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *RTSPPublisher) OnPacket(m *description.Media, f format.Format, pack *rtp.Packet) { | 		v = strings.TrimSpace(v) | ||||||
| 	if t, ok := p.Tracks[m]; ok { | 		vals := strings.Split(v, " ") | ||||||
| 		t.WriteRTPPack(pack) | 		if len(vals) != 2 { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		timeScale := 0 | ||||||
|  | 		keyval := strings.Split(vals[1], "/") | ||||||
|  | 		if i, err := strconv.Atoi(keyval[1]); err == nil { | ||||||
|  | 			timeScale = i | ||||||
|  | 		} | ||||||
|  | 		if len(keyval) >= 2 { | ||||||
|  | 			Printf("track %d is %s", i, keyval[0]) | ||||||
|  | 			switch strings.ToLower(keyval[0]) { | ||||||
|  | 			case "h264": | ||||||
|  | 				vt := p.NewRTPVideo(7) | ||||||
|  | 				if conf, err := track.ExtractConfigH264(); err == nil { | ||||||
|  | 					vt.PushNalu(0, 0, conf.SPS, conf.PPS) | ||||||
|  | 				} | ||||||
|  | 				p.processFunc[i] = vt.Push | ||||||
|  | 			case "h265", "hevc": | ||||||
|  | 				vt := p.NewRTPVideo(12) | ||||||
|  | 				if v, ok := fmtp["sprop-vps"]; ok { | ||||||
|  | 					vps, _ := base64.StdEncoding.DecodeString(v) | ||||||
|  | 					vt.PushNalu(0, 0, vps) | ||||||
|  | 				} | ||||||
|  | 				if v, ok := fmtp["sprop-sps"]; ok { | ||||||
|  | 					sps, _ := base64.StdEncoding.DecodeString(v) | ||||||
|  | 					vt.PushNalu(0, 0, sps) | ||||||
|  | 				} | ||||||
|  | 				if v, ok := fmtp["sprop-pps"]; ok { | ||||||
|  | 					pps, _ := base64.StdEncoding.DecodeString(v) | ||||||
|  | 					vt.PushNalu(0, 0, pps) | ||||||
|  | 				} | ||||||
|  | 				p.processFunc[i] = vt.Push | ||||||
|  | 			case "pcma": | ||||||
|  | 				at := p.NewRTPAudio(7) | ||||||
|  | 				at.SoundRate = timeScale | ||||||
|  | 				at.SoundSize = 16 | ||||||
|  | 				if len(keyval) >= 3 { | ||||||
|  | 					x, _ := strconv.Atoi(keyval[2]) | ||||||
|  | 					at.Channels = byte(x) | ||||||
|  | 				} else { | ||||||
|  | 					at.Channels = 1 | ||||||
|  | 				} | ||||||
|  | 				at.ExtraData = []byte{(at.CodecID << 4) | (1 << 1)} | ||||||
|  | 				p.processFunc[i] = at.Push | ||||||
|  | 			case "pcmu": | ||||||
|  | 				at := p.NewRTPAudio(8) | ||||||
|  | 				at.SoundRate = timeScale | ||||||
|  | 				at.SoundSize = 16 | ||||||
|  | 				if len(keyval) >= 3 { | ||||||
|  | 					x, _ := strconv.Atoi(keyval[2]) | ||||||
|  | 					at.Channels = byte(x) | ||||||
|  | 				} else { | ||||||
|  | 					at.Channels = 1 | ||||||
|  | 				} | ||||||
|  | 				at.ExtraData = []byte{(at.CodecID << 4) | (1 << 1)} | ||||||
|  | 				p.processFunc[i] = at.Push | ||||||
|  | 			case "mpeg4-generic": | ||||||
|  | 				at := p.NewRTPAudio(0) | ||||||
|  | 				if config, ok := fmtp["config"]; ok { | ||||||
|  | 					asc, _ := hex.DecodeString(config) | ||||||
|  | 					at.SetASC(asc) | ||||||
|  | 				} else { | ||||||
|  | 					Println("aac no config") | ||||||
|  | 				} | ||||||
|  | 				at.SoundSize = 16 | ||||||
|  | 				p.processFunc[i] = at.Push | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										325
									
								
								server.go
									
									
									
									
									
								
							
							
						
						
									
										325
									
								
								server.go
									
									
									
									
									
								
							| @@ -1,121 +1,264 @@ | |||||||
| package rtsp | package rtsp | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"github.com/bluenviron/gortsplib/v4" | 	"fmt" | ||||||
| 	"github.com/bluenviron/gortsplib/v4/pkg/base" | 	"sync" | ||||||
| 	"github.com/bluenviron/gortsplib/v4/pkg/description" | 	"unsafe" | ||||||
| 	"go.uber.org/zap" |  | ||||||
| 	. "m7s.live/engine/v4" | 	"github.com/Monibuca/engine/v3" | ||||||
|  | 	. "github.com/Monibuca/utils/v3" | ||||||
|  | 	"github.com/Monibuca/utils/v3/codec" | ||||||
|  | 	"github.com/aler9/gortsplib" | ||||||
|  | 	"github.com/aler9/gortsplib/pkg/aac" | ||||||
|  | 	"github.com/aler9/gortsplib/pkg/base" | ||||||
|  | 	"github.com/pion/rtp" | ||||||
|  | 	"github.com/pion/rtp/codecs" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type RTSPIO struct { | // 接收RTSP推流:OnConnOpen->OnAnnounce->OnSetup->OnSessionOpen | ||||||
| 	server     *gortsplib.Server | // 接收RTSP拉流:OnConnOpen->OnDescribe->OnSetup->OnSessionOpen | ||||||
| 	session    *description.Session | type RTSPServer struct { | ||||||
| 	tracks     []*description.Media | 	sync.Map | ||||||
| 	stream     *gortsplib.ServerStream | } | ||||||
| 	audioTrack *description.Media | type RTSPSubscriber struct { | ||||||
| 	videoTrack *description.Media | 	stream *gortsplib.ServerStream | ||||||
|  | 	engine.Subscriber | ||||||
|  | 	vt *engine.VideoTrack | ||||||
|  | 	at *engine.AudioTrack | ||||||
| } | } | ||||||
|  |  | ||||||
| func (conf *RTSPConfig) OnConnOpen(ctx *gortsplib.ServerHandlerOnConnOpenCtx) { | // called after a connection is opened. | ||||||
| 	RTSPPlugin.Debug("conn opened") | func (sh *RTSPServer) OnConnOpen(ctx *gortsplib.ServerHandlerOnConnOpenCtx) { | ||||||
|  | 	Printf("rtsp conn opened") | ||||||
| } | } | ||||||
|  |  | ||||||
| func (conf *RTSPConfig) OnConnClose(ctx *gortsplib.ServerHandlerOnConnCloseCtx) { | // called after a connection is closed. | ||||||
| 	RTSPPlugin.Debug("conn closed") | func (sh *RTSPServer) OnConnClose(ctx *gortsplib.ServerHandlerOnConnCloseCtx) { | ||||||
| 	if p, ok := conf.LoadAndDelete(ctx.Conn); ok { | 	Printf("rtsp conn closed (%v)", ctx.Error) | ||||||
| 		p.(IIO).Stop(zap.String("conn", "closed")) | 	if p, ok := sh.Load(ctx.Conn); ok { | ||||||
|  | 		switch v := p.(type) { | ||||||
|  | 		case *RTSPublisher: | ||||||
|  | 			v.Close() | ||||||
|  | 		case *RTSPSubscriber: | ||||||
|  | 			v.Close() | ||||||
|  | 		} | ||||||
|  | 		sh.Delete(ctx.Conn) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (conf *RTSPConfig) OnSessionOpen(ctx *gortsplib.ServerHandlerOnSessionOpenCtx) { | // called after a session is opened. | ||||||
| 	RTSPPlugin.Debug("session opened") | func (sh *RTSPServer) OnSessionOpen(ctx *gortsplib.ServerHandlerOnSessionOpenCtx) { | ||||||
|  | 	Printf("rtsp session opened") | ||||||
| } | } | ||||||
|  |  | ||||||
| func (conf *RTSPConfig) OnSessionClose(ctx *gortsplib.ServerHandlerOnSessionCloseCtx) { | // called after a session is closed. | ||||||
| 	RTSPPlugin.Debug("session closed") | func (sh *RTSPServer) OnSessionClose(ctx *gortsplib.ServerHandlerOnSessionCloseCtx) { | ||||||
| 	conf.Delete(ctx.Session) | 	Printf("rtsp session closed") | ||||||
|  | 	if v, ok := sh.LoadAndDelete(ctx.Session); ok { | ||||||
|  | 		switch v := v.(type) { | ||||||
|  | 		case *RTSPublisher: | ||||||
|  | 			v.Close() | ||||||
|  | 		case *RTSPSubscriber: | ||||||
|  | 			v.Close() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| // called after receiving a DESCRIBE request. | // called after receiving a DESCRIBE request. | ||||||
| func (conf *RTSPConfig) OnDescribe(ctx *gortsplib.ServerHandlerOnDescribeCtx) (*base.Response, *gortsplib.ServerStream, error) { | func (sh *RTSPServer) OnDescribe(ctx *gortsplib.ServerHandlerOnDescribeCtx) (*base.Response, *gortsplib.ServerStream, error) { | ||||||
| 	RTSPPlugin.Debug("describe request") | 	Printf("describe request") | ||||||
| 	var suber RTSPSubscriber | 	var err error | ||||||
| 	suber.server = conf.server | 	if s := engine.FindStream(ctx.Path); s != nil { | ||||||
| 	suber.RemoteAddr = ctx.Conn.NetConn().RemoteAddr().String() | 		var tracks gortsplib.Tracks | ||||||
| 	suber.SetIO(ctx.Conn.NetConn()) | 		var stream *gortsplib.ServerStream | ||||||
| 	streamPath := ctx.Path | 		var sub RTSPSubscriber | ||||||
| 	if ctx.Query != "" { | 		sub.Type = "RTSP pull" | ||||||
| 		streamPath = streamPath + "?" + ctx.Query | 		sub.vt = s.WaitVideoTrack("h264", "h265") | ||||||
| 	} | 		sub.at = s.WaitAudioTrack("aac", "pcma", "pcmu") | ||||||
| 	if err := RTSPPlugin.Subscribe(streamPath, &suber); err == nil { | 		ssrc := uintptr(unsafe.Pointer(&stream)) | ||||||
| 		RTSPPlugin.Debug("describe replay ok") | 		var trackIds = 0 | ||||||
| 		conf.Store(ctx.Conn, &suber) | 		if sub.vt != nil { | ||||||
|  | 			trackId := trackIds | ||||||
|  | 			var vtrack *gortsplib.Track | ||||||
|  | 			var vpacketer rtp.Packetizer | ||||||
|  | 			switch sub.vt.CodecID { | ||||||
|  | 			case codec.CodecID_H264: | ||||||
|  | 				if vtrack, err = gortsplib.NewTrackH264(96, &gortsplib.TrackConfigH264{ | ||||||
|  | 					SPS: sub.vt.ExtraData.NALUs[0], | ||||||
|  | 					PPS: sub.vt.ExtraData.NALUs[1], | ||||||
|  | 				}); err == nil { | ||||||
|  | 					vpacketer = rtp.NewPacketizer(1200, 96, uint32(ssrc), &codecs.H264Payloader{}, rtp.NewFixedSequencer(1), 90000) | ||||||
|  | 				} else { | ||||||
|  | 					return nil, nil, err | ||||||
|  | 				} | ||||||
|  | 			case codec.CodecID_H265: | ||||||
|  | 				vtrack = NewH265Track(96, sub.vt.ExtraData.NALUs) | ||||||
|  | 				vpacketer = rtp.NewPacketizer(1200, 96, uint32(ssrc), &H265Payloader{}, rtp.NewFixedSequencer(1), 90000) | ||||||
|  | 			} | ||||||
|  | 			var st uint32 | ||||||
|  | 			onVideo := func(ts uint32, pack *engine.VideoPack) { | ||||||
|  | 				for i, nalu := range pack.NALUs { | ||||||
|  | 					var samples uint32 | ||||||
|  | 					if i == len(pack.NALUs)-1 { | ||||||
|  | 						samples = (ts - st) * 90 | ||||||
|  | 					} else { | ||||||
|  | 						samples = 0 | ||||||
|  | 					} | ||||||
|  | 					packs := vpacketer.Packetize(nalu, samples) | ||||||
|  | 					for j, rtpack := range packs { | ||||||
|  | 						rtpack.Marker = i == len(pack.NALUs)-1 && j == len(packs)-1 | ||||||
|  | 						rtp, _ := rtpack.Marshal() | ||||||
|  | 						stream.WritePacketRTP(trackId, rtp) | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				st = ts | ||||||
|  | 			} | ||||||
|  | 			sub.OnVideo = func(ts uint32, pack *engine.VideoPack) { | ||||||
|  | 				if st = ts; st != 0 { | ||||||
|  | 					sub.OnVideo = onVideo | ||||||
|  | 				} | ||||||
|  | 				onVideo(ts, pack) | ||||||
|  | 			} | ||||||
|  | 			tracks = append(tracks, vtrack) | ||||||
|  | 			trackIds++ | ||||||
|  | 		} | ||||||
|  | 		if sub.at != nil { | ||||||
|  | 			var st uint32 | ||||||
|  | 			trackId := trackIds | ||||||
|  | 			switch sub.at.CodecID { | ||||||
|  | 			case codec.CodecID_PCMA, codec.CodecID_PCMU: | ||||||
|  | 				atrack := NewG711Track(97, map[byte]string{7: "pcma", 8: "pcmu"}[sub.at.CodecID]) | ||||||
|  | 				apacketizer := rtp.NewPacketizer(1200, 97, uint32(ssrc), &codecs.G711Payloader{}, rtp.NewFixedSequencer(1), 8000) | ||||||
|  | 				sub.OnAudio = func(ts uint32, pack *engine.AudioPack) { | ||||||
|  | 					for _, pack := range apacketizer.Packetize(pack.Raw, (ts-st)*8) { | ||||||
|  | 						buf, _ := pack.Marshal() | ||||||
|  | 						stream.WritePacketRTP(trackId, buf) | ||||||
|  | 					} | ||||||
|  | 					st = ts | ||||||
|  | 				} | ||||||
|  | 				tracks = append(tracks, atrack) | ||||||
|  | 			case codec.CodecID_AAC: | ||||||
|  | 				var mpegConf aac.MPEG4AudioConfig | ||||||
|  | 				mpegConf.Decode(sub.at.ExtraData[2:]) | ||||||
|  | 				conf := &gortsplib.TrackConfigAAC{ | ||||||
|  | 					Type:              int(mpegConf.Type), | ||||||
|  | 					SampleRate:        mpegConf.SampleRate, | ||||||
|  | 					ChannelCount:      mpegConf.ChannelCount, | ||||||
|  | 					AOTSpecificConfig: mpegConf.AOTSpecificConfig, | ||||||
|  | 				} | ||||||
|  | 				if atrack, err := gortsplib.NewTrackAAC(97, conf); err == nil { | ||||||
|  | 					apacketizer := rtp.NewPacketizer(1200, 97, uint32(ssrc), &AACPayloader{}, rtp.NewFixedSequencer(1), uint32(mpegConf.SampleRate)) | ||||||
|  | 					sub.OnAudio = func(ts uint32, pack *engine.AudioPack) { | ||||||
|  | 						for _, pack := range apacketizer.Packetize(pack.Raw, (ts-st)*uint32(mpegConf.SampleRate)/1000) { | ||||||
|  | 							buf, _ := pack.Marshal() | ||||||
|  | 							stream.WritePacketRTP(trackId, buf) | ||||||
|  | 						} | ||||||
|  | 						st = ts | ||||||
|  | 					} | ||||||
|  | 					tracks = append(tracks, atrack) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		stream = gortsplib.NewServerStream(tracks) | ||||||
|  | 		sub.stream = stream | ||||||
|  | 		sh.Store(ctx.Conn, &sub) | ||||||
| 		return &base.Response{ | 		return &base.Response{ | ||||||
| 			StatusCode: base.StatusOK, | 			StatusCode: base.StatusOK, | ||||||
| 		}, suber.stream, nil | 		}, stream, nil | ||||||
| 	} else { | 		// if stream, ok := s.ExtraProp.(*gortsplib.ServerStream); ok { | ||||||
| 		return &base.Response{ | 		// 	return &base.Response{ | ||||||
| 			StatusCode: base.StatusNotFound, | 		// 		StatusCode: base.StatusOK, | ||||||
| 		}, suber.stream, nil | 		// 	}, stream, nil | ||||||
| 	} | 		// } | ||||||
| } |  | ||||||
|  |  | ||||||
| func (conf *RTSPConfig) OnSetup(ctx *gortsplib.ServerHandlerOnSetupCtx) (*base.Response, *gortsplib.ServerStream, error) { |  | ||||||
| 	var resp base.Response |  | ||||||
| 	resp.StatusCode = base.StatusOK |  | ||||||
| 	if p, ok := conf.Load(ctx.Conn); ok { |  | ||||||
| 		switch v := p.(type) { |  | ||||||
| 		case *RTSPSubscriber: |  | ||||||
| 			return &resp, v.stream, nil |  | ||||||
| 		case *RTSPPublisher: |  | ||||||
| 			return &resp, v.stream, nil |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	resp.StatusCode = base.StatusNotFound |  | ||||||
| 	return &resp, nil, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (conf *RTSPConfig) OnPlay(ctx *gortsplib.ServerHandlerOnPlayCtx) (*base.Response, error) { |  | ||||||
| 	var resp base.Response |  | ||||||
| 	resp.StatusCode = base.StatusNotFound |  | ||||||
| 	if p, ok := conf.Load(ctx.Conn); ok { |  | ||||||
| 		switch v := p.(type) { |  | ||||||
| 		case *RTSPSubscriber: |  | ||||||
| 			resp.StatusCode = base.StatusOK |  | ||||||
| 			go func() { |  | ||||||
| 				v.PlayRTP() |  | ||||||
| 				ctx.Session.Close() |  | ||||||
| 			}() |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return &resp, nil |  | ||||||
| } |  | ||||||
| func (conf *RTSPConfig) OnRecord(ctx *gortsplib.ServerHandlerOnRecordCtx) (*base.Response, error) { |  | ||||||
| 	if p, ok := conf.Load(ctx.Session); ok { |  | ||||||
| 		ctx.Session.OnPacketRTPAny(p.(*RTSPPublisher).OnPacket) |  | ||||||
| 	} | 	} | ||||||
| 	return &base.Response{ | 	return &base.Response{ | ||||||
| 		StatusCode: base.StatusOK, | 		StatusCode: base.StatusNotFound, | ||||||
| 	}, nil | 	}, nil, nil | ||||||
| } | } | ||||||
| func (conf *RTSPConfig) OnAnnounce(ctx *gortsplib.ServerHandlerOnAnnounceCtx) (*base.Response, error) { |  | ||||||
| 	p := &RTSPPublisher{} | // called after receiving an ANNOUNCE request. | ||||||
| 	p.SetIO(ctx.Conn.NetConn()) | func (sh *RTSPServer) OnAnnounce(ctx *gortsplib.ServerHandlerOnAnnounceCtx) (*base.Response, error) { | ||||||
| 	if err := RTSPPlugin.Publish(ctx.Path, p); err == nil { | 	Printf("announce request") | ||||||
| 		p.session = ctx.Description | 	p := &RTSPublisher{ | ||||||
| 		p.stream = gortsplib.NewServerStream(conf.server, ctx.Description) | 		Stream: &engine.Stream{ | ||||||
| 		if err = p.SetTracks(); err != nil { | 			StreamPath: ctx.Path, | ||||||
| 			return nil, err | 			Type:       "RTSP push", | ||||||
| 		} | 		}, | ||||||
| 		conf.Store(ctx.Conn, p) | 	} | ||||||
| 		conf.Store(ctx.Session, p) | 	p.ExtraProp = p | ||||||
|  | 	p.URL = ctx.Req.URL.String() | ||||||
|  | 	if p.Publish() { | ||||||
|  | 		p.setTracks(ctx.Tracks) | ||||||
|  | 		p.stream = gortsplib.NewServerStream(ctx.Tracks) | ||||||
|  | 		sh.Store(ctx.Conn, p) | ||||||
|  | 		sh.Store(ctx.Session, p) | ||||||
| 	} else { | 	} else { | ||||||
| 		return &base.Response{ | 		return &base.Response{ | ||||||
| 			StatusCode: base.StatusBadRequest, | 			StatusCode: base.StatusBadRequest, | ||||||
| 		}, nil | 		}, fmt.Errorf("streamPath is already exist") | ||||||
| 	} | 	} | ||||||
| 	return &base.Response{ | 	return &base.Response{ | ||||||
| 		StatusCode: base.StatusOK, | 		StatusCode: base.StatusOK, | ||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // called after receiving a SETUP request. | ||||||
|  | func (sh *RTSPServer) OnSetup(ctx *gortsplib.ServerHandlerOnSetupCtx) (*base.Response, *gortsplib.ServerStream, error) { | ||||||
|  | 	Printf("setup request") | ||||||
|  | 	if p, ok := sh.Load(ctx.Conn); ok { | ||||||
|  | 		switch v := p.(type) { | ||||||
|  | 		case *RTSPublisher: | ||||||
|  | 			return &base.Response{ | ||||||
|  | 				StatusCode: base.StatusOK, | ||||||
|  | 			}, v.stream, nil | ||||||
|  | 		case *RTSPSubscriber: | ||||||
|  | 			return &base.Response{ | ||||||
|  | 				StatusCode: base.StatusOK, | ||||||
|  | 			}, v.stream, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return &base.Response{ | ||||||
|  | 		StatusCode: base.StatusNotFound, | ||||||
|  | 	}, nil, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // called after receiving a PLAY request. | ||||||
|  | func (sh *RTSPServer) OnPlay(ctx *gortsplib.ServerHandlerOnPlayCtx) (*base.Response, error) { | ||||||
|  | 	Printf("play request") | ||||||
|  | 	if p, ok := sh.Load(ctx.Conn); ok { | ||||||
|  | 		if sub := p.(*RTSPSubscriber); sub.Subscribe(ctx.Path) == nil { | ||||||
|  | 			go func() { | ||||||
|  | 				sub.Play(sub.at, sub.vt) | ||||||
|  | 				ctx.Conn.Close() | ||||||
|  | 			}() | ||||||
|  | 			return &base.Response{ | ||||||
|  | 				StatusCode: base.StatusOK, | ||||||
|  | 			}, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return &base.Response{ | ||||||
|  | 		StatusCode: base.StatusNotFound, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // called after receiving a RECORD request. | ||||||
|  | func (sh *RTSPServer) OnRecord(ctx *gortsplib.ServerHandlerOnRecordCtx) (*base.Response, error) { | ||||||
|  | 	Printf("record request") | ||||||
|  | 	return &base.Response{ | ||||||
|  | 		StatusCode: base.StatusOK, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // called after receiving a frame. | ||||||
|  | func (sh *RTSPServer) OnPacketRTP(ctx *gortsplib.ServerHandlerOnPacketRTPCtx) { | ||||||
|  | 	if p, ok := sh.Load(ctx.Session); ok { | ||||||
|  | 		rtsp := p.(*RTSPublisher) | ||||||
|  | 		if rtsp.Err() != nil { | ||||||
|  | 			ctx.Session.Close() | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if f := rtsp.processFunc[ctx.TrackID]; f != nil { | ||||||
|  | 			f(ctx.Payload) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										105
									
								
								subscriber.go
									
									
									
									
									
								
							
							
						
						
									
										105
									
								
								subscriber.go
									
									
									
									
									
								
							| @@ -1,105 +0,0 @@ | |||||||
| package rtsp |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"github.com/bluenviron/gortsplib/v4" |  | ||||||
| 	"github.com/bluenviron/gortsplib/v4/pkg/description" |  | ||||||
| 	"github.com/bluenviron/gortsplib/v4/pkg/format" |  | ||||||
| 	"github.com/bluenviron/mediacommon/pkg/codecs/mpeg4audio" |  | ||||||
| 	. "m7s.live/engine/v4" |  | ||||||
| 	"m7s.live/engine/v4/codec" |  | ||||||
| 	"m7s.live/engine/v4/track" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type RTSPSubscriber struct { |  | ||||||
| 	Subscriber |  | ||||||
| 	RTSPIO |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (s *RTSPSubscriber) OnEvent(event any) { |  | ||||||
| 	switch v := event.(type) { |  | ||||||
| 	case *track.Video: |  | ||||||
| 		if s.Video != nil { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		switch v.CodecID { |  | ||||||
| 		case codec.CodecID_H264: |  | ||||||
| 			video := &description.Media{ |  | ||||||
| 				Type: description.MediaTypeVideo, |  | ||||||
| 				Formats: []format.Format{&format.H264{ |  | ||||||
| 					PacketizationMode: 1, |  | ||||||
| 					PayloadTyp:        v.PayloadType, |  | ||||||
| 					SPS:               v.ParamaterSets[0], |  | ||||||
| 					PPS:               v.ParamaterSets[1], |  | ||||||
| 				}}, |  | ||||||
| 			} |  | ||||||
| 			s.videoTrack = video |  | ||||||
| 			s.tracks = append(s.tracks, video) |  | ||||||
| 		case codec.CodecID_H265: |  | ||||||
| 			video := &description.Media{ |  | ||||||
| 				Type: description.MediaTypeVideo, |  | ||||||
| 				Formats: []format.Format{&format.H265{ |  | ||||||
| 					PayloadTyp: v.PayloadType, |  | ||||||
| 					VPS:        v.ParamaterSets[0], |  | ||||||
| 					SPS:        v.ParamaterSets[1], |  | ||||||
| 					PPS:        v.ParamaterSets[2], |  | ||||||
| 				}}, |  | ||||||
| 			} |  | ||||||
| 			s.videoTrack = video |  | ||||||
| 			s.tracks = append(s.tracks, video) |  | ||||||
| 		} |  | ||||||
| 		s.AddTrack(v) |  | ||||||
| 	case *track.Audio: |  | ||||||
| 		if s.Audio != nil { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		switch v.CodecID { |  | ||||||
| 		case codec.CodecID_AAC: |  | ||||||
| 			audio := &description.Media{ |  | ||||||
| 				Type: description.MediaTypeAudio, |  | ||||||
| 				Formats: []format.Format{&format.MPEG4Audio{ |  | ||||||
| 					PayloadTyp: v.PayloadType, |  | ||||||
| 					Config: &mpeg4audio.Config{ |  | ||||||
| 						Type:         mpeg4audio.ObjectTypeAACLC, |  | ||||||
| 						SampleRate:   int(v.SampleRate), |  | ||||||
| 						ChannelCount: int(v.Channels), |  | ||||||
| 					}, |  | ||||||
| 					SizeLength:       v.SizeLength, |  | ||||||
| 					IndexLength:      v.IndexLength, |  | ||||||
| 					IndexDeltaLength: v.IndexDeltaLength, |  | ||||||
| 				}}, |  | ||||||
| 			} |  | ||||||
| 			s.audioTrack = audio |  | ||||||
| 			s.tracks = append(s.tracks, audio) |  | ||||||
| 		case codec.CodecID_PCMA: |  | ||||||
| 			audio := &description.Media{ |  | ||||||
| 				Type:    description.MediaTypeAudio, |  | ||||||
| 				Formats: []format.Format{&format.G711{}}, |  | ||||||
| 			} |  | ||||||
| 			s.audioTrack = audio |  | ||||||
| 			s.tracks = append(s.tracks, audio) |  | ||||||
| 		case codec.CodecID_PCMU: |  | ||||||
| 			audio := &description.Media{ |  | ||||||
| 				Type: description.MediaTypeAudio, |  | ||||||
| 				Formats: []format.Format{&format.G711{ |  | ||||||
| 					MULaw: true, |  | ||||||
| 				}}, |  | ||||||
| 			} |  | ||||||
| 			s.audioTrack = audio |  | ||||||
| 			s.tracks = append(s.tracks, audio) |  | ||||||
| 		} |  | ||||||
| 		s.AddTrack(v) |  | ||||||
| 	case ISubscriber: |  | ||||||
| 		s.session = &description.Session{ |  | ||||||
| 			Medias:  s.tracks, |  | ||||||
| 		} |  | ||||||
| 		if s.server != nil { |  | ||||||
| 			s.stream = gortsplib.NewServerStream(s.server, s.session) |  | ||||||
| 		} |  | ||||||
| 	case VideoRTP: |  | ||||||
| 		s.stream.WritePacketRTP(s.videoTrack, v.Packet) |  | ||||||
| 	case AudioRTP: |  | ||||||
| 		s.stream.WritePacketRTP(s.audioTrack, v.Packet) |  | ||||||
| 	default: |  | ||||||
| 		s.Subscriber.OnEvent(event) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
							
								
								
									
										87
									
								
								track.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								track.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | |||||||
|  | package rtsp | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/base64" | ||||||
|  | 	"fmt" | ||||||
|  | 	"strconv" | ||||||
|  |  | ||||||
|  | 	"github.com/aler9/gortsplib" | ||||||
|  | 	psdp "github.com/pion/sdp/v3" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // func NewTrackAAC(payloadType uint8, conf *gortsplib.TrackConfigAAC) (*gortsplib.Track, error) { | ||||||
|  | // 	mpegConf, err := aac.MPEG4AudioConfig{ | ||||||
|  | // 		Type:              aac.MPEG4AudioType(conf.Type), | ||||||
|  | // 		SampleRate:        conf.SampleRate, | ||||||
|  | // 		ChannelCount:      conf.ChannelCount, | ||||||
|  | // 		AOTSpecificConfig: conf.AOTSpecificConfig, | ||||||
|  | // 	}.Encode() | ||||||
|  | // 	if err != nil { | ||||||
|  | // 		return nil, err | ||||||
|  | // 	} | ||||||
|  |  | ||||||
|  | // 	typ := strconv.FormatInt(int64(payloadType), 10) | ||||||
|  |  | ||||||
|  | // 	return &gortsplib.Track{ | ||||||
|  | // 		Media: &psdp.MediaDescription{ | ||||||
|  | // 			MediaName: psdp.MediaName{ | ||||||
|  | // 				Media:   "audio", | ||||||
|  | // 				Protos:  []string{"RTP", "AVP"}, | ||||||
|  | // 				Formats: []string{typ}, | ||||||
|  | // 			}, | ||||||
|  | // 			Attributes: []psdp.Attribute{ | ||||||
|  | // 				{ | ||||||
|  | // 					Key: "rtpmap", | ||||||
|  | // 					Value: typ + " mpeg4-generic/" + strconv.FormatInt(int64(conf.SampleRate), 10) + | ||||||
|  | // 						"/" + strconv.FormatInt(int64(conf.ChannelCount), 10), | ||||||
|  | // 				}, | ||||||
|  | // 				{ | ||||||
|  | // 					Key: "fmtp", | ||||||
|  | // 					Value: typ + " profile-level-id=1; " + | ||||||
|  | // 						"mode=AAC-hbr; " + | ||||||
|  | // 						"sizelength=6; " + | ||||||
|  | // 						"indexlength=2; " + | ||||||
|  | // 						"indexdeltalength=2; " + | ||||||
|  | // 						"config=" + hex.EncodeToString(mpegConf), | ||||||
|  | // 				}, | ||||||
|  | // 			}, | ||||||
|  | // 		}, | ||||||
|  | // 	}, nil | ||||||
|  | // } | ||||||
|  | func NewG711Track(payloadType uint8, law string) *gortsplib.Track { | ||||||
|  | 	return &gortsplib.Track{ | ||||||
|  | 		Media: &psdp.MediaDescription{ | ||||||
|  | 			MediaName: psdp.MediaName{ | ||||||
|  | 				Media:   "audio", | ||||||
|  | 				Protos:  []string{"RTP", "AVP"}, | ||||||
|  | 				Formats: []string{strconv.FormatInt(int64(payloadType), 10)}}, | ||||||
|  | 			Attributes: []psdp.Attribute{ | ||||||
|  | 				{ | ||||||
|  | 					Key:   "rtpmap", | ||||||
|  | 					Value: fmt.Sprintf("%d %s/8000/1", payloadType, law), | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | func NewH265Track(payloadType uint8, sprop [][]byte) *gortsplib.Track { | ||||||
|  | 	return &gortsplib.Track{ | ||||||
|  | 		Media: &psdp.MediaDescription{ | ||||||
|  | 			MediaName: psdp.MediaName{ | ||||||
|  | 				Media:   "video", | ||||||
|  | 				Protos:  []string{"RTP", "AVP"}, | ||||||
|  | 				Formats: []string{fmt.Sprintf("%d", payloadType)}, | ||||||
|  | 			}, | ||||||
|  | 			Attributes: []psdp.Attribute{ | ||||||
|  | 				{ | ||||||
|  | 					Key:   "rtpmap", | ||||||
|  | 					Value: fmt.Sprintf("%d H265/90000", payloadType), | ||||||
|  | 				}, | ||||||
|  | 				{ | ||||||
|  | 					Key:   "fmtp", | ||||||
|  | 					Value: fmt.Sprintf("%d packetization-mode=1;sprop-vps=%s;sprop-sps=%s;sprop-pps=%s;", payloadType, base64.StdEncoding.EncodeToString(sprop[0]), base64.StdEncoding.EncodeToString(sprop[1]), base64.StdEncoding.EncodeToString(sprop[2])), | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user