mirror of
https://github.com/zitadel/zitadel.git
synced 2025-01-10 16:13:40 +00:00
1a49b7d298
* project quota added * project quota removed * add periods table * make log record generic * accumulate usage * query usage * count action run seconds * fix filter in ReportQuotaUsage * fix existing tests * fix logstore tests * fix typo * fix: add quota unit tests command side * fix: add quota unit tests command side * fix: add quota unit tests command side * move notifications into debouncer and improve limit querying * cleanup * comment * fix: add quota unit tests command side * fix remaining quota usage query * implement InmemLogStorage * cleanup and linting * improve test * fix: add quota unit tests command side * fix: add quota unit tests command side * fix: add quota unit tests command side * fix: add quota unit tests command side * action notifications and fixes for notifications query * revert console prefix * fix: add quota unit tests command side * fix: add quota integration tests * improve accountable requests * improve accountable requests * fix: add quota integration tests * fix: add quota integration tests * fix: add quota integration tests * comment * remove ability to store logs in db and other changes requested from review * changes requested from review * changes requested from review * Update internal/api/http/middleware/access_interceptor.go Co-authored-by: Silvan <silvan.reusser@gmail.com> * tests: fix quotas integration tests * improve incrementUsageStatement * linting * fix: delete e2e tests as intergation tests cover functionality * Update internal/api/http/middleware/access_interceptor.go Co-authored-by: Silvan <silvan.reusser@gmail.com> * backup * fix conflict * create rc * create prerelease * remove issue release labeling * fix tracing --------- Co-authored-by: Livio Spring <livio.a@gmail.com> Co-authored-by: Stefan Benz <stefan@caos.ch> Co-authored-by: adlerhurst <silvan.reusser@gmail.com>
287 lines
7.1 KiB
Go
287 lines
7.1 KiB
Go
// The library github.com/benbjohnson/clock fails when race is enabled
|
|
// https://github.com/benbjohnson/clock/issues/44
|
|
//go:build !race
|
|
|
|
package logstore_test
|
|
|
|
import (
|
|
"context"
|
|
"reflect"
|
|
"runtime"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/benbjohnson/clock"
|
|
|
|
"github.com/zitadel/zitadel/internal/logstore"
|
|
emittermock "github.com/zitadel/zitadel/internal/logstore/mock"
|
|
"github.com/zitadel/zitadel/internal/query"
|
|
)
|
|
|
|
const (
|
|
tick = time.Second
|
|
ticks = 60
|
|
)
|
|
|
|
type args struct {
|
|
mainSink *logstore.EmitterConfig
|
|
secondarySink *logstore.EmitterConfig
|
|
config *query.Quota
|
|
}
|
|
|
|
type want struct {
|
|
enabled bool
|
|
remaining *uint64
|
|
mainSink wantSink
|
|
secondarySink wantSink
|
|
}
|
|
|
|
type wantSink struct {
|
|
bulks []int
|
|
len int
|
|
}
|
|
|
|
func TestService(t *testing.T) {
|
|
// tests should run on a single thread
|
|
// important for deterministic results
|
|
beforeProcs := runtime.GOMAXPROCS(1)
|
|
defer runtime.GOMAXPROCS(beforeProcs)
|
|
|
|
tests := []struct {
|
|
name string
|
|
args args
|
|
want want
|
|
}{{
|
|
name: "max and min debouncing works",
|
|
args: args{
|
|
mainSink: emitterConfig(withDebouncerConfig(&logstore.DebouncerConfig{
|
|
MinFrequency: 1 * time.Minute,
|
|
MaxBulkSize: 60,
|
|
})),
|
|
secondarySink: emitterConfig(),
|
|
config: quotaConfig(),
|
|
},
|
|
want: want{
|
|
enabled: true,
|
|
remaining: nil,
|
|
mainSink: wantSink{
|
|
bulks: repeat(60, 1),
|
|
len: 60,
|
|
},
|
|
secondarySink: wantSink{
|
|
bulks: repeat(1, 60),
|
|
len: 60,
|
|
},
|
|
},
|
|
}, {
|
|
name: "mixed debouncing works",
|
|
args: args{
|
|
mainSink: emitterConfig(withDebouncerConfig(&logstore.DebouncerConfig{
|
|
MinFrequency: 0,
|
|
MaxBulkSize: 6,
|
|
})),
|
|
secondarySink: emitterConfig(withDebouncerConfig(&logstore.DebouncerConfig{
|
|
MinFrequency: 10 * time.Second,
|
|
MaxBulkSize: 0,
|
|
})),
|
|
config: quotaConfig(),
|
|
},
|
|
want: want{
|
|
enabled: true,
|
|
remaining: nil,
|
|
mainSink: wantSink{
|
|
bulks: repeat(6, 10),
|
|
len: 60,
|
|
},
|
|
secondarySink: wantSink{
|
|
bulks: repeat(10, 6),
|
|
len: 60,
|
|
},
|
|
},
|
|
}, {
|
|
name: "when disabling main sink, secondary sink still works",
|
|
args: args{
|
|
mainSink: emitterConfig(withDisabled()),
|
|
secondarySink: emitterConfig(),
|
|
config: quotaConfig(),
|
|
},
|
|
want: want{
|
|
enabled: true,
|
|
remaining: nil,
|
|
mainSink: wantSink{
|
|
bulks: repeat(99, 0),
|
|
len: 0,
|
|
},
|
|
secondarySink: wantSink{
|
|
bulks: repeat(1, 60),
|
|
len: 60,
|
|
},
|
|
},
|
|
}, {
|
|
name: "when all sink are disabled, the service is disabled",
|
|
args: args{
|
|
mainSink: emitterConfig(withDisabled()),
|
|
secondarySink: emitterConfig(withDisabled()),
|
|
config: quotaConfig(),
|
|
},
|
|
want: want{
|
|
enabled: false,
|
|
remaining: nil,
|
|
mainSink: wantSink{
|
|
bulks: repeat(99, 0),
|
|
len: 0,
|
|
},
|
|
secondarySink: wantSink{
|
|
bulks: repeat(99, 0),
|
|
len: 0,
|
|
},
|
|
},
|
|
}, {
|
|
name: "when quota has a limit of 90, 30 are remaining",
|
|
args: args{
|
|
mainSink: emitterConfig(),
|
|
secondarySink: emitterConfig(),
|
|
config: quotaConfig(withLimiting()),
|
|
},
|
|
want: want{
|
|
enabled: true,
|
|
remaining: uint64Ptr(30),
|
|
mainSink: wantSink{
|
|
bulks: repeat(1, 60),
|
|
len: 60,
|
|
},
|
|
secondarySink: wantSink{
|
|
bulks: repeat(1, 60),
|
|
len: 60,
|
|
},
|
|
},
|
|
}, {
|
|
name: "when quota has a limit of 30, 0 are remaining",
|
|
args: args{
|
|
mainSink: emitterConfig(),
|
|
secondarySink: emitterConfig(),
|
|
config: quotaConfig(withLimiting(), withAmountAndInterval(30)),
|
|
},
|
|
want: want{
|
|
enabled: true,
|
|
remaining: uint64Ptr(0),
|
|
mainSink: wantSink{
|
|
bulks: repeat(1, 60),
|
|
len: 60,
|
|
},
|
|
secondarySink: wantSink{
|
|
bulks: repeat(1, 60),
|
|
len: 60,
|
|
},
|
|
},
|
|
}, {
|
|
name: "when quota has amount of 30 but is not limited, remaining is nil",
|
|
args: args{
|
|
mainSink: emitterConfig(),
|
|
secondarySink: emitterConfig(),
|
|
config: quotaConfig(withAmountAndInterval(30)),
|
|
},
|
|
want: want{
|
|
enabled: true,
|
|
remaining: nil,
|
|
mainSink: wantSink{
|
|
bulks: repeat(1, 60),
|
|
len: 60,
|
|
},
|
|
secondarySink: wantSink{
|
|
bulks: repeat(1, 60),
|
|
len: 60,
|
|
},
|
|
},
|
|
}}
|
|
for _, tt := range tests {
|
|
runTest(t, tt.name, tt.args, tt.want)
|
|
}
|
|
}
|
|
|
|
func runTest(t *testing.T, name string, args args, want want) bool {
|
|
return t.Run("Given over a minute, each second a log record is emitted", func(tt *testing.T) {
|
|
tt.Run(name, func(t *testing.T) {
|
|
ctx, clock, mainStorage, secondaryStorage, svc := given(t, args, want)
|
|
remaining := when(svc, ctx, clock)
|
|
then(t, mainStorage, secondaryStorage, remaining, want)
|
|
})
|
|
})
|
|
}
|
|
|
|
func given(t *testing.T, args args, want want) (context.Context, *clock.Mock, *emittermock.InmemLogStorage, *emittermock.InmemLogStorage, *logstore.Service[*emittermock.Record]) {
|
|
ctx := context.Background()
|
|
clock := clock.NewMock()
|
|
|
|
clock.Set(args.config.From)
|
|
|
|
mainStorage := emittermock.NewInMemoryStorage(clock, args.config)
|
|
mainEmitter, err := logstore.NewEmitter[*emittermock.Record](ctx, clock, args.mainSink, mainStorage)
|
|
if err != nil {
|
|
t.Errorf("expected no error but got %v", err)
|
|
}
|
|
secondaryStorage := emittermock.NewInMemoryStorage(clock, args.config)
|
|
secondaryEmitter, err := logstore.NewEmitter[*emittermock.Record](ctx, clock, args.secondarySink, secondaryStorage)
|
|
if err != nil {
|
|
t.Errorf("expected no error but got %v", err)
|
|
}
|
|
svc := logstore.New[*emittermock.Record](
|
|
mainStorage,
|
|
mainEmitter,
|
|
secondaryEmitter)
|
|
|
|
if svc.Enabled() != want.enabled {
|
|
t.Errorf("wantet service enabled to be %t but is %t", want.enabled, svc.Enabled())
|
|
}
|
|
return ctx, clock, mainStorage, secondaryStorage, svc
|
|
}
|
|
|
|
func when(svc *logstore.Service[*emittermock.Record], ctx context.Context, clock *clock.Mock) *uint64 {
|
|
var remaining *uint64
|
|
for i := 0; i < ticks; i++ {
|
|
svc.Handle(ctx, emittermock.NewRecord(clock))
|
|
runtime.Gosched()
|
|
remaining = svc.Limit(ctx, "non-empty-instance-id")
|
|
clock.Add(tick)
|
|
}
|
|
time.Sleep(time.Millisecond)
|
|
runtime.Gosched()
|
|
return remaining
|
|
}
|
|
|
|
func then(t *testing.T, mainStorage, secondaryStorage *emittermock.InmemLogStorage, remaining *uint64, want want) {
|
|
mainBulks := mainStorage.Bulks()
|
|
if !reflect.DeepEqual(want.mainSink.bulks, mainBulks) {
|
|
t.Errorf("wanted main storage to have bulks %v, but got %v", want.mainSink.bulks, mainBulks)
|
|
}
|
|
|
|
mainLen := mainStorage.Len()
|
|
if !reflect.DeepEqual(want.mainSink.len, mainLen) {
|
|
t.Errorf("wanted main storage to have len %d, but got %d", want.mainSink.len, mainLen)
|
|
}
|
|
|
|
secondaryBulks := secondaryStorage.Bulks()
|
|
if !reflect.DeepEqual(want.secondarySink.bulks, secondaryBulks) {
|
|
t.Errorf("wanted secondary storage to have bulks %v, but got %v", want.secondarySink.bulks, secondaryBulks)
|
|
}
|
|
|
|
secondaryLen := secondaryStorage.Len()
|
|
if !reflect.DeepEqual(want.secondarySink.len, secondaryLen) {
|
|
t.Errorf("wanted secondary storage to have len %d, but got %d", want.secondarySink.len, secondaryLen)
|
|
}
|
|
|
|
if remaining == nil && want.remaining == nil {
|
|
return
|
|
}
|
|
|
|
if remaining == nil && want.remaining != nil ||
|
|
remaining != nil && want.remaining == nil {
|
|
t.Errorf("wantet remaining nil %t but got %t", want.remaining == nil, remaining == nil)
|
|
return
|
|
}
|
|
if *remaining != *want.remaining {
|
|
t.Errorf("wantet remaining %d but got %d", *want.remaining, *remaining)
|
|
return
|
|
}
|
|
}
|