package migration import ( "context" "encoding/json" "github.com/zitadel/zitadel/internal/api/authz" "github.com/zitadel/zitadel/internal/api/service" "github.com/zitadel/zitadel/internal/errors" "github.com/zitadel/zitadel/internal/eventstore" "github.com/zitadel/zitadel/internal/eventstore/repository" ) //SetupStep is the command pushed on the eventstore type SetupStep struct { eventstore.BaseEvent `json:"-"` migration Migration Name string `json:"name"` Error error `json:"error,omitempty"` } func (s *SetupStep) UnmarshalJSON(data []byte) error { fields := struct { Name string `json:"name,"` Error *errors.CaosError `json:"error"` }{} if err := json.Unmarshal(data, &fields); err != nil { return err } s.Name = fields.Name s.Error = fields.Error return nil } func setupStartedCmd(migration Migration) eventstore.Command { ctx := authz.SetCtxData(service.WithService(context.Background(), "system"), authz.CtxData{UserID: "system", OrgID: "SYSTEM", ResourceOwner: "SYSTEM"}) return &SetupStep{ BaseEvent: *eventstore.NewBaseEventForPush( ctx, eventstore.NewAggregate(ctx, aggregateID, aggregateType, "v1"), startedType), migration: migration, Name: migration.String(), } } func setupDoneCmd(migration Migration, err error) eventstore.Command { ctx := authz.SetCtxData(service.WithService(context.Background(), "system"), authz.CtxData{UserID: "system", OrgID: "SYSTEM", ResourceOwner: "SYSTEM"}) s := &SetupStep{ migration: migration, Name: migration.String(), Error: err, } typ := doneType if err != nil { typ = failedType } s.BaseEvent = *eventstore.NewBaseEventForPush( ctx, eventstore.NewAggregate(ctx, aggregateID, aggregateType, "v1"), typ) return s } func (s *SetupStep) Data() interface{} { return s } func (s *SetupStep) UniqueConstraints() []*eventstore.EventUniqueConstraint { switch s.Type() { case startedType: return []*eventstore.EventUniqueConstraint{ eventstore.NewAddGlobalEventUniqueConstraint("migration_started", s.migration.String(), "Errors.Step.Started.AlreadyExists"), } case failedType: return []*eventstore.EventUniqueConstraint{ eventstore.NewRemoveGlobalEventUniqueConstraint("migration_started", s.migration.String()), } default: return []*eventstore.EventUniqueConstraint{ eventstore.NewAddGlobalEventUniqueConstraint("migration_done", s.migration.String(), "Errors.Step.Done.AlreadyExists"), } } } func RegisterMappers(es *eventstore.Eventstore) { es.RegisterFilterEventMapper(startedType, SetupMapper) es.RegisterFilterEventMapper(doneType, SetupMapper) es.RegisterFilterEventMapper(failedType, SetupMapper) } func SetupMapper(event *repository.Event) (eventstore.Event, error) { step := &SetupStep{ BaseEvent: *eventstore.BaseEventFromRepo(event), } if len(event.Data) == 0 { return step, nil } err := json.Unmarshal(event.Data, step) if err != nil { return nil, errors.ThrowInternal(err, "IAM-hYp7M", "unable to unmarshal step") } return step, nil }