2023-05-05 17:34:53 +02:00
package query
import (
"context"
"database/sql"
"database/sql/driver"
"errors"
"fmt"
"regexp"
"testing"
sq "github.com/Masterminds/squirrel"
"github.com/stretchr/testify/require"
"github.com/zitadel/zitadel/internal/domain"
errs "github.com/zitadel/zitadel/internal/errors"
)
var (
2023-06-27 14:36:07 +02:00
expectedSessionQuery = regexp . QuoteMeta ( ` SELECT projections.sessions3.id, ` +
` projections.sessions3.creation_date, ` +
` projections.sessions3.change_date, ` +
` projections.sessions3.sequence, ` +
` projections.sessions3.state, ` +
` projections.sessions3.resource_owner, ` +
` projections.sessions3.creator, ` +
` projections.sessions3.domain, ` +
` projections.sessions3.user_id, ` +
` projections.sessions3.user_checked_at, ` +
2023-05-05 17:34:53 +02:00
` projections.login_names2.login_name, ` +
` projections.users8_humans.display_name, ` +
2023-06-27 14:36:07 +02:00
` projections.sessions3.password_checked_at, ` +
` projections.sessions3.intent_checked_at, ` +
` projections.sessions3.passkey_checked_at, ` +
` projections.sessions3.metadata, ` +
` projections.sessions3.token_id ` +
` FROM projections.sessions3 ` +
` LEFT JOIN projections.login_names2 ON projections.sessions3.user_id = projections.login_names2.user_id AND projections.sessions3.instance_id = projections.login_names2.instance_id ` +
` LEFT JOIN projections.users8_humans ON projections.sessions3.user_id = projections.users8_humans.user_id AND projections.sessions3.instance_id = projections.users8_humans.instance_id ` +
2023-05-05 17:34:53 +02:00
` AS OF SYSTEM TIME '-1 ms' ` )
2023-06-27 14:36:07 +02:00
expectedSessionsQuery = regexp . QuoteMeta ( ` SELECT projections.sessions3.id, ` +
` projections.sessions3.creation_date, ` +
` projections.sessions3.change_date, ` +
` projections.sessions3.sequence, ` +
` projections.sessions3.state, ` +
` projections.sessions3.resource_owner, ` +
` projections.sessions3.creator, ` +
` projections.sessions3.domain, ` +
` projections.sessions3.user_id, ` +
` projections.sessions3.user_checked_at, ` +
2023-05-05 17:34:53 +02:00
` projections.login_names2.login_name, ` +
` projections.users8_humans.display_name, ` +
2023-06-27 14:36:07 +02:00
` projections.sessions3.password_checked_at, ` +
` projections.sessions3.intent_checked_at, ` +
` projections.sessions3.passkey_checked_at, ` +
` projections.sessions3.metadata, ` +
2023-05-05 17:34:53 +02:00
` COUNT(*) OVER () ` +
2023-06-27 14:36:07 +02:00
` FROM projections.sessions3 ` +
` LEFT JOIN projections.login_names2 ON projections.sessions3.user_id = projections.login_names2.user_id AND projections.sessions3.instance_id = projections.login_names2.instance_id ` +
` LEFT JOIN projections.users8_humans ON projections.sessions3.user_id = projections.users8_humans.user_id AND projections.sessions3.instance_id = projections.users8_humans.instance_id ` +
2023-05-05 17:34:53 +02:00
` AS OF SYSTEM TIME '-1 ms' ` )
sessionCols = [ ] string {
"id" ,
"creation_date" ,
"change_date" ,
"sequence" ,
"state" ,
"resource_owner" ,
"creator" ,
2023-06-27 14:36:07 +02:00
"domain" ,
2023-05-05 17:34:53 +02:00
"user_id" ,
"user_checked_at" ,
"login_name" ,
"display_name" ,
"password_checked_at" ,
2023-06-21 16:06:18 +02:00
"intent_checked_at" ,
2023-06-07 17:28:42 +02:00
"passkey_checked_at" ,
2023-05-05 17:34:53 +02:00
"metadata" ,
"token" ,
}
sessionsCols = [ ] string {
"id" ,
"creation_date" ,
"change_date" ,
"sequence" ,
"state" ,
"resource_owner" ,
"creator" ,
2023-06-27 14:36:07 +02:00
"domain" ,
2023-05-05 17:34:53 +02:00
"user_id" ,
"user_checked_at" ,
"login_name" ,
"display_name" ,
"password_checked_at" ,
2023-06-21 16:06:18 +02:00
"intent_checked_at" ,
2023-06-07 17:28:42 +02:00
"passkey_checked_at" ,
2023-05-05 17:34:53 +02:00
"metadata" ,
"count" ,
}
)
func Test_SessionsPrepare ( t * testing . T ) {
type want struct {
sqlExpectations sqlExpectation
err checkErr
}
tests := [ ] struct {
name string
prepare interface { }
want want
object interface { }
} {
{
name : "prepareSessionsQuery no result" ,
prepare : prepareSessionsQuery ,
want : want {
sqlExpectations : mockQueries (
expectedSessionsQuery ,
nil ,
nil ,
) ,
} ,
object : & Sessions { Sessions : [ ] * Session { } } ,
} ,
{
name : "prepareSessionQuery" ,
prepare : prepareSessionsQuery ,
want : want {
sqlExpectations : mockQueries (
expectedSessionsQuery ,
sessionsCols ,
[ ] [ ] driver . Value {
{
"session-id" ,
testNow ,
testNow ,
uint64 ( 20211109 ) ,
domain . SessionStateActive ,
"ro" ,
"creator" ,
2023-06-27 14:36:07 +02:00
"domain" ,
2023-05-05 17:34:53 +02:00
"user-id" ,
testNow ,
"login-name" ,
"display-name" ,
testNow ,
2023-06-07 17:28:42 +02:00
testNow ,
2023-06-21 16:06:18 +02:00
testNow ,
2023-05-05 17:34:53 +02:00
[ ] byte ( ` { "key": "dmFsdWU="} ` ) ,
} ,
} ,
) ,
} ,
object : & Sessions {
SearchResponse : SearchResponse {
Count : 1 ,
} ,
Sessions : [ ] * Session {
{
ID : "session-id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
Sequence : 20211109 ,
State : domain . SessionStateActive ,
ResourceOwner : "ro" ,
Creator : "creator" ,
2023-06-27 14:36:07 +02:00
Domain : "domain" ,
2023-05-05 17:34:53 +02:00
UserFactor : SessionUserFactor {
UserID : "user-id" ,
UserCheckedAt : testNow ,
LoginName : "login-name" ,
DisplayName : "display-name" ,
} ,
PasswordFactor : SessionPasswordFactor {
PasswordCheckedAt : testNow ,
} ,
2023-06-21 16:06:18 +02:00
IntentFactor : SessionIntentFactor {
IntentCheckedAt : testNow ,
} ,
2023-06-07 17:28:42 +02:00
PasskeyFactor : SessionPasskeyFactor {
PasskeyCheckedAt : testNow ,
} ,
2023-05-05 17:34:53 +02:00
Metadata : map [ string ] [ ] byte {
"key" : [ ] byte ( "value" ) ,
} ,
} ,
} ,
} ,
} ,
{
name : "prepareSessionsQuery multiple result" ,
prepare : prepareSessionsQuery ,
want : want {
sqlExpectations : mockQueries (
expectedSessionsQuery ,
sessionsCols ,
[ ] [ ] driver . Value {
{
"session-id" ,
testNow ,
testNow ,
uint64 ( 20211109 ) ,
domain . SessionStateActive ,
"ro" ,
"creator" ,
2023-06-27 14:36:07 +02:00
"domain" ,
2023-05-05 17:34:53 +02:00
"user-id" ,
testNow ,
"login-name" ,
"display-name" ,
testNow ,
2023-06-07 17:28:42 +02:00
testNow ,
2023-06-21 16:06:18 +02:00
testNow ,
2023-05-05 17:34:53 +02:00
[ ] byte ( ` { "key": "dmFsdWU="} ` ) ,
} ,
{
"session-id2" ,
testNow ,
testNow ,
uint64 ( 20211109 ) ,
domain . SessionStateActive ,
"ro" ,
"creator2" ,
2023-06-27 14:36:07 +02:00
"domain" ,
2023-05-05 17:34:53 +02:00
"user-id2" ,
testNow ,
"login-name2" ,
"display-name2" ,
testNow ,
2023-06-07 17:28:42 +02:00
testNow ,
2023-06-21 16:06:18 +02:00
testNow ,
2023-05-05 17:34:53 +02:00
[ ] byte ( ` { "key": "dmFsdWU="} ` ) ,
} ,
} ,
) ,
} ,
object : & Sessions {
SearchResponse : SearchResponse {
Count : 2 ,
} ,
Sessions : [ ] * Session {
{
ID : "session-id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
Sequence : 20211109 ,
State : domain . SessionStateActive ,
ResourceOwner : "ro" ,
Creator : "creator" ,
2023-06-27 14:36:07 +02:00
Domain : "domain" ,
2023-05-05 17:34:53 +02:00
UserFactor : SessionUserFactor {
UserID : "user-id" ,
UserCheckedAt : testNow ,
LoginName : "login-name" ,
DisplayName : "display-name" ,
} ,
PasswordFactor : SessionPasswordFactor {
PasswordCheckedAt : testNow ,
} ,
2023-06-21 16:06:18 +02:00
IntentFactor : SessionIntentFactor {
IntentCheckedAt : testNow ,
} ,
2023-06-07 17:28:42 +02:00
PasskeyFactor : SessionPasskeyFactor {
PasskeyCheckedAt : testNow ,
} ,
2023-05-05 17:34:53 +02:00
Metadata : map [ string ] [ ] byte {
"key" : [ ] byte ( "value" ) ,
} ,
} ,
{
ID : "session-id2" ,
CreationDate : testNow ,
ChangeDate : testNow ,
Sequence : 20211109 ,
State : domain . SessionStateActive ,
ResourceOwner : "ro" ,
Creator : "creator2" ,
2023-06-27 14:36:07 +02:00
Domain : "domain" ,
2023-05-05 17:34:53 +02:00
UserFactor : SessionUserFactor {
UserID : "user-id2" ,
UserCheckedAt : testNow ,
LoginName : "login-name2" ,
DisplayName : "display-name2" ,
} ,
PasswordFactor : SessionPasswordFactor {
PasswordCheckedAt : testNow ,
} ,
2023-06-21 16:06:18 +02:00
IntentFactor : SessionIntentFactor {
IntentCheckedAt : testNow ,
} ,
2023-06-07 17:28:42 +02:00
PasskeyFactor : SessionPasskeyFactor {
PasskeyCheckedAt : testNow ,
} ,
2023-05-05 17:34:53 +02:00
Metadata : map [ string ] [ ] byte {
"key" : [ ] byte ( "value" ) ,
} ,
} ,
} ,
} ,
} ,
{
name : "prepareSessionsQuery sql err" ,
prepare : prepareSessionsQuery ,
want : want {
sqlExpectations : mockQueryErr (
expectedSessionsQuery ,
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 , defaultPrepareArgs ... )
} )
}
}
func Test_SessionPrepare ( t * testing . T ) {
type want struct {
sqlExpectations sqlExpectation
err checkErr
}
tests := [ ] struct {
name string
prepare interface { }
want want
object interface { }
} {
{
name : "prepareSessionQuery no result" ,
prepare : prepareSessionQueryTesting ( t , "" ) ,
want : want {
sqlExpectations : mockQueries (
expectedSessionQuery ,
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 : ( * Session ) ( nil ) ,
} ,
{
name : "prepareSessionQuery found" ,
prepare : prepareSessionQueryTesting ( t , "tokenID" ) ,
want : want {
sqlExpectations : mockQuery (
expectedSessionQuery ,
sessionCols ,
[ ] driver . Value {
"session-id" ,
testNow ,
testNow ,
uint64 ( 20211109 ) ,
domain . SessionStateActive ,
"ro" ,
"creator" ,
2023-06-27 14:36:07 +02:00
"domain" ,
2023-05-05 17:34:53 +02:00
"user-id" ,
testNow ,
"login-name" ,
"display-name" ,
testNow ,
2023-06-07 17:28:42 +02:00
testNow ,
2023-06-21 16:06:18 +02:00
testNow ,
2023-05-05 17:34:53 +02:00
[ ] byte ( ` { "key": "dmFsdWU="} ` ) ,
"tokenID" ,
} ,
) ,
} ,
object : & Session {
ID : "session-id" ,
CreationDate : testNow ,
ChangeDate : testNow ,
Sequence : 20211109 ,
State : domain . SessionStateActive ,
ResourceOwner : "ro" ,
Creator : "creator" ,
2023-06-27 14:36:07 +02:00
Domain : "domain" ,
2023-05-05 17:34:53 +02:00
UserFactor : SessionUserFactor {
UserID : "user-id" ,
UserCheckedAt : testNow ,
LoginName : "login-name" ,
DisplayName : "display-name" ,
} ,
PasswordFactor : SessionPasswordFactor {
PasswordCheckedAt : testNow ,
} ,
2023-06-21 16:06:18 +02:00
IntentFactor : SessionIntentFactor {
IntentCheckedAt : testNow ,
} ,
2023-06-07 17:28:42 +02:00
PasskeyFactor : SessionPasskeyFactor {
PasskeyCheckedAt : testNow ,
} ,
2023-05-05 17:34:53 +02:00
Metadata : map [ string ] [ ] byte {
"key" : [ ] byte ( "value" ) ,
} ,
} ,
} ,
{
name : "prepareSessionQuery sql err" ,
prepare : prepareSessionQueryTesting ( t , "" ) ,
want : want {
sqlExpectations : mockQueryErr (
expectedSessionQuery ,
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 , defaultPrepareArgs ... )
} )
}
}
func prepareSessionQueryTesting ( t * testing . T , token string ) func ( context . Context , prepareDatabase ) ( sq . SelectBuilder , func ( * sql . Row ) ( * Session , error ) ) {
return func ( ctx context . Context , db prepareDatabase ) ( sq . SelectBuilder , func ( * sql . Row ) ( * Session , error ) ) {
builder , scan := prepareSessionQuery ( ctx , db )
return builder , func ( row * sql . Row ) ( * Session , error ) {
session , tokenID , err := scan ( row )
require . Equal ( t , tokenID , token )
return session , err
}
}
}