Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
107682: plpgsql: add support for CONSTANT variable declarations r=DrewKimball a=DrewKimball

#### plpgsql: validate PLpgSQL functions during creation

This patch ensures that PLpgSQL functions are built during function
creation, not just during invocation. This ensures that compile-time
errors are thrown when the function is created. This commit also tests
and corrects the codes for some of the errors that can be thrown while
building a PLpgSQL function.

Note that some of the existing tests had to be removed since they called
UDFs within other UDFs. We didn't catch this before because the restriction
is only checked during function creation when the body statements are first
built.

Fixes cockroachdb#107681

Release note: None

#### plpgsql: add support for CONSTANT variable declarations

It is possible to declare variables as CONSTANT in PLpgSQL. This will cause
any attempt to assign to the variable to result in a compile-time error.
This patch implements CONSTANT variables by adding a check whenever an
assignment is built that the assigned variable is non-constant.

Fixes cockroachdb#105241

Release note (sql change): Added support for CONSTANT variable
declarations in PLpgSQL routines. Any assignment to a variable declared
with the CONSTANT keyword will raise a compile-time error.

Co-authored-by: Drew Kimball <[email protected]>
  • Loading branch information
craig[bot] and DrewKimball committed Jul 30, 2023
2 parents f295bd8 + a181794 commit e57e974
Show file tree
Hide file tree
Showing 6 changed files with 224 additions and 198 deletions.
325 changes: 151 additions & 174 deletions pkg/sql/logictest/testdata/logic_test/udf_plpgsql
Original file line number Diff line number Diff line change
Expand Up @@ -325,172 +325,16 @@ SELECT f(1, 5), f(-5, 5), f(0, 1)
----
10 10 0

# Dijkstra's Algorithm
#
# ┌─┬────8──┬─┬──7────┬─┐
# ┌─────┤1│ │2│ │3├─────┐
# │ └┬┘ └┬┴───┐ └┬┘ │
# │ │ │ │ │ │
# 4 │ 2 │ │ 9
# │ │ │ │ │ │
# ┌┴┐ │ ┌┴┐ │ │ ┌┴┐
# │0│ 11 ┌───┤8│ │ 14 │4│
# └┬┘ │ │ └┬┘ │ │ └┬┘
# │ │ 7 │ │ │ │
# 8 │ │ 6 │ │ 10
# │ │ │ │ │ │ │
# │ ┌┴┬───┘ ┌┴┐ └───┬┴┐ │
# └─────┤7│ │6│ │5├─────┘
# └─┴────1──┴─┴──4────┴─┘
#
# Encode the graph as a series of undirected edges, where "a" and "b" are the
# "to" and "from" nodes and "weight" is the weight of the edge.
statement ok
CREATE TABLE edges (a INT, b INT, weight INT);
INSERT INTO edges VALUES
(0, 1, 4),
(0, 7, 8),
(1, 7, 11),
(1, 2, 8),
(2, 8, 2),
(7, 8, 7),
(7, 6, 1),
(6, 8, 6),
(2, 5, 4),
(5, 6, 2),
(2, 3, 7),
(3, 5, 14),
(3, 4, 9),
(4, 5, 10);

# Get the number of vertexes in the graph.
statement ok
CREATE FUNCTION vertexes() RETURNS INT AS $$ SELECT max(greatest(a, b)) + 1 FROM edges $$ LANGUAGE SQL;

# Get the maximum int32 value.
statement ok
CREATE FUNCTION max_int() RETURNS INT AS $$ SELECT 2147483647 $$ LANGUAGE SQL;

# Get the weight of the edge between the two given nodes, if any.
statement ok
CREATE FUNCTION graph(x INT, y INT) RETURNS INT AS $$
SELECT coalesce((SELECT weight FROM edges WHERE (a = x AND b = y) OR (a = y AND b = x) LIMIT 1), 0);
$$ LANGUAGE SQL;

# Replace the element at the given index of the array with the given value.
statement ok
CREATE FUNCTION replace(arr INT[], idx INT, val INT) RETURNS INT[] AS $$
DECLARE
i INT;
n INT := array_length(arr, 1);
res INT[] := ARRAY[]::INT[];
BEGIN
i := 0;
LOOP
IF i = idx THEN
res := res || val;
ELSE
res := res || arr[i+1];
END IF;
i := i + 1;
IF i >= n THEN EXIT; END IF;
END LOOP;
RETURN res;
END
$$ LANGUAGE PLpgSQL;

# Return the node with the minimum distance from the source node known so far
# out of the nodes that don't already have a shortest path calculated.
statement ok
CREATE FUNCTION min_distance(dist INT[], spt_set INT[]) RETURNS INT AS $$
DECLARE
n INT := vertexes();
i INT;
min INT := max_int();
min_index INT := 0;
BEGIN
i := 0;
LOOP
IF spt_set[i+1] = 0 AND dist[i+1] <= min THEN
min := dist[i+1];
min_index := i;
END IF;
i := i + 1;
IF i >= n THEN EXIT; END IF;
END LOOP;
RETURN min_index;
END
$$ LANGUAGE PLPGSQL;

# Implement dijkstra's algorithm using the "edges" table.
statement ok
CREATE FUNCTION dijkstra(src INT) RETURNS INT[] AS $$
DECLARE
n INT := vertexes();
i INT;
count INT;
dist INT[] := ARRAY[]::INT[];
spt_set INT[] := ARRAY[]::INT[];
u INT;
BEGIN
i := 0;
LOOP
dist := dist || max_int();
spt_set := spt_set || 0;
i := i + 1;
IF i >= n THEN EXIT; END IF;
END LOOP;
dist := replace(dist, src, 0);
count := 0;
LOOP
u := min_distance(dist, spt_set);
spt_set := replace(spt_set, u, 1);
i := 0;
LOOP
IF
spt_set[i+1] = 0 AND
graph(u, i) > 0 AND
dist[u+1] <> max_int() AND
dist[u+1] + graph(u, i) < dist[i+1]
THEN
dist := replace(dist, i, dist[u+1] + graph(u, i));
END IF;
i := i + 1;
IF i >= n THEN EXIT; END IF;
END LOOP;
count := count + 1;
IF count >= n THEN EXIT; END IF;
END LOOP;
RETURN dist;
END
$$ LANGUAGE PLPGSQL;
# TODO(drewk): add back the dijkstra test once UDFs calling other UDFs is
# allowed.

# Run dijkstra's algorithm using node 0 as the source.
query II nosort,colnames
SELECT i AS "Vertex", dist[i+1] AS "Distance From Source"
FROM generate_series(0, vertexes() - 1) f(i), dijkstra(0) g(dist);
----
Vertex Distance From Source
0 0
1 4
2 12
3 19
4 21
5 11
6 9
7 8
8 14

statement ok
statement error pgcode 2F005 control reached end of function without RETURN
CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
BEGIN
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 2F005 control reached end of function without RETURN
SELECT f(1, 2);

statement ok
CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
DECLARE
i INT;
Expand All @@ -500,9 +344,6 @@ CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
$$ LANGUAGE PLpgSQL;

statement error pgcode 2F005 control reached end of function without RETURN
SELECT f(1, 2);

statement ok
CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
BEGIN
IF a < b THEN
Expand All @@ -512,9 +353,6 @@ CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
$$ LANGUAGE PLpgSQL;

statement error pgcode 2F005 control reached end of function without RETURN
SELECT f(1, 2);

statement ok
CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
DECLARE
i INT;
Expand All @@ -528,9 +366,6 @@ CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
$$ LANGUAGE PLpgSQL;

statement error pgcode 2F005 control reached end of function without RETURN
SELECT f(1, 2);

