|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/tls"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/go-kratos/kratos/v2/errors"
|
|
|
|
"github.com/go-kratos/kratos/v2/middleware"
|
|
|
|
|
|
|
|
"github.com/go-kratos/kratos/v2/internal/host"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
)
|
|
|
|
|
|
|
|
type testKey struct{}
|
|
|
|
|
|
|
|
type testData struct {
|
|
|
|
Path string `json:"path"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestServer(t *testing.T) {
|
|
|
|
fn := func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
_ = json.NewEncoder(w).Encode(testData{Path: r.RequestURI})
|
|
|
|
}
|
|
|
|
ctx := context.Background()
|
|
|
|
srv := NewServer()
|
|
|
|
srv.HandleFunc("/index", fn)
|
|
|
|
srv.HandleFunc("/index/{id:[0-9]+}", fn)
|
|
|
|
|
|
|
|
if e, err := srv.Endpoint(); err != nil || e == nil || strings.HasSuffix(e.Host, ":0") {
|
|
|
|
t.Fatal(e, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
if err := srv.Start(ctx); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
time.Sleep(time.Second)
|
|
|
|
testClient(t, srv)
|
|
|
|
assert.NoError(t, srv.Stop(ctx))
|
|
|
|
}
|
|
|
|
|
|
|
|
func testClient(t *testing.T, srv *Server) {
|
|
|
|
tests := []struct {
|
|
|
|
method string
|
|
|
|
path string
|
|
|
|
}{
|
|
|
|
{"GET", "/index"},
|
|
|
|
{"PUT", "/index"},
|
|
|
|
{"POST", "/index"},
|
|
|
|
{"PATCH", "/index"},
|
|
|
|
{"DELETE", "/index"},
|
|
|
|
|
|
|
|
{"GET", "/index/1"},
|
|
|
|
{"PUT", "/index/1"},
|
|
|
|
{"POST", "/index/1"},
|
|
|
|
{"PATCH", "/index/1"},
|
|
|
|
{"DELETE", "/index/1"},
|
|
|
|
|
|
|
|
{"GET", "/index/notfound"},
|
|
|
|
}
|
|
|
|
e, err := srv.Endpoint()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
client, err := NewClient(context.Background(), WithEndpoint(e.Host))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer client.Close()
|
|
|
|
for _, test := range tests {
|
|
|
|
var res testData
|
|
|
|
url := fmt.Sprintf(e.String() + test.path)
|
|
|
|
req, err := http.NewRequest(test.method, url, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
if test.path == "/index/notfound" && err != nil {
|
|
|
|
if e, ok := err.(*errors.Error); ok && e.Code == http.StatusNotFound {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
t.Fatalf("http status got %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
content, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("read resp error %v", err)
|
|
|
|
}
|
|
|
|
err = json.Unmarshal(content, &res)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unmarshal resp error %v", err)
|
|
|
|
}
|
|
|
|
if res.Path != test.path {
|
|
|
|
t.Errorf("expected %s got %s", test.path, res.Path)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, test := range tests {
|
|
|
|
var res testData
|
|
|
|
err := client.Invoke(context.Background(), test.method, test.path, nil, &res)
|
|
|
|
if test.path == "/index/notfound" && err != nil {
|
|
|
|
if e, ok := err.(*errors.Error); ok && e.Code == http.StatusNotFound {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("invoke error %v", err)
|
|
|
|
}
|
|
|
|
if res.Path != test.path {
|
|
|
|
t.Errorf("expected %s got %s", test.path, res.Path)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkServer(b *testing.B) {
|
|
|
|
fn := func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
data := &testData{Path: r.RequestURI}
|
|
|
|
_ = json.NewEncoder(w).Encode(data)
|
|
|
|
if r.Context().Value(testKey{}) != "test" {
|
|
|
|
w.WriteHeader(500)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ctx := context.Background()
|
|
|
|
ctx = context.WithValue(ctx, testKey{}, "test")
|
|
|
|
srv := NewServer()
|
|
|
|
srv.HandleFunc("/index", fn)
|
|
|
|
go func() {
|
|
|
|
if err := srv.Start(ctx); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
time.Sleep(time.Second)
|
|
|
|
port, ok := host.Port(srv.lis)
|
|
|
|
assert.True(b, ok)
|
|
|
|
client, err := NewClient(context.Background(), WithEndpoint(fmt.Sprintf("127.0.0.1:%d", port)))
|
|
|
|
assert.NoError(b, err)
|
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
var res testData
|
|
|
|
err := client.Invoke(context.Background(), "POST", "/index", nil, &res)
|
|
|
|
assert.NoError(b, err)
|
|
|
|
}
|
|
|
|
_ = srv.Stop(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNetwork(t *testing.T) {
|
|
|
|
o := &Server{}
|
|
|
|
v := "abc"
|
|
|
|
Network(v)(o)
|
|
|
|
assert.Equal(t, v, o.network)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddress(t *testing.T) {
|
|
|
|
o := &Server{}
|
|
|
|
v := "abc"
|
|
|
|
Address(v)(o)
|
|
|
|
assert.Equal(t, v, o.address)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestTimeout(t *testing.T) {
|
|
|
|
o := &Server{}
|
|
|
|
v := time.Duration(123)
|
|
|
|
Timeout(v)(o)
|
|
|
|
assert.Equal(t, v, o.timeout)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLogger(t *testing.T) {
|
|
|
|
// todo
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestEndpoint(t *testing.T) {
|
|
|
|
u, err := url.Parse("http://hello/world")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
o := &Server{}
|
|
|
|
Endpoint(u)(o)
|
|
|
|
assert.Equal(t, "hello", o.endpoint.Host)
|
|
|
|
assert.Equal(t, "http", o.endpoint.Scheme)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMiddleware(t *testing.T) {
|
|
|
|
o := &Server{}
|
|
|
|
v := []middleware.Middleware{
|
|
|
|
func(middleware.Handler) middleware.Handler { return nil },
|
|
|
|
}
|
|
|
|
Middleware(v...)(o)
|
|
|
|
assert.Equal(t, v, o.ms)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRequestDecoder(t *testing.T) {
|
|
|
|
o := &Server{}
|
|
|
|
v := func(*http.Request, interface{}) error { return nil }
|
|
|
|
RequestDecoder(v)(o)
|
|
|
|
assert.NotNil(t, o.dec)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResponseEncoder(t *testing.T) {
|
|
|
|
o := &Server{}
|
|
|
|
v := func(http.ResponseWriter, *http.Request, interface{}) error { return nil }
|
|
|
|
ResponseEncoder(v)(o)
|
|
|
|
assert.NotNil(t, o.enc)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestErrorEncoder(t *testing.T) {
|
|
|
|
o := &Server{}
|
|
|
|
v := func(http.ResponseWriter, *http.Request, error) {}
|
|
|
|
ErrorEncoder(v)(o)
|
|
|
|
assert.NotNil(t, o.ene)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestTLSConfig(t *testing.T) {
|
|
|
|
o := &Server{}
|
|
|
|
v := &tls.Config{}
|
|
|
|
TLSConfig(v)(o)
|
|
|
|
assert.Equal(t, v, o.tlsConf)
|
|
|
|
}
|