tags:

views:

4091

answers:

6

I have a transaction that contains multiple SQL Statements (INSERT, UPDATE and/or DELETES). When executing, I want to ignore Duplicate Error statements and continue onto the next statement. What's the best way of doing that?

Thanks in Advance.

+1  A: 

Keys must be unique. Don't do that. Redesign as needed.

(if you are trying to insert, then delete, and the insert fails... just do the delete first. Rollback on error in either statement).

SquareCog
No I don't think I made my question clear. INSERT INTO X VALUES(Y,Z)INSERT INTO X2 VALUES(Y2,Z2)INSERT INTO X3 VALUES(Y3,Z3)Let's say the second statement causes a duplicate error. I want to ignore it and continue execute the 3rd statement. The behavior in a Transaction is, it throws an error and it exists (Rolls back).
+3  A: 

If by "Ignore Duplicate Error statments", to abort the current statement and continue to the next statement without aborting the trnsaction then just put BEGIN TRY.. END TRY around each statement:

BEGIN TRY
    INSERT ...
END TRY
BEGIN CATCH /*required, but you dont have to do anything */ END CATCH
...
Unfortunately, we're using SQL Server 2000 right now so I don't think the Try .. Catch statements are available in that version.
+1  A: 

I think you are looking for the IGNORE_DUP_KEY option on your index. Have a look at http://sqlkit.com/2009/06/17/ for an explanation.

Canoehead
Bad idea - you do not want duplicate data in a unique index.
Jonathan Leffler
@Jonathan Leffler: `IGNORE_DUP_KEY` won't cause duplicate data. It causes SQL Server to ignore the duplicate key: not inserting it into the database.
Ian Boyd
@Ian Boyd: OK - living and learning is part of SO culture. Thanks for the information. (I'm not sure whether I could write reliable code if an INSERT succeeded without inserting what I asked it to insert and without telling me; presumably, there is a warning passed back, at least. However, I see what the feature does.)
Jonathan Leffler
@Jonathan Leffler: It's not your fault for mis-understanding the feature. It's is horribly documented: "*If you create a unique index, you can set this option to ensure each value in an indexed column is unique.*". For years i thought the option was the difference between an error being raised right when you insert, or delayed when you call COMMIT. Turns out it's neither of those. It is better to be named "Ignore, skip, and do not insert, duplicate rows"
Ian Boyd
+3  A: 

Expanding on your comment to SquareCog's reply, you could do:

INSERT INTO X VALUES(Y,Z)    WHERE Y  NOT IN (SELECT Y FROM X)
INSERT INTO X2 VALUES(Y2,Z2) WHERE Y2 NOT IN (SELECT Y FROM X2)
INSERT INTO X3 VALUES(Y3,Z3) WHERE Y3 NOT IN (SELECT Y FROM X3)

Here, I assume that column Y is present in all three tables. Note that performance will be poor if the tables are not indexed on Y.

Oh yeah, Y has a unique constraint on it--so they're indexed, and this should perform optimally.

Philip Kelley
I like this - our deployment scripts are designed to be run many times safely, so INSERT's always have criteria to avoid collisions.
n8wrl
+2  A: 

Although my emphatic advice to you is to structure your sql so as to not attempt duplicate inserts (Philip Kelley's snippet is probably what you need), I want to mention that an error on a statement doesn't necessarily cause a rollback.

Unless XACT_ABORT is ON, a transaction will not automatically rollback if an error is encountered unless it's severe enough to kill the connection. XACT_ABORT defaults to OFF.

For example, the following sql successfully inserts three values into the table:

create table x ( y int not null primary key )

begin transaction
insert into x(y)
values(1)
insert into x(y)
values(2)
insert into x(y)
values(2)
insert into x(y)
values(3)
commit

Unless you're setting XACT_ABORT, an error is being raised on the client and causing the rollback. If for some horrible reason you can't avoid inserting duplicates, you ought to be able to trap the error on the client and ignore it.

zinglon
+1  A: 

OK. After trying out some error handling, I figured out how to solve the issue I was having.

Here's an example of how to make this work (let me know if there's something I'm missing) :

SET XACT_ABORT OFF ; -- > really important to set that to OFF
BEGIN
DECLARE @Any_error int
DECLARE @SSQL varchar(4000)
BEGIN TRANSACTION
    INSERT INTO Table1(Value1) VALUES('Value1')
    SELECT @Any_error = @@ERROR
    IF @Any_error<> 0 AND @Any_error<>2627 GOTO ErrorHandler

    INSERT INTO Table1(Value1) VALUES('Value1')
    SELECT @Any_error = @@ERROR
    IF @Any_error<> 0 AND @Any_error<>2627 GOTO ErrorHandler

    INSERT INTO Table1(Value1) VALUES('Value2')
    SELECT @Any_error = @@ERROR
    IF @Any_error<> 0 AND @Any_error<>2627 GOTO ErrorHandler

    ErrorHandler: 
       IF @Any_error = 0 OR @Any_error=2627
       BEGIN 
           PRINT @ssql 
           COMMIT TRAN
       END
       ELSE 
       BEGIN 
           PRINT @ssql 
           ROLLBACK TRAN 
       END
END

As a result of the above Transaction, Table1 will have the following values Value1, Value2.

2627 is the error code for Duplicate Key by the way.

Thank you all for the prompt reply and helpful suggestions.