statement ok
CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
BEGIN
LOOP
Expand All @@ -540,9 +375,6 @@ CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
$$ LANGUAGE PLpgSQL;

statement error pgcode 2F005 control reached end of function without RETURN
SELECT f(1, 2);

statement ok
CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
BEGIN
LOOP
Expand All @@ -554,9 +386,6 @@ CREATE OR REPLACE FUNCTION f(a INT, b INT) RETURNS INT AS $$
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 2F005 control reached end of function without RETURN
SELECT f(1, 2);

statement error pgcode 0A000 PL/pgSQL functions with RECORD input arguments are not yet supported
CREATE FUNCTION f_err(p1 RECORD) RETURNS RECORD AS $$
BEGIN
Expand Down Expand Up @@ -831,3 +660,151 @@ $$ LANGUAGE PLpgSQL;

query error pgcode P0001 pq: foo
SELECT f();

statement error pgcode 42601 pq: too few parameters specified for RAISE
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
BEGIN
RAISE 'foo% % %', 1, 2;
RETURN 0;
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 42601 pq: too many parameters specified for RAISE
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
BEGIN
RAISE 'foo%', 1, 2;
RETURN 0;
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 42601 pq: RAISE option already specified: ERRCODE
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
BEGIN
RAISE EXCEPTION USING ERRCODE = '22012', ERRCODE = '22013';
return 0;
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 42601 pq: \"i\" is not a known variable
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
BEGIN
i := 0;
RETURN i;
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 42601 CONTINUE cannot be used outside a loop
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
BEGIN
CONTINUE;
RETURN i;
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 42601 EXIT cannot be used outside a loop, unless it has a label
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
BEGIN
EXIT;
RETURN i;
END
$$ LANGUAGE PLpgSQL;

statement ok
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
DECLARE
i CONSTANT INT;
BEGIN
RETURN i;
END
$$ LANGUAGE PLpgSQL;

query I
SELECT f();
----
NULL

statement ok
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
DECLARE
i CONSTANT INT := 0;
BEGIN
RETURN i;
END
$$ LANGUAGE PLpgSQL;

query I
SELECT f();
----
0

statement ok
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
DECLARE
i CONSTANT INT := (SELECT x FROM xy ORDER BY x LIMIT 1);
BEGIN
RETURN i;
END
$$ LANGUAGE PLpgSQL;

query I
SELECT f();
----
1

statement ok
CREATE OR REPLACE FUNCTION f(n INT) RETURNS INT AS $$
DECLARE
i CONSTANT INT := n;
BEGIN
RETURN i;
END
$$ LANGUAGE PLpgSQL;

query IIIIII
SELECT f(-100), f(-1), f(0), f(1), f(100), f(NULL);
----
-100 -1 0 1 100 NULL

statement error pgcode 22005 pq: variable \"i\" is declared CONSTANT
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
DECLARE
i CONSTANT INT;
BEGIN
i := i + 1;
RETURN i;
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 22005 pq: variable \"i\" is declared CONSTANT
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
DECLARE
i CONSTANT INT := 0;
BEGIN
i := i + 1;
RETURN i;
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 22005 pq: variable \"i\" is declared CONSTANT
CREATE OR REPLACE FUNCTION f(n INT) RETURNS INT AS $$
DECLARE
i CONSTANT INT := 0;
BEGIN
IF n > 0 THEN
i := i + 1;
END IF;
RETURN i;
END
$$ LANGUAGE PLpgSQL;

statement error pgcode 22005 pq: variable \"i\" is declared CONSTANT
CREATE OR REPLACE FUNCTION f() RETURNS INT AS $$
DECLARE
i CONSTANT INT := 0;
BEGIN
LOOP IF i >= 10 THEN EXIT; END IF;
i := i + 1;
END LOOP;
RETURN i;
END
$$ LANGUAGE PLpgSQL;
Loading

0 comments on commit e57e974

Please sign in to comment.