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 }