Skip to main content

Log Shipping in SQL Server Express 2008

When you entrust your Applications and Data to the Cloud Based Service Providers such as GoGrid and Amazon EC2 it becomes absolutely critical to employ a strict Disaster Recovery strategy.

As part of of our strategy it was imperative that we have a failover for our SQL Server. The only problem was that SQL Server Express 2008 doesn’t support Log Shipping.
Well technically it does but SQL Server Express does not have SQL Server Agent.

Now I’m a big fan of SQL Server, but when it comes to doing something a bit more complicated the Express editions limitations become a bit of a problem, that’s why it’s time to find creative solutions.

To set up Log Shipping the first step is to Backup your database and then Restore it on another instance making sure set the Recovery state as “RESTORE WITH STANDBY”. See below

restore

To get this going you’ll need three stored procedures, one in the Master database and two in the slave database.

Now when you are trying to do this across Networks and Hosting Providers you will need a reliable way of transferring the Logs from the Master server to the Slave. For this we use Windows Live Sync, but that’s a whole other blog post.

Here are the stored procedures you need to get this going.
NB: These only work in SQL Server 2008 and our not backward compatible.

BackupLog

All you do here is specify the path for the log and the name of the database you want the transaction log for.

USE [master] GO CREATE PROCEDURE [dbo].[BackupLog] @LogPath nvarchar(500), @DatabaseName nvarchar(100) AS BEGIN -- SET NOCOUNT ON added to prevent extra result sets from -- interfering with SELECT statements. SET NOCOUNT ON; BACKUP LOG @DatabaseName to disk=@LogPath with NOINIT, NOSKIP, NOFORMAT END GO

RestoreLog

All you do here is specify the path for the log and the name of the database you want to restore the transaction log for.

CREATE PROCEDURE [dbo].[RestoreLog] @LogPath nvarchar(500), @DatabaseName nvarchar(100) AS BEGIN SET NOCOUNT ON; DECLARE @StandbyFile nvarchar(1000); SET @StandbyFile = 'C:\Program Files\Microsoft SQL Server\MSSQL10.SQL2008\MSSQL\Backup\' + @DatabaseName + '.undo'; DECLARE @spid as varchar(10) DECLARE @CMD as varchar(1000) DECLARE cur CURSOR FOR SELECT spid FROM master.dbo.sysprocesses WHERE dbid = (SELECT dbid FROM sysdatabases WHERE name = @DatabaseName) open cur FETCH NEXT FROM cur INTO @spid WHILE @@FETCH_STATUS = 0 BEGIN SET @CMD = 'kill ' + @spid EXEC (@CMD) FETCH NEXT FROM cur INTO @spid END CLOSE cur DEALLOCATE cur DECLARE @i int DECLARE @j int SET @j = 1 SET @i = 0 RESTORE HEADERONLY FROM DISK=@LogPath SET @i = @@ROWCOUNT WHILE @i+1 > (@j) BEGIN RESTORE LOG @DatabaseName FROM DISK=@LogPath WITH FILE = @j, STANDBY = @StandbyFile SET @j = @j + 1 END SET @j = 1 SET @i = 0 CREATE TABLE #HeaderTable (BackupName varchar(255), BackupDescription varchar(255), BackupType int, ExpirationDate datetime, Compressed int, Position int, DeviceType int, UserName VarChar(255), ServerName varchar(255), DatabaseName varchar(255), DatabaseVersion int, DatabaseCreationDate datetime, BackupSize numeric(24,0), FirstLsn numeric(24,0), LastLsn numeric(24,0), CheckpointLsn numeric(24,0), DatabaseBackupLsn numeric(24,0), BackupStartDate datetime, BackupFinishDate datetime, SortOrder int, CodePage int, UnicodeLocaleID int, UnicodeComparisonStyle int, CompatibilityLevel int, SoftwareVendorID int, SoftwareVersionMajor int, SoftwareVersionMinor int, SoftwareVersionBuild int, MachineName varchar(255), Flags int, BindingID uniqueidentifier, RecoveryForkID uniqueidentifier, Collation varchar(255), FamilyGUID uniqueidentifier, HasBulkLoggedData bit, IsSnapshot bit, IsReadOnly bit, IsSingleUser bit, HasBackupChecksums bit, IsDamaged bit, BeginsLogChain bit, HasIncompleteMetaData bit, IsForceOffline bit, IsCopyOnly bit, FirstRecoveryForkID uniqueidentifier, ForkPointLSN numeric(24, 0), RecoveryModel varchar(256), DifferentialBaseLSN numeric(24, 0), DifferentialBaseGUID uniqueidentifier, BackupTypeDescription varchar(256), BackupSetGUID uniqueidentifier, CompressedBackupSize int ) INSERT INTO #HeaderTable EXEC master.dbo.GetRestoreCount @LogPath SET @i = (SELECT COUNT(*) FROM #HeaderTable) DROP TABLE #HeaderTable WHILE @i+1 > (@j) BEGIN RESTORE LOG @DatabaseName FROM DISK=@LogPath WITH FILE = @j, STANDBY = @StandbyFile SET @j = @j + 1 END END GO


