mirror of
https://github.com/status-im/go-libp2p-webrtc-direct.git
synced 2025-01-19 10:13:23 +00:00
40d97a0ca5
Resolves #20
157 lines
3.3 KiB
Go
157 lines
3.3 KiB
Go
package libp2pwebrtcdirect
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"net/http"
|
|
|
|
tpt "github.com/libp2p/go-libp2p-transport"
|
|
ma "github.com/multiformats/go-multiaddr"
|
|
manet "github.com/multiformats/go-multiaddr-net"
|
|
)
|
|
|
|
// Listener is an interface closely resembling the net.Listener interface.
|
|
type Listener struct {
|
|
config *connConfig
|
|
accept chan *Conn
|
|
|
|
srv *http.Server
|
|
}
|
|
|
|
func newListener(config *connConfig) (*Listener, error) {
|
|
|
|
ln, err := net.Listen(config.addr.Network(), config.addr.String())
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to listen: %v", err)
|
|
}
|
|
|
|
// Update the addr after listening
|
|
tcpMa, err := manet.FromNetAddr(ln.Addr())
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed create ma: %v", err)
|
|
}
|
|
|
|
httpMa := tcpMa.Encapsulate(httpma)
|
|
maAddr := httpMa.Encapsulate(webrtcma)
|
|
|
|
config.addr = ln.Addr()
|
|
config.maAddr = maAddr
|
|
|
|
l := &Listener{
|
|
config: config,
|
|
accept: make(chan *Conn),
|
|
}
|
|
|
|
mux := http.NewServeMux()
|
|
mux.HandleFunc("/", l.handler)
|
|
|
|
srv := &http.Server{
|
|
Handler: mux,
|
|
}
|
|
|
|
go func() {
|
|
srvErr := srv.Serve(ln)
|
|
if srvErr != nil {
|
|
log.Warningf("failed to start server: %v", srvErr)
|
|
}
|
|
}()
|
|
|
|
l.srv = srv
|
|
return l, nil
|
|
}
|
|
|
|
func (l *Listener) handler(w http.ResponseWriter, r *http.Request) {
|
|
r.ParseForm()
|
|
signals, ok := r.Form["signal"]
|
|
if !ok || len(signals) != 1 {
|
|
log.Warningf("failed to handle request: failed to parse signal")
|
|
return
|
|
}
|
|
|
|
answer, err := l.handleSignal(signals[0])
|
|
if err != nil {
|
|
log.Warningf("failed to handle request: failed to setup connection: %v", err)
|
|
return
|
|
}
|
|
|
|
w.Header().Set("Access-Control-Allow-Origin", "*")
|
|
|
|
_, err = fmt.Fprint(w, answer)
|
|
if err != nil {
|
|
log.Warningf("failed to handle request: failed to send answer: %v", err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func (l *Listener) handleSignal(offerStr string) (string, error) {
|
|
offer, err := decodeSignal(offerStr)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to decode offer: %v", err)
|
|
}
|
|
|
|
api := l.config.transport.api
|
|
pc, err := api.NewPeerConnection(l.config.transport.webrtcOptions)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if err := pc.SetRemoteDescription(offer); err != nil {
|
|
return "", fmt.Errorf("failed to set remote description: %v", err)
|
|
}
|
|
|
|
answer, err := pc.CreateAnswer(nil)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to create answer: %v", err)
|
|
}
|
|
|
|
err = pc.SetLocalDescription(answer)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to set local description: %v", err)
|
|
}
|
|
|
|
answerEnc, err := encodeSignal(answer)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to encode answer: %v", err)
|
|
}
|
|
|
|
c := newConn(l.config, pc, nil)
|
|
l.accept <- c
|
|
|
|
return answerEnc, nil
|
|
}
|
|
|
|
// Accept waits for and returns the next connection to the listener.
|
|
func (l *Listener) Accept() (tpt.Conn, error) {
|
|
conn, ok := <-l.accept
|
|
if !ok {
|
|
return nil, errors.New("Listener closed")
|
|
}
|
|
|
|
return conn, nil
|
|
}
|
|
|
|
// Close closes the listener.
|
|
// Any blocked Accept operations will be unblocked and return errors.
|
|
func (l *Listener) Close() error {
|
|
err := l.srv.Shutdown(context.Background())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
close(l.accept)
|
|
|
|
return nil
|
|
}
|
|
|
|
// Addr returns the listener's network address.
|
|
func (l *Listener) Addr() net.Addr {
|
|
return l.config.addr
|
|
}
|
|
|
|
// Multiaddr returns the listener's network Multi address.
|
|
func (l *Listener) Multiaddr() ma.Multiaddr {
|
|
return l.config.maAddr
|
|
}
|