diff --git a/cmd/ooniprobe/internal/nettests/dnscheck.go b/cmd/ooniprobe/internal/nettests/dnscheck.go index cb5356015..2c8e79862 100644 --- a/cmd/ooniprobe/internal/nettests/dnscheck.go +++ b/cmd/ooniprobe/internal/nettests/dnscheck.go @@ -4,23 +4,21 @@ import ( "context" "github.com/ooni/probe-cli/v3/internal/model" - "github.com/ooni/probe-cli/v3/internal/targetloading" ) // DNSCheck nettest implementation. type DNSCheck struct{} -func (n DNSCheck) lookupURLs(ctl *Controller) ([]model.ExperimentTarget, error) { - targetloader := &targetloading.Loader{ +func (n DNSCheck) lookupURLs(ctl *Controller, builder model.ExperimentBuilder) ([]model.ExperimentTarget, error) { + config := &model.ExperimentTargetLoaderConfig{ CheckInConfig: &model.OOAPICheckInConfig{ // not needed because we have default static input in the engine }, - ExperimentName: "dnscheck", - InputPolicy: model.InputOrStaticDefault, - Session: ctl.Session, - SourceFiles: ctl.InputFiles, - StaticInputs: ctl.Inputs, + Session: ctl.Session, + SourceFiles: ctl.InputFiles, + StaticInputs: ctl.Inputs, } + targetloader := builder.NewTargetLoader(config) testlist, err := targetloader.Load(context.Background()) if err != nil { return nil, err @@ -34,7 +32,7 @@ func (n DNSCheck) Run(ctl *Controller) error { if err != nil { return err } - urls, err := n.lookupURLs(ctl) + urls, err := n.lookupURLs(ctl, builder) if err != nil { return err } diff --git a/cmd/ooniprobe/internal/nettests/stunreachability.go b/cmd/ooniprobe/internal/nettests/stunreachability.go index 743d53e8f..c6313d96c 100644 --- a/cmd/ooniprobe/internal/nettests/stunreachability.go +++ b/cmd/ooniprobe/internal/nettests/stunreachability.go @@ -4,23 +4,21 @@ import ( "context" "github.com/ooni/probe-cli/v3/internal/model" - "github.com/ooni/probe-cli/v3/internal/targetloading" ) // STUNReachability nettest implementation. type STUNReachability struct{} -func (n STUNReachability) lookupURLs(ctl *Controller) ([]model.ExperimentTarget, error) { - targetloader := &targetloading.Loader{ +func (n STUNReachability) lookupURLs(ctl *Controller, builder model.ExperimentBuilder) ([]model.ExperimentTarget, error) { + config := &model.ExperimentTargetLoaderConfig{ CheckInConfig: &model.OOAPICheckInConfig{ // not needed because we have default static input in the engine }, - ExperimentName: "stunreachability", - InputPolicy: model.InputOrStaticDefault, - Session: ctl.Session, - SourceFiles: ctl.InputFiles, - StaticInputs: ctl.Inputs, + Session: ctl.Session, + SourceFiles: ctl.InputFiles, + StaticInputs: ctl.Inputs, } + targetloader := builder.NewTargetLoader(config) testlist, err := targetloader.Load(context.Background()) if err != nil { return nil, err @@ -34,7 +32,7 @@ func (n STUNReachability) Run(ctl *Controller) error { if err != nil { return err } - urls, err := n.lookupURLs(ctl) + urls, err := n.lookupURLs(ctl, builder) if err != nil { return err } diff --git a/cmd/ooniprobe/internal/nettests/web_connectivity.go b/cmd/ooniprobe/internal/nettests/web_connectivity.go index 7e4bcad99..0da3be56d 100644 --- a/cmd/ooniprobe/internal/nettests/web_connectivity.go +++ b/cmd/ooniprobe/internal/nettests/web_connectivity.go @@ -5,11 +5,11 @@ import ( "github.com/apex/log" "github.com/ooni/probe-cli/v3/internal/model" - "github.com/ooni/probe-cli/v3/internal/targetloading" ) -func (n WebConnectivity) lookupURLs(ctl *Controller, categories []string) ([]model.ExperimentTarget, error) { - targetloader := &targetloading.Loader{ +func (n WebConnectivity) lookupURLs( + ctl *Controller, builder model.ExperimentBuilder, categories []string) ([]model.ExperimentTarget, error) { + config := &model.ExperimentTargetLoaderConfig{ CheckInConfig: &model.OOAPICheckInConfig{ // Setting Charging and OnWiFi to true causes the CheckIn // API to return to us as much URL as possible with the @@ -21,12 +21,11 @@ func (n WebConnectivity) lookupURLs(ctl *Controller, categories []string) ([]mod CategoryCodes: categories, }, }, - ExperimentName: "web_connectivity", - InputPolicy: model.InputOrQueryBackend, - Session: ctl.Session, - SourceFiles: ctl.InputFiles, - StaticInputs: ctl.Inputs, + Session: ctl.Session, + SourceFiles: ctl.InputFiles, + StaticInputs: ctl.Inputs, } + targetloader := builder.NewTargetLoader(config) testlist, err := targetloader.Load(context.Background()) if err != nil { return nil, err @@ -39,12 +38,12 @@ type WebConnectivity struct{} // Run starts the test func (n WebConnectivity) Run(ctl *Controller) error { - log.Debugf("Enabled category codes are the following %v", ctl.Probe.Config().Nettests.WebsitesEnabledCategoryCodes) - urls, err := n.lookupURLs(ctl, ctl.Probe.Config().Nettests.WebsitesEnabledCategoryCodes) + builder, err := ctl.Session.NewExperimentBuilder("web_connectivity") if err != nil { return err } - builder, err := ctl.Session.NewExperimentBuilder("web_connectivity") + log.Debugf("Enabled category codes are the following %v", ctl.Probe.Config().Nettests.WebsitesEnabledCategoryCodes) + urls, err := n.lookupURLs(ctl, builder, ctl.Probe.Config().Nettests.WebsitesEnabledCategoryCodes) if err != nil { return err } diff --git a/internal/engine/experimentbuilder.go b/internal/engine/experimentbuilder.go index a60d0a50d..330777957 100644 --- a/internal/engine/experimentbuilder.go +++ b/internal/engine/experimentbuilder.go @@ -22,6 +22,8 @@ type experimentBuilder struct { session *Session } +var _ model.ExperimentBuilder = &experimentBuilder{} + // Interruptible implements ExperimentBuilder.Interruptible. func (b *experimentBuilder) Interruptible() bool { return b.factory.Interruptible() @@ -60,6 +62,11 @@ func (b *experimentBuilder) NewExperiment() model.Experiment { return experiment } +// NewTargetLoader creates a new [model.ExperimentTargetLoader] instance. +func (b *experimentBuilder) NewTargetLoader(config *model.ExperimentTargetLoaderConfig) model.ExperimentTargetLoader { + return b.factory.NewTargetLoader(config) +} + // newExperimentBuilder creates a new experimentBuilder instance. func newExperimentBuilder(session *Session, name string) (*experimentBuilder, error) { factory, err := registry.NewFactory(name, session.kvStore, session.logger) diff --git a/internal/engine/experimentbuilder_test.go b/internal/engine/experimentbuilder_test.go index 00a22ef62..d81e0bb7b 100644 --- a/internal/engine/experimentbuilder_test.go +++ b/internal/engine/experimentbuilder_test.go @@ -1 +1,51 @@ package engine + +import ( + "context" + "errors" + "testing" + + "github.com/ooni/probe-cli/v3/internal/model" +) + +func TestExperimentBuilderEngineWebConnectivity(t *testing.T) { + // create a session for testing that does not use the network at all + sess := newSessionForTestingNoLookups(t) + + // create an experiment builder for Web Connectivity + builder, err := sess.NewExperimentBuilder("WebConnectivity") + if err != nil { + t.Fatal(err) + } + + // create suitable loader config + config := &model.ExperimentTargetLoaderConfig{ + CheckInConfig: &model.OOAPICheckInConfig{ + // nothing + }, + Session: sess, + StaticInputs: nil, + SourceFiles: nil, + } + + // create the loader + loader := builder.NewTargetLoader(config) + + // create cancelled context to interrupt immediately so that we + // don't use the network when running this test + ctx, cancel := context.WithCancel(context.Background()) + cancel() + + // attempt to load targets + targets, err := loader.Load(ctx) + + // make sure we've got the expected error + if !errors.Is(err, context.Canceled) { + t.Fatal("unexpected err", err) + } + + // make sure there are no targets + if len(targets) != 0 { + t.Fatal("expected zero length targets") + } +} diff --git a/internal/experimentname/experimentname.go b/internal/experimentname/experimentname.go new file mode 100644 index 000000000..193edc613 --- /dev/null +++ b/internal/experimentname/experimentname.go @@ -0,0 +1,25 @@ +// Package experimentname contains code to manipulate experiment names. +package experimentname + +import "github.com/ooni/probe-cli/v3/internal/strcasex" + +// Canonicalize allows code to provide experiment names +// in a more flexible way, where we have aliases. +// +// Because we allow for uppercase experiment names for backwards +// compatibility with MK, we need to add some exceptions here when +// mapping (e.g., DNSCheck => dnscheck). +func Canonicalize(name string) string { + switch name = strcasex.ToSnake(name); name { + case "ndt_7": + name = "ndt" // since 2020-03-18, we use ndt7 to implement ndt by default + case "dns_check": + name = "dnscheck" + case "stun_reachability": + name = "stunreachability" + case "web_connectivity@v_0_5": + name = "web_connectivity@v0.5" + default: + } + return name +} diff --git a/internal/experimentname/experimentname_test.go b/internal/experimentname/experimentname_test.go new file mode 100644 index 000000000..df191d768 --- /dev/null +++ b/internal/experimentname/experimentname_test.go @@ -0,0 +1,55 @@ +// Package experimentname contains code to manipulate experiment names. +package experimentname + +import "testing" + +func TestCanonicalize(t *testing.T) { + tests := []struct { + input string + expect string + }{ + { + input: "example", + expect: "example", + }, + { + input: "Example", + expect: "example", + }, + { + input: "ndt7", + expect: "ndt", + }, + { + input: "Ndt7", + expect: "ndt", + }, + { + input: "DNSCheck", + expect: "dnscheck", + }, + { + input: "dns_check", + expect: "dnscheck", + }, + { + input: "STUNReachability", + expect: "stunreachability", + }, + { + input: "stun_reachability", + expect: "stunreachability", + }, + { + input: "WebConnectivity@v0.5", + expect: "web_connectivity@v0.5", + }, + } + for _, tt := range tests { + t.Run(tt.input, func(t *testing.T) { + if got := Canonicalize(tt.input); got != tt.expect { + t.Errorf("Canonicalize() = %v, want %v", got, tt.expect) + } + }) + } +} diff --git a/internal/mocks/experimentbuilder.go b/internal/mocks/experimentbuilder.go index 16763d471..1f6a27187 100644 --- a/internal/mocks/experimentbuilder.go +++ b/internal/mocks/experimentbuilder.go @@ -17,8 +17,12 @@ type ExperimentBuilder struct { MockSetCallbacks func(callbacks model.ExperimentCallbacks) MockNewExperiment func() model.Experiment + + MockNewTargetLoader func(config *model.ExperimentTargetLoaderConfig) model.ExperimentTargetLoader } +var _ model.ExperimentBuilder = &ExperimentBuilder{} + func (eb *ExperimentBuilder) Interruptible() bool { return eb.MockInterruptible() } @@ -46,3 +50,7 @@ func (eb *ExperimentBuilder) SetCallbacks(callbacks model.ExperimentCallbacks) { func (eb *ExperimentBuilder) NewExperiment() model.Experiment { return eb.MockNewExperiment() } + +func (eb *ExperimentBuilder) NewTargetLoader(config *model.ExperimentTargetLoaderConfig) model.ExperimentTargetLoader { + return eb.MockNewTargetLoader(config) +} diff --git a/internal/mocks/experimentbuilder_test.go b/internal/mocks/experimentbuilder_test.go index 6fd1ce532..55ba1783f 100644 --- a/internal/mocks/experimentbuilder_test.go +++ b/internal/mocks/experimentbuilder_test.go @@ -96,4 +96,16 @@ func TestExperimentBuilder(t *testing.T) { t.Fatal("invalid result") } }) + + t.Run("NewTargetLoader", func(t *testing.T) { + tloader := &ExperimentTargetLoader{} + eb := &ExperimentBuilder{ + MockNewTargetLoader: func(*model.ExperimentTargetLoaderConfig) model.ExperimentTargetLoader { + return tloader + }, + } + if out := eb.NewTargetLoader(&model.ExperimentTargetLoaderConfig{}); out != tloader { + t.Fatal("invalid result") + } + }) } diff --git a/internal/model/experiment.go b/internal/model/experiment.go index 9ae3dbcce..8bd309f32 100644 --- a/internal/model/experiment.go +++ b/internal/model/experiment.go @@ -232,8 +232,46 @@ type ExperimentBuilder interface { // SetCallbacks sets the experiment's interactive callbacks. SetCallbacks(callbacks ExperimentCallbacks) - // NewExperiment creates the experiment instance. + // NewExperiment creates the [Experiment] instance. NewExperiment() Experiment + + // NewTargetLoader creates the [ExperimentTargetLoader] instance. + NewTargetLoader(config *ExperimentTargetLoaderConfig) ExperimentTargetLoader +} + +// ExperimentTargetLoaderConfig is the configuration to create a new [ExperimentTargetLoader]. +// +// The zero value is not ready to use; please, init the MANDATORY fields. +type ExperimentTargetLoaderConfig struct { + // CheckInConfig contains OPTIONAL options for the CheckIn API. If not set, then we'll create a + // default config. If set but there are fields inside it that are not set, then we will set them + // to a default value. + CheckInConfig *OOAPICheckInConfig + + // Session is the MANDATORY current measurement session. + Session ExperimentTargetLoaderSession + + // StaticInputs contains OPTIONAL input to be added + // to the resulting input list if possible. + StaticInputs []string + + // SourceFiles contains OPTIONAL files to read input + // from. Each file should contain a single input string + // per line. We will fail if any file is unreadable + // as well as if any file is empty. + SourceFiles []string +} + +// ExperimentTargetLoaderSession is the session according to [ExperimentTargetLoader]. +type ExperimentTargetLoaderSession interface { + // CheckIn invokes the check-in API. + CheckIn(ctx context.Context, config *OOAPICheckInConfig) (*OOAPICheckInResult, error) + + // FetchOpenVPNConfig fetches the OpenVPN experiment configuration. + FetchOpenVPNConfig(ctx context.Context, provider, cc string) (*OOAPIVPNProviderConfig, error) + + // Logger returns the logger to use. + Logger() Logger } // ExperimentOptionInfo contains info about an experiment option. diff --git a/internal/oonirun/experiment.go b/internal/oonirun/experiment.go index 09deedaed..758db9e0c 100644 --- a/internal/oonirun/experiment.go +++ b/internal/oonirun/experiment.go @@ -14,7 +14,6 @@ import ( "github.com/ooni/probe-cli/v3/internal/humanize" "github.com/ooni/probe-cli/v3/internal/model" - "github.com/ooni/probe-cli/v3/internal/targetloading" ) // experimentShuffledInputs counts how many times we shuffled inputs @@ -61,7 +60,7 @@ type Experiment struct { newExperimentBuilderFn func(experimentName string) (model.ExperimentBuilder, error) // newTargetLoaderFn is OPTIONAL and used for testing. - newTargetLoaderFn func(inputPolicy model.InputPolicy) targetLoader + newTargetLoaderFn func(builder model.ExperimentBuilder) targetLoader // newSubmitterFn is OPTIONAL and used for testing. newSubmitterFn func(ctx context.Context) (model.Submitter, error) @@ -84,7 +83,7 @@ func (ed *Experiment) Run(ctx context.Context) error { } // 2. create input loader and load input for this experiment - targetLoader := ed.newTargetLoader(builder.InputPolicy()) + targetLoader := ed.newTargetLoader(builder) targetList, err := targetLoader.Load(ctx) if err != nil { return err @@ -196,22 +195,20 @@ func (ed *Experiment) newExperimentBuilder(experimentName string) (model.Experim type targetLoader = model.ExperimentTargetLoader // newTargetLoader creates a new [model.ExperimentTargetLoader]. -func (ed *Experiment) newTargetLoader(inputPolicy model.InputPolicy) targetLoader { +func (ed *Experiment) newTargetLoader(builder model.ExperimentBuilder) targetLoader { if ed.newTargetLoaderFn != nil { - return ed.newTargetLoaderFn(inputPolicy) + return ed.newTargetLoaderFn(builder) } - return &targetloading.Loader{ + return builder.NewTargetLoader(&model.ExperimentTargetLoaderConfig{ CheckInConfig: &model.OOAPICheckInConfig{ RunType: model.RunTypeManual, OnWiFi: true, // meaning: not on 4G Charging: true, }, - ExperimentName: ed.Name, - InputPolicy: inputPolicy, - StaticInputs: ed.Inputs, - SourceFiles: ed.InputFilePaths, - Session: ed.Session, - } + StaticInputs: ed.Inputs, + SourceFiles: ed.InputFilePaths, + Session: ed.Session, + }) } // experimentOptionsToStringList convers the options to []string, which is diff --git a/internal/oonirun/experiment_test.go b/internal/oonirun/experiment_test.go index 29764d9c8..d438f47ce 100644 --- a/internal/oonirun/experiment_test.go +++ b/internal/oonirun/experiment_test.go @@ -67,6 +67,16 @@ func TestExperimentRunWithFailureToSubmitAndShuffle(t *testing.T) { } return exp }, + MockNewTargetLoader: func(config *model.ExperimentTargetLoaderConfig) model.ExperimentTargetLoader { + return &mocks.ExperimentTargetLoader{ + MockLoad: func(ctx context.Context) ([]model.ExperimentTarget, error) { + // Implementation note: the convention for input-less experiments is that + // they require a single entry containing an empty input. + entry := model.NewOOAPIURLInfoWithDefaultCategoryAndCountry("") + return []model.ExperimentTarget{entry}, nil + }, + } + }, } return eb, nil }, @@ -165,7 +175,7 @@ func TestExperimentRun(t *testing.T) { ReportFile string Session Session newExperimentBuilderFn func(experimentName string) (model.ExperimentBuilder, error) - newTargetLoaderFn func(inputPolicy model.InputPolicy) targetLoader + newTargetLoaderFn func(builder model.ExperimentBuilder) targetLoader newSubmitterFn func(ctx context.Context) (model.Submitter, error) newSaverFn func() (model.Saver, error) newInputProcessorFn func(experiment model.Experiment, @@ -199,7 +209,7 @@ func TestExperimentRun(t *testing.T) { } return eb, nil }, - newTargetLoaderFn: func(inputPolicy model.InputPolicy) targetLoader { + newTargetLoaderFn: func(builder model.ExperimentBuilder) targetLoader { return &mocks.ExperimentTargetLoader{ MockLoad: func(ctx context.Context) ([]model.ExperimentTarget, error) { return nil, errMocked @@ -223,7 +233,7 @@ func TestExperimentRun(t *testing.T) { } return eb, nil }, - newTargetLoaderFn: func(inputPolicy model.InputPolicy) targetLoader { + newTargetLoaderFn: func(builder model.ExperimentBuilder) targetLoader { return &mocks.ExperimentTargetLoader{ MockLoad: func(ctx context.Context) ([]model.ExperimentTarget, error) { return []model.ExperimentTarget{}, nil @@ -263,7 +273,7 @@ func TestExperimentRun(t *testing.T) { } return eb, nil }, - newTargetLoaderFn: func(inputPolicy model.InputPolicy) targetLoader { + newTargetLoaderFn: func(builder model.ExperimentBuilder) targetLoader { return &mocks.ExperimentTargetLoader{ MockLoad: func(ctx context.Context) ([]model.ExperimentTarget, error) { return []model.ExperimentTarget{}, nil @@ -306,7 +316,7 @@ func TestExperimentRun(t *testing.T) { } return eb, nil }, - newTargetLoaderFn: func(inputPolicy model.InputPolicy) targetLoader { + newTargetLoaderFn: func(builder model.ExperimentBuilder) targetLoader { return &mocks.ExperimentTargetLoader{ MockLoad: func(ctx context.Context) ([]model.ExperimentTarget, error) { return []model.ExperimentTarget{}, nil @@ -352,7 +362,7 @@ func TestExperimentRun(t *testing.T) { } return eb, nil }, - newTargetLoaderFn: func(inputPolicy model.InputPolicy) targetLoader { + newTargetLoaderFn: func(builder model.ExperimentBuilder) targetLoader { return &mocks.ExperimentTargetLoader{ MockLoad: func(ctx context.Context) ([]model.ExperimentTarget, error) { return []model.ExperimentTarget{}, nil diff --git a/internal/oonirun/v1_test.go b/internal/oonirun/v1_test.go index c23455a9f..910d910b0 100644 --- a/internal/oonirun/v1_test.go +++ b/internal/oonirun/v1_test.go @@ -44,6 +44,16 @@ func newMinimalFakeSession() *mocks.Session { } return exp }, + MockNewTargetLoader: func(config *model.ExperimentTargetLoaderConfig) model.ExperimentTargetLoader { + return &mocks.ExperimentTargetLoader{ + MockLoad: func(ctx context.Context) ([]model.ExperimentTarget, error) { + // Implementation note: the convention for input-less experiments is that + // they require a single entry containing an empty input. + entry := model.NewOOAPIURLInfoWithDefaultCategoryAndCountry("") + return []model.ExperimentTarget{entry}, nil + }, + } + }, } return eb, nil }, diff --git a/internal/oonirun/v2_test.go b/internal/oonirun/v2_test.go index c4afa60e9..d849c6d08 100644 --- a/internal/oonirun/v2_test.go +++ b/internal/oonirun/v2_test.go @@ -392,6 +392,16 @@ func TestV2MeasureDescriptor(t *testing.T) { } return exp }, + MockNewTargetLoader: func(config *model.ExperimentTargetLoaderConfig) model.ExperimentTargetLoader { + return &mocks.ExperimentTargetLoader{ + MockLoad: func(ctx context.Context) ([]model.ExperimentTarget, error) { + // Implementation note: the convention for input-less experiments is that + // they require a single entry containing an empty input. + entry := model.NewOOAPIURLInfoWithDefaultCategoryAndCountry("") + return []model.ExperimentTarget{entry}, nil + }, + } + }, } return eb, nil } diff --git a/internal/registry/dash.go b/internal/registry/dash.go index 812619361..e8a2ad116 100644 --- a/internal/registry/dash.go +++ b/internal/registry/dash.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["dash"] = func() *Factory { + const canonicalName = "dash" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return dash.NewExperimentMeasurer( *config.(*dash.Config), ) }, + canonicalName: canonicalName, config: &dash.Config{}, enabledByDefault: true, interruptible: true, diff --git a/internal/registry/dnscheck.go b/internal/registry/dnscheck.go index b9355773f..2890ca19a 100644 --- a/internal/registry/dnscheck.go +++ b/internal/registry/dnscheck.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["dnscheck"] = func() *Factory { + const canonicalName = "dnscheck" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return dnscheck.NewExperimentMeasurer( *config.(*dnscheck.Config), ) }, + canonicalName: canonicalName, config: &dnscheck.Config{}, enabledByDefault: true, inputPolicy: model.InputOrStaticDefault, diff --git a/internal/registry/dnsping.go b/internal/registry/dnsping.go index 92d52456e..fb01d1ab8 100644 --- a/internal/registry/dnsping.go +++ b/internal/registry/dnsping.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["dnsping"] = func() *Factory { + const canonicalName = "dnsping" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return dnsping.NewExperimentMeasurer( *config.(*dnsping.Config), ) }, + canonicalName: canonicalName, config: &dnsping.Config{}, enabledByDefault: true, inputPolicy: model.InputOrStaticDefault, diff --git a/internal/registry/dslxtutorial.go b/internal/registry/dslxtutorial.go index 41aef7d40..199e36077 100644 --- a/internal/registry/dslxtutorial.go +++ b/internal/registry/dslxtutorial.go @@ -10,15 +10,17 @@ import ( ) func init() { - AllExperiments["simple_sni"] = func() *Factory { + const canonicalName = "simple_sni" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return chapter02.NewExperimentMeasurer( *config.(*chapter02.Config), ) }, - config: &chapter02.Config{}, - inputPolicy: model.InputOrQueryBackend, + canonicalName: canonicalName, + config: &chapter02.Config{}, + inputPolicy: model.InputOrQueryBackend, } } } diff --git a/internal/registry/echcheck.go b/internal/registry/echcheck.go index b091d58af..939cf3d15 100644 --- a/internal/registry/echcheck.go +++ b/internal/registry/echcheck.go @@ -10,15 +10,17 @@ import ( ) func init() { - AllExperiments["echcheck"] = func() *Factory { + const canonicalName = "echcheck" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return echcheck.NewExperimentMeasurer( *config.(*echcheck.Config), ) }, - config: &echcheck.Config{}, - inputPolicy: model.InputOptional, + canonicalName: canonicalName, + config: &echcheck.Config{}, + inputPolicy: model.InputOptional, } } } diff --git a/internal/registry/example.go b/internal/registry/example.go index aa2d13ee1..94c55ca59 100644 --- a/internal/registry/example.go +++ b/internal/registry/example.go @@ -12,13 +12,15 @@ import ( ) func init() { - AllExperiments["example"] = func() *Factory { + const canonicalName = "example" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return example.NewExperimentMeasurer( *config.(*example.Config), "example", ) }, + canonicalName: canonicalName, config: &example.Config{ Message: "Good day from the example experiment!", SleepTime: int64(time.Second), diff --git a/internal/registry/factory.go b/internal/registry/factory.go index f0baa3480..0b37a79ee 100644 --- a/internal/registry/factory.go +++ b/internal/registry/factory.go @@ -13,8 +13,9 @@ import ( "strconv" "github.com/ooni/probe-cli/v3/internal/checkincache" + "github.com/ooni/probe-cli/v3/internal/experimentname" "github.com/ooni/probe-cli/v3/internal/model" - "github.com/ooni/probe-cli/v3/internal/strcasex" + "github.com/ooni/probe-cli/v3/internal/targetloading" ) // Factory allows to construct an experiment measurer. @@ -22,6 +23,9 @@ type Factory struct { // build is the constructor that build an experiment with the given config. build func(config interface{}) model.ExperimentMeasurer + // canonicalName is the canonical name of the experiment. + canonicalName string + // config contains the experiment's config. config any @@ -35,6 +39,22 @@ type Factory struct { interruptible bool } +// Session is the session definition according to this package. +type Session = model.ExperimentTargetLoaderSession + +// NewTargetLoader creates a new [model.ExperimentTargetLoader] instance. +func (b *Factory) NewTargetLoader(config *model.ExperimentTargetLoaderConfig) model.ExperimentTargetLoader { + return &targetloading.Loader{ + CheckInConfig: config.CheckInConfig, // OPTIONAL + ExperimentName: b.canonicalName, + InputPolicy: b.inputPolicy, + Logger: config.Session.Logger(), + Session: config.Session, + StaticInputs: config.StaticInputs, + SourceFiles: config.SourceFiles, + } +} + // Interruptible returns whether the experiment is interruptible. func (b *Factory) Interruptible() bool { return b.interruptible @@ -218,27 +238,6 @@ func (b *Factory) NewExperimentMeasurer() model.ExperimentMeasurer { return b.build(b.config) } -// CanonicalizeExperimentName allows code to provide experiment names -// in a more flexible way, where we have aliases. -// -// Because we allow for uppercase experiment names for backwards -// compatibility with MK, we need to add some exceptions here when -// mapping (e.g., DNSCheck => dnscheck). -func CanonicalizeExperimentName(name string) string { - switch name = strcasex.ToSnake(name); name { - case "ndt_7": - name = "ndt" // since 2020-03-18, we use ndt7 to implement ndt by default - case "dns_check": - name = "dnscheck" - case "stun_reachability": - name = "stunreachability" - case "web_connectivity@v_0_5": - name = "web_connectivity@v0.5" - default: - } - return name -} - // ErrNoSuchExperiment indicates a given experiment does not exist. var ErrNoSuchExperiment = errors.New("no such experiment") @@ -285,7 +284,7 @@ const OONI_FORCE_ENABLE_EXPERIMENT = "OONI_FORCE_ENABLE_EXPERIMENT" func NewFactory(name string, kvStore model.KeyValueStore, logger model.Logger) (*Factory, error) { // Make sure we are deadling with the canonical experiment name. Historically MK used // names such as WebConnectivity and we want to continue supporting this use case. - name = CanonicalizeExperimentName(name) + name = experimentname.Canonicalize(name) // Handle A/B testing where we dynamically choose LTE for some users. The current policy // only relates to a few users to collect data. diff --git a/internal/registry/factory_test.go b/internal/registry/factory_test.go index dd32ddafb..60105d135 100644 --- a/internal/registry/factory_test.go +++ b/internal/registry/factory_test.go @@ -1,16 +1,20 @@ package registry import ( + "context" "errors" "fmt" "math" "os" "testing" + "github.com/apex/log" "github.com/google/go-cmp/cmp" "github.com/ooni/probe-cli/v3/internal/checkincache" "github.com/ooni/probe-cli/v3/internal/experiment/webconnectivitylte" + "github.com/ooni/probe-cli/v3/internal/experimentname" "github.com/ooni/probe-cli/v3/internal/kvstore" + "github.com/ooni/probe-cli/v3/internal/mocks" "github.com/ooni/probe-cli/v3/internal/model" ) @@ -294,6 +298,15 @@ func TestExperimentBuilderSetOptionAny(t *testing.T) { FieldValue: 1.11, ExpectErr: ErrCannotSetIntegerOption, ExpectConfig: &fakeExperimentConfig{}, + }, { + TestCaseName: "[int] for float64 with zero fractional value", + InitialConfig: &fakeExperimentConfig{}, + FieldName: "Value", + FieldValue: float64(16.0), + ExpectErr: nil, + ExpectConfig: &fakeExperimentConfig{ + Value: 16, + }, }, { TestCaseName: "[string] for serialized bool value while setting a string value", InitialConfig: &fakeExperimentConfig{}, @@ -691,7 +704,7 @@ func TestNewFactory(t *testing.T) { // get experiment expectations -- note that here we must canonicalize the // experiment name otherwise we won't find it into the map when testing non-canonical names - expectations := expectationsMap[CanonicalizeExperimentName(tc.experimentName)] + expectations := expectationsMap[experimentname.Canonicalize(tc.experimentName)] if expectations == nil { t.Fatal("no expectations for", tc.experimentName) } @@ -801,3 +814,49 @@ func TestNewFactory(t *testing.T) { } }) } + +// Make sure the target loader for web connectivity is WAI when using no static inputs. +func TestFactoryNewTargetLoaderWebConnectivity(t *testing.T) { + // construct the proper factory instance + store := &kvstore.Memory{} + factory, err := NewFactory("web_connectivity", store, log.Log) + if err != nil { + t.Fatal(err) + } + + // define the expected error. + expected := errors.New("antani") + + // create suitable loader config. + config := &model.ExperimentTargetLoaderConfig{ + CheckInConfig: &model.OOAPICheckInConfig{ + // nothing + }, + Session: &mocks.Session{ + MockCheckIn: func(ctx context.Context, config *model.OOAPICheckInConfig) (*model.OOAPICheckInResult, error) { + return nil, expected + }, + MockLogger: func() model.Logger { + return log.Log + }, + }, + StaticInputs: nil, + SourceFiles: nil, + } + + // obtain the loader + loader := factory.NewTargetLoader(config) + + // attempt to load targets + targets, err := loader.Load(context.Background()) + + // make sure we've got the expected error + if !errors.Is(err, expected) { + t.Fatal("unexpected error", err) + } + + // make sure there are no targets + if len(targets) != 0 { + t.Fatal("expected zero length targets") + } +} diff --git a/internal/registry/fbmessenger.go b/internal/registry/fbmessenger.go index d3f3233e3..cd4651982 100644 --- a/internal/registry/fbmessenger.go +++ b/internal/registry/fbmessenger.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["facebook_messenger"] = func() *Factory { + const canonicalName = "facebook_messenger" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return fbmessenger.NewExperimentMeasurer( *config.(*fbmessenger.Config), ) }, + canonicalName: canonicalName, config: &fbmessenger.Config{}, enabledByDefault: true, inputPolicy: model.InputNone, diff --git a/internal/registry/hhfm.go b/internal/registry/hhfm.go index 0820a8647..a86ce9819 100644 --- a/internal/registry/hhfm.go +++ b/internal/registry/hhfm.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["http_header_field_manipulation"] = func() *Factory { + const canonicalName = "http_header_field_manipulation" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return hhfm.NewExperimentMeasurer( *config.(*hhfm.Config), ) }, + canonicalName: canonicalName, config: &hhfm.Config{}, enabledByDefault: true, inputPolicy: model.InputNone, diff --git a/internal/registry/hirl.go b/internal/registry/hirl.go index 846493bc8..84dbea7cc 100644 --- a/internal/registry/hirl.go +++ b/internal/registry/hirl.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["http_invalid_request_line"] = func() *Factory { + const canonicalName = "http_invalid_request_line" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return hirl.NewExperimentMeasurer( *config.(*hirl.Config), ) }, + canonicalName: canonicalName, config: &hirl.Config{}, enabledByDefault: true, inputPolicy: model.InputNone, diff --git a/internal/registry/httphostheader.go b/internal/registry/httphostheader.go index c1ecffd76..c3a02b655 100644 --- a/internal/registry/httphostheader.go +++ b/internal/registry/httphostheader.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["http_host_header"] = func() *Factory { + const canonicalName = "http_host_header" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return httphostheader.NewExperimentMeasurer( *config.(*httphostheader.Config), ) }, + canonicalName: canonicalName, config: &httphostheader.Config{}, enabledByDefault: true, inputPolicy: model.InputOrQueryBackend, diff --git a/internal/registry/ndt.go b/internal/registry/ndt.go index e6891c7c7..71ea5562a 100644 --- a/internal/registry/ndt.go +++ b/internal/registry/ndt.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["ndt"] = func() *Factory { + const canonicalName = "ndt" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return ndt7.NewExperimentMeasurer( *config.(*ndt7.Config), ) }, + canonicalName: canonicalName, config: &ndt7.Config{}, enabledByDefault: true, interruptible: true, diff --git a/internal/registry/openvpn.go b/internal/registry/openvpn.go index cf9244786..8bf107630 100644 --- a/internal/registry/openvpn.go +++ b/internal/registry/openvpn.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["openvpn"] = func() *Factory { + const canonicalName = "openvpn" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return openvpn.NewExperimentMeasurer( *config.(*openvpn.Config), "openvpn", ) }, + canonicalName: canonicalName, config: &openvpn.Config{}, enabledByDefault: true, interruptible: true, diff --git a/internal/registry/portfiltering.go b/internal/registry/portfiltering.go index 56937eaca..8c6a857df 100644 --- a/internal/registry/portfiltering.go +++ b/internal/registry/portfiltering.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["portfiltering"] = func() *Factory { + const canonicalName = "portfiltering" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config any) model.ExperimentMeasurer { return portfiltering.NewExperimentMeasurer( config.(portfiltering.Config), ) }, + canonicalName: canonicalName, config: portfiltering.Config{}, enabledByDefault: true, interruptible: false, diff --git a/internal/registry/psiphon.go b/internal/registry/psiphon.go index 48dc7888a..517bb2205 100644 --- a/internal/registry/psiphon.go +++ b/internal/registry/psiphon.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["psiphon"] = func() *Factory { + const canonicalName = "psiphon" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return psiphon.NewExperimentMeasurer( *config.(*psiphon.Config), ) }, + canonicalName: canonicalName, config: &psiphon.Config{}, enabledByDefault: true, inputPolicy: model.InputOptional, diff --git a/internal/registry/quicping.go b/internal/registry/quicping.go index 090037ad7..77a1d3ebf 100644 --- a/internal/registry/quicping.go +++ b/internal/registry/quicping.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["quicping"] = func() *Factory { + const canonicalName = "quicping" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return quicping.NewExperimentMeasurer( *config.(*quicping.Config), ) }, + canonicalName: canonicalName, config: &quicping.Config{}, enabledByDefault: true, inputPolicy: model.InputStrictlyRequired, diff --git a/internal/registry/riseupvpn.go b/internal/registry/riseupvpn.go index 950a86b79..6527f3a04 100644 --- a/internal/registry/riseupvpn.go +++ b/internal/registry/riseupvpn.go @@ -10,15 +10,17 @@ import ( ) func init() { - AllExperiments["riseupvpn"] = func() *Factory { + const canonicalName = "riseupvpn" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return riseupvpn.NewExperimentMeasurer( *config.(*riseupvpn.Config), ) }, - config: &riseupvpn.Config{}, - inputPolicy: model.InputNone, + canonicalName: canonicalName, + config: &riseupvpn.Config{}, + inputPolicy: model.InputNone, } } } diff --git a/internal/registry/signal.go b/internal/registry/signal.go index 615d44b73..5890936b3 100644 --- a/internal/registry/signal.go +++ b/internal/registry/signal.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["signal"] = func() *Factory { + const canonicalName = "signal" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return signal.NewExperimentMeasurer( *config.(*signal.Config), ) }, + canonicalName: canonicalName, config: &signal.Config{}, enabledByDefault: true, inputPolicy: model.InputNone, diff --git a/internal/registry/simplequicping.go b/internal/registry/simplequicping.go index 8eb3e7c54..c83b8cec8 100644 --- a/internal/registry/simplequicping.go +++ b/internal/registry/simplequicping.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["simplequicping"] = func() *Factory { + const canonicalName = "simplequicping" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return simplequicping.NewExperimentMeasurer( *config.(*simplequicping.Config), ) }, + canonicalName: canonicalName, config: &simplequicping.Config{}, enabledByDefault: true, inputPolicy: model.InputStrictlyRequired, diff --git a/internal/registry/sniblocking.go b/internal/registry/sniblocking.go index 8af8214d6..a433de3dc 100644 --- a/internal/registry/sniblocking.go +++ b/internal/registry/sniblocking.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["sni_blocking"] = func() *Factory { + const canonicalName = "sni_blocking" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return sniblocking.NewExperimentMeasurer( *config.(*sniblocking.Config), ) }, + canonicalName: canonicalName, config: &sniblocking.Config{}, enabledByDefault: true, inputPolicy: model.InputOrQueryBackend, diff --git a/internal/registry/stunreachability.go b/internal/registry/stunreachability.go index dfa331c3f..db35b5984 100644 --- a/internal/registry/stunreachability.go +++ b/internal/registry/stunreachability.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["stunreachability"] = func() *Factory { + const canonicalName = "stunreachability" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return stunreachability.NewExperimentMeasurer( *config.(*stunreachability.Config), ) }, + canonicalName: canonicalName, config: &stunreachability.Config{}, enabledByDefault: true, inputPolicy: model.InputOrStaticDefault, diff --git a/internal/registry/tcpping.go b/internal/registry/tcpping.go index d0ca93249..029e6e2d1 100644 --- a/internal/registry/tcpping.go +++ b/internal/registry/tcpping.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["tcpping"] = func() *Factory { + const canonicalName = "tcpping" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return tcpping.NewExperimentMeasurer( *config.(*tcpping.Config), ) }, + canonicalName: canonicalName, config: &tcpping.Config{}, enabledByDefault: true, inputPolicy: model.InputStrictlyRequired, diff --git a/internal/registry/telegram.go b/internal/registry/telegram.go index 8f61d78ba..987e64093 100644 --- a/internal/registry/telegram.go +++ b/internal/registry/telegram.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["telegram"] = func() *Factory { + const canonicalName = "telegram" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config any) model.ExperimentMeasurer { return telegram.NewExperimentMeasurer( config.(telegram.Config), ) }, + canonicalName: canonicalName, config: telegram.Config{}, enabledByDefault: true, interruptible: false, diff --git a/internal/registry/tlsmiddlebox.go b/internal/registry/tlsmiddlebox.go index a97de82c8..73e1ecfa3 100644 --- a/internal/registry/tlsmiddlebox.go +++ b/internal/registry/tlsmiddlebox.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["tlsmiddlebox"] = func() *Factory { + const canonicalName = "tlsmiddlebox" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return tlsmiddlebox.NewExperimentMeasurer( *config.(*tlsmiddlebox.Config), ) }, + canonicalName: canonicalName, config: &tlsmiddlebox.Config{}, enabledByDefault: true, inputPolicy: model.InputStrictlyRequired, diff --git a/internal/registry/tlsping.go b/internal/registry/tlsping.go index a40723d20..6ec5048d5 100644 --- a/internal/registry/tlsping.go +++ b/internal/registry/tlsping.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["tlsping"] = func() *Factory { + const canonicalName = "tlsping" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return tlsping.NewExperimentMeasurer( *config.(*tlsping.Config), ) }, + canonicalName: canonicalName, config: &tlsping.Config{}, enabledByDefault: true, inputPolicy: model.InputStrictlyRequired, diff --git a/internal/registry/tlstool.go b/internal/registry/tlstool.go index 8bb70983f..0fe2625a5 100644 --- a/internal/registry/tlstool.go +++ b/internal/registry/tlstool.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["tlstool"] = func() *Factory { + const canonicalName = "tlstool" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return tlstool.NewExperimentMeasurer( *config.(*tlstool.Config), ) }, + canonicalName: canonicalName, config: &tlstool.Config{}, enabledByDefault: true, inputPolicy: model.InputOrQueryBackend, diff --git a/internal/registry/tor.go b/internal/registry/tor.go index 73098bf77..2e2a61326 100644 --- a/internal/registry/tor.go +++ b/internal/registry/tor.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["tor"] = func() *Factory { + const canonicalName = "tor" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return tor.NewExperimentMeasurer( *config.(*tor.Config), ) }, + canonicalName: canonicalName, config: &tor.Config{}, enabledByDefault: true, inputPolicy: model.InputNone, diff --git a/internal/registry/torsf.go b/internal/registry/torsf.go index 178f01d5e..309090044 100644 --- a/internal/registry/torsf.go +++ b/internal/registry/torsf.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["torsf"] = func() *Factory { + const canonicalName = "torsf" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return torsf.NewExperimentMeasurer( *config.(*torsf.Config), ) }, + canonicalName: canonicalName, config: &torsf.Config{}, enabledByDefault: false, inputPolicy: model.InputNone, diff --git a/internal/registry/urlgetter.go b/internal/registry/urlgetter.go index 63dba20f9..ae3391bd6 100644 --- a/internal/registry/urlgetter.go +++ b/internal/registry/urlgetter.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["urlgetter"] = func() *Factory { + const canonicalName = "urlgetter" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return urlgetter.NewExperimentMeasurer( *config.(*urlgetter.Config), ) }, + canonicalName: canonicalName, config: &urlgetter.Config{}, enabledByDefault: true, inputPolicy: model.InputStrictlyRequired, diff --git a/internal/registry/vanillator.go b/internal/registry/vanillator.go index 1af548660..a1835e22c 100644 --- a/internal/registry/vanillator.go +++ b/internal/registry/vanillator.go @@ -10,14 +10,16 @@ import ( ) func init() { - AllExperiments["vanilla_tor"] = func() *Factory { + const canonicalName = "vanilla_tor" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return vanillator.NewExperimentMeasurer( *config.(*vanillator.Config), ) }, - config: &vanillator.Config{}, + canonicalName: canonicalName, + config: &vanillator.Config{}, // We discussed this topic with @aanorbel. On Android this experiment crashes // frequently because of https://github.com/ooni/probe/issues/2406. So, it seems // more cautious to disable it by default and let the check-in API decide. diff --git a/internal/registry/webconnectivity.go b/internal/registry/webconnectivity.go index 1ea720d58..470bad802 100644 --- a/internal/registry/webconnectivity.go +++ b/internal/registry/webconnectivity.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["web_connectivity"] = func() *Factory { + const canonicalName = "web_connectivity" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config any) model.ExperimentMeasurer { return webconnectivity.NewExperimentMeasurer( config.(webconnectivity.Config), ) }, + canonicalName: canonicalName, config: webconnectivity.Config{}, enabledByDefault: true, interruptible: false, diff --git a/internal/registry/webconnectivityv05.go b/internal/registry/webconnectivityv05.go index 0881b4448..3a56511a6 100644 --- a/internal/registry/webconnectivityv05.go +++ b/internal/registry/webconnectivityv05.go @@ -12,13 +12,15 @@ import ( ) func init() { - AllExperiments["web_connectivity@v0.5"] = func() *Factory { + const canonicalName = "web_connectivity@v0.5" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config any) model.ExperimentMeasurer { return webconnectivitylte.NewExperimentMeasurer( config.(*webconnectivitylte.Config), ) }, + canonicalName: canonicalName, config: &webconnectivitylte.Config{}, enabledByDefault: true, interruptible: false, diff --git a/internal/registry/whatsapp.go b/internal/registry/whatsapp.go index 84acb5789..24a27a2ac 100644 --- a/internal/registry/whatsapp.go +++ b/internal/registry/whatsapp.go @@ -10,13 +10,15 @@ import ( ) func init() { - AllExperiments["whatsapp"] = func() *Factory { + const canonicalName = "whatsapp" + AllExperiments[canonicalName] = func() *Factory { return &Factory{ build: func(config interface{}) model.ExperimentMeasurer { return whatsapp.NewExperimentMeasurer( *config.(*whatsapp.Config), ) }, + canonicalName: canonicalName, config: &whatsapp.Config{}, enabledByDefault: true, inputPolicy: model.InputNone, diff --git a/internal/targetloading/targetloading.go b/internal/targetloading/targetloading.go index 6590c2c4a..92e6a3b99 100644 --- a/internal/targetloading/targetloading.go +++ b/internal/targetloading/targetloading.go @@ -11,9 +11,9 @@ import ( "github.com/apex/log" "github.com/ooni/probe-cli/v3/internal/experiment/openvpn" + "github.com/ooni/probe-cli/v3/internal/experimentname" "github.com/ooni/probe-cli/v3/internal/fsx" "github.com/ooni/probe-cli/v3/internal/model" - "github.com/ooni/probe-cli/v3/internal/registry" "github.com/ooni/probe-cli/v3/internal/stuninput" ) @@ -26,12 +26,8 @@ var ( ErrNoStaticInput = errors.New("no static input for this experiment") ) -// Session is the session according to a [*Loader]. -type Session interface { - CheckIn(ctx context.Context, config *model.OOAPICheckInConfig) (*model.OOAPICheckInResult, error) - FetchOpenVPNConfig(ctx context.Context, - provider, cc string) (*model.OOAPIVPNProviderConfig, error) -} +// Session is the session according to a [*Loader] instance. +type Session = model.ExperimentTargetLoaderSession // Logger is the [model.Logger] according to a [*Loader]. type Logger interface { @@ -230,7 +226,7 @@ func StaticBareInputForExperiment(name string) ([]string, error) { // // TODO(https://github.com/ooni/probe/issues/2557): server STUNReachability // inputs using richer input (aka check-in v2). - switch registry.CanonicalizeExperimentName(name) { + switch experimentname.Canonicalize(name) { case "dnscheck": return dnsCheckDefaultInput, nil case "stunreachability": @@ -305,7 +301,7 @@ func (il *Loader) readfile(filepath string, open openFunc) ([]model.ExperimentTa // loadRemote loads inputs from a remote source. func (il *Loader) loadRemote(ctx context.Context) ([]model.ExperimentTarget, error) { - switch registry.CanonicalizeExperimentName(il.ExperimentName) { + switch experimentname.Canonicalize(il.ExperimentName) { case "openvpn": // TODO(ainghazal): given the semantics of the current API call, in an ideal world we'd need to pass // the desired provider here, if known. We only have one provider for now, so I'm acting like YAGNI. Another diff --git a/internal/targetloading/targetloading_test.go b/internal/targetloading/targetloading_test.go index 9710ce1b3..82684a820 100644 --- a/internal/targetloading/targetloading_test.go +++ b/internal/targetloading/targetloading_test.go @@ -11,6 +11,7 @@ import ( "testing" "time" + "github.com/apex/log" "github.com/google/go-cmp/cmp" "github.com/ooni/probe-cli/v3/internal/mocks" "github.com/ooni/probe-cli/v3/internal/model" @@ -535,7 +536,7 @@ type TargetLoaderMockableSession struct { Error error } -// CheckIn implements TargetLoaderSession.CheckIn. +// CheckIn implements [Session]. func (sess *TargetLoaderMockableSession) CheckIn( ctx context.Context, config *model.OOAPICheckInConfig) (*model.OOAPICheckInResult, error) { if sess.Output == nil && sess.Error == nil { @@ -544,13 +545,19 @@ func (sess *TargetLoaderMockableSession) CheckIn( return sess.Output, sess.Error } -// FetchOpenVPNConfig implements TargetLoaderSession.FetchOpenVPNConfig. +// FetchOpenVPNConfig implements [Session]. func (sess *TargetLoaderMockableSession) FetchOpenVPNConfig( ctx context.Context, provider, cc string) (*model.OOAPIVPNProviderConfig, error) { runtimex.Assert(!(sess.Error == nil && sess.FetchOpenVPNConfigOutput == nil), "both FetchOpenVPNConfig and Error are nil") return sess.FetchOpenVPNConfigOutput, sess.Error } +// Logger implements [Session]. +func (sess *TargetLoaderMockableSession) Logger() model.Logger { + // Such that we see some logs when running tests + return log.Log +} + func TestTargetLoaderCheckInFailure(t *testing.T) { il := &Loader{ Session: &TargetLoaderMockableSession{