Understanding CleanupTime in Ola Hallengren's SQL Server scripts in relation to FULL backups and LOG backups

Understanding CleanupTime in Ola Hallengren's SQL Server scripts in relation to FULL backups and LOG backups



I am having trouble understanding what exactly to expect from the CleanupTime option in the Ola Hallengren Server Maintenance Solution. I'm finding some related questions, and elaborate answers, but the explanations still puzzle me a bit.


CleanupTime



Specifically:



I am doing a weekly FULL backup, a daily DIFF backup, and an hourly LOG backup.
The FULL backup is using the default CleanupTime of 24h. The DIFF and LOG backup have NULL as CleanupTime.


CleanupTime


CleanupTime



From the documentation of the CleanupTime paramter, I fail to understand if setting the CleanupTime setting for a backup of BackupType FULL, will also delete older DIFF and LOG backup files, or only FULL backup files.


CleanupTime


BackupType



Specify the time, in hours, after which the backup files are deleted.
If no time is specified, then no backup files are deleted.



The latter paragraph makes me think that setting CleanupTime on backups of BackupType FULL will also delete older transaction logs. Yet it is unclear if this paragraph only applies to backups of the BackupType LOG, or also to backups of the BackupType FULL.


CleanupTime


BackupType


BackupType


BackupType



DatabaseBackup has a check to verify that transaction log backups that
are newer than the most recent full or differential backup are not
deleted.



What I am trying to achieve, is that I can do a point-in-time recovery up to 1 week. (We have a very slowly changing database, so this is feasible) The way I understand it now, this would require a week old full backup, and a weeks worth of Transaction log backup. Since the full, and differential backups can only be used to restore to one specific point in time.



So, should I just set the CleanupTime option of my FULL backup job to 24*7? What I'm guessing now is that setting it to 24h, will cause the next FULL backup to delete all older Full, diff and Transaction log backup files, leaving me with a point-in-time recovery window of ... 0 hours. Right?


CleanupTime


24*7




2 Answers
2



The @CleanupTime is always specified for a specific backup job. For example, if you create a Full backup job, a Differential backup job and a Transaction Log backup job, then the @CleanupTime always relates to the extension of the job.


@CleanupTime


@CleanupTime



Let's take a look at a Full backup example.



If you create a full backup job, then you will normally add one or more of the following parameters:


@Databases


@Directory


@BackupType


@CleanupTime


@FileExtensionFull



So you have a backup job in place that will create a full backup according to the schedule you defined for the at job. Let's assume the following:


@FileExtensionFull


'BAK'


@Directory


'F:SQLBACKUP'


@CleanupTime


24



If we look at the MaintenanceSolution.sql file then you will find the description for the parameter:


MaintenanceSolution.sql


SET @CleanupTime = NULL
-- Time in hours, after which backup files are deleted. If no time is specified, then no backup files are deleted.



Well, that's not helping much. Same as in the official documentation on the site. Let's dig further. If you scan through the script you will eventually find a section that looks like this:



The script has been wrapped to increase readability



So Ola is basically using the built-in xp_delete_file function of SQL Server to delete a file at a certain time according to:


xp_delete_file function


@CurrentDirectoryPath


@CurrentFileExtension


@CurrentCleanupDate



But wait what would for example, the @CurrentCleanupDate be? If we go back a bit in the script you can find a section that looks like this:


@CurrentCleanupDate



Ah, so the @CurrentCleanupDate is a date addition which is calculated from the @CleanupTime and the current time GETDATE(). Cool.
(...and we might have just found a typo in the code, because the sections for normal and mirror database both contain Mirror in the code.)


@CurrentCleanupDate


@CleanupTime


GETDATE()


Mirror



What is the relevant section then for @CurrentFileExtension? Let's search around a bit again. And we find:


@CurrentFileExtension



So there you have it.



If the parameters for your Full backup job are set as @FileExtensionFull='BAK' and you have set a @CleanupTime=24 then the procedure will delete all Full backup files that are at least a day old (24 hours).


