Skip to content

Latest commit

 

History

History

httpclient

Http Client Module

ci go report codecov Deps PkgGoDev

Http client module based on net/http.

Installation

go get github.com/ankorstore/yokai/httpclient

Documentation

To create a http.Client:

package main

import (
	"time"

	"github.com/ankorstore/yokai/httpclient"
	"github.com/ankorstore/yokai/httpclient/transport"
)

var client, _ = httpclient.NewDefaultHttpClientFactory().Create()

// equivalent to:
var client, _ = httpclient.NewDefaultHttpClientFactory().Create(
	httpclient.WithTransport(transport.NewBaseTransport()), // base http transport (optimized)
	httpclient.WithTimeout(30*time.Second),                 // 30 seconds timeout
	httpclient.WithCheckRedirect(nil),                      // default redirection checks
	httpclient.WithCookieJar(nil),                          // default cookie jar
)

Requests

This module provide some request helpers to ease client requests headers propagation from an incoming request:

  • CopyObservabilityRequestHeaders to copy x-request-id and traceparent headers
  • CopyRequestHeaders to choose a list of headers to copy

For example:

package main

import (
	"net/http"

	"github.com/ankorstore/yokai/httpclient"
)

func exampleHandler(w http.ResponseWriter, r *http.Request) {
	// create http client
	client, _ := httpclient.NewDefaultHttpClientFactory().Create()

	// build a request to send with the client
	rc, _ := http.NewRequest(http.MethodGet, "https://example.com", nil)

	// propagate observability headers: x-request-id and traceparent
	httpclient.CopyObservabilityRequestHeaders(r, rc)

	// client call
	resp, _ := client.Do(rc)

	// propagate response code
	w.WriteHeader(resp.StatusCode)
}

func main() {
	http.HandleFunc("/", exampleHandler)
	http.ListenAndServe(":8080", nil)
}

Transports

BaseTransport

This module provide a BaseTransport, optimized regarding max connections handling.

To use it:

package main

import (
	"github.com/ankorstore/yokai/httpclient"
	"github.com/ankorstore/yokai/httpclient/transport"
)

var client, _ = httpclient.NewDefaultHttpClientFactory().Create(
	httpclient.WithTransport(transport.NewBaseTransport()),
)

// equivalent to:
var client, _ = httpclient.NewDefaultHttpClientFactory().Create(
	httpclient.WithTransport(
		transport.NewBaseTransportWithConfig(&transport.BaseTransportConfig{
			MaxIdleConnections:        100,
			MaxConnectionsPerHost:     100,
			MaxIdleConnectionsPerHost: 100,
		}),
	),
)

LoggerTransport

This module provide a LoggerTransport, able to decorate any http.RoundTripper to add logging:

  • with requests and response details (and optionally body)
  • with configurable log level for each

To use it:

package main

import (
	"github.com/ankorstore/yokai/httpclient"
	"github.com/ankorstore/yokai/httpclient/transport"
	"github.com/rs/zerolog"
)

var client, _ = httpclient.NewDefaultHttpClientFactory().Create(
	httpclient.WithTransport(transport.NewLoggerTransport(nil)),
)

// equivalent to:
var client, _ = httpclient.NewDefaultHttpClientFactory().Create(
	httpclient.WithTransport(
		transport.NewLoggerTransportWithConfig(
			transport.NewBaseTransport(),
			&transport.LoggerTransportConfig{
				LogRequest:                       false,             // to log request details
				LogResponse:                      false,             // to log response details
				LogRequestBody:                   false,             // to log request body (if request details logging enabled)
				LogResponseBody:                  false,             // to log response body (if response details logging enabled)
				LogRequestLevel:                  zerolog.InfoLevel, // log level for request log
				LogResponseLevel:                 zerolog.InfoLevel, // log level for response log
				LogResponseLevelFromResponseCode: false,             // to use response code for response log level
			},
		),
	),
)

Note: if no transport is provided for decoration in transport.NewLoggerTransport(nil), the BaseTransport will be used as base transport.

MetricsTransport

This module provide a MetricsTransport, able to decorate any http.RoundTripper to add metrics:

  • about requests total count (labelled by url, http method and status code)
  • about requests duration (labelled by url and http method)

To use it:

package main

