Files
zitadel/backend/v3/domain/invoke.go

159 lines
4.3 KiB
Go
Raw Normal View History

2025-04-29 06:03:47 +02:00
package domain
import (
"context"
"fmt"
"github.com/zitadel/zitadel/backend/v3/storage/eventstore"
)
2025-05-08 19:01:55 +02:00
// Invoke provides a way to execute commands within the domain package.
// It uses a chain of responsibility pattern to handle the command execution.
// The default chain includes logging, tracing, and event publishing.
// If you want to invoke multiple commands in a single transaction, you can use the [commandBatch].
2025-04-29 06:03:47 +02:00
func Invoke(ctx context.Context, cmd Commander) error {
2025-05-08 15:30:06 +02:00
invoker := newEventStoreInvoker(newLoggingInvoker(newTraceInvoker(nil)))
2025-04-29 06:03:47 +02:00
opts := &CommandOpts{
Invoker: invoker.collector,
2025-05-08 15:30:06 +02:00
DB: pool,
2025-04-29 06:03:47 +02:00
}
return invoker.Invoke(ctx, cmd, opts)
}
2025-05-08 19:01:55 +02:00
// eventStoreInvoker checks if the command implements the [eventer] interface.
// If it does, it collects the events and publishes them to the event store.
2025-04-29 06:03:47 +02:00
type eventStoreInvoker struct {
collector *eventCollector
}
func newEventStoreInvoker(next Invoker) *eventStoreInvoker {
return &eventStoreInvoker{collector: &eventCollector{next: next}}
}
func (i *eventStoreInvoker) Invoke(ctx context.Context, command Commander, opts *CommandOpts) (err error) {
err = i.collector.Invoke(ctx, command, opts)
if err != nil {
return err
}
if len(i.collector.events) > 0 {
err = eventstore.Publish(ctx, i.collector.events, opts.DB)
if err != nil {
return err
}
}
return nil
}
2025-05-08 19:01:55 +02:00
// eventCollector collects events from all commands. The [eventStoreInvoker] pushes the collected events after all commands are executed.
2025-04-29 06:03:47 +02:00
type eventCollector struct {
next Invoker
events []*eventstore.Event
}
type eventer interface {
Events() []*eventstore.Event
}
func (i *eventCollector) Invoke(ctx context.Context, command Commander, opts *CommandOpts) (err error) {
if e, ok := command.(eventer); ok && len(e.Events()) > 0 {
// we need to ensure all commands are executed in the same transaction
close, err := opts.EnsureTx(ctx)
if err != nil {
return err
}
defer func() { err = close(ctx, err) }()
i.events = append(i.events, e.Events()...)
}
if i.next != nil {
2025-05-08 15:30:06 +02:00
return i.next.Invoke(ctx, command, opts)
2025-04-29 06:03:47 +02:00
}
2025-05-08 15:30:06 +02:00
return command.Execute(ctx, opts)
2025-04-29 06:03:47 +02:00
}
2025-05-08 19:01:55 +02:00
// traceInvoker decorates each command with tracing.
2025-04-29 06:03:47 +02:00
type traceInvoker struct {
next Invoker
}
func newTraceInvoker(next Invoker) *traceInvoker {
return &traceInvoker{next: next}
}
func (i *traceInvoker) Invoke(ctx context.Context, command Commander, opts *CommandOpts) (err error) {
ctx, span := tracer.Start(ctx, fmt.Sprintf("%T", command))
2025-05-08 15:30:06 +02:00
defer func() {
if err != nil {
span.RecordError(err)
}
span.End()
}()
if i.next != nil {
return i.next.Invoke(ctx, command, opts)
}
return command.Execute(ctx, opts)
}
2025-05-08 19:01:55 +02:00
// loggingInvoker decorates each command with logging.
// It is an example implementation and logs the command name at the beginning and success or failure after the command got executed.
2025-05-08 15:30:06 +02:00
type loggingInvoker struct {
next Invoker
}
func newLoggingInvoker(next Invoker) *loggingInvoker {
return &loggingInvoker{next: next}
}
func (i *loggingInvoker) Invoke(ctx context.Context, command Commander, opts *CommandOpts) (err error) {
logger.InfoContext(ctx, "Invoking command", "command", command.String())
2025-04-29 06:03:47 +02:00
if i.next != nil {
err = i.next.Invoke(ctx, command, opts)
} else {
err = command.Execute(ctx, opts)
}
2025-05-08 15:30:06 +02:00
2025-04-29 06:03:47 +02:00
if err != nil {
2025-05-08 15:30:06 +02:00
logger.ErrorContext(ctx, "Command invocation failed", "command", command.String(), "error", err)
return err
}
logger.InfoContext(ctx, "Command invocation succeeded", "command", command.String())
return nil
}
type noopInvoker struct {
next Invoker
}
func (i *noopInvoker) Invoke(ctx context.Context, command Commander, opts *CommandOpts) error {
if i.next != nil {
return i.next.Invoke(ctx, command, opts)
}
return command.Execute(ctx, opts)
}
2025-05-08 19:01:55 +02:00
// cacheInvoker could be used in the future to do the caching.
// My goal would be to have two interfaces:
// - cacheSetter: which caches an object
// - cacheGetter: which gets an object from the cache, this should also skip the command execution
2025-05-08 15:30:06 +02:00
type cacheInvoker struct {
next Invoker
}
type cacher interface {
Cache(opts *CommandOpts)
}
func (i *cacheInvoker) Invoke(ctx context.Context, command Commander, opts *CommandOpts) (err error) {
if c, ok := command.(cacher); ok {
c.Cache(opts)
}
if i.next != nil {
err = i.next.Invoke(ctx, command, opts)
} else {
err = command.Execute(ctx, opts)
2025-04-29 06:03:47 +02:00
}
return err
}