Re: "record" datatype - plpgsql - Mailing list pgsql-sql

From Jr.
Subject Re: "record" datatype - plpgsql
Date
Msg-id 3ED76CEC.5000200@digitalwizardry.net
Whole thread Raw
In response to "record" datatype - plpgsql  (Brian Knox <laotse@aol.net>)
List pgsql-sql
<br /> I attempted the same thing in pl/pgsql but was unable to find a satisfactory solution using it.  I eventually
startedusing tcl as the procedural language to get this type of effect. Tcl casts NEW and OLD into arrays in a manner
thatmakes it possible.<br /><br /> Original post:<br /> Subject: PL/Pgsql Trigger fcuntion issue..<br /> This is
supposeto pull all the columns of the table that initiated the trigger func from the sys catalogs, loop through them
andput everything that has changed between OLD and NEW into a comma delimited string for input into a log like table
forfuture analysis via middleware (php,perl..,etc).  Here is the problem, OLD.A results in 'old does not have field A',
whichis true. I cant get the OLD and NEW record objects to realize that I want OLD.<string value of A> for the
columnname instead of an explicit A as the column name.  The only way I can find to make this work is by using TCL for
theprocedural language because of the way it casts the OLD and NEW into an associative array instead of a RECORD
object,but by using TCL I will lose functionallity in the "complete" version of the following function which has been
strippedto show my specific problem so using TCL is currently not in my list of options.  Any insight will be greatly
appreciated.<br /> create or replace function hmm() returns TRIGGER as ' <br /> DECLARE <br /> table_cols RECORD; <br
/>attribs VARCHAR; <br /> A VARCHAR; <br /> BEGIN <br /> IF TG_OP = ''UPDATE'' THEN <br />     FOR table_cols IN select
attnamefrom pg_attribute where attrelid = TG_RELID and attnum > -1 LOOP <br />      A := table_cols.attname; <br />
    IF OLD.A != NEW.A THEN  --Begin problem <br />       IF attribs != '''' THEN <br />       attribs := attribs ||
'',''|| table_cols.attname || ''='' || OLD.A || ''->'' || NEW.A; <br />       ELSE <br />       attribs :=
table_cols.attname|| ''='' || OLD.A || ''->'' || NEW.A; <br />       END IF; <br />      END IF; <br />     END
LOOP;<br /> END IF; <br /> RAISE EXCEPTION ''%'', attribs; <br /> RETURN NULL; <br /> END; <br /> ' Language 'plpgsql';
<br/><br /><br /> 1 of my TCL functions (note: a person with good TCL experience may be able to make this shorter or
faster,first thing I ever really did in TCL but it works fast and reliable on a large database):<br /><br /> create
functiontouch_loggerINSUPD() returns OPAQUE as '<br /> spi_exec "select current_user as tguser"<br /> spi_exec "select
relnameas tgname from pg_class where relfilenode = $TG_relid"<br /> if {[string equal -nocase $tguser audit] } {return
OK}<br /> if {[string equal -nocase $TG_op INSERT] } {<br />   set forins ""<br />  foreach i $TG_relatts {<br />   set
forins"$forins^[array get NEW $i]"<br />  }<br />  set themid [lindex [split [array get NEW acid] " "] 1]<br />
 spi_exec-array C "INSERT INTO TOUCHLOG (username,tablename,recid,modtime,action,attributes) values
(''$tguser'',''$tgname'',''$themid'',''now'',''$TG_op'',''$forins'')"<br/> }<br /> if {[string equal -nocase $TG_op
UPDATE]} {<br />   set forins ""<br />   set toadd ""<br />  foreach i $TG_relatts {<br />    if {[string equal -nocase
[arrayget NEW $i] [array get OLD $i]] == 0} {<br />     set toadd "[array get OLD $i]-[array get NEW $i]"<br />     set
forins"$forins^[string trim $toadd \']"<br />    }<br />  }<br />  set themid [lindex [split [array get NEW acid] " "]
1]<br/>  spi_exec -array C "INSERT INTO TOUCHLOG (username,tablename,recid,modtime,action,attributes) values
(''$tguser'',''$tgname'',''$themid'',''now'',''$TG_op'',''$forins'')"<br/> }<br /> return OK<br /> ' LANGUAGE
'pltcl';<br/><br /><br /> Brian Knox wrote:<br /><blockquote
cite="midPine.LNX.4.53.0305300851380.13440@tao.office.aol.com"type="cite"><pre wrap="">That's not what I was asking. I
knowI can access the values of each
 
column using NEW.columnname. What I was asking is if there's anyway to use
the NEW record to get a list of the columnnames in it without knowing them
beforehand.

Brian Knox

On Thu, 29 May 2003, George Weaver wrote:
 </pre><blockquote type="cite"><pre wrap="">Hi Brian;

Assuming "NEW" has been declared as   foo%rowtype,  you can access the
columns thus
   NEW.xxx where xxx is the column name

HTH.
George

----- Original Message -----
From: "Brian Knox" <a class="moz-txt-link-rfc2396E" href="mailto:laotse@aol.net"><laotse@aol.net></a>
To: <a class="moz-txt-link-rfc2396E" href="mailto:pgsql-sql@postgresql.org"><pgsql-sql@postgresql.org></a>
Sent: Thursday, May 29, 2003 11:11 AM
Subject: [SQL] "record" datatype - plpgsql

   </pre><blockquote type="cite"><pre wrap="">Given a variable of the "record" data type in pl/pgsql, is it possible
to
get the names of the columns ( attributes ) of that record?

eg, given record "NEW" for table "foo", is there a way to get information
concerning the columns that make up that record?

Brian Knox

---------------------------(end of broadcast)---------------------------
TIP 5: Have you checked our extensive FAQ?

<a class="moz-txt-link-freetext"
href="http://www.postgresql.org/docs/faqs/FAQ.html">http://www.postgresql.org/docs/faqs/FAQ.html</a>
     </pre></blockquote></blockquote><pre wrap="">
---------------------------(end of broadcast)---------------------------
TIP 1: subscribe and unsubscribe commands go to <a class="moz-txt-link-abbreviated"
href="mailto:majordomo@postgresql.org">majordomo@postgresql.org</a>

 </pre></blockquote><br />

pgsql-sql by date:

Previous
From: Tom Lane
Date:
Subject: Re: "record" datatype - plpgsql
Next
From: C F
Date:
Subject: SQL Help