mirror of
https://github.com/zitadel/zitadel.git
synced 2025-07-17 18:58:52 +00:00

This PR summarizes multiple changes specifically only available with ZITADEL v3: - feat: Web Keys management (https://github.com/zitadel/zitadel/pull/9526) - fix(cmd): ensure proper working of mirror (https://github.com/zitadel/zitadel/pull/9509) - feat(Authz): system user support for permission check v2 (https://github.com/zitadel/zitadel/pull/9640) - chore(license): change from Apache to AGPL (https://github.com/zitadel/zitadel/pull/9597) - feat(console): list v2 sessions (https://github.com/zitadel/zitadel/pull/9539) - fix(console): add loginV2 feature flag (https://github.com/zitadel/zitadel/pull/9682) - fix(feature flags): allow reading "own" flags (https://github.com/zitadel/zitadel/pull/9649) - feat(console): add Actions V2 UI (https://github.com/zitadel/zitadel/pull/9591) BREAKING CHANGE - feat(webkey): migrate to v2beta API (https://github.com/zitadel/zitadel/pull/9445) - chore!: remove CockroachDB Support (https://github.com/zitadel/zitadel/pull/9444) - feat(actions): migrate to v2beta API (https://github.com/zitadel/zitadel/pull/9489) --------- Co-authored-by: Livio Spring <livio.a@gmail.com> Co-authored-by: Stefan Benz <46600784+stebenz@users.noreply.github.com> Co-authored-by: Silvan <27845747+adlerhurst@users.noreply.github.com> Co-authored-by: Ramon <mail@conblem.me> Co-authored-by: Elio Bischof <elio@zitadel.com> Co-authored-by: Kenta Yamaguchi <56732734+KEY60228@users.noreply.github.com> Co-authored-by: Harsha Reddy <harsha.reddy@klaviyo.com> Co-authored-by: Livio Spring <livio@zitadel.com> Co-authored-by: Max Peintner <max@caos.ch> Co-authored-by: Iraq <66622793+kkrime@users.noreply.github.com> Co-authored-by: Florian Forster <florian@zitadel.com> Co-authored-by: Tim Möhlmann <tim+github@zitadel.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Max Peintner <peintnerm@gmail.com>
203 lines
4.5 KiB
Go
203 lines
4.5 KiB
Go
package eventstore
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"encoding/json"
|
|
"errors"
|
|
"sync"
|
|
|
|
"github.com/DATA-DOG/go-sqlmock"
|
|
"github.com/jackc/pgx/v5"
|
|
"github.com/jackc/pgx/v5/pgtype"
|
|
"github.com/jackc/pgx/v5/stdlib"
|
|
"github.com/zitadel/logging"
|
|
|
|
"github.com/zitadel/zitadel/internal/database"
|
|
"github.com/zitadel/zitadel/internal/database/dialect"
|
|
"github.com/zitadel/zitadel/internal/eventstore"
|
|
)
|
|
|
|
func init() {
|
|
dialect.RegisterAfterConnect(RegisterEventstoreTypes)
|
|
}
|
|
|
|
var (
|
|
// pushPlaceholderFmt defines how data are inserted into the events table
|
|
pushPlaceholderFmt = "($%d, $%d, $%d, $%d, $%d, $%d, $%d, $%d, $%d, statement_timestamp(), EXTRACT(EPOCH FROM clock_timestamp()), $%d)"
|
|
// uniqueConstraintPlaceholderFmt defines the format of the unique constraint error returned from the database
|
|
uniqueConstraintPlaceholderFmt = "(%s, %s, %s)"
|
|
|
|
_ eventstore.Pusher = (*Eventstore)(nil)
|
|
)
|
|
|
|
type Eventstore struct {
|
|
client *database.DB
|
|
}
|
|
|
|
var (
|
|
textType = &pgtype.Type{
|
|
Name: "text",
|
|
OID: pgtype.TextOID,
|
|
Codec: pgtype.TextCodec{},
|
|
}
|
|
commandType = &pgtype.Type{
|
|
Codec: &pgtype.CompositeCodec{
|
|
Fields: []pgtype.CompositeCodecField{
|
|
{
|
|
Name: "instance_id",
|
|
Type: textType,
|
|
},
|
|
{
|
|
Name: "aggregate_type",
|
|
Type: textType,
|
|
},
|
|
{
|
|
Name: "aggregate_id",
|
|
Type: textType,
|
|
},
|
|
{
|
|
Name: "command_type",
|
|
Type: textType,
|
|
},
|
|
{
|
|
Name: "revision",
|
|
Type: &pgtype.Type{
|
|
Name: "int2",
|
|
OID: pgtype.Int2OID,
|
|
Codec: pgtype.Int2Codec{},
|
|
},
|
|
},
|
|
{
|
|
Name: "payload",
|
|
Type: &pgtype.Type{
|
|
Name: "jsonb",
|
|
OID: pgtype.JSONBOID,
|
|
Codec: &pgtype.JSONBCodec{
|
|
Marshal: json.Marshal,
|
|
Unmarshal: json.Unmarshal,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Name: "creator",
|
|
Type: textType,
|
|
},
|
|
{
|
|
Name: "owner",
|
|
Type: textType,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
commandArrayCodec = &pgtype.Type{
|
|
Codec: &pgtype.ArrayCodec{
|
|
ElementType: commandType,
|
|
},
|
|
}
|
|
)
|
|
|
|
var typeMu sync.Mutex
|
|
|
|
func RegisterEventstoreTypes(ctx context.Context, conn *pgx.Conn) error {
|
|
// conn.TypeMap is not thread safe
|
|
typeMu.Lock()
|
|
defer typeMu.Unlock()
|
|
|
|
m := conn.TypeMap()
|
|
|
|
var cmd *command
|
|
if _, ok := m.TypeForValue(cmd); ok {
|
|
return nil
|
|
}
|
|
|
|
if commandType.OID == 0 || commandArrayCodec.OID == 0 {
|
|
err := conn.QueryRow(ctx, "select oid, typarray from pg_type where typname = $1 and typnamespace = (select oid from pg_namespace where nspname = $2)", "command", "eventstore").
|
|
Scan(&commandType.OID, &commandArrayCodec.OID)
|
|
if err != nil {
|
|
logging.WithError(err).Debug("failed to get oid for command type")
|
|
return nil
|
|
}
|
|
if commandType.OID == 0 || commandArrayCodec.OID == 0 {
|
|
logging.Debug("oid for command type not found")
|
|
return nil
|
|
}
|
|
}
|
|
|
|
m.RegisterTypes([]*pgtype.Type{
|
|
{
|
|
Name: "eventstore.command",
|
|
Codec: commandType.Codec,
|
|
OID: commandType.OID,
|
|
},
|
|
{
|
|
Name: "command",
|
|
Codec: commandType.Codec,
|
|
OID: commandType.OID,
|
|
},
|
|
{
|
|
Name: "eventstore._command",
|
|
Codec: commandArrayCodec.Codec,
|
|
OID: commandArrayCodec.OID,
|
|
},
|
|
{
|
|
Name: "_command",
|
|
Codec: commandArrayCodec.Codec,
|
|
OID: commandArrayCodec.OID,
|
|
},
|
|
})
|
|
dialect.RegisterDefaultPgTypeVariants[command](m, "eventstore.command", "eventstore._command")
|
|
dialect.RegisterDefaultPgTypeVariants[command](m, "command", "_command")
|
|
|
|
return nil
|
|
}
|
|
|
|
// Client implements the [eventstore.Pusher]
|
|
func (es *Eventstore) Client() *database.DB {
|
|
return es.client
|
|
}
|
|
|
|
func NewEventstore(client *database.DB) *Eventstore {
|
|
return &Eventstore{client: client}
|
|
}
|
|
|
|
func (es *Eventstore) Health(ctx context.Context) error {
|
|
return es.client.PingContext(ctx)
|
|
}
|
|
|
|
var errTypesNotFound = errors.New("types not found")
|
|
|
|
func CheckExecutionPlan(ctx context.Context, conn *sql.Conn) error {
|
|
return conn.Raw(func(driverConn any) error {
|
|
if _, ok := driverConn.(sqlmock.SqlmockCommon); ok {
|
|
return nil
|
|
}
|
|
conn, ok := driverConn.(*stdlib.Conn)
|
|
if !ok {
|
|
return errTypesNotFound
|
|
}
|
|
|
|
return RegisterEventstoreTypes(ctx, conn.Conn())
|
|
})
|
|
}
|
|
|
|
func (es *Eventstore) pushTx(ctx context.Context, client database.ContextQueryExecuter) (tx database.Tx, deferrable func(err error) error, err error) {
|
|
tx, ok := client.(database.Tx)
|
|
if ok {
|
|
return tx, nil, nil
|
|
}
|
|
beginner, ok := client.(database.Beginner)
|
|
if !ok {
|
|
beginner = es.client
|
|
}
|
|
|
|
tx, err = beginner.BeginTx(ctx, &sql.TxOptions{
|
|
Isolation: sql.LevelReadCommitted,
|
|
ReadOnly: false,
|
|
})
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
return tx, func(err error) error { return database.CloseTransaction(tx, err) }, nil
|
|
}
|