Showing posts with label Triggers. Show all posts
Showing posts with label Triggers. Show all posts

(ref) System Event Triggers


System Event Triggers

About:
  • Created on either a schema or the database
  • A SCHEMA trigger is created on a schema and fires whenever the user who owns it is the current user and initiates the triggering event.
  • A DATABASE trigger is created on the database and fires whenever any database user initiates the triggering event.

Syntax:
CREATE [OR REPLACE] TRIGGER trigger_name
{BEFORE | AFTER} database_event ON {database | schema}
[DECLARE]
declaration_statements;
BEGIN
execution_statements;
[EXCEPTION ...]
END [trigger_name];
/


Valid events are listed below.
  • DDL Events: ALTER, ANALYZE, ASSOCIATE STATISTICS, AUDIT, COMMENT, CREATE, DISASSOCIATE STATISTICS, DROP, GRANT, NOAUDIT, RENAME, REVOKE, TRUNCATE, DDL
  • Database Events: AFTER STARTUP, BEFORE SHUTDOWN, AFTER DB_ROLE_CHANGE, AFTER SERVERERROR, AFTER LOGON, BEFORE LOGOFF, AFTER SUSPEND.


Trigger on Schema
- The trigger fires to stop any drop of DB objects belonging to the HR schema.
CREATE OR REPLACE TRIGGER stop_drop_tg
 BEFORE DROP ON hr.SCHEMA
BEGIN
  RAISE_APPLICATION_ERROR (
    num => -20000,
    msg => 'Cannot drop object');
END;
/

Trigger on Database
- When user connects, the trigger runs the connecting procedure in the user_connecting package.
- Before logoff the procedure disconnecting is executed.
CREATE OR REPLACE TRIGGER connect_tg
AFTER LOGON ON DATABASE
BEGIN
user_connection.connecting(sys.login_user);
END;
/

CREATE OR REPLACE TRIGGER disconnect_tg
BEFORE LOGOFF ON DATABASE
BEGIN
user_connection.disconnecting(sys.login_user);
END;
/

Providing Fine-Grained Access Control

As suggested in the examples above, you can also use LOGON triggers to run a package associated with an application context.
An application context captures session-related information about the user who is logging in to the database. From there, your application can control how much access this user has, based on his or her session information.

Note:
If you have very specific logon requirements, such as preventing users from logging in from outside the firewall or after work hours, consider using Oracle Database Vault instead of LOGON triggers. With Oracle Database Vault, you can create custom rules to strictly control user access.

(ref) On Compound Triggers

Compound Triggers

About Compound triggers:
  • First appeared in Oracle 11g.
  • Can fire at multiple timing points.
  • Each timing point section has its own executable parte and optional exection-handling part.
  • Act as both statement- and row-level triggers.
  • Can be defined on either a table or a view
  • Do not support filtering actions with the WHEN clause or the use of the autonomous transaction PRAGMA
  • The firing order of compound triggers is not guaranteed because they can be interleaved (mixed between) with the firing of standalone triggers.
  • There is a declaration section for the trigger as a whole. Individual timing points share this single global declaration section, whose state is maintained for the lifetime of the statement. In addition, each timing point section has its own local declaration section.
  • Compound triggers do not support an EXCEPTION block, but you can implement EXCEPTION
    blocks in any of the subordinate timing point blocks.
  • Compound triggers do not fire when (a) the DML statement doesn’t change any rows and (b) the trigger does not implement at least a BEFORE STATEMENT or AFTER STATEMENT block.
  • In previous releases this type of functionality was only possible by defining multiple triggers whose code and global variables were defined in a separate package.

Common uses of Compound DML Triggers:
  • Accumulate rows - with values obtained or generated at each "EACH ROW" firing - so that these can be inserted with BULK-INSERT
  • To avoid the mutating-table error (ORA-04091)

Restrictions:
  • OLD, NEW, and PARENT cannot appear in the declarative part, the BEFORE STATEMENT section, or the AFTER STATEMENT section.
  • Only the BEFORE EACH ROW section can change the value of NEW.
  • A timing-point section cannot handle exceptions raised in another timing-point section.
  • If a timing-point section includes a GOTO statement, the target of the GOTO statement must be in the same timing-point section.


Syntax:
CREATE [OR REPLACE] TRIGGER trigger_name
 FOR {INSERT | UPDATE | UPDATE OF column1 [, column2 [, column(n+1)]] | DELETE}
 ON table_name
 COMPOUND TRIGGER
[BEFORE STATEMENT IS
  [declaration_statement;]
BEGIN
  execution_statement;
END BEFORE STATEMENT;]
[BEFORE EACH ROW IS
  [declaration_statement;]
BEGIN
  execution_statement;
END BEFORE EACH ROW;]
[AFTER EACH ROW IS
  [declaration_statement;]
BEGIN
  execution_statement;
END AFTER EACH ROW;]
[AFTER STATEMENT IS
  [declaration_statement;]
BEGIN
  execution_statement;
END AFTER STATEMENT;]
END [trigger_name];
/


Mutating-Table error: what is it?

  • A mutating table is a table that is being modified by a DML statement (i.e. by the effects of a DELETE CASCADE constraint).
  • If a trigger is fired as the result of a DML statement in table A, it cannot query or modify table A (the table that the triggering statement is modifying.)
  • When a row-level trigger encounters a mutating table, ORA-04091 occurs, the effects of the trigger and triggering statement are rolled back, and control returns to the user or application that issued the triggering statement.

Example:

(a) create table ELOG to log updates in the table EMPLOYEES
(b) create trigger to insert into ELOG the values (:OLD) that are modified when an UPDATE is issued against EMPLOYEES.
(c) The trigger should (1) Insert the OLD values into ELOG and (2) Print a statement with the number of rows remaining on EMPLOYEES.
  • For this, it has to perform a SELECT in the table that is being modified (EMPLOYEES). And there the bicho pega...

create table elog (
  emp_id number(6),
  l_name varchar2(25),
  f_name varchar2(20));

create or replace trigger log_deletions
  after delete on employees
  for each row
declare
 n integer;
begin
 insert into log values
   (:old.employee_id,
    :old.last_name, 
    :old.first_name);
 select count(*) into n from employees;
 dbms_output.put_line('there are now ' || n || ' employees.');
end;

delete from employees where employee_id=197;

SQL Error: ORA-04091: table EMPLOYEES is mutating, trigger/function may not see it
ORA-06512: at "LOG_DELETIONS", line 8
ORA-04088: error during execution of trigger 'LOG_DELETIONS'
04091. 00000 -  "table %s.%s is mutating, trigger/function may not see it"
*Cause:    A trigger (or a user defined plsql function that is referenced in
           this statement) attempted to look at (or modify) a table that was
           in the middle of being modified by the statement which fired it.
