Files
webrtc/examples/pion-to-pion/answer/main.go
2018-09-28 09:39:51 -07:00

133 lines
3.4 KiB
Go

package main
import (
"encoding/json"
"flag"
"fmt"
"math/rand"
"net/http"
"time"
"github.com/pions/webrtc"
"github.com/pions/webrtc/pkg/datachannel"
"github.com/pions/webrtc/pkg/ice"
)
func main() {
addr := flag.String("address", ":50000", "Address to host the HTTP server on.")
flag.Parse()
offerChan, answerChan := mustSignalViaHTTP(*addr)
/* Everything below is the pion-WebRTC API, thanks for using it! */
// Prepare the configuration
config := webrtc.RTCConfiguration{
IceServers: []webrtc.RTCIceServer{
{
URLs: []string{"stun:stun.l.google.com:19302"},
},
},
}
// Create a new RTCPeerConnection
peerConnection, err := webrtc.New(config)
check(err)
// Set the handler for ICE connection state
// This will notify you when the peer has connected/disconnected
peerConnection.OnICEConnectionStateChange = func(connectionState ice.ConnectionState) {
fmt.Printf("ICE Connection State has changed: %s\n", connectionState.String())
}
// Register data channel creation handling
peerConnection.OnDataChannel = func(d *webrtc.RTCDataChannel) {
fmt.Printf("New DataChannel %s %d\n", d.Label, d.ID)
d.Lock()
defer d.Unlock()
// Register channel opening handling
d.OnOpen = func() {
fmt.Printf("Data channel '%s'-'%d' open. Random messages will now be sent to any connected DataChannels every 5 seconds\n", d.Label, d.ID)
for range time.NewTicker(5 * time.Second).C {
message := randSeq(15)
fmt.Printf("Sending %s \n", message)
err := d.Send(datachannel.PayloadString{Data: []byte(message)})
check(err)
}
}
// Register message handling
d.Onmessage = func(payload datachannel.Payload) {
switch p := payload.(type) {
case *datachannel.PayloadString:
fmt.Printf("Message '%s' from DataChannel '%s' payload '%s'\n", p.PayloadType().String(), d.Label, string(p.Data))
case *datachannel.PayloadBinary:
fmt.Printf("Message '%s' from DataChannel '%s' payload '% 02x'\n", p.PayloadType().String(), d.Label, p.Data)
default:
fmt.Printf("Message '%s' from DataChannel '%s' no payload \n", p.PayloadType().String(), d.Label)
}
}
}
// Wait for the remote SessionDescription
offer := <-offerChan
err = peerConnection.SetRemoteDescription(offer)
check(err)
// Sets the LocalDescription, and starts our UDP listeners
answer, err := peerConnection.CreateAnswer(nil)
check(err)
// Send the answer
answerChan <- answer
// Block forever
select {}
}
// mustSignalViaHTTP exchange the SDP offer and answer using an HTTP server.
func mustSignalViaHTTP(address string) (offerOut chan webrtc.RTCSessionDescription, answerIn chan webrtc.RTCSessionDescription) {
offerOut = make(chan webrtc.RTCSessionDescription)
answerIn = make(chan webrtc.RTCSessionDescription)
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
var offer webrtc.RTCSessionDescription
err := json.NewDecoder(r.Body).Decode(&offer)
check(err)
offerOut <- offer
answer := <-answerIn
err = json.NewEncoder(w).Encode(answer)
check(err)
})
go http.ListenAndServe(address, nil)
fmt.Println("Listening on", address)
return
}
func randSeq(n int) string {
r := rand.New(rand.NewSource(time.Now().UnixNano()))
letters := []rune("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ")
b := make([]rune, n)
for i := range b {
b[i] = letters[r.Intn(len(letters))]
}
return string(b)
}
// check is used to panic in an error occurs.
func check(err error) {
if err != nil {
panic(err)
}
}