-
Notifications
You must be signed in to change notification settings - Fork 79
/
versioning_function_nochecks.sql
156 lines (144 loc) · 5.18 KB
/
versioning_function_nochecks.sql
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
-- version 1.0.1
CREATE OR REPLACE FUNCTION versioning()
RETURNS TRIGGER AS $$
DECLARE
sys_period text;
history_table text;
manipulate jsonb;
mitigate_update_conflicts text;
ignore_unchanged_values bool;
include_current_version_in_history bool;
commonColumns text[];
time_stamp_to_use timestamptz;
range_lower timestamptz;
existing_range tstzrange;
newVersion record;
oldVersion record;
user_defined_system_time text;
BEGIN
-- set custom system time if exists
BEGIN
SELECT current_setting('user_defined.system_time') INTO user_defined_system_time;
IF NOT FOUND OR (user_defined_system_time <> '') IS NOT TRUE THEN
time_stamp_to_use := CURRENT_TIMESTAMP;
ELSE
SELECT TO_TIMESTAMP(
user_defined_system_time,
'YYYY-MM-DD HH24:MI:SS.MS.US'
) INTO time_stamp_to_use;
END IF;
EXCEPTION WHEN OTHERS THEN
time_stamp_to_use := CURRENT_TIMESTAMP;
END;
sys_period := TG_ARGV[0];
history_table := TG_ARGV[1];
mitigate_update_conflicts := TG_ARGV[2];
ignore_unchanged_values := COALESCE(TG_ARGV[3],'false');
include_current_version_in_history := COALESCE(TG_ARGV[4],'false');
IF ignore_unchanged_values AND TG_OP = 'UPDATE' THEN
IF NEW IS NOT DISTINCT FROM OLD THEN
RETURN OLD;
END IF;
END IF;
IF TG_OP = 'UPDATE' OR TG_OP = 'DELETE' OR (include_current_version_in_history = 'true' AND TG_OP = 'INSERT') THEN
IF include_current_version_in_history <> 'true' THEN
-- Ignore rows already modified in the current transaction
IF OLD.xmin::text = (txid_current() % (2^32)::bigint)::text THEN
IF TG_OP = 'DELETE' THEN
RETURN OLD;
END IF;
RETURN NEW;
END IF;
END IF;
-- If we we are performing an update or delete we might want to optionally mitigate update conflicts
IF TG_OP = 'UPDATE' OR TG_OP = 'DELETE' THEN
EXECUTE format('SELECT $1.%I', sys_period) USING OLD INTO existing_range;
range_lower := lower(existing_range);
IF mitigate_update_conflicts = 'true' THEN
-- mitigate update conflicts
IF range_lower >= time_stamp_to_use THEN
time_stamp_to_use := range_lower + interval '1 microseconds';
END IF;
END IF;
END IF;
WITH history AS
(SELECT attname
FROM pg_attribute
WHERE attrelid = history_table::regclass
AND attnum > 0
AND NOT attisdropped),
main AS
(SELECT attname
FROM pg_attribute
WHERE attrelid = TG_RELID
AND attnum > 0
AND NOT attisdropped)
SELECT array_agg(quote_ident(history.attname)) INTO commonColumns
FROM history
INNER JOIN main
ON history.attname = main.attname
AND history.attname != sys_period;
-- skip version if it would be identical to the previous version
IF ignore_unchanged_values AND TG_OP = 'UPDATE' AND array_length(commonColumns, 1) > 0 THEN EXECUTE 'SELECT ROW($1.' || array_to_string(commonColumns , ', $1.') || ')'
USING NEW
INTO newVersion;
EXECUTE 'SELECT ROW($1.' || array_to_string(commonColumns , ', $1.') || ')'
USING OLD
INTO oldVersion;
IF newVersion IS NOT DISTINCT FROM oldVersion THEN
RETURN NEW;
END IF;
END IF;
-- If we are including the current version in the history and the operation is an update or delete, we need to update the previous version in the history table
IF include_current_version_in_history = 'true' THEN
IF TG_OP = 'UPDATE' OR TG_OP = 'DELETE' THEN
EXECUTE (
'UPDATE ' ||
history_table ||
' SET ' ||
quote_ident(sys_period) ||
' = tstzrange($2, $3, ''[)'')' ||
' WHERE (' ||
array_to_string(commonColumns , ',') ||
') IS NOT DISTINCT FROM ($1.' ||
array_to_string(commonColumns, ',$1.') ||
') AND ' ||
quote_ident(sys_period) ||
' = $1.' ||
quote_ident(sys_period)
)
USING OLD, range_lower, time_stamp_to_use;
END IF;
-- If we are including the current version in the history and the operation is an insert or update, we need to insert the current version in the history table
IF TG_OP = 'UPDATE' OR TG_OP = 'INSERT' THEN
EXECUTE ('INSERT INTO ' ||
history_table ||
'(' ||
array_to_string(commonColumns , ',') ||
',' ||
quote_ident(sys_period) ||
') VALUES ($1.' ||
array_to_string(commonColumns, ',$1.') ||
',tstzrange($2, NULL, ''[)''))')
USING NEW, time_stamp_to_use;
END IF;
ELSE
EXECUTE ('INSERT INTO ' ||
history_table ||
'(' ||
array_to_string(commonColumns , ',') ||
',' ||
quote_ident(sys_period) ||
') VALUES ($1.' ||
array_to_string(commonColumns, ',$1.') ||
',tstzrange($2, $3, ''[)''))')
USING OLD, range_lower, time_stamp_to_use;
END IF;
END IF;
IF TG_OP = 'UPDATE' OR TG_OP = 'INSERT' THEN
manipulate := jsonb_set('{}'::jsonb, ('{' || sys_period || '}')::text[], to_jsonb(tstzrange(time_stamp_to_use, null, '[)')));
RETURN jsonb_populate_record(NEW, manipulate);
END IF;
RETURN OLD;
END;
$$ LANGUAGE plpgsql;