*Action:   Rewrite the trigger (or function) so it does not read that table.

(?) How to solve this error?

(1) You may Use a temporary table
Instead of using one AFTER each row trigger that updates the mutating table, use two triggers
(a) an AFTER each row trigger that updates the temporary table and
(b) an AFTER statement trigger that updates the mutating table with the values from the temporary table.

(2) Use a compound DML trigger:


Compound Trigger: Bulk Logging updates to solve ORA-04091
The example below uses a compound trigger to log salary updates in the employees table, and capture old and new values for the updated columns.
(a) create table emp_log_update
(b) create sequence emp_log_update_seq
(c) Create trigger emp_log_tg
(d) Configure client information (use sys_context to set client info)
(e) Update rows on employees to fire the compound trigger
(f) check the inserted row on emp_log_update

-- create table to log updates. The table will be used by the compound trigger.
create table emp_log_update
 (emp_log_id       number,
  emp_id           number,
  oldval           varchar2(20),
  newval           varchar2(20),
  created_by       varchar2(20), 
  creation_date    date,
  last_updated_by  varchar2(20),
  last_update_date date);

create sequence emp_log_update_seq;

-- create compound trigger. 
-- The For each row section: stores :old and :new values for each updated row
-- The statement section: performs a bulk insert in the log table.

CREATE OR REPLACE TRIGGER employee_update_tg
 FOR UPDATE of SALARY on EMPLOYEES
 COMPOUND TRIGGER
 TYPE emp_rec IS RECORD
  (emp_log_id      emp_log_update.emp_log_id%type,
   emp_id          emp_log_update.emp_id%type,
   oldval          emp_log_update.oldval%type,
   newval          emp_log_update.newval%type,
   created_by      emp_log_update.created_by%type, 
   creation_date   emp_log_update.creation_date%type,
   last_updated_by emp_log_update.last_updated_by%type,
   last_update_date emp_log_update.last_update_date%type);
 
 TYPE emp_list IS TABLE OF emp_rec;
 emp_updates emp_list := emp_list();

BEFORE EACH ROW IS
 -- declare local timing point variables
 c number;
 -- stores the client info (identification, set up in the application)
 user_id varchar2(20) := nvl((sys_context('userenv','client_info')), -1);
BEGIN
 emp_updates.EXTEND;
 c := emp_updates.LAST;
 emp_updates(c).emp_log_id := emp_log_update_seq.nextval; 
 emp_updates(c).emp_id := :old.employee_id; 
 emp_updates(c).oldval := to_char(:old.salary); 
 emp_updates(c).newval := to_char(:new.salary); 
 emp_updates(c).created_by := user_id; 
 emp_updates(c).creation_date := sysdate; 
 emp_updates(c).last_updated_by := user_id; 
 emp_updates(c).last_update_date := sysdate; 
END BEFORE EACH ROW;
AFTER STATEMENT IS
BEGIN
 FORALL i in emp_updates.FIRST..emp_updates.LAST
   insert into emp_log_update
   values (
      emp_updates(i).emp_log_id, 
      emp_updates(i).emp_id, 
      emp_updates(i).oldval, 
      emp_updates(i).newval, 
      emp_updates(i).created_by,
      emp_updates(i).creation_date,
      emp_updates(i).last_updated_by,
      emp_updates(i).last_update_date
      );
END AFTER STATEMENT;
END;

Trigger created.

-- Set the client info: 
SQL> EXEC dbms_application_info.set_client_info('dev2');

PL/SQL procedure successfully completed.

SQL>update employees set salary=salary*1.15 where department_id=60;
5 rows updated.

SQL> select * from emp_event_log;
(formatted result..)
ID  EMP_ID  OLDVAL    NEWVAL   CREATED_BY   CREATION  LAST_UPDATED  LAST_UPDATED
                                              DATE         BY          DATE
--- ------- --------- -------- ------------ --------- ------------- ------------
  2     103 9000      10350    dev2         26-AUG-11 dev2          26-AUG-11
  3     104 6000      6900     dev2         26-AUG-11 dev2          26-AUG-11
  4     105 4800      5520     dev2         26-AUG-11 dev2          26-AUG-11
  5     106 4800      5520     dev2         26-AUG-11 dev2          26-AUG-11
  6     107 4200      4830     dev2         26-AUG-11 dev2          26-AUG-11
DBMS_APPLICATION_INFO package
  • Use DBMS_APPLICATION_INFO.{SET_CLIENT_INFO | READ_CLIENT_INFO} to maintain client info
  • Read/Write column CLIENT_INFO on V$SESSION and V$SQL_AREA
  • You can set module name or (trans)action name.
  • Also useful with Oracle Trace and SQL Trace: to record names of modules or transactions for performance analysis or debugging.
  • set action name before the start of a transaction.
  • DBMSAPIN.SQL run by catproc

#set module and transaction name
CREATE or replace PROCEDURE add_employee( 
  name VARCHAR2, 
  salary NUMBER, 
  manager NUMBER, 
  title VARCHAR2, 
  commission NUMBER, 
  department NUMBER) AS 
BEGIN 
  DBMS_APPLICATION_INFO.SET_MODULE( 
    module_name => 'add_employee', 
    action_name => 'insert into emp'); 
  INSERT INTO emp 
    (ename, empno, sal, mgr, job, hiredate, comm, deptno) 
    VALUES (name, emp_seq.nextval, salary, manager, title, SYSDATE, 
            commission, department); 
  DBMS_APPLICATION_INFO.SET_MODULE(null,null); 
END;


# Set transaction name
CREATE OR REPLACE PROCEDURE bal_tran (amt IN NUMBER(7,2)) AS 
BEGIN 
-- balance transfer transaction 
   DBMS_APPLICATION_INFO.SET_ACTION(
      action_name => 'transfer from chk to sav'); 
   UPDATE chk SET bal = bal + :amt 
      WHERE acct# = :acct; 
   UPDATE sav SET bal = bal - :amt 
      WHERE acct# = :acct; 
   COMMIT; 
   DBMS_APPLICATION_INFO.SET_ACTION(null); 
END;  

Other example:
Session 1 (anonymous block):
(a) set action and client info.
(b) delete rows from employees

Session 2:
(a) query V$SESSION

#SID 38: 
SQL> BEGIN
    dbms_application_info.set_action(action_name => 'delete from temp_emp');
    dbms_application_info.set_client_info(client_info => 'Issued from terminal 2');
    delete from temp_emp where city='Denver';
    END;
SQL> /

PL/SQL procedure successfully completed

#SID 2:
SQL> select sid, serial#, username, osuser, module, action, client_info from v$session;

(formatted output)
 SID  SERIAL# USERNAME  OSUSER   MODULE     ACTION                CLIENT_INFO
---- -------- --------- -------- ---------- --------------------- ----------------------
   2       10 SYSTEM    jdba     SQL*Plus
