gateway: open timeout function cabinet (#3047)
Co-authored-by: Kevin Wan <wanjunfeng@gmail.com>
This commit is contained in:
@@ -4,3 +4,4 @@ comment:
|
||||
require_changes: true
|
||||
ignore:
|
||||
- "tools"
|
||||
- "internal/mock"
|
||||
@@ -23,22 +23,28 @@ import (
|
||||
type (
|
||||
// Server is a gateway server.
|
||||
Server struct {
|
||||
c GatewayConf
|
||||
*rest.Server
|
||||
upstreams []Upstream
|
||||
upstreams []*upstream
|
||||
timeout time.Duration
|
||||
processHeader func(http.Header) []string
|
||||
}
|
||||
|
||||
// Option defines the method to customize Server.
|
||||
Option func(svr *Server)
|
||||
|
||||
upstream struct {
|
||||
Upstream
|
||||
client zrpc.Client
|
||||
}
|
||||
)
|
||||
|
||||
// MustNewServer creates a new gateway server.
|
||||
func MustNewServer(c GatewayConf, opts ...Option) *Server {
|
||||
svr := &Server{
|
||||
Server: rest.MustNewServer(c.RestConf),
|
||||
upstreams: c.Upstreams,
|
||||
timeout: time.Duration(c.Timeout) * time.Millisecond,
|
||||
c: c,
|
||||
Server: rest.MustNewServer(c.RestConf),
|
||||
timeout: time.Duration(c.Timeout) * time.Millisecond,
|
||||
}
|
||||
for _, opt := range opts {
|
||||
opt(svr)
|
||||
@@ -59,17 +65,47 @@ func (s *Server) Stop() {
|
||||
}
|
||||
|
||||
func (s *Server) build() error {
|
||||
if err := s.buildClient(); err != nil {
|
||||
return err
|
||||
}
|
||||
return s.buildUpstream()
|
||||
}
|
||||
|
||||
func (s *Server) buildClient() error {
|
||||
if err := s.ensureUpstreamNames(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return mr.MapReduceVoid(func(source chan<- Upstream) {
|
||||
for _, up := range s.c.Upstreams {
|
||||
source <- up
|
||||
}
|
||||
}, func(up Upstream, writer mr.Writer[*upstream], cancel func(error)) {
|
||||
target, err := up.Grpc.BuildTarget()
|
||||
if err != nil {
|
||||
cancel(err)
|
||||
}
|
||||
up.Name = target
|
||||
cli := zrpc.MustNewClient(up.Grpc)
|
||||
writer.Write(&upstream{
|
||||
Upstream: up,
|
||||
client: cli,
|
||||
})
|
||||
}, func(pipe <-chan *upstream, cancel func(error)) {
|
||||
for up := range pipe {
|
||||
s.upstreams = append(s.upstreams, up)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func (s *Server) buildUpstream() error {
|
||||
return mr.MapReduceVoid(func(source chan<- *upstream) {
|
||||
for _, up := range s.upstreams {
|
||||
source <- up
|
||||
}
|
||||
}, func(up Upstream, writer mr.Writer[rest.Route], cancel func(error)) {
|
||||
cli := zrpc.MustNewClient(up.Grpc)
|
||||
source, err := s.createDescriptorSource(cli, up)
|
||||
}, func(up *upstream, writer mr.Writer[rest.Route], cancel func(error)) {
|
||||
cli := up.client
|
||||
source, err := s.createDescriptorSource(cli, up.Upstream)
|
||||
if err != nil {
|
||||
cancel(fmt.Errorf("%s: %w", up.Name, err))
|
||||
return
|
||||
@@ -161,7 +197,7 @@ func (s *Server) createDescriptorSource(cli zrpc.Client, up Upstream) (grpcurl.D
|
||||
}
|
||||
|
||||
func (s *Server) ensureUpstreamNames() error {
|
||||
for _, up := range s.upstreams {
|
||||
for _, up := range s.c.Upstreams {
|
||||
target, err := up.Grpc.BuildTarget()
|
||||
if err != nil {
|
||||
return err
|
||||
|
||||
95
gateway/server_test.go
Normal file
95
gateway/server_test.go
Normal file
@@ -0,0 +1,95 @@
|
||||
package gateway
|
||||
|
||||
import (
|
||||
"context"
|
||||
"log"
|
||||
"net"
|
||||
"net/http"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
"github.com/zeromicro/go-zero/core/conf"
|
||||
"github.com/zeromicro/go-zero/core/logx"
|
||||
"github.com/zeromicro/go-zero/internal/mock"
|
||||
"github.com/zeromicro/go-zero/rest/httpc"
|
||||
"github.com/zeromicro/go-zero/zrpc"
|
||||
"google.golang.org/grpc"
|
||||
"google.golang.org/grpc/reflection"
|
||||
"google.golang.org/grpc/test/bufconn"
|
||||
)
|
||||
|
||||
func init() {
|
||||
logx.Disable()
|
||||
}
|
||||
|
||||
func dialer() func(context.Context, string) (net.Conn, error) {
|
||||
listener := bufconn.Listen(1024 * 1024)
|
||||
server := grpc.NewServer()
|
||||
mock.RegisterDepositServiceServer(server, &mock.DepositServer{})
|
||||
|
||||
reflection.Register(server)
|
||||
|
||||
go func() {
|
||||
if err := server.Serve(listener); err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
}()
|
||||
|
||||
return func(context.Context, string) (net.Conn, error) {
|
||||
return listener.Dial()
|
||||
}
|
||||
}
|
||||
|
||||
func TestMustNewServer(t *testing.T) {
|
||||
var c GatewayConf
|
||||
assert.NoError(t, conf.FillDefault(&c))
|
||||
c.Port = 18881
|
||||
|
||||
s := MustNewServer(c)
|
||||
|
||||
s.upstreams = []*upstream{
|
||||
{
|
||||
Upstream: Upstream{
|
||||
Mappings: []RouteMapping{
|
||||
{
|
||||
Method: "get",
|
||||
Path: "/deposit/:amount",
|
||||
RpcPath: "mock.DepositService/Deposit",
|
||||
},
|
||||
},
|
||||
},
|
||||
client: zrpc.MustNewClient(zrpc.RpcClientConf{
|
||||
Endpoints: []string{"foo"},
|
||||
Timeout: 1000,
|
||||
Middlewares: zrpc.ClientMiddlewaresConf{
|
||||
Trace: true,
|
||||
Duration: true,
|
||||
Prometheus: true,
|
||||
Breaker: true,
|
||||
Timeout: true,
|
||||
},
|
||||
},
|
||||
zrpc.WithDialOption(grpc.WithContextDialer(dialer())),
|
||||
zrpc.WithUnaryClientInterceptor(func(ctx context.Context, method string, req, reply any,
|
||||
cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error {
|
||||
return invoker(ctx, method, req, reply, cc, opts...)
|
||||
})),
|
||||
},
|
||||
}
|
||||
|
||||
assert.NoError(t, s.buildUpstream())
|
||||
go s.Server.Start()
|
||||
|
||||
time.Sleep(time.Millisecond * 100)
|
||||
|
||||
ctx := context.Background()
|
||||
|
||||
resp, err := httpc.Do(ctx, http.MethodGet, "http://localhost:18881/deposit/100", nil)
|
||||
assert.NoError(t, err)
|
||||
assert.Equal(t, http.StatusOK, resp.StatusCode)
|
||||
|
||||
resp, err = httpc.Do(ctx, http.MethodGet, "http://localhost:18881/deposit_fail/100", nil)
|
||||
assert.NoError(t, err)
|
||||
assert.Equal(t, http.StatusNotFound, resp.StatusCode)
|
||||
}
|
||||
@@ -11,7 +11,7 @@ import (
|
||||
"github.com/stretchr/testify/assert"
|
||||
"github.com/zeromicro/go-zero/core/discov"
|
||||
"github.com/zeromicro/go-zero/core/logx"
|
||||
"github.com/zeromicro/go-zero/zrpc/internal/mock"
|
||||
"github.com/zeromicro/go-zero/internal/mock"
|
||||
"google.golang.org/grpc"
|
||||
"google.golang.org/grpc/codes"
|
||||
"google.golang.org/grpc/credentials/insecure"
|
||||
|
||||
@@ -8,7 +8,7 @@ import (
|
||||
"github.com/stretchr/testify/assert"
|
||||
"github.com/zeromicro/go-zero/core/proc"
|
||||
"github.com/zeromicro/go-zero/core/stat"
|
||||
"github.com/zeromicro/go-zero/zrpc/internal/mock"
|
||||
"github.com/zeromicro/go-zero/internal/mock"
|
||||
"google.golang.org/grpc"
|
||||
)
|
||||
|
||||
|
||||
@@ -6,7 +6,7 @@ import (
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
"github.com/zeromicro/go-zero/zrpc/internal/mock"
|
||||
"github.com/zeromicro/go-zero/internal/mock"
|
||||
"google.golang.org/grpc"
|
||||
"google.golang.org/grpc/codes"
|
||||
"google.golang.org/grpc/credentials/insecure"
|
||||
|
||||
Reference in New Issue
Block a user