This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
methods.go
94 lines (89 loc) · 1.99 KB
/
methods.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
package main
import (
"errors"
"net"
"sync"
"time"
"github.com/ethereum/go-ethereum/rlp"
"github.com/geph-official/geph2/libs/bdclient"
"github.com/geph-official/geph2/libs/warpfront"
log "github.com/sirupsen/logrus"
)
func connThroughBridge(bridgeConn net.Conn) (exitConn net.Conn, err error) {
bridgeConn.SetDeadline(time.Now().Add(time.Second * 30))
rlp.Encode(bridgeConn, "conn/feedback")
rlp.Encode(bridgeConn, exitName)
_, err = bridgeConn.Read(make([]byte, 1))
if err != nil {
bridgeConn.Close()
//log.Debugln("conn in", bi.Host, "failed!", err)
return
}
bridgeConn.SetDeadline(time.Time{})
exitConn = bridgeConn
return
}
func getSingleTCP(bridges []bdclient.BridgeInfo) (conn net.Conn, err error) {
bridgeRace := make(chan net.Conn)
bridgeDeadWait := new(sync.WaitGroup)
bridgeDeadWait.Add(len(bridges))
go func() {
bridgeDeadWait.Wait()
close(bridgeRace)
}()
syncer := make(chan bool)
go func() {
time.Sleep(time.Second * 3)
close(syncer)
}()
for _, bi := range bridges {
bi := bi
go func() {
defer bridgeDeadWait.Done()
bridgeConn, err := dialBridge(bi.Host, bi.Cookie)
if err != nil {
log.Debugln("dialing to", bi.Host, "failed!", err)
return
}
<-syncer
realConn, err := connThroughBridge(bridgeConn)
if err != nil {
bridgeConn.Close()
log.Debugln("conn in", bi.Host, "failed!", err)
return
}
select {
case bridgeRace <- realConn:
default:
bridgeConn.Close()
}
}()
}
zz, ok := <-bridgeRace
if !ok {
err = errors.New("singlepath timed out")
return
}
conn = zz
return
}
func getWarpfront(host2front map[string]string) (conn net.Conn, err error) {
for host, front := range host2front {
log.Println("> WF", host, front)
rc, e := warpfront.Connect(cleanHTTPClient, front, host)
if e != nil {
err = e
log.Debugf("WF failed 1/2 %v", e)
continue
}
c, e := connThroughBridge(rc)
if e != nil {
log.Debugf("WF failed 2/2 %v", e)
err = e
continue
}
conn = c
return
}
return
}