(...) 
  38        7 JFARR     jdev     SQL*Plus   delete from temp_emp  Issued from terminal 2
(...)



Oracle DML Triggers

DML Triggers

DML triggers:
  • Simple
  • Compound
    • can fire at one, some, or all of the preceding timing points
  • INSTEAD OF
    • Created on a noneditioning view, or on a nested table column of a noneditioning view.
    • Used to perform a DML action on the underlying tables of a view.
    • The database fires the INSTEAD OF trigger instead of running the triggering DML statement.
  • Crossedition
    • For use only in edition-based redefinition

  • Triggering event: DELETE, INSERT, and UPDATE.
  • Simple DML trigger fires at exactly one of these timing points:
  • Before the triggering statement runs (BEFORE trigger).
  • After the triggering statement runs (AFTER trigger).
    • Statement triggers are always executed, independently of whether the triggering event actually affected any rows in the underlying table.
    • Statement-level triggers are also known as table-level triggers.
    • You cannot use a WHEN clause in a statement-level trigger.
    • You also cannot reference the new or old pseudo-records in a statement-level trigger. Doing so results in a compile-time error (ORA-04082: NEW or OLD references not allowed in table level triggers).
  • Before each row that the triggering statement affects (BEFORE each row trigger).
  • After each row that the triggering statement affects (AFTER each row trigger).
    • If the DML statement does not affect any row the trigger is NOT executed.

DML Triggers and DCL (Commit, Rollback, Savepoint)
  • You cannot use SQL Data Control Language (DCL) in them, unless you declare the trigger as autonomous.
  • When triggers run inside the scope of a transaction, they disallow setting a SAVEPOINT or performing either a ROLLBACK or COMMIT statement.
  • Also, No function or procedure called by a trigger can have a DCL statement in their execution path.

DML Trigger: Syntax
CREATE [OR REPLACE] TRIGGER trigger_name
 {BEFORE | AFTER}
 {INSERT | UPDATE | UPDATE OF column1 [, column2 [, column(n+1)]] | DELETE}
 ON table_name
 [FOR EACH ROW]
 [WHEN (logical_expression)]
[DECLARE]
  [PRAGMA AUTONOMOUS_TRANSACTION;]
  declaration_statements;
BEGIN
  execution_statements;
END [trigger_name];
/

Trigger firing sequence for any DML (I/U/D)
  1. Before DML FOR EACH STATEMENT.
  2. Before DML FOR EACH ROW.
  3. DML statement executed.
  4. After DML FOR EACH ROW.
  5. After DML FOR EACH STATEMENT.


(?) How many triggers can you have for INSERT on Employees?
  • No restrictions. You can create many for a single event.
  • However, the ORDER of firing is not guaranteed.
  • Consolidate triggers, if sequence in important.

Before UPDATE Statement-level Triggers
  • Useful to ensure integrity rules (i.e. new value should be within some range) or business requirements (i.e. ensure that user has required privileges for operation).
  • Can be also used to check whether the triggering event should be allowed to complete successfully. If the statement is found to be not authorized, the trigger could, for example, log it on an audit table, send notification messages, and raise an exception so that the statement fails.

SQL> CREATE or REPLACE TRIGGER audit_emp_hours
 BEFORE INSERT OR UPDATE OR DELETE 
 ON employees
BEGIN
  IF TO_CHAR (SYSDATE, 'HH24:MI') NOT BETWEEN '08:00' AND '18:00'
        OR TO_CHAR (SYSDATE, 'DY') IN ('SAT', 'SUN') THEN
    RAISE_APPLICATION_ERROR (-20205,
		'You may only make changes during normal office hours');
  END IF;
END;
/
Trigger Created.
SQL> 

SQL> update employees set salary=43000 where employee_id=210;
update employees set salary=43000 where employee_id=210
         *
ERROR at line 1:
ORA-20205: You may only make changes during normal office hours
ORA-06512: at "HR.AUDIT_EMP_HOURS", line 4
ORA-04088: error during execution of trigger 'HR.AUDIT_EMP_HOURS'


Using CONDITIONAL PREDICATES to audit actions/events:
  • INSERING
  • UPDATING
  • DELETING
{ TRUE | FALSE }

The trigger below writes into an audit table (AUDITBL) when any DML statement is executed on the EMPLOYEES table.

(a) Create the Audit table (AUDITBL)

SQL> CREATE TABLE AudiTbl ( 
     id_audit NUMBER  NOT NULL , 
     username VARCHAR2 (20) , 
     action VARCHAR2 (400) , 
     dt_action DATE , 
     table_name VARCHAR2 (100) , 
     old_value VARCHAR2 (400) , 
     new_value VARCHAR2 (400) 
    );
Table created.

SQL>ALTER TABLE AudiTbl 
    ADD CONSTRAINT "AudiTblPK" PRIMARY KEY (id_audit);
Table altered.

SQL> CREATE SEQUENCE seq_audit START WITH 1;
Sequence created.
SQL> 

(b) Create Conditional AFTER UPDATE/INSERT/DELETE Trigger on (EMPLOYEES).

SQL> Set Serveroutput on;

SQL> CCREATE OR REPLACE TRIGGER audit_employees
 AFTER
    INSERT OR
    UPDATE OF salary OR
    UPDATE OF department_id OR
    DELETE
  ON employees
  FOR EACH ROW
BEGIN
  CASE
    WHEN INSERTING THEN
      INSERT into AudiTbl VALUES
       (seq_audit.nextval, user, 'INSERT', sysdate, 'employees', null, null);
      DBMS_OUTPUT.PUT_LINE('Inserting on employees');
    WHEN UPDATING('salary') THEN
      DBMS_OUTPUT.PUT_LINE('Updating salary');
      INSERT into AudiTbl VALUES
       (seq_audit.nextval, user, 'UPDATE salary', sysdate, 'employees', 
        :OLD.salary, :NEW.salary);
    WHEN UPDATING('department_id') THEN
      DBMS_OUTPUT.PUT_LINE('Updating department ID');
      INSERT into AudiTbl VALUES
       (seq_audit.nextval, user, 'UPDATE Dept_id', sysdate, 'employees', 
        :OLD.department_id, :NEW.department_id);
    WHEN DELETING THEN
      DBMS_OUTPUT.PUT_LINE('Deleting');
      INSERT into AudiTbl VALUES
       (seq_audit.nextval, user, 'DELETE', sysdate, 'employees', 
        :OLD.employee_id, null);
  END CASE;
END;
/

Notes:
  • OLD, NEW and PARENT are correlation names. :OLD and :NEW qualifiers can only be used in row level triggers. These reference the old and new values of a column, respectively.







(c) Test the INSERT trigger on (EMPLOYEES).

SQL> insert into employees (employee_id, first_name, last_name, email, 
         phone_number, hire_date, job_id, salary, commission_pct, 
         manager_id, department_id)
values (employees_seq.nextval, 'John', 'Fergunson', 'email@email.com', 
        null, sysdate, 'IT_PROG', 20000, null, null, null);
Inserting on employees

1 row created.
SQL>
SQL> Select * from auditbl;

ID_AUDIT USERNAME ACTION DT_ACTION TABLE_NAME OLD_VALUE NEW_VALUE
1 JOHNF INSERT 18-AUG-10 employees    

(d) Test the UPDATE (salary) trigger

SQL> update employees set salary=salary*2 where employee_id=210;
Updating salary

1 row updated.
SQL>
SQL> Select * from auditbl;

ID_AUDIT USERNAME ACTION DT_ACTION TABLE_NAME OLD_VALUE NEW_VALUE
1 JOHNF INSERT 18-AUG-10 employees    
2 MARCJ UPDATE 11-Sep-10 employees 15000 30000



Row-Level Update Trigger
  • The trigger below is fired when an employee is assigned to the department 10.
  • It fires AFTER an UPDATE on emploees, but ONLY WHEN the update was on the department_id COLUMN AND the new value for that column is set to 10.
  • Note that in the WHEN clause you use NEW. In the trigger bodyyou use :NEW.
  • Inside a trigger body, you preface the pseudo-records with a colon (:). The colon let you reference the externally scoped pseudo-records in the trigger body.

SQL> CREATE OR REPLACE TRIGGER audit_dept10
  AFTER UPDATE OF department_id
  ON employees
  FOR EACH ROW
  WHEN (new.department_id = 10)
BEGIN
  DBMS_OUTPUT.PUT_LINE('Update on employees');
  INSERT into AudiTbl VALUES
    (seq_audit.nextval, user, 'UPDATE', sysdate, 'employees', 
        :old.department_id, :new.department_id);
END;
/

Trigger created.
SQL> set serveroutput on
SQL> update employees 
        set department_id = 10 
        where employee_id in (201, 202, 203, 210);

4 rows updated.
Update on employees
Update on employees
Update on employees
Update on employees

 
SQL> select * from auditbl;



ID_AUDIT USERNAME ACTION DT_ACTION TABLE_NAME OLD_VALUE NEW_VALUE
18 JAFFN UPDATE 26-AUG-11 employees 70 10
18 JAFFN UPDATE 26-AUG-11 employees 20 10
18 JAFFN UPDATE 26-AUG-11 employees 20 10
18 JAFFN UPDATE 26-AUG-11 employees 40 10


Exception handling in Triggers







(ref) Oracle DDL Triggers

DDL Triggers
Can be used to:
  • Monitor significant events in the database.
  • Monitor errant code that can that can corrupt or destabilize your database.
  • Use these in development, test, and stage systems to understand and monitor the dynamics of database activities.
  • Also useful when you patch your application code. They can let you find
    potential changes between releases.
  • During an upgrade: use instead-of create trigger to enforce table creation storage clauses or partitioning rules.
  • Track the creation and modification of tables by application programs that lead to database fragmentation.
  • Effective security tools: monitor GRANT and REVOKE privilege statements.

When can they fire? DDL Events
ALTER, ANALYZE, ASSOCIATE STATISTICS, AUDIT, COMMENT, CREATE, DDL (means: ANY DDL event), DISASSOCIATE STATISTICS, DROP, GRANT, NOAUDIT, RENAME, REVOKE, TRUNCATE

System-defined event attribute functions
ORA_CLIENT_IP_ADDRESS Returns the client IP address as a VARCHAR2.
DECLARE
 ip_address VARCHAR2(11);
BEGIN
 IF ora_sysevent = 'LOGON' THEN
    ip_address := ora_client_ip_address;
 END IF;
END;
ORA_DATABASE_NAME Returns the database name.
BEGIN
(...)
   db_name := ora_database_name;
(...)
END;
ORA_DES_ENCRYPTED_PASSWORD - Returns the DES-encrypted password as VARCHAR2.
- Equivalent to the value in the SYS.USER$ table PASSWORD col (11g).
- Passwds are no longer accessible in DBA_USERS or ALL_USERS
BEGIN
 IF ora_dict_obj_type = 'USER' THEN
   password := ora_des_encrypted_password;
 END IF;
END;
ORA_DICT_OBJ_NAME Returns the name of the object target of the DDL statement.
DECLARE
 database VARCHAR2(50);
BEGIN
 database := ora_obj_name;
END; 
ORA_DICT_OBJ_NAME_LIST The function returns the number of elements in the list as a PLS_INTEGER datatype. The name_list contains the list of object names touched by the triggering event.
DECLARE
  name_list DBMS_STANDARD.ORA_NAME_LIST_T;
  counter PLS_INTEGER;
BEGIN
  IF ora_sysevent = 'ASSOCIATE_STATISTICS' THEN
    counter := ora_dict_obj_name_list(name_list);
  END IF;
END;
ORA_DICT_OBJ_OWNER returns an owner of the object acted upon by the event.
database := ora_dict_obj_owner;
ORA_DICT_OBJ_OWNER_LIST formal parameter datatype is defined
in the DBMS_STANDARD package as ORA_NAME_LIST_T (table of varchar2(64)).
The function returns the number of elements in the list.
The owner_list contains the list of owners of objects
affected by the event.
DECLARE
  owner_list DBMS_STANDARD.ORA_NAME_LIST_T;
  counter PLS_INTEGER;
BEGIN
  IF ora_sysevent = 'ASSOCIATE_STATISTICS' THEN
     counter := ora_dict_obj_owner_list(owner_list);
  END IF;
END;
ORA_DICT_OBJ_TYPE
ORA_GRANTEE
ORA_INSTANCE_NUM
ORA_IS_ALTER_COLUMN
ORA_IS_CREATING_NESTED_TABLE
BEGIN
  IF ora_sysevent = 'CREATE' AND
    ora_dict_obj_type = 'TABLE' AND
    ora_is_creating_nested_table THEN
      INSERT INTO logging_table
      VALUES (ora_dict_obj_name||'.'||' 
            created with nested table.');
   END IF;
END;
ORA_IS_DROP_COLUMN
ORA_IS_SERVERERROR
ORA_LOGIN_USER>
ORA_PARTITION_POS returns the numeric position with the SQL text where you can insert a partition clause. This is only available in an INSTEAD OF CREATE trigger.
DECLARE
  sql_text ORA_NAME_LIST_T;
  sql_stmt VARCHAR2(32767);
  partition VARCHAR2(32767) := 'partitioning_clause';
BEGIN
  FOR i IN 1..ora_sql_txt(sql_text) LOOP
    sql_stmt := sql_stmt || sql_text(i);
  END LOOP;
  sql_stmt := SUBSTR(sql_text,1,ora_partition_pos – 1)
            ||' ' || partition||' '||
            SUBSTR(sql_test,ora_partition_pos);
  -- Add logic to prepend schema because 
  -- this runs under SYSTEM.
  sql_stmt := REPLACE(UPPER(sql_stmt),'CREATE TABLE '
            ,'CREATE TABLE '||ora_login_user||'.');
   EXECUTE IMMEDIATE sql_stmt;
END;
ORA_PRIVILEGE_LIST Returns list of privileges granted/revoked.
ORA_REVOKEE
ORA_SERVER_ERROR
ORA_SERVER_ERROR_DEPTH
ORA_SERVER_ERROR_MSG
ORA_SERVER_ERROR_NUM_PARAMS
ORA_SERVER_ERROR_PARAM
ORA_SQL_TXT Returns the substring of the processed SQL statement that triggered the event.
ORA_SYSEVENT
ORA_WITH_GRANT_OPTION
SPACE_ERROR_INFO

DDL Triggers: Syntax
CREATE [OR REPLACE] TRIGGER trigger_name
{BEFORE | AFTER | INSTEAD OF} ddl_event ON {DATABASE | SCHEMA}
[WHEN (logical_expression)]
[DECLARE]
 declaration_statements;
BEGIN
 execution_statements;
END [trigger_name];
/



DDL Trigger on Creation statements

(a) create table audit_creation
(b) create sequence audit_creation_s1
(c) Create trigger audit_creation
(d) create a synonym to fire the trigger
(e) check the inserted row on audit_creation


CREATE TABLE audit_creation
( audit_creation_id NUMBER PRIMARY KEY,
  audit_owner_name VARCHAR2(30) NOT NULL,
  audit_obj_name   VARCHAR2(30) NOT NULL,
  audit_date       DATE NOT NULL);

CREATE SEQUENCE audit_creation_s1;



CREATE OR REPLACE TRIGGER audit_creation
 BEFORE CREATE ON SCHEMA
BEGIN
  insert into audit_creation values
  (audit_creation_s1.nextval, 
   ORA_DICT_OBJ_OWNER,
   ORA_DICT_OBJ_NAME,
   sysdate);
END audit_creation;

SQL> Create synonym empsym for hr.employees;

synonym created.
SQL> Select * from audit_creation;

AUDIT_CREATION_ID    AUDIT_OWNER_NAME       AUDIT_OBJ_NAME      AUDIT_DATE
-------------------  ---------------------- ------------------- ---------------
1                    HR                     EMPSYN              22-AUG-10 













(ref) INSTEAD OF triggers

NON-UPDATABLE Views contain any:
  • One or more Table Joins.
  • A GROUP BY clause.
  • START WITH ...CONNECT BY.
  • DISTINCT.
  • Aggregate function (i.e. SUM, AVG, MIN, MAX, COUNT).
  • Set operators.
  • Derived columns (i.e. concatenated, pseudocolumns, etc).

INSTEAD OF triggers

INSTEAD OF triggers are created on
  • (usually) a non-updatable view
  • a nested table column of a non-updatable view

What?
  • INSTEAD OF trigger cannot be conditional.
  • Is the only way to update a view that is not inherently updatable
  • Is always a row-level trigger
  • Can read :OLD and :NEW, but cannot change them.
  • The timing events (BEFORE and AFTER) is not applicable for INSTEAD OF triggers.
  • INSTEAD OF trigger can fire for all three DML statements (I/U/D).
  • Prior to Oracle8.1.6 INSTEAD OF triggers functionality was included in the Enterprise Edition only.

Why use?
  • Used to intercept DML statements and replace them with alternative code.
  • Useful when application does not need (or should not) see the underlying tables (i.e. for security reasons). In this case:
    1. The application issues DMLs against the view
    2. The INSTEAD OF trigger intercepts and rewrite the DMLs to the underlying tables.
  • Useful also when the system uses object-relational functionality.
  • Updates against complex views may result in "ORA-01776-cannot modify more than one base table through a join view" or "ORA-01779-cannot modify a column which maps to a non key-preserved table"
How?
  • The database fires the INSTEAD OF trigger instead of running the triggering DML statement.
  • The trigger should: (a) determine what operation was intended and (b) perform the appropriate DML operations on the underlying tables.
Creation Syntax:
CREATE [OR REPLACE] TRIGGER trigger_name
 INSTEAD OF {dml_statement }
 ON {object_name | database | schema}
 FOR EACH ROW
 [DISABLE]
 [WHEN (logical_expression)]
 [DECLARE]
  declaration_statements;
BEGIN
  execution_statements;
END [trigger_name];
/
In the example below (taken from here) you:

(1) Create the base tables Custormer and Orders.
(2) Create a view order_info.
(3) Create a INSTEAD OF trigger order_info_insert that translates the DML statement into statements acting on the base tables (Orders and Customers).
(4) Insert data on the order_info view. Check the insertion on the base tables.

(1-2) Create base tables Customer, Orders and view.
SQL> CREATE TABLE customers ( 
     customer_id NUMBER (6)  NOT NULL  PRIMARY KEY, 
     cust_first_name VARCHAR2 (20)  NOT NULL , 
     cust_last_name VARCHAR2 (20)  NOT NULL , 
     cust_address VARCHAR2 (200) , 
     phone_numbers VARCHAR2 (12) , 
     nls_language VARCHAR2 (3) , 
     nls_territory VARCHAR2 (30) , 
     credit_limit NUMBER (9,2) , 
     cust_email VARCHAR2 (30) , 
     account_mgr_id NUMBER (6));

Table created.
SQL> CREATE TABLE orders ( 
     order_id NUMBER (12)  NOT NULL Primary key, 
     order_date TIMESTAMP  NOT NULL , 
     order_mode VARCHAR2 (8) , 
     customer_id NUMBER (6)  NOT NULL , 
     order_status NUMBER (2) , 
     order_total NUMBER (8,2) , 
     sales_rep_id NUMBER (6) , 
     promotion_id NUMBER (6));

Table created.
SQL> CREATE OR REPLACE VIEW order_info AS
   SELECT c.customer_id, c.cust_last_name, c.cust_first_name,
          o.order_id, o.order_date, o.order_status
   FROM customers c, orders o
   WHERE c.customer_id = o.customer_id;

View created.
SQL>

Now, if you try to perform an INSERT in the view order_info, it will fail with an ORA-01779 error:

SQL>INSERT INTO order_info 
    values (345, 'Rogers', 'John', 1250, sysdate, 1);

Error: ORA-01779: "cannot modify a column which maps to a non key-preserved table"
*Cause:    An attempt was made to insert or update columns of a join view which
           map to a non-key-preserved table.
*Action:   Modify the underlying base tables directly.


A Key preserved table means that the row from the base table will appear AT MOST ONCE in the output view on that table. Here one customer clearly may have multiple orders, so customer_id will most probably appear multiple times on order_info.

So, if you issue an INSERT against order_info, how can the database decide what to do with it?
Should it insert into orders AND customers? Only into orders? Or only into customers?

Well, you need to write an instead of trigger to intercept the DML against the trigger and issue DMLs against the base table(s) according to your business rules..

In this example, the trigger will split the INSERT coming from the application and insert on each table accordingly:


(3) Create a INSTEAD OF trigger order_info_insert that translates the DML statement into statements acting on the base tables (Orders and Customers).
SQL> CREATE OR REPLACE TRIGGER order_info_insert
  INSTEAD OF INSERT ON order_info
DECLARE
 duplicate_info EXCEPTION;
 PRAGMA EXCEPTION_INIT (duplicate_info, -00001);
BEGIN
 INSERT INTO Customers
   (customer_id, cust_last_name, cust_first_name)
 VALUES (:new.customer_id, :new.cust_last_name, :new.cust_first_name);
 INSERT INTO Orders (order_id, order_date, customer_id, order_status)
 VALUES (:new.order_id, :new.order_date, :new.customer_id, :new.order_status);
EXCEPTION
 WHEN duplicate_info THEN
   raise_application_error (
     num=> -20107,
     msg=> 'Duplicate customer or order ID');
END order_info_insert;
/
Trigger created.
SQL>

(4) Insert data on the order_info view. Check the insertion on the base tables.
SQL> INSERT into order_info 
 (Customer_id, cust_last_name, cust_first_name, order_id, order_date, order_status)
VALUES
 (1, 'Smith', 'John', 2500, '13-AUG-09', 0);
1 rows inserted.


SQL> Select Customer_id, cust_last_name, cust_first_name from customers;

CUSTOMER_ID  CUST_FIRST_NAME  CUST_LAST_NAME       
------------ ---------------- ----------------- 
           1 John             Smith 

SQL> Select order_id, order_date, order_status from orders;

ORDER_ID    ORDER_DATE                              ORDER_STATUS
---------- ---------------------------------------- ------------------------
     2500  13-AUG-09 12.00.00.000000 AM             0

SQL>

INSTEAD OF triggers on Nested Table Column of View


Nested Table Column of View (?)

You can create a view that has a nested table as one of its columns.
Now if you want to be able to update such a view, you need a instead of trigger.
It will intercept the object-relational INSERT and translate it to one that inserts into the base table.



(1) To create a view with a nested table as one of its columns you can:

(a) create the object type for the elements of the nested table. In this case, the type nte captures attributes of employees.

CREATE OR REPLACE TYPE nte AUTHID DEFINER IS
 OBJECT (
   emp_id   number(6),
   lastname varchar2(25),
   job      varchar2(10),
   sal      number(8,2));


(b) create the object table type. Here it will define a table of objects of the type nte.

CREATE OR REPLACE TYPE emp_list_ IS TABLE OF nte;

(c) Now you define the view containing the nested table in one of its columns (emplist).

CREATE OR REPLACE VIEW dept_view AS
  SELECT d.department_id,
  d.department_name,
  CAST (MULTISET (SELECT e.employee_id, e.last_name, e.job_id, i.salary
                  FROM employees e
                  WHERE e.department_id = d.department_id) 
                  AS emp_list_
                 ) emplist
  FROM departments d; 

(d) You can select the contents of emplist, and compare with a direct select of the base table:

SQL> select emplist from dept_view where department_id=10;

EMPLIST(EMP_ID, LASTNAME, JOB, SAL)
-----------------------------------------------
EMP_LIST_(NTE(200, 'Whalen', 'AD_ASST', 4400)


SQL> SELECT employee_id, last_name, job_id, salary
     FROM employees
     WHERE department_id = 10;

EMPLOYEE_ID LAST_NAME		      JOB_ID	     SALARY
----------- ------------------------- ---------- ----------
200 Whalen		      AD_ASST	       4400

(e) Now, if you try to insert directly into the dept_view, you'll receive an ORA-25015.

SQL> INSERT into table (
                        select d.emplist
                        from dept_view d
                        where department_id = 10
     ) 
     VALUES (1001, 'Glenn', 'AC_MGR', 10000);

SQL Error: ORA-25015: cannot perform DML on this nested table view column
*Cause:    DML cannot be performed on a nested table view column except through
an INSTEAD OF trigger
*Action:   Create an INSTEAD OF trigger over the nested table view column
and then perform the DML.

(f) So you need an INSTEAD OF trigger:

CREATE OR REPLACE TRIGGER dept_emplist_tr
 INSTEAD OF INSERT ON NESTED TABLE emplist OF dept_view
 REFERENCING NEW as Employee
             PARENT as Department
 FOR EACH ROW
BEGIN
  -- Intercept the insert into the nested table.
  -- Translate it to an insert into the base table.
  INSERT INTO employees (
              employee_id, last_name, email, hire_date, job_id, salary, department_id)
  VALUES (
     :Employee.emp_id,
     :Employee.lastname,
     :Employee..lastname || '@company.com', 
     SYSDATE,
     :Employee.job,
     :Employee.sal,
     :Department.department_id);
END;


(g) Now you if repeat the insert statement, the trigger will fire and perform the insert in the base table employees:

SQL> INSERT into table (
                        select d.emplist
                        from dept_view d
                        where department_id = 10
     ) 
     VALUES (1001, 'Glenn', 'AC_MGR', 10000);

1 rows inserted.

(h) You can check the results by querying the dept_view view and employees table:

SQL> select emplist from dept_view where department_id = 10;

EMPLIST(EMP_ID, LASTNAME, JOB, SAL)
--------------------------------------------------------------------------------
EMP_LIST_(NTE(200, 'Whalen', 'AD_ASST', 4400), NTE(1001, 'Glenn', 'AC_MGR', 10000))


SQL> SELECT employee_id, last_name, job_id, salary
     FROM employees
     WHERE department_id = 10;

EMPLOYEE_ID LAST_NAME		      JOB_ID	     SALARY
----------- ------------------------- ---------- ----------
	200 Whalen		      AD_ASST	       4400
       1001 Glenn		      AC_MGR	      10000

2 rows selected.



(ref) Trigger on Object Table

You can create triggers on object tables. In this case, the trigger can reference the pseudocolumn OBJECT_VALUE.
In the example below you:

(1) Create an OBJECT_TYPE to define a table.
(2) Create an OBJECT TABLE obj_employee
(3) Insert data on obj_employee
(4) Create a table emp_hist for logging updates to obj_employee
(5) Create a trigger emp_obj_trg.
  • The trigger will execute for each row of obj_employee affected by a DML statement.
  • The old and new values of the object emp in obj_employee will be written in emp_history.
  • The old and new values are :OLD.OBJECT_VALUE and :NEW.OBJECT_VALUE


(1-3) Create OBJECT_TYPE, OBJECT TABLE and Insert data.

SQL> CREATE or REPLACE TYPE EmployeeType AS OBJECT (
  Id             number,
  first_name     varchar2(15),
  last_name      varchar2(15),
  hire_date      date,
  salary         number(8,2)
  );

Type created.
SQL> CREATE TABLE obj_employee OF EmployeeType;

Table created.
SQL> INSERT INTO obj_employee VALUES (
    EmployeeType(1, 'John', 'Martic', '02-AUG-03', 95000))
1 row created.

SQL> INSERT INTO obj_employee VALUES (
    EmployeeType(2, 'Mary', 'Kempft', '02-JUL-99', 98000)))
1 row created.

SQL> INSERT INTO obj_employee VALUES (
    EmployeeType(3, 'Greg', 'Bloom', '02-AUG-09', 55000))
1 row created.

SQL> commit;
Commit complete.

SQL> select * from obj_employee;

    ID FIRST_NAME      LAST_NAME       HIRE_DATE    SALARY
------ --------------- --------------- ------------ ----------
     1 John            Martic          02-AUG-03         95000
     2 Mary            Kempft          02-JUL-99         98000
     3 Greg            Bloom           02-AUG-09         55000



(4) Create history table emp_hist:
SQL> CREATE TABLE emp_hist 
  (dt          date,
   username    varchar2(20),
   old_obj_emp EmployeeType,
   new_obj_emp EmployeeType);

table created.
SQL>

Compare emp_hist and obj_employee:

  • obj_employee is an object table. Each of its rows is an object of the type EmployeeType
  • emp_hist is a traditional table. Two of its columns are objects of the type EmployeeType
(?) How is the translation/mapping OO-Relational done?
SQL> desc obj_employee
 Name               Null?  Type
 ------------------ ------ ------------------
 ID                         NUMBER
 FIRST_NAME                 VARCHAR2(15)
 LAST_NAME                  VARCHAR2(15)
 HIRE_DATE                  DATE
 SALARY                     NUMBER(8,2)

SQL> desc emp_hist
 Name               Null?  Type
 ------------------ ------ ------------------
 DT                        DATE
 USERNAME                  VARCHAR2(20)
 OLD_OBJ_EMP               EMPLOYEETYPE
 NEW_OBJ_EMP               EMPLOYEETYPE


(5) Create a trigger emp_obj_trg.
  • The trigger will execute for each row of obj_employee affected by a DML statement.
  • The old and new values of the object emp in obj_employee will be written in emp_history.
  • The old and new values are :OLD.OBJECT_VALUE and :NEW.OBJECT_VALUE

SQL> CREATE OR REPLACE TRIGGER emp_obj_trg
 AFTER UPDATE on obj_employee
 FOR EACH ROW
BEGIN
 DBMS_OUTPUT.PUT_LINE('Logging Update');
 INSERT INTO emp_hist (dt, username, old_obj_emp, new_obj_emp)
 VALUES (sysdate, user, :OLD.OBJECT_VALUE, :NEW.OBJECT_VALUE);
END emp_obj_trg;

Trigger created.
SQL> 

SQL> set serveroutput on 
SQL> update obj_employee set salary=salary*1.10 where id=1;
Logging Update

1 row updated.

SQL> update obj_employee set salary=salary*.98;
Logging Update
Logging Update
Logging Update

3 rows updated.


(6) Check the audit table (using implicit cursor):

SQL> set serveroutput on
SQL> BEGIN
 for j in (select * from emp_hist) loop
  dbms_output.put_line (
   'Date: ' || j.dt ||'-- User '|| j.username);
   DBMS_OUTPUT.PUT_LINE(
   '(old rec)-- ID:'|| j.old_obj_emp.id || '-- OLD SALARY: '|| j.old_obj_emp.salary);
  DBMS_OUTPUT.PUT_LINE(
   '(new rec)-- ID:'|| j.new_obj_emp.id || '-- NEW SALARY: '|| j.new_obj_emp.salary);
 end loop;
end;
/

Date: 22-AUG-11-- User FHAT
(old rec)-- ID:1-- OLD SALARY: 95000
(new rec)-- ID:1-- NEW SALARY: 104500
Date: 22-AUG-11-- User FHAT
(old rec)-- ID:1-- OLD SALARY: 104500
(new rec)-- ID:1-- NEW SALARY: 102410
Date: 22-AUG-11-- User FHAT
(old rec)-- ID:2-- OLD SALARY: 98000
(new rec)-- ID:2-- NEW SALARY: 96040
Date: 22-AUG-11-- User FHAT
(old rec)-- ID:3-- OLD SALARY: 55000
(new rec)-- ID:3-- NEW SALARY: 53900

PL/SQL procedure successfully completed.
SQL>








(17) On Triggers



Introduction
  • Named PL/SQL unit. Can be enabled or disabled, but cannot be explicitly invoked.
  • Trigger is created on or defined on the item (to which it will be "attached"): table, view, schema or database.
  • Firing criteria is based on a triggering event (DML, DDL, System) and on a timing specification (before, after, instead of). A conditional clause (WHEN) may also be used to further specify the triggering rules.
  • Triggers do not accept arguments.
  • Triggers can be written in PL/SQL or JAVA.
  • Starting on Oracle 11g, triggers can now be created in the disabled state.

Triggers: what for?
Customization of database management; centralization of some business or validation rules; logging and audit.
  • Overcome the mutating-table error.
  • Maintain referential integrity between parent and child.
  • Generate calculated column values
  • Log events (connections, user actions, table updates, etc)
  • Gather statistics on table access
  • Modify table data when DML statements are issued against views
  • Enforce referential integrity when child and parent tables are on different nodes of a distributed database
  • Publish information about database events, user events, and SQL statements to subscribing applications
  • Enforce complex security authorizations: (i.e. prevent DML operations on a table after regular business hours)
  • Prevent invalid transactions
  • Enforce complex business or referential integrity rules that you cannot define with constraints
  • Control the behavior of DDL statements, as by altering, creating, or renaming objects
  • when they change data in a view>
  • Audit information of system access and behavior by creating transparent logs
(but, however, nonetheless, take heed:),
  • if the trigger code turns out to be very(?) long, you will more likely have better performance using a stored procedure instead. In fact, a trigger cannot be larger than 32Kb (because stored on LONG column). If you need to write something longer, use a stored procedure instead.
  • You can’t control the sequence of or synchronize calls to triggers, and this can present problems if you rely too heavily on triggers
  • A trigger can call a SQL statement that in turn fires another trigger: The number of cascading triggers is limited to 32, after which an exception is thrown. (11g and earlier)

Triggers: How much is too much?
  • DML statements on tables with DML Triggers are likely to have decreased perform.
  • You may choose to disable triggers before loading data. Of course the cost to this is the work you'll have to perform latter what the disabled triggers did not do.
  • If the task is complex, you may spread it across multiple triggers. However, this will make it maintenance more difficult, since it is likely to make the entire process harder to follow later.
  • Triggers may get disabled by accident: For example, DDLs on objects touched by a trigger may render it unusable. If you don't catch this, you may end up with missing/corrupt data.


Five types of Triggers
DDL Triggers
  • On CREATE, ALTER, DROP
  • Useful to control or monitor DDL statements.
  • An instead-of create table trigger allow for:
    • Ensuring that table creation meets development standards (i.e. proper storage or partitioning clauses.
    • Monitor poor programming practices (i.e. programs create and drop temporary tables rather than use Oracle collections. Temporary tables can fragmentdisk space and degrade database performance over time.
DML Triggers
  • Statement-level or row-level
  • Audit, check, save, and replace values before they are changed.
  • Automatic numbering of numeric primary keys (through row-level trigger).
Compound Triggers
  • Act as statement- and row-level triggers.
  • Lets you capture information at four timing points:
    (a) before the firing statement;
    (b) before each row change from the firing statement;
    (c) after each row change from the firing statement; and
    (d) after the firing statement.
  • Audit, check, save, and replace values before they are changed when you need to take action at both the statement and row event levels.
Instead-of Triggers
  • Enable you to stop performance of a DML statement and redirect the DML statement.
  • Often used to manage how you write to non-updatable views: They apply business rules and directly insert,update, or delete rows in tables that define updatable views.
  • Alternatively, they insert, update, or delete rows in tables unrelated to the view.
System Database event Triggers
  • Fire when a system activity occurs in the database (i.e. logon and logoff).
  • Useful for auditing information of system access. (You can track system events and map them to users).


Constraints
  • Apply to OLD and NEW data.
  • Easier to write and less error-prone.
Triggers
  • Apply only to NEW data.
  • Can enforce complex business rules.
  • Enforce ref integrity on distributed databases.


Using DML triggers

Using Compound triggers (11g only)

Using triggers on object tables
You can create triggers on object tables. In this case, the trigger can reference the pseudocolumn OBJECT_VALUE. Check an example here.

Using INSTEAD OF triggers

INSTEAD OF triggers are created on views. This allow DML statements to be issued against non-updatable views. Check an example here.

Privileges required to use Triggers

- CREATE TRIGGER: For your own objects.
- CREATE ANY TRIGGER + ALTER ANY TABLE.
- EXECUTE:to fire triggers on other schemas

Trigger Design Guidelines
  • Do not create triggers that duplicate database features. For example, do not create a trigger to reject invalid data if you can do the same with constraints.
  • Do not create triggers that depend on the order in which a SQL statement processes rows (which can vary).

    For example, do not assign a value to a global package variable in a row trigger if the current value of the variable depends on the row being processed by the row trigger. If a trigger updates global package variables, initialize those variables in a BEFORE statement trigger.
  • If the triggering statement of a BEFORE statement trigger is an UPDATE or DELETE statement that conflicts with an UPDATE statement that is running, then the database does a transparent ROLLBACK to SAVEPOINT and restarts the triggering statement. The database can do this many times before the triggering statement completes successfully. Each time the database restarts the triggering statement, the trigger fires. The ROLLBACK to SAVEPOINT does not undo changes to package variables that the trigger references. To detect this situation, include a counter variable in the package.
  • Do not create recursive triggers. The trigger fires recursively until it runs out of memory.
  • If you create a trigger that includes a statement that accesses a remote database, then put the exception handler for that statement in a stored subprogram and invoke the subprogram from the trigger.
  • Use DATABASE triggers judiciously. They fire every time any database user initiates a triggering event.
  • Only committed triggers fire. A trigger is committed, implicitly, after the CREATE TRIGGER statement that creates it succeeds.

Trigger Restrictions
  • Maximum Trigger Size
    • Max 32Kb. If needed, you can move code into functions, procedures or packages. In this case, the code could also be reused. stored modules can also be wrapped.
    • If the logic for your trigger requires much more than 60 lines of PL/SQL source text, then put most of the source text in a stored subprogram and invoke the subprogram from the trigger
  • DCL and DDL Restrictions
    • Only an autonomous trigger can run TCL or DDL statements
    • Nonsystem trigger bodies can’t contain DDL statements. They also can’t contain Transaction Control Language (TCL) commands, like ROLLBACK, SAVEPOINT,or COMMIT.
    • A trigger cannot invoke a subprogram that runs transaction control statements, because the subprogram runs in the context of the trigger body.
    • If you declare a trigger as autonomous, nonsystem trigger bodies can contain Data Control Language commands because they don’t alter the transaction scope.
    • To enable a trigger to work outside the scope of a triggering statement you use include in its DECLARE block: PRAGMA AUTONOMOUS_TRANSACTION;
    • A larger problem with SQL statements exists with remote transactions. If you call a remote
      schema-level function or procedure from a trigger body, it is possible that you may encounter a
      timestamp or signature mismatch. A mismatch invalidates the trigger and causes the triggering
      SQL statement to fail.
  • LONG and LONG RAW Datatypes
    • The LONG and LONG RAW datatypes are legacy components. Migrate out of them.
    • A trigger cannot declare a variable of the LONG or LONG RAW data type.
    • You may, however, insert into a LONG or LONG RAW column when the value can be converted CHAR or VARCHAR2.
    • Row-level triggers cannot use a :new,:old or parent with a LONG or LONG RAW column.
  • Triggers will fail if try to access a mutating table.
  • Oracle 11g has relaxed some mutating table restrictions

Triggers and data transfers
These utilities may fire triggers:
SQL*LOoader (sqlldr), Data Pump Import (impdp) and Original import (imp)
SQL*Loader (sqlldr):
- During a SQL*Loader conventional load, INSERT triggers fire.
- Before a SQL*Loader direct load, triggers are disabled.
Data Pump Import (impdp):
- If a table to be imported does not exist on the target database, or if you specify TABLE_EXISTS_ACTION=REPLACE, then impdp creates and loads the table before creating any triggers, so no triggers fire.
- If a table to be imported exists on the target database, and you specify either TABLE_EXISTS_ACTION=APPEND or TABLE_EXISTS_ACTION=TRUNCATE, then impdp loads rows into the existing table, and INSERT triggers created on the table fire.
Original Import (imp):
- If a table to be imported does not exist on the target database, then imp creates and loads the table before creating any triggers, so no triggers fire.
- If a table to be imported exists on the target database, then the Import IGNORE parameter determines whether triggers fire during import operations:
- If IGNORE=n (default), then imp does not change the table and no triggers fire.
- If IGNORE=y, then imp loads rows into the existing table, and INSERT triggers created on the table fire.