Skip to main content

Violation of PRIMARY KEY constraint in Log Shipping

·         I have SQL Server 2005 log shipping setup with primary/secondary configuration.  I can confirm from the logs that log shipping is working without issue, however, reports generated from the monitor server show this message:
Violation of PRIMARY KEY constraint 'PK__#log_shipping_mo__3ABBDC91'. Cannot insert duplicate key in object 'dbo.#log_shipping_monitor'.  The statement has been terminated.
There is nothing special about the configuration.  Any ideas?
·         I think the problem you are seeing is related to some old information being present in the tables used to store log shipping configuration.  There are some scenarios where this can happen and it causes the problem you reported in your first post.  We are working on correcting this in a future release.
 As you can tell from the error, the problem is caused by an insert to a temp table causing a PK constraint violation.  The PK for the temp table is server name and database name.  This error is normally caused by old configuration being present in the tables log_shipping_monitor_primary and/or log_shipping_monitor_secondary.  You can view the contents of these tables directly (in msdb) or use some supplied help SP's (see BOL topic titled "Log Shipping Tables and Stored Procedures").
 The workaround is to remove the old rows from log_shipping_monitor_primary and/or log_shipping_monitor_secondary tables.  Can you determine that you do indeed have stale data in the log shipping tables.  If this is the case I can work with you on how to remove the old rows.
 The old configuration data is probably causing the incorrect alerts you are seeing.

Problem Description
===============
·         After role reversal in a log shipping setup, the stored procedure sp_help_log_shipping_monitor may return the following error OR we may find the error in the log shipping report in SQL Server Management Studio.
Msg 2627, Level 14, State 1, Procedure sp_help_log_shipping_monitor, Line 148
Violation of PRIMARY KEY constraint 'PK__#log_shipping_mo__15502E78'. Cannot insert duplicate key in object 'dbo.#log_shipping_monitor'.
The statement has been terminated.

·         The stored procedure sp_help_log_shipping_monitor fetches data from log_shipping_monitor_primary and log_shipping_monitor_secondary tables and puts them into a temp table called #log_shipping_monitor. This temp table has a primary key with the ServerName and the DatabaseName.
log_shipping_monitor_primary - Stores one monitor record per primary database in each log shipping configuration
log_shipping_monitor_secondary - Stores one monitor record per secondary database in a log shipping configuration

·         The problem may happen under few scenarios such as:
1.       A manual failover was done for the log shipped database. Since, the roles are not switched on the monitor server, there would be two entries in the Log_shipping_monitor_secondary table. This would cause the stored procedure sp_help_log_shipping_monitor to fail.

2.       There could be incorrect metadata cleanup of a previous log shipping setup for the same log shipped database.

Resolution
========
Note: Any incorrect update/deletion in the metadata table may lead to inconsistencies in the logshipping setup,

Let us say :
Logshipping Database name : TESTDB
Primary Server name : Server\Primary
Secondary Server name : Server\Secondary

Check the entries under –

1) select primary_server, primary_database from db.dbo.log_shipping_monitor_primary  -

Output :
primary_server     primary_database
------------------ ----------------
Server\Primary         TestDB

Primary server should have an entry corresponding to its primary database. It should show Primary_server as Server\Primary and Primary_database as TESTDB

2) select secondary_server,secondary_database,primary_server, primary_database from msdb.dbo.log_shipping_monitor_secondary –

 Output:
secondary_server     secondary_database  primary_server   primary_database
-------------------- ------------------- ---------------- -----------------
Server\Secondary                TestDB            Server\Primary    TestDB

Secondary server should have a correct entry corresponding to it’s primary server and primary database.
It should show Secondary_server as Server\Secondary, secondary_database as TESTDB, primary_server as Server\Primary, Primary_database as TESTDB

Now if you find any mismatch in the above output, you need to update/delete the incorrect entries manually using Delete or Update command depending on the scenario.

For example –

Scenario 1 :
select secondary_server,secondary_database,primary_server, primary_database from msdb.dbo.log_shipping_monitor_secondary –

Output:
secondary_server     secondary_database  primary_server   primary_database
-------------------- ------------------- ---------------- -----------------
Server\Secondary                TestDB            Server\other_server    TestDB

The output here shows incorrect server under Primary_Server column, To resolve this update the msdb.dbo.log_shipping_monitor_secondary manually as -

Update  msdb.dbo.log_shipping_monitor_secondary Set Primary_server = ‘Server\Primary’ where Primary_server = ‘Server\Other_server’


Scenario 2 :
select secondary_server,secondary_database,primary_server, primary_database from msdb.dbo.log_shipping_monitor_secondary –

Output:
secondary_server     secondary_database  primary_server   primary_database
-------------------- ------------------- ---------------- -----------------
Server\Secondary                TestDB            Server\Primary          TestDB
Server\Secondary                TestDB            Server\other_server     TestDB

In this scenario, It shows uncleaned metadata which might be from previous Log shipping setup so delete the entry from msdb.dbo.log_shipping_monitor_secondary as –

Delete from  msdb.dbo.log_shipping_monitor_secondary where Primary_server = ‘Server\Other_server’

Note: “There was a design change that was done on SQL Server 2008 to prevent this issue from occurring.” 

Comments

Popular posts from this blog

Digging into a long running job

Revisiting the blogger to finally start again blogging about the SQL Server after a long long time.  This post is on analyzing and digging into what is causing a job significant time to complete. This will be a basic level of approach and I wish, I could have done better. But, its was enough in my case to get to a conclusion, with the knowledge I equip.   Issue : A job that's scheduled to run every three hours, was running for more than 48hours. Average job run time, 1.5hrs.  Analysis : First and foremost thing to do, is to find out on what session id is the job running on the instance. You may use the below query to fetch the session id of the job from the job name. declare @Var1 varchar(36) select @Var1=substring(REPLACE(CAST(job_id AS VARCHAR(36)),'-',''),17,16) from sysjobs where name =' JOB_NAME ' select * from sys.sysprocesses where program_name like '%'+@Var1+'%' You may get a single row or multiple rows with ...

Deferred prepare could not be completed

This is usually an error that is encountered while querying a remote database on a different server instance which says OLEDB provider "SQLNCLI" for linked server "ABCD" returned message "Deffered prepare could not be completed" Msg 8180, Level 16, State 1, Line 1 Statement(s) could not be prepared. This is the way I could rectify, though this is not the lone mistake that gives this error every time. Just set the default to the database that is being querying to the user through which the Linked server is connecting to the database.

Incorrect SET Options with Filtered Indexes

Rebuild Index|Create Index|DML on a Table with Filtered Index fails with the error: The following SET options have incorrect settings: 'QUOTED_IDENTIFIER'. Verify that SET options are correct for use with indexed views and/or indexes on computed columns and/or filtered indexes and/or query notifications and/or XML data type methods and/or spatial index operations. [SQLSTATE 42000] (Error 1934).  The step failed. The Operation above might succeed when its through SSMS. But, would fail when the operation is from a SQL Agent Job. This is because, SQL Agent by default, do not have the SET Options enabled (ON). They are OFF. Reason: The following are the SET Options that are needed while operation on a table with a filtered index. Required SET Options for Filtered Indexes The SET options in the Required Value column are required whenever any of the following conditions occur: Create a filtered index. INSERT, UPDATE, DELETE, or MERGE operation modifies the data in a...