Thanks to visit codestin.com
Credit goes to doxygen.postgresql.org

PostgreSQL Source Code git master
libpq-be-fe-helpers.h
Go to the documentation of this file.
1/*-------------------------------------------------------------------------
2 *
3 * libpq-be-fe-helpers.h
4 * Helper functions for using libpq in extensions
5 *
6 * Code built directly into the backend is not allowed to link to libpq
7 * directly. Extension code is allowed to use libpq however. However, libpq
8 * used in extensions has to be careful not to block inside libpq, otherwise
9 * interrupts will not be processed, leading to issues like unresolvable
10 * deadlocks. Backend code also needs to take care to acquire/release an
11 * external fd for the connection, otherwise fd.c's accounting of fd's is
12 * broken.
13 *
14 * This file provides helper functions to make it easier to comply with these
15 * rules. It is a header only library as it needs to be linked into each
16 * extension using libpq, and it seems too small to be worth adding a
17 * dedicated static library for.
18 *
19 * TODO: For historical reasons the connections established here are not put
20 * into non-blocking mode. That can lead to blocking even when only the async
21 * libpq functions are used. This should be fixed.
22 *
23 * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
24 * Portions Copyright (c) 1994, Regents of the University of California
25 *
26 * src/include/libpq/libpq-be-fe-helpers.h
27 *
28 *-------------------------------------------------------------------------
29 */
30#ifndef LIBPQ_BE_FE_HELPERS_H
31#define LIBPQ_BE_FE_HELPERS_H
32
33#include "libpq/libpq-be-fe.h"
34#include "miscadmin.h"
35#include "storage/fd.h"
36#include "storage/latch.h"
37#include "utils/timestamp.h"
38#include "utils/wait_event.h"
39
40
41static inline void libpqsrv_connect_prepare(void);
42static inline void libpqsrv_connect_internal(PGconn *conn, uint32 wait_event_info);
43static inline PGresult *libpqsrv_get_result_last(PGconn *conn, uint32 wait_event_info);
44static inline PGresult *libpqsrv_get_result(PGconn *conn, uint32 wait_event_info);
45
46
47/*
48 * PQconnectdb() wrapper that reserves a file descriptor and processes
49 * interrupts during connection establishment.
50 *
51 * Throws an error if AcquireExternalFD() fails, but does not throw if
52 * connection establishment itself fails. Callers need to use PQstatus() to
53 * check if connection establishment succeeded.
54 */
55static inline PGconn *
56libpqsrv_connect(const char *conninfo, uint32 wait_event_info)
57{
58 PGconn *conn = NULL;
59
61
62 conn = PQconnectStart(conninfo);
63
64 libpqsrv_connect_internal(conn, wait_event_info);
65
66 return conn;
67}
68
69/*
70 * Like libpqsrv_connect(), except that this is a wrapper for
71 * PQconnectdbParams().
72 */
73static inline PGconn *
75 const char *const *values,
76 int expand_dbname,
77 uint32 wait_event_info)
78{
79 PGconn *conn = NULL;
80
82
83 conn = PQconnectStartParams(keywords, values, expand_dbname);
84
85 libpqsrv_connect_internal(conn, wait_event_info);
86
87 return conn;
88}
89
90/*
91 * PQfinish() wrapper that additionally releases the reserved file descriptor.
92 *
93 * It is allowed to call this with a NULL pgconn iff NULL was returned by
94 * libpqsrv_connect*.
95 */
96static inline void
98{
99 /*
100 * If no connection was established, we haven't reserved an FD for it (or
101 * already released it). This rule makes it easier to write PG_CATCH()
102 * handlers for this facility's users.
103 *
104 * See also libpqsrv_connect_internal().
105 */
106 if (conn == NULL)
107 return;
108
110 PQfinish(conn);
111}
112
113
114/* internal helper functions follow */
115
116
117/*
118 * Helper function for all connection establishment functions.
119 */
120static inline void
122{
123 /*
124 * We must obey fd.c's limit on non-virtual file descriptors. Assume that
125 * a PGconn represents one long-lived FD. (Doing this here also ensures
126 * that VFDs are closed if needed to make room.)
127 */
128 if (!AcquireExternalFD())
129 {
130#ifndef WIN32 /* can't write #if within ereport() macro */
132 (errcode(ERRCODE_SQLCLIENT_UNABLE_TO_ESTABLISH_SQLCONNECTION),
133 errmsg("could not establish connection"),
134 errdetail("There are too many open files on the local server."),
135 errhint("Raise the server's \"max_files_per_process\" and/or \"ulimit -n\" limits.")));
136#else
138 (errcode(ERRCODE_SQLCLIENT_UNABLE_TO_ESTABLISH_SQLCONNECTION),
139 errmsg("could not establish connection"),
140 errdetail("There are too many open files on the local server."),
141 errhint("Raise the server's \"max_files_per_process\" setting.")));
142#endif
143 }
144}
145
146/*
147 * Helper function for all connection establishment functions.
148 */
149static inline void
151{
152 /*
153 * With conn == NULL libpqsrv_disconnect() wouldn't release the FD. So do
154 * that here.
155 */
156 if (conn == NULL)
157 {
159 return;
160 }
161
162 /*
163 * Can't wait without a socket. Note that we don't want to close the libpq
164 * connection yet, so callers can emit a useful error.
165 */
167 return;
168
169 /*
170 * WaitLatchOrSocket() can conceivably fail, handle that case here instead
171 * of requiring all callers to do so.
172 */
173 PG_TRY();
174 {
176
177 /*
178 * Poll connection until we have OK or FAILED status.
179 *
180 * Per spec for PQconnectPoll, first wait till socket is write-ready.
181 */
182 status = PGRES_POLLING_WRITING;
183 while (status != PGRES_POLLING_OK && status != PGRES_POLLING_FAILED)
184 {
185 int io_flag;
186 int rc;
187
188 if (status == PGRES_POLLING_READING)
189 io_flag = WL_SOCKET_READABLE;
190#ifdef WIN32
191
192 /*
193 * Windows needs a different test while waiting for
194 * connection-made
195 */
196 else if (PQstatus(conn) == CONNECTION_STARTED)
197 io_flag = WL_SOCKET_CONNECTED;
198#endif
199 else
200 io_flag = WL_SOCKET_WRITEABLE;
201
204 PQsocket(conn),
205 0,
206 wait_event_info);
207
208 /* Interrupted? */
209 if (rc & WL_LATCH_SET)
210 {
213 }
214
215 /* If socket is ready, advance the libpq state machine */
216 if (rc & io_flag)
217 status = PQconnectPoll(conn);
218 }
219 }
220 PG_CATCH();
221 {
222 /*
223 * If an error is thrown here, the callers won't call
224 * libpqsrv_disconnect() with a conn, so release resources
225 * immediately.
226 */
228 PQfinish(conn);
229
230 PG_RE_THROW();
231 }
232 PG_END_TRY();
233}
234
235/*
236 * PQexec() wrapper that processes interrupts.
237 *
238 * Unless PQsetnonblocking(conn, 1) is in effect, this can't process
239 * interrupts while pushing the query text to the server. Consider that
240 * setting if query strings can be long relative to TCP buffer size.
241 *
242 * This has the preconditions of PQsendQuery(), not those of PQexec(). Most
243 * notably, PQexec() would silently discard any prior query results.
244 */
245static inline PGresult *
246libpqsrv_exec(PGconn *conn, const char *query, uint32 wait_event_info)
247{
248 if (!PQsendQuery(conn, query))
249 return NULL;
250 return libpqsrv_get_result_last(conn, wait_event_info);
251}
252
253/*
254 * PQexecParams() wrapper that processes interrupts.
255 *
256 * See notes at libpqsrv_exec().
257 */
258static inline PGresult *
260 const char *command,
261 int nParams,
262 const Oid *paramTypes,
263 const char *const *paramValues,
264 const int *paramLengths,
265 const int *paramFormats,
266 int resultFormat,
267 uint32 wait_event_info)
268{
269 if (!PQsendQueryParams(conn, command, nParams, paramTypes, paramValues,
270 paramLengths, paramFormats, resultFormat))
271 return NULL;
272 return libpqsrv_get_result_last(conn, wait_event_info);
273}
274
275/*
276 * Like PQexec(), loop over PQgetResult() until it returns NULL or another
277 * terminal state. Return the last non-NULL result or the terminal state.
278 */
279static inline PGresult *
281{
282 PGresult *lastResult = NULL;
283
284 for (;;)
285 {
286 /* Wait for, and collect, the next PGresult. */
287 PGresult *result;
288
289 result = libpqsrv_get_result(conn, wait_event_info);
290 if (result == NULL)
291 break; /* query is complete, or failure */
292
293 /*
294 * Emulate PQexec()'s behavior of returning the last result when there
295 * are many.
296 */
297 PQclear(lastResult);
298 lastResult = result;
299
300 if (PQresultStatus(lastResult) == PGRES_COPY_IN ||
301 PQresultStatus(lastResult) == PGRES_COPY_OUT ||
302 PQresultStatus(lastResult) == PGRES_COPY_BOTH ||
304 break;
305 }
306 return lastResult;
307}
308
309/*
310 * Perform the equivalent of PQgetResult(), but watch for interrupts.
311 */
312static inline PGresult *
314{
315 /*
316 * Collect data until PQgetResult is ready to get the result without
317 * blocking.
318 */
319 while (PQisBusy(conn))
320 {
321 int rc;
322
326 PQsocket(conn),
327 0,
328 wait_event_info);
329
330 /* Interrupted? */
331 if (rc & WL_LATCH_SET)
332 {
335 }
336
337 /* Consume whatever data is available from the socket */
338 if (PQconsumeInput(conn) == 0)
339 {
340 /* trouble; expect PQgetResult() to return NULL */
341 break;
342 }
343 }
344
345 /* Now we can collect and return the next PGresult */
346 return PQgetResult(conn);
347}
348
349/*
350 * Submit a cancel request to the given connection, waiting only until
351 * the given time.
352 *
353 * We sleep interruptibly until we receive confirmation that the cancel
354 * request has been accepted, and if it is, return NULL; if the cancel
355 * request fails, return an error message string (which is not to be
356 * freed).
357 *
358 * For other problems (to wit: OOM when strdup'ing an error message from
359 * libpq), this function can ereport(ERROR).
360 *
361 * Note: this function leaks a string's worth of memory when reporting
362 * libpq errors. Make sure to call it in a transient memory context.
363 */
364static inline const char *
366{
367 PGcancelConn *cancel_conn;
368 const char *error = NULL;
369
370 cancel_conn = PQcancelCreate(conn);
371 if (cancel_conn == NULL)
372 return "out of memory";
373
374 /* In what follows, do not leak any PGcancelConn on any errors. */
375
376 PG_TRY();
377 {
378 if (!PQcancelStart(cancel_conn))
379 {
380 error = pchomp(PQcancelErrorMessage(cancel_conn));
381 goto exit;
382 }
383
384 for (;;)
385 {
388 long cur_timeout;
389 int waitEvents = WL_LATCH_SET | WL_TIMEOUT | WL_EXIT_ON_PM_DEATH;
390
391 pollres = PQcancelPoll(cancel_conn);
392 if (pollres == PGRES_POLLING_OK)
393 break; /* success! */
394
395 /* If timeout has expired, give up, else get sleep time. */
397 cur_timeout = TimestampDifferenceMilliseconds(now, endtime);
398 if (cur_timeout <= 0)
399 {
400 error = "cancel request timed out";
401 break;
402 }
403
404 switch (pollres)
405 {
407 waitEvents |= WL_SOCKET_READABLE;
408 break;
410 waitEvents |= WL_SOCKET_WRITEABLE;
411 break;
412 default:
413 error = pchomp(PQcancelErrorMessage(cancel_conn));
414 goto exit;
415 }
416
417 /* Sleep until there's something to do */
418 WaitLatchOrSocket(MyLatch, waitEvents, PQcancelSocket(cancel_conn),
419 cur_timeout, PG_WAIT_CLIENT);
420
422
424 }
425exit: ;
426 }
427 PG_FINALLY();
428 {
429 PQcancelFinish(cancel_conn);
430 }
431 PG_END_TRY();
432
433 return error;
434}
435
436/*
437 * libpqsrv_notice_receiver
438 *
439 * Custom notice receiver for libpq connections.
440 *
441 * This function is intended to be set via PQsetNoticeReceiver() so that
442 * NOTICE, WARNING, and similar messages from the connection are reported via
443 * ereport(), instead of being printed to stderr.
444 *
445 * Because this will be called from libpq with a "real" (not wrapped)
446 * PGresult, we need to temporarily ignore libpq-be-fe.h's wrapper macros
447 * for PGresult and also PQresultErrorMessage, and put back the wrappers
448 * afterwards. That's not pretty, but there seems no better alternative.
449 */
450#undef PGresult
451#undef PQresultErrorMessage
452
453static inline void
455{
456 const char *message;
457 int len;
458 const char *prefix = (const char *) arg;
459
460 /*
461 * Trim the trailing newline from the message text returned from
462 * PQresultErrorMessage(), as it always includes one, to produce cleaner
463 * log output.
464 */
465 message = PQresultErrorMessage(res);
466 len = strlen(message);
467 if (len > 0 && message[len - 1] == '\n')
468 len--;
469
470 ereport(LOG,
471 errmsg_internal("%s: %.*s", prefix, len, message));
472}
473
474#define PGresult libpqsrv_PGresult
475#define PQresultErrorMessage libpqsrv_PQresultErrorMessage
476
477#endif /* LIBPQ_BE_FE_HELPERS_H */
long TimestampDifferenceMilliseconds(TimestampTz start_time, TimestampTz stop_time)
Definition: timestamp.c:1757
TimestampTz GetCurrentTimestamp(void)
Definition: timestamp.c:1645
Datum now(PG_FUNCTION_ARGS)
Definition: timestamp.c:1609
static Datum values[MAXATTR]
Definition: bootstrap.c:153
uint32_t uint32
Definition: c.h:539
int64 TimestampTz
Definition: timestamp.h:39
int errmsg_internal(const char *fmt,...)
Definition: elog.c:1161
int errdetail(const char *fmt,...)
Definition: elog.c:1207
int errhint(const char *fmt,...)
Definition: elog.c:1321
int errcode(int sqlerrcode)
Definition: elog.c:854
int errmsg(const char *fmt,...)
Definition: elog.c:1071
#define LOG
Definition: elog.h:31
#define PG_RE_THROW()
Definition: elog.h:405
#define PG_TRY(...)
Definition: elog.h:372
#define PG_END_TRY(...)
Definition: elog.h:397
#define ERROR
Definition: elog.h:39
#define PG_CATCH(...)
Definition: elog.h:382
#define PG_FINALLY(...)
Definition: elog.h:389
#define ereport(elevel,...)
Definition: elog.h:150
void ReleaseExternalFD(void)
Definition: fd.c:1238
bool AcquireExternalFD(void)
Definition: fd.c:1185
PGcancelConn * PQcancelCreate(PGconn *conn)
Definition: fe-cancel.c:68
PostgresPollingStatusType PQcancelPoll(PGcancelConn *cancelConn)
Definition: fe-cancel.c:226
void PQcancelFinish(PGcancelConn *cancelConn)
Definition: fe-cancel.c:353
int PQcancelSocket(const PGcancelConn *cancelConn)
Definition: fe-cancel.c:313
char * PQcancelErrorMessage(const PGcancelConn *cancelConn)
Definition: fe-cancel.c:325
int PQcancelStart(PGcancelConn *cancelConn)
Definition: fe-cancel.c:204
PostgresPollingStatusType PQconnectPoll(PGconn *conn)
Definition: fe-connect.c:2911
ConnStatusType PQstatus(const PGconn *conn)
Definition: fe-connect.c:7616
PGconn * PQconnectStart(const char *conninfo)
Definition: fe-connect.c:951
PGconn * PQconnectStartParams(const char *const *keywords, const char *const *values, int expand_dbname)
Definition: fe-connect.c:870
void PQfinish(PGconn *conn)
Definition: fe-connect.c:5305
int PQsocket(const PGconn *conn)
Definition: fe-connect.c:7705
int PQsendQueryParams(PGconn *conn, const char *command, int nParams, const Oid *paramTypes, const char *const *paramValues, const int *paramLengths, const int *paramFormats, int resultFormat)
Definition: fe-exec.c:1503
int PQconsumeInput(PGconn *conn)
Definition: fe-exec.c:1995
int PQsendQuery(PGconn *conn, const char *query)
Definition: fe-exec.c:1427
int PQisBusy(PGconn *conn)
Definition: fe-exec.c:2042
struct Latch * MyLatch
Definition: globals.c:63
static const JsonPathKeyword keywords[]
int WaitLatchOrSocket(Latch *latch, int wakeEvents, pgsocket sock, long timeout, uint32 wait_event_info)
Definition: latch.c:223
void ResetLatch(Latch *latch)
Definition: latch.c:374
static const char * libpqsrv_cancel(PGconn *conn, TimestampTz endtime)
static PGresult * libpqsrv_exec(PGconn *conn, const char *query, uint32 wait_event_info)
static PGconn * libpqsrv_connect_params(const char *const *keywords, const char *const *values, int expand_dbname, uint32 wait_event_info)
static PGconn * libpqsrv_connect(const char *conninfo, uint32 wait_event_info)
static void libpqsrv_connect_prepare(void)
static PGresult * libpqsrv_get_result(PGconn *conn, uint32 wait_event_info)
static void libpqsrv_notice_receiver(void *arg, const PGresult *res)
static void libpqsrv_connect_internal(PGconn *conn, uint32 wait_event_info)
static void libpqsrv_disconnect(PGconn *conn)
#define PQresultErrorMessage
static PGresult * libpqsrv_exec_params(PGconn *conn, const char *command, int nParams, const Oid *paramTypes, const char *const *paramValues, const int *paramLengths, const int *paramFormats, int resultFormat, uint32 wait_event_info)
static PGresult * libpqsrv_get_result_last(PGconn *conn, uint32 wait_event_info)
#define PQgetResult
Definition: libpq-be-fe.h:246
#define PQclear
Definition: libpq-be-fe.h:245
#define PQresultStatus
Definition: libpq-be-fe.h:247
@ CONNECTION_STARTED
Definition: libpq-fe.h:92
@ CONNECTION_BAD
Definition: libpq-fe.h:85
@ PGRES_COPY_IN
Definition: libpq-fe.h:132
@ PGRES_COPY_BOTH
Definition: libpq-fe.h:137
@ PGRES_COPY_OUT
Definition: libpq-fe.h:131
PostgresPollingStatusType
Definition: libpq-fe.h:114
@ PGRES_POLLING_OK
Definition: libpq-fe.h:118
@ PGRES_POLLING_READING
Definition: libpq-fe.h:116
@ PGRES_POLLING_WRITING
Definition: libpq-fe.h:117
@ PGRES_POLLING_FAILED
Definition: libpq-fe.h:115
char * pchomp(const char *in)
Definition: mcxt.c:1787
#define CHECK_FOR_INTERRUPTS()
Definition: miscadmin.h:122
void * arg
const void size_t len
unsigned int Oid
Definition: postgres_ext.h:32
static void error(void)
Definition: sql-dyntest.c:147
PGconn * conn
Definition: streamutil.c:52
#define PG_WAIT_CLIENT
Definition: wait_classes.h:22
#define WL_SOCKET_READABLE
Definition: waiteventset.h:35
#define WL_TIMEOUT
Definition: waiteventset.h:37
#define WL_EXIT_ON_PM_DEATH
Definition: waiteventset.h:39
#define WL_LATCH_SET
Definition: waiteventset.h:34
#define WL_SOCKET_CONNECTED
Definition: waiteventset.h:44
#define WL_SOCKET_WRITEABLE
Definition: waiteventset.h:36