-
Notifications
You must be signed in to change notification settings - Fork 146
/
Copy pathis_sql.sql
117 lines (108 loc) · 4.14 KB
/
is_sql.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
CREATE OR REPLACE FUNCTION public.is_sql(sql text, is_warning boolean default false)
returns boolean
returns null on null input
parallel unsafe --(ERROR: cannot start subtransactions during a parallel operation)
language plpgsql
set search_path = ''
set client_min_messages = warning --suppress notice [42622] identifier "..." will be truncated to "..."
cost 5
AS
$$
DECLARE
exception_sqlstate text;
exception_message text;
exception_context text;
id text;
BEGIN
BEGIN
--Speed improves. Shortest commands are "abort" or "do ''"
IF octet_length(sql) < 5 OR sql !~ '[A-Za-z]{2}' THEN
return false;
END IF;
-- add ";" in end of string, if ";" does not exist
sql := regexp_replace(sql, $regexp$
;?
((?: #1
--[^\r\n]*? #singe-line comment
| /\* #multi-line comment (can be nested)
[^*/]*? #speed improves
(?: [^*/]+?
| \*[^/] #not end comment
| /[^*] #not begin comment
| #recursive:
/\* #multi-line comment (can be nested)
[^*/]*? #speed improves
(?: [^*/]+?
| \*[^/] #not end comment
| /[^*] #not begin comment
| #recursive:
/\* #multi-line comment (can be nested)
[^*/]*? #speed improves
(?: [^*/]+?
| \*[^/] #not end comment
| /[^*] #not begin comment
#| #recursive
)*?
\*/
)*?
\*/
)*?
\*/
| \s+?
)*?)
$
$regexp$, ';\1', 'x');
id := to_char(now(), 'YYYYMMDDHH24MISSMS');
sql := E'DO $IS_SQL' || id || E'$ BEGIN\nRETURN;\n' || sql || E'\nEND; $IS_SQL' || id || E'$;';
EXECUTE sql;
EXCEPTION WHEN others THEN
IF is_warning THEN
GET STACKED DIAGNOSTICS
exception_sqlstate := RETURNED_SQLSTATE,
exception_message := MESSAGE_TEXT,
exception_context := PG_EXCEPTION_CONTEXT;
RAISE WARNING 'exception_sqlstate = %', exception_sqlstate;
RAISE WARNING 'exception_context = %', exception_context;
RAISE WARNING 'exception_message = %', exception_message;
END IF;
RETURN FALSE;
END;
RETURN TRUE;
END
$$;
COMMENT ON FUNCTION public.is_sql(sql text, is_warning boolean) IS 'Check SQL syntax exactly in your PostgreSQL version';
-- TEST
do $do$
begin
--positive
assert public.is_sql('SELECT x/*--;*/ ; ');
assert public.is_sql('SELECT x ; --');
assert public.is_sql('SELECT -- ; ');
assert public.is_sql('SELECT ; /*select ;*/ --');
assert public.is_sql('ABORT');
assert public.is_sql($$do ''$$);
assert public.is_sql(pg_catalog.pg_get_functiondef('public.is_sql'::regproc::oid)); --self test
--negative
assert not public.is_sql('');
assert not public.is_sql('do');
assert not public.is_sql('123');
assert not public.is_sql('SELECT !');
assert not public.is_sql('-------');
assert not public.is_sql('SELECT ;;');
assert not public.is_sql('SELECT ; ; /*select ;*/ --');
assert not public.is_sql(' --select 1 ');
assert not public.is_sql(' /*select 1*/ ');
assert not public.is_sql('return unknown');
end;
$do$;
--HINT
/*
alter table db_migration
add constraint db_migration_sql_up_check check (is_sql(sql_up)) not valid,
add constraint db_migration_sql_down_check check (
sql_down ~ '^\s*$' OR
sql_comments_remove(sql_down) ~ '^\s*$' OR
is_sql(sql_down)
) not valid;
*/
--see also https://github.com/okbob/plpgsql_check/