2019-02-25 22:08:06 +01:00

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
}