mirror of
https://github.com/elyby/chrly.git
synced 2025-01-10 05:47:47 +05:30
165 lines
4.1 KiB
Go
165 lines
4.1 KiB
Go
|
package otel
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"errors"
|
||
|
"log/slog"
|
||
|
"time"
|
||
|
|
||
|
"github.com/agoda-com/opentelemetry-go/otelslog"
|
||
|
logsOtel "github.com/agoda-com/opentelemetry-logs-go"
|
||
|
"github.com/agoda-com/opentelemetry-logs-go/exporters/otlp/otlplogs"
|
||
|
"github.com/agoda-com/opentelemetry-logs-go/sdk/logs"
|
||
|
logsSdk "github.com/agoda-com/opentelemetry-logs-go/sdk/logs"
|
||
|
runtimeMetrics "go.opentelemetry.io/contrib/instrumentation/runtime"
|
||
|
"go.opentelemetry.io/otel"
|
||
|
"go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp"
|
||
|
"go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp"
|
||
|
"go.opentelemetry.io/otel/propagation"
|
||
|
"go.opentelemetry.io/otel/sdk/metric"
|
||
|
"go.opentelemetry.io/otel/sdk/resource"
|
||
|
"go.opentelemetry.io/otel/sdk/trace"
|
||
|
"go.opentelemetry.io/otel/semconv/v1.4.0"
|
||
|
|
||
|
"ely.by/chrly/internal/version"
|
||
|
)
|
||
|
|
||
|
func SetupOTelSDK(ctx context.Context) (shutdown func(context.Context) error, err error) {
|
||
|
var shutdownFuncs []func(context.Context) error
|
||
|
|
||
|
// shutdown calls cleanup functions registered via shutdownFuncs.
|
||
|
// The errors from the calls are joined.
|
||
|
// Each registered cleanup will be invoked once
|
||
|
shutdown = func(ctx context.Context) error {
|
||
|
var err error
|
||
|
for _, fn := range shutdownFuncs {
|
||
|
err = errors.Join(err, fn(ctx))
|
||
|
}
|
||
|
|
||
|
shutdownFuncs = nil
|
||
|
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
// handleErr calls shutdown for cleanup and makes sure that all errors are returned
|
||
|
handleErr := func(inErr error) {
|
||
|
err = errors.Join(inErr, shutdown(ctx))
|
||
|
}
|
||
|
|
||
|
// Set up propagator
|
||
|
prop := newPropagator()
|
||
|
otel.SetTextMapPropagator(prop)
|
||
|
|
||
|
// Set up resource
|
||
|
res, err := newResource(ctx)
|
||
|
if err != nil {
|
||
|
handleErr(err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Set up logs provider
|
||
|
logsProvider, err := newLoggerProvider(ctx, res)
|
||
|
if err != nil {
|
||
|
handleErr(err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
shutdownFuncs = append(shutdownFuncs, logsProvider.Shutdown)
|
||
|
logsOtel.SetLoggerProvider(logsProvider)
|
||
|
|
||
|
otelSlog := slog.New(otelslog.NewOtelHandler(logsProvider, &otelslog.HandlerOptions{Level: slog.LevelDebug}))
|
||
|
slog.SetDefault(otelSlog)
|
||
|
|
||
|
// Set up trace provider
|
||
|
tracerProvider, err := newTraceProvider(ctx, res)
|
||
|
if err != nil {
|
||
|
handleErr(err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
shutdownFuncs = append(shutdownFuncs, tracerProvider.Shutdown)
|
||
|
otel.SetTracerProvider(tracerProvider)
|
||
|
|
||
|
// Set up meter provider
|
||
|
meterProvider, err := newMeterProvider(ctx, res)
|
||
|
if err != nil {
|
||
|
handleErr(err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
shutdownFuncs = append(shutdownFuncs, meterProvider.Shutdown)
|
||
|
otel.SetMeterProvider(meterProvider)
|
||
|
|
||
|
err = runtimeMetrics.Start(runtimeMetrics.WithMinimumReadMemStatsInterval(time.Second))
|
||
|
if err != nil {
|
||
|
handleErr(err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func newPropagator() propagation.TextMapPropagator {
|
||
|
return propagation.NewCompositeTextMapPropagator(
|
||
|
propagation.TraceContext{},
|
||
|
propagation.Baggage{},
|
||
|
)
|
||
|
}
|
||
|
|
||
|
func newResource(ctx context.Context) (*resource.Resource, error) {
|
||
|
return resource.New(
|
||
|
ctx,
|
||
|
resource.WithFromEnv(),
|
||
|
resource.WithTelemetrySDK(),
|
||
|
resource.WithOS(),
|
||
|
resource.WithContainer(),
|
||
|
resource.WithHost(),
|
||
|
resource.WithAttributes(
|
||
|
semconv.ServiceNameKey.String("chrly"),
|
||
|
semconv.ServiceVersionKey.String(version.Version()),
|
||
|
),
|
||
|
)
|
||
|
}
|
||
|
|
||
|
func newLoggerProvider(ctx context.Context, res *resource.Resource) (*logs.LoggerProvider, error) {
|
||
|
exporter, err := otlplogs.NewExporter(ctx)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
loggerProvider := logsSdk.NewLoggerProvider(
|
||
|
logsSdk.WithBatcher(exporter),
|
||
|
logsSdk.WithResource(res),
|
||
|
)
|
||
|
|
||
|
return loggerProvider, nil
|
||
|
}
|
||
|
|
||
|
func newTraceProvider(ctx context.Context, res *resource.Resource) (*trace.TracerProvider, error) {
|
||
|
traceExporter, err := otlptracehttp.New(ctx)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
traceProvider := trace.NewTracerProvider(
|
||
|
trace.WithResource(res),
|
||
|
trace.WithBatcher(traceExporter),
|
||
|
)
|
||
|
|
||
|
return traceProvider, nil
|
||
|
}
|
||
|
|
||
|
func newMeterProvider(ctx context.Context, res *resource.Resource) (*metric.MeterProvider, error) {
|
||
|
metricExporter, err := otlpmetrichttp.New(ctx)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
meterProvider := metric.NewMeterProvider(
|
||
|
metric.WithResource(res),
|
||
|
metric.WithReader(metric.NewPeriodicReader(metricExporter)),
|
||
|
)
|
||
|
|
||
|
return meterProvider, nil
|
||
|
}
|