mirror of
https://github.com/status-im/go-libp2p-webrtc-direct.git
synced 2025-01-19 02:01:35 +00:00
391 lines
7.7 KiB
Go
391 lines
7.7 KiB
Go
package libp2pwebrtcdirect
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net"
|
|
"net/http"
|
|
"sync"
|
|
"time"
|
|
|
|
ic "github.com/libp2p/go-libp2p-crypto"
|
|
peer "github.com/libp2p/go-libp2p-peer"
|
|
tpt "github.com/libp2p/go-libp2p-transport"
|
|
smux "github.com/libp2p/go-stream-muxer"
|
|
ma "github.com/multiformats/go-multiaddr"
|
|
manet "github.com/multiformats/go-multiaddr-net"
|
|
"github.com/pions/datachannel"
|
|
"github.com/pions/webrtc"
|
|
)
|
|
|
|
type connConfig struct {
|
|
transport *Transport
|
|
maAddr ma.Multiaddr
|
|
addr net.Addr
|
|
isServer bool
|
|
remoteID peer.ID
|
|
}
|
|
|
|
func newConnConfig(transport *Transport, maAddr ma.Multiaddr, isServer bool) (*connConfig, error) {
|
|
httpMa := maAddr.Decapsulate(webrtcma)
|
|
|
|
tcpMa := httpMa.Decapsulate(httpma)
|
|
addr, err := manet.ToNetAddr(tcpMa)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get net addr: %v", err)
|
|
}
|
|
|
|
return &connConfig{
|
|
transport: transport,
|
|
maAddr: maAddr,
|
|
addr: addr,
|
|
isServer: isServer,
|
|
}, nil
|
|
}
|
|
|
|
// Conn is a stream-multiplexing connection to a remote peer.
|
|
type Conn struct {
|
|
config *connConfig
|
|
|
|
peerConnection *webrtc.RTCPeerConnection
|
|
initChannel *datachannel.DataChannel
|
|
|
|
lock sync.RWMutex
|
|
accept chan chan detachResult
|
|
isMuxed bool
|
|
muxedConn smux.Conn
|
|
}
|
|
|
|
func newConn(config *connConfig, pc *webrtc.RTCPeerConnection, initChannel *datachannel.DataChannel) *Conn {
|
|
conn := &Conn{
|
|
config: config,
|
|
peerConnection: pc,
|
|
initChannel: initChannel,
|
|
accept: make(chan chan detachResult),
|
|
isMuxed: config.transport.muxer != nil,
|
|
}
|
|
|
|
pc.OnDataChannel(func(dc *webrtc.RTCDataChannel) {
|
|
// We have to detach in OnDataChannel
|
|
detachRes := detachChannel(dc)
|
|
conn.accept <- detachRes
|
|
})
|
|
|
|
return conn
|
|
}
|
|
|
|
func dial(ctx context.Context, config *connConfig) (*Conn, error) {
|
|
api := config.transport.api
|
|
pc, err := api.NewRTCPeerConnection(config.transport.webrtcOptions)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
dc, err := pc.CreateDataChannel("data", nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
detachRes := detachChannel(dc)
|
|
|
|
offer, err := pc.CreateOffer(nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
offerEnc, err := encodeSignal(offer)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
req, err := http.NewRequest("GET", "http://"+config.addr.String()+"/?signal="+offerEnc, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
req = req.WithContext(ctx)
|
|
|
|
var client = &http.Client{}
|
|
resp, err := client.Do(req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
answerEnc, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil && err != io.EOF {
|
|
return nil, err
|
|
}
|
|
|
|
answer, err := decodeSignal(string(answerEnc))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := pc.SetRemoteDescription(answer); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
select {
|
|
case res := <-detachRes:
|
|
if res.err != nil {
|
|
return nil, res.err
|
|
}
|
|
return newConn(config, pc, res.dc), nil
|
|
|
|
case <-ctx.Done():
|
|
return newConn(config, pc, nil), ctx.Err()
|
|
}
|
|
}
|
|
|
|
type detachResult struct {
|
|
dc *datachannel.DataChannel
|
|
err error
|
|
}
|
|
|
|
func detachChannel(dc *webrtc.RTCDataChannel) chan detachResult {
|
|
onOpenRes := make(chan detachResult)
|
|
dc.OnOpen(func() {
|
|
// Detach the data channel
|
|
raw, err := dc.Detach()
|
|
onOpenRes <- detachResult{raw, err}
|
|
})
|
|
|
|
return onOpenRes
|
|
}
|
|
|
|
// Close closes the stream muxer and the the underlying net.Conn.
|
|
func (c *Conn) Close() error {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
var err error
|
|
if c.peerConnection != nil {
|
|
err = c.peerConnection.Close()
|
|
}
|
|
c.peerConnection = nil
|
|
|
|
close(c.accept)
|
|
|
|
return err
|
|
}
|
|
|
|
// IsClosed returns whether a connection is fully closed, so it can
|
|
// be garbage collected.
|
|
func (c *Conn) IsClosed() bool {
|
|
c.lock.RLock()
|
|
pc := c.peerConnection
|
|
c.lock.RUnlock()
|
|
return pc == nil
|
|
}
|
|
|
|
// OpenStream creates a new stream.
|
|
func (c *Conn) OpenStream() (smux.Stream, error) {
|
|
muxed, err := c.getMuxed()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if muxed != nil {
|
|
return muxed.OpenStream()
|
|
}
|
|
|
|
rawDC := c.checkInitChannel()
|
|
if rawDC == nil {
|
|
pc, err := c.getPC()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
dc, err := pc.CreateDataChannel("data", nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
detachRes := detachChannel(dc)
|
|
|
|
res := <-detachRes
|
|
if res.err != nil {
|
|
return nil, res.err
|
|
}
|
|
rawDC = res.dc
|
|
}
|
|
|
|
return newStream(rawDC), nil
|
|
}
|
|
|
|
func (c *Conn) getPC() (*webrtc.RTCPeerConnection, error) {
|
|
c.lock.RLock()
|
|
pc := c.peerConnection
|
|
c.lock.RUnlock()
|
|
|
|
if pc == nil {
|
|
return nil, errors.New("Conn closed")
|
|
}
|
|
|
|
return pc, nil
|
|
}
|
|
|
|
func (c *Conn) getMuxed() (smux.Conn, error) {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
|
|
if !c.isMuxed {
|
|
return nil, nil
|
|
}
|
|
|
|
if c.muxedConn != nil {
|
|
return c.muxedConn, nil
|
|
}
|
|
|
|
rawDC := c.initChannel
|
|
if rawDC == nil {
|
|
var err error
|
|
rawDC, err = c.awaitAccept()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
err := c.useMuxer(&dcWrapper{rawDC, c.config.addr}, c.config.transport.muxer)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c.muxedConn, nil
|
|
}
|
|
|
|
// Note: caller should hold the conn lock.
|
|
func (c *Conn) useMuxer(conn net.Conn, muxer smux.Transport) error {
|
|
muxed, err := muxer.NewConn(conn, c.config.isServer)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
c.muxedConn = muxed
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *Conn) checkInitChannel() *datachannel.DataChannel {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
// Since a WebRTC offer can't be empty the offering side will have
|
|
// an initial data channel opened. We return it here, the first time
|
|
// OpenStream is called.
|
|
if c.initChannel != nil {
|
|
ch := c.initChannel
|
|
c.initChannel = nil
|
|
return ch
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// AcceptStream accepts a stream opened by the other side.
|
|
func (c *Conn) AcceptStream() (smux.Stream, error) {
|
|
muxed, err := c.getMuxed()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if muxed != nil {
|
|
return muxed.AcceptStream()
|
|
}
|
|
|
|
rawDC := c.checkInitChannel()
|
|
if rawDC == nil {
|
|
rawDC, err = c.awaitAccept()
|
|
}
|
|
|
|
return newStream(rawDC), nil
|
|
}
|
|
|
|
func (c *Conn) awaitAccept() (*datachannel.DataChannel, error) {
|
|
detachRes, ok := <-c.accept
|
|
if !ok {
|
|
return nil, errors.New("Conn closed")
|
|
}
|
|
|
|
res := <-detachRes
|
|
return res.dc, res.err
|
|
}
|
|
|
|
// LocalPeer returns our peer ID
|
|
func (c *Conn) LocalPeer() peer.ID {
|
|
// TODO: Base on WebRTC security?
|
|
return c.config.transport.localID
|
|
}
|
|
|
|
// LocalPrivateKey returns our private key
|
|
func (c *Conn) LocalPrivateKey() ic.PrivKey {
|
|
// TODO: Base on WebRTC security?
|
|
return nil
|
|
|
|
}
|
|
|
|
// RemotePeer returns the peer ID of the remote peer.
|
|
func (c *Conn) RemotePeer() peer.ID {
|
|
// TODO: Base on WebRTC security?
|
|
return c.config.remoteID
|
|
}
|
|
|
|
// RemotePublicKey returns the public key of the remote peer.
|
|
func (c *Conn) RemotePublicKey() ic.PubKey {
|
|
// TODO: Base on WebRTC security?
|
|
return nil
|
|
}
|
|
|
|
// LocalMultiaddr returns the local Multiaddr associated
|
|
// with this connection
|
|
func (c *Conn) LocalMultiaddr() ma.Multiaddr {
|
|
return c.config.maAddr
|
|
}
|
|
|
|
// RemoteMultiaddr returns the remote Multiaddr associated
|
|
// with this connection
|
|
func (c *Conn) RemoteMultiaddr() ma.Multiaddr {
|
|
return c.config.maAddr
|
|
}
|
|
|
|
// Transport returns the transport to which this connection belongs.
|
|
func (c *Conn) Transport() tpt.Transport {
|
|
return c.config.transport
|
|
}
|
|
|
|
// dcWrapper wraps datachannel.DataChannel to form a net.Conn
|
|
type dcWrapper struct {
|
|
channel *datachannel.DataChannel
|
|
addr net.Addr
|
|
}
|
|
|
|
func (w *dcWrapper) Read(p []byte) (int, error) {
|
|
return w.channel.Read(p)
|
|
}
|
|
|
|
func (w *dcWrapper) Write(p []byte) (n int, err error) {
|
|
return w.channel.Write(p)
|
|
}
|
|
|
|
func (w *dcWrapper) Close() error {
|
|
return w.channel.Close()
|
|
}
|
|
|
|
func (w *dcWrapper) LocalAddr() net.Addr {
|
|
return w.addr
|
|
}
|
|
|
|
func (w *dcWrapper) RemoteAddr() net.Addr {
|
|
return w.addr
|
|
}
|
|
|
|
func (w *dcWrapper) SetDeadline(t time.Time) error {
|
|
return nil
|
|
}
|
|
|
|
func (w *dcWrapper) SetReadDeadline(t time.Time) error {
|
|
return nil
|
|
}
|
|
|
|
func (w *dcWrapper) SetWriteDeadline(t time.Time) error {
|
|
return nil
|
|
}
|