# Chapter II: creating an empty experiment In this chapter we will create an empty experiment and replace the code calling the real `torsf` experiment in `main.go` to call our empty experiment instead. (This file is auto-generated from the corresponding source file, so make sure you don't edit it manually.) ## Changes in main.go In `main.go` we will simply replace the call to the `torsf.NewExperimentMeasurer` function with a call to a `NewExperimentMeasurer` function that we are going to implement as part of this chapter. After you do this, you also need to remove the now-unneded import of the `torsf` package. There are no additional changes to `main.go`. ```Go m := NewExperimentMeasurer(Config{}) ``` ## The torsf.go file This file will contain the implementation of the `NewExperimentMeasurer` function. As usual we start with the `package` declaration and with the few imports we need to add. ```Go package main import ( "context" "time" "github.com/ooni/probe-cli/v3/internal/model" ) ``` ### Data structures Next, we define data structures. Config contains config for the torsf experiment. As for the real `torsf` experiment, we don't have any specific config, so we keep the structure empty. We still need to define a `Config` struct here, because, by convention, all OONI experiments have a `Config`. ```Go type Config struct{} ``` Measurer is the torsf measurer. This structure implements the `model.ExperimentMeasurer` interface, as we will see below. Most OONI experiments have a measurer that contains as its unique field the specific configuration. Here we do the same. ```Go type Measurer struct { config Config } ``` NewExperimentMeasurer creates a new model.ExperimentMeasurer instance for performing `torsf` measurements. This function will just assemble a new instance of `Measurer` with the `config` that was passed as an argument. ```Go func NewExperimentMeasurer(config Config) model.ExperimentMeasurer { return &Measurer{config: config} } ``` ### Implementing the model.ExperimentMeasurer. Now it's time to implement the methods required by the `model`'s `ExperimentMeasurer` interface. ExperimentName implements ExperimentMeasurer.ExperimentName. This function returns the name of the experiment. This code is used by generic code manipulating the experiment to print the experiment name. ```Go func (m *Measurer) ExperimentName() string { return "torsf" } ``` ExperimentVersion implements ExperimentMeasurer.ExperimentVersion. This function returns the version of the experiment. This code is also used by generic code manipulating the experiment to print the experiment version. ```Go func (m *Measurer) ExperimentVersion() string { return "0.1.0" } ``` Run implements ExperimentMeasurer.Run. This is the most interesting function, where we run the experiment proper. In the previous chapter we learned how to call this function from a `main.go` file. Here, instead, we're going to create a minimal stub. In the subsequent chapters, finally, we will modify this function until it is a minimal implementation of the `torsf` experiment. ```Go func (m *Measurer) Run( ctx context.Context, sess model.ExperimentSession, measurement *model.Measurement, callbacks model.ExperimentCallbacks, ) error { ``` As you can see, this is just a stub implementation that sleeps for one second and prints a logging message. ```Go time.Sleep(time.Second) sess.Logger().Info("hello from the torsf experiment!") return nil } ``` ### Summary keys Before concluding this chapter, we also need to create the `SummaryKeys` for this experiment. For historical reasons, the `TestKeys` of each experiment is an `interface{}`. Every experiment also defines a `SummaryKeys` data structure and a `GetSummaryKeys` method to convert the opaque result of a measurement to the summary for such an experiment. The experiment summary is *only* used by the OONI Probe CLI. SummaryKeys contains summary keys for this experiment. Because this is just an illustrative tutorial, we will just include a single key, named `IsAnomaly`. This key is not exported as JSON and is used by the OONI Probe CLI to inform the user of whether this measurement is ordinary or anomalous. All OONI experiments' `SummaryKeys` contain such a field. ```Go type SummaryKeys struct { IsAnomaly bool `json:"-"` } ``` GetSummaryKeys implements model.ExperimentMeasurer.GetSummaryKeys. This method just converts the `TestKeys` inside `measurement` to an instance of the `SummaryKeys` structure. For now, we'll just implement a stub returning fake `SummaryKeys` declaring there was no anomaly. ```Go func (m *Measurer) GetSummaryKeys(measurement *model.Measurement) (interface{}, error) { return &SummaryKeys{IsAnomaly: false}, nil } ``` ## Running the code We can run the code written in this chapter as follows: ``` $ go run ./experiment/torsf/chapter02 2021/06/21 20:48:32 info hello from the torsf experiment! { "data_format_version": "", "input": null, "measurement_start_time": "", "probe_asn": "", "probe_cc": "", "probe_network_name": "", "report_id": "", "resolver_asn": "", "resolver_ip": "", "resolver_network_name": "", "software_name": "", "software_version": "", "test_keys": null, "test_name": "", "test_runtime": 0, "test_start_time": "", "test_version": "" } ``` Here you see that we're printing the log message and that the `test_keys` are `null`. The OONI data processing popeline will not be so happy if we pass it a `null` settings, because there is not much interesting data in there. We will thus start filling it in the next chapter.