mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Detect and Log multiple_unique_conflicts type conflict.
Introduce a new conflict type, multiple_unique_conflicts, to handle cases where an incoming row during logical replication violates multiple UNIQUE constraints. Previously, the apply worker detected and reported only the first encountered key conflict (insert_exists/update_exists), causing repeated failures as each constraint violation needs to be handled one by one making the process slow and error-prone. With this patch, the apply worker checks all unique constraints upfront once the first key conflict is detected and reports multiple_unique_conflicts if multiple violations exist. This allows users to resolve all conflicts at once by deleting all conflicting tuples rather than dealing with them individually or skipping the transaction. In the future, this will also allow us to specify different resolution handlers for such a conflict type. Add the stats for this conflict type in pg_stat_subscription_stats. Author: Nisha Moond <nisha.moond412@gmail.com> Author: Zhijie Hou <houzj.fnst@fujitsu.com> Reviewed-by: Amit Kapila <amit.kapila16@gmail.com> Reviewed-by: Peter Smith <smithpb2250@gmail.com> Reviewed-by: Dilip Kumar <dilipbalaut@gmail.com> Discussion: https://postgr.es/m/CABdArM7FW-_dnthGkg2s0fy1HhUB8C3ELA0gZX1kkbs1ZZoV3Q@mail.gmail.com
This commit is contained in:
		| @@ -41,6 +41,9 @@ typedef enum | ||||
| 	/* The row to be deleted is missing */ | ||||
| 	CT_DELETE_MISSING, | ||||
|  | ||||
| 	/* The row to be inserted/updated violates multiple unique constraint */ | ||||
| 	CT_MULTIPLE_UNIQUE_CONFLICTS, | ||||
|  | ||||
| 	/* | ||||
| 	 * Other conflicts, such as exclusion constraint violations, involve more | ||||
| 	 * complex rules than simple equality checks. These conflicts are left for | ||||
| @@ -48,7 +51,23 @@ typedef enum | ||||
| 	 */ | ||||
| } ConflictType; | ||||
|  | ||||
| #define CONFLICT_NUM_TYPES (CT_DELETE_MISSING + 1) | ||||
| #define CONFLICT_NUM_TYPES (CT_MULTIPLE_UNIQUE_CONFLICTS + 1) | ||||
|  | ||||
| /* | ||||
|  * Information for the existing local tuple that caused the conflict. | ||||
|  */ | ||||
| typedef struct ConflictTupleInfo | ||||
| { | ||||
| 	TupleTableSlot *slot;		/* tuple slot holding the conflicting local | ||||
| 								 * tuple */ | ||||
| 	Oid			indexoid;		/* OID of the index where the conflict | ||||
| 								 * occurred */ | ||||
| 	TransactionId xmin;			/* transaction ID of the modification causing | ||||
| 								 * the conflict */ | ||||
| 	RepOriginId origin;			/* origin identifier of the modification */ | ||||
| 	TimestampTz ts;				/* timestamp of when the modification on the | ||||
| 								 * conflicting local tuple occurred */ | ||||
| } ConflictTupleInfo; | ||||
|  | ||||
| extern bool GetTupleTransactionInfo(TupleTableSlot *localslot, | ||||
| 									TransactionId *xmin, | ||||
| @@ -57,10 +76,7 @@ extern bool GetTupleTransactionInfo(TupleTableSlot *localslot, | ||||
| extern void ReportApplyConflict(EState *estate, ResultRelInfo *relinfo, | ||||
| 								int elevel, ConflictType type, | ||||
| 								TupleTableSlot *searchslot, | ||||
| 								TupleTableSlot *localslot, | ||||
| 								TupleTableSlot *remoteslot, | ||||
| 								Oid indexoid, TransactionId localxmin, | ||||
| 								RepOriginId localorigin, TimestampTz localts); | ||||
| 								List *conflicttuples); | ||||
| extern void InitConflictIndexes(ResultRelInfo *relInfo); | ||||
|  | ||||
| #endif | ||||
|   | ||||
		Reference in New Issue
	
	Block a user