on 2012 Jun 11 2:39 PM
I am stuggling to create a trigger that drops a constraint using ALTER TABLE I have read that this command cannot be used within a trigger so have included it as a string. All the code has been tested outside of a trigger and works correctly, any help would be appreciated.
The error I am currently getting is 'commit rollback not alowed within atomic operation'
CREATE TRIGGER updates_equipment_type BEFORE INSERT ON trained_on REFERENCING NEW AS new_trained_on FOR EACH ROW WHEN (new_trained_on.type NOT IN (SELECT type FROM equipment_type)) BEGIN EXECUTE IMMEDIATE 'ALTER TABLE trained_on DROP CONSTRAINT relationship_fixed_by ALTER TABLE equipment_type DROP CONSTRAINT mandatory_participation_in_fixed_by'; INSERT INTO equipment_type VALUES (new_trained_on.type); EXECUTE IMMEDIATE 'ALTER TABLE trained_on ADD CONSTRAINT relationship_fixed_by FOREIGN KEY (type) REFERENCES equipment_type ALTER TABLE equipment_type ADD CONSTRAINT mandatory_participation_in_fixed_by CHECK (type IN ( SELECT type FROM trained_on))'; END
Try this...
Change the trigger to AFTER.
Delete the EXECUTE IMMEDIATE ... ALTER TABLE statements, so only the INSERT equipment_type remains inside the trigger.
Use SET TEMPORARY WAIT_FOR_COMMIT = 'ON' before your INSERT trained_on statements.
Don't code INSERT equipment_type outside the trigger; let the trigger do it.
CREATE TABLE equipment_type ( type INTEGER NOT NULL PRIMARY KEY, CONSTRAINT mandatory_participation_in_fixed_by CHECK ( type IN ( SELECT type FROM trained_on ) ) ); CREATE TABLE trained_on ( id INTEGER NOT NULL PRIMARY KEY, type INTEGER NOT NULL, CONSTRAINT relationship_fixed_by FOREIGN KEY ( type ) REFERENCES equipment_type ); CREATE TRIGGER updates_equipment_type AFTER INSERT ON trained_on REFERENCING NEW AS new_trained_on FOR EACH ROW WHEN ( new_trained_on.type NOT IN ( SELECT type FROM equipment_type ) ) BEGIN INSERT INTO equipment_type VALUES ( new_trained_on.type ); END; SET TEMPORARY OPTION WAIT_FOR_COMMIT = 'ON'; INSERT trained_on VALUES ( 1, 100 ); COMMIT; INSERT trained_on VALUES ( 2, 100 ); COMMIT; INSERT trained_on VALUES ( 3, 200 ); COMMIT; INSERT trained_on VALUES ( 4, 200 ); COMMIT; SET TEMPORARY OPTION WAIT_FOR_COMMIT = 'OFF'; SELECT * FROM equipment_type; SELECT * FROM trained_on; type 100 200 id,type 1,100 2,100 3,200 4,200
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
"Wow - surely better" than all other replies so far!
And it's even real code:)
BTW: Instead of the "wait_on_commit" option, one might also declare the FK with the CHECK ON COMMIT clause, in case the "automatic type addition" should work always and not only when that particular option is set:
... FOREIGN KEY ( type ) REFERENCES equipment_type CHECK ON COMMIT)
And Breck has summed up the whole story of this nice thread in his blog:
Back up, start over, find another way
An "act of penance" worth reading, methinks:)
Not being able to ALTER a table is not just a syntax issue, it is a semantic one: you cannot execute a COMMIT from within a trigger execution, and an ALTER implies a COMMIT. So, the EXECUTE IMMEDIATE does not help, it's still "within the trigger execution".
Do not confuse "TRIGGER EVENT" with CREATE TRIGGER, they don't have anything to do with one another.
The CREATE EVENT needs a name, but no schedule or condition, because you are going to use an explicit TRIGGER EVENT.
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
Thanks very much for taking the time to write such detailed replies, It has been a great help.
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
In case any answer is preferrable from your point of, feel free to accept that one - cf. this little HOWTO...
Based on Brecks great example here a solution with a Instead of trigger.
INSTEAD OF triggers differ from BEFORE and AFTER triggers because when an INSTEAD OF trigger fires, the triggering action is skipped and the specified action is performed instead.
CREATE TABLE equipment_type ( type INTEGER NOT NULL PRIMARY KEY); CREATE TABLE trained_on ( id INTEGER NOT NULL PRIMARY KEY, type INTEGER NOT NULL, CONSTRAINT relationship_fixed_by FOREIGN KEY ( type ) REFERENCES equipment_type ); CREATE TRIGGER updates_equipment_type INSTEAD OF INSERT ON trained_on REFERENCING NEW AS new_trained_on FOR EACH ROW BEGIN INSERT INTO equipment_type on existing skip VALUES ( new_trained_on.type ); INSERT INTO trained_on VALUES ( new_trained_on.id, new_trained_on.type ); END; INSERT trained_on VALUES ( 1, 100 ); COMMIT; INSERT trained_on VALUES ( 2, 100 ); COMMIT; INSERT trained_on VALUES ( 3, 200 ); COMMIT; INSERT trained_on VALUES ( 4, 200 ); COMMIT; SELECT * FROM equipment_type; SELECT * FROM trained_on; type 100 200 id,type 1,100 2,100 3,200 4,200
I have left out the Constraint mandatory_participation_in_fixed_by on the equipment type table is not possible with this solution.
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
An interesting approach - I surely stumbled on the recursive part (the instead of trigger on trained_on does itself insert into that table), but the docs truly allow that (i.e. prevent a recursive calling) for base tables:
Whether an INSTEAD OF trigger performs recursion depends on whether the target of the trigger is a base table or a view. Recursion occurs for views, but not for base tables. That is, if an INSTEAD OF trigger performs DML operations on the base table on which the trigger is defined, those operations do not cause triggers to fire (including BEFORE or AFTER triggers). If the target is a view, all triggers fire for the operations performed on the view.
While I fully follow Breck's advice in general, in the current case, this won't work as expected IMHO:
You seem to want to disable a CHECK constraint and a FOREIGN KEY constraint for the current table while entering "violating" data, so the desired solution to
requires correct timing, which won't be guaranteed with the asynchronous EVENT solution.
There are possibly different approaches if you need to disable any logic during particular DML statements on a regular basis - which would not require to ALTER TABLE in between (which will not scale very well, anyway, in case that might matter...).
E.g. you might use a particular configuration table that can be used to disable checks for particular users or connections. Cf. this FAQ for a similar question on disabling trigger actions...
We could surely give some sample code if you can elaborate what exactly you're trying to achieve. Ideas would include delayed foreign key checks with the "wait_for_commit" option (to add the missing parent entry "on demand") or a check including a CREATE VARIABLE to disable the check temporarily...
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
OK, but would that not conflict with the INSERT statement itself?
Re-reading the question, I would think that
would be much simpler - but that's just my understanding of the question...
First of all, wait_for_commit doesn't affect the CHECK constraint that is dropped and added... if the CHECK is going to fail, it will still fail.
Second of all, dropping and recreating a foreign key around a single INSERT seems pointless... if the INSERT causes a violation, the ALTER TABLE ADD foreign key will raise it. Generally, dropping and recreating foreign keys, or wait_for_commit, is ONLY useful if you have two or more operations... one which violates, and another which repairs the violation. That is not the case here, which implies the dropping and recreating the foreign key is pointless.
I suspect more work needs to be done here... BUT, please tell me how moving all the code into an event would cause a conflict (or at least, a conflict that isn't already there 🙂
...by "all code" I mean the four ALTERs and one INSERT.
Yes, "wait_for_commit" would not resolve the CHECK constraint issue - therefore I've hinted at a "check including a variable to disable the check temporarily...".
The conflict might arise between the "INSERT trained_on ..." and the event code when triggered :
Obviously we're dealing with an "INSERT trained_on ... " that tries to insert an equipment_type which does not already exist in the latter table.
In that case the according FK constraint will fail, unless the FK is already dropped from within the event before the INSERT statement is finished - or, in case "wait_for_commit" is set, before it is committed. That's the dangling timing issue I see. - In other words: I would think the event code must be run before the INSERT is finished, and that would not be guaranteed simply because of the asynchronous nature of event connections.
Correction: IMHO the ALTER TABLE cannot succeed timely, as the following situation clearly would be satisfied:
"ALTER TABLE is prevented whenever the statement affects a table that is currently being used by another connection".
User | Count |
---|---|
66 | |
11 | |
10 | |
10 | |
9 | |
7 | |
7 | |
6 | |
5 | |
4 |
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.