Note: forwarded message attached.



Do You Yahoo!?
注册世界一流品质的雅虎免费电邮
--- Begin Message ---
The reason I drop the table directly is that:
 
I am developing a graphical information system. There are many tables inherit from object in the system, such as "map", "line", "rect", "circle", etc. In most time, I use tables inherit from "object" instead of table "object".
 
I group the "objects" in such manners as "map" contains the drawable objects, such as "rect", "circle". And some "object" maybe consists of other "objects", such as a "rect" is assembled by four "line". Besides there are also many relationships between "objects".
 
Our principle is to construct a model that reflects the association among objects. We use table "relationship" and tables that inherits from it to record the OID of objects that participate this "relationship". And also, every "object" has a field to record the "relationship"s it participates in.  To maintain the above constraint automatically, I define many triggers on every "object" table and every "relationship" table.
 
To make the system extensible, we provides interface make_object_type( ) to permit the user to create the "object type" inherits from "object". And if the user think the existing table is unappropriate, he may drop it using function destroy_object_type( ) if he has sufficient priviledge.
 
In the first version, the function destroy_object_type( ) just drop the table only. But when I find the triggers havn't been actived yet,  I add the delete setense. This time it becomes even worse, the system throws error message.ERROR:  could not open relation with OID 1390714.
 
To avoid the fatal error, I have no choice but to rollback the function to the original version.
 
I have written a simplified test cast to verify the system, but it performs as I expected and it's wrong.
 
Could you help me?
Thanks!


Michael Fuhr <[EMAIL PROTECTED]> wrote:
On Tue, Dec 07, 2004 at 11:44:58AM +0800, cheng shan wrote:

> I have one table named "object" and i many tables that are related
> to the table "object". When a record in "object" is deleted, the
> relative records in other tables should also be erased. To implement
> the above goal, i define many triggers on table "object" to make
> operations automatically.
>
> But i find that the triggers don't activate when i dropping the
> table "object". So there is many garbage information left in the
> system which makes the system unstable.

If you want to fire the triggers then delete all records from the
table before dropping it. But why are you dropping the table?

--
Michael Fuhr
http://www.fuhr.org/~mfuhr/

---------------------------(end of broadcast)---------------------------
TIP 9: the p lanner will ignore your desire to choose an index scan if your
joining column's datatypes do not match



Do You Yahoo!?
注册世界一流品质的雅虎免费电邮
/****************************************************************************
Function Description:
        _test_setup(): Initialize the environment
        _test_teardown(): Clean up
        _test_object_insert(): Insert records into "object"
        _test_object_update(): Update records of "object"
        _test_object_delete(): Delete records from "object"
        _test_containment_insert(): Insert records into "containment"
        _test_containmnet_delete(): Delete records from "containment"
        _trigger_update_containment(): The trigger on table "object" 
                that is used to update table "containment"
        _trigger_update_object(): The trigger on table "containment" 
                that is used to update table "object"
        _test_delete_and_drop(): Delete records from "object" and drop table 
"object"

        Notice the "!!!!!!"

Usage:
        Before running any test, you should reload test.sql use the command 
        "\i test.sql" for the function will be analysed and cached once it run. 
        If the table was deleted, there will be errors.
        See the results use the command "SELECCT * FROM object;" and 
        "SELECT * FROM containment" to examine the records in table "object"
        and table "containment".
        
        Test inserting records into table "object":
        \i test.sql;
        SELECT _test_setup();
        SELECT _test_object_insert();
        SELECT * FROM object;
        SELECT * FROM containment;
        
        Test updating table "object":
        \i test.sql;
        SELECT _test_setup();
        SELECT _test_object_insert();
        SELECT _test_object_update();
        SELECT * FROM object;
        SELECT * FROM containment;
        
        Test deleting records from "object":
        \i test.sql;
        SELECT _test_setup();
        SELECT _test_object_insert();
        SELECT _test_object_delete();
        SELECT * FROM object;
        SELECT * FROM containment;

        Test inserting records into table "containment":
        \i test.sql;
        SELECT _test_setup();
        SELECT _test_object_insert();
        SELECT _test_containment_insert();
        SELECT * FROM object;
        SELECT * FROM containment;
        
        Test deleting records from table "containment":
        \i test.sql;
        SELECT _test_setup();
        SELECT _test_object_insert();
        SELECT _test_containment_delete();
        SELECT * FROM object;
        SELECT * FROM containment;

!!!!!!  All above test cases can pass. But the following is an exception:
        \i test.sql;
        SELECT _test_setup();
        SELECT _test_object_insert();
        SELECT _test_delete_and_drop();

****************************************************************************/

