refactor(engine): more abstract Experiment{,Builder} (#838)
This diff modifies the engine package to make Experiment and ExperimentBuilder interfaces rather than structs. The previosuly existing structs are now named experiment{,Builder}. This diff helps https://github.com/ooni/probe/issues/2184 because it allows us to write unit tests more easily. There should be no functional change. While there, I removed a bunch of deprecated functions, which were unnecessarily complicate the implementation and could be easily replaced by passing them a context.Context or context.Background().
This commit is contained in:
parent
5b27df1a37
commit
97864b324f
|
@ -1,6 +1,7 @@
|
||||||
package nettests
|
package nettests
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"database/sql"
|
"database/sql"
|
||||||
"fmt"
|
"fmt"
|
||||||
"time"
|
"time"
|
||||||
|
@ -122,7 +123,7 @@ func (c *Controller) SetNettestIndex(i, n int) {
|
||||||
//
|
//
|
||||||
// This function will continue to run in most cases but will
|
// This function will continue to run in most cases but will
|
||||||
// immediately halt if something's wrong with the file system.
|
// immediately halt if something's wrong with the file system.
|
||||||
func (c *Controller) Run(builder *engine.ExperimentBuilder, inputs []string) error {
|
func (c *Controller) Run(builder engine.ExperimentBuilder, inputs []string) error {
|
||||||
// This will configure the controller as handler for the callbacks
|
// This will configure the controller as handler for the callbacks
|
||||||
// called by ooni/probe-engine/experiment.Experiment.
|
// called by ooni/probe-engine/experiment.Experiment.
|
||||||
builder.SetCallbacks(model.ExperimentCallbacks(c))
|
builder.SetCallbacks(model.ExperimentCallbacks(c))
|
||||||
|
@ -143,7 +144,7 @@ func (c *Controller) Run(builder *engine.ExperimentBuilder, inputs []string) err
|
||||||
log.Debug(color.RedString("status.started"))
|
log.Debug(color.RedString("status.started"))
|
||||||
|
|
||||||
if c.Probe.Config().Sharing.UploadResults {
|
if c.Probe.Config().Sharing.UploadResults {
|
||||||
if err := exp.OpenReport(); err != nil {
|
if err := exp.OpenReportContext(context.Background()); err != nil {
|
||||||
log.Debugf(
|
log.Debugf(
|
||||||
"%s: %s", color.RedString("failure.report_create"), err.Error(),
|
"%s: %s", color.RedString("failure.report_create"), err.Error(),
|
||||||
)
|
)
|
||||||
|
@ -197,7 +198,7 @@ func (c *Controller) Run(builder *engine.ExperimentBuilder, inputs []string) err
|
||||||
if input != "" {
|
if input != "" {
|
||||||
c.OnProgress(0, fmt.Sprintf("processing input: %s", input))
|
c.OnProgress(0, fmt.Sprintf("processing input: %s", input))
|
||||||
}
|
}
|
||||||
measurement, err := exp.Measure(input)
|
measurement, err := exp.MeasureWithContext(context.Background(), input)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.WithError(err).Debug(color.RedString("failure.measurement"))
|
log.WithError(err).Debug(color.RedString("failure.measurement"))
|
||||||
if err := c.msmts[idx64].Failed(c.Probe.DB(), err.Error()); err != nil {
|
if err := c.msmts[idx64].Failed(c.Probe.DB(), err.Error()); err != nil {
|
||||||
|
@ -218,7 +219,7 @@ func (c *Controller) Run(builder *engine.ExperimentBuilder, inputs []string) err
|
||||||
// Implementation note: SubmitMeasurement will fail here if we did fail
|
// Implementation note: SubmitMeasurement will fail here if we did fail
|
||||||
// to open the report but we still want to continue. There will be a
|
// to open the report but we still want to continue. There will be a
|
||||||
// bit of a spew in the logs, perhaps, but stopping seems less efficient.
|
// bit of a spew in the logs, perhaps, but stopping seems less efficient.
|
||||||
if err := exp.SubmitAndUpdateMeasurement(measurement); err != nil {
|
if err := exp.SubmitAndUpdateMeasurementContext(context.Background(), measurement); err != nil {
|
||||||
log.Debug(color.RedString("failure.measurement_submission"))
|
log.Debug(color.RedString("failure.measurement_submission"))
|
||||||
if err := c.msmts[idx64].UploadFailed(c.Probe.DB(), err.Error()); err != nil {
|
if err := c.msmts[idx64].UploadFailed(c.Probe.DB(), err.Error()); err != nil {
|
||||||
return errors.Wrap(err, "failed to mark upload as failed")
|
return errors.Wrap(err, "failed to mark upload as failed")
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
package engine
|
package engine
|
||||||
|
|
||||||
|
//
|
||||||
|
// List of all implemented experiments
|
||||||
|
//
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -32,11 +36,11 @@ import (
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/experiment/whatsapp"
|
"github.com/ooni/probe-cli/v3/internal/engine/experiment/whatsapp"
|
||||||
)
|
)
|
||||||
|
|
||||||
var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
var experimentsByName = map[string]func(*Session) *experimentBuilder{
|
||||||
"dash": func(session *Session) *ExperimentBuilder {
|
"dash": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, dash.NewExperimentMeasurer(
|
return newExperiment(session, dash.NewExperimentMeasurer(
|
||||||
*config.(*dash.Config),
|
*config.(*dash.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -46,10 +50,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"dnscheck": func(session *Session) *ExperimentBuilder {
|
"dnscheck": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, dnscheck.NewExperimentMeasurer(
|
return newExperiment(session, dnscheck.NewExperimentMeasurer(
|
||||||
*config.(*dnscheck.Config),
|
*config.(*dnscheck.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -58,10 +62,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"dnsping": func(session *Session) *ExperimentBuilder {
|
"dnsping": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, dnsping.NewExperimentMeasurer(
|
return newExperiment(session, dnsping.NewExperimentMeasurer(
|
||||||
*config.(*dnsping.Config),
|
*config.(*dnsping.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -70,10 +74,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"example": func(session *Session) *ExperimentBuilder {
|
"example": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, example.NewExperimentMeasurer(
|
return newExperiment(session, example.NewExperimentMeasurer(
|
||||||
*config.(*example.Config), "example",
|
*config.(*example.Config), "example",
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -86,10 +90,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"facebook_messenger": func(session *Session) *ExperimentBuilder {
|
"facebook_messenger": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, fbmessenger.NewExperimentMeasurer(
|
return newExperiment(session, fbmessenger.NewExperimentMeasurer(
|
||||||
*config.(*fbmessenger.Config),
|
*config.(*fbmessenger.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -98,10 +102,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"http_header_field_manipulation": func(session *Session) *ExperimentBuilder {
|
"http_header_field_manipulation": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, hhfm.NewExperimentMeasurer(
|
return newExperiment(session, hhfm.NewExperimentMeasurer(
|
||||||
*config.(*hhfm.Config),
|
*config.(*hhfm.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -110,10 +114,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"http_host_header": func(session *Session) *ExperimentBuilder {
|
"http_host_header": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, httphostheader.NewExperimentMeasurer(
|
return newExperiment(session, httphostheader.NewExperimentMeasurer(
|
||||||
*config.(*httphostheader.Config),
|
*config.(*httphostheader.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -122,10 +126,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"http_invalid_request_line": func(session *Session) *ExperimentBuilder {
|
"http_invalid_request_line": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, hirl.NewExperimentMeasurer(
|
return newExperiment(session, hirl.NewExperimentMeasurer(
|
||||||
*config.(*hirl.Config),
|
*config.(*hirl.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -134,10 +138,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"ndt": func(session *Session) *ExperimentBuilder {
|
"ndt": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, ndt7.NewExperimentMeasurer(
|
return newExperiment(session, ndt7.NewExperimentMeasurer(
|
||||||
*config.(*ndt7.Config),
|
*config.(*ndt7.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -147,10 +151,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"psiphon": func(session *Session) *ExperimentBuilder {
|
"psiphon": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, psiphon.NewExperimentMeasurer(
|
return newExperiment(session, psiphon.NewExperimentMeasurer(
|
||||||
*config.(*psiphon.Config),
|
*config.(*psiphon.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -159,10 +163,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"quicping": func(session *Session) *ExperimentBuilder {
|
"quicping": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, quicping.NewExperimentMeasurer(
|
return newExperiment(session, quicping.NewExperimentMeasurer(
|
||||||
*config.(*quicping.Config),
|
*config.(*quicping.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -171,10 +175,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"riseupvpn": func(session *Session) *ExperimentBuilder {
|
"riseupvpn": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, riseupvpn.NewExperimentMeasurer(
|
return newExperiment(session, riseupvpn.NewExperimentMeasurer(
|
||||||
*config.(*riseupvpn.Config),
|
*config.(*riseupvpn.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -183,10 +187,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"run": func(session *Session) *ExperimentBuilder {
|
"run": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, run.NewExperimentMeasurer(
|
return newExperiment(session, run.NewExperimentMeasurer(
|
||||||
*config.(*run.Config),
|
*config.(*run.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -195,10 +199,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"simplequicping": func(session *Session) *ExperimentBuilder {
|
"simplequicping": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, simplequicping.NewExperimentMeasurer(
|
return newExperiment(session, simplequicping.NewExperimentMeasurer(
|
||||||
*config.(*simplequicping.Config),
|
*config.(*simplequicping.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -207,10 +211,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"signal": func(session *Session) *ExperimentBuilder {
|
"signal": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, signal.NewExperimentMeasurer(
|
return newExperiment(session, signal.NewExperimentMeasurer(
|
||||||
*config.(*signal.Config),
|
*config.(*signal.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -219,10 +223,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"sni_blocking": func(session *Session) *ExperimentBuilder {
|
"sni_blocking": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, sniblocking.NewExperimentMeasurer(
|
return newExperiment(session, sniblocking.NewExperimentMeasurer(
|
||||||
*config.(*sniblocking.Config),
|
*config.(*sniblocking.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -231,10 +235,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"stunreachability": func(session *Session) *ExperimentBuilder {
|
"stunreachability": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, stunreachability.NewExperimentMeasurer(
|
return newExperiment(session, stunreachability.NewExperimentMeasurer(
|
||||||
*config.(*stunreachability.Config),
|
*config.(*stunreachability.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -243,10 +247,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"tcpping": func(session *Session) *ExperimentBuilder {
|
"tcpping": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, tcpping.NewExperimentMeasurer(
|
return newExperiment(session, tcpping.NewExperimentMeasurer(
|
||||||
*config.(*tcpping.Config),
|
*config.(*tcpping.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -255,10 +259,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"tlsping": func(session *Session) *ExperimentBuilder {
|
"tlsping": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, tlsping.NewExperimentMeasurer(
|
return newExperiment(session, tlsping.NewExperimentMeasurer(
|
||||||
*config.(*tlsping.Config),
|
*config.(*tlsping.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -267,10 +271,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"telegram": func(session *Session) *ExperimentBuilder {
|
"telegram": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, telegram.NewExperimentMeasurer(
|
return newExperiment(session, telegram.NewExperimentMeasurer(
|
||||||
*config.(*telegram.Config),
|
*config.(*telegram.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -279,10 +283,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"tlstool": func(session *Session) *ExperimentBuilder {
|
"tlstool": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, tlstool.NewExperimentMeasurer(
|
return newExperiment(session, tlstool.NewExperimentMeasurer(
|
||||||
*config.(*tlstool.Config),
|
*config.(*tlstool.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -291,10 +295,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"tor": func(session *Session) *ExperimentBuilder {
|
"tor": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, tor.NewExperimentMeasurer(
|
return newExperiment(session, tor.NewExperimentMeasurer(
|
||||||
*config.(*tor.Config),
|
*config.(*tor.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -303,10 +307,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"torsf": func(session *Session) *ExperimentBuilder {
|
"torsf": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, torsf.NewExperimentMeasurer(
|
return newExperiment(session, torsf.NewExperimentMeasurer(
|
||||||
*config.(*torsf.Config),
|
*config.(*torsf.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -315,10 +319,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"urlgetter": func(session *Session) *ExperimentBuilder {
|
"urlgetter": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, urlgetter.NewExperimentMeasurer(
|
return newExperiment(session, urlgetter.NewExperimentMeasurer(
|
||||||
*config.(*urlgetter.Config),
|
*config.(*urlgetter.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -327,10 +331,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"vanilla_tor": func(session *Session) *ExperimentBuilder {
|
"vanilla_tor": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, vanillator.NewExperimentMeasurer(
|
return newExperiment(session, vanillator.NewExperimentMeasurer(
|
||||||
*config.(*vanillator.Config),
|
*config.(*vanillator.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -339,10 +343,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"web_connectivity": func(session *Session) *ExperimentBuilder {
|
"web_connectivity": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, webconnectivity.NewExperimentMeasurer(
|
return newExperiment(session, webconnectivity.NewExperimentMeasurer(
|
||||||
*config.(*webconnectivity.Config),
|
*config.(*webconnectivity.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
@ -351,10 +355,10 @@ var experimentsByName = map[string]func(*Session) *ExperimentBuilder{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
"whatsapp": func(session *Session) *ExperimentBuilder {
|
"whatsapp": func(session *Session) *experimentBuilder {
|
||||||
return &ExperimentBuilder{
|
return &experimentBuilder{
|
||||||
build: func(config interface{}) *Experiment {
|
build: func(config interface{}) *experiment {
|
||||||
return NewExperiment(session, whatsapp.NewExperimentMeasurer(
|
return newExperiment(session, whatsapp.NewExperimentMeasurer(
|
||||||
*config.(*whatsapp.Config),
|
*config.(*whatsapp.Config),
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
package engine
|
package engine
|
||||||
|
|
||||||
|
//
|
||||||
|
// Experiment definition and implementation.
|
||||||
|
//
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
@ -23,7 +27,77 @@ func formatTimeNowUTC() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Experiment is an experiment instance.
|
// Experiment is an experiment instance.
|
||||||
type Experiment struct {
|
type Experiment interface {
|
||||||
|
// KibiBytesReceived accounts for the KibiBytes received by the experiment.
|
||||||
|
KibiBytesReceived() float64
|
||||||
|
|
||||||
|
// KibiBytesSent is like KibiBytesReceived but for the bytes sent.
|
||||||
|
KibiBytesSent() float64
|
||||||
|
|
||||||
|
// Name returns the experiment name.
|
||||||
|
Name() string
|
||||||
|
|
||||||
|
// GetSummaryKeys returns a data structure containing a
|
||||||
|
// summary of the test keys for ooniprobe.
|
||||||
|
GetSummaryKeys(m *model.Measurement) (any, error)
|
||||||
|
|
||||||
|
// ReportID returns the open report's ID, if we have opened a report
|
||||||
|
// successfully before, or an empty string, otherwise.
|
||||||
|
//
|
||||||
|
// Deprecated: new code should use a Submitter.
|
||||||
|
ReportID() string
|
||||||
|
|
||||||
|
// MeasureAsync runs an async measurement. This operation could post
|
||||||
|
// one or more measurements onto the returned channel. We'll close the
|
||||||
|
// channel when we've emitted all the measurements.
|
||||||
|
//
|
||||||
|
// Arguments:
|
||||||
|
//
|
||||||
|
// - ctx is the context for deadline/cancellation/timeout;
|
||||||
|
//
|
||||||
|
// - input is the input (typically a URL but it could also be
|
||||||
|
// just an endpoint or an empty string for input-less experiments
|
||||||
|
// such as, e.g., ndt7 and dash).
|
||||||
|
//
|
||||||
|
// Return value:
|
||||||
|
//
|
||||||
|
// - on success, channel where to post measurements (the channel
|
||||||
|
// will be closed when done) and nil error;
|
||||||
|
//
|
||||||
|
// - on failure, nil channel and non-nil error.
|
||||||
|
MeasureAsync(ctx context.Context, input string) (<-chan *model.Measurement, error)
|
||||||
|
|
||||||
|
// MeasureWithContext performs a synchronous measurement.
|
||||||
|
//
|
||||||
|
// Return value: strictly either a non-nil measurement and
|
||||||
|
// a nil error or a nil measurement and a non-nil error.
|
||||||
|
//
|
||||||
|
// CAVEAT: while this API is perfectly fine for experiments that
|
||||||
|
// return a single measurement, it will only return the first measurement
|
||||||
|
// when used with an asynchronous experiment.
|
||||||
|
MeasureWithContext(ctx context.Context, input string) (measurement *model.Measurement, err error)
|
||||||
|
|
||||||
|
// SaveMeasurement saves a measurement on the specified file path.
|
||||||
|
//
|
||||||
|
// Deprecated: new code should use a Saver.
|
||||||
|
SaveMeasurement(measurement *model.Measurement, filePath string) error
|
||||||
|
|
||||||
|
// SubmitAndUpdateMeasurementContext submits a measurement and updates the
|
||||||
|
// fields whose value has changed as part of the submission.
|
||||||
|
//
|
||||||
|
// Deprecated: new code should use a Submitter.
|
||||||
|
SubmitAndUpdateMeasurementContext(
|
||||||
|
ctx context.Context, measurement *model.Measurement) error
|
||||||
|
|
||||||
|
// OpenReportContext will open a report using the given context
|
||||||
|
// to possibly limit the lifetime of this operation.
|
||||||
|
//
|
||||||
|
// Deprecated: new code should use a Submitter.
|
||||||
|
OpenReportContext(ctx context.Context) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// experiment implements Experiment.
|
||||||
|
type experiment struct {
|
||||||
byteCounter *bytecounter.Counter
|
byteCounter *bytecounter.Counter
|
||||||
callbacks model.ExperimentCallbacks
|
callbacks model.ExperimentCallbacks
|
||||||
measurer model.ExperimentMeasurer
|
measurer model.ExperimentMeasurer
|
||||||
|
@ -34,11 +108,9 @@ type Experiment struct {
|
||||||
testVersion string
|
testVersion string
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewExperiment creates a new experiment given a measurer. The preferred
|
// newExperiment creates a new experiment given a measurer.
|
||||||
// way to create an experiment is the ExperimentBuilder. Though this function
|
func newExperiment(sess *Session, measurer model.ExperimentMeasurer) *experiment {
|
||||||
// allows the programmer to create a custom, external experiment.
|
return &experiment{
|
||||||
func NewExperiment(sess *Session, measurer model.ExperimentMeasurer) *Experiment {
|
|
||||||
return &Experiment{
|
|
||||||
byteCounter: bytecounter.New(),
|
byteCounter: bytecounter.New(),
|
||||||
callbacks: model.NewPrinterCallbacks(sess.Logger()),
|
callbacks: model.NewPrinterCallbacks(sess.Logger()),
|
||||||
measurer: measurer,
|
measurer: measurer,
|
||||||
|
@ -49,64 +121,37 @@ func NewExperiment(sess *Session, measurer model.ExperimentMeasurer) *Experiment
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// KibiBytesReceived accounts for the KibiBytes received by the HTTP clients
|
// KibiBytesReceived implements Experiment.KibiBytesReceived.
|
||||||
// managed by this session so far, including experiments.
|
func (e *experiment) KibiBytesReceived() float64 {
|
||||||
func (e *Experiment) KibiBytesReceived() float64 {
|
|
||||||
return e.byteCounter.KibiBytesReceived()
|
return e.byteCounter.KibiBytesReceived()
|
||||||
}
|
}
|
||||||
|
|
||||||
// KibiBytesSent is like KibiBytesReceived but for the bytes sent.
|
// KibiBytesSent implements Experiment.KibiBytesSent.
|
||||||
func (e *Experiment) KibiBytesSent() float64 {
|
func (e *experiment) KibiBytesSent() float64 {
|
||||||
return e.byteCounter.KibiBytesSent()
|
return e.byteCounter.KibiBytesSent()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Name returns the experiment name.
|
// Name implements Experiment.Name.
|
||||||
func (e *Experiment) Name() string {
|
func (e *experiment) Name() string {
|
||||||
return e.testName
|
return e.testName
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetSummaryKeys returns a data structure containing a
|
// GetSummaryKeys implements Experiment.GetSummaryKeys.
|
||||||
// summary of the test keys for probe-cli.
|
func (e *experiment) GetSummaryKeys(m *model.Measurement) (interface{}, error) {
|
||||||
func (e *Experiment) GetSummaryKeys(m *model.Measurement) (interface{}, error) {
|
|
||||||
return e.measurer.GetSummaryKeys(m)
|
return e.measurer.GetSummaryKeys(m)
|
||||||
}
|
}
|
||||||
|
|
||||||
// OpenReport is an idempotent method to open a report. We assume that
|
// ReportID implements Experiment.ReportID.
|
||||||
// you have configured the available probe services, either manually or
|
func (e *experiment) ReportID() string {
|
||||||
// through using the session's MaybeLookupBackends method.
|
|
||||||
func (e *Experiment) OpenReport() (err error) {
|
|
||||||
return e.OpenReportContext(context.Background())
|
|
||||||
}
|
|
||||||
|
|
||||||
// ReportID returns the open reportID, if we have opened a report
|
|
||||||
// successfully before, or an empty string, otherwise.
|
|
||||||
func (e *Experiment) ReportID() string {
|
|
||||||
if e.report == nil {
|
if e.report == nil {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return e.report.ReportID()
|
return e.report.ReportID()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Measure performs a measurement with input. We assume that you have
|
|
||||||
// configured the available test helpers, either manually or by calling
|
|
||||||
// the session's MaybeLookupBackends() method.
|
|
||||||
//
|
|
||||||
// Return value: strictly either a non-nil measurement and
|
|
||||||
// a nil error or a nil measurement and a non-nil error.
|
|
||||||
//
|
|
||||||
// CAVEAT: while this API is perfectly fine for experiments that
|
|
||||||
// return a single measurement, it will only return the first measurement
|
|
||||||
// when used with an asynchronous experiment. We plan on eventually
|
|
||||||
// migrating all experiments to run in asynchronous fashion.
|
|
||||||
//
|
|
||||||
// Deprecated: use MeasureWithContext instead, please.
|
|
||||||
func (e *Experiment) Measure(input string) (*model.Measurement, error) {
|
|
||||||
return e.MeasureWithContext(context.Background(), input)
|
|
||||||
}
|
|
||||||
|
|
||||||
// experimentAsyncWrapper makes a sync experiment behave like it was async
|
// experimentAsyncWrapper makes a sync experiment behave like it was async
|
||||||
type experimentAsyncWrapper struct {
|
type experimentAsyncWrapper struct {
|
||||||
*Experiment
|
*experiment
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ model.ExperimentMeasurerAsync = &experimentAsyncWrapper{}
|
var _ model.ExperimentMeasurerAsync = &experimentAsyncWrapper{}
|
||||||
|
@ -116,9 +161,9 @@ func (eaw *experimentAsyncWrapper) RunAsync(
|
||||||
ctx context.Context, sess model.ExperimentSession, input string,
|
ctx context.Context, sess model.ExperimentSession, input string,
|
||||||
callbacks model.ExperimentCallbacks) (<-chan *model.ExperimentAsyncTestKeys, error) {
|
callbacks model.ExperimentCallbacks) (<-chan *model.ExperimentAsyncTestKeys, error) {
|
||||||
out := make(chan *model.ExperimentAsyncTestKeys)
|
out := make(chan *model.ExperimentAsyncTestKeys)
|
||||||
measurement := eaw.Experiment.newMeasurement(input)
|
measurement := eaw.experiment.newMeasurement(input)
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
err := eaw.Experiment.measurer.Run(ctx, eaw.session, measurement, eaw.callbacks)
|
err := eaw.experiment.measurer.Run(ctx, eaw.session, measurement, eaw.callbacks)
|
||||||
stop := time.Now()
|
stop := time.Now()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -136,25 +181,8 @@ func (eaw *experimentAsyncWrapper) RunAsync(
|
||||||
return out, nil
|
return out, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// MeasureAsync runs an async measurement. This operation could post
|
// MeasureAsync implements Experiment.MeasureAsync.
|
||||||
// one or more measurements onto the returned channel. We'll close the
|
func (e *experiment) MeasureAsync(
|
||||||
// channel when we've emitted all the measurements.
|
|
||||||
//
|
|
||||||
// Arguments:
|
|
||||||
//
|
|
||||||
// - ctx is the context for deadline/cancellation/timeout;
|
|
||||||
//
|
|
||||||
// - input is the input (typically a URL but it could also be
|
|
||||||
// just an endpoint or an empty string for input-less experiments
|
|
||||||
// such as, e.g., ndt7 and dash).
|
|
||||||
//
|
|
||||||
// Return value:
|
|
||||||
//
|
|
||||||
// - on success, channel where to post measurements (the channel
|
|
||||||
// will be closed when done) and nil error;
|
|
||||||
//
|
|
||||||
// - on failure, nil channel and non-nil error.
|
|
||||||
func (e *Experiment) MeasureAsync(
|
|
||||||
ctx context.Context, input string) (<-chan *model.Measurement, error) {
|
ctx context.Context, input string) (<-chan *model.Measurement, error) {
|
||||||
err := e.session.MaybeLookupLocationContext(ctx) // this already tracks session bytes
|
err := e.session.MaybeLookupLocationContext(ctx) // this already tracks session bytes
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -195,16 +223,8 @@ func (e *Experiment) MeasureAsync(
|
||||||
return out, nil
|
return out, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// MeasureWithContext is like Measure but with context.
|
// MeasureWithContext implements Experiment.MeasureWithContext.
|
||||||
//
|
func (e *experiment) MeasureWithContext(
|
||||||
// Return value: strictly either a non-nil measurement and
|
|
||||||
// a nil error or a nil measurement and a non-nil error.
|
|
||||||
//
|
|
||||||
// CAVEAT: while this API is perfectly fine for experiments that
|
|
||||||
// return a single measurement, it will only return the first measurement
|
|
||||||
// when used with an asynchronous experiment. We plan on eventually
|
|
||||||
// migrating all experiments to run in asynchronous fashion.
|
|
||||||
func (e *Experiment) MeasureWithContext(
|
|
||||||
ctx context.Context, input string,
|
ctx context.Context, input string,
|
||||||
) (measurement *model.Measurement, err error) {
|
) (measurement *model.Measurement, err error) {
|
||||||
out, err := e.MeasureAsync(ctx, input)
|
out, err := e.MeasureAsync(ctx, input)
|
||||||
|
@ -222,8 +242,8 @@ func (e *Experiment) MeasureWithContext(
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// SaveMeasurement saves a measurement on the specified file path.
|
// SaveMeasurement implements Experiment.SaveMeasurement.
|
||||||
func (e *Experiment) SaveMeasurement(measurement *model.Measurement, filePath string) error {
|
func (e *experiment) SaveMeasurement(measurement *model.Measurement, filePath string) error {
|
||||||
return e.saveMeasurement(
|
return e.saveMeasurement(
|
||||||
measurement, filePath, json.Marshal, os.OpenFile,
|
measurement, filePath, json.Marshal, os.OpenFile,
|
||||||
func(fp *os.File, b []byte) (int, error) {
|
func(fp *os.File, b []byte) (int, error) {
|
||||||
|
@ -232,15 +252,8 @@ func (e *Experiment) SaveMeasurement(measurement *model.Measurement, filePath st
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SubmitAndUpdateMeasurement submits a measurement and updates the
|
// SubmitAndUpdateMeasurementContext implements Experiment.SubmitAndUpdateMeasurementContext.
|
||||||
// fields whose value has changed as part of the submission.
|
func (e *experiment) SubmitAndUpdateMeasurementContext(
|
||||||
func (e *Experiment) SubmitAndUpdateMeasurement(measurement *model.Measurement) error {
|
|
||||||
return e.SubmitAndUpdateMeasurementContext(context.Background(), measurement)
|
|
||||||
}
|
|
||||||
|
|
||||||
// SubmitAndUpdateMeasurementContext submits a measurement and updates the
|
|
||||||
// fields whose value has changed as part of the submission.
|
|
||||||
func (e *Experiment) SubmitAndUpdateMeasurementContext(
|
|
||||||
ctx context.Context, measurement *model.Measurement) error {
|
ctx context.Context, measurement *model.Measurement) error {
|
||||||
if e.report == nil {
|
if e.report == nil {
|
||||||
return errors.New("report is not open")
|
return errors.New("report is not open")
|
||||||
|
@ -249,7 +262,7 @@ func (e *Experiment) SubmitAndUpdateMeasurementContext(
|
||||||
}
|
}
|
||||||
|
|
||||||
// newMeasurement creates a new measurement for this experiment with the given input.
|
// newMeasurement creates a new measurement for this experiment with the given input.
|
||||||
func (e *Experiment) newMeasurement(input string) *model.Measurement {
|
func (e *experiment) newMeasurement(input string) *model.Measurement {
|
||||||
utctimenow := time.Now().UTC()
|
utctimenow := time.Now().UTC()
|
||||||
m := &model.Measurement{
|
m := &model.Measurement{
|
||||||
DataFormatVersion: probeservices.DefaultDataFormatVersion,
|
DataFormatVersion: probeservices.DefaultDataFormatVersion,
|
||||||
|
@ -277,9 +290,8 @@ func (e *Experiment) newMeasurement(input string) *model.Measurement {
|
||||||
return m
|
return m
|
||||||
}
|
}
|
||||||
|
|
||||||
// OpenReportContext will open a report using the given context
|
// OpenReportContext implements Experiment.OpenReportContext.
|
||||||
// to possibly limit the lifetime of this operation.
|
func (e *experiment) OpenReportContext(ctx context.Context) error {
|
||||||
func (e *Experiment) OpenReportContext(ctx context.Context) error {
|
|
||||||
if e.report != nil {
|
if e.report != nil {
|
||||||
return nil // already open
|
return nil // already open
|
||||||
}
|
}
|
||||||
|
@ -305,7 +317,7 @@ func (e *Experiment) OpenReportContext(ctx context.Context) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *Experiment) newReportTemplate() probeservices.ReportTemplate {
|
func (e *experiment) newReportTemplate() probeservices.ReportTemplate {
|
||||||
return probeservices.ReportTemplate{
|
return probeservices.ReportTemplate{
|
||||||
DataFormatVersion: probeservices.DefaultDataFormatVersion,
|
DataFormatVersion: probeservices.DefaultDataFormatVersion,
|
||||||
Format: probeservices.DefaultFormat,
|
Format: probeservices.DefaultFormat,
|
||||||
|
@ -319,7 +331,7 @@ func (e *Experiment) newReportTemplate() probeservices.ReportTemplate {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *Experiment) saveMeasurement(
|
func (e *experiment) saveMeasurement(
|
||||||
measurement *model.Measurement, filePath string,
|
measurement *model.Measurement, filePath string,
|
||||||
marshal func(v interface{}) ([]byte, error),
|
marshal func(v interface{}) ([]byte, error),
|
||||||
openFile func(name string, flag int, perm os.FileMode) (*os.File, error),
|
openFile func(name string, flag int, perm os.FileMode) (*os.File, error),
|
||||||
|
|
|
@ -162,7 +162,7 @@ func TestSetCallbacks(t *testing.T) {
|
||||||
}
|
}
|
||||||
register := ®isterCallbacksCalled{}
|
register := ®isterCallbacksCalled{}
|
||||||
builder.SetCallbacks(register)
|
builder.SetCallbacks(register)
|
||||||
if _, err := builder.NewExperiment().Measure(""); err != nil {
|
if _, err := builder.NewExperiment().MeasureWithContext(context.Background(), ""); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
if register.onProgressCalled == false {
|
if register.onProgressCalled == false {
|
||||||
|
@ -206,7 +206,7 @@ func TestMeasurementFailure(t *testing.T) {
|
||||||
if err := builder.SetOptionAny("ReturnError", true); err != nil {
|
if err := builder.SetOptionAny("ReturnError", true); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
measurement, err := builder.NewExperiment().Measure("")
|
measurement, err := builder.NewExperiment().MeasureWithContext(context.Background(), "")
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatal("expected an error here")
|
t.Fatal("expected an error here")
|
||||||
}
|
}
|
||||||
|
@ -288,7 +288,7 @@ func TestUseOptions(t *testing.T) {
|
||||||
if err := builder.SetOptionAny("Message", "antani"); err != nil {
|
if err := builder.SetOptionAny("Message", "antani"); err != nil {
|
||||||
t.Fatal("cannot set Message field")
|
t.Fatal("cannot set Message field")
|
||||||
}
|
}
|
||||||
config := builder.config.(*example.Config)
|
config := builder.(*experimentBuilder).config.(*example.Config)
|
||||||
if config.ReturnError != true {
|
if config.ReturnError != true {
|
||||||
t.Fatal("config.ReturnError was not changed")
|
t.Fatal("config.ReturnError was not changed")
|
||||||
}
|
}
|
||||||
|
@ -313,15 +313,16 @@ func TestRunHHFM(t *testing.T) {
|
||||||
runexperimentflow(t, builder.NewExperiment(), "")
|
runexperimentflow(t, builder.NewExperiment(), "")
|
||||||
}
|
}
|
||||||
|
|
||||||
func runexperimentflow(t *testing.T, experiment *Experiment, input string) {
|
func runexperimentflow(t *testing.T, experiment Experiment, input string) {
|
||||||
err := experiment.OpenReport()
|
ctx := context.Background()
|
||||||
|
err := experiment.OpenReportContext(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
if experiment.ReportID() == "" {
|
if experiment.ReportID() == "" {
|
||||||
t.Fatal("reportID should not be empty here")
|
t.Fatal("reportID should not be empty here")
|
||||||
}
|
}
|
||||||
measurement, err := experiment.Measure(input)
|
measurement, err := experiment.MeasureWithContext(ctx, input)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -335,7 +336,7 @@ func runexperimentflow(t *testing.T, experiment *Experiment, input string) {
|
||||||
if data == nil {
|
if data == nil {
|
||||||
t.Fatal("data is nil")
|
t.Fatal("data is nil")
|
||||||
}
|
}
|
||||||
err = experiment.SubmitAndUpdateMeasurement(measurement)
|
err = experiment.SubmitAndUpdateMeasurementContext(ctx, measurement)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -364,14 +365,14 @@ func TestSaveMeasurementErrors(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
exp := builder.NewExperiment()
|
exp := builder.NewExperiment().(*experiment)
|
||||||
dirname, err := ioutil.TempDir("", "ooniprobe-engine-save-measurement")
|
dirname, err := ioutil.TempDir("", "ooniprobe-engine-save-measurement")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
filename := filepath.Join(dirname, "report.jsonl")
|
filename := filepath.Join(dirname, "report.jsonl")
|
||||||
m := new(model.Measurement)
|
m := new(model.Measurement)
|
||||||
err = exp.SaveMeasurementEx(
|
err = exp.saveMeasurement(
|
||||||
m, filename, func(v interface{}) ([]byte, error) {
|
m, filename, func(v interface{}) ([]byte, error) {
|
||||||
return nil, errors.New("mocked error")
|
return nil, errors.New("mocked error")
|
||||||
}, os.OpenFile, func(fp *os.File, b []byte) (int, error) {
|
}, os.OpenFile, func(fp *os.File, b []byte) (int, error) {
|
||||||
|
@ -381,7 +382,7 @@ func TestSaveMeasurementErrors(t *testing.T) {
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatal("expected an error here")
|
t.Fatal("expected an error here")
|
||||||
}
|
}
|
||||||
err = exp.SaveMeasurementEx(
|
err = exp.saveMeasurement(
|
||||||
m, filename, json.Marshal,
|
m, filename, json.Marshal,
|
||||||
func(name string, flag int, perm os.FileMode) (*os.File, error) {
|
func(name string, flag int, perm os.FileMode) (*os.File, error) {
|
||||||
return nil, errors.New("mocked error")
|
return nil, errors.New("mocked error")
|
||||||
|
@ -392,7 +393,7 @@ func TestSaveMeasurementErrors(t *testing.T) {
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatal("expected an error here")
|
t.Fatal("expected an error here")
|
||||||
}
|
}
|
||||||
err = exp.SaveMeasurementEx(
|
err = exp.saveMeasurement(
|
||||||
m, filename, json.Marshal, os.OpenFile,
|
m, filename, json.Marshal, os.OpenFile,
|
||||||
func(fp *os.File, b []byte) (int, error) {
|
func(fp *os.File, b []byte) (int, error) {
|
||||||
return 0, errors.New("mocked error")
|
return 0, errors.New("mocked error")
|
||||||
|
@ -417,10 +418,11 @@ func TestOpenReportIdempotent(t *testing.T) {
|
||||||
if exp.ReportID() != "" {
|
if exp.ReportID() != "" {
|
||||||
t.Fatal("unexpected initial report ID")
|
t.Fatal("unexpected initial report ID")
|
||||||
}
|
}
|
||||||
if err := exp.SubmitAndUpdateMeasurement(&model.Measurement{}); err == nil {
|
ctx := context.Background()
|
||||||
|
if err := exp.SubmitAndUpdateMeasurementContext(ctx, &model.Measurement{}); err == nil {
|
||||||
t.Fatal("we should not be able to submit before OpenReport")
|
t.Fatal("we should not be able to submit before OpenReport")
|
||||||
}
|
}
|
||||||
err = exp.OpenReport()
|
err = exp.OpenReportContext(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -428,7 +430,7 @@ func TestOpenReportIdempotent(t *testing.T) {
|
||||||
if rid == "" {
|
if rid == "" {
|
||||||
t.Fatal("invalid report ID")
|
t.Fatal("invalid report ID")
|
||||||
}
|
}
|
||||||
err = exp.OpenReport()
|
err = exp.OpenReportContext(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -453,12 +455,12 @@ func TestOpenReportFailure(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
exp := builder.NewExperiment()
|
exp := builder.NewExperiment().(*experiment)
|
||||||
exp.session.selectedProbeService = &model.OOAPIService{
|
exp.session.selectedProbeService = &model.OOAPIService{
|
||||||
Address: server.URL,
|
Address: server.URL,
|
||||||
Type: "https",
|
Type: "https",
|
||||||
}
|
}
|
||||||
err = exp.OpenReport()
|
err = exp.OpenReportContext(context.Background())
|
||||||
if !strings.HasPrefix(err.Error(), "httpx: request failed") {
|
if !strings.HasPrefix(err.Error(), "httpx: request failed") {
|
||||||
t.Fatal("not the error we expected")
|
t.Fatal("not the error we expected")
|
||||||
}
|
}
|
||||||
|
@ -474,12 +476,12 @@ func TestOpenReportNewClientFailure(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
exp := builder.NewExperiment()
|
exp := builder.NewExperiment().(*experiment)
|
||||||
exp.session.selectedProbeService = &model.OOAPIService{
|
exp.session.selectedProbeService = &model.OOAPIService{
|
||||||
Address: "antani:///",
|
Address: "antani:///",
|
||||||
Type: "antani",
|
Type: "antani",
|
||||||
}
|
}
|
||||||
err = exp.OpenReport()
|
err = exp.OpenReportContext(context.Background())
|
||||||
if err.Error() != "probe services: unsupported endpoint type" {
|
if err.Error() != "probe services: unsupported endpoint type" {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -497,7 +499,7 @@ func TestSubmitAndUpdateMeasurementWithClosedReport(t *testing.T) {
|
||||||
}
|
}
|
||||||
exp := builder.NewExperiment()
|
exp := builder.NewExperiment()
|
||||||
m := new(model.Measurement)
|
m := new(model.Measurement)
|
||||||
err = exp.SubmitAndUpdateMeasurement(m)
|
err = exp.SubmitAndUpdateMeasurementContext(context.Background(), m)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatal("expected an error here")
|
t.Fatal("expected an error here")
|
||||||
}
|
}
|
||||||
|
@ -509,7 +511,7 @@ func TestMeasureLookupLocationFailure(t *testing.T) {
|
||||||
}
|
}
|
||||||
sess := newSessionForTestingNoLookups(t)
|
sess := newSessionForTestingNoLookups(t)
|
||||||
defer sess.Close()
|
defer sess.Close()
|
||||||
exp := NewExperiment(sess, new(antaniMeasurer))
|
exp := newExperiment(sess, new(antaniMeasurer))
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
cancel() // so we fail immediately
|
cancel() // so we fail immediately
|
||||||
if _, err := exp.MeasureWithContext(ctx, "xx"); err == nil {
|
if _, err := exp.MeasureWithContext(ctx, "xx"); err == nil {
|
||||||
|
@ -529,8 +531,8 @@ func TestOpenReportNonHTTPS(t *testing.T) {
|
||||||
Type: "mascetti",
|
Type: "mascetti",
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
exp := NewExperiment(sess, new(antaniMeasurer))
|
exp := newExperiment(sess, new(antaniMeasurer))
|
||||||
if err := exp.OpenReport(); err == nil {
|
if err := exp.OpenReportContext(context.Background()); err == nil {
|
||||||
t.Fatal("expected an error here")
|
t.Fatal("expected an error here")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,16 +0,0 @@
|
||||||
package engine
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/model"
|
|
||||||
)
|
|
||||||
|
|
||||||
func (e *Experiment) SaveMeasurementEx(
|
|
||||||
measurement *model.Measurement, filePath string,
|
|
||||||
marshal func(v interface{}) ([]byte, error),
|
|
||||||
openFile func(name string, flag int, perm os.FileMode) (*os.File, error),
|
|
||||||
write func(fp *os.File, b []byte) (n int, err error),
|
|
||||||
) error {
|
|
||||||
return e.saveMeasurement(measurement, filePath, marshal, openFile, write)
|
|
||||||
}
|
|
|
@ -14,7 +14,7 @@ func TestExperimentHonoursSharingDefaults(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
exp := builder.NewExperiment()
|
exp := builder.NewExperiment().(*experiment)
|
||||||
return exp.newMeasurement("")
|
return exp.newMeasurement("")
|
||||||
}
|
}
|
||||||
type spec struct {
|
type spec struct {
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
package engine
|
package engine
|
||||||
|
|
||||||
|
//
|
||||||
|
// ExperimentBuilder definition and implementation
|
||||||
|
//
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
@ -42,23 +46,60 @@ const (
|
||||||
InputOrStaticDefault = InputPolicy("or_static_default")
|
InputOrStaticDefault = InputPolicy("or_static_default")
|
||||||
)
|
)
|
||||||
|
|
||||||
// ExperimentBuilder is an experiment builder.
|
// ExperimentBuilder builds an experiment.
|
||||||
type ExperimentBuilder struct {
|
type ExperimentBuilder interface {
|
||||||
build func(interface{}) *Experiment
|
// Interruptible tells you whether this is an interruptible experiment. This kind
|
||||||
callbacks model.ExperimentCallbacks
|
// of experiments (e.g. ndt7) may be interrupted mid way.
|
||||||
config interface{}
|
Interruptible() bool
|
||||||
inputPolicy InputPolicy
|
|
||||||
|
// InputPolicy returns the experiment input policy.
|
||||||
|
InputPolicy() InputPolicy
|
||||||
|
|
||||||
|
// Options returns information about the experiment's options.
|
||||||
|
Options() (map[string]OptionInfo, error)
|
||||||
|
|
||||||
|
// SetOptionAny sets an option whose value is an any value. We will use reasonable
|
||||||
|
// heuristics to convert the any value to the proper type of the field whose name is
|
||||||
|
// contained by the key variable. If we cannot convert the provided any value to
|
||||||
|
// the proper type, then this function returns an error.
|
||||||
|
SetOptionAny(key string, value any) error
|
||||||
|
|
||||||
|
// SetOptionsAny sets options from a map[string]any. See the documentation of
|
||||||
|
// the SetOptionAny method for more information.
|
||||||
|
SetOptionsAny(options map[string]any) error
|
||||||
|
|
||||||
|
// SetCallbacks sets the experiment's interactive callbacks.
|
||||||
|
SetCallbacks(callbacks model.ExperimentCallbacks)
|
||||||
|
|
||||||
|
// NewExperiment creates the experiment instance.
|
||||||
|
NewExperiment() Experiment
|
||||||
|
}
|
||||||
|
|
||||||
|
// experimentBuilder implements ExperimentBuilder.
|
||||||
|
type experimentBuilder struct {
|
||||||
|
// build is the constructor that build an experiment with the given config.
|
||||||
|
build func(config interface{}) *experiment
|
||||||
|
|
||||||
|
// callbacks contains callbacks for the new experiment.
|
||||||
|
callbacks model.ExperimentCallbacks
|
||||||
|
|
||||||
|
// config contains the experiment's config.
|
||||||
|
config interface{}
|
||||||
|
|
||||||
|
// inputPolicy contains the experiment's InputPolicy.
|
||||||
|
inputPolicy InputPolicy
|
||||||
|
|
||||||
|
// interruptible indicates whether the experiment is interruptible.
|
||||||
interruptible bool
|
interruptible bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Interruptible tells you whether this is an interruptible experiment. This kind
|
// Interruptible implements ExperimentBuilder.Interruptible.
|
||||||
// of experiments (e.g. ndt7) may be interrupted mid way.
|
func (b *experimentBuilder) Interruptible() bool {
|
||||||
func (b *ExperimentBuilder) Interruptible() bool {
|
|
||||||
return b.interruptible
|
return b.interruptible
|
||||||
}
|
}
|
||||||
|
|
||||||
// InputPolicy returns the experiment input policy
|
// InputPolicy implements ExperimentBuilder.InputPolicy.
|
||||||
func (b *ExperimentBuilder) InputPolicy() InputPolicy {
|
func (b *experimentBuilder) InputPolicy() InputPolicy {
|
||||||
return b.inputPolicy
|
return b.inputPolicy
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,8 +137,8 @@ var (
|
||||||
ErrUnsupportedOptionType = errors.New("unsupported option type")
|
ErrUnsupportedOptionType = errors.New("unsupported option type")
|
||||||
)
|
)
|
||||||
|
|
||||||
// Options returns info about all options
|
// Options implements ExperimentBuilder.Options.
|
||||||
func (b *ExperimentBuilder) Options() (map[string]OptionInfo, error) {
|
func (b *experimentBuilder) Options() (map[string]OptionInfo, error) {
|
||||||
result := make(map[string]OptionInfo)
|
result := make(map[string]OptionInfo)
|
||||||
ptrinfo := reflect.ValueOf(b.config)
|
ptrinfo := reflect.ValueOf(b.config)
|
||||||
if ptrinfo.Kind() != reflect.Ptr {
|
if ptrinfo.Kind() != reflect.Ptr {
|
||||||
|
@ -118,7 +159,7 @@ func (b *ExperimentBuilder) Options() (map[string]OptionInfo, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// setOptionBool sets a bool option.
|
// setOptionBool sets a bool option.
|
||||||
func (b *ExperimentBuilder) setOptionBool(field reflect.Value, value any) error {
|
func (b *experimentBuilder) setOptionBool(field reflect.Value, value any) error {
|
||||||
switch v := value.(type) {
|
switch v := value.(type) {
|
||||||
case bool:
|
case bool:
|
||||||
field.SetBool(v)
|
field.SetBool(v)
|
||||||
|
@ -135,7 +176,7 @@ func (b *ExperimentBuilder) setOptionBool(field reflect.Value, value any) error
|
||||||
}
|
}
|
||||||
|
|
||||||
// setOptionInt sets an int option
|
// setOptionInt sets an int option
|
||||||
func (b *ExperimentBuilder) setOptionInt(field reflect.Value, value any) error {
|
func (b *experimentBuilder) setOptionInt(field reflect.Value, value any) error {
|
||||||
switch v := value.(type) {
|
switch v := value.(type) {
|
||||||
case int64:
|
case int64:
|
||||||
field.SetInt(v)
|
field.SetInt(v)
|
||||||
|
@ -165,7 +206,7 @@ func (b *ExperimentBuilder) setOptionInt(field reflect.Value, value any) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
// setOptionString sets a string option
|
// setOptionString sets a string option
|
||||||
func (b *ExperimentBuilder) setOptionString(field reflect.Value, value any) error {
|
func (b *experimentBuilder) setOptionString(field reflect.Value, value any) error {
|
||||||
switch v := value.(type) {
|
switch v := value.(type) {
|
||||||
case string:
|
case string:
|
||||||
field.SetString(v)
|
field.SetString(v)
|
||||||
|
@ -175,11 +216,8 @@ func (b *ExperimentBuilder) setOptionString(field reflect.Value, value any) erro
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetOptionAny sets an option whose value is an any value. We will use reasonable
|
// SetOptionAny implements ExperimentBuilder.SetOptionAny.
|
||||||
// heuristics to convert the any value to the proper type of the field whose name is
|
func (b *experimentBuilder) SetOptionAny(key string, value any) error {
|
||||||
// contained by the key variable. If we cannot convert the provided any value to
|
|
||||||
// the proper type, then this function returns an error.
|
|
||||||
func (b *ExperimentBuilder) SetOptionAny(key string, value any) error {
|
|
||||||
field, err := b.fieldbyname(b.config, key)
|
field, err := b.fieldbyname(b.config, key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -196,9 +234,8 @@ func (b *ExperimentBuilder) SetOptionAny(key string, value any) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetOptionsAny sets options from a map[string]any. See the documentation of
|
// SetOptionsAny implements ExperimentBuilder.SetOptionsAny.
|
||||||
// the SetOptionAny function for more information.
|
func (b *experimentBuilder) SetOptionsAny(options map[string]any) error {
|
||||||
func (b *ExperimentBuilder) SetOptionsAny(options map[string]any) error {
|
|
||||||
for key, value := range options {
|
for key, value := range options {
|
||||||
if err := b.SetOptionAny(key, value); err != nil {
|
if err := b.SetOptionAny(key, value); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -207,12 +244,13 @@ func (b *ExperimentBuilder) SetOptionsAny(options map[string]any) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetCallbacks sets the interactive callbacks
|
// SetCallbacks implements ExperimentBuilder.SetCallbacks.
|
||||||
func (b *ExperimentBuilder) SetCallbacks(callbacks model.ExperimentCallbacks) {
|
func (b *experimentBuilder) SetCallbacks(callbacks model.ExperimentCallbacks) {
|
||||||
b.callbacks = callbacks
|
b.callbacks = callbacks
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *ExperimentBuilder) fieldbyname(v interface{}, key string) (reflect.Value, error) {
|
// fieldbyname return v's field whose name is equal to the given key.
|
||||||
|
func (b *experimentBuilder) fieldbyname(v interface{}, key string) (reflect.Value, error) {
|
||||||
// See https://stackoverflow.com/a/6396678/4354461
|
// See https://stackoverflow.com/a/6396678/4354461
|
||||||
ptrinfo := reflect.ValueOf(v)
|
ptrinfo := reflect.ValueOf(v)
|
||||||
if ptrinfo.Kind() != reflect.Ptr {
|
if ptrinfo.Kind() != reflect.Ptr {
|
||||||
|
@ -230,7 +268,7 @@ func (b *ExperimentBuilder) fieldbyname(v interface{}, key string) (reflect.Valu
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewExperiment creates the experiment
|
// NewExperiment creates the experiment
|
||||||
func (b *ExperimentBuilder) NewExperiment() *Experiment {
|
func (b *experimentBuilder) NewExperiment() Experiment {
|
||||||
experiment := b.build(b.config)
|
experiment := b.build(b.config)
|
||||||
experiment.callbacks = b.callbacks
|
experiment.callbacks = b.callbacks
|
||||||
return experiment
|
return experiment
|
||||||
|
@ -255,7 +293,8 @@ func canonicalizeExperimentName(name string) string {
|
||||||
return name
|
return name
|
||||||
}
|
}
|
||||||
|
|
||||||
func newExperimentBuilder(session *Session, name string) (*ExperimentBuilder, error) {
|
// newExperimentBuilder creates a new experimentBuilder instance.
|
||||||
|
func newExperimentBuilder(session *Session, name string) (*experimentBuilder, error) {
|
||||||
factory := experimentsByName[canonicalizeExperimentName(name)]
|
factory := experimentsByName[canonicalizeExperimentName(name)]
|
||||||
if factory == nil {
|
if factory == nil {
|
||||||
return nil, fmt.Errorf("no such experiment: %s", name)
|
return nil, fmt.Errorf("no such experiment: %s", name)
|
||||||
|
|
|
@ -16,7 +16,7 @@ type fakeExperimentConfig struct {
|
||||||
|
|
||||||
func TestExperimentBuilderOptions(t *testing.T) {
|
func TestExperimentBuilderOptions(t *testing.T) {
|
||||||
t.Run("when config is not a pointer", func(t *testing.T) {
|
t.Run("when config is not a pointer", func(t *testing.T) {
|
||||||
b := &ExperimentBuilder{
|
b := &experimentBuilder{
|
||||||
config: 17,
|
config: 17,
|
||||||
}
|
}
|
||||||
options, err := b.Options()
|
options, err := b.Options()
|
||||||
|
@ -30,7 +30,7 @@ func TestExperimentBuilderOptions(t *testing.T) {
|
||||||
|
|
||||||
t.Run("when config is not a struct", func(t *testing.T) {
|
t.Run("when config is not a struct", func(t *testing.T) {
|
||||||
number := 17
|
number := 17
|
||||||
b := &ExperimentBuilder{
|
b := &experimentBuilder{
|
||||||
config: &number,
|
config: &number,
|
||||||
}
|
}
|
||||||
options, err := b.Options()
|
options, err := b.Options()
|
||||||
|
@ -44,7 +44,7 @@ func TestExperimentBuilderOptions(t *testing.T) {
|
||||||
|
|
||||||
t.Run("when config is a pointer to struct", func(t *testing.T) {
|
t.Run("when config is a pointer to struct", func(t *testing.T) {
|
||||||
config := &fakeExperimentConfig{}
|
config := &fakeExperimentConfig{}
|
||||||
b := &ExperimentBuilder{
|
b := &experimentBuilder{
|
||||||
config: config,
|
config: config,
|
||||||
}
|
}
|
||||||
options, err := b.Options()
|
options, err := b.Options()
|
||||||
|
@ -291,7 +291,7 @@ func TestExperimentBuilderSetOptionAny(t *testing.T) {
|
||||||
for _, input := range inputs {
|
for _, input := range inputs {
|
||||||
t.Run(input.TestCaseName, func(t *testing.T) {
|
t.Run(input.TestCaseName, func(t *testing.T) {
|
||||||
ec := input.InitialConfig
|
ec := input.InitialConfig
|
||||||
b := &ExperimentBuilder{config: ec}
|
b := &experimentBuilder{config: ec}
|
||||||
err := b.SetOptionAny(input.FieldName, input.FieldValue)
|
err := b.SetOptionAny(input.FieldName, input.FieldValue)
|
||||||
if !errors.Is(err, input.ExpectErr) {
|
if !errors.Is(err, input.ExpectErr) {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
|
@ -304,7 +304,7 @@ func TestExperimentBuilderSetOptionAny(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestExperimentBuilderSetOptionsAny(t *testing.T) {
|
func TestExperimentBuilderSetOptionsAny(t *testing.T) {
|
||||||
b := &ExperimentBuilder{config: &fakeExperimentConfig{}}
|
b := &experimentBuilder{config: &fakeExperimentConfig{}}
|
||||||
|
|
||||||
t.Run("we correctly handle an empty map", func(t *testing.T) {
|
t.Run("we correctly handle an empty map", func(t *testing.T) {
|
||||||
if err := b.SetOptionsAny(nil); err != nil {
|
if err := b.SetOptionsAny(nil); err != nil {
|
||||||
|
@ -314,7 +314,7 @@ func TestExperimentBuilderSetOptionsAny(t *testing.T) {
|
||||||
|
|
||||||
t.Run("we correctly handle a map containing options", func(t *testing.T) {
|
t.Run("we correctly handle a map containing options", func(t *testing.T) {
|
||||||
f := &fakeExperimentConfig{}
|
f := &fakeExperimentConfig{}
|
||||||
privateb := &ExperimentBuilder{config: f}
|
privateb := &experimentBuilder{config: f}
|
||||||
opts := map[string]any{
|
opts := map[string]any{
|
||||||
"String": "yoloyolo",
|
"String": "yoloyolo",
|
||||||
"Value": "174",
|
"Value": "174",
|
||||||
|
|
|
@ -390,7 +390,7 @@ var ErrAlreadyUsingProxy = errors.New(
|
||||||
// NewExperimentBuilder returns a new experiment builder
|
// NewExperimentBuilder returns a new experiment builder
|
||||||
// for the experiment with the given name, or an error if
|
// for the experiment with the given name, or an error if
|
||||||
// there's no such experiment with the given name
|
// there's no such experiment with the given name
|
||||||
func (s *Session) NewExperimentBuilder(name string) (*ExperimentBuilder, error) {
|
func (s *Session) NewExperimentBuilder(name string) (ExperimentBuilder, error) {
|
||||||
return newExperimentBuilder(s, name)
|
return newExperimentBuilder(s, name)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -67,7 +67,7 @@ type experimentBuilder interface {
|
||||||
|
|
||||||
// experimentBuilderWrapper wraps *ExperimentBuilder
|
// experimentBuilderWrapper wraps *ExperimentBuilder
|
||||||
type experimentBuilderWrapper struct {
|
type experimentBuilderWrapper struct {
|
||||||
eb *engine.ExperimentBuilder
|
eb engine.ExperimentBuilder
|
||||||
}
|
}
|
||||||
|
|
||||||
// newExperiment implements experimentBuilder.newExperiment
|
// newExperiment implements experimentBuilder.newExperiment
|
||||||
|
|
|
@ -59,7 +59,7 @@ func (sess *taskSessionEngine) NewExperimentBuilderByName(
|
||||||
// taskExperimentBuilderEngine wraps ./internal/engine's
|
// taskExperimentBuilderEngine wraps ./internal/engine's
|
||||||
// ExperimentBuilder type.
|
// ExperimentBuilder type.
|
||||||
type taskExperimentBuilderEngine struct {
|
type taskExperimentBuilderEngine struct {
|
||||||
*engine.ExperimentBuilder
|
engine.ExperimentBuilder
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ taskExperimentBuilder = &taskExperimentBuilderEngine{}
|
var _ taskExperimentBuilder = &taskExperimentBuilderEngine{}
|
||||||
|
@ -72,7 +72,7 @@ func (b *taskExperimentBuilderEngine) NewExperimentInstance() taskExperiment {
|
||||||
|
|
||||||
// taskExperimentEngine wraps ./internal/engine's Experiment.
|
// taskExperimentEngine wraps ./internal/engine's Experiment.
|
||||||
type taskExperimentEngine struct {
|
type taskExperimentEngine struct {
|
||||||
*engine.Experiment
|
engine.Experiment
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ taskExperiment = &taskExperimentEngine{}
|
var _ taskExperiment = &taskExperimentEngine{}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user