Skip to content

Commit

Permalink
Add some more tests
Browse files Browse the repository at this point in the history
  • Loading branch information
guseggert committed May 9, 2023
1 parent 4f44e87 commit 51b9d17
Show file tree
Hide file tree
Showing 5 changed files with 176 additions and 87 deletions.
38 changes: 20 additions & 18 deletions routing/http/client/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,8 +29,15 @@ import (
)

var (
_ contentrouter.Client = &client{}
logger = logging.Logger("service/delegatedrouting")
_ contentrouter.Client = &client{}
logger = logging.Logger("service/delegatedrouting")
defaultHTTPClient = &http.Client{
Transport: &ResponseBodyLimitedTransport{
RoundTripper: http.DefaultTransport,
LimitBytes: 1 << 20,
UserAgent: defaultUserAgent,
},
}
)

const (
Expand Down Expand Up @@ -65,21 +72,21 @@ type httpClient interface {
Do(req *http.Request) (*http.Response, error)
}

type option func(*client)
type Option func(*client)

func WithIdentity(identity crypto.PrivKey) option {
func WithIdentity(identity crypto.PrivKey) Option {
return func(c *client) {
c.identity = identity
}
}

func WithHTTPClient(h httpClient) option {
func WithHTTPClient(h httpClient) Option {
return func(c *client) {
c.httpClient = h
}
}

func WithUserAgent(ua string) option {
func WithUserAgent(ua string) Option {
return func(c *client) {
if ua == "" {
return
Expand All @@ -96,7 +103,7 @@ func WithUserAgent(ua string) option {
}
}

func WithProviderInfo(peerID peer.ID, addrs []multiaddr.Multiaddr) option {
func WithProviderInfo(peerID peer.ID, addrs []multiaddr.Multiaddr) Option {
return func(c *client) {
c.peerID = peerID
for _, a := range addrs {
Expand All @@ -105,22 +112,15 @@ func WithProviderInfo(peerID peer.ID, addrs []multiaddr.Multiaddr) option {
}
}

func WithStreamResultsRequired() option {
func WithStreamResultsRequired() Option {
return func(c *client) {
c.accepts = mediaTypeNDJSON
}
}

// New creates a content routing API client.
// The Provider and identity parameters are option. If they are nil, the `Provide` method will not function.
func New(baseURL string, opts ...option) (*client, error) {
defaultHTTPClient := &http.Client{
Transport: &ResponseBodyLimitedTransport{
RoundTripper: http.DefaultTransport,
LimitBytes: 1 << 20,
UserAgent: defaultUserAgent,
},
}
func New(baseURL string, opts ...Option) (*client, error) {
client := &client{
baseURL: baseURL,
httpClient: defaultHTTPClient,
Expand Down Expand Up @@ -171,6 +171,7 @@ func (c *client) FindProviders(ctx context.Context, key cid.Cid) (provs iter.Res
if err != nil {
return nil, err
}
req.Header.Set("Accept", c.accepts)

m.host = req.Host

Expand All @@ -189,13 +190,14 @@ func (c *client) FindProviders(ctx context.Context, key cid.Cid) (provs iter.Res
if resp.StatusCode == http.StatusNotFound {
resp.Body.Close()
m.record(ctx)
return nil, nil
return iter.FromSlice[iter.Result[types.ProviderResponse]](nil), nil
}

if resp.StatusCode != http.StatusOK {
err := httpError(resp.StatusCode, resp.Body)
resp.Body.Close()
m.record(ctx)
return nil, httpError(resp.StatusCode, resp.Body)
return nil, err
}

respContentType := resp.Header.Get("Content-Type")
Expand Down
186 changes: 134 additions & 52 deletions routing/http/client/client_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -42,45 +42,76 @@ func (m *mockContentRouter) Provide(ctx context.Context, req *server.WriteProvid
}

type testDeps struct {
router *mockContentRouter
server *httptest.Server
peerID peer.ID
addrs []multiaddr.Multiaddr
client *client
// recordingHandler records requests received on the server side
recordingHandler *recordingHandler
// recordingHTTPClient records responses received on the client side
recordingHTTPClient *recordingHTTPClient
router *mockContentRouter
server *httptest.Server
peerID peer.ID
addrs []multiaddr.Multiaddr
client *client
}

func makeTestDeps(t *testing.T) testDeps {
type recordingHandler struct {
http.Handler
f []func(*http.Request)
}

func (h *recordingHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
for _, f := range h.f {
f(r)
}
h.Handler.ServeHTTP(w, r)
}

type recordingHTTPClient struct {
httpClient
f []func(*http.Response)
}

func (c *recordingHTTPClient) Do(req *http.Request) (*http.Response, error) {
resp, err := c.httpClient.Do(req)
for _, f := range c.f {
f(resp)
}
return resp, err
}

func makeTestDeps(t *testing.T, clientsOpts []Option, serverOpts []server.Option) testDeps {
const testUserAgent = "testUserAgent"
peerID, addrs, identity := makeProviderAndIdentity()
router := &mockContentRouter{}
server := httptest.NewServer(server.Handler(router))
recordingHandler := &recordingHandler{
Handler: server.Handler(router, serverOpts...),
f: []func(*http.Request){
func(r *http.Request) {
assert.Equal(t, testUserAgent, r.Header.Get("User-Agent"))
},
},
}
server := httptest.NewServer(recordingHandler)
t.Cleanup(server.Close)
serverAddr := "http://" + server.Listener.Addr().String()
c, err := New(serverAddr, WithProviderInfo(peerID, addrs), WithIdentity(identity), WithUserAgent(testUserAgent))
recordingHTTPClient := &recordingHTTPClient{httpClient: defaultHTTPClient}
defaultClientOpts := []Option{
WithProviderInfo(peerID, addrs),
WithIdentity(identity),
WithUserAgent(testUserAgent),
WithHTTPClient(recordingHTTPClient),
}
c, err := New(serverAddr, append(defaultClientOpts, clientsOpts...)...)
if err != nil {
panic(err)
}
assertUserAgentOverride(t, c, testUserAgent)
return testDeps{
router: router,
server: server,
peerID: peerID,
addrs: addrs,
client: c,
}
}

func assertUserAgentOverride(t *testing.T, c *client, expected string) {
httpClient, ok := c.httpClient.(*http.Client)
if !ok {
t.Error("invalid c.httpClient")
}
transport, ok := httpClient.Transport.(*ResponseBodyLimitedTransport)
if !ok {
t.Error("invalid httpClient.Transport")
}
if transport.UserAgent != expected {
t.Error("invalid httpClient.Transport.UserAgent")
recordingHandler: recordingHandler,
recordingHTTPClient: recordingHTTPClient,
router: router,
server: server,
peerID: peerID,
addrs: addrs,
client: c,
}
}

Expand Down Expand Up @@ -150,6 +181,10 @@ type osErrContains struct {
}

func (e *osErrContains) errContains(t *testing.T, err error) {
if e.expContains == "" && e.expContainsWin == "" {
assert.NoError(t, err)
return
}
if runtime.GOOS == "windows" && len(e.expContainsWin) != 0 {
assert.ErrorContains(t, err, e.expContainsWin)
} else {
Expand All @@ -164,37 +199,90 @@ func TestClient_FindProviders(t *testing.T) {
}

cases := []struct {
name string
httpStatusCode int
stopServer bool
routerProvs []iter.Result[types.ProviderResponse]
routerErr error

expProvs []iter.Result[types.ProviderResponse]
expErrContains []osErrContains
name string
httpStatusCode int
stopServer bool
routerProvs []iter.Result[types.ProviderResponse]
routerErr error
clientRequiresStreaming bool
serverStreamingDisabled bool

expErrContains osErrContains
expProvs []iter.Result[types.ProviderResponse]
expStreamingResponse bool
expJSONResponse bool
}{
{
name: "happy case",
routerProvs: bitswapProvs,
expProvs: bitswapProvs,
name: "happy case",
routerProvs: bitswapProvs,
expProvs: bitswapProvs,
expStreamingResponse: true,
},
{
name: "server doesn't support streaming",
routerProvs: bitswapProvs,
expProvs: bitswapProvs,
serverStreamingDisabled: true,
expJSONResponse: true,
},
{
name: "client requires streaming but server doesn't support it",
serverStreamingDisabled: true,
clientRequiresStreaming: true,
expErrContains: osErrContains{expContains: "HTTP error with StatusCode=400: no supported content types"},
},
{
name: "returns an error if there's a non-200 response",
httpStatusCode: 500,
expErrContains: []osErrContains{{expContains: "HTTP error with StatusCode=500: "}},
expErrContains: osErrContains{expContains: "HTTP error with StatusCode=500"},
},
{
name: "returns an error if the HTTP client returns a non-HTTP error",
stopServer: true,
expErrContains: []osErrContains{{
expErrContains: osErrContains{
expContains: "connect: connection refused",
expContainsWin: "connectex: No connection could be made because the target machine actively refused it.",
}},
},
},
{
name: "returns no providers if the HTTP server returns a 404 respones",
httpStatusCode: 404,
expProvs: nil,
},
}
for _, c := range cases {
t.Run(c.name, func(t *testing.T) {
deps := makeTestDeps(t)
var clientOpts []Option
var serverOpts []server.Option
var onRespReceived []func(*http.Response)
var onReqReceived []func(*http.Request)

if c.serverStreamingDisabled {
serverOpts = append(serverOpts, server.WithStreamingResultsDisabled())
}
if c.clientRequiresStreaming {
clientOpts = append(clientOpts, WithStreamResultsRequired())
onReqReceived = append(onReqReceived, func(r *http.Request) {
assert.Equal(t, mediaTypeNDJSON, r.Header.Get("Accept"))
})
}

if c.expStreamingResponse {
onRespReceived = append(onRespReceived, func(r *http.Response) {
assert.Equal(t, mediaTypeNDJSON, r.Header.Get("Content-Type"))
})
}
if c.expJSONResponse {
onRespReceived = append(onRespReceived, func(r *http.Response) {
assert.Equal(t, mediaTypeJSON, r.Header.Get("Content-Type"))
})
}

deps := makeTestDeps(t, clientOpts, serverOpts)

deps.recordingHTTPClient.f = append(deps.recordingHTTPClient.f, onRespReceived...)
deps.recordingHandler.f = append(deps.recordingHandler.f, onReqReceived...)

client := deps.client
router := deps.router

Expand All @@ -219,12 +307,7 @@ func TestClient_FindProviders(t *testing.T) {

provsIter, err := client.FindProviders(ctx, cid)

for _, exp := range c.expErrContains {
exp.errContains(t, err)
}
if len(c.expErrContains) == 0 {
require.NoError(t, err)
}
c.expErrContains.errContains(t, err)

provs := iter.ReadAll[iter.Result[types.ProviderResponse]](provsIter)
assert.Equal(t, c.expProvs, provs)
Expand Down Expand Up @@ -264,8 +347,7 @@ func TestClient_Provide(t *testing.T) {
name: "should return a 403 if the payload signature verification fails",
cids: []cid.Cid{},
mangleSignature: true,

expErrContains: "HTTP error with StatusCode=403",
expErrContains: "HTTP error with StatusCode=403",
},
{
name: "should return error if identity is not provided",
Expand All @@ -291,7 +373,7 @@ func TestClient_Provide(t *testing.T) {
}
for _, c := range cases {
t.Run(c.name, func(t *testing.T) {
deps := makeTestDeps(t)
deps := makeTestDeps(t, nil, nil)
client := deps.client
router := deps.router

Expand Down
Loading

0 comments on commit 51b9d17

Please sign in to comment.