Chapter I: main.go using the real torsf implementation
In this chapter we will write together a main.go
file that
uses the real torsf
implementation to run the experiment.
(This file is auto-generated from the corresponding source file,
so make sure you don't edit it manually.)
The torsf experiment
This experiment attempts to bootstrap the tor
binary using
Snowflake as the pluggable transport.
You can read the specification
of the torsf
experiment in the ooni/spec
repository. (The ooni/spec
repository is the repository
containing the specification of all OONI nettests, as well
as of the data formats used by OONI.)
The main.go file
We define main.go
file using package main
.
package main
Imports
Then we add the required imports.
import (
These are standard library imports.
"context"
"encoding/json"
"fmt"
"io/ioutil"
The apex/log library is the logging library used by OONI Probe.
"github.com/apex/log"
The torsf package contains the implementation of the torsf experiment.
"github.com/ooni/probe-cli/v3/internal/engine/experiment/torsf"
The mockable package contains widely used mocks.
"github.com/ooni/probe-cli/v3/internal/engine/mockable"
The model package contains the data model used by OONI experiments.
"github.com/ooni/probe-cli/v3/internal/model"
We will need the execabs library to check whether there is
a binary called tor
in the PATH
.
"golang.org/x/sys/execabs"
)
Main function
Finally, here's the code of the main function
.
func main() {
We start by checking whether there is an executable named "tor"
in
the PATH
. If there is no such executable, we fail with an error.
if _, err := execabs.LookPath("tor"); err != nil {
log.Fatal("cannot find the tor executable in path")
}
Then, we create a temporary directory to hold any state that may be
required either by the tor
or by the Snowflake pluggable transport.
tempdir, err := ioutil.TempDir("", "")
if err != nil {
log.WithError(err).Fatal("cannot create temporary directory")
}
Creating the experiment measurer
All OONI experiments implement a function called
NewExprimentMeasurer
that allows you to make
an ExperimentMeasurer
instance. The ExperimentMeasurer
is an interface
defined by the model
package we
imported above. Because we don't want to configure
any setting (and the experiment does not support any
setting anyway), here we're passing to the
NewExperimentMeasurer
factory an empty Config
.
m := torsf.NewExperimentMeasurer(torsf.Config{})
Creating the measurement
Next, we create an empty Measurement
. OONI measurements
are JSON data structures that contain generic fields common
to all OONI experiments and experiment-specific data. The
experiment-specific data is contained by a the test_keys
field of the Measurement
.
In the real OONI implementation, there is common code
that fills the several fields of a Measurement
. For
example, it will fill the country code and the autonomous
system number of the network in which the OONI Probe is
running. Because this is just an example to illustrate
how to write experiments, we will not bother with doing
that. Instead, we will pass to the experiment just an
emtpy measurement where no field has been set.
measurement := &model.Measurement{}
Creating the callbacks
Then, we create an instance of the experiment callbacks. The
experiment callbacks historically groups a set of callbacks
called when the measurer is running. At the moment of writing
this note, the model.ExperimentCallbacks
contains just a
single method called OnDataUsage
, which is used to tell the
caller which is the amount of data used by the experiment.
Because this is an example for illustrative purposes, here
we construct an implementation of ExperimentCallbacks
that
just prints the data usage using the log.Log
logger.
callbacks := model.NewPrinterCallbacks(log.Log)
Creating a session
The ExperimentMeasurer
also wants a Session
. In normal
OONI code, the Session
is a data structure containing
information regarding the current measurement session. Since
this is just an illustrative example, rather than creating
a real Session
instance, we use much-simpler mock.
The interface required by a Session
is called
ExperimentSession
and is part of the model
package.
Here we configure this mockable session to use log.Log
as a logger and the previously computed temp dir.
sess := &mockable.Session{
MockableLogger: log.Log,
MockableTempDir: tempdir,
}
Running the experiment
At last, it's time to run the experiment using all the
previously constructed data structures. The Run
function
is the main function you need to implement when you are
defining a new OONI experiment.
By convention, the Run
function only returns an error
when some precondition required by the experiment is
not met. Say that, for example, the experiment needs a
port listening on the local host. If we cannot create
such a port, we will return an error to the caller.
For network errors, instead, we return nil. Consider the
case where we connect to a remote host and the connection
fails. This is not really an error, rather it's a result
that we will include into the measurement.
Apart from the other arguments that we discussed previously,
the Run
function also wants a context.Context
as its
first argument. The context is used to interrupt long running
functions early, and our code (mostly) honours contexts.
Since here we are just writing a simple example, we don't
need any fancy context and we pass a context.Background
to Run
.
ctx := context.Background()
args := &model.ExperimentArgs{
Callbacks: callbacks,
Measurement: measurement,
Session: sess,
}
if err = m.Run(ctx, args); err != nil {
log.WithError(err).Fatal("torsf experiment failed")
}
Printing the measurement result
The Run
function modifies the TestKeys
(test_keys
in JSON)
field of the measurement. The real OONI implementation would
now submit this measurement. Because this is an illustrative example,
we will just pretty-print the measurement on the stdout
.
data, err := json.Marshal(measurement)
if err != nil {
log.WithError(err).Fatal("json.Marshal failed")
}
fmt.Printf("%s\n", data)
}
Running the code
You can now run this code as follows:
$ go run ./experiment/torsf/chapter01 | jq
[snip]
{
"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": {
"bootstrap_time": 68.909067459,
"failure": null
},
"test_name": "",
"test_runtime": 0,
"test_start_time": "",
"test_version": ""
}
We have snipped through logs and we have used jq
to
pretty print the measurement. You see that all the fields
except the test_keys
are empty.
Let us now analyze the content of the test_keys
:
-
the bootstrap_time
field contains the time (in seconds) to
bootstrap tor
using the Snowflake transport;
-
the failure
field contains the error that occurred, if
any, or null
if no error occurred.
This is all you need to know in terms of minimal code for
running an OONI experiment. In the remainder of this tutorial,
we will show how to reimplement the torsf
experiment.
Apart from minor changes, the main.go
file would basically
not change for the remainder of this tutorial.