2022-01-26 17:23:55 -05:00
|
|
|
package httpserver
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"net/http"
|
|
|
|
|
"testing"
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
"github.com/qdm12/govalid/address"
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func Test_Settings_SetDefaults(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
const defaultTimeout = 3 * time.Second
|
|
|
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
|
settings Settings
|
|
|
|
|
expected Settings
|
|
|
|
|
}{
|
|
|
|
|
"empty settings": {
|
|
|
|
|
settings: Settings{},
|
|
|
|
|
expected: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8000",
|
|
|
|
|
ReadHeaderTimeout: defaultTimeout,
|
|
|
|
|
ReadTimeout: defaultTimeout,
|
|
|
|
|
ShutdownTimeout: defaultTimeout,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
"filled settings": {
|
|
|
|
|
settings: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
expected: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for name, testCase := range testCases {
|
|
|
|
|
testCase := testCase
|
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
testCase.settings.SetDefaults()
|
|
|
|
|
|
|
|
|
|
assert.Equal(t, testCase.expected, testCase.settings)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Test_Settings_Copy(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
someHandler := http.NewServeMux()
|
|
|
|
|
someLogger := &testLogger{}
|
|
|
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
|
settings Settings
|
|
|
|
|
expected Settings
|
|
|
|
|
}{
|
|
|
|
|
"empty settings": {},
|
|
|
|
|
"filled settings": {
|
|
|
|
|
settings: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
expected: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for name, testCase := range testCases {
|
|
|
|
|
testCase := testCase
|
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
copied := testCase.settings.Copy()
|
|
|
|
|
|
|
|
|
|
assert.Equal(t, testCase.expected, copied)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Test_Settings_MergeWith(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
someHandler := http.NewServeMux()
|
|
|
|
|
someLogger := &testLogger{}
|
|
|
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
|
settings Settings
|
|
|
|
|
other Settings
|
|
|
|
|
expected Settings
|
|
|
|
|
}{
|
|
|
|
|
"merge empty with empty": {},
|
|
|
|
|
"merge empty with filled": {
|
|
|
|
|
other: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
expected: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
"merge filled with empty": {
|
|
|
|
|
settings: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
expected: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for name, testCase := range testCases {
|
|
|
|
|
testCase := testCase
|
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
testCase.settings.MergeWith(testCase.other)
|
|
|
|
|
|
|
|
|
|
assert.Equal(t, testCase.expected, testCase.settings)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Test_Settings_OverrideWith(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
someHandler := http.NewServeMux()
|
|
|
|
|
someLogger := &testLogger{}
|
|
|
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
|
settings Settings
|
|
|
|
|
other Settings
|
|
|
|
|
expected Settings
|
|
|
|
|
}{
|
|
|
|
|
"override empty with empty": {},
|
|
|
|
|
"override empty with filled": {
|
|
|
|
|
other: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
expected: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
"override filled with empty": {
|
|
|
|
|
settings: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
expected: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
"override filled with filled": {
|
|
|
|
|
settings: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8001",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Second,
|
|
|
|
|
ReadTimeout: time.Second,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
other: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8002",
|
|
|
|
|
ReadHeaderTimeout: time.Hour,
|
|
|
|
|
ReadTimeout: time.Hour,
|
|
|
|
|
ShutdownTimeout: time.Hour,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
expected: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8002",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Hour,
|
|
|
|
|
ReadTimeout: time.Hour,
|
|
|
|
|
ShutdownTimeout: time.Hour,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for name, testCase := range testCases {
|
|
|
|
|
testCase := testCase
|
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
testCase.settings.OverrideWith(testCase.other)
|
|
|
|
|
|
|
|
|
|
assert.Equal(t, testCase.expected, testCase.settings)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Test_Settings_Validate(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
someHandler := http.NewServeMux()
|
|
|
|
|
someLogger := &testLogger{}
|
|
|
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
|
settings Settings
|
|
|
|
|
errWrapped error
|
|
|
|
|
errMessage string
|
|
|
|
|
}{
|
|
|
|
|
"bad address": {
|
|
|
|
|
settings: Settings{
|
|
|
|
|
Address: "noport",
|
|
|
|
|
},
|
|
|
|
|
errWrapped: address.ErrValueNotValid,
|
|
|
|
|
errMessage: "value is not valid: address noport: missing port in address",
|
|
|
|
|
},
|
|
|
|
|
"nil handler": {
|
|
|
|
|
settings: Settings{
|
|
|
|
|
Address: ":8000",
|
|
|
|
|
},
|
|
|
|
|
errWrapped: ErrHandlerIsNotSet,
|
|
|
|
|
errMessage: ErrHandlerIsNotSet.Error(),
|
|
|
|
|
},
|
|
|
|
|
"nil logger": {
|
|
|
|
|
settings: Settings{
|
|
|
|
|
Address: ":8000",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
},
|
|
|
|
|
errWrapped: ErrLoggerIsNotSet,
|
|
|
|
|
errMessage: ErrLoggerIsNotSet.Error(),
|
|
|
|
|
},
|
2022-07-28 21:55:10 +00:00
|
|
|
"read header timeout too small": {
|
|
|
|
|
settings: Settings{
|
|
|
|
|
Address: ":8000",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Nanosecond,
|
|
|
|
|
},
|
|
|
|
|
errWrapped: ErrReadHeaderTimeoutTooSmall,
|
|
|
|
|
errMessage: "read header timeout is too small: 1ns must be at least 1ms",
|
|
|
|
|
},
|
|
|
|
|
"read timeout too small": {
|
|
|
|
|
settings: Settings{
|
|
|
|
|
Address: ":8000",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Millisecond,
|
|
|
|
|
ReadTimeout: time.Nanosecond,
|
|
|
|
|
},
|
|
|
|
|
errWrapped: ErrReadTimeoutTooSmall,
|
|
|
|
|
errMessage: "read timeout is too small: 1ns must be at least 1ms",
|
|
|
|
|
},
|
2022-01-26 17:23:55 -05:00
|
|
|
"shutdown timeout too small": {
|
|
|
|
|
settings: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8000",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Millisecond,
|
|
|
|
|
ReadTimeout: time.Millisecond,
|
|
|
|
|
ShutdownTimeout: time.Millisecond,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
errWrapped: ErrShutdownTimeoutTooSmall,
|
|
|
|
|
errMessage: "shutdown timeout is too small: 1ms must be at least 5ms",
|
|
|
|
|
},
|
|
|
|
|
"valid settings": {
|
|
|
|
|
settings: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8000",
|
|
|
|
|
Handler: someHandler,
|
|
|
|
|
Logger: someLogger,
|
|
|
|
|
ReadHeaderTimeout: time.Millisecond,
|
|
|
|
|
ReadTimeout: time.Millisecond,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for name, testCase := range testCases {
|
|
|
|
|
testCase := testCase
|
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
err := testCase.settings.Validate()
|
|
|
|
|
|
|
|
|
|
assert.ErrorIs(t, err, testCase.errWrapped)
|
|
|
|
|
if err != nil {
|
|
|
|
|
assert.EqualError(t, err, testCase.errMessage)
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Test_Settings_String(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
|
settings Settings
|
|
|
|
|
s string
|
|
|
|
|
}{
|
|
|
|
|
"all values": {
|
|
|
|
|
settings: Settings{
|
2022-07-28 21:55:10 +00:00
|
|
|
Address: ":8000",
|
|
|
|
|
ReadHeaderTimeout: time.Millisecond,
|
|
|
|
|
ReadTimeout: time.Millisecond,
|
|
|
|
|
ShutdownTimeout: time.Second,
|
2022-01-26 17:23:55 -05:00
|
|
|
},
|
2022-03-30 07:41:23 +00:00
|
|
|
s: `HTTP server settings:
|
2022-01-26 17:23:55 -05:00
|
|
|
├── Listening address: :8000
|
2022-07-28 21:55:10 +00:00
|
|
|
├── Read header timeout: 1ms
|
|
|
|
|
├── Read timeout: 1ms
|
2022-01-26 17:23:55 -05:00
|
|
|
└── Shutdown timeout: 1s`,
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for name, testCase := range testCases {
|
|
|
|
|
testCase := testCase
|
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
|
|
s := testCase.settings.String()
|
|
|
|
|
|
|
|
|
|
assert.Equal(t, testCase.s, s)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|