Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
69 changes: 65 additions & 4 deletions pkg/aux_/store/memstore/dss.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,24 +2,85 @@ package memstore

import (
"context"
"database/sql"
"time"

auxmodels "github.com/interuss/dss/pkg/aux_/models"
dsserr "github.com/interuss/dss/pkg/errors"
"github.com/interuss/stacktrace"
)

func (r *repo) SaveOwnMetadata(_ context.Context, locality string, publicEndpoint string) error {
return stacktrace.NewErrorWithCode(dsserr.NotImplemented, "SaveOwnMetadata not implemented for memstore")
if locality == "" {
return stacktrace.NewErrorWithCode(dsserr.BadRequest, "Locality not set")
}
if publicEndpoint == "" {
return stacktrace.NewErrorWithCode(dsserr.BadRequest, "Public endpoint not set")
}

r.state.Participants[locality] = &participant{
PublicEndpoint: publicEndpoint,
UpdatedAt: time.Now().UTC(),
}
return nil
}

func (r *repo) GetDSSMetadata(_ context.Context) ([]*auxmodels.DSSMetadata, error) {
return nil, stacktrace.NewErrorWithCode(dsserr.NotImplemented, "GetDSSMetadata not implemented for memstore")
metadata := make([]*auxmodels.DSSMetadata, 0, len(r.state.Participants))
for locality, p := range r.state.Participants {
updatedAt := p.UpdatedAt
m := &auxmodels.DSSMetadata{
Locality: locality,
PublicEndpoint: p.PublicEndpoint,
UpdatedAt: &updatedAt,
}

// Find the latest heartbeat across all sources for this locality.
var latest auxmodels.Heartbeat
found := false
for key, hb := range r.state.Heartbeats {
if key.Locality != locality {
continue
}
if !found || hb.Timestamp.After(*latest.Timestamp) {
latest = hb
found = true
}
}

if found {
m.LatestTimestamp.Source = sql.NullString{String: latest.Source, Valid: true}
m.LatestTimestamp.Timestamp = latest.Timestamp
m.LatestTimestamp.NextHeartbeatExpectedBefore = latest.NextHeartbeatExpectedBefore
m.LatestTimestamp.Reporter = sql.NullString{String: latest.Reporter, Valid: true}
}

metadata = append(metadata, m)
}
return metadata, nil
}

func (r *repo) RecordHeartbeat(_ context.Context, heartbeat auxmodels.Heartbeat) error {
return stacktrace.NewErrorWithCode(dsserr.NotImplemented, "RecordHeartbeat not implemented for memstore")
if heartbeat.Locality == "" {
return stacktrace.NewErrorWithCode(dsserr.BadRequest, "Locality not set")
}
if heartbeat.Source == "" {
return stacktrace.NewErrorWithCode(dsserr.BadRequest, "Source not set")
}

if heartbeat.Timestamp == nil {
now := time.Now().UTC()
heartbeat.Timestamp = &now
}

if heartbeat.NextHeartbeatExpectedBefore != nil && heartbeat.NextHeartbeatExpectedBefore.Before(*heartbeat.Timestamp) {
return stacktrace.NewErrorWithCode(dsserr.BadRequest, "Cannot expect the timestamp of the next heartbeat before the timestamp of the new heartbeat")
}

r.state.Heartbeats[heartbeatKey{Locality: heartbeat.Locality, Source: heartbeat.Source}] = heartbeat
return nil
}

func (r *repo) GetDSSAirspaceRepresentationID(_ context.Context) (string, error) {
return "", stacktrace.NewErrorWithCode(dsserr.NotImplemented, "GetDSSAirspaceRepresentationID not implemented for memstore")
return "", stacktrace.NewErrorWithCode(dsserr.NotImplemented, "GetDSSAirspaceRepresentationID not implementable for memstore")
}
125 changes: 125 additions & 0 deletions pkg/aux_/store/memstore/dss_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,125 @@
package memstore

import (
"context"
"testing"
"time"

auxmodels "github.com/interuss/dss/pkg/aux_/models"
dsserr "github.com/interuss/dss/pkg/errors"
"github.com/interuss/stacktrace"
"github.com/stretchr/testify/require"
)

func TestSaveOwnMetadataValidation(t *testing.T) {
ctx := context.Background()
r := newRepo()

require.Equal(t, dsserr.BadRequest, stacktrace.GetCode(r.SaveOwnMetadata(ctx, "", "https://example.com")))
require.Equal(t, dsserr.BadRequest, stacktrace.GetCode(r.SaveOwnMetadata(ctx, "dss-1", "")))
}

func TestSaveOwnMetadataRoundTrip(t *testing.T) {
ctx := context.Background()
r := newRepo()

require.NoError(t, r.SaveOwnMetadata(ctx, "dss-1", "https://example.com"))

md, err := r.GetDSSMetadata(ctx)
require.NoError(t, err)

require.Len(t, md, 1)
require.Equal(t, "dss-1", md[0].Locality)
require.Equal(t, "https://example.com", md[0].PublicEndpoint)
require.NotNil(t, md[0].UpdatedAt)

// No heartbeat recorded yet.
require.False(t, md[0].LatestTimestamp.Source.Valid)
require.Nil(t, md[0].LatestTimestamp.Timestamp)
}

