PostgreSQL Source Code git master
logicalfuncs.c File Reference
#include "postgres.h"
#include <unistd.h>
#include "access/xlogrecovery.h"
#include "access/xlogutils.h"
#include "catalog/pg_type.h"
#include "fmgr.h"
#include "funcapi.h"
#include "mb/pg_wchar.h"
#include "miscadmin.h"
#include "nodes/makefuncs.h"
#include "replication/decode.h"
#include "replication/logical.h"
#include "replication/message.h"
#include "utils/array.h"
#include "utils/builtins.h"
#include "utils/inval.h"
#include "utils/memutils.h"
#include "utils/pg_lsn.h"
#include "utils/regproc.h"
#include "utils/resowner.h"
Include dependency graph for logicalfuncs.c:

Go to the source code of this file.

Data Structures

struct  DecodingOutputState
 

Typedefs

typedef struct DecodingOutputState DecodingOutputState
 

Functions

static void LogicalOutputPrepareWrite (LogicalDecodingContext *ctx, XLogRecPtr lsn, TransactionId xid, bool last_write)
 
static void LogicalOutputWrite (LogicalDecodingContext *ctx, XLogRecPtr lsn, TransactionId xid, bool last_write)
 
static Datum pg_logical_slot_get_changes_guts (FunctionCallInfo fcinfo, bool confirm, bool binary)
 
Datum pg_logical_slot_get_changes (PG_FUNCTION_ARGS)
 
Datum pg_logical_slot_peek_changes (PG_FUNCTION_ARGS)
 
Datum pg_logical_slot_get_binary_changes (PG_FUNCTION_ARGS)
 
Datum pg_logical_slot_peek_binary_changes (PG_FUNCTION_ARGS)
 
Datum pg_logical_emit_message_bytea (PG_FUNCTION_ARGS)
 
Datum pg_logical_emit_message_text (PG_FUNCTION_ARGS)
 

Typedef Documentation

◆ DecodingOutputState

Function Documentation

◆ LogicalOutputPrepareWrite()

static void LogicalOutputPrepareWrite ( LogicalDecodingContext ctx,
XLogRecPtr  lsn,
TransactionId  xid,
bool  last_write 
)
static

Definition at line 52 of file logicalfuncs.c.

54{
55 resetStringInfo(ctx->out);
56}
void resetStringInfo(StringInfo str)
Definition: stringinfo.c:126
StringInfo out
Definition: logical.h:71

References LogicalDecodingContext::out, and resetStringInfo().

Referenced by pg_logical_slot_get_changes_guts().

◆ LogicalOutputWrite()

static void LogicalOutputWrite ( LogicalDecodingContext ctx,
XLogRecPtr  lsn,
TransactionId  xid,
bool  last_write 
)
static

Definition at line 62 of file logicalfuncs.c.

64{
65 Datum values[3];
66 bool nulls[3];
68
69 /* SQL Datums can only be of a limited length... */
70 if (ctx->out->len > MaxAllocSize - VARHDRSZ)
71 elog(ERROR, "too much output for sql interface");
72
74
75 memset(nulls, 0, sizeof(nulls));
76 values[0] = LSNGetDatum(lsn);
78
79 /*
80 * Assert ctx->out is in database encoding when we're writing textual
81 * output.
82 */
83 if (!p->binary_output)
85 ctx->out->data, ctx->out->len,
86 false));
87
88 /* ick, but cstring_to_text_with_len works for bytea perfectly fine */
90
92 p->returned_rows++;
93}
static Datum values[MAXATTR]
Definition: bootstrap.c:151
#define VARHDRSZ
Definition: c.h:649
#define Assert(condition)
Definition: c.h:815
#define ERROR
Definition: elog.h:39
#define elog(elevel,...)
Definition: elog.h:225
#define MaxAllocSize
Definition: fe_memutils.h:22
int GetDatabaseEncoding(void)
Definition: mbutils.c:1261
bool pg_verify_mbstr(int encoding, const char *mbstr, int len, bool noError)
Definition: mbutils.c:1566
static Datum LSNGetDatum(XLogRecPtr X)
Definition: pg_lsn.h:28
static Datum PointerGetDatum(const void *X)
Definition: postgres.h:327
static Datum TransactionIdGetDatum(TransactionId X)
Definition: postgres.h:277
uintptr_t Datum
Definition: postgres.h:69
Tuplestorestate * tupstore
Definition: logicalfuncs.c:42
void * output_writer_private
Definition: logical.h:81
void tuplestore_putvalues(Tuplestorestate *state, TupleDesc tdesc, const Datum *values, const bool *isnull)
Definition: tuplestore.c:784
text * cstring_to_text_with_len(const char *s, int len)
Definition: varlena.c:196

