2021-02-02 12:05:47 +01:00
|
|
|
package oonimkall_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/engine/model"
|
refactor: start building an Android package (#205)
* refactor: start building an Android package
Part of https://github.com/ooni/probe/issues/1335.
This seems also a good moment to move some packages out of the
engine, e.g., oonimkall. This package, for example, is a consumer
of the engine, so it makes sense it's not _inside_ it.
* fix: committed some stuff I didn't need to commit
* fix: oonimkall needs to be public to build
The side effect is that we will probably need to bump the major
version number every time we change one of these APIs.
(We can also of course choose to violate the basic guidelines of Go
software, but I believe this is bad form.)
I have no problem in bumping the major quite frequently and in
any case this monorepo solution is convinving me more than continuing
to keep a split between engine and cli. The need to embed assets to
make the probe more reliable trumps the negative effects of having to
~frequently bump major because we expose a public API.
* fix: let's not forget about libooniffi
Honestly, I don't know what to do with this library. I added it
to provide a drop in replacement for MK but I have no idea whether
it's used and useful. I would not feel comfortable exposing it,
unlike oonimkall, since we're not using it.
It may be that the right thing to do here is just to delete the
package and reduce the amount of code we're maintaining?
* woops, we're still missing the publish android script
* fix(publish-android.bash): add proper API key
* ouch fix another place where the name changed
2021-02-03 10:51:14 +01:00
|
|
|
"github.com/ooni/probe-cli/v3/pkg/oonimkall"
|
2021-02-03 19:48:23 +01:00
|
|
|
"github.com/ooni/probe-cli/v3/pkg/oonimkall/internal/tasks"
|
2021-02-02 12:05:47 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type eventlike struct {
|
|
|
|
Key string `json:"key"`
|
|
|
|
Value map[string]interface{} `json:"value"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGood(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"log_level": "DEBUG",
|
|
|
|
"name": "Example",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// interrupt the task so we also exercise this functionality
|
|
|
|
go func() {
|
|
|
|
<-time.After(time.Second)
|
|
|
|
task.Interrupt()
|
|
|
|
}()
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
t.Fatal("unexpected failure.startup event")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// make sure we only see task_terminated at this point
|
|
|
|
for {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key != "task_terminated" {
|
|
|
|
t.Fatalf("unexpected event.Key: %s", event.Key)
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWithMeasurementFailure(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"log_level": "DEBUG",
|
|
|
|
"name": "ExampleWithFailure",
|
|
|
|
"options": {
|
|
|
|
"no_geoip": true,
|
|
|
|
"no_resolver_lookup": true,
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
t.Fatal("unexpected failure.startup event")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInvalidJSON(t *testing.T) {
|
|
|
|
task, err := oonimkall.StartTask(`{`)
|
|
|
|
var syntaxerr *json.SyntaxError
|
|
|
|
if !errors.As(err, &syntaxerr) {
|
|
|
|
t.Fatal("not the expected error")
|
|
|
|
}
|
|
|
|
if task != nil {
|
|
|
|
t.Fatal("task is not nil")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUnsupportedSetting(t *testing.T) {
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"log_level": "DEBUG",
|
|
|
|
"name": "Example",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state"
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var seen bool
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
if strings.Contains(eventstr, tasks.FailureInvalidVersion) {
|
|
|
|
seen = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !seen {
|
|
|
|
t.Fatal("did not see failure.startup with invalid version info")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestEmptyStateDir(t *testing.T) {
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"log_level": "DEBUG",
|
|
|
|
"name": "Example",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var seen bool
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
if strings.Contains(eventstr, "mkdir : no such file or directory") {
|
|
|
|
seen = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !seen {
|
|
|
|
t.Fatal("did not see failure.startup with info that state dir is empty")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestEmptyAssetsDir(t *testing.T) {
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"log_level": "DEBUG",
|
|
|
|
"name": "Example",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var seen bool
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
if strings.Contains(eventstr, "AssetsDir is empty") {
|
|
|
|
seen = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !seen {
|
|
|
|
t.Fatal("did not see failure.startup")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUnknownExperiment(t *testing.T) {
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"log_level": "DEBUG",
|
|
|
|
"name": "Antani",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var seen bool
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
if strings.Contains(eventstr, "no such experiment: ") {
|
|
|
|
seen = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !seen {
|
|
|
|
t.Fatal("did not see failure.startup")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInputIsRequired(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"log_level": "DEBUG",
|
|
|
|
"name": "ExampleWithInput",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var seen bool
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
if strings.Contains(eventstr, "no input provided") {
|
|
|
|
seen = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !seen {
|
|
|
|
t.Fatal("did not see failure.startup")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMaxRuntime(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
begin := time.Now()
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"inputs": ["a", "b", "c"],
|
|
|
|
"name": "ExampleWithInput",
|
|
|
|
"options": {
|
|
|
|
"max_runtime": 1,
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
t.Fatal(eventstr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// The runtime is long because of ancillary operations and is even more
|
|
|
|
// longer because of self shaping we may be performing (especially in
|
|
|
|
// CI builds) using `-tags shaping`). We have experimentally determined
|
|
|
|
// that ~10 seconds is the typical CI test run time. See:
|
|
|
|
//
|
|
|
|
// 1. https://github.com/ooni/probe-cli/v3/internal/engine/pull/588/checks?check_run_id=667263788
|
|
|
|
//
|
|
|
|
// 2. https://github.com/ooni/probe-cli/v3/internal/engine/pull/588/checks?check_run_id=667263855
|
|
|
|
//
|
|
|
|
// In case there are further timeouts, e.g. in the sessionresolver, the
|
|
|
|
// time used by the experiment will be much more. This is for example the
|
|
|
|
// case in https://github.com/ooni/probe-cli/v3/internal/engine/issues/1005.
|
|
|
|
if time.Now().Sub(begin) > 10*time.Second {
|
|
|
|
t.Fatal("expected shorter runtime")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInterruptExampleWithInput(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
t.Skip("Skipping broken test; see https://github.com/ooni/probe-cli/v3/internal/engine/issues/992")
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"inputs": [
|
|
|
|
"http://www.kernel.org/",
|
|
|
|
"http://www.x.org/",
|
|
|
|
"http://www.microsoft.com/",
|
|
|
|
"http://www.slashdot.org/",
|
|
|
|
"http://www.repubblica.it/",
|
|
|
|
"http://www.google.it/",
|
|
|
|
"http://ooni.org/"
|
|
|
|
],
|
|
|
|
"name": "ExampleWithInputNonInterruptible",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var keys []string
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
switch event.Key {
|
|
|
|
case "failure.startup":
|
|
|
|
t.Fatal(eventstr)
|
|
|
|
case "status.measurement_start":
|
|
|
|
go task.Interrupt()
|
|
|
|
}
|
|
|
|
// We compress the keys. What matters is basically that we
|
|
|
|
// see just one of the many possible measurements here.
|
|
|
|
if keys == nil || keys[len(keys)-1] != event.Key {
|
|
|
|
keys = append(keys, event.Key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
expect := []string{
|
|
|
|
"status.queued",
|
|
|
|
"status.started",
|
|
|
|
"status.progress",
|
|
|
|
"status.geoip_lookup",
|
|
|
|
"status.resolver_lookup",
|
|
|
|
"status.progress",
|
|
|
|
"status.report_create",
|
|
|
|
"status.measurement_start",
|
|
|
|
"log",
|
|
|
|
"status.progress",
|
|
|
|
"measurement",
|
|
|
|
"status.measurement_submission",
|
|
|
|
"status.measurement_done",
|
|
|
|
"status.end",
|
|
|
|
"task_terminated",
|
|
|
|
}
|
|
|
|
if diff := cmp.Diff(expect, keys); diff != "" {
|
|
|
|
t.Fatal(diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInterruptNdt7(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"name": "Ndt7",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
<-time.After(11 * time.Second)
|
|
|
|
task.Interrupt()
|
|
|
|
}()
|
|
|
|
var keys []string
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if event.Key == "failure.startup" {
|
|
|
|
t.Fatal(eventstr)
|
|
|
|
}
|
|
|
|
// We compress the keys because we don't know how many
|
|
|
|
// status.progress we will see. What matters is that we
|
|
|
|
// don't see a measurement submission, since it means
|
|
|
|
// that we have interrupted the measurement.
|
|
|
|
if keys == nil || keys[len(keys)-1] != event.Key {
|
|
|
|
keys = append(keys, event.Key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
expect := []string{
|
|
|
|
"status.queued",
|
|
|
|
"status.started",
|
|
|
|
"status.progress",
|
|
|
|
"status.geoip_lookup",
|
|
|
|
"status.resolver_lookup",
|
|
|
|
"status.progress",
|
|
|
|
"status.report_create",
|
|
|
|
"status.measurement_start",
|
|
|
|
"status.progress",
|
|
|
|
"status.end",
|
|
|
|
"task_terminated",
|
|
|
|
}
|
|
|
|
if diff := cmp.Diff(expect, keys); diff != "" {
|
|
|
|
t.Fatal(diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCountBytesForExample(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"name": "Example",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var downloadKB, uploadKB float64
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
switch event.Key {
|
|
|
|
case "failure.startup":
|
|
|
|
t.Fatal(eventstr)
|
|
|
|
case "status.end":
|
|
|
|
downloadKB = event.Value["downloaded_kb"].(float64)
|
|
|
|
uploadKB = event.Value["uploaded_kb"].(float64)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if downloadKB == 0 {
|
|
|
|
t.Fatal("downloadKB is zero")
|
|
|
|
}
|
|
|
|
if uploadKB == 0 {
|
|
|
|
t.Fatal("uploadKB is zero")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPrivacyAndScrubbing(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"name": "Example",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var m *model.Measurement
|
|
|
|
for !task.IsDone() {
|
|
|
|
eventstr := task.WaitForNextEvent()
|
|
|
|
var event eventlike
|
|
|
|
if err := json.Unmarshal([]byte(eventstr), &event); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
switch event.Key {
|
|
|
|
case "failure.startup":
|
|
|
|
t.Fatal(eventstr)
|
|
|
|
case "measurement":
|
|
|
|
v := []byte(event.Value["json_str"].(string))
|
|
|
|
m = new(model.Measurement)
|
|
|
|
if err := json.Unmarshal(v, &m); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if m == nil {
|
|
|
|
t.Fatal("measurement is nil")
|
|
|
|
}
|
|
|
|
if m.ProbeASN == "AS0" || m.ProbeCC == "ZZ" || m.ProbeIP != "127.0.0.1" {
|
|
|
|
t.Fatal("unexpected result")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNonblock(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
task, err := oonimkall.StartTask(`{
|
|
|
|
"assets_dir": "../testdata/oonimkall/assets",
|
|
|
|
"name": "Example",
|
|
|
|
"options": {
|
|
|
|
"software_name": "oonimkall-test",
|
|
|
|
"software_version": "0.1.0"
|
|
|
|
},
|
|
|
|
"state_dir": "../testdata/oonimkall/state",
|
|
|
|
"version": 1
|
|
|
|
}`)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if !task.IsRunning() {
|
|
|
|
t.Fatal("The runner should be running at this point")
|
|
|
|
}
|
|
|
|
// If the task blocks because it emits too much events, this test
|
|
|
|
// will run forever and will be killed. Because we have room for up
|
|
|
|
// to 128 events in the buffer, we should hopefully be fine.
|
|
|
|
for task.IsRunning() {
|
|
|
|
time.Sleep(time.Second)
|
|
|
|
}
|
|
|
|
for !task.IsDone() {
|
|
|
|
task.WaitForNextEvent()
|
|
|
|
}
|
|
|
|
}
|