99 lines
2.9 KiB
Go
99 lines
2.9 KiB
Go
|
package main
|
||
|
|
||
|
//
|
||
|
// Generates IP and endpoint information.
|
||
|
//
|
||
|
|
||
|
import (
|
||
|
"net"
|
||
|
"net/url"
|
||
|
"sort"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/ooni/probe-cli/v3/internal/engine/experiment/webconnectivity"
|
||
|
"github.com/ooni/probe-cli/v3/internal/engine/geolocate"
|
||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||
|
)
|
||
|
|
||
|
// newIPInfo creates an IP to IPInfo mapping from addresses resolved
|
||
|
// by the probe (inside [creq]) or the TH (inside [addrs]).
|
||
|
func newIPInfo(creq *ctrlRequest, addrs []string) map[string]*webconnectivity.ControlIPInfo {
|
||
|
discoveredby := make(map[string]int64)
|
||
|
for _, epnt := range creq.TCPConnect {
|
||
|
addr, _, err := net.SplitHostPort(epnt)
|
||
|
if err != nil || net.ParseIP(addr) == nil {
|
||
|
continue
|
||
|
}
|
||
|
discoveredby[addr] |= webconnectivity.ControlIPInfoFlagResolvedByProbe
|
||
|
}
|
||
|
for _, addr := range addrs {
|
||
|
if net.ParseIP(addr) != nil {
|
||
|
discoveredby[addr] |= webconnectivity.ControlIPInfoFlagResolvedByTH
|
||
|
}
|
||
|
}
|
||
|
ipinfo := make(map[string]*webconnectivity.ControlIPInfo)
|
||
|
for addr, flags := range discoveredby {
|
||
|
if netxlite.IsBogon(addr) { // note: we already excluded non-IP addrs above
|
||
|
flags |= webconnectivity.ControlIPInfoFlagIsBogon
|
||
|
}
|
||
|
asn, _, _ := geolocate.LookupASN(addr) // AS0 on failure
|
||
|
ipinfo[addr] = &webconnectivity.ControlIPInfo{
|
||
|
ASN: int64(asn),
|
||
|
Flags: flags,
|
||
|
}
|
||
|
}
|
||
|
return ipinfo
|
||
|
}
|
||
|
|
||
|
// endpointInfo contains info about an endpoint to measure
|
||
|
type endpointInfo struct {
|
||
|
// Addr is the address to measure
|
||
|
Addr string
|
||
|
|
||
|
// Epnt is the endpoint to measure
|
||
|
Epnt string
|
||
|
}
|
||
|
|
||
|
// ipInfoToEndpoints takes in input the [ipinfo] returned by newIPInfo
|
||
|
// and the [URL] provided by the probe to generate the list of endpoints
|
||
|
// to measure. We choose ports as follows:
|
||
|
//
|
||
|
// 1. if the input URL contains a port, we use such a port;
|
||
|
//
|
||
|
// 2. if the input URL scheme is "https", we choose port 443;
|
||
|
//
|
||
|
// 3. if the input URL scheme is "http", we use both 443 and 80, which
|
||
|
// allows us to include in the measurement information useful to determine
|
||
|
// whether an IP address is valid for a domain;
|
||
|
//
|
||
|
// 4. otherwise, we don't generate any endpoint to measure.
|
||
|
func ipInfoToEndpoints(URL *url.URL, ipinfo map[string]*webconnectivity.ControlIPInfo) []endpointInfo {
|
||
|
var ports []string
|
||
|
if port := URL.Port(); port != "" {
|
||
|
ports = []string{port} // as documented
|
||
|
} else if URL.Scheme == "https" {
|
||
|
ports = []string{"443"} // as documented
|
||
|
} else if URL.Scheme == "http" {
|
||
|
ports = []string{"80", "443"} // as documented
|
||
|
}
|
||
|
out := []endpointInfo{}
|
||
|
for addr, info := range ipinfo {
|
||
|
if (info.Flags & webconnectivity.ControlIPInfoFlagIsBogon) != 0 {
|
||
|
continue // as documented
|
||
|
}
|
||
|
for _, port := range ports {
|
||
|
epnt := net.JoinHostPort(addr, port)
|
||
|
out = append(out, endpointInfo{
|
||
|
Addr: addr,
|
||
|
Epnt: epnt,
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
// sort the output to make testing work deterministically since iterating
|
||
|
// a map in golang isn't guaranteed to return ordered keys
|
||
|
sort.SliceStable(out, func(i, j int) bool {
|
||
|
return strings.Compare(out[i].Epnt, out[j].Epnt) < 0
|
||
|
})
|
||
|
return out
|
||
|
}
|