2020-07-26 17:09:05 +08:00
|
|
|
package httpx
|
|
|
|
|
|
|
|
import (
|
2022-12-03 18:48:02 +08:00
|
|
|
"context"
|
2020-08-21 23:09:35 +08:00
|
|
|
"errors"
|
2023-06-16 01:04:58 +08:00
|
|
|
"fmt"
|
2020-07-26 17:09:05 +08:00
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
2022-01-04 15:51:32 +08:00
|
|
|
"github.com/zeromicro/go-zero/core/logx"
|
2022-06-11 23:07:26 +08:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type message struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
logx.Disable()
|
|
|
|
}
|
|
|
|
|
2020-08-21 23:09:35 +08:00
|
|
|
func TestError(t *testing.T) {
|
2020-11-17 18:04:48 +08:00
|
|
|
const (
|
|
|
|
body = "foo"
|
|
|
|
wrappedBody = `"foo"`
|
|
|
|
)
|
|
|
|
|
|
|
|
tests := []struct {
|
2021-09-01 19:52:56 +08:00
|
|
|
name string
|
|
|
|
input string
|
2023-01-24 16:32:02 +08:00
|
|
|
errorHandler func(error) (int, any)
|
2021-09-01 19:52:56 +08:00
|
|
|
expectHasBody bool
|
|
|
|
expectBody string
|
|
|
|
expectCode int
|
2020-11-17 18:04:48 +08:00
|
|
|
}{
|
|
|
|
{
|
2021-09-01 19:52:56 +08:00
|
|
|
name: "default error handler",
|
|
|
|
input: body,
|
|
|
|
expectHasBody: true,
|
|
|
|
expectBody: body,
|
|
|
|
expectCode: http.StatusBadRequest,
|
2020-11-17 18:04:48 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "customized error handler return string",
|
|
|
|
input: body,
|
2023-01-24 16:32:02 +08:00
|
|
|
errorHandler: func(err error) (int, any) {
|
2020-11-17 18:04:48 +08:00
|
|
|
return http.StatusForbidden, err.Error()
|
|
|
|
},
|
2021-09-01 19:52:56 +08:00
|
|
|
expectHasBody: true,
|
|
|
|
expectBody: wrappedBody,
|
|
|
|
expectCode: http.StatusForbidden,
|
2020-11-17 18:04:48 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "customized error handler return error",
|
|
|
|
input: body,
|
2023-01-24 16:32:02 +08:00
|
|
|
errorHandler: func(err error) (int, any) {
|
2020-11-17 18:04:48 +08:00
|
|
|
return http.StatusForbidden, err
|
|
|
|
},
|
2021-09-01 19:52:56 +08:00
|
|
|
expectHasBody: true,
|
|
|
|
expectBody: body,
|
|
|
|
expectCode: http.StatusForbidden,
|
2021-09-01 19:33:33 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "customized error handler return nil",
|
|
|
|
input: body,
|
2023-01-24 16:32:02 +08:00
|
|
|
errorHandler: func(err error) (int, any) {
|
2021-09-01 19:33:33 +08:00
|
|
|
return http.StatusForbidden, nil
|
|
|
|
},
|
2021-09-01 19:52:56 +08:00
|
|
|
expectHasBody: false,
|
|
|
|
expectBody: "",
|
|
|
|
expectCode: http.StatusForbidden,
|
2020-11-17 18:04:48 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
if test.errorHandler != nil {
|
2023-06-16 01:04:58 +08:00
|
|
|
errorLock.RLock()
|
2020-11-17 18:04:48 +08:00
|
|
|
prev := errorHandler
|
2023-06-16 01:04:58 +08:00
|
|
|
errorLock.RUnlock()
|
2020-11-17 18:04:48 +08:00
|
|
|
SetErrorHandler(test.errorHandler)
|
|
|
|
defer func() {
|
2023-06-16 01:04:58 +08:00
|
|
|
errorLock.Lock()
|
2020-11-17 18:04:48 +08:00
|
|
|
errorHandler = prev
|
2023-06-16 01:04:58 +08:00
|
|
|
errorLock.Unlock()
|
2020-11-17 18:04:48 +08:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
Error(&w, errors.New(test.input))
|
|
|
|
assert.Equal(t, test.expectCode, w.code)
|
2021-09-01 19:52:56 +08:00
|
|
|
assert.Equal(t, test.expectHasBody, w.hasBody)
|
2020-11-17 18:04:48 +08:00
|
|
|
assert.Equal(t, test.expectBody, strings.TrimSpace(w.builder.String()))
|
|
|
|
})
|
2020-08-21 23:09:35 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-11 23:07:26 +08:00
|
|
|
func TestErrorWithGrpcError(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
Error(&w, status.Error(codes.Unavailable, "foo"))
|
|
|
|
assert.Equal(t, http.StatusServiceUnavailable, w.code)
|
|
|
|
assert.True(t, w.hasBody)
|
|
|
|
assert.True(t, strings.Contains(w.builder.String(), "foo"))
|
|
|
|
}
|
|
|
|
|
2021-12-29 21:34:28 +08:00
|
|
|
func TestErrorWithHandler(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
Error(&w, errors.New("foo"), func(w http.ResponseWriter, err error) {
|
|
|
|
http.Error(w, err.Error(), 499)
|
|
|
|
})
|
|
|
|
assert.Equal(t, 499, w.code)
|
|
|
|
assert.True(t, w.hasBody)
|
|
|
|
assert.Equal(t, "foo", strings.TrimSpace(w.builder.String()))
|
|
|
|
}
|
|
|
|
|
2020-08-21 23:09:35 +08:00
|
|
|
func TestOk(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
Ok(&w)
|
|
|
|
assert.Equal(t, http.StatusOK, w.code)
|
|
|
|
}
|
|
|
|
|
2020-07-26 17:09:05 +08:00
|
|
|
func TestOkJson(t *testing.T) {
|
2023-06-16 01:04:58 +08:00
|
|
|
t.Run("no handler", func(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
msg := message{Name: "anyone"}
|
|
|
|
OkJson(&w, msg)
|
|
|
|
assert.Equal(t, http.StatusOK, w.code)
|
|
|
|
assert.Equal(t, "{\"name\":\"anyone\"}", w.builder.String())
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("with handler", func(t *testing.T) {
|
2023-06-26 00:27:26 +08:00
|
|
|
okLock.RLock()
|
|
|
|
prev := okHandler
|
|
|
|
okLock.RUnlock()
|
2023-06-16 01:04:58 +08:00
|
|
|
t.Cleanup(func() {
|
2023-06-26 00:27:26 +08:00
|
|
|
okLock.Lock()
|
|
|
|
okHandler = prev
|
|
|
|
okLock.Unlock()
|
2023-06-16 01:04:58 +08:00
|
|
|
})
|
|
|
|
|
2023-06-26 00:27:26 +08:00
|
|
|
SetOkHandler(func(_ context.Context, v interface{}) any {
|
2023-06-16 01:04:58 +08:00
|
|
|
return fmt.Sprintf("hello %s", v.(message).Name)
|
|
|
|
})
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
msg := message{Name: "anyone"}
|
|
|
|
OkJson(&w, msg)
|
|
|
|
assert.Equal(t, http.StatusOK, w.code)
|
|
|
|
assert.Equal(t, `"hello anyone"`, w.builder.String())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOkJsonCtx(t *testing.T) {
|
|
|
|
t.Run("no handler", func(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
msg := message{Name: "anyone"}
|
|
|
|
OkJsonCtx(context.Background(), &w, msg)
|
|
|
|
assert.Equal(t, http.StatusOK, w.code)
|
|
|
|
assert.Equal(t, "{\"name\":\"anyone\"}", w.builder.String())
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("with handler", func(t *testing.T) {
|
2023-06-26 00:27:26 +08:00
|
|
|
okLock.RLock()
|
|
|
|
prev := okHandler
|
|
|
|
okLock.RUnlock()
|
2023-06-16 01:04:58 +08:00
|
|
|
t.Cleanup(func() {
|
2023-06-26 00:27:26 +08:00
|
|
|
okLock.Lock()
|
|
|
|
okHandler = prev
|
|
|
|
okLock.Unlock()
|
2023-06-16 01:04:58 +08:00
|
|
|
})
|
|
|
|
|
2023-06-26 00:27:26 +08:00
|
|
|
SetOkHandler(func(_ context.Context, v interface{}) any {
|
2023-06-16 01:04:58 +08:00
|
|
|
return fmt.Sprintf("hello %s", v.(message).Name)
|
|
|
|
})
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
msg := message{Name: "anyone"}
|
|
|
|
OkJsonCtx(context.Background(), &w, msg)
|
|
|
|
assert.Equal(t, http.StatusOK, w.code)
|
|
|
|
assert.Equal(t, `"hello anyone"`, w.builder.String())
|
|
|
|
})
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestWriteJsonTimeout(t *testing.T) {
|
|
|
|
// only log it and ignore
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
2022-05-02 21:24:20 +08:00
|
|
|
err: http.ErrHandlerTimeout,
|
|
|
|
}
|
|
|
|
msg := message{Name: "anyone"}
|
|
|
|
WriteJson(&w, http.StatusOK, msg)
|
|
|
|
assert.Equal(t, http.StatusOK, w.code)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWriteJsonError(t *testing.T) {
|
|
|
|
// only log it and ignore
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
err: errors.New("foo"),
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
msg := message{Name: "anyone"}
|
|
|
|
WriteJson(&w, http.StatusOK, msg)
|
|
|
|
assert.Equal(t, http.StatusOK, w.code)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWriteJsonLessWritten(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
lessWritten: true,
|
|
|
|
}
|
|
|
|
msg := message{Name: "anyone"}
|
|
|
|
WriteJson(&w, http.StatusOK, msg)
|
|
|
|
assert.Equal(t, http.StatusOK, w.code)
|
|
|
|
}
|
|
|
|
|
2022-04-21 21:55:01 +08:00
|
|
|
func TestWriteJsonMarshalFailed(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
2023-01-24 16:32:02 +08:00
|
|
|
WriteJson(&w, http.StatusOK, map[string]any{
|
2022-04-21 21:55:01 +08:00
|
|
|
"Data": complex(0, 0),
|
|
|
|
})
|
|
|
|
assert.Equal(t, http.StatusInternalServerError, w.code)
|
|
|
|
}
|
|
|
|
|
2020-07-26 17:09:05 +08:00
|
|
|
type tracedResponseWriter struct {
|
|
|
|
headers map[string][]string
|
|
|
|
builder strings.Builder
|
2021-09-01 19:52:56 +08:00
|
|
|
hasBody bool
|
2020-07-26 17:09:05 +08:00
|
|
|
code int
|
|
|
|
lessWritten bool
|
2022-04-21 21:55:01 +08:00
|
|
|
wroteHeader bool
|
2022-05-02 21:24:20 +08:00
|
|
|
err error
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (w *tracedResponseWriter) Header() http.Header {
|
|
|
|
return w.headers
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *tracedResponseWriter) Write(bytes []byte) (n int, err error) {
|
2022-05-02 21:24:20 +08:00
|
|
|
if w.err != nil {
|
|
|
|
return 0, w.err
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
n, err = w.builder.Write(bytes)
|
|
|
|
if w.lessWritten {
|
2022-10-14 22:45:48 +08:00
|
|
|
n--
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
2021-09-01 19:52:56 +08:00
|
|
|
w.hasBody = true
|
|
|
|
|
2020-07-26 17:09:05 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *tracedResponseWriter) WriteHeader(code int) {
|
2022-04-21 21:55:01 +08:00
|
|
|
if w.wroteHeader {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
w.wroteHeader = true
|
2020-07-26 17:09:05 +08:00
|
|
|
w.code = code
|
|
|
|
}
|
2022-12-03 18:48:02 +08:00
|
|
|
|
|
|
|
func TestErrorCtx(t *testing.T) {
|
|
|
|
const (
|
|
|
|
body = "foo"
|
|
|
|
wrappedBody = `"foo"`
|
|
|
|
)
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
input string
|
2023-01-24 16:32:02 +08:00
|
|
|
errorHandlerCtx func(context.Context, error) (int, any)
|
2022-12-03 18:48:02 +08:00
|
|
|
expectHasBody bool
|
|
|
|
expectBody string
|
|
|
|
expectCode int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "default error handler",
|
|
|
|
input: body,
|
|
|
|
expectHasBody: true,
|
|
|
|
expectBody: body,
|
|
|
|
expectCode: http.StatusBadRequest,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "customized error handler return string",
|
|
|
|
input: body,
|
2023-01-24 16:32:02 +08:00
|
|
|
errorHandlerCtx: func(ctx context.Context, err error) (int, any) {
|
2022-12-03 18:48:02 +08:00
|
|
|
return http.StatusForbidden, err.Error()
|
|
|
|
},
|
|
|
|
expectHasBody: true,
|
|
|
|
expectBody: wrappedBody,
|
|
|
|
expectCode: http.StatusForbidden,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "customized error handler return error",
|
|
|
|
input: body,
|
2023-01-24 16:32:02 +08:00
|
|
|
errorHandlerCtx: func(ctx context.Context, err error) (int, any) {
|
2022-12-03 18:48:02 +08:00
|
|
|
return http.StatusForbidden, err
|
|
|
|
},
|
|
|
|
expectHasBody: true,
|
|
|
|
expectBody: body,
|
|
|
|
expectCode: http.StatusForbidden,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "customized error handler return nil",
|
|
|
|
input: body,
|
2023-01-24 16:32:02 +08:00
|
|
|
errorHandlerCtx: func(context.Context, error) (int, any) {
|
2022-12-03 18:48:02 +08:00
|
|
|
return http.StatusForbidden, nil
|
|
|
|
},
|
|
|
|
expectHasBody: false,
|
|
|
|
expectBody: "",
|
|
|
|
expectCode: http.StatusForbidden,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
if test.errorHandlerCtx != nil {
|
2023-06-16 01:04:58 +08:00
|
|
|
errorLock.RLock()
|
|
|
|
prev := errorHandler
|
|
|
|
errorLock.RUnlock()
|
2022-12-03 18:48:02 +08:00
|
|
|
SetErrorHandlerCtx(test.errorHandlerCtx)
|
|
|
|
defer func() {
|
2023-06-16 01:04:58 +08:00
|
|
|
errorLock.Lock()
|
2022-12-03 18:48:02 +08:00
|
|
|
test.errorHandlerCtx = prev
|
2023-06-16 01:04:58 +08:00
|
|
|
errorLock.Unlock()
|
2022-12-03 18:48:02 +08:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
ErrorCtx(context.Background(), &w, errors.New(test.input))
|
|
|
|
assert.Equal(t, test.expectCode, w.code)
|
|
|
|
assert.Equal(t, test.expectHasBody, w.hasBody)
|
|
|
|
assert.Equal(t, test.expectBody, strings.TrimSpace(w.builder.String()))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-01-24 16:32:02 +08:00
|
|
|
// The current handler is a global event,Set default values to avoid impacting subsequent unit tests
|
2022-12-03 18:48:02 +08:00
|
|
|
SetErrorHandlerCtx(nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestErrorWithGrpcErrorCtx(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
ErrorCtx(context.Background(), &w, status.Error(codes.Unavailable, "foo"))
|
|
|
|
assert.Equal(t, http.StatusServiceUnavailable, w.code)
|
|
|
|
assert.True(t, w.hasBody)
|
|
|
|
assert.True(t, strings.Contains(w.builder.String(), "foo"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestErrorWithHandlerCtx(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
|
|
|
ErrorCtx(context.Background(), &w, errors.New("foo"), func(w http.ResponseWriter, err error) {
|
|
|
|
http.Error(w, err.Error(), 499)
|
|
|
|
})
|
|
|
|
assert.Equal(t, 499, w.code)
|
|
|
|
assert.True(t, w.hasBody)
|
|
|
|
assert.Equal(t, "foo", strings.TrimSpace(w.builder.String()))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWriteJsonCtxMarshalFailed(t *testing.T) {
|
|
|
|
w := tracedResponseWriter{
|
|
|
|
headers: make(map[string][]string),
|
|
|
|
}
|
2023-01-24 16:32:02 +08:00
|
|
|
WriteJsonCtx(context.Background(), &w, http.StatusOK, map[string]any{
|
2022-12-03 18:48:02 +08:00
|
|
|
"Data": complex(0, 0),
|
|
|
|
})
|
|
|
|
assert.Equal(t, http.StatusInternalServerError, w.code)
|
|
|
|
}
|