func TestSaveOwnMetadataUpsert(t *testing.T) {
ctx := context.Background()
r := newRepo()

require.NoError(t, r.SaveOwnMetadata(ctx, "dss-1", "https://old.example.com"))
require.NoError(t, r.SaveOwnMetadata(ctx, "dss-1", "https://new.example.com"))

md, err := r.GetDSSMetadata(ctx)
require.NoError(t, err)

require.Len(t, md, 1)
require.Equal(t, "https://new.example.com", md[0].PublicEndpoint)
}

func TestRecordHeartbeatValidation(t *testing.T) {
ctx := context.Background()
r := newRepo()

require.Equal(t, dsserr.BadRequest, stacktrace.GetCode(r.RecordHeartbeat(ctx, auxmodels.Heartbeat{Source: "source1"})))
require.Equal(t, dsserr.BadRequest, stacktrace.GetCode(r.RecordHeartbeat(ctx, auxmodels.Heartbeat{Locality: "dss-1"})))

ts := time.Now()
before := ts.Add(-time.Minute)
err := r.RecordHeartbeat(ctx, auxmodels.Heartbeat{
Locality: "dss-1",
Source: "source1",
Timestamp: &ts,
NextHeartbeatExpectedBefore: &before,
})

require.Equal(t, dsserr.BadRequest, stacktrace.GetCode(err))
}

func TestRecordHeartbeatDefaultsTimestamp(t *testing.T) {
ctx := context.Background()
r := newRepo()

require.NoError(t, r.SaveOwnMetadata(ctx, "dss-1", "https://example.com"))
require.NoError(t, r.RecordHeartbeat(ctx, auxmodels.Heartbeat{Locality: "dss-1", Source: "source1"}))

md, err := r.GetDSSMetadata(ctx)
require.NoError(t, err)

require.Len(t, md, 1)
require.True(t, md[0].LatestTimestamp.Source.Valid)
require.NotNil(t, md[0].LatestTimestamp.Timestamp)
}

func TestGetDSSMetadataPicksLatestHeartbeat(t *testing.T) {
ctx := context.Background()
r := newRepo()

require.NoError(t, r.SaveOwnMetadata(ctx, "dss-1", "https://example.com"))

older := time.Now().Add(-time.Hour)
newer := time.Now()
require.NoError(t, r.RecordHeartbeat(ctx, auxmodels.Heartbeat{Locality: "dss-1", Source: "source1", Timestamp: &older, Reporter: "uss1"}))
require.NoError(t, r.RecordHeartbeat(ctx, auxmodels.Heartbeat{Locality: "dss-1", Source: "source2", Timestamp: &newer, Reporter: "uss2"}))

md, err := r.GetDSSMetadata(ctx)
require.NoError(t, err)

require.Len(t, md, 1)
require.True(t, md[0].LatestTimestamp.Timestamp.Equal(newer))
require.Equal(t, "source2", md[0].LatestTimestamp.Source.String)
require.Equal(t, "uss2", md[0].LatestTimestamp.Reporter.String)
}

func TestGetDSSMetadataUpdatesHeartbeatPerSource(t *testing.T) {
ctx := context.Background()
r := newRepo()

require.NoError(t, r.SaveOwnMetadata(ctx, "dss-1", "https://example.com"))

first := time.Now().Add(-time.Hour)
second := time.Now()
require.NoError(t, r.RecordHeartbeat(ctx, auxmodels.Heartbeat{Locality: "dss-1", Source: "source1", Timestamp: &first}))
require.NoError(t, r.RecordHeartbeat(ctx, auxmodels.Heartbeat{Locality: "dss-1", Source: "source1", Timestamp: &second}))

md, err := r.GetDSSMetadata(ctx)
require.NoError(t, err)

require.Len(t, md, 1)
require.True(t, md[0].LatestTimestamp.Timestamp.Equal(second))
}
35 changes: 35 additions & 0 deletions pkg/aux_/store/memstore/snapshot.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
package memstore

import (
"bytes"
"encoding/gob"

"github.com/interuss/stacktrace"
)

const snapshotVersion = 1

type snapshotEnvelope struct {
Version int
State state
}

func (r *repo) GetSnapshot() ([]byte, error) {
var buf bytes.Buffer
if err := gob.NewEncoder(&buf).Encode(snapshotEnvelope{Version: snapshotVersion, State: r.state}); err != nil {
return nil, stacktrace.Propagate(err, "Failed to encode memstore snapshot")
}
return buf.Bytes(), nil
}

