mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-05 12:57:38 +03:00
Add service registry and ntp service
This commit is contained in:
parent
b7cd741872
commit
20b4148381
5 changed files with 237 additions and 0 deletions
16
common/ntp/context.go
Normal file
16
common/ntp/context.go
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
package ntp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/sagernet/sing/common/service"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TimeFuncFromContext(ctx context.Context) func() time.Time {
|
||||||
|
timeService := service.FromContext[TimeService](ctx)
|
||||||
|
if timeService == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return timeService.TimeFunc()
|
||||||
|
}
|
123
common/ntp/service.go
Normal file
123
common/ntp/service.go
Normal file
|
@ -0,0 +1,123 @@
|
||||||
|
package ntp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"os"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/sagernet/sing/common"
|
||||||
|
E "github.com/sagernet/sing/common/exceptions"
|
||||||
|
"github.com/sagernet/sing/common/logger"
|
||||||
|
M "github.com/sagernet/sing/common/metadata"
|
||||||
|
N "github.com/sagernet/sing/common/network"
|
||||||
|
)
|
||||||
|
|
||||||
|
const TimeLayout = "2006-01-02 15:04:05 -0700"
|
||||||
|
|
||||||
|
type TimeService interface {
|
||||||
|
TimeFunc() func() time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
type Options struct {
|
||||||
|
Context context.Context
|
||||||
|
Server M.Socksaddr
|
||||||
|
Interval time.Duration
|
||||||
|
Dialer N.Dialer
|
||||||
|
Logger logger.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ TimeService = (*Service)(nil)
|
||||||
|
|
||||||
|
type Service struct {
|
||||||
|
ctx context.Context
|
||||||
|
cancel common.ContextCancelCauseFunc
|
||||||
|
server M.Socksaddr
|
||||||
|
dialer N.Dialer
|
||||||
|
logger logger.Logger
|
||||||
|
ticker *time.Ticker
|
||||||
|
clockOffset time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewService(options Options) *Service {
|
||||||
|
ctx := options.Context
|
||||||
|
if ctx == nil {
|
||||||
|
ctx = context.Background()
|
||||||
|
}
|
||||||
|
ctx, cancel := common.ContextWithCancelCause(ctx)
|
||||||
|
destination := options.Server
|
||||||
|
if !destination.IsValid() {
|
||||||
|
destination = M.Socksaddr{
|
||||||
|
Fqdn: "time.google.com",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if destination.Port == 0 {
|
||||||
|
destination.Port = 123
|
||||||
|
}
|
||||||
|
var interval time.Duration
|
||||||
|
if options.Interval > 0 {
|
||||||
|
interval = options.Interval
|
||||||
|
} else {
|
||||||
|
interval = 30 * time.Minute
|
||||||
|
}
|
||||||
|
var dialer N.Dialer
|
||||||
|
if options.Dialer != nil {
|
||||||
|
dialer = options.Dialer
|
||||||
|
} else {
|
||||||
|
dialer = N.SystemDialer
|
||||||
|
}
|
||||||
|
return &Service{
|
||||||
|
ctx: ctx,
|
||||||
|
cancel: cancel,
|
||||||
|
server: destination,
|
||||||
|
dialer: dialer,
|
||||||
|
logger: options.Logger,
|
||||||
|
ticker: time.NewTicker(interval),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Service) Start() error {
|
||||||
|
err := s.update()
|
||||||
|
if err != nil {
|
||||||
|
return E.Cause(err, "initialize time")
|
||||||
|
}
|
||||||
|
s.logger.Info("updated time: ", s.TimeFunc()().Local().Format(TimeLayout))
|
||||||
|
go s.loopUpdate()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Service) Close() error {
|
||||||
|
s.ticker.Stop()
|
||||||
|
s.cancel(os.ErrClosed)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Service) TimeFunc() func() time.Time {
|
||||||
|
return func() time.Time {
|
||||||
|
return time.Now().Add(s.clockOffset)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Service) loopUpdate() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-s.ctx.Done():
|
||||||
|
return
|
||||||
|
case <-s.ticker.C:
|
||||||
|
}
|
||||||
|
err := s.update()
|
||||||
|
if err == nil {
|
||||||
|
s.logger.Debug("updated time: ", s.TimeFunc()().Local().Format(TimeLayout))
|
||||||
|
} else {
|
||||||
|
s.logger.Warn("update time: ", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Service) update() error {
|
||||||
|
response, err := Exchange(s.ctx, s.dialer, s.server)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
s.clockOffset = response.ClockOffset
|
||||||
|
return nil
|
||||||
|
}
|
63
common/service/context.go
Normal file
63
common/service/context.go
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"github.com/sagernet/sing/common"
|
||||||
|
)
|
||||||
|
|
||||||
|
func ContextWithRegistry(ctx context.Context, registry Registry) context.Context {
|
||||||
|
return context.WithValue(ctx, common.DefaultValue[*Registry](), registry)
|
||||||
|
}
|
||||||
|
|
||||||
|
func RegistryFromContext(ctx context.Context) Registry {
|
||||||
|
registry := ctx.Value(common.DefaultValue[*Registry]())
|
||||||
|
if registry == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return registry.(Registry)
|
||||||
|
}
|
||||||
|
|
||||||
|
func FromContext[T any](ctx context.Context) T {
|
||||||
|
registry := RegistryFromContext(ctx)
|
||||||
|
if registry == nil {
|
||||||
|
return common.DefaultValue[T]()
|
||||||
|
}
|
||||||
|
service := registry.Get(common.DefaultValue[*T]())
|
||||||
|
if service == nil {
|
||||||
|
return common.DefaultValue[T]()
|
||||||
|
}
|
||||||
|
return service.(T)
|
||||||
|
}
|
||||||
|
|
||||||
|
func PtrFromContext[T any](ctx context.Context) *T {
|
||||||
|
registry := RegistryFromContext(ctx)
|
||||||
|
if registry == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
servicePtr := registry.Get(common.DefaultValue[*T]())
|
||||||
|
if servicePtr == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return servicePtr.(*T)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ContextWith[T any](ctx context.Context, service T) context.Context {
|
||||||
|
registry := RegistryFromContext(ctx)
|
||||||
|
if registry == nil {
|
||||||
|
registry = NewRegistry()
|
||||||
|
ctx = ContextWithRegistry(ctx, registry)
|
||||||
|
}
|
||||||
|
registry.Register(common.DefaultValue[*T](), service)
|
||||||
|
return ctx
|
||||||
|
}
|
||||||
|
|
||||||
|
func ContextWithPtr[T any](ctx context.Context, servicePtr *T) context.Context {
|
||||||
|
registry := RegistryFromContext(ctx)
|
||||||
|
if registry == nil {
|
||||||
|
registry = NewRegistry()
|
||||||
|
ctx = ContextWithRegistry(ctx, registry)
|
||||||
|
}
|
||||||
|
registry.Register(common.DefaultValue[*T](), servicePtr)
|
||||||
|
return ctx
|
||||||
|
}
|
35
common/service/registry.go
Normal file
35
common/service/registry.go
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Registry interface {
|
||||||
|
Register(serviceType any, service any) any
|
||||||
|
Get(serviceType any) any
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewRegistry() Registry {
|
||||||
|
return &defaultRegistry{
|
||||||
|
serviceTypes: make(map[any]any),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type defaultRegistry struct {
|
||||||
|
serviceTypes map[any]any
|
||||||
|
access sync.RWMutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *defaultRegistry) Register(serviceType any, service any) any {
|
||||||
|
r.access.Lock()
|
||||||
|
defer r.access.Unlock()
|
||||||
|
oldService := r.serviceTypes[serviceType]
|
||||||
|
r.serviceTypes[serviceType] = service
|
||||||
|
return oldService
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *defaultRegistry) Get(serviceType any) any {
|
||||||
|
r.access.RLock()
|
||||||
|
defer r.access.RUnlock()
|
||||||
|
return r.serviceTypes[serviceType]
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue