mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Extend the output plugin API to allow decoding of prepared xacts.
This adds six methods to the output plugin API, adding support for streaming changes of two-phase transactions at prepare time. * begin_prepare * filter_prepare * prepare * commit_prepared * rollback_prepared * stream_prepare Most of this is a simple extension of the existing methods, with the semantic difference that the transaction is not yet committed and maybe aborted later. Until now two-phase transactions were translated into regular transactions on the subscriber, and the GID was not forwarded to it. None of the two-phase commands were communicated to the subscriber. This patch provides the infrastructure for logical decoding plugins to be informed of two-phase commands Like PREPARE TRANSACTION, COMMIT PREPARED and ROLLBACK PREPARED commands with the corresponding GID. This also extends the 'test_decoding' plugin, implementing these new methods. This commit simply adds these new APIs and the upcoming patch to "allow the decoding at prepare time in ReorderBuffer" will use these APIs. Author: Ajin Cherian and Amit Kapila based on previous work by Nikhil Sontakke and Stas Kelvich Reviewed-by: Amit Kapila, Peter Smith, Sawada Masahiko, and Dilip Kumar Discussion: https://postgr.es/m/02DA5F5E-CECE-4D9C-8B4B-418077E2C010@postgrespro.ru https://postgr.es/m/CAMGcDxeqEpWj3fTXwqhSwBdXd2RS9jzwWscO-XbeCfso6ts3+Q@mail.gmail.com
This commit is contained in:
		| @@ -389,9 +389,15 @@ typedef struct OutputPluginCallbacks | ||||