func (r *repo) RestoreFromSnapshot(data []byte) error {
var env snapshotEnvelope
if err := gob.NewDecoder(bytes.NewReader(data)).Decode(&env); err != nil {
return stacktrace.Propagate(err, "Failed to decode memstore snapshot")
}
if env.Version != snapshotVersion {
return stacktrace.NewError("Unsupported memstore snapshot version %d, expected %d", env.Version, snapshotVersion)
}
r.state = env.State
return nil
}
59 changes: 59 additions & 0 deletions pkg/aux_/store/memstore/snapshot_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
package memstore

import (
"bytes"
"context"
"encoding/gob"
"testing"
"time"

auxmodels "github.com/interuss/dss/pkg/aux_/models"
"github.com/stretchr/testify/require"
)

func TestSnapshotRoundTrip(t *testing.T) {
ctx := context.Background()
src := newRepo()
require.NoError(t, src.SaveOwnMetadata(ctx, "dss-1", "https://example.com"))
ts := time.Now().UTC()
require.NoError(t, src.RecordHeartbeat(ctx, auxmodels.Heartbeat{Locality: "dss-1", Source: "source-1", Timestamp: &ts, Reporter: "uss-1"}))

data, err := src.GetSnapshot()
require.NoError(t, err)

dst := newRepo()
require.NoError(t, dst.RestoreFromSnapshot(data))

want, err := src.GetDSSMetadata(ctx)
require.NoError(t, err)
got, err := dst.GetDSSMetadata(ctx)
require.NoError(t, err)
require.Equal(t, want, got)
}

func TestRestoreFromSnapshotReplacesState(t *testing.T) {
ctx := context.Background()
src := newRepo()
require.NoError(t, src.SaveOwnMetadata(ctx, "dss-1", "https://example.com"))
data, err := src.GetSnapshot()
require.NoError(t, err)

dst := newRepo()
require.NoError(t, dst.SaveOwnMetadata(ctx, "dss-2", "https://other.example.com"))
require.NoError(t, dst.RestoreFromSnapshot(data))

md, err := dst.GetDSSMetadata(ctx)
require.NoError(t, err)
require.Len(t, md, 1)
require.Equal(t, "dss-1", md[0].Locality)
}

func TestRestoreFromSnapshotInvalidData(t *testing.T) {
require.Error(t, newRepo().RestoreFromSnapshot([]byte("random value that is definitely not valid")))
}

func TestRestoreFromSnapshotVersionMismatch(t *testing.T) {
var buf bytes.Buffer
require.NoError(t, gob.NewEncoder(&buf).Encode(snapshotEnvelope{Version: snapshotVersion + 1}))
require.Error(t, newRepo().RestoreFromSnapshot(buf.Bytes()))
}
38 changes: 36 additions & 2 deletions pkg/aux_/store/memstore/store.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,51 @@ package memstore

import (
"context"
"time"

auxmodels "github.com/interuss/dss/pkg/aux_/models"
"github.com/interuss/dss/pkg/aux_/repos"
"github.com/interuss/dss/pkg/memstore"
"go.uber.org/zap"
)

// repo is a full implementation of aux_.repos.Repository for memory-based storage.
type repo struct{}
type repo struct {
state state
}

// state is the serializable in-memory state.
type state struct {
// Participants holds pool participants metadata, keyed by locality.
Participants map[string]*participant
// Heartbeats holds the latest heartbeat per (locality, source).
Heartbeats map[heartbeatKey]auxmodels.Heartbeat
// participants holds pool participants metadata, keyed by locality.
participants map[string]*participant
// heartbeats holds the latest heartbeat per (locality, source).
heartbeats map[heartbeatKey]auxmodels.Heartbeat
}

type participant struct {
PublicEndpoint string
UpdatedAt time.Time
}

type heartbeatKey struct {
Locality string
Source string
}

func newRepo() *repo {
return &repo{
state: state{
Participants: map[string]*participant{},
Heartbeats: map[heartbeatKey]auxmodels.Heartbeat{},
}}
}

func Init(ctx context.Context, logger *zap.Logger) (*memstore.Store[repos.Repository], error) {
return memstore.Init(ctx, logger, "aux_", &repo{})
return memstore.Init(ctx, logger, "aux_", newRepo())
}

func (r *repo) GetRepo() repos.Repository { return r }
2 changes: 2 additions & 0 deletions pkg/memstore/store.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@ import (

type MemRepo[R any] interface {
GetRepo() R
GetSnapshot() ([]byte, error)
RestoreFromSnapshot([]byte) error
}

type Store[R any] struct {
Expand Down
13 changes: 13 additions & 0 deletions pkg/rid/store/memstore/snapshot.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package memstore

import (
"github.com/interuss/stacktrace"
)

func (r *repo) GetSnapshot() ([]byte, error) {
return nil, stacktrace.NewError("GetSnapshot not yet implemented for rid")
}

func (r *repo) RestoreFromSnapshot(data []byte) error {
return stacktrace.NewError("RestoreFromSnapshot not yet implemented for rid")
}
Loading
Loading