mirror of
https://github.com/postgres/postgres.git
synced 2025-11-22 12:22:45 +03:00
Redesign archive modules
A new callback named startup_cb, called shortly after a module is
loaded, is added. This makes possible the initialization of any
additional state data required by a module. This initial state data can
be saved in a ArchiveModuleState, that is now passed down to all the
callbacks that can be defined in a module. With this design, it is
possible to have a per-module state, aimed at opening the door to the
support of more than one archive module.
The initialization of the callbacks is changed so as
_PG_archive_module_init() does not anymore give in input a
ArchiveModuleCallbacks that a module has to fill in with callback
definitions. Instead, a module now needs to return a const
ArchiveModuleCallbacks.
All the structure and callback definitions of archive modules are moved
into their own header, named archive_module.h, from pgarch.h.
Command-based archiving follows the same line, with a new set of files
named shell_archive.{c,h}.
There are a few more items that are under discussion to improve the
design of archive modules, like the fact that basic_archive calls
sigsetjmp() by itself to define its own error handling flow. These will
be adjusted later, the changes done here cover already a good portion
of what has been discussed.
Any modules created for v15 will need to be adjusted to this new
design.
Author: Nathan Bossart
Reviewed-by: Andres Freund
Discussion: https://postgr.es/m/20230130194810.6fztfgbn32e7qarj@awork3.anarazel.de
This commit is contained in:
@@ -31,6 +31,8 @@
|
||||
|
||||
#include "access/xlog.h"
|
||||
#include "access/xlog_internal.h"
|
||||
#include "archive/archive_module.h"
|
||||
#include "archive/shell_archive.h"
|
||||
#include "lib/binaryheap.h"
|
||||
#include "libpq/pqsignal.h"
|
||||
#include "pgstat.h"
|
||||
@@ -97,7 +99,8 @@ char *XLogArchiveLibrary = "";
|
||||
*/
|
||||
static time_t last_sigterm_time = 0;
|
||||
static PgArchData *PgArch = NULL;
|
||||
static ArchiveModuleCallbacks ArchiveContext;
|
||||
static const ArchiveModuleCallbacks *ArchiveCallbacks;
|
||||
static ArchiveModuleState *archive_module_state;
|
||||
|
||||
|
||||
/*
|
||||
@@ -406,8 +409,8 @@ pgarch_ArchiverCopyLoop(void)
|
||||
HandlePgArchInterrupts();
|
||||
|
||||
/* can't do anything if not configured ... */
|
||||
if (ArchiveContext.check_configured_cb != NULL &&
|
||||
!ArchiveContext.check_configured_cb())
|
||||
if (ArchiveCallbacks->check_configured_cb != NULL &&
|
||||
!ArchiveCallbacks->check_configured_cb(archive_module_state))
|
||||
{
|
||||
ereport(WARNING,
|
||||
(errmsg("archive_mode enabled, yet archiving is not configured")));
|
||||
@@ -508,7 +511,7 @@ pgarch_archiveXlog(char *xlog)
|
||||
snprintf(activitymsg, sizeof(activitymsg), "archiving %s", xlog);
|
||||
set_ps_display(activitymsg);
|
||||
|
||||
ret = ArchiveContext.archive_file_cb(xlog, pathname);
|
||||
ret = ArchiveCallbacks->archive_file_cb(archive_module_state, xlog, pathname);
|
||||
if (ret)
|
||||
snprintf(activitymsg, sizeof(activitymsg), "last was %s", xlog);
|
||||
else
|
||||
@@ -814,7 +817,7 @@ HandlePgArchInterrupts(void)
|
||||
/*
|
||||
* LoadArchiveLibrary
|
||||
*
|
||||
* Loads the archiving callbacks into our local ArchiveContext.
|
||||
* Loads the archiving callbacks into our local ArchiveCallbacks.
|
||||
*/
|
||||
static void
|
||||
LoadArchiveLibrary(void)
|
||||
@@ -827,8 +830,6 @@ LoadArchiveLibrary(void)
|
||||
errmsg("both archive_command and archive_library set"),
|
||||
errdetail("Only one of archive_command, archive_library may be set.")));
|
||||
|
||||
memset(&ArchiveContext, 0, sizeof(ArchiveModuleCallbacks));
|
||||
|
||||
/*
|
||||
* If shell archiving is enabled, use our special initialization function.
|
||||
* Otherwise, load the library and call its _PG_archive_module_init().
|
||||
@@ -844,12 +845,16 @@ LoadArchiveLibrary(void)
|
||||
ereport(ERROR,
|
||||
(errmsg("archive modules have to define the symbol %s", "_PG_archive_module_init")));
|
||||
|
||||
(*archive_init) (&ArchiveContext);
|
||||
ArchiveCallbacks = (*archive_init) ();
|
||||
|
||||
if (ArchiveContext.archive_file_cb == NULL)
|
||||
if (ArchiveCallbacks->archive_file_cb == NULL)
|
||||
ereport(ERROR,
|
||||
(errmsg("archive modules must register an archive callback")));
|
||||
|
||||
archive_module_state = (ArchiveModuleState *) palloc0(sizeof(ArchiveModuleState));
|
||||
if (ArchiveCallbacks->startup_cb != NULL)
|
||||
ArchiveCallbacks->startup_cb(archive_module_state);
|
||||
|
||||
before_shmem_exit(pgarch_call_module_shutdown_cb, 0);
|
||||
}
|
||||
|
||||
@@ -859,6 +864,6 @@ LoadArchiveLibrary(void)
|
||||
static void
|
||||
pgarch_call_module_shutdown_cb(int code, Datum arg)
|
||||
{
|
||||
if (ArchiveContext.shutdown_cb != NULL)
|
||||
ArchiveContext.shutdown_cb();
|
||||
if (ArchiveCallbacks->shutdown_cb != NULL)
|
||||
ArchiveCallbacks->shutdown_cb(archive_module_state);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user