Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * pg_backup_archiver.c
4 : : *
5 : : * Private implementation of the archiver routines.
6 : : *
7 : : * See the headers to pg_restore for more details.
8 : : *
9 : : * Copyright (c) 2000, Philip Warner
10 : : * Rights are granted to use this software in any way so long
11 : : * as this notice is not removed.
12 : : *
13 : : * The author is not responsible for loss or damages that may
14 : : * result from its use.
15 : : *
16 : : *
17 : : * IDENTIFICATION
18 : : * src/bin/pg_dump/pg_backup_archiver.c
19 : : *
20 : : *-------------------------------------------------------------------------
21 : : */
22 : : #include "postgres_fe.h"
23 : :
24 : : #include <ctype.h>
25 : : #include <fcntl.h>
26 : : #include <unistd.h>
27 : : #include <sys/stat.h>
28 : : #include <sys/wait.h>
29 : : #ifdef WIN32
30 : : #include <io.h>
31 : : #endif
32 : :
33 : : #include "catalog/pg_class_d.h"
34 : : #include "catalog/pg_largeobject_metadata_d.h"
35 : : #include "catalog/pg_shdepend_d.h"
36 : : #include "common/string.h"
37 : : #include "compress_io.h"
38 : : #include "dumputils.h"
39 : : #include "fe_utils/string_utils.h"
40 : : #include "lib/binaryheap.h"
41 : : #include "lib/stringinfo.h"
42 : : #include "libpq/libpq-fs.h"
43 : : #include "parallel.h"
44 : : #include "pg_backup_archiver.h"
45 : : #include "pg_backup_db.h"
46 : : #include "pg_backup_utils.h"
47 : : #include "pgtar.h"
48 : :
49 : : #define TEXT_DUMP_HEADER "--\n-- PostgreSQL database dump\n--\n\n"
50 : : #define TEXT_DUMPALL_HEADER "--\n-- PostgreSQL database cluster dump\n--\n\n"
51 : :
52 : : #define TOC_PREFIX_NONE ""
53 : : #define TOC_PREFIX_DATA "Data for "
54 : : #define TOC_PREFIX_STATS "Statistics for "
55 : :
56 : : static ArchiveHandle *_allocAH(const char *FileSpec, const ArchiveFormat fmt,
57 : : const pg_compress_specification compression_spec,
58 : : bool dosync, ArchiveMode mode,
59 : : SetupWorkerPtrType setupWorkerPtr,
60 : : DataDirSyncMethod sync_method);
61 : : static void _getObjectDescription(PQExpBuffer buf, const TocEntry *te);
62 : : static void _printTocEntry(ArchiveHandle *AH, TocEntry *te, const char *pfx);
63 : : static void _doSetFixedOutputState(ArchiveHandle *AH);
64 : : static void _doSetSessionAuth(ArchiveHandle *AH, const char *user);
65 : : static void _reconnectToDB(ArchiveHandle *AH, const char *dbname);
66 : : static void _becomeUser(ArchiveHandle *AH, const char *user);
67 : : static void _becomeOwner(ArchiveHandle *AH, TocEntry *te);
68 : : static void _selectOutputSchema(ArchiveHandle *AH, const char *schemaName);
69 : : static void _selectTablespace(ArchiveHandle *AH, const char *tablespace);
70 : : static void _selectTableAccessMethod(ArchiveHandle *AH, const char *tableam);
71 : : static void _printTableAccessMethodNoStorage(ArchiveHandle *AH,
72 : : TocEntry *te);
73 : : static void processEncodingEntry(ArchiveHandle *AH, TocEntry *te);
74 : : static void processStdStringsEntry(ArchiveHandle *AH, TocEntry *te);
75 : : static void processSearchPathEntry(ArchiveHandle *AH, TocEntry *te);
76 : : static int _tocEntryRequired(TocEntry *te, teSection curSection, ArchiveHandle *AH);
77 : : static RestorePass _tocEntryRestorePass(TocEntry *te);
78 : : static bool _tocEntryIsACL(TocEntry *te);
79 : : static void _disableTriggersIfNecessary(ArchiveHandle *AH, TocEntry *te);
80 : : static void _enableTriggersIfNecessary(ArchiveHandle *AH, TocEntry *te);
81 : : static bool is_load_via_partition_root(TocEntry *te);
82 : : static void buildTocEntryArrays(ArchiveHandle *AH);
83 : : static void _moveBefore(TocEntry *pos, TocEntry *te);
84 : : static int _discoverArchiveFormat(ArchiveHandle *AH);
85 : :
86 : : static int RestoringToDB(ArchiveHandle *AH);
87 : : static void dump_lo_buf(ArchiveHandle *AH);
88 : : static void dumpTimestamp(ArchiveHandle *AH, const char *msg, time_t tim);
89 : : static void SetOutput(ArchiveHandle *AH, const char *filename,
90 : : const pg_compress_specification compression_spec);
91 : : static CompressFileHandle *SaveOutput(ArchiveHandle *AH);
92 : : static void RestoreOutput(ArchiveHandle *AH, CompressFileHandle *savedOutput);
93 : :
94 : : static int restore_toc_entry(ArchiveHandle *AH, TocEntry *te, bool is_parallel);
95 : : static void restore_toc_entries_prefork(ArchiveHandle *AH,
96 : : TocEntry *pending_list);
97 : : static void restore_toc_entries_parallel(ArchiveHandle *AH,
98 : : ParallelState *pstate,
99 : : TocEntry *pending_list);
100 : : static void restore_toc_entries_postfork(ArchiveHandle *AH,
101 : : TocEntry *pending_list);
102 : : static void pending_list_header_init(TocEntry *l);
103 : : static void pending_list_append(TocEntry *l, TocEntry *te);
104 : : static void pending_list_remove(TocEntry *te);
105 : : static int TocEntrySizeCompareQsort(const void *p1, const void *p2);
106 : : static int TocEntrySizeCompareBinaryheap(void *p1, void *p2, void *arg);
107 : : static void move_to_ready_heap(TocEntry *pending_list,
108 : : binaryheap *ready_heap,
109 : : RestorePass pass);
110 : : static TocEntry *pop_next_work_item(binaryheap *ready_heap,
111 : : ParallelState *pstate);
112 : : static void mark_dump_job_done(ArchiveHandle *AH,
113 : : TocEntry *te,
114 : : int status,
115 : : void *callback_data);
116 : : static void mark_restore_job_done(ArchiveHandle *AH,
117 : : TocEntry *te,
118 : : int status,
119 : : void *callback_data);
120 : : static void fix_dependencies(ArchiveHandle *AH);
121 : : static bool has_lock_conflicts(TocEntry *te1, TocEntry *te2);
122 : : static void repoint_table_dependencies(ArchiveHandle *AH);
123 : : static void identify_locking_dependencies(ArchiveHandle *AH, TocEntry *te);
124 : : static void reduce_dependencies(ArchiveHandle *AH, TocEntry *te,
125 : : binaryheap *ready_heap);
126 : : static void mark_create_done(ArchiveHandle *AH, TocEntry *te);
127 : : static void inhibit_data_for_failed_table(ArchiveHandle *AH, TocEntry *te);
128 : :
129 : : static void StrictNamesCheck(RestoreOptions *ropt);
130 : :
131 : :
132 : : /*
133 : : * Allocate a new DumpOptions block containing all default values.
134 : : */
135 : : DumpOptions *
4277 alvherre@alvh.no-ip. 136 :CBC 67 : NewDumpOptions(void)
137 : : {
137 michael@paquier.xyz 138 :GNC 67 : DumpOptions *opts = pg_malloc_object(DumpOptions);
139 : :
4188 tgl@sss.pgh.pa.us 140 :CBC 67 : InitDumpOptions(opts);
141 : 67 : return opts;
142 : : }
143 : :
144 : : /*
145 : : * Initialize a DumpOptions struct to all default values
146 : : */
147 : : void
148 : 303 : InitDumpOptions(DumpOptions *opts)
149 : : {
150 : 303 : memset(opts, 0, sizeof(DumpOptions));
151 : : /* set any fields that shouldn't default to zeroes */
4277 alvherre@alvh.no-ip. 152 : 303 : opts->include_everything = true;
2105 tgl@sss.pgh.pa.us 153 : 303 : opts->cparams.promptPassword = TRI_DEFAULT;
4277 alvherre@alvh.no-ip. 154 : 303 : opts->dumpSections = DUMP_UNSECTIONED;
582 nathan@postgresql.or 155 : 303 : opts->dumpSchema = true;
156 : 303 : opts->dumpData = true;
399 jdavis@postgresql.or 157 : 303 : opts->dumpStatistics = false;
4277 alvherre@alvh.no-ip. 158 : 303 : }
159 : :
160 : : /*
161 : : * Create a freshly allocated DumpOptions with options equivalent to those
162 : : * found in the given RestoreOptions.
163 : : */
164 : : DumpOptions *
165 : 67 : dumpOptionsFromRestoreOptions(RestoreOptions *ropt)
166 : : {
167 : 67 : DumpOptions *dopt = NewDumpOptions();
168 : :
169 : : /* this is the inverse of what's at the end of pg_dump.c's main() */
2105 tgl@sss.pgh.pa.us 170 [ + + ]: 67 : dopt->cparams.dbname = ropt->cparams.dbname ? pg_strdup(ropt->cparams.dbname) : NULL;
171 [ + + ]: 67 : dopt->cparams.pgport = ropt->cparams.pgport ? pg_strdup(ropt->cparams.pgport) : NULL;
172 [ + + ]: 67 : dopt->cparams.pghost = ropt->cparams.pghost ? pg_strdup(ropt->cparams.pghost) : NULL;
173 [ + + ]: 67 : dopt->cparams.username = ropt->cparams.username ? pg_strdup(ropt->cparams.username) : NULL;
174 : 67 : dopt->cparams.promptPassword = ropt->cparams.promptPassword;
4277 alvherre@alvh.no-ip. 175 : 67 : dopt->outputClean = ropt->dropSchema;
582 nathan@postgresql.or 176 : 67 : dopt->dumpData = ropt->dumpData;
177 : 67 : dopt->dumpSchema = ropt->dumpSchema;
495 jdavis@postgresql.or 178 : 67 : dopt->dumpSections = ropt->dumpSections;
179 : 67 : dopt->dumpStatistics = ropt->dumpStatistics;
4277 alvherre@alvh.no-ip. 180 : 67 : dopt->if_exists = ropt->if_exists;
181 : 67 : dopt->column_inserts = ropt->column_inserts;
182 : 67 : dopt->aclsSkip = ropt->aclsSkip;
183 : 67 : dopt->outputSuperuser = ropt->superuser;
184 : 67 : dopt->outputCreateDB = ropt->createDB;
185 : 67 : dopt->outputNoOwner = ropt->noOwner;
1625 michael@paquier.xyz 186 : 67 : dopt->outputNoTableAm = ropt->noTableAm;
4277 alvherre@alvh.no-ip. 187 : 67 : dopt->outputNoTablespaces = ropt->noTablespace;
188 : 67 : dopt->disable_triggers = ropt->disable_triggers;
189 : 67 : dopt->use_setsessauth = ropt->use_setsessauth;
190 : 67 : dopt->disable_dollar_quoting = ropt->disable_dollar_quoting;
191 : 67 : dopt->dump_inserts = ropt->dump_inserts;
3078 tgl@sss.pgh.pa.us 192 : 67 : dopt->no_comments = ropt->no_comments;
471 193 : 67 : dopt->no_policies = ropt->no_policies;
3336 peter_e@gmx.net 194 : 67 : dopt->no_publications = ropt->no_publications;
4277 alvherre@alvh.no-ip. 195 : 67 : dopt->no_security_labels = ropt->no_security_labels;
3339 peter_e@gmx.net 196 : 67 : dopt->no_subscriptions = ropt->no_subscriptions;
4277 alvherre@alvh.no-ip. 197 : 67 : dopt->lockWaitTimeout = ropt->lockWaitTimeout;
198 : 67 : dopt->include_everything = ropt->include_everything;
199 : 67 : dopt->enable_row_security = ropt->enable_row_security;
3598 peter_e@gmx.net 200 : 67 : dopt->sequence_data = ropt->sequence_data;
323 nathan@postgresql.or 201 [ + + ]: 67 : dopt->restrict_key = ropt->restrict_key ? pg_strdup(ropt->restrict_key) : NULL;
202 : :
4277 alvherre@alvh.no-ip. 203 : 67 : return dopt;
204 : : }
205 : :
206 : :
207 : : /*
208 : : * Wrapper functions.
209 : : *
210 : : * The objective is to make writing new formats and dumpers as simple
211 : : * as possible, if necessary at the expense of extra function calls etc.
212 : : *
213 : : */
214 : :
215 : : /*
216 : : * The dump worker setup needs lots of knowledge of the internals of pg_dump,
217 : : * so it's defined in pg_dump.c and passed into OpenArchive. The restore worker
218 : : * setup doesn't need to know anything much, so it's defined here.
219 : : */
220 : : static void
3821 tgl@sss.pgh.pa.us 221 : 10 : setupRestoreWorker(Archive *AHX)
222 : : {
4846 andrew@dunslane.net 223 : 10 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
224 : :
3218 peter_e@gmx.net 225 : 10 : AH->ReopenPtr(AH);
4846 andrew@dunslane.net 226 : 10 : }
227 : :
228 : :
229 : : /* Create a new archive */
230 : : /* Public */
231 : : Archive *
9231 bruce@momjian.us 232 : 211 : CreateArchive(const char *FileSpec, const ArchiveFormat fmt,
233 : : const pg_compress_specification compression_spec,
234 : : bool dosync, ArchiveMode mode,
235 : : SetupWorkerPtrType setupDumpWorker,
236 : : DataDirSyncMethod sync_method)
237 : :
238 : : {
1306 michael@paquier.xyz 239 : 211 : ArchiveHandle *AH = _allocAH(FileSpec, fmt, compression_spec,
240 : : dosync, mode, setupDumpWorker, sync_method);
241 : :
9231 bruce@momjian.us 242 : 210 : return (Archive *) AH;
243 : : }
244 : :
245 : : /* Open an existing archive */
246 : : /* Public */
247 : : Archive *
248 : 65 : OpenArchive(const char *FileSpec, const ArchiveFormat fmt)
249 : : {
250 : : ArchiveHandle *AH;
1306 michael@paquier.xyz 251 : 65 : pg_compress_specification compression_spec = {0};
252 : :
253 : 65 : compression_spec.algorithm = PG_COMPRESSION_NONE;
254 : 65 : AH = _allocAH(FileSpec, fmt, compression_spec, true,
255 : : archModeRead, setupRestoreWorker,
256 : : DATA_DIR_SYNC_METHOD_FSYNC);
257 : :
9231 bruce@momjian.us 258 : 65 : return (Archive *) AH;
259 : : }
260 : :
261 : : /* Public */
262 : : void
3821 tgl@sss.pgh.pa.us 263 : 255 : CloseArchive(Archive *AHX)
264 : : {
9231 bruce@momjian.us 265 : 255 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
266 : :
3218 peter_e@gmx.net 267 : 255 : AH->ClosePtr(AH);
268 : :
269 : : /* Close the output */
1223 tomas.vondra@postgre 270 : 255 : errno = 0;
1195 271 [ - + ]: 255 : if (!EndCompressFileHandle(AH->OF))
1544 tgl@sss.pgh.pa.us 272 :UBC 0 : pg_fatal("could not close output file: %m");
9492 bruce@momjian.us 273 :CBC 255 : }
274 : :
275 : : /* Public */
276 : : void
3821 tgl@sss.pgh.pa.us 277 : 470 : SetArchiveOptions(Archive *AH, DumpOptions *dopt, RestoreOptions *ropt)
278 : : {
279 : : /* Caller can omit dump options, in which case we synthesize them */
280 [ + + + - ]: 470 : if (dopt == NULL && ropt != NULL)
281 : 67 : dopt = dumpOptionsFromRestoreOptions(ropt);
282 : :
283 : : /* Save options for later access */
284 : 470 : AH->dopt = dopt;
5145 285 : 470 : AH->ropt = ropt;
3821 286 : 470 : }
287 : :
288 : : /* Public */
289 : : void
290 : 252 : ProcessArchiveRestoreOptions(Archive *AHX)
291 : : {
292 : 252 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
293 : 252 : RestoreOptions *ropt = AH->public.ropt;
294 : : TocEntry *te;
295 : : teSection curSection;
296 : :
297 : : /* Decide which TOC entries will be dumped/restored, and mark them */
5145 298 : 252 : curSection = SECTION_PRE_DATA;
299 [ + + ]: 52068 : for (te = AH->toc->next; te != AH->toc; te = te->next)
300 : : {
301 : : /*
302 : : * When writing an archive, we also take this opportunity to check
303 : : * that we have generated the entries in a sane order that respects
304 : : * the section divisions. When reading, don't complain, since buggy
305 : : * old versions of pg_dump might generate out-of-order archives.
306 : : */
5118 307 [ + + ]: 51816 : if (AH->mode != archModeRead)
308 : : {
309 [ + + + + : 44022 : switch (te->section)
- ]
310 : : {
311 : 8734 : case SECTION_NONE:
312 : : /* ok to be anywhere */
313 : 8734 : break;
314 : 20309 : case SECTION_PRE_DATA:
315 [ - + ]: 20309 : if (curSection != SECTION_PRE_DATA)
2647 peter@eisentraut.org 316 :UBC 0 : pg_log_warning("archive items not in correct section order");
5118 tgl@sss.pgh.pa.us 317 :CBC 20309 : break;
318 : 7564 : case SECTION_DATA:
319 [ - + ]: 7564 : if (curSection == SECTION_POST_DATA)
2647 peter@eisentraut.org 320 :UBC 0 : pg_log_warning("archive items not in correct section order");
5118 tgl@sss.pgh.pa.us 321 :CBC 7564 : break;
322 : 7415 : case SECTION_POST_DATA:
323 : : /* ok no matter which section we were in */
324 : 7415 : break;
5118 tgl@sss.pgh.pa.us 325 :UBC 0 : default:
1544 326 : 0 : pg_fatal("unexpected section code %d",
327 : : (int) te->section);
328 : : break;
329 : : }
330 : : }
331 : :
5145 tgl@sss.pgh.pa.us 332 [ + + ]:CBC 51816 : if (te->section != SECTION_NONE)
333 : 42232 : curSection = te->section;
334 : :
3078 335 : 51816 : te->reqs = _tocEntryRequired(te, curSection, AH);
336 : : }
337 : :
338 : : /* Enforce strict names checking */
3942 teodor@sigaev.ru 339 [ - + ]: 252 : if (ropt->strict_names)
3942 teodor@sigaev.ru 340 :UBC 0 : StrictNamesCheck(ropt);
5145 tgl@sss.pgh.pa.us 341 :CBC 252 : }
342 : :
343 : : /* Public */
344 : : void
15 andrew@dunslane.net 345 : 191 : RestoreArchive(Archive *AHX)
346 : : {
5145 tgl@sss.pgh.pa.us 347 : 191 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
3821 348 : 191 : RestoreOptions *ropt = AH->public.ropt;
349 : : bool parallel_mode;
350 : : TocEntry *te;
351 : : CompressFileHandle *sav;
352 : :
7984 bruce@momjian.us 353 : 191 : AH->stage = STAGE_INITIALIZING;
354 : :
355 : : /*
356 : : * If we're going to do parallel restore, there are some restrictions.
357 : : */
4846 andrew@dunslane.net 358 [ + + + + ]: 191 : parallel_mode = (AH->public.numWorkers > 1 && ropt->useDB);
5420 tgl@sss.pgh.pa.us 359 [ + + ]: 191 : if (parallel_mode)
360 : : {
361 : : /* We haven't got round to making this work for all archive formats */
362 [ + - - + ]: 4 : if (AH->ClonePtr == NULL || AH->ReopenPtr == NULL)
1544 tgl@sss.pgh.pa.us 363 :UBC 0 : pg_fatal("parallel restore is not supported with this archive file format");
364 : :
365 : : /* Doesn't work if the archive represents dependencies as OIDs */
5420 tgl@sss.pgh.pa.us 366 [ - + ]:CBC 4 : if (AH->version < K_VERS_1_8)
1544 tgl@sss.pgh.pa.us 367 :UBC 0 : pg_fatal("parallel restore is not supported with archives made by pre-8.0 pg_dump");
368 : :
369 : : /*
370 : : * It's also not gonna work if we can't reopen the input file, so
371 : : * let's try that immediately.
372 : : */
3218 peter_e@gmx.net 373 :CBC 4 : AH->ReopenPtr(AH);
374 : : }
375 : :
376 : : /*
377 : : * Make sure we won't need (de)compression we haven't got
378 : : */
1223 tomas.vondra@postgre 379 [ + - ]: 191 : if (AH->PrintTocDataPtr != NULL)
380 : : {
6357 andrew@dunslane.net 381 [ + + ]: 30335 : for (te = AH->toc->next; te != AH->toc; te = te->next)
382 : : {
5145 tgl@sss.pgh.pa.us 383 [ + + + + ]: 30278 : if (te->hadDumper && (te->reqs & REQ_DATA) != 0)
384 : : {
1138 385 : 134 : char *errmsg = supports_compression(AH->compression_spec);
386 : :
1223 tomas.vondra@postgre 387 [ - + ]: 134 : if (errmsg)
1223 tomas.vondra@postgre 388 :UBC 0 : pg_fatal("cannot restore from compressed archive (%s)",
389 : : errmsg);
390 : : else
1223 tomas.vondra@postgre 391 :CBC 134 : break;
392 : : }
393 : : }
394 : : }
395 : :
396 : : /*
397 : : * Prepare index arrays, so we can assume we have them throughout restore.
398 : : * It's possible we already did this, though.
399 : : */
5146 tgl@sss.pgh.pa.us 400 [ + + ]: 191 : if (AH->tocsByDumpId == NULL)
401 : 187 : buildTocEntryArrays(AH);
402 : :
403 : : /*
404 : : * If we're using a DB connection, then connect it.
405 : : */
9475 pjw@rhyme.com.au 406 [ + + ]: 191 : if (ropt->useDB)
407 : : {
2647 peter@eisentraut.org 408 : 36 : pg_log_info("connecting to database for restore");
9475 pjw@rhyme.com.au 409 [ - + ]: 36 : if (AH->version < K_VERS_1_3)
1544 tgl@sss.pgh.pa.us 410 :UBC 0 : pg_fatal("direct database connections are not supported in pre-1.3 archives");
411 : :
412 : : /*
413 : : * We don't want to guess at whether the dump will successfully
414 : : * restore; allow the attempt regardless of the version of the restore
415 : : * target.
416 : : */
4566 kgrittn@postgresql.o 417 :CBC 36 : AHX->minRemoteVersion = 0;
3548 tgl@sss.pgh.pa.us 418 : 36 : AHX->maxRemoteVersion = 9999999;
419 : :
452 andrew@dunslane.net 420 : 36 : ConnectDatabaseAhx(AHX, &ropt->cparams, false);
421 : :
422 : : /*
423 : : * If we're talking to the DB directly, don't send comments since they
424 : : * obscure SQL when displaying errors
425 : : */
7984 bruce@momjian.us 426 : 36 : AH->noTocComments = 1;
427 : : }
428 : :
429 : : /*
430 : : * Work out if we have an implied schema-less restore. This can happen if
431 : : * the dump excluded the schema or the user has used a toc list to exclude
432 : : * all of the schema data. All we do is look for schema entries - if none
433 : : * are found then we unset the dumpSchema flag.
434 : : *
435 : : * We could scan for wanted TABLE entries, but that is not the same as
436 : : * data-only. At this stage, it seems unnecessary (6-Mar-2001).
437 : : */
582 nathan@postgresql.or 438 [ + + ]: 191 : if (ropt->dumpSchema)
439 : : {
495 jdavis@postgresql.or 440 : 179 : bool no_schema_found = true;
441 : :
7826 tgl@sss.pgh.pa.us 442 [ + + ]: 1491 : for (te = AH->toc->next; te != AH->toc; te = te->next)
443 : : {
5145 444 [ + + ]: 1471 : if ((te->reqs & REQ_SCHEMA) != 0)
445 : : {
495 jdavis@postgresql.or 446 : 159 : no_schema_found = false;
9247 pjw@rhyme.com.au 447 : 159 : break;
448 : : }
449 : : }
495 jdavis@postgresql.or 450 [ + + ]: 179 : if (no_schema_found)
451 : : {
582 nathan@postgresql.or 452 : 20 : ropt->dumpSchema = false;
495 jdavis@postgresql.or 453 : 20 : pg_log_info("implied no-schema restore");
454 : : }
455 : : }
456 : :
457 : : /*
458 : : * Setup the output file if necessary.
459 : : */
5638 tgl@sss.pgh.pa.us 460 : 191 : sav = SaveOutput(AH);
1306 michael@paquier.xyz 461 [ + + - + ]: 191 : if (ropt->filename || ropt->compression_spec.algorithm != PG_COMPRESSION_NONE)
15 andrew@dunslane.net 462 : 151 : SetOutput(AH, ropt->filename, ropt->compression_spec);
463 : :
8717 peter_e@gmx.net 464 : 191 : ahprintf(AH, "--\n-- PostgreSQL database dump\n--\n\n");
465 : :
466 : : /*
467 : : * If generating plain-text output, enter restricted mode to block any
468 : : * unexpected psql meta-commands. A malicious source might try to inject
469 : : * a variety of things via bogus responses to queries. While we cannot
470 : : * prevent such sources from affecting the destination at restore time, we
471 : : * can block psql meta-commands so that the client machine that runs psql
472 : : * with the dump output remains unaffected.
473 : : */
323 nathan@postgresql.or 474 [ + + ]: 191 : if (ropt->restrict_key)
475 : 153 : ahprintf(AH, "\\restrict %s\n\n", ropt->restrict_key);
476 : :
4376 tgl@sss.pgh.pa.us 477 [ + - ]: 191 : if (AH->archiveRemoteVersion)
478 : 191 : ahprintf(AH, "-- Dumped from database version %s\n",
479 : : AH->archiveRemoteVersion);
480 [ + - ]: 191 : if (AH->archiveDumpVersion)
481 : 191 : ahprintf(AH, "-- Dumped by pg_dump version %s\n",
482 : : AH->archiveDumpVersion);
483 : :
484 : 191 : ahprintf(AH, "\n");
485 : :
5971 bruce@momjian.us 486 [ + + ]: 191 : if (AH->public.verbose)
7746 tgl@sss.pgh.pa.us 487 : 43 : dumpTimestamp(AH, "Started on", AH->createDate);
488 : :
7442 489 [ - + ]: 191 : if (ropt->single_txn)
490 : : {
7441 tgl@sss.pgh.pa.us 491 [ # # ]:UBC 0 : if (AH->connection)
4277 alvherre@alvh.no-ip. 492 : 0 : StartTransaction(AHX);
493 : : else
7441 tgl@sss.pgh.pa.us 494 : 0 : ahprintf(AH, "BEGIN;\n\n");
495 : : }
496 : :
497 : : /*
498 : : * Establish important parameter values right away.
499 : : */
8162 tgl@sss.pgh.pa.us 500 :CBC 191 : _doSetFixedOutputState(AH);
501 : :
7984 bruce@momjian.us 502 : 191 : AH->stage = STAGE_PROCESSING;
503 : :
504 : : /*
505 : : * Drop the items at the start, in reverse order
506 : : */
9231 507 [ + + ]: 191 : if (ropt->dropSchema)
508 : : {
7746 tgl@sss.pgh.pa.us 509 [ + + ]: 1534 : for (te = AH->toc->prev; te != AH->toc; te = te->prev)
510 : : {
511 : 1507 : AH->currentTE = te;
512 : :
513 : : /*
514 : : * In createDB mode, issue a DROP *only* for the database as a
515 : : * whole. Issuing drops against anything else would be wrong,
516 : : * because at this point we're connected to the wrong database.
517 : : * (The DATABASE PROPERTIES entry, if any, should be treated like
518 : : * the DATABASE entry.)
519 : : */
5001 520 [ + + ]: 1507 : if (ropt->createDB)
521 : : {
3081 522 [ + + ]: 682 : if (strcmp(te->desc, "DATABASE") != 0 &&
523 [ + + ]: 660 : strcmp(te->desc, "DATABASE PROPERTIES") != 0)
5001 524 : 640 : continue;
525 : : }
526 : :
527 : : /* Otherwise, drop anything that's selected and has a dropStmt */
5145 528 [ + + + + ]: 867 : if (((te->reqs & (REQ_SCHEMA | REQ_DATA)) != 0) && te->dropStmt)
529 : : {
820 530 : 362 : bool not_allowed_in_txn = false;
531 : :
2647 peter@eisentraut.org 532 : 362 : pg_log_info("dropping %s %s", te->desc, te->tag);
533 : :
534 : : /*
535 : : * In --transaction-size mode, we have to temporarily exit our
536 : : * transaction block to drop objects that can't be dropped
537 : : * within a transaction.
538 : : */
820 tgl@sss.pgh.pa.us 539 [ + + ]: 362 : if (ropt->txn_size > 0)
540 : : {
541 [ + + ]: 40 : if (strcmp(te->desc, "DATABASE") == 0 ||
542 [ + - ]: 20 : strcmp(te->desc, "DATABASE PROPERTIES") == 0)
543 : : {
544 : 40 : not_allowed_in_txn = true;
545 [ + - ]: 40 : if (AH->connection)
546 : 40 : CommitTransaction(AHX);
547 : : else
820 tgl@sss.pgh.pa.us 548 :UBC 0 : ahprintf(AH, "COMMIT;\n");
549 : : }
550 : : }
551 : :
552 : : /* Select owner and schema as necessary */
8316 tgl@sss.pgh.pa.us 553 :CBC 362 : _becomeOwner(AH, te);
8817 554 : 362 : _selectOutputSchema(AH, te->namespace);
555 : :
556 : : /*
557 : : * Now emit the DROP command, if the object has one. Note we
558 : : * don't necessarily emit it verbatim; at this point we add an
559 : : * appropriate IF EXISTS clause, if the user requested it.
560 : : */
820 561 [ + + ]: 362 : if (strcmp(te->desc, "BLOB METADATA") == 0)
562 : : {
563 : : /* We must generate the per-blob commands */
564 [ + + ]: 4 : if (ropt->if_exists)
565 : 2 : IssueCommandPerBlob(AH, te,
566 : : "SELECT pg_catalog.lo_unlink(oid) "
567 : : "FROM pg_catalog.pg_largeobject_metadata "
568 : : "WHERE oid = '", "'");
569 : : else
570 : 2 : IssueCommandPerBlob(AH, te,
571 : : "SELECT pg_catalog.lo_unlink('",
572 : : "')");
573 : : }
574 [ + + ]: 358 : else if (*te->dropStmt != '\0')
575 : : {
1258 576 [ + + ]: 332 : if (!ropt->if_exists ||
577 [ + + ]: 150 : strncmp(te->dropStmt, "--", 2) == 0)
578 : : {
579 : : /*
580 : : * Without --if-exists, or if it's just a comment (as
581 : : * happens for the public schema), print the dropStmt
582 : : * as-is.
583 : : */
4502 alvherre@alvh.no-ip. 584 : 183 : ahprintf(AH, "%s", te->dropStmt);
585 : : }
586 : : else
587 : : {
588 : : /*
589 : : * Inject an appropriate spelling of "if exists". For
590 : : * old-style large objects, we have a routine that
591 : : * knows how to do it, without depending on
592 : : * te->dropStmt; use that. For other objects we need
593 : : * to parse the command.
594 : : */
820 tgl@sss.pgh.pa.us 595 [ - + ]: 149 : if (strcmp(te->desc, "BLOB") == 0)
596 : : {
1303 peter@eisentraut.org 597 :UBC 0 : DropLOIfExists(AH, te->catalogId.oid);
598 : : }
599 : : else
600 : : {
4291 alvherre@alvh.no-ip. 601 :CBC 149 : char *dropStmt = pg_strdup(te->dropStmt);
3512 tgl@sss.pgh.pa.us 602 : 149 : char *dropStmtOrig = dropStmt;
4291 alvherre@alvh.no-ip. 603 : 149 : PQExpBuffer ftStmt = createPQExpBuffer();
604 : :
605 : : /*
606 : : * Need to inject IF EXISTS clause after ALTER
607 : : * TABLE part in ALTER TABLE .. DROP statement
608 : : */
609 [ + + ]: 149 : if (strncmp(dropStmt, "ALTER TABLE", 11) == 0)
610 : : {
2553 drowley@postgresql.o 611 : 19 : appendPQExpBufferStr(ftStmt,
612 : : "ALTER TABLE IF EXISTS");
4291 alvherre@alvh.no-ip. 613 : 19 : dropStmt = dropStmt + 11;
614 : : }
615 : :
616 : : /*
617 : : * ALTER TABLE..ALTER COLUMN..DROP DEFAULT does
618 : : * not support the IF EXISTS clause, and therefore
619 : : * we simply emit the original command for DEFAULT
620 : : * objects (modulo the adjustment made above).
621 : : *
622 : : * Likewise, don't mess with DATABASE PROPERTIES.
623 : : *
624 : : * If we used CREATE OR REPLACE VIEW as a means of
625 : : * quasi-dropping an ON SELECT rule, that should
626 : : * be emitted unchanged as well.
627 : : *
628 : : * For other object types, we need to extract the
629 : : * first part of the DROP which includes the
630 : : * object type. Most of the time this matches
631 : : * te->desc, so search for that; however for the
632 : : * different kinds of CONSTRAINTs, we know to
633 : : * search for hardcoded "DROP CONSTRAINT" instead.
634 : : */
3512 tgl@sss.pgh.pa.us 635 [ + + ]: 149 : if (strcmp(te->desc, "DEFAULT") == 0 ||
3081 636 [ + - ]: 146 : strcmp(te->desc, "DATABASE PROPERTIES") == 0 ||
3512 637 [ - + ]: 146 : strncmp(dropStmt, "CREATE OR REPLACE VIEW", 22) == 0)
4016 heikki.linnakangas@i 638 : 3 : appendPQExpBufferStr(ftStmt, dropStmt);
639 : : else
640 : : {
641 : : char buffer[40];
642 : : char *mark;
643 : :
4291 alvherre@alvh.no-ip. 644 [ + + ]: 146 : if (strcmp(te->desc, "CONSTRAINT") == 0 ||
3296 tgl@sss.pgh.pa.us 645 [ + - ]: 132 : strcmp(te->desc, "CHECK CONSTRAINT") == 0 ||
4291 alvherre@alvh.no-ip. 646 [ + + ]: 132 : strcmp(te->desc, "FK CONSTRAINT") == 0)
647 : 16 : strcpy(buffer, "DROP CONSTRAINT");
648 : : else
649 : 130 : snprintf(buffer, sizeof(buffer), "DROP %s",
650 : : te->desc);
651 : :
652 : 146 : mark = strstr(dropStmt, buffer);
653 : :
3512 tgl@sss.pgh.pa.us 654 [ + - ]: 146 : if (mark)
655 : : {
656 : 146 : *mark = '\0';
657 : 146 : appendPQExpBuffer(ftStmt, "%s%s IF EXISTS%s",
658 : : dropStmt, buffer,
659 : 146 : mark + strlen(buffer));
660 : : }
661 : : else
662 : : {
663 : : /* complain and emit unmodified command */
2647 peter@eisentraut.org 664 :UBC 0 : pg_log_warning("could not find where to insert IF EXISTS in statement \"%s\"",
665 : : dropStmtOrig);
3512 tgl@sss.pgh.pa.us 666 : 0 : appendPQExpBufferStr(ftStmt, dropStmt);
667 : : }
668 : : }
669 : :
4291 alvherre@alvh.no-ip. 670 :CBC 149 : ahprintf(AH, "%s", ftStmt->data);
671 : :
672 : 149 : destroyPQExpBuffer(ftStmt);
3512 tgl@sss.pgh.pa.us 673 : 149 : pg_free(dropStmtOrig);
674 : : }
675 : : }
676 : : }
677 : :
678 : : /*
679 : : * In --transaction-size mode, re-establish the transaction
680 : : * block if needed; otherwise, commit after every N drops.
681 : : */
820 682 [ + + ]: 362 : if (ropt->txn_size > 0)
683 : : {
684 [ + - ]: 40 : if (not_allowed_in_txn)
685 : : {
686 [ + - ]: 40 : if (AH->connection)
687 : 40 : StartTransaction(AHX);
688 : : else
820 tgl@sss.pgh.pa.us 689 :UBC 0 : ahprintf(AH, "BEGIN;\n");
820 tgl@sss.pgh.pa.us 690 :CBC 40 : AH->txnCount = 0;
691 : : }
820 tgl@sss.pgh.pa.us 692 [ # # ]:UBC 0 : else if (++AH->txnCount >= ropt->txn_size)
693 : : {
694 [ # # ]: 0 : if (AH->connection)
695 : : {
696 : 0 : CommitTransaction(AHX);
697 : 0 : StartTransaction(AHX);
698 : : }
699 : : else
700 : 0 : ahprintf(AH, "COMMIT;\nBEGIN;\n");
701 : 0 : AH->txnCount = 0;
702 : : }
703 : : }
704 : : }
705 : : }
706 : :
707 : : /*
708 : : * _selectOutputSchema may have set currSchema to reflect the effect
709 : : * of a "SET search_path" command it emitted. However, by now we may
710 : : * have dropped that schema; or it might not have existed in the first
711 : : * place. In either case the effective value of search_path will not
712 : : * be what we think. Forcibly reset currSchema so that we will
713 : : * re-establish the search_path setting when needed (after creating
714 : : * the schema).
715 : : *
716 : : * If we treated users as pg_dump'able objects then we'd need to reset
717 : : * currUser here too.
718 : : */
1475 peter@eisentraut.org 719 :CBC 27 : free(AH->currSchema);
6357 andrew@dunslane.net 720 : 27 : AH->currSchema = NULL;
721 : : }
722 : :
5420 tgl@sss.pgh.pa.us 723 [ + + ]: 191 : if (parallel_mode)
724 : : {
725 : : /*
726 : : * In parallel mode, turn control over to the parallel-restore logic.
727 : : */
728 : : ParallelState *pstate;
729 : : TocEntry pending_list;
730 : :
731 : : /* The archive format module may need some setup for this */
2846 732 [ + - ]: 4 : if (AH->PrepParallelRestorePtr)
733 : 4 : AH->PrepParallelRestorePtr(AH);
734 : :
735 : 4 : pending_list_header_init(&pending_list);
736 : :
737 : : /* This runs PRE_DATA items and then disconnects from the database */
3253 738 : 4 : restore_toc_entries_prefork(AH, &pending_list);
4846 andrew@dunslane.net 739 [ - + ]: 4 : Assert(AH->connection == NULL);
740 : :
741 : : /* ParallelBackupStart() will actually fork the processes */
3821 tgl@sss.pgh.pa.us 742 : 4 : pstate = ParallelBackupStart(AH);
4846 andrew@dunslane.net 743 : 4 : restore_toc_entries_parallel(AH, pstate, &pending_list);
744 : 4 : ParallelBackupEnd(AH, pstate);
745 : :
746 : : /* reconnect the leader and see if we missed something */
747 : 4 : restore_toc_entries_postfork(AH, &pending_list);
748 [ - + ]: 4 : Assert(AH->connection != NULL);
749 : : }
750 : : else
751 : : {
752 : : /*
753 : : * In serial mode, process everything in three phases: normal items,
754 : : * then ACLs, then post-ACL items. We might be able to skip one or
755 : : * both extra phases in some cases, eg data-only restores.
756 : : */
3253 tgl@sss.pgh.pa.us 757 : 187 : bool haveACL = false;
2304 758 : 187 : bool havePostACL = false;
759 : :
6357 andrew@dunslane.net 760 [ + + ]: 44530 : for (te = AH->toc->next; te != AH->toc; te = te->next)
761 : : {
495 jdavis@postgresql.or 762 [ + + ]: 44344 : if ((te->reqs & (REQ_SCHEMA | REQ_DATA | REQ_STATS)) == 0)
3253 tgl@sss.pgh.pa.us 763 : 2326 : continue; /* ignore if not to be dumped at all */
764 : :
452 nathan@postgresql.or 765 [ + + + - ]: 42018 : switch (_tocEntryRestorePass(te))
766 : : {
3253 tgl@sss.pgh.pa.us 767 : 38613 : case RESTORE_PASS_MAIN:
768 : 38613 : (void) restore_toc_entry(AH, te, false);
769 : 38612 : break;
770 : 1935 : case RESTORE_PASS_ACL:
771 : 1935 : haveACL = true;
772 : 1935 : break;
2304 773 : 1470 : case RESTORE_PASS_POST_ACL:
774 : 1470 : havePostACL = true;
3253 775 : 1470 : break;
776 : : }
777 : : }
778 : :
779 [ + + ]: 186 : if (haveACL)
780 : : {
781 [ + + ]: 42056 : for (te = AH->toc->next; te != AH->toc; te = te->next)
782 : : {
495 jdavis@postgresql.or 783 [ + + + + ]: 82876 : if ((te->reqs & (REQ_SCHEMA | REQ_DATA | REQ_STATS)) != 0 &&
452 nathan@postgresql.or 784 : 40897 : _tocEntryRestorePass(te) == RESTORE_PASS_ACL)
3253 tgl@sss.pgh.pa.us 785 : 1935 : (void) restore_toc_entry(AH, te, false);
786 : : }
787 : : }
788 : :
2304 789 [ + + ]: 186 : if (havePostACL)
790 : : {
3253 791 [ + + ]: 28003 : for (te = AH->toc->next; te != AH->toc; te = te->next)
792 : : {
495 jdavis@postgresql.or 793 [ + + + + ]: 55032 : if ((te->reqs & (REQ_SCHEMA | REQ_DATA | REQ_STATS)) != 0 &&
452 nathan@postgresql.or 794 : 27082 : _tocEntryRestorePass(te) == RESTORE_PASS_POST_ACL)
3253 tgl@sss.pgh.pa.us 795 : 1470 : (void) restore_toc_entry(AH, te, false);
796 : : }
797 : : }
798 : : }
799 : :
800 : : /*
801 : : * Close out any persistent transaction we may have. While these two
802 : : * cases are started in different places, we can end both cases here.
803 : : */
820 804 [ + - + + ]: 190 : if (ropt->single_txn || ropt->txn_size > 0)
805 : : {
7441 806 [ + - ]: 32 : if (AH->connection)
4277 alvherre@alvh.no-ip. 807 : 32 : CommitTransaction(AHX);
808 : : else
7441 tgl@sss.pgh.pa.us 809 :UBC 0 : ahprintf(AH, "COMMIT;\n\n");
810 : : }
811 : :
7746 tgl@sss.pgh.pa.us 812 [ + + ]:CBC 190 : if (AH->public.verbose)
813 : 43 : dumpTimestamp(AH, "Completed on", time(NULL));
814 : :
7774 815 : 190 : ahprintf(AH, "--\n-- PostgreSQL database dump complete\n--\n\n");
816 : :
817 : : /*
818 : : * If generating plain-text output, exit restricted mode at the very end
819 : : * of the script. This is not pro forma; in particular, pg_dumpall
820 : : * requires this when transitioning from one database to another.
821 : : */
323 nathan@postgresql.or 822 [ + + ]: 190 : if (ropt->restrict_key)
823 : 152 : ahprintf(AH, "\\unrestrict %s\n\n", ropt->restrict_key);
824 : :
825 : : /*
826 : : * Clean up & we're done.
827 : : */
7984 bruce@momjian.us 828 : 190 : AH->stage = STAGE_FINALIZING;
829 : :
1306 michael@paquier.xyz 830 [ + + - + ]: 190 : if (ropt->filename || ropt->compression_spec.algorithm != PG_COMPRESSION_NONE)
5638 tgl@sss.pgh.pa.us 831 : 151 : RestoreOutput(AH, sav);
832 : :
8154 833 [ + + ]: 190 : if (ropt->useDB)
5248 rhaas@postgresql.org 834 : 36 : DisconnectDatabase(&AH->public);
9492 bruce@momjian.us 835 : 190 : }
836 : :
837 : : /*
838 : : * Restore a single TOC item. Used in both parallel and non-parallel restore;
839 : : * is_parallel is true if we are in a worker child process.
840 : : *
841 : : * Returns 0 normally, but WORKER_CREATE_DONE or WORKER_INHIBIT_DATA if
842 : : * the parallel parent has to make the corresponding status update.
843 : : */
844 : : static int
3821 tgl@sss.pgh.pa.us 845 : 42114 : restore_toc_entry(ArchiveHandle *AH, TocEntry *te, bool is_parallel)
846 : : {
847 : 42114 : RestoreOptions *ropt = AH->public.ropt;
4846 andrew@dunslane.net 848 : 42114 : int status = WORKER_OK;
849 : : int reqs;
850 : : bool defnDumped;
851 : :
6357 852 : 42114 : AH->currentTE = te;
853 : :
854 : : /* Dump any relevant dump warnings to stderr */
855 [ + + - + ]: 42114 : if (!ropt->suppressDumpWarnings && strcmp(te->desc, "WARNING") == 0)
856 : : {
582 nathan@postgresql.or 857 [ # # # # :UBC 0 : if (ropt->dumpSchema && te->defn != NULL && strlen(te->defn) != 0)
# # ]
2647 peter@eisentraut.org 858 : 0 : pg_log_warning("warning from original dump file: %s", te->defn);
6357 andrew@dunslane.net 859 [ # # # # ]: 0 : else if (te->copyStmt != NULL && strlen(te->copyStmt) != 0)
2647 peter@eisentraut.org 860 : 0 : pg_log_warning("warning from original dump file: %s", te->copyStmt);
861 : : }
862 : :
863 : : /* Work out what, if anything, we want from this entry */
3078 tgl@sss.pgh.pa.us 864 :CBC 42114 : reqs = te->reqs;
865 : :
6357 andrew@dunslane.net 866 : 42114 : defnDumped = false;
867 : :
868 : : /*
869 : : * If it has a schema component that we want, then process that
870 : : */
3253 tgl@sss.pgh.pa.us 871 [ + + ]: 42114 : if ((reqs & REQ_SCHEMA) != 0)
872 : : {
820 873 : 33528 : bool object_is_db = false;
874 : :
875 : : /*
876 : : * In --transaction-size mode, must exit our transaction block to
877 : : * create a database or set its properties.
878 : : */
879 [ + + ]: 33528 : if (strcmp(te->desc, "DATABASE") == 0 ||
880 [ + + ]: 33464 : strcmp(te->desc, "DATABASE PROPERTIES") == 0)
881 : : {
882 : 100 : object_is_db = true;
883 [ + + ]: 100 : if (ropt->txn_size > 0)
884 : : {
885 [ + - ]: 64 : if (AH->connection)
886 : 64 : CommitTransaction(&AH->public);
887 : : else
820 tgl@sss.pgh.pa.us 888 :UBC 0 : ahprintf(AH, "COMMIT;\n\n");
889 : : }
890 : : }
891 : :
892 : : /* Show namespace in log message if available */
4326 heikki.linnakangas@i 893 [ + + ]:CBC 33528 : if (te->namespace)
2647 peter@eisentraut.org 894 : 31968 : pg_log_info("creating %s \"%s.%s\"",
895 : : te->desc, te->namespace, te->tag);
896 : : else
897 : 1560 : pg_log_info("creating %s \"%s\"",
898 : : te->desc, te->tag);
899 : :
495 jdavis@postgresql.or 900 : 33528 : _printTocEntry(AH, te, TOC_PREFIX_NONE);
6357 andrew@dunslane.net 901 : 33528 : defnDumped = true;
902 : :
903 [ + + ]: 33528 : if (strcmp(te->desc, "TABLE") == 0)
904 : : {
905 [ - + ]: 5659 : if (AH->lastErrorTE == te)
906 : : {
907 : : /*
908 : : * We failed to create the table. If
909 : : * --no-data-for-failed-tables was given, mark the
910 : : * corresponding TABLE DATA to be ignored.
911 : : *
912 : : * In the parallel case this must be done in the parent, so we
913 : : * just set the return value.
914 : : */
6357 andrew@dunslane.net 915 [ # # ]:UBC 0 : if (ropt->noDataForFailedTables)
916 : : {
917 [ # # ]: 0 : if (is_parallel)
4846 918 : 0 : status = WORKER_INHIBIT_DATA;
919 : : else
6357 920 : 0 : inhibit_data_for_failed_table(AH, te);
921 : : }
922 : : }
923 : : else
924 : : {
925 : : /*
926 : : * We created the table successfully. Mark the corresponding
927 : : * TABLE DATA for possible truncation.
928 : : *
929 : : * In the parallel case this must be done in the parent, so we
930 : : * just set the return value.
931 : : */
6357 andrew@dunslane.net 932 [ - + ]:CBC 5659 : if (is_parallel)
4846 andrew@dunslane.net 933 :UBC 0 : status = WORKER_CREATE_DONE;
934 : : else
6357 andrew@dunslane.net 935 :CBC 5659 : mark_create_done(AH, te);
936 : : }
937 : : }
938 : :
939 : : /*
940 : : * If we created a DB, connect to it. Also, if we changed DB
941 : : * properties, reconnect to ensure that relevant GUC settings are
942 : : * applied to our session. (That also restarts the transaction block
943 : : * in --transaction-size mode.)
944 : : */
820 tgl@sss.pgh.pa.us 945 [ + + ]: 33528 : if (object_is_db)
946 : : {
2647 peter@eisentraut.org 947 : 100 : pg_log_info("connecting to new database \"%s\"", te->tag);
6357 andrew@dunslane.net 948 : 100 : _reconnectToDB(AH, te->tag);
949 : : }
950 : : }
951 : :
952 : : /*
953 : : * If it has a data component that we want, then process that
954 : : */
955 [ + + ]: 42114 : if ((reqs & REQ_DATA) != 0)
956 : : {
957 : : /*
958 : : * hadDumper will be set if there is genuine data component for this
959 : : * node. Otherwise, we need to check the defn field for statements
960 : : * that need to be executed in data-only restores.
961 : : */
962 [ + + ]: 5078 : if (te->hadDumper)
963 : : {
964 : : /*
965 : : * If we can output the data, then restore it.
966 : : */
3331 bruce@momjian.us 967 [ + - ]: 4535 : if (AH->PrintTocDataPtr != NULL)
968 : : {
495 jdavis@postgresql.or 969 : 4535 : _printTocEntry(AH, te, TOC_PREFIX_DATA);
970 : :
6357 andrew@dunslane.net 971 [ + + ]: 4535 : if (strcmp(te->desc, "BLOBS") == 0 ||
972 [ - + ]: 4461 : strcmp(te->desc, "BLOB COMMENTS") == 0)
973 : : {
2647 peter@eisentraut.org 974 : 74 : pg_log_info("processing %s", te->desc);
975 : :
6357 andrew@dunslane.net 976 : 74 : _selectOutputSchema(AH, "pg_catalog");
977 : :
978 : : /* Send BLOB COMMENTS data to ExecuteSimpleCommands() */
4401 tgl@sss.pgh.pa.us 979 [ - + ]: 74 : if (strcmp(te->desc, "BLOB COMMENTS") == 0)
4401 tgl@sss.pgh.pa.us 980 :UBC 0 : AH->outputKind = OUTPUT_OTHERDATA;
981 : :
3218 peter_e@gmx.net 982 :CBC 74 : AH->PrintTocDataPtr(AH, te);
983 : :
4401 tgl@sss.pgh.pa.us 984 : 74 : AH->outputKind = OUTPUT_SQLCMDS;
985 : : }
986 : : else
987 : : {
988 : : bool use_truncate;
989 : :
3821 990 : 4461 : _disableTriggersIfNecessary(AH, te);
991 : :
992 : : /* Select owner and schema as necessary */
6357 andrew@dunslane.net 993 : 4461 : _becomeOwner(AH, te);
994 : 4461 : _selectOutputSchema(AH, te->namespace);
995 : :
2647 peter@eisentraut.org 996 : 4461 : pg_log_info("processing data for table \"%s.%s\"",
997 : : te->namespace, te->tag);
998 : :
999 : : /*
1000 : : * In parallel restore, if we created the table earlier in
1001 : : * this run (so that we know it is empty) and we are not
1002 : : * restoring a load-via-partition-root data item then we
1003 : : * wrap the COPY in a transaction and precede it with a
1004 : : * TRUNCATE. If wal_level is set to minimal this prevents
1005 : : * WAL-logging the COPY. This obtains a speedup similar
1006 : : * to that from using single_txn mode in non-parallel
1007 : : * restores.
1008 : : *
1009 : : * We mustn't do this for load-via-partition-root cases
1010 : : * because some data might get moved across partition
1011 : : * boundaries, risking deadlock and/or loss of previously
1012 : : * loaded data. (We assume that all partitions of a
1013 : : * partitioned table will be treated the same way.)
1014 : : */
1201 tgl@sss.pgh.pa.us 1015 [ + + + - ]: 4477 : use_truncate = is_parallel && te->created &&
1016 [ + + ]: 16 : !is_load_via_partition_root(te);
1017 : :
1018 [ + + ]: 4461 : if (use_truncate)
1019 : : {
1020 : : /*
1021 : : * Parallel restore is always talking directly to a
1022 : : * server, so no need to see if we should issue BEGIN.
1023 : : */
4277 alvherre@alvh.no-ip. 1024 : 10 : StartTransaction(&AH->public);
1025 : :
1026 : : /*
1027 : : * Issue TRUNCATE with ONLY so that child tables are
1028 : : * not wiped.
1029 : : */
1659 tgl@sss.pgh.pa.us 1030 : 10 : ahprintf(AH, "TRUNCATE TABLE ONLY %s;\n\n",
2874 1031 : 10 : fmtQualifiedId(te->namespace, te->tag));
1032 : : }
1033 : :
1034 : : /*
1035 : : * If we have a copy statement, use it.
1036 : : */
6357 andrew@dunslane.net 1037 [ + + + - ]: 4461 : if (te->copyStmt && strlen(te->copyStmt) > 0)
1038 : : {
1039 : 4374 : ahprintf(AH, "%s", te->copyStmt);
5289 tgl@sss.pgh.pa.us 1040 : 4374 : AH->outputKind = OUTPUT_COPYDATA;
1041 : : }
1042 : : else
1043 : 87 : AH->outputKind = OUTPUT_OTHERDATA;
1044 : :
3218 peter_e@gmx.net 1045 : 4461 : AH->PrintTocDataPtr(AH, te);
1046 : :
1047 : : /*
1048 : : * Terminate COPY if needed.
1049 : : */
5289 tgl@sss.pgh.pa.us 1050 [ + + + + ]: 8833 : if (AH->outputKind == OUTPUT_COPYDATA &&
1051 : 4373 : RestoringToDB(AH))
4277 alvherre@alvh.no-ip. 1052 : 41 : EndDBCopyMode(&AH->public, te->tag);
5289 tgl@sss.pgh.pa.us 1053 : 4460 : AH->outputKind = OUTPUT_SQLCMDS;
1054 : :
1055 : : /* close out the transaction started above */
1201 1056 [ + + ]: 4460 : if (use_truncate)
4277 alvherre@alvh.no-ip. 1057 : 10 : CommitTransaction(&AH->public);
1058 : :
3821 tgl@sss.pgh.pa.us 1059 : 4460 : _enableTriggersIfNecessary(AH, te);
1060 : : }
1061 : : }
1062 : : }
6357 andrew@dunslane.net 1063 [ + - ]: 543 : else if (!defnDumped)
1064 : : {
1065 : : /* If we haven't already dumped the defn part, do so now */
2647 peter@eisentraut.org 1066 : 543 : pg_log_info("executing %s %s", te->desc, te->tag);
495 jdavis@postgresql.or 1067 : 543 : _printTocEntry(AH, te, TOC_PREFIX_NONE);
1068 : : }
1069 : : }
1070 : :
1071 : : /*
1072 : : * If it has a statistics component that we want, then process that
1073 : : */
1074 [ + + ]: 42113 : if ((reqs & REQ_STATS) != 0)
1075 : 3493 : _printTocEntry(AH, te, TOC_PREFIX_STATS);
1076 : :
1077 : : /*
1078 : : * If we emitted anything for this TOC entry, that counts as one action
1079 : : * against the transaction-size limit. Commit if it's time to.
1080 : : */
1081 [ + + + + ]: 42113 : if ((reqs & (REQ_SCHEMA | REQ_DATA | REQ_STATS)) != 0 && ropt->txn_size > 0)
1082 : : {
820 tgl@sss.pgh.pa.us 1083 [ + + ]: 3693 : if (++AH->txnCount >= ropt->txn_size)
1084 : : {
1085 [ + - ]: 11 : if (AH->connection)
1086 : : {
1087 : 11 : CommitTransaction(&AH->public);
1088 : 11 : StartTransaction(&AH->public);
1089 : : }
1090 : : else
820 tgl@sss.pgh.pa.us 1091 :UBC 0 : ahprintf(AH, "COMMIT;\nBEGIN;\n\n");
820 tgl@sss.pgh.pa.us 1092 :CBC 11 : AH->txnCount = 0;
1093 : : }
1094 : : }
1095 : :
4846 andrew@dunslane.net 1096 [ - + - - ]: 42113 : if (AH->public.n_errors > 0 && status == WORKER_OK)
4846 andrew@dunslane.net 1097 :UBC 0 : status = WORKER_IGNORED_ERRORS;
1098 : :
4846 andrew@dunslane.net 1099 :CBC 42113 : return status;
1100 : : }
1101 : :
1102 : : /*
1103 : : * Allocate a new RestoreOptions block.
1104 : : * This is mainly so we can initialize it, but also for future expansion,
1105 : : */
1106 : : RestoreOptions *
9231 bruce@momjian.us 1107 : 294 : NewRestoreOptions(void)
1108 : : {
1109 : : RestoreOptions *opts;
1110 : :
137 michael@paquier.xyz 1111 :GNC 294 : opts = pg_malloc0_object(RestoreOptions);
1112 : :
1113 : : /* set any fields that shouldn't default to zeroes */
9492 bruce@momjian.us 1114 :CBC 294 : opts->format = archUnknown;
2105 tgl@sss.pgh.pa.us 1115 : 294 : opts->cparams.promptPassword = TRI_DEFAULT;
5310 andrew@dunslane.net 1116 : 294 : opts->dumpSections = DUMP_UNSECTIONED;
1306 michael@paquier.xyz 1117 : 294 : opts->compression_spec.algorithm = PG_COMPRESSION_NONE;
1118 : 294 : opts->compression_spec.level = 0;
582 nathan@postgresql.or 1119 : 294 : opts->dumpSchema = true;
1120 : 294 : opts->dumpData = true;
495 jdavis@postgresql.or 1121 : 294 : opts->dumpStatistics = true;
1122 : :
9492 bruce@momjian.us 1123 : 294 : return opts;
1124 : : }
1125 : :
1126 : : static void
3821 tgl@sss.pgh.pa.us 1127 : 4461 : _disableTriggersIfNecessary(ArchiveHandle *AH, TocEntry *te)
1128 : : {
1129 : 4461 : RestoreOptions *ropt = AH->public.ropt;
1130 : :
1131 : : /* This hack is only needed in a data-only restore */
582 nathan@postgresql.or 1132 [ + + + + ]: 4461 : if (ropt->dumpSchema || !ropt->disable_triggers)
9464 pjw@rhyme.com.au 1133 : 4422 : return;
1134 : :
2647 peter@eisentraut.org 1135 : 39 : pg_log_info("disabling triggers for %s", te->tag);
1136 : :
1137 : : /*
1138 : : * Become superuser if possible, since they are the only ones who can
1139 : : * disable constraint triggers. If -S was not given, assume the initial
1140 : : * user identity is a superuser. (XXX would it be better to become the
1141 : : * table owner?)
1142 : : */
8316 tgl@sss.pgh.pa.us 1143 : 39 : _becomeUser(AH, ropt->superuser);
1144 : :
1145 : : /*
1146 : : * Disable them.
1147 : : */
7616 1148 : 39 : ahprintf(AH, "ALTER TABLE %s DISABLE TRIGGER ALL;\n\n",
2874 1149 : 39 : fmtQualifiedId(te->namespace, te->tag));
1150 : : }
1151 : :
1152 : : static void
3821 1153 : 4460 : _enableTriggersIfNecessary(ArchiveHandle *AH, TocEntry *te)
1154 : : {
1155 : 4460 : RestoreOptions *ropt = AH->public.ropt;
1156 : :
1157 : : /* This hack is only needed in a data-only restore */
582 nathan@postgresql.or 1158 [ + + + + ]: 4460 : if (ropt->dumpSchema || !ropt->disable_triggers)
9464 pjw@rhyme.com.au 1159 : 4421 : return;
1160 : :
2647 peter@eisentraut.org 1161 : 39 : pg_log_info("enabling triggers for %s", te->tag);
1162 : :
1163 : : /*
1164 : : * Become superuser if possible, since they are the only ones who can
1165 : : * disable constraint triggers. If -S was not given, assume the initial
1166 : : * user identity is a superuser. (XXX would it be better to become the
1167 : : * table owner?)
1168 : : */
8316 tgl@sss.pgh.pa.us 1169 : 39 : _becomeUser(AH, ropt->superuser);
1170 : :
1171 : : /*
1172 : : * Enable them.
1173 : : */
7616 1174 : 39 : ahprintf(AH, "ALTER TABLE %s ENABLE TRIGGER ALL;\n\n",
2874 1175 : 39 : fmtQualifiedId(te->namespace, te->tag));
1176 : : }
1177 : :
1178 : : /*
1179 : : * Detect whether a TABLE DATA TOC item is performing "load via partition
1180 : : * root", that is the target table is an ancestor partition rather than the
1181 : : * table the TOC item is nominally for.
1182 : : *
1183 : : * In newer archive files this can be detected by checking for a special
1184 : : * comment placed in te->defn. In older files we have to fall back to seeing
1185 : : * if the COPY statement targets the named table or some other one. This
1186 : : * will not work for data dumped as INSERT commands, so we could give a false
1187 : : * negative in that case; fortunately, that's a rarely-used option.
1188 : : */
1189 : : static bool
1201 1190 : 16 : is_load_via_partition_root(TocEntry *te)
1191 : : {
1192 [ + + ]: 16 : if (te->defn &&
1193 [ + - ]: 6 : strncmp(te->defn, "-- load via partition root ", 27) == 0)
1194 : 6 : return true;
1195 [ + + + - ]: 10 : if (te->copyStmt && *te->copyStmt)
1196 : : {
1197 : 6 : PQExpBuffer copyStmt = createPQExpBuffer();
1198 : : bool result;
1199 : :
1200 : : /*
1201 : : * Build the initial part of the COPY as it would appear if the
1202 : : * nominal target table is the actual target. If we see anything
1203 : : * else, it must be a load-via-partition-root case.
1204 : : */
1205 : 6 : appendPQExpBuffer(copyStmt, "COPY %s ",
1206 : 6 : fmtQualifiedId(te->namespace, te->tag));
1207 : 6 : result = strncmp(te->copyStmt, copyStmt->data, copyStmt->len) != 0;
1208 : 6 : destroyPQExpBuffer(copyStmt);
1209 : 6 : return result;
1210 : : }
1211 : : /* Assume it's not load-via-partition-root */
1212 : 4 : return false;
1213 : : }
1214 : :
1215 : : /*
1216 : : * This is a routine that is part of the dumper interface, hence the 'Archive*' parameter.
1217 : : */
1218 : :
1219 : : /* Public */
1220 : : void
8715 peter_e@gmx.net 1221 : 1848424 : WriteData(Archive *AHX, const void *data, size_t dLen)
1222 : : {
9231 bruce@momjian.us 1223 : 1848424 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
1224 : :
9475 pjw@rhyme.com.au 1225 [ - + ]: 1848424 : if (!AH->currToc)
1544 tgl@sss.pgh.pa.us 1226 :UBC 0 : pg_fatal("internal error -- WriteData cannot be called outside the context of a DataDumper routine");
1227 : :
3218 peter_e@gmx.net 1228 :CBC 1848424 : AH->WriteDataPtr(AH, data, dLen);
9492 bruce@momjian.us 1229 : 1848424 : }
1230 : :
1231 : : /*
1232 : : * Create a new TOC entry. The TOC was designed as a TOC, but is now the
1233 : : * repository for all metadata. But the name has stuck.
1234 : : *
1235 : : * The new entry is added to the Archive's TOC list. Most callers can ignore
1236 : : * the result value because nothing else need be done, but a few want to
1237 : : * manipulate the TOC entry further.
1238 : : */
1239 : :
1240 : : /* Public */
1241 : : TocEntry *
2706 alvherre@alvh.no-ip. 1242 : 44022 : ArchiveEntry(Archive *AHX, CatalogId catalogId, DumpId dumpId,
1243 : : ArchiveOpts *opts)
1244 : : {
9231 bruce@momjian.us 1245 : 44022 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
1246 : : TocEntry *newToc;
1247 : :
137 michael@paquier.xyz 1248 :GNC 44022 : newToc = pg_malloc0_object(TocEntry);
1249 : :
8242 tgl@sss.pgh.pa.us 1250 :CBC 44022 : AH->tocCount++;
1251 [ + + ]: 44022 : if (dumpId > AH->maxDumpId)
1252 : 13807 : AH->maxDumpId = dumpId;
1253 : :
9231 bruce@momjian.us 1254 : 44022 : newToc->prev = AH->toc->prev;
1255 : 44022 : newToc->next = AH->toc;
1256 : 44022 : AH->toc->prev->next = newToc;
1257 : 44022 : AH->toc->prev = newToc;
1258 : :
8242 tgl@sss.pgh.pa.us 1259 : 44022 : newToc->catalogId = catalogId;
1260 : 44022 : newToc->dumpId = dumpId;
2706 alvherre@alvh.no-ip. 1261 : 44022 : newToc->section = opts->section;
1262 : :
1263 : 44022 : newToc->tag = pg_strdup(opts->tag);
1264 [ + + ]: 44022 : newToc->namespace = opts->namespace ? pg_strdup(opts->namespace) : NULL;
1265 [ + + ]: 44022 : newToc->tablespace = opts->tablespace ? pg_strdup(opts->tablespace) : NULL;
2673 andres@anarazel.de 1266 [ + + ]: 44022 : newToc->tableam = opts->tableam ? pg_strdup(opts->tableam) : NULL;
799 michael@paquier.xyz 1267 : 44022 : newToc->relkind = opts->relkind;
2622 alvherre@alvh.no-ip. 1268 [ + + ]: 44022 : newToc->owner = opts->owner ? pg_strdup(opts->owner) : NULL;
2706 1269 : 44022 : newToc->desc = pg_strdup(opts->description);
2622 1270 [ + + ]: 44022 : newToc->defn = opts->createStmt ? pg_strdup(opts->createStmt) : NULL;
1271 [ + + ]: 44022 : newToc->dropStmt = opts->dropStmt ? pg_strdup(opts->dropStmt) : NULL;
2706 1272 [ + + ]: 44022 : newToc->copyStmt = opts->copyStmt ? pg_strdup(opts->copyStmt) : NULL;
1273 : :
1274 [ + + ]: 44022 : if (opts->nDeps > 0)
1275 : : {
137 michael@paquier.xyz 1276 :GNC 18022 : newToc->dependencies = pg_malloc_array(DumpId, opts->nDeps);
2706 alvherre@alvh.no-ip. 1277 :CBC 18022 : memcpy(newToc->dependencies, opts->deps, opts->nDeps * sizeof(DumpId));
1278 : 18022 : newToc->nDeps = opts->nDeps;
1279 : : }
1280 : : else
1281 : : {
8242 tgl@sss.pgh.pa.us 1282 : 26000 : newToc->dependencies = NULL;
1283 : 26000 : newToc->nDeps = 0;
1284 : : }
1285 : :
2706 alvherre@alvh.no-ip. 1286 : 44022 : newToc->dataDumper = opts->dumpFn;
1287 : 44022 : newToc->dataDumperArg = opts->dumpArg;
1288 : 44022 : newToc->hadDumper = opts->dumpFn ? true : false;
1289 : :
452 nathan@postgresql.or 1290 : 44022 : newToc->defnDumper = opts->defnFn;
1291 : 44022 : newToc->defnDumperArg = opts->defnArg;
1292 : :
8242 tgl@sss.pgh.pa.us 1293 : 44022 : newToc->formatData = NULL;
2846 1294 : 44022 : newToc->dataLength = 0;
1295 : :
3331 bruce@momjian.us 1296 [ + + ]: 44022 : if (AH->ArchiveEntryPtr != NULL)
3218 peter_e@gmx.net 1297 : 7791 : AH->ArchiveEntryPtr(AH, newToc);
1298 : :
2846 tgl@sss.pgh.pa.us 1299 : 44022 : return newToc;
1300 : : }
1301 : :
1302 : : /* Public */
1303 : : void
3821 1304 : 4 : PrintTOCSummary(Archive *AHX)
1305 : : {
9231 bruce@momjian.us 1306 : 4 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
3821 tgl@sss.pgh.pa.us 1307 : 4 : RestoreOptions *ropt = AH->public.ropt;
1308 : : TocEntry *te;
1306 michael@paquier.xyz 1309 : 4 : pg_compress_specification out_compression_spec = {0};
1310 : : teSection curSection;
1311 : : CompressFileHandle *sav;
1312 : : const char *fmtName;
1313 : : char stamp_str[64];
1314 : :
1315 : : /* TOC is always uncompressed */
1316 : 4 : out_compression_spec.algorithm = PG_COMPRESSION_NONE;
1317 : :
5638 tgl@sss.pgh.pa.us 1318 : 4 : sav = SaveOutput(AH);
9231 bruce@momjian.us 1319 [ - + ]: 4 : if (ropt->filename)
15 andrew@dunslane.net 1320 :UBC 0 : SetOutput(AH, ropt->filename, out_compression_spec);
1321 : :
4265 tgl@sss.pgh.pa.us 1322 [ - + ]:CBC 4 : if (strftime(stamp_str, sizeof(stamp_str), PGDUMP_STRFTIME_FMT,
1323 : 4 : localtime(&AH->createDate)) == 0)
4265 tgl@sss.pgh.pa.us 1324 :UBC 0 : strcpy(stamp_str, "[unknown]");
1325 : :
4316 bruce@momjian.us 1326 :CBC 4 : ahprintf(AH, ";\n; Archive created at %s\n", stamp_str);
1223 tomas.vondra@postgre 1327 : 8 : ahprintf(AH, "; dbname: %s\n; TOC Entries: %d\n; Compression: %s\n",
2706 alvherre@alvh.no-ip. 1328 : 4 : sanitize_line(AH->archdbname, false),
1329 : : AH->tocCount,
1330 : : get_compress_algorithm_name(AH->compression_spec.algorithm));
1331 : :
9231 bruce@momjian.us 1332 [ + + - - ]: 4 : switch (AH->format)
1333 : : {
9475 pjw@rhyme.com.au 1334 : 3 : case archCustom:
1335 : 3 : fmtName = "CUSTOM";
1336 : 3 : break;
4762 fujii@postgresql.org 1337 : 1 : case archDirectory:
1338 : 1 : fmtName = "DIRECTORY";
1339 : 1 : break;
9475 pjw@rhyme.com.au 1340 :UBC 0 : case archTar:
1341 : 0 : fmtName = "TAR";
1342 : 0 : break;
1343 : 0 : default:
1344 : 0 : fmtName = "UNKNOWN";
1345 : : }
1346 : :
3535 peter_e@gmx.net 1347 :CBC 4 : ahprintf(AH, "; Dump Version: %d.%d-%d\n",
1348 : 4 : ARCHIVE_MAJOR(AH->version), ARCHIVE_MINOR(AH->version), ARCHIVE_REV(AH->version));
8652 bruce@momjian.us 1349 : 4 : ahprintf(AH, "; Format: %s\n", fmtName);
203 peter@eisentraut.org 1350 :GNC 4 : ahprintf(AH, "; Integer: %zu bytes\n", AH->intSize);
1351 : 4 : ahprintf(AH, "; Offset: %zu bytes\n", AH->offSize);
7906 tgl@sss.pgh.pa.us 1352 [ + - ]:CBC 4 : if (AH->archiveRemoteVersion)
1353 : 4 : ahprintf(AH, "; Dumped from database version: %s\n",
1354 : : AH->archiveRemoteVersion);
1355 [ + - ]: 4 : if (AH->archiveDumpVersion)
1356 : 4 : ahprintf(AH, "; Dumped by pg_dump version: %s\n",
1357 : : AH->archiveDumpVersion);
1358 : :
8652 bruce@momjian.us 1359 : 4 : ahprintf(AH, ";\n;\n; Selected TOC Entries:\n;\n");
1360 : :
5145 tgl@sss.pgh.pa.us 1361 : 4 : curSection = SECTION_PRE_DATA;
6357 andrew@dunslane.net 1362 [ + + ]: 824 : for (te = AH->toc->next; te != AH->toc; te = te->next)
1363 : : {
1364 : : /* This bit must match ProcessArchiveRestoreOptions' marking logic */
5145 tgl@sss.pgh.pa.us 1365 [ + + ]: 820 : if (te->section != SECTION_NONE)
1366 : 664 : curSection = te->section;
784 1367 : 820 : te->reqs = _tocEntryRequired(te, curSection, AH);
1368 : : /* Now, should we print it? */
5145 1369 [ + - ]: 820 : if (ropt->verbose ||
495 jdavis@postgresql.or 1370 [ + + ]: 820 : (te->reqs & (REQ_SCHEMA | REQ_DATA | REQ_STATS)) != 0)
1371 : : {
1372 : : char *sanitized_name;
1373 : : char *sanitized_schema;
1374 : : char *sanitized_owner;
1375 : :
1376 : : /*
1377 : : */
2706 alvherre@alvh.no-ip. 1378 : 800 : sanitized_name = sanitize_line(te->tag, false);
1379 : 800 : sanitized_schema = sanitize_line(te->namespace, true);
1380 : 800 : sanitized_owner = sanitize_line(te->owner, false);
1381 : :
7935 tgl@sss.pgh.pa.us 1382 : 800 : ahprintf(AH, "%d; %u %u %s %s %s %s\n", te->dumpId,
1383 : : te->catalogId.tableoid, te->catalogId.oid,
1384 : : te->desc, sanitized_schema, sanitized_name,
1385 : : sanitized_owner);
1386 : :
3399 1387 : 800 : free(sanitized_name);
1388 : 800 : free(sanitized_schema);
1389 : 800 : free(sanitized_owner);
1390 : : }
6357 andrew@dunslane.net 1391 [ - + - - ]: 820 : if (ropt->verbose && te->nDeps > 0)
1392 : : {
1393 : : int i;
1394 : :
6357 andrew@dunslane.net 1395 :UBC 0 : ahprintf(AH, ";\tdepends on:");
1396 [ # # ]: 0 : for (i = 0; i < te->nDeps; i++)
1397 : 0 : ahprintf(AH, " %d", te->dependencies[i]);
1398 : 0 : ahprintf(AH, "\n");
1399 : : }
1400 : : }
1401 : :
1402 : : /* Enforce strict names checking */
3942 teodor@sigaev.ru 1403 [ - + ]:CBC 4 : if (ropt->strict_names)
3942 teodor@sigaev.ru 1404 :UBC 0 : StrictNamesCheck(ropt);
1405 : :
9231 bruce@momjian.us 1406 [ - + ]:CBC 4 : if (ropt->filename)
5638 tgl@sss.pgh.pa.us 1407 :UBC 0 : RestoreOutput(AH, sav);
9492 bruce@momjian.us 1408 :CBC 4 : }
1409 : :
1410 : : /***********
1411 : : * Large Object Archival
1412 : : ***********/
1413 : :
1414 : : /* Called by a dumper to signal start of a LO */
1415 : : int
1303 peter@eisentraut.org 1416 : 85 : StartLO(Archive *AHX, Oid oid)
1417 : : {
9231 bruce@momjian.us 1418 : 85 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
1419 : :
1303 peter@eisentraut.org 1420 [ - + ]: 85 : if (!AH->StartLOPtr)
1544 tgl@sss.pgh.pa.us 1421 :UBC 0 : pg_fatal("large-object output not supported in chosen format");
1422 : :
1303 peter@eisentraut.org 1423 :CBC 85 : AH->StartLOPtr(AH, AH->currToc, oid);
1424 : :
9231 bruce@momjian.us 1425 : 85 : return 1;
1426 : : }
1427 : :
1428 : : /* Called by a dumper to signal end of a LO */
1429 : : int
1303 peter@eisentraut.org 1430 : 85 : EndLO(Archive *AHX, Oid oid)
1431 : : {
9231 bruce@momjian.us 1432 : 85 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
1433 : :
1303 peter@eisentraut.org 1434 [ + - ]: 85 : if (AH->EndLOPtr)
1435 : 85 : AH->EndLOPtr(AH, AH->currToc, oid);
1436 : :
9231 bruce@momjian.us 1437 : 85 : return 1;
1438 : : }
1439 : :
1440 : : /**********
1441 : : * Large Object Restoration
1442 : : **********/
1443 : :
1444 : : /*
1445 : : * Called by a format handler before a group of LOs is restored
1446 : : */
1447 : : void
1303 peter@eisentraut.org 1448 : 16 : StartRestoreLOs(ArchiveHandle *AH)
1449 : : {
3821 tgl@sss.pgh.pa.us 1450 : 16 : RestoreOptions *ropt = AH->public.ropt;
1451 : :
1452 : : /*
1453 : : * LOs must be restored within a transaction block, since we need the LO
1454 : : * handle to stay open while we write it. Establish a transaction unless
1455 : : * there's one being used globally.
1456 : : */
820 1457 [ + - + - ]: 16 : if (!(ropt->single_txn || ropt->txn_size > 0))
1458 : : {
7441 1459 [ - + ]: 16 : if (AH->connection)
4277 alvherre@alvh.no-ip. 1460 :UBC 0 : StartTransaction(&AH->public);
1461 : : else
7441 tgl@sss.pgh.pa.us 1462 :CBC 16 : ahprintf(AH, "BEGIN;\n\n");
1463 : : }
1464 : :
1303 peter@eisentraut.org 1465 : 16 : AH->loCount = 0;
9373 pjw@rhyme.com.au 1466 : 16 : }
1467 : :
1468 : : /*
1469 : : * Called by a format handler after a group of LOs is restored
1470 : : */
1471 : : void
1303 peter@eisentraut.org 1472 : 16 : EndRestoreLOs(ArchiveHandle *AH)
1473 : : {
3821 tgl@sss.pgh.pa.us 1474 : 16 : RestoreOptions *ropt = AH->public.ropt;
1475 : :
820 1476 [ + - + - ]: 16 : if (!(ropt->single_txn || ropt->txn_size > 0))
1477 : : {
7441 1478 [ - + ]: 16 : if (AH->connection)
4277 alvherre@alvh.no-ip. 1479 :UBC 0 : CommitTransaction(&AH->public);
1480 : : else
7441 tgl@sss.pgh.pa.us 1481 :CBC 16 : ahprintf(AH, "COMMIT;\n\n");
1482 : : }
1483 : :
2647 peter@eisentraut.org 1484 : 16 : pg_log_info(ngettext("restored %d large object",
1485 : : "restored %d large objects",
1486 : : AH->loCount),
1487 : : AH->loCount);
9373 pjw@rhyme.com.au 1488 : 16 : }
1489 : :
1490 : :
1491 : : /*
1492 : : * Called by a format handler to initiate restoration of a LO
1493 : : */
1494 : : void
1303 peter@eisentraut.org 1495 : 16 : StartRestoreLO(ArchiveHandle *AH, Oid oid, bool drop)
1496 : : {
1497 : 16 : bool old_lo_style = (AH->version < K_VERS_1_12);
1498 : : Oid loOid;
1499 : :
1500 : 16 : AH->loCount++;
1501 : :
1502 : : /* Initialize the LO Buffer */
820 tgl@sss.pgh.pa.us 1503 [ + + ]: 16 : if (AH->lo_buf == NULL)
1504 : : {
1505 : : /* First time through (in this process) so allocate the buffer */
1506 : 10 : AH->lo_buf_size = LOBBUFSIZE;
579 peter@eisentraut.org 1507 : 10 : AH->lo_buf = pg_malloc(LOBBUFSIZE);
1508 : : }
8833 bruce@momjian.us 1509 : 16 : AH->lo_buf_used = 0;
1510 : :
2647 peter@eisentraut.org 1511 : 16 : pg_log_info("restoring large object with OID %u", oid);
1512 : :
1513 : : /* With an old archive we must do drop and create logic here */
1303 1514 [ - + - - ]: 16 : if (old_lo_style && drop)
1303 peter@eisentraut.org 1515 :UBC 0 : DropLOIfExists(AH, oid);
1516 : :
7679 tgl@sss.pgh.pa.us 1517 [ - + ]:CBC 16 : if (AH->connection)
1518 : : {
1303 peter@eisentraut.org 1519 [ # # ]:UBC 0 : if (old_lo_style)
1520 : : {
5976 tgl@sss.pgh.pa.us 1521 : 0 : loOid = lo_create(AH->connection, oid);
1522 [ # # # # ]: 0 : if (loOid == 0 || loOid != oid)
1544 1523 : 0 : pg_fatal("could not create large object %u: %s",
1524 : : oid, PQerrorMessage(AH->connection));
1525 : : }
7679 1526 : 0 : AH->loFd = lo_open(AH->connection, oid, INV_WRITE);
1527 [ # # ]: 0 : if (AH->loFd == -1)
1544 1528 : 0 : pg_fatal("could not open large object %u: %s",
1529 : : oid, PQerrorMessage(AH->connection));
1530 : : }
1531 : : else
1532 : : {
1303 peter@eisentraut.org 1533 [ - + ]:CBC 16 : if (old_lo_style)
5976 tgl@sss.pgh.pa.us 1534 :UBC 0 : ahprintf(AH, "SELECT pg_catalog.lo_open(pg_catalog.lo_create('%u'), %d);\n",
1535 : : oid, INV_WRITE);
1536 : : else
5976 tgl@sss.pgh.pa.us 1537 :CBC 16 : ahprintf(AH, "SELECT pg_catalog.lo_open('%u', %d);\n",
1538 : : oid, INV_WRITE);
1539 : : }
1540 : :
1303 peter@eisentraut.org 1541 : 16 : AH->writingLO = true;
9475 pjw@rhyme.com.au 1542 : 16 : }
1543 : :
1544 : : void
1303 peter@eisentraut.org 1545 : 16 : EndRestoreLO(ArchiveHandle *AH, Oid oid)
1546 : : {
8798 tgl@sss.pgh.pa.us 1547 [ + + ]: 16 : if (AH->lo_buf_used > 0)
1548 : : {
1549 : : /* Write remaining bytes from the LO buffer */
7679 1550 : 10 : dump_lo_buf(AH);
1551 : : }
1552 : :
1303 peter@eisentraut.org 1553 : 16 : AH->writingLO = false;
1554 : :
7679 tgl@sss.pgh.pa.us 1555 [ - + ]: 16 : if (AH->connection)
1556 : : {
7679 tgl@sss.pgh.pa.us 1557 :UBC 0 : lo_close(AH->connection, AH->loFd);
1558 : 0 : AH->loFd = -1;
1559 : : }
1560 : : else
1561 : : {
6188 tgl@sss.pgh.pa.us 1562 :CBC 16 : ahprintf(AH, "SELECT pg_catalog.lo_close(0);\n\n");
1563 : : }
9475 pjw@rhyme.com.au 1564 : 16 : }
1565 : :
1566 : : /***********
1567 : : * Sorting and Reordering
1568 : : ***********/
1569 : :
1570 : : void
3821 tgl@sss.pgh.pa.us 1571 :UBC 0 : SortTocFromFile(Archive *AHX)
1572 : : {
9231 bruce@momjian.us 1573 : 0 : ArchiveHandle *AH = (ArchiveHandle *) AHX;
3821 tgl@sss.pgh.pa.us 1574 : 0 : RestoreOptions *ropt = AH->public.ropt;
1575 : : FILE *fh;
1576 : : StringInfoData linebuf;
1577 : :
1578 : : /* Allocate space for the 'wanted' array, and init it */
137 michael@paquier.xyz 1579 :UNC 0 : ropt->idWanted = pg_malloc0_array(bool, AH->maxDumpId);
1580 : :
1581 : : /* Setup the file */
9231 bruce@momjian.us 1582 :UBC 0 : fh = fopen(ropt->tocFile, PG_BINARY_R);
1583 [ # # ]: 0 : if (!fh)
1544 tgl@sss.pgh.pa.us 1584 : 0 : pg_fatal("could not open TOC file \"%s\": %m", ropt->tocFile);
1585 : :
2107 1586 : 0 : initStringInfo(&linebuf);
1587 : :
1588 [ # # ]: 0 : while (pg_get_line_buf(fh, &linebuf))
1589 : : {
1590 : : char *cmnt;
1591 : : char *endptr;
1592 : : DumpId id;
1593 : : TocEntry *te;
1594 : :
1595 : : /* Truncate line at comment, if any */
1596 : 0 : cmnt = strchr(linebuf.data, ';');
9231 bruce@momjian.us 1597 [ # # ]: 0 : if (cmnt != NULL)
1598 : : {
1599 : 0 : cmnt[0] = '\0';
2107 tgl@sss.pgh.pa.us 1600 : 0 : linebuf.len = cmnt - linebuf.data;
1601 : : }
1602 : :
1603 : : /* Ignore if all blank */
1604 [ # # ]: 0 : if (strspn(linebuf.data, " \t\r\n") == linebuf.len)
9231 bruce@momjian.us 1605 : 0 : continue;
1606 : :
1607 : : /* Get an ID, check it's valid and not already seen */
2107 tgl@sss.pgh.pa.us 1608 : 0 : id = strtol(linebuf.data, &endptr, 10);
1609 [ # # # # : 0 : if (endptr == linebuf.data || id <= 0 || id > AH->maxDumpId ||
# # ]
7714 1610 [ # # ]: 0 : ropt->idWanted[id - 1])
1611 : : {
2107 1612 : 0 : pg_log_warning("line ignored: %s", linebuf.data);
9231 bruce@momjian.us 1613 : 0 : continue;
1614 : : }
1615 : :
1616 : : /* Find TOC entry */
8242 tgl@sss.pgh.pa.us 1617 : 0 : te = getTocEntryByDumpId(AH, id);
9231 bruce@momjian.us 1618 [ # # ]: 0 : if (!te)
1544 tgl@sss.pgh.pa.us 1619 : 0 : pg_fatal("could not find entry for ID %d",
1620 : : id);
1621 : :
1622 : : /* Mark it wanted */
8242 1623 : 0 : ropt->idWanted[id - 1] = true;
1624 : :
1625 : : /*
1626 : : * Move each item to the end of the list as it is selected, so that
1627 : : * they are placed in the desired order. Any unwanted items will end
1628 : : * up at the front of the list, which may seem unintuitive but it's
1629 : : * what we need. In an ordinary serial restore that makes no
1630 : : * difference, but in a parallel restore we need to mark unrestored
1631 : : * items' dependencies as satisfied before we start examining
1632 : : * restorable items. Otherwise they could have surprising
1633 : : * side-effects on the order in which restorable items actually get
1634 : : * restored.
1635 : : */
2135 peter@eisentraut.org 1636 : 0 : _moveBefore(AH->toc, te);
1637 : : }
1638 : :
2107 tgl@sss.pgh.pa.us 1639 : 0 : pg_free(linebuf.data);
1640 : :
9231 bruce@momjian.us 1641 [ # # ]: 0 : if (fclose(fh) != 0)
1544 tgl@sss.pgh.pa.us 1642 : 0 : pg_fatal("could not close TOC file: %m");
9492 bruce@momjian.us 1643 : 0 : }
1644 : :
1645 : : /**********************
1646 : : * Convenience functions that look like standard IO functions
1647 : : * for writing data when in dump mode.
1648 : : **********************/
1649 : :
1650 : : /* Public */
1651 : : void
9231 bruce@momjian.us 1652 :CBC 24430 : archputs(const char *s, Archive *AH)
1653 : : {
4439 1654 : 24430 : WriteData(AH, s, strlen(s));
9492 1655 : 24430 : }
1656 : :
1657 : : /* Public */
1658 : : int
48 tgl@sss.pgh.pa.us 1659 :GNC 4427 : archprintf(Archive *AH, const char *fmt, ...)
1660 : : {
2834 tgl@sss.pgh.pa.us 1661 :CBC 4427 : int save_errno = errno;
1662 : : char *p;
4632 1663 : 4427 : size_t len = 128; /* initial assumption about buffer size */
1664 : : size_t cnt;
1665 : :
1666 : : for (;;)
9492 bruce@momjian.us 1667 :UBC 0 : {
1668 : : va_list args;
1669 : :
1670 : : /* Allocate work buffer. */
4632 tgl@sss.pgh.pa.us 1671 :CBC 4427 : p = (char *) pg_malloc(len);
1672 : :
1673 : : /* Try to format the data. */
2834 1674 : 4427 : errno = save_errno;
4632 1675 : 4427 : va_start(args, fmt);
1676 : 4427 : cnt = pvsnprintf(p, len, fmt, args);
1677 : 4427 : va_end(args);
1678 : :
1679 [ + - ]: 4427 : if (cnt < len)
1680 : 4427 : break; /* success */
1681 : :
1682 : : /* Release buffer and loop around to try again with larger len. */
4632 tgl@sss.pgh.pa.us 1683 :UBC 0 : free(p);
1684 : 0 : len = cnt;
1685 : : }
1686 : :
9231 bruce@momjian.us 1687 :CBC 4427 : WriteData(AH, p, cnt);
1688 : 4427 : free(p);
4632 tgl@sss.pgh.pa.us 1689 : 4427 : return (int) cnt;
1690 : : }
1691 : :
1692 : :
1693 : : /*******************************
1694 : : * Stuff below here should be 'private' to the archiver routines
1695 : : *******************************/
1696 : :
1697 : : static void
1306 michael@paquier.xyz 1698 : 151 : SetOutput(ArchiveHandle *AH, const char *filename,
1699 : : const pg_compress_specification compression_spec)
1700 : : {
1701 : : CompressFileHandle *CFH;
1702 : : const char *mode;
1223 tomas.vondra@postgre 1703 : 151 : int fn = -1;
1704 : :
9231 bruce@momjian.us 1705 [ + - ]: 151 : if (filename)
1706 : : {
2644 alvherre@alvh.no-ip. 1707 [ - + ]: 151 : if (strcmp(filename, "-") == 0)
2644 alvherre@alvh.no-ip. 1708 :UBC 0 : fn = fileno(stdout);
1709 : : }
9231 bruce@momjian.us 1710 [ # # ]: 0 : else if (AH->FH)
1711 : 0 : fn = fileno(AH->FH);
1712 [ # # ]: 0 : else if (AH->fSpec)
1713 : : {
1714 : 0 : filename = AH->fSpec;
1715 : : }
1716 : : else
1717 : 0 : fn = fileno(stdout);
1718 : :
15 andrew@dunslane.net 1719 [ + + ]:CBC 151 : if (AH->mode == archModeAppend)
1223 tomas.vondra@postgre 1720 : 48 : mode = PG_BINARY_A;
1721 : : else
1722 : 103 : mode = PG_BINARY_W;
1723 : :
1724 : 151 : CFH = InitCompressFileHandle(compression_spec);
1725 : :
1195 1726 [ - + ]: 151 : if (!CFH->open_func(filename, fn, mode, CFH))
1727 : : {
6820 tgl@sss.pgh.pa.us 1728 [ # # ]:UBC 0 : if (filename)
1544 1729 : 0 : pg_fatal("could not open output file \"%s\": %m", filename);
1730 : : else
1731 : 0 : pg_fatal("could not open output file: %m");
1732 : : }
1733 : :
1223 tomas.vondra@postgre 1734 :CBC 151 : AH->OF = CFH;
5638 tgl@sss.pgh.pa.us 1735 : 151 : }
1736 : :
1737 : : static CompressFileHandle *
1738 : 195 : SaveOutput(ArchiveHandle *AH)
1739 : : {
1223 tomas.vondra@postgre 1740 : 195 : return (CompressFileHandle *) AH->OF;
1741 : : }
1742 : :
1743 : : static void
1744 : 151 : RestoreOutput(ArchiveHandle *AH, CompressFileHandle *savedOutput)
1745 : : {
1746 : 151 : errno = 0;
1195 1747 [ - + ]: 151 : if (!EndCompressFileHandle(AH->OF))
1544 tgl@sss.pgh.pa.us 1748 :UBC 0 : pg_fatal("could not close output file: %m");
1749 : :
1223 tomas.vondra@postgre 1750 :CBC 151 : AH->OF = savedOutput;
9492 bruce@momjian.us 1751 : 151 : }
1752 : :
1753 : :
1754 : :
1755 : : /*
1756 : : * Print formatted text to the output file (usually stdout).
1757 : : */
1758 : : int
48 tgl@sss.pgh.pa.us 1759 :GNC 224973 : ahprintf(ArchiveHandle *AH, const char *fmt, ...)
1760 : : {
2834 tgl@sss.pgh.pa.us 1761 :CBC 224973 : int save_errno = errno;
1762 : : char *p;
4632 1763 : 224973 : size_t len = 128; /* initial assumption about buffer size */
1764 : : size_t cnt;
1765 : :
1766 : : for (;;)
9258 1767 : 14498 : {
1768 : : va_list args;
1769 : :
1770 : : /* Allocate work buffer. */
4632 1771 : 239471 : p = (char *) pg_malloc(len);
1772 : :
1773 : : /* Try to format the data. */
2834 1774 : 239471 : errno = save_errno;
4632 1775 : 239471 : va_start(args, fmt);
1776 : 239471 : cnt = pvsnprintf(p, len, fmt, args);
1777 : 239471 : va_end(args);
1778 : :
1779 [ + + ]: 239471 : if (cnt < len)
1780 : 224973 : break; /* success */
1781 : :
1782 : : /* Release buffer and loop around to try again with larger len. */
1783 : 14498 : free(p);
1784 : 14498 : len = cnt;
1785 : : }
1786 : :
9231 bruce@momjian.us 1787 : 224973 : ahwrite(p, 1, cnt, AH);
1788 : 224973 : free(p);
4632 tgl@sss.pgh.pa.us 1789 : 224973 : return (int) cnt;
1790 : : }
1791 : :
1792 : : /*
1793 : : * Single place for logic which says 'We are restoring to a direct DB connection'.
1794 : : */
1795 : : static int
9231 bruce@momjian.us 1796 : 2050045 : RestoringToDB(ArchiveHandle *AH)
1797 : : {
3821 tgl@sss.pgh.pa.us 1798 : 2050045 : RestoreOptions *ropt = AH->public.ropt;
1799 : :
1800 [ + - + + : 2050045 : return (ropt && ropt->useDB && AH->connection);
+ - ]
1801 : : }
1802 : :
1803 : : /*
1804 : : * Dump the current contents of the LO data buffer while writing a LO
1805 : : */
1806 : : static void
7679 1807 : 10 : dump_lo_buf(ArchiveHandle *AH)
1808 : : {
1809 [ - + ]: 10 : if (AH->connection)
1810 : : {
1811 : : int res;
1812 : :
7679 tgl@sss.pgh.pa.us 1813 :UBC 0 : res = lo_write(AH->connection, AH->loFd, AH->lo_buf, AH->lo_buf_used);
2081 1814 [ # # ]: 0 : pg_log_debug(ngettext("wrote %zu byte of large object data (result = %d)",
1815 : : "wrote %zu bytes of large object data (result = %d)",
1816 : : AH->lo_buf_used),
1817 : : AH->lo_buf_used, res);
1818 : : /* We assume there are no short writes, only errors */
7679 1819 [ # # ]: 0 : if (res != AH->lo_buf_used)
2081 1820 : 0 : warn_or_exit_horribly(AH, "could not write to large object: %s",
1821 : 0 : PQerrorMessage(AH->connection));
1822 : : }
1823 : : else
1824 : : {
6174 tgl@sss.pgh.pa.us 1825 :CBC 10 : PQExpBuffer buf = createPQExpBuffer();
1826 : :
1827 : 10 : appendByteaLiteralAHX(buf,
1828 : : (const unsigned char *) AH->lo_buf,
1829 : : AH->lo_buf_used,
1830 : : AH);
1831 : :
1832 : : /* Hack: turn off writingLO so ahwrite doesn't recurse to here */
1303 peter@eisentraut.org 1833 : 10 : AH->writingLO = false;
6174 tgl@sss.pgh.pa.us 1834 : 10 : ahprintf(AH, "SELECT pg_catalog.lowrite(0, %s);\n", buf->data);
1303 peter@eisentraut.org 1835 : 10 : AH->writingLO = true;
1836 : :
6174 tgl@sss.pgh.pa.us 1837 : 10 : destroyPQExpBuffer(buf);
1838 : : }
7679 1839 : 10 : AH->lo_buf_used = 0;
1840 : 10 : }
1841 : :
1842 : :
1843 : : /*
1844 : : * Write buffer to the output file (usually stdout). This is used for
1845 : : * outputting 'restore' scripts etc. It is even possible for an archive
1846 : : * format to create a custom output routine to 'fake' a restore if it
1847 : : * wants to generate a script (see TAR output).
1848 : : */
1849 : : void
9231 bruce@momjian.us 1850 : 2047381 : ahwrite(const void *ptr, size_t size, size_t nmemb, ArchiveHandle *AH)
1851 : : {
4438 1852 : 2047381 : int bytes_written = 0;
1853 : :
1303 peter@eisentraut.org 1854 [ + + ]: 2047381 : if (AH->writingLO)
1855 : : {
7563 bruce@momjian.us 1856 : 13 : size_t remaining = size * nmemb;
1857 : :
7679 tgl@sss.pgh.pa.us 1858 [ - + ]: 13 : while (AH->lo_buf_used + remaining > AH->lo_buf_size)
1859 : : {
7679 tgl@sss.pgh.pa.us 1860 :UBC 0 : size_t avail = AH->lo_buf_size - AH->lo_buf_used;
1861 : :
1862 : 0 : memcpy((char *) AH->lo_buf + AH->lo_buf_used, ptr, avail);
571 peter@eisentraut.org 1863 : 0 : ptr = (const char *) ptr + avail;
7679 tgl@sss.pgh.pa.us 1864 : 0 : remaining -= avail;
1865 : 0 : AH->lo_buf_used += avail;
1866 : 0 : dump_lo_buf(AH);
1867 : : }
1868 : :
7679 tgl@sss.pgh.pa.us 1869 :CBC 13 : memcpy((char *) AH->lo_buf + AH->lo_buf_used, ptr, remaining);
1870 : 13 : AH->lo_buf_used += remaining;
1871 : :
4439 bruce@momjian.us 1872 : 13 : bytes_written = size * nmemb;
1873 : : }
9231 1874 [ + + ]: 2047368 : else if (AH->CustomOutPtr)
3331 1875 : 2369 : bytes_written = AH->CustomOutPtr(AH, ptr, size * nmemb);
1876 : :
1877 : : /*
1878 : : * If we're doing a restore, and it's direct to DB, and we're connected
1879 : : * then send it to the DB.
1880 : : */
1223 tomas.vondra@postgre 1881 [ + + ]: 2044999 : else if (RestoringToDB(AH))
1882 : 6943 : bytes_written = ExecuteSqlCommandBuf(&AH->public, (const char *) ptr, size * nmemb);
1883 : : else
1884 : : {
1885 : 2038056 : CompressFileHandle *CFH = (CompressFileHandle *) AH->OF;
1886 : :
305 dgustafsson@postgres 1887 : 2038056 : CFH->write_func(ptr, size * nmemb, CFH);
1888 : 2038056 : bytes_written = size * nmemb;
1889 : : }
1890 : :
4439 bruce@momjian.us 1891 [ - + ]: 2047381 : if (bytes_written != size * nmemb)
4439 bruce@momjian.us 1892 :UBC 0 : WRITE_ERROR_EXIT;
9231 bruce@momjian.us 1893 :CBC 2047381 : }
1894 : :
1895 : : /* on some error, we may decide to go on... */
1896 : : void
48 tgl@sss.pgh.pa.us 1897 :UNC 0 : warn_or_exit_horribly(ArchiveHandle *AH, const char *fmt, ...)
1898 : : {
1899 : : va_list ap;
1900 : :
7975 bruce@momjian.us 1901 [ # # # # :UBC 0 : switch (AH->stage)
# ]
1902 : : {
1903 : :
7984 1904 : 0 : case STAGE_NONE:
1905 : : /* Do nothing special */
1906 : 0 : break;
1907 : :
1908 : 0 : case STAGE_INITIALIZING:
7975 1909 [ # # ]: 0 : if (AH->stage != AH->lastErrorStage)
1544 tgl@sss.pgh.pa.us 1910 : 0 : pg_log_info("while INITIALIZING:");
7984 bruce@momjian.us 1911 : 0 : break;
1912 : :
1913 : 0 : case STAGE_PROCESSING:
7975 1914 [ # # ]: 0 : if (AH->stage != AH->lastErrorStage)
1544 tgl@sss.pgh.pa.us 1915 : 0 : pg_log_info("while PROCESSING TOC:");
7984 bruce@momjian.us 1916 : 0 : break;
1917 : :
1918 : 0 : case STAGE_FINALIZING:
7975 1919 [ # # ]: 0 : if (AH->stage != AH->lastErrorStage)
1544 tgl@sss.pgh.pa.us 1920 : 0 : pg_log_info("while FINALIZING:");
7984 bruce@momjian.us 1921 : 0 : break;
1922 : : }
7975 1923 [ # # # # ]: 0 : if (AH->currentTE != NULL && AH->currentTE != AH->lastErrorTE)
1924 : : {
1544 tgl@sss.pgh.pa.us 1925 [ # # # # : 0 : pg_log_info("from TOC entry %d; %u %u %s %s %s",
# # ]
1926 : : AH->currentTE->dumpId,
1927 : : AH->currentTE->catalogId.tableoid,
1928 : : AH->currentTE->catalogId.oid,
1929 : : AH->currentTE->desc ? AH->currentTE->desc : "(no desc)",
1930 : : AH->currentTE->tag ? AH->currentTE->tag : "(no tag)",
1931 : : AH->currentTE->owner ? AH->currentTE->owner : "(no owner)");
1932 : : }
7984 bruce@momjian.us 1933 : 0 : AH->lastErrorStage = AH->stage;
1934 : 0 : AH->lastErrorTE = AH->currentTE;
1935 : :
8104 1936 : 0 : va_start(ap, fmt);
1544 tgl@sss.pgh.pa.us 1937 : 0 : pg_log_generic_v(PG_LOG_ERROR, PG_LOG_PRIMARY, fmt, ap);
5215 alvherre@alvh.no-ip. 1938 : 0 : va_end(ap);
1939 : :
7984 bruce@momjian.us 1940 [ # # ]: 0 : if (AH->public.exit_on_error)
5215 alvherre@alvh.no-ip. 1941 : 0 : exit_nicely(1);
1942 : : else
8104 bruce@momjian.us 1943 : 0 : AH->public.n_errors++;
1944 : 0 : }
1945 : :
1946 : : #ifdef NOT_USED
1947 : :
1948 : : static void
1949 : : _moveAfter(ArchiveHandle *AH, TocEntry *pos, TocEntry *te)
1950 : : {
1951 : : /* Unlink te from list */
1952 : : te->prev->next = te->next;
1953 : : te->next->prev = te->prev;
1954 : :
1955 : : /* and insert it after "pos" */
1956 : : te->prev = pos;
1957 : : te->next = pos->next;
1958 : : pos->next->prev = te;
1959 : : pos->next = te;
1960 : : }
1961 : : #endif
1962 : :
1963 : : static void
2135 peter@eisentraut.org 1964 : 0 : _moveBefore(TocEntry *pos, TocEntry *te)
1965 : : {
1966 : : /* Unlink te from list */
9231 bruce@momjian.us 1967 : 0 : te->prev->next = te->next;
1968 : 0 : te->next->prev = te->prev;
1969 : :
1970 : : /* and insert it before "pos" */
1971 : 0 : te->prev = pos->prev;
1972 : 0 : te->next = pos;
1973 : 0 : pos->prev->next = te;
1974 : 0 : pos->prev = te;
9492 1975 : 0 : }
1976 : :
1977 : : /*
1978 : : * Build index arrays for the TOC list
1979 : : *
1980 : : * This should be invoked only after we have created or read in all the TOC
1981 : : * items.
1982 : : *
1983 : : * The arrays are indexed by dump ID (so entry zero is unused). Note that the
1984 : : * array entries run only up to maxDumpId. We might see dependency dump IDs
1985 : : * beyond that (if the dump was partial); so always check the array bound
1986 : : * before trying to touch an array entry.
1987 : : */
1988 : : static void
5146 tgl@sss.pgh.pa.us 1989 :CBC 233 : buildTocEntryArrays(ArchiveHandle *AH)
1990 : : {
1991 : 233 : DumpId maxDumpId = AH->maxDumpId;
1992 : : TocEntry *te;
1993 : :
137 michael@paquier.xyz 1994 :GNC 233 : AH->tocsByDumpId = pg_malloc0_array(TocEntry *, (maxDumpId + 1));
1995 : 233 : AH->tableDataId = pg_malloc0_array(DumpId, (maxDumpId + 1));
1996 : :
6357 andrew@dunslane.net 1997 [ + + ]:CBC 51797 : for (te = AH->toc->next; te != AH->toc; te = te->next)
1998 : : {
1999 : : /* this check is purely paranoia, maxDumpId should be correct */
5146 tgl@sss.pgh.pa.us 2000 [ + - - + ]: 51564 : if (te->dumpId <= 0 || te->dumpId > maxDumpId)
1544 tgl@sss.pgh.pa.us 2001 :UBC 0 : pg_fatal("bad dumpId");
2002 : :
2003 : : /* tocsByDumpId indexes all TOCs by their dump ID */
5146 tgl@sss.pgh.pa.us 2004 :CBC 51564 : AH->tocsByDumpId[te->dumpId] = te;
2005 : :
2006 : : /*
2007 : : * tableDataId provides the TABLE DATA item's dump ID for each TABLE
2008 : : * TOC entry that has a DATA item. We compute this by reversing the
2009 : : * TABLE DATA item's dependency, knowing that a TABLE DATA item has
2010 : : * just one dependency and it is the TABLE item.
2011 : : */
2012 [ + + + - ]: 51564 : if (strcmp(te->desc, "TABLE DATA") == 0 && te->nDeps > 0)
2013 : : {
2014 : 4992 : DumpId tableId = te->dependencies[0];
2015 : :
2016 : : /*
2017 : : * The TABLE item might not have been in the archive, if this was
2018 : : * a data-only dump; but its dump ID should be less than its data
2019 : : * item's dump ID, so there should be a place for it in the array.
2020 : : */
2021 [ + - - + ]: 4992 : if (tableId <= 0 || tableId > maxDumpId)
1544 tgl@sss.pgh.pa.us 2022 :UBC 0 : pg_fatal("bad table dumpId for TABLE DATA item");
2023 : :
5146 tgl@sss.pgh.pa.us 2024 :CBC 4992 : AH->tableDataId[tableId] = te->dumpId;
2025 : : }
2026 : : }
2027 : 233 : }
2028 : :
2029 : : TocEntry *
2030 : 12404 : getTocEntryByDumpId(ArchiveHandle *AH, DumpId id)
2031 : : {
2032 : : /* build index arrays if we didn't already */
2033 [ + + ]: 12404 : if (AH->tocsByDumpId == NULL)
2034 : 46 : buildTocEntryArrays(AH);
2035 : :
2036 [ + - + - ]: 12404 : if (id > 0 && id <= AH->maxDumpId)
2037 : 12404 : return AH->tocsByDumpId[id];
2038 : :
9231 bruce@momjian.us 2039 :UBC 0 : return NULL;
2040 : : }
2041 : :
2042 : : int
5145 tgl@sss.pgh.pa.us 2043 :CBC 11813 : TocIDRequired(ArchiveHandle *AH, DumpId id)
2044 : : {
8242 2045 : 11813 : TocEntry *te = getTocEntryByDumpId(AH, id);
2046 : :
9231 bruce@momjian.us 2047 [ + + ]: 11813 : if (!te)
2048 : 5416 : return 0;
2049 : :
5145 tgl@sss.pgh.pa.us 2050 : 6397 : return te->reqs;
2051 : : }
2052 : :
2053 : : size_t
7071 magnus@hagander.net 2054 : 8445 : WriteOffset(ArchiveHandle *AH, pgoff_t o, int wasSet)
2055 : : {
2056 : : int off;
2057 : :
2058 : : /* Save the flag */
3218 peter_e@gmx.net 2059 : 8445 : AH->WriteBytePtr(AH, wasSet);
2060 : :
2061 : : /* Write out pgoff_t smallest byte first, prevents endian mismatch */
7071 magnus@hagander.net 2062 [ + + ]: 76005 : for (off = 0; off < sizeof(pgoff_t); off++)
2063 : : {
3218 peter_e@gmx.net 2064 : 67560 : AH->WriteBytePtr(AH, o & 0xFF);
8652 bruce@momjian.us 2065 : 67560 : o >>= 8;
2066 : : }
7071 magnus@hagander.net 2067 : 8445 : return sizeof(pgoff_t) + 1;
2068 : : }
2069 : :
2070 : : int
229 michael@paquier.xyz 2071 :GNC 6806 : ReadOffset(ArchiveHandle *AH, pgoff_t *o)
2072 : : {
2073 : : int i;
2074 : : int off;
2075 : : int offsetFlg;
2076 : :
2077 : : /* Initialize to zero */
8652 bruce@momjian.us 2078 :CBC 6806 : *o = 0;
2079 : :
2080 : : /* Check for old version */
2081 [ - + ]: 6806 : if (AH->version < K_VERS_1_7)
2082 : : {
2083 : : /* Prior versions wrote offsets using WriteInt */
8652 bruce@momjian.us 2084 :UBC 0 : i = ReadInt(AH);
2085 : : /* -1 means not set */
2086 [ # # ]: 0 : if (i < 0)
8366 2087 : 0 : return K_OFFSET_POS_NOT_SET;
8652 2088 [ # # ]: 0 : else if (i == 0)
8366 2089 : 0 : return K_OFFSET_NO_DATA;
2090 : :
2091 : : /* Cast to pgoff_t because it was written as an int. */
7071 magnus@hagander.net 2092 : 0 : *o = (pgoff_t) i;
8652 bruce@momjian.us 2093 : 0 : return K_OFFSET_POS_SET;
2094 : : }
2095 : :
2096 : : /*
2097 : : * Read the flag indicating the state of the data pointer. Check if valid
2098 : : * and die if not.
2099 : : *
2100 : : * This used to be handled by a negative or zero pointer, now we use an
2101 : : * extra byte specifically for the state.
2102 : : */
3218 peter_e@gmx.net 2103 :CBC 6806 : offsetFlg = AH->ReadBytePtr(AH) & 0xFF;
2104 : :
8652 bruce@momjian.us 2105 [ + - ]: 6806 : switch (offsetFlg)
2106 : : {
2107 : 6806 : case K_OFFSET_POS_NOT_SET:
2108 : : case K_OFFSET_NO_DATA:
2109 : : case K_OFFSET_POS_SET:
2110 : :
8366 2111 : 6806 : break;
2112 : :
8652 bruce@momjian.us 2113 :UBC 0 : default:
1544 tgl@sss.pgh.pa.us 2114 : 0 : pg_fatal("unexpected data offset flag %d", offsetFlg);
2115 : : }
2116 : :
2117 : : /*
2118 : : * Read the bytes
2119 : : */
8652 bruce@momjian.us 2120 [ + + ]:CBC 61254 : for (off = 0; off < AH->offSize; off++)
2121 : : {
7071 magnus@hagander.net 2122 [ + - ]: 54448 : if (off < sizeof(pgoff_t))
3218 peter_e@gmx.net 2123 : 54448 : *o |= ((pgoff_t) (AH->ReadBytePtr(AH))) << (off * 8);
2124 : : else
2125 : : {
3218 peter_e@gmx.net 2126 [ # # ]:UBC 0 : if (AH->ReadBytePtr(AH) != 0)
1544 tgl@sss.pgh.pa.us 2127 : 0 : pg_fatal("file offset in dump file is too large");
2128 : : }
2129 : : }
2130 : :
8652 bruce@momjian.us 2131 :CBC 6806 : return offsetFlg;
2132 : : }
2133 : :
2134 : : size_t
9231 2135 : 186164 : WriteInt(ArchiveHandle *AH, int i)
2136 : : {
2137 : : int b;
2138 : :
2139 : : /*
2140 : : * This is a bit yucky, but I don't want to make the binary format very
2141 : : * dependent on representation, and not knowing much about it, I write out
2142 : : * a sign byte. If you change this, don't forget to change the file
2143 : : * version #, and modify ReadInt to read the new format AS WELL AS the old
2144 : : * formats.
2145 : : */
2146 : :
2147 : : /* SIGN byte */
2148 [ + + ]: 186164 : if (i < 0)
2149 : : {
3218 peter_e@gmx.net 2150 : 45066 : AH->WriteBytePtr(AH, 1);
9475 pjw@rhyme.com.au 2151 : 45066 : i = -i;
2152 : : }
2153 : : else
3218 peter_e@gmx.net 2154 : 141098 : AH->WriteBytePtr(AH, 0);
2155 : :
9231 bruce@momjian.us 2156 [ + + ]: 930820 : for (b = 0; b < AH->intSize; b++)
2157 : : {
3218 peter_e@gmx.net 2158 : 744656 : AH->WriteBytePtr(AH, i & 0xFF);
8798 tgl@sss.pgh.pa.us 2159 : 744656 : i >>= 8;
2160 : : }
2161 : :
9231 bruce@momjian.us 2162 : 186164 : return AH->intSize + 1;
2163 : : }
2164 : :
2165 : : int
2166 : 163246 : ReadInt(ArchiveHandle *AH)
2167 : : {
2168 : 163246 : int res = 0;
2169 : : int bv,
2170 : : b;
2171 : 163246 : int sign = 0; /* Default positive */
2172 : 163246 : int bitShift = 0;
2173 : :
2174 [ + - ]: 163246 : if (AH->version > K_VERS_1_0)
2175 : : /* Read a sign byte */
3218 peter_e@gmx.net 2176 : 163246 : sign = AH->ReadBytePtr(AH);
2177 : :
9231 bruce@momjian.us 2178 [ + + ]: 816230 : for (b = 0; b < AH->intSize; b++)
2179 : : {
3218 peter_e@gmx.net 2180 : 652984 : bv = AH->ReadBytePtr(AH) & 0xFF;
9475 pjw@rhyme.com.au 2181 [ + + ]: 652984 : if (bv != 0)
2182 : 155715 : res = res + (bv << bitShift);
2183 : 652984 : bitShift += 8;
2184 : : }
2185 : :
9231 bruce@momjian.us 2186 [ + + ]: 163246 : if (sign)
2187 : 39139 : res = -res;
2188 : :
2189 : 163246 : return res;
2190 : : }
2191 : :
2192 : : size_t
9221 pjw@rhyme.com.au 2193 : 145340 : WriteStr(ArchiveHandle *AH, const char *c)
2194 : : {
2195 : : size_t res;
2196 : :
9475 2197 [ + + ]: 145340 : if (c)
2198 : : {
4438 bruce@momjian.us 2199 : 100274 : int len = strlen(c);
2200 : :
4439 2201 : 100274 : res = WriteInt(AH, len);
3218 peter_e@gmx.net 2202 : 100274 : AH->WriteBufPtr(AH, c, len);
4439 bruce@momjian.us 2203 : 100274 : res += len;
2204 : : }
2205 : : else
9475 pjw@rhyme.com.au 2206 : 45066 : res = WriteInt(AH, -1);
2207 : :
9231 bruce@momjian.us 2208 : 145340 : return res;
2209 : : }
2210 : :
2211 : : char *
2212 : 127730 : ReadStr(ArchiveHandle *AH)
2213 : : {
2214 : : char *buf;
2215 : : int l;
2216 : :
2217 : 127730 : l = ReadInt(AH);
6903 tgl@sss.pgh.pa.us 2218 [ + + ]: 127730 : if (l < 0)
9475 pjw@rhyme.com.au 2219 : 39139 : buf = NULL;
2220 : : else
2221 : : {
5331 bruce@momjian.us 2222 : 88591 : buf = (char *) pg_malloc(l + 1);
579 peter@eisentraut.org 2223 : 88591 : AH->ReadBufPtr(AH, buf, l);
2224 : :
9475 pjw@rhyme.com.au 2225 : 88591 : buf[l] = '\0';
2226 : : }
2227 : :
9231 bruce@momjian.us 2228 : 127730 : return buf;
2229 : : }
2230 : :
2231 : : static bool
1223 tomas.vondra@postgre 2232 : 11 : _fileExistsInDirectory(const char *dir, const char *filename)
2233 : : {
2234 : : struct stat st;
2235 : : char buf[MAXPGPATH];
2236 : :
2237 [ - + ]: 11 : if (snprintf(buf, MAXPGPATH, "%s/%s", dir, filename) >= MAXPGPATH)
1223 tomas.vondra@postgre 2238 :UBC 0 : pg_fatal("directory name too long: \"%s\"", dir);
2239 : :
1223 tomas.vondra@postgre 2240 [ + + + - ]:CBC 11 : return (stat(buf, &st) == 0 && S_ISREG(st.st_mode));
2241 : : }
2242 : :
2243 : : static int
9231 bruce@momjian.us 2244 : 50 : _discoverArchiveFormat(ArchiveHandle *AH)
2245 : : {
2246 : : FILE *fh;
2247 : : char sig[6]; /* More than enough */
2248 : : size_t cnt;
2249 : 50 : int wantClose = 0;
2250 : :
2647 peter@eisentraut.org 2251 [ - + ]: 50 : pg_log_debug("attempting to ascertain archive format");
2252 : :
1475 2253 : 50 : free(AH->lookahead);
2254 : :
1916 tgl@sss.pgh.pa.us 2255 : 50 : AH->readHeader = 0;
9475 pjw@rhyme.com.au 2256 : 50 : AH->lookaheadSize = 512;
5019 tgl@sss.pgh.pa.us 2257 : 50 : AH->lookahead = pg_malloc0(512);
9475 pjw@rhyme.com.au 2258 : 50 : AH->lookaheadLen = 0;
2259 : 50 : AH->lookaheadPos = 0;
2260 : :
9231 bruce@momjian.us 2261 [ + - ]: 50 : if (AH->fSpec)
2262 : : {
2263 : : struct stat st;
2264 : :
9475 pjw@rhyme.com.au 2265 : 50 : wantClose = 1;
2266 : :
2267 : : /*
2268 : : * Check if the specified archive is a directory. If so, check if
2269 : : * there's a "toc.dat" (or "toc.dat.{gz,lz4,zst}") file in it.
2270 : : */
5637 heikki.linnakangas@i 2271 [ + - + + ]: 50 : if (stat(AH->fSpec, &st) == 0 && S_ISDIR(st.st_mode))
2272 : : {
1223 tomas.vondra@postgre 2273 : 10 : AH->format = archDirectory;
2274 [ + + ]: 10 : if (_fileExistsInDirectory(AH->fSpec, "toc.dat"))
5637 heikki.linnakangas@i 2275 : 10 : return AH->format;
2276 : : #ifdef HAVE_LIBZ
1223 tomas.vondra@postgre 2277 [ + - ]:GBC 1 : if (_fileExistsInDirectory(AH->fSpec, "toc.dat.gz"))
5637 heikki.linnakangas@i 2278 : 1 : return AH->format;
2279 : : #endif
2280 : : #ifdef USE_LZ4
1223 tomas.vondra@postgre 2281 [ # # ]:UBC 0 : if (_fileExistsInDirectory(AH->fSpec, "toc.dat.lz4"))
2282 : 0 : return AH->format;
2283 : : #endif
2284 : : #ifdef USE_ZSTD
2285 : : if (_fileExistsInDirectory(AH->fSpec, "toc.dat.zst"))
2286 : : return AH->format;
2287 : : #endif
1544 tgl@sss.pgh.pa.us 2288 : 0 : pg_fatal("directory \"%s\" does not appear to be a valid archive (\"toc.dat\" does not exist)",
2289 : : AH->fSpec);
2290 : : fh = NULL; /* keep compiler quiet */
2291 : : }
2292 : : else
2293 : : {
5637 heikki.linnakangas@i 2294 :CBC 40 : fh = fopen(AH->fSpec, PG_BINARY_R);
2295 [ - + ]: 40 : if (!fh)
1544 tgl@sss.pgh.pa.us 2296 :UBC 0 : pg_fatal("could not open input file \"%s\": %m", AH->fSpec);
2297 : : }
2298 : : }
2299 : : else
2300 : : {
9475 pjw@rhyme.com.au 2301 : 0 : fh = stdin;
6820 tgl@sss.pgh.pa.us 2302 [ # # ]: 0 : if (!fh)
1544 2303 : 0 : pg_fatal("could not open input file: %m");
2304 : : }
2305 : :
4439 bruce@momjian.us 2306 [ - + ]:CBC 40 : if ((cnt = fread(sig, 1, 5, fh)) != 5)
2307 : : {
9134 peter_e@gmx.net 2308 [ # # ]:UBC 0 : if (ferror(fh))
1544 tgl@sss.pgh.pa.us 2309 : 0 : pg_fatal("could not read input file: %m");
2310 : : else
203 peter@eisentraut.org 2311 :UNC 0 : pg_fatal("input file is too short (read %zu, expected 5)", cnt);
2312 : : }
2313 : :
2314 : : /* Save it, just in case we need it later */
4175 tgl@sss.pgh.pa.us 2315 :CBC 40 : memcpy(&AH->lookahead[0], sig, 5);
9475 pjw@rhyme.com.au 2316 : 40 : AH->lookaheadLen = 5;
2317 : :
9231 bruce@momjian.us 2318 [ + + ]: 40 : if (strncmp(sig, "PGDMP", 5) == 0)
2319 : : {
2320 : : /* It's custom format, stop here */
1916 tgl@sss.pgh.pa.us 2321 : 39 : AH->format = archCustom;
2322 : 39 : AH->readHeader = 1;
2323 : : }
2324 : : else
2325 : : {
2326 : : /*
2327 : : * *Maybe* we have a tar archive format file or a text dump ... So,
2328 : : * read first 512 byte header...
2329 : : */
9475 pjw@rhyme.com.au 2330 : 1 : cnt = fread(&AH->lookahead[AH->lookaheadLen], 1, 512 - AH->lookaheadLen, fh);
2331 : : /* read failure is checked below */
2332 : 1 : AH->lookaheadLen += cnt;
2333 : :
5292 andrew@dunslane.net 2334 [ + - ]: 1 : if (AH->lookaheadLen >= strlen(TEXT_DUMPALL_HEADER) &&
2335 [ + - ]: 1 : (strncmp(AH->lookahead, TEXT_DUMP_HEADER, strlen(TEXT_DUMP_HEADER)) == 0 ||
2336 [ - + ]: 1 : strncmp(AH->lookahead, TEXT_DUMPALL_HEADER, strlen(TEXT_DUMPALL_HEADER)) == 0))
2337 : : {
2338 : : /*
2339 : : * looks like it's probably a text format dump. so suggest they
2340 : : * try psql
2341 : : */
1544 tgl@sss.pgh.pa.us 2342 :UBC 0 : pg_fatal("input file appears to be a text format dump. Please use psql.");
2343 : : }
2344 : :
4438 bruce@momjian.us 2345 [ - + ]:CBC 1 : if (AH->lookaheadLen != 512)
2346 : : {
4438 bruce@momjian.us 2347 [ # # ]:UBC 0 : if (feof(fh))
1544 tgl@sss.pgh.pa.us 2348 : 0 : pg_fatal("input file does not appear to be a valid archive (too short?)");
2349 : : else
4438 bruce@momjian.us 2350 [ # # ]: 0 : READ_ERROR_EXIT(fh);
2351 : : }
2352 : :
9475 pjw@rhyme.com.au 2353 [ - + ]:CBC 1 : if (!isValidTarHeader(AH->lookahead))
89 tgl@sss.pgh.pa.us 2354 :UBC 0 : pg_fatal("input file does not appear to be a valid tar archive");
2355 : :
9475 pjw@rhyme.com.au 2356 :CBC 1 : AH->format = archTar;
2357 : : }
2358 : :
2359 : : /* Close the file if we opened it */
9231 bruce@momjian.us 2360 [ + - ]: 40 : if (wantClose)
2361 : : {
9300 pjw@rhyme.com.au 2362 [ - + ]: 40 : if (fclose(fh) != 0)
1544 tgl@sss.pgh.pa.us 2363 :UBC 0 : pg_fatal("could not close input file: %m");
2364 : : /* Forget lookahead, since we'll re-read header after re-opening */
1916 tgl@sss.pgh.pa.us 2365 :CBC 40 : AH->readHeader = 0;
2366 : 40 : AH->lookaheadLen = 0;
2367 : : }
2368 : :
9231 bruce@momjian.us 2369 : 40 : return AH->format;
2370 : : }
2371 : :
2372 : :
2373 : : /*
2374 : : * Allocate an archive handle
2375 : : */
2376 : : static ArchiveHandle *
2377 : 276 : _allocAH(const char *FileSpec, const ArchiveFormat fmt,
2378 : : const pg_compress_specification compression_spec,
2379 : : bool dosync, ArchiveMode mode,
2380 : : SetupWorkerPtrType setupWorkerPtr, DataDirSyncMethod sync_method)
2381 : : {
2382 : : ArchiveHandle *AH;
2383 : : CompressFileHandle *CFH;
1223 tomas.vondra@postgre 2384 : 276 : pg_compress_specification out_compress_spec = {0};
2385 : :
2112 tgl@sss.pgh.pa.us 2386 [ - + - - ]: 276 : pg_log_debug("allocating AH for %s, format %d",
2387 : : FileSpec ? FileSpec : "(stdio)", fmt);
2388 : :
137 michael@paquier.xyz 2389 :GNC 276 : AH = pg_malloc0_object(ArchiveHandle);
2390 : :
3535 peter_e@gmx.net 2391 :CBC 276 : AH->version = K_VERS_SELF;
2392 : :
2393 : : /* initialize for backwards compatible string processing */
6835 tgl@sss.pgh.pa.us 2394 : 276 : AH->public.encoding = 0; /* PG_SQL_ASCII */
15 andrew@dunslane.net 2395 : 276 : AH->public.std_strings = false;
2396 : :
2397 : : /* sql error handling */
7338 tgl@sss.pgh.pa.us 2398 : 276 : AH->public.exit_on_error = true;
2399 : 276 : AH->public.n_errors = 0;
2400 : :
5970 2401 : 276 : AH->archiveDumpVersion = PG_VERSION;
2402 : :
9475 pjw@rhyme.com.au 2403 : 276 : AH->createDate = time(NULL);
2404 : :
9231 bruce@momjian.us 2405 : 276 : AH->intSize = sizeof(int);
7071 magnus@hagander.net 2406 : 276 : AH->offSize = sizeof(pgoff_t);
9231 bruce@momjian.us 2407 [ + + ]: 276 : if (FileSpec)
2408 : : {
5331 2409 : 258 : AH->fSpec = pg_strdup(FileSpec);
2410 : :
2411 : : /*
2412 : : * Not used; maybe later....
2413 : : *
2414 : : * AH->workDir = pg_strdup(FileSpec); for(i=strlen(FileSpec) ; i > 0 ;
2415 : : * i--) if (AH->workDir[i-1] == '/')
2416 : : */
2417 : : }
2418 : : else
9475 pjw@rhyme.com.au 2419 : 18 : AH->fSpec = NULL;
2420 : :
6357 andrew@dunslane.net 2421 : 276 : AH->currUser = NULL; /* unknown */
2422 : 276 : AH->currSchema = NULL; /* ditto */
2423 : 276 : AH->currTablespace = NULL; /* ditto */
2596 tgl@sss.pgh.pa.us 2424 : 276 : AH->currTableAm = NULL; /* ditto */
2425 : :
137 michael@paquier.xyz 2426 :GNC 276 : AH->toc = pg_malloc0_object(TocEntry);
2427 : :
9231 bruce@momjian.us 2428 :CBC 276 : AH->toc->next = AH->toc;
2429 : 276 : AH->toc->prev = AH->toc;
2430 : :
2431 : 276 : AH->mode = mode;
1306 michael@paquier.xyz 2432 : 276 : AH->compression_spec = compression_spec;
3387 andrew@dunslane.net 2433 : 276 : AH->dosync = dosync;
1028 nathan@postgresql.or 2434 : 276 : AH->sync_method = sync_method;
2435 : :
5289 tgl@sss.pgh.pa.us 2436 : 276 : memset(&(AH->sqlparse), 0, sizeof(AH->sqlparse));
2437 : :
2438 : : /* Open stdout with no compression for AH output handle */
1223 tomas.vondra@postgre 2439 : 276 : out_compress_spec.algorithm = PG_COMPRESSION_NONE;
2440 : 276 : CFH = InitCompressFileHandle(out_compress_spec);
1195 2441 [ - + ]: 276 : if (!CFH->open_func(NULL, fileno(stdout), PG_BINARY_A, CFH))
1223 tomas.vondra@postgre 2442 :UBC 0 : pg_fatal("could not open stdout for appending: %m");
1223 tomas.vondra@postgre 2443 :CBC 276 : AH->OF = CFH;
2444 : :
2445 : : /*
2446 : : * On Windows, we need to use binary mode to read/write non-text files,
2447 : : * which include all archive formats as well as compressed plain text.
2448 : : * Force stdin/stdout into binary mode if that is what we are using.
2449 : : */
2450 : : #ifdef WIN32
2451 : : if ((fmt != archNull || compression_spec.algorithm != PG_COMPRESSION_NONE) &&
2452 : : (AH->fSpec == NULL || strcmp(AH->fSpec, "") == 0))
2453 : : {
2454 : : if (mode == archModeWrite)
2455 : : _setmode(fileno(stdout), O_BINARY);
2456 : : else
2457 : : _setmode(fileno(stdin), O_BINARY);
2458 : : }
2459 : : #endif
2460 : :
4846 andrew@dunslane.net 2461 : 276 : AH->SetupWorkerPtr = setupWorkerPtr;
2462 : :
9231 bruce@momjian.us 2463 [ + + ]: 276 : if (fmt == archUnknown)
9475 pjw@rhyme.com.au 2464 : 50 : AH->format = _discoverArchiveFormat(AH);
2465 : : else
2466 : 226 : AH->format = fmt;
2467 : :
9231 bruce@momjian.us 2468 [ + + + + : 276 : switch (AH->format)
- ]
2469 : : {
9475 pjw@rhyme.com.au 2470 : 103 : case archCustom:
2471 : 103 : InitArchiveFmt_Custom(AH);
2472 : 103 : break;
2473 : :
2474 : 147 : case archNull:
2475 : 147 : InitArchiveFmt_Null(AH);
2476 : 147 : break;
2477 : :
5637 heikki.linnakangas@i 2478 : 21 : case archDirectory:
2479 : 21 : InitArchiveFmt_Directory(AH);
2480 : 21 : break;
2481 : :
9475 pjw@rhyme.com.au 2482 : 5 : case archTar:
2483 : 5 : InitArchiveFmt_Tar(AH);
2484 : 4 : break;
2485 : :
9475 pjw@rhyme.com.au 2486 :UBC 0 : default:
439 fujii@postgresql.org 2487 : 0 : pg_fatal("unrecognized file format \"%d\"", AH->format);
2488 : : }
2489 : :
9231 bruce@momjian.us 2490 :CBC 275 : return AH;
2491 : : }
2492 : :
2493 : : /*
2494 : : * Write out all data (tables & LOs)
2495 : : */
2496 : : void
3821 tgl@sss.pgh.pa.us 2497 : 63 : WriteDataChunks(ArchiveHandle *AH, ParallelState *pstate)
2498 : : {
2499 : : TocEntry *te;
2500 : :
2846 2501 [ + + + + ]: 63 : if (pstate && pstate->numWorkers > 1)
9231 bruce@momjian.us 2502 : 8 : {
2503 : : /*
2504 : : * In parallel mode, this code runs in the leader process. We
2505 : : * construct an array of candidate TEs, then sort it into decreasing
2506 : : * size order, then dispatch each TE to a data-transfer worker. By
2507 : : * dumping larger tables first, we avoid getting into a situation
2508 : : * where we're down to one job and it's big, losing parallelism.
2509 : : */
2510 : : TocEntry **tes;
2511 : : int ntes;
2512 : :
137 michael@paquier.xyz 2513 :GNC 8 : tes = pg_malloc_array(TocEntry *, AH->tocCount);
2846 tgl@sss.pgh.pa.us 2514 :CBC 8 : ntes = 0;
2515 [ + + ]: 602 : for (te = AH->toc->next; te != AH->toc; te = te->next)
2516 : : {
2517 : : /* Consider only TEs with dataDumper functions ... */
2518 [ + + ]: 594 : if (!te->dataDumper)
2519 : 525 : continue;
2520 : : /* ... and ignore ones not enabled for dump */
2521 [ - + ]: 69 : if ((te->reqs & REQ_DATA) == 0)
2846 tgl@sss.pgh.pa.us 2522 :UBC 0 : continue;
2523 : :
2846 tgl@sss.pgh.pa.us 2524 :CBC 69 : tes[ntes++] = te;
2525 : : }
2526 : :
2527 [ + + ]: 8 : if (ntes > 1)
1015 nathan@postgresql.or 2528 : 7 : qsort(tes, ntes, sizeof(TocEntry *), TocEntrySizeCompareQsort);
2529 : :
2846 tgl@sss.pgh.pa.us 2530 [ + + ]: 77 : for (int i = 0; i < ntes; i++)
2531 : 69 : DispatchJobForTocEntry(AH, pstate, tes[i], ACT_DUMP,
2532 : : mark_dump_job_done, NULL);
2533 : :
2534 : 8 : pg_free(tes);
2535 : :
2536 : : /* Now wait for workers to finish. */
3563 2537 : 8 : WaitForWorkers(AH, pstate, WFW_ALL_IDLE);
2538 : : }
2539 : : else
2540 : : {
2541 : : /* Non-parallel mode: just dump all candidate TEs sequentially. */
2846 2542 [ + + ]: 7252 : for (te = AH->toc->next; te != AH->toc; te = te->next)
2543 : : {
2544 : : /* Must have same filter conditions as above */
2545 [ + + ]: 7197 : if (!te->dataDumper)
2546 : 6834 : continue;
2547 [ + + ]: 363 : if ((te->reqs & REQ_DATA) == 0)
2548 : 7 : continue;
2549 : :
2550 : 356 : WriteDataChunksForTocEntry(AH, te);
2551 : : }
2552 : : }
4846 andrew@dunslane.net 2553 : 63 : }
2554 : :
2555 : :
2556 : : /*
2557 : : * Callback function that's invoked in the leader process after a step has
2558 : : * been parallel dumped.
2559 : : *
2560 : : * We don't need to do anything except check for worker failure.
2561 : : */
2562 : : static void
3563 tgl@sss.pgh.pa.us 2563 : 69 : mark_dump_job_done(ArchiveHandle *AH,
2564 : : TocEntry *te,
2565 : : int status,
2566 : : void *callback_data)
2567 : : {
2647 peter@eisentraut.org 2568 : 69 : pg_log_info("finished item %d %s %s",
2569 : : te->dumpId, te->desc, te->tag);
2570 : :
3563 tgl@sss.pgh.pa.us 2571 [ - + ]: 69 : if (status != 0)
1544 tgl@sss.pgh.pa.us 2572 :UBC 0 : pg_fatal("worker process failed: exit code %d",
2573 : : status);
3563 tgl@sss.pgh.pa.us 2574 :CBC 69 : }
2575 : :
2576 : :
2577 : : void
3821 2578 : 425 : WriteDataChunksForTocEntry(ArchiveHandle *AH, TocEntry *te)
2579 : : {
2580 : : StartDataPtrType startPtr;
2581 : : EndDataPtrType endPtr;
2582 : :
4846 andrew@dunslane.net 2583 : 425 : AH->currToc = te;
2584 : :
2585 [ + + ]: 425 : if (strcmp(te->desc, "BLOBS") == 0)
2586 : : {
1303 peter@eisentraut.org 2587 : 21 : startPtr = AH->StartLOsPtr;
2588 : 21 : endPtr = AH->EndLOsPtr;
2589 : : }
2590 : : else
2591 : : {
4846 andrew@dunslane.net 2592 : 404 : startPtr = AH->StartDataPtr;
2593 : 404 : endPtr = AH->EndDataPtr;
2594 : : }
2595 : :
2596 [ + - ]: 425 : if (startPtr != NULL)
2597 : 425 : (*startPtr) (AH, te);
2598 : :
2599 : : /*
2600 : : * The user-provided DataDumper routine needs to call AH->WriteData
2601 : : */
3218 peter_e@gmx.net 2602 : 425 : te->dataDumper((Archive *) AH, te->dataDumperArg);
2603 : :
4846 andrew@dunslane.net 2604 [ + - ]: 425 : if (endPtr != NULL)
2605 : 425 : (*endPtr) (AH, te);
2606 : :
2607 : 425 : AH->currToc = NULL;
9492 bruce@momjian.us 2608 : 425 : }
2609 : :
2610 : : void
9231 2611 : 74 : WriteToc(ArchiveHandle *AH)
2612 : : {
2613 : : TocEntry *te;
2614 : : char workbuf[32];
2615 : : int tocCount;
2616 : : int i;
2617 : :
2618 : : /* count entries that will actually be dumped */
5145 tgl@sss.pgh.pa.us 2619 : 74 : tocCount = 0;
2620 [ + + ]: 9951 : for (te = AH->toc->next; te != AH->toc; te = te->next)
2621 : : {
495 jdavis@postgresql.or 2622 [ + + ]: 9877 : if ((te->reqs & (REQ_SCHEMA | REQ_DATA | REQ_STATS | REQ_SPECIAL)) != 0)
5145 tgl@sss.pgh.pa.us 2623 : 9869 : tocCount++;
2624 : : }
2625 : :
2626 : : /* printf("%d TOC Entries to save\n", tocCount); */
2627 : :
2628 : 74 : WriteInt(AH, tocCount);
2629 : :
8242 2630 [ + + ]: 9951 : for (te = AH->toc->next; te != AH->toc; te = te->next)
2631 : : {
495 jdavis@postgresql.or 2632 [ + + ]: 9877 : if ((te->reqs & (REQ_SCHEMA | REQ_DATA | REQ_STATS | REQ_SPECIAL)) == 0)
5145 tgl@sss.pgh.pa.us 2633 : 8 : continue;
2634 : :
8242 2635 : 9869 : WriteInt(AH, te->dumpId);
9231 bruce@momjian.us 2636 : 9869 : WriteInt(AH, te->dataDumper ? 1 : 0);
2637 : :
2638 : : /* OID is recorded as a string for historical reasons */
8242 tgl@sss.pgh.pa.us 2639 : 9869 : sprintf(workbuf, "%u", te->catalogId.tableoid);
2640 : 9869 : WriteStr(AH, workbuf);
2641 : 9869 : sprintf(workbuf, "%u", te->catalogId.oid);
2642 : 9869 : WriteStr(AH, workbuf);
2643 : :
8762 bruce@momjian.us 2644 : 9869 : WriteStr(AH, te->tag);
9231 2645 : 9869 : WriteStr(AH, te->desc);
6357 andrew@dunslane.net 2646 : 9869 : WriteInt(AH, te->section);
2647 : :
452 nathan@postgresql.or 2648 [ + + ]: 9869 : if (te->defnLen)
2649 : : {
2650 : : /*
2651 : : * defnLen should only be set for custom format's second call to
2652 : : * WriteToc(), which rewrites the TOC in place to update data
2653 : : * offsets. Instead of calling the defnDumper a second time
2654 : : * (which could involve re-executing queries), just skip writing
2655 : : * the entry. While regenerating the definition should
2656 : : * theoretically produce the same result as before, it's expensive
2657 : : * and feels risky.
2658 : : *
2659 : : * The custom format only calls WriteToc() a second time if
2660 : : * fseeko() is usable (see _CloseArchive() in pg_backup_custom.c),
2661 : : * so we can safely use it without checking. For other formats,
2662 : : * we fail because one of our assumptions must no longer hold
2663 : : * true.
2664 : : *
2665 : : * XXX This is a layering violation, but the alternative is an
2666 : : * awkward and complicated callback infrastructure for this
2667 : : * special case. This might be worth revisiting in the future.
2668 : : */
2669 [ - + ]: 369 : if (AH->format != archCustom)
452 nathan@postgresql.or 2670 :UBC 0 : pg_fatal("unexpected TOC entry in WriteToc(): %d %s %s",
2671 : : te->dumpId, te->desc, te->tag);
2672 : :
392 noah@leadboat.com 2673 [ - + ]:CBC 369 : if (fseeko(AH->FH, te->defnLen, SEEK_CUR) != 0)
452 nathan@postgresql.or 2674 :UBC 0 : pg_fatal("error during file seek: %m");
2675 : : }
452 nathan@postgresql.or 2676 [ + + ]:CBC 9500 : else if (te->defnDumper)
2677 : : {
2678 : 1868 : char *defn = te->defnDumper((Archive *) AH, te->defnDumperArg, te);
2679 : :
2680 : 1868 : te->defnLen = WriteStr(AH, defn);
2681 : 1868 : pg_free(defn);
2682 : : }
2683 : : else
2684 : 7632 : WriteStr(AH, te->defn);
2685 : :
9231 bruce@momjian.us 2686 : 9869 : WriteStr(AH, te->dropStmt);
2687 : 9869 : WriteStr(AH, te->copyStmt);
8817 tgl@sss.pgh.pa.us 2688 : 9869 : WriteStr(AH, te->namespace);
7906 2689 : 9869 : WriteStr(AH, te->tablespace);
2673 andres@anarazel.de 2690 : 9869 : WriteStr(AH, te->tableam);
799 michael@paquier.xyz 2691 : 9869 : WriteInt(AH, te->relkind);
9231 bruce@momjian.us 2692 : 9869 : WriteStr(AH, te->owner);
2779 andres@anarazel.de 2693 : 9869 : WriteStr(AH, "false");
2694 : :
2695 : : /* Dump list of dependencies */
8242 tgl@sss.pgh.pa.us 2696 [ + + ]: 25668 : for (i = 0; i < te->nDeps; i++)
2697 : : {
2698 : 15799 : sprintf(workbuf, "%d", te->dependencies[i]);
2699 : 15799 : WriteStr(AH, workbuf);
2700 : : }
9014 bruce@momjian.us 2701 : 9869 : WriteStr(AH, NULL); /* Terminate List */
2702 : :
9231 2703 [ + - ]: 9869 : if (AH->WriteExtraTocPtr)
3218 peter_e@gmx.net 2704 : 9869 : AH->WriteExtraTocPtr(AH, te);
2705 : : }
9492 bruce@momjian.us 2706 : 74 : }
2707 : :
2708 : : void
9231 2709 : 65 : ReadToc(ArchiveHandle *AH)
2710 : : {
2711 : : int i;
2712 : : char *tmp;
2713 : : DumpId *deps;
2714 : : int depIdx;
2715 : : int depSize;
2716 : : TocEntry *te;
2717 : : bool is_supported;
2718 : :
2719 : 65 : AH->tocCount = ReadInt(AH);
8242 tgl@sss.pgh.pa.us 2720 : 65 : AH->maxDumpId = 0;
2721 : :
9231 bruce@momjian.us 2722 [ + + ]: 8679 : for (i = 0; i < AH->tocCount; i++)
2723 : : {
137 michael@paquier.xyz 2724 :GNC 8614 : te = pg_malloc0_object(TocEntry);
8242 tgl@sss.pgh.pa.us 2725 :CBC 8614 : te->dumpId = ReadInt(AH);
2726 : :
2727 [ + + ]: 8614 : if (te->dumpId > AH->maxDumpId)
2728 : 3537 : AH->maxDumpId = te->dumpId;
2729 : :
2730 : : /* Sanity check */
2731 [ - + ]: 8614 : if (te->dumpId <= 0)
1544 tgl@sss.pgh.pa.us 2732 :UBC 0 : pg_fatal("entry ID %d out of range -- perhaps a corrupt TOC",
2733 : : te->dumpId);
2734 : :
9475 pjw@rhyme.com.au 2735 :CBC 8614 : te->hadDumper = ReadInt(AH);
2736 : :
8242 tgl@sss.pgh.pa.us 2737 [ + - ]: 8614 : if (AH->version >= K_VERS_1_8)
2738 : : {
2739 : 8614 : tmp = ReadStr(AH);
2740 : 8614 : sscanf(tmp, "%u", &te->catalogId.tableoid);
2741 : 8614 : free(tmp);
2742 : : }
2743 : : else
8242 tgl@sss.pgh.pa.us 2744 :UBC 0 : te->catalogId.tableoid = InvalidOid;
8242 tgl@sss.pgh.pa.us 2745 :CBC 8614 : tmp = ReadStr(AH);
2746 : 8614 : sscanf(tmp, "%u", &te->catalogId.oid);
2747 : 8614 : free(tmp);
2748 : :
8762 bruce@momjian.us 2749 : 8614 : te->tag = ReadStr(AH);
9475 pjw@rhyme.com.au 2750 : 8614 : te->desc = ReadStr(AH);
2751 : :
6357 andrew@dunslane.net 2752 [ + - ]: 8614 : if (AH->version >= K_VERS_1_11)
2753 : : {
2754 : 8614 : te->section = ReadInt(AH);
2755 : : }
2756 : : else
2757 : : {
2758 : : /*
2759 : : * Rules for pre-8.4 archives wherein pg_dump hasn't classified
2760 : : * the entries into sections. This list need not cover entry
2761 : : * types added later than 8.4.
2762 : : */
6357 andrew@dunslane.net 2763 [ # # ]:UBC 0 : if (strcmp(te->desc, "COMMENT") == 0 ||
6112 tgl@sss.pgh.pa.us 2764 [ # # ]: 0 : strcmp(te->desc, "ACL") == 0 ||
5976 2765 [ # # ]: 0 : strcmp(te->desc, "ACL LANGUAGE") == 0)
6357 andrew@dunslane.net 2766 : 0 : te->section = SECTION_NONE;
2767 [ # # ]: 0 : else if (strcmp(te->desc, "TABLE DATA") == 0 ||
2768 [ # # ]: 0 : strcmp(te->desc, "BLOBS") == 0 ||
2769 [ # # ]: 0 : strcmp(te->desc, "BLOB COMMENTS") == 0)
2770 : 0 : te->section = SECTION_DATA;
2771 [ # # ]: 0 : else if (strcmp(te->desc, "CONSTRAINT") == 0 ||
2772 [ # # ]: 0 : strcmp(te->desc, "CHECK CONSTRAINT") == 0 ||
2773 [ # # ]: 0 : strcmp(te->desc, "FK CONSTRAINT") == 0 ||
2774 [ # # ]: 0 : strcmp(te->desc, "INDEX") == 0 ||
2775 [ # # ]: 0 : strcmp(te->desc, "RULE") == 0 ||
2776 [ # # ]: 0 : strcmp(te->desc, "TRIGGER") == 0)
2777 : 0 : te->section = SECTION_POST_DATA;
2778 : : else
2779 : 0 : te->section = SECTION_PRE_DATA;
2780 : : }
2781 : :
9475 pjw@rhyme.com.au 2782 :CBC 8614 : te->defn = ReadStr(AH);
2783 : 8614 : te->dropStmt = ReadStr(AH);
2784 : :
2785 [ + - ]: 8614 : if (AH->version >= K_VERS_1_3)
2786 : 8614 : te->copyStmt = ReadStr(AH);
2787 : :
8817 tgl@sss.pgh.pa.us 2788 [ + - ]: 8614 : if (AH->version >= K_VERS_1_6)
2789 : 8614 : te->namespace = ReadStr(AH);
2790 : :
7906 2791 [ + - ]: 8614 : if (AH->version >= K_VERS_1_10)
2792 : 8614 : te->tablespace = ReadStr(AH);
2793 : :
2673 andres@anarazel.de 2794 [ + - ]: 8614 : if (AH->version >= K_VERS_1_14)
2795 : 8614 : te->tableam = ReadStr(AH);
2796 : :
799 michael@paquier.xyz 2797 [ + - ]: 8614 : if (AH->version >= K_VERS_1_16)
2798 : 8614 : te->relkind = ReadInt(AH);
2799 : :
9475 pjw@rhyme.com.au 2800 : 8614 : te->owner = ReadStr(AH);
1602 dgustafsson@postgres 2801 : 8614 : is_supported = true;
2802 [ - + ]: 8614 : if (AH->version < K_VERS_1_9)
1602 dgustafsson@postgres 2803 :UBC 0 : is_supported = false;
2804 : : else
2805 : : {
1510 tgl@sss.pgh.pa.us 2806 :CBC 8614 : tmp = ReadStr(AH);
2807 : :
2808 [ - + ]: 8614 : if (strcmp(tmp, "true") == 0)
1510 tgl@sss.pgh.pa.us 2809 :UBC 0 : is_supported = false;
2810 : :
1510 tgl@sss.pgh.pa.us 2811 :CBC 8614 : free(tmp);
2812 : : }
2813 : :
1602 dgustafsson@postgres 2814 [ - + ]: 8614 : if (!is_supported)
2647 peter@eisentraut.org 2815 :UBC 0 : pg_log_warning("restoring tables WITH OIDS is not supported anymore");
2816 : :
2817 : : /* Read TOC entry dependencies */
9221 pjw@rhyme.com.au 2818 [ + - ]:CBC 8614 : if (AH->version >= K_VERS_1_5)
2819 : : {
2820 : 8614 : depSize = 100;
137 michael@paquier.xyz 2821 :GNC 8614 : deps = pg_malloc_array(DumpId, depSize);
9221 pjw@rhyme.com.au 2822 :CBC 8614 : depIdx = 0;
2823 : : for (;;)
2824 : : {
8242 tgl@sss.pgh.pa.us 2825 : 22359 : tmp = ReadStr(AH);
2826 [ + + ]: 22359 : if (!tmp)
2827 : 8614 : break; /* end of list */
8459 2828 [ - + ]: 13745 : if (depIdx >= depSize)
2829 : : {
9221 pjw@rhyme.com.au 2830 :UBC 0 : depSize *= 2;
137 michael@paquier.xyz 2831 :UNC 0 : deps = pg_realloc_array(deps, DumpId, depSize);
2832 : : }
8242 tgl@sss.pgh.pa.us 2833 :CBC 13745 : sscanf(tmp, "%d", &deps[depIdx]);
2834 : 13745 : free(tmp);
2835 : 13745 : depIdx++;
2836 : : }
2837 : :
2838 [ + + ]: 8614 : if (depIdx > 0) /* We have a non-null entry */
2839 : : {
137 michael@paquier.xyz 2840 :GNC 7111 : deps = pg_realloc_array(deps, DumpId, depIdx);
8242 tgl@sss.pgh.pa.us 2841 :CBC 7111 : te->dependencies = deps;
2842 : 7111 : te->nDeps = depIdx;
2843 : : }
2844 : : else
2845 : : {
8459 2846 : 1503 : free(deps);
8242 2847 : 1503 : te->dependencies = NULL;
2848 : 1503 : te->nDeps = 0;
2849 : : }
2850 : : }
2851 : : else
2852 : : {
8242 tgl@sss.pgh.pa.us 2853 :UBC 0 : te->dependencies = NULL;
2854 : 0 : te->nDeps = 0;
2855 : : }
2846 tgl@sss.pgh.pa.us 2856 :CBC 8614 : te->dataLength = 0;
2857 : :
9231 bruce@momjian.us 2858 [ + - ]: 8614 : if (AH->ReadExtraTocPtr)
3218 peter_e@gmx.net 2859 : 8614 : AH->ReadExtraTocPtr(AH, te);
2860 : :
2647 peter@eisentraut.org 2861 [ - + ]: 8614 : pg_log_debug("read TOC entry %d (ID %d) for %s %s",
2862 : : i, te->dumpId, te->desc, te->tag);
2863 : :
2864 : : /* link completed entry into TOC circular list */
9475 pjw@rhyme.com.au 2865 : 8614 : te->prev = AH->toc->prev;
2866 : 8614 : AH->toc->prev->next = te;
2867 : 8614 : AH->toc->prev = te;
2868 : 8614 : te->next = AH->toc;
2869 : :
2870 : : /* special processing immediately upon read for some items */
7338 tgl@sss.pgh.pa.us 2871 [ + + ]: 8614 : if (strcmp(te->desc, "ENCODING") == 0)
2872 : 65 : processEncodingEntry(AH, te);
2873 [ + + ]: 8549 : else if (strcmp(te->desc, "STDSTRINGS") == 0)
2874 : 65 : processStdStringsEntry(AH, te);
3046 2875 [ + + ]: 8484 : else if (strcmp(te->desc, "SEARCHPATH") == 0)
2876 : 65 : processSearchPathEntry(AH, te);
2877 : : }
9492 bruce@momjian.us 2878 : 65 : }
2879 : :
2880 : : static void
7338 tgl@sss.pgh.pa.us 2881 : 65 : processEncodingEntry(ArchiveHandle *AH, TocEntry *te)
2882 : : {
2883 : : /* te->defn should have the form SET client_encoding = 'foo'; */
5331 bruce@momjian.us 2884 : 65 : char *defn = pg_strdup(te->defn);
2885 : : char *ptr1;
7338 tgl@sss.pgh.pa.us 2886 : 65 : char *ptr2 = NULL;
2887 : : int encoding;
2888 : :
2889 : 65 : ptr1 = strchr(defn, '\'');
2890 [ + - ]: 65 : if (ptr1)
2891 : 65 : ptr2 = strchr(++ptr1, '\'');
2892 [ + - ]: 65 : if (ptr2)
2893 : : {
2894 : 65 : *ptr2 = '\0';
2895 : 65 : encoding = pg_char_to_encoding(ptr1);
2896 [ - + ]: 65 : if (encoding < 0)
1544 tgl@sss.pgh.pa.us 2897 :UBC 0 : pg_fatal("unrecognized encoding \"%s\"",
2898 : : ptr1);
7338 tgl@sss.pgh.pa.us 2899 :CBC 65 : AH->public.encoding = encoding;
505 andres@anarazel.de 2900 : 65 : setFmtEncoding(encoding);
2901 : : }
2902 : : else
1544 tgl@sss.pgh.pa.us 2903 :UBC 0 : pg_fatal("invalid ENCODING item: %s",
2904 : : te->defn);
2905 : :
7338 tgl@sss.pgh.pa.us 2906 :CBC 65 : free(defn);
2907 : 65 : }
2908 : :
2909 : : static void
2910 : 65 : processStdStringsEntry(ArchiveHandle *AH, TocEntry *te)
2911 : : {
2912 : : /* te->defn should have the form SET standard_conforming_strings = 'x'; */
2913 : : char *ptr1;
2914 : :
2915 : 65 : ptr1 = strchr(te->defn, '\'');
2916 [ + - + - ]: 65 : if (ptr1 && strncmp(ptr1, "'on'", 4) == 0)
2917 : 65 : AH->public.std_strings = true;
7338 tgl@sss.pgh.pa.us 2918 [ # # # # ]:UBC 0 : else if (ptr1 && strncmp(ptr1, "'off'", 5) == 0)
2919 : 0 : AH->public.std_strings = false;
2920 : : else
1544 2921 : 0 : pg_fatal("invalid STDSTRINGS item: %s",
2922 : : te->defn);
7338 tgl@sss.pgh.pa.us 2923 :CBC 65 : }
2924 : :
2925 : : static void
3046 2926 : 65 : processSearchPathEntry(ArchiveHandle *AH, TocEntry *te)
2927 : : {
2928 : : /*
2929 : : * te->defn should contain a command to set search_path. We just copy it
2930 : : * verbatim for use later.
2931 : : */
2932 : 65 : AH->public.searchpath = pg_strdup(te->defn);
2933 : 65 : }
2934 : :
2935 : : static void
3942 teodor@sigaev.ru 2936 :UBC 0 : StrictNamesCheck(RestoreOptions *ropt)
2937 : : {
2938 : : const char *missing_name;
2939 : :
2940 [ # # ]: 0 : Assert(ropt->strict_names);
2941 : :
2942 [ # # ]: 0 : if (ropt->schemaNames.head != NULL)
2943 : : {
2944 : 0 : missing_name = simple_string_list_not_touched(&ropt->schemaNames);
2945 [ # # ]: 0 : if (missing_name != NULL)
1544 tgl@sss.pgh.pa.us 2946 : 0 : pg_fatal("schema \"%s\" not found", missing_name);
2947 : : }
2948 : :
3942 teodor@sigaev.ru 2949 [ # # ]: 0 : if (ropt->tableNames.head != NULL)
2950 : : {
2951 : 0 : missing_name = simple_string_list_not_touched(&ropt->tableNames);
2952 [ # # ]: 0 : if (missing_name != NULL)
1544 tgl@sss.pgh.pa.us 2953 : 0 : pg_fatal("table \"%s\" not found", missing_name);
2954 : : }
2955 : :
3942 teodor@sigaev.ru 2956 [ # # ]: 0 : if (ropt->indexNames.head != NULL)
2957 : : {
2958 : 0 : missing_name = simple_string_list_not_touched(&ropt->indexNames);
2959 [ # # ]: 0 : if (missing_name != NULL)
1544 tgl@sss.pgh.pa.us 2960 : 0 : pg_fatal("index \"%s\" not found", missing_name);
2961 : : }
2962 : :
3942 teodor@sigaev.ru 2963 [ # # ]: 0 : if (ropt->functionNames.head != NULL)
2964 : : {
2965 : 0 : missing_name = simple_string_list_not_touched(&ropt->functionNames);
2966 [ # # ]: 0 : if (missing_name != NULL)
1544 tgl@sss.pgh.pa.us 2967 : 0 : pg_fatal("function \"%s\" not found", missing_name);
2968 : : }
2969 : :
3942 teodor@sigaev.ru 2970 [ # # ]: 0 : if (ropt->triggerNames.head != NULL)
2971 : : {
2972 : 0 : missing_name = simple_string_list_not_touched(&ropt->triggerNames);
2973 [ # # ]: 0 : if (missing_name != NULL)
1544 tgl@sss.pgh.pa.us 2974 : 0 : pg_fatal("trigger \"%s\" not found", missing_name);
2975 : : }
3942 teodor@sigaev.ru 2976 : 0 : }
2977 : :
2978 : : /*
2979 : : * Determine whether we want to restore this TOC entry.
2980 : : *
2981 : : * Returns 0 if entry should be skipped, or some combination of the
2982 : : * REQ_SCHEMA, REQ_DATA, and REQ_STATS bits if we want to restore schema, data
2983 : : * and/or statistics portions of this TOC entry, or REQ_SPECIAL if it's a
2984 : : * special entry.
2985 : : */
2986 : : static int
3078 tgl@sss.pgh.pa.us 2987 :CBC 52636 : _tocEntryRequired(TocEntry *te, teSection curSection, ArchiveHandle *AH)
2988 : : {
2027 peter@eisentraut.org 2989 : 52636 : int res = REQ_SCHEMA | REQ_DATA;
3078 tgl@sss.pgh.pa.us 2990 : 52636 : RestoreOptions *ropt = AH->public.ropt;
2991 : :
2992 : : /*
2993 : : * For binary upgrade mode, dump pg_largeobject_metadata and the
2994 : : * associated pg_shdepend rows. This is faster to restore than the
2995 : : * equivalent set of large object commands.
2996 : : */
134 nathan@postgresql.or 2997 [ + + + + ]:GNC 52636 : if (ropt->binary_upgrade && strcmp(te->desc, "TABLE DATA") == 0 &&
347 2998 [ + - ]: 41 : (te->catalogId.oid == LargeObjectMetadataRelationId ||
2999 [ + + ]: 41 : te->catalogId.oid == SharedDependRelationId))
3000 : 40 : return REQ_DATA;
3001 : :
3002 : : /* These items are treated specially */
7338 tgl@sss.pgh.pa.us 3003 [ + + ]:CBC 52596 : if (strcmp(te->desc, "ENCODING") == 0 ||
3046 3004 [ + + ]: 52340 : strcmp(te->desc, "STDSTRINGS") == 0 ||
1860 3005 [ + + ]: 52084 : strcmp(te->desc, "SEARCHPATH") == 0)
5145 3006 : 768 : return REQ_SPECIAL;
3007 : :
154 michael@paquier.xyz 3008 [ + + ]:GNC 51828 : if ((strcmp(te->desc, "STATISTICS DATA") == 0) ||
3009 [ + + ]: 46139 : (strcmp(te->desc, "EXTENDED STATISTICS DATA") == 0))
3010 : : {
495 jdavis@postgresql.or 3011 [ - + ]:CBC 5744 : if (!ropt->dumpStatistics)
495 jdavis@postgresql.or 3012 :UBC 0 : return 0;
3013 : :
481 jdavis@postgresql.or 3014 :CBC 5744 : res = REQ_STATS;
3015 : : }
3016 : :
3017 : : /*
3018 : : * DATABASE and DATABASE PROPERTIES also have a special rule: they are
3019 : : * restored in createDB mode, and not restored otherwise, independently of
3020 : : * all else.
3021 : : */
3078 tgl@sss.pgh.pa.us 3022 [ + + ]: 51828 : if (strcmp(te->desc, "DATABASE") == 0 ||
3023 [ + + ]: 51671 : strcmp(te->desc, "DATABASE PROPERTIES") == 0)
3024 : : {
3025 [ + + ]: 233 : if (ropt->createDB)
3026 : 200 : return REQ_SCHEMA;
3027 : : else
3028 : 33 : return 0;
3029 : : }
3030 : :
3031 : : /*
3032 : : * Process exclusions that affect certain classes of TOC entries.
3033 : : */
3034 : :
3035 : : /* If it's an ACL, maybe ignore it */
5145 3036 [ + + - + ]: 51595 : if (ropt->aclsSkip && _tocEntryIsACL(te))
9475 pjw@rhyme.com.au 3037 :UBC 0 : return 0;
3038 : :
3039 : : /* If it's a comment, maybe ignore it */
3078 tgl@sss.pgh.pa.us 3040 [ - + - - ]:CBC 51595 : if (ropt->no_comments && strcmp(te->desc, "COMMENT") == 0)
3078 tgl@sss.pgh.pa.us 3041 :UBC 0 : return 0;
3042 : :
3043 : : /* If it's a policy, maybe ignore it */
471 tgl@sss.pgh.pa.us 3044 [ + + ]:CBC 51595 : if (ropt->no_policies &&
3045 [ + + ]: 771 : (strcmp(te->desc, "POLICY") == 0 ||
3046 [ + + ]: 765 : strcmp(te->desc, "ROW SECURITY") == 0))
3047 : 7 : return 0;
3048 : :
3049 : : /*
3050 : : * If it's a comment on a policy, a publication, or a subscription, maybe
3051 : : * ignore it.
3052 : : */
287 fujii@postgresql.org 3053 [ + + ]: 51588 : if (strcmp(te->desc, "COMMENT") == 0)
3054 : : {
3055 [ + + ]: 7126 : if (ropt->no_policies &&
3056 [ + + ]: 48 : strncmp(te->tag, "POLICY", strlen("POLICY")) == 0)
3057 : 1 : return 0;
3058 : :
3059 [ - + ]: 7125 : if (ropt->no_publications &&
287 fujii@postgresql.org 3060 [ # # ]:UBC 0 : strncmp(te->tag, "PUBLICATION", strlen("PUBLICATION")) == 0)
3061 : 0 : return 0;
3062 : :
287 fujii@postgresql.org 3063 [ + + ]:CBC 7125 : if (ropt->no_subscriptions &&
3064 [ + + ]: 48 : strncmp(te->tag, "SUBSCRIPTION", strlen("SUBSCRIPTION")) == 0)
3065 : 1 : return 0;
3066 : : }
3067 : :
3068 : : /*
3069 : : * If it's a publication or a table part of a publication, maybe ignore
3070 : : * it.
3071 : : */
2835 michael@paquier.xyz 3072 [ - + ]: 51586 : if (ropt->no_publications &&
2835 michael@paquier.xyz 3073 [ # # ]:UBC 0 : (strcmp(te->desc, "PUBLICATION") == 0 ||
1707 akapila@postgresql.o 3074 [ # # ]: 0 : strcmp(te->desc, "PUBLICATION TABLE") == 0 ||
3075 [ # # ]: 0 : strcmp(te->desc, "PUBLICATION TABLES IN SCHEMA") == 0))
3336 peter_e@gmx.net 3076 : 0 : return 0;
3077 : :
3078 : : /* If it's a security label, maybe ignore it */
5521 peter_e@gmx.net 3079 [ - + - - ]:CBC 51586 : if (ropt->no_security_labels && strcmp(te->desc, "SECURITY LABEL") == 0)
5755 rhaas@postgresql.org 3080 :UBC 0 : return 0;
3081 : :
3082 : : /*
3083 : : * If it's a security label on a publication or a subscription, maybe
3084 : : * ignore it.
3085 : : */
285 fujii@postgresql.org 3086 [ + + ]:CBC 51586 : if (strcmp(te->desc, "SECURITY LABEL") == 0)
3087 : : {
285 fujii@postgresql.org 3088 [ - + ]:GBC 9 : if (ropt->no_publications &&
285 fujii@postgresql.org 3089 [ # # ]:UBC 0 : strncmp(te->tag, "PUBLICATION", strlen("PUBLICATION")) == 0)
3090 : 0 : return 0;
3091 : :
285 fujii@postgresql.org 3092 [ - + ]:GBC 9 : if (ropt->no_subscriptions &&
285 fujii@postgresql.org 3093 [ # # ]:UBC 0 : strncmp(te->tag, "SUBSCRIPTION", strlen("SUBSCRIPTION")) == 0)
3094 : 0 : return 0;
3095 : : }
3096 : :
3097 : : /* If it's a subscription, maybe ignore it */
3339 peter_e@gmx.net 3098 [ + + + + ]:CBC 51586 : if (ropt->no_subscriptions && strcmp(te->desc, "SUBSCRIPTION") == 0)
3099 : 3 : return 0;
3100 : :
3101 : : /* Ignore it if section is not to be dumped/restored */
5145 tgl@sss.pgh.pa.us 3102 [ + + + - ]: 51583 : switch (curSection)
3103 : : {
3104 : 32104 : case SECTION_PRE_DATA:
3105 [ + + ]: 32104 : if (!(ropt->dumpSections & DUMP_PRE_DATA))
3106 : 384 : return 0;
3107 : 31720 : break;
3108 : 9659 : case SECTION_DATA:
3109 [ + + ]: 9659 : if (!(ropt->dumpSections & DUMP_DATA))
3110 : 192 : return 0;
3111 : 9467 : break;
3112 : 9820 : case SECTION_POST_DATA:
3113 [ + + ]: 9820 : if (!(ropt->dumpSections & DUMP_POST_DATA))
3114 : 234 : return 0;
3115 : 9586 : break;
5145 tgl@sss.pgh.pa.us 3116 :UBC 0 : default:
3117 : : /* shouldn't get here, really, but ignore it */
5310 andrew@dunslane.net 3118 : 0 : return 0;
3119 : : }
3120 : :
3121 : : /* Ignore it if rejected by idWanted[] (cf. SortTocFromFile) */
3078 tgl@sss.pgh.pa.us 3122 [ - + - - ]:CBC 50773 : if (ropt->idWanted && !ropt->idWanted[te->dumpId - 1])
3570 peter_e@gmx.net 3123 :UBC 0 : return 0;
3124 : :
3125 : : /*
3126 : : * Check options for selective dump/restore.
3127 : : */
3078 tgl@sss.pgh.pa.us 3128 [ + + ]:CBC 50773 : if (strcmp(te->desc, "ACL") == 0 ||
3129 [ + + ]: 48284 : strcmp(te->desc, "COMMENT") == 0 ||
3130 [ + + ]: 41210 : strcmp(te->desc, "SECURITY LABEL") == 0)
3131 : : {
3132 : : /* Database properties react to createDB, not selectivity options. */
3133 [ + + ]: 18969 : if (strncmp(te->tag, "DATABASE ", 9) == 0)
3134 : : {
3135 [ + + ]: 114 : if (!ropt->createDB)
9475 pjw@rhyme.com.au 3136 : 23 : return 0;
3137 : : }
3078 tgl@sss.pgh.pa.us 3138 [ + + ]: 9458 : else if (ropt->schemaNames.head != NULL ||
3139 [ + - ]: 9306 : ropt->schemaExcludeNames.head != NULL ||
3140 [ - + ]: 9306 : ropt->selTypes)
3141 : : {
3142 : : /*
3143 : : * In a selective dump/restore, we want to restore these dependent
3144 : : * TOC entry types only if their parent object is being restored.
3145 : : * Without selectivity options, we let through everything in the
3146 : : * archive. Note there may be such entries with no parent, eg
3147 : : * non-default ACLs for built-in objects. Also, we make
3148 : : * per-column ACLs additionally depend on the table's ACL if any
3149 : : * to ensure correct restore order, so those dependencies should
3150 : : * be ignored in this check.
3151 : : *
3152 : : * This code depends on the parent having been marked already,
3153 : : * which should be the case; if it isn't, perhaps due to
3154 : : * SortTocFromFile rearrangement, skipping the dependent entry
3155 : : * seems prudent anyway.
3156 : : *
3157 : : * Ideally we'd handle, eg, table CHECK constraints this way too.
3158 : : * But it's hard to tell which of their dependencies is the one to
3159 : : * consult.
3160 : : */
1002 3161 : 152 : bool dumpthis = false;
3162 : :
3163 [ + + ]: 366 : for (int i = 0; i < te->nDeps; i++)
3164 : : {
3165 : 214 : TocEntry *pte = getTocEntryByDumpId(AH, te->dependencies[i]);
3166 : :
3167 [ + + ]: 214 : if (!pte)
3168 : 68 : continue; /* probably shouldn't happen */
3169 [ + + ]: 146 : if (strcmp(pte->desc, "ACL") == 0)
3170 : 62 : continue; /* ignore dependency on another ACL */
3171 [ + - ]: 84 : if (pte->reqs == 0)
3172 : 84 : continue; /* this object isn't marked, so ignore it */
3173 : : /* Found a parent to be dumped, so we want to dump this too */
1002 tgl@sss.pgh.pa.us 3174 :UBC 0 : dumpthis = true;
3175 : 0 : break;
3176 : : }
1002 tgl@sss.pgh.pa.us 3177 [ + - ]:CBC 152 : if (!dumpthis)
9475 pjw@rhyme.com.au 3178 : 152 : return 0;
3179 : : }
3180 : : }
3181 : : else
3182 : : {
3183 : : /* Apply selective-restore rules for standalone TOC entries. */
3078 tgl@sss.pgh.pa.us 3184 [ + + ]: 41201 : if (ropt->schemaNames.head != NULL)
3185 : : {
3186 : : /* If no namespace is specified, it means all. */
3187 [ + + ]: 704 : if (!te->namespace)
9475 pjw@rhyme.com.au 3188 : 64 : return 0;
3078 tgl@sss.pgh.pa.us 3189 [ + + ]: 640 : if (!simple_string_list_member(&ropt->schemaNames, te->namespace))
9475 pjw@rhyme.com.au 3190 : 128 : return 0;
3191 : : }
3192 : :
3078 tgl@sss.pgh.pa.us 3193 [ + + ]: 41009 : if (ropt->schemaExcludeNames.head != NULL &&
3194 [ + + + + ]: 38 : te->namespace &&
3195 : 18 : simple_string_list_member(&ropt->schemaExcludeNames, te->namespace))
3196 : 4 : return 0;
3197 : :
3198 [ + + ]: 41005 : if (ropt->selTypes)
3199 : : {
14 michael@paquier.xyz 3200 [ + + ]: 528 : if (strcmp(te->desc, "STATISTICS DATA") == 0)
3201 : : {
3202 : 114 : bool dumpthis = false;
3203 : :
3204 : : /*
3205 : : * Statistics data entries can be for tables or indexes. Check
3206 : : * the parent dependency to determine which type this entry
3207 : : * belongs to, then apply the appropriate name filter.
3208 : : */
3209 [ + + ]: 376 : for (int i = 0; i < te->nDeps; i++)
3210 : : {
3211 : 262 : TocEntry *pte = getTocEntryByDumpId(AH, te->dependencies[i]);
3212 : :
3213 [ + + ]: 262 : if (!pte)
3214 : 134 : continue;
3215 : :
3216 [ + + ]: 128 : if (ropt->selTable &&
3217 [ + + ]: 64 : (strcmp(pte->desc, "TABLE") == 0 ||
3218 [ + - ]: 28 : strcmp(pte->desc, "VIEW") == 0 ||
3219 [ + + ]: 28 : strcmp(pte->desc, "FOREIGN TABLE") == 0 ||
3220 [ + + ]: 27 : strcmp(pte->desc, "MATERIALIZED VIEW") == 0))
3221 : : {
3222 [ + - + + ]: 92 : if (ropt->tableNames.head == NULL ||
3223 : 46 : simple_string_list_member(&ropt->tableNames, pte->tag))
3224 : 1 : dumpthis = true;
3225 : : }
3226 : :
3227 [ + + ]: 128 : if (ropt->selIndex &&
3228 [ + + ]: 64 : strcmp(pte->desc, "INDEX") == 0)
3229 : : {
3230 [ + - + + ]: 6 : if (ropt->indexNames.head == NULL ||
3231 : 3 : simple_string_list_member(&ropt->indexNames, pte->tag))
3232 : 1 : dumpthis = true;
3233 : : }
3234 : : }
3235 [ + + ]: 114 : if (!dumpthis)
3236 : 112 : return 0;
3237 : : }
3238 [ + + ]: 414 : else if (strcmp(te->desc, "TABLE") == 0 ||
3239 [ + + ]: 322 : strcmp(te->desc, "TABLE DATA") == 0 ||
3240 [ + + ]: 236 : strcmp(te->desc, "VIEW") == 0 ||
3241 [ + + ]: 234 : strcmp(te->desc, "FOREIGN TABLE") == 0 ||
3242 [ + + ]: 232 : strcmp(te->desc, "MATERIALIZED VIEW") == 0 ||
3243 [ + + ]: 214 : strcmp(te->desc, "MATERIALIZED VIEW DATA") == 0 ||
3244 [ + + ]: 200 : strcmp(te->desc, "SEQUENCE") == 0 ||
3245 [ + + ]: 191 : strcmp(te->desc, "SEQUENCE SET") == 0)
3246 : : {
3078 tgl@sss.pgh.pa.us 3247 [ + + ]: 232 : if (!ropt->selTable)
3248 : 123 : return 0;
3249 [ + - ]: 109 : if (ropt->tableNames.head != NULL &&
3250 [ + + ]: 109 : !simple_string_list_member(&ropt->tableNames, te->tag))
3251 : 105 : return 0;
3252 : : }
3253 [ + + ]: 182 : else if (strcmp(te->desc, "INDEX") == 0)
3254 : : {
3255 [ + + ]: 12 : if (!ropt->selIndex)
3256 : 7 : return 0;
3257 [ + - ]: 5 : if (ropt->indexNames.head != NULL &&
3258 [ + + ]: 5 : !simple_string_list_member(&ropt->indexNames, te->tag))
3259 : 3 : return 0;
3260 : : }
3261 [ + + ]: 170 : else if (strcmp(te->desc, "FUNCTION") == 0 ||
3262 [ + + ]: 144 : strcmp(te->desc, "AGGREGATE") == 0 ||
3263 [ + + ]: 142 : strcmp(te->desc, "PROCEDURE") == 0)
3264 : : {
3265 [ + + ]: 30 : if (!ropt->selFunction)
3266 : 22 : return 0;
3267 [ + - ]: 8 : if (ropt->functionNames.head != NULL &&
3268 [ + + ]: 8 : !simple_string_list_member(&ropt->functionNames, te->tag))
3269 : 6 : return 0;
3270 : : }
3271 [ + + ]: 140 : else if (strcmp(te->desc, "TRIGGER") == 0)
3272 : : {
3273 [ + + ]: 10 : if (!ropt->selTrigger)
3274 : 8 : return 0;
3275 [ + - ]: 2 : if (ropt->triggerNames.head != NULL &&
3276 [ + + ]: 2 : !simple_string_list_member(&ropt->triggerNames, te->tag))
3277 : 1 : return 0;
3278 : : }
3279 : : else
9475 pjw@rhyme.com.au 3280 : 130 : return 0;
3281 : : }
3282 : : }
3283 : :
3284 : :
3285 : : /*
3286 : : * Determine whether the TOC entry contains schema and/or data components,
3287 : : * and mask off inapplicable REQ bits. If it had a dataDumper, assume
3288 : : * it's both schema and data. Otherwise it's probably schema-only, but
3289 : : * there are exceptions.
3290 : : */
9003 bruce@momjian.us 3291 [ + + ]: 49885 : if (!te->hadDumper)
3292 : : {
3293 : : /*
3294 : : * Special Case: If 'SEQUENCE SET' or anything to do with LOs, then it
3295 : : * is considered a data entry. We don't need to check for BLOBS or
3296 : : * old-style BLOB COMMENTS entries, because they will have hadDumper =
3297 : : * true ... but we do need to check new-style BLOB ACLs, comments,
3298 : : * etc.
3299 : : */
5976 tgl@sss.pgh.pa.us 3300 [ + + ]: 44911 : if (strcmp(te->desc, "SEQUENCE SET") == 0 ||
3301 [ + - ]: 44433 : strcmp(te->desc, "BLOB") == 0 ||
820 3302 [ + + ]: 44433 : strcmp(te->desc, "BLOB METADATA") == 0 ||
5976 3303 [ + + ]: 44334 : (strcmp(te->desc, "ACL") == 0 &&
820 3304 [ + + ]: 2385 : strncmp(te->tag, "LARGE OBJECT", 12) == 0) ||
5976 3305 [ + + ]: 44294 : (strcmp(te->desc, "COMMENT") == 0 &&
820 3306 [ + + ]: 7003 : strncmp(te->tag, "LARGE OBJECT", 12) == 0) ||
5755 rhaas@postgresql.org 3307 [ + + ]: 44230 : (strcmp(te->desc, "SECURITY LABEL") == 0 &&
820 tgl@sss.pgh.pa.us 3308 [ + - ]:GBC 9 : strncmp(te->tag, "LARGE OBJECT", 12) == 0))
9003 bruce@momjian.us 3309 :CBC 690 : res = res & REQ_DATA;
3310 : : else
9004 pjw@rhyme.com.au 3311 : 44221 : res = res & ~REQ_DATA;
3312 : : }
3313 : :
3314 : : /*
3315 : : * If there's no definition command, there's no schema component. Treat
3316 : : * "load via partition root" comments as not schema.
3317 : : */
1201 tgl@sss.pgh.pa.us 3318 [ + + + - ]: 49885 : if (!te->defn || !te->defn[0] ||
3319 [ + + ]: 41471 : strncmp(te->defn, "-- load via partition root ", 27) == 0)
3078 3320 : 8508 : res = res & ~REQ_SCHEMA;
3321 : :
3322 : : /*
3323 : : * Special case: <Init> type with <Max OID> tag; this is obsolete and we
3324 : : * always ignore it.
3325 : : */
8762 bruce@momjian.us 3326 [ - + - - ]: 49885 : if ((strcmp(te->desc, "<Init>") == 0) && (strcmp(te->tag, "Max OID") == 0))
7627 tgl@sss.pgh.pa.us 3327 :UBC 0 : return 0;
3328 : :
3329 : : /* Mask it if we don't want data */
582 nathan@postgresql.or 3330 [ + + ]:CBC 49885 : if (!ropt->dumpData)
3331 : : {
3332 : : /*
3333 : : * The sequence_data option overrides dumpData for SEQUENCE SET.
3334 : : *
3335 : : * In binary-upgrade mode, even with dumpData unset, we do not mask
3336 : : * out large objects. (Only large object definitions, comments and
3337 : : * other metadata should be generated in binary-upgrade mode, not the
3338 : : * actual data, but that need not concern us here.)
3339 : : */
3403 sfrost@snowman.net 3340 [ + + + + ]: 4570 : if (!(ropt->sequence_data && strcmp(te->desc, "SEQUENCE SET") == 0) &&
3081 tgl@sss.pgh.pa.us 3341 [ + + ]: 4505 : !(ropt->binary_upgrade &&
3342 [ + - ]: 3963 : (strcmp(te->desc, "BLOB") == 0 ||
820 3343 [ + - ]: 3963 : strcmp(te->desc, "BLOB METADATA") == 0 ||
3081 3344 [ + + ]: 3963 : (strcmp(te->desc, "ACL") == 0 &&
820 3345 [ + - ]: 101 : strncmp(te->tag, "LARGE OBJECT", 12) == 0) ||
3081 3346 [ + + ]: 3963 : (strcmp(te->desc, "COMMENT") == 0 &&
820 3347 [ + + ]: 115 : strncmp(te->tag, "LARGE OBJECT", 12) == 0) ||
3081 3348 [ + + ]: 3955 : (strcmp(te->desc, "SECURITY LABEL") == 0 &&
820 tgl@sss.pgh.pa.us 3349 [ - + ]:GBC 5 : strncmp(te->tag, "LARGE OBJECT", 12) == 0))))
495 jdavis@postgresql.or 3350 :CBC 4492 : res = res & (REQ_SCHEMA | REQ_STATS);
3351 : : }
3352 : :
3353 : : /* Mask it if we don't want schema */
582 nathan@postgresql.or 3354 [ + + ]: 49885 : if (!ropt->dumpSchema)
495 jdavis@postgresql.or 3355 : 568 : res = res & (REQ_DATA | REQ_STATS);
3356 : :
9231 bruce@momjian.us 3357 : 49885 : return res;
3358 : : }
3359 : :
3360 : : /*
3361 : : * Identify which pass we should restore this TOC entry in.
3362 : : *
3363 : : * See notes with the RestorePass typedef in pg_backup_archiver.h.
3364 : : */
3365 : : static RestorePass
452 nathan@postgresql.or 3366 : 110187 : _tocEntryRestorePass(TocEntry *te)
3367 : : {
3368 : : /* "ACL LANGUAGE" was a crock emitted only in PG 7.4 */
3253 tgl@sss.pgh.pa.us 3369 [ + + ]: 110187 : if (strcmp(te->desc, "ACL") == 0 ||
3370 [ + - ]: 104978 : strcmp(te->desc, "ACL LANGUAGE") == 0 ||
3371 [ + + ]: 104978 : strcmp(te->desc, "DEFAULT ACL") == 0)
3372 : 5593 : return RESTORE_PASS_ACL;
2304 3373 [ + + ]: 104594 : if (strcmp(te->desc, "EVENT TRIGGER") == 0 ||
3374 [ + + ]: 104490 : strcmp(te->desc, "MATERIALIZED VIEW DATA") == 0)
3375 : 814 : return RESTORE_PASS_POST_ACL;
3376 : :
3377 : : /*
3378 : : * Comments and security labels need to be emitted in the same pass as
3379 : : * their parent objects. ACLs haven't got comments and security labels,
3380 : : * and neither do matview data objects, but event triggers do.
3381 : : * (Fortunately, event triggers haven't got ACLs, or we'd need yet another
3382 : : * weird special case.)
3383 : : */
287 fujii@postgresql.org 3384 [ + + ]: 103780 : if ((strcmp(te->desc, "COMMENT") == 0 ||
3385 [ + + ]: 89746 : strcmp(te->desc, "SECURITY LABEL") == 0) &&
2274 tgl@sss.pgh.pa.us 3386 [ - + ]: 14038 : strncmp(te->tag, "EVENT TRIGGER ", 14) == 0)
2274 tgl@sss.pgh.pa.us 3387 :UBC 0 : return RESTORE_PASS_POST_ACL;
3388 : :
3389 : : /*
3390 : : * If statistics data is dependent on materialized view data, it must be
3391 : : * deferred to RESTORE_PASS_POST_ACL. Those entries are already marked as
3392 : : * SECTION_POST_DATA, and some other stats entries (e.g., index stats)
3393 : : * will also be marked as SECTION_POST_DATA. Additionally, our lookahead
3394 : : * code in fetchAttributeStats() assumes that we dump all statistics data
3395 : : * entries in TOC order. To ensure this assumption holds, we move all
3396 : : * statistics data entries in SECTION_POST_DATA to RESTORE_PASS_POST_ACL.
3397 : : */
452 nathan@postgresql.or 3398 [ + + ]:CBC 103780 : if (strcmp(te->desc, "STATISTICS DATA") == 0 &&
3399 [ + + ]: 9991 : te->section == SECTION_POST_DATA)
3400 : 3494 : return RESTORE_PASS_POST_ACL;
3401 : :
3402 : : /* All else can be handled in the main pass. */
3253 tgl@sss.pgh.pa.us 3403 : 100286 : return RESTORE_PASS_MAIN;
3404 : : }
3405 : :
3406 : : /*
3407 : : * Identify TOC entries that are ACLs.
3408 : : *
3409 : : * Note: it seems worth duplicating some code here to avoid a hard-wired
3410 : : * assumption that these are exactly the same entries that we restore during
3411 : : * the RESTORE_PASS_ACL phase.
3412 : : */
3413 : : static bool
5976 3414 : 42442 : _tocEntryIsACL(TocEntry *te)
3415 : : {
3416 : : /* "ACL LANGUAGE" was a crock emitted only in PG 7.4 */
3417 [ + + ]: 42442 : if (strcmp(te->desc, "ACL") == 0 ||
3418 [ + - ]: 40635 : strcmp(te->desc, "ACL LANGUAGE") == 0 ||
3419 [ + + ]: 40635 : strcmp(te->desc, "DEFAULT ACL") == 0)
3420 : 1935 : return true;
3421 : 40507 : return false;
3422 : : }
3423 : :
3424 : : /*
3425 : : * Issue SET commands for parameters that we want to have set the same way
3426 : : * at all times during execution of a restore script.
3427 : : */
3428 : : static void
8162 3429 : 305 : _doSetFixedOutputState(ArchiveHandle *AH)
3430 : : {
3821 3431 : 305 : RestoreOptions *ropt = AH->public.ropt;
3432 : :
3433 : : /*
3434 : : * Disable timeouts to allow for slow commands, idle parallel workers, etc
3435 : : */
6631 andrew@dunslane.net 3436 : 305 : ahprintf(AH, "SET statement_timeout = 0;\n");
4854 tgl@sss.pgh.pa.us 3437 : 305 : ahprintf(AH, "SET lock_timeout = 0;\n");
3667 3438 : 305 : ahprintf(AH, "SET idle_in_transaction_session_timeout = 0;\n");
866 akorotkov@postgresql 3439 : 305 : ahprintf(AH, "SET transaction_timeout = 0;\n");
3440 : :
3441 : : /* Select the correct character set encoding */
7338 tgl@sss.pgh.pa.us 3442 : 305 : ahprintf(AH, "SET client_encoding = '%s';\n",
3443 : : pg_encoding_to_char(AH->public.encoding));
3444 : :
3445 : : /* Select the correct string literal syntax */
3446 : 305 : ahprintf(AH, "SET standard_conforming_strings = %s;\n",
3447 [ + - ]: 305 : AH->public.std_strings ? "on" : "off");
3448 : :
3449 : : /* Select the role to be used during restore */
3821 3450 [ + - - + ]: 305 : if (ropt && ropt->use_role)
3821 tgl@sss.pgh.pa.us 3451 :UBC 0 : ahprintf(AH, "SET ROLE %s;\n", fmtId(ropt->use_role));
3452 : :
3453 : : /* Select the dump-time search_path */
3046 tgl@sss.pgh.pa.us 3454 [ + - ]:CBC 305 : if (AH->public.searchpath)
3455 : 305 : ahprintf(AH, "%s", AH->public.searchpath);
3456 : :
3457 : : /* Make sure function checking is disabled */
8162 3458 : 305 : ahprintf(AH, "SET check_function_bodies = false;\n");
3459 : :
3460 : : /* Ensure that all valid XML data will be accepted */
2656 3461 : 305 : ahprintf(AH, "SET xmloption = content;\n");
3462 : :
3463 : : /* Avoid annoying notices etc */
7984 bruce@momjian.us 3464 : 305 : ahprintf(AH, "SET client_min_messages = warning;\n");
3465 : :
3466 : : /* Adjust row-security state */
3821 tgl@sss.pgh.pa.us 3467 [ + - - + ]: 305 : if (ropt && ropt->enable_row_security)
4150 tgl@sss.pgh.pa.us 3468 :UBC 0 : ahprintf(AH, "SET row_security = on;\n");
3469 : : else
4150 tgl@sss.pgh.pa.us 3470 :CBC 305 : ahprintf(AH, "SET row_security = off;\n");
3471 : :
3472 : : /*
3473 : : * In --transaction-size mode, we should always be in a transaction when
3474 : : * we begin to restore objects.
3475 : : */
820 3476 [ + - + + ]: 305 : if (ropt && ropt->txn_size > 0)
3477 : : {
3478 [ + - ]: 96 : if (AH->connection)
3479 : 96 : StartTransaction(&AH->public);
3480 : : else
820 tgl@sss.pgh.pa.us 3481 :UBC 0 : ahprintf(AH, "\nBEGIN;\n");
820 tgl@sss.pgh.pa.us 3482 :CBC 96 : AH->txnCount = 0;
3483 : : }
3484 : :
8162 3485 : 305 : ahprintf(AH, "\n");
3486 : 305 : }
3487 : :
3488 : : /*
3489 : : * Issue a SET SESSION AUTHORIZATION command. Caller is responsible
3490 : : * for updating state if appropriate. If user is NULL or an empty string,
3491 : : * the specification DEFAULT will be used.
3492 : : */
3493 : : static void
8717 peter_e@gmx.net 3494 : 1 : _doSetSessionAuth(ArchiveHandle *AH, const char *user)
3495 : : {
3496 : 1 : PQExpBuffer cmd = createPQExpBuffer();
3497 : :
4607 heikki.linnakangas@i 3498 : 1 : appendPQExpBufferStr(cmd, "SET SESSION AUTHORIZATION ");
3499 : :
3500 : : /*
3501 : : * SQL requires a string literal here. Might as well be correct.
3502 : : */
8316 tgl@sss.pgh.pa.us 3503 [ + - + - ]: 1 : if (user && *user)
7338 3504 : 1 : appendStringLiteralAHX(cmd, user, AH);
3505 : : else
4607 heikki.linnakangas@i 3506 :UBC 0 : appendPQExpBufferStr(cmd, "DEFAULT");
4607 heikki.linnakangas@i 3507 :CBC 1 : appendPQExpBufferChar(cmd, ';');
3508 : :
8817 tgl@sss.pgh.pa.us 3509 [ - + ]: 1 : if (RestoringToDB(AH))
3510 : : {
3511 : : PGresult *res;
3512 : :
8717 peter_e@gmx.net 3513 :UBC 0 : res = PQexec(AH->connection, cmd->data);
3514 : :
8817 tgl@sss.pgh.pa.us 3515 [ # # # # ]: 0 : if (!res || PQresultStatus(res) != PGRES_COMMAND_OK)
3516 : : /* NOT warn_or_exit_horribly... use -O instead to skip this. */
1544 3517 : 0 : pg_fatal("could not set session user to \"%s\": %s",
3518 : : user, PQerrorMessage(AH->connection));
3519 : :
8817 3520 : 0 : PQclear(res);
3521 : : }
3522 : : else
8717 peter_e@gmx.net 3523 :CBC 1 : ahprintf(AH, "%s\n\n", cmd->data);
3524 : :
3525 : 1 : destroyPQExpBuffer(cmd);
8817 tgl@sss.pgh.pa.us 3526 : 1 : }
3527 : :
3528 : :
3529 : : /*
3530 : : * Issue the commands to connect to the specified database.
3531 : : *
3532 : : * If we're currently restoring right into a database, this will
3533 : : * actually establish a connection. Otherwise it puts a \connect into
3534 : : * the script output.
3535 : : */
3536 : : static void
7963 3537 : 100 : _reconnectToDB(ArchiveHandle *AH, const char *dbname)
3538 : : {
8316 3539 [ + + ]: 100 : if (RestoringToDB(AH))
2105 3540 : 65 : ReconnectToServer(AH, dbname);
3541 : : else
3542 : : {
3543 : : PQExpBufferData connectbuf;
323 nathan@postgresql.or 3544 : 35 : RestoreOptions *ropt = AH->public.ropt;
3545 : :
3546 : : /*
3547 : : * We must temporarily exit restricted mode for \connect, etc.
3548 : : * Anything added between this line and the following \restrict must
3549 : : * be careful to avoid any possible meta-command injection vectors.
3550 : : */
3551 : 35 : ahprintf(AH, "\\unrestrict %s\n", ropt->restrict_key);
3552 : :
2105 tgl@sss.pgh.pa.us 3553 : 35 : initPQExpBuffer(&connectbuf);
3554 : 35 : appendPsqlMetaConnect(&connectbuf, dbname);
323 nathan@postgresql.or 3555 : 35 : ahprintf(AH, "%s", connectbuf.data);
2105 tgl@sss.pgh.pa.us 3556 : 35 : termPQExpBuffer(&connectbuf);
3557 : :
323 nathan@postgresql.or 3558 : 35 : ahprintf(AH, "\\restrict %s\n\n", ropt->restrict_key);
3559 : : }
3560 : :
3561 : : /*
3562 : : * NOTE: currUser keeps track of what the imaginary session user in our
3563 : : * script is. It's now effectively reset to the original userID.
3564 : : */
1475 peter@eisentraut.org 3565 : 100 : free(AH->currUser);
6357 andrew@dunslane.net 3566 : 100 : AH->currUser = NULL;
3567 : :
3568 : : /* don't assume we still know the output schema, tablespace, etc either */
1475 peter@eisentraut.org 3569 : 100 : free(AH->currSchema);
6357 andrew@dunslane.net 3570 : 100 : AH->currSchema = NULL;
3571 : :
1475 peter@eisentraut.org 3572 : 100 : free(AH->currTableAm);
1625 michael@paquier.xyz 3573 : 100 : AH->currTableAm = NULL;
3574 : :
1475 peter@eisentraut.org 3575 : 100 : free(AH->currTablespace);
6357 andrew@dunslane.net 3576 : 100 : AH->currTablespace = NULL;
3577 : :
3578 : : /* re-establish fixed state */
8162 tgl@sss.pgh.pa.us 3579 : 100 : _doSetFixedOutputState(AH);
9464 pjw@rhyme.com.au 3580 : 100 : }
3581 : :
3582 : : /*
3583 : : * Become the specified user, and update state to avoid redundant commands
3584 : : *
3585 : : * NULL or empty argument is taken to mean restoring the session default
3586 : : */
3587 : : static void
8316 tgl@sss.pgh.pa.us 3588 : 78 : _becomeUser(ArchiveHandle *AH, const char *user)
3589 : : {
3590 [ - + ]: 78 : if (!user)
8316 tgl@sss.pgh.pa.us 3591 :UBC 0 : user = ""; /* avoid null pointers */
3592 : :
8316 tgl@sss.pgh.pa.us 3593 [ + + + - ]:CBC 78 : if (AH->currUser && strcmp(AH->currUser, user) == 0)
3594 : 77 : return; /* no need to do anything */
3595 : :
3596 : 1 : _doSetSessionAuth(AH, user);
3597 : :
3598 : : /*
3599 : : * NOTE: currUser keeps track of what the imaginary session user in our
3600 : : * script is
3601 : : */
1475 peter@eisentraut.org 3602 : 1 : free(AH->currUser);
5331 bruce@momjian.us 3603 : 1 : AH->currUser = pg_strdup(user);
3604 : : }
3605 : :
3606 : : /*
3607 : : * Become the owner of the given TOC entry object. If
3608 : : * changes in ownership are not allowed, this doesn't do anything.
3609 : : */
3610 : : static void
8316 tgl@sss.pgh.pa.us 3611 : 46922 : _becomeOwner(ArchiveHandle *AH, TocEntry *te)
3612 : : {
3821 3613 : 46922 : RestoreOptions *ropt = AH->public.ropt;
3614 : :
3615 [ + - + + : 46922 : if (ropt && (ropt->noOwner || !ropt->use_setsessauth))
+ - ]
9464 pjw@rhyme.com.au 3616 : 46922 : return;
3617 : :
8316 tgl@sss.pgh.pa.us 3618 :UBC 0 : _becomeUser(AH, te->owner);
3619 : : }
3620 : :
3621 : :
3622 : : /*
3623 : : * Issue the commands to select the specified schema as the current schema
3624 : : * in the target database.
3625 : : */
3626 : : static void
8817 tgl@sss.pgh.pa.us 3627 :CBC 46996 : _selectOutputSchema(ArchiveHandle *AH, const char *schemaName)
3628 : : {
3629 : : PQExpBuffer qry;
3630 : :
3631 : : /*
3632 : : * If there was a SEARCHPATH TOC entry, we're supposed to just stay with
3633 : : * that search_path rather than switching to entry-specific paths.
3634 : : * Otherwise, it's an old archive that will not restore correctly unless
3635 : : * we set the search_path as it's expecting.
3636 : : */
3046 3637 [ + - ]: 46996 : if (AH->public.searchpath)
3638 : 46996 : return;
3639 : :
8817 tgl@sss.pgh.pa.us 3640 [ # # # # ]:UBC 0 : if (!schemaName || *schemaName == '\0' ||
7377 3641 [ # # # # ]: 0 : (AH->currSchema && strcmp(AH->currSchema, schemaName) == 0))
8817 3642 : 0 : return; /* no need to do anything */
3643 : :
8799 3644 : 0 : qry = createPQExpBuffer();
3645 : :
3646 : 0 : appendPQExpBuffer(qry, "SET search_path = %s",
3647 : : fmtId(schemaName));
3648 [ # # ]: 0 : if (strcmp(schemaName, "pg_catalog") != 0)
4607 heikki.linnakangas@i 3649 : 0 : appendPQExpBufferStr(qry, ", pg_catalog");
3650 : :
8817 tgl@sss.pgh.pa.us 3651 [ # # ]: 0 : if (RestoringToDB(AH))
3652 : : {
3653 : : PGresult *res;
3654 : :
3655 : 0 : res = PQexec(AH->connection, qry->data);
3656 : :
3657 [ # # # # ]: 0 : if (!res || PQresultStatus(res) != PGRES_COMMAND_OK)
2647 peter@eisentraut.org 3658 : 0 : warn_or_exit_horribly(AH,
3659 : : "could not set \"search_path\" to \"%s\": %s",
5215 alvherre@alvh.no-ip. 3660 : 0 : schemaName, PQerrorMessage(AH->connection));
3661 : :
8817 tgl@sss.pgh.pa.us 3662 : 0 : PQclear(res);
3663 : : }
3664 : : else
8799 3665 : 0 : ahprintf(AH, "%s;\n\n", qry->data);
3666 : :
1475 peter@eisentraut.org 3667 : 0 : free(AH->currSchema);
5331 bruce@momjian.us 3668 : 0 : AH->currSchema = pg_strdup(schemaName);
3669 : :
8799 tgl@sss.pgh.pa.us 3670 : 0 : destroyPQExpBuffer(qry);
3671 : : }
3672 : :
3673 : : /*
3674 : : * Issue the commands to select the specified tablespace as the current one
3675 : : * in the target database.
3676 : : */
3677 : : static void
7906 tgl@sss.pgh.pa.us 3678 :CBC 42099 : _selectTablespace(ArchiveHandle *AH, const char *tablespace)
3679 : : {
3821 3680 : 42099 : RestoreOptions *ropt = AH->public.ropt;
3681 : : PQExpBuffer qry;
3682 : : const char *want,
3683 : : *have;
3684 : :
3685 : : /* do nothing in --no-tablespaces mode */
3686 [ - + ]: 42099 : if (ropt->noTablespace)
6676 tgl@sss.pgh.pa.us 3687 :UBC 0 : return;
3688 : :
7906 tgl@sss.pgh.pa.us 3689 :CBC 42099 : have = AH->currTablespace;
3690 : 42099 : want = tablespace;
3691 : :
3692 : : /* no need to do anything for non-tablespace object */
3693 [ + + ]: 42099 : if (!want)
3694 : 33484 : return;
3695 : :
3696 [ + + + + ]: 8615 : if (have && strcmp(want, have) == 0)
3697 : 8384 : return; /* no need to do anything */
3698 : :
3699 : 231 : qry = createPQExpBuffer();
3700 : :
3701 [ + + ]: 231 : if (strcmp(want, "") == 0)
3702 : : {
3703 : : /* We want the tablespace to be the database's default */
4607 heikki.linnakangas@i 3704 : 173 : appendPQExpBufferStr(qry, "SET default_tablespace = ''");
3705 : : }
3706 : : else
3707 : : {
3708 : : /* We want an explicit tablespace */
7906 tgl@sss.pgh.pa.us 3709 : 58 : appendPQExpBuffer(qry, "SET default_tablespace = %s", fmtId(want));
3710 : : }
3711 : :
3712 [ + + ]: 231 : if (RestoringToDB(AH))
3713 : : {
3714 : : PGresult *res;
3715 : :
3716 : 29 : res = PQexec(AH->connection, qry->data);
3717 : :
3718 [ + - - + ]: 29 : if (!res || PQresultStatus(res) != PGRES_COMMAND_OK)
2647 peter@eisentraut.org 3719 :UBC 0 : warn_or_exit_horribly(AH,
3720 : : "could not set \"default_tablespace\" to %s: %s",
3296 tgl@sss.pgh.pa.us 3721 : 0 : fmtId(want), PQerrorMessage(AH->connection));
3722 : :
7906 tgl@sss.pgh.pa.us 3723 :CBC 29 : PQclear(res);
3724 : : }
3725 : : else
3726 : 202 : ahprintf(AH, "%s;\n\n", qry->data);
3727 : :
1475 peter@eisentraut.org 3728 : 231 : free(AH->currTablespace);
5331 bruce@momjian.us 3729 : 231 : AH->currTablespace = pg_strdup(want);
3730 : :
7906 tgl@sss.pgh.pa.us 3731 : 231 : destroyPQExpBuffer(qry);
3732 : : }
3733 : :
3734 : : /*
3735 : : * Set the proper default_table_access_method value for the table.
3736 : : */
3737 : : static void
2673 andres@anarazel.de 3738 : 41507 : _selectTableAccessMethod(ArchiveHandle *AH, const char *tableam)
3739 : : {
1625 michael@paquier.xyz 3740 : 41507 : RestoreOptions *ropt = AH->public.ropt;
3741 : : PQExpBuffer cmd;
3742 : : const char *want,
3743 : : *have;
3744 : :
3745 : : /* do nothing in --no-table-access-method mode */
3746 [ + + ]: 41507 : if (ropt->noTableAm)
3747 : 385 : return;
3748 : :
2673 andres@anarazel.de 3749 : 41122 : have = AH->currTableAm;
3750 : 41122 : want = tableam;
3751 : :
3752 [ + + ]: 41122 : if (!want)
3753 : 35778 : return;
3754 : :
3755 [ + + + + ]: 5344 : if (have && strcmp(want, have) == 0)
3756 : 5033 : return;
3757 : :
3758 : 311 : cmd = createPQExpBuffer();
3759 : 311 : appendPQExpBuffer(cmd, "SET default_table_access_method = %s;", fmtId(want));
3760 : :
3761 [ + + ]: 311 : if (RestoringToDB(AH))
3762 : : {
3763 : : PGresult *res;
3764 : :
3765 : 24 : res = PQexec(AH->connection, cmd->data);
3766 : :
3767 [ + - - + ]: 24 : if (!res || PQresultStatus(res) != PGRES_COMMAND_OK)
2647 peter@eisentraut.org 3768 :UBC 0 : warn_or_exit_horribly(AH,
3769 : : "could not set \"default_table_access_method\": %s",
2673 andres@anarazel.de 3770 : 0 : PQerrorMessage(AH->connection));
3771 : :
2673 andres@anarazel.de 3772 :CBC 24 : PQclear(res);
3773 : : }
3774 : : else
3775 : 287 : ahprintf(AH, "%s\n\n", cmd->data);
3776 : :
3777 : 311 : destroyPQExpBuffer(cmd);
3778 : :
1475 peter@eisentraut.org 3779 : 311 : free(AH->currTableAm);
2673 andres@anarazel.de 3780 : 311 : AH->currTableAm = pg_strdup(want);
3781 : : }
3782 : :
3783 : : /*
3784 : : * Set the proper default table access method for a table without storage.
3785 : : * Currently, this is required only for partitioned tables with a table AM.
3786 : : */
3787 : : static void
799 michael@paquier.xyz 3788 : 592 : _printTableAccessMethodNoStorage(ArchiveHandle *AH, TocEntry *te)
3789 : : {
3790 : 592 : RestoreOptions *ropt = AH->public.ropt;
3791 : 592 : const char *tableam = te->tableam;
3792 : : PQExpBuffer cmd;
3793 : :
3794 : : /* do nothing in --no-table-access-method mode */
3795 [ + + ]: 592 : if (ropt->noTableAm)
3796 : 4 : return;
3797 : :
3798 [ + + ]: 588 : if (!tableam)
3799 : 558 : return;
3800 : :
3801 [ - + ]: 30 : Assert(te->relkind == RELKIND_PARTITIONED_TABLE);
3802 : :
3803 : 30 : cmd = createPQExpBuffer();
3804 : :
3805 : 30 : appendPQExpBufferStr(cmd, "ALTER TABLE ");
3806 : 30 : appendPQExpBuffer(cmd, "%s ", fmtQualifiedId(te->namespace, te->tag));
3807 : 30 : appendPQExpBuffer(cmd, "SET ACCESS METHOD %s;",
3808 : : fmtId(tableam));
3809 : :
3810 [ - + ]: 30 : if (RestoringToDB(AH))
3811 : : {
3812 : : PGresult *res;
3813 : :
799 michael@paquier.xyz 3814 :UBC 0 : res = PQexec(AH->connection, cmd->data);
3815 : :
3816 [ # # # # ]: 0 : if (!res || PQresultStatus(res) != PGRES_COMMAND_OK)
3817 : 0 : warn_or_exit_horribly(AH,
3818 : : "could not alter table access method: %s",
3819 : 0 : PQerrorMessage(AH->connection));
3820 : 0 : PQclear(res);
3821 : : }
3822 : : else
799 michael@paquier.xyz 3823 :CBC 30 : ahprintf(AH, "%s\n\n", cmd->data);
3824 : :
3825 : 30 : destroyPQExpBuffer(cmd);
3826 : : }
3827 : :
3828 : : /*
3829 : : * Extract an object description for a TOC entry, and append it to buf.
3830 : : *
3831 : : * This is used for ALTER ... OWNER TO.
3832 : : *
3833 : : * If the object type has no owner, do nothing.
3834 : : */
3835 : : static void
1336 peter@eisentraut.org 3836 : 21373 : _getObjectDescription(PQExpBuffer buf, const TocEntry *te)
3837 : : {
7840 tgl@sss.pgh.pa.us 3838 : 21373 : const char *type = te->desc;
3839 : :
3840 : : /* objects that don't require special decoration */
5617 peter_e@gmx.net 3841 [ + + ]: 21373 : if (strcmp(type, "COLLATION") == 0 ||
3842 [ + + ]: 18660 : strcmp(type, "CONVERSION") == 0 ||
7840 tgl@sss.pgh.pa.us 3843 [ + + ]: 18336 : strcmp(type, "DOMAIN") == 0 ||
5659 rhaas@postgresql.org 3844 [ + + ]: 18166 : strcmp(type, "FOREIGN TABLE") == 0 ||
1336 peter@eisentraut.org 3845 [ + + ]: 18134 : strcmp(type, "MATERIALIZED VIEW") == 0 ||
106 peter@eisentraut.org 3846 [ + + ]:GNC 17818 : strcmp(type, "PROPERTY GRAPH") == 0 ||
1336 peter@eisentraut.org 3847 [ + + ]:CBC 17723 : strcmp(type, "SEQUENCE") == 0 ||
3848 [ + + ]: 17495 : strcmp(type, "STATISTICS") == 0 ||
3849 [ + + ]: 17356 : strcmp(type, "TABLE") == 0 ||
6888 tgl@sss.pgh.pa.us 3850 [ + + ]: 11742 : strcmp(type, "TEXT SEARCH DICTIONARY") == 0 ||
4704 bruce@momjian.us 3851 [ + + ]: 11571 : strcmp(type, "TEXT SEARCH CONFIGURATION") == 0 ||
1336 peter@eisentraut.org 3852 [ + + ]: 11425 : strcmp(type, "TYPE") == 0 ||
3853 [ + + ]: 10804 : strcmp(type, "VIEW") == 0 ||
3854 : : /* non-schema-specified objects */
4704 bruce@momjian.us 3855 [ + + ]: 10238 : strcmp(type, "DATABASE") == 0 ||
7036 tgl@sss.pgh.pa.us 3856 [ + + ]: 10174 : strcmp(type, "PROCEDURAL LANGUAGE") == 0 ||
6402 peter_e@gmx.net 3857 [ + + ]: 10145 : strcmp(type, "SCHEMA") == 0 ||
3265 tgl@sss.pgh.pa.us 3858 [ + + ]: 9946 : strcmp(type, "EVENT TRIGGER") == 0 ||
6402 peter_e@gmx.net 3859 [ + + ]: 9912 : strcmp(type, "FOREIGN DATA WRAPPER") == 0 ||
3860 [ + + ]: 9870 : strcmp(type, "SERVER") == 0 ||
3449 3861 [ + + ]: 9826 : strcmp(type, "PUBLICATION") == 0 ||
1336 peter@eisentraut.org 3862 [ + + ]: 9510 : strcmp(type, "SUBSCRIPTION") == 0)
3863 : : {
3046 tgl@sss.pgh.pa.us 3864 : 11946 : appendPQExpBuffer(buf, "%s ", type);
3865 [ + + + - ]: 11946 : if (te->namespace && *te->namespace)
3866 : 11135 : appendPQExpBuffer(buf, "%s.", fmtId(te->namespace));
3867 : 11946 : appendPQExpBufferStr(buf, fmtId(te->tag));
3868 : : }
3869 : : /* LOs just have a name, but it's numeric so must not use fmtId */
1336 peter@eisentraut.org 3870 [ - + ]: 9427 : else if (strcmp(type, "BLOB") == 0)
3871 : : {
5976 tgl@sss.pgh.pa.us 3872 :UBC 0 : appendPQExpBuffer(buf, "LARGE OBJECT %s", te->tag);
3873 : : }
3874 : :
3875 : : /*
3876 : : * These object types require additional decoration. Fortunately, the
3877 : : * information needed is exactly what's in the DROP command.
3878 : : */
1336 peter@eisentraut.org 3879 [ + + ]:CBC 9427 : else if (strcmp(type, "AGGREGATE") == 0 ||
3880 [ + + ]: 9161 : strcmp(type, "FUNCTION") == 0 ||
3881 [ + + ]: 7535 : strcmp(type, "OPERATOR") == 0 ||
3882 [ + + ]: 5035 : strcmp(type, "OPERATOR CLASS") == 0 ||
3883 [ + + ]: 4393 : strcmp(type, "OPERATOR FAMILY") == 0 ||
3884 [ + + ]: 3854 : strcmp(type, "PROCEDURE") == 0)
3885 : : {
3886 : : /* Chop "DROP " off the front and make a modifiable copy */
5331 bruce@momjian.us 3887 : 5663 : char *first = pg_strdup(te->dropStmt + 5);
3888 : : char *last;
3889 : :
3890 : : /* point to last character in string */
7840 tgl@sss.pgh.pa.us 3891 : 5663 : last = first + strlen(first) - 1;
3892 : :
3893 : : /* Strip off any ';' or '\n' at the end */
3894 [ + - + + : 16989 : while (last >= first && (*last == '\n' || *last == ';'))
+ + ]
3895 : 11326 : last--;
3896 : 5663 : *(last + 1) = '\0';
3897 : :
3898 : 5663 : appendPQExpBufferStr(buf, first);
3899 : :
7975 bruce@momjian.us 3900 : 5663 : free(first);
7840 tgl@sss.pgh.pa.us 3901 : 5663 : return;
3902 : : }
3903 : : /* these object types don't have separate owners */
1336 peter@eisentraut.org 3904 [ + - ]: 3764 : else if (strcmp(type, "CAST") == 0 ||
3905 [ + + ]: 3764 : strcmp(type, "CHECK CONSTRAINT") == 0 ||
3906 [ + + ]: 3714 : strcmp(type, "CONSTRAINT") == 0 ||
3907 [ + + ]: 2077 : strcmp(type, "DATABASE PROPERTIES") == 0 ||
3908 [ + + ]: 2067 : strcmp(type, "DEFAULT") == 0 ||
3909 [ + + ]: 1907 : strcmp(type, "FK CONSTRAINT") == 0 ||
3910 [ + + ]: 1692 : strcmp(type, "INDEX") == 0 ||
3911 [ + + ]: 833 : strcmp(type, "RULE") == 0 ||
3912 [ + + ]: 622 : strcmp(type, "TRIGGER") == 0 ||
3913 [ + - ]: 242 : strcmp(type, "ROW SECURITY") == 0 ||
3914 [ + + ]: 242 : strcmp(type, "POLICY") == 0 ||
3915 [ - + ]: 29 : strcmp(type, "USER MAPPING") == 0)
3916 : : {
3917 : : /* do nothing */
3918 : : }
3919 : : else
1336 peter@eisentraut.org 3920 :UBC 0 : pg_fatal("don't know how to set owner for object type \"%s\"", type);
3921 : : }
3922 : :
3923 : : /*
3924 : : * Emit the SQL commands to create the object represented by a TOC entry
3925 : : *
3926 : : * This now also includes issuing an ALTER OWNER command to restore the
3927 : : * object's ownership, if wanted. But note that the object's permissions
3928 : : * will remain at default, until the matching ACL TOC entry is restored.
3929 : : */
3930 : : static void
495 jdavis@postgresql.or 3931 :CBC 42099 : _printTocEntry(ArchiveHandle *AH, TocEntry *te, const char *pfx)
3932 : : {
3821 tgl@sss.pgh.pa.us 3933 : 42099 : RestoreOptions *ropt = AH->public.ropt;
3934 : :
3935 : : /*
3936 : : * Select owner, schema, tablespace and default AM as necessary. The
3937 : : * default access method for partitioned tables is handled after
3938 : : * generating the object definition, as it requires an ALTER command
3939 : : * rather than SET.
3940 : : */
7991 3941 : 42099 : _becomeOwner(AH, te);
3942 : 42099 : _selectOutputSchema(AH, te->namespace);
7906 3943 : 42099 : _selectTablespace(AH, te->tablespace);
799 michael@paquier.xyz 3944 [ + + ]: 42099 : if (te->relkind != RELKIND_PARTITIONED_TABLE)
3945 : 41507 : _selectTableAccessMethod(AH, te->tableam);
3946 : :
3947 : : /* Emit header comment for item */
7974 tgl@sss.pgh.pa.us 3948 [ + + ]: 42099 : if (!AH->noTocComments)
3949 : : {
3950 : : char *sanitized_name;
3951 : : char *sanitized_schema;
3952 : : char *sanitized_owner;
3953 : :
3954 : 38325 : ahprintf(AH, "--\n");
3955 [ + + ]: 38325 : if (AH->public.verbose)
3956 : : {
3957 : 1176 : ahprintf(AH, "-- TOC entry %d (class %u OID %u)\n",
3958 : : te->dumpId, te->catalogId.tableoid, te->catalogId.oid);
3959 [ + + ]: 1176 : if (te->nDeps > 0)
3960 : : {
3961 : : int i;
3962 : :
3963 : 772 : ahprintf(AH, "-- Dependencies:");
3964 [ + + ]: 2102 : for (i = 0; i < te->nDeps; i++)
3965 : 1330 : ahprintf(AH, " %d", te->dependencies[i]);
3966 : 772 : ahprintf(AH, "\n");
3967 : : }
3968 : : }
3969 : :
2706 alvherre@alvh.no-ip. 3970 : 38325 : sanitized_name = sanitize_line(te->tag, false);
3971 : 38325 : sanitized_schema = sanitize_line(te->namespace, true);
3972 [ + + ]: 38325 : sanitized_owner = sanitize_line(ropt->noOwner ? NULL : te->owner, true);
3973 : :
7906 tgl@sss.pgh.pa.us 3974 : 38325 : ahprintf(AH, "-- %sName: %s; Type: %s; Schema: %s; Owner: %s",
3975 : : pfx, sanitized_name, te->desc, sanitized_schema,
3976 : : sanitized_owner);
3977 : :
5241 3978 : 38325 : free(sanitized_name);
3979 : 38325 : free(sanitized_schema);
3980 : 38325 : free(sanitized_owner);
3981 : :
4068 bruce@momjian.us 3982 [ + + + + : 38325 : if (te->tablespace && strlen(te->tablespace) > 0 && !ropt->noTablespace)
+ - ]
3983 : : {
3984 : : char *sanitized_tablespace;
3985 : :
2706 alvherre@alvh.no-ip. 3986 : 102 : sanitized_tablespace = sanitize_line(te->tablespace, false);
5241 tgl@sss.pgh.pa.us 3987 : 102 : ahprintf(AH, "; Tablespace: %s", sanitized_tablespace);
3988 : 102 : free(sanitized_tablespace);
3989 : : }
7906 3990 : 38325 : ahprintf(AH, "\n");
3991 : :
3331 bruce@momjian.us 3992 [ + + ]: 38325 : if (AH->PrintExtraTocPtr != NULL)
3218 peter_e@gmx.net 3993 : 3318 : AH->PrintExtraTocPtr(AH, te);
7974 tgl@sss.pgh.pa.us 3994 : 38325 : ahprintf(AH, "--\n\n");
3995 : : }
3996 : :
3997 : : /*
3998 : : * Actually print the definition. Normally we can just print the defn
3999 : : * string if any, but we have four special cases:
4000 : : *
4001 : : * 1. A crude hack for suppressing AUTHORIZATION clause that old pg_dump
4002 : : * versions put into CREATE SCHEMA. Don't mutate the variant for schema
4003 : : * "public" that is a comment. We have to do this when --no-owner mode is
4004 : : * selected. This is ugly, but I see no other good way ...
4005 : : *
4006 : : * 2. BLOB METADATA entries need special processing since their defn
4007 : : * strings are just lists of OIDs, not complete SQL commands.
4008 : : *
4009 : : * 3. ACL LARGE OBJECTS entries need special processing because they
4010 : : * contain only one copy of the ACL GRANT/REVOKE commands, which we must
4011 : : * apply to each large object listed in the associated BLOB METADATA.
4012 : : *
4013 : : * 4. Entries with a defnDumper need to call it to generate the
4014 : : * definition. This is primarily intended to provide a way to save memory
4015 : : * for objects that would otherwise need a lot of it (e.g., statistics
4016 : : * data).
4017 : : */
1828 noah@leadboat.com 4018 [ + + ]: 42099 : if (ropt->noOwner &&
4019 [ + + + + ]: 405 : strcmp(te->desc, "SCHEMA") == 0 && strncmp(te->defn, "--", 2) != 0)
4020 : : {
7840 tgl@sss.pgh.pa.us 4021 : 2 : ahprintf(AH, "CREATE SCHEMA %s;\n\n\n", fmtId(te->tag));
4022 : : }
820 4023 [ + + ]: 42097 : else if (strcmp(te->desc, "BLOB METADATA") == 0)
4024 : : {
4025 : 74 : IssueCommandPerBlob(AH, te, "SELECT pg_catalog.lo_create('", "')");
4026 : : }
4027 [ + + ]: 42023 : else if (strcmp(te->desc, "ACL") == 0 &&
4028 [ - + ]: 1807 : strncmp(te->tag, "LARGE OBJECTS", 13) == 0)
4029 : : {
820 tgl@sss.pgh.pa.us 4030 :UBC 0 : IssueACLPerBlob(AH, te);
4031 : : }
452 nathan@postgresql.or 4032 [ + + - + ]:CBC 42023 : else if (te->defnLen && AH->format != archTar)
4033 : : {
4034 : : /*
4035 : : * If defnLen is set, the defnDumper has already been called for this
4036 : : * TOC entry. We don't normally expect a defnDumper to be called for
4037 : : * a TOC entry a second time in _printTocEntry(), but there's an
4038 : : * exception. The tar format first calls WriteToc(), which scans the
4039 : : * entire TOC, and then it later calls RestoreArchive() to generate
4040 : : * restore.sql, which scans the TOC again. There doesn't appear to be
4041 : : * a good way to prevent a second defnDumper call in this case without
4042 : : * storing the definition in memory, which defeats the purpose. This
4043 : : * second defnDumper invocation should generate the same output as the
4044 : : * first, but even if it doesn't, the worst-case scenario is that
4045 : : * restore.sql might have different statistics data than the archive.
4046 : : *
4047 : : * In all other cases, encountering a TOC entry a second time in
4048 : : * _printTocEntry() is unexpected, so we fail because one of our
4049 : : * assumptions must no longer hold true.
4050 : : *
4051 : : * XXX This is a layering violation, but the alternative is an awkward
4052 : : * and complicated callback infrastructure for this special case. This
4053 : : * might be worth revisiting in the future.
4054 : : */
452 nathan@postgresql.or 4055 :UBC 0 : pg_fatal("unexpected TOC entry in _printTocEntry(): %d %s %s",
4056 : : te->dumpId, te->desc, te->tag);
4057 : : }
452 nathan@postgresql.or 4058 [ + + ]:CBC 42023 : else if (te->defnDumper)
4059 : : {
4060 : 1738 : char *defn = te->defnDumper((Archive *) AH, te->defnDumperArg, te);
4061 : :
4062 : 1738 : te->defnLen = ahprintf(AH, "%s\n\n", defn);
4063 : 1738 : pg_free(defn);
4064 : : }
701 tgl@sss.pgh.pa.us 4065 [ + + + - ]: 40285 : else if (te->defn && strlen(te->defn) > 0)
4066 : : {
4067 : 35820 : ahprintf(AH, "%s\n\n", te->defn);
4068 : :
4069 : : /*
4070 : : * If the defn string contains multiple SQL commands, txn_size mode
4071 : : * should count it as N actions not one. But rather than build a full
4072 : : * SQL parser, approximate this by counting semicolons. One case
4073 : : * where that tends to be badly fooled is function definitions, so
4074 : : * ignore them. (restore_toc_entry will count one action anyway.)
4075 : : */
4076 [ + + ]: 35820 : if (ropt->txn_size > 0 &&
4077 [ + + ]: 3661 : strcmp(te->desc, "FUNCTION") != 0 &&
4078 [ + + ]: 3378 : strcmp(te->desc, "PROCEDURE") != 0)
4079 : : {
4080 : 3366 : const char *p = te->defn;
4081 : 3366 : int nsemis = 0;
4082 : :
4083 [ + + ]: 14910 : while ((p = strchr(p, ';')) != NULL)
4084 : : {
4085 : 11544 : nsemis++;
4086 : 11544 : p++;
4087 : : }
4088 [ + + ]: 3366 : if (nsemis > 1)
4089 : 1644 : AH->txnCount += nsemis - 1;
4090 : : }
4091 : : }
4092 : :
4093 : : /*
4094 : : * If we aren't using SET SESSION AUTH to determine ownership, we must
4095 : : * instead issue an ALTER OWNER command. Schema "public" is special; when
4096 : : * a dump emits a comment in lieu of creating it, we use ALTER OWNER even
4097 : : * when using SET SESSION for all other objects. We assume that anything
4098 : : * without a DROP command is not a separately ownable object.
4099 : : */
1828 noah@leadboat.com 4100 [ + + ]: 42099 : if (!ropt->noOwner &&
4101 [ - + ]: 41694 : (!ropt->use_setsessauth ||
1828 noah@leadboat.com 4102 [ # # ]:UBC 0 : (strcmp(te->desc, "SCHEMA") == 0 &&
4103 [ # # ]: 0 : strncmp(te->defn, "--", 2) == 0)) &&
2622 alvherre@alvh.no-ip. 4104 [ + + + + ]:CBC 41694 : te->owner && strlen(te->owner) > 0 &&
4105 [ + + + + ]: 37888 : te->dropStmt && strlen(te->dropStmt) > 0)
4106 : : {
820 tgl@sss.pgh.pa.us 4107 [ + + ]: 21445 : if (strcmp(te->desc, "BLOB METADATA") == 0)
4108 : : {
4109 : : /* BLOB METADATA needs special code to handle multiple LOs */
4110 : 72 : char *cmdEnd = psprintf(" OWNER TO %s", fmtId(te->owner));
4111 : :
4112 : 72 : IssueCommandPerBlob(AH, te, "ALTER LARGE OBJECT ", cmdEnd);
4113 : 72 : pg_free(cmdEnd);
4114 : : }
4115 : : else
4116 : : {
4117 : : /* For all other cases, we can use _getObjectDescription */
4118 : : PQExpBufferData temp;
4119 : :
4120 : 21373 : initPQExpBuffer(&temp);
4121 : 21373 : _getObjectDescription(&temp, te);
4122 : :
4123 : : /*
4124 : : * If _getObjectDescription() didn't fill the buffer, then there
4125 : : * is no owner.
4126 : : */
4127 [ + + ]: 21373 : if (temp.data[0])
4128 : 17609 : ahprintf(AH, "ALTER %s OWNER TO %s;\n\n",
4129 : 17609 : temp.data, fmtId(te->owner));
4130 : 21373 : termPQExpBuffer(&temp);
4131 : : }
4132 : : }
4133 : :
4134 : : /*
4135 : : * Select a partitioned table's default AM, once the table definition has
4136 : : * been generated.
4137 : : */
799 michael@paquier.xyz 4138 [ + + ]: 42099 : if (te->relkind == RELKIND_PARTITIONED_TABLE)
4139 : 592 : _printTableAccessMethodNoStorage(AH, te);
4140 : :
4141 : : /*
4142 : : * If it's an ACL entry, it might contain SET SESSION AUTHORIZATION
4143 : : * commands, so we can no longer assume we know the current auth setting.
4144 : : */
3253 tgl@sss.pgh.pa.us 4145 [ + + ]: 42099 : if (_tocEntryIsACL(te))
4146 : : {
1475 peter@eisentraut.org 4147 : 1935 : free(AH->currUser);
8016 tgl@sss.pgh.pa.us 4148 : 1935 : AH->currUser = NULL;
4149 : : }
9492 bruce@momjian.us 4150 : 42099 : }
4151 : :
4152 : : /*
4153 : : * Write the file header for a custom-format archive
4154 : : */
4155 : : void
9231 4156 : 63 : WriteHead(ArchiveHandle *AH)
4157 : : {
4158 : : struct tm crtm;
4159 : :
3218 peter_e@gmx.net 4160 : 63 : AH->WriteBufPtr(AH, "PGDMP", 5); /* Magic code */
4161 : 63 : AH->WriteBytePtr(AH, ARCHIVE_MAJOR(AH->version));
4162 : 63 : AH->WriteBytePtr(AH, ARCHIVE_MINOR(AH->version));
4163 : 63 : AH->WriteBytePtr(AH, ARCHIVE_REV(AH->version));
4164 : 63 : AH->WriteBytePtr(AH, AH->intSize);
4165 : 63 : AH->WriteBytePtr(AH, AH->offSize);
4166 : 63 : AH->WriteBytePtr(AH, AH->format);
1223 tomas.vondra@postgre 4167 : 63 : AH->WriteBytePtr(AH, AH->compression_spec.algorithm);
9475 pjw@rhyme.com.au 4168 : 63 : crtm = *localtime(&AH->createDate);
4169 : 63 : WriteInt(AH, crtm.tm_sec);
4170 : 63 : WriteInt(AH, crtm.tm_min);
4171 : 63 : WriteInt(AH, crtm.tm_hour);
4172 : 63 : WriteInt(AH, crtm.tm_mday);
4173 : 63 : WriteInt(AH, crtm.tm_mon);
4174 : 63 : WriteInt(AH, crtm.tm_year);
4175 : 63 : WriteInt(AH, crtm.tm_isdst);
8725 peter_e@gmx.net 4176 : 63 : WriteStr(AH, PQdb(AH->connection));
7906 tgl@sss.pgh.pa.us 4177 : 63 : WriteStr(AH, AH->public.remoteVersionStr);
4178 : 63 : WriteStr(AH, PG_VERSION);
9492 bruce@momjian.us 4179 : 63 : }
4180 : :
4181 : : void
9231 4182 : 65 : ReadHead(ArchiveHandle *AH)
4183 : : {
4184 : : char *errmsg;
4185 : : char vmaj,
4186 : : vmin,
4187 : : vrev;
4188 : : int fmt;
4189 : :
4190 : : /*
4191 : : * If we haven't already read the header, do so.
4192 : : *
4193 : : * NB: this code must agree with _discoverArchiveFormat(). Maybe find a
4194 : : * way to unify the cases?
4195 : : */
4196 [ + - ]: 65 : if (!AH->readHeader)
4197 : : {
4198 : : char tmpMag[7];
4199 : :
3218 peter_e@gmx.net 4200 : 65 : AH->ReadBufPtr(AH, tmpMag, 5);
4201 : :
9231 bruce@momjian.us 4202 [ - + ]: 65 : if (strncmp(tmpMag, "PGDMP", 5) != 0)
1544 tgl@sss.pgh.pa.us 4203 :UBC 0 : pg_fatal("did not find magic string in file header");
4204 : : }
4205 : :
1916 tgl@sss.pgh.pa.us 4206 :CBC 65 : vmaj = AH->ReadBytePtr(AH);
4207 : 65 : vmin = AH->ReadBytePtr(AH);
4208 : :
4209 [ + - + - : 65 : if (vmaj > 1 || (vmaj == 1 && vmin > 0)) /* Version > 1.0 */
+ - ]
4210 : 65 : vrev = AH->ReadBytePtr(AH);
4211 : : else
1916 tgl@sss.pgh.pa.us 4212 :UBC 0 : vrev = 0;
4213 : :
1916 tgl@sss.pgh.pa.us 4214 :CBC 65 : AH->version = MAKE_ARCHIVE_VERSION(vmaj, vmin, vrev);
4215 : :
4216 [ + - - + ]: 65 : if (AH->version < K_VERS_1_0 || AH->version > K_VERS_MAX)
1544 tgl@sss.pgh.pa.us 4217 :UBC 0 : pg_fatal("unsupported version (%d.%d) in file header",
4218 : : vmaj, vmin);
4219 : :
1916 tgl@sss.pgh.pa.us 4220 :CBC 65 : AH->intSize = AH->ReadBytePtr(AH);
4221 [ - + ]: 65 : if (AH->intSize > 32)
203 peter@eisentraut.org 4222 :UNC 0 : pg_fatal("sanity check on integer size (%zu) failed", AH->intSize);
4223 : :
1916 tgl@sss.pgh.pa.us 4224 [ - + ]:CBC 65 : if (AH->intSize > sizeof(int))
1916 tgl@sss.pgh.pa.us 4225 :UBC 0 : pg_log_warning("archive was made on a machine with larger integers, some operations might fail");
4226 : :
1916 tgl@sss.pgh.pa.us 4227 [ + - ]:CBC 65 : if (AH->version >= K_VERS_1_7)
4228 : 65 : AH->offSize = AH->ReadBytePtr(AH);
4229 : : else
1916 tgl@sss.pgh.pa.us 4230 :UBC 0 : AH->offSize = AH->intSize;
4231 : :
1916 tgl@sss.pgh.pa.us 4232 :CBC 65 : fmt = AH->ReadBytePtr(AH);
4233 : :
4234 [ - + ]: 65 : if (AH->format != fmt)
1544 tgl@sss.pgh.pa.us 4235 :UBC 0 : pg_fatal("expected format (%d) differs from format found in file (%d)",
4236 : : AH->format, fmt);
4237 : :
1223 tomas.vondra@postgre 4238 [ + - ]:CBC 65 : if (AH->version >= K_VERS_1_15)
4239 : 65 : AH->compression_spec.algorithm = AH->ReadBytePtr(AH);
1223 tomas.vondra@postgre 4240 [ # # ]:UBC 0 : else if (AH->version >= K_VERS_1_2)
4241 : : {
4242 : : /* Guess the compression method based on the level */
9475 pjw@rhyme.com.au 4243 [ # # ]: 0 : if (AH->version < K_VERS_1_4)
1306 michael@paquier.xyz 4244 : 0 : AH->compression_spec.level = AH->ReadBytePtr(AH);
4245 : : else
4246 : 0 : AH->compression_spec.level = ReadInt(AH);
4247 : :
4248 [ # # ]: 0 : if (AH->compression_spec.level != 0)
4249 : 0 : AH->compression_spec.algorithm = PG_COMPRESSION_GZIP;
4250 : : }
4251 : : else
4252 : 0 : AH->compression_spec.algorithm = PG_COMPRESSION_GZIP;
4253 : :
1223 tomas.vondra@postgre 4254 :CBC 65 : errmsg = supports_compression(AH->compression_spec);
4255 [ - + ]: 65 : if (errmsg)
4256 : : {
1223 tomas.vondra@postgre 4257 :UBC 0 : pg_log_warning("archive is compressed, but this installation does not support compression (%s) -- no data will be available",
4258 : : errmsg);
4259 : 0 : pg_free(errmsg);
4260 : : }
4261 : :
9475 pjw@rhyme.com.au 4262 [ + - ]:CBC 65 : if (AH->version >= K_VERS_1_4)
4263 : : {
4264 : : struct tm crtm;
4265 : :
4266 : 65 : crtm.tm_sec = ReadInt(AH);
4267 : 65 : crtm.tm_min = ReadInt(AH);
4268 : 65 : crtm.tm_hour = ReadInt(AH);
4269 : 65 : crtm.tm_mday = ReadInt(AH);
4270 : 65 : crtm.tm_mon = ReadInt(AH);
4271 : 65 : crtm.tm_year = ReadInt(AH);
4272 : 65 : crtm.tm_isdst = ReadInt(AH);
4273 : :
4274 : : /*
4275 : : * Newer versions of glibc have mktime() report failure if tm_isdst is
4276 : : * inconsistent with the prevailing timezone, e.g. tm_isdst = 1 when
4277 : : * TZ=UTC. This is problematic when restoring an archive under a
4278 : : * different timezone setting. If we get a failure, try again with
4279 : : * tm_isdst set to -1 ("don't know").
4280 : : *
4281 : : * XXX with or without this hack, we reconstruct createDate
4282 : : * incorrectly when the prevailing timezone is different from
4283 : : * pg_dump's. Next time we bump the archive version, we should flush
4284 : : * this representation and store a plain seconds-since-the-Epoch
4285 : : * timestamp instead.
4286 : : */
4287 : 65 : AH->createDate = mktime(&crtm);
9231 bruce@momjian.us 4288 [ - + ]: 65 : if (AH->createDate == (time_t) -1)
4289 : : {
1843 tgl@sss.pgh.pa.us 4290 :UBC 0 : crtm.tm_isdst = -1;
4291 : 0 : AH->createDate = mktime(&crtm);
4292 [ # # ]: 0 : if (AH->createDate == (time_t) -1)
4293 : 0 : pg_log_warning("invalid creation date in header");
4294 : : }
4295 : : }
4296 : :
1843 tgl@sss.pgh.pa.us 4297 [ + - ]:CBC 65 : if (AH->version >= K_VERS_1_4)
4298 : : {
4299 : 65 : AH->archdbname = ReadStr(AH);
4300 : : }
4301 : :
7906 4302 [ + - ]: 65 : if (AH->version >= K_VERS_1_10)
4303 : : {
4304 : 65 : AH->archiveRemoteVersion = ReadStr(AH);
4305 : 65 : AH->archiveDumpVersion = ReadStr(AH);
4306 : : }
9492 bruce@momjian.us 4307 : 65 : }
4308 : :
4309 : :
4310 : : /*
4311 : : * checkSeek
4312 : : * check to see if ftell/fseek can be performed.
4313 : : */
4314 : : bool
8649 4315 : 108 : checkSeek(FILE *fp)
4316 : : {
4317 : : pgoff_t tpos;
4318 : :
4319 : : /* Check that ftello works on this file */
5846 tgl@sss.pgh.pa.us 4320 : 108 : tpos = ftello(fp);
4524 sfrost@snowman.net 4321 [ + + ]: 108 : if (tpos < 0)
5846 tgl@sss.pgh.pa.us 4322 : 1 : return false;
4323 : :
4324 : : /*
4325 : : * Check that fseeko(SEEK_SET) works, too. NB: we used to try to test
4326 : : * this with fseeko(fp, 0, SEEK_CUR). But some platforms treat that as a
4327 : : * successful no-op even on files that are otherwise unseekable.
4328 : : */
4329 [ - + ]: 107 : if (fseeko(fp, tpos, SEEK_SET) != 0)
5846 tgl@sss.pgh.pa.us 4330 :UBC 0 : return false;
4331 : :
5846 tgl@sss.pgh.pa.us 4332 :CBC 107 : return true;
4333 : : }
4334 : :
4335 : :
4336 : : /*
4337 : : * dumpTimestamp
4338 : : */
4339 : : static void
7746 4340 : 86 : dumpTimestamp(ArchiveHandle *AH, const char *msg, time_t tim)
4341 : : {
4342 : : char buf[64];
4343 : :
4265 4344 [ + - ]: 86 : if (strftime(buf, sizeof(buf), PGDUMP_STRFTIME_FMT, localtime(&tim)) != 0)
7746 4345 : 86 : ahprintf(AH, "-- %s %s\n\n", msg, buf);
4346 : 86 : }
4347 : :
4348 : : /*
4349 : : * Main engine for parallel restore.
4350 : : *
4351 : : * Parallel restore is done in three phases. In this first phase,
4352 : : * we'll process all SECTION_PRE_DATA TOC entries that are allowed to be
4353 : : * processed in the RESTORE_PASS_MAIN pass. (In practice, that's all
4354 : : * PRE_DATA items other than ACLs.) Entries we can't process now are
4355 : : * added to the pending_list for later phases to deal with.
4356 : : */
4357 : : static void
3253 4358 : 4 : restore_toc_entries_prefork(ArchiveHandle *AH, TocEntry *pending_list)
4359 : : {
4360 : : bool skipped_some;
4361 : : TocEntry *next_work_item;
4362 : :
2647 peter@eisentraut.org 4363 [ - + ]: 4 : pg_log_debug("entering restore_toc_entries_prefork");
4364 : :
4365 : : /* Adjust dependency information */
6357 andrew@dunslane.net 4366 : 4 : fix_dependencies(AH);
4367 : :
4368 : : /*
4369 : : * Do all the early stuff in a single connection in the parent. There's no
4370 : : * great point in running it in parallel, in fact it will actually run
4371 : : * faster in a single connection because we avoid all the connection and
4372 : : * setup overhead. Also, pre-9.2 pg_dump versions were not very good
4373 : : * about showing all the dependencies of SECTION_PRE_DATA items, so we do
4374 : : * not risk trying to process them out-of-order.
4375 : : *
4376 : : * Stuff that we can't do immediately gets added to the pending_list.
4377 : : * Note: we don't yet filter out entries that aren't going to be restored.
4378 : : * They might participate in dependency chains connecting entries that
4379 : : * should be restored, so we treat them as live until we actually process
4380 : : * them.
4381 : : *
4382 : : * Note: as of 9.2, it should be guaranteed that all PRE_DATA items appear
4383 : : * before DATA items, and all DATA items before POST_DATA items. That is
4384 : : * not certain to be true in older archives, though, and in any case use
4385 : : * of a list file would destroy that ordering (cf. SortTocFromFile). So
4386 : : * this loop cannot assume that it holds.
4387 : : */
3253 tgl@sss.pgh.pa.us 4388 : 4 : AH->restorePass = RESTORE_PASS_MAIN;
5611 4389 : 4 : skipped_some = false;
6171 4390 [ + + ]: 100 : for (next_work_item = AH->toc->next; next_work_item != AH->toc; next_work_item = next_work_item->next)
4391 : : {
3253 4392 : 96 : bool do_now = true;
4393 : :
5611 4394 [ + + ]: 96 : if (next_work_item->section != SECTION_PRE_DATA)
4395 : : {
4396 : : /* DATA and POST_DATA items are just ignored for now */
4397 [ + + ]: 46 : if (next_work_item->section == SECTION_DATA ||
4398 [ + - ]: 30 : next_work_item->section == SECTION_POST_DATA)
4399 : : {
3253 4400 : 46 : do_now = false;
5611 4401 : 46 : skipped_some = true;
4402 : : }
4403 : : else
4404 : : {
4405 : : /*
4406 : : * SECTION_NONE items, such as comments, can be processed now
4407 : : * if we are still in the PRE_DATA part of the archive. Once
4408 : : * we've skipped any items, we have to consider whether the
4409 : : * comment's dependencies are satisfied, so skip it for now.
4410 : : */
5611 tgl@sss.pgh.pa.us 4411 [ # # ]:UBC 0 : if (skipped_some)
3253 4412 : 0 : do_now = false;
4413 : : }
4414 : : }
4415 : :
4416 : : /*
4417 : : * Also skip items that need to be forced into later passes. We need
4418 : : * not set skipped_some in this case, since by assumption no main-pass
4419 : : * items could depend on these.
4420 : : */
452 nathan@postgresql.or 4421 [ - + ]:CBC 96 : if (_tocEntryRestorePass(next_work_item) != RESTORE_PASS_MAIN)
3253 tgl@sss.pgh.pa.us 4422 :UBC 0 : do_now = false;
4423 : :
3253 tgl@sss.pgh.pa.us 4424 [ + + ]:CBC 96 : if (do_now)
4425 : : {
4426 : : /* OK, restore the item and update its dependencies */
2647 peter@eisentraut.org 4427 : 50 : pg_log_info("processing item %d %s %s",
4428 : : next_work_item->dumpId,
4429 : : next_work_item->desc, next_work_item->tag);
4430 : :
3253 tgl@sss.pgh.pa.us 4431 : 50 : (void) restore_toc_entry(AH, next_work_item, false);
4432 : :
4433 : : /* Reduce dependencies, but don't move anything to ready_heap */
4434 : 50 : reduce_dependencies(AH, next_work_item, NULL);
4435 : : }
4436 : : else
4437 : : {
4438 : : /* Nope, so add it to pending_list */
2846 4439 : 46 : pending_list_append(pending_list, next_work_item);
4440 : : }
4441 : : }
4442 : :
4443 : : /*
4444 : : * In --transaction-size mode, we must commit the open transaction before
4445 : : * dropping the database connection. This also ensures that child workers
4446 : : * can see the objects we've created so far.
4447 : : */
820 4448 [ - + ]: 4 : if (AH->public.ropt->txn_size > 0)
820 tgl@sss.pgh.pa.us 4449 :UBC 0 : CommitTransaction(&AH->public);
4450 : :
4451 : : /*
4452 : : * Now close parent connection in prep for parallel steps. We do this
4453 : : * mainly to ensure that we don't exceed the specified number of parallel
4454 : : * connections.
4455 : : */
5248 rhaas@postgresql.org 4456 :CBC 4 : DisconnectDatabase(&AH->public);
4457 : :
4458 : : /* blow away any transient state from the old connection */
1475 peter@eisentraut.org 4459 : 4 : free(AH->currUser);
6357 andrew@dunslane.net 4460 : 4 : AH->currUser = NULL;
1475 peter@eisentraut.org 4461 : 4 : free(AH->currSchema);
6357 andrew@dunslane.net 4462 : 4 : AH->currSchema = NULL;
1475 peter@eisentraut.org 4463 : 4 : free(AH->currTablespace);
6357 andrew@dunslane.net 4464 : 4 : AH->currTablespace = NULL;
1475 peter@eisentraut.org 4465 : 4 : free(AH->currTableAm);
2673 andres@anarazel.de 4466 : 4 : AH->currTableAm = NULL;
4846 andrew@dunslane.net 4467 : 4 : }
4468 : :
4469 : : /*
4470 : : * Main engine for parallel restore.
4471 : : *
4472 : : * Parallel restore is done in three phases. In this second phase,
4473 : : * we process entries by dispatching them to parallel worker children
4474 : : * (processes on Unix, threads on Windows), each of which connects
4475 : : * separately to the database. Inter-entry dependencies are respected,
4476 : : * and so is the RestorePass multi-pass structure. When we can no longer
4477 : : * make any entries ready to process, we exit. Normally, there will be
4478 : : * nothing left to do; but if there is, the third phase will mop up.
4479 : : */
4480 : : static void
4481 : 4 : restore_toc_entries_parallel(ArchiveHandle *AH, ParallelState *pstate,
4482 : : TocEntry *pending_list)
4483 : : {
4484 : : binaryheap *ready_heap;
4485 : : TocEntry *next_work_item;
4486 : :
2647 peter@eisentraut.org 4487 [ - + ]: 4 : pg_log_debug("entering restore_toc_entries_parallel");
4488 : :
4489 : : /* Set up ready_heap with enough room for all known TocEntrys */
1015 nathan@postgresql.or 4490 : 4 : ready_heap = binaryheap_allocate(AH->tocCount,
4491 : : TocEntrySizeCompareBinaryheap,
4492 : : NULL);
4493 : :
4494 : : /*
4495 : : * The pending_list contains all items that we need to restore. Move all
4496 : : * items that are available to process immediately into the ready_heap.
4497 : : * After this setup, the pending list is everything that needs to be done
4498 : : * but is blocked by one or more dependencies, while the ready heap
4499 : : * contains items that have no remaining dependencies and are OK to
4500 : : * process in the current restore pass.
4501 : : */
3253 tgl@sss.pgh.pa.us 4502 : 4 : AH->restorePass = RESTORE_PASS_MAIN;
452 nathan@postgresql.or 4503 : 4 : move_to_ready_heap(pending_list, ready_heap, AH->restorePass);
4504 : :
4505 : : /*
4506 : : * main parent loop
4507 : : *
4508 : : * Keep going until there is no worker still running AND there is no work
4509 : : * left to be done. Note invariant: at top of loop, there should always
4510 : : * be at least one worker available to dispatch a job to.
4511 : : */
2647 peter@eisentraut.org 4512 : 4 : pg_log_info("entering main parallel loop");
4513 : :
4514 : : for (;;)
4515 : : {
4516 : : /* Look for an item ready to be dispatched to a worker */
1015 nathan@postgresql.or 4517 : 69 : next_work_item = pop_next_work_item(ready_heap, pstate);
6357 andrew@dunslane.net 4518 [ + + ]: 69 : if (next_work_item != NULL)
4519 : : {
4520 : : /* If not to be restored, don't waste time launching a worker */
495 jdavis@postgresql.or 4521 [ - + ]: 46 : if ((next_work_item->reqs & (REQ_SCHEMA | REQ_DATA | REQ_STATS)) == 0)
4522 : : {
2647 peter@eisentraut.org 4523 :UBC 0 : pg_log_info("skipping item %d %s %s",
4524 : : next_work_item->dumpId,
4525 : : next_work_item->desc, next_work_item->tag);
4526 : : /* Update its dependencies as though we'd completed it */
1015 nathan@postgresql.or 4527 : 0 : reduce_dependencies(AH, next_work_item, ready_heap);
4528 : : /* Loop around to see if anything else can be dispatched */
6357 andrew@dunslane.net 4529 : 0 : continue;
4530 : : }
4531 : :
2647 peter@eisentraut.org 4532 :CBC 46 : pg_log_info("launching item %d %s %s",
4533 : : next_work_item->dumpId,
4534 : : next_work_item->desc, next_work_item->tag);
4535 : :
4536 : : /* Dispatch to some worker */
3563 tgl@sss.pgh.pa.us 4537 : 46 : DispatchJobForTocEntry(AH, pstate, next_work_item, ACT_RESTORE,
4538 : : mark_restore_job_done, ready_heap);
4539 : : }
3253 4540 [ + + ]: 23 : else if (IsEveryWorkerIdle(pstate))
4541 : : {
4542 : : /*
4543 : : * Nothing is ready and no worker is running, so we're done with
4544 : : * the current pass or maybe with the whole process.
4545 : : */
4546 [ + + ]: 12 : if (AH->restorePass == RESTORE_PASS_LAST)
4547 : 4 : break; /* No more parallel processing is possible */
4548 : :
4549 : : /* Advance to next restore pass */
4550 : 8 : AH->restorePass++;
4551 : : /* That probably allows some stuff to be made ready */
452 nathan@postgresql.or 4552 : 8 : move_to_ready_heap(pending_list, ready_heap, AH->restorePass);
4553 : : /* Loop around to see if anything's now ready */
3253 tgl@sss.pgh.pa.us 4554 : 8 : continue;
4555 : : }
4556 : : else
4557 : : {
4558 : : /*
4559 : : * We have nothing ready, but at least one child is working, so
4560 : : * wait for some subjob to finish.
4561 : : */
4562 : : }
4563 : :
4564 : : /*
4565 : : * Before dispatching another job, check to see if anything has
4566 : : * finished. We should check every time through the loop so as to
4567 : : * reduce dependencies as soon as possible. If we were unable to
4568 : : * dispatch any job this time through, wait until some worker finishes
4569 : : * (and, hopefully, unblocks some pending item). If we did dispatch
4570 : : * something, continue as soon as there's at least one idle worker.
4571 : : * Note that in either case, there's guaranteed to be at least one
4572 : : * idle worker when we return to the top of the loop. This ensures we
4573 : : * won't block inside DispatchJobForTocEntry, which would be
4574 : : * undesirable: we'd rather postpone dispatching until we see what's
4575 : : * been unblocked by finished jobs.
4576 : : */
3563 4577 [ + + ]: 57 : WaitForWorkers(AH, pstate,
4578 : : next_work_item ? WFW_ONE_IDLE : WFW_GOT_STATUS);
4579 : : }
4580 : :
4581 : : /* There should now be nothing in ready_heap. */
1015 nathan@postgresql.or 4582 [ - + ]: 4 : Assert(binaryheap_empty(ready_heap));
4583 : :
4584 : 4 : binaryheap_free(ready_heap);
4585 : :
2647 peter@eisentraut.org 4586 : 4 : pg_log_info("finished main parallel loop");
4846 andrew@dunslane.net 4587 : 4 : }
4588 : :
4589 : : /*
4590 : : * Main engine for parallel restore.
4591 : : *
4592 : : * Parallel restore is done in three phases. In this third phase,
4593 : : * we mop up any remaining TOC entries by processing them serially.
4594 : : * This phase normally should have nothing to do, but if we've somehow
4595 : : * gotten stuck due to circular dependencies or some such, this provides
4596 : : * at least some chance of completing the restore successfully.
4597 : : */
4598 : : static void
4599 : 4 : restore_toc_entries_postfork(ArchiveHandle *AH, TocEntry *pending_list)
4600 : : {
3821 tgl@sss.pgh.pa.us 4601 : 4 : RestoreOptions *ropt = AH->public.ropt;
4602 : : TocEntry *te;
4603 : :
2647 peter@eisentraut.org 4604 [ - + ]: 4 : pg_log_debug("entering restore_toc_entries_postfork");
4605 : :
4606 : : /*
4607 : : * Now reconnect the single parent connection.
4608 : : */
452 andrew@dunslane.net 4609 : 4 : ConnectDatabaseAhx((Archive *) AH, &ropt->cparams, true);
4610 : :
4611 : : /* re-establish fixed state */
6357 4612 : 4 : _doSetFixedOutputState(AH);
4613 : :
4614 : : /*
4615 : : * Make sure there is no work left due to, say, circular dependencies, or
4616 : : * some other pathological condition. If so, do it in the single parent
4617 : : * connection. We don't sweat about RestorePass ordering; it's likely we
4618 : : * already violated that.
4619 : : */
2846 tgl@sss.pgh.pa.us 4620 [ - + ]: 4 : for (te = pending_list->pending_next; te != pending_list; te = te->pending_next)
4621 : : {
2647 peter@eisentraut.org 4622 :UBC 0 : pg_log_info("processing missed item %d %s %s",
4623 : : te->dumpId, te->desc, te->tag);
3821 tgl@sss.pgh.pa.us 4624 : 0 : (void) restore_toc_entry(AH, te, false);
4625 : : }
6357 andrew@dunslane.net 4626 :CBC 4 : }
4627 : :
4628 : : /*
4629 : : * Check if te1 has an exclusive lock requirement for an item that te2 also
4630 : : * requires, whether or not te2's requirement is for an exclusive lock.
4631 : : */
4632 : : static bool
6288 4633 : 152 : has_lock_conflicts(TocEntry *te1, TocEntry *te2)
4634 : : {
4635 : : int j,
4636 : : k;
4637 : :
4638 [ + + ]: 371 : for (j = 0; j < te1->nLockDeps; j++)
4639 : : {
4640 [ + + ]: 919 : for (k = 0; k < te2->nDeps; k++)
4641 : : {
4642 [ + + ]: 700 : if (te1->lockDeps[j] == te2->dependencies[k])
4643 : 2 : return true;
4644 : : }
4645 : : }
4646 : 150 : return false;
4647 : : }
4648 : :
4649 : :
4650 : : /*
4651 : : * Initialize the header of the pending-items list.
4652 : : *
4653 : : * This is a circular list with a dummy TocEntry as header, just like the
4654 : : * main TOC list; but we use separate list links so that an entry can be in
4655 : : * the main TOC list as well as in the pending list.
4656 : : */
4657 : : static void
2846 tgl@sss.pgh.pa.us 4658 : 4 : pending_list_header_init(TocEntry *l)
4659 : : {
4660 : 4 : l->pending_prev = l->pending_next = l;
4661 : 4 : }
4662 : :
4663 : : /* Append te to the end of the pending-list headed by l */
4664 : : static void
4665 : 46 : pending_list_append(TocEntry *l, TocEntry *te)
4666 : : {
4667 : 46 : te->pending_prev = l->pending_prev;
4668 : 46 : l->pending_prev->pending_next = te;
4669 : 46 : l->pending_prev = te;
4670 : 46 : te->pending_next = l;
4671 : 46 : }
4672 : :
4673 : : /* Remove te from the pending-list */
4674 : : static void
4675 : 46 : pending_list_remove(TocEntry *te)
4676 : : {
4677 : 46 : te->pending_prev->pending_next = te->pending_next;
4678 : 46 : te->pending_next->pending_prev = te->pending_prev;
4679 : 46 : te->pending_prev = NULL;
4680 : 46 : te->pending_next = NULL;
4681 : 46 : }
4682 : :
4683 : :
4684 : : /* qsort comparator for sorting TocEntries by dataLength */
4685 : : static int
1015 nathan@postgresql.or 4686 : 414 : TocEntrySizeCompareQsort(const void *p1, const void *p2)
4687 : : {
2846 tgl@sss.pgh.pa.us 4688 : 414 : const TocEntry *te1 = *(const TocEntry *const *) p1;
4689 : 414 : const TocEntry *te2 = *(const TocEntry *const *) p2;
4690 : :
4691 : : /* Sort by decreasing dataLength */
4692 [ + + ]: 414 : if (te1->dataLength > te2->dataLength)
4693 : 58 : return -1;
4694 [ + + ]: 356 : if (te1->dataLength < te2->dataLength)
4695 : 98 : return 1;
4696 : :
4697 : : /* For equal dataLengths, sort by dumpId, just to be stable */
4698 [ + + ]: 258 : if (te1->dumpId < te2->dumpId)
4699 : 105 : return -1;
4700 [ + + ]: 153 : if (te1->dumpId > te2->dumpId)
4701 : 140 : return 1;
4702 : :
4703 : 13 : return 0;
4704 : : }
4705 : :
4706 : : /* binaryheap comparator for sorting TocEntries by dataLength */
4707 : : static int
1015 nathan@postgresql.or 4708 : 143 : TocEntrySizeCompareBinaryheap(void *p1, void *p2, void *arg)
4709 : : {
4710 : : /* return opposite of qsort comparator for max-heap */
4711 : 143 : return -TocEntrySizeCompareQsort(&p1, &p2);
4712 : : }
4713 : :
4714 : :
4715 : : /*
4716 : : * Move all immediately-ready items from pending_list to ready_heap.
4717 : : *
4718 : : * Items are considered ready if they have no remaining dependencies and
4719 : : * they belong in the current restore pass. (See also reduce_dependencies,
4720 : : * which applies the same logic one-at-a-time.)
4721 : : */
4722 : : static void
452 4723 : 12 : move_to_ready_heap(TocEntry *pending_list,
4724 : : binaryheap *ready_heap,
4725 : : RestorePass pass)
4726 : : {
4727 : : TocEntry *te;
4728 : : TocEntry *next_te;
4729 : :
2846 tgl@sss.pgh.pa.us 4730 [ + + ]: 58 : for (te = pending_list->pending_next; te != pending_list; te = next_te)
4731 : : {
4732 : : /* must save list link before possibly removing te from list */
4733 : 46 : next_te = te->pending_next;
4734 : :
3253 4735 [ + + + - ]: 66 : if (te->depCount == 0 &&
452 nathan@postgresql.or 4736 : 20 : _tocEntryRestorePass(te) == pass)
4737 : : {
4738 : : /* Remove it from pending_list ... */
2846 tgl@sss.pgh.pa.us 4739 : 20 : pending_list_remove(te);
4740 : : /* ... and add to ready_heap */
1015 nathan@postgresql.or 4741 : 20 : binaryheap_add(ready_heap, te);
4742 : : }
4743 : : }
3253 tgl@sss.pgh.pa.us 4744 : 12 : }
4745 : :
4746 : : /*
4747 : : * Find the next work item (if any) that is capable of being run now,
4748 : : * and remove it from the ready_heap.
4749 : : *
4750 : : * Returns the item, or NULL if nothing is runnable.
4751 : : *
4752 : : * To qualify, the item must have no remaining dependencies
4753 : : * and no requirements for locks that are incompatible with
4754 : : * items currently running. Items in the ready_heap are known to have
4755 : : * no remaining dependencies, but we have to check for lock conflicts.
4756 : : */
4757 : : static TocEntry *
1015 nathan@postgresql.or 4758 : 69 : pop_next_work_item(binaryheap *ready_heap,
4759 : : ParallelState *pstate)
4760 : : {
4761 : : /*
4762 : : * Search the ready_heap until we find a suitable item. Note that we do a
4763 : : * sequential scan through the heap nodes, so even though we will first
4764 : : * try to choose the highest-priority item, we might end up picking
4765 : : * something with a much lower priority. However, we expect that we will
4766 : : * typically be able to pick one of the first few items, which should
4767 : : * usually have a relatively high priority.
4768 : : */
4769 [ + + ]: 71 : for (int i = 0; i < binaryheap_size(ready_heap); i++)
4770 : : {
4771 : 48 : TocEntry *te = (TocEntry *) binaryheap_get_node(ready_heap, i);
6228 bruce@momjian.us 4772 : 48 : bool conflicts = false;
4773 : :
4774 : : /*
4775 : : * Check to see if the item would need exclusive lock on something
4776 : : * that a currently running item also needs lock on, or vice versa. If
4777 : : * so, we don't want to schedule them together.
4778 : : */
2846 tgl@sss.pgh.pa.us 4779 [ + + ]: 186 : for (int k = 0; k < pstate->numWorkers; k++)
4780 : : {
4781 : 140 : TocEntry *running_te = pstate->te[k];
4782 : :
3563 4783 [ + + ]: 140 : if (running_te == NULL)
6357 andrew@dunslane.net 4784 : 63 : continue;
6288 4785 [ + + - + ]: 152 : if (has_lock_conflicts(te, running_te) ||
4786 : 75 : has_lock_conflicts(running_te, te))
4787 : : {
4788 : 2 : conflicts = true;
4789 : 2 : break;
4790 : : }
4791 : : }
4792 : :
6357 4793 [ + + ]: 48 : if (conflicts)
4794 : 2 : continue;
4795 : :
4796 : : /* passed all tests, so this item can run */
1015 nathan@postgresql.or 4797 : 46 : binaryheap_remove_node(ready_heap, i);
6357 andrew@dunslane.net 4798 : 46 : return te;
4799 : : }
4800 : :
2647 peter@eisentraut.org 4801 [ - + ]: 23 : pg_log_debug("no item ready");
6357 andrew@dunslane.net 4802 : 23 : return NULL;
4803 : : }
4804 : :
4805 : :
4806 : : /*
4807 : : * Restore a single TOC item in parallel with others
4808 : : *
4809 : : * this is run in the worker, i.e. in a thread (Windows) or a separate process
4810 : : * (everything else). A worker process executes several such work items during
4811 : : * a parallel backup or restore. Once we terminate here and report back that
4812 : : * our work is finished, the leader process will assign us a new work item.
4813 : : */
4814 : : int
3563 tgl@sss.pgh.pa.us 4815 : 46 : parallel_restore(ArchiveHandle *AH, TocEntry *te)
4816 : : {
4817 : : int status;
4818 : :
4846 andrew@dunslane.net 4819 [ - + ]: 46 : Assert(AH->connection != NULL);
4820 : :
4821 : : /* Count only errors associated with this TOC entry */
4822 : 46 : AH->public.n_errors = 0;
4823 : :
4824 : : /* Restore the TOC item */
3821 tgl@sss.pgh.pa.us 4825 : 46 : status = restore_toc_entry(AH, te, true);
4826 : :
4846 andrew@dunslane.net 4827 : 46 : return status;
4828 : : }
4829 : :
4830 : :
4831 : : /*
4832 : : * Callback function that's invoked in the leader process after a step has
4833 : : * been parallel restored.
4834 : : *
4835 : : * Update status and reduce the dependency count of any dependent items.
4836 : : */
4837 : : static void
3563 tgl@sss.pgh.pa.us 4838 : 46 : mark_restore_job_done(ArchiveHandle *AH,
4839 : : TocEntry *te,
4840 : : int status,
4841 : : void *callback_data)
4842 : : {
1015 nathan@postgresql.or 4843 : 46 : binaryheap *ready_heap = (binaryheap *) callback_data;
4844 : :
2647 peter@eisentraut.org 4845 : 46 : pg_log_info("finished item %d %s %s",
4846 : : te->dumpId, te->desc, te->tag);
4847 : :
6357 andrew@dunslane.net 4848 [ - + ]: 46 : if (status == WORKER_CREATE_DONE)
6357 andrew@dunslane.net 4849 :UBC 0 : mark_create_done(AH, te);
6357 andrew@dunslane.net 4850 [ - + ]:CBC 46 : else if (status == WORKER_INHIBIT_DATA)
4851 : : {
6357 andrew@dunslane.net 4852 :UBC 0 : inhibit_data_for_failed_table(AH, te);
4853 : 0 : AH->public.n_errors++;
4854 : : }
6357 andrew@dunslane.net 4855 [ - + ]:CBC 46 : else if (status == WORKER_IGNORED_ERRORS)
6357 andrew@dunslane.net 4856 :UBC 0 : AH->public.n_errors++;
6357 andrew@dunslane.net 4857 [ - + ]:CBC 46 : else if (status != 0)
1544 tgl@sss.pgh.pa.us 4858 :UBC 0 : pg_fatal("worker process failed: exit code %d",
4859 : : status);
4860 : :
1015 nathan@postgresql.or 4861 :CBC 46 : reduce_dependencies(AH, te, ready_heap);
6357 andrew@dunslane.net 4862 : 46 : }
4863 : :
4864 : :
4865 : : /*
4866 : : * Process the dependency information into a form useful for parallel restore.
4867 : : *
4868 : : * This function takes care of fixing up some missing or badly designed
4869 : : * dependencies, and then prepares subsidiary data structures that will be
4870 : : * used in the main parallel-restore logic, including:
4871 : : * 1. We build the revDeps[] arrays of incoming dependency dumpIds.
4872 : : * 2. We set up depCount fields that are the number of as-yet-unprocessed
4873 : : * dependencies for each TOC entry.
4874 : : *
4875 : : * We also identify locking dependencies so that we can avoid trying to
4876 : : * schedule conflicting items at the same time.
4877 : : */
4878 : : static void
4879 : 4 : fix_dependencies(ArchiveHandle *AH)
4880 : : {
4881 : : TocEntry *te;
4882 : : int i;
4883 : :
4884 : : /*
4885 : : * Initialize the depCount/revDeps/nRevDeps fields, and make sure the TOC
4886 : : * items are marked as not being in any parallel-processing list.
4887 : : */
4888 [ + + ]: 100 : for (te = AH->toc->next; te != AH->toc; te = te->next)
4889 : : {
4890 : 96 : te->depCount = te->nDeps;
5682 tgl@sss.pgh.pa.us 4891 : 96 : te->revDeps = NULL;
4892 : 96 : te->nRevDeps = 0;
2846 4893 : 96 : te->pending_prev = NULL;
4894 : 96 : te->pending_next = NULL;
4895 : : }
4896 : :
4897 : : /*
4898 : : * POST_DATA items that are shown as depending on a table need to be
4899 : : * re-pointed to depend on that table's data, instead. This ensures they
4900 : : * won't get scheduled until the data has been loaded.
4901 : : */
5146 4902 : 4 : repoint_table_dependencies(AH);
4903 : :
4904 : : /*
4905 : : * Pre-8.4 versions of pg_dump neglected to set up a dependency from BLOB
4906 : : * COMMENTS to BLOBS. Cope. (We assume there's only one BLOBS and only
4907 : : * one BLOB COMMENTS in such files.)
4908 : : */
6357 andrew@dunslane.net 4909 [ - + ]: 4 : if (AH->version < K_VERS_1_11)
4910 : : {
6357 andrew@dunslane.net 4911 [ # # ]:UBC 0 : for (te = AH->toc->next; te != AH->toc; te = te->next)
4912 : : {
4913 [ # # # # ]: 0 : if (strcmp(te->desc, "BLOB COMMENTS") == 0 && te->nDeps == 0)
4914 : : {
4915 : : TocEntry *te2;
4916 : :
4917 [ # # ]: 0 : for (te2 = AH->toc->next; te2 != AH->toc; te2 = te2->next)
4918 : : {
4919 [ # # ]: 0 : if (strcmp(te2->desc, "BLOBS") == 0)
4920 : : {
137 michael@paquier.xyz 4921 :UNC 0 : te->dependencies = pg_malloc_object(DumpId);
6357 andrew@dunslane.net 4922 :UBC 0 : te->dependencies[0] = te2->dumpId;
4923 : 0 : te->nDeps++;
4924 : 0 : te->depCount++;
4925 : 0 : break;
4926 : : }
4927 : : }
4928 : 0 : break;
4929 : : }
4930 : : }
4931 : : }
4932 : :
4933 : : /*
4934 : : * At this point we start to build the revDeps reverse-dependency arrays,
4935 : : * so all changes of dependencies must be complete.
4936 : : */
4937 : :
4938 : : /*
4939 : : * Count the incoming dependencies for each item. Also, it is possible
4940 : : * that the dependencies list items that are not in the archive at all
4941 : : * (that should not happen in 9.2 and later, but is highly likely in older
4942 : : * archives). Subtract such items from the depCounts.
4943 : : */
6357 andrew@dunslane.net 4944 [ + + ]:CBC 100 : for (te = AH->toc->next; te != AH->toc; te = te->next)
4945 : : {
4946 [ + + ]: 288 : for (i = 0; i < te->nDeps; i++)
4947 : : {
6006 tgl@sss.pgh.pa.us 4948 : 192 : DumpId depid = te->dependencies[i];
4949 : :
5146 4950 [ + - + - ]: 192 : if (depid <= AH->maxDumpId && AH->tocsByDumpId[depid] != NULL)
4951 : 192 : AH->tocsByDumpId[depid]->nRevDeps++;
4952 : : else
6357 andrew@dunslane.net 4953 :UBC 0 : te->depCount--;
4954 : : }
4955 : : }
4956 : :
4957 : : /*
4958 : : * Allocate space for revDeps[] arrays, and reset nRevDeps so we can use
4959 : : * it as a counter below.
4960 : : */
5682 tgl@sss.pgh.pa.us 4961 [ + + ]:CBC 100 : for (te = AH->toc->next; te != AH->toc; te = te->next)
4962 : : {
4963 [ + + ]: 96 : if (te->nRevDeps > 0)
137 michael@paquier.xyz 4964 :GNC 52 : te->revDeps = pg_malloc_array(DumpId, te->nRevDeps);
5682 tgl@sss.pgh.pa.us 4965 :CBC 96 : te->nRevDeps = 0;
4966 : : }
4967 : :
4968 : : /*
4969 : : * Build the revDeps[] arrays of incoming-dependency dumpIds. This had
4970 : : * better agree with the loops above.
4971 : : */
4972 [ + + ]: 100 : for (te = AH->toc->next; te != AH->toc; te = te->next)
4973 : : {
4974 [ + + ]: 288 : for (i = 0; i < te->nDeps; i++)
4975 : : {
4976 : 192 : DumpId depid = te->dependencies[i];
4977 : :
5146 4978 [ + - + - ]: 192 : if (depid <= AH->maxDumpId && AH->tocsByDumpId[depid] != NULL)
4979 : : {
4980 : 192 : TocEntry *otherte = AH->tocsByDumpId[depid];
4981 : :
5682 4982 : 192 : otherte->revDeps[otherte->nRevDeps++] = te->dumpId;
4983 : : }
4984 : : }
4985 : : }
4986 : :
4987 : : /*
4988 : : * Lastly, work out the locking dependencies.
4989 : : */
6357 andrew@dunslane.net 4990 [ + + ]: 100 : for (te = AH->toc->next; te != AH->toc; te = te->next)
4991 : : {
4992 : 96 : te->lockDeps = NULL;
4993 : 96 : te->nLockDeps = 0;
5146 tgl@sss.pgh.pa.us 4994 : 96 : identify_locking_dependencies(AH, te);
4995 : : }
6357 andrew@dunslane.net 4996 : 4 : }
4997 : :
4998 : : /*
4999 : : * Change dependencies on table items to depend on table data items instead,
5000 : : * but only in POST_DATA items.
5001 : : *
5002 : : * Also, for any item having such dependency(s), set its dataLength to the
5003 : : * largest dataLength of the table data items it depends on. This ensures
5004 : : * that parallel restore will prioritize larger jobs (index builds, FK
5005 : : * constraint checks, etc) over smaller ones, avoiding situations where we
5006 : : * end a restore with only one active job working on a large table.
5007 : : */
5008 : : static void
5146 tgl@sss.pgh.pa.us 5009 : 4 : repoint_table_dependencies(ArchiveHandle *AH)
5010 : : {
5011 : : TocEntry *te;
5012 : : int i;
5013 : : DumpId olddep;
5014 : :
6357 andrew@dunslane.net 5015 [ + + ]: 100 : for (te = AH->toc->next; te != AH->toc; te = te->next)
5016 : : {
5017 [ + + ]: 96 : if (te->section != SECTION_POST_DATA)
5018 : 66 : continue;
5019 [ + + ]: 160 : for (i = 0; i < te->nDeps; i++)
5020 : : {
5146 tgl@sss.pgh.pa.us 5021 : 130 : olddep = te->dependencies[i];
5022 [ + - ]: 130 : if (olddep <= AH->maxDumpId &&
5023 [ + + ]: 130 : AH->tableDataId[olddep] != 0)
5024 : : {
2846 5025 : 62 : DumpId tabledataid = AH->tableDataId[olddep];
5026 : 62 : TocEntry *tabledatate = AH->tocsByDumpId[tabledataid];
5027 : :
5028 : 62 : te->dependencies[i] = tabledataid;
5029 : 62 : te->dataLength = Max(te->dataLength, tabledatate->dataLength);
2647 peter@eisentraut.org 5030 [ - + ]: 62 : pg_log_debug("transferring dependency %d -> %d to %d",
5031 : : te->dumpId, olddep, tabledataid);
5032 : : }
5033 : : }
5034 : : }
6357 andrew@dunslane.net 5035 : 4 : }
5036 : :
5037 : : /*
5038 : : * Identify which objects we'll need exclusive lock on in order to restore
5039 : : * the given TOC entry (*other* than the one identified by the TOC entry
5040 : : * itself). Record their dump IDs in the entry's lockDeps[] array.
5041 : : */
5042 : : static void
5146 tgl@sss.pgh.pa.us 5043 : 96 : identify_locking_dependencies(ArchiveHandle *AH, TocEntry *te)
5044 : : {
5045 : : DumpId *lockids;
5046 : : int nlockids;
5047 : : int i;
5048 : :
5049 : : /*
5050 : : * We only care about this for POST_DATA items. PRE_DATA items are not
5051 : : * run in parallel, and DATA items are all independent by assumption.
5052 : : */
2863 5053 [ + + ]: 96 : if (te->section != SECTION_POST_DATA)
5054 : 66 : return;
5055 : :
5056 : : /* Quick exit if no dependencies at all */
6357 andrew@dunslane.net 5057 [ - + ]: 30 : if (te->nDeps == 0)
6357 andrew@dunslane.net 5058 :UBC 0 : return;
5059 : :
5060 : : /*
5061 : : * Most POST_DATA items are ALTER TABLEs or some moral equivalent of that,
5062 : : * and hence require exclusive lock. However, we know that CREATE INDEX
5063 : : * does not. (Maybe someday index-creating CONSTRAINTs will fall in that
5064 : : * category too ... but today is not that day.)
5065 : : */
2863 tgl@sss.pgh.pa.us 5066 [ - + ]:CBC 30 : if (strcmp(te->desc, "INDEX") == 0)
6357 andrew@dunslane.net 5067 :UBC 0 : return;
5068 : :
5069 : : /*
5070 : : * We assume the entry requires exclusive lock on each TABLE or TABLE DATA
5071 : : * item listed among its dependencies. Originally all of these would have
5072 : : * been TABLE items, but repoint_table_dependencies would have repointed
5073 : : * them to the TABLE DATA items if those are present (which they might not
5074 : : * be, eg in a schema-only dump). Note that all of the entries we are
5075 : : * processing here are POST_DATA; otherwise there might be a significant
5076 : : * difference between a dependency on a table and a dependency on its
5077 : : * data, so that closer analysis would be needed here.
5078 : : */
137 michael@paquier.xyz 5079 :GNC 30 : lockids = pg_malloc_array(DumpId, te->nDeps);
6357 andrew@dunslane.net 5080 :CBC 30 : nlockids = 0;
5081 [ + + ]: 160 : for (i = 0; i < te->nDeps; i++)
5082 : : {
6228 bruce@momjian.us 5083 : 130 : DumpId depid = te->dependencies[i];
5084 : :
5146 tgl@sss.pgh.pa.us 5085 [ + - + - ]: 130 : if (depid <= AH->maxDumpId && AH->tocsByDumpId[depid] != NULL &&
4295 rhaas@postgresql.org 5086 [ + + ]: 130 : ((strcmp(AH->tocsByDumpId[depid]->desc, "TABLE DATA") == 0) ||
4274 tgl@sss.pgh.pa.us 5087 [ + + ]: 68 : strcmp(AH->tocsByDumpId[depid]->desc, "TABLE") == 0))
6357 andrew@dunslane.net 5088 : 82 : lockids[nlockids++] = depid;
5089 : : }
5090 : :
5091 [ - + ]: 30 : if (nlockids == 0)
5092 : : {
6357 andrew@dunslane.net 5093 :UBC 0 : free(lockids);
5094 : 0 : return;
5095 : : }
5096 : :
137 michael@paquier.xyz 5097 :GNC 30 : te->lockDeps = pg_realloc_array(lockids, DumpId, nlockids);
6357 andrew@dunslane.net 5098 :CBC 30 : te->nLockDeps = nlockids;
5099 : : }
5100 : :
5101 : : /*
5102 : : * Remove the specified TOC entry from the depCounts of items that depend on
5103 : : * it, thereby possibly making them ready-to-run. Any pending item that
5104 : : * becomes ready should be moved to the ready_heap, if that's provided.
5105 : : */
5106 : : static void
2846 tgl@sss.pgh.pa.us 5107 : 96 : reduce_dependencies(ArchiveHandle *AH, TocEntry *te,
5108 : : binaryheap *ready_heap)
5109 : : {
5110 : : int i;
5111 : :
2647 peter@eisentraut.org 5112 [ - + ]: 96 : pg_log_debug("reducing dependencies for %d", te->dumpId);
5113 : :
5682 tgl@sss.pgh.pa.us 5114 [ + + ]: 288 : for (i = 0; i < te->nRevDeps; i++)
5115 : : {
5146 5116 : 192 : TocEntry *otherte = AH->tocsByDumpId[te->revDeps[i]];
5117 : :
3253 5118 [ - + ]: 192 : Assert(otherte->depCount > 0);
5682 5119 : 192 : otherte->depCount--;
5120 : :
5121 : : /*
5122 : : * It's ready if it has no remaining dependencies, and it belongs in
5123 : : * the current restore pass, and it is currently a member of the
5124 : : * pending list (that check is needed to prevent double restore in
5125 : : * some cases where a list-file forces out-of-order restoring).
5126 : : * However, if ready_heap == NULL then caller doesn't want any list
5127 : : * memberships changed.
5128 : : */
3253 5129 [ + + ]: 192 : if (otherte->depCount == 0 &&
452 nathan@postgresql.or 5130 [ + - ]: 74 : _tocEntryRestorePass(otherte) == AH->restorePass &&
2846 tgl@sss.pgh.pa.us 5131 [ + + + - ]: 74 : otherte->pending_prev != NULL &&
5132 : : ready_heap != NULL)
5133 : : {
5134 : : /* Remove it from pending list ... */
5135 : 26 : pending_list_remove(otherte);
5136 : : /* ... and add to ready_heap */
1015 nathan@postgresql.or 5137 : 26 : binaryheap_add(ready_heap, otherte);
5138 : : }
5139 : : }
6357 andrew@dunslane.net 5140 : 96 : }
5141 : :
5142 : : /*
5143 : : * Set the created flag on the DATA member corresponding to the given
5144 : : * TABLE member
5145 : : */
5146 : : static void
5147 : 5659 : mark_create_done(ArchiveHandle *AH, TocEntry *te)
5148 : : {
5146 tgl@sss.pgh.pa.us 5149 [ + + ]: 5659 : if (AH->tableDataId[te->dumpId] != 0)
5150 : : {
5151 : 4251 : TocEntry *ted = AH->tocsByDumpId[AH->tableDataId[te->dumpId]];
5152 : :
5153 : 4251 : ted->created = true;
5154 : : }
6357 andrew@dunslane.net 5155 : 5659 : }
5156 : :
5157 : : /*
5158 : : * Mark the DATA member corresponding to the given TABLE member
5159 : : * as not wanted
5160 : : */
5161 : : static void
6357 andrew@dunslane.net 5162 :UBC 0 : inhibit_data_for_failed_table(ArchiveHandle *AH, TocEntry *te)
5163 : : {
2647 peter@eisentraut.org 5164 : 0 : pg_log_info("table \"%s\" could not be created, will not restore its data",
5165 : : te->tag);
5166 : :
5146 tgl@sss.pgh.pa.us 5167 [ # # ]: 0 : if (AH->tableDataId[te->dumpId] != 0)
5168 : : {
5145 5169 : 0 : TocEntry *ted = AH->tocsByDumpId[AH->tableDataId[te->dumpId]];
5170 : :
5171 : 0 : ted->reqs = 0;
5172 : : }
6357 andrew@dunslane.net 5173 : 0 : }
5174 : :
5175 : : /*
5176 : : * Clone and de-clone routines used in parallel restoration.
5177 : : *
5178 : : * Enough of the structure is cloned to ensure that there is no
5179 : : * conflict between different threads each with their own clone.
5180 : : */
5181 : : ArchiveHandle *
6357 andrew@dunslane.net 5182 :CBC 26 : CloneArchive(ArchiveHandle *AH)
5183 : : {
5184 : : ArchiveHandle *clone;
5185 : :
5186 : : /* Make a "flat" copy */
137 michael@paquier.xyz 5187 :GNC 26 : clone = pg_malloc_object(ArchiveHandle);
6357 andrew@dunslane.net 5188 :CBC 26 : memcpy(clone, AH, sizeof(ArchiveHandle));
5189 : :
5190 : : /* Likewise flat-copy the RestoreOptions, so we can alter them locally */
137 michael@paquier.xyz 5191 :GNC 26 : clone->public.ropt = pg_malloc_object(RestoreOptions);
820 tgl@sss.pgh.pa.us 5192 :CBC 26 : memcpy(clone->public.ropt, AH->public.ropt, sizeof(RestoreOptions));
5193 : :
5194 : : /* Handle format-independent fields */
5289 5195 : 26 : memset(&(clone->sqlparse), 0, sizeof(clone->sqlparse));
5196 : :
5197 : : /* The clone will have its own connection, so disregard connection state */
6357 andrew@dunslane.net 5198 : 26 : clone->connection = NULL;
3680 tgl@sss.pgh.pa.us 5199 : 26 : clone->connCancel = NULL;
6357 andrew@dunslane.net 5200 : 26 : clone->currUser = NULL;
5201 : 26 : clone->currSchema = NULL;
1625 michael@paquier.xyz 5202 : 26 : clone->currTableAm = NULL;
6357 andrew@dunslane.net 5203 : 26 : clone->currTablespace = NULL;
5204 : :
5205 : : /* savedPassword must be local in case we change it while connecting */
5206 [ - + ]: 26 : if (clone->savedPassword)
5331 bruce@momjian.us 5207 :UBC 0 : clone->savedPassword = pg_strdup(clone->savedPassword);
5208 : :
5209 : : /* clone has its own error count, too */
6357 andrew@dunslane.net 5210 :CBC 26 : clone->public.n_errors = 0;
5211 : :
5212 : : /* clones should not share lo_buf */
820 tgl@sss.pgh.pa.us 5213 : 26 : clone->lo_buf = NULL;
5214 : :
5215 : : /*
5216 : : * Clone connections disregard --transaction-size; they must commit after
5217 : : * each command so that the results are immediately visible to other
5218 : : * workers.
5219 : : */
5220 : 26 : clone->public.ropt->txn_size = 0;
5221 : :
5222 : : /*
5223 : : * Connect our new clone object to the database, using the same connection
5224 : : * parameters used for the original connection.
5225 : : */
452 andrew@dunslane.net 5226 : 26 : ConnectDatabaseAhx((Archive *) clone, &clone->public.ropt->cparams, true);
5227 : :
5228 : : /* re-establish fixed state */
2105 tgl@sss.pgh.pa.us 5229 [ + + ]: 26 : if (AH->mode == archModeRead)
3681 5230 : 10 : _doSetFixedOutputState(clone);
5231 : : /* in write case, setupDumpWorker will fix up connection state */
5232 : :
5233 : : /* Let the format-specific code have a chance too */
3218 peter_e@gmx.net 5234 : 26 : clone->ClonePtr(clone);
5235 : :
4846 andrew@dunslane.net 5236 [ - + ]: 26 : Assert(clone->connection != NULL);
6357 5237 : 26 : return clone;
5238 : : }
5239 : :
5240 : : /*
5241 : : * Release clone-local storage.
5242 : : *
5243 : : * Note: we assume any clone-local connection was already closed.
5244 : : */
5245 : : void
5246 : 26 : DeCloneArchive(ArchiveHandle *AH)
5247 : : {
5248 : : /* Should not have an open database connection */
3680 tgl@sss.pgh.pa.us 5249 [ - + ]: 26 : Assert(AH->connection == NULL);
5250 : :
5251 : : /* Clear format-specific state */
3218 peter_e@gmx.net 5252 : 26 : AH->DeClonePtr(AH);
5253 : :
5254 : : /* Clear state allocated by CloneArchive */
5289 tgl@sss.pgh.pa.us 5255 [ + + ]: 26 : if (AH->sqlparse.curCmd)
5256 : 3 : destroyPQExpBuffer(AH->sqlparse.curCmd);
5257 : :
5258 : : /* Clear any connection-local state */
1475 peter@eisentraut.org 5259 : 26 : free(AH->currUser);
5260 : 26 : free(AH->currSchema);
5261 : 26 : free(AH->currTablespace);
5262 : 26 : free(AH->currTableAm);
5263 : 26 : free(AH->savedPassword);
5264 : :
6357 andrew@dunslane.net 5265 : 26 : free(AH);
5266 : 26 : }
|