snippetsqlModerate
How to conditionally stop a psql script (based on a variable value)?
Viewed 0 times
conditionallyscriptvaluestoppsqlbasedhowvariable
Problem
Let's consider the following example (from the start of a psql script):
Now if it is run this by the command
then it just runs and the user is happy. But what if (s)he decides to specify
So the question arises: how to check the variables passed to a script and stop further processing based on their value?
\c :db_to_run_on
TRUNCATE the_most_important_table;
-- tried to avoid similarities to anything that exists out thereNow if it is run this by the command
psql [connection details] -v db_to_run_on=\'dev_database\'then it just runs and the user is happy. But what if (s)he decides to specify
-v db_to_run_on=production_database? (Let's assume that this can happen, just like people run rm -rf / # don't try this at home!!! ocassionally.) Hopefully there is a fresh backup of that table...So the question arises: how to check the variables passed to a script and stop further processing based on their value?
Solution
There is an option in
The problem is that we have to test the variable and produce an error somehow. Since one can't use control structures in
*: You can use any shell commands after
psql which stops executing commands on error, this is ON_ERROR_STOP. If we could raise an error somehow, this would do what we want. The problem is that we have to test the variable and produce an error somehow. Since one can't use control structures in
psql (because there are none)*, my only idea was to use SQL for testing. Well, producing an error conditionally is something which pl/pgsql is quite good at, so I wrote a function which would generate an error. I can now call this function from a simple CASE structure. A simple example:-- let's assume for clarity that there is no function with this name in the database
CREATE OR REPLACE FUNCTION error_generator()
RETURNS boolean AS
$body$
BEGIN
RAISE 'Meaningful error message here';
RETURN FALSE; -- just for aesthetical purposes
END;
$body$
LANGUAGE plpgsql;
\set ON_ERROR_STOP on
BEGIN;
-- test for the variable value
-- notice that if :var is not set, it fails as well (with a syntax error)
SELECT CASE WHEN 1 = :var THEN error_generator() ELSE TRUE END;
INSERT INTO test_table (integer_value, text_value)
VALUES (:var, 'something');
COMMIT;
*: You can use any shell commands after
\! and conditionals of the shell, but since \! opens a new shell, executing anything there does not have any effect for the current psql script.Context
StackExchange Database Administrators Q#24518, answer score: 18
Revisions (0)
No revisions yet.