I created a table to hold some forum messages :

table messages (
        id                      serial primary key,

        parent_folder   integer not null
                                references folders(id), --in another table

        -- for replies, this stores the message which we reply to
        reply_to_id             integer null
                                references messages( id ),

        thread_depth    integer not null default 0

        other fields :
        message_title, message_text, author, etc.
);

I want to know which depth the messages in a thread are, simply to shift them a little more to the right in the display, and I want to store this in the "thread_depth" field so I don't have to compute it at each SELECT I do.

My first thought was to use an "ON INSERT" rule which would change the "thread_depth" of messages inserted with a not-null "reply_to_id" by inserting an extra UPDATE after the INSERT. Problem was, the "NEW.id" which got passed to me by postgresql contained something like :

(nextval('messages_id_seq')::text)::integer

Aside from the funky double cast, trying to read the value incremented the sequence and the UPDATE failed because it tried to update a non-existent row.

So, NEW contained unevaluated default values, BUT reading it evaluates the values (of course) so it consumes an extra sequence number...

        Is this normal ?

        In the end I used a trigger which looks cleaner :

create or replace function vf.messages_set_depth()
returns trigger
as 'begin
if new.reply_to_id is not null then
new.thread_depth=1+(select thread_depth from messages where id=new.reply_to_id limit 1);
end if;
return new;
end;'
language plpgsql;


create trigger messages_set_depth_trigger
        before insert or update on messages
        for each row execute procedure messages_set_depth();

And it works well this way. I don't intend to do mass message insertion so the fact that the trigger is called on every insert does not disturb me. However, the strange behaviour of the rule still bothers me.

        Any comments ?















---------------------------(end of broadcast)---------------------------
TIP 8: explain analyze is your friend

Reply via email to