2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								package  eventstore  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  (  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"context" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"database/sql" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									_  "embed" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"fmt" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"strings" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"github.com/zitadel/logging" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									"github.com/zitadel/zitadel/internal/api/authz" 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-22 17:25:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									"github.com/zitadel/zitadel/internal/database" 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									"github.com/zitadel/zitadel/internal/eventstore" 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-08 16:30:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									"github.com/zitadel/zitadel/internal/zerrors" 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								)  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								type  latestSequence  struct  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									aggregate  * eventstore . Aggregate 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									sequence   uint64 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								//go:embed sequences_query.sql  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								var  latestSequencesStmt  string  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-11-22 17:25:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								func  latestSequences ( ctx  context . Context ,  tx  database . Tx ,  commands  [ ] eventstore . Command )  ( [ ] * latestSequence ,  error )  {  
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									sequences  :=  commandsToSequences ( ctx ,  commands ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									conditions ,  args  :=  sequencesToSql ( sequences ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-31 15:30:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									rows ,  err  :=  tx . QueryContext ( ctx ,  fmt . Sprintf ( latestSequencesStmt ,  strings . Join ( conditions ,  " UNION ALL " ) ) ,  args ... ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  err  !=  nil  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-08 16:30:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										return  nil ,  zerrors . ThrowInternal ( err ,  "V3-5jU5z" ,  "Errors.Internal" ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									defer  rows . Close ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  rows . Next ( )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  err  :=  scanToSequence ( rows ,  sequences ) ;  err  !=  nil  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-08 16:30:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											return  nil ,  zerrors . ThrowInternal ( err ,  "V3-Ydiwv" ,  "Errors.Internal" ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  rows . Err ( )  !=  nil  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-08 16:30:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										return  nil ,  zerrors . ThrowInternal ( rows . Err ( ) ,  "V3-XApDk" ,  "Errors.Internal" ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  sequences ,  nil 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								func  searchSequenceByCommand ( sequences  [ ] * latestSequence ,  command  eventstore . Command )  * latestSequence  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  _ ,  sequence  :=  range  sequences  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  sequence . aggregate . Type  ==  command . Aggregate ( ) . Type  && 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											sequence . aggregate . ID  ==  command . Aggregate ( ) . ID  && 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											sequence . aggregate . InstanceID  ==  command . Aggregate ( ) . InstanceID  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  sequence 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  nil 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								func  searchSequence ( sequences  [ ] * latestSequence ,  aggregateType  eventstore . AggregateType ,  aggregateID ,  instanceID  string )  * latestSequence  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  _ ,  sequence  :=  range  sequences  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  sequence . aggregate . Type  ==  aggregateType  && 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											sequence . aggregate . ID  ==  aggregateID  && 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											sequence . aggregate . InstanceID  ==  instanceID  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  sequence 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  nil 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								func  commandsToSequences ( ctx  context . Context ,  commands  [ ] eventstore . Command )  [ ] * latestSequence  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									sequences  :=  make ( [ ] * latestSequence ,  0 ,  len ( commands ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  _ ,  command  :=  range  commands  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  searchSequenceByCommand ( sequences ,  command )  !=  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											continue 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  command . Aggregate ( ) . InstanceID  ==  ""  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											command . Aggregate ( ) . InstanceID  =  authz . GetInstance ( ctx ) . InstanceID ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										sequences  =  append ( sequences ,  & latestSequence { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											aggregate :  command . Aggregate ( ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  sequences 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const  argsPerCondition  =  3  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								func  sequencesToSql ( sequences  [ ] * latestSequence )  ( conditions  [ ] string ,  args  [ ] any )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									args  =  make ( [ ] interface { } ,  0 ,  len ( sequences ) * argsPerCondition ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									conditions  =  make ( [ ] string ,  len ( sequences ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  i ,  sequence  :=  range  sequences  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-31 15:30:25 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										conditions [ i ]  =  fmt . Sprintf ( ` (SELECT instance_id, aggregate_type, aggregate_id, "sequence" FROM eventstore.events2 WHERE instance_id = $%d AND aggregate_type = $%d AND aggregate_id = $%d ORDER BY "sequence" DESC LIMIT 1) ` , 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											i * argsPerCondition + 1 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											i * argsPerCondition + 2 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											i * argsPerCondition + 3 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										args  =  append ( args ,  sequence . aggregate . InstanceID ,  sequence . aggregate . Type ,  sequence . aggregate . ID ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  conditions ,  args 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								func  scanToSequence ( rows  * sql . Rows ,  sequences  [ ] * latestSequence )  error  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									var  aggregateType  eventstore . AggregateType 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									var  aggregateID ,  instanceID  string 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									var  currentSequence  uint64 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									var  resourceOwner  string 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  err  :=  rows . Scan ( & instanceID ,  & resourceOwner ,  & aggregateType ,  & aggregateID ,  & currentSequence ) ;  err  !=  nil  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-08 16:30:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										return  zerrors . ThrowInternal ( err ,  "V3-OIWqj" ,  "Errors.Internal" ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									sequence  :=  searchSequence ( sequences ,  aggregateType ,  aggregateID ,  instanceID ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  sequence  ==  nil  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										logging . WithFields ( 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"aggType" ,  aggregateType , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"aggID" ,  aggregateID , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"instance" ,  instanceID , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										) . Panic ( "no sequence found" ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// added return for linting 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  nil 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									sequence . sequence  =  currentSequence 
							 
						 
					
						
							
								
									
										
										
										
											2025-01-15 11:22:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  resourceOwner  !=  ""  &&  sequence . aggregate . ResourceOwner  !=  ""  &&  sequence . aggregate . ResourceOwner  !=  resourceOwner  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										logging . WithFields ( 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"current_sequence" ,  sequence . sequence , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"instance_id" ,  sequence . aggregate . InstanceID , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"agg_type" ,  sequence . aggregate . Type , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"agg_id" ,  sequence . aggregate . ID , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"current_owner" ,  resourceOwner , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											"provided_owner" ,  sequence . aggregate . ResourceOwner , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										) . Info ( "would have set wrong resource owner" ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// set resource owner from previous events 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  resourceOwner  !=  ""  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 12:19:10 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										sequence . aggregate . ResourceOwner  =  resourceOwner 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  nil 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}