References Assert, DecodingOutputState::binary_output, cstring_to_text_with_len(), StringInfoData::data, elog, ERROR, GetDatabaseEncoding(), StringInfoData::len, LSNGetDatum(), MaxAllocSize, LogicalDecodingContext::out, LogicalDecodingContext::output_writer_private, pg_verify_mbstr(), PointerGetDatum(), DecodingOutputState::returned_rows, TransactionIdGetDatum(), DecodingOutputState::tupdesc, tuplestore_putvalues(), DecodingOutputState::tupstore, values, and VARHDRSZ.

Referenced by pg_logical_slot_get_changes_guts().

◆ pg_logical_emit_message_bytea()

Datum pg_logical_emit_message_bytea ( PG_FUNCTION_ARGS  )

Definition at line 368 of file logicalfuncs.c.

369{
370 bool transactional = PG_GETARG_BOOL(0);
371 char *prefix = text_to_cstring(PG_GETARG_TEXT_PP(1));
373 bool flush = PG_GETARG_BOOL(3);
374 XLogRecPtr lsn;
375
377 transactional, flush);
378 PG_RETURN_LSN(lsn);
379}
#define PG_GETARG_BYTEA_PP(n)
Definition: fmgr.h:308
#define PG_GETARG_TEXT_PP(n)
Definition: fmgr.h:309
#define PG_GETARG_BOOL(n)
Definition: fmgr.h:274
XLogRecPtr LogLogicalMessage(const char *prefix, const char *message, size_t size, bool transactional, bool flush)
Definition: message.c:43
const void * data
#define PG_RETURN_LSN(x)
Definition: pg_lsn.h:34
Definition: c.h:644
#define VARDATA_ANY(PTR)
Definition: varatt.h:324
#define VARSIZE_ANY_EXHDR(PTR)
Definition: varatt.h:317
char * text_to_cstring(const text *t)
Definition: varlena.c:217
uint64 XLogRecPtr
Definition: xlogdefs.h:21

References data, LogLogicalMessage(), PG_GETARG_BOOL, PG_GETARG_BYTEA_PP, PG_GETARG_TEXT_PP, PG_RETURN_LSN, text_to_cstring(), VARDATA_ANY, and VARSIZE_ANY_EXHDR.

Referenced by pg_logical_emit_message_text().

◆ pg_logical_emit_message_text()

Datum pg_logical_emit_message_text ( PG_FUNCTION_ARGS  )

Definition at line 382 of file logicalfuncs.c.

383{
384 /* bytea and text are compatible */
385 return pg_logical_emit_message_bytea(fcinfo);
386}
Datum pg_logical_emit_message_bytea(PG_FUNCTION_ARGS)
Definition: logicalfuncs.c:368

References pg_logical_emit_message_bytea().

◆ pg_logical_slot_get_binary_changes()

Datum pg_logical_slot_get_binary_changes ( PG_FUNCTION_ARGS  )

Definition at line 349 of file logicalfuncs.c.

350{
351 return pg_logical_slot_get_changes_guts(fcinfo, true, true);
352}
static Datum pg_logical_slot_get_changes_guts(FunctionCallInfo fcinfo, bool confirm, bool binary)
Definition: logicalfuncs.c:99

References pg_logical_slot_get_changes_guts().

◆ pg_logical_slot_get_changes()

Datum pg_logical_slot_get_changes ( PG_FUNCTION_ARGS  )

Definition at line 331 of file logicalfuncs.c.

332{
333 return pg_logical_slot_get_changes_guts(fcinfo, true, false);
334}

References pg_logical_slot_get_changes_guts().

◆ pg_logical_slot_get_changes_guts()

static Datum pg_logical_slot_get_changes_guts ( FunctionCallInfo  fcinfo,
bool  confirm,
bool  binary 
)
static

Definition at line 99 of file logicalfuncs.c.

100{
101 Name name;
102 XLogRecPtr upto_lsn;
103 int32 upto_nchanges;
104 ReturnSetInfo *rsinfo = (ReturnSetInfo *) fcinfo->resultinfo;
105 MemoryContext per_query_ctx;
106 MemoryContext oldcontext;
107 XLogRecPtr end_of_wal;
108 XLogRecPtr wait_for_wal_lsn;
110 ResourceOwner old_resowner = CurrentResourceOwner;
111 ArrayType *arr;
112 Size ndim;
113 List *options = NIL;
115
117
119
120 if (PG_ARGISNULL(0))
122 (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
123 errmsg("slot name must not be null")));
124 name = PG_GETARG_NAME(0);
125
126 if (PG_ARGISNULL(1))
127 upto_lsn = InvalidXLogRecPtr;
128 else
129 upto_lsn = PG_GETARG_LSN(1);
130
131 if (PG_ARGISNULL(2))
132 upto_nchanges = InvalidXLogRecPtr;
133 else
134 upto_nchanges = PG_GETARG_INT32(2);
135
136 if (PG_ARGISNULL(3))
138 (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
139 errmsg("options array must not be null")));
140 arr = PG_GETARG_ARRAYTYPE_P(3);
141
142 /* state to write output to */
143 p = palloc0(sizeof(DecodingOutputState));
144
145 p->binary_output = binary;
146
147 per_query_ctx = rsinfo->econtext->ecxt_per_query_memory;
148 oldcontext = MemoryContextSwitchTo(per_query_ctx);
149
150 /* Deconstruct options array */
151 ndim = ARR_NDIM(arr);
152 if (ndim > 1)
153 {
155 (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
156 errmsg("array must be one-dimensional")));
157 }
158 else if (array_contains_nulls(arr))
159 {
161 (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
162 errmsg("array must not contain nulls")));
163 }
164 else if (ndim == 1)
165 {
166 int nelems;
167 Datum *datum_opts;
168 int i;
169
170 Assert(ARR_ELEMTYPE(arr) == TEXTOID);
171
172 deconstruct_array_builtin(arr, TEXTOID, &datum_opts, NULL, &nelems);
173
174 if (nelems % 2 != 0)
176 (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
177 errmsg("array must have even number of elements")));
178
179 for (i = 0; i < nelems; i += 2)
180 {
181 char *optname = TextDatumGetCString(datum_opts[i]);
182 char *opt = TextDatumGetCString(datum_opts[i + 1]);
183
184 options = lappend(options, makeDefElem(optname, (Node *) makeString(opt), -1));
185 }
186 }
187
188 InitMaterializedSRF(fcinfo, 0);
189 p->tupstore = rsinfo->setResult;
190 p->tupdesc = rsinfo->setDesc;
191
192 /*
193 * Compute the current end-of-wal.
194 */
195 if (!RecoveryInProgress())
196 end_of_wal = GetFlushRecPtr(NULL);
197 else
198 end_of_wal = GetXLogReplayRecPtr(NULL);
199
201
202 PG_TRY();
203 {
204 /* restart at slot's confirmed_flush */
206 options,
207 false,
208 XL_ROUTINE(.page_read = read_local_xlog_page,
209 .segment_open = wal_segment_open,
210 .segment_close = wal_segment_close),
212 LogicalOutputWrite, NULL);
213
214 MemoryContextSwitchTo(oldcontext);
215
216 /*
217 * Check whether the output plugin writes textual output if that's
218 * what we need.
219 */
220 if (!binary &&
221 ctx->options.output_type !=OUTPUT_PLUGIN_TEXTUAL_OUTPUT)
223 (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
224 errmsg("logical decoding output plugin \"%s\" produces binary output, but function \"%s\" expects textual data",
226 format_procedure(fcinfo->flinfo->fn_oid))));
227
228 /*
229 * Wait for specified streaming replication standby servers (if any)
230 * to confirm receipt of WAL up to wait_for_wal_lsn.
231 */
232 if (XLogRecPtrIsInvalid(upto_lsn))
233 wait_for_wal_lsn = end_of_wal;
234 else
235 wait_for_wal_lsn = Min(upto_lsn, end_of_wal);
236
237 WaitForStandbyConfirmation(wait_for_wal_lsn);
238
239 ctx->output_writer_private = p;
240
241 /*
242 * Decoding of WAL must start at restart_lsn so that the entirety of
243 * xacts that committed after the slot's confirmed_flush can be
244 * accumulated into reorder buffers.
245 */
247
248 /* invalidate non-timetravel entries */
250
251 /* Decode until we run out of records */
252 while (ctx->reader->EndRecPtr < end_of_wal)
253 {
254 XLogRecord *record;
255 char *errm = NULL;
256
257 record = XLogReadRecord(ctx->reader, &errm);
258 if (errm)
259 elog(ERROR, "could not find record for logical decoding: %s", errm);
260
261 /*
262 * The {begin_txn,change,commit_txn}_wrapper callbacks above will
263 * store the description into our tuplestore.
264 */
265 if (record != NULL)
266 LogicalDecodingProcessRecord(ctx, ctx->reader);
267
268 /* check limits */
269 if (upto_lsn != InvalidXLogRecPtr &&
270 upto_lsn <= ctx->reader->EndRecPtr)
271 break;
272 if (upto_nchanges != 0 &&
273 upto_nchanges <= p->returned_rows)
274 break;
276 }
277
278 /*
279 * Logical decoding could have clobbered CurrentResourceOwner during
280 * transaction management, so restore the executor's value. (This is
281 * a kluge, but it's not worth cleaning up right now.)
282 */
283 CurrentResourceOwner = old_resowner;
284
285 /*
286 * Next time, start where we left off. (Hunting things, the family
287 * business..)
288 */
289 if (ctx->reader->EndRecPtr != InvalidXLogRecPtr && confirm)
290 {
291 LogicalConfirmReceivedLocation(ctx->reader->EndRecPtr);
292
293 /*
294 * If only the confirmed_flush_lsn has changed the slot won't get
295 * marked as dirty by the above. Callers on the walsender
296 * interface are expected to keep track of their own progress and
297 * don't need it written out. But SQL-interface users cannot
298 * specify their own start positions and it's harder for them to
299 * keep track of their progress, so we should make more of an
300 * effort to save it for them.
301 *
302 * Dirty the slot so it's written out at the next checkpoint.
303 * We'll still lose its position on crash, as documented, but it's
304 * better than always losing the position even on clean restart.
305 */
307 }
308
309 /* free context, call shutdown callback */
311
314 }
315 PG_CATCH();
316 {
317 /* clear all timetravel entries */
319
320 PG_RE_THROW();
321 }
322 PG_END_TRY();
323
324 return (Datum) 0;
325}
#define ARR_NDIM(a)
Definition: array.h:290
#define PG_GETARG_ARRAYTYPE_P(n)
Definition: array.h:263
#define ARR_ELEMTYPE(a)
Definition: array.h:292
bool array_contains_nulls(ArrayType *array)
Definition: arrayfuncs.c:3767
void deconstruct_array_builtin(ArrayType *array, Oid elmtype, Datum **elemsp, bool **nullsp, int *nelemsp)
Definition: arrayfuncs.c:3697
#define TextDatumGetCString(d)
Definition: builtins.h:98
#define NameStr(name)
Definition: c.h:703
#define Min(x, y)
Definition: c.h:961
int32_t int32
Definition: c.h:484
size_t Size
Definition: c.h:562
void LogicalDecodingProcessRecord(LogicalDecodingContext *ctx, XLogReaderState *record)
Definition: decode.c:88
int errcode(int sqlerrcode)
Definition: elog.c:853
int errmsg(const char *fmt,...)
Definition: elog.c:1070
#define PG_RE_THROW()
Definition: elog.h:412
#define PG_TRY(...)
Definition: elog.h:371
#define PG_END_TRY(...)
Definition: elog.h:396
#define PG_CATCH(...)
Definition: elog.h:381
#define ereport(elevel,...)
Definition: elog.h:149
#define PG_ARGISNULL(n)
Definition: fmgr.h:209
#define PG_GETARG_NAME(n)
Definition: fmgr.h:278
#define PG_GETARG_INT32(n)
Definition: fmgr.h:269
void InitMaterializedSRF(FunctionCallInfo fcinfo, bits32 flags)
Definition: funcapi.c:76
void InvalidateSystemCaches(void)
Definition: inval.c:849
int i
Definition: isn.c:72
List * lappend(List *list, void *datum)
Definition: list.c:339
void LogicalConfirmReceivedLocation(XLogRecPtr lsn)
Definition: logical.c:1837
void FreeDecodingContext(LogicalDecodingContext *ctx)
Definition: logical.c:694
LogicalDecodingContext * CreateDecodingContext(XLogRecPtr start_lsn, List *output_plugin_options, bool fast_forward, XLogReaderRoutine *xl_routine, LogicalOutputPluginWriterPrepareWrite prepare_write, LogicalOutputPluginWriterWrite do_write, LogicalOutputPluginWriterUpdateProgress update_progress)
Definition: logical.c:496
void CheckLogicalDecodingRequirements(void)
Definition: logical.c:109
static void LogicalOutputPrepareWrite(LogicalDecodingContext *ctx, XLogRecPtr lsn, TransactionId xid, bool last_write)
Definition: logicalfuncs.c:52
static void LogicalOutputWrite(LogicalDecodingContext *ctx, XLogRecPtr lsn, TransactionId xid, bool last_write)
Definition: logicalfuncs.c:62
DefElem * makeDefElem(char *name, Node *arg, int location)
Definition: makefuncs.c:590
void * palloc0(Size size)
Definition: mcxt.c:1347
#define CHECK_FOR_INTERRUPTS()
Definition: miscadmin.h:122
@ OUTPUT_PLUGIN_TEXTUAL_OUTPUT
Definition: output_plugin.h:20
static MemoryContext MemoryContextSwitchTo(MemoryContext context)
Definition: palloc.h:124
#define NIL
Definition: pg_list.h:68
#define PG_GETARG_LSN(n)
Definition: pg_lsn.h:33
char * format_procedure(Oid procedure_oid)
Definition: regproc.c:299
ResourceOwner CurrentResourceOwner
Definition: resowner.c:165
void ReplicationSlotMarkDirty(void)
Definition: slot.c:1038
void ReplicationSlotAcquire(const char *name, bool nowait)
Definition: slot.c:540
ReplicationSlot * MyReplicationSlot
Definition: slot.c:138
void CheckSlotPermissions(void)
Definition: slot.c:1412
void ReplicationSlotRelease(void)
Definition: slot.c:652
void WaitForStandbyConfirmation(XLogRecPtr wait_for_lsn)
Definition: slot.c:2759
MemoryContext ecxt_per_query_memory
Definition: execnodes.h:274
Oid fn_oid
Definition: fmgr.h:59
fmNodePtr resultinfo
Definition: fmgr.h:89
FmgrInfo * flinfo
Definition: fmgr.h:87
Definition: pg_list.h:54
Definition: nodes.h:129
XLogRecPtr restart_lsn
Definition: slot.h:96
ReplicationSlotPersistentData data
Definition: slot.h:181
ExprContext * econtext
Definition: execnodes.h:350
TupleDesc setDesc
Definition: execnodes.h:358
Tuplestorestate * setResult
Definition: execnodes.h:357
Definition: c.h:698
String * makeString(char *str)
Definition: value.c:63
const char * name
bool RecoveryInProgress(void)
Definition: xlog.c:6334
XLogRecPtr GetFlushRecPtr(TimeLineID *insertTLI)
Definition: xlog.c:6499
#define XLogRecPtrIsInvalid(r)
Definition: xlogdefs.h:29
#define InvalidXLogRecPtr
Definition: xlogdefs.h:28
XLogRecord * XLogReadRecord(XLogReaderState *state, char **errormsg)
Definition: xlogreader.c:389
void XLogBeginRead(XLogReaderState *state, XLogRecPtr RecPtr)
Definition: xlogreader.c:231
#define XL_ROUTINE(...)
Definition: xlogreader.h:117
XLogRecPtr GetXLogReplayRecPtr(TimeLineID *replayTLI)
void wal_segment_close(XLogReaderState *state)
Definition: xlogutils.c:842
void wal_segment_open(XLogReaderState *state, XLogSegNo nextSegNo, TimeLineID *tli_p)
Definition: xlogutils.c:817
int read_local_xlog_page(XLogReaderState *state, XLogRecPtr targetPagePtr, int reqLen, XLogRecPtr targetRecPtr, char *cur_page)
Definition: xlogutils.c:861

References ARR_ELEMTYPE, ARR_NDIM, array_contains_nulls(), Assert, CHECK_FOR_INTERRUPTS, CheckLogicalDecodingRequirements(), CheckSlotPermissions(), CreateDecodingContext(), CurrentResourceOwner, ReplicationSlot::data, deconstruct_array_builtin(), ReturnSetInfo::econtext, ExprContext::ecxt_per_query_memory, elog, ereport, errcode(), errmsg(), ERROR, FunctionCallInfoBaseData::flinfo, FmgrInfo::fn_oid, format_procedure(), FreeDecodingContext(), GetFlushRecPtr(), GetXLogReplayRecPtr(), i, InitMaterializedSRF(), InvalidateSystemCaches(), InvalidXLogRecPtr, lappend(), LogicalConfirmReceivedLocation(), LogicalDecodingProcessRecord(), LogicalOutputPrepareWrite(), LogicalOutputWrite(), makeDefElem(), makeString(), MemoryContextSwitchTo(), Min, MyReplicationSlot, name, NameStr, NIL, OUTPUT_PLUGIN_TEXTUAL_OUTPUT, palloc0(), PG_ARGISNULL, PG_CATCH, PG_END_TRY, PG_GETARG_ARRAYTYPE_P, PG_GETARG_INT32, PG_GETARG_LSN, PG_GETARG_NAME, PG_RE_THROW, PG_TRY, ReplicationSlotPersistentData::plugin, read_local_xlog_page(), RecoveryInProgress(), ReplicationSlotAcquire(), ReplicationSlotMarkDirty(), ReplicationSlotRelease(), ReplicationSlotPersistentData::restart_lsn, FunctionCallInfoBaseData::resultinfo, ReturnSetInfo::setDesc, ReturnSetInfo::setResult, TextDatumGetCString, WaitForStandbyConfirmation(), wal_segment_close(), wal_segment_open(), XL_ROUTINE, XLogBeginRead(), XLogReadRecord(), and XLogRecPtrIsInvalid.

Referenced by pg_logical_slot_get_binary_changes(), pg_logical_slot_get_changes(), pg_logical_slot_peek_binary_changes(), and pg_logical_slot_peek_changes().

◆ pg_logical_slot_peek_binary_changes()

Datum pg_logical_slot_peek_binary_changes ( PG_FUNCTION_ARGS  )

Definition at line 358 of file logicalfuncs.c.

359{
360 return pg_logical_slot_get_changes_guts(fcinfo, false, true);
361}

References pg_logical_slot_get_changes_guts().

◆ pg_logical_slot_peek_changes()

Datum pg_logical_slot_peek_changes ( PG_FUNCTION_ARGS  )

Definition at line 340 of file logicalfuncs.c.

341{
342 return pg_logical_slot_get_changes_guts(fcinfo, false, false);
343}

References pg_logical_slot_get_changes_guts().