forked from trpc-group/trpc-go
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconfig.go
783 lines (703 loc) · 24.3 KB
/
config.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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
//
//
// 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 trpc
import (
"errors"
"flag"
"fmt"
"net"
"os"
"strconv"
"strings"
"sync/atomic"
"time"
yaml "gopkg.in/yaml.v3"
"trpc.group/trpc-go/trpc-go/internal/expandenv"
trpcpb "trpc.group/trpc/trpc-protocol/pb/go/trpc"
"trpc.group/trpc-go/trpc-go/client"
"trpc.group/trpc-go/trpc-go/codec"
"trpc.group/trpc-go/trpc-go/errs"
"trpc.group/trpc-go/trpc-go/internal/rand"
"trpc.group/trpc-go/trpc-go/plugin"
"trpc.group/trpc-go/trpc-go/rpcz"
)
// ServerConfigPath is the file path of trpc server config file.
// By default, it's ./trpc_go.yaml. It can be set by the flag -conf.
var ServerConfigPath = defaultConfigPath
const (
defaultConfigPath = "./trpc_go.yaml"
defaultIdleTimeout = 60000 // in ms
)
// serverConfigPath returns the file path of trpc server config file.
// With the highest priority: modifying value of ServerConfigPath.
// With second-highest priority: setting by flag --conf or -conf.
// With third-highest priority: using ./trpc_go.yaml as default path.
func serverConfigPath() string {
if ServerConfigPath == defaultConfigPath && !flag.Parsed() {
flag.StringVar(&ServerConfigPath, "conf", defaultConfigPath, "server config path")
flag.Parse()
}
return ServerConfigPath
}
// Config is the configuration for trpc, which can be divided into 4 parts:
// 1. Global config.
// 2. Server config.
// 3. Client config.
// 4. Plugins config.
type Config struct {
Global struct {
Namespace string `yaml:"namespace"` // Namespace for the configuration.
EnvName string `yaml:"env_name"` // Environment name.
ContainerName string `yaml:"container_name"` // Container name.
LocalIP string `yaml:"local_ip"` // Local IP address.
EnableSet string `yaml:"enable_set"` // Y/N. Whether to enable Set. Default is N.
// Full set name with the format: [set name].[set region].[set group name].
FullSetName string `yaml:"full_set_name"`
// Size of the read buffer in bytes. <=0 means read buffer disabled. Default value will be used if not set.
ReadBufferSize *int `yaml:"read_buffer_size,omitempty"`
}
Server struct {
App string `yaml:"app"` // Application name.
Server string `yaml:"server"` // Server name.
BinPath string `yaml:"bin_path"` // Binary file path.
DataPath string `yaml:"data_path"` // Data file path.
ConfPath string `yaml:"conf_path"` // Configuration file path.
Admin struct {
IP string `yaml:"ip"` // NIC IP to bind, e.g., 127.0.0.1.
Nic string `yaml:"nic"` // NIC to bind.
Port uint16 `yaml:"port"` // Port to bind, e.g., 80. Default is 9028.
ReadTimeout int `yaml:"read_timeout"` // Read timeout in milliseconds for admin HTTP server.
WriteTimeout int `yaml:"write_timeout"` // Write timeout in milliseconds for admin HTTP server.
EnableTLS bool `yaml:"enable_tls"` // Whether to enable TLS.
RPCZ *RPCZConfig `yaml:"rpcz"` // RPCZ configuration.
}
Transport string `yaml:"transport"` // Transport type.
Network string `yaml:"network"` // Network type for all services. Default is tcp.
Protocol string `yaml:"protocol"` // Protocol type for all services. Default is trpc.
Filter []string `yaml:"filter"` // Filters for all services.
StreamFilter []string `yaml:"stream_filter"` // Stream filters for all services.
Service []*ServiceConfig `yaml:"service"` // Configuration for each individual service.
// Minimum waiting time in milliseconds when closing the server to wait for deregister finish.
CloseWaitTime int `yaml:"close_wait_time"`
// Maximum waiting time in milliseconds when closing the server to wait for requests to finish.
MaxCloseWaitTime int `yaml:"max_close_wait_time"`
Timeout int `yaml:"timeout"` // Timeout in milliseconds.
}
Client ClientConfig `yaml:"client"` // Client configuration.
Plugins plugin.Config `yaml:"plugins"` // Plugins configuration.
}
// RPCZConfig is the config for rpcz.GlobalRPCZ, and is a field of Config.Admin.
type RPCZConfig struct {
Fraction float64 `yaml:"fraction"`
Capacity uint32 `yaml:"capacity"`
RecordWhen *RecordWhenConfig `yaml:"record_when"`
}
func (c *RPCZConfig) generate() *rpcz.Config {
if c.Capacity == 0 {
const defaultCapacity uint32 = 10000
c.Capacity = defaultCapacity
}
config := &rpcz.Config{
Fraction: c.Fraction,
Capacity: c.Capacity,
}
if c.RecordWhen != nil {
config.ShouldRecord = c.RecordWhen.shouldRecord()
}
return config
}
type node interface {
yaml.Unmarshaler
shouldRecorder
}
type nodeKind string
const (
kindAND nodeKind = "AND"
kindOR nodeKind = "OR"
kindNOT nodeKind = "NOT"
kindMinDuration nodeKind = "__min_duration"
kindMinRequestSize nodeKind = "__min_request_size"
kindMinResponseSize nodeKind = "__min_response_size"
kindRPCName nodeKind = "__rpc_name"
kindErrorCodes nodeKind = "__error_code"
kindErrorMessages nodeKind = "__error_message"
kindSamplingFraction nodeKind = "__sampling_fraction"
kindHasAttributes nodeKind = "__has_attribute"
)
var kindToNode = map[nodeKind]func() node{
kindAND: func() node { return &andNode{} },
kindOR: func() node { return &orNode{} },
kindNOT: func() node { return ¬Node{} },
kindMinDuration: func() node { return &minMinDurationNode{} },
kindMinRequestSize: func() node { return &minRequestSizeNode{} },
kindMinResponseSize: func() node { return &minResponseSizeNode{} },
kindRPCName: func() node { return &rpcNameNode{} },
kindErrorCodes: func() node { return &errorCodeNode{} },
kindErrorMessages: func() node { return &errorMessageNode{} },
kindSamplingFraction: func() node { return &samplingFractionNode{} },
kindHasAttributes: func() node { return &hasAttributeNode{} },
}
var kinds = func() []nodeKind {
ks := make([]nodeKind, 0, len(kindToNode))
for k := range kindToNode {
ks = append(ks, k)
}
return ks
}()
func generate(k nodeKind) (node, error) {
if fn, ok := kindToNode[k]; ok {
return fn(), nil
}
return nil, fmt.Errorf("unknown node: %s, valid node must be one of %v", k, kinds)
}
type shouldRecorder interface {
shouldRecord() rpcz.ShouldRecord
}
type recorder struct {
rpcz.ShouldRecord
}
func (n *recorder) shouldRecord() rpcz.ShouldRecord {
return n.ShouldRecord
}
// RecordWhenConfig stores the RecordWhenConfig field of Config.
type RecordWhenConfig struct {
andNode
}
// UnmarshalYAML customizes RecordWhenConfig's behavior when being unmarshalled from a YAML document.
func (c *RecordWhenConfig) UnmarshalYAML(node *yaml.Node) error {
if err := node.Decode(&c.andNode); err != nil {
return fmt.Errorf("decoding RecordWhenConfig's andNode: %w", err)
}
return nil
}
type nodeList struct {
shouldRecords []rpcz.ShouldRecord
}
func (nl *nodeList) UnmarshalYAML(node *yaml.Node) error {
var nodes []map[nodeKind]yaml.Node
if err := node.Decode(&nodes); err != nil {
return fmt.Errorf("decoding []map[nodeKind]yaml.Node: %w", err)
}
nl.shouldRecords = make([]rpcz.ShouldRecord, 0, len(nodes))
for _, n := range nodes {
if size := len(n); size != 1 {
return fmt.Errorf("%v node has %d element currently, "+
"but the valid number of elements can only be 1", n, size)
}
for nodeKind, value := range n {
if valueEmpty(value) {
return fmt.Errorf("decoding %s node: value is empty", nodeKind)
}
node, err := generate(nodeKind)
if err != nil {
return fmt.Errorf("generating %s node: %w", nodeKind, err)
}
if err := value.Decode(node); err != nil {
return fmt.Errorf("decoding %s node: %w", nodeKind, err)
}
nl.shouldRecords = append(nl.shouldRecords, node.shouldRecord())
}
}
return nil
}
func valueEmpty(node yaml.Node) bool {
return len(node.Content) == 0 && len(node.Value) == 0
}
type andNode struct {
recorder
}
func (n *andNode) UnmarshalYAML(node *yaml.Node) error {
nl := &nodeList{}
if err := node.Decode(nl); err != nil {
return fmt.Errorf("decoding andNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
if len(nl.shouldRecords) == 0 {
return false
}
for _, r := range nl.shouldRecords {
if !r(s) {
return false
}
}
return true
}
return nil
}
type orNode struct {
recorder
}
func (n *orNode) UnmarshalYAML(node *yaml.Node) error {
nl := &nodeList{}
if err := node.Decode(nl); err != nil {
return fmt.Errorf("decoding orNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
for _, r := range nl.shouldRecords {
if r(s) {
return true
}
}
return false
}
return nil
}
type notNode struct {
recorder
}
func (n *notNode) UnmarshalYAML(node *yaml.Node) error {
var not map[nodeKind]yaml.Node
if err := node.Decode(¬); err != nil {
return fmt.Errorf("decoding notNode: %w", err)
}
const numInvalidChildren = 1
if n := len(not); n != numInvalidChildren {
return fmt.Errorf("NOT node has %d child node currently, "+
"but the valid number of child node can only be %d", n, numInvalidChildren)
}
for nodeKind, value := range not {
node, err := generate(nodeKind)
if err != nil {
return fmt.Errorf("generating %s node: %w", nodeKind, err)
}
if err := value.Decode(node); err != nil {
return fmt.Errorf("decoding %s node: %w", nodeKind, err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
return !node.shouldRecord()(s)
}
}
return nil
}
type hasAttributeNode struct {
recorder
}
func (n *hasAttributeNode) UnmarshalYAML(node *yaml.Node) error {
var attribute string
if err := node.Decode(&attribute); err != nil {
return fmt.Errorf("decoding hasAttributeNode: %w", err)
}
key, value, err := parse(attribute)
if err != nil {
return fmt.Errorf("parsing attribute %s : %w", attribute, err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
v, ok := s.Attribute(key)
return ok && strings.Contains(fmt.Sprintf("%s", v), value)
}
return nil
}
var errInvalidAttribute = errors.New("invalid attribute form [ valid attribute form: (key, value), " +
"only one space character after comma character, and key can't contain comma(',') character ]")
func parse(attribute string) (key string, value string, err error) {
if len(attribute) == 0 || attribute[0] != '(' {
return "", "", errInvalidAttribute
}
attribute = attribute[1:]
if n := len(attribute); n == 0 || attribute[n-1] != ')' {
return "", "", errInvalidAttribute
}
attribute = attribute[:len(attribute)-1]
const delimiter = ", "
i := strings.Index(attribute, delimiter)
if i == -1 {
return "", "", errInvalidAttribute
}
return attribute[:i], attribute[i+len(delimiter):], nil
}
type minRequestSizeNode struct {
recorder
}
func (n *minRequestSizeNode) UnmarshalYAML(node *yaml.Node) error {
var minRequestSize int
if err := node.Decode(&minRequestSize); err != nil {
return fmt.Errorf("decoding minRequestSizeNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
size, ok := s.Attribute(rpcz.TRPCAttributeRequestSize)
if !ok {
return false
}
if size, ok := size.(int); !ok || size < minRequestSize {
return false
}
return true
}
return nil
}
type minResponseSizeNode struct {
recorder
}
func (n *minResponseSizeNode) UnmarshalYAML(node *yaml.Node) error {
var minResponseSize int
if err := node.Decode(&minResponseSize); err != nil {
return fmt.Errorf("decoding minResponseSizeNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
responseSize, ok := s.Attribute(rpcz.TRPCAttributeResponseSize)
if !ok {
return false
}
if size, ok := responseSize.(int); !ok || size < minResponseSize {
return false
}
return true
}
return nil
}
type minMinDurationNode struct {
recorder
}
func (n *minMinDurationNode) UnmarshalYAML(node *yaml.Node) error {
var dur time.Duration
if err := node.Decode(&dur); err != nil {
return fmt.Errorf("decoding minMinDurationNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
if dur == 0 {
return true
}
et := s.EndTime()
return et.IsZero() || et.Sub(s.StartTime()) >= dur
}
return nil
}
type rpcNameNode struct {
recorder
}
func (n *rpcNameNode) UnmarshalYAML(node *yaml.Node) error {
var rpcName string
if err := node.Decode(&rpcName); err != nil {
return fmt.Errorf("decoding rpcNameNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
name, ok := s.Attribute(rpcz.TRPCAttributeRPCName)
if !ok {
return false
}
if name, ok := name.(string); !ok || !strings.Contains(name, rpcName) {
return false
}
return true
}
return nil
}
type samplingFractionNode struct {
recorder
}
var safeRand = rand.NewSafeRand(time.Now().UnixNano())
func (n *samplingFractionNode) UnmarshalYAML(node *yaml.Node) error {
var f float64
if err := node.Decode(&f); err != nil {
return fmt.Errorf("decoding samplingFractionNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
return f > safeRand.Float64()
}
return nil
}
type errorCodeNode struct {
recorder
}
func (n *errorCodeNode) UnmarshalYAML(node *yaml.Node) error {
var code trpcpb.TrpcRetCode
if err := node.Decode(&code); err != nil {
return fmt.Errorf("decoding errorCodeNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
err, ok := extractError(s)
if !ok {
return false
}
c := errs.Code(err)
return c == code
}
return nil
}
type errorMessageNode struct {
recorder
}
func (n *errorMessageNode) UnmarshalYAML(node *yaml.Node) error {
var message string
if err := node.Decode(&message); err != nil {
return fmt.Errorf("decoding errorMessageNode: %w", err)
}
n.ShouldRecord = func(s rpcz.Span) bool {
err, ok := extractError(s)
if !ok {
return false
}
return strings.Contains(message, errs.Msg(err))
}
return nil
}
func extractError(span rpcz.Span) (error, bool) {
err, ok := span.Attribute(rpcz.TRPCAttributeError)
if !ok {
return nil, false
}
e, ok := err.(error)
return e, ok
}
// ServiceConfig is a configuration for a single service. A server process might have multiple services.
type ServiceConfig struct {
// Disable request timeout inherited from upstream service.
DisableRequestTimeout bool `yaml:"disable_request_timeout"`
IP string `yaml:"ip"` // IP address to listen to.
// Service name in the format: trpc.app.server.service. Used for naming the service.
Name string `yaml:"name"`
Nic string `yaml:"nic"` // Network Interface Card (NIC) to listen to. No need to configure.
Port uint16 `yaml:"port"` // Port to listen to.
// Address to listen to. If set, ipport will be ignored. Otherwise, ipport will be used.
Address string `yaml:"address"`
Network string `yaml:"network"` // Network type like tcp/udp.
Protocol string `yaml:"protocol"` // Protocol type like trpc.
// Longest time in milliseconds for a handler to handle a request.
Timeout int `yaml:"timeout"`
// Maximum idle time in milliseconds for a server connection. Default is 1 minute.
Idletime int `yaml:"idletime"`
DisableKeepAlives bool `yaml:"disable_keep_alives"` // Disables keep-alives.
Registry string `yaml:"registry"` // Registry to use, e.g., polaris.
Filter []string `yaml:"filter"` // Filters for the service.
StreamFilter []string `yaml:"stream_filter"` // Stream filters for the service.
TLSKey string `yaml:"tls_key"` // Server TLS key.
TLSCert string `yaml:"tls_cert"` // Server TLS certificate.
CACert string `yaml:"ca_cert"` // CA certificate to validate client certificate.
ServerAsync *bool `yaml:"server_async,omitempty"` // Whether to enable server asynchronous mode.
// MaxRoutines is the maximum number of goroutines for server asynchronous mode.
// Requests exceeding MaxRoutines will be queued. Prolonged overages may lead to OOM!
// MaxRoutines is not the solution to alleviate server overloading.
MaxRoutines int `yaml:"max_routines"`
Writev *bool `yaml:"writev,omitempty"` // Whether to enable writev.
Transport string `yaml:"transport"` // Transport type.
}
// ClientConfig is the configuration for the client to request backends.
type ClientConfig struct {
Network string `yaml:"network"` // Network for all backends. Default is tcp.
Protocol string `yaml:"protocol"` // Protocol for all backends. Default is trpc.
Filter []string `yaml:"filter"` // Filters for all backends.
StreamFilter []string `yaml:"stream_filter"` // Stream filters for all backends.
Namespace string `yaml:"namespace"` // Namespace for all backends.
Transport string `yaml:"transport"` // Transport type.
Timeout int `yaml:"timeout"` // Timeout in milliseconds.
Discovery string `yaml:"discovery"` // Discovery mechanism.
ServiceRouter string `yaml:"servicerouter"` // Service router.
Loadbalance string `yaml:"loadbalance"` // Load balancing algorithm.
Circuitbreaker string `yaml:"circuitbreaker"` // Circuit breaker configuration.
Service []*client.BackendConfig `yaml:"service"` // Configuration for each individual backend.
}
// trpc server config, set after the framework setup and the yaml config file is parsed.
var globalConfig atomic.Value
func init() {
globalConfig.Store(defaultConfig())
}
func defaultConfig() *Config {
cfg := &Config{}
cfg.Global.EnableSet = "N"
cfg.Server.Network = "tcp"
cfg.Server.Protocol = "trpc"
cfg.Client.Network = "tcp"
cfg.Client.Protocol = "trpc"
return cfg
}
// GlobalConfig returns the global Config.
func GlobalConfig() *Config {
return globalConfig.Load().(*Config)
}
// SetGlobalConfig set the global Config.
func SetGlobalConfig(cfg *Config) {
globalConfig.Store(cfg)
}
// LoadGlobalConfig loads a Config from the config file path and sets it as the global Config.
func LoadGlobalConfig(configPath string) error {
cfg, err := LoadConfig(configPath)
if err != nil {
return err
}
SetGlobalConfig(cfg)
return nil
}
// LoadConfig loads a Config from the config file path.
func LoadConfig(configPath string) (*Config, error) {
cfg, err := parseConfigFromFile(configPath)
if err != nil {
return nil, err
}
if err := RepairConfig(cfg); err != nil {
return nil, err
}
return cfg, nil
}
func parseConfigFromFile(configPath string) (*Config, error) {
buf, err := os.ReadFile(configPath)
if err != nil {
return nil, err
}
cfg := defaultConfig()
if err := yaml.Unmarshal(expandenv.ExpandEnv(buf), cfg); err != nil {
return nil, err
}
return cfg, nil
}
// Setup registers client config and setups plugins according to the Config.
func Setup(cfg *Config) error {
if _, err := SetupPlugins(cfg.Plugins); err != nil {
return err
}
if err := SetupClients(&cfg.Client); err != nil {
return err
}
return nil
}
// SetupPlugins sets up all plugins and returns a function to close them.
func SetupPlugins(cfg plugin.Config) (func() error, error) {
if cfg == nil {
return func() error { return nil }, nil
}
return cfg.SetupClosables()
}
// SetupClients sets up all backends and the wildcard.
func SetupClients(cfg *ClientConfig) error {
for _, backendCfg := range cfg.Service {
if err := client.RegisterClientConfig(backendCfg.Callee, backendCfg); err != nil {
return err
}
}
// * represents general config for all backends
if _, ok := client.DefaultClientConfig()["*"]; !ok { // should not be covered if already registered by some plugins
if err := client.RegisterClientConfig("*", &client.BackendConfig{
Network: cfg.Network,
Protocol: cfg.Protocol,
Namespace: cfg.Namespace,
Transport: cfg.Transport,
Timeout: cfg.Timeout,
Filter: cfg.Filter,
StreamFilter: cfg.StreamFilter,
Discovery: cfg.Discovery,
ServiceRouter: cfg.ServiceRouter,
Loadbalance: cfg.Loadbalance,
Circuitbreaker: cfg.Circuitbreaker,
}); err != nil {
return err
}
}
return nil
}
// RepairConfig repairs the Config by filling in some fields with default values.
func RepairConfig(cfg *Config) error {
// nic -> ip
if err := repairServiceIPWithNic(cfg); err != nil {
return err
}
// set default read buffer size
if cfg.Global.ReadBufferSize == nil {
readerSize := codec.DefaultReaderSize
cfg.Global.ReadBufferSize = &readerSize
}
codec.SetReaderSize(*cfg.Global.ReadBufferSize)
// nic -> ip
if err := repairServiceIPWithNic(cfg); err != nil {
return err
}
// Set empty ip to "0.0.0.0" to prevent malformed key matching
// for passed listeners during hot restart.
const defaultIP = "0.0.0.0"
setDefault(&cfg.Global.LocalIP, defaultIP)
setDefault(&cfg.Server.Admin.IP, cfg.Global.LocalIP)
// protocol network ip empty
for _, serviceCfg := range cfg.Server.Service {
setDefault(&serviceCfg.Protocol, cfg.Server.Protocol)
setDefault(&serviceCfg.Network, cfg.Server.Network)
setDefault(&serviceCfg.IP, cfg.Global.LocalIP)
setDefault(&serviceCfg.Transport, cfg.Server.Transport)
setDefault(&serviceCfg.Address, net.JoinHostPort(serviceCfg.IP, strconv.Itoa(int(serviceCfg.Port))))
// server async mode by default
if serviceCfg.ServerAsync == nil {
enableServerAsync := true
serviceCfg.ServerAsync = &enableServerAsync
}
// writev disabled by default
if serviceCfg.Writev == nil {
enableWritev := false
serviceCfg.Writev = &enableWritev
}
if serviceCfg.Timeout == 0 {
serviceCfg.Timeout = cfg.Server.Timeout
}
if serviceCfg.Idletime == 0 {
serviceCfg.Idletime = defaultIdleTimeout
if serviceCfg.Timeout > defaultIdleTimeout {
serviceCfg.Idletime = serviceCfg.Timeout
}
}
}
setDefault(&cfg.Client.Namespace, cfg.Global.Namespace)
for _, backendCfg := range cfg.Client.Service {
repairClientConfig(backendCfg, &cfg.Client)
}
return nil
}
// repairServiceIPWithNic repairs the Config when service ip is empty according to the nic.
func repairServiceIPWithNic(cfg *Config) error {
for index, item := range cfg.Server.Service {
if item.IP == "" {
ip := getIP(item.Nic)
if ip == "" && item.Nic != "" {
return fmt.Errorf("can't find service IP by the NIC: %s", item.Nic)
}
cfg.Server.Service[index].IP = ip
}
setDefault(&cfg.Global.LocalIP, item.IP)
}
if cfg.Server.Admin.IP == "" {
ip := getIP(cfg.Server.Admin.Nic)
if ip == "" && cfg.Server.Admin.Nic != "" {
return fmt.Errorf("can't find admin IP by the NIC: %s", cfg.Server.Admin.Nic)
}
cfg.Server.Admin.IP = ip
}
return nil
}
func repairClientConfig(backendCfg *client.BackendConfig, clientCfg *ClientConfig) {
// service name in proto file will be used as key for backend config by default
// generally, service name in proto file is the same as the backend service name.
// therefore, no need to config backend service name
setDefault(&backendCfg.Callee, backendCfg.ServiceName)
setDefault(&backendCfg.ServiceName, backendCfg.Callee)
setDefault(&backendCfg.Namespace, clientCfg.Namespace)
setDefault(&backendCfg.Network, clientCfg.Network)
setDefault(&backendCfg.Protocol, clientCfg.Protocol)
setDefault(&backendCfg.Transport, clientCfg.Transport)
if backendCfg.Target == "" {
setDefault(&backendCfg.Discovery, clientCfg.Discovery)
setDefault(&backendCfg.ServiceRouter, clientCfg.ServiceRouter)
setDefault(&backendCfg.Loadbalance, clientCfg.Loadbalance)
setDefault(&backendCfg.Circuitbreaker, clientCfg.Circuitbreaker)
}
if backendCfg.Timeout == 0 {
backendCfg.Timeout = clientCfg.Timeout
}
// Global filter is at front and is deduplicated.
backendCfg.Filter = deduplicate(clientCfg.Filter, backendCfg.Filter)
backendCfg.StreamFilter = deduplicate(clientCfg.StreamFilter, backendCfg.StreamFilter)
}
// getMillisecond returns time.Duration by the input value in milliseconds.
func getMillisecond(sec int) time.Duration {
return time.Millisecond * time.Duration(sec)
}
// setDefault points dst to def if dst is not nil and points to empty string.
func setDefault(dst *string, def string) {
if dst != nil && *dst == "" {
*dst = def
}
}