You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

72 lines
1.2 KiB

4 years ago
package watcher
import (
"context"
"fmt"
"log"
"time"
"gitoa.ru/go-4devs/config"
)
var (
_ config.Provider = (*Provider)(nil)
_ config.WatchProvider = (*Provider)(nil)
)
func New(duration time.Duration, provider config.Provider, opts ...Option) *Provider {
p := &Provider{
Provider: provider,
ticker: time.NewTicker(duration),
logger: func(_ context.Context, msg string) {
log.Print(msg)
},
}
for _, opt := range opts {
opt(p)
}
return p
}
func WithLogger(l func(context.Context, string)) Option {
return func(p *Provider) {
p.logger = l
}
}
type Option func(*Provider)
type Provider struct {
config.Provider
ticker *time.Ticker
logger func(context.Context, string)
}
func (p *Provider) Watch(ctx context.Context, key config.Key, callback config.WatchCallback) error {
oldVar, err := p.Provider.Read(ctx, key)
if err != nil {
return fmt.Errorf("%s: failed watch variable: %w", p.Provider.Name(), err)
}
go func() {
for {
select {
case <-p.ticker.C:
newVar, err := p.Provider.Read(ctx, key)
if err != nil {
p.logger(ctx, err.Error())
} else if !newVar.IsEquals(oldVar) {
callback(ctx, oldVar, newVar)
oldVar = newVar
}
case <-ctx.Done():
return
}
}
}()
return nil
}