forked from keenser/bdr
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbdr_executor.c
579 lines (473 loc) · 14.6 KB
/
bdr_executor.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
/* -------------------------------------------------------------------------
*
* bdr_executor.c
* Relation and index access and maintenance routines required by bdr
*
* BDR does a lot of direct access to indexes and relations, some of which
* isn't handled by simple calls into the backend. Most of it lives here.
*
* Copyright (C) 2012-2015, PostgreSQL Global Development Group
*
* IDENTIFICATION
* bdr_executor.c
*
* -------------------------------------------------------------------------
*/
#include "postgres.h"
#include "bdr.h"
#include "access/xact.h"
#include "catalog/indexing.h"
#include "catalog/pg_namespace.h"
#include "executor/executor.h"
#include "miscadmin.h"
#include "nodes/makefuncs.h"
#include "parser/parse_relation.h"
#include "parser/parsetree.h"
#include "replication/origin.h"
#include "storage/bufmgr.h"
#include "storage/lmgr.h"
#include "utils/builtins.h"
#include "utils/fmgroids.h"
#include "utils/lsyscache.h"
#include "utils/syscache.h"
#if PG_VERSION_NUM/100 == 904
/* Needed for InitDirtySnapshot on 9.4 */
#include "utils/tqual.h"
/* Needed for PushActiveSnapshot, GetLatestSnapshot and PopActiveSnapshot on 9.4 */
#include "utils/snapmgr.h"
#endif
static void BdrExecutorStart(QueryDesc *queryDesc, int eflags);
static ExecutorStart_hook_type PrevExecutorStart_hook = NULL;
static bool bdr_always_allow_writes = false;
PGDLLEXPORT Datum bdr_node_set_read_only(PG_FUNCTION_ARGS);
PG_FUNCTION_INFO_V1(bdr_node_set_read_only);
EState *
bdr_create_rel_estate(Relation rel)
{
EState *estate;
ResultRelInfo *resultRelInfo;
estate = CreateExecutorState();
resultRelInfo = makeNode(ResultRelInfo);
resultRelInfo->ri_RangeTableIndex = 1; /* dummy */
resultRelInfo->ri_RelationDesc = rel;
resultRelInfo->ri_TrigInstrument = NULL;
estate->es_result_relations = resultRelInfo;
estate->es_num_result_relations = 1;
estate->es_result_relation_info = resultRelInfo;
return estate;
}
void
UserTableUpdateIndexes(EState *estate, TupleTableSlot *slot)
{
/* HOT update does not require index inserts */
if (HeapTupleIsHeapOnly(slot->tts_tuple))
return;
ExecOpenIndices(estate->es_result_relation_info, false);
UserTableUpdateOpenIndexes(estate, slot);
ExecCloseIndices(estate->es_result_relation_info);
}
void
UserTableUpdateOpenIndexes(EState *estate, TupleTableSlot *slot)
{
List *recheckIndexes = NIL;
/* HOT update does not require index inserts */
if (HeapTupleIsHeapOnly(slot->tts_tuple))
return;
if (estate->es_result_relation_info->ri_NumIndices > 0)
{
recheckIndexes = ExecInsertIndexTuples(slot,
&slot->tts_tuple->t_self,
estate,
false, NULL, NIL);
if (recheckIndexes != NIL)
ereport(ERROR,
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
errmsg("bdr doesn't support index rechecks")));
}
/* FIXME: recheck the indexes */
list_free(recheckIndexes);
}
void
build_index_scan_keys(EState *estate, ScanKey *scan_keys, BDRTupleData *tup)
{
ResultRelInfo *relinfo;
int i;
relinfo = estate->es_result_relation_info;
/* build scankeys for each index */
for (i = 0; i < relinfo->ri_NumIndices; i++)
{
IndexInfo *ii = relinfo->ri_IndexRelationInfo[i];
/*
* Only unique indexes are of interest here, and we can't deal with
* expression indexes so far. FIXME: predicates should be handled
* better.
*/
if (!ii->ii_Unique || ii->ii_Expressions != NIL)
{
scan_keys[i] = NULL;
continue;
}
scan_keys[i] = palloc(ii->ii_NumIndexAttrs * sizeof(ScanKeyData));
/*
* Only return index if we could build a key without NULLs.
*/
if (build_index_scan_key(scan_keys[i],
relinfo->ri_RelationDesc,
relinfo->ri_IndexRelationDescs[i],
tup))
{
pfree(scan_keys[i]);
scan_keys[i] = NULL;
continue;
}
}
}
/*
* Setup a ScanKey for a search in the relation 'rel' for a tuple 'key' that
* is setup to match 'rel' (*NOT* idxrel!).
*
* Returns whether any column contains NULLs.
*/
bool
build_index_scan_key(ScanKey skey, Relation rel, Relation idxrel, BDRTupleData *tup)
{
int attoff;
Datum indclassDatum;
Datum indkeyDatum;
bool isnull;
oidvector *opclass;
int2vector *indkey;
bool hasnulls = false;
indclassDatum = SysCacheGetAttr(INDEXRELID, idxrel->rd_indextuple,
Anum_pg_index_indclass, &isnull);
Assert(!isnull);
opclass = (oidvector *) DatumGetPointer(indclassDatum);
indkeyDatum = SysCacheGetAttr(INDEXRELID, idxrel->rd_indextuple,
Anum_pg_index_indkey, &isnull);
Assert(!isnull);
indkey = (int2vector *) DatumGetPointer(indkeyDatum);
for (attoff = 0; attoff < RelationGetNumberOfAttributes(idxrel); attoff++)
{
Oid operator;
Oid opfamily;
RegProcedure regop;
int pkattno = attoff + 1;
int mainattno = indkey->values[attoff];
Oid atttype = attnumTypeId(rel, mainattno);
Oid optype = get_opclass_input_type(opclass->values[attoff]);
opfamily = get_opclass_family(opclass->values[attoff]);
operator = get_opfamily_member(opfamily, optype,
optype,
BTEqualStrategyNumber);
if (!OidIsValid(operator))
elog(ERROR,
"could not lookup equality operator for type %u, optype %u in opfamily %u",
atttype, optype, opfamily);
regop = get_opcode(operator);
/* FIXME: convert type? */
ScanKeyInit(&skey[attoff],
pkattno,
BTEqualStrategyNumber,
regop,
tup->values[mainattno - 1]);
if (tup->isnull[mainattno - 1])
{
hasnulls = true;
skey[attoff].sk_flags |= SK_ISNULL;
}
}
return hasnulls;
}
/*
* Search the index 'idxrel' for a tuple identified by 'skey' in 'rel'.
*
* If a matching tuple is found setup 'tid' to point to it and return true,
* false is returned otherwise.
*
* Populates 'slot' with a materialized copy of the found tuple in the memory
* context of the passed slot.
*/
bool
find_pkey_tuple(ScanKey skey, BDRRelation *rel, Relation idxrel,
TupleTableSlot *slot, bool lock, LockTupleMode mode)
{
HeapTuple scantuple;
bool found;
IndexScanDesc scan;
SnapshotData snap;
TransactionId xwait;
InitDirtySnapshot(snap);
retry:
found = false;
scan = index_beginscan(rel->rel, idxrel,
&snap,
RelationGetNumberOfAttributes(idxrel),
0);
index_rescan(scan, skey, RelationGetNumberOfAttributes(idxrel), NULL, 0);
if ((scantuple = index_getnext(scan, ForwardScanDirection)) != NULL)
{
found = true;
/*
* Store a copied physical tuple that doesn't reference shmem or hold
* any buffer pin, so it can live past the index scan. Any old tuple
* from a prior loop is cleared first.
*/
/* FIXME: Improve TupleSlot to not require copying the whole tuple */
ExecStoreTuple(scantuple, slot, InvalidBuffer, false);
ExecMaterializeSlot(slot);
xwait = TransactionIdIsValid(snap.xmin) ? snap.xmin : snap.xmax;
if (TransactionIdIsValid(xwait))
{
XactLockTableWait(xwait, NULL, NULL, XLTW_None);
index_endscan(scan);
goto retry;
}
}
if (lock && found)
{
Buffer buf;
HeapUpdateFailureData hufd;
HTSU_Result res;
HeapTupleData locktup;
ItemPointerCopy(&slot->tts_tuple->t_self, &locktup.t_self);
PushActiveSnapshot(GetLatestSnapshot());
res = heap_lock_tuple(rel->rel, &locktup, GetCurrentCommandId(false), mode,
false /* wait */,
false /* don't follow updates */,
&buf, &hufd);
/* the tuple slot already has the buffer pinned */
ReleaseBuffer(buf);
PopActiveSnapshot();
switch (res)
{
case HeapTupleMayBeUpdated:
break;
case HeapTupleUpdated:
/* XXX: Improve handling here */
ereport(LOG,
(errcode(ERRCODE_T_R_SERIALIZATION_FAILURE),
errmsg("concurrent update, retrying")));
index_endscan(scan);
goto retry;
default:
elog(ERROR, "unexpected HTSU_Result after locking: %u", res);
break;
}
}
index_endscan(scan);
return found;
}
void
bdr_node_set_read_only_internal(char *node_name, bool read_only, bool force)
{
HeapTuple tuple = NULL;
Relation rel;
RangeVar *rv;
SnapshotData SnapshotDirty;
SysScanDesc scan;
ScanKeyData key;
BdrNodeStatus status;
Assert(IsTransactionState());
/*
* We don't allow the user to clear read-only status
* while the local node is initing.
*/
status = bdr_local_node_status();
if ((status != BDR_NODE_STATUS_READY && status != BDR_NODE_STATUS_KILLED) && !force)
{
ereport(ERROR,
(errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
errmsg("local node is still starting up, cannot change read-only status.")));
}
InitDirtySnapshot(SnapshotDirty);
rv = makeRangeVar("bdr", "bdr_nodes", -1);
rel = heap_openrv(rv, RowExclusiveLock);
ScanKeyInit(&key,
get_attnum(rel->rd_id, "node_name"),
BTEqualStrategyNumber, F_TEXTEQ,
PointerGetDatum(cstring_to_text(node_name)));
scan = systable_beginscan(rel, InvalidOid,
true,
&SnapshotDirty,
1, &key);
tuple = systable_getnext(scan);
if (HeapTupleIsValid(tuple))
{
HeapTuple newtuple;
Datum *values;
bool *nulls;
TupleDesc tupDesc;
AttrNumber attnum = get_attnum(rel->rd_id, "node_read_only");
tupDesc = RelationGetDescr(rel);
values = (Datum *) palloc(tupDesc->natts * sizeof(Datum));
nulls = (bool *) palloc(tupDesc->natts * sizeof(bool));
heap_deform_tuple(tuple, tupDesc, values, nulls);
values[attnum - 1] = BoolGetDatum(read_only);
newtuple = heap_form_tuple(RelationGetDescr(rel),
values, nulls);
//simple_heap_update(rel, &tuple->t_self, newtuple);
CatalogTupleUpdate(rel, &tuple->t_self, newtuple);
}
else
elog(ERROR, "Node %s not found.", node_name);
systable_endscan(scan);
CommandCounterIncrement();
/* now release lock again, */
heap_close(rel, RowExclusiveLock);
bdr_connections_changed(NULL);
}
/*
* Set node_read_only field in bdr_nodes entry for given node.
*
* This has to be C function to avoid being subject to the executor read-only
* filtering.
*/
Datum
bdr_node_set_read_only(PG_FUNCTION_ARGS)
{
char *node_name = text_to_cstring(PG_GETARG_TEXT_PP(0));
bool read_only = PG_GETARG_BOOL(1);
bdr_node_set_read_only_internal(node_name, read_only, false);
PG_RETURN_VOID();
}
void
bdr_executor_always_allow_writes(bool always_allow)
{
Assert(IsUnderPostmaster);
bdr_always_allow_writes = always_allow;
}
static const char *
CreateWritableStmtTag(PlannedStmt *plannedstmt)
{
if (plannedstmt->commandType == CMD_SELECT)
return "DML"; /* SELECT INTO/WCTE */
return CreateCommandTag((Node *) plannedstmt);
}
/*
* The BDR ExecutorStart_hook that does DDL lock checks and forbids
* writing into tables without replica identity index.
*
* Runs in all backends and workers.
*/
static void
BdrExecutorStart(QueryDesc *queryDesc, int eflags)
{
bool performs_writes = false;
bool read_only_node;
ListCell *l;
List *rangeTable;
PlannedStmt *plannedstmt = queryDesc->plannedstmt;
if (bdr_always_allow_writes)
goto done;
/* don't perform filtering while replaying */
if (replorigin_session_origin != InvalidRepOriginId)
goto done;
/* identify whether this is a modifying statement */
if (plannedstmt != NULL &&
(plannedstmt->hasModifyingCTE ||
plannedstmt->rowMarks != NIL))
performs_writes = true;
else if (queryDesc->operation != CMD_SELECT)
performs_writes = true;
if (!performs_writes)
goto done;
if (!bdr_is_bdr_activated_db(MyDatabaseId))
goto done;
read_only_node = bdr_local_node_read_only() && !bdr_permit_unsafe_commands;
/* check for concurrent global DDL locks */
bdr_locks_check_dml();
/*
* Are we in bdr.replicate_ddl_command? If so, it's not safe to do DML,
* since this will basically do statement based replication that'll mess up
* volatile functions etc. If we skipped replicating it as rows and just
* replicated statements, we'd get wrong sequences and so on.
*
* We can't just ignore the DML and leave it in the command string, then
* replicate its effects with rows, either. Otherwise DDL like this would
* break:
*
* bdr.replicate_ddl_command($$
* ALTER TABLE foo ADD COLUMN bar ...;
* UPDATE foo SET bar = baz WHERE ...;
* ALTER TABLE foo DROP COLUMN baz;
* $$);
*
* ... because we'd apply the DROP COLUMN before we replicated
* the rows, since we execute a DDL string as a single operation. Then
* row apply would fail because the incoming rows would have data for
* dropped column 'baz'.
*/
if (in_bdr_replicate_ddl_command && !bdr_in_extension)
{
if (queryDesc->operation == CMD_INSERT
|| queryDesc->operation == CMD_UPDATE
|| queryDesc->operation == CMD_DELETE)
{
ereport(ERROR,
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
errmsg("row-data-modifying statements INSERT, UPDATE and DELETE are not permitted inside bdr.replicate_ddl_command"),
errhint("Split up scripts, putting DDL in bdr.replicate_ddl_command and DML as normal statements")));
}
}
/* plain INSERTs are ok beyond this point if node is not read-only */
if (queryDesc->operation == CMD_INSERT &&
!plannedstmt->hasModifyingCTE && !read_only_node)
goto done;
/* Fail if query tries to UPDATE or DELETE any of tables without PK */
rangeTable = plannedstmt->rtable;
foreach(l, plannedstmt->resultRelations)
{
Index rtei = lfirst_int(l);
RangeTblEntry *rte = rt_fetch(rtei, rangeTable);
Relation rel;
rel = RelationIdGetRelation(rte->relid);
/* Skip UNLOGGED and TEMP tables */
if (!RelationNeedsWAL(rel))
{
RelationClose(rel);
continue;
}
/*
* Since changes to pg_catalog aren't replicated directly there's
* no strong need to suppress direct UPDATEs on them. The usual
* rule of "it's dumb to modify the catalogs directly if you don't
* know what you're doing" applies.
*/
if (RelationGetNamespace(rel) == PG_CATALOG_NAMESPACE)
{
RelationClose(rel);
continue;
}
if (read_only_node)
ereport(ERROR,
(errcode(ERRCODE_READ_ONLY_SQL_TRANSACTION),
errmsg("%s may only affect UNLOGGED or TEMPORARY tables "\
"on read-only BDR node; %s is a regular table",
CreateWritableStmtTag(plannedstmt),
RelationGetRelationName(rel))));
if (rel->rd_indexvalid == 0)
RelationGetIndexList(rel);
if (OidIsValid(rel->rd_replidindex))
{
RelationClose(rel);
continue;
}
ereport(ERROR,
(errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
errmsg("Cannot run UPDATE or DELETE on table %s because it does not have a PRIMARY KEY.",
RelationGetRelationName(rel)),
errhint("Add a PRIMARY KEY to the table")));
RelationClose(rel);
}
done:
if (PrevExecutorStart_hook)
(*PrevExecutorStart_hook) (queryDesc, eflags);
else
standard_ExecutorStart(queryDesc, eflags);
}
void
bdr_executor_init(void)
{
PrevExecutorStart_hook = ExecutorStart_hook;
ExecutorStart_hook = BdrExecutorStart;
}