package model_test import ( "bytes" "encoding/json" "errors" "fmt" "testing" "github.com/ooni/probe-cli/v3/internal/engine/model" ) func TestMeasurementTargetMarshalJSON(t *testing.T) { var mt model.MeasurementTarget data, err := json.Marshal(mt) if err != nil { t.Fatal(err) } if string(data) != "null" { t.Fatal("unexpected serialization") } mt = "xx" data, err = json.Marshal(mt) if err != nil { t.Fatal(err) } if string(data) != `"xx"` { t.Fatal("unexpected serialization") } } type fakeTestKeys struct { ClientResolver string `json:"client_resolver"` Body string `json:"body"` } func TestAddAnnotations(t *testing.T) { m := &model.Measurement{} m.AddAnnotations(map[string]string{ "foo": "bar", "f": "b", }) m.AddAnnotations(map[string]string{ "foobar": "bar", "f": "b", }) if len(m.Annotations) != 3 { t.Fatal("unexpected number of annotations") } if m.Annotations["foo"] != "bar" { t.Fatal("unexpected annotation") } if m.Annotations["f"] != "b" { t.Fatal("unexpected annotation") } if m.Annotations["foobar"] != "bar" { t.Fatal("unexpected annotation") } } type makeMeasurementConfig struct { ProbeIP string ProbeASN string ProbeNetworkName string ProbeCC string ResolverIP string ResolverNetworkName string ResolverASN string } func makeMeasurement(config makeMeasurementConfig) model.Measurement { return model.Measurement{ DataFormatVersion: "0.3.0", ID: "bdd20d7a-bba5-40dd-a111-9863d7908572", MeasurementStartTime: "2018-11-01 15:33:20", ProbeIP: config.ProbeIP, ProbeASN: config.ProbeASN, ProbeNetworkName: config.ProbeNetworkName, ProbeCC: config.ProbeCC, ReportID: "", ResolverIP: config.ResolverIP, ResolverNetworkName: config.ResolverNetworkName, ResolverASN: config.ResolverASN, SoftwareName: "probe-engine", SoftwareVersion: "0.1.0", TestKeys: &fakeTestKeys{ ClientResolver: "91.80.37.104", Body: fmt.Sprintf(` Your IP is %s

Hey you, I see your IP and it's %s!

`, config.ProbeIP, config.ProbeIP), }, TestName: "dummy", MeasurementRuntime: 5.0565230846405, TestStartTime: "2018-11-01 15:33:17", TestVersion: "0.1.0", } } func TestScrubWeAreScrubbing(t *testing.T) { config := makeMeasurementConfig{ ProbeIP: "130.192.91.211", ProbeASN: "AS137", ProbeCC: "IT", ProbeNetworkName: "Vodafone Italia S.p.A.", ResolverIP: "8.8.8.8", ResolverNetworkName: "Google LLC", ResolverASN: "AS12345", } m := makeMeasurement(config) if err := m.Scrub(config.ProbeIP); err != nil { t.Fatal(err) } if m.ProbeASN != config.ProbeASN { t.Fatal("ProbeASN has been scrubbed") } if m.ProbeCC != config.ProbeCC { t.Fatal("ProbeCC has been scrubbed") } if m.ProbeIP == config.ProbeIP { t.Fatal("ProbeIP HAS NOT been scrubbed") } if m.ProbeNetworkName != config.ProbeNetworkName { t.Fatal("ProbeNetworkName has been scrubbed") } if m.ResolverIP != config.ResolverIP { t.Fatal("ResolverIP has been scrubbed") } if m.ResolverNetworkName != config.ResolverNetworkName { t.Fatal("ResolverNetworkName has been scrubbed") } if m.ResolverASN != config.ResolverASN { t.Fatal("ResolverASN has been scrubbed") } data, err := json.Marshal(m) if err != nil { t.Fatal(err) } if bytes.Count(data, []byte(config.ProbeIP)) != 0 { t.Fatal("ProbeIP not fully redacted") } } func TestScrubNoScrubbingRequired(t *testing.T) { config := makeMeasurementConfig{ ProbeIP: "130.192.91.211", ProbeASN: "AS137", ProbeCC: "IT", ProbeNetworkName: "Vodafone Italia S.p.A.", ResolverIP: "8.8.8.8", ResolverNetworkName: "Google LLC", ResolverASN: "AS12345", } m := makeMeasurement(config) m.TestKeys.(*fakeTestKeys).Body = "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum." if err := m.Scrub(config.ProbeIP); err != nil { t.Fatal(err) } if m.ProbeASN != config.ProbeASN { t.Fatal("ProbeASN has been scrubbed") } if m.ProbeCC != config.ProbeCC { t.Fatal("ProbeCC has been scrubbed") } if m.ProbeIP == config.ProbeIP { t.Fatal("ProbeIP HAS NOT been scrubbed") } if m.ProbeNetworkName != config.ProbeNetworkName { t.Fatal("ProbeNetworkName has been scrubbed") } if m.ResolverIP != config.ResolverIP { t.Fatal("ResolverIP has been scrubbed") } if m.ResolverNetworkName != config.ResolverNetworkName { t.Fatal("ResolverNetworkName has been scrubbed") } if m.ResolverASN != config.ResolverASN { t.Fatal("ResolverASN has been scrubbed") } data, err := json.Marshal(m) if err != nil { t.Fatal(err) } if bytes.Count(data, []byte(model.Scrubbed)) > 0 { t.Fatal("We should not see any scrubbing") } } func TestScrubInvalidIP(t *testing.T) { m := &model.Measurement{ ProbeASN: "AS1234", ProbeCC: "IT", } err := m.Scrub("") // invalid IP if !errors.Is(err, model.ErrInvalidProbeIP) { t.Fatal("not the error we expected") } } func TestScrubMarshalError(t *testing.T) { expected := errors.New("mocked error") m := &model.Measurement{ ProbeASN: "AS1234", ProbeCC: "IT", } err := m.MaybeRewriteTestKeys( "8.8.8.8", func(v interface{}) ([]byte, error) { return nil, expected }) if !errors.Is(err, expected) { t.Fatal("not the error we expected") } } func TestDiscardLoggerWorksAsIntended(t *testing.T) { logger := model.DiscardLogger logger.Debug("foo") logger.Debugf("%s", "foo") logger.Info("foo") logger.Infof("%s", "foo") logger.Warn("foo") logger.Warnf("%s", "foo") }