|
|
|
package selector
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/go-kratos/kratos/v2/middleware"
|
|
|
|
"github.com/go-kratos/kratos/v2/transport"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
transporter func(ctx context.Context) (transport.Transporter, bool)
|
|
|
|
MatchFunc func(ctx context.Context, operation string) bool
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// serverTransporter is get server transport.Transporter from ctx
|
|
|
|
serverTransporter transporter = func(ctx context.Context) (transport.Transporter, bool) {
|
|
|
|
return transport.FromServerContext(ctx)
|
|
|
|
}
|
|
|
|
// clientTransporter is get client transport.Transporter from ctx
|
|
|
|
clientTransporter transporter = func(ctx context.Context) (transport.Transporter, bool) {
|
|
|
|
return transport.FromClientContext(ctx)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// Builder is a selector builder
|
|
|
|
type Builder struct {
|
|
|
|
client bool
|
|
|
|
|
|
|
|
prefix []string
|
|
|
|
regex []string
|
|
|
|
path []string
|
|
|
|
match MatchFunc
|
|
|
|
|
|
|
|
ms []middleware.Middleware
|
|
|
|
}
|
|
|
|
|
|
|
|
// Server selector middleware
|
|
|
|
func Server(ms ...middleware.Middleware) *Builder {
|
|
|
|
return &Builder{ms: ms}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Client selector middleware
|
|
|
|
func Client(ms ...middleware.Middleware) *Builder {
|
|
|
|
return &Builder{client: true, ms: ms}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prefix is with Builder's prefix
|
|
|
|
func (b *Builder) Prefix(prefix ...string) *Builder {
|
|
|
|
b.prefix = prefix
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
// Regex is with Builder's regex
|
|
|
|
func (b *Builder) Regex(regex ...string) *Builder {
|
|
|
|
b.regex = regex
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
// Path is with Builder's path
|
|
|
|
func (b *Builder) Path(path ...string) *Builder {
|
|
|
|
b.path = path
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
// Match is with Builder's match
|
|
|
|
func (b *Builder) Match(fn MatchFunc) *Builder {
|
|
|
|
b.match = fn
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build is Builder's Build, for example: Server().Path(m1,m2).Build()
|
|
|
|
func (b *Builder) Build() middleware.Middleware {
|
|
|
|
var transporter func(ctx context.Context) (transport.Transporter, bool)
|
|
|
|
if b.client {
|
|
|
|
transporter = clientTransporter
|
|
|
|
} else {
|
|
|
|
transporter = serverTransporter
|
|
|
|
}
|
|
|
|
return selector(transporter, b.matchs, b.ms...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// matchs is match operation compliance Builder
|
|
|
|
func (b *Builder) matchs(ctx context.Context, transporter transporter) bool {
|
|
|
|
info, ok := transporter(ctx)
|
|
|
|
if !ok {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
operation := info.Operation()
|
|
|
|
for _, prefix := range b.prefix {
|
|
|
|
if prefixMatch(prefix, operation) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, regex := range b.regex {
|
|
|
|
if regexMatch(regex, operation) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, path := range b.path {
|
|
|
|
if pathMatch(path, operation) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if b.match != nil {
|
|
|
|
if b.match(ctx, operation) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// selector middleware
|
|
|
|
func selector(transporter transporter, match func(context.Context, transporter) bool, ms ...middleware.Middleware) middleware.Middleware {
|
|
|
|
return func(handler middleware.Handler) middleware.Handler {
|
|
|
|
return func(ctx context.Context, req interface{}) (reply interface{}, err error) {
|
|
|
|
if !match(ctx, transporter) {
|
|
|
|
return handler(ctx, req)
|
|
|
|
}
|
|
|
|
return middleware.Chain(ms...)(handler)(ctx, req)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func pathMatch(path string, operation string) bool {
|
|
|
|
return path == operation
|
|
|
|
}
|
|
|
|
|
|
|
|
func prefixMatch(prefix string, operation string) bool {
|
|
|
|
return strings.HasPrefix(operation, prefix)
|
|
|
|
}
|
|
|
|
|
|
|
|
func regexMatch(regex string, operation string) bool {
|
|
|
|
r, err := regexp.Compile(regex)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return r.FindString(operation) == operation
|
|
|
|
}
|