@FileExtensionFull='BAK'


@CleanupTime=24



The @CurrentCommand02 that gets executed is basically:


@CurrentCommand02


xp_delete_file 0, 'F:SQLBACKUP', 'BAK', '2018-08-20 20:00:00.045'



So it doesn't touch on any other backup files. (Unless of course you have defined 'BAK' to be the extension of all backup types, in which case you lose).


'BAK'





I hadn't even considered digging into the script to figure out what's happening under the hood. Thanks a lot for pointing it out, taking the time to do it, and then communicating your findings so clearly. I love this community...
– Wouter
Aug 20 at 15:18



I upvoted @hot2use's answer as it covers this question in detail, but I did want to share an easy way to test this stuff.



It might help you (as it helped me) to fully understand how the script works if you:


hh


minute


hh



Run a FULL, DIFF and LOG backup of a test database and note the files created in the individual folders. This is what I used (note the CleanupTime of 1 minute due to alter the script from hours to minutes):


CleanupTime


exec [dbo].[DatabaseBackup]
@Databases = 'test',
@Directory = 'C:OlaBackupTest',
@BackupType = 'full',
@Verify = 'N',
@CleanupTime = 1,
@CleanupMode = 'AFTER_BACKUP'

exec [dbo].[DatabaseBackup]
@Databases = 'test',
@Directory = 'C:OlaBackupTest',
@BackupType = 'diff',
@Verify = 'N',
@CleanupTime = 1,
@CleanupMode = 'AFTER_BACKUP'

exec [dbo].[DatabaseBackup]
@Databases = 'test',
@Directory = 'C:OlaBackupTest',
@BackupType = 'log',
@Verify = 'N',
@CleanupTime = 1,
@CleanupMode = 'AFTER_BACKUP'



My testing revealed the following observations:



FULL



Every execution of a FULL backup created a new FULL backup and deleted any FULL backup files older than 1 minute. No DIFF or LOG backup files were ever affected.


FULL


FULL


FULL


DIFF


LOG



DIFF



Every execution of a DIFF backup created a new DIFF backup and deleted any DIFF backup files older than 1 minute. No FULL or LOG backup files were ever affected.


DIFF


DIFF


DIFF


FULL


LOG



LOG



Every execution of a LOG backup created a new LOG backup. Continuous LOG backups (without intervening FULL or DIFF backups) simply continued to accumulate in the LOG backup folder without regard to the cleanup time. If a FULL or DIFF backup was eventually taken, the NEXT run of the LOG backup deleted any LOG backups older than the latest FULL or DIFF and also older than 1 minute.


LOG


LOG


LOG


FULL


DIFF


LOG


FULL


DIFF


LOG


LOG


FULL


DIFF



No FULL or DIFF backup files were ever affected while running LOG backups.


FULL


DIFF


LOG



I would recommend keeping more than 1 week of FULL backups in the event you need to go back in time to restore. Assuming 2 weeks of FULL backups, you'd need a CleanupTime of 336 hours.


FULL


FULL


CleanupTime



During your testing of the 1 minute stuff, you'll see that:


FULL


DIFF


LOG


DIFF


FULL


LOG


LOG


FULL


DIFF





I wish I could accept two answers as valid answers. This, in combination with @hot2use's answer is just perfect... thanks!
– Wouter
Aug 20 at 15:17





@Wouter - I completely understand - the answer given by hot2use is excellent and deserves the 'win'.
– Scott Hodgin
Aug 20 at 15:19





Well, you got my vote on the simple version. Sometimes I just think too far.
– hot2use
Aug 20 at 18:01






By clicking "Post Your Answer", you acknowledge that you have read our updated terms of service, privacy policy and cookie policy, and that your continued use of the website is subject to these policies.

Popular posts from this blog

Help:Category

How can temperature be calculated given relative humidity and dew point?

I have a recursive function to validate tree graph and need a return condition