2018-02-12 17:29:03 +01:00
|
|
|
package database
|
|
|
|
|
2018-02-13 17:11:22 +01:00
|
|
|
import (
|
2018-03-20 12:38:33 +01:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2018-02-13 17:11:22 +01:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/apex/log"
|
|
|
|
"github.com/jmoiron/sqlx"
|
2018-05-03 14:59:55 +02:00
|
|
|
"github.com/ooni/probe-cli/utils"
|
2018-02-13 17:11:22 +01:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
2018-02-12 17:29:03 +01:00
|
|
|
|
2018-03-20 14:19:19 +01:00
|
|
|
// ResultSummaryFunc is the function used to generate result summaries
|
|
|
|
type ResultSummaryFunc func(SummaryMap) (string, error)
|
|
|
|
|
|
|
|
// SummaryMap contains a mapping from test name to serialized summary for it
|
2018-03-23 14:58:33 +01:00
|
|
|
type SummaryMap map[string][]string
|
2018-03-20 14:19:19 +01:00
|
|
|
|
2018-03-19 16:23:30 +01:00
|
|
|
// UpdateOne will run the specified update query and check that it only affected one row
|
|
|
|
func UpdateOne(db *sqlx.DB, query string, arg interface{}) error {
|
|
|
|
res, err := db.NamedExec(query, arg)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "updating table")
|
|
|
|
}
|
|
|
|
count, err := res.RowsAffected()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "updating table")
|
|
|
|
}
|
|
|
|
if count != 1 {
|
|
|
|
return errors.New("inconsistent update count")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-06-22 12:12:35 +02:00
|
|
|
// ListMeasurements given a result ID
|
|
|
|
func ListMeasurements(db *sqlx.DB, resultID int64) ([]*Measurement, error) {
|
|
|
|
measurements := []*Measurement{}
|
|
|
|
|
|
|
|
rows, err := db.Query(`SELECT id, name,
|
|
|
|
start_time, runtime,
|
|
|
|
country,
|
|
|
|
asn,
|
|
|
|
summary,
|
|
|
|
input
|
|
|
|
FROM measurements
|
|
|
|
WHERE result_id = ?
|
|
|
|
ORDER BY start_time;`, resultID)
|
|
|
|
if err != nil {
|
|
|
|
return measurements, errors.Wrap(err, "failed to get measurement list")
|
|
|
|
}
|
|
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
msmt := Measurement{}
|
|
|
|
err = rows.Scan(&msmt.ID, &msmt.Name,
|
|
|
|
&msmt.StartTime, &msmt.Runtime,
|
|
|
|
&msmt.CountryCode,
|
|
|
|
&msmt.ASN,
|
|
|
|
&msmt.Summary, &msmt.Input,
|
|
|
|
//&result.DataUsageUp, &result.DataUsageDown)
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("failed to fetch a row")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
measurements = append(measurements, &msmt)
|
|
|
|
}
|
|
|
|
|
|
|
|
return measurements, nil
|
|
|
|
}
|
|
|
|
|
2018-02-12 17:29:03 +01:00
|
|
|
// Measurement model
|
|
|
|
type Measurement struct {
|
2018-02-13 17:11:22 +01:00
|
|
|
ID int64 `db:"id"`
|
2018-02-12 17:29:03 +01:00
|
|
|
Name string `db:"name"`
|
2018-02-13 17:11:22 +01:00
|
|
|
StartTime time.Time `db:"start_time"`
|
2018-03-20 12:38:33 +01:00
|
|
|
Runtime float64 `db:"runtime"` // Fractional number of seconds
|
2018-02-12 17:29:03 +01:00
|
|
|
Summary string `db:"summary"` // XXX this should be JSON
|
2018-03-19 16:23:30 +01:00
|
|
|
ASN string `db:"asn"`
|
2018-02-12 17:29:03 +01:00
|
|
|
IP string `db:"ip"`
|
|
|
|
CountryCode string `db:"country"`
|
|
|
|
State string `db:"state"`
|
|
|
|
Failure string `db:"failure"`
|
2018-03-19 16:23:30 +01:00
|
|
|
UploadFailure string `db:"upload_failure"`
|
|
|
|
Uploaded bool `db:"uploaded"`
|
2018-02-13 17:11:22 +01:00
|
|
|
ReportFilePath string `db:"report_file"`
|
|
|
|
ReportID string `db:"report_id"`
|
2018-02-12 17:29:03 +01:00
|
|
|
Input string `db:"input"`
|
2018-03-19 16:23:30 +01:00
|
|
|
ResultID int64 `db:"result_id"`
|
2018-02-13 17:11:22 +01:00
|
|
|
}
|
|
|
|
|
2018-03-08 13:46:21 +01:00
|
|
|
// SetGeoIPInfo for the Measurement
|
|
|
|
func (m *Measurement) SetGeoIPInfo() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-03-19 16:23:30 +01:00
|
|
|
// Failed writes the error string to the measurement
|
|
|
|
func (m *Measurement) Failed(db *sqlx.DB, failure string) error {
|
|
|
|
m.Failure = failure
|
|
|
|
|
|
|
|
err := UpdateOne(db, `UPDATE measurements
|
|
|
|
SET failure = :failure, state = :state
|
|
|
|
WHERE id = :id`, m)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "updating measurement")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Done marks the measurement as completed
|
|
|
|
func (m *Measurement) Done(db *sqlx.DB) error {
|
2018-03-20 12:38:33 +01:00
|
|
|
runtime := time.Now().UTC().Sub(m.StartTime)
|
|
|
|
m.Runtime = runtime.Seconds()
|
2018-03-19 16:23:30 +01:00
|
|
|
m.State = "done"
|
|
|
|
|
|
|
|
err := UpdateOne(db, `UPDATE measurements
|
2018-03-20 12:38:33 +01:00
|
|
|
SET state = :state, runtime = :runtime
|
2018-03-19 16:23:30 +01:00
|
|
|
WHERE id = :id`, m)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "updating measurement")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UploadFailed writes the error string for the upload failure to the measurement
|
|
|
|
func (m *Measurement) UploadFailed(db *sqlx.DB, failure string) error {
|
|
|
|
m.UploadFailure = failure
|
|
|
|
m.Uploaded = false
|
|
|
|
|
|
|
|
err := UpdateOne(db, `UPDATE measurements
|
|
|
|
SET upload_failure = :upload_failure
|
|
|
|
WHERE id = :id`, m)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "updating measurement")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UploadSucceeded writes the error string for the upload failure to the measurement
|
|
|
|
func (m *Measurement) UploadSucceeded(db *sqlx.DB) error {
|
|
|
|
m.Uploaded = true
|
|
|
|
|
|
|
|
err := UpdateOne(db, `UPDATE measurements
|
|
|
|
SET uploaded = :uploaded
|
|
|
|
WHERE id = :id`, m)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "updating measurement")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteSummary writes the summary to the measurement
|
|
|
|
func (m *Measurement) WriteSummary(db *sqlx.DB, summary string) error {
|
|
|
|
m.Summary = summary
|
|
|
|
|
|
|
|
err := UpdateOne(db, `UPDATE measurements
|
|
|
|
SET summary = :summary
|
|
|
|
WHERE id = :id`, m)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "updating measurement")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-03-20 14:31:05 +01:00
|
|
|
// AddToResult adds a measurement to a result
|
2018-03-20 12:38:33 +01:00
|
|
|
func (m *Measurement) AddToResult(db *sqlx.DB, result *Result) error {
|
2018-03-23 13:17:39 +01:00
|
|
|
var err error
|
|
|
|
|
2018-03-20 12:38:33 +01:00
|
|
|
m.ResultID = result.ID
|
|
|
|
finalPath := filepath.Join(result.MeasurementDir,
|
|
|
|
filepath.Base(m.ReportFilePath))
|
|
|
|
|
2018-03-23 13:17:39 +01:00
|
|
|
// If the finalPath already exists, it means it has already been moved there.
|
|
|
|
// This happens in multi input reports
|
|
|
|
if _, err = os.Stat(finalPath); os.IsNotExist(err) {
|
|
|
|
err = os.Rename(m.ReportFilePath, finalPath)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "moving report file")
|
|
|
|
}
|
2018-03-20 12:38:33 +01:00
|
|
|
}
|
|
|
|
m.ReportFilePath = finalPath
|
|
|
|
|
|
|
|
err = UpdateOne(db, `UPDATE measurements
|
|
|
|
SET result_id = :result_id, report_file = :report_file
|
|
|
|
WHERE id = :id`, m)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "updating measurement")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-02-13 17:11:22 +01:00
|
|
|
// CreateMeasurement writes the measurement to the database a returns a pointer
|
|
|
|
// to the Measurement
|
2018-03-19 16:23:30 +01:00
|
|
|
func CreateMeasurement(db *sqlx.DB, m Measurement, i string) (*Measurement, error) {
|
|
|
|
// XXX Do we want to have this be part of something else?
|
|
|
|
m.StartTime = time.Now().UTC()
|
|
|
|
m.Input = i
|
|
|
|
m.State = "active"
|
|
|
|
|
2018-02-13 17:11:22 +01:00
|
|
|
res, err := db.NamedExec(`INSERT INTO measurements
|
|
|
|
(name, start_time,
|
2018-03-19 16:23:30 +01:00
|
|
|
asn, ip, country,
|
2018-02-13 17:11:22 +01:00
|
|
|
state, failure, report_file,
|
2018-03-19 19:28:32 +01:00
|
|
|
report_id, input,
|
2018-02-13 17:11:22 +01:00
|
|
|
result_id)
|
|
|
|
VALUES (:name,:start_time,
|
2018-02-21 16:06:30 +01:00
|
|
|
:asn,:ip,:country,
|
2018-02-13 17:11:22 +01:00
|
|
|
:state,:failure,:report_file,
|
2018-02-21 16:06:30 +01:00
|
|
|
:report_id,:input,
|
2018-02-13 17:11:22 +01:00
|
|
|
:result_id)`,
|
|
|
|
m)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "creating measurement")
|
|
|
|
}
|
|
|
|
id, err := res.LastInsertId()
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "creating measurement")
|
|
|
|
}
|
|
|
|
m.ID = id
|
|
|
|
return &m, nil
|
|
|
|
}
|
|
|
|
|
2018-02-12 17:29:03 +01:00
|
|
|
// Result model
|
|
|
|
type Result struct {
|
2018-03-20 12:38:33 +01:00
|
|
|
ID int64 `db:"id"`
|
|
|
|
Name string `db:"name"`
|
|
|
|
StartTime time.Time `db:"start_time"`
|
2018-05-03 18:40:52 +02:00
|
|
|
Country string `db:"country"`
|
|
|
|
ASN string `db:"asn"`
|
|
|
|
NetworkName string `db:"network_name"`
|
2018-03-20 12:38:33 +01:00
|
|
|
Runtime float64 `db:"runtime"` // Runtime is expressed in fractional seconds
|
|
|
|
Summary string `db:"summary"` // XXX this should be JSON
|
|
|
|
Done bool `db:"done"`
|
|
|
|
DataUsageUp int64 `db:"data_usage_up"`
|
|
|
|
DataUsageDown int64 `db:"data_usage_down"`
|
|
|
|
MeasurementDir string `db:"measurement_dir"`
|
2018-03-08 13:46:21 +01:00
|
|
|
}
|
2018-03-08 11:53:04 +01:00
|
|
|
|
2018-05-03 18:40:52 +02:00
|
|
|
// ListResults return the list of results
|
|
|
|
func ListResults(db *sqlx.DB) ([]*Result, []*Result, error) {
|
|
|
|
doneResults := []*Result{}
|
|
|
|
incompleteResults := []*Result{}
|
|
|
|
|
|
|
|
rows, err := db.Query(`SELECT id, name,
|
|
|
|
start_time, runtime,
|
|
|
|
network_name, country,
|
|
|
|
asn,
|
|
|
|
summary, done
|
|
|
|
FROM results
|
|
|
|
WHERE done = 1
|
|
|
|
ORDER BY start_time;`)
|
|
|
|
if err != nil {
|
|
|
|
return doneResults, incompleteResults, errors.Wrap(err, "failed to get result done list")
|
|
|
|
}
|
|
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
result := Result{}
|
|
|
|
err = rows.Scan(&result.ID, &result.Name,
|
|
|
|
&result.StartTime, &result.Runtime,
|
|
|
|
&result.NetworkName, &result.Country,
|
|
|
|
&result.ASN,
|
|
|
|
&result.Summary, &result.Done,
|
|
|
|
//&result.DataUsageUp, &result.DataUsageDown)
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("failed to fetch a row")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
doneResults = append(doneResults, &result)
|
|
|
|
}
|
|
|
|
|
|
|
|
rows, err = db.Query(`SELECT
|
|
|
|
id, name,
|
|
|
|
start_time,
|
|
|
|
network_name, country,
|
|
|
|
asn
|
|
|
|
FROM results
|
|
|
|
WHERE done != 1
|
|
|
|
ORDER BY start_time;`)
|
|
|
|
if err != nil {
|
|
|
|
return doneResults, incompleteResults, errors.Wrap(err, "failed to get result done list")
|
|
|
|
}
|
|
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
result := Result{Done: false}
|
|
|
|
err = rows.Scan(&result.ID, &result.Name, &result.StartTime,
|
|
|
|
&result.NetworkName, &result.Country,
|
|
|
|
&result.ASN)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("failed to fetch a row")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
incompleteResults = append(incompleteResults, &result)
|
|
|
|
}
|
|
|
|
return doneResults, incompleteResults, nil
|
|
|
|
}
|
|
|
|
|
2018-03-20 14:19:19 +01:00
|
|
|
// MakeSummaryMap return a mapping of test names to summaries for the given
|
|
|
|
// result
|
|
|
|
func MakeSummaryMap(db *sqlx.DB, r *Result) (SummaryMap, error) {
|
|
|
|
summaryMap := SummaryMap{}
|
|
|
|
|
|
|
|
msmts := []Measurement{}
|
|
|
|
// XXX maybe we only want to select some of the columns
|
|
|
|
err := db.Select(&msmts, "SELECT name, summary FROM measurements WHERE result_id = $1", r.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to get measurements")
|
|
|
|
}
|
|
|
|
for _, msmt := range msmts {
|
2018-03-23 14:58:33 +01:00
|
|
|
val, ok := summaryMap[msmt.Name]
|
|
|
|
if ok {
|
|
|
|
summaryMap[msmt.Name] = append(val, msmt.Summary)
|
|
|
|
} else {
|
|
|
|
summaryMap[msmt.Name] = []string{msmt.Summary}
|
|
|
|
}
|
2018-03-20 14:19:19 +01:00
|
|
|
}
|
|
|
|
return summaryMap, nil
|
|
|
|
}
|
|
|
|
|
2018-03-08 13:46:21 +01:00
|
|
|
// Finished marks the result as done and sets the runtime
|
2018-03-20 14:19:19 +01:00
|
|
|
func (r *Result) Finished(db *sqlx.DB, makeSummary ResultSummaryFunc) error {
|
2018-03-08 13:46:21 +01:00
|
|
|
if r.Done == true || r.Runtime != 0 {
|
|
|
|
return errors.New("Result is already finished")
|
|
|
|
}
|
2018-03-20 12:38:33 +01:00
|
|
|
r.Runtime = time.Now().UTC().Sub(r.StartTime).Seconds()
|
2018-03-08 13:46:21 +01:00
|
|
|
r.Done = true
|
2018-03-20 12:38:33 +01:00
|
|
|
// XXX add in here functionality to compute the summary
|
2018-03-20 14:19:19 +01:00
|
|
|
summaryMap, err := MakeSummaryMap(db, r)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
r.Summary, err = makeSummary(summaryMap)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-03-08 13:46:21 +01:00
|
|
|
|
2018-03-20 14:19:19 +01:00
|
|
|
err = UpdateOne(db, `UPDATE results
|
|
|
|
SET done = :done, runtime = :runtime, summary = :summary
|
2018-03-08 13:46:21 +01:00
|
|
|
WHERE id = :id`, r)
|
|
|
|
if err != nil {
|
2018-03-20 12:38:33 +01:00
|
|
|
return errors.Wrap(err, "updating finished result")
|
2018-03-08 13:46:21 +01:00
|
|
|
}
|
2018-02-13 17:11:22 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateResult writes the Result to the database a returns a pointer
|
|
|
|
// to the Result
|
2018-03-23 12:10:14 +01:00
|
|
|
func CreateResult(db *sqlx.DB, homePath string, r Result) (*Result, error) {
|
2018-03-08 11:53:04 +01:00
|
|
|
log.Debugf("Creating result %v", r)
|
2018-03-20 12:38:33 +01:00
|
|
|
|
2018-03-27 15:09:34 +02:00
|
|
|
p, err := utils.MakeResultsDir(homePath, r.Name, r.StartTime)
|
2018-03-20 12:38:33 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
r.MeasurementDir = p
|
2018-02-13 17:11:22 +01:00
|
|
|
res, err := db.NamedExec(`INSERT INTO results
|
2018-05-03 18:40:52 +02:00
|
|
|
(name, start_time, country, network_name, asn)
|
|
|
|
VALUES (:name,:start_time,:country,:network_name,:asn)`,
|
2018-02-13 17:11:22 +01:00
|
|
|
r)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "creating result")
|
|
|
|
}
|
|
|
|
id, err := res.LastInsertId()
|
|
|
|
if err != nil {
|
2018-03-08 13:46:21 +01:00
|
|
|
return nil, errors.Wrap(err, "creating result")
|
2018-02-13 17:11:22 +01:00
|
|
|
}
|
|
|
|
r.ID = id
|
|
|
|
return &r, nil
|
2018-02-12 17:29:03 +01:00
|
|
|
}
|