205 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			205 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package database
 | |
| 
 | |
| import (
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/apex/log"
 | |
| 	"github.com/jmoiron/sqlx"
 | |
| 	"github.com/pkg/errors"
 | |
| )
 | |
| 
 | |
| // 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
 | |
| }
 | |
| 
 | |
| // Measurement model
 | |
| type Measurement struct {
 | |
| 	ID             int64     `db:"id"`
 | |
| 	Name           string    `db:"name"`
 | |
| 	StartTime      time.Time `db:"start_time"`
 | |
| 	Runtime        float64   `db:"runtime"`
 | |
| 	Summary        string    `db:"summary"` // XXX this should be JSON
 | |
| 	ASN            string    `db:"asn"`
 | |
| 	IP             string    `db:"ip"`
 | |
| 	CountryCode    string    `db:"country"`
 | |
| 	State          string    `db:"state"`
 | |
| 	Failure        string    `db:"failure"`
 | |
| 	UploadFailure  string    `db:"upload_failure"`
 | |
| 	Uploaded       bool      `db:"uploaded"`
 | |
| 	ReportFilePath string    `db:"report_file"`
 | |
| 	ReportID       string    `db:"report_id"`
 | |
| 	Input          string    `db:"input"`
 | |
| 	ResultID       int64     `db:"result_id"`
 | |
| }
 | |
| 
 | |
| // SetGeoIPInfo for the Measurement
 | |
| func (m *Measurement) SetGeoIPInfo() error {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // 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 {
 | |
| 	m.State = "done"
 | |
| 
 | |
| 	err := UpdateOne(db, `UPDATE measurements
 | |
| 		SET state = :state
 | |
| 		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
 | |
| }
 | |
| 
 | |
| // CreateMeasurement writes the measurement to the database a returns a pointer
 | |
| // to the Measurement
 | |
| 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"
 | |
| 
 | |
| 	res, err := db.NamedExec(`INSERT INTO measurements
 | |
| 		(name, start_time,
 | |
| 			asn, ip, country,
 | |
| 			state, failure, report_file,
 | |
| 			report_id, input,
 | |
| 			result_id)
 | |
| 		VALUES (:name,:start_time,
 | |
| 			:asn,:ip,:country,
 | |
| 			:state,:failure,:report_file,
 | |
| 			:report_id,:input,
 | |
| 			: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
 | |
| }
 | |
| 
 | |
| // Result model
 | |
| type Result struct {
 | |
| 	ID            int64     `db:"id"`
 | |
| 	Name          string    `db:"name"`
 | |
| 	StartTime     time.Time `db:"start_time"`
 | |
| 	Runtime       float64   `db:"runtime"` // Runtime is expressed in Microseconds
 | |
| 	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"`
 | |
| 
 | |
| 	started time.Time
 | |
| }
 | |
| 
 | |
| // Started marks the Result as having started
 | |
| func (r *Result) Started(db *sqlx.DB) error {
 | |
| 	r.started = time.Now()
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // Finished marks the result as done and sets the runtime
 | |
| func (r *Result) Finished(db *sqlx.DB) error {
 | |
| 	if r.Done == true || r.Runtime != 0 {
 | |
| 		return errors.New("Result is already finished")
 | |
| 	}
 | |
| 	r.Runtime = float64(time.Now().Sub(r.started)) / float64(time.Microsecond)
 | |
| 	r.Done = true
 | |
| 
 | |
| 	err := UpdateOne(db, `UPDATE results
 | |
| 		SET done = true, runtime = :runtime
 | |
| 		WHERE id = :id`, r)
 | |
| 	if err != nil {
 | |
| 		return errors.Wrap(err, "updating result")
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // CreateResult writes the Result to the database a returns a pointer
 | |
| // to the Result
 | |
| func CreateResult(db *sqlx.DB, r Result) (*Result, error) {
 | |
| 	log.Debugf("Creating result %v", r)
 | |
| 	res, err := db.NamedExec(`INSERT INTO results
 | |
| 		(name, start_time)
 | |
| 		VALUES (:name,:start_time)`,
 | |
| 		r)
 | |
| 	if err != nil {
 | |
| 		return nil, errors.Wrap(err, "creating result")
 | |
| 	}
 | |
| 	id, err := res.LastInsertId()
 | |
| 	if err != nil {
 | |
| 		return nil, errors.Wrap(err, "creating result")
 | |
| 	}
 | |
| 	r.ID = id
 | |
| 	return &r, nil
 | |
| }
 |