-
Notifications
You must be signed in to change notification settings - Fork 54
/
Copy pathiplookup.go
117 lines (104 loc) · 2.86 KB
/
iplookup.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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
package enginelocate
import (
"context"
"errors"
"fmt"
"math/rand"
"net"
"net/http"
"time"
"github.com/ooni/probe-cli/v3/internal/model"
"github.com/ooni/probe-cli/v3/internal/multierror"
"github.com/ooni/probe-cli/v3/internal/netxlite"
)
var (
// ErrAllIPLookuppersFailed indicates that we failed with looking
// up the probe IP for with all the lookuppers that we tried.
ErrAllIPLookuppersFailed = errors.New("all IP lookuppers failed")
// ErrInvalidIPAddress indicates that the code returned to us a
// string that actually isn't a valid IP address.
ErrInvalidIPAddress = errors.New("lookupper did not return a valid IP")
)
type lookupFunc func(
ctx context.Context, client *http.Client,
logger model.Logger, userAgent string,
resolver model.Resolver,
) (string, error)
type method struct {
name string
fn lookupFunc
}
var (
methods = []method{
{
name: "cloudflare",
fn: cloudflareIPLookup,
},
{
name: "stun_ekiga",
fn: stunEkigaIPLookup,
},
{
name: "stun_google",
fn: stunGoogleIPLookup,
},
{
name: "ubuntu",
fn: ubuntuIPLookup,
},
}
)
type ipLookupClient struct {
// Resolver is the resolver to use for HTTP.
Resolver model.Resolver
// Logger is the logger to use
Logger model.Logger
// UserAgent is the user agent to use
UserAgent string
}
func makeSlice() []method {
r := rand.New(rand.NewSource(time.Now().UnixNano()))
ret := make([]method, len(methods))
perm := r.Perm(len(methods))
for idx, randIdx := range perm {
ret[idx] = methods[randIdx]
}
return ret
}
func (c ipLookupClient) doWithCustomFunc(
ctx context.Context, fn lookupFunc,
) (string, error) {
// Reliability fix: let these mechanisms timeout earlier.
const timeout = 7 * time.Second
ctx, cancel := context.WithTimeout(ctx, timeout)
defer cancel()
// Implementation note: we MUST use an HTTP client that we're
// sure IS NOT using any proxy. To this end, we construct a
// client ourself that we know is not proxied.
// TODO(https://github.com/ooni/probe/issues/2534): the NewHTTPTransportWithResolver has QUIRKS but
// we don't care about them in this context
txp := netxlite.NewHTTPTransportWithResolver(c.Logger, c.Resolver)
clnt := &http.Client{Transport: txp}
defer clnt.CloseIdleConnections()
ip, err := fn(ctx, clnt, c.Logger, c.UserAgent, c.Resolver)
if err != nil {
return model.DefaultProbeIP, err
}
if net.ParseIP(ip) == nil {
return model.DefaultProbeIP, fmt.Errorf("%w: %s", ErrInvalidIPAddress, ip)
}
c.Logger.Debugf("iplookup: IP: %s", ip)
return ip, nil
}
func (c ipLookupClient) LookupProbeIP(ctx context.Context) (string, error) {
union := multierror.New(ErrAllIPLookuppersFailed)
for _, method := range makeSlice() {
c.Logger.Infof("iplookup: using %s", method.name)
ip, err := c.doWithCustomFunc(ctx, method.fn)
if err == nil {
return ip, nil
}
union.Add(err)
}
return model.DefaultProbeIP, union
}