move recorder to interface
This commit is contained in:
parent
e9b70fe6e0
commit
a5abbbec1f
2
main.go
2
main.go
@ -82,7 +82,7 @@ func prepareApp(ctx context.Context, aw *ambient.AmbientWeather) *app.App {
|
|||||||
{
|
{
|
||||||
Name: "Weather Service",
|
Name: "Weather Service",
|
||||||
Type: &weatherpb.AmbientLocalWeatherService_ServiceDesc,
|
Type: &weatherpb.AmbientLocalWeatherService_ServiceDesc,
|
||||||
Service: weathergrpc.NewGRPCWeather(ctx, aw.GetState()),
|
Service: weathergrpc.NewGRPCWeather(ctx, aw.GetRecorder()),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -30,13 +30,13 @@ type AmbientWeather struct {
|
|||||||
// when either "AmbientWeather" or "Wunderground" are selected
|
// when either "AmbientWeather" or "Wunderground" are selected
|
||||||
// in the "Custom" section of the AWNet app, or the web UI
|
// in the "Custom" section of the AWNet app, or the web UI
|
||||||
// of an Ambient WeatherHub
|
// of an Ambient WeatherHub
|
||||||
Config *config.AmbientLocalExporterConfig
|
Config *config.AmbientLocalExporterConfig
|
||||||
awnProvider provider.AmbientProvider
|
awnProvider provider.AmbientProvider
|
||||||
wuProvider provider.AmbientProvider
|
wuProvider provider.AmbientProvider
|
||||||
weatherState *recorder.WeatherRecorder
|
weatherRecorder *recorder.WeatherRecorder
|
||||||
appCtx context.Context
|
appCtx context.Context
|
||||||
metrics *weather.WeatherMetrics
|
metrics *weather.WeatherMetrics
|
||||||
l *zerolog.Logger
|
l *zerolog.Logger
|
||||||
*sync.RWMutex
|
*sync.RWMutex
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -70,7 +70,8 @@ func (aw *AmbientWeather) Init() *AmbientWeather {
|
|||||||
}
|
}
|
||||||
span.SetAttributes(attribute.Int("updatesToKeep", updatesToKeep))
|
span.SetAttributes(attribute.Int("updatesToKeep", updatesToKeep))
|
||||||
|
|
||||||
aw.weatherState = recorder.NewWeatherRecorder(&recorder.Opts{
|
// TODO: Support other recorders (don't rely on default)
|
||||||
|
aw.weatherRecorder = recorder.NewWeatherRecorder(&recorder.Opts{
|
||||||
Ctx: aw.appCtx,
|
Ctx: aw.appCtx,
|
||||||
KeepLast: updatesToKeep,
|
KeepLast: updatesToKeep,
|
||||||
})
|
})
|
||||||
@ -142,7 +143,7 @@ func (aw *AmbientWeather) handleProviderRequest(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Record state
|
// Record state
|
||||||
aw.weatherState.Set(ctx, update)
|
aw.weatherRecorder.Set(ctx, update)
|
||||||
|
|
||||||
// Update metrics
|
// Update metrics
|
||||||
aw.metricsUpdate(ctx, p, update)
|
aw.metricsUpdate(ctx, p, update)
|
||||||
@ -277,9 +278,9 @@ func (aw *AmbientWeather) enrichStation(update *weather.WeatherUpdate) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (aw *AmbientWeather) GetState() *recorder.WeatherRecorder {
|
func (aw *AmbientWeather) GetRecorder() *recorder.WeatherRecorder {
|
||||||
aw.RLock()
|
aw.RLock()
|
||||||
defer aw.RUnlock()
|
defer aw.RUnlock()
|
||||||
|
|
||||||
return aw.weatherState
|
return aw.weatherRecorder
|
||||||
}
|
}
|
||||||
|
17
pkg/util/util.go
Normal file
17
pkg/util/util.go
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
package util
|
||||||
|
|
||||||
|
import "k8s.io/utils/ptr"
|
||||||
|
|
||||||
|
func DerefStr(s *string) string {
|
||||||
|
if s == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return *s
|
||||||
|
}
|
||||||
|
|
||||||
|
func Int32ptr(i *int) *int32 {
|
||||||
|
if i == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return ptr.To(int32(*i))
|
||||||
|
}
|
@ -1,9 +1,8 @@
|
|||||||
package grpc
|
package grpc
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"k8s.io/utils/ptr"
|
|
||||||
|
|
||||||
pb "gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/api/v1alpha1/weather"
|
pb "gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/api/v1alpha1/weather"
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/util"
|
||||||
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -18,18 +17,18 @@ func UpdatesToPbUpdates(u []*weather.WeatherUpdate) []*pb.WeatherUpdate {
|
|||||||
func UpdateToPbUpdate(u *weather.WeatherUpdate) *pb.WeatherUpdate {
|
func UpdateToPbUpdate(u *weather.WeatherUpdate) *pb.WeatherUpdate {
|
||||||
return &pb.WeatherUpdate{
|
return &pb.WeatherUpdate{
|
||||||
StationName: u.StationConfig.Name,
|
StationName: u.StationConfig.Name,
|
||||||
StationType: derefStr(u.StationType),
|
StationType: util.DerefStr(u.StationType),
|
||||||
StationId: derefStr(u.StationID),
|
StationId: util.DerefStr(u.StationID),
|
||||||
TempOutdoorF: u.TempOutdoorF,
|
TempOutdoorF: u.TempOutdoorF,
|
||||||
TempIndoorF: u.TempIndoorF,
|
TempIndoorF: u.TempIndoorF,
|
||||||
HumidityOutdoor: int32ptr(u.HumidityOudoor),
|
HumidityOutdoor: util.Int32ptr(u.HumidityOudoor),
|
||||||
HumidityIndoor: int32ptr(u.HumidityIndoor),
|
HumidityIndoor: util.Int32ptr(u.HumidityIndoor),
|
||||||
WindSpeedMph: u.WindSpeedMPH,
|
WindSpeedMph: u.WindSpeedMPH,
|
||||||
WindGustMph: u.WindGustMPH,
|
WindGustMph: u.WindGustMPH,
|
||||||
MaxDailyGust: u.MaxDailyGust,
|
MaxDailyGust: u.MaxDailyGust,
|
||||||
WindDir: int32ptr(u.WindDir),
|
WindDir: util.Int32ptr(u.WindDir),
|
||||||
WindDirAvg_10M: int32ptr(u.WindDirAvg10m),
|
WindDirAvg_10M: util.Int32ptr(u.WindDirAvg10m),
|
||||||
Uv: int32ptr(u.UV),
|
Uv: util.Int32ptr(u.UV),
|
||||||
SolarRadiation: u.SolarRadiation,
|
SolarRadiation: u.SolarRadiation,
|
||||||
HourlyRainIn: u.HourlyRainIn,
|
HourlyRainIn: u.HourlyRainIn,
|
||||||
EventRainIn: u.EventRainIn,
|
EventRainIn: u.EventRainIn,
|
||||||
@ -52,7 +51,7 @@ func batteriesToPbBatteries(batteries []weather.BatteryStatus) []*pb.BatteryStat
|
|||||||
for i, b := range batteries {
|
for i, b := range batteries {
|
||||||
pbBatteries[i] = &pb.BatteryStatus{
|
pbBatteries[i] = &pb.BatteryStatus{
|
||||||
Component: b.Component,
|
Component: b.Component,
|
||||||
Status: int32ptr(b.Status),
|
Status: util.Int32ptr(b.Status),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return pbBatteries
|
return pbBatteries
|
||||||
@ -64,22 +63,8 @@ func thSensorsToPbSensors(sensors []*weather.TempHumiditySensor) []*pb.TempHumid
|
|||||||
pbSensors[i] = &pb.TempHumiditySensor{
|
pbSensors[i] = &pb.TempHumiditySensor{
|
||||||
Name: s.Name,
|
Name: s.Name,
|
||||||
TempF: s.TempF,
|
TempF: s.TempF,
|
||||||
Humidity: int32ptr(s.Humidity),
|
Humidity: util.Int32ptr(s.Humidity),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return pbSensors
|
return pbSensors
|
||||||
}
|
}
|
||||||
|
|
||||||
func derefStr(s *string) string {
|
|
||||||
if s == nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
return *s
|
|
||||||
}
|
|
||||||
|
|
||||||
func int32ptr(i *int) *int32 {
|
|
||||||
if i == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return ptr.To(int32(*i))
|
|
||||||
}
|
|
||||||
|
@ -46,7 +46,7 @@ func (w *GRPCWeather) GetWeather(ctx context.Context, req *pb.GetWeatherRequest)
|
|||||||
|
|
||||||
span.SetAttributes(attribute.Int("limit", limit))
|
span.SetAttributes(attribute.Int("limit", limit))
|
||||||
|
|
||||||
updates, err := w.recorder.Get(ctx, limit)
|
updates, err := w.recorder.Get(ctx, req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
span.RecordError(err)
|
span.RecordError(err)
|
||||||
span.SetStatus(otelcodes.Error, err.Error())
|
span.SetStatus(otelcodes.Error, err.Error())
|
||||||
|
@ -9,20 +9,21 @@ import (
|
|||||||
|
|
||||||
"gitea.libretechconsulting.com/rmcguire/go-app/pkg/otel"
|
"gitea.libretechconsulting.com/rmcguire/go-app/pkg/otel"
|
||||||
|
|
||||||
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather/recorder/recorders"
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather/recorder/recorders/memory"
|
||||||
)
|
)
|
||||||
|
|
||||||
type WeatherRecorder struct {
|
type WeatherRecorder struct {
|
||||||
updates []*weather.WeatherUpdate
|
recorder recorders.Recorder
|
||||||
keep int
|
ctx context.Context
|
||||||
ctx context.Context
|
tracer trace.Tracer
|
||||||
tracer trace.Tracer
|
meter metric.Meter
|
||||||
meter metric.Meter
|
|
||||||
*sync.RWMutex
|
*sync.RWMutex
|
||||||
}
|
}
|
||||||
|
|
||||||
type Opts struct {
|
type Opts struct {
|
||||||
Ctx context.Context
|
Ctx context.Context
|
||||||
|
Recorder recorders.Recorder // If nil, will use memory recorder
|
||||||
KeepLast int
|
KeepLast int
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -31,12 +32,20 @@ func NewWeatherRecorder(opts *Opts) *WeatherRecorder {
|
|||||||
opts.KeepLast = 1
|
opts.KeepLast = 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if opts.Recorder == nil {
|
||||||
|
opts.Recorder = &memory.MemoryRecorder{}
|
||||||
|
}
|
||||||
|
|
||||||
|
opts.Recorder.Init(opts.Ctx, &recorders.RecorderOpts{
|
||||||
|
RetainLast: opts.KeepLast,
|
||||||
|
BaseCtx: opts.Ctx,
|
||||||
|
})
|
||||||
|
|
||||||
return &WeatherRecorder{
|
return &WeatherRecorder{
|
||||||
updates: make([]*weather.WeatherUpdate, 0, opts.KeepLast),
|
ctx: opts.Ctx,
|
||||||
keep: opts.KeepLast,
|
recorder: opts.Recorder,
|
||||||
ctx: opts.Ctx,
|
tracer: otel.GetTracer(opts.Ctx, "weatherRecorder"),
|
||||||
tracer: otel.GetTracer(opts.Ctx, "weatherRecorder"),
|
meter: otel.GetMeter(opts.Ctx, "weatherRecorder"),
|
||||||
meter: otel.GetMeter(opts.Ctx, "weatherRecorder"),
|
RWMutex: &sync.RWMutex{},
|
||||||
RWMutex: &sync.RWMutex{},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,35 +2,41 @@ package recorder
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
|
||||||
|
|
||||||
"go.opentelemetry.io/otel/attribute"
|
"go.opentelemetry.io/otel/attribute"
|
||||||
"go.opentelemetry.io/otel/codes"
|
"go.opentelemetry.io/otel/codes"
|
||||||
"go.opentelemetry.io/otel/trace"
|
"k8s.io/utils/ptr"
|
||||||
|
|
||||||
|
pb "gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/api/v1alpha1/weather"
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/util"
|
||||||
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Returns last requested number of weather updates
|
// Returns last requested number of weather updates
|
||||||
// If negative number given, will return all weather observations
|
// If negative number given, will return all weather observations
|
||||||
func (w *WeatherRecorder) Get(ctx context.Context, last int) (
|
func (w *WeatherRecorder) Get(ctx context.Context, req *pb.GetWeatherRequest) (
|
||||||
[]*weather.WeatherUpdate, error,
|
[]*weather.WeatherUpdate, error,
|
||||||
) {
|
) {
|
||||||
if last < 0 {
|
if req == nil {
|
||||||
last = w.keep
|
req = &pb.GetWeatherRequest{Limit: ptr.To(int32(-1))}
|
||||||
} else if last < 1 {
|
}
|
||||||
last = 1
|
|
||||||
|
if req.Limit == nil || *req.Limit < 0 {
|
||||||
|
req.Limit = ptr.To(int32(-1))
|
||||||
|
} else if *req.Limit == 0 {
|
||||||
|
req.Limit = ptr.To(int32(1))
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, span := w.tracer.Start(ctx, "getWeatherRecorder")
|
ctx, span := w.tracer.Start(ctx, "getWeatherRecorder")
|
||||||
span.SetAttributes(
|
span.SetAttributes(
|
||||||
attribute.Int("last", last),
|
attribute.String("stationNameFilter", util.DerefStr(req.Opts.StationName)),
|
||||||
attribute.Int("keep", w.keep),
|
attribute.String("stationTypeFilter", util.DerefStr(req.Opts.StationType)),
|
||||||
|
attribute.Int("last", int(*req.Limit)),
|
||||||
attribute.Int("currentSize", w.Count(ctx)),
|
attribute.Int("currentSize", w.Count(ctx)),
|
||||||
)
|
)
|
||||||
defer span.End()
|
defer span.End()
|
||||||
|
|
||||||
updates, err := w.get(ctx, last)
|
updates, err := w.recorder.Get(ctx, req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
span.RecordError(err)
|
span.RecordError(err)
|
||||||
span.SetStatus(codes.Error, err.Error())
|
span.SetStatus(codes.Error, err.Error())
|
||||||
@ -41,50 +47,14 @@ func (w *WeatherRecorder) Get(ctx context.Context, last int) (
|
|||||||
return updates, err
|
return updates, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *WeatherRecorder) get(ctx context.Context, last int) (
|
|
||||||
[]*weather.WeatherUpdate, error,
|
|
||||||
) {
|
|
||||||
span := trace.SpanFromContext(ctx)
|
|
||||||
|
|
||||||
w.RLock()
|
|
||||||
defer w.RUnlock()
|
|
||||||
|
|
||||||
span.AddEvent("acquired lock on recorder cache")
|
|
||||||
|
|
||||||
updates := w.updates
|
|
||||||
|
|
||||||
if w.count() == 0 {
|
|
||||||
err := errors.New("no recorded updates to get")
|
|
||||||
span.RecordError(err)
|
|
||||||
return nil, err
|
|
||||||
} else if w.count() <= last {
|
|
||||||
span.RecordError(errors.New("requested more updates than recorded"))
|
|
||||||
} else {
|
|
||||||
updates = updates[len(updates)-last:]
|
|
||||||
}
|
|
||||||
|
|
||||||
span.SetAttributes(attribute.Int("retrieved", len(updates)))
|
|
||||||
span.SetStatus(codes.Ok, "")
|
|
||||||
|
|
||||||
return updates, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns count of retained weather updates
|
// Returns count of retained weather updates
|
||||||
func (w *WeatherRecorder) Count(ctx context.Context) int {
|
func (w *WeatherRecorder) Count(ctx context.Context) int {
|
||||||
_, span := w.tracer.Start(ctx, "countWeatherRecorder")
|
ctx, span := w.tracer.Start(ctx, "countWeatherRecorder")
|
||||||
defer span.End()
|
defer span.End()
|
||||||
|
|
||||||
count := w.count()
|
count := w.recorder.Count(ctx)
|
||||||
|
|
||||||
span.SetAttributes(attribute.Int("count", count))
|
span.SetAttributes(attribute.Int("count", count))
|
||||||
span.SetStatus(codes.Ok, "")
|
|
||||||
|
|
||||||
|
span.SetStatus(codes.Ok, "")
|
||||||
return count
|
return count
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *WeatherRecorder) count() int {
|
|
||||||
w.RLock()
|
|
||||||
defer w.RUnlock()
|
|
||||||
|
|
||||||
return len(w.updates)
|
|
||||||
}
|
|
||||||
|
@ -6,30 +6,16 @@ import (
|
|||||||
"go.opentelemetry.io/otel/attribute"
|
"go.opentelemetry.io/otel/attribute"
|
||||||
"go.opentelemetry.io/otel/trace"
|
"go.opentelemetry.io/otel/trace"
|
||||||
|
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/util"
|
||||||
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (w *WeatherRecorder) Set(ctx context.Context, u *weather.WeatherUpdate) error {
|
func (w *WeatherRecorder) Set(ctx context.Context, u *weather.WeatherUpdate) error {
|
||||||
ctx, span := w.tracer.Start(ctx, "recordWeatherUpdate")
|
ctx, span := w.tracer.Start(ctx, "setRecorderUpdate", trace.WithAttributes(
|
||||||
span.SetAttributes(
|
attribute.String("stationName", u.StationConfig.Name),
|
||||||
attribute.Int("countWeatherUpdates", w.Count(ctx)),
|
attribute.String("stationType", util.DerefStr(u.StationType)),
|
||||||
attribute.Int("keepUpdates", w.keep),
|
attribute.String("stationEquipment", u.StationConfig.Equipment),
|
||||||
)
|
))
|
||||||
defer span.End()
|
defer span.End()
|
||||||
|
return w.recorder.Set(ctx, u)
|
||||||
return w.set(span, u)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *WeatherRecorder) set(span trace.Span, u *weather.WeatherUpdate) error {
|
|
||||||
w.Lock()
|
|
||||||
defer w.Unlock()
|
|
||||||
|
|
||||||
if len(w.updates) > w.keep {
|
|
||||||
w.updates = w.updates[1:]
|
|
||||||
span.AddEvent("trimmed recorded updates by 1")
|
|
||||||
}
|
|
||||||
|
|
||||||
w.updates = append(w.updates, u)
|
|
||||||
span.AddEvent("recorded weather update")
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
75
pkg/weather/recorder/recorders/memory/get.go
Normal file
75
pkg/weather/recorder/recorders/memory/get.go
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
package memory
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"slices"
|
||||||
|
|
||||||
|
"go.opentelemetry.io/otel/attribute"
|
||||||
|
"go.opentelemetry.io/otel/codes"
|
||||||
|
|
||||||
|
pb "gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/api/v1alpha1/weather"
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (r *MemoryRecorder) Get(ctx context.Context, req *pb.GetWeatherRequest) (
|
||||||
|
[]*weather.WeatherUpdate, error,
|
||||||
|
) {
|
||||||
|
ctx, span := r.tracer.Start(ctx, "memoryRecorder.Get")
|
||||||
|
defer span.End()
|
||||||
|
|
||||||
|
r.RLock()
|
||||||
|
defer r.RUnlock()
|
||||||
|
|
||||||
|
span.AddEvent("acquired lock on recorder cache")
|
||||||
|
|
||||||
|
updates := r.updates
|
||||||
|
|
||||||
|
if r.count() == 0 {
|
||||||
|
err := errors.New("no recorded updates to get")
|
||||||
|
span.RecordError(err)
|
||||||
|
return nil, err
|
||||||
|
} else if r.count() <= int(*req.Limit) {
|
||||||
|
span.RecordError(errors.New("requested more updates than recorded"))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Filter by Station Name if requested
|
||||||
|
if req.Opts.StationName != nil && *req.Opts.StationName != "" {
|
||||||
|
updates = slices.DeleteFunc(updates, func(u *weather.WeatherUpdate) bool {
|
||||||
|
return u.StationConfig == nil || u.StationConfig.Name != *req.Opts.StationName
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Filter by Station Type if requested
|
||||||
|
if req.Opts.StationType != nil && *req.Opts.StationType != "" {
|
||||||
|
updates = slices.DeleteFunc(updates, func(u *weather.WeatherUpdate) bool {
|
||||||
|
return u.StationType == nil || *u.StationType != *req.Opts.StationType
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Limit results
|
||||||
|
if len(updates) > int(*req.Limit) {
|
||||||
|
updates = updates[len(updates)-int(*req.Limit):]
|
||||||
|
}
|
||||||
|
|
||||||
|
span.SetAttributes(attribute.Int("retrieved", len(updates)))
|
||||||
|
span.SetStatus(codes.Ok, "")
|
||||||
|
|
||||||
|
return updates, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *MemoryRecorder) Count(ctx context.Context) int {
|
||||||
|
_, span := r.tracer.Start(ctx, "countWeatherRecorder")
|
||||||
|
defer span.End()
|
||||||
|
|
||||||
|
count := r.count()
|
||||||
|
|
||||||
|
span.SetAttributes(attribute.Int("count", count))
|
||||||
|
span.SetStatus(codes.Ok, "")
|
||||||
|
|
||||||
|
return count
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *MemoryRecorder) count() int {
|
||||||
|
return len(r.updates)
|
||||||
|
}
|
35
pkg/weather/recorder/recorders/memory/memory.go
Normal file
35
pkg/weather/recorder/recorders/memory/memory.go
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
package memory
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"go.opentelemetry.io/otel/trace"
|
||||||
|
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/go-app/pkg/otel"
|
||||||
|
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather/recorder/recorders"
|
||||||
|
)
|
||||||
|
|
||||||
|
const defRetainLast = 120
|
||||||
|
|
||||||
|
type MemoryRecorder struct {
|
||||||
|
baseCtx context.Context
|
||||||
|
updates []*weather.WeatherUpdate
|
||||||
|
tracer trace.Tracer
|
||||||
|
keep int
|
||||||
|
*sync.RWMutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *MemoryRecorder) Init(ctx context.Context, opts *recorders.RecorderOpts) {
|
||||||
|
if opts.RetainLast < 1 {
|
||||||
|
opts.RetainLast = defRetainLast
|
||||||
|
}
|
||||||
|
|
||||||
|
r.updates = make([]*weather.WeatherUpdate, 0, opts.RetainLast)
|
||||||
|
r.keep = opts.RetainLast
|
||||||
|
r.baseCtx = opts.BaseCtx
|
||||||
|
r.RWMutex = &sync.RWMutex{}
|
||||||
|
r.tracer = otel.GetTracer(r.baseCtx, "memoryRecorder")
|
||||||
|
}
|
38
pkg/weather/recorder/recorders/memory/set.go
Normal file
38
pkg/weather/recorder/recorders/memory/set.go
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
package memory
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"go.opentelemetry.io/otel/attribute"
|
||||||
|
"go.opentelemetry.io/otel/trace"
|
||||||
|
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (r *MemoryRecorder) Set(ctx context.Context, u *weather.WeatherUpdate) error {
|
||||||
|
r.Lock()
|
||||||
|
defer r.Unlock()
|
||||||
|
|
||||||
|
ctx, span := r.tracer.Start(ctx, "memoryRecorder.Set")
|
||||||
|
span.SetAttributes(
|
||||||
|
attribute.Int("countWeatherUpdates", r.Count(ctx)),
|
||||||
|
attribute.Int("keepUpdates", r.keep),
|
||||||
|
)
|
||||||
|
defer span.End()
|
||||||
|
|
||||||
|
return r.set(ctx, u)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *MemoryRecorder) set(ctx context.Context, u *weather.WeatherUpdate) error {
|
||||||
|
span := trace.SpanFromContext(ctx)
|
||||||
|
|
||||||
|
if len(r.updates) > r.keep {
|
||||||
|
r.updates = r.updates[1:]
|
||||||
|
span.AddEvent("trimmed recorded updates by 1")
|
||||||
|
}
|
||||||
|
|
||||||
|
r.updates = append(r.updates, u)
|
||||||
|
span.AddEvent("recorded weather update")
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
20
pkg/weather/recorder/recorders/recorders.go
Normal file
20
pkg/weather/recorder/recorders/recorders.go
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
package recorders
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
pb "gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/api/v1alpha1/weather"
|
||||||
|
"gitea.libretechconsulting.com/rmcguire/ambient-local-exporter/pkg/weather"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RecorderOpts struct {
|
||||||
|
RetainLast int
|
||||||
|
BaseCtx context.Context
|
||||||
|
}
|
||||||
|
|
||||||
|
type Recorder interface {
|
||||||
|
Init(context.Context, *RecorderOpts)
|
||||||
|
Set(context.Context, *weather.WeatherUpdate) error
|
||||||
|
Get(context.Context, *pb.GetWeatherRequest) ([]*weather.WeatherUpdate, error)
|
||||||
|
Count(context.Context) int // Best Effort
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user