2020-10-23 16:16:46 +02:00
|
|
|
package eventstore_test
|
|
|
|
|
|
|
|
import (
|
2023-10-19 12:19:10 +02:00
|
|
|
"context"
|
|
|
|
"encoding/json"
|
2020-10-23 16:16:46 +02:00
|
|
|
"os"
|
|
|
|
"testing"
|
2023-02-27 22:36:43 +01:00
|
|
|
"time"
|
2020-10-23 16:16:46 +02:00
|
|
|
|
|
|
|
"github.com/cockroachdb/cockroach-go/v2/testserver"
|
2024-09-06 11:19:19 +02:00
|
|
|
pgxdecimal "github.com/jackc/pgx-shopspring-decimal"
|
|
|
|
"github.com/jackc/pgx/v5"
|
|
|
|
"github.com/jackc/pgx/v5/pgxpool"
|
|
|
|
"github.com/jackc/pgx/v5/stdlib"
|
2022-04-27 01:01:45 +02:00
|
|
|
"github.com/zitadel/logging"
|
2022-03-15 07:19:02 +01:00
|
|
|
|
2022-06-27 12:32:34 +02:00
|
|
|
"github.com/zitadel/zitadel/cmd/initialise"
|
2022-08-31 09:52:43 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/database"
|
|
|
|
"github.com/zitadel/zitadel/internal/database/cockroach"
|
2023-10-19 12:19:10 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/eventstore"
|
|
|
|
es_sql "github.com/zitadel/zitadel/internal/eventstore/repository/sql"
|
|
|
|
new_es "github.com/zitadel/zitadel/internal/eventstore/v3"
|
2020-10-23 16:16:46 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2023-02-27 22:36:43 +01:00
|
|
|
testCRDBClient *database.DB
|
2023-10-19 12:19:10 +02:00
|
|
|
queriers map[string]eventstore.Querier = make(map[string]eventstore.Querier)
|
|
|
|
pushers map[string]eventstore.Pusher = make(map[string]eventstore.Pusher)
|
|
|
|
clients map[string]*database.DB = make(map[string]*database.DB)
|
2020-10-23 16:16:46 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
2024-03-11 11:43:56 +01:00
|
|
|
opts := make([]testserver.TestServerOpt, 0, 1)
|
|
|
|
if version := os.Getenv("ZITADEL_CRDB_VERSION"); version != "" {
|
|
|
|
opts = append(opts, testserver.CustomVersionOpt(version))
|
|
|
|
}
|
|
|
|
ts, err := testserver.NewTestServer(opts...)
|
2020-10-23 16:16:46 +02:00
|
|
|
if err != nil {
|
2022-03-15 07:19:02 +01:00
|
|
|
logging.WithFields("error", err).Fatal("unable to start db")
|
2020-10-23 16:16:46 +02:00
|
|
|
}
|
|
|
|
|
2023-02-27 22:36:43 +01:00
|
|
|
testCRDBClient = &database.DB{
|
|
|
|
Database: new(testDB),
|
|
|
|
}
|
2024-09-06 11:19:19 +02:00
|
|
|
config, err := pgxpool.ParseConfig(ts.PGURL().String())
|
2020-10-23 16:16:46 +02:00
|
|
|
if err != nil {
|
2024-09-06 11:19:19 +02:00
|
|
|
logging.WithFields("error", err).Fatal("unable to parse db config")
|
2022-03-15 07:19:02 +01:00
|
|
|
}
|
2024-09-06 11:19:19 +02:00
|
|
|
config.AfterConnect = func(ctx context.Context, conn *pgx.Conn) error {
|
|
|
|
pgxdecimal.Register(conn.TypeMap())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
pool, err := pgxpool.NewWithConfig(context.Background(), config)
|
|
|
|
if err != nil {
|
|
|
|
logging.WithFields("error", err).Fatal("unable to create db pool")
|
|
|
|
}
|
|
|
|
testCRDBClient.DB = stdlib.OpenDBFromPool(pool)
|
2022-03-15 07:19:02 +01:00
|
|
|
if err = testCRDBClient.Ping(); err != nil {
|
|
|
|
logging.WithFields("error", err).Fatal("unable to ping db")
|
2020-10-23 16:16:46 +02:00
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
v2 := &es_sql.CRDB{DB: testCRDBClient}
|
|
|
|
queriers["v2(inmemory)"] = v2
|
|
|
|
clients["v2(inmemory)"] = testCRDBClient
|
|
|
|
|
|
|
|
pushers["v3(inmemory)"] = new_es.NewEventstore(testCRDBClient)
|
|
|
|
clients["v3(inmemory)"] = testCRDBClient
|
|
|
|
|
|
|
|
if localDB, err := connectLocalhost(); err == nil {
|
|
|
|
if err = initDB(localDB); err != nil {
|
|
|
|
logging.WithFields("error", err).Fatal("migrations failed")
|
|
|
|
}
|
|
|
|
pushers["v3(singlenode)"] = new_es.NewEventstore(localDB)
|
|
|
|
clients["v3(singlenode)"] = localDB
|
|
|
|
}
|
|
|
|
|
|
|
|
// pushers["v2(inmemory)"] = v2
|
|
|
|
|
2020-10-23 16:16:46 +02:00
|
|
|
defer func() {
|
|
|
|
testCRDBClient.Close()
|
|
|
|
ts.Stop()
|
|
|
|
}()
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
if err = initDB(testCRDBClient); err != nil {
|
2022-03-15 07:19:02 +01:00
|
|
|
logging.WithFields("error", err).Fatal("migrations failed")
|
2020-10-23 16:16:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
os.Exit(m.Run())
|
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
func initDB(db *database.DB) error {
|
2022-08-31 09:52:43 +02:00
|
|
|
initialise.ReadStmts("cockroach")
|
|
|
|
config := new(database.Config)
|
|
|
|
config.SetConnector(&cockroach.Config{
|
|
|
|
User: cockroach.User{
|
|
|
|
Username: "zitadel",
|
|
|
|
},
|
|
|
|
Database: "zitadel",
|
|
|
|
})
|
|
|
|
err := initialise.Init(db,
|
|
|
|
initialise.VerifyUser(config.Username(), ""),
|
2023-02-27 22:36:43 +01:00
|
|
|
initialise.VerifyDatabase(config.DatabaseName()),
|
2024-05-27 11:03:34 +02:00
|
|
|
initialise.VerifyGrant(config.DatabaseName(), config.Username()),
|
|
|
|
initialise.VerifySettings(config.DatabaseName(), config.Username()))
|
2020-10-23 16:16:46 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-01-04 17:12:20 +01:00
|
|
|
err = initialise.VerifyZitadel(context.Background(), db, *config)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// create old events
|
|
|
|
_, err = db.Exec(oldEventsTable)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func connectLocalhost() (*database.DB, error) {
|
2024-09-06 11:19:19 +02:00
|
|
|
config, err := pgxpool.ParseConfig("postgresql://root@localhost:26257/defaultdb?sslmode=disable")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
config.AfterConnect = func(ctx context.Context, conn *pgx.Conn) error {
|
|
|
|
pgxdecimal.Register(conn.TypeMap())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
pool, err := pgxpool.NewWithConfig(context.Background(), config)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-09-06 11:19:19 +02:00
|
|
|
client := stdlib.OpenDBFromPool(pool)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err = client.Ping(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &database.DB{
|
|
|
|
DB: client,
|
|
|
|
Database: new(testDB),
|
|
|
|
}, nil
|
2020-10-23 16:16:46 +02:00
|
|
|
}
|
2023-02-27 22:36:43 +01:00
|
|
|
|
|
|
|
type testDB struct{}
|
|
|
|
|
|
|
|
func (_ *testDB) Timetravel(time.Duration) string { return " AS OF SYSTEM TIME '-1 ms' " }
|
|
|
|
|
|
|
|
func (*testDB) DatabaseName() string { return "db" }
|
|
|
|
|
|
|
|
func (*testDB) Username() string { return "user" }
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
func (*testDB) Type() string { return "cockroach" }
|
|
|
|
|
|
|
|
func generateCommand(aggregateType eventstore.AggregateType, aggregateID string, opts ...func(*testEvent)) eventstore.Command {
|
|
|
|
e := &testEvent{
|
|
|
|
BaseEvent: eventstore.BaseEvent{
|
|
|
|
Agg: &eventstore.Aggregate{
|
|
|
|
ID: aggregateID,
|
|
|
|
Type: aggregateType,
|
|
|
|
ResourceOwner: "ro",
|
|
|
|
Version: "v1",
|
|
|
|
},
|
|
|
|
Service: "svc",
|
|
|
|
EventType: "test.created",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, opt := range opts {
|
|
|
|
opt(e)
|
|
|
|
}
|
|
|
|
|
|
|
|
return e
|
|
|
|
}
|
|
|
|
|
|
|
|
type testEvent struct {
|
|
|
|
eventstore.BaseEvent
|
|
|
|
uniqueConstraints []*eventstore.UniqueConstraint
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *testEvent) Payload() any {
|
|
|
|
return e.BaseEvent.Data
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *testEvent) UniqueConstraints() []*eventstore.UniqueConstraint {
|
|
|
|
return e.uniqueConstraints
|
|
|
|
}
|
|
|
|
|
|
|
|
func canceledCtx() context.Context {
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
cancel()
|
|
|
|
return ctx
|
|
|
|
}
|
|
|
|
|
|
|
|
func fillUniqueData(unique_type, field, instanceID string) error {
|
|
|
|
_, err := testCRDBClient.Exec("INSERT INTO eventstore.unique_constraints (unique_type, unique_field, instance_id) VALUES ($1, $2, $3)", unique_type, field, instanceID)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func generateAddUniqueConstraint(table, uniqueField string) func(e *testEvent) {
|
|
|
|
return func(e *testEvent) {
|
|
|
|
e.uniqueConstraints = append(e.uniqueConstraints,
|
|
|
|
&eventstore.UniqueConstraint{
|
|
|
|
UniqueType: table,
|
|
|
|
UniqueField: uniqueField,
|
|
|
|
Action: eventstore.UniqueConstraintAdd,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func generateRemoveUniqueConstraint(table, uniqueField string) func(e *testEvent) {
|
|
|
|
return func(e *testEvent) {
|
|
|
|
e.uniqueConstraints = append(e.uniqueConstraints,
|
|
|
|
&eventstore.UniqueConstraint{
|
|
|
|
UniqueType: table,
|
|
|
|
UniqueField: uniqueField,
|
|
|
|
Action: eventstore.UniqueConstraintRemove,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func withTestData(data any) func(e *testEvent) {
|
|
|
|
return func(e *testEvent) {
|
|
|
|
d, err := json.Marshal(data)
|
|
|
|
if err != nil {
|
|
|
|
panic("marshal data failed")
|
|
|
|
}
|
|
|
|
e.BaseEvent.Data = d
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func cleanupEventstore(client *database.DB) func() {
|
|
|
|
return func() {
|
|
|
|
_, err := client.Exec("TRUNCATE eventstore.events")
|
|
|
|
if err != nil {
|
|
|
|
logging.Warnf("unable to truncate events: %v", err)
|
|
|
|
}
|
|
|
|
_, err = client.Exec("TRUNCATE eventstore.events2")
|
|
|
|
if err != nil {
|
|
|
|
logging.Warnf("unable to truncate events: %v", err)
|
|
|
|
}
|
|
|
|
_, err = client.Exec("TRUNCATE eventstore.unique_constraints")
|
|
|
|
if err != nil {
|
|
|
|
logging.Warnf("unable to truncate unique constraints: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const oldEventsTable = `CREATE TABLE IF NOT EXISTS eventstore.events (
|
|
|
|
id UUID DEFAULT gen_random_uuid()
|
|
|
|
, event_type TEXT NOT NULL
|
|
|
|
, aggregate_type TEXT NOT NULL
|
|
|
|
, aggregate_id TEXT NOT NULL
|
|
|
|
, aggregate_version TEXT NOT NULL
|
|
|
|
, event_sequence BIGINT NOT NULL
|
|
|
|
, previous_aggregate_sequence BIGINT
|
|
|
|
, previous_aggregate_type_sequence INT8
|
|
|
|
, creation_date TIMESTAMPTZ NOT NULL DEFAULT now()
|
|
|
|
, created_at TIMESTAMPTZ NOT NULL DEFAULT clock_timestamp()
|
|
|
|
, event_data JSONB
|
|
|
|
, editor_user TEXT NOT NULL
|
|
|
|
, editor_service TEXT
|
|
|
|
, resource_owner TEXT NOT NULL
|
|
|
|
, instance_id TEXT NOT NULL
|
|
|
|
, "position" DECIMAL NOT NULL
|
|
|
|
, in_tx_order INTEGER NOT NULL
|
|
|
|
|
|
|
|
, PRIMARY KEY (instance_id, aggregate_type, aggregate_id, event_sequence DESC)
|
|
|
|
);`
|