forked from trpc-group/trpc-go
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtrpc_selector.go
114 lines (99 loc) · 3.39 KB
/
trpc_selector.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
//
//
// Tencent is pleased to support the open source community by making tRPC available.
//
// Copyright (C) 2023 THL A29 Limited, a Tencent company.
// All rights reserved.
//
// If you have downloaded a copy of the tRPC source code from Tencent,
// please note that tRPC source code is licensed under the Apache 2.0 License,
// A copy of the Apache 2.0 License is included in this file.
//
//
package selector
import (
"errors"
"time"
"trpc.group/trpc-go/trpc-go/naming/circuitbreaker"
"trpc.group/trpc-go/trpc-go/naming/discovery"
"trpc.group/trpc-go/trpc-go/naming/loadbalance"
"trpc.group/trpc-go/trpc-go/naming/registry"
"trpc.group/trpc-go/trpc-go/naming/servicerouter"
)
// Errors when route failed.
var (
ErrReportNodeEmpty = errors.New("selector report node empty")
ErrReportMetaDataEmpty = errors.New("selector report metadata empty")
ErrReportNoCircuitBreaker = errors.New("selector report not circuitbreaker")
ErrReportInvalidCircuitBreaker = errors.New("selector report circuitbreaker invalid")
)
// DefaultSelector is the default Selector.
var DefaultSelector Selector = &TrpcSelector{}
// TrpcSelector is the default Selector implementation of tRPC. It automatically combines pluggable
// modules, like, service discovery, load balance and circuit breaker.
type TrpcSelector struct{}
// Select returns an available node by service name.
func (s *TrpcSelector) Select(serviceName string, opt ...Option) (*registry.Node, error) {
if serviceName == "" {
return nil, errors.New("service name empty")
}
opts := &Options{
Discovery: discovery.DefaultDiscovery,
DiscoveryOptions: make([]discovery.Option, 0, defaultDiscoveryOptionsSize),
LoadBalancer: loadbalance.DefaultLoadBalancer,
LoadBalanceOptions: make([]loadbalance.Option, 0, defaultLoadBalanceOptionsSize),
ServiceRouter: servicerouter.DefaultServiceRouter,
ServiceRouterOptions: make([]servicerouter.Option, 0, defaultServiceRouterOptionsSize),
CircuitBreaker: circuitbreaker.DefaultCircuitBreaker,
}
for _, o := range opt {
o(opts)
}
if opts.Discovery == nil {
return nil, errors.New("discovery not exists")
}
list, err := opts.Discovery.List(serviceName, opts.DiscoveryOptions...)
if err != nil {
return nil, err
}
if opts.ServiceRouter == nil {
return nil, errors.New("servicerouter not exists")
}
list, err = opts.ServiceRouter.Filter(serviceName, list, opts.ServiceRouterOptions...)
if err != nil {
return nil, err
}
if opts.LoadBalancer == nil {
return nil, errors.New("loadbalancer not exists")
}
if opts.CircuitBreaker == nil {
return nil, errors.New("circuitbreaker not exists")
}
node, err := opts.LoadBalancer.Select(serviceName, list, opts.LoadBalanceOptions...)
if err != nil {
return nil, err
}
if len(node.Metadata) == 0 {
node.Metadata = make(map[string]interface{})
}
node.Metadata["circuitbreaker"] = opts.CircuitBreaker
return node, nil
}
// Report reports result.
func (s *TrpcSelector) Report(node *registry.Node, cost time.Duration, err error) error {
if node == nil {
return ErrReportNodeEmpty
}
if node.Metadata == nil {
return ErrReportMetaDataEmpty
}
breaker, ok := node.Metadata["circuitbreaker"]
if !ok {
return ErrReportNoCircuitBreaker
}
circuitbreaker, ok := breaker.(circuitbreaker.CircuitBreaker)
if !ok {
return ErrReportInvalidCircuitBreaker
}
return circuitbreaker.Report(node, cost, err)
}