kratos/transport/http/resolver_test.go

176 lines
4.8 KiB

package http
import (
"context"
"errors"
"fmt"
"reflect"
"strconv"
"testing"
"time"
"github.com/go-kratos/kratos/v2/registry"
"github.com/go-kratos/kratos/v2/selector"
)
func TestParseTarget(t *testing.T) {
target, err := parseTarget("localhost:8000", true)
if err != nil {
t.Errorf("expect %v, got %v", nil, err)
}
if !reflect.DeepEqual(&Target{Scheme: "http", Authority: "localhost:8000"}, target) {
t.Errorf("expect %v, got %v", &Target{Scheme: "http", Authority: "localhost:8000"}, target)
}
target, err = parseTarget("discovery:///demo", true)
if err != nil {
t.Errorf("expect %v, got %v", nil, err)
}
if !reflect.DeepEqual(&Target{Scheme: "discovery", Authority: "", Endpoint: "demo"}, target) {
t.Errorf("expect %v, got %v", &Target{Scheme: "discovery", Authority: "", Endpoint: "demo"}, target)
}
target, err = parseTarget("127.0.0.1:8000", true)
if err != nil {
t.Errorf("expect %v, got %v", nil, err)
}
if !reflect.DeepEqual(&Target{Scheme: "http", Authority: "127.0.0.1:8000"}, target) {
t.Errorf("expect %v, got %v", &Target{Scheme: "http", Authority: "127.0.0.1:8000"}, target)
}
target, err = parseTarget("https://127.0.0.1:8000", false)
if err != nil {
t.Errorf("expect %v, got %v", nil, err)
}
if !reflect.DeepEqual(&Target{Scheme: "https", Authority: "127.0.0.1:8000"}, target) {
t.Errorf("expect %v, got %v", &Target{Scheme: "https", Authority: "127.0.0.1:8000"}, target)
}
target, err = parseTarget("127.0.0.1:8000", false)
if err != nil {
t.Errorf("expect %v, got %v", nil, err)
}
if !reflect.DeepEqual(&Target{Scheme: "https", Authority: "127.0.0.1:8000"}, target) {
t.Errorf("expect %v, got %v", &Target{Scheme: "https", Authority: "127.0.0.1:8000"}, target)
}
}
type mockRebalancer struct{}
func (m *mockRebalancer) Apply(nodes []selector.Node) {}
type mockDiscoveries struct {
isSecure bool
nextErr bool
stopErr bool
}
func (d *mockDiscoveries) GetService(ctx context.Context, serviceName string) ([]*registry.ServiceInstance, error) {
return nil, nil
}
const errServiceName = "needErr"
func (d *mockDiscoveries) Watch(ctx context.Context, serviceName string) (registry.Watcher, error) {
if serviceName == errServiceName {
return nil, errors.New("mock test service name watch err")
}
return &mockWatch{ctx: ctx, isSecure: d.isSecure, nextErr: d.nextErr, stopErr: d.stopErr}, nil
}
type mockWatch struct {
ctx context.Context
isSecure bool
count int
nextErr bool
stopErr bool
}
func (m *mockWatch) Next() ([]*registry.ServiceInstance, error) {
select {
case <-m.ctx.Done():
return nil, m.ctx.Err()
default:
}
if m.nextErr {
return nil, errors.New("mock test error")
}
if m.count == 1 {
return nil, errors.New("mock test error")
}
m.count++
instance := &registry.ServiceInstance{
ID: "1",
Name: "kratos",
Version: "v1",
Metadata: map[string]string{},
Endpoints: []string{fmt.Sprintf("http://127.0.0.1:9001?isSecure=%s", strconv.FormatBool(m.isSecure))},
}
if m.count > 3 {
time.Sleep(time.Millisecond * 500)
}
return []*registry.ServiceInstance{instance}, nil
}
func (m *mockWatch) Stop() error {
if m.stopErr {
return errors.New("mock test error")
}
// 标记 next 需要报错
m.nextErr = true
return nil
}
func TestResolver(t *testing.T) {
ta, err := parseTarget("discovery://helloworld", true)
if err != nil {
t.Errorf("parse err %v", err)
return
}
// 异步 无需报错
_, err = newResolver(context.Background(), &mockDiscoveries{true, false, false}, ta, &mockRebalancer{}, false, false)
if err != nil {
t.Errorf("expect %v, got %v", nil, err)
}
// 同步 一切正常运行
_, err = newResolver(context.Background(), &mockDiscoveries{false, false, false}, ta, &mockRebalancer{}, true, true)
if err != nil {
t.Errorf("expect %v, got %v", nil, err)
}
// 同步 但是 next 出错 以及 stop 出错
_, err = newResolver(context.Background(), &mockDiscoveries{false, true, true}, ta, &mockRebalancer{}, true, true)
if err == nil {
t.Errorf("expect err, got nil")
}
// 同步 service name watch 失败
_, err = newResolver(context.Background(), &mockDiscoveries{false, true, true}, &Target{
Scheme: "discovery",
Endpoint: errServiceName,
}, &mockRebalancer{}, true, true)
if err == nil {
t.Errorf("expect err, got nil")
}
cancelCtx, cancel := context.WithCancel(context.Background())
cancel()
// 此处应该打印出来 context.Canceled
r, err := newResolver(cancelCtx, &mockDiscoveries{false, false, false}, ta, &mockRebalancer{}, false, false)
if err != nil {
t.Errorf("expect %v, got %v", nil, err)
}
_ = r.Close()
// 同步 但是服务取消,此时需要报错
_, err = newResolver(cancelCtx, &mockDiscoveries{false, false, true}, ta, &mockRebalancer{}, true, true)
if err == nil {
t.Errorf("expect ctx cancel err, got nil")
}
}