diff --git a/internal/testingx/oonibackendwithlogin.go b/internal/testingx/oonibackendwithlogin.go index 169e80c68..5c8779c9a 100644 --- a/internal/testingx/oonibackendwithlogin.go +++ b/internal/testingx/oonibackendwithlogin.go @@ -25,8 +25,8 @@ type OONIBackendWithLoginFlowUserRecord struct { Token string } -// OONIBackendWithLoginFlow is an [http.Handler] that implements the register and -// loging workflow and serves psiphon and tor config. +// OONIBackendWithLoginFlow implements the register and login workflows +// and serves the psiphon config and tor targets. // // The zero value is ready to use. // diff --git a/internal/testingx/oonicollector.go b/internal/testingx/oonicollector.go new file mode 100644 index 000000000..62c6b2cc3 --- /dev/null +++ b/internal/testingx/oonicollector.go @@ -0,0 +1,245 @@ +package testingx + +import ( + "encoding/json" + "io" + "log" + "net/http" + "strings" + "sync" + + "github.com/google/go-cmp/cmp" + "github.com/google/uuid" + "github.com/ooni/probe-cli/v3/internal/model" + "github.com/ooni/probe-cli/v3/internal/must" + "github.com/ooni/probe-cli/v3/internal/runtimex" +) + +// OONICollector implements the OONI collector for testing. +// +// The zero value is ready to use. +// +// This struct methods panics for several errors. Only use for testing purposes! +type OONICollector struct { + // EditOpenReportResponse is an OPTIONAL callback to edit the response + // before the server actually sends it to the client. + EditOpenReportResponse func(resp *model.OOAPICollectorOpenResponse) + + // EditUpdateResponse is an OPTIONAL callback to edit the response + // before the server actually sends it to the client. + EditUpdateResponse func(resp *model.OOAPICollectorUpdateResponse) + + // ValidateMeasurement is an OPTIONAL callback to validate the incoming measurement + // beyond checks that ensure it is consistent with the original template. + ValidateMeasurement func(meas *model.Measurement) error + + // ValidateReportTemplate is an OPTIONAL callback to validate the incoming report + // template beyond the data format version and format fields values. + ValidateReportTemplate func(rt *model.OOAPIReportTemplate) error + + // mu provides mutual exclusion. + mu sync.Mutex + + // reports contains the open reports. + reports map[string]*model.OOAPIReportTemplate +} + +// OpenReport opens a report for the given report ID and template. +// +// This method is safe to call concurrently with other methods. +func (oc *OONICollector) OpenReport(reportID string, template *model.OOAPIReportTemplate) { + oc.mu.Lock() + if oc.reports == nil { + oc.reports = make(map[string]*model.OOAPIReportTemplate) + } + oc.reports[reportID] = template + oc.mu.Unlock() +} + +// ServeHTTP implements [http.Handler]. +// +// This method is safe to call concurrently with other methods. +func (oc *OONICollector) ServeHTTP(w http.ResponseWriter, r *http.Request) { + // make sure that the method is POST + if r.Method != "POST" { + log.Printf("OONICollector: invalid method") + w.WriteHeader(http.StatusNotImplemented) + return + } + + // make sure the URL path starts with /report + if !strings.HasPrefix(r.URL.Path, "/report") { + log.Printf("OONICollector: invalid URL path prefix") + w.WriteHeader(http.StatusBadRequest) + return + } + + // make sure that the content-type is application/json + if r.Header.Get("Content-Type") != "application/json" { + log.Printf("OONICollector: missing content-type header") + w.WriteHeader(http.StatusBadRequest) + return + } + + // read the raw request body or panic if we cannot read it + body := runtimex.Try1(io.ReadAll(r.Body)) + + log.Printf("OONICollector: URLPath %+v", r.URL.Path) + log.Printf("OONICollector: request body %s", string(body)) + + // handle the case where the user wants to open a new report + if r.URL.Path == "/report" { + log.Printf("OONICollector: opening new report") + oc.openReport(w, body) + return + } + + // handle the case where the user wants to append to an existing report + log.Printf("OONICollector: updating existing report") + oc.updateReport(w, r.URL.Path, body) +} + +// openReport handles opening a new OONI report. +func (oc *OONICollector) openReport(w http.ResponseWriter, body []byte) { + // make sure we can parse the incoming request + var template model.OOAPIReportTemplate + if err := json.Unmarshal(body, &template); err != nil { + log.Printf("OONICollector: cannot unmarshal JSON: %s", err.Error()) + w.WriteHeader(http.StatusBadRequest) + return + } + + // make sure the data format version is OK + if template.DataFormatVersion != model.OOAPIReportDefaultDataFormatVersion { + log.Printf("OONICollector: invalid data format version") + w.WriteHeader(http.StatusBadRequest) + return + } + + // make sure the format is also OK + if template.Format != model.OOAPIReportDefaultFormat { + log.Printf("OONICollector: invalid format") + w.WriteHeader(http.StatusBadRequest) + return + } + + // optionally allow the user to validate the report template + if oc.ValidateReportTemplate != nil { + if err := oc.ValidateReportTemplate(&template); err != nil { + log.Printf("OONICollector: invalid report template: %s", err.Error()) + w.WriteHeader(http.StatusBadRequest) + return + } + } + + // create the response + response := &model.OOAPICollectorOpenResponse{ + BackendVersion: "1.3.0", + ReportID: uuid.Must(uuid.NewRandom()).String(), + SupportedFormats: []string{ + model.OOAPIReportDefaultFormat, + }, + } + + // optionally allow the user to modify the response + if oc.EditOpenReportResponse != nil { + oc.EditOpenReportResponse(response) + } + + // make sure we know that this report ID now exists - note that this must + // happen after the client code has edited the response + oc.OpenReport(response.ReportID, &template) + + // set the content-type header + w.Header().Set("Content-Type", "application/json") + + // serialize and send + w.Write(must.MarshalJSON(response)) +} + +// updateReport handles updating an existing OONI report. +func (oc *OONICollector) updateReport(w http.ResponseWriter, urlpath string, body []byte) { + // get the report ID + reportID := strings.TrimPrefix(urlpath, "/report/") + + // obtain the report template + oc.mu.Lock() + template := oc.reports[reportID] + oc.mu.Unlock() + + // handle the case of missing template + if template == nil { + log.Printf("OONICollector: the report does not exist: %s", reportID) + w.WriteHeader(http.StatusBadRequest) + return + } + + // make sure we can parse the incoming request + var request model.OOAPICollectorUpdateRequest + if err := json.Unmarshal(body, &request); err != nil { + log.Printf("OONICollector: cannot unmarshal JSON: %s", err.Error()) + w.WriteHeader(http.StatusBadRequest) + return + } + + // make sure the measurement is encoded as JSON + if request.Format != "json" { + log.Printf("OONICollector: invalid request format: %s", request.Format) + w.WriteHeader(http.StatusBadRequest) + return + } + + // make sure we can parse the content + // + // note: we unmarshaled into a map[string]any so we need to marshal + // and unmarshal again to get a measurement structure + var measurement model.Measurement + if err := json.Unmarshal(must.MarshalJSON(request.Content), &measurement); err != nil { + log.Printf("OONICollector: cannot unmarshal JSON: %s", err.Error()) + w.WriteHeader(http.StatusBadRequest) + return + } + + // make sure all the required fields match + mt := &model.OOAPIReportTemplate{ + DataFormatVersion: measurement.DataFormatVersion, + Format: request.Format, + ProbeASN: measurement.ProbeASN, + ProbeCC: measurement.ProbeCC, + SoftwareName: measurement.SoftwareName, + SoftwareVersion: measurement.SoftwareVersion, + TestName: measurement.TestName, + TestStartTime: measurement.TestStartTime, + TestVersion: measurement.TestVersion, + } + if diff := cmp.Diff(template, mt); diff != "" { + log.Printf("OONICollector: measurement differs from template %s", diff) + w.WriteHeader(http.StatusBadRequest) + return + } + + // give the user a chance to validate the measurement + if oc.ValidateMeasurement != nil { + if err := oc.ValidateMeasurement(&measurement); err != nil { + log.Printf("OONICollector: invalid measurement: %s", err.Error()) + w.WriteHeader(http.StatusBadRequest) + return + } + } + + // create the response + response := &model.OOAPICollectorUpdateResponse{ + MeasurementUID: uuid.Must(uuid.NewRandom()).String(), + } + + // optionally allow the user to modify the response + if oc.EditUpdateResponse != nil { + oc.EditUpdateResponse(response) + } + + // set the content-type header + w.Header().Set("Content-Type", "application/json") + + // serialize and send + w.Write(must.MarshalJSON(response)) +} diff --git a/internal/testingx/oonicollector_test.go b/internal/testingx/oonicollector_test.go new file mode 100644 index 000000000..70f39edc1 --- /dev/null +++ b/internal/testingx/oonicollector_test.go @@ -0,0 +1,891 @@ +package testingx + +import ( + "bytes" + "errors" + "io" + "net/http" + "net/url" + "slices" + "sync" + "testing" + + "github.com/google/go-cmp/cmp" + "github.com/ooni/probe-cli/v3/internal/model" + "github.com/ooni/probe-cli/v3/internal/must" + "github.com/ooni/probe-cli/v3/internal/runtimex" +) + +// This function tests the OONICollector type. +func TestOONICollector(t *testing.T) { + t.Run("common: when method is not POST", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // create request + req := runtimex.Try1(http.NewRequest("GET", srv.URL, nil)) + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 501 + if resp.StatusCode != http.StatusNotImplemented { + t.Fatal("unexpected status code") + } + }) + + t.Run("common: when the URL path does not start with report", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // create request + // + // note: the server URL has / as its path so this URL is good to go + req := runtimex.Try1(http.NewRequest("POST", srv.URL, nil)) + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("common: when the Content-Type header is missing", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // rewrite the URL to have a path starting with /report + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report" + + // create request + // + // note: the request has no Content-Type so we should be good here + req := runtimex.Try1(http.NewRequest("POST", URL.String(), nil)) + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("openReport: when we cannot unmarshal the body", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // rewrite the URL to be exactly /report + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report" + + // create request + // + // note: the body is empty so parsing should fail + req := runtimex.Try1(http.NewRequest("POST", URL.String(), nil)) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("openReport: with invalid data format version", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // rewrite the URL to be exactly /report + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report" + + // create the request body + // + // note: we're using an invalid data format version to trigger error + const invalidDataFormatVersion = "0.3.0" + request := &model.OOAPIReportTemplate{ + DataFormatVersion: invalidDataFormatVersion, + } + rawreqbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawreqbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("openReport: with invalid format", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // rewrite the URL to be exactly /report + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report" + + // create the request body + // + // note: we're using an invalid format to trigger error + const validDataFormatVersion = "0.2.0" + request := &model.OOAPIReportTemplate{ + DataFormatVersion: validDataFormatVersion, + Format: "yaml", + } + rawreqbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawreqbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("openReport: we can invoke a callback to see the incoming template", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // rewrite the URL to be exactly /report + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report" + + // create the request body + // + // note: we're using the fake filler to randomly fill and then we're + // editing to avoid failures, but this means we have a request that we + // can later compare to using google/cmp-go/cmp.Diff. + const validDataFormatVersion = "0.2.0" + request := &model.OOAPIReportTemplate{} + ff := &FakeFiller{} + ff.Fill(&request) + request.DataFormatVersion = validDataFormatVersion + request.Format = "json" + rawreqbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawreqbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // prepare to obtain the incoming report template + mu := &sync.Mutex{} + var incoming *model.OOAPIReportTemplate + + // set the callback to verify the request + // + // we save the original request and return an error to trigger failure + collector.ValidateReportTemplate = func(rt *model.OOAPIReportTemplate) error { + mu.Lock() + incoming = rt + mu.Unlock() + return errors.New("mocked error") + } + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + + // make sure we got what we sent + if diff := cmp.Diff(request, incoming); diff != "" { + t.Fatal(diff) + } + }) + + t.Run("openReport: we can edit the outgoing response", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // rewrite the URL to be exactly /report + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report" + + // create the request body + const validDataFormatVersion = "0.2.0" + request := &model.OOAPIReportTemplate{} + ff := &FakeFiller{} + ff.Fill(&request) + request.DataFormatVersion = validDataFormatVersion + request.Format = "json" + rawreqbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawreqbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // make sure we can edit the response + collector.EditOpenReportResponse = func(resp *model.OOAPICollectorOpenResponse) { + resp.BackendVersion = "antani-antani-antani" + } + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 200 + if resp.StatusCode != http.StatusOK { + t.Fatal("unexpected status code") + } + + // read response body + rawrespbody := runtimex.Try1(io.ReadAll(resp.Body)) + var response model.OOAPICollectorOpenResponse + must.UnmarshalJSON(rawrespbody, &response) + + // make sure we've got the edited BackendVersion + if response.BackendVersion != "antani-antani-antani" { + t.Fatal("did not edit the response") + } + }) + + t.Run("openReport: we get a reportID back and format=json", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // rewrite the URL to be exactly /report + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report" + + // create the request body + const validDataFormatVersion = "0.2.0" + request := &model.OOAPIReportTemplate{} + ff := &FakeFiller{} + ff.Fill(&request) + request.DataFormatVersion = validDataFormatVersion + request.Format = "json" + rawreqbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawreqbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 200 + if resp.StatusCode != http.StatusOK { + t.Fatal("unexpected status code") + } + + // read response body + rawrespbody := runtimex.Try1(io.ReadAll(resp.Body)) + var response model.OOAPICollectorOpenResponse + must.UnmarshalJSON(rawrespbody, &response) + + // make sure the fields are okay + if response.BackendVersion != "1.3.0" { + t.Fatal("unexpected backend version") + } + if response.ReportID == "" { + t.Fatal("empty report ID") + } + if !slices.Contains(response.SupportedFormats, "json") { + t.Fatal("SupportedFormats does not contain the json format") + } + }) + + // This is a convenience function to open a report before submitting + openreport := func(t *testing.T, stringURL string) (*model.OOAPIReportTemplate, *model.OOAPICollectorOpenResponse) { + // rewrite the URL to be exactly /report + URL := runtimex.Try1(url.Parse(stringURL)) + URL.Path = "/report" + + // create the request body + const validDataFormatVersion = "0.2.0" + request := &model.OOAPIReportTemplate{} + ff := &FakeFiller{} + ff.Fill(&request) + request.DataFormatVersion = validDataFormatVersion + request.Format = "json" + rawreqbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawreqbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 200 + if resp.StatusCode != http.StatusOK { + t.Fatal("unexpected status code") + } + + // read response body + rawrespbody := runtimex.Try1(io.ReadAll(resp.Body)) + var response model.OOAPICollectorOpenResponse + must.UnmarshalJSON(rawrespbody, &response) + + return request, &response + } + + t.Run("submit: when the report ID does not exist", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // rewrite the URL to be exactly /report/${reportID} + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report/" + "blah" + + // create request + // + // note: the body is empty so parsing should fail + req := runtimex.Try1(http.NewRequest("POST", URL.String(), nil)) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("submit: when we cannot unmarshal the body", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // first of all let's open a report + _, reportInfo := openreport(t, srv.URL) + + // rewrite the URL to be exactly /report/${reportID} + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report/" + reportInfo.ReportID + + // create request + // + // note: the body is empty so parsing should fail + req := runtimex.Try1(http.NewRequest("POST", URL.String(), nil)) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("submit: with invalid format", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // first of all let's open a report + _, reportInfo := openreport(t, srv.URL) + + // rewrite the URL to be exactly /report/${reportID} + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report/" + reportInfo.ReportID + + // create the request body + // + // note: the YAML format here is invalid + request := &model.OOAPICollectorUpdateRequest{ + Format: "yaml", + Content: nil, + } + rawrequestbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawrequestbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("submit: when we cannot unmarshal the nested inside body", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // first of all let's open a report + _, reportInfo := openreport(t, srv.URL) + + // rewrite the URL to be exactly /report/${reportID} + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report/" + reportInfo.ReportID + + // create the request body + // + // note: the content is empty so we cannot parse a JSON + request := &model.OOAPICollectorUpdateRequest{ + Format: "json", + Content: "", + } + rawrequestbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawrequestbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("submit: when the template does not match the expectations", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // first of all let's open a report + template, reportInfo := openreport(t, srv.URL) + + // rewrite the URL to be exactly /report/${reportID} + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report/" + reportInfo.ReportID + + // create the measurement + // + // note: we're changing the test name here + measurement := &model.Measurement{ + DataFormatVersion: template.DataFormatVersion, + InputHashes: []string{}, + MeasurementStartTime: template.TestStartTime, + ProbeASN: template.ProbeASN, + ProbeCC: template.ProbeCC, + ReportID: reportInfo.ReportID, + TestKeys: nil, + TestName: template.TestName + "blahblah", + TestStartTime: template.TestStartTime, + TestVersion: template.TestVersion, + } + + // create the request body + request := &model.OOAPICollectorUpdateRequest{ + Format: "json", + Content: measurement, + } + rawrequestbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawrequestbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + }) + + t.Run("submit: we can invoke a callback to further validate the measurement", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // first of all let's open a report + template, reportInfo := openreport(t, srv.URL) + + // rewrite the URL to be exactly /report/${reportID} + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report/" + reportInfo.ReportID + + // create the measurement + measurement := &model.Measurement{ + DataFormatVersion: template.DataFormatVersion, + MeasurementStartTime: template.TestStartTime, + ProbeASN: template.ProbeASN, + ProbeCC: template.ProbeCC, + ReportID: reportInfo.ReportID, + SoftwareName: template.SoftwareName, + SoftwareVersion: template.SoftwareVersion, + TestKeys: nil, + TestName: template.TestName, + TestStartTime: template.TestStartTime, + TestVersion: template.TestVersion, + } + + // create the request body + request := &model.OOAPICollectorUpdateRequest{ + Format: "json", + Content: measurement, + } + rawrequestbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawrequestbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // prepare to obtain the incoming measurement + mu := &sync.Mutex{} + var incoming *model.Measurement + + // setup a callback so we can get the incoming measurement + // + // note: here we return an error to make the API fail but we save the measurement for later + collector.ValidateMeasurement = func(meas *model.Measurement) error { + mu.Lock() + incoming = meas + mu.Unlock() + return errors.New("mocked error") + } + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 400 + if resp.StatusCode != http.StatusBadRequest { + t.Fatal("unexpected status code") + } + + // make sure the measurement received by the API is the expected one + if diff := cmp.Diff(measurement, incoming); diff != "" { + t.Fatal(diff) + } + }) + + t.Run("submit: we can edit the outgoing response", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // first of all let's open a report + template, reportInfo := openreport(t, srv.URL) + + // rewrite the URL to be exactly /report/${reportID} + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report/" + reportInfo.ReportID + + // create the measurement + measurement := &model.Measurement{ + DataFormatVersion: template.DataFormatVersion, + MeasurementStartTime: template.TestStartTime, + ProbeASN: template.ProbeASN, + ProbeCC: template.ProbeCC, + ReportID: reportInfo.ReportID, + SoftwareName: template.SoftwareName, + SoftwareVersion: template.SoftwareVersion, + TestKeys: nil, + TestName: template.TestName, + TestStartTime: template.TestStartTime, + TestVersion: template.TestVersion, + } + + // create the request body + request := &model.OOAPICollectorUpdateRequest{ + Format: "json", + Content: measurement, + } + rawrequestbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawrequestbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // make sure we can edit the response + collector.EditUpdateResponse = func(resp *model.OOAPICollectorUpdateResponse) { + resp.MeasurementUID = "blablah" + } + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 200 + if resp.StatusCode != http.StatusOK { + t.Fatal("unexpected status code") + } + + // read and parse response body + rawrespbody := runtimex.Try1(io.ReadAll(resp.Body)) + var response model.OOAPICollectorUpdateResponse + must.UnmarshalJSON(rawrespbody, &response) + + // make sure the measurement UID has been edited + if response.MeasurementUID != "blablah" { + t.Fatal("did not exit the measurement UID") + } + }) + + t.Run("submit: we get a measurement ID back", func(t *testing.T) { + // create and expose the testing collector + collector := &OONICollector{} + srv := MustNewHTTPServer(collector) + defer srv.Close() + + // first of all let's open a report + template, reportInfo := openreport(t, srv.URL) + + // rewrite the URL to be exactly /report/${reportID} + URL := runtimex.Try1(url.Parse(srv.URL)) + URL.Path = "/report/" + reportInfo.ReportID + + // create the measurement + measurement := &model.Measurement{ + DataFormatVersion: template.DataFormatVersion, + MeasurementStartTime: template.TestStartTime, + ProbeASN: template.ProbeASN, + ProbeCC: template.ProbeCC, + ReportID: reportInfo.ReportID, + SoftwareName: template.SoftwareName, + SoftwareVersion: template.SoftwareVersion, + TestKeys: nil, + TestName: template.TestName, + TestStartTime: template.TestStartTime, + TestVersion: template.TestVersion, + } + + // create the request body + request := &model.OOAPICollectorUpdateRequest{ + Format: "json", + Content: measurement, + } + rawrequestbody := must.MarshalJSON(request) + + // create request + req := runtimex.Try1(http.NewRequest("POST", URL.String(), bytes.NewReader(rawrequestbody))) + + // make sure there's content-type + req.Header.Set("Content-Type", "application/json") + + // issue the request + resp, err := http.DefaultClient.Do(req) + + // we don't expect error + if err != nil { + t.Fatal(err) + } + + // make sure we close the body + defer resp.Body.Close() + + // we expect to see 200 + if resp.StatusCode != http.StatusOK { + t.Fatal("unexpected status code") + } + + // read and parse response body + rawrespbody := runtimex.Try1(io.ReadAll(resp.Body)) + var response model.OOAPICollectorUpdateResponse + must.UnmarshalJSON(rawrespbody, &response) + + // make sure the measurement UID is not empty + if response.MeasurementUID == "" { + t.Fatal("the measurement UID is unexpectedly empty") + } + }) +} diff --git a/script/nocopyreadall.bash b/script/nocopyreadall.bash index be14d720d..dcfd3f1d3 100755 --- a/script/nocopyreadall.bash +++ b/script/nocopyreadall.bash @@ -91,6 +91,18 @@ for file in $(find . -type f -name \*.go); do continue fi + if [ "$file" = "./internal/testingx/oonicollector.go" ]; then + # We're allowed to use ReadAll and Copy in this file because + # it's code that we only use for testing purposes. + continue + fi + + if [ "$file" = "./internal/testingx/oonicollector_test.go" ]; then + # We're allowed to use ReadAll and Copy in this file because + # it's code that we only use for testing purposes. + continue + fi + if [ "$file" = "./internal/testingx/tlssniproxy.go" ]; then # We're allowed to use ReadAll and Copy in this file because # it's code that we only use for testing purposes.