feat(httpx): improve testing using the fakefiller (#649)
This diff extracts the fakefiller inside of internal/ooapi (a currently unused package) into its own package. The fakefiller knows how to fill many fields that are typically shared as data structures across processes. It is not perfect in that it cannot fill logger or http client fields, but still helps with better filling and testing. So, here we're using the fakefiller to improve testing of httpx and, nicely enough, we've already catched a bug in the way in which APIClientTemplate.Build misses to forward Authorization from the original template. Yay! Work part of https://github.com/ooni/probe/issues/1951
This commit is contained in:
@@ -0,0 +1,139 @@
|
||||
// Package fakefill contains code to fill structs for testing.
|
||||
//
|
||||
// This package is quite limited in scope and we can fill only the
|
||||
// structures you typically send over as JSONs.
|
||||
//
|
||||
// As part of future work, we aim to investigate whether we can
|
||||
// replace this implementation with https://go.dev/blog/fuzz-beta.
|
||||
package fakefill
|
||||
|
||||
import (
|
||||
"math/rand"
|
||||
"reflect"
|
||||
"sync"
|
||||
"time"
|
||||
)
|
||||
|
||||
// Filler fills specific data structures with random data. The only
|
||||
// exception to this behaviour is time.Time, which is instead filled
|
||||
// with the current time plus a small random number of seconds.
|
||||
//
|
||||
// We use this implementation to initialize data in our model. The code
|
||||
// has been written with that in mind. It will require some hammering in
|
||||
// case we extend the model with new field types.
|
||||
//
|
||||
// Caveat: this kind of fillter does not support filling interfaces
|
||||
// and channels and other complex types. The current behavior when this
|
||||
// kind of data types is encountered is to just ignore them.
|
||||
type Filler struct {
|
||||
// mu provides mutual exclusion
|
||||
mu sync.Mutex
|
||||
|
||||
// Now is OPTIONAL and allows to mock the current time
|
||||
Now func() time.Time
|
||||
|
||||
// rnd is the random number generator and is
|
||||
// automatically initialized on first use
|
||||
rnd *rand.Rand
|
||||
}
|
||||
|
||||
func (ff *Filler) getRandLocked() *rand.Rand {
|
||||
if ff.rnd == nil {
|
||||
now := time.Now
|
||||
if ff.Now != nil {
|
||||
now = ff.Now
|
||||
}
|
||||
ff.rnd = rand.New(rand.NewSource(now().UnixNano()))
|
||||
}
|
||||
return ff.rnd
|
||||
}
|
||||
|
||||
func (ff *Filler) getRandomString() string {
|
||||
defer ff.mu.Unlock()
|
||||
ff.mu.Lock()
|
||||
rnd := ff.getRandLocked()
|
||||
n := rnd.Intn(63) + 1
|
||||
// See https://stackoverflow.com/a/31832326
|
||||
var letterRunes = []rune("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ")
|
||||
b := make([]rune, n)
|
||||
for i := range b {
|
||||
b[i] = letterRunes[rnd.Intn(len(letterRunes))]
|
||||
}
|
||||
return string(b)
|
||||
}
|
||||
|
||||
func (ff *Filler) getRandomInt64() int64 {
|
||||
defer ff.mu.Unlock()
|
||||
ff.mu.Lock()
|
||||
rnd := ff.getRandLocked()
|
||||
return rnd.Int63()
|
||||
}
|
||||
|
||||
func (ff *Filler) getRandomBool() bool {
|
||||
defer ff.mu.Unlock()
|
||||
ff.mu.Lock()
|
||||
rnd := ff.getRandLocked()
|
||||
return rnd.Float64() >= 0.5
|
||||
}
|
||||
|
||||
func (ff *Filler) getRandomSmallPositiveInt() int {
|
||||
defer ff.mu.Unlock()
|
||||
ff.mu.Lock()
|
||||
rnd := ff.getRandLocked()
|
||||
return int(rnd.Int63n(8)) + 1 // safe cast
|
||||
}
|
||||
|
||||
func (ff *Filler) doFill(v reflect.Value) {
|
||||
for v.Type().Kind() == reflect.Ptr {
|
||||
if v.IsNil() {
|
||||
// if the pointer is nil, allocate an element
|
||||
v.Set(reflect.New(v.Type().Elem()))
|
||||
}
|
||||
// switch to the element
|
||||
v = v.Elem()
|
||||
}
|
||||
switch v.Type().Kind() {
|
||||
case reflect.String:
|
||||
v.SetString(ff.getRandomString())
|
||||
case reflect.Int64:
|
||||
v.SetInt(ff.getRandomInt64())
|
||||
case reflect.Bool:
|
||||
v.SetBool(ff.getRandomBool())
|
||||
case reflect.Struct:
|
||||
if v.Type().String() == "time.Time" {
|
||||
// Implementation note: we treat the time specially
|
||||
// and we avoid attempting to set its fields.
|
||||
v.Set(reflect.ValueOf(time.Now().Add(
|
||||
time.Duration(ff.getRandomSmallPositiveInt()) * time.Second)))
|
||||
return
|
||||
}
|
||||
for idx := 0; idx < v.NumField(); idx++ {
|
||||
ff.doFill(v.Field(idx)) // visit all fields
|
||||
}
|
||||
case reflect.Slice:
|
||||
kind := v.Type().Elem()
|
||||
total := ff.getRandomSmallPositiveInt()
|
||||
for idx := 0; idx < total; idx++ {
|
||||
value := reflect.New(kind) // make a new element
|
||||
ff.doFill(value)
|
||||
v.Set(reflect.Append(v, value.Elem())) // append to slice
|
||||
}
|
||||
case reflect.Map:
|
||||
if v.Type().Key().Kind() != reflect.String {
|
||||
panic("fakefill: we only support string key types")
|
||||
}
|
||||
v.Set(reflect.MakeMap(v.Type())) // we need to init the map
|
||||
total := ff.getRandomSmallPositiveInt()
|
||||
kind := v.Type().Elem()
|
||||
for idx := 0; idx < total; idx++ {
|
||||
value := reflect.New(kind)
|
||||
ff.doFill(value)
|
||||
v.SetMapIndex(reflect.ValueOf(ff.getRandomString()), value.Elem())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Fill fills the input structure or pointer with random data.
|
||||
func (ff *Filler) Fill(in interface{}) {
|
||||
ff.doFill(reflect.ValueOf(in))
|
||||
}
|
||||
@@ -0,0 +1,91 @@
|
||||
package fakefill
|
||||
|
||||
import (
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
// exampleStructure is an example structure we fill.
|
||||
type exampleStructure struct {
|
||||
CategoryCodes string
|
||||
CountryCode string
|
||||
Enabled bool
|
||||
MaxResults int64
|
||||
Now time.Time
|
||||
}
|
||||
|
||||
func TestFakeFillWorksWithCustomTime(t *testing.T) {
|
||||
var req *exampleStructure
|
||||
ff := &Filler{
|
||||
Now: func() time.Time {
|
||||
return time.Date(1992, time.January, 24, 17, 53, 0, 0, time.UTC)
|
||||
},
|
||||
}
|
||||
ff.Fill(&req)
|
||||
if req == nil {
|
||||
t.Fatal("we expected non nil here")
|
||||
}
|
||||
}
|
||||
|
||||
func TestFakeFillAllocatesIntoAPointerToPointer(t *testing.T) {
|
||||
var req *exampleStructure
|
||||
ff := &Filler{}
|
||||
ff.Fill(&req)
|
||||
if req == nil {
|
||||
t.Fatal("we expected non nil here")
|
||||
}
|
||||
}
|
||||
|
||||
func TestFakeFillAllocatesIntoAMapLikeWithStringKeys(t *testing.T) {
|
||||
var resp map[string]*exampleStructure
|
||||
ff := &Filler{}
|
||||
ff.Fill(&resp)
|
||||
if resp == nil {
|
||||
t.Fatal("we expected non nil here")
|
||||
}
|
||||
if len(resp) < 1 {
|
||||
t.Fatal("we expected some data here")
|
||||
}
|
||||
for _, value := range resp {
|
||||
if value == nil {
|
||||
t.Fatal("expected non-nil here")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestFakeFillAllocatesIntoAMapLikeWithNonStringKeys(t *testing.T) {
|
||||
var panicmsg string
|
||||
func() {
|
||||
defer func() {
|
||||
if v := recover(); v != nil {
|
||||
panicmsg = v.(string)
|
||||
}
|
||||
}()
|
||||
var resp map[int64]*exampleStructure
|
||||
ff := &Filler{}
|
||||
ff.Fill(&resp)
|
||||
if resp != nil {
|
||||
t.Fatal("we expected nil here")
|
||||
}
|
||||
}()
|
||||
if panicmsg != "fakefill: we only support string key types" {
|
||||
t.Fatal("unexpected panic message", panicmsg)
|
||||
}
|
||||
}
|
||||
|
||||
func TestFakeFillAllocatesIntoASlice(t *testing.T) {
|
||||
var resp *[]*exampleStructure
|
||||
ff := &Filler{}
|
||||
ff.Fill(&resp)
|
||||
if resp == nil {
|
||||
t.Fatal("we expected non nil here")
|
||||
}
|
||||
if len(*resp) < 1 {
|
||||
t.Fatal("we expected some data here")
|
||||
}
|
||||
for _, entry := range *resp {
|
||||
if entry == nil {
|
||||
t.Fatal("expected non-nil here")
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user