2018-02-13 10:48:46 +01:00
|
|
|
package nettests
|
|
|
|
|
|
|
|
import (
|
2018-09-06 16:13:04 +02:00
|
|
|
"database/sql"
|
2018-03-19 16:23:30 +01:00
|
|
|
"encoding/json"
|
2018-03-22 15:22:29 +01:00
|
|
|
"fmt"
|
2018-05-11 16:06:28 +02:00
|
|
|
"os"
|
2018-03-23 15:26:25 +01:00
|
|
|
"path/filepath"
|
2018-09-07 12:55:27 +02:00
|
|
|
"time"
|
2018-03-19 16:23:30 +01:00
|
|
|
|
2018-02-13 10:48:46 +01:00
|
|
|
"github.com/apex/log"
|
2018-06-29 16:50:05 +02:00
|
|
|
"github.com/fatih/color"
|
2018-02-13 10:48:46 +01:00
|
|
|
"github.com/measurement-kit/go-measurement-kit"
|
2018-05-03 14:59:55 +02:00
|
|
|
ooni "github.com/ooni/probe-cli"
|
2018-09-10 16:29:14 +02:00
|
|
|
"github.com/ooni/probe-cli/internal/crashreport"
|
2018-05-03 14:59:55 +02:00
|
|
|
"github.com/ooni/probe-cli/internal/database"
|
|
|
|
"github.com/ooni/probe-cli/internal/output"
|
|
|
|
"github.com/ooni/probe-cli/utils"
|
2018-09-10 16:29:14 +02:00
|
|
|
"github.com/ooni/probe-cli/utils/strcase"
|
2018-02-13 10:48:46 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// Nettest interface. Every Nettest should implement this.
|
|
|
|
type Nettest interface {
|
|
|
|
Run(*Controller) error
|
2018-09-10 12:41:28 +02:00
|
|
|
GetTestKeys(map[string]interface{}) interface{}
|
2018-02-13 10:48:46 +01:00
|
|
|
LogSummary(string) error
|
|
|
|
}
|
|
|
|
|
2018-02-13 16:16:23 +01:00
|
|
|
// NewController creates a nettest controller
|
2018-09-07 12:55:27 +02:00
|
|
|
func NewController(nt Nettest, ctx *ooni.Context, res *database.Result) *Controller {
|
|
|
|
msmtPath := filepath.Join(ctx.TempDir,
|
|
|
|
fmt.Sprintf("msmt-%T-%s.jsonl", nt,
|
|
|
|
time.Now().UTC().Format(utils.ResultTimestamp)))
|
2018-02-13 10:48:46 +01:00
|
|
|
return &Controller{
|
2018-03-19 16:23:30 +01:00
|
|
|
Ctx: ctx,
|
|
|
|
nt: nt,
|
|
|
|
res: res,
|
|
|
|
msmtPath: msmtPath,
|
2018-02-13 10:48:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-13 16:16:23 +01:00
|
|
|
// Controller is passed to the run method of every Nettest
|
2018-03-19 13:20:42 +01:00
|
|
|
// each nettest instance has one controller
|
2018-02-13 16:16:23 +01:00
|
|
|
type Controller struct {
|
2018-09-07 12:55:27 +02:00
|
|
|
Ctx *ooni.Context
|
|
|
|
res *database.Result
|
|
|
|
nt Nettest
|
|
|
|
msmts map[int64]*database.Measurement
|
|
|
|
msmtPath string // XXX maybe we can drop this and just use a temporary file
|
|
|
|
inputIdxMap map[int64]int64 // Used to map mk idx to database id
|
2018-02-13 16:16:23 +01:00
|
|
|
}
|
|
|
|
|
2018-05-22 02:33:59 +02:00
|
|
|
func getCaBundlePath() string {
|
2018-05-11 16:06:28 +02:00
|
|
|
path := os.Getenv("SSL_CERT_FILE")
|
|
|
|
if path != "" {
|
|
|
|
return path
|
|
|
|
}
|
|
|
|
return "/etc/ssl/cert.pem"
|
|
|
|
}
|
|
|
|
|
2018-09-07 12:55:27 +02:00
|
|
|
func (c *Controller) SetInputIdxMap(inputIdxMap map[int64]int64) error {
|
|
|
|
c.inputIdxMap = inputIdxMap
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-09-12 13:42:16 +02:00
|
|
|
type StatusEnd struct {
|
|
|
|
DownloadedKB float64 `json:"download_kb"`
|
|
|
|
UploadedKB float64 `json:"uploaded_kb"`
|
|
|
|
Failure string `json:"failure"`
|
|
|
|
}
|
|
|
|
|
2018-02-13 10:48:46 +01:00
|
|
|
// Init should be called once to initialise the nettest
|
2018-03-19 13:20:42 +01:00
|
|
|
func (c *Controller) Init(nt *mk.Nettest) error {
|
2018-03-08 11:53:04 +01:00
|
|
|
log.Debugf("Init: %v", nt)
|
2018-03-23 12:41:06 +01:00
|
|
|
c.Ctx.LocationLookup()
|
|
|
|
|
2018-03-19 19:28:32 +01:00
|
|
|
c.msmts = make(map[int64]*database.Measurement)
|
2018-03-19 13:20:42 +01:00
|
|
|
|
2018-09-07 12:55:27 +02:00
|
|
|
// These values are shared by every measurement
|
|
|
|
reportID := sql.NullString{String: "", Valid: false}
|
2018-09-10 16:29:14 +02:00
|
|
|
testName := strcase.ToSnake(nt.Name)
|
2018-09-07 12:55:27 +02:00
|
|
|
resultID := c.res.ID
|
|
|
|
reportFilePath := c.msmtPath
|
2018-03-19 16:23:30 +01:00
|
|
|
|
2018-05-31 11:07:16 +02:00
|
|
|
// This is to workaround homedirs having UTF-8 characters in them.
|
|
|
|
// See: https://github.com/measurement-kit/measurement-kit/issues/1635
|
|
|
|
geoIPCountryPath := filepath.Join(utils.GeoIPDir(c.Ctx.Home), "GeoIP.dat")
|
|
|
|
geoIPASNPath := filepath.Join(utils.GeoIPDir(c.Ctx.Home), "GeoIPASNum.dat")
|
|
|
|
caBundlePath := getCaBundlePath()
|
|
|
|
msmtPath := c.msmtPath
|
|
|
|
|
2018-07-09 19:17:59 +02:00
|
|
|
userHome, err := utils.GetOONIHome()
|
2018-05-31 11:07:16 +02:00
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("failed to figure out the homedir")
|
|
|
|
return err
|
|
|
|
}
|
2018-07-09 19:17:59 +02:00
|
|
|
// Get the parent of it
|
|
|
|
userHome = filepath.Dir(userHome)
|
2018-05-31 11:07:16 +02:00
|
|
|
|
|
|
|
relPath, err := filepath.Rel(userHome, caBundlePath)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("caBundlePath is not relative to the users home")
|
|
|
|
} else {
|
|
|
|
caBundlePath = relPath
|
|
|
|
}
|
|
|
|
relPath, err = filepath.Rel(userHome, geoIPASNPath)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("geoIPASNPath is not relative to the users home")
|
|
|
|
} else {
|
|
|
|
geoIPASNPath = relPath
|
|
|
|
}
|
|
|
|
relPath, err = filepath.Rel(userHome, geoIPCountryPath)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("geoIPCountryPath is not relative to the users home")
|
|
|
|
} else {
|
|
|
|
geoIPCountryPath = relPath
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debugf("Chdir to: %s", userHome)
|
|
|
|
if err := os.Chdir(userHome); err != nil {
|
|
|
|
log.WithError(err).Errorf("failed to chdir to %s", userHome)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debugf("OutputPath: %s", msmtPath)
|
2018-02-13 16:16:23 +01:00
|
|
|
nt.Options = mk.NettestOptions{
|
2018-03-23 12:41:06 +01:00
|
|
|
IncludeIP: c.Ctx.Config.Sharing.IncludeIP,
|
|
|
|
IncludeASN: c.Ctx.Config.Sharing.IncludeASN,
|
|
|
|
IncludeCountry: c.Ctx.Config.Advanced.IncludeCountry,
|
2018-09-10 12:41:28 +02:00
|
|
|
LogLevel: "DEBUG",
|
2018-03-23 12:41:06 +01:00
|
|
|
|
|
|
|
ProbeCC: c.Ctx.Location.CountryCode,
|
|
|
|
ProbeASN: fmt.Sprintf("AS%d", c.Ctx.Location.ASN),
|
|
|
|
ProbeIP: c.Ctx.Location.IP,
|
|
|
|
|
2018-05-31 12:02:32 +02:00
|
|
|
DisableReportFile: false,
|
|
|
|
DisableCollector: false,
|
2018-09-10 17:37:47 +02:00
|
|
|
RandomizeInput: false, // It's important to disable input randomization to ensure the URLs are written in sync to the DB
|
2018-05-31 12:02:32 +02:00
|
|
|
SoftwareName: "ooniprobe",
|
2018-07-14 16:58:08 +02:00
|
|
|
SoftwareVersion: ooni.Version,
|
2018-02-13 16:16:23 +01:00
|
|
|
|
2018-05-31 11:07:16 +02:00
|
|
|
OutputPath: msmtPath,
|
|
|
|
GeoIPCountryPath: geoIPCountryPath,
|
|
|
|
GeoIPASNPath: geoIPASNPath,
|
|
|
|
CaBundlePath: caBundlePath,
|
2018-02-13 16:16:23 +01:00
|
|
|
}
|
2018-05-31 11:07:16 +02:00
|
|
|
log.Debugf("GeoIPASNPath: %s", nt.Options.GeoIPASNPath)
|
|
|
|
log.Debugf("GeoIPCountryPath: %s", nt.Options.GeoIPCountryPath)
|
2018-03-19 13:20:42 +01:00
|
|
|
|
2018-03-08 13:46:21 +01:00
|
|
|
nt.On("log", func(e mk.Event) {
|
2018-03-19 19:28:32 +01:00
|
|
|
level := e.Value.LogLevel
|
|
|
|
msg := e.Value.Message
|
2018-03-08 13:46:21 +01:00
|
|
|
|
|
|
|
switch level {
|
|
|
|
case "ERROR":
|
2018-09-10 12:41:28 +02:00
|
|
|
log.Errorf("%v: %s", color.RedString("mklog"), msg)
|
2018-03-08 13:46:21 +01:00
|
|
|
case "INFO":
|
2018-09-10 12:41:28 +02:00
|
|
|
log.Infof("%v: %s", color.BlueString("mklog"), msg)
|
2018-03-08 13:46:21 +01:00
|
|
|
default:
|
2018-09-10 12:41:28 +02:00
|
|
|
log.Debugf("%v: %s", color.WhiteString("mklog"), msg)
|
2018-02-21 16:06:30 +01:00
|
|
|
}
|
2018-03-08 13:46:21 +01:00
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("status.queued", func(e mk.Event) {
|
|
|
|
log.Debugf("%s", e.Key)
|
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("status.started", func(e mk.Event) {
|
|
|
|
log.Debugf("%s", e.Key)
|
2018-02-13 16:16:23 +01:00
|
|
|
})
|
2018-03-08 13:46:21 +01:00
|
|
|
|
|
|
|
nt.On("status.report_created", func(e mk.Event) {
|
|
|
|
log.Debugf("%s", e.Key)
|
2018-03-19 16:23:30 +01:00
|
|
|
|
2018-09-07 12:55:27 +02:00
|
|
|
reportID = sql.NullString{String: e.Value.ReportID, Valid: true}
|
2018-03-08 13:46:21 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("status.geoip_lookup", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-03-19 16:23:30 +01:00
|
|
|
})
|
|
|
|
|
2018-05-31 12:02:32 +02:00
|
|
|
nt.On("status.measurement_start", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-03-19 16:23:30 +01:00
|
|
|
|
2018-03-19 19:28:32 +01:00
|
|
|
idx := e.Value.Idx
|
2018-09-07 12:55:27 +02:00
|
|
|
urlID := sql.NullInt64{Int64: 0, Valid: false}
|
|
|
|
if c.inputIdxMap != nil {
|
|
|
|
urlID = sql.NullInt64{Int64: c.inputIdxMap[idx], Valid: true}
|
|
|
|
}
|
|
|
|
msmt, err := database.CreateMeasurement(c.Ctx.DB, reportID, testName, resultID, reportFilePath, urlID)
|
2018-03-19 16:23:30 +01:00
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("Failed to create measurement")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.msmts[idx] = msmt
|
2018-03-08 13:46:21 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("status.progress", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-03-19 19:28:32 +01:00
|
|
|
c.OnProgress(e.Value.Percentage, e.Value.Message)
|
2018-03-08 13:46:21 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("status.update.*", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-03-08 13:46:21 +01:00
|
|
|
})
|
|
|
|
|
2018-06-29 15:41:12 +02:00
|
|
|
// XXX should these be made into permanent failures?
|
|
|
|
nt.On("failure.asn_lookup", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-06-29 15:41:12 +02:00
|
|
|
log.Debugf("%v", e.Value)
|
|
|
|
})
|
|
|
|
nt.On("failure.cc_lookup", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-06-29 15:41:12 +02:00
|
|
|
log.Debugf("%v", e.Value)
|
|
|
|
})
|
|
|
|
nt.On("failure.ip_lookup", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-06-29 15:41:12 +02:00
|
|
|
log.Debugf("%v", e.Value)
|
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("failure.resolver_lookup", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-06-29 15:41:12 +02:00
|
|
|
log.Debugf("%v", e.Value)
|
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("failure.report_create", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-06-29 15:41:12 +02:00
|
|
|
log.Debugf("%v", e.Value)
|
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("failure.report_close", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-06-29 15:41:12 +02:00
|
|
|
log.Debugf("%v", e.Value)
|
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("failure.startup", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-06-29 15:41:12 +02:00
|
|
|
|
|
|
|
c.msmts[e.Value.Idx].Failed(c.Ctx.DB, e.Value.Failure)
|
|
|
|
})
|
|
|
|
|
2018-03-08 13:46:21 +01:00
|
|
|
nt.On("failure.measurement", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-03-19 16:23:30 +01:00
|
|
|
|
2018-03-19 19:28:32 +01:00
|
|
|
c.msmts[e.Value.Idx].Failed(c.Ctx.DB, e.Value.Failure)
|
2018-03-19 16:23:30 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("failure.measurement_submission", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-03-19 16:23:30 +01:00
|
|
|
|
2018-03-19 19:28:32 +01:00
|
|
|
failure := e.Value.Failure
|
|
|
|
c.msmts[e.Value.Idx].UploadFailed(c.Ctx.DB, failure)
|
2018-03-19 16:23:30 +01:00
|
|
|
})
|
|
|
|
|
2018-05-31 12:02:32 +02:00
|
|
|
nt.On("status.measurement_submission", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-03-19 16:23:30 +01:00
|
|
|
|
2018-03-20 12:38:33 +01:00
|
|
|
if err := c.msmts[e.Value.Idx].UploadSucceeded(c.Ctx.DB); err != nil {
|
|
|
|
log.WithError(err).Error("failed to mark msmt as uploaded")
|
|
|
|
}
|
2018-03-08 13:46:21 +01:00
|
|
|
})
|
|
|
|
|
2018-03-19 16:23:30 +01:00
|
|
|
nt.On("status.measurement_done", func(e mk.Event) {
|
2018-06-29 16:50:05 +02:00
|
|
|
log.Debugf(color.RedString(e.Key))
|
2018-03-19 16:23:30 +01:00
|
|
|
|
2018-03-20 12:38:33 +01:00
|
|
|
if err := c.msmts[e.Value.Idx].Done(c.Ctx.DB); err != nil {
|
|
|
|
log.WithError(err).Error("failed to mark msmt as done")
|
|
|
|
}
|
2018-03-08 13:46:21 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
nt.On("measurement", func(e mk.Event) {
|
2018-06-29 15:41:12 +02:00
|
|
|
log.Debugf("status.end")
|
|
|
|
|
2018-09-10 16:29:14 +02:00
|
|
|
crashreport.CapturePanicAndWait(func() {
|
|
|
|
c.OnEntry(e.Value.Idx, e.Value.JSONStr)
|
|
|
|
}, nil)
|
2018-03-08 13:46:21 +01:00
|
|
|
})
|
|
|
|
|
2018-03-20 12:38:33 +01:00
|
|
|
nt.On("status.end", func(e mk.Event) {
|
|
|
|
log.Debugf("status.end")
|
2018-09-12 13:42:16 +02:00
|
|
|
|
2018-03-20 12:38:33 +01:00
|
|
|
for idx, msmt := range c.msmts {
|
|
|
|
log.Debugf("adding msmt#%d to result", idx)
|
|
|
|
if err := msmt.AddToResult(c.Ctx.DB, c.res); err != nil {
|
|
|
|
log.WithError(err).Error("failed to add to result")
|
|
|
|
}
|
|
|
|
}
|
2018-09-12 13:42:16 +02:00
|
|
|
|
|
|
|
var endMsg StatusEnd
|
|
|
|
err := json.Unmarshal([]byte(e.Value.JSONStr), &endMsg)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Errorf("failed to extract status.end message %s", e.Value.JSONStr)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if endMsg.Failure != "" {
|
|
|
|
log.Errorf("Failure in status.end: %s", endMsg.Failure)
|
|
|
|
}
|
|
|
|
|
|
|
|
c.res.DataUsageDown += endMsg.DownloadedKB
|
|
|
|
c.res.DataUsageDown += endMsg.UploadedKB
|
2018-03-19 13:20:42 +01:00
|
|
|
})
|
|
|
|
|
2018-09-10 12:41:28 +02:00
|
|
|
log.Debugf("Registered all the handlers")
|
2018-03-19 13:20:42 +01:00
|
|
|
return nil
|
2018-02-13 10:48:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// OnProgress should be called when a new progress event is available.
|
2018-03-08 13:46:21 +01:00
|
|
|
func (c *Controller) OnProgress(perc float64, msg string) {
|
2018-02-13 10:48:46 +01:00
|
|
|
log.Debugf("OnProgress: %f - %s", perc, msg)
|
2018-03-22 15:22:29 +01:00
|
|
|
|
|
|
|
key := fmt.Sprintf("%T", c.nt)
|
|
|
|
output.Progress(key, perc, msg)
|
2018-02-13 10:48:46 +01:00
|
|
|
}
|
|
|
|
|
2018-03-19 16:23:30 +01:00
|
|
|
// Entry is an opaque measurement entry
|
|
|
|
type Entry struct {
|
|
|
|
TestKeys map[string]interface{} `json:"test_keys"`
|
|
|
|
}
|
|
|
|
|
2018-02-13 10:48:46 +01:00
|
|
|
// OnEntry should be called every time there is a new entry
|
2018-03-19 16:23:30 +01:00
|
|
|
func (c *Controller) OnEntry(idx int64, jsonStr string) {
|
2018-03-19 19:28:32 +01:00
|
|
|
log.Debugf("OnEntry")
|
2018-03-19 16:23:30 +01:00
|
|
|
|
|
|
|
var entry Entry
|
2018-09-12 13:42:16 +02:00
|
|
|
if err := json.Unmarshal([]byte(jsonStr), &entry); err != nil {
|
|
|
|
log.WithError(err).Error("failed to parse onEntry")
|
|
|
|
return
|
|
|
|
}
|
2018-09-10 12:41:28 +02:00
|
|
|
tk := c.nt.GetTestKeys(entry.TestKeys)
|
|
|
|
|
|
|
|
log.Debugf("Fetching: %s %v", idx, c.msmts[idx])
|
|
|
|
err := database.AddTestKeys(c.Ctx.DB, c.msmts[idx], tk)
|
2018-03-19 16:23:30 +01:00
|
|
|
if err != nil {
|
2018-09-10 12:41:28 +02:00
|
|
|
log.WithError(err).Error("failed to add test keys to summary")
|
2018-03-19 16:23:30 +01:00
|
|
|
}
|
2018-02-13 10:48:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// MKStart is the interface for the mk.Nettest Start() function
|
|
|
|
type MKStart func(name string) (chan bool, error)
|
|
|
|
|
2018-02-21 16:06:30 +01:00
|
|
|
// Start should be called to start the test
|
2018-02-13 10:48:46 +01:00
|
|
|
func (c *Controller) Start(f MKStart) {
|
|
|
|
log.Debugf("MKStart: %s", f)
|
|
|
|
}
|