2025-02-19 00:35:53 +05:30
|
|
|
package factory
|
2024-08-22 14:24:02 +05:30
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
2025-02-19 00:35:53 +05:30
|
|
|
"log/slog"
|
2024-08-22 14:24:02 +05:30
|
|
|
"os"
|
|
|
|
|
"os/signal"
|
|
|
|
|
"syscall"
|
2025-09-09 15:44:27 +05:30
|
|
|
|
|
|
|
|
"github.com/SigNoz/signoz/pkg/errors"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var (
|
|
|
|
|
ErrCodeInvalidRegistry = errors.MustNewCode("invalid_registry")
|
2024-08-22 14:24:02 +05:30
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type Registry struct {
|
2025-02-19 00:35:53 +05:30
|
|
|
services NamedMap[NamedService]
|
|
|
|
|
logger *slog.Logger
|
2024-08-22 14:24:02 +05:30
|
|
|
startCh chan error
|
|
|
|
|
stopCh chan error
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// New creates a new registry of services. It needs at least one service in the input.
|
2025-02-19 00:35:53 +05:30
|
|
|
func NewRegistry(logger *slog.Logger, services ...NamedService) (*Registry, error) {
|
2024-08-22 14:24:02 +05:30
|
|
|
if logger == nil {
|
2025-09-09 15:44:27 +05:30
|
|
|
return nil, errors.Newf(errors.TypeInvalidInput, ErrCodeInvalidRegistry, "cannot build registry, logger is required")
|
2024-08-22 14:24:02 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if len(services) == 0 {
|
2025-09-09 15:44:27 +05:30
|
|
|
return nil, errors.Newf(errors.TypeInvalidInput, ErrCodeInvalidRegistry, "cannot build registry, at least one service is required")
|
2024-08-22 14:24:02 +05:30
|
|
|
}
|
|
|
|
|
|
2025-02-19 00:35:53 +05:30
|
|
|
m, err := NewNamedMap(services...)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-08-22 14:24:02 +05:30
|
|
|
return &Registry{
|
2025-03-24 19:30:14 +05:30
|
|
|
logger: logger.With("pkg", "go.signoz.io/pkg/factory"),
|
2025-02-19 00:35:53 +05:30
|
|
|
services: m,
|
2024-08-22 14:24:02 +05:30
|
|
|
startCh: make(chan error, 1),
|
|
|
|
|
stopCh: make(chan error, len(services)),
|
|
|
|
|
}, nil
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-10 01:30:42 +05:30
|
|
|
func (r *Registry) Start(ctx context.Context) {
|
2025-02-19 00:35:53 +05:30
|
|
|
for _, s := range r.services.GetInOrder() {
|
|
|
|
|
go func(s NamedService) {
|
|
|
|
|
r.logger.InfoContext(ctx, "starting service", "service", s.Name())
|
2024-08-22 14:24:02 +05:30
|
|
|
err := s.Start(ctx)
|
|
|
|
|
r.startCh <- err
|
|
|
|
|
}(s)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (r *Registry) Wait(ctx context.Context) error {
|
|
|
|
|
interrupt := make(chan os.Signal, 1)
|
|
|
|
|
signal.Notify(interrupt, syscall.SIGINT, syscall.SIGTERM)
|
|
|
|
|
|
|
|
|
|
select {
|
|
|
|
|
case <-ctx.Done():
|
2025-02-19 00:35:53 +05:30
|
|
|
r.logger.InfoContext(ctx, "caught context error, exiting", "error", ctx.Err())
|
2024-08-22 14:24:02 +05:30
|
|
|
case s := <-interrupt:
|
2025-02-19 00:35:53 +05:30
|
|
|
r.logger.InfoContext(ctx, "caught interrupt signal, exiting", "signal", s)
|
2024-08-22 14:24:02 +05:30
|
|
|
case err := <-r.startCh:
|
2025-02-19 00:35:53 +05:30
|
|
|
r.logger.ErrorContext(ctx, "caught service error, exiting", "error", err)
|
2024-08-22 14:24:02 +05:30
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (r *Registry) Stop(ctx context.Context) error {
|
2025-02-19 00:35:53 +05:30
|
|
|
for _, s := range r.services.GetInOrder() {
|
|
|
|
|
go func(s NamedService) {
|
|
|
|
|
r.logger.InfoContext(ctx, "stopping service", "service", s.Name())
|
2024-08-22 14:24:02 +05:30
|
|
|
err := s.Stop(ctx)
|
|
|
|
|
r.stopCh <- err
|
|
|
|
|
}(s)
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-19 00:35:53 +05:30
|
|
|
errs := make([]error, len(r.services.GetInOrder()))
|
|
|
|
|
for i := 0; i < len(r.services.GetInOrder()); i++ {
|
2024-08-22 14:24:02 +05:30
|
|
|
err := <-r.stopCh
|
|
|
|
|
if err != nil {
|
|
|
|
|
errs = append(errs, err)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return errors.Join(errs...)
|
|
|
|
|
}
|