mirror of
https://github.com/zitadel/zitadel.git
synced 2025-01-07 08:07:46 +00:00
1573 lines
35 KiB
Go
1573 lines
35 KiB
Go
package crdb
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"errors"
|
|
"reflect"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/DATA-DOG/go-sqlmock"
|
|
|
|
"github.com/caos/zitadel/internal/eventstore"
|
|
"github.com/caos/zitadel/internal/eventstore/handler"
|
|
"github.com/caos/zitadel/internal/eventstore/repository"
|
|
es_repo_mock "github.com/caos/zitadel/internal/eventstore/repository/mock"
|
|
)
|
|
|
|
var (
|
|
errFilter = errors.New("filter err")
|
|
errReduce = errors.New("reduce err")
|
|
)
|
|
|
|
var _ eventstore.Event = &testEvent{}
|
|
|
|
type testEvent struct {
|
|
eventstore.BaseEvent
|
|
sequence uint64
|
|
previousSequence uint64
|
|
aggregateType eventstore.AggregateType
|
|
}
|
|
|
|
func (e *testEvent) Sequence() uint64 {
|
|
return e.sequence
|
|
}
|
|
|
|
func (e *testEvent) Aggregate() eventstore.Aggregate {
|
|
return eventstore.Aggregate{
|
|
Type: e.aggregateType,
|
|
}
|
|
}
|
|
|
|
func (e *testEvent) PreviousAggregateTypeSequence() uint64 {
|
|
return e.previousSequence
|
|
}
|
|
|
|
func TestProjectionHandler_SearchQuery(t *testing.T) {
|
|
type want struct {
|
|
SearchQueryBuilder *eventstore.SearchQueryBuilder
|
|
limit uint64
|
|
isErr func(error) bool
|
|
expectations []mockExpectation
|
|
}
|
|
type fields struct {
|
|
sequenceTable string
|
|
projectionName string
|
|
aggregates []eventstore.AggregateType
|
|
bulkLimit uint64
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
want want
|
|
}{
|
|
{
|
|
name: "error in current sequence",
|
|
fields: fields{
|
|
sequenceTable: "my_sequences",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
bulkLimit: 5,
|
|
},
|
|
want: want{
|
|
limit: 0,
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrTxDone)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectCurrentSequenceErr("my_sequences", "my_projection", sql.ErrTxDone),
|
|
},
|
|
SearchQueryBuilder: nil,
|
|
},
|
|
},
|
|
{
|
|
name: "only aggregates",
|
|
fields: fields{
|
|
sequenceTable: "my_sequences",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
bulkLimit: 5,
|
|
},
|
|
want: want{
|
|
limit: 5,
|
|
isErr: func(err error) bool {
|
|
return err == nil
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectCurrentSequence("my_sequences", "my_projection", 5, "testAgg"),
|
|
},
|
|
SearchQueryBuilder: eventstore.
|
|
NewSearchQueryBuilder(eventstore.ColumnsEvent).
|
|
AddQuery().
|
|
AggregateTypes("testAgg").
|
|
SequenceGreater(5).
|
|
Builder().
|
|
Limit(5),
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
client, mock, err := sqlmock.New()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer client.Close()
|
|
|
|
h := NewStatementHandler(context.Background(), StatementHandlerConfig{
|
|
ProjectionHandlerConfig: handler.ProjectionHandlerConfig{
|
|
ProjectionName: tt.fields.projectionName,
|
|
},
|
|
SequenceTable: tt.fields.sequenceTable,
|
|
BulkLimit: tt.fields.bulkLimit,
|
|
Client: client,
|
|
})
|
|
|
|
h.aggregates = tt.fields.aggregates
|
|
|
|
for _, expectation := range tt.want.expectations {
|
|
expectation(mock)
|
|
}
|
|
|
|
query, limit, err := h.SearchQuery()
|
|
if !tt.want.isErr(err) {
|
|
t.Errorf("ProjectionHandler.prepareBulkStmts() error = %v", err)
|
|
return
|
|
}
|
|
if !reflect.DeepEqual(query, tt.want.SearchQueryBuilder) {
|
|
t.Errorf("unexpected query: expected %v, got %v", tt.want.SearchQueryBuilder, query)
|
|
}
|
|
if tt.want.limit != limit {
|
|
t.Errorf("unexpected limit: got: %d want %d", limit, tt.want.limit)
|
|
}
|
|
|
|
mock.MatchExpectationsInOrder(true)
|
|
if err := mock.ExpectationsWereMet(); err != nil {
|
|
t.Errorf("expectations not met: %v", err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestStatementHandler_Update(t *testing.T) {
|
|
type fields struct {
|
|
eventstore *eventstore.Eventstore
|
|
aggregates []eventstore.AggregateType
|
|
}
|
|
type want struct {
|
|
expectations []mockExpectation
|
|
isErr func(error) bool
|
|
stmtsLen int
|
|
}
|
|
type args struct {
|
|
ctx context.Context
|
|
stmts []*handler.Statement
|
|
reduce handler.Reduce
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
want want
|
|
args args
|
|
}{
|
|
{
|
|
name: "begin fails",
|
|
args: args{
|
|
ctx: context.Background(),
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBeginErr(sql.ErrConnDone),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrConnDone)
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "current sequence fails",
|
|
args: args{
|
|
ctx: context.Background(),
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBegin(),
|
|
expectCurrentSequenceErr("my_sequences", "my_projection", sql.ErrTxDone),
|
|
expectRollback(),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrTxDone)
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "fetch previous fails",
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t).
|
|
ExpectFilterEventsError(errFilter),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
args: args{
|
|
ctx: context.Background(),
|
|
stmts: []*handler.Statement{
|
|
NewNoOpStatement(&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 6,
|
|
previousSequence: 0,
|
|
}),
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBegin(),
|
|
expectCurrentSequence("my_sequences", "my_projection", 5, "testAgg"),
|
|
expectRollback(),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, errFilter)
|
|
},
|
|
stmtsLen: 1,
|
|
},
|
|
},
|
|
{
|
|
name: "no successful stmts",
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
args: args{
|
|
ctx: context.Background(),
|
|
stmts: []*handler.Statement{
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "testAgg",
|
|
sequence: 7,
|
|
previousSequence: 6,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBegin(),
|
|
expectCurrentSequence("my_sequences", "my_projection", 5, "testAgg"),
|
|
expectCommit(),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, handler.ErrSomeStmtsFailed)
|
|
},
|
|
stmtsLen: 1,
|
|
},
|
|
},
|
|
{
|
|
name: "update current sequence fails",
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
ctx: context.Background(),
|
|
stmts: []*handler.Statement{
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 7,
|
|
previousSequence: 5,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBegin(),
|
|
expectCurrentSequence("my_sequences", "my_projection", 5, "agg"),
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
expectUpdateCurrentSequenceNoRows("my_sequences", "my_projection", 7, "agg"),
|
|
expectRollback(),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, errSeqNotUpdated)
|
|
},
|
|
stmtsLen: 1,
|
|
},
|
|
},
|
|
{
|
|
name: "commit fails",
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
ctx: context.Background(),
|
|
stmts: []*handler.Statement{
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 7,
|
|
previousSequence: 5,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBegin(),
|
|
expectCurrentSequence("my_sequences", "my_projection", 5, "agg"),
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
expectUpdateCurrentSequence("my_sequences", "my_projection", 7, "agg"),
|
|
expectCommitErr(sql.ErrConnDone),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrConnDone)
|
|
},
|
|
stmtsLen: 1,
|
|
},
|
|
},
|
|
{
|
|
name: "correct",
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
args: args{
|
|
ctx: context.Background(),
|
|
stmts: []*handler.Statement{
|
|
NewNoOpStatement(&testEvent{
|
|
aggregateType: "testAgg",
|
|
sequence: 7,
|
|
previousSequence: 5,
|
|
}),
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBegin(),
|
|
expectCurrentSequence("my_sequences", "my_projection", 5, "testAgg"),
|
|
expectUpdateCurrentSequence("my_sequences", "my_projection", 7, "testAgg"),
|
|
expectCommit(),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, nil)
|
|
},
|
|
stmtsLen: 0,
|
|
},
|
|
},
|
|
{
|
|
name: "fetch previous stmts no additional stmts",
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t).ExpectFilterEvents(),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
args: args{
|
|
ctx: context.Background(),
|
|
stmts: []*handler.Statement{
|
|
NewNoOpStatement(&testEvent{
|
|
aggregateType: "testAgg",
|
|
sequence: 7,
|
|
previousSequence: 0,
|
|
}),
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBegin(),
|
|
expectCurrentSequence("my_sequences", "my_projection", 5, "testAgg"),
|
|
expectUpdateCurrentSequence("my_sequences", "my_projection", 7, "testAgg"),
|
|
expectCommit(),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, nil)
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "fetch previous stmts additional events",
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t).ExpectFilterEvents(
|
|
&repository.Event{
|
|
AggregateType: "testAgg",
|
|
Sequence: 6,
|
|
PreviousAggregateSequence: 5,
|
|
},
|
|
),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
args: args{
|
|
ctx: context.Background(),
|
|
stmts: []*handler.Statement{
|
|
NewNoOpStatement(&testEvent{
|
|
aggregateType: "testAgg",
|
|
sequence: 7,
|
|
previousSequence: 0,
|
|
}),
|
|
},
|
|
reduce: testReduce(),
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectBegin(),
|
|
expectCurrentSequence("my_sequences", "my_projection", 5, "testAgg"),
|
|
expectUpdateCurrentSequence("my_sequences", "my_projection", 7, "testAgg"),
|
|
expectCommit(),
|
|
},
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, nil)
|
|
},
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
client, mock, err := sqlmock.New()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer client.Close()
|
|
|
|
h := NewStatementHandler(context.Background(), StatementHandlerConfig{
|
|
ProjectionHandlerConfig: handler.ProjectionHandlerConfig{
|
|
ProjectionName: "my_projection",
|
|
HandlerConfig: handler.HandlerConfig{
|
|
Eventstore: tt.fields.eventstore,
|
|
},
|
|
RequeueEvery: 0,
|
|
},
|
|
SequenceTable: "my_sequences",
|
|
Client: client,
|
|
})
|
|
|
|
h.aggregates = tt.fields.aggregates
|
|
|
|
for _, expectation := range tt.want.expectations {
|
|
expectation(mock)
|
|
}
|
|
|
|
stmts, err := h.Update(tt.args.ctx, tt.args.stmts, tt.args.reduce)
|
|
if !tt.want.isErr(err) {
|
|
t.Errorf("StatementHandler.Update() error = %v", err)
|
|
}
|
|
if tt.want.stmtsLen != len(stmts) {
|
|
t.Errorf("wrong stmts length: want: %d got %d", tt.want.stmtsLen, len(stmts))
|
|
}
|
|
|
|
mock.MatchExpectationsInOrder(true)
|
|
if err := mock.ExpectationsWereMet(); err != nil {
|
|
t.Errorf("expectations not met: %v", err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestProjectionHandler_fetchPreviousStmts(t *testing.T) {
|
|
type args struct {
|
|
ctx context.Context
|
|
stmtSeq uint64
|
|
sequences currentSequences
|
|
reduce handler.Reduce
|
|
}
|
|
type want struct {
|
|
stmtCount int
|
|
isErr func(error) bool
|
|
}
|
|
type fields struct {
|
|
eventstore *eventstore.Eventstore
|
|
aggregates []eventstore.AggregateType
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
args args
|
|
fields fields
|
|
want want
|
|
}{
|
|
{
|
|
name: "no queries",
|
|
args: args{
|
|
ctx: context.Background(),
|
|
reduce: testReduce(),
|
|
},
|
|
fields: fields{
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, nil)
|
|
},
|
|
stmtCount: 0,
|
|
},
|
|
},
|
|
{
|
|
name: "eventstore returns err",
|
|
args: args{
|
|
ctx: context.Background(),
|
|
reduce: testReduce(),
|
|
sequences: currentSequences{
|
|
"testAgg": 5,
|
|
},
|
|
stmtSeq: 6,
|
|
},
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t).ExpectFilterEventsError(errFilter),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, errFilter)
|
|
},
|
|
stmtCount: 0,
|
|
},
|
|
},
|
|
{
|
|
name: "no events found",
|
|
args: args{
|
|
ctx: context.Background(),
|
|
reduce: testReduce(),
|
|
sequences: currentSequences{
|
|
"testAgg": 5,
|
|
},
|
|
stmtSeq: 6,
|
|
},
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t).ExpectFilterEvents(),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return err == nil
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "found events",
|
|
args: args{
|
|
ctx: context.Background(),
|
|
reduce: testReduce(),
|
|
sequences: currentSequences{
|
|
"testAgg": 5,
|
|
},
|
|
stmtSeq: 10,
|
|
},
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t).ExpectFilterEvents(
|
|
&repository.Event{
|
|
ID: "id",
|
|
Sequence: 7,
|
|
PreviousAggregateSequence: 0,
|
|
CreationDate: time.Now(),
|
|
Type: "test.added",
|
|
Version: "v1",
|
|
AggregateID: "testid",
|
|
AggregateType: "testAgg",
|
|
},
|
|
&repository.Event{
|
|
ID: "id",
|
|
Sequence: 9,
|
|
PreviousAggregateSequence: 7,
|
|
CreationDate: time.Now(),
|
|
Type: "test.changed",
|
|
Version: "v1",
|
|
AggregateID: "testid",
|
|
AggregateType: "testAgg",
|
|
},
|
|
),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
want: want{
|
|
stmtCount: 2,
|
|
isErr: func(err error) bool {
|
|
return err == nil
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "reduce fails",
|
|
args: args{
|
|
ctx: context.Background(),
|
|
reduce: testReduceErr(errReduce),
|
|
sequences: currentSequences{
|
|
"testAgg": 5,
|
|
},
|
|
stmtSeq: 10,
|
|
},
|
|
fields: fields{
|
|
eventstore: eventstore.NewEventstore(
|
|
es_repo_mock.NewRepo(t).ExpectFilterEvents(
|
|
&repository.Event{
|
|
ID: "id",
|
|
Sequence: 7,
|
|
PreviousAggregateSequence: 0,
|
|
CreationDate: time.Now(),
|
|
Type: "test.added",
|
|
Version: "v1",
|
|
AggregateID: "testid",
|
|
AggregateType: "testAgg",
|
|
},
|
|
),
|
|
),
|
|
aggregates: []eventstore.AggregateType{"testAgg"},
|
|
},
|
|
want: want{
|
|
stmtCount: 0,
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, errReduce)
|
|
},
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
h := &StatementHandler{
|
|
ProjectionHandler: handler.NewProjectionHandler(handler.ProjectionHandlerConfig{
|
|
HandlerConfig: handler.HandlerConfig{
|
|
Eventstore: tt.fields.eventstore,
|
|
},
|
|
ProjectionName: "my_projection",
|
|
RequeueEvery: 0,
|
|
}),
|
|
aggregates: tt.fields.aggregates,
|
|
}
|
|
stmts, err := h.fetchPreviousStmts(tt.args.ctx, tt.args.stmtSeq, tt.args.sequences, tt.args.reduce)
|
|
if !tt.want.isErr(err) {
|
|
t.Errorf("ProjectionHandler.prepareBulkStmts() error = %v", err)
|
|
return
|
|
}
|
|
if tt.want.stmtCount != len(stmts) {
|
|
t.Errorf("unexpected length of stmts: got: %d want %d", len(stmts), tt.want.stmtCount)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestStatementHandler_executeStmts(t *testing.T) {
|
|
type fields struct {
|
|
projectionName string
|
|
maxFailureCount uint
|
|
failedEventsTable string
|
|
}
|
|
type args struct {
|
|
stmts []*handler.Statement
|
|
sequences currentSequences
|
|
}
|
|
type want struct {
|
|
expectations []mockExpectation
|
|
idx int
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
args args
|
|
want want
|
|
}{
|
|
{
|
|
name: "already inserted",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmts: []*handler.Statement{
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 5,
|
|
previousSequence: 2,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val1",
|
|
},
|
|
}),
|
|
},
|
|
sequences: currentSequences{
|
|
"agg": 5,
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{},
|
|
idx: -1,
|
|
},
|
|
},
|
|
{
|
|
name: "previous sequence higher than sequence",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmts: []*handler.Statement{
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 5,
|
|
previousSequence: 0,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col1",
|
|
Value: "val1",
|
|
},
|
|
}),
|
|
NewCreateStatement(
|
|
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 8,
|
|
previousSequence: 7,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col2",
|
|
Value: "val2",
|
|
},
|
|
}),
|
|
},
|
|
sequences: currentSequences{
|
|
"agg": 2,
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col1"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
},
|
|
idx: 0,
|
|
},
|
|
},
|
|
{
|
|
name: "execute fails not continue",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
maxFailureCount: 5,
|
|
failedEventsTable: "failed_events",
|
|
},
|
|
args: args{
|
|
stmts: []*handler.Statement{
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 5,
|
|
previousSequence: 0,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 6,
|
|
previousSequence: 5,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 7,
|
|
previousSequence: 6,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
},
|
|
sequences: currentSequences{
|
|
"agg": 2,
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
expectSavePoint(),
|
|
expectCreateErr("my_projection", []string{"col"}, []string{"$1"}, sql.ErrConnDone),
|
|
expectSavePointRollback(),
|
|
expectFailureCount("failed_events", "my_projection", 6, 3),
|
|
expectUpdateFailureCount("failed_events", "my_projection", 6, 4),
|
|
},
|
|
idx: 0,
|
|
},
|
|
},
|
|
{
|
|
name: "execute fails continue",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
maxFailureCount: 5,
|
|
failedEventsTable: "failed_events",
|
|
},
|
|
args: args{
|
|
stmts: []*handler.Statement{
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 5,
|
|
previousSequence: 0,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col1",
|
|
Value: "val1",
|
|
},
|
|
}),
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 6,
|
|
previousSequence: 5,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col2",
|
|
Value: "val2",
|
|
},
|
|
}),
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 7,
|
|
previousSequence: 6,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col3",
|
|
Value: "val3",
|
|
},
|
|
}),
|
|
},
|
|
sequences: currentSequences{
|
|
"agg": 2,
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col1"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
expectSavePoint(),
|
|
expectCreateErr("my_projection", []string{"col2"}, []string{"$1"}, sql.ErrConnDone),
|
|
expectSavePointRollback(),
|
|
expectFailureCount("failed_events", "my_projection", 6, 4),
|
|
expectUpdateFailureCount("failed_events", "my_projection", 6, 5),
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col3"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
},
|
|
idx: 2,
|
|
},
|
|
},
|
|
{
|
|
name: "correct",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmts: []*handler.Statement{
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 5,
|
|
previousSequence: 0,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 6,
|
|
previousSequence: 5,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 7,
|
|
previousSequence: 6,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
NewMultiStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 8,
|
|
previousSequence: 7,
|
|
},
|
|
AddCreateStatement(
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
},
|
|
),
|
|
AddCreateStatement(
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
},
|
|
),
|
|
),
|
|
},
|
|
sequences: currentSequences{
|
|
"agg": 2,
|
|
},
|
|
},
|
|
want: want{
|
|
expectations: []mockExpectation{
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
},
|
|
idx: 3,
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
client, mock, err := sqlmock.New()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer client.Close()
|
|
|
|
h := NewStatementHandler(
|
|
context.Background(),
|
|
StatementHandlerConfig{
|
|
ProjectionHandlerConfig: handler.ProjectionHandlerConfig{
|
|
HandlerConfig: handler.HandlerConfig{
|
|
Eventstore: nil,
|
|
},
|
|
ProjectionName: tt.fields.projectionName,
|
|
RequeueEvery: 0,
|
|
},
|
|
Client: client,
|
|
FailedEventsTable: tt.fields.failedEventsTable,
|
|
MaxFailureCount: tt.fields.maxFailureCount,
|
|
},
|
|
)
|
|
|
|
mock.ExpectBegin()
|
|
|
|
for _, expectation := range tt.want.expectations {
|
|
expectation(mock)
|
|
}
|
|
|
|
mock.ExpectCommit()
|
|
|
|
tx, err := client.Begin()
|
|
if err != nil {
|
|
t.Fatalf("unexpected err in begin: %v", err)
|
|
}
|
|
|
|
idx := h.executeStmts(tx, tt.args.stmts, tt.args.sequences)
|
|
if idx != tt.want.idx {
|
|
t.Errorf("unexpected index want: %d got %d", tt.want.idx, idx)
|
|
}
|
|
|
|
if err := tx.Commit(); err != nil {
|
|
t.Fatalf("unexpected err in commit: %v", err)
|
|
}
|
|
|
|
if err := mock.ExpectationsWereMet(); err != nil {
|
|
t.Errorf("expectations not met: %v", err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestStatementHandler_executeStmt(t *testing.T) {
|
|
type fields struct {
|
|
projectionName string
|
|
}
|
|
type args struct {
|
|
stmt *handler.Statement
|
|
}
|
|
type want struct {
|
|
expectations []mockExpectation
|
|
isErr func(error) bool
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
args args
|
|
want want
|
|
}{
|
|
{
|
|
name: "create savepoint fails",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmt: NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 1,
|
|
previousSequence: 0,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrConnDone)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectSavePointErr(sql.ErrConnDone),
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "execute fails",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmt: NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 1,
|
|
previousSequence: 0,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrNoRows)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectSavePoint(),
|
|
expectCreateErr("my_projection", []string{"col"}, []string{"$1"}, sql.ErrNoRows),
|
|
expectSavePointRollback(),
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "rollback savepoint fails",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmt: NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 1,
|
|
previousSequence: 0,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrConnDone)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectSavePoint(),
|
|
expectCreateErr("my_projection", []string{"col"}, []string{"$1"}, sql.ErrNoRows),
|
|
expectSavePointRollbackErr(sql.ErrConnDone),
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "no op",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmt: NewNoOpStatement(&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 1,
|
|
previousSequence: 0,
|
|
}),
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return err == nil
|
|
},
|
|
expectations: []mockExpectation{},
|
|
},
|
|
},
|
|
{
|
|
name: "with op",
|
|
fields: fields{
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmt: NewCreateStatement(
|
|
&testEvent{
|
|
aggregateType: "agg",
|
|
sequence: 1,
|
|
previousSequence: 0,
|
|
},
|
|
[]handler.Column{
|
|
{
|
|
Name: "col",
|
|
Value: "val",
|
|
},
|
|
}),
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return err == nil
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectSavePoint(),
|
|
expectCreate("my_projection", []string{"col"}, []string{"$1"}),
|
|
expectSavePointRelease(),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
h := &StatementHandler{
|
|
ProjectionHandler: &handler.ProjectionHandler{
|
|
ProjectionName: tt.fields.projectionName,
|
|
},
|
|
}
|
|
|
|
client, mock, err := sqlmock.New()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer client.Close()
|
|
|
|
mock.ExpectBegin()
|
|
|
|
for _, expectation := range tt.want.expectations {
|
|
expectation(mock)
|
|
}
|
|
|
|
mock.ExpectCommit()
|
|
|
|
tx, err := client.Begin()
|
|
if err != nil {
|
|
t.Fatalf("unexpected err in begin: %v", err)
|
|
}
|
|
|
|
err = h.executeStmt(tx, tt.args.stmt)
|
|
if !tt.want.isErr(err) {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
|
|
if err = tx.Commit(); err != nil {
|
|
t.Fatalf("unexpected err in begin: %v", err)
|
|
}
|
|
|
|
mock.MatchExpectationsInOrder(true)
|
|
if err := mock.ExpectationsWereMet(); err != nil {
|
|
t.Errorf("expectations not met: %v", err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestStatementHandler_currentSequence(t *testing.T) {
|
|
type fields struct {
|
|
sequenceTable string
|
|
projectionName string
|
|
aggregates []eventstore.AggregateType
|
|
}
|
|
type args struct {
|
|
stmt handler.Statement
|
|
}
|
|
type want struct {
|
|
expectations []mockExpectation
|
|
isErr func(error) bool
|
|
sequences currentSequences
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
args args
|
|
want want
|
|
}{
|
|
{
|
|
name: "error in query",
|
|
fields: fields{
|
|
sequenceTable: "my_table",
|
|
projectionName: "my_projection",
|
|
},
|
|
args: args{
|
|
stmt: handler.Statement{},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrConnDone)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectCurrentSequenceErr("my_table", "my_projection", sql.ErrConnDone),
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "no rows",
|
|
fields: fields{
|
|
sequenceTable: "my_table",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
stmt: handler.Statement{},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, nil)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectCurrentSequenceNoRows("my_table", "my_projection"),
|
|
},
|
|
sequences: currentSequences{
|
|
"agg": 0,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "scan err",
|
|
fields: fields{
|
|
sequenceTable: "my_table",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
stmt: handler.Statement{},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrTxDone)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectCurrentSequenceScanErr("my_table", "my_projection"),
|
|
},
|
|
sequences: currentSequences{
|
|
"agg": 0,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "found",
|
|
fields: fields{
|
|
sequenceTable: "my_table",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
stmt: handler.Statement{},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, nil)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectCurrentSequence("my_table", "my_projection", 5, "agg"),
|
|
},
|
|
sequences: currentSequences{
|
|
"agg": 5,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
h := NewStatementHandler(context.Background(), StatementHandlerConfig{
|
|
ProjectionHandlerConfig: handler.ProjectionHandlerConfig{
|
|
ProjectionName: tt.fields.projectionName,
|
|
},
|
|
SequenceTable: tt.fields.sequenceTable,
|
|
})
|
|
|
|
h.aggregates = tt.fields.aggregates
|
|
|
|
client, mock, err := sqlmock.New()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer client.Close()
|
|
|
|
mock.ExpectBegin()
|
|
|
|
for _, expectation := range tt.want.expectations {
|
|
expectation(mock)
|
|
}
|
|
|
|
mock.ExpectCommit()
|
|
|
|
tx, err := client.Begin()
|
|
if err != nil {
|
|
t.Fatalf("unexpected err in begin: %v", err)
|
|
}
|
|
|
|
seq, err := h.currentSequences(tx.Query)
|
|
if !tt.want.isErr(err) {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
|
|
if err = tx.Commit(); err != nil {
|
|
t.Fatalf("unexpected err in commit: %v", err)
|
|
}
|
|
|
|
mock.MatchExpectationsInOrder(true)
|
|
if err := mock.ExpectationsWereMet(); err != nil {
|
|
t.Errorf("expectations not met: %v", err)
|
|
}
|
|
|
|
for _, aggregateType := range tt.fields.aggregates {
|
|
if seq[aggregateType] != tt.want.sequences[aggregateType] {
|
|
t.Errorf("unexpected sequence in aggregate type %s: want %d got %d", aggregateType, tt.want.sequences[aggregateType], seq[aggregateType])
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestStatementHandler_updateCurrentSequence(t *testing.T) {
|
|
type fields struct {
|
|
sequenceTable string
|
|
projectionName string
|
|
aggregates []eventstore.AggregateType
|
|
}
|
|
type args struct {
|
|
sequences currentSequences
|
|
}
|
|
type want struct {
|
|
expectations []mockExpectation
|
|
isErr func(error) bool
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
args args
|
|
want want
|
|
}{
|
|
{
|
|
name: "update sequence fails",
|
|
fields: fields{
|
|
sequenceTable: "my_table",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
sequences: currentSequences{
|
|
"agg": 5,
|
|
},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.Is(err, sql.ErrConnDone)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectUpdateCurrentSequenceErr("my_table", "my_projection", 5, sql.ErrConnDone, "agg"),
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "update sequence returns no rows",
|
|
fields: fields{
|
|
sequenceTable: "my_table",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
sequences: currentSequences{
|
|
"agg": 5,
|
|
},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return errors.As(err, &errSeqNotUpdated)
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectUpdateCurrentSequenceNoRows("my_table", "my_projection", 5, "agg"),
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "correct",
|
|
fields: fields{
|
|
sequenceTable: "my_table",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
sequences: currentSequences{
|
|
"agg": 5,
|
|
},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return err == nil
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectUpdateCurrentSequence("my_table", "my_projection", 5, "agg"),
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "multiple sequences",
|
|
fields: fields{
|
|
sequenceTable: "my_table",
|
|
projectionName: "my_projection",
|
|
aggregates: []eventstore.AggregateType{"agg"},
|
|
},
|
|
args: args{
|
|
sequences: currentSequences{
|
|
"agg": 5,
|
|
"agg2": 6,
|
|
},
|
|
},
|
|
want: want{
|
|
isErr: func(err error) bool {
|
|
return err == nil
|
|
},
|
|
expectations: []mockExpectation{
|
|
expectUpdateTwoCurrentSequence("my_table", "my_projection", currentSequences{
|
|
"agg": 5,
|
|
"agg2": 6},
|
|
),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
h := NewStatementHandler(context.Background(), StatementHandlerConfig{
|
|
ProjectionHandlerConfig: handler.ProjectionHandlerConfig{
|
|
ProjectionName: tt.fields.projectionName,
|
|
},
|
|
SequenceTable: tt.fields.sequenceTable,
|
|
})
|
|
|
|
h.aggregates = tt.fields.aggregates
|
|
|
|
client, mock, err := sqlmock.New()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer client.Close()
|
|
|
|
mock.ExpectBegin()
|
|
for _, expectation := range tt.want.expectations {
|
|
expectation(mock)
|
|
}
|
|
mock.ExpectCommit()
|
|
|
|
tx, err := client.Begin()
|
|
if err != nil {
|
|
t.Fatalf("unexpected error in begin: %v", err)
|
|
}
|
|
|
|
err = h.updateCurrentSequences(tx, tt.args.sequences)
|
|
if !tt.want.isErr(err) {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
|
|
err = tx.Commit()
|
|
if err != nil {
|
|
t.Fatalf("unexpected error in commit: %v", err)
|
|
}
|
|
|
|
mock.MatchExpectationsInOrder(true)
|
|
if err := mock.ExpectationsWereMet(); err != nil {
|
|
t.Errorf("expectations not met: %v", err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func testReduce() handler.Reduce {
|
|
return func(event eventstore.Event) (*handler.Statement, error) {
|
|
return NewNoOpStatement(event), nil
|
|
}
|
|
}
|
|
|
|
func testReduceErr(err error) handler.Reduce {
|
|
return func(event eventstore.Event) (*handler.Statement, error) {
|
|
return nil, err
|
|
}
|
|
}
|