2015-05-26 22:39:48 +01:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2018-03-03 11:53:52 +00:00
|
|
|
"context"
|
2020-02-15 15:10:26 +00:00
|
|
|
"crypto/tls"
|
2020-12-15 11:52:05 +03:00
|
|
|
"net"
|
2019-05-27 21:17:57 +08:00
|
|
|
"sync"
|
2016-01-26 23:32:27 +00:00
|
|
|
"time"
|
|
|
|
|
2020-08-19 17:47:17 +03:00
|
|
|
"github.com/unistack-org/micro/v3/auth"
|
|
|
|
"github.com/unistack-org/micro/v3/broker"
|
|
|
|
"github.com/unistack-org/micro/v3/codec"
|
2020-08-29 17:44:49 +03:00
|
|
|
"github.com/unistack-org/micro/v3/logger"
|
2020-11-18 16:50:41 +03:00
|
|
|
"github.com/unistack-org/micro/v3/metadata"
|
2021-01-22 23:32:33 +03:00
|
|
|
"github.com/unistack-org/micro/v3/meter"
|
2020-10-09 16:15:36 +03:00
|
|
|
"github.com/unistack-org/micro/v3/network/transport"
|
2021-01-29 13:17:32 +03:00
|
|
|
"github.com/unistack-org/micro/v3/register"
|
2020-09-10 00:06:29 +03:00
|
|
|
"github.com/unistack-org/micro/v3/tracer"
|
2015-05-26 22:39:48 +01:00
|
|
|
)
|
|
|
|
|
2020-12-08 00:38:37 +03:00
|
|
|
// Option func
|
2020-11-26 01:13:05 +03:00
|
|
|
type Option func(*Options)
|
|
|
|
|
2020-11-02 13:25:29 +03:00
|
|
|
// Options server struct
|
2015-12-31 18:11:46 +00:00
|
|
|
type Options struct {
|
2020-11-23 16:18:47 +03:00
|
|
|
Codecs map[string]codec.Codec
|
2015-12-31 18:11:46 +00:00
|
|
|
Broker broker.Broker
|
2021-01-29 13:17:32 +03:00
|
|
|
Register register.Register
|
2020-09-10 00:06:29 +03:00
|
|
|
Tracer tracer.Tracer
|
2020-02-03 08:16:02 +00:00
|
|
|
Auth auth.Auth
|
2020-08-29 17:44:49 +03:00
|
|
|
Logger logger.Logger
|
2021-01-22 23:32:33 +03:00
|
|
|
Meter meter.Meter
|
2015-12-31 18:11:46 +00:00
|
|
|
Transport transport.Transport
|
2020-11-18 16:50:41 +03:00
|
|
|
Metadata metadata.Metadata
|
2015-12-31 18:11:46 +00:00
|
|
|
Name string
|
|
|
|
Address string
|
|
|
|
Advertise string
|
|
|
|
Id string
|
2020-06-19 09:24:32 +01:00
|
|
|
Namespace string
|
2015-12-31 18:11:46 +00:00
|
|
|
Version string
|
|
|
|
HdlrWrappers []HandlerWrapper
|
|
|
|
SubWrappers []SubscriberWrapper
|
|
|
|
|
2019-05-13 01:39:42 +03:00
|
|
|
// RegisterCheck runs a check function before registering the service
|
|
|
|
RegisterCheck func(context.Context) error
|
2019-01-08 20:32:47 +00:00
|
|
|
// The register expiry time
|
2016-01-27 12:23:18 +00:00
|
|
|
RegisterTTL time.Duration
|
2019-01-24 13:22:17 +00:00
|
|
|
// The interval on which to register
|
|
|
|
RegisterInterval time.Duration
|
2020-10-16 09:38:57 +03:00
|
|
|
// RegisterAttempts specify how many times try to register
|
|
|
|
RegisterAttempts int
|
|
|
|
// DeegisterAttempts specify how many times try to deregister
|
|
|
|
DeregisterAttempts int
|
2016-01-27 12:23:18 +00:00
|
|
|
|
2019-01-08 20:32:47 +00:00
|
|
|
// The router for requests
|
|
|
|
Router Router
|
|
|
|
|
2020-02-15 15:10:26 +00:00
|
|
|
// TLSConfig specifies tls.Config for secure serving
|
|
|
|
TLSConfig *tls.Config
|
|
|
|
|
2020-09-20 15:48:07 +03:00
|
|
|
Wait *sync.WaitGroup
|
2020-12-15 11:46:26 +03:00
|
|
|
|
2020-12-15 11:52:05 +03:00
|
|
|
// Listener may be passed if already created
|
|
|
|
Listener net.Listener
|
2020-12-15 11:46:26 +03:00
|
|
|
// MaxConn limit connections to server
|
|
|
|
MaxConn int
|
2016-01-06 16:25:12 +00:00
|
|
|
// Other options for implementations of the interface
|
|
|
|
// can be stored in a context
|
|
|
|
Context context.Context
|
2015-12-31 18:11:46 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 13:25:29 +03:00
|
|
|
// NewOptions returns new options struct with default or passed values
|
2020-09-10 00:25:33 +03:00
|
|
|
func NewOptions(opts ...Option) Options {
|
|
|
|
options := Options{
|
2020-09-07 13:38:52 +03:00
|
|
|
Auth: auth.DefaultAuth,
|
2020-11-23 16:18:47 +03:00
|
|
|
Codecs: make(map[string]codec.Codec),
|
2020-09-05 02:11:29 +03:00
|
|
|
Context: context.Background(),
|
2020-11-18 16:50:41 +03:00
|
|
|
Metadata: metadata.New(0),
|
2019-09-23 17:59:34 +01:00
|
|
|
RegisterInterval: DefaultRegisterInterval,
|
|
|
|
RegisterTTL: DefaultRegisterTTL,
|
2020-09-05 02:11:29 +03:00
|
|
|
RegisterCheck: DefaultRegisterCheck,
|
2020-10-16 09:38:57 +03:00
|
|
|
Logger: logger.DefaultLogger,
|
2021-01-22 23:32:33 +03:00
|
|
|
Meter: meter.DefaultMeter,
|
2020-10-16 09:38:57 +03:00
|
|
|
Tracer: tracer.DefaultTracer,
|
2020-09-07 13:38:52 +03:00
|
|
|
Broker: broker.DefaultBroker,
|
2021-01-29 13:17:32 +03:00
|
|
|
Register: register.DefaultRegister,
|
2020-11-05 22:35:05 +03:00
|
|
|
Transport: transport.DefaultTransport,
|
2020-09-05 02:11:29 +03:00
|
|
|
Address: DefaultAddress,
|
|
|
|
Name: DefaultName,
|
|
|
|
Version: DefaultVersion,
|
|
|
|
Id: DefaultId,
|
2020-11-05 22:35:05 +03:00
|
|
|
Namespace: DefaultNamespace,
|
2015-11-25 19:50:05 +00:00
|
|
|
}
|
2020-09-10 00:25:33 +03:00
|
|
|
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
return options
|
2015-05-26 22:39:48 +01:00
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// Name sets the server name option
|
2015-06-03 01:25:37 +01:00
|
|
|
func Name(n string) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Name = n
|
2015-06-03 01:25:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-19 09:24:32 +01:00
|
|
|
// Namespace to register handlers in
|
|
|
|
func Namespace(n string) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.Namespace = n
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// Logger sets the logger option
|
2020-08-29 17:44:49 +03:00
|
|
|
func Logger(l logger.Logger) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.Logger = l
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-22 23:32:33 +03:00
|
|
|
// Meter sets the meter option
|
|
|
|
func Meter(m meter.Meter) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.Meter = m
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-02 13:25:29 +03:00
|
|
|
// Id unique server id
|
2015-06-03 01:25:37 +01:00
|
|
|
func Id(id string) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Id = id
|
2015-06-03 01:25:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-26 00:23:36 +00:00
|
|
|
// Version of the service
|
2015-06-03 01:25:37 +01:00
|
|
|
func Version(v string) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Version = v
|
2015-06-03 01:25:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-26 00:23:36 +00:00
|
|
|
// Address to bind to - host:port
|
2015-06-03 01:25:37 +01:00
|
|
|
func Address(a string) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Address = a
|
2015-06-03 01:25:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// Advertise the address to advertise for discovery - host:port
|
2015-11-11 18:22:04 +00:00
|
|
|
func Advertise(a string) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Advertise = a
|
2015-11-11 18:22:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-26 00:23:36 +00:00
|
|
|
// Broker to use for pub/sub
|
2015-06-12 19:52:27 +01:00
|
|
|
func Broker(b broker.Broker) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Broker = b
|
2015-06-12 19:52:27 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-26 00:23:36 +00:00
|
|
|
// Codec to use to encode/decode requests for a given content type
|
2020-11-23 16:18:47 +03:00
|
|
|
func Codec(contentType string, c codec.Codec) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Codecs[contentType] = c
|
2015-11-25 19:50:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-05 19:37:03 +03:00
|
|
|
// Context specifies a context for the service.
|
|
|
|
// Can be used to signal shutdown of the service
|
|
|
|
// Can be used for extra option values.
|
|
|
|
func Context(ctx context.Context) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.Context = ctx
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-29 13:17:32 +03:00
|
|
|
// Register used for discovery
|
|
|
|
func Register(r register.Register) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
2021-01-29 13:17:32 +03:00
|
|
|
o.Register = r
|
2015-06-03 01:25:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-29 15:45:11 +00:00
|
|
|
// Tracer mechanism for distributed tracking
|
2020-09-10 00:06:29 +03:00
|
|
|
func Tracer(t tracer.Tracer) Option {
|
2020-01-29 15:45:11 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Tracer = t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-03 08:16:02 +00:00
|
|
|
// Auth mechanism for role based access control
|
|
|
|
func Auth(a auth.Auth) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.Auth = a
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-26 00:23:36 +00:00
|
|
|
// Transport mechanism for communication e.g http, rabbitmq, etc
|
2015-06-03 01:25:37 +01:00
|
|
|
func Transport(t transport.Transport) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.Transport = t
|
2015-06-03 01:25:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-26 00:23:36 +00:00
|
|
|
// Metadata associated with the server
|
2020-11-18 16:50:41 +03:00
|
|
|
func Metadata(md metadata.Metadata) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
2020-11-18 16:50:41 +03:00
|
|
|
o.Metadata = metadata.Copy(md)
|
2015-06-03 01:25:37 +01:00
|
|
|
}
|
|
|
|
}
|
2015-12-02 00:47:52 +00:00
|
|
|
|
2021-01-29 13:17:32 +03:00
|
|
|
// RegisterCheck run func before register service
|
2019-05-13 01:39:42 +03:00
|
|
|
func RegisterCheck(fn func(context.Context) error) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.RegisterCheck = fn
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// RegisterTTL registers service with a TTL
|
2016-01-27 12:23:18 +00:00
|
|
|
func RegisterTTL(t time.Duration) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.RegisterTTL = t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// RegisterInterval registers service with at interval
|
2019-01-24 13:22:17 +00:00
|
|
|
func RegisterInterval(t time.Duration) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.RegisterInterval = t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-15 15:10:26 +00:00
|
|
|
// TLSConfig specifies a *tls.Config
|
|
|
|
func TLSConfig(t *tls.Config) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
// set the internal tls
|
|
|
|
o.TLSConfig = t
|
2020-02-17 09:28:48 +00:00
|
|
|
|
|
|
|
// set the default transport if one is not
|
|
|
|
// already set. Required for Init call below.
|
|
|
|
|
2020-02-15 15:10:26 +00:00
|
|
|
// set the transport tls
|
|
|
|
o.Transport.Init(
|
|
|
|
transport.TLSConfig(t),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-08 20:32:47 +00:00
|
|
|
// WithRouter sets the request router
|
|
|
|
func WithRouter(r Router) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.Router = r
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-31 19:47:50 +01:00
|
|
|
// Wait tells the server to wait for requests to finish before exiting
|
2019-05-27 21:17:57 +08:00
|
|
|
// If `wg` is nil, server only wait for completion of rpc handler.
|
|
|
|
// For user need finer grained control, pass a concrete `wg` here, server will
|
|
|
|
// wait against it on stop.
|
|
|
|
func Wait(wg *sync.WaitGroup) Option {
|
2017-05-31 19:47:50 +01:00
|
|
|
return func(o *Options) {
|
2019-05-27 21:17:57 +08:00
|
|
|
if wg == nil {
|
|
|
|
wg = new(sync.WaitGroup)
|
|
|
|
}
|
2020-09-20 15:48:07 +03:00
|
|
|
o.Wait = wg
|
2017-05-31 19:47:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// WrapHandler adds a handler Wrapper to a list of options passed into the server
|
2015-12-02 11:54:36 +00:00
|
|
|
func WrapHandler(w HandlerWrapper) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.HdlrWrappers = append(o.HdlrWrappers, w)
|
2015-12-02 00:47:52 +00:00
|
|
|
}
|
|
|
|
}
|
2015-12-02 11:54:36 +00:00
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// WrapSubscriber adds a subscriber Wrapper to a list of options passed into the server
|
2015-12-02 11:54:36 +00:00
|
|
|
func WrapSubscriber(w SubscriberWrapper) Option {
|
2015-12-31 18:11:46 +00:00
|
|
|
return func(o *Options) {
|
|
|
|
o.SubWrappers = append(o.SubWrappers, w)
|
2015-12-02 11:54:36 +00:00
|
|
|
}
|
|
|
|
}
|
2020-10-16 09:38:57 +03:00
|
|
|
|
2020-12-15 11:52:05 +03:00
|
|
|
// MaxConn specifies maximum number of max simultaneous connections to server
|
|
|
|
func MaxConn(n int) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.MaxConn = n
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Listener specifies the net.Listener to use instead of the default
|
|
|
|
func Listener(l net.Listener) Option {
|
|
|
|
return func(o *Options) {
|
|
|
|
o.Listener = l
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// HandlerOption func
|
2020-10-16 09:38:57 +03:00
|
|
|
type HandlerOption func(*HandlerOptions)
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// HandlerOptions struct
|
2020-10-16 09:38:57 +03:00
|
|
|
type HandlerOptions struct {
|
|
|
|
Internal bool
|
2020-11-18 16:50:41 +03:00
|
|
|
Metadata map[string]metadata.Metadata
|
2020-10-16 09:38:57 +03:00
|
|
|
Context context.Context
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// NewHandlerOptions creates new HandlerOptions
|
2020-10-16 09:38:57 +03:00
|
|
|
func NewHandlerOptions(opts ...HandlerOption) HandlerOptions {
|
|
|
|
options := HandlerOptions{
|
2021-01-10 03:36:23 +03:00
|
|
|
Context: context.Background(),
|
|
|
|
Metadata: make(map[string]metadata.Metadata),
|
2020-10-16 09:38:57 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
return options
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// SubscriberOption func
|
2020-10-16 09:38:57 +03:00
|
|
|
type SubscriberOption func(*SubscriberOptions)
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// SubscriberOptions struct
|
2020-10-16 09:38:57 +03:00
|
|
|
type SubscriberOptions struct {
|
|
|
|
// AutoAck defaults to true. When a handler returns
|
|
|
|
// with a nil error the message is acked.
|
|
|
|
AutoAck bool
|
|
|
|
Queue string
|
|
|
|
Internal bool
|
|
|
|
Context context.Context
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// NewSubscriberOptions create new SubscriberOptions
|
2020-10-16 09:38:57 +03:00
|
|
|
func NewSubscriberOptions(opts ...SubscriberOption) SubscriberOptions {
|
|
|
|
options := SubscriberOptions{
|
|
|
|
AutoAck: true,
|
|
|
|
Context: context.Background(),
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
return options
|
|
|
|
}
|
|
|
|
|
|
|
|
// EndpointMetadata is a Handler option that allows metadata to be added to
|
|
|
|
// individual endpoints.
|
2020-11-18 16:50:41 +03:00
|
|
|
func EndpointMetadata(name string, md metadata.Metadata) HandlerOption {
|
2020-10-16 09:38:57 +03:00
|
|
|
return func(o *HandlerOptions) {
|
2020-11-18 16:50:41 +03:00
|
|
|
o.Metadata[name] = metadata.Copy(md)
|
2020-10-16 09:38:57 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// InternalHandler options specifies that a handler is not advertised
|
2020-10-16 09:38:57 +03:00
|
|
|
// to the discovery system. In the future this may also limit request
|
|
|
|
// to the internal network or authorised user.
|
|
|
|
func InternalHandler(b bool) HandlerOption {
|
|
|
|
return func(o *HandlerOptions) {
|
|
|
|
o.Internal = b
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// InternalSubscriber options specifies that a subscriber is not advertised
|
2020-10-16 09:38:57 +03:00
|
|
|
// to the discovery system.
|
|
|
|
func InternalSubscriber(b bool) SubscriberOption {
|
|
|
|
return func(o *SubscriberOptions) {
|
|
|
|
o.Internal = b
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DisableAutoAck will disable auto acking of messages
|
|
|
|
// after they have been handled.
|
|
|
|
func DisableAutoAck() SubscriberOption {
|
|
|
|
return func(o *SubscriberOptions) {
|
|
|
|
o.AutoAck = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 01:08:23 +03:00
|
|
|
// SubscriberQueue sets the shared queue name distributed messages across subscribers
|
2020-10-16 09:38:57 +03:00
|
|
|
func SubscriberQueue(n string) SubscriberOption {
|
|
|
|
return func(o *SubscriberOptions) {
|
|
|
|
o.Queue = n
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// SubscriberContext set context options to allow broker SubscriberOption passed
|
|
|
|
func SubscriberContext(ctx context.Context) SubscriberOption {
|
|
|
|
return func(o *SubscriberOptions) {
|
|
|
|
o.Context = ctx
|
|
|
|
}
|
|
|
|
}
|