import (
	"github.com/ankorstore/yokai/httpclient"
	"github.com/ankorstore/yokai/httpclient/transport"
	"github.com/prometheus/client_golang/prometheus"
	"github.com/rs/zerolog"
)

var client, _ = httpclient.NewDefaultHttpClientFactory().Create(
	httpclient.WithTransport(transport.NewMetricsTransport(nil)),
)

// equivalent to:
var client, _ = httpclient.NewDefaultHttpClientFactory().Create(
	httpclient.WithTransport(
		transport.NewMetricsTransportWithConfig(
			transport.NewBaseTransport(),
			&transport.MetricsTransportConfig{
				Registry:                  prometheus.DefaultRegisterer, // metrics registry
				Namespace:                 "",                           // metrics namespace
				Subsystem:                 "",                           // metrics subsystem
				Buckets:                   prometheus.DefBuckets,        // metrics duration buckets
				NormalizeRequestPath:      false,                        // normalize the request path following the masks given in NormalizePathMasks
				NormalizeRequestPathMasks: map[string]string{},          // request path normalization masks (key: regex to match, value: mask to apply)
				NormalizeResponseStatus:   true,                         // normalize the response HTTP code (ex: 201 => 2xx)
			},
		),
	),
)

If no transport is provided for decoration in transport.NewMetricsTransport(nil), the BaseTransport will be used as base transport.

If no registry is provided in the config in transport.NewMetricsTransportWithConfig(nil, config), the prometheus.DefaultRegisterer will be used a metrics registry

If the provided config provides NormalizeRequestPath to true and with the following NormalizeRequestPathMasks:

map[string]string{
    `/foo/(.+)/bar\?page=(.+)`: "/foo/{fooId}/bar?page={pageId}",
},

Then if the request path is /foo/1/bar?page=2, the metric path label will be masked with /foo/{fooId}/bar?page={pageId}.

Testing

This module provides a httpclienttest.NewTestHTTPServer() helper for testing your clients against a test server, that allows you:

  • to define test HTTP roundtrips: a couple of test aware functions to define the request and the response behavior
  • to configure several test HTTP roundtrips if you need to test successive calls

To use it:

package main_test

import (
	"errors"
	"net/http"
	"testing"

	"github.com/ankorstore/yokai/httpclient"
	"github.com/ankorstore/yokai/httpclient/httpclienttest"
	"github.com/stretchr/testify/assert"
)

func TestHTTPClient(t *testing.T) {
	t.Parallel()

	// client
	client, err := httpclient.NewDefaultHttpClientFactory().Create()
	assert.NoError(t, err)

	// test server preparation
	testServer := httpclienttest.NewTestHTTPServer(
		t,
		// configures a roundtrip for the 1st client call
		httpclienttest.WithTestHTTPRoundTrip(
			// func to configure / assert on the client request
			func(tb testing.TB, req *http.Request) error {
				tb.Helper()

				// performs some assertions
				assert.Equal(tb, "/foo", req.URL.Path)

				// returning an error here will make the test fail, if needed
				return nil
			},
			// func to configure / assert on the response for the client
			func(tb testing.TB, w http.ResponseWriter) error {
				tb.Helper()

				// prepares the response for the client
				w.Header.Set("foo", "bar")

				// performs some assertions
				assert.Equal(tb, "bar", w.Header.Get("foo"))

				// returning an error here will make the test fail, if needed
				return nil
			},
		),
		// configures a roundtrip for the 2nd client call
		httpclienttest.WithTestHTTPRoundTrip(
			// func to configure / assert on the client request
			func(tb testing.TB, req *http.Request) error {
				tb.Helper()

				assert.Equal(tb, "/bar", req.URL.Path)
				
				return nil
			},
			// func to configure / assert on the response for the client
			func(tb testing.TB, w http.ResponseWriter) error {
				tb.Helper()

				w.WriteHeader(http.StatusInternalServerError)
				
				return nil
			},
		),
	)

	// 1st client call
	resp, err := client.Get(testServer.URL + "/foo")
	assert.NoError(t, err)
	assert.Equal(t, http.StatusOK, resp.StatusCode)
	assert.Equal(t, "bar", resp.Header.Get("foo"))

	// 2nd client call
	resp, err = client.Get(testServer.URL + "/bar")
	assert.NoError(t, err)
	assert.Equal(t, http.StatusInternalServerError, resp.StatusCode)
}

You can find more complete examples in the tests.