Fix xslt_process() to ensure that it inserts a NULL terminator after the
[PostgreSQL.git] / src / test / regress / expected / triggers.out
blob2a5cb909d7027ec9ee0b6d9e940354c0e107378a
1 --
2 -- TRIGGERS
3 --
4 create table pkeys (pkey1 int4 not null, pkey2 text not null);
5 create table fkeys (fkey1 int4, fkey2 text, fkey3 int);
6 create table fkeys2 (fkey21 int4, fkey22 text, pkey23 int not null);
7 create index fkeys_i on fkeys (fkey1, fkey2);
8 create index fkeys2_i on fkeys2 (fkey21, fkey22);
9 create index fkeys2p_i on fkeys2 (pkey23);
10 insert into pkeys values (10, '1');
11 insert into pkeys values (20, '2');
12 insert into pkeys values (30, '3');
13 insert into pkeys values (40, '4');
14 insert into pkeys values (50, '5');
15 insert into pkeys values (60, '6');
16 create unique index pkeys_i on pkeys (pkey1, pkey2);
18 -- For fkeys:
19 --      (fkey1, fkey2)  --> pkeys (pkey1, pkey2)
20 --      (fkey3)         --> fkeys2 (pkey23)
22 create trigger check_fkeys_pkey_exist 
23         before insert or update on fkeys 
24         for each row 
25         execute procedure 
26         check_primary_key ('fkey1', 'fkey2', 'pkeys', 'pkey1', 'pkey2');
27 create trigger check_fkeys_pkey2_exist 
28         before insert or update on fkeys 
29         for each row 
30         execute procedure check_primary_key ('fkey3', 'fkeys2', 'pkey23');
32 -- For fkeys2:
33 --      (fkey21, fkey22)        --> pkeys (pkey1, pkey2)
35 create trigger check_fkeys2_pkey_exist 
36         before insert or update on fkeys2 
37         for each row 
38         execute procedure 
39         check_primary_key ('fkey21', 'fkey22', 'pkeys', 'pkey1', 'pkey2');
40 -- Test comments
41 COMMENT ON TRIGGER check_fkeys2_pkey_bad ON fkeys2 IS 'wrong';
42 ERROR:  trigger "check_fkeys2_pkey_bad" for table "fkeys2" does not exist
43 COMMENT ON TRIGGER check_fkeys2_pkey_exist ON fkeys2 IS 'right';
44 COMMENT ON TRIGGER check_fkeys2_pkey_exist ON fkeys2 IS NULL;
46 -- For pkeys:
47 --      ON DELETE/UPDATE (pkey1, pkey2) CASCADE:
48 --              fkeys (fkey1, fkey2) and fkeys2 (fkey21, fkey22)
50 create trigger check_pkeys_fkey_cascade
51         before delete or update on pkeys 
52         for each row 
53         execute procedure 
54         check_foreign_key (2, 'cascade', 'pkey1', 'pkey2', 
55         'fkeys', 'fkey1', 'fkey2', 'fkeys2', 'fkey21', 'fkey22');
57 -- For fkeys2:
58 --      ON DELETE/UPDATE (pkey23) RESTRICT:
59 --              fkeys (fkey3)
61 create trigger check_fkeys2_fkey_restrict 
62         before delete or update on fkeys2
63         for each row 
64         execute procedure check_foreign_key (1, 'restrict', 'pkey23', 'fkeys', 'fkey3');
65 insert into fkeys2 values (10, '1', 1);
66 insert into fkeys2 values (30, '3', 2);
67 insert into fkeys2 values (40, '4', 5);
68 insert into fkeys2 values (50, '5', 3);
69 -- no key in pkeys
70 insert into fkeys2 values (70, '5', 3);
71 ERROR:  tuple references non-existent key
72 DETAIL:  Trigger "check_fkeys2_pkey_exist" found tuple referencing non-existent key in "pkeys".
73 insert into fkeys values (10, '1', 2);
74 insert into fkeys values (30, '3', 3);
75 insert into fkeys values (40, '4', 2);
76 insert into fkeys values (50, '5', 2);
77 -- no key in pkeys
78 insert into fkeys values (70, '5', 1);
79 ERROR:  tuple references non-existent key
80 DETAIL:  Trigger "check_fkeys_pkey_exist" found tuple referencing non-existent key in "pkeys".
81 -- no key in fkeys2
82 insert into fkeys values (60, '6', 4);
83 ERROR:  tuple references non-existent key
84 DETAIL:  Trigger "check_fkeys_pkey2_exist" found tuple referencing non-existent key in "fkeys2".
85 delete from pkeys where pkey1 = 30 and pkey2 = '3';
86 NOTICE:  check_pkeys_fkey_cascade: 1 tuple(s) of fkeys are deleted
87 ERROR:  "check_fkeys2_fkey_restrict": tuple is referenced in "fkeys"
88 CONTEXT:  SQL statement "delete from fkeys2 where fkey21 = $1 and fkey22 = $2 "
89 delete from pkeys where pkey1 = 40 and pkey2 = '4';
90 NOTICE:  check_pkeys_fkey_cascade: 1 tuple(s) of fkeys are deleted
91 NOTICE:  check_pkeys_fkey_cascade: 1 tuple(s) of fkeys2 are deleted
92 update pkeys set pkey1 = 7, pkey2 = '70' where pkey1 = 50 and pkey2 = '5';
93 NOTICE:  check_pkeys_fkey_cascade: 1 tuple(s) of fkeys are deleted
94 ERROR:  "check_fkeys2_fkey_restrict": tuple is referenced in "fkeys"
95 CONTEXT:  SQL statement "delete from fkeys2 where fkey21 = $1 and fkey22 = $2 "
96 update pkeys set pkey1 = 7, pkey2 = '70' where pkey1 = 10 and pkey2 = '1';
97 NOTICE:  check_pkeys_fkey_cascade: 1 tuple(s) of fkeys are deleted
98 NOTICE:  check_pkeys_fkey_cascade: 1 tuple(s) of fkeys2 are deleted
99 DROP TABLE pkeys;
100 DROP TABLE fkeys;
101 DROP TABLE fkeys2;
102 -- -- I've disabled the funny_dup17 test because the new semantics
103 -- -- of AFTER ROW triggers, which get now fired at the end of a
104 -- -- query always, cause funny_dup17 to enter an endless loop.
105 -- --
106 -- --      Jan
108 -- create table dup17 (x int4);
109 -- 
110 -- create trigger dup17_before 
111 --      before insert on dup17
112 --      for each row 
113 --      execute procedure 
114 --      funny_dup17 ()
115 -- ;
116 -- 
117 -- insert into dup17 values (17);
118 -- select count(*) from dup17;
119 -- insert into dup17 values (17);
120 -- select count(*) from dup17;
121 -- 
122 -- drop trigger dup17_before on dup17;
123 -- 
124 -- create trigger dup17_after
125 --      after insert on dup17
126 --      for each row 
127 --      execute procedure 
128 --      funny_dup17 ()
129 -- ;
130 -- insert into dup17 values (13);
131 -- select count(*) from dup17 where x = 13;
132 -- insert into dup17 values (13);
133 -- select count(*) from dup17 where x = 13;
134 -- 
135 -- DROP TABLE dup17;
136 create sequence ttdummy_seq increment 10 start 0 minvalue 0;
137 create table tttest (
138         price_id        int4, 
139         price_val       int4, 
140         price_on        int4,
141         price_off       int4 default 999999
143 create trigger ttdummy 
144         before delete or update on tttest
145         for each row 
146         execute procedure 
147         ttdummy (price_on, price_off);
148 create trigger ttserial 
149         before insert or update on tttest
150         for each row 
151         execute procedure 
152         autoinc (price_on, ttdummy_seq);
153 insert into tttest values (1, 1, null);
154 insert into tttest values (2, 2, null);
155 insert into tttest values (3, 3, 0);
156 select * from tttest;
157  price_id | price_val | price_on | price_off 
158 ----------+-----------+----------+-----------
159         1 |         1 |       10 |    999999
160         2 |         2 |       20 |    999999
161         3 |         3 |       30 |    999999
162 (3 rows)
164 delete from tttest where price_id = 2;
165 select * from tttest;
166  price_id | price_val | price_on | price_off 
167 ----------+-----------+----------+-----------
168         1 |         1 |       10 |    999999
169         3 |         3 |       30 |    999999
170         2 |         2 |       20 |        40
171 (3 rows)
173 -- what do we see ?
174 -- get current prices
175 select * from tttest where price_off = 999999;
176  price_id | price_val | price_on | price_off 
177 ----------+-----------+----------+-----------
178         1 |         1 |       10 |    999999
179         3 |         3 |       30 |    999999
180 (2 rows)
182 -- change price for price_id == 3
183 update tttest set price_val = 30 where price_id = 3;
184 select * from tttest;
185  price_id | price_val | price_on | price_off 
186 ----------+-----------+----------+-----------
187         1 |         1 |       10 |    999999
188         2 |         2 |       20 |        40
189         3 |        30 |       50 |    999999
190         3 |         3 |       30 |        50
191 (4 rows)
193 -- now we want to change pric_id in ALL tuples
194 -- this gets us not what we need
195 update tttest set price_id = 5 where price_id = 3;
196 select * from tttest;
197  price_id | price_val | price_on | price_off 
198 ----------+-----------+----------+-----------
199         1 |         1 |       10 |    999999
200         2 |         2 |       20 |        40
201         3 |         3 |       30 |        50
202         5 |        30 |       60 |    999999
203         3 |        30 |       50 |        60
204 (5 rows)
206 -- restore data as before last update:
207 select set_ttdummy(0);
208  set_ttdummy 
209 -------------
210            1
211 (1 row)
213 delete from tttest where price_id = 5;
214 update tttest set price_off = 999999 where price_val = 30;
215 select * from tttest;
216  price_id | price_val | price_on | price_off 
217 ----------+-----------+----------+-----------
218         1 |         1 |       10 |    999999
219         2 |         2 |       20 |        40
220         3 |         3 |       30 |        50
221         3 |        30 |       50 |    999999
222 (4 rows)
224 -- and try change price_id now!
225 update tttest set price_id = 5 where price_id = 3;
226 select * from tttest;
227  price_id | price_val | price_on | price_off 
228 ----------+-----------+----------+-----------
229         1 |         1 |       10 |    999999
230         2 |         2 |       20 |        40
231         5 |         3 |       30 |        50
232         5 |        30 |       50 |    999999
233 (4 rows)
235 -- isn't it what we need ?
236 select set_ttdummy(1);
237  set_ttdummy 
238 -------------
239            0
240 (1 row)
242 -- we want to correct some "date"
243 update tttest set price_on = -1 where price_id = 1;
244 ERROR:  ttdummy (tttest): you cannot change price_on and/or price_off columns (use set_ttdummy)
245 -- but this doesn't work
246 -- try in this way
247 select set_ttdummy(0);
248  set_ttdummy 
249 -------------
250            1
251 (1 row)
253 update tttest set price_on = -1 where price_id = 1;
254 select * from tttest;
255  price_id | price_val | price_on | price_off 
256 ----------+-----------+----------+-----------
257         2 |         2 |       20 |        40
258         5 |         3 |       30 |        50
259         5 |        30 |       50 |    999999
260         1 |         1 |       -1 |    999999
261 (4 rows)
263 -- isn't it what we need ?
264 -- get price for price_id == 5 as it was @ "date" 35
265 select * from tttest where price_on <= 35 and price_off > 35 and price_id = 5;
266  price_id | price_val | price_on | price_off 
267 ----------+-----------+----------+-----------
268         5 |         3 |       30 |        50
269 (1 row)
271 drop table tttest;
272 drop sequence ttdummy_seq;
274 -- tests for per-statement triggers
276 CREATE TABLE log_table (tstamp timestamp default timeofday()::timestamp);
277 CREATE TABLE main_table (a int, b int);
278 COPY main_table (a,b) FROM stdin;
279 CREATE FUNCTION trigger_func() RETURNS trigger LANGUAGE plpgsql AS '
280 BEGIN
281         RAISE NOTICE ''trigger_func() called: action = %, when = %, level = %'', TG_OP, TG_WHEN, TG_LEVEL;
282         RETURN NULL;
283 END;';
284 CREATE TRIGGER before_ins_stmt_trig BEFORE INSERT ON main_table
285 FOR EACH STATEMENT EXECUTE PROCEDURE trigger_func();
286 CREATE TRIGGER after_ins_stmt_trig AFTER INSERT ON main_table
287 FOR EACH STATEMENT EXECUTE PROCEDURE trigger_func();
289 -- if neither 'FOR EACH ROW' nor 'FOR EACH STATEMENT' was specified,
290 -- CREATE TRIGGER should default to 'FOR EACH STATEMENT'
292 CREATE TRIGGER before_upd_stmt_trig AFTER UPDATE ON main_table
293 EXECUTE PROCEDURE trigger_func();
294 CREATE TRIGGER before_upd_row_trig AFTER UPDATE ON main_table
295 FOR EACH ROW EXECUTE PROCEDURE trigger_func();
296 INSERT INTO main_table DEFAULT VALUES;
297 NOTICE:  trigger_func() called: action = INSERT, when = BEFORE, level = STATEMENT
298 NOTICE:  trigger_func() called: action = INSERT, when = AFTER, level = STATEMENT
299 UPDATE main_table SET a = a + 1 WHERE b < 30;
300 NOTICE:  trigger_func() called: action = UPDATE, when = AFTER, level = ROW
301 NOTICE:  trigger_func() called: action = UPDATE, when = AFTER, level = ROW
302 NOTICE:  trigger_func() called: action = UPDATE, when = AFTER, level = ROW
303 NOTICE:  trigger_func() called: action = UPDATE, when = AFTER, level = ROW
304 NOTICE:  trigger_func() called: action = UPDATE, when = AFTER, level = STATEMENT
305 -- UPDATE that effects zero rows should still call per-statement trigger
306 UPDATE main_table SET a = a + 2 WHERE b > 100;
307 NOTICE:  trigger_func() called: action = UPDATE, when = AFTER, level = STATEMENT
308 -- COPY should fire per-row and per-statement INSERT triggers
309 COPY main_table (a, b) FROM stdin;
310 NOTICE:  trigger_func() called: action = INSERT, when = BEFORE, level = STATEMENT
311 NOTICE:  trigger_func() called: action = INSERT, when = AFTER, level = STATEMENT
312 SELECT * FROM main_table ORDER BY a, b;
313  a  | b  
314 ----+----
315   6 | 10
316  21 | 20
317  30 | 40
318  31 | 10
319  50 | 35
320  50 | 60
321  81 | 15
322     |   
323 (8 rows)
325 -- Test enable/disable triggers
326 create table trigtest (i serial primary key);
327 NOTICE:  CREATE TABLE will create implicit sequence "trigtest_i_seq" for serial column "trigtest.i"
328 NOTICE:  CREATE TABLE / PRIMARY KEY will create implicit index "trigtest_pkey" for table "trigtest"
329 -- test that disabling RI triggers works
330 create table trigtest2 (i int references trigtest(i) on delete cascade);
331 create function trigtest() returns trigger as $$
332 begin
333         raise notice '% % % %', TG_RELNAME, TG_OP, TG_WHEN, TG_LEVEL;
334         return new;
335 end;$$ language plpgsql;
336 create trigger trigtest_b_row_tg before insert or update or delete on trigtest
337 for each row execute procedure trigtest();
338 create trigger trigtest_a_row_tg after insert or update or delete on trigtest
339 for each row execute procedure trigtest();
340 create trigger trigtest_b_stmt_tg before insert or update or delete on trigtest
341 for each statement execute procedure trigtest();
342 create trigger trigtest_a_stmt_tg after insert or update or delete on trigtest
343 for each statement execute procedure trigtest();
344 insert into trigtest default values;
345 NOTICE:  trigtest INSERT BEFORE STATEMENT
346 NOTICE:  trigtest INSERT BEFORE ROW
347 NOTICE:  trigtest INSERT AFTER ROW
348 NOTICE:  trigtest INSERT AFTER STATEMENT
349 alter table trigtest disable trigger trigtest_b_row_tg;
350 insert into trigtest default values;
351 NOTICE:  trigtest INSERT BEFORE STATEMENT
352 NOTICE:  trigtest INSERT AFTER ROW
353 NOTICE:  trigtest INSERT AFTER STATEMENT
354 alter table trigtest disable trigger user;
355 insert into trigtest default values;
356 alter table trigtest enable trigger trigtest_a_stmt_tg;
357 insert into trigtest default values;
358 NOTICE:  trigtest INSERT AFTER STATEMENT
359 insert into trigtest2 values(1);
360 insert into trigtest2 values(2);
361 delete from trigtest where i=2;
362 NOTICE:  trigtest DELETE AFTER STATEMENT
363 select * from trigtest2;
364  i 
367 (1 row)
369 alter table trigtest disable trigger all;
370 delete from trigtest where i=1;
371 select * from trigtest2;
372  i 
375 (1 row)
377 -- ensure we still insert, even when all triggers are disabled
378 insert into trigtest default values;
379 select *  from trigtest;
380  i 
385 (3 rows)
387 drop table trigtest2;
388 drop table trigtest;
389 -- dump trigger data
390 CREATE TABLE trigger_test (
391         i int,
392         v varchar
394 CREATE OR REPLACE FUNCTION trigger_data()  RETURNS trigger 
395 LANGUAGE plpgsql AS $$
397 declare
399         argstr text;
400         relid text;
402 begin
404         relid := TG_relid::regclass;
406         -- plpgsql can't discover its trigger data in a hash like perl and python
407         -- can, or by a sort of reflection like tcl can, 
408         -- so we have to hard code the names.
409         raise NOTICE 'TG_NAME: %', TG_name;
410         raise NOTICE 'TG_WHEN: %', TG_when;
411         raise NOTICE 'TG_LEVEL: %', TG_level;
412         raise NOTICE 'TG_OP: %', TG_op;
413         raise NOTICE 'TG_RELID::regclass: %', relid;
414         raise NOTICE 'TG_RELNAME: %', TG_relname;
415         raise NOTICE 'TG_TABLE_NAME: %', TG_table_name;
416         raise NOTICE 'TG_TABLE_SCHEMA: %', TG_table_schema;
417         raise NOTICE 'TG_NARGS: %', TG_nargs;
419         argstr := '[';
420         for i in 0 .. TG_nargs - 1 loop
421                 if i > 0 then
422                         argstr := argstr || ', ';
423                 end if;
424                 argstr := argstr || TG_argv[i];
425         end loop;
426         argstr := argstr || ']';
427         raise NOTICE 'TG_ARGV: %', argstr;
429         if TG_OP != 'INSERT' then
430                 raise NOTICE 'OLD: %', OLD;
431         end if;
433         if TG_OP != 'DELETE' then
434                 raise NOTICE 'NEW: %', NEW;
435         end if;
437         if TG_OP = 'DELETE' then
438                 return OLD;
439         else
440                 return NEW;
441         end if;
443 end;
445 CREATE TRIGGER show_trigger_data_trig 
446 BEFORE INSERT OR UPDATE OR DELETE ON trigger_test
447 FOR EACH ROW EXECUTE PROCEDURE trigger_data(23,'skidoo');
448 insert into trigger_test values(1,'insert');
449 NOTICE:  TG_NAME: show_trigger_data_trig
450 NOTICE:  TG_WHEN: BEFORE
451 NOTICE:  TG_LEVEL: ROW
452 NOTICE:  TG_OP: INSERT
453 NOTICE:  TG_RELID::regclass: trigger_test
454 NOTICE:  TG_RELNAME: trigger_test
455 NOTICE:  TG_TABLE_NAME: trigger_test
456 NOTICE:  TG_TABLE_SCHEMA: public
457 NOTICE:  TG_NARGS: 2
458 NOTICE:  TG_ARGV: [23, skidoo]
459 NOTICE:  NEW: (1,insert)
460 update trigger_test set v = 'update' where i = 1;
461 NOTICE:  TG_NAME: show_trigger_data_trig
462 NOTICE:  TG_WHEN: BEFORE
463 NOTICE:  TG_LEVEL: ROW
464 NOTICE:  TG_OP: UPDATE
465 NOTICE:  TG_RELID::regclass: trigger_test
466 NOTICE:  TG_RELNAME: trigger_test
467 NOTICE:  TG_TABLE_NAME: trigger_test
468 NOTICE:  TG_TABLE_SCHEMA: public
469 NOTICE:  TG_NARGS: 2
470 NOTICE:  TG_ARGV: [23, skidoo]
471 NOTICE:  OLD: (1,insert)
472 NOTICE:  NEW: (1,update)
473 delete from trigger_test;
474 NOTICE:  TG_NAME: show_trigger_data_trig
475 NOTICE:  TG_WHEN: BEFORE
476 NOTICE:  TG_LEVEL: ROW
477 NOTICE:  TG_OP: DELETE
478 NOTICE:  TG_RELID::regclass: trigger_test
479 NOTICE:  TG_RELNAME: trigger_test
480 NOTICE:  TG_TABLE_NAME: trigger_test
481 NOTICE:  TG_TABLE_SCHEMA: public
482 NOTICE:  TG_NARGS: 2
483 NOTICE:  TG_ARGV: [23, skidoo]
484 NOTICE:  OLD: (1,update)
485       
486 DROP TRIGGER show_trigger_data_trig on trigger_test;
487       
488 DROP FUNCTION trigger_data();
489 DROP TABLE trigger_test;
491 -- Test use of row comparisons on OLD/NEW
493 CREATE TABLE trigger_test (f1 int, f2 text, f3 text);
494 -- this is the obvious (and wrong...) way to compare rows
495 CREATE FUNCTION mytrigger() RETURNS trigger LANGUAGE plpgsql as $$
496 begin
497         if row(old.*) = row(new.*) then
498                 raise notice 'row % not changed', new.f1;
499         else
500                 raise notice 'row % changed', new.f1;
501         end if;
502         return new;
503 end$$;
504 CREATE TRIGGER t
505 BEFORE UPDATE ON trigger_test
506 FOR EACH ROW EXECUTE PROCEDURE mytrigger();
507 INSERT INTO trigger_test VALUES(1, 'foo', 'bar');
508 INSERT INTO trigger_test VALUES(2, 'baz', 'quux');
509 UPDATE trigger_test SET f3 = 'bar';
510 NOTICE:  row 1 not changed
511 NOTICE:  row 2 changed
512 UPDATE trigger_test SET f3 = NULL;
513 NOTICE:  row 1 changed
514 NOTICE:  row 2 changed
515 -- this demonstrates that the above isn't really working as desired:
516 UPDATE trigger_test SET f3 = NULL;
517 NOTICE:  row 1 changed
518 NOTICE:  row 2 changed
519 -- the right way when considering nulls is
520 CREATE OR REPLACE FUNCTION mytrigger() RETURNS trigger LANGUAGE plpgsql as $$
521 begin
522         if row(old.*) is distinct from row(new.*) then
523                 raise notice 'row % changed', new.f1;
524         else
525                 raise notice 'row % not changed', new.f1;
526         end if;
527         return new;
528 end$$;
529 UPDATE trigger_test SET f3 = 'bar';
530 NOTICE:  row 1 changed
531 NOTICE:  row 2 changed
532 UPDATE trigger_test SET f3 = NULL;
533 NOTICE:  row 1 changed
534 NOTICE:  row 2 changed
535 UPDATE trigger_test SET f3 = NULL;
536 NOTICE:  row 1 not changed
537 NOTICE:  row 2 not changed
538 DROP TABLE trigger_test;
539 DROP FUNCTION mytrigger();
540 -- minimal update trigger
541 CREATE TABLE min_updates_test (
542         f1      text,
543         f2 int,
544         f3 int);
545 CREATE TABLE min_updates_test_oids (
546         f1      text,
547         f2 int,
548         f3 int) WITH OIDS;
549 INSERT INTO min_updates_test VALUES ('a',1,2),('b','2',null);
550 INSERT INTO min_updates_test_oids VALUES ('a',1,2),('b','2',null);
551 CREATE TRIGGER z_min_update 
552 BEFORE UPDATE ON min_updates_test
553 FOR EACH ROW EXECUTE PROCEDURE suppress_redundant_updates_trigger();
554 CREATE TRIGGER z_min_update 
555 BEFORE UPDATE ON min_updates_test_oids
556 FOR EACH ROW EXECUTE PROCEDURE suppress_redundant_updates_trigger();
557 \set QUIET false
558 UPDATE min_updates_test SET f1 = f1;
559 UPDATE 0
560 UPDATE min_updates_test SET f2 = f2 + 1;
561 UPDATE 2
562 UPDATE min_updates_test SET f3 = 2 WHERE f3 is null;
563 UPDATE 1
564 UPDATE min_updates_test_oids SET f1 = f1;
565 UPDATE 0
566 UPDATE min_updates_test_oids SET f2 = f2 + 1;
567 UPDATE 2
568 UPDATE min_updates_test_oids SET f3 = 2 WHERE f3 is null;
569 UPDATE 1
570 \set QUIET true
571 SELECT * FROM min_updates_test;
572  f1 | f2 | f3 
573 ----+----+----
574  a  |  2 |  2
575  b  |  3 |  2
576 (2 rows)
578 SELECT * FROM min_updates_test_oids;
579  f1 | f2 | f3 
580 ----+----+----
581  a  |  2 |  2
582  b  |  3 |  2
583 (2 rows)
585 DROP TABLE min_updates_test;
586 DROP TABLE min_updates_test_oids;