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
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"><html xmlns="http://www.w3.org/1999/xhtml"><head><meta http-equiv="Content-Type" content="text/html; charset=UTF-8" /><title>39.1. Overview of Trigger Behavior</title><link rel="stylesheet" type="text/css" href="stylesheet.css" /><link rev="made" href="pgsql-docs@lists.postgresql.org" /><meta name="generator" content="DocBook XSL Stylesheets Vsnapshot" /><link rel="prev" href="triggers.html" title="Chapter 39. Triggers" /><link rel="next" href="trigger-datachanges.html" title="39.2. Visibility of Data Changes" /></head><body id="docContent" class="container-fluid col-10"><div class="navheader"><table width="100%" summary="Navigation header"><tr><th colspan="5" align="center">39.1. Overview of Trigger Behavior</th></tr><tr><td width="10%" align="left"><a accesskey="p" href="triggers.html" title="Chapter 39. Triggers">Prev</a> </td><td width="10%" align="left"><a accesskey="u" href="triggers.html" title="Chapter 39. Triggers">Up</a></td><th width="60%" align="center">Chapter 39. Triggers</th><td width="10%" align="right"><a accesskey="h" href="index.html" title="PostgreSQL 16.3 Documentation">Home</a></td><td width="10%" align="right"> <a accesskey="n" href="trigger-datachanges.html" title="39.2. Visibility of Data Changes">Next</a></td></tr></table><hr /></div><div class="sect1" id="TRIGGER-DEFINITION"><div class="titlepage"><div><div><h2 class="title" style="clear: both">39.1. Overview of Trigger Behavior <a href="#TRIGGER-DEFINITION" class="id_link">#</a></h2></div></div></div><p>
A trigger is a specification that the database should automatically
execute a particular function whenever a certain type of operation is
performed. Triggers can be attached to tables (partitioned or not),
views, and foreign tables.
</p><p>
On tables and foreign tables, triggers can be defined to execute either
before or after any <code class="command">INSERT</code>, <code class="command">UPDATE</code>,
or <code class="command">DELETE</code> operation, either once per modified row,
or once per <acronym class="acronym">SQL</acronym> statement.
<code class="command">UPDATE</code> triggers can moreover be set to fire only if
certain columns are mentioned in the <code class="literal">SET</code> clause of
the <code class="command">UPDATE</code> statement. Triggers can also fire
for <code class="command">TRUNCATE</code> statements. If a trigger event occurs,
the trigger's function is called at the appropriate time to handle the
event.
</p><p>
On views, triggers can be defined to execute instead of
<code class="command">INSERT</code>, <code class="command">UPDATE</code>, or
<code class="command">DELETE</code> operations.
Such <code class="literal">INSTEAD OF</code> triggers
are fired once for each row that needs to be modified in the view.
It is the responsibility of the
trigger's function to perform the necessary modifications to the view's
underlying base table(s) and, where appropriate, return the modified
row as it will appear in the view. Triggers on views can also be defined
to execute once per <acronym class="acronym">SQL</acronym> statement, before or after
<code class="command">INSERT</code>, <code class="command">UPDATE</code>, or
<code class="command">DELETE</code> operations.
However, such triggers are fired only if there is also
an <code class="literal">INSTEAD OF</code> trigger on the view. Otherwise,
any statement targeting the view must be rewritten into a statement
affecting its underlying base table(s), and then the triggers
that will be fired are the ones attached to the base table(s).
</p><p>
The trigger function must be defined before the trigger itself can be
created. The trigger function must be declared as a
function taking no arguments and returning type <code class="literal">trigger</code>.
(The trigger function receives its input through a specially-passed
<code class="structname">TriggerData</code> structure, not in the form of ordinary function
arguments.)
</p><p>
Once a suitable trigger function has been created, the trigger is
established with
<a class="xref" href="sql-createtrigger.html" title="CREATE TRIGGER"><span class="refentrytitle">CREATE TRIGGER</span></a>.
The same trigger function can be used for multiple triggers.
</p><p>
<span class="productname">PostgreSQL</span> offers both <em class="firstterm">per-row</em>
triggers and <em class="firstterm">per-statement</em> triggers. With a per-row
trigger, the trigger function
is invoked once for each row that is affected by the statement
that fired the trigger. In contrast, a per-statement trigger is
invoked only once when an appropriate statement is executed,
regardless of the number of rows affected by that statement. In
particular, a statement that affects zero rows will still result
in the execution of any applicable per-statement triggers. These
two types of triggers are sometimes called <em class="firstterm">row-level</em>
triggers and <em class="firstterm">statement-level</em> triggers,
respectively. Triggers on <code class="command">TRUNCATE</code> may only be
defined at statement level, not per-row.
</p><p>
Triggers are also classified according to whether they fire
<em class="firstterm">before</em>, <em class="firstterm">after</em>, or
<em class="firstterm">instead of</em> the operation. These are referred to
as <code class="literal">BEFORE</code> triggers, <code class="literal">AFTER</code> triggers, and
<code class="literal">INSTEAD OF</code> triggers respectively.
Statement-level <code class="literal">BEFORE</code> triggers naturally fire before the
statement starts to do anything, while statement-level <code class="literal">AFTER</code>
triggers fire at the very end of the statement. These types of
triggers may be defined on tables, views, or foreign tables. Row-level
<code class="literal">BEFORE</code> triggers fire immediately before a particular row is
operated on, while row-level <code class="literal">AFTER</code> triggers fire at the end of
the statement (but before any statement-level <code class="literal">AFTER</code> triggers).
These types of triggers may only be defined on tables and
foreign tables, not views.
<code class="literal">INSTEAD OF</code> triggers may only be
defined on views, and only at row level; they fire immediately as each
row in the view is identified as needing to be operated on.
</p><p>
The execution of an <code class="literal">AFTER</code> trigger can be deferred
to the end of the transaction, rather than the end of the statement,
if it was defined as a <em class="firstterm">constraint trigger</em>.
In all cases, a trigger is executed as part of the same transaction as
the statement that triggered it, so if either the statement or the
trigger causes an error, the effects of both will be rolled back.
</p><p>
A statement that targets a parent table in an inheritance or partitioning
hierarchy does not cause the statement-level triggers of affected child
tables to be fired; only the parent table's statement-level triggers are
fired. However, row-level triggers of any affected child tables will be
fired.
</p><p>
If an <code class="command">INSERT</code> contains an <code class="literal">ON CONFLICT
DO UPDATE</code> clause, it is possible that the effects of
row-level <code class="literal">BEFORE</code> <code class="command">INSERT</code> triggers and
row-level <code class="literal">BEFORE</code> <code class="command">UPDATE</code> triggers can
both be applied in a way that is apparent from the final state of
the updated row, if an <code class="varname">EXCLUDED</code> column is referenced.
There need not be an <code class="varname">EXCLUDED</code> column reference for
both sets of row-level <code class="literal">BEFORE</code> triggers to execute,
though. The
possibility of surprising outcomes should be considered when there
are both <code class="literal">BEFORE</code> <code class="command">INSERT</code> and
<code class="literal">BEFORE</code> <code class="command">UPDATE</code> row-level triggers
that change a row being inserted/updated (this can be
problematic even if the modifications are more or less equivalent, if
they're not also idempotent). Note that statement-level
<code class="command">UPDATE</code> triggers are executed when <code class="literal">ON
CONFLICT DO UPDATE</code> is specified, regardless of whether or not
any rows were affected by the <code class="command">UPDATE</code> (and
regardless of whether the alternative <code class="command">UPDATE</code>
path was ever taken). An <code class="command">INSERT</code> with an
<code class="literal">ON CONFLICT DO UPDATE</code> clause will execute
statement-level <code class="literal">BEFORE</code> <code class="command">INSERT</code>
triggers first, then statement-level <code class="literal">BEFORE</code>
<code class="command">UPDATE</code> triggers, followed by statement-level
<code class="literal">AFTER</code> <code class="command">UPDATE</code> triggers and finally
statement-level <code class="literal">AFTER</code> <code class="command">INSERT</code>
triggers.
</p><p>
If an <code class="command">UPDATE</code> on a partitioned table causes a row to move
to another partition, it will be performed as a <code class="command">DELETE</code>
from the original partition followed by an <code class="command">INSERT</code> into
the new partition. In this case, all row-level <code class="literal">BEFORE</code>
<code class="command">UPDATE</code> triggers and all row-level
<code class="literal">BEFORE</code> <code class="command">DELETE</code> triggers are fired on
the original partition. Then all row-level <code class="literal">BEFORE</code>
<code class="command">INSERT</code> triggers are fired on the destination partition.
The possibility of surprising outcomes should be considered when all these
triggers affect the row being moved. As far as <code class="literal">AFTER ROW</code>
triggers are concerned, <code class="literal">AFTER</code> <code class="command">DELETE</code>
and <code class="literal">AFTER</code> <code class="command">INSERT</code> triggers are
applied; but <code class="literal">AFTER</code> <code class="command">UPDATE</code> triggers
are not applied because the <code class="command">UPDATE</code> has been converted to
a <code class="command">DELETE</code> and an <code class="command">INSERT</code>. As far as
statement-level triggers are concerned, none of the
<code class="command">DELETE</code> or <code class="command">INSERT</code> triggers are fired,
even if row movement occurs; only the <code class="command">UPDATE</code> triggers
defined on the target table used in the <code class="command">UPDATE</code> statement
will be fired.
</p><p>
No separate triggers are defined for <code class="command">MERGE</code>. Instead,
statement-level or row-level <code class="command">UPDATE</code>,
<code class="command">DELETE</code>, and <code class="command">INSERT</code> triggers are fired
depending on (for statement-level triggers) what actions are specified in
the <code class="command">MERGE</code> query and (for row-level triggers) what
actions are performed.
</p><p>
While running a <code class="command">MERGE</code> command, statement-level
<code class="literal">BEFORE</code> and <code class="literal">AFTER</code> triggers are
fired for events specified in the actions of the <code class="command">MERGE</code>
command, irrespective of whether or not the action is ultimately performed.
This is the same as an <code class="command">UPDATE</code> statement that updates
no rows, yet statement-level triggers are fired.
The row-level triggers are fired only when a row is actually updated,
inserted or deleted. So it's perfectly legal that while statement-level
triggers are fired for certain types of action, no row-level triggers
are fired for the same kind of action.
</p><p>
Trigger functions invoked by per-statement triggers should always
return <code class="symbol">NULL</code>. Trigger functions invoked by per-row
triggers can return a table row (a value of
type <code class="structname">HeapTuple</code>) to the calling executor,
if they choose. A row-level trigger fired before an operation has
the following choices:
</p><div class="itemizedlist"><ul class="itemizedlist" style="list-style-type: disc; "><li class="listitem"><p>
It can return <code class="symbol">NULL</code> to skip the operation for the
current row. This instructs the executor to not perform the
row-level operation that invoked the trigger (the insertion,
modification, or deletion of a particular table row).
</p></li><li class="listitem"><p>
For row-level <code class="command">INSERT</code>
and <code class="command">UPDATE</code> triggers only, the returned row
becomes the row that will be inserted or will replace the row
being updated. This allows the trigger function to modify the
row being inserted or updated.
</p></li></ul></div><p>
A row-level <code class="literal">BEFORE</code> trigger that does not intend to cause
either of these behaviors must be careful to return as its result the same
row that was passed in (that is, the <code class="varname">NEW</code> row
for <code class="command">INSERT</code> and <code class="command">UPDATE</code>
triggers, the <code class="varname">OLD</code> row for
<code class="command">DELETE</code> triggers).
</p><p>
A row-level <code class="literal">INSTEAD OF</code> trigger should either return
<code class="symbol">NULL</code> to indicate that it did not modify any data from
the view's underlying base tables, or it should return the view
row that was passed in (the <code class="varname">NEW</code> row
for <code class="command">INSERT</code> and <code class="command">UPDATE</code>
operations, or the <code class="varname">OLD</code> row for
<code class="command">DELETE</code> operations). A nonnull return value is
used to signal that the trigger performed the necessary data
modifications in the view. This will cause the count of the number
of rows affected by the command to be incremented. For
<code class="command">INSERT</code> and <code class="command">UPDATE</code> operations only, the trigger
may modify the <code class="varname">NEW</code> row before returning it. This will
change the data returned by
<code class="command">INSERT RETURNING</code> or <code class="command">UPDATE RETURNING</code>,
and is useful when the view will not show exactly the same data
that was provided.
</p><p>
The return value is ignored for row-level triggers fired after an
operation, and so they can return <code class="symbol">NULL</code>.
</p><p>
Some considerations apply for generated
columns.<a id="id-1.8.4.5.18.1" class="indexterm"></a> Stored generated columns are computed after
<code class="literal">BEFORE</code> triggers and before <code class="literal">AFTER</code>
triggers. Therefore, the generated value can be inspected in
<code class="literal">AFTER</code> triggers. In <code class="literal">BEFORE</code> triggers,
the <code class="literal">OLD</code> row contains the old generated value, as one
would expect, but the <code class="literal">NEW</code> row does not yet contain the
new generated value and should not be accessed. In the C language
interface, the content of the column is undefined at this point; a
higher-level programming language should prevent access to a stored
generated column in the <code class="literal">NEW</code> row in a
<code class="literal">BEFORE</code> trigger. Changes to the value of a generated
column in a <code class="literal">BEFORE</code> trigger are ignored and will be
overwritten.
</p><p>
If more than one trigger is defined for the same event on the same
relation, the triggers will be fired in alphabetical order by
trigger name. In the case of <code class="literal">BEFORE</code> and
<code class="literal">INSTEAD OF</code> triggers, the possibly-modified row returned by
each trigger becomes the input to the next trigger. If any
<code class="literal">BEFORE</code> or <code class="literal">INSTEAD OF</code> trigger returns
<code class="symbol">NULL</code>, the operation is abandoned for that row and subsequent
triggers are not fired (for that row).
</p><p>
A trigger definition can also specify a Boolean <code class="literal">WHEN</code>
condition, which will be tested to see whether the trigger should
be fired. In row-level triggers the <code class="literal">WHEN</code> condition can
examine the old and/or new values of columns of the row. (Statement-level
triggers can also have <code class="literal">WHEN</code> conditions, although the feature
is not so useful for them.) In a <code class="literal">BEFORE</code> trigger, the
<code class="literal">WHEN</code>
condition is evaluated just before the function is or would be executed,
so using <code class="literal">WHEN</code> is not materially different from testing the
same condition at the beginning of the trigger function. However, in
an <code class="literal">AFTER</code> trigger, the <code class="literal">WHEN</code> condition is evaluated
just after the row update occurs, and it determines whether an event is
queued to fire the trigger at the end of statement. So when an
<code class="literal">AFTER</code> trigger's
<code class="literal">WHEN</code> condition does not return true, it is not necessary
to queue an event nor to re-fetch the row at end of statement. This
can result in significant speedups in statements that modify many
rows, if the trigger only needs to be fired for a few of the rows.
<code class="literal">INSTEAD OF</code> triggers do not support
<code class="literal">WHEN</code> conditions.
</p><p>
Typically, row-level <code class="literal">BEFORE</code> triggers are used for checking or
modifying the data that will be inserted or updated. For example,
a <code class="literal">BEFORE</code> trigger might be used to insert the current time into a
<code class="type">timestamp</code> column, or to check that two elements of the row are
consistent. Row-level <code class="literal">AFTER</code> triggers are most sensibly
used to propagate the updates to other tables, or make consistency
checks against other tables. The reason for this division of labor is
that an <code class="literal">AFTER</code> trigger can be certain it is seeing the final
value of the row, while a <code class="literal">BEFORE</code> trigger cannot; there might
be other <code class="literal">BEFORE</code> triggers firing after it. If you have no
specific reason to make a trigger <code class="literal">BEFORE</code> or
<code class="literal">AFTER</code>, the <code class="literal">BEFORE</code> case is more efficient, since
the information about
the operation doesn't have to be saved until end of statement.
</p><p>
If a trigger function executes SQL commands then these
commands might fire triggers again. This is known as cascading
triggers. There is no direct limitation on the number of cascade
levels. It is possible for cascades to cause a recursive invocation
of the same trigger; for example, an <code class="command">INSERT</code>
trigger might execute a command that inserts an additional row
into the same table, causing the <code class="command">INSERT</code> trigger
to be fired again. It is the trigger programmer's responsibility
to avoid infinite recursion in such scenarios.
</p><p>
<a id="id-1.8.4.5.23.1" class="indexterm"></a>
When a trigger is being defined, arguments can be specified for
it. The purpose of including arguments in the
trigger definition is to allow different triggers with similar
requirements to call the same function. As an example, there
could be a generalized trigger function that takes as its
arguments two column names and puts the current user in one and
the current time stamp in the other. Properly written, this
trigger function would be independent of the specific table it is
triggering on. So the same function could be used for
<code class="command">INSERT</code> events on any table with suitable
columns, to automatically track creation of records in a
transaction table for example. It could also be used to track
last-update events if defined as an <code class="command">UPDATE</code>
trigger.
</p><p>
Each programming language that supports triggers has its own method
for making the trigger input data available to the trigger function.
This input data includes the type of trigger event (e.g.,
<code class="command">INSERT</code> or <code class="command">UPDATE</code>) as well as any
arguments that were listed in <code class="command">CREATE TRIGGER</code>.
For a row-level trigger, the input data also includes the
<code class="varname">NEW</code> row for <code class="command">INSERT</code> and
<code class="command">UPDATE</code> triggers, and/or the <code class="varname">OLD</code> row
for <code class="command">UPDATE</code> and <code class="command">DELETE</code> triggers.
</p><p>
By default, statement-level triggers do not have any way to examine the
individual row(s) modified by the statement. But an <code class="literal">AFTER
STATEMENT</code> trigger can request that <em class="firstterm">transition tables</em>
be created to make the sets of affected rows available to the trigger.
<code class="literal">AFTER ROW</code> triggers can also request transition tables, so
that they can see the total changes in the table as well as the change in
the individual row they are currently being fired for. The method for
examining the transition tables again depends on the programming language
that is being used, but the typical approach is to make the transition
tables act like read-only temporary tables that can be accessed by SQL
commands issued within the trigger function.
</p></div><div class="navfooter"><hr /><table width="100%" summary="Navigation footer"><tr><td width="40%" align="left"><a accesskey="p" href="triggers.html" title="Chapter 39. Triggers">Prev</a> </td><td width="20%" align="center"><a accesskey="u" href="triggers.html" title="Chapter 39. Triggers">Up</a></td><td width="40%" align="right"> <a accesskey="n" href="trigger-datachanges.html" title="39.2. Visibility of Data Changes">Next</a></td></tr><tr><td width="40%" align="left" valign="top">Chapter 39. Triggers </td><td width="20%" align="center"><a accesskey="h" href="index.html" title="PostgreSQL 16.3 Documentation">Home</a></td><td width="40%" align="right" valign="top"> 39.2. Visibility of Data Changes</td></tr></table></div></body></html>
|