LCOV - code coverage report
Current view: top level - src/backend/replication/logical - proto.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 447 493 90.7 %
Date: 2025-01-18 04:15:08 Functions: 45 46 97.8 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * proto.c
       4             :  *      logical replication protocol functions
       5             :  *
       6             :  * Copyright (c) 2015-2025, PostgreSQL Global Development Group
       7             :  *
       8             :  * IDENTIFICATION
       9             :  *      src/backend/replication/logical/proto.c
      10             :  *
      11             :  *-------------------------------------------------------------------------
      12             :  */
      13             : #include "postgres.h"
      14             : 
      15             : #include "access/sysattr.h"
      16             : #include "catalog/pg_namespace.h"
      17             : #include "catalog/pg_type.h"
      18             : #include "libpq/pqformat.h"
      19             : #include "replication/logicalproto.h"
      20             : #include "utils/lsyscache.h"
      21             : #include "utils/syscache.h"
      22             : 
      23             : /*
      24             :  * Protocol message flags.
      25             :  */
      26             : #define LOGICALREP_IS_REPLICA_IDENTITY 1
      27             : 
      28             : #define MESSAGE_TRANSACTIONAL (1<<0)
      29             : #define TRUNCATE_CASCADE        (1<<0)
      30             : #define TRUNCATE_RESTART_SEQS   (1<<1)
      31             : 
      32             : static void logicalrep_write_attrs(StringInfo out, Relation rel,
      33             :                                    Bitmapset *columns, bool include_gencols);
      34             : static void logicalrep_write_tuple(StringInfo out, Relation rel,
      35             :                                    TupleTableSlot *slot,
      36             :                                    bool binary, Bitmapset *columns,
      37             :                                    bool include_gencols);
      38             : static void logicalrep_read_attrs(StringInfo in, LogicalRepRelation *rel);
      39             : static void logicalrep_read_tuple(StringInfo in, LogicalRepTupleData *tuple);
      40             : 
      41             : static void logicalrep_write_namespace(StringInfo out, Oid nspid);
      42             : static const char *logicalrep_read_namespace(StringInfo in);
      43             : 
      44             : /*
      45             :  * Write BEGIN to the output stream.
      46             :  */
      47             : void
      48         820 : logicalrep_write_begin(StringInfo out, ReorderBufferTXN *txn)
      49             : {
      50         820 :     pq_sendbyte(out, LOGICAL_REP_MSG_BEGIN);
      51             : 
      52             :     /* fixed fields */
      53         820 :     pq_sendint64(out, txn->final_lsn);
      54         820 :     pq_sendint64(out, txn->xact_time.commit_time);
      55         820 :     pq_sendint32(out, txn->xid);
      56         820 : }
      57             : 
      58             : /*
      59             :  * Read transaction BEGIN from the stream.
      60             :  */
      61             : void
      62         888 : logicalrep_read_begin(StringInfo in, LogicalRepBeginData *begin_data)
      63             : {
      64             :     /* read fields */
      65         888 :     begin_data->final_lsn = pq_getmsgint64(in);
      66         888 :     if (begin_data->final_lsn == InvalidXLogRecPtr)
      67           0 :         elog(ERROR, "final_lsn not set in begin message");
      68         888 :     begin_data->committime = pq_getmsgint64(in);
      69         888 :     begin_data->xid = pq_getmsgint(in, 4);
      70         888 : }
      71             : 
      72             : 
      73             : /*
      74             :  * Write COMMIT to the output stream.
      75             :  */
      76             : void
      77         818 : logicalrep_write_commit(StringInfo out, ReorderBufferTXN *txn,
      78             :                         XLogRecPtr commit_lsn)
      79             : {
      80         818 :     uint8       flags = 0;
      81             : 
      82         818 :     pq_sendbyte(out, LOGICAL_REP_MSG_COMMIT);
      83             : 
      84             :     /* send the flags field (unused for now) */
      85         818 :     pq_sendbyte(out, flags);
      86             : 
      87             :     /* send fields */
      88         818 :     pq_sendint64(out, commit_lsn);
      89         818 :     pq_sendint64(out, txn->end_lsn);
      90         818 :     pq_sendint64(out, txn->xact_time.commit_time);
      91         818 : }
      92             : 
      93             : /*
      94             :  * Read transaction COMMIT from the stream.
      95             :  */
      96             : void
      97         834 : logicalrep_read_commit(StringInfo in, LogicalRepCommitData *commit_data)
      98             : {
      99             :     /* read flags (unused for now) */
     100         834 :     uint8       flags = pq_getmsgbyte(in);
     101             : 
     102         834 :     if (flags != 0)
     103           0 :         elog(ERROR, "unrecognized flags %u in commit message", flags);
     104             : 
     105             :     /* read fields */
     106         834 :     commit_data->commit_lsn = pq_getmsgint64(in);
     107         834 :     commit_data->end_lsn = pq_getmsgint64(in);
     108         834 :     commit_data->committime = pq_getmsgint64(in);
     109         834 : }
     110             : 
     111             : /*
     112             :  * Write BEGIN PREPARE to the output stream.
     113             :  */
     114             : void
     115          40 : logicalrep_write_begin_prepare(StringInfo out, ReorderBufferTXN *txn)
     116             : {
     117          40 :     pq_sendbyte(out, LOGICAL_REP_MSG_BEGIN_PREPARE);
     118             : 
     119             :     /* fixed fields */
     120          40 :     pq_sendint64(out, txn->final_lsn);
     121          40 :     pq_sendint64(out, txn->end_lsn);
     122          40 :     pq_sendint64(out, txn->xact_time.prepare_time);
     123          40 :     pq_sendint32(out, txn->xid);
     124             : 
     125             :     /* send gid */
     126          40 :     pq_sendstring(out, txn->gid);
     127          40 : }
     128             : 
     129             : /*
     130             :  * Read transaction BEGIN PREPARE from the stream.
     131             :  */
     132             : void
     133          32 : logicalrep_read_begin_prepare(StringInfo in, LogicalRepPreparedTxnData *begin_data)
     134             : {
     135             :     /* read fields */
     136          32 :     begin_data->prepare_lsn = pq_getmsgint64(in);
     137          32 :     if (begin_data->prepare_lsn == InvalidXLogRecPtr)
     138           0 :         elog(ERROR, "prepare_lsn not set in begin prepare message");
     139          32 :     begin_data->end_lsn = pq_getmsgint64(in);
     140          32 :     if (begin_data->end_lsn == InvalidXLogRecPtr)
     141           0 :         elog(ERROR, "end_lsn not set in begin prepare message");
     142          32 :     begin_data->prepare_time = pq_getmsgint64(in);
     143          32 :     begin_data->xid = pq_getmsgint(in, 4);
     144             : 
     145             :     /* read gid (copy it into a pre-allocated buffer) */
     146          32 :     strlcpy(begin_data->gid, pq_getmsgstring(in), sizeof(begin_data->gid));
     147          32 : }
     148             : 
     149             : /*
     150             :  * The core functionality for logicalrep_write_prepare and
     151             :  * logicalrep_write_stream_prepare.
     152             :  */
     153             : static void
     154          68 : logicalrep_write_prepare_common(StringInfo out, LogicalRepMsgType type,
     155             :                                 ReorderBufferTXN *txn, XLogRecPtr prepare_lsn)
     156             : {
     157          68 :     uint8       flags = 0;
     158             : 
     159          68 :     pq_sendbyte(out, type);
     160             : 
     161             :     /*
     162             :      * This should only ever happen for two-phase commit transactions, in
     163             :      * which case we expect to have a valid GID.
     164             :      */
     165             :     Assert(txn->gid != NULL);
     166             :     Assert(rbtxn_prepared(txn));
     167             :     Assert(TransactionIdIsValid(txn->xid));
     168             : 
     169             :     /* send the flags field */
     170          68 :     pq_sendbyte(out, flags);
     171             : 
     172             :     /* send fields */
     173          68 :     pq_sendint64(out, prepare_lsn);
     174          68 :     pq_sendint64(out, txn->end_lsn);
     175          68 :     pq_sendint64(out, txn->xact_time.prepare_time);
     176          68 :     pq_sendint32(out, txn->xid);
     177             : 
     178             :     /* send gid */
     179          68 :     pq_sendstring(out, txn->gid);
     180          68 : }
     181             : 
     182             : /*
     183             :  * Write PREPARE to the output stream.
     184             :  */
     185             : void
     186          40 : logicalrep_write_prepare(StringInfo out, ReorderBufferTXN *txn,
     187             :                          XLogRecPtr prepare_lsn)
     188             : {
     189          40 :     logicalrep_write_prepare_common(out, LOGICAL_REP_MSG_PREPARE,
     190             :                                     txn, prepare_lsn);
     191          40 : }
     192             : 
     193             : /*
     194             :  * The core functionality for logicalrep_read_prepare and
     195             :  * logicalrep_read_stream_prepare.
     196             :  */
     197             : static void
     198          52 : logicalrep_read_prepare_common(StringInfo in, char *msgtype,
     199             :                                LogicalRepPreparedTxnData *prepare_data)
     200             : {
     201             :     /* read flags */
     202          52 :     uint8       flags = pq_getmsgbyte(in);
     203             : 
     204          52 :     if (flags != 0)
     205           0 :         elog(ERROR, "unrecognized flags %u in %s message", flags, msgtype);
     206             : 
     207             :     /* read fields */
     208          52 :     prepare_data->prepare_lsn = pq_getmsgint64(in);
     209          52 :     if (prepare_data->prepare_lsn == InvalidXLogRecPtr)
     210           0 :         elog(ERROR, "prepare_lsn is not set in %s message", msgtype);
     211          52 :     prepare_data->end_lsn = pq_getmsgint64(in);
     212          52 :     if (prepare_data->end_lsn == InvalidXLogRecPtr)
     213           0 :         elog(ERROR, "end_lsn is not set in %s message", msgtype);
     214          52 :     prepare_data->prepare_time = pq_getmsgint64(in);
     215          52 :     prepare_data->xid = pq_getmsgint(in, 4);
     216          52 :     if (prepare_data->xid == InvalidTransactionId)
     217           0 :         elog(ERROR, "invalid two-phase transaction ID in %s message", msgtype);
     218             : 
     219             :     /* read gid (copy it into a pre-allocated buffer) */
     220          52 :     strlcpy(prepare_data->gid, pq_getmsgstring(in), sizeof(prepare_data->gid));
     221          52 : }
     222             : 
     223             : /*
     224             :  * Read transaction PREPARE from the stream.
     225             :  */
     226             : void
     227          30 : logicalrep_read_prepare(StringInfo in, LogicalRepPreparedTxnData *prepare_data)
     228             : {
     229          30 :     logicalrep_read_prepare_common(in, "prepare", prepare_data);
     230          30 : }
     231             : 
     232             : /*
     233             :  * Write COMMIT PREPARED to the output stream.
     234             :  */
     235             : void
     236          48 : logicalrep_write_commit_prepared(StringInfo out, ReorderBufferTXN *txn,
     237             :                                  XLogRecPtr commit_lsn)
     238             : {
     239          48 :     uint8       flags = 0;
     240             : 
     241          48 :     pq_sendbyte(out, LOGICAL_REP_MSG_COMMIT_PREPARED);
     242             : 
     243             :     /*
     244             :      * This should only ever happen for two-phase commit transactions, in
     245             :      * which case we expect to have a valid GID.
     246             :      */
     247             :     Assert(txn->gid != NULL);
     248             : 
     249             :     /* send the flags field */
     250          48 :     pq_sendbyte(out, flags);
     251             : 
     252             :     /* send fields */
     253          48 :     pq_sendint64(out, commit_lsn);
     254          48 :     pq_sendint64(out, txn->end_lsn);
     255          48 :     pq_sendint64(out, txn->xact_time.commit_time);
     256          48 :     pq_sendint32(out, txn->xid);
     257             : 
     258             :     /* send gid */
     259          48 :     pq_sendstring(out, txn->gid);
     260          48 : }
     261             : 
     262             : /*
     263             :  * Read transaction COMMIT PREPARED from the stream.
     264             :  */
     265             : void
     266          40 : logicalrep_read_commit_prepared(StringInfo in, LogicalRepCommitPreparedTxnData *prepare_data)
     267             : {
     268             :     /* read flags */
     269          40 :     uint8       flags = pq_getmsgbyte(in);
     270             : 
     271          40 :     if (flags != 0)
     272           0 :         elog(ERROR, "unrecognized flags %u in commit prepared message", flags);
     273             : 
     274             :     /* read fields */
     275          40 :     prepare_data->commit_lsn = pq_getmsgint64(in);
     276          40 :     if (prepare_data->commit_lsn == InvalidXLogRecPtr)
     277           0 :         elog(ERROR, "commit_lsn is not set in commit prepared message");
     278          40 :     prepare_data->end_lsn = pq_getmsgint64(in);
     279          40 :     if (prepare_data->end_lsn == InvalidXLogRecPtr)
     280           0 :         elog(ERROR, "end_lsn is not set in commit prepared message");
     281          40 :     prepare_data->commit_time = pq_getmsgint64(in);
     282          40 :     prepare_data->xid = pq_getmsgint(in, 4);
     283             : 
     284             :     /* read gid (copy it into a pre-allocated buffer) */
     285          40 :     strlcpy(prepare_data->gid, pq_getmsgstring(in), sizeof(prepare_data->gid));
     286          40 : }
     287             : 
     288             : /*
     289             :  * Write ROLLBACK PREPARED to the output stream.
     290             :  */
     291             : void
     292          18 : logicalrep_write_rollback_prepared(StringInfo out, ReorderBufferTXN *txn,
     293             :                                    XLogRecPtr prepare_end_lsn,
     294             :                                    TimestampTz prepare_time)
     295             : {
     296          18 :     uint8       flags = 0;
     297             : 
     298          18 :     pq_sendbyte(out, LOGICAL_REP_MSG_ROLLBACK_PREPARED);
     299             : 
     300             :     /*
     301             :      * This should only ever happen for two-phase commit transactions, in
     302             :      * which case we expect to have a valid GID.
     303             :      */
     304             :     Assert(txn->gid != NULL);
     305             : 
     306             :     /* send the flags field */
     307          18 :     pq_sendbyte(out, flags);
     308             : 
     309             :     /* send fields */
     310          18 :     pq_sendint64(out, prepare_end_lsn);
     311          18 :     pq_sendint64(out, txn->end_lsn);
     312          18 :     pq_sendint64(out, prepare_time);
     313          18 :     pq_sendint64(out, txn->xact_time.commit_time);
     314          18 :     pq_sendint32(out, txn->xid);
     315             : 
     316             :     /* send gid */
     317          18 :     pq_sendstring(out, txn->gid);
     318          18 : }
     319             : 
     320             : /*
     321             :  * Read transaction ROLLBACK PREPARED from the stream.
     322             :  */
     323             : void
     324          10 : logicalrep_read_rollback_prepared(StringInfo in,
     325             :                                   LogicalRepRollbackPreparedTxnData *rollback_data)
     326             : {
     327             :     /* read flags */
     328          10 :     uint8       flags = pq_getmsgbyte(in);
     329             : 
     330          10 :     if (flags != 0)
     331           0 :         elog(ERROR, "unrecognized flags %u in rollback prepared message", flags);
     332             : 
     333             :     /* read fields */
     334          10 :     rollback_data->prepare_end_lsn = pq_getmsgint64(in);
     335          10 :     if (rollback_data->prepare_end_lsn == InvalidXLogRecPtr)
     336           0 :         elog(ERROR, "prepare_end_lsn is not set in rollback prepared message");
     337          10 :     rollback_data->rollback_end_lsn = pq_getmsgint64(in);
     338          10 :     if (rollback_data->rollback_end_lsn == InvalidXLogRecPtr)
     339           0 :         elog(ERROR, "rollback_end_lsn is not set in rollback prepared message");
     340          10 :     rollback_data->prepare_time = pq_getmsgint64(in);
     341          10 :     rollback_data->rollback_time = pq_getmsgint64(in);
     342          10 :     rollback_data->xid = pq_getmsgint(in, 4);
     343             : 
     344             :     /* read gid (copy it into a pre-allocated buffer) */
     345          10 :     strlcpy(rollback_data->gid, pq_getmsgstring(in), sizeof(rollback_data->gid));
     346          10 : }
     347             : 
     348             : /*
     349             :  * Write STREAM PREPARE to the output stream.
     350             :  */
     351             : void
     352          28 : logicalrep_write_stream_prepare(StringInfo out,
     353             :                                 ReorderBufferTXN *txn,
     354             :                                 XLogRecPtr prepare_lsn)
     355             : {
     356          28 :     logicalrep_write_prepare_common(out, LOGICAL_REP_MSG_STREAM_PREPARE,
     357             :                                     txn, prepare_lsn);
     358          28 : }
     359             : 
     360             : /*
     361             :  * Read STREAM PREPARE from the stream.
     362             :  */
     363             : void
     364          22 : logicalrep_read_stream_prepare(StringInfo in, LogicalRepPreparedTxnData *prepare_data)
     365             : {
     366          22 :     logicalrep_read_prepare_common(in, "stream prepare", prepare_data);
     367          22 : }
     368             : 
     369             : /*
     370             :  * Write ORIGIN to the output stream.
     371             :  */
     372             : void
     373          18 : logicalrep_write_origin(StringInfo out, const char *origin,
     374             :                         XLogRecPtr origin_lsn)
     375             : {
     376          18 :     pq_sendbyte(out, LOGICAL_REP_MSG_ORIGIN);
     377             : 
     378             :     /* fixed fields */
     379          18 :     pq_sendint64(out, origin_lsn);
     380             : 
     381             :     /* origin string */
     382          18 :     pq_sendstring(out, origin);
     383          18 : }
     384             : 
     385             : /*
     386             :  * Read ORIGIN from the output stream.
     387             :  */
     388             : char *
     389           0 : logicalrep_read_origin(StringInfo in, XLogRecPtr *origin_lsn)
     390             : {
     391             :     /* fixed fields */
     392           0 :     *origin_lsn = pq_getmsgint64(in);
     393             : 
     394             :     /* return origin */
     395           0 :     return pstrdup(pq_getmsgstring(in));
     396             : }
     397             : 
     398             : /*
     399             :  * Write INSERT to the output stream.
     400             :  */
     401             : void
     402      211774 : logicalrep_write_insert(StringInfo out, TransactionId xid, Relation rel,
     403             :                         TupleTableSlot *newslot, bool binary,
     404             :                         Bitmapset *columns, bool include_gencols)
     405             : {
     406      211774 :     pq_sendbyte(out, LOGICAL_REP_MSG_INSERT);
     407             : 
     408             :     /* transaction ID (if not valid, we're not streaming) */
     409      211774 :     if (TransactionIdIsValid(xid))
     410      200168 :         pq_sendint32(out, xid);
     411             : 
     412             :     /* use Oid as relation identifier */
     413      211774 :     pq_sendint32(out, RelationGetRelid(rel));
     414             : 
     415      211774 :     pq_sendbyte(out, 'N');      /* new tuple follows */
     416      211774 :     logicalrep_write_tuple(out, rel, newslot, binary, columns, include_gencols);
     417      211774 : }
     418             : 
     419             : /*
     420             :  * Read INSERT from stream.
     421             :  *
     422             :  * Fills the new tuple.
     423             :  */
     424             : LogicalRepRelId
     425      152796 : logicalrep_read_insert(StringInfo in, LogicalRepTupleData *newtup)
     426             : {
     427             :     char        action;
     428             :     LogicalRepRelId relid;
     429             : 
     430             :     /* read the relation id */
     431      152796 :     relid = pq_getmsgint(in, 4);
     432             : 
     433      152796 :     action = pq_getmsgbyte(in);
     434      152796 :     if (action != 'N')
     435           0 :         elog(ERROR, "expected new tuple but got %d",
     436             :              action);
     437             : 
     438      152796 :     logicalrep_read_tuple(in, newtup);
     439             : 
     440      152796 :     return relid;
     441             : }
     442             : 
     443             : /*
     444             :  * Write UPDATE to the output stream.
     445             :  */
     446             : void
     447       68878 : logicalrep_write_update(StringInfo out, TransactionId xid, Relation rel,
     448             :                         TupleTableSlot *oldslot, TupleTableSlot *newslot,
     449             :                         bool binary, Bitmapset *columns, bool include_gencols)
     450             : {
     451       68878 :     pq_sendbyte(out, LOGICAL_REP_MSG_UPDATE);
     452             : 
     453             :     Assert(rel->rd_rel->relreplident == REPLICA_IDENTITY_DEFAULT ||
     454             :            rel->rd_rel->relreplident == REPLICA_IDENTITY_FULL ||
     455             :            rel->rd_rel->relreplident == REPLICA_IDENTITY_INDEX);
     456             : 
     457             :     /* transaction ID (if not valid, we're not streaming) */
     458       68878 :     if (TransactionIdIsValid(xid))
     459       68464 :         pq_sendint32(out, xid);
     460             : 
     461             :     /* use Oid as relation identifier */
     462       68878 :     pq_sendint32(out, RelationGetRelid(rel));
     463             : 
     464       68878 :     if (oldslot != NULL)
     465             :     {
     466         254 :         if (rel->rd_rel->relreplident == REPLICA_IDENTITY_FULL)
     467         122 :             pq_sendbyte(out, 'O');  /* old tuple follows */
     468             :         else
     469         132 :             pq_sendbyte(out, 'K');  /* old key follows */
     470         254 :         logicalrep_write_tuple(out, rel, oldslot, binary, columns,
     471             :                                include_gencols);
     472             :     }
     473             : 
     474       68878 :     pq_sendbyte(out, 'N');      /* new tuple follows */
     475       68878 :     logicalrep_write_tuple(out, rel, newslot, binary, columns, include_gencols);
     476       68878 : }
     477             : 
     478             : /*
     479             :  * Read UPDATE from stream.
     480             :  */
     481             : LogicalRepRelId
     482       63866 : logicalrep_read_update(StringInfo in, bool *has_oldtuple,
     483             :                        LogicalRepTupleData *oldtup,
     484             :                        LogicalRepTupleData *newtup)
     485             : {
     486             :     char        action;
     487             :     LogicalRepRelId relid;
     488             : 
     489             :     /* read the relation id */
     490       63866 :     relid = pq_getmsgint(in, 4);
     491             : 
     492             :     /* read and verify action */
     493       63866 :     action = pq_getmsgbyte(in);
     494       63866 :     if (action != 'K' && action != 'O' && action != 'N')
     495           0 :         elog(ERROR, "expected action 'N', 'O' or 'K', got %c",
     496             :              action);
     497             : 
     498             :     /* check for old tuple */
     499       63866 :     if (action == 'K' || action == 'O')
     500             :     {
     501         258 :         logicalrep_read_tuple(in, oldtup);
     502         258 :         *has_oldtuple = true;
     503             : 
     504         258 :         action = pq_getmsgbyte(in);
     505             :     }
     506             :     else
     507       63608 :         *has_oldtuple = false;
     508             : 
     509             :     /* check for new  tuple */
     510       63866 :     if (action != 'N')
     511           0 :         elog(ERROR, "expected action 'N', got %c",
     512             :              action);
     513             : 
     514       63866 :     logicalrep_read_tuple(in, newtup);
     515             : 
     516       63866 :     return relid;
     517             : }
     518             : 
     519             : /*
     520             :  * Write DELETE to the output stream.
     521             :  */
     522             : void
     523       83766 : logicalrep_write_delete(StringInfo out, TransactionId xid, Relation rel,
     524             :                         TupleTableSlot *oldslot, bool binary,
     525             :                         Bitmapset *columns, bool include_gencols)
     526             : {
     527             :     Assert(rel->rd_rel->relreplident == REPLICA_IDENTITY_DEFAULT ||
     528             :            rel->rd_rel->relreplident == REPLICA_IDENTITY_FULL ||
     529             :            rel->rd_rel->relreplident == REPLICA_IDENTITY_INDEX);
     530             : 
     531       83766 :     pq_sendbyte(out, LOGICAL_REP_MSG_DELETE);
     532             : 
     533             :     /* transaction ID (if not valid, we're not streaming) */
     534       83766 :     if (TransactionIdIsValid(xid))
     535       83250 :         pq_sendint32(out, xid);
     536             : 
     537             :     /* use Oid as relation identifier */
     538       83766 :     pq_sendint32(out, RelationGetRelid(rel));
     539             : 
     540       83766 :     if (rel->rd_rel->relreplident == REPLICA_IDENTITY_FULL)
     541         254 :         pq_sendbyte(out, 'O');  /* old tuple follows */
     542             :     else
     543       83512 :         pq_sendbyte(out, 'K');  /* old key follows */
     544             : 
     545       83766 :     logicalrep_write_tuple(out, rel, oldslot, binary, columns, include_gencols);
     546       83766 : }
     547             : 
     548             : /*
     549             :  * Read DELETE from stream.
     550             :  *
     551             :  * Fills the old tuple.
     552             :  */
     553             : LogicalRepRelId
     554       80636 : logicalrep_read_delete(StringInfo in, LogicalRepTupleData *oldtup)
     555             : {
     556             :     char        action;
     557             :     LogicalRepRelId relid;
     558             : 
     559             :     /* read the relation id */
     560       80636 :     relid = pq_getmsgint(in, 4);
     561             : 
     562             :     /* read and verify action */
     563       80636 :     action = pq_getmsgbyte(in);
     564       80636 :     if (action != 'K' && action != 'O')
     565           0 :         elog(ERROR, "expected action 'O' or 'K', got %c", action);
     566             : 
     567       80636 :     logicalrep_read_tuple(in, oldtup);
     568             : 
     569       80636 :     return relid;
     570             : }
     571             : 
     572             : /*
     573             :  * Write TRUNCATE to the output stream.
     574             :  */
     575             : void
     576          20 : logicalrep_write_truncate(StringInfo out,
     577             :                           TransactionId xid,
     578             :                           int nrelids,
     579             :                           Oid relids[],
     580             :                           bool cascade, bool restart_seqs)
     581             : {
     582             :     int         i;
     583          20 :     uint8       flags = 0;
     584             : 
     585          20 :     pq_sendbyte(out, LOGICAL_REP_MSG_TRUNCATE);
     586             : 
     587             :     /* transaction ID (if not valid, we're not streaming) */
     588          20 :     if (TransactionIdIsValid(xid))
     589           0 :         pq_sendint32(out, xid);
     590             : 
     591          20 :     pq_sendint32(out, nrelids);
     592             : 
     593             :     /* encode and send truncate flags */
     594          20 :     if (cascade)
     595           0 :         flags |= TRUNCATE_CASCADE;
     596          20 :     if (restart_seqs)
     597           0 :         flags |= TRUNCATE_RESTART_SEQS;
     598          20 :     pq_sendint8(out, flags);
     599             : 
     600          52 :     for (i = 0; i < nrelids; i++)
     601          32 :         pq_sendint32(out, relids[i]);
     602          20 : }
     603             : 
     604             : /*
     605             :  * Read TRUNCATE from stream.
     606             :  */
     607             : List *
     608          38 : logicalrep_read_truncate(StringInfo in,
     609             :                          bool *cascade, bool *restart_seqs)
     610             : {
     611             :     int         i;
     612             :     int         nrelids;
     613          38 :     List       *relids = NIL;
     614             :     uint8       flags;
     615             : 
     616          38 :     nrelids = pq_getmsgint(in, 4);
     617             : 
     618             :     /* read and decode truncate flags */
     619          38 :     flags = pq_getmsgint(in, 1);
     620          38 :     *cascade = (flags & TRUNCATE_CASCADE) > 0;
     621          38 :     *restart_seqs = (flags & TRUNCATE_RESTART_SEQS) > 0;
     622             : 
     623          94 :     for (i = 0; i < nrelids; i++)
     624          56 :         relids = lappend_oid(relids, pq_getmsgint(in, 4));
     625             : 
     626          38 :     return relids;
     627             : }
     628             : 
     629             : /*
     630             :  * Write MESSAGE to stream
     631             :  */
     632             : void
     633          10 : logicalrep_write_message(StringInfo out, TransactionId xid, XLogRecPtr lsn,
     634             :                          bool transactional, const char *prefix, Size sz,
     635             :                          const char *message)
     636             : {
     637          10 :     uint8       flags = 0;
     638             : 
     639          10 :     pq_sendbyte(out, LOGICAL_REP_MSG_MESSAGE);
     640             : 
     641             :     /* encode and send message flags */
     642          10 :     if (transactional)
     643           4 :         flags |= MESSAGE_TRANSACTIONAL;
     644             : 
     645             :     /* transaction ID (if not valid, we're not streaming) */
     646          10 :     if (TransactionIdIsValid(xid))
     647           0 :         pq_sendint32(out, xid);
     648             : 
     649          10 :     pq_sendint8(out, flags);
     650          10 :     pq_sendint64(out, lsn);
     651          10 :     pq_sendstring(out, prefix);
     652          10 :     pq_sendint32(out, sz);
     653          10 :     pq_sendbytes(out, message, sz);
     654          10 : }
     655             : 
     656             : /*
     657             :  * Write relation description to the output stream.
     658             :  */
     659             : void
     660         676 : logicalrep_write_rel(StringInfo out, TransactionId xid, Relation rel,
     661             :                      Bitmapset *columns, bool include_gencols)
     662             : {
     663             :     char       *relname;
     664             : 
     665         676 :     pq_sendbyte(out, LOGICAL_REP_MSG_RELATION);
     666             : 
     667             :     /* transaction ID (if not valid, we're not streaming) */
     668         676 :     if (TransactionIdIsValid(xid))
     669         142 :         pq_sendint32(out, xid);
     670             : 
     671             :     /* use Oid as relation identifier */
     672         676 :     pq_sendint32(out, RelationGetRelid(rel));
     673             : 
     674             :     /* send qualified relation name */
     675         676 :     logicalrep_write_namespace(out, RelationGetNamespace(rel));
     676         676 :     relname = RelationGetRelationName(rel);
     677         676 :     pq_sendstring(out, relname);
     678             : 
     679             :     /* send replica identity */
     680         676 :     pq_sendbyte(out, rel->rd_rel->relreplident);
     681             : 
     682             :     /* send the attribute info */
     683         676 :     logicalrep_write_attrs(out, rel, columns, include_gencols);
     684         676 : }
     685             : 
     686             : /*
     687             :  * Read the relation info from stream and return as LogicalRepRelation.
     688             :  */
     689             : LogicalRepRelation *
     690         788 : logicalrep_read_rel(StringInfo in)
     691             : {
     692         788 :     LogicalRepRelation *rel = palloc(sizeof(LogicalRepRelation));
     693             : 
     694         788 :     rel->remoteid = pq_getmsgint(in, 4);
     695             : 
     696             :     /* Read relation name from stream */
     697         788 :     rel->nspname = pstrdup(logicalrep_read_namespace(in));
     698         788 :     rel->relname = pstrdup(pq_getmsgstring(in));
     699             : 
     700             :     /* Read the replica identity. */
     701         788 :     rel->replident = pq_getmsgbyte(in);
     702             : 
     703             :     /* Get attribute description */
     704         788 :     logicalrep_read_attrs(in, rel);
     705             : 
     706         788 :     return rel;
     707             : }
     708             : 
     709             : /*
     710             :  * Write type info to the output stream.
     711             :  *
     712             :  * This function will always write base type info.
     713             :  */
     714             : void
     715          36 : logicalrep_write_typ(StringInfo out, TransactionId xid, Oid typoid)
     716             : {
     717          36 :     Oid         basetypoid = getBaseType(typoid);
     718             :     HeapTuple   tup;
     719             :     Form_pg_type typtup;
     720             : 
     721          36 :     pq_sendbyte(out, LOGICAL_REP_MSG_TYPE);
     722             : 
     723             :     /* transaction ID (if not valid, we're not streaming) */
     724          36 :     if (TransactionIdIsValid(xid))
     725           0 :         pq_sendint32(out, xid);
     726             : 
     727          36 :     tup = SearchSysCache1(TYPEOID, ObjectIdGetDatum(basetypoid));
     728          36 :     if (!HeapTupleIsValid(tup))
     729           0 :         elog(ERROR, "cache lookup failed for type %u", basetypoid);
     730          36 :     typtup = (Form_pg_type) GETSTRUCT(tup);
     731             : 
     732             :     /* use Oid as type identifier */
     733          36 :     pq_sendint32(out, typoid);
     734             : 
     735             :     /* send qualified type name */
     736          36 :     logicalrep_write_namespace(out, typtup->typnamespace);
     737          36 :     pq_sendstring(out, NameStr(typtup->typname));
     738             : 
     739          36 :     ReleaseSysCache(tup);
     740          36 : }
     741             : 
     742             : /*
     743             :  * Read type info from the output stream.
     744             :  */
     745             : void
     746          36 : logicalrep_read_typ(StringInfo in, LogicalRepTyp *ltyp)
     747             : {
     748          36 :     ltyp->remoteid = pq_getmsgint(in, 4);
     749             : 
     750             :     /* Read type name from stream */
     751          36 :     ltyp->nspname = pstrdup(logicalrep_read_namespace(in));
     752          36 :     ltyp->typname = pstrdup(pq_getmsgstring(in));
     753          36 : }
     754             : 
     755             : /*
     756             :  * Write a tuple to the outputstream, in the most efficient format possible.
     757             :  */
     758             : static void
     759      364672 : logicalrep_write_tuple(StringInfo out, Relation rel, TupleTableSlot *slot,
     760             :                        bool binary, Bitmapset *columns, bool include_gencols)
     761             : {
     762             :     TupleDesc   desc;
     763             :     Datum      *values;
     764             :     bool       *isnull;
     765             :     int         i;
     766      364672 :     uint16      nliveatts = 0;
     767             : 
     768      364672 :     desc = RelationGetDescr(rel);
     769             : 
     770     1096984 :     for (i = 0; i < desc->natts; i++)
     771             :     {
     772      732312 :         Form_pg_attribute att = TupleDescAttr(desc, i);
     773             : 
     774      732312 :         if (!logicalrep_should_publish_column(att, columns, include_gencols))
     775         262 :             continue;
     776             : 
     777      732050 :         nliveatts++;
     778             :     }
     779      364672 :     pq_sendint16(out, nliveatts);
     780             : 
     781      364672 :     slot_getallattrs(slot);
     782      364672 :     values = slot->tts_values;
     783      364672 :     isnull = slot->tts_isnull;
     784             : 
     785             :     /* Write the values */
     786     1096984 :     for (i = 0; i < desc->natts; i++)
     787             :     {
     788             :         HeapTuple   typtup;
     789             :         Form_pg_type typclass;
     790      732312 :         Form_pg_attribute att = TupleDescAttr(desc, i);
     791             : 
     792      732312 :         if (!logicalrep_should_publish_column(att, columns, include_gencols))
     793         262 :             continue;
     794             : 
     795      732050 :         if (isnull[i])
     796             :         {
     797      103826 :             pq_sendbyte(out, LOGICALREP_COLUMN_NULL);
     798      103826 :             continue;
     799             :         }
     800             : 
     801      628224 :         if (att->attlen == -1 && VARATT_IS_EXTERNAL_ONDISK(values[i]))
     802             :         {
     803             :             /*
     804             :              * Unchanged toasted datum.  (Note that we don't promise to detect
     805             :              * unchanged data in general; this is just a cheap check to avoid
     806             :              * sending large values unnecessarily.)
     807             :              */
     808           6 :             pq_sendbyte(out, LOGICALREP_COLUMN_UNCHANGED);
     809           6 :             continue;
     810             :         }
     811             : 
     812      628218 :         typtup = SearchSysCache1(TYPEOID, ObjectIdGetDatum(att->atttypid));
     813      628218 :         if (!HeapTupleIsValid(typtup))
     814           0 :             elog(ERROR, "cache lookup failed for type %u", att->atttypid);
     815      628218 :         typclass = (Form_pg_type) GETSTRUCT(typtup);
     816             : 
     817             :         /*
     818             :          * Send in binary if requested and type has suitable send function.
     819             :          */
     820      628218 :         if (binary && OidIsValid(typclass->typsend))
     821      230090 :         {
     822             :             bytea      *outputbytes;
     823             :             int         len;
     824             : 
     825      230090 :             pq_sendbyte(out, LOGICALREP_COLUMN_BINARY);
     826      230090 :             outputbytes = OidSendFunctionCall(typclass->typsend, values[i]);
     827      230090 :             len = VARSIZE(outputbytes) - VARHDRSZ;
     828      230090 :             pq_sendint(out, len, 4);    /* length */
     829      230090 :             pq_sendbytes(out, VARDATA(outputbytes), len);   /* data */
     830      230090 :             pfree(outputbytes);
     831             :         }
     832             :         else
     833             :         {
     834             :             char       *outputstr;
     835             : 
     836      398128 :             pq_sendbyte(out, LOGICALREP_COLUMN_TEXT);
     837      398128 :             outputstr = OidOutputFunctionCall(typclass->typoutput, values[i]);
     838      398128 :             pq_sendcountedtext(out, outputstr, strlen(outputstr));
     839      398128 :             pfree(outputstr);
     840             :         }
     841             : 
     842      628218 :         ReleaseSysCache(typtup);
     843             :     }
     844      364672 : }
     845             : 
     846             : /*
     847             :  * Read tuple in logical replication format from stream.
     848             :  */
     849             : static void
     850      297556 : logicalrep_read_tuple(StringInfo in, LogicalRepTupleData *tuple)
     851             : {
     852             :     int         i;
     853             :     int         natts;
     854             : 
     855             :     /* Get number of attributes */
     856      297556 :     natts = pq_getmsgint(in, 2);
     857             : 
     858             :     /* Allocate space for per-column values; zero out unused StringInfoDatas */
     859      297556 :     tuple->colvalues = (StringInfoData *) palloc0(natts * sizeof(StringInfoData));
     860      297556 :     tuple->colstatus = (char *) palloc(natts * sizeof(char));
     861      297556 :     tuple->ncols = natts;
     862             : 
     863             :     /* Read the data */
     864      904340 :     for (i = 0; i < natts; i++)
     865             :     {
     866             :         char       *buff;
     867             :         char        kind;
     868             :         int         len;
     869      606784 :         StringInfo  value = &tuple->colvalues[i];
     870             : 
     871      606784 :         kind = pq_getmsgbyte(in);
     872      606784 :         tuple->colstatus[i] = kind;
     873             : 
     874      606784 :         switch (kind)
     875             :         {
     876      100718 :             case LOGICALREP_COLUMN_NULL:
     877             :                 /* nothing more to do */
     878      100718 :                 break;
     879           6 :             case LOGICALREP_COLUMN_UNCHANGED:
     880             :                 /* we don't receive the value of an unchanged column */
     881           6 :                 break;
     882      506060 :             case LOGICALREP_COLUMN_TEXT:
     883             :             case LOGICALREP_COLUMN_BINARY:
     884      506060 :                 len = pq_getmsgint(in, 4);  /* read length */
     885             : 
     886             :                 /* and data */
     887      506060 :                 buff = palloc(len + 1);
     888      506060 :                 pq_copymsgbytes(in, buff, len);
     889             : 
     890             :                 /*
     891             :                  * NUL termination is required for LOGICALREP_COLUMN_TEXT mode
     892             :                  * as input functions require that.  For
     893             :                  * LOGICALREP_COLUMN_BINARY it's not technically required, but
     894             :                  * it's harmless.
     895             :                  */
     896      506060 :                 buff[len] = '\0';
     897             : 
     898      506060 :                 initStringInfoFromString(value, buff, len);
     899      506060 :                 break;
     900           0 :             default:
     901           0 :                 elog(ERROR, "unrecognized data representation type '%c'", kind);
     902             :         }
     903             :     }
     904      297556 : }
     905             : 
     906             : /*
     907             :  * Write relation attribute metadata to the stream.
     908             :  */
     909             : static void
     910         676 : logicalrep_write_attrs(StringInfo out, Relation rel, Bitmapset *columns,
     911             :                        bool include_gencols)
     912             : {
     913             :     TupleDesc   desc;
     914             :     int         i;
     915         676 :     uint16      nliveatts = 0;
     916         676 :     Bitmapset  *idattrs = NULL;
     917             :     bool        replidentfull;
     918             : 
     919         676 :     desc = RelationGetDescr(rel);
     920             : 
     921             :     /* send number of live attributes */
     922        2060 :     for (i = 0; i < desc->natts; i++)
     923             :     {
     924        1384 :         Form_pg_attribute att = TupleDescAttr(desc, i);
     925             : 
     926        1384 :         if (!logicalrep_should_publish_column(att, columns, include_gencols))
     927         136 :             continue;
     928             : 
     929        1248 :         nliveatts++;
     930             :     }
     931         676 :     pq_sendint16(out, nliveatts);
     932             : 
     933             :     /* fetch bitmap of REPLICATION IDENTITY attributes */
     934         676 :     replidentfull = (rel->rd_rel->relreplident == REPLICA_IDENTITY_FULL);
     935         676 :     if (!replidentfull)
     936         574 :         idattrs = RelationGetIdentityKeyBitmap(rel);
     937             : 
     938             :     /* send the attributes */
     939        2060 :     for (i = 0; i < desc->natts; i++)
     940             :     {
     941        1384 :         Form_pg_attribute att = TupleDescAttr(desc, i);
     942        1384 :         uint8       flags = 0;
     943             : 
     944        1384 :         if (!logicalrep_should_publish_column(att, columns, include_gencols))
     945         136 :             continue;
     946             : 
     947             :         /* REPLICA IDENTITY FULL means all columns are sent as part of key. */
     948        2344 :         if (replidentfull ||
     949        1096 :             bms_is_member(att->attnum - FirstLowInvalidHeapAttributeNumber,
     950             :                           idattrs))
     951         614 :             flags |= LOGICALREP_IS_REPLICA_IDENTITY;
     952             : 
     953        1248 :         pq_sendbyte(out, flags);
     954             : 
     955             :         /* attribute name */
     956        1248 :         pq_sendstring(out, NameStr(att->attname));
     957             : 
     958             :         /* attribute type id */
     959        1248 :         pq_sendint32(out, (int) att->atttypid);
     960             : 
     961             :         /* attribute mode */
     962        1248 :         pq_sendint32(out, att->atttypmod);
     963             :     }
     964             : 
     965         676 :     bms_free(idattrs);
     966         676 : }
     967             : 
     968             : /*
     969             :  * Read relation attribute metadata from the stream.
     970             :  */
     971             : static void
     972         788 : logicalrep_read_attrs(StringInfo in, LogicalRepRelation *rel)
     973             : {
     974             :     int         i;
     975             :     int         natts;
     976             :     char      **attnames;
     977             :     Oid        *atttyps;
     978         788 :     Bitmapset  *attkeys = NULL;
     979             : 
     980         788 :     natts = pq_getmsgint(in, 2);
     981         788 :     attnames = palloc(natts * sizeof(char *));
     982         788 :     atttyps = palloc(natts * sizeof(Oid));
     983             : 
     984             :     /* read the attributes */
     985        2222 :     for (i = 0; i < natts; i++)
     986             :     {
     987             :         uint8       flags;
     988             : 
     989             :         /* Check for replica identity column */
     990        1434 :         flags = pq_getmsgbyte(in);
     991        1434 :         if (flags & LOGICALREP_IS_REPLICA_IDENTITY)
     992         696 :             attkeys = bms_add_member(attkeys, i);
     993             : 
     994             :         /* attribute name */
     995        1434 :         attnames[i] = pstrdup(pq_getmsgstring(in));
     996             : 
     997             :         /* attribute type id */
     998        1434 :         atttyps[i] = (Oid) pq_getmsgint(in, 4);
     999             : 
    1000             :         /* we ignore attribute mode for now */
    1001        1434 :         (void) pq_getmsgint(in, 4);
    1002             :     }
    1003             : 
    1004         788 :     rel->attnames = attnames;
    1005         788 :     rel->atttyps = atttyps;
    1006         788 :     rel->attkeys = attkeys;
    1007         788 :     rel->natts = natts;
    1008         788 : }
    1009             : 
    1010             : /*
    1011             :  * Write the namespace name or empty string for pg_catalog (to save space).
    1012             :  */
    1013             : static void
    1014         712 : logicalrep_write_namespace(StringInfo out, Oid nspid)
    1015             : {
    1016         712 :     if (nspid == PG_CATALOG_NAMESPACE)
    1017           2 :         pq_sendbyte(out, '\0');
    1018             :     else
    1019             :     {
    1020         710 :         char       *nspname = get_namespace_name(nspid);
    1021             : 
    1022         710 :         if (nspname == NULL)
    1023           0 :             elog(ERROR, "cache lookup failed for namespace %u",
    1024             :                  nspid);
    1025             : 
    1026         710 :         pq_sendstring(out, nspname);
    1027             :     }
    1028         712 : }
    1029             : 
    1030             : /*
    1031             :  * Read the namespace name while treating empty string as pg_catalog.
    1032             :  */
    1033             : static const char *
    1034         824 : logicalrep_read_namespace(StringInfo in)
    1035             : {
    1036         824 :     const char *nspname = pq_getmsgstring(in);
    1037             : 
    1038         824 :     if (nspname[0] == '\0')
    1039           2 :         nspname = "pg_catalog";
    1040             : 
    1041         824 :     return nspname;
    1042             : }
    1043             : 
    1044             : /*
    1045             :  * Write the information for the start stream message to the output stream.
    1046             :  */
    1047             : void
    1048        1274 : logicalrep_write_stream_start(StringInfo out,
    1049             :                               TransactionId xid, bool first_segment)
    1050             : {
    1051        1274 :     pq_sendbyte(out, LOGICAL_REP_MSG_STREAM_START);
    1052             : 
    1053             :     Assert(TransactionIdIsValid(xid));
    1054             : 
    1055             :     /* transaction ID (we're starting to stream, so must be valid) */
    1056        1274 :     pq_sendint32(out, xid);
    1057             : 
    1058             :     /* 1 if this is the first streaming segment for this xid */
    1059        1274 :     pq_sendbyte(out, first_segment ? 1 : 0);
    1060        1274 : }
    1061             : 
    1062             : /*
    1063             :  * Read the information about the start stream message from output stream.
    1064             :  */
    1065             : TransactionId
    1066        1676 : logicalrep_read_stream_start(StringInfo in, bool *first_segment)
    1067             : {
    1068             :     TransactionId xid;
    1069             : 
    1070             :     Assert(first_segment);
    1071             : 
    1072        1676 :     xid = pq_getmsgint(in, 4);
    1073        1676 :     *first_segment = (pq_getmsgbyte(in) == 1);
    1074             : 
    1075        1676 :     return xid;
    1076             : }
    1077             : 
    1078             : /*
    1079             :  * Write the stop stream message to the output stream.
    1080             :  */
    1081             : void
    1082        1274 : logicalrep_write_stream_stop(StringInfo out)
    1083             : {
    1084        1274 :     pq_sendbyte(out, LOGICAL_REP_MSG_STREAM_STOP);
    1085        1274 : }
    1086             : 
    1087             : /*
    1088             :  * Write STREAM COMMIT to the output stream.
    1089             :  */
    1090             : void
    1091          92 : logicalrep_write_stream_commit(StringInfo out, ReorderBufferTXN *txn,
    1092             :                                XLogRecPtr commit_lsn)
    1093             : {
    1094          92 :     uint8       flags = 0;
    1095             : 
    1096          92 :     pq_sendbyte(out, LOGICAL_REP_MSG_STREAM_COMMIT);
    1097             : 
    1098             :     Assert(TransactionIdIsValid(txn->xid));
    1099             : 
    1100             :     /* transaction ID */
    1101          92 :     pq_sendint32(out, txn->xid);
    1102             : 
    1103             :     /* send the flags field (unused for now) */
    1104          92 :     pq_sendbyte(out, flags);
    1105             : 
    1106             :     /* send fields */
    1107          92 :     pq_sendint64(out, commit_lsn);
    1108          92 :     pq_sendint64(out, txn->end_lsn);
    1109          92 :     pq_sendint64(out, txn->xact_time.commit_time);
    1110          92 : }
    1111             : 
    1112             : /*
    1113             :  * Read STREAM COMMIT from the output stream.
    1114             :  */
    1115             : TransactionId
    1116         122 : logicalrep_read_stream_commit(StringInfo in, LogicalRepCommitData *commit_data)
    1117             : {
    1118             :     TransactionId xid;
    1119             :     uint8       flags;
    1120             : 
    1121         122 :     xid = pq_getmsgint(in, 4);
    1122             : 
    1123             :     /* read flags (unused for now) */
    1124         122 :     flags = pq_getmsgbyte(in);
    1125             : 
    1126         122 :     if (flags != 0)
    1127           0 :         elog(ERROR, "unrecognized flags %u in commit message", flags);
    1128             : 
    1129             :     /* read fields */
    1130         122 :     commit_data->commit_lsn = pq_getmsgint64(in);
    1131         122 :     commit_data->end_lsn = pq_getmsgint64(in);
    1132         122 :     commit_data->committime = pq_getmsgint64(in);
    1133             : 
    1134         122 :     return xid;
    1135             : }
    1136             : 
    1137             : /*
    1138             :  * Write STREAM ABORT to the output stream. Note that xid and subxid will be
    1139             :  * same for the top-level transaction abort.
    1140             :  *
    1141             :  * If write_abort_info is true, send the abort_lsn and abort_time fields,
    1142             :  * otherwise don't.
    1143             :  */
    1144             : void
    1145          52 : logicalrep_write_stream_abort(StringInfo out, TransactionId xid,
    1146             :                               TransactionId subxid, XLogRecPtr abort_lsn,
    1147             :                               TimestampTz abort_time, bool write_abort_info)
    1148             : {
    1149          52 :     pq_sendbyte(out, LOGICAL_REP_MSG_STREAM_ABORT);
    1150             : 
    1151             :     Assert(TransactionIdIsValid(xid) && TransactionIdIsValid(subxid));
    1152             : 
    1153             :     /* transaction ID */
    1154          52 :     pq_sendint32(out, xid);
    1155          52 :     pq_sendint32(out, subxid);
    1156             : 
    1157          52 :     if (write_abort_info)
    1158             :     {
    1159          24 :         pq_sendint64(out, abort_lsn);
    1160          24 :         pq_sendint64(out, abort_time);
    1161             :     }
    1162          52 : }
    1163             : 
    1164             : /*
    1165             :  * Read STREAM ABORT from the output stream.
    1166             :  *
    1167             :  * If read_abort_info is true, read the abort_lsn and abort_time fields,
    1168             :  * otherwise don't.
    1169             :  */
    1170             : void
    1171          76 : logicalrep_read_stream_abort(StringInfo in,
    1172             :                              LogicalRepStreamAbortData *abort_data,
    1173             :                              bool read_abort_info)
    1174             : {
    1175             :     Assert(abort_data);
    1176             : 
    1177          76 :     abort_data->xid = pq_getmsgint(in, 4);
    1178          76 :     abort_data->subxid = pq_getmsgint(in, 4);
    1179             : 
    1180          76 :     if (read_abort_info)
    1181             :     {
    1182          48 :         abort_data->abort_lsn = pq_getmsgint64(in);
    1183          48 :         abort_data->abort_time = pq_getmsgint64(in);
    1184             :     }
    1185             :     else
    1186             :     {
    1187          28 :         abort_data->abort_lsn = InvalidXLogRecPtr;
    1188          28 :         abort_data->abort_time = 0;
    1189             :     }
    1190          76 : }
    1191             : 
    1192             : /*
    1193             :  * Get string representing LogicalRepMsgType.
    1194             :  */
    1195             : const char *
    1196         704 : logicalrep_message_type(LogicalRepMsgType action)
    1197             : {
    1198             :     static char err_unknown[20];
    1199             : 
    1200         704 :     switch (action)
    1201             :     {
    1202           2 :         case LOGICAL_REP_MSG_BEGIN:
    1203           2 :             return "BEGIN";
    1204           2 :         case LOGICAL_REP_MSG_COMMIT:
    1205           2 :             return "COMMIT";
    1206           0 :         case LOGICAL_REP_MSG_ORIGIN:
    1207           0 :             return "ORIGIN";
    1208          72 :         case LOGICAL_REP_MSG_INSERT:
    1209          72 :             return "INSERT";
    1210          30 :         case LOGICAL_REP_MSG_UPDATE:
    1211          30 :             return "UPDATE";
    1212          22 :         case LOGICAL_REP_MSG_DELETE:
    1213          22 :             return "DELETE";
    1214           0 :         case LOGICAL_REP_MSG_TRUNCATE:
    1215           0 :             return "TRUNCATE";
    1216           4 :         case LOGICAL_REP_MSG_RELATION:
    1217           4 :             return "RELATION";
    1218           0 :         case LOGICAL_REP_MSG_TYPE:
    1219           0 :             return "TYPE";
    1220           0 :         case LOGICAL_REP_MSG_MESSAGE:
    1221           0 :             return "MESSAGE";
    1222           2 :         case LOGICAL_REP_MSG_BEGIN_PREPARE:
    1223           2 :             return "BEGIN PREPARE";
    1224           4 :         case LOGICAL_REP_MSG_PREPARE:
    1225           4 :             return "PREPARE";
    1226           0 :         case LOGICAL_REP_MSG_COMMIT_PREPARED:
    1227           0 :             return "COMMIT PREPARED";
    1228           0 :         case LOGICAL_REP_MSG_ROLLBACK_PREPARED:
    1229           0 :             return "ROLLBACK PREPARED";
    1230          26 :         case LOGICAL_REP_MSG_STREAM_START:
    1231          26 :             return "STREAM START";
    1232         458 :         case LOGICAL_REP_MSG_STREAM_STOP:
    1233         458 :             return "STREAM STOP";
    1234          40 :         case LOGICAL_REP_MSG_STREAM_COMMIT:
    1235          40 :             return "STREAM COMMIT";
    1236          38 :         case LOGICAL_REP_MSG_STREAM_ABORT:
    1237          38 :             return "STREAM ABORT";
    1238           4 :         case LOGICAL_REP_MSG_STREAM_PREPARE:
    1239           4 :             return "STREAM PREPARE";
    1240             :     }
    1241             : 
    1242             :     /*
    1243             :      * This message provides context in the error raised when applying a
    1244             :      * logical message. So we can't throw an error here. Return an unknown
    1245             :      * indicator value so that the original error is still reported.
    1246             :      */
    1247           0 :     snprintf(err_unknown, sizeof(err_unknown), "??? (%d)", action);
    1248             : 
    1249           0 :     return err_unknown;
    1250             : }
    1251             : 
    1252             : /*
    1253             :  * Check if the column 'att' of a table should be published.
    1254             :  *
    1255             :  * 'columns' represents the publication column list (if any) for that table.
    1256             :  *
    1257             :  * 'include_gencols' flag indicates whether generated columns should be
    1258             :  * published when there is no column list. Typically, this will have the same
    1259             :  * value as the 'publish_generated_columns' publication parameter.
    1260             :  *
    1261             :  * Note that generated columns can be published only when present in a
    1262             :  * publication column list, or when include_gencols is true.
    1263             :  */
    1264             : bool
    1265     1468776 : logicalrep_should_publish_column(Form_pg_attribute att, Bitmapset *columns,
    1266             :                                  bool include_gencols)
    1267             : {
    1268     1468776 :     if (att->attisdropped)
    1269         102 :         return false;
    1270             : 
    1271             :     /* If a column list is provided, publish only the cols in that list. */
    1272     1468674 :     if (columns)
    1273        1888 :         return bms_is_member(att->attnum, columns);
    1274             : 
    1275             :     /* All non-generated columns are always published. */
    1276     1466786 :     return att->attgenerated ? include_gencols : true;
    1277             : }

Generated by: LCOV version 1.14