-----------------------------------------------------------------------------
-- Tests
-----------------------------------------------------------------------------

-- Initialize the test environment
CREATE OR REPLACE FUNCTION _test_setup()
RETURNS void AS $$
BEGIN
        -- Drop table "object"
        IF EXISTS(SELECT * FROM pg_class WHERE relname = 'object') THEN
                DROP TABLE object;
        END IF; 

        CREATE TABLE object
        (
                name name,
                pid oid
        );
        
        -- Drop table "object"
        IF EXISTS(SELECT * FROM pg_class WHERE relname = 'containment') THEN
                DROP TABLE containment;
        END IF; 
        
        CREATE TABLE containment
        (
                parent oid,
                child oid
        );

        CREATE TRIGGER _trigger_update_containment AFTER DELETE OR INSERT OR 
UPDATE ON object 
                FOR EACH ROW EXECUTE PROCEDURE _trigger_update_containment();
        CREATE TRIGGER _trigger_update_object AFTER DELETE OR INSERT ON 
containment 
                FOR EACH ROW EXECUTE PROCEDURE _trigger_update_object();        
        RETURN;
END;
$$ LANGUAGE plpgsql;

-- Clean up
CREATE OR REPLACE FUNCTION _test_teardown()
RETURNS void AS $$
BEGIN
        DROP TABLE object;
        DROP TABLE containment;
        RETURN;
END;
$$ LANGUAGE plpgsql;

-- Insert records into "object"
CREATE OR REPLACE FUNCTION _test_object_insert()
RETURNS void AS $$
DECLARE
        l_pid oid;
BEGIN
        INSERT INTO object(name) VALUES('obj1');
        GET DIAGNOSTICS l_pid = RESULT_OID;
        INSERT INTO object(name, pid) VALUES('obj11', l_pid);
        INSERT INTO object(name, pid) VALUES('obj12', l_pid);
        INSERT INTO object(name) VALUES('obj21');
        INSERT INTO object(name, pid) VALUES('obj22', l_pid);
        INSERT INTO object(name) VALUES('obj23');
        INSERT INTO object(name, pid) VALUES('obj24', l_pid);
        INSERT INTO object(name, pid) VALUES('obj25', l_pid);
        RETURN;
END;
$$ LANGUAGE plpgsql;

-- Update records of "object"
CREATE OR REPLACE FUNCTION _test_object_update()
RETURNS void AS $$
DECLARE
        l_obj1 oid;
        l_obj21 oid;
BEGIN
        SELECT oid INTO l_obj1 FROM object WHERE name = 'obj1';
        SELECT oid INTO l_obj21 FROM object WHERE name = 'obj21';

        -- NEW.pid = OLD.pid = NULL
        UPDATE object SET pid = NULL WHERE name = 'obj21';

        -- NEW.pid = OLD.pid
        UPDATE object SET pid = l_obj1 WHERE name = 'obj22';

        -- OLD.pid = NULL AND NEW.pid <> NULL
        UPDATE object SET pid = l_obj21 WHERE name = 'obj23';

        -- OLD.pid <> NULL AND NEW.pid = NULL
        UPDATE object SET pid = NULL WHERE name = 'obj24';

        -- OLD.pid <> NULL AND NEW.pid <> NULL
        UPDATE object SET pid = l_obj21 WHERE name = 'obj25';

        RETURN;
