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.3 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.NamedProvider, opts ...Option) *Provider {
2 years ago
prov := &Provider{
NamedProvider: provider,
ticker: time.NewTicker(duration),
4 years ago
logger: func(_ context.Context, msg string) {
log.Print(msg)
},
}
for _, opt := range opts {
2 years ago
opt(prov)
4 years ago
}
2 years ago
return prov
4 years ago
}
func WithLogger(l func(context.Context, string)) Option {
return func(p *Provider) {
p.logger = l
}
}
type Option func(*Provider)
type Provider struct {
config.NamedProvider
4 years ago
ticker *time.Ticker
logger func(context.Context, string)
}
func (p *Provider) Watch(ctx context.Context, callback config.WatchCallback, key ...string) error {
oldVar, err := p.NamedProvider.Value(ctx, key...)
4 years ago
if err != nil {
return fmt.Errorf("failed watch variable: %w", err)
4 years ago
}
go func() {
for {
select {
case <-p.ticker.C:
newVar, err := p.NamedProvider.Value(ctx, key...)
4 years ago
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
}