|     LogicalDecodeMessageCB message_cb; | ||||
|     LogicalDecodeFilterByOriginCB filter_by_origin_cb; | ||||
|     LogicalDecodeShutdownCB shutdown_cb; | ||||
|     LogicalDecodeFilterPrepareCB filter_prepare_cb; | ||||
|     LogicalDecodeBeginPrepareCB begin_prepare_cb; | ||||
|     LogicalDecodePrepareCB prepare_cb; | ||||
|     LogicalDecodeCommitPreparedCB commit_prepared_cb; | ||||
|     LogicalDecodeRollbackPreparedCB rollback_prepared_cb; | ||||
|     LogicalDecodeStreamStartCB stream_start_cb; | ||||
|     LogicalDecodeStreamStopCB stream_stop_cb; | ||||
|     LogicalDecodeStreamAbortCB stream_abort_cb; | ||||
|     LogicalDecodeStreamPrepareCB stream_prepare_cb; | ||||
|     LogicalDecodeStreamCommitCB stream_commit_cb; | ||||
|     LogicalDecodeStreamChangeCB stream_change_cb; | ||||
|     LogicalDecodeStreamMessageCB stream_message_cb; | ||||
| @@ -413,10 +419,20 @@ typedef void (*LogicalOutputPluginInit) (struct OutputPluginCallbacks *cb); | ||||
|      An output plugin may also define functions to support streaming of large, | ||||
|      in-progress transactions. The <function>stream_start_cb</function>, | ||||
|      <function>stream_stop_cb</function>, <function>stream_abort_cb</function>, | ||||
|      <function>stream_commit_cb</function> and <function>stream_change_cb</function> | ||||
|      <function>stream_commit_cb</function>, <function>stream_change_cb</function>, | ||||
|      and <function>stream_prepare_cb</function> | ||||
|      are required, while <function>stream_message_cb</function> and | ||||
|      <function>stream_truncate_cb</function> are optional. | ||||
|     </para> | ||||
|  | ||||
|     <para> | ||||
|     An output plugin may also define functions to support two-phase commits, | ||||
|     which allows actions to be decoded on the <command>PREPARE TRANSACTION</command>. | ||||
|     The <function>begin_prepare_cb</function>, <function>prepare_cb</function>,  | ||||
|     <function>stream_prepare_cb</function>, | ||||
|     <function>commit_prepared_cb</function> and <function>rollback_prepared_cb</function> | ||||
|     callbacks are required, while <function>filter_prepare_cb</function> is optional. | ||||
|     </para> | ||||
|    </sect2> | ||||
|  | ||||
|    <sect2 id="logicaldecoding-capabilities"> | ||||
| @@ -477,7 +493,15 @@ CREATE TABLE another_catalog_table(data text) WITH (user_catalog_table = true); | ||||
|      never get | ||||
|      decoded. Successful savepoints are | ||||
|      folded into the transaction containing them in the order they were | ||||
|      executed within that transaction. | ||||
|      executed within that transaction. A transaction that is prepared for | ||||
|      a two-phase commit using <command>PREPARE TRANSACTION</command> will | ||||
|      also be decoded if the output plugin callbacks needed for decoding | ||||
|      them are provided. It is possible that the current transaction which | ||||
|      is being decoded is aborted concurrently via a <command>ROLLBACK PREPARED</command> | ||||
|      command. In that case, the logical decoding of this transaction will | ||||
|      be aborted too. We will skip all the changes of such a transaction once | ||||
|      the abort is detected and abort the transaction when we read WAL for | ||||
|      <command>ROLLBACK PREPARED</command>. | ||||
|     </para> | ||||
|  | ||||
|     <note> | ||||
| @@ -587,7 +611,13 @@ typedef void (*LogicalDecodeCommitCB) (struct LogicalDecodingContext *ctx, | ||||
|       an <command>INSERT</command>, <command>UPDATE</command>, | ||||
|       or <command>DELETE</command>. Even if the original command modified | ||||
|       several rows at once the callback will be called individually for each | ||||
|       row. | ||||
|       row. The <function>change_cb</function> callback may access system or | ||||
|       user catalog tables to aid in the process of outputting the row | ||||
|       modification details. In case of decoding a prepared (but yet | ||||
|       uncommitted) transaction or decoding of an uncommitted transaction, this | ||||
|       change callback might also error out due to simultaneous rollback of | ||||
|       this very same transaction. In that case, the logical decoding of this | ||||
|       aborted transaction is stopped gracefully. | ||||
| <programlisting> | ||||
| typedef void (*LogicalDecodeChangeCB) (struct LogicalDecodingContext *ctx, | ||||
|                                        ReorderBufferTXN *txn, | ||||
| @@ -685,7 +715,13 @@ typedef void (*LogicalDecodeMessageCB) (struct LogicalDecodingContext *ctx, | ||||
|       non-transactional and the XID was not assigned yet in the transaction | ||||
|       which logged the message. The <parameter>lsn</parameter> has WAL | ||||
|       location of the message. The <parameter>transactional</parameter> says | ||||
|       if the message was sent as transactional or not. | ||||
|       if the message was sent as transactional or not. Similar to the change | ||||
|       callback, in case of decoding a prepared (but yet uncommitted) | ||||
|       transaction or decoding of an uncommitted transaction, this message | ||||
|       callback might also error out due to simultaneous rollback of | ||||
|       this very same transaction. In that case, the logical decoding of this | ||||
|       aborted transaction is stopped gracefully. | ||||
|  | ||||
|       The <parameter>prefix</parameter> is arbitrary null-terminated prefix | ||||
|       which can be used for identifying interesting messages for the current | ||||
|       plugin. And finally the <parameter>message</parameter> parameter holds | ||||
| @@ -698,6 +734,111 @@ typedef void (*LogicalDecodeMessageCB) (struct LogicalDecodingContext *ctx, | ||||
|      </para> | ||||
|     </sect3> | ||||
|  | ||||
|     <sect3 id="logicaldecoding-output-plugin-filter-prepare"> | ||||
|      <title>Prepare Filter Callback</title> | ||||
|  | ||||
|      <para> | ||||
|        The optional <function>filter_prepare_cb</function> callback | ||||
|        is called to determine whether data that is part of the current | ||||
|        two-phase commit transaction should be considered for decode | ||||
|        at this prepare stage or as a regular one-phase transaction at | ||||
|        <command>COMMIT PREPARED</command> time later. To signal that | ||||
|        decoding should be skipped, return <literal>true</literal>; | ||||
|        <literal>false</literal> otherwise. When the callback is not | ||||
|        defined, <literal>false</literal> is assumed (i.e. nothing is | ||||
|        filtered). | ||||
| <programlisting> | ||||
| typedef bool (*LogicalDecodeFilterPrepareCB) (struct LogicalDecodingContext *ctx, | ||||
|                                               const char *gid); | ||||
| </programlisting> | ||||
|       The <parameter>ctx</parameter> parameter has the same contents as for the | ||||
|       other callbacks. The <parameter>gid</parameter> is the identifier that later | ||||
|       identifies this transaction for <command>COMMIT PREPARED</command> or | ||||
|       <command>ROLLBACK PREPARED</command>. | ||||
|      </para> | ||||
|      <para> | ||||
|       The callback has to provide the same static answer for a given | ||||
|       <parameter>gid</parameter> every time it is called. | ||||
|      </para> | ||||
|      </sect3> | ||||
|  | ||||
|     <sect3 id="logicaldecoding-output-plugin-begin-prepare"> | ||||
|      <title>Transaction Begin Prepare Callback</title> | ||||
|  | ||||
|      <para> | ||||
|       The required <function>begin_prepare_cb</function> callback is called | ||||
|       whenever the start of a prepared transaction has been decoded. The | ||||
|       <parameter>gid</parameter> field, which is part of the | ||||
|       <parameter>txn</parameter> parameter can be used in this callback to | ||||
|       check if the plugin has already received this prepare in which case it | ||||
|       can skip the remaining changes of the transaction. This can only happen | ||||
|       if the user restarts the decoding after receiving the prepare for a | ||||
|       transaction but before receiving the commit prepared say because of some | ||||
|       error. | ||||
|       <programlisting> | ||||
|        typedef void (*LogicalDecodeBeginPrepareCB) (struct LogicalDecodingContext *ctx, | ||||
|                                                     ReorderBufferTXN *txn); | ||||
|       </programlisting> | ||||
|      </para> | ||||
|     </sect3> | ||||
|  | ||||
|     <sect3 id="logicaldecoding-output-plugin-prepare"> | ||||
|      <title>Transaction Prepare Callback</title> | ||||
|  | ||||
|      <para> | ||||
|       The required <function>prepare_cb</function> callback is called whenever | ||||
|       a transaction which is prepared for two-phase commit has been | ||||
|       decoded. The <function>change_cb</function> callback for all modified | ||||
|       rows will have been called before this, if there have been any modified | ||||
|       rows. The <parameter>gid</parameter> field, which is part of the | ||||
|       <parameter>txn</parameter> parameter can be used in this callback. | ||||
|       <programlisting> | ||||
|        typedef void (*LogicalDecodePrepareCB) (struct LogicalDecodingContext *ctx, | ||||
|                                                ReorderBufferTXN *txn, | ||||
|                                                XLogRecPtr prepare_lsn); | ||||
|       </programlisting> | ||||
|      </para> | ||||
|     </sect3> | ||||
|  | ||||
|     <sect3 id="logicaldecoding-output-plugin-commit-prepared"> | ||||
|      <title>Transaction Commit Prepared Callback</title> | ||||
|  | ||||
|      <para> | ||||
|       The required <function>commit_prepared_cb</function> callback is called | ||||
|       whenever a transaction commit prepared has been decoded. The | ||||
|       <parameter>gid</parameter> field, which is part of the | ||||
|       <parameter>txn</parameter> parameter can be used in this callback. | ||||
|       <programlisting> | ||||
|        typedef void (*LogicalDecodeCommitPreparedCB) (struct LogicalDecodingContext *ctx, | ||||
|                                                       ReorderBufferTXN *txn, | ||||
|                                                       XLogRecPtr commit_lsn); | ||||
|       </programlisting> | ||||
|      </para> | ||||
|     </sect3> | ||||
|  | ||||
|     <sect3 id="logicaldecoding-output-plugin-rollback-prepared"> | ||||
|      <title>Transaction Rollback Prepared Callback</title> | ||||
|  | ||||
|      <para> | ||||
|       The required <function>rollback_prepared_cb</function> callback is called | ||||
|       whenever a transaction rollback prepared has been decoded. The | ||||
|       <parameter>gid</parameter> field, which is part of the | ||||
|       <parameter>txn</parameter> parameter can be used in this callback. The | ||||
|       parameters <parameter>prepare_end_lsn</parameter> and | ||||
|       <parameter>prepare_time</parameter> can be used to check if the plugin | ||||
|       has received this prepare transaction in which case it can apply the | ||||
|       rollback, otherwise, it can skip the rollback operation. The | ||||
|       <parameter>gid</parameter> alone is not sufficient because the downstream | ||||
|       node can have prepared transaction with same identifier. | ||||
|       <programlisting> | ||||
|        typedef void (*LogicalDecodeRollbackPreparedCB) (struct LogicalDecodingContext *ctx, | ||||
|                                                         ReorderBufferTXN *txn, | ||||
|                                                         XLogRecPtr preapre_end_lsn, | ||||
|                                                         TimestampTz prepare_time); | ||||
|       </programlisting> | ||||
|      </para> | ||||
|     </sect3> | ||||
|  | ||||
|     <sect3 id="logicaldecoding-output-plugin-stream-start"> | ||||
|      <title>Stream Start Callback</title> | ||||
|      <para> | ||||
| @@ -735,6 +876,19 @@ typedef void (*LogicalDecodeStreamAbortCB) (struct LogicalDecodingContext *ctx, | ||||
|      </para> | ||||
|     </sect3> | ||||
|  | ||||
|     <sect3 id="logicaldecoding-output-plugin-stream-prepare"> | ||||
|      <title>Stream Prepare Callback</title> | ||||
|      <para> | ||||
|       The <function>stream_prepare_cb</function> callback is called to prepare | ||||
|       a previously streamed transaction as part of a two-phase commit. | ||||
| <programlisting> | ||||
| typedef void (*LogicalDecodeStreamPrepareCB) (struct LogicalDecodingContext *ctx, | ||||
|                                               ReorderBufferTXN *txn, | ||||
|                                               XLogRecPtr prepare_lsn); | ||||
| </programlisting> | ||||
|      </para> | ||||
|     </sect3> | ||||
|  | ||||
|     <sect3 id="logicaldecoding-output-plugin-stream-commit"> | ||||
|      <title>Stream Commit Callback</title> | ||||
|      <para> | ||||
| @@ -913,9 +1067,13 @@ OutputPluginWrite(ctx, true); | ||||
|     When streaming an in-progress transaction, the changes (and messages) are | ||||
|     streamed in blocks demarcated by <function>stream_start_cb</function> | ||||
|     and <function>stream_stop_cb</function> callbacks. Once all the decoded | ||||
|     changes are transmitted, the transaction is committed using the | ||||
|     <function>stream_commit_cb</function> callback (or possibly aborted using | ||||
|     the <function>stream_abort_cb</function> callback). | ||||
|     changes are transmitted, the transaction can be committed using the | ||||
|     the <function>stream_commit_cb</function> callback | ||||
|     (or possibly aborted using the <function>stream_abort_cb</function> callback). | ||||
|     If two-phase commits are supported, the transaction can be prepared using the | ||||
|     <function>stream_prepare_cb</function> callback, commit prepared using the | ||||
|     <function>commit_prepared_cb</function> callback or aborted using the | ||||
|     <function>rollback_prepared_cb</function>. | ||||
|    </para> | ||||
|  | ||||
|    <para> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user