END;
$$ LANGUAGE plpgsql;

-- Delete records of "object"
CREATE OR REPLACE FUNCTION _test_object_delete()
RETURNS void AS $$
BEGIN
        DELETE FROM object WHERE name = 'obj1';
        RETURN;
END;
$$ LANGUAGE plpgsql;

-- Delete records in "containment"
CREATE OR REPLACE FUNCTION _test_containment_delete()
RETURNS void AS $$
DECLARE
        l_obj12 oid;
BEGIN
        SELECT oid INTO l_obj12 FROM object WHERE name = 'obj12';
        DELETE FROM containment WHERE child = l_obj12;
        RETURN;
END;
$$ LANGUAGE plpgsql;

-- Insert records to "containment"
CREATE OR REPLACE FUNCTION _test_containment_insert()
RETURNS void AS $$
DECLARE
        l_obj11 oid;
        l_obj12 oid;
BEGIN
        SELECT oid INTO l_obj11 FROM object WHERE name = 'obj11';
        SELECT oid INTO l_obj12 FROM object WHERE name = 'obj12';
        INSERT INTO containment(parent, child) VALUES(l_obj11, l_obj12);
        RETURN;
END;
$$ LANGUAGE plpgsql;

--
-- A special test
-- 
CREATE OR REPLACE FUNCTION _test_delete_and_drop()
RETURNS void AS $$
BEGIN
        DELETE FROM object;
        DROP TABLE object;
        RETURN;
END;
$$ LANGUAGE plpgsql;

-----------------------------------------------------------------------------
-- Triggers
-----------------------------------------------------------------------------
-- The trigger on table "object" that is used to update table "containment"
CREATE OR REPLACE FUNCTION _trigger_update_containment()
RETURNS trigger AS $$
DECLARE
        l_rec RECORD;
BEGIN
        IF TG_OP = 'DELETE' THEN
                -- clear all descendants
                FOR l_rec IN SELECT child FROM containment WHERE parent = 
OLD.oid LOOP
                        DELETE FROM object WHERE oid = l_rec.child;
                END LOOP;
                -- delete corresponding record in table containment
                DELETE FROM containment WHERE child = OLD.oid;
        ELSIF TG_OP = 'INSERT' THEN
                -- add record in table containment
                IF NEW.pid NOTNULL THEN
                        INSERT INTO containment(parent, child) VALUES(NEW.pid, 
NEW.oid);
                END IF;         
        ELSIF TG_OP = 'UPDATE' THEN
                -- NEW is equral to OLD
                IF (NEW.pid ISNULL AND OLD.pid ISNULL) OR NEW.pid = OLD.pid THEN
                        RETURN null;
                END IF;
        
                IF OLD.pid ISNULL THEN
                        IF NOT EXISTS(SELECT * FROM containment WHERE child = 
NEW.oid) THEN
                                INSERT INTO containment(parent, child) 
VALUES(NEW.pid, NEW.oid);
                        END IF;
                ELSE
                        IF NEW.pid IS NULL THEN
                                DELETE FROM containment WHERE child = NEW.oid;
                        ELSE
                                UPDATE containment SET parent = NEW.pid WHERE 
child = NEW.oid;
                        END IF;
                END IF;                 
        END IF;
        RETURN null;
END;
$$ LANGUAGE plpgsql;

-- The trigger on table "containment" that is used to update table "object"
CREATE OR REPLACE FUNCTION _trigger_update_object()
RETURNS trigger AS $$
BEGIN
        IF TG_OP = 'DELETE' THEN
                UPDATE object SET pid = null WHERE oid = OLD.child;
        ELSIF TG_OP = 'INSERT' THEN
                UPDATE object SET pid = NEW.parent WHERE oid = NEW.child;
        END IF;
        RETURN null;
END;
$$ LANGUAGE plpgsql;

--- End Message ---
---------------------------(end of broadcast)---------------------------
TIP 7: don't forget to increase your free space map settings

Reply via email to