ooni-probe-cli/nettests/nettests.go

312 lines
7.9 KiB
Go
Raw Normal View History

2018-02-13 10:48:46 +01:00
package nettests
import (
"encoding/json"
"fmt"
"os"
"path/filepath"
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"
ooni "github.com/ooni/probe-cli"
"github.com/ooni/probe-cli/internal/database"
"github.com/ooni/probe-cli/internal/output"
"github.com/ooni/probe-cli/utils"
2018-02-13 10:48:46 +01:00
)
// Nettest interface. Every Nettest should implement this.
type Nettest interface {
Run(*Controller) error
2018-02-13 16:16:23 +01:00
Summary(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
func NewController(nt Nettest, ctx *ooni.Context, res *database.Result, msmtPath string) *Controller {
2018-02-13 10:48:46 +01:00
return &Controller{
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
// each nettest instance has one controller
2018-02-13 16:16:23 +01:00
type Controller struct {
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
2018-02-13 16:16:23 +01:00
}
func getCaBundlePath() string {
path := os.Getenv("SSL_CERT_FILE")
if path != "" {
return path
}
return "/etc/ssl/cert.pem"
}
2018-02-13 10:48:46 +01:00
// Init should be called once to initialise the nettest
func (c *Controller) Init(nt *mk.Nettest) error {
2018-03-08 11:53:04 +01:00
log.Debugf("Init: %v", nt)
c.Ctx.LocationLookup()
2018-03-19 19:28:32 +01:00
c.msmts = make(map[int64]*database.Measurement)
msmtTemplate := database.Measurement{
ReportID: "",
TestName: nt.Name,
ResultID: c.res.ID,
ReportFilePath: c.msmtPath,
}
// 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
userHome, err := utils.GetOONIHome()
if err != nil {
log.WithError(err).Error("failed to figure out the homedir")
return err
}
// Get the parent of it
userHome = filepath.Dir(userHome)
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{
IncludeIP: c.Ctx.Config.Sharing.IncludeIP,
IncludeASN: c.Ctx.Config.Sharing.IncludeASN,
IncludeCountry: c.Ctx.Config.Advanced.IncludeCountry,
2018-05-31 12:02:32 +02:00
LogLevel: "INFO",
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,
SoftwareName: "ooniprobe",
2018-07-14 16:58:08 +02:00
SoftwareVersion: ooni.Version,
2018-02-13 16:16:23 +01:00
OutputPath: msmtPath,
GeoIPCountryPath: geoIPCountryPath,
GeoIPASNPath: geoIPASNPath,
CaBundlePath: caBundlePath,
2018-02-13 16:16:23 +01:00
}
log.Debugf("GeoIPASNPath: %s", nt.Options.GeoIPASNPath)
log.Debugf("GeoIPCountryPath: %s", nt.Options.GeoIPCountryPath)
2018-03-08 13:46:21 +01:00
nt.On("log", 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
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":
log.Error(msg)
case "INFO":
log.Info(msg)
default:
log.Debug(msg)
}
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 19:28:32 +01:00
msmtTemplate.ReportID = e.Value.ReportID
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))
/* FIXME
Put this into the network table
2018-03-19 19:28:32 +01:00
msmtTemplate.ASN = e.Value.ProbeASN
msmtTemplate.IP = e.Value.ProbeIP
msmtTemplate.CountryCode = e.Value.ProbeCC
*/
})
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 19:28:32 +01:00
idx := e.Value.Idx
msmt, err := database.CreateMeasurement(c.Ctx.DB, msmtTemplate, e.Value.Input)
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 19:28:32 +01:00
c.msmts[e.Value.Idx].Failed(c.Ctx.DB, e.Value.Failure)
})
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 19:28:32 +01:00
failure := e.Value.Failure
c.msmts[e.Value.Idx].UploadFailed(c.Ctx.DB, failure)
})
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))
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
})
nt.On("status.measurement_done", func(e mk.Event) {
2018-06-29 16:50:05 +02:00
log.Debugf(color.RedString(e.Key))
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-03-19 19:28:32 +01:00
c.OnEntry(e.Value.Idx, e.Value.JSONStr)
2018-03-08 13:46:21 +01:00
})
nt.On("status.end", func(e mk.Event) {
log.Debugf("status.end")
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")
}
}
})
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)
key := fmt.Sprintf("%T", c.nt)
output.Progress(key, perc, msg)
2018-02-13 10:48:46 +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
func (c *Controller) OnEntry(idx int64, jsonStr string) {
2018-03-19 19:28:32 +01:00
log.Debugf("OnEntry")
var entry Entry
json.Unmarshal([]byte(jsonStr), &entry)
summary := c.nt.Summary(entry.TestKeys)
summaryBytes, err := json.Marshal(summary)
if err != nil {
log.WithError(err).Error("failed to serialize summary")
}
2018-03-19 19:28:32 +01:00
log.Debugf("Fetching: %s %v", idx, c.msmts[idx])
c.msmts[idx].WriteSummary(c.Ctx.DB, string(summaryBytes))
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)
// 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)
}