GetRestoreCount
This is used internally by RestoreLog

USE [master] GO SET ANSI_NULLS ON GO SET QUOTED_IDENTIFIER ON GO CREATE PROCEDURE [dbo].[GetRestoreCount] @FileName AS varchar(1000) AS RESTORE HEADERONLY FROM DISK=@FileName GO

Comments

  1. Hi! Nice Post!

    I'll give it a try, if i dare! This is exactly my problem! How can they say SQL Log Shipping is supported when it's only half way true? But should one expect from free software..

    ReplyDelete

Post a Comment

Popular posts from this blog

Freeing Disk Space on C:\ Windows Server 2008

I just spent the last little while trying to clear space on our servers in order to install .NET 4.5. Decided to post so my future self can find the information when I next have to do this. I performed all the usual tasks: Deleting any files/folders from C:\windows\temp and C:\Users\%UserName%\AppData\Local\TempDelete all EventViewer logs Save to another Disk if you want to keep themRemove any unused programs, e.g. FirefoxRemove anything in C:\inetpub\logsRemove any file/folders C:\Windows\System32\LogFilesRemove any file/folders from C:\Users\%UserName%\DownloadsRemove any file/folders able to be removed from C:\Users\%UserName%\DesktopRemove any file/folders able to be removed from C:\Users\%UserName%\My DocumentsStop Windows Update service and remove all files/folders from C:\Windows\SoftwareDistributionDeleting an Event Logs Run COMPCLN.exe Move the Virtual Memory file to another disk However this wasn’t enough & I found the most space was cleared by using the Disk Cleanup to…

CPF Contribution Rates for new Singapore Permanent Residents (SPR’s)

Recently my wife and I applied and got approved for Singapore Permanent Residency. After completing the formalities the most significant immediate change is the contribution to CPF which is Singapore’s mandatory social security savings scheme requiring contributions from employers and employees. CPF contributions start from the date you obtain SPR status, which is the date of the entry permit.   Being a relentless budgeter I needed to know exactly how much I and my employer would have to contribute so that I could adjust my budget accordingly as the employee contributions get deducted from the monthly salary. After doing some research I discovered that there is a “graduated” approach to CPF contributions for new SPR’s where the contributions gradually increase in the first and second year and then upon reaching the third year are at the full amount. Note: There is an option for employers to contribute the full amount for year 1 and year 2 and the employee can use the graduated rate, b…

Implementing Custom Castle Windsor Facilities

If you’ve been following my posts you would know that I love Castle Windsor. One of the many useful features I have found is the Facility and I’m going to try and give a good example how you can make use of this. In a recent post I showed how you can add Cross-Cutting concerns to your application by using Interceptors.Now when configuring the Container you can explicitly configure each Interceptor per Service but when you have lot’s of components it can get pretty hard to maintain after a while and can also introduce subtle issues if someone forgets to configure it correctly.Below is how you would configure your Container without using a Facility. On the last line we are specifying the Interceptor explicitly. public void Configure() { container = new WindsorContainer(); container.Register( Component.For<CacheInterceptor>(), Component.For<ICacheProvider>() .ImplementedBy<WebCache…