2022-08-04 14:37:51 -06:00
|
|
|
package es
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/sour-is/ev/pkg/es/driver"
|
2022-08-09 16:23:33 -06:00
|
|
|
"github.com/sour-is/ev/pkg/es/event"
|
2022-08-04 14:37:51 -06:00
|
|
|
"github.com/sour-is/ev/pkg/locker"
|
|
|
|
)
|
|
|
|
|
|
|
|
type config struct {
|
|
|
|
drivers map[string]driver.Driver
|
|
|
|
}
|
|
|
|
|
2022-08-09 16:23:33 -06:00
|
|
|
const AppendOnly = ^uint64(0)
|
|
|
|
const AllEvents = int64(AppendOnly >> 1)
|
|
|
|
|
2022-08-04 14:37:51 -06:00
|
|
|
var (
|
2022-08-06 09:52:36 -06:00
|
|
|
drivers = locker.New(&config{drivers: make(map[string]driver.Driver)})
|
2022-08-04 14:37:51 -06:00
|
|
|
)
|
|
|
|
|
2022-08-06 09:52:36 -06:00
|
|
|
func Register(ctx context.Context, name string, d driver.Driver) error {
|
|
|
|
return drivers.Modify(ctx, func(c *config) error {
|
2022-08-04 14:37:51 -06:00
|
|
|
if _, set := c.drivers[name]; set {
|
|
|
|
return fmt.Errorf("driver %s already set", name)
|
|
|
|
}
|
|
|
|
c.drivers[name] = d
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-08-09 16:23:33 -06:00
|
|
|
type EventStore struct {
|
|
|
|
driver.Driver
|
|
|
|
}
|
|
|
|
|
|
|
|
func Open(ctx context.Context, dsn string, options ...Option) (*EventStore, error) {
|
2022-08-04 14:37:51 -06:00
|
|
|
name, _, ok := strings.Cut(dsn, ":")
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("%w: no scheme", ErrNoDriver)
|
|
|
|
}
|
|
|
|
|
|
|
|
var d driver.Driver
|
2022-08-09 16:23:33 -06:00
|
|
|
c, err := drivers.Copy(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
d, ok = c.drivers[name]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("%w: %s not registered", ErrNoDriver, name)
|
|
|
|
}
|
|
|
|
|
|
|
|
conn, err := d.Open(ctx, dsn)
|
|
|
|
|
|
|
|
es := &EventStore{Driver: conn}
|
|
|
|
for _, o := range options {
|
|
|
|
o.Apply(es)
|
|
|
|
}
|
|
|
|
|
|
|
|
return es, err
|
|
|
|
}
|
|
|
|
|
|
|
|
type Option interface {
|
|
|
|
Apply(*EventStore)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (es *EventStore) Save(ctx context.Context, agg event.Aggregate) (uint64, error) {
|
|
|
|
l, err := es.EventLog(ctx, agg.StreamID())
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
events := agg.Events(true)
|
|
|
|
|
|
|
|
count, err := l.Append(ctx, events, agg.StreamVersion())
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
agg.Commit()
|
|
|
|
return count, err
|
|
|
|
}
|
|
|
|
func (es *EventStore) Load(ctx context.Context, agg event.Aggregate) error {
|
|
|
|
l, err := es.Driver.EventLog(ctx, agg.StreamID())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
events, err := l.Read(ctx, 0, AllEvents)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
event.Append(agg, events...)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
func (es *EventStore) Read(ctx context.Context, streamID string, pos, count int64) (event.Events, error) {
|
|
|
|
l, err := es.Driver.EventLog(ctx, streamID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return l.Read(ctx, pos, count)
|
|
|
|
}
|
|
|
|
func (es *EventStore) Append(ctx context.Context, streamID string, events event.Events) (uint64, error) {
|
|
|
|
l, err := es.Driver.EventLog(ctx, streamID)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return l.Append(ctx, events, AppendOnly)
|
|
|
|
}
|
|
|
|
func (es *EventStore) FirstIndex(ctx context.Context, streamID string) (uint64, error) {
|
|
|
|
l, err := es.Driver.EventLog(ctx, streamID)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return l.FirstIndex(ctx)
|
|
|
|
}
|
|
|
|
func (es *EventStore) LastIndex(ctx context.Context, streamID string) (uint64, error) {
|
|
|
|
l, err := es.Driver.EventLog(ctx, streamID)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return l.LastIndex(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (es *EventStore) EventStream() driver.EventStream {
|
|
|
|
d := es.Driver
|
|
|
|
for d != nil {
|
|
|
|
if d, ok := d.(driver.EventStream); ok {
|
|
|
|
return d
|
2022-08-04 14:37:51 -06:00
|
|
|
}
|
|
|
|
|
2022-08-09 16:23:33 -06:00
|
|
|
d = Unwrap(d)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Unwrap[T any](t T) T {
|
|
|
|
if unwrap, ok := any(t).(interface{Unwrap() T}); ok {
|
|
|
|
return unwrap.Unwrap()
|
|
|
|
} else {
|
|
|
|
var zero T
|
|
|
|
return zero
|
|
|
|
}
|
2022-08-04 14:37:51 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
var ErrNoDriver = errors.New("no driver")
|