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:
Simone Basso 2022-07-08 12:29:23 +02:00 committed by GitHub
parent 5b27df1a37
commit 97864b324f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 325 additions and 283 deletions

View File

@ -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")

View File

@ -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),
)) ))
}, },

View File

@ -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),

View File

@ -162,7 +162,7 @@ func TestSetCallbacks(t *testing.T) {
} }
register := &registerCallbacksCalled{} register := &registerCallbacksCalled{}
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")
} }
} }

View File

@ -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)
}

View File

@ -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 {

View File

@ -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)

View File

@ -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",

View File

@ -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)
} }

View File

@ -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

View File

@ -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{}