2021-02-02 12:05:47 +01:00
|
|
|
package tasks_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
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
|
|
|
)
|
|
|
|
|
|
|
|
func TestRunnerMaybeLookupBackendsFailure(t *testing.T) {
|
|
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
w.WriteHeader(500)
|
|
|
|
}))
|
|
|
|
defer server.Close()
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
Name: "Example",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
ProbeServicesBaseURL: server.URL,
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var failures []string
|
|
|
|
for ev := range out {
|
|
|
|
switch ev.Key {
|
|
|
|
case "failure.startup":
|
|
|
|
failure := ev.Value.(tasks.EventFailure).Failure
|
|
|
|
failures = append(failures, failure)
|
|
|
|
case "status.queued", "status.started", "log", "status.end":
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("unexpected key: %s", ev.Key))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(failures) != 1 {
|
|
|
|
t.Fatal("unexpected number of failures")
|
|
|
|
}
|
|
|
|
if failures[0] != "all available probe services failed" {
|
|
|
|
t.Fatal("invalid failure")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerOpenReportFailure(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
var (
|
|
|
|
nreq int64
|
|
|
|
mu sync.Mutex
|
|
|
|
)
|
|
|
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
mu.Lock()
|
|
|
|
defer mu.Unlock()
|
|
|
|
nreq++
|
|
|
|
if nreq == 1 {
|
|
|
|
w.Write([]byte(`{}`))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
w.WriteHeader(500)
|
|
|
|
}))
|
|
|
|
defer server.Close()
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
Name: "Example",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
ProbeServicesBaseURL: server.URL,
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
seench := make(chan int64)
|
|
|
|
go func() {
|
|
|
|
var seen int64
|
|
|
|
for ev := range out {
|
|
|
|
switch ev.Key {
|
|
|
|
case "failure.report_create":
|
|
|
|
seen++
|
|
|
|
case "status.progress":
|
|
|
|
evv := ev.Value.(tasks.EventStatusProgress)
|
|
|
|
if evv.Percentage >= 0.4 {
|
|
|
|
panic(fmt.Sprintf("too much progress: %+v", ev))
|
|
|
|
}
|
|
|
|
case "status.queued", "status.started", "log", "status.end",
|
|
|
|
"status.geoip_lookup", "status.resolver_lookup":
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("unexpected key: %s", ev.Key))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
seench <- seen
|
|
|
|
}()
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
if n := <-seench; n != 1 {
|
|
|
|
t.Fatal("unexpected number of events")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerGood(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
LogLevel: "DEBUG",
|
|
|
|
Name: "Example",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var found bool
|
|
|
|
for ev := range out {
|
|
|
|
if ev.Key == "status.end" {
|
|
|
|
found = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Fatal("status.end event not found")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerWithUnsupportedSettings(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
LogLevel: "DEBUG",
|
|
|
|
Name: "Example",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var failures []string
|
|
|
|
for ev := range out {
|
|
|
|
if ev.Key == "failure.startup" {
|
|
|
|
failure := ev.Value.(tasks.EventFailure).Failure
|
|
|
|
failures = append(failures, failure)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(failures) != 1 {
|
|
|
|
t.Fatal("invalid number of failures")
|
|
|
|
}
|
|
|
|
if failures[0] != tasks.FailureInvalidVersion {
|
|
|
|
t.Fatal("not the failure we expected")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerWithInvalidKVStorePath(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
LogLevel: "DEBUG",
|
|
|
|
Name: "Example",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "", // must be empty to cause the failure below
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var failures []string
|
|
|
|
for ev := range out {
|
|
|
|
if ev.Key == "failure.startup" {
|
|
|
|
failure := ev.Value.(tasks.EventFailure).Failure
|
|
|
|
failures = append(failures, failure)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(failures) != 1 {
|
|
|
|
t.Fatal("invalid number of failures")
|
|
|
|
}
|
|
|
|
if failures[0] != "mkdir : no such file or directory" {
|
|
|
|
t.Fatal("not the failure we expected")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerWithInvalidExperimentName(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
LogLevel: "DEBUG",
|
|
|
|
Name: "Nonexistent",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var failures []string
|
|
|
|
for ev := range out {
|
|
|
|
if ev.Key == "failure.startup" {
|
|
|
|
failure := ev.Value.(tasks.EventFailure).Failure
|
|
|
|
failures = append(failures, failure)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(failures) != 1 {
|
|
|
|
t.Fatal("invalid number of failures")
|
|
|
|
}
|
|
|
|
if failures[0] != "no such experiment: Nonexistent" {
|
|
|
|
t.Fatalf("not the failure we expected: %s", failures[0])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerWithMissingInput(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
LogLevel: "DEBUG",
|
|
|
|
Name: "ExampleWithInput",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var failures []string
|
|
|
|
for ev := range out {
|
|
|
|
if ev.Key == "failure.startup" {
|
|
|
|
failure := ev.Value.(tasks.EventFailure).Failure
|
|
|
|
failures = append(failures, failure)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(failures) != 1 {
|
|
|
|
t.Fatal("invalid number of failures")
|
|
|
|
}
|
|
|
|
if failures[0] != "no input provided" {
|
|
|
|
t.Fatalf("not the failure we expected: %s", failures[0])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerWithMaxRuntime(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
Inputs: []string{"a", "b", "c", "d", "e", "f", "g", "h", "i"},
|
|
|
|
LogLevel: "DEBUG",
|
|
|
|
Name: "ExampleWithInput",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
MaxRuntime: 1,
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
begin := time.Now()
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var found bool
|
|
|
|
for ev := range out {
|
|
|
|
if ev.Key == "status.end" {
|
|
|
|
found = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Fatal("status.end event not found")
|
|
|
|
}
|
|
|
|
// 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
|
|
|
|
if time.Now().Sub(begin) > 10*time.Second {
|
|
|
|
t.Fatal("expected shorter runtime")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerWithMaxRuntimeNonInterruptible(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
Inputs: []string{"a", "b", "c", "d", "e", "f", "g", "h", "i"},
|
|
|
|
LogLevel: "DEBUG",
|
|
|
|
Name: "ExampleWithInputNonInterruptible",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
MaxRuntime: 1,
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
begin := time.Now()
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var found bool
|
|
|
|
for ev := range out {
|
|
|
|
if ev.Key == "status.end" {
|
|
|
|
found = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Fatal("status.end event not found")
|
|
|
|
}
|
|
|
|
// 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
|
|
|
|
if time.Now().Sub(begin) > 10*time.Second {
|
|
|
|
t.Fatal("expected shorter runtime")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunnerWithFailedMeasurement(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
out := make(chan *tasks.Event)
|
|
|
|
settings := &tasks.Settings{
|
|
|
|
AssetsDir: "../../testdata/oonimkall/assets",
|
|
|
|
Inputs: []string{"a", "b", "c", "d", "e", "f", "g", "h", "i"},
|
|
|
|
LogLevel: "DEBUG",
|
|
|
|
Name: "ExampleWithFailure",
|
|
|
|
Options: tasks.SettingsOptions{
|
|
|
|
MaxRuntime: 1,
|
|
|
|
SoftwareName: "oonimkall-test",
|
|
|
|
SoftwareVersion: "0.1.0",
|
|
|
|
},
|
|
|
|
StateDir: "../../testdata/oonimkall/state",
|
|
|
|
Version: 1,
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
tasks.Run(context.Background(), settings, out)
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
var found bool
|
|
|
|
for ev := range out {
|
|
|
|
if ev.Key == "failure.measurement" {
|
|
|
|
found = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Fatal("failure.measurement event not found")
|
|
|
|
}
|
|
|
|
}
|