PL / pgSQL: general way to update N columns in triggers?

I am trying to create a function that will take a general table and convert N columns to uppercase. I had no luck finding a solution to this problem, but I was able to come up with the following:

create or replace function uc_on_insert()
returns trigger as
$$
declare
p_tbl varchar = TG_TABLE_NAME;
p_sch varchar = TG_TABLE_SCHEMA;
i varchar;
begin
    for i in 

    (select column_name
    from  INFORMATION_SCHEMA.COLUMNS
    where 1=1
    and table_name ilike p_tbl
    and table_schema ilike p_sch
    and data_type ='character varying')

    loop
           execute 'new.' || i || ' = upper(new.' || i || ');';
       return new;
    end loop;

end;
$$ language plpgsql;

      

I am currently getting this error:

ERROR:  syntax error at or near "new"
LINE 1: new.c1 = upper(new.c1);
        ^
QUERY:  new.c1 = upper(new.c1);

      

The expected input would be, on any table I have this trigger:

insert into table_one('a', 'b');

>> A, B

      

and if i put this trigger in another table:

insert into table_two ('a', 3);
>> A, 3

      

etc..

+3
types plpgsql triggers postgresql dynamic-sql


source to share


1 answer


That's a very difficult question.

Your attempt is in error because the current string variable is NEW

not visible internally EXECUTE

. And even if it were, it NEW

is a string type (record), not an array. Unlike array elements, string fields cannot be referenced by a numeric index. This will cause all sorts of problems in SQL because (unlike an array) each field can have a different datatype, and SQL expects to know the datatype to process in advance. Very difficult.

Generic method for selected types only

Fortunately, we ran into a similar problem earlier:

  • How to set the value of a composite variable field using dynamic SQL

You will find a sufficient explanation here.
Adapted for trigger function and depending on the data type of the columns, it might look like this:

Trigger function

CREATE OR REPLACE FUNCTION trg_uc_on_insert()
  RETURNS trigger AS
$func$
BEGIN

EXECUTE 'SELECT ' || array_to_string(ARRAY(
      SELECT CASE WHEN atttypid = 'varchar'::regtype
                    -- atttypid = ANY('{text, bpchar, varchar}'::regtype[])
                THEN 'upper(($1).' || quote_ident(attname)
                                   || ')::' || atttypid::regtype::text 
                ELSE '($1).' || quote_ident(attname)
             END AS fld
      FROM   pg_catalog.pg_attribute
      WHERE  attrelid = pg_typeof(NEW)::text::regclass
      AND    attnum > 0
      AND    attisdropped = FALSE
      ORDER  BY attnum
      ), ',')
USING  NEW
INTO   NEW;

RETURN NEW;

END
$func$ LANGUAGE plpgsql;

      

If you want to apply the same rule to other basic character types, use the commented alternative.

Trigger

CREATE TRIGGER trg_t_insbef
BEFORE INSERT
ON t              -- works for any table
FOR EACH ROW
EXECUTE PROCEDURE trg_uc_on_insert();

      



SQL Fiddle.

Simple, non-specific method

As long as you only use simple types in your tables and want to have all character data in uppercase , there is another rough, quick and simple method: cast the whole string to text

, uppercase text representation, revert to the string type, and update NEW

with the result. Details of the table row type

  • Check entire table for one value

Trigger function

CREATE OR REPLACE FUNCTION trg_uc_simple_on_insert()
  RETURNS trigger AS
$func$
BEGIN

EXECUTE 'SELECT ($1::' || pg_typeof(NEW) || ').*'
   USING upper(NEW::text)
   INTO NEW;

RETURN NEW;

END
$func$ LANGUAGE plpgsql;

      

SQL Fiddle.

We have to decompose the line type because it SELECT INTO

assigns separate target line type fields one by one. We cannot assign the entire line at once. If you look closely, the "generic" solution does the same, less obvious one.

While character data is case sensitive in textual representation, other underlying numeric or date data types are not. Thus, the simple method works reliably. Probably with most other types as well. But I haven't tested others and there is definitely an exception. You need to check the data types used.

Also, while the code is much shorter and simpler than the generic method, it is not necessarily faster, especially with a lot of unaffected columns. This is probably much faster in simple cases.

+3


source to share







All Articles
Loading...
X
Show
Funny
Dev
Pics