![]() |
ALLBASE/SQL Reference Manual
> Chapter 4 Constraints, Procedures, and RulesUsing Rules |
|||||||||||||||||||||||
|
Understanding RulesRules allow you to define generalized constraints by invoking procedures whenever specified operations are performed on a table. The rule fires, that is, invokes a procedure, each time the specified operation (such as INSERT, UPDATE, or DELETE) is performed and the rule's search condition is satisfied. Rules tie procedures to particular kinds of data manipulation statements on a table. This permits data processing to be carried out by the DBEnvironment itself. The effect is less application coding and more efficient use of resources. This is especially important for networked systems. Rules will fire under the following conditions:
Creating RulesA rule is defined in a CREATE RULE statement, which identifies a table, types of data manipulation statements, a firing condition, and a procedure to be executed whenever the condition evaluates to TRUE and the data manipulation statement is of the right type. The following is a simple example of a rule tied to deletions from the Parts table: CREATE RULE PurchDB.RemovePart AFTER DELETE FROM PurchDB.Parts WHERE SUBSTRING(PartNumber,1,4) < > 'XXXX' EXECUTE PROCEDURE PurchDB.ListDeletes (OLD.PartNumber);The table on which the rule is defined is PurchDB.Parts. The statement type required to trigger the procedure is the DELETE operation. The search condition that must be satisfied in addition to the statement type of DELETE is that the first four characters in PartNumber must not be "XXXX." The procedure to be executed is PurchDB.ListDeletes, shown in the following: CREATE PROCEDURE PurchDB.ListDeletes (PartNumber CHAR(16) NOT NULL) AS BEGIN INSERT INTO PurchDB.Deletions VALUES (:PartNumber, CURRENT_DATETIME); END;When a row containing a part number that does not start with XXXX is deleted from the Parts table, its number is inserted along with the current date and time, in the PurchDB.Deletions table. Techniques for Using Procedures with RulesOne common use of the rule-and-procedure combination is to enforce integrity within a DBEnvironment. This can be done in different ways, depending on your needs. The following sections contrast two approaches to integrity enforcement:
Using a Chained Set of Procedures and RulesThe following example uses a chained set of procedures and rules to remove all references to a part number once it has been deleted from the database. In this case a rule fires a procedure, which causes another delete, which causes another rule to invoke an additional procedure, and so on.CREATE PROCEDURE PurchDB.RemovePart (PartNum CHAR(16) NOT NULL) AS BEGIN DELETE FROM PurchDB.Inventory WHERE PartNumber = :PartNum; DELETE FROM PurchDB.SupplyPrice WHERE PartNumber = :PartNum; END; CREATE RULE PurchDB.RemovePart AFTER DELETE FROM PurchDB.Parts EXECUTE PROCEDURE PurchDB.RemovePart (OLD.PartNumber); CREATE PROCEDURE PurchDB.RemoveVendPart (VendPartNum CHAR(16) NOT NULL) AS BEGIN DELETE FROM PurchDB.OrderItems WHERE VendPartNumber = :VendPartNum; DELETE FROM ManufDB.SupplyBatches WHERE VendPartNumber = :VendPartNum; END; CREATE RULE PurchDB.RemoveVendPart AFTER DELETE FROM PurchDB.SupplyPrice EXECUTE PROCEDURE PurchDB.RemoveVendPart (OLD.VendPartNumber); CREATE PROCEDURE ManufDB.RemoveBatchStamp (BatchStamp DATETIME NOT NULL) AS BEGIN DELETE FROM ManufDB.TestData WHERE BatchStamp = :BatchStamp; END; CREATE RULE ManufDB.RemoveBatchStamp AFTER DELETE FROM ManufDB.SupplyBatches EXECUTE PROCEDURE ManufDB.RemoveBatchStamp (OLD.BatchStamp); Executing the Chained Set of Procedures and RulesWhenever a user performs a DELETE operation on PurchDB.Parts, the procedures and rules are executed on each row of each table for the identified part number in the following order:
Using a Single Procedure with CursorsThe following example uses a single rule and one procedure to remove all references to a part number once it has been deleted from the database. In this case, a single procedure RemovePart determines which rows need to be deleted in the other tables once a part number is deleted from the Parts table. Since this method only uses one rule and one procedure, it would be effective only when a DELETE is done from the Parts table. Deletions of part numbers from other tables would not trigger any rules at all. The single procedure uses two cursors to scan the PurchDB.SupplyPrice and ManufDB.SupplyBatches tables for entries that correspond to a deleted part number. The procedure then performs deletions of qualifying rows in PurchDB.OrderItems and ManufDB.TestData.CREATE PROCEDURE PurchDB.RemovePart(PartNum CHAR(16) NOT NULL) AS BEGIN DECLARE VendPartNum CHAR(16) NOT NULL; DECLARE BatchStamp DATETIME NOT NULL; DECLARE SupplyCursor CURSOR FOR SELECT VendPartNumber FROM PurchDB.SupplyPrice WHERE PartNumber = :PartNum; DECLARE BatchCursor CURSOR FOR SELECT BatchStamp FROM ManufDB.SupplyBatches WHERE VendPartNumber = :VendPartNum; DELETE FROM PurchDB.Inventory WHERE PartNumber = :PartNum;Open the first cursor: OPEN SupplyCursor; FETCH SupplyCursor INTO :VendPartNum; WHILE ::sqlerrd2 = 1 DO DELETE FROM PurchDB.OrderItems WHERE VendPartNumber = :VendPartNum;Open the second cursor: OPEN BatchCursor; FETCH BatchCursor INTO :BatchStamp; WHILE ::sqlerrd2 = 1 DO DELETE FROM ManufDB.TestData WHERE BatchStamp = :BatchStamp; FETCH BatchCursor INTO :BatchStamp; ENDWHILE; CLOSE BatchCursor; DELETE FROM ManufDB.SupplyBatches WHERE VendPartNumber = :VendPartNum; FETCH SupplyCursor INTO :VendPartNum; ENDWHILE; CLOSE SupplyCursor; DELETE FROM PurchDB.SupplyPrice WHERE PartNumber = :PartNum; END;The single rule that invokes the above procedure is as follows: CREATE RULE PurchDB.RemovePart AFTER DELETE FROM PurchDB.Parts EXECUTE PROCEDURE PurchDB.RemovePart (OLD.PartNumber); Error Handling in Procedures Invoked by RulesWhen invoked by a rule, a procedure is executed inside the execution of a data manipulation statement. Therefore, if the procedure encounters an error, the effect of the procedure and the effect of the data manipulation statement as a whole are undone. Statements that may fire rules always execute with statement atomicity, regardless of the current general error checking level set by the SET DML ATOMICITY statement. Inside procedures invoked by rules, SQL errors have the usual effect of issuing messages, halting execution of the current statement, rolling back a transaction, or ending a connection. In addition, even if the error does not result in rolling back a transaction or losing a connection, it results in the undoing of the effects of all procedures invoked in a chain by the current statement, and it results in the undoing of the effects of all rules triggered by the current statement. Thus the entire execution of the statement is undone. Using RAISE ERROR in Procedures Invoked by RulesWithin a procedure which is triggered by a rule, the RAISE ERROR statement can be used to generate an error, which causes an immediate return and undoes the statement that triggered the rule. The text of the RAISE ERROR message can provide useful information to the user such as the procedure name, the exact reason for the error, the location in the procedure, or the name of the rule that invoked the procedure (if the procedure is only fired by one rule). Suppose the following rule executes whenever a user attempts to delete a row in the Vendors table: CREATE RULE PurchDB.CheckVendor AFTER DELETE FROM PurchDB.Vendors EXECUTE PROCEDURE PurchDB.DelVendor (OLD.VendorNumber);The procedure PurchDB.DelVendor checks for the existence of the use of a vendor number elsewhere in the database, and if it finds that the number is being used, it rolls back the delete on the Vendors table. The procedure is coded as follows: CREATE PROCEDURE PurchDB.DelVendor (VendorNumber INTEGER NOT NULL) AS BEGIN DECLARE rows INTEGER NOT NULL; SELECT COUNT(*) INTO :rows FROM PurchDB.Orders WHERE VendorNumber = :VendorNumber; IF :rows <> 0 THEN RAISE ERROR 1 MESSAGE 'Vendor number exists in the "Orders" table.'; ENDIF; SELECT COUNT(*) INTO :rows FROM PurchDB.SupplyPrice WHERE VendorNumber = :VendorNumber; IF :rows <> 0 THEN RAISE ERROR 1 MESSAGE 'Vendor number exists in "SupplyPrice" table.'; ENDIF; END;PurchDB.DelVendor checks for the existence of the use of a vendor number in two tables: PurchDB.Orders and PurchDB.SupplyPrice. If it retrieves any rows containing the vendor number, it returns an error code and a string of text to the caller by means of the RAISE ERROR statement. The following shows the effect of the rule and procedure when you attempt to delete a row from the Vendors table in ISQL: isql=> DELETE FROM purchdb.vendors WHERE vendornumber = 9006; Vendor number exists in the "Orders" table. Error occurred executing procedure PURCHDB.DELVENDOR statement 3. (DBERR 2235) INSERT/UPDATE/DELETE statement had no effect due to execution errors. (DBERR 2292) Number of rows processed is 0 isql=>The DELETE statement triggers the rule, which executes the procedure PurchDB.DelVendor. If the vendor number that is to be deleted is not found in either of the two tables, sqlcode is 0, and no messages are displayed. When a procedure is called through the use of a rule, the procedure exits as soon as an error occurs. This can be either an ordinary SQL error (but not a warning), or a user-defined error produced with the RAISE ERROR statement. After an error return, the statement that fired the rule is undone, and the operation of all other rules fired by the statement is also undone. In application programs, you use SQLEXPLAIN to retrieve the messages generated by RAISE ERROR and other SQL statements. Enabling and Disabling RulesRule processing takes place by default in the DBEnvironment. However, the DBA can use the following statement to disable the operation of rules in the current session: isql=> disable rules;This statement, which is useful in debugging, should be employed only with great care, since it can affect the integrity of the database, if rules are being used to control data integrity. To restore the operation of rules in the session, use the following statement: isql=> enable rules;Rules are not fired retroactively when the ENABLE RULES statement is issued after the DISABLE RULES statement has been issued. Special Considerations for Procedures Invoked by RulesProcedures operate somewhat differently when invoked by rules than when invoked directly by a user. The differences are most pronounced in several areas:
Transaction Handling in RulesSince rules are fired by data manipulation statements that are already being executed, a transaction is always active when a rule invokes a procedure. Therefore, BEGIN WORK and BEGIN ARCHIVE statements will result in errors in a procedure invoked by a rule. The error will cause the rule to fail and the user's statement to be undone. COMMIT WORK, COMMIT ARCHIVE, ROLLBACK WORK, ROLLBACK ARCHIVE, SAVEPOINT, and ROLLBACK TO SAVEPOINT statements will generate errors when encountered in procedures triggered by rules. The error causes the user's statement and all subsequent rule-driven statements to be undone. If you wish to include COMMIT WORK, COMMIT ARCHIVE, ROLLBACK WORK, ROLLBACK ARCHIVE, SAVEPOINT, or ROLLBACK TO SAVEPOINT statements in the procedure, because the procedure will be executed by users directly as well as by rules, you should include these statements within a condition that will only be true for non-rule invocation. To do this, add a flag parameter to the procedure. Have users invoking the procedure pass in a fixed value (such as 0), and have rules invoking the procedure pass in a different value (such as 1). Then the procedure can be coded with IF statements like the following:if :Flag = 0 then commit work; endif;The flag check ensures that the rule will not execute statements that would cause it to generate an error when the procedure is invoked by a rule, while user calls can commit or roll back changes automatically. Effects of Rule ChainingProcedures invoked by rules can include data manipulation statements that invoke rules that trigger the execution of other procedures. Excessive chaining of rules in this fashion uses additional system resources. When the chain length exceeds 20, an error occurs, which causes the user's statement to be undone. To avoid problems, be sure to trace the dependencies of statements within procedures invoked by rules so as to:
Invalidation of SectionsProcedures can include data definition statements that affect the execution of procedures and rules by invalidating sections. Use care when issuing the following statements inside procedures:
Changing Session AttributesProcedures should avoid the following statements, which change the attributes of transactions or sessions:
Performance ConsiderationsThe placement of conditions on execution of statements within the firing of a rule should be examined carefully. Firing conditions placed in the WHERE clause can avoid the overhead of loading and invoking the procedure, since the WHERE condition is checked before the procedure is invoked. Thus, it might be better to develop several rules with separate conditions and procedures with well-defined actions rather than a single rule with no condition and a single procedure that makes checks before deciding what steps to carry out. To determine the best design for your needs, weigh the overhead of frequent loading and executing of a procedure against the overhead of maintaining several procedures and rules.Differences between Rules and Integrity ConstraintsRules are similar to integrity constraints in that when a rule is created, all existing INSERT, UPDATE, and DELETE statements will be affected by the rule (if the statement type is appropriate to the rule). Rules are viewed as changes to the table definition, and so all existing sections depending on the table are invalidated when a rule is created. When these sections are next revalidated, the rule definition is picked up and compared to the section; appropriate rules are then included in the revalidated section for checking at statement execution time. The following are some of the most important ways in which rules differ from integrity constraints:
|