package command import ( "context" "testing" "github.com/stretchr/testify/assert" "github.com/zitadel/zitadel/internal/domain" caos_errs "github.com/zitadel/zitadel/internal/errors" "github.com/zitadel/zitadel/internal/eventstore" "github.com/zitadel/zitadel/internal/eventstore/repository" "github.com/zitadel/zitadel/internal/eventstore/v1/models" "github.com/zitadel/zitadel/internal/id" id_mock "github.com/zitadel/zitadel/internal/id/mock" "github.com/zitadel/zitadel/internal/repository/org" "github.com/zitadel/zitadel/internal/repository/user" ) func TestCommandSide_AddMachine(t *testing.T) { type fields struct { eventstore *eventstore.Eventstore idGenerator id.Generator } type args struct { ctx context.Context orgID string machine *domain.Machine } type res struct { want *domain.Machine err func(error) bool } tests := []struct { name string fields fields args args res res }{ { name: "user invalid, invalid argument error", fields: fields{ eventstore: eventstoreExpect( t, ), }, args: args{ ctx: context.Background(), orgID: "org1", machine: &domain.Machine{ Username: "username", }, }, res: res{ err: caos_errs.IsErrorInvalidArgument, }, }, { name: "org policy not found, precondition error", fields: fields{ eventstore: eventstoreExpect( t, expectFilter(), expectFilter(), ), }, args: args{ ctx: context.Background(), orgID: "org1", machine: &domain.Machine{ Username: "username", Name: "name", }, }, res: res{ err: caos_errs.IsPreconditionFailed, }, }, { name: "org policy global, precondition error", fields: fields{ eventstore: eventstoreExpect( t, expectFilter( eventFromEventPusher( org.NewDomainPolicyAddedEvent(context.Background(), &user.NewAggregate("user1", "org1").Aggregate, false, true, ), ), ), ), }, args: args{ ctx: context.Background(), orgID: "org1", machine: &domain.Machine{ Username: "username", Name: "name", }, }, res: res{ err: caos_errs.IsPreconditionFailed, }, }, { name: "add machine, ok", fields: fields{ eventstore: eventstoreExpect( t, expectFilter( eventFromEventPusher( org.NewDomainPolicyAddedEvent(context.Background(), &user.NewAggregate("user1", "org1").Aggregate, true, true, ), ), ), expectPush( []*repository.Event{ eventFromEventPusher( user.NewMachineAddedEvent(context.Background(), &user.NewAggregate("user1", "org1").Aggregate, "username", "name", "description", true, ), ), }, uniqueConstraintsFromEventConstraint(user.NewAddUsernameUniqueConstraint("username", "org1", true)), ), ), idGenerator: id_mock.NewIDGeneratorExpectIDs(t, "user1"), }, args: args{ ctx: context.Background(), orgID: "org1", machine: &domain.Machine{ Username: "username", Description: "description", Name: "name", }, }, res: res{ want: &domain.Machine{ ObjectRoot: models.ObjectRoot{ AggregateID: "user1", ResourceOwner: "org1", }, Username: "username", Name: "name", Description: "description", State: domain.UserStateActive, }, }, }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { r := &Commands{ eventstore: tt.fields.eventstore, idGenerator: tt.fields.idGenerator, } got, err := r.AddMachine(tt.args.ctx, tt.args.orgID, tt.args.machine) if tt.res.err == nil { assert.NoError(t, err) } if tt.res.err != nil && !tt.res.err(err) { t.Errorf("got wrong err: %v ", err) } if tt.res.err == nil { assert.Equal(t, tt.res.want, got) } }) } } func TestCommandSide_ChangeMachine(t *testing.T) { type fields struct { eventstore *eventstore.Eventstore } type args struct { ctx context.Context orgID string machine *domain.Machine } type res struct { want *domain.Machine err func(error) bool } tests := []struct { name string fields fields args args res res }{ { name: "user invalid, invalid argument error", fields: fields{ eventstore: eventstoreExpect( t, ), }, args: args{ ctx: context.Background(), orgID: "org1", machine: &domain.Machine{ Username: "username", }, }, res: res{ err: caos_errs.IsErrorInvalidArgument, }, }, { name: "user not existing, precondition error", fields: fields{ eventstore: eventstoreExpect( t, expectFilter(), ), }, args: args{ ctx: context.Background(), orgID: "org1", machine: &domain.Machine{ ObjectRoot: models.ObjectRoot{ AggregateID: "user1", }, Username: "username", Name: "name", }, }, res: res{ err: caos_errs.IsNotFound, }, }, { name: "no changes, precondition error", fields: fields{ eventstore: eventstoreExpect( t, expectFilter( eventFromEventPusher( user.NewMachineAddedEvent(context.Background(), &user.NewAggregate("user1", "org1").Aggregate, "username", "name", "description", true, ), ), ), ), }, args: args{ ctx: context.Background(), orgID: "org1", machine: &domain.Machine{ ObjectRoot: models.ObjectRoot{ AggregateID: "user1", }, Name: "name", Description: "description", }, }, res: res{ err: caos_errs.IsPreconditionFailed, }, }, { name: "change machine, ok", fields: fields{ eventstore: eventstoreExpect( t, expectFilter( eventFromEventPusher( user.NewMachineAddedEvent(context.Background(), &user.NewAggregate("user1", "org1").Aggregate, "username", "name", "description", true, ), ), ), expectPush( []*repository.Event{ eventFromEventPusher( newMachineChangedEvent(context.Background(), "user1", "org1", "name1", "description1"), ), }, ), ), }, args: args{ ctx: context.Background(), orgID: "org1", machine: &domain.Machine{ ObjectRoot: models.ObjectRoot{ AggregateID: "user1", }, Description: "description1", Name: "name1", }, }, res: res{ want: &domain.Machine{ ObjectRoot: models.ObjectRoot{ AggregateID: "user1", ResourceOwner: "org1", }, Username: "username", Name: "name1", Description: "description1", State: domain.UserStateActive, }, }, }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { r := &Commands{ eventstore: tt.fields.eventstore, } got, err := r.ChangeMachine(tt.args.ctx, tt.args.machine) if tt.res.err == nil { assert.NoError(t, err) } if tt.res.err != nil && !tt.res.err(err) { t.Errorf("got wrong err: %v ", err) } if tt.res.err == nil { assert.Equal(t, tt.res.want, got) } }) } } func newMachineChangedEvent(ctx context.Context, userID, resourceOwner, name, description string) *user.MachineChangedEvent { event, _ := user.NewMachineChangedEvent(ctx, &user.NewAggregate(userID, resourceOwner).Aggregate, []user.MachineChanges{ user.ChangeName(name), user.ChangeDescription(description), }, ) return event }