mirror of
https://github.com/pion/webrtc.git
synced 2025-09-27 03:25:58 +08:00
235 lines
7.5 KiB
Go
235 lines
7.5 KiB
Go
// SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
//go:build !js
|
|
// +build !js
|
|
|
|
// show-network-usage shows the amount of packets flowing through the vnet
|
|
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"net"
|
|
"os"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/pion/logging"
|
|
"github.com/pion/transport/v3/vnet"
|
|
"github.com/pion/webrtc/v4"
|
|
)
|
|
|
|
/* VNet Configuration
|
|
+ - - - - - - - - - - - - - - - - - - - - - - - +
|
|
VNet
|
|
| +-------------------------------------------+ |
|
|
| wan:vnet.Router |
|
|
| +---------+----------------------+----------+ |
|
|
| |
|
|
| +---------+----------+ +---------+----------+ |
|
|
| offerVNet:vnet.Net | |answerVNet:vnet.Net |
|
|
| +---------+----------+ +---------+----------+ |
|
|
| |
|
|
+ - - - - - + - - - - - - - - - - -+- - - - - - +
|
|
| |
|
|
+---------+----------+ +---------+----------+
|
|
|offerPeerConnection | |answerPeerConnection|
|
|
+--------------------+ +--------------------+
|
|
*/
|
|
|
|
// nolint:cyclop
|
|
func main() {
|
|
var inboundBytes int32 // for offerPeerConnection
|
|
var outboundBytes int32 // for offerPeerConnection
|
|
|
|
// Create a root router
|
|
wan, err := vnet.NewRouter(&vnet.RouterConfig{
|
|
CIDR: "1.2.3.0/24",
|
|
LoggerFactory: logging.NewDefaultLoggerFactory(),
|
|
})
|
|
panicIfError(err)
|
|
|
|
// Add a filter that monitors the traffic on the router
|
|
wan.AddChunkFilter(func(chunk vnet.Chunk) bool {
|
|
netType := chunk.SourceAddr().Network()
|
|
if netType == "udp" {
|
|
dstAddr := chunk.DestinationAddr().String()
|
|
host, _, err2 := net.SplitHostPort(dstAddr)
|
|
panicIfError(err2)
|
|
if host == "1.2.3.4" {
|
|
// c.UserData() returns a []byte of UDP payload
|
|
atomic.AddInt32(&inboundBytes, int32(len(chunk.UserData()))) //nolint:gosec // G115
|
|
}
|
|
srcAddr := chunk.SourceAddr().String()
|
|
host, _, err2 = net.SplitHostPort(srcAddr)
|
|
panicIfError(err2)
|
|
if host == "1.2.3.4" {
|
|
// c.UserData() returns a []byte of UDP payload
|
|
atomic.AddInt32(&outboundBytes, int32(len(chunk.UserData()))) //nolint:gosec // G115
|
|
}
|
|
}
|
|
|
|
return true
|
|
})
|
|
|
|
// Log throughput every 3 seconds
|
|
go func() {
|
|
duration := 2 * time.Second
|
|
for {
|
|
time.Sleep(duration)
|
|
|
|
inBytes := atomic.SwapInt32(&inboundBytes, 0) // read & reset
|
|
outBytes := atomic.SwapInt32(&outboundBytes, 0) // read & reset
|
|
inboundThroughput := float64(inBytes) / duration.Seconds()
|
|
outboundThroughput := float64(outBytes) / duration.Seconds()
|
|
log.Printf("inbound throughput : %.01f [Byte/s]\n", inboundThroughput)
|
|
log.Printf("outbound throughput: %.01f [Byte/s]\n", outboundThroughput)
|
|
}
|
|
}()
|
|
|
|
// Create a network interface for offerer
|
|
offerVNet, err := vnet.NewNet(&vnet.NetConfig{
|
|
StaticIPs: []string{"1.2.3.4"},
|
|
})
|
|
panicIfError(err)
|
|
|
|
// Add the network interface to the router
|
|
panicIfError(wan.AddNet(offerVNet))
|
|
|
|
offerSettingEngine := webrtc.SettingEngine{}
|
|
offerSettingEngine.SetNet(offerVNet)
|
|
offerAPI := webrtc.NewAPI(webrtc.WithSettingEngine(offerSettingEngine))
|
|
|
|
// Create a network interface for answerer
|
|
answerVNet, err := vnet.NewNet(&vnet.NetConfig{
|
|
StaticIPs: []string{"1.2.3.5"},
|
|
})
|
|
panicIfError(err)
|
|
|
|
// Add the network interface to the router
|
|
panicIfError(wan.AddNet(answerVNet))
|
|
|
|
answerSettingEngine := webrtc.SettingEngine{}
|
|
answerSettingEngine.SetNet(answerVNet)
|
|
answerAPI := webrtc.NewAPI(webrtc.WithSettingEngine(answerSettingEngine))
|
|
|
|
// Start the virtual network by calling Start() on the root router
|
|
panicIfError(wan.Start())
|
|
|
|
offerPeerConnection, err := offerAPI.NewPeerConnection(webrtc.Configuration{})
|
|
panicIfError(err)
|
|
defer func() {
|
|
if cErr := offerPeerConnection.Close(); cErr != nil {
|
|
fmt.Printf("cannot close offerPeerConnection: %v\n", cErr)
|
|
}
|
|
}()
|
|
|
|
answerPeerConnection, err := answerAPI.NewPeerConnection(webrtc.Configuration{})
|
|
panicIfError(err)
|
|
defer func() {
|
|
if cErr := answerPeerConnection.Close(); cErr != nil {
|
|
fmt.Printf("cannot close answerPeerConnection: %v\n", cErr)
|
|
}
|
|
}()
|
|
|
|
// Set the handler for Peer connection state
|
|
// This will notify you when the peer has connected/disconnected
|
|
offerPeerConnection.OnConnectionStateChange(func(state webrtc.PeerConnectionState) {
|
|
fmt.Printf("Peer Connection State has changed: %s (offerer)\n", state.String())
|
|
|
|
if state == webrtc.PeerConnectionStateFailed {
|
|
// Wait until PeerConnection has had no network activity for 30 seconds or another failure.
|
|
// It may be reconnected using an ICE Restart.
|
|
// Use webrtc.PeerConnectionStateDisconnected if you are interested in detecting faster timeout.
|
|
// Note that the PeerConnection may come back from PeerConnectionStateDisconnected.
|
|
fmt.Println("Peer Connection has gone to failed exiting")
|
|
os.Exit(0)
|
|
}
|
|
|
|
if state == webrtc.PeerConnectionStateClosed {
|
|
// PeerConnection was explicitly closed. This usually happens from a DTLS CloseNotify
|
|
fmt.Println("Peer Connection has gone to closed exiting")
|
|
os.Exit(0)
|
|
}
|
|
})
|
|
|
|
// Set the handler for Peer connection state
|
|
// This will notify you when the peer has connected/disconnected
|
|
answerPeerConnection.OnConnectionStateChange(func(state webrtc.PeerConnectionState) {
|
|
fmt.Printf("Peer Connection State has changed: %s (answerer)\n", state.String())
|
|
|
|
if state == webrtc.PeerConnectionStateFailed {
|
|
// Wait until PeerConnection has had no network activity for 30 seconds or another failure.
|
|
// It may be reconnected using an ICE Restart.
|
|
// Use webrtc.PeerConnectionStateDisconnected if you are interested in detecting faster timeout.
|
|
// Note that the PeerConnection may come back from PeerConnectionStateDisconnected.
|
|
fmt.Println("Peer Connection has gone to failed exiting")
|
|
os.Exit(0)
|
|
}
|
|
|
|
if state == webrtc.PeerConnectionStateClosed {
|
|
// PeerConnection was explicitly closed. This usually happens from a DTLS CloseNotify
|
|
fmt.Println("Peer Connection has gone to closed exiting")
|
|
os.Exit(0)
|
|
}
|
|
})
|
|
|
|
// Set ICE Candidate handler. As soon as a PeerConnection has gathered a candidate
|
|
// send it to the other peer
|
|
answerPeerConnection.OnICECandidate(func(candidate *webrtc.ICECandidate) {
|
|
if candidate != nil {
|
|
panicIfError(offerPeerConnection.AddICECandidate(candidate.ToJSON()))
|
|
}
|
|
})
|
|
|
|
// Set ICE Candidate handler. As soon as a PeerConnection has gathered a candidate
|
|
// send it to the other peer
|
|
offerPeerConnection.OnICECandidate(func(candidate *webrtc.ICECandidate) {
|
|
if candidate != nil {
|
|
panicIfError(answerPeerConnection.AddICECandidate(candidate.ToJSON()))
|
|
}
|
|
})
|
|
|
|
offerDataChannel, err := offerPeerConnection.CreateDataChannel("label", nil)
|
|
panicIfError(err)
|
|
|
|
msgSendLoop := func(dc *webrtc.DataChannel, interval time.Duration) {
|
|
for {
|
|
time.Sleep(interval)
|
|
panicIfError(dc.SendText("My DataChannel Message"))
|
|
}
|
|
}
|
|
|
|
offerDataChannel.OnOpen(func() {
|
|
// Send test from offerer every 100 msec
|
|
msgSendLoop(offerDataChannel, 100*time.Millisecond)
|
|
})
|
|
|
|
answerPeerConnection.OnDataChannel(func(answerDataChannel *webrtc.DataChannel) {
|
|
answerDataChannel.OnOpen(func() {
|
|
// Send test from answerer every 200 msec
|
|
msgSendLoop(answerDataChannel, 200*time.Millisecond)
|
|
})
|
|
})
|
|
|
|
offer, err := offerPeerConnection.CreateOffer(nil)
|
|
panicIfError(err)
|
|
panicIfError(offerPeerConnection.SetLocalDescription(offer))
|
|
panicIfError(answerPeerConnection.SetRemoteDescription(offer))
|
|
|
|
answer, err := answerPeerConnection.CreateAnswer(nil)
|
|
panicIfError(err)
|
|
panicIfError(answerPeerConnection.SetLocalDescription(answer))
|
|
panicIfError(offerPeerConnection.SetRemoteDescription(answer))
|
|
|
|
// Block forever
|
|
select {}
|
|
}
|
|
|
|
func panicIfError(err error) {
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|