2018-02-13 10:48:46 +01:00
|
|
|
package websites
|
|
|
|
|
|
|
|
import (
|
2018-03-23 12:41:06 +01:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
|
2018-09-07 12:55:27 +02:00
|
|
|
"github.com/apex/log"
|
2018-02-13 16:16:23 +01:00
|
|
|
"github.com/measurement-kit/go-measurement-kit"
|
2018-09-07 12:55:27 +02:00
|
|
|
"github.com/ooni/probe-cli/internal/database"
|
2018-05-03 14:59:55 +02:00
|
|
|
"github.com/ooni/probe-cli/nettests"
|
2018-03-23 12:41:06 +01:00
|
|
|
"github.com/pkg/errors"
|
2018-02-13 10:48:46 +01:00
|
|
|
)
|
|
|
|
|
2018-03-23 13:17:39 +01:00
|
|
|
// URLInfo contains the URL and the citizenlab category code for that URL
|
2018-03-23 12:41:06 +01:00
|
|
|
type URLInfo struct {
|
|
|
|
URL string `json:"url"`
|
2018-09-07 12:55:27 +02:00
|
|
|
CountryCode string `json:"country_code"`
|
2018-03-23 12:41:06 +01:00
|
|
|
CategoryCode string `json:"category_code"`
|
|
|
|
}
|
|
|
|
|
2018-03-23 13:17:39 +01:00
|
|
|
// URLResponse is the orchestrate url response containing a list of URLs
|
2018-03-23 12:41:06 +01:00
|
|
|
type URLResponse struct {
|
|
|
|
Results []URLInfo `json:"results"`
|
|
|
|
}
|
|
|
|
|
|
|
|
const orchestrateBaseURL = "https://events.proteus.test.ooni.io"
|
|
|
|
|
2018-09-07 12:55:27 +02:00
|
|
|
func lookupURLs(ctl *nettests.Controller) ([]string, map[int64]int64, error) {
|
2018-03-23 12:41:06 +01:00
|
|
|
var (
|
2018-09-10 12:41:28 +02:00
|
|
|
parsed = new(URLResponse)
|
|
|
|
urls []string
|
2018-03-23 12:41:06 +01:00
|
|
|
)
|
2018-09-10 12:41:28 +02:00
|
|
|
urlIDMap := make(map[int64]int64)
|
|
|
|
log.Debug("Looking up URLs")
|
2018-06-27 15:45:35 +02:00
|
|
|
// XXX pass in the configuration for category codes
|
2018-03-23 12:41:06 +01:00
|
|
|
reqURL := fmt.Sprintf("%s/api/v1/urls?probe_cc=%s",
|
|
|
|
orchestrateBaseURL,
|
|
|
|
ctl.Ctx.Location.CountryCode)
|
|
|
|
|
|
|
|
resp, err := http.Get(reqURL)
|
|
|
|
if err != nil {
|
2018-09-07 12:55:27 +02:00
|
|
|
return urls, urlIDMap, errors.Wrap(err, "failed to perform request")
|
2018-03-23 12:41:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
2018-09-07 12:55:27 +02:00
|
|
|
return urls, urlIDMap, errors.Wrap(err, "failed to read response body")
|
2018-03-23 12:41:06 +01:00
|
|
|
}
|
|
|
|
err = json.Unmarshal([]byte(body), &parsed)
|
|
|
|
if err != nil {
|
2018-09-07 12:55:27 +02:00
|
|
|
return urls, urlIDMap, errors.Wrap(err, "failed to parse json")
|
2018-03-23 12:41:06 +01:00
|
|
|
}
|
|
|
|
|
2018-09-07 12:55:27 +02:00
|
|
|
for idx, url := range parsed.Results {
|
2018-09-10 12:41:28 +02:00
|
|
|
log.Debugf("Going over URL %d", idx)
|
2018-09-07 15:23:29 +02:00
|
|
|
urlID, err := database.CreateOrUpdateURL(ctl.Ctx.DB, url.URL, url.CategoryCode, url.CountryCode)
|
2018-09-07 12:55:27 +02:00
|
|
|
if err != nil {
|
2018-09-07 15:23:29 +02:00
|
|
|
log.Error("failed to add to the URL table")
|
2018-09-07 12:55:27 +02:00
|
|
|
}
|
|
|
|
urlIDMap[int64(idx)] = urlID
|
2018-03-23 12:41:06 +01:00
|
|
|
urls = append(urls, url.URL)
|
|
|
|
}
|
2018-09-07 12:55:27 +02:00
|
|
|
return urls, urlIDMap, nil
|
2018-03-23 12:41:06 +01:00
|
|
|
}
|
|
|
|
|
2018-02-13 10:48:46 +01:00
|
|
|
// WebConnectivity test implementation
|
|
|
|
type WebConnectivity struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run starts the test
|
|
|
|
func (n WebConnectivity) Run(ctl *nettests.Controller) error {
|
2018-03-19 17:29:32 +01:00
|
|
|
nt := mk.NewNettest("WebConnectivity")
|
|
|
|
ctl.Init(nt)
|
2018-03-23 12:41:06 +01:00
|
|
|
|
2018-09-07 12:55:27 +02:00
|
|
|
urls, urlIDMap, err := lookupURLs(ctl)
|
2018-03-23 12:41:06 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-09-07 12:55:27 +02:00
|
|
|
ctl.SetInputIdxMap(urlIDMap)
|
2018-03-23 12:41:06 +01:00
|
|
|
nt.Options.Inputs = urls
|
|
|
|
|
2018-02-13 16:16:23 +01:00
|
|
|
return nt.Run()
|
2018-02-13 10:48:46 +01:00
|
|
|
}
|
|
|
|
|
2018-09-10 12:41:28 +02:00
|
|
|
// WebConnectivityTestKeys for the test
|
|
|
|
type WebConnectivityTestKeys struct {
|
|
|
|
Accessible bool `json:"accessible"`
|
|
|
|
Blocking string `json:"blocking"`
|
|
|
|
IsAnomaly bool `json:"-"`
|
2018-03-23 13:17:39 +01:00
|
|
|
}
|
|
|
|
|
2018-09-10 12:41:28 +02:00
|
|
|
// GetTestKeys generates a summary for a test run
|
|
|
|
func (n WebConnectivity) GetTestKeys(tk map[string]interface{}) interface{} {
|
2018-03-23 13:17:39 +01:00
|
|
|
var (
|
|
|
|
blocked bool
|
|
|
|
blocking string
|
|
|
|
accessible bool
|
|
|
|
)
|
|
|
|
|
|
|
|
// We need to do these complicated type assertions, because some of the fields
|
|
|
|
// are "nullable" and/or can be of different types
|
|
|
|
switch v := tk["blocking"].(type) {
|
|
|
|
case bool:
|
|
|
|
blocked = false
|
|
|
|
blocking = "none"
|
|
|
|
case string:
|
|
|
|
blocked = true
|
|
|
|
blocking = v
|
|
|
|
default:
|
|
|
|
blocked = false
|
|
|
|
blocking = "none"
|
|
|
|
}
|
|
|
|
|
|
|
|
if tk["accessible"] == nil {
|
|
|
|
accessible = false
|
|
|
|
} else {
|
|
|
|
accessible = tk["accessible"].(bool)
|
|
|
|
}
|
|
|
|
|
2018-09-10 12:41:28 +02:00
|
|
|
return WebConnectivityTestKeys{
|
2018-03-23 13:17:39 +01:00
|
|
|
Accessible: accessible,
|
|
|
|
Blocking: blocking,
|
2018-09-10 12:41:28 +02:00
|
|
|
IsAnomaly: blocked,
|
2018-03-23 13:17:39 +01:00
|
|
|
}
|
2018-02-13 10:48:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// LogSummary writes the summary to the standard output
|
|
|
|
func (n WebConnectivity) LogSummary(s string) error {
|
|
|
|
return nil
|
|
|
|
}
|