2016-05-03 23:06:19 +02:00
|
|
|
package selector
|
|
|
|
|
|
|
|
import (
|
2022-10-20 13:00:50 +02:00
|
|
|
"sync"
|
2018-12-29 17:44:51 +02:00
|
|
|
"time"
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
2024-06-04 22:40:43 +02:00
|
|
|
"go-micro.dev/v5/registry"
|
|
|
|
"go-micro.dev/v5/registry/cache"
|
2016-05-03 23:06:19 +02:00
|
|
|
)
|
|
|
|
|
2018-12-29 17:44:51 +02:00
|
|
|
type registrySelector struct {
|
2019-02-13 11:47:31 +02:00
|
|
|
so Options
|
2019-05-31 17:00:44 +02:00
|
|
|
rc cache.Cache
|
2022-10-20 13:00:50 +02:00
|
|
|
mu sync.RWMutex
|
2018-12-29 17:44:51 +02:00
|
|
|
}
|
|
|
|
|
2019-05-31 17:00:44 +02:00
|
|
|
func (c *registrySelector) newCache() cache.Cache {
|
2021-05-17 09:16:52 +02:00
|
|
|
opts := make([]cache.Option, 0, 1)
|
2022-10-20 13:00:50 +02:00
|
|
|
|
2019-02-13 11:47:31 +02:00
|
|
|
if c.so.Context != nil {
|
|
|
|
if t, ok := c.so.Context.Value("selector_ttl").(time.Duration); ok {
|
2021-05-17 09:16:52 +02:00
|
|
|
opts = append(opts, cache.WithTTL(t))
|
2018-12-29 17:44:51 +02:00
|
|
|
}
|
|
|
|
}
|
2022-10-20 13:00:50 +02:00
|
|
|
|
2021-05-17 09:16:52 +02:00
|
|
|
return cache.New(c.so.Registry, opts...)
|
2018-12-29 17:44:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *registrySelector) Init(opts ...Option) error {
|
2022-10-20 13:00:50 +02:00
|
|
|
c.mu.Lock()
|
|
|
|
defer c.mu.Unlock()
|
|
|
|
|
2016-05-03 23:06:19 +02:00
|
|
|
for _, o := range opts {
|
2018-12-29 17:44:51 +02:00
|
|
|
o(&c.so)
|
2016-05-03 23:06:19 +02:00
|
|
|
}
|
2018-12-29 17:44:51 +02:00
|
|
|
|
2019-02-13 11:47:31 +02:00
|
|
|
c.rc.Stop()
|
2019-05-31 17:00:44 +02:00
|
|
|
c.rc = c.newCache()
|
2018-12-29 17:44:51 +02:00
|
|
|
|
2016-05-03 23:06:19 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-12-29 17:44:51 +02:00
|
|
|
func (c *registrySelector) Options() Options {
|
|
|
|
return c.so
|
2016-05-03 23:06:19 +02:00
|
|
|
}
|
|
|
|
|
2018-12-29 17:44:51 +02:00
|
|
|
func (c *registrySelector) Select(service string, opts ...SelectOption) (Next, error) {
|
2022-10-20 13:00:50 +02:00
|
|
|
c.mu.RLock()
|
|
|
|
defer c.mu.RUnlock()
|
|
|
|
|
2016-05-03 23:06:19 +02:00
|
|
|
sopts := SelectOptions{
|
2018-12-29 17:44:51 +02:00
|
|
|
Strategy: c.so.Strategy,
|
2016-05-03 23:06:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, opt := range opts {
|
|
|
|
opt(&sopts)
|
|
|
|
}
|
|
|
|
|
|
|
|
// get the service
|
2018-12-29 17:44:51 +02:00
|
|
|
// try the cache first
|
|
|
|
// if that fails go directly to the registry
|
2019-02-13 11:47:31 +02:00
|
|
|
services, err := c.rc.GetService(service)
|
2016-05-03 23:06:19 +02:00
|
|
|
if err != nil {
|
2022-10-20 13:00:50 +02:00
|
|
|
if errors.Is(err, registry.ErrNotFound) {
|
2019-08-11 04:14:41 +02:00
|
|
|
return nil, ErrNotFound
|
|
|
|
}
|
2022-10-20 13:00:50 +02:00
|
|
|
|
2016-05-03 23:06:19 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply the filters
|
|
|
|
for _, filter := range sopts.Filters {
|
|
|
|
services = filter(services)
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there's nothing left, return
|
|
|
|
if len(services) == 0 {
|
2016-05-07 01:04:08 +02:00
|
|
|
return nil, ErrNoneAvailable
|
2016-05-03 23:06:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return sopts.Strategy(services), nil
|
|
|
|
}
|
|
|
|
|
2018-12-29 17:44:51 +02:00
|
|
|
func (c *registrySelector) Mark(service string, node *registry.Node, err error) {
|
2016-05-03 23:06:19 +02:00
|
|
|
}
|
|
|
|
|
2018-12-29 17:44:51 +02:00
|
|
|
func (c *registrySelector) Reset(service string) {
|
2016-05-03 23:06:19 +02:00
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// Close stops the watcher and destroys the cache.
|
2018-12-29 17:44:51 +02:00
|
|
|
func (c *registrySelector) Close() error {
|
2019-02-13 11:47:31 +02:00
|
|
|
c.rc.Stop()
|
2018-12-29 17:44:51 +02:00
|
|
|
|
2016-05-03 23:06:19 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-12-29 17:44:51 +02:00
|
|
|
func (c *registrySelector) String() string {
|
|
|
|
return "registry"
|
2016-05-03 23:06:19 +02:00
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
// NewSelector creates a new default selector.
|
2018-12-29 17:44:51 +02:00
|
|
|
func NewSelector(opts ...Option) Selector {
|
2016-05-03 23:06:19 +02:00
|
|
|
sopts := Options{
|
|
|
|
Strategy: Random,
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, opt := range opts {
|
|
|
|
opt(&sopts)
|
|
|
|
}
|
|
|
|
|
|
|
|
if sopts.Registry == nil {
|
|
|
|
sopts.Registry = registry.DefaultRegistry
|
|
|
|
}
|
|
|
|
|
2019-02-13 11:47:31 +02:00
|
|
|
s := ®istrySelector{
|
|
|
|
so: sopts,
|
2018-12-29 17:44:51 +02:00
|
|
|
}
|
2019-05-31 17:00:44 +02:00
|
|
|
s.rc = s.newCache()
|
2018-12-29 17:44:51 +02:00
|
|
|
|
2019-02-13 11:47:31 +02:00
|
|
|
return s
|
2016-05-03 23:06:19 +02:00
|
|
|
}
|