mirror of
https://github.com/zitadel/zitadel.git
synced 2025-08-11 21:47:32 +00:00
perf: query data AS OF SYSTEM TIME
(#5231)
Queries the data in the storage layser at the timestamp when the call hit the API layer
This commit is contained in:
38
internal/api/call/duration.go
Normal file
38
internal/api/call/duration.go
Normal file
@@ -0,0 +1,38 @@
|
||||
package call
|
||||
|
||||
import (
|
||||
"context"
|
||||
"time"
|
||||
)
|
||||
|
||||
type durationKey struct{}
|
||||
|
||||
var key *durationKey = (*durationKey)(nil)
|
||||
|
||||
// WithTimestamp sets [time.Now()] adds the call field to the context
|
||||
// if it's not already set
|
||||
func WithTimestamp(parent context.Context) context.Context {
|
||||
if parent.Value(key) != nil {
|
||||
return parent
|
||||
}
|
||||
return context.WithValue(parent, key, time.Now())
|
||||
}
|
||||
|
||||
// FromContext returns the [time.Time] the call hit the api
|
||||
func FromContext(ctx context.Context) (t time.Time) {
|
||||
value := ctx.Value(key)
|
||||
if t, ok := value.(time.Time); ok {
|
||||
return t
|
||||
}
|
||||
|
||||
return t
|
||||
}
|
||||
|
||||
// Took returns the time the call took so far
|
||||
func Took(ctx context.Context) time.Duration {
|
||||
start := FromContext(ctx)
|
||||
if start.IsZero() {
|
||||
return 0
|
||||
}
|
||||
return time.Since(start)
|
||||
}
|
119
internal/api/call/duration_test.go
Normal file
119
internal/api/call/duration_test.go
Normal file
@@ -0,0 +1,119 @@
|
||||
package call
|
||||
|
||||
import (
|
||||
"context"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func TestTook(t *testing.T) {
|
||||
type args struct {
|
||||
ctx context.Context
|
||||
}
|
||||
tests := []struct {
|
||||
name string
|
||||
args args
|
||||
startIsZero bool
|
||||
}{
|
||||
{
|
||||
name: "no start",
|
||||
args: args{
|
||||
ctx: context.Background(),
|
||||
},
|
||||
startIsZero: true,
|
||||
},
|
||||
{
|
||||
name: "with start",
|
||||
args: args{
|
||||
ctx: WithTimestamp(context.Background()),
|
||||
},
|
||||
startIsZero: false,
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got := Took(tt.args.ctx)
|
||||
if tt.startIsZero && got != 0 {
|
||||
t.Errorf("Duration should be 0 but was %v", got)
|
||||
}
|
||||
if !tt.startIsZero && got <= 0 {
|
||||
t.Errorf("Duration should be greater 0 but was %d", got)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestFromContext(t *testing.T) {
|
||||
type args struct {
|
||||
ctx context.Context
|
||||
}
|
||||
tests := []struct {
|
||||
name string
|
||||
args args
|
||||
isZero bool
|
||||
}{
|
||||
{
|
||||
name: "no start",
|
||||
args: args{
|
||||
ctx: context.Background(),
|
||||
},
|
||||
isZero: true,
|
||||
},
|
||||
{
|
||||
name: "with start",
|
||||
args: args{
|
||||
ctx: WithTimestamp(context.Background()),
|
||||
},
|
||||
isZero: false,
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got := FromContext(tt.args.ctx)
|
||||
if tt.isZero != got.IsZero() {
|
||||
t.Errorf("Time is zero should be %v but was %v", tt.isZero, got.IsZero())
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestWithTimestamp(t *testing.T) {
|
||||
start := time.Date(2019, 4, 29, 0, 0, 0, 0, time.UTC)
|
||||
|
||||
type args struct {
|
||||
ctx context.Context
|
||||
}
|
||||
tests := []struct {
|
||||
name string
|
||||
args args
|
||||
noPrevious bool
|
||||
}{
|
||||
{
|
||||
name: "fresh context",
|
||||
args: args{
|
||||
ctx: context.WithValue(context.Background(), key, start),
|
||||
},
|
||||
noPrevious: true,
|
||||
},
|
||||
{
|
||||
name: "with start",
|
||||
args: args{
|
||||
ctx: WithTimestamp(context.Background()),
|
||||
},
|
||||
noPrevious: false,
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got := WithTimestamp(tt.args.ctx)
|
||||
val := got.Value(key).(time.Time)
|
||||
|
||||
if !tt.noPrevious && val.Before(start) {
|
||||
t.Errorf("time should be now not %v", val)
|
||||
}
|
||||
if tt.noPrevious && val.After(start) {
|
||||
t.Errorf("time should be start not %v", val)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user