2021-06-15 19:25:09 +02:00
|
|
|
// Package mlablocatev2 implements m-lab locate services API v2. This
|
|
|
|
// API currently only allows you to get servers for ndt7. Use the
|
|
|
|
// mlablocate package for all other m-lab tools.
|
2021-02-02 12:05:47 +01:00
|
|
|
package mlablocatev2
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"regexp"
|
|
|
|
|
2021-06-15 11:57:40 +02:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/iox"
|
2021-02-02 12:05:47 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// ndt7URLPath is the URL path to be used for ndt
|
|
|
|
ndt7URLPath = "v2/nearest/ndt/ndt7"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// ErrRequestFailed indicates that the response is not "200 Ok"
|
|
|
|
ErrRequestFailed = errors.New("mlablocatev2: request failed")
|
|
|
|
|
|
|
|
// ErrEmptyResponse indicates that no hosts were returned
|
|
|
|
ErrEmptyResponse = errors.New("mlablocatev2: empty response")
|
|
|
|
)
|
|
|
|
|
2021-06-15 19:25:09 +02:00
|
|
|
// Logger is the logger expected by this package.
|
|
|
|
type Logger interface {
|
|
|
|
// Debugf formats and emits a debug message.
|
|
|
|
Debugf(format string, v ...interface{})
|
|
|
|
}
|
|
|
|
|
|
|
|
// HTTPClient is anything that looks like an http.Client.
|
|
|
|
type HTTPClient interface {
|
|
|
|
// Do behaves like http.Client.Do.
|
|
|
|
Do(req *http.Request) (*http.Response, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Client is a client for v2 of the locate services. Please use the
|
|
|
|
// NewClient factory to construct a new instance of client, otherwise
|
|
|
|
// you MUST fill all the fields marked as MANDATORY.
|
2021-02-02 12:05:47 +01:00
|
|
|
type Client struct {
|
2021-06-15 19:25:09 +02:00
|
|
|
// HTTPClient is the MANDATORY http client to use
|
|
|
|
HTTPClient HTTPClient
|
|
|
|
|
|
|
|
// Hostname is the MANDATORY hostname of the mlablocate API.
|
|
|
|
Hostname string
|
|
|
|
|
|
|
|
// Logger is the MANDATORY logger to use.
|
|
|
|
Logger Logger
|
|
|
|
|
|
|
|
// Scheme is the MANDATORY scheme to use (http or https).
|
|
|
|
Scheme string
|
|
|
|
|
|
|
|
// UserAgent is the MANDATORY user-agent to use.
|
|
|
|
UserAgent string
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewClient creates a client for v2 of the locate services.
|
2021-06-15 19:25:09 +02:00
|
|
|
func NewClient(httpClient HTTPClient, logger Logger, userAgent string) Client {
|
2021-02-02 12:05:47 +01:00
|
|
|
return Client{
|
|
|
|
HTTPClient: httpClient,
|
|
|
|
Hostname: "locate.measurementlab.net",
|
|
|
|
Logger: logger,
|
|
|
|
Scheme: "https",
|
|
|
|
UserAgent: userAgent,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// entryRecord describes one of the boxes returned by v2 of
|
|
|
|
// the locate service. It gives you the FQDN of the specific
|
2021-06-15 19:25:09 +02:00
|
|
|
// box along with URLs for each experiment phase. You MUST
|
|
|
|
// use the URLs directly because they contain access tokens.
|
2021-02-02 12:05:47 +01:00
|
|
|
type entryRecord struct {
|
|
|
|
Machine string `json:"machine"`
|
|
|
|
URLs map[string]string `json:"urls"`
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
// siteRegexp is the regexp to extract the site from the
|
|
|
|
// machine name when the domain is a v2 domain.
|
|
|
|
//
|
|
|
|
// Example: mlab3-mil04.mlab-oti.measurement-lab.org.
|
|
|
|
siteRegexp = regexp.MustCompile(
|
|
|
|
`^(mlab[1-4]d?)-([a-z]{3}[0-9tc]{2})\.([a-z0-9-]{1,16})\.(measurement-lab\.org)$`)
|
|
|
|
)
|
|
|
|
|
|
|
|
// Site returns the site name. If it is not possible to determine
|
|
|
|
// the site name, we return the empty string.
|
|
|
|
func (er entryRecord) Site() string {
|
|
|
|
m := siteRegexp.FindAllStringSubmatch(er.Machine, -1)
|
|
|
|
if len(m) != 1 || len(m[0]) != 5 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return m[0][2]
|
|
|
|
}
|
|
|
|
|
|
|
|
// resultRecord is a result of a query to locate.measurementlab.net.
|
|
|
|
type resultRecord struct {
|
|
|
|
Results []entryRecord `json:"results"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// query performs a locate.measurementlab.net query
|
|
|
|
// using v2 of the locate protocol.
|
|
|
|
func (c Client) query(ctx context.Context, path string) (resultRecord, error) {
|
2021-06-15 19:25:09 +02:00
|
|
|
// TODO(bassosimone): this code should probably be
|
|
|
|
// refactored to use the httpx package.
|
2021-02-02 12:05:47 +01:00
|
|
|
URL := &url.URL{
|
|
|
|
Scheme: c.Scheme,
|
|
|
|
Host: c.Hostname,
|
|
|
|
Path: path,
|
|
|
|
}
|
|
|
|
req, err := http.NewRequestWithContext(ctx, "GET", URL.String(), nil)
|
|
|
|
if err != nil {
|
|
|
|
return resultRecord{}, err
|
|
|
|
}
|
|
|
|
req.Header.Add("User-Agent", c.UserAgent)
|
|
|
|
c.Logger.Debugf("mlablocatev2: GET %s", URL.String())
|
|
|
|
resp, err := c.HTTPClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return resultRecord{}, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return resultRecord{}, fmt.Errorf("%w: %d", ErrRequestFailed, resp.StatusCode)
|
|
|
|
}
|
2021-06-15 11:57:40 +02:00
|
|
|
data, err := iox.ReadAllContext(ctx, resp.Body)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
return resultRecord{}, err
|
|
|
|
}
|
|
|
|
c.Logger.Debugf("mlablocatev2: %s", string(data))
|
|
|
|
var result resultRecord
|
|
|
|
if err := json.Unmarshal(data, &result); err != nil {
|
|
|
|
return resultRecord{}, err
|
|
|
|
}
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NDT7Result is the result of a v2 locate services query for ndt7.
|
|
|
|
type NDT7Result struct {
|
2021-06-15 19:25:09 +02:00
|
|
|
// Hostname is an informative field containing the hostname
|
|
|
|
// to which you're connected. Because there are access tokens,
|
|
|
|
// you cannot use this field directly.
|
|
|
|
Hostname string
|
|
|
|
|
|
|
|
// Site is an informative field containing the site
|
|
|
|
// to which the server belongs to.
|
|
|
|
Site string
|
|
|
|
|
|
|
|
// WSSDownloadURL is the WebSocket URL to be used for
|
|
|
|
// performing a download over HTTPS. Note that the URL
|
|
|
|
// typically includes the required access token.
|
2021-02-02 12:05:47 +01:00
|
|
|
WSSDownloadURL string
|
2021-06-15 19:25:09 +02:00
|
|
|
|
|
|
|
// WSSUploadURL is like WSSDownloadURL but for the upload.
|
|
|
|
WSSUploadURL string
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// QueryNDT7 performs a v2 locate services query for ndt7.
|
|
|
|
func (c Client) QueryNDT7(ctx context.Context) ([]NDT7Result, error) {
|
|
|
|
out, err := c.query(ctx, ndt7URLPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
var result []NDT7Result
|
|
|
|
for _, entry := range out.Results {
|
|
|
|
r := NDT7Result{
|
|
|
|
WSSDownloadURL: entry.URLs["wss:///ndt/v7/download"],
|
|
|
|
WSSUploadURL: entry.URLs["wss:///ndt/v7/upload"],
|
|
|
|
}
|
|
|
|
if r.WSSDownloadURL == "" || r.WSSUploadURL == "" {
|
|
|
|
continue
|
|
|
|
}
|
2021-06-15 19:25:09 +02:00
|
|
|
// Implementation note: we extract the hostname from the
|
|
|
|
// download URL, under the assumption that the download and
|
|
|
|
// the upload URLs have the same hostname.
|
2021-02-02 12:05:47 +01:00
|
|
|
url, err := url.Parse(r.WSSDownloadURL)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
r.Site = entry.Site()
|
|
|
|
r.Hostname = url.Hostname()
|
|
|
|
result = append(result, r)
|
|
|
|
}
|
|
|
|
if len(result) <= 0 {
|
|
|
|
return nil, ErrEmptyResponse
|
|
|
|
}
|
|
|
|
return result, nil
|
|
|
|
}
|