2022-01-20 15:40:25 +01:00
package query
import (
"database/sql"
"database/sql/driver"
"errors"
"fmt"
"regexp"
"testing"
2022-03-23 09:02:39 +01:00
sq "github.com/Masterminds/squirrel"
2022-01-20 15:40:25 +01:00
"golang.org/x/text/language"
2022-08-31 09:52:43 +02:00
"github.com/zitadel/zitadel/internal/database"
2022-04-27 01:01:45 +02:00
"github.com/zitadel/zitadel/internal/domain"
errs "github.com/zitadel/zitadel/internal/errors"
2022-01-20 15:40:25 +01:00
)
var (
2022-11-30 17:01:17 +01:00
loginNamesQuery = ` SELECT login_names.user_id, ARRAY_AGG(login_names.login_name)::TEXT[] AS loginnames, login_names.instance_id, login_names.user_owner_removed, login_names.policy_owner_removed, login_names.domain_owner_removed ` +
` FROM projections.login_names2 AS login_names ` +
` GROUP BY login_names.user_id, login_names.instance_id, login_names.user_owner_removed, login_names.policy_owner_removed, login_names.domain_owner_removed `
preferredLoginNameQuery = ` SELECT preferred_login_name.user_id, preferred_login_name.login_name, preferred_login_name.instance_id, preferred_login_name.user_owner_removed, preferred_login_name.policy_owner_removed, preferred_login_name.domain_owner_removed ` +
` FROM projections.login_names2 AS preferred_login_name ` +
` WHERE preferred_login_name.is_primary = $1 `
userQuery = ` SELECT projections.users6.id, ` +
` projections.users6.creation_date, ` +
` projections.users6.change_date, ` +
` projections.users6.resource_owner, ` +
` projections.users6.sequence, ` +
` projections.users6.state, ` +
` projections.users6.type, ` +
` projections.users6.username, ` +
2022-02-02 08:21:54 +01:00
` login_names.loginnames, ` +
2022-01-20 15:40:25 +01:00
` preferred_login_name.login_name, ` +
2022-11-30 17:01:17 +01:00
` projections.users6_humans.user_id, ` +
` projections.users6_humans.first_name, ` +
` projections.users6_humans.last_name, ` +
` projections.users6_humans.nick_name, ` +
` projections.users6_humans.display_name, ` +
` projections.users6_humans.preferred_language, ` +
` projections.users6_humans.gender, ` +
` projections.users6_humans.avatar_key, ` +
` projections.users6_humans.email, ` +
` projections.users6_humans.is_email_verified, ` +
` projections.users6_humans.phone, ` +
` projections.users6_humans.is_phone_verified, ` +
` projections.users6_machines.user_id, ` +
` projections.users6_machines.name, ` +
` projections.users6_machines.description, ` +
2022-10-17 21:19:15 +02:00
` COUNT(*) OVER () ` +
2022-11-30 17:01:17 +01:00
` FROM projections.users6 ` +
` LEFT JOIN projections.users6_humans ON projections.users6.id = projections.users6_humans.user_id AND projections.users6.instance_id = projections.users6_humans.instance_id ` +
` LEFT JOIN projections.users6_machines ON projections.users6.id = projections.users6_machines.user_id AND projections.users6.instance_id = projections.users6_machines.instance_id ` +
2022-01-20 15:40:25 +01:00
` LEFT JOIN ` +
2022-11-30 17:01:17 +01:00
` ( ` + loginNamesQuery + ` ) AS login_names ` +
` ON login_names.user_id = projections.users6.id AND login_names.instance_id = projections.users6.instance_id ` +
2022-01-20 15:40:25 +01:00
` LEFT JOIN ` +
2022-11-30 17:01:17 +01:00
` ( ` + preferredLoginNameQuery + ` ) AS preferred_login_name ` +
` ON preferred_login_name.user_id = projections.users6.id AND preferred_login_name.instance_id = projections.users6.instance_id `
2022-01-20 15:40:25 +01:00
userCols = [ ] string {
"id" ,
"creation_date" ,
"change_date" ,
"resource_owner" ,
"sequence" ,
"state" ,
"type" ,
"username" ,
2022-02-02 08:21:54 +01:00
"loginnames" ,
2022-01-20 15:40:25 +01:00
"login_name" ,
//human
"user_id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"preferred_language" ,
"gender" ,
"avatar_key" ,
"email" ,
"is_email_verified" ,
"phone" ,
"is_phone_verified" ,
//machine
"user_id" ,
"name" ,
"description" ,
2022-10-17 21:19:15 +02:00
"count" ,
2022-01-20 15:40:25 +01:00
}
2022-11-30 17:01:17 +01:00
profileQuery = ` SELECT projections.users6.id, ` +
` projections.users6.creation_date, ` +
` projections.users6.change_date, ` +
` projections.users6.resource_owner, ` +
` projections.users6.sequence, ` +
` projections.users6_humans.user_id, ` +
` projections.users6_humans.first_name, ` +
` projections.users6_humans.last_name, ` +
` projections.users6_humans.nick_name, ` +
` projections.users6_humans.display_name, ` +
` projections.users6_humans.preferred_language, ` +
` projections.users6_humans.gender, ` +
` projections.users6_humans.avatar_key ` +
` FROM projections.users6 ` +
` LEFT JOIN projections.users6_humans ON projections.users6.id = projections.users6_humans.user_id AND projections.users6.instance_id = projections.users6_humans.instance_id `
2022-01-20 15:40:25 +01:00
profileCols = [ ] string {
"id" ,
"creation_date" ,
"change_date" ,
"resource_owner" ,
"sequence" ,
"user_id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"preferred_language" ,
"gender" ,
"avatar_key" ,
}
2022-11-30 17:01:17 +01:00
emailQuery = ` SELECT projections.users6.id, ` +
` projections.users6.creation_date, ` +
` projections.users6.change_date, ` +
` projections.users6.resource_owner, ` +
` projections.users6.sequence, ` +
` projections.users6_humans.user_id, ` +
` projections.users6_humans.email, ` +
` projections.users6_humans.is_email_verified ` +
` FROM projections.users6 ` +
` LEFT JOIN projections.users6_humans ON projections.users6.id = projections.users6_humans.user_id AND projections.users6.instance_id = projections.users6_humans.instance_id `
2022-01-20 15:40:25 +01:00
emailCols = [ ] string {
"id" ,
"creation_date" ,
"change_date" ,
"resource_owner" ,
"sequence" ,
"user_id" ,
"email" ,
"is_email_verified" ,
}
2022-11-30 17:01:17 +01:00
phoneQuery = ` SELECT projections.users6.id, ` +
` projections.users6.creation_date, ` +
` projections.users6.change_date, ` +
` projections.users6.resource_owner, ` +
` projections.users6.sequence, ` +
` projections.users6_humans.user_id, ` +
` projections.users6_humans.phone, ` +
` projections.users6_humans.is_phone_verified ` +
` FROM projections.users6 ` +
` LEFT JOIN projections.users6_humans ON projections.users6.id = projections.users6_humans.user_id AND projections.users6.instance_id = projections.users6_humans.instance_id `
2022-01-20 15:40:25 +01:00
phoneCols = [ ] string {
"id" ,
"creation_date" ,
"change_date" ,
"resource_owner" ,
"sequence" ,
"user_id" ,
"phone" ,
"is_phone_verified" ,
}
2022-11-30 17:01:17 +01:00
userUniqueQuery = ` SELECT projections.users6.id, ` +
` projections.users6.state, ` +
` projections.users6.username, ` +
` projections.users6_humans.user_id, ` +
` projections.users6_humans.email, ` +
` projections.users6_humans.is_email_verified ` +
` FROM projections.users6 ` +
` LEFT JOIN projections.users6_humans ON projections.users6.id = projections.users6_humans.user_id AND projections.users6.instance_id = projections.users6_humans.instance_id `
2022-01-20 15:40:25 +01:00
userUniqueCols = [ ] string {
"id" ,
"state" ,
"username" ,
"user_id" ,
"email" ,
"is_email_verified" ,
}
2022-11-30 17:01:17 +01:00
notifyUserQuery = ` SELECT projections.users6.id, ` +
` projections.users6.creation_date, ` +
` projections.users6.change_date, ` +
` projections.users6.resource_owner, ` +
` projections.users6.sequence, ` +
` projections.users6.state, ` +
` projections.users6.type, ` +
` projections.users6.username, ` +
2022-07-06 14:09:49 +02:00
` login_names.loginnames, ` +
` preferred_login_name.login_name, ` +
2022-11-30 17:01:17 +01:00
` projections.users6_humans.user_id, ` +
` projections.users6_humans.first_name, ` +
` projections.users6_humans.last_name, ` +
` projections.users6_humans.nick_name, ` +
` projections.users6_humans.display_name, ` +
` projections.users6_humans.preferred_language, ` +
` projections.users6_humans.gender, ` +
` projections.users6_humans.avatar_key, ` +
` projections.users6_notifications.user_id, ` +
` projections.users6_notifications.last_email, ` +
` projections.users6_notifications.verified_email, ` +
` projections.users6_notifications.last_phone, ` +
` projections.users6_notifications.verified_phone, ` +
` projections.users6_notifications.password_set, ` +
2022-10-17 21:19:15 +02:00
` COUNT(*) OVER () ` +
2022-11-30 17:01:17 +01:00
` FROM projections.users6 ` +
` LEFT JOIN projections.users6_humans ON projections.users6.id = projections.users6_humans.user_id AND projections.users6.instance_id = projections.users6_humans.instance_id ` +
` LEFT JOIN projections.users6_notifications ON projections.users6.id = projections.users6_notifications.user_id AND projections.users6.instance_id = projections.users6_notifications.instance_id ` +
2022-07-06 14:09:49 +02:00
` LEFT JOIN ` +
2022-11-30 17:01:17 +01:00
` ( ` + loginNamesQuery + ` ) AS login_names ` +
` ON login_names.user_id = projections.users6.id AND login_names.instance_id = projections.users6.instance_id ` +
2022-07-06 14:09:49 +02:00
` LEFT JOIN ` +
2022-11-30 17:01:17 +01:00
` ( ` + preferredLoginNameQuery + ` ) AS preferred_login_name ` +
` ON preferred_login_name.user_id = projections.users6.id AND preferred_login_name.instance_id = projections.users6.instance_id `
2022-07-06 14:09:49 +02:00
notifyUserCols = [ ] string {
"id" ,
"creation_date" ,
"change_date" ,
"resource_owner" ,
"sequence" ,
"state" ,
"type" ,
"username" ,
"loginnames" ,
"login_name" ,
//human
"user_id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"preferred_language" ,
"gender" ,
"avatar_key" ,
//machine
"user_id" ,
"last_email" ,
"verified_email" ,
"last_phone" ,
"verified_phone" ,
"password_set" ,
2022-10-17 21:19:15 +02:00
"count" ,
2022-07-06 14:09:49 +02:00
}
2022-11-30 17:01:17 +01:00
usersQuery = ` SELECT projections.users6.id, ` +
` projections.users6.creation_date, ` +
` projections.users6.change_date, ` +
` projections.users6.resource_owner, ` +
` projections.users6.sequence, ` +
` projections.users6.state, ` +
` projections.users6.type, ` +
` projections.users6.username, ` +
2022-02-02 08:21:54 +01:00
` login_names.loginnames, ` +
2022-01-20 15:40:25 +01:00
` preferred_login_name.login_name, ` +
2022-11-30 17:01:17 +01:00
` projections.users6_humans.user_id, ` +
` projections.users6_humans.first_name, ` +
` projections.users6_humans.last_name, ` +
` projections.users6_humans.nick_name, ` +
` projections.users6_humans.display_name, ` +
` projections.users6_humans.preferred_language, ` +
` projections.users6_humans.gender, ` +
` projections.users6_humans.avatar_key, ` +
` projections.users6_humans.email, ` +
` projections.users6_humans.is_email_verified, ` +
` projections.users6_humans.phone, ` +
` projections.users6_humans.is_phone_verified, ` +
` projections.users6_machines.user_id, ` +
` projections.users6_machines.name, ` +
` projections.users6_machines.description, ` +
2022-01-20 15:40:25 +01:00
` COUNT(*) OVER () ` +
2022-11-30 17:01:17 +01:00
` FROM projections.users6 ` +
` LEFT JOIN projections.users6_humans ON projections.users6.id = projections.users6_humans.user_id AND projections.users6.instance_id = projections.users6_humans.instance_id ` +
` LEFT JOIN projections.users6_machines ON projections.users6.id = projections.users6_machines.user_id AND projections.users6.instance_id = projections.users6_machines.instance_id ` +
2022-01-20 15:40:25 +01:00
` LEFT JOIN ` +
2022-11-30 17:01:17 +01:00
` ( ` + loginNamesQuery + ` ) AS login_names ` +
` ON login_names.user_id = projections.users6.id AND login_names.instance_id = projections.users6.instance_id ` +
2022-01-20 15:40:25 +01:00
` LEFT JOIN ` +
2022-11-30 17:01:17 +01:00
` ( ` + preferredLoginNameQuery + ` ) AS preferred_login_name ` +
` ON preferred_login_name.user_id = projections.users6.id AND preferred_login_name.instance_id = projections.users6.instance_id `
2022-01-20 15:40:25 +01:00
usersCols = [ ] string {
"id" ,
"creation_date" ,
"change_date" ,
"resource_owner" ,
"sequence" ,
"state" ,
"type" ,
"username" ,
2022-02-02 08:21:54 +01:00
"loginnames" ,
2022-01-20 15:40:25 +01:00
"login_name" ,
//human
"user_id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"preferred_language" ,
"gender" ,
"avatar_key" ,
"email" ,
"is_email_verified" ,
"phone" ,
"is_phone_verified" ,
//machine
"user_id" ,
"name" ,
"description" ,
"count" ,
}
)
func Test_UserPrepares ( t * testing . T ) {
type want struct {
sqlExpectations sqlExpectation
err checkErr
}
tests := [ ] struct {
name string
prepare interface { }
want want
object interface { }
} {
{
2022-03-23 09:02:39 +01:00
name : "prepareUserQuery no result" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Row ) ( * User , error ) ) {
2022-11-07 10:33:22 +01:00
return prepareUserQuery ( )
2022-03-23 09:02:39 +01:00
} ,
2022-01-20 15:40:25 +01:00
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( userQuery ) ,
nil ,
nil ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsNotFound ( err ) {
return fmt . Errorf ( "err should be zitadel.NotFoundError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * User ) ( nil ) ,
} ,
{
2022-03-23 09:02:39 +01:00
name : "prepareUserQuery human found" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Row ) ( * User , error ) ) {
2022-11-07 10:33:22 +01:00
return prepareUserQuery ( )
2022-03-23 09:02:39 +01:00
} ,
2022-01-20 15:40:25 +01:00
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( userQuery ) ,
userCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
domain . UserStateActive ,
domain . UserTypeHuman ,
"username" ,
2022-08-31 09:52:43 +02:00
database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
"login_name1" ,
//human
"id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"de" ,
domain . GenderUnspecified ,
"avatar_key" ,
"email" ,
true ,
"phone" ,
true ,
//machine
nil ,
nil ,
nil ,
2022-10-17 21:19:15 +02:00
1 ,
2022-01-20 15:40:25 +01:00
} ,
) ,
} ,
object : & User {
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
State : domain . UserStateActive ,
Type : domain . UserTypeHuman ,
Username : "username" ,
2022-08-31 09:52:43 +02:00
LoginNames : database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
PreferredLoginName : "login_name1" ,
Human : & Human {
FirstName : "first_name" ,
LastName : "last_name" ,
NickName : "nick_name" ,
DisplayName : "display_name" ,
AvatarKey : "avatar_key" ,
PreferredLanguage : language . German ,
Gender : domain . GenderUnspecified ,
Email : "email" ,
IsEmailVerified : true ,
Phone : "phone" ,
IsPhoneVerified : true ,
} ,
} ,
} ,
{
2022-03-23 09:02:39 +01:00
name : "prepareUserQuery machine found" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Row ) ( * User , error ) ) {
2022-11-07 10:33:22 +01:00
return prepareUserQuery ( )
2022-03-23 09:02:39 +01:00
} ,
2022-01-20 15:40:25 +01:00
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( userQuery ) ,
userCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
domain . UserStateActive ,
domain . UserTypeMachine ,
"username" ,
2022-08-31 09:52:43 +02:00
database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
"login_name1" ,
//human
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
//machine
"id" ,
"name" ,
"description" ,
2022-10-17 21:19:15 +02:00
1 ,
2022-01-20 15:40:25 +01:00
} ,
) ,
} ,
object : & User {
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
State : domain . UserStateActive ,
Type : domain . UserTypeMachine ,
Username : "username" ,
2022-08-31 09:52:43 +02:00
LoginNames : database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
PreferredLoginName : "login_name1" ,
Machine : & Machine {
Name : "name" ,
Description : "description" ,
} ,
} ,
} ,
{
2022-03-23 09:02:39 +01:00
name : "prepareUserQuery sql err" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Row ) ( * User , error ) ) {
2022-11-07 10:33:22 +01:00
return prepareUserQuery ( )
2022-03-23 09:02:39 +01:00
} ,
2022-01-20 15:40:25 +01:00
want : want {
sqlExpectations : mockQueryErr (
regexp . QuoteMeta ( userQuery ) ,
sql . ErrConnDone ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errors . Is ( err , sql . ErrConnDone ) {
return fmt . Errorf ( "err should be sql.ErrConnDone got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : nil ,
} ,
{
name : "prepareProfileQuery no result" ,
prepare : prepareProfileQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( profileQuery ) ,
nil ,
nil ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsNotFound ( err ) {
return fmt . Errorf ( "err should be zitadel.NotFoundError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * Profile ) ( nil ) ,
} ,
{
name : "prepareProfileQuery human found" ,
prepare : prepareProfileQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( profileQuery ) ,
profileCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
"id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"de" ,
domain . GenderUnspecified ,
"avatar_key" ,
} ,
) ,
} ,
object : & Profile {
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
FirstName : "first_name" ,
LastName : "last_name" ,
NickName : "nick_name" ,
DisplayName : "display_name" ,
AvatarKey : "avatar_key" ,
PreferredLanguage : language . German ,
Gender : domain . GenderUnspecified ,
} ,
} ,
{
name : "prepareProfileQuery not human found (error)" ,
prepare : prepareProfileQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( profileQuery ) ,
profileCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
} ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsPreconditionFailed ( err ) {
return fmt . Errorf ( "err should be zitadel.PredconditionError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * Profile ) ( nil ) ,
} ,
{
name : "prepareProfileQuery sql err" ,
prepare : prepareProfileQuery ,
want : want {
sqlExpectations : mockQueryErr (
regexp . QuoteMeta ( profileQuery ) ,
sql . ErrConnDone ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errors . Is ( err , sql . ErrConnDone ) {
return fmt . Errorf ( "err should be sql.ErrConnDone got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : nil ,
} ,
{
name : "prepareEmailQuery no result" ,
prepare : prepareEmailQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( emailQuery ) ,
nil ,
nil ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsNotFound ( err ) {
return fmt . Errorf ( "err should be zitadel.NotFoundError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * Email ) ( nil ) ,
} ,
{
name : "prepareEmailQuery human found" ,
prepare : prepareEmailQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( emailQuery ) ,
emailCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
//domain.UserStateActive,
"id" ,
"email" ,
true ,
} ,
) ,
} ,
object : & Email {
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
//State: domain.UserStateActive,
Email : "email" ,
IsVerified : true ,
} ,
} ,
{
name : "prepareEmailQuery not human found (error)" ,
prepare : prepareEmailQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( emailQuery ) ,
emailCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
//domain.UserStateActive,
nil ,
nil ,
nil ,
} ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsPreconditionFailed ( err ) {
return fmt . Errorf ( "err should be zitadel.PredconditionError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * Email ) ( nil ) ,
} ,
{
name : "prepareEmailQuery sql err" ,
prepare : prepareEmailQuery ,
want : want {
sqlExpectations : mockQueryErr (
regexp . QuoteMeta ( emailQuery ) ,
sql . ErrConnDone ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errors . Is ( err , sql . ErrConnDone ) {
return fmt . Errorf ( "err should be sql.ErrConnDone got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : nil ,
} ,
{
name : "preparePhoneQuery no result" ,
prepare : preparePhoneQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( phoneQuery ) ,
nil ,
nil ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsNotFound ( err ) {
return fmt . Errorf ( "err should be zitadel.NotFoundError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * Phone ) ( nil ) ,
} ,
{
name : "preparePhoneQuery human found" ,
prepare : preparePhoneQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( phoneQuery ) ,
phoneCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
//domain.UserStateActive,
"id" ,
"phone" ,
true ,
} ,
) ,
} ,
object : & Phone {
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
//State: domain.UserStateActive,
Phone : "phone" ,
IsVerified : true ,
} ,
} ,
{
name : "preparePhoneQuery not human found (error)" ,
prepare : preparePhoneQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( phoneQuery ) ,
phoneCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
//domain.UserStateActive,
nil ,
nil ,
nil ,
} ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsPreconditionFailed ( err ) {
return fmt . Errorf ( "err should be zitadel.PredconditionError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * Phone ) ( nil ) ,
} ,
{
name : "preparePhoneQuery sql err" ,
prepare : preparePhoneQuery ,
want : want {
sqlExpectations : mockQueryErr (
regexp . QuoteMeta ( phoneQuery ) ,
sql . ErrConnDone ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errors . Is ( err , sql . ErrConnDone ) {
return fmt . Errorf ( "err should be sql.ErrConnDone got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : nil ,
} ,
{
name : "prepareUserUniqueQuery no result" ,
prepare : prepareUserUniqueQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( userUniqueQuery ) ,
nil ,
nil ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsNotFound ( err ) {
return fmt . Errorf ( "err should be zitadel.NotFoundError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : true ,
} ,
{
name : "prepareUserUniqueQuery found" ,
prepare : prepareUserUniqueQuery ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( userUniqueQuery ) ,
userUniqueCols ,
[ ] driver . Value {
"id" ,
domain . UserStateActive ,
"username" ,
"id" ,
"email" ,
true ,
} ,
) ,
} ,
object : false ,
} ,
{
name : "prepareUserUniqueQuery sql err" ,
prepare : prepareUserUniqueQuery ,
want : want {
sqlExpectations : mockQueryErr (
regexp . QuoteMeta ( userUniqueQuery ) ,
sql . ErrConnDone ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errors . Is ( err , sql . ErrConnDone ) {
return fmt . Errorf ( "err should be sql.ErrConnDone got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : nil ,
} ,
2022-07-06 14:09:49 +02:00
{
name : "prepareNotifyUserQuery no result" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Row ) ( * NotifyUser , error ) ) {
2022-11-07 10:33:22 +01:00
return prepareNotifyUserQuery ( )
2022-07-06 14:09:49 +02:00
} ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( notifyUserQuery ) ,
nil ,
nil ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsNotFound ( err ) {
return fmt . Errorf ( "err should be zitadel.NotFoundError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * NotifyUser ) ( nil ) ,
} ,
{
name : "prepareNotifyUserQuery notify found" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Row ) ( * NotifyUser , error ) ) {
2022-11-07 10:33:22 +01:00
return prepareNotifyUserQuery ( )
2022-07-06 14:09:49 +02:00
} ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( notifyUserQuery ) ,
notifyUserCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
domain . UserStateActive ,
domain . UserTypeHuman ,
"username" ,
2022-08-31 09:52:43 +02:00
database . StringArray { "login_name1" , "login_name2" } ,
2022-07-06 14:09:49 +02:00
"login_name1" ,
//human
"id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"de" ,
domain . GenderUnspecified ,
"avatar_key" ,
//notify
"id" ,
"lastEmail" ,
"verifiedEmail" ,
"lastPhone" ,
"verifiedPhone" ,
true ,
2022-10-17 21:19:15 +02:00
1 ,
2022-07-06 14:09:49 +02:00
} ,
) ,
} ,
object : & NotifyUser {
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
State : domain . UserStateActive ,
Type : domain . UserTypeHuman ,
Username : "username" ,
2022-08-31 09:52:43 +02:00
LoginNames : database . StringArray { "login_name1" , "login_name2" } ,
2022-07-06 14:09:49 +02:00
PreferredLoginName : "login_name1" ,
FirstName : "first_name" ,
LastName : "last_name" ,
NickName : "nick_name" ,
DisplayName : "display_name" ,
AvatarKey : "avatar_key" ,
PreferredLanguage : language . German ,
Gender : domain . GenderUnspecified ,
LastEmail : "lastEmail" ,
VerifiedEmail : "verifiedEmail" ,
LastPhone : "lastPhone" ,
VerifiedPhone : "verifiedPhone" ,
PasswordSet : true ,
} ,
} ,
{
name : "prepareNotifyUserQuery not notify found (error)" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Row ) ( * NotifyUser , error ) ) {
2022-11-07 10:33:22 +01:00
return prepareNotifyUserQuery ( )
2022-07-06 14:09:49 +02:00
} ,
want : want {
sqlExpectations : mockQuery (
regexp . QuoteMeta ( notifyUserQuery ) ,
notifyUserCols ,
[ ] driver . Value {
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
domain . UserStateActive ,
domain . UserTypeHuman ,
"username" ,
2022-08-31 09:52:43 +02:00
database . StringArray { "login_name1" , "login_name2" } ,
2022-07-06 14:09:49 +02:00
"login_name1" ,
//human
"id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"de" ,
domain . GenderUnspecified ,
"avatar_key" ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
2022-10-17 21:19:15 +02:00
1 ,
2022-07-06 14:09:49 +02:00
} ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsPreconditionFailed ( err ) {
return fmt . Errorf ( "err should be zitadel.PredconditionError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : ( * NotifyUser ) ( nil ) ,
} ,
{
name : "prepareNotifyUserQuery sql err" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Row ) ( * NotifyUser , error ) ) {
2022-11-07 10:33:22 +01:00
return prepareNotifyUserQuery ( )
2022-07-06 14:09:49 +02:00
} ,
want : want {
sqlExpectations : mockQueryErr (
regexp . QuoteMeta ( notifyUserQuery ) ,
sql . ErrConnDone ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errors . Is ( err , sql . ErrConnDone ) {
return fmt . Errorf ( "err should be sql.ErrConnDone got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : nil ,
} ,
2022-01-20 15:40:25 +01:00
{
2022-11-07 10:33:22 +01:00
name : "prepareUsersQuery no result" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Rows ) ( * Users , error ) ) {
return prepareUsersQuery ( )
} ,
2022-01-20 15:40:25 +01:00
want : want {
sqlExpectations : mockQueries (
regexp . QuoteMeta ( usersQuery ) ,
nil ,
nil ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errs . IsNotFound ( err ) {
return fmt . Errorf ( "err should be zitadel.NotFoundError got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : & Users { Users : [ ] * User { } } ,
} ,
{
2022-11-07 10:33:22 +01:00
name : "prepareUsersQuery one result" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Rows ) ( * Users , error ) ) {
return prepareUsersQuery ( )
} ,
2022-01-20 15:40:25 +01:00
want : want {
sqlExpectations : mockQueries (
regexp . QuoteMeta ( usersQuery ) ,
usersCols ,
[ ] [ ] driver . Value {
{
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
domain . UserStateActive ,
domain . UserTypeHuman ,
"username" ,
2022-08-31 09:52:43 +02:00
database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
"login_name1" ,
//human
"id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"de" ,
domain . GenderUnspecified ,
"avatar_key" ,
"email" ,
true ,
"phone" ,
true ,
//machine
nil ,
nil ,
nil ,
} ,
} ,
) ,
} ,
object : & Users {
SearchResponse : SearchResponse {
Count : 1 ,
} ,
Users : [ ] * User {
{
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
State : domain . UserStateActive ,
Type : domain . UserTypeHuman ,
Username : "username" ,
2022-08-31 09:52:43 +02:00
LoginNames : database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
PreferredLoginName : "login_name1" ,
Human : & Human {
FirstName : "first_name" ,
LastName : "last_name" ,
NickName : "nick_name" ,
DisplayName : "display_name" ,
AvatarKey : "avatar_key" ,
PreferredLanguage : language . German ,
Gender : domain . GenderUnspecified ,
Email : "email" ,
IsEmailVerified : true ,
Phone : "phone" ,
IsPhoneVerified : true ,
} ,
} ,
} ,
} ,
} ,
{
2022-11-07 10:33:22 +01:00
name : "prepareUsersQuery multiple results" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Rows ) ( * Users , error ) ) {
return prepareUsersQuery ( )
} ,
2022-01-20 15:40:25 +01:00
want : want {
sqlExpectations : mockQueries (
regexp . QuoteMeta ( usersQuery ) ,
usersCols ,
[ ] [ ] driver . Value {
{
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
domain . UserStateActive ,
domain . UserTypeHuman ,
"username" ,
2022-08-31 09:52:43 +02:00
database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
"login_name1" ,
//human
"id" ,
"first_name" ,
"last_name" ,
"nick_name" ,
"display_name" ,
"de" ,
domain . GenderUnspecified ,
"avatar_key" ,
"email" ,
true ,
"phone" ,
true ,
//machine
nil ,
nil ,
nil ,
} ,
{
"id" ,
testNow ,
testNow ,
"resource_owner" ,
uint64 ( 20211108 ) ,
domain . UserStateActive ,
domain . UserTypeMachine ,
"username" ,
2022-08-31 09:52:43 +02:00
database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
"login_name1" ,
//human
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
nil ,
//machine
"id" ,
"name" ,
"description" ,
} ,
} ,
) ,
} ,
object : & Users {
SearchResponse : SearchResponse {
Count : 2 ,
} ,
Users : [ ] * User {
{
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
State : domain . UserStateActive ,
Type : domain . UserTypeHuman ,
Username : "username" ,
2022-08-31 09:52:43 +02:00
LoginNames : database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
PreferredLoginName : "login_name1" ,
Human : & Human {
FirstName : "first_name" ,
LastName : "last_name" ,
NickName : "nick_name" ,
DisplayName : "display_name" ,
AvatarKey : "avatar_key" ,
PreferredLanguage : language . German ,
Gender : domain . GenderUnspecified ,
Email : "email" ,
IsEmailVerified : true ,
Phone : "phone" ,
IsPhoneVerified : true ,
} ,
} ,
{
ID : "id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
ResourceOwner : "resource_owner" ,
Sequence : 20211108 ,
State : domain . UserStateActive ,
Type : domain . UserTypeMachine ,
Username : "username" ,
2022-08-31 09:52:43 +02:00
LoginNames : database . StringArray { "login_name1" , "login_name2" } ,
2022-01-20 15:40:25 +01:00
PreferredLoginName : "login_name1" ,
Machine : & Machine {
Name : "name" ,
Description : "description" ,
} ,
} ,
} ,
} ,
} ,
{
2022-11-07 10:33:22 +01:00
name : "prepareUsersQuery sql err" ,
prepare : func ( ) ( sq . SelectBuilder , func ( * sql . Rows ) ( * Users , error ) ) {
return prepareUsersQuery ( )
} ,
2022-01-20 15:40:25 +01:00
want : want {
sqlExpectations : mockQueryErr (
regexp . QuoteMeta ( usersQuery ) ,
sql . ErrConnDone ,
) ,
err : func ( err error ) ( error , bool ) {
if ! errors . Is ( err , sql . ErrConnDone ) {
return fmt . Errorf ( "err should be sql.ErrConnDone got: %w" , err ) , false
}
return nil , true
} ,
} ,
object : nil ,
} ,
}
for _ , tt := range tests {
t . Run ( tt . name , func ( t * testing . T ) {
assertPrepare ( t , tt . prepare , tt . object , tt . want . sqlExpectations , tt . want . err )
} )
}
}