mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-24 01:29:19 +03:00 
			
		
		
		
	Fix decoding of MULTI_INSERTs when rows other than the last are toasted.
When decoding the results of a HEAP2_MULTI_INSERT (currently only generated by COPY FROM) toast columns for all but the last tuple weren't replaced by their actual contents before being handed to the output plugin. The reassembled toast datums where disregarded after every REORDER_BUFFER_CHANGE_(INSERT|UPDATE|DELETE) which is correct for plain inserts, updates, deletes, but not multi inserts - there we generate several REORDER_BUFFER_CHANGE_INSERTs for a single xl_heap_multi_insert record. To solve the problem add a clear_toast_afterwards boolean to ReorderBufferChange's union member that's used by modifications. All row changes but multi_inserts always set that to true, but multi_insert sets it only for the last change generated. Add a regression test covering decoding of multi_inserts - there was none at all before. Backpatch to 9.4 where logical decoding was introduced. Bug found by Petr Jelinek.
This commit is contained in:
		| @@ -40,6 +40,14 @@ UPDATE toasted_key SET toasted_col2 = toasted_col1; | |||||||
| -- test update of a toasted key, changing it | -- test update of a toasted key, changing it | ||||||
| UPDATE toasted_key SET toasted_key = toasted_key || '1'; | UPDATE toasted_key SET toasted_key = toasted_key || '1'; | ||||||
| DELETE FROM toasted_key; | DELETE FROM toasted_key; | ||||||
|  | -- Test that HEAP2_MULTI_INSERT insertions with and without toasted | ||||||
|  | -- columns are handled correctly | ||||||
|  | CREATE TABLE toasted_copy ( | ||||||
|  |     id int primary key, -- no default, copy didn't use to handle that with multi inserts | ||||||
|  |     data text | ||||||
|  | ); | ||||||
|  | ALTER TABLE toasted_copy ALTER COLUMN data SET STORAGE EXTERNAL; | ||||||
|  | \copy toasted_copy FROM STDIN | ||||||
| SELECT substr(data, 1, 200) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0'); | SELECT substr(data, 1, 200) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0'); | ||||||
|                                                                                                   substr                                                                                                   |                                                                                                   substr                                                                                                   | ||||||
| ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ||||||
| @@ -80,7 +88,17 @@ SELECT substr(data, 1, 200) FROM pg_logical_slot_get_changes('regression_slot', | |||||||
|  BEGIN |  BEGIN | ||||||
|  table public.toasted_key: DELETE: toasted_key[text]:'123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567 |  table public.toasted_key: DELETE: toasted_key[text]:'123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567 | ||||||
|  COMMIT |  COMMIT | ||||||
| (37 rows) |  BEGIN | ||||||
|  |  COMMIT | ||||||
|  |  BEGIN | ||||||
|  |  COMMIT | ||||||
|  |  BEGIN | ||||||
|  |  table public.toasted_copy: INSERT: id[integer]:1 data[text]:'untoasted1' | ||||||
|  |  table public.toasted_copy: INSERT: id[integer]:2 data[text]:'toasted1-1234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890 | ||||||
|  |  table public.toasted_copy: INSERT: id[integer]:3 data[text]:'untoasted2' | ||||||
|  |  table public.toasted_copy: INSERT: id[integer]:4 data[text]:'toasted2-1234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890 | ||||||
|  |  COMMIT | ||||||
|  | (47 rows) | ||||||
|  |  | ||||||
| SELECT pg_drop_replication_slot('regression_slot'); | SELECT pg_drop_replication_slot('regression_slot'); | ||||||
|  pg_drop_replication_slot  |  pg_drop_replication_slot  | ||||||
|   | |||||||
| @@ -47,5 +47,18 @@ UPDATE toasted_key SET toasted_key = toasted_key || '1'; | |||||||
|  |  | ||||||
| DELETE FROM toasted_key; | DELETE FROM toasted_key; | ||||||
|  |  | ||||||
|  | -- Test that HEAP2_MULTI_INSERT insertions with and without toasted | ||||||
|  | -- columns are handled correctly | ||||||
|  | CREATE TABLE toasted_copy ( | ||||||
|  |     id int primary key, -- no default, copy didn't use to handle that with multi inserts | ||||||
|  |     data text | ||||||
|  | ); | ||||||
|  | ALTER TABLE toasted_copy ALTER COLUMN data SET STORAGE EXTERNAL; | ||||||
|  | \copy toasted_copy FROM STDIN | ||||||
|  | 1	untoasted1 | ||||||
|  | 2	toasted1-12345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890 | ||||||
|  | 3	untoasted2 | ||||||
|  | 4	toasted2-12345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890 | ||||||
|  | \. | ||||||
| SELECT substr(data, 1, 200) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0'); | SELECT substr(data, 1, 200) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0'); | ||||||
| SELECT pg_drop_replication_slot('regression_slot'); | SELECT pg_drop_replication_slot('regression_slot'); | ||||||
|   | |||||||
| @@ -608,6 +608,8 @@ DecodeInsert(LogicalDecodingContext *ctx, XLogRecordBuffer *buf) | |||||||
| 						change->data.tp.newtuple); | 						change->data.tp.newtuple); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	change->data.tp.clear_toast_afterwards = true; | ||||||
|  |  | ||||||
| 	ReorderBufferQueueChange(ctx->reorder, r->xl_xid, buf->origptr, change); | 	ReorderBufferQueueChange(ctx->reorder, r->xl_xid, buf->origptr, change); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -673,6 +675,8 @@ DecodeUpdate(LogicalDecodingContext *ctx, XLogRecordBuffer *buf) | |||||||
| #endif | #endif | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	change->data.tp.clear_toast_afterwards = true; | ||||||
|  |  | ||||||
| 	ReorderBufferQueueChange(ctx->reorder, r->xl_xid, buf->origptr, change); | 	ReorderBufferQueueChange(ctx->reorder, r->xl_xid, buf->origptr, change); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -710,6 +714,9 @@ DecodeDelete(LogicalDecodingContext *ctx, XLogRecordBuffer *buf) | |||||||
| 						r->xl_len - SizeOfHeapDelete, | 						r->xl_len - SizeOfHeapDelete, | ||||||
| 						change->data.tp.oldtuple); | 						change->data.tp.oldtuple); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	change->data.tp.clear_toast_afterwards = true; | ||||||
|  |  | ||||||
| 	ReorderBufferQueueChange(ctx->reorder, r->xl_xid, buf->origptr, change); | 	ReorderBufferQueueChange(ctx->reorder, r->xl_xid, buf->origptr, change); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -795,6 +802,9 @@ DecodeMultiInsert(LogicalDecodingContext *ctx, XLogRecordBuffer *buf) | |||||||
| 			tuple->header.t_hoff = xlhdr->t_hoff; | 			tuple->header.t_hoff = xlhdr->t_hoff; | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		/* reset toast reassembly only after the last chunk */ | ||||||
|  | 		change->data.tp.clear_toast_afterwards = (i + 1) == xlrec->ntuples; | ||||||
|  |  | ||||||
| 		ReorderBufferQueueChange(ctx->reorder, r->xl_xid, | 		ReorderBufferQueueChange(ctx->reorder, r->xl_xid, | ||||||
| 								 buf->origptr, change); | 								 buf->origptr, change); | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -1383,7 +1383,14 @@ ReorderBufferCommit(ReorderBuffer *rb, TransactionId xid, | |||||||
| 						{ | 						{ | ||||||
| 							ReorderBufferToastReplace(rb, txn, relation, change); | 							ReorderBufferToastReplace(rb, txn, relation, change); | ||||||
| 							rb->apply_change(rb, txn, relation, change); | 							rb->apply_change(rb, txn, relation, change); | ||||||
| 							ReorderBufferToastReset(rb, txn); |  | ||||||
|  | 							/* | ||||||
|  | 							 * Only clear reassembled toast chunks if we're | ||||||
|  | 							 * sure they're not required anymore. The creator | ||||||
|  | 							 * of the tuple tells us. | ||||||
|  | 							 */ | ||||||
|  | 							if (change->data.tp.clear_toast_afterwards) | ||||||
|  | 								ReorderBufferToastReset(rb, txn); | ||||||
| 						} | 						} | ||||||
| 						/* we're not interested in toast deletions */ | 						/* we're not interested in toast deletions */ | ||||||
| 						else if (change->action == REORDER_BUFFER_CHANGE_INSERT) | 						else if (change->action == REORDER_BUFFER_CHANGE_INSERT) | ||||||
|   | |||||||
| @@ -75,6 +75,10 @@ typedef struct ReorderBufferChange | |||||||
| 		{ | 		{ | ||||||
| 			/* relation that has been changed */ | 			/* relation that has been changed */ | ||||||
| 			RelFileNode relnode; | 			RelFileNode relnode; | ||||||
|  |  | ||||||
|  | 			/* no previously reassembled toast chunks are necessary anymore */ | ||||||
|  | 			bool clear_toast_afterwards; | ||||||
|  |  | ||||||
| 			/* valid for DELETE || UPDATE */ | 			/* valid for DELETE || UPDATE */ | ||||||
| 			ReorderBufferTupleBuf *oldtuple; | 			ReorderBufferTupleBuf *oldtuple; | ||||||
| 			/* valid for INSERT || UPDATE */ | 			/* valid for INSERT || UPDATE */ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user