535a5d3e00
In https://github.com/ooni/probe-cli/pull/832's initial diff, I mentioned it would be cool to flatten oohelperd's hier. I'm doing this now, and just for the master branch. This diff is mostly a mechanical refactoring with very light and apparently rather safe manual changes.
166 lines
3.9 KiB
Go
166 lines
3.9 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/model"
|
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
|
)
|
|
|
|
const simplerequest = `{
|
|
"http_request": "https://dns.google",
|
|
"http_request_headers": {
|
|
"Accept": [
|
|
"*/*"
|
|
],
|
|
"Accept-Language": [
|
|
"en-US;q=0.8,en;q=0.5"
|
|
],
|
|
"User-Agent": [
|
|
"Mozilla/5.0"
|
|
]
|
|
},
|
|
"tcp_connect": [
|
|
"8.8.8.8:443"
|
|
]
|
|
}`
|
|
|
|
const requestWithoutDomainName = `{
|
|
"http_request": "https://8.8.8.8",
|
|
"http_request_headers": {
|
|
"Accept": [
|
|
"*/*"
|
|
],
|
|
"Accept-Language": [
|
|
"en-US;q=0.8,en;q=0.5"
|
|
],
|
|
"User-Agent": [
|
|
"Mozilla/5.0"
|
|
]
|
|
},
|
|
"tcp_connect": [
|
|
"8.8.8.8:443"
|
|
]
|
|
}`
|
|
|
|
func TestWorkingAsIntended(t *testing.T) {
|
|
handler := &handler{
|
|
MaxAcceptableBody: 1 << 24,
|
|
NewClient: func() model.HTTPClient {
|
|
return http.DefaultClient
|
|
},
|
|
NewDialer: func() model.Dialer {
|
|
return netxlite.NewDialerWithStdlibResolver(model.DiscardLogger)
|
|
},
|
|
NewResolver: func() model.Resolver {
|
|
return netxlite.NewUnwrappedStdlibResolver()
|
|
},
|
|
}
|
|
srv := httptest.NewServer(handler)
|
|
defer srv.Close()
|
|
type expectationSpec struct {
|
|
name string
|
|
reqMethod string
|
|
reqContentType string
|
|
reqBody string
|
|
respStatusCode int
|
|
respContentType string
|
|
parseBody bool
|
|
}
|
|
expectations := []expectationSpec{{
|
|
name: "check for invalid method",
|
|
reqMethod: "GET",
|
|
respStatusCode: 400,
|
|
}, {
|
|
name: "check for invalid content-type",
|
|
reqMethod: "POST",
|
|
respStatusCode: 400,
|
|
}, {
|
|
name: "check for invalid request body",
|
|
reqMethod: "POST",
|
|
reqContentType: "application/json",
|
|
reqBody: "{",
|
|
respStatusCode: 400,
|
|
}, {
|
|
name: "with measurement failure",
|
|
reqMethod: "POST",
|
|
reqContentType: "application/json",
|
|
reqBody: `{"http_request": "http://[::1]aaaa"}`,
|
|
respStatusCode: 400,
|
|
}, {
|
|
name: "with reasonably good request",
|
|
reqMethod: "POST",
|
|
reqContentType: "application/json",
|
|
reqBody: simplerequest,
|
|
respStatusCode: 200,
|
|
respContentType: "application/json",
|
|
parseBody: true,
|
|
}, {
|
|
name: "when there's no domain name in the request",
|
|
reqMethod: "POST",
|
|
reqContentType: "application/json",
|
|
reqBody: requestWithoutDomainName,
|
|
respStatusCode: 200,
|
|
respContentType: "application/json",
|
|
parseBody: true,
|
|
}}
|
|
for _, expect := range expectations {
|
|
t.Run(expect.name, func(t *testing.T) {
|
|
body := strings.NewReader(expect.reqBody)
|
|
req, err := http.NewRequest(expect.reqMethod, srv.URL, body)
|
|
if err != nil {
|
|
t.Fatalf("%s: %+v", expect.name, err)
|
|
}
|
|
if expect.reqContentType != "" {
|
|
req.Header.Add("content-type", expect.reqContentType)
|
|
}
|
|
resp, err := http.DefaultClient.Do(req)
|
|
if err != nil {
|
|
t.Fatalf("%s: %+v", expect.name, err)
|
|
}
|
|
defer resp.Body.Close()
|
|
if resp.StatusCode != expect.respStatusCode {
|
|
t.Fatalf("unexpected status code: %+v", resp.StatusCode)
|
|
}
|
|
if v := resp.Header.Get("content-type"); v != expect.respContentType {
|
|
t.Fatalf("unexpected content-type: %s", v)
|
|
}
|
|
data, err := netxlite.ReadAllContext(context.Background(), resp.Body)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !expect.parseBody {
|
|
return
|
|
}
|
|
var v interface{}
|
|
if err := json.Unmarshal(data, &v); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestHandlerWithRequestBodyReadingError(t *testing.T) {
|
|
expected := errors.New("mocked error")
|
|
handler := handler{MaxAcceptableBody: 1 << 24}
|
|
rw := NewFakeResponseWriter()
|
|
req := &http.Request{
|
|
Method: "POST",
|
|
Header: map[string][]string{
|
|
"Content-Type": {"application/json"},
|
|
"Content-Length": {"2048"},
|
|
},
|
|
Body: &FakeBody{Err: expected},
|
|
}
|
|
handler.ServeHTTP(rw, req)
|
|
if rw.StatusCode != 400 {
|
|
t.Fatal("unexpected status code")
|
|
}
|
|
}
|