Let's suppose we have a simple script (taken from a recent
post on stackoverflow.com) to backup all databases on a server.
DECLARE @name VARCHAR(50); -- Database name
DECLARE @path VARCHAR(256); -- Path for backup files
DECLARE @fileName VARCHAR(256); -- Filename for backup
DECLARE @fileDate VARCHAR(20); -- Used for file name
-- Path to
backups.
SET @path = 'D:\Backup\';
-- Get date to
include in file name.
SELECT @fileDate = CONVERT(VARCHAR(20),GETDATE(),112);
-- Dynamically
get each database on the server.
DECLARE db_cursor CURSOR FOR
SELECT name
FROM master.sys.databases
WHERE name NOT IN ('master','model','msdb','tempdb');
OPEN db_cursor;
FETCH NEXT FROM db_cursor INTO @name;
-- Loop through
the list to backup each database.
WHILE @@FETCH_STATUS = 0
BEGIN
-- Build the path and file name.
SET @fileName = @path + @name + '_' + @fileDate + '.BAK';
-- Backup the database.
BACKUP DATABASE
@name TO DISK = @fileName WITH INIT;
-- Loop to the next database.
FETCH NEXT
FROM db_cursor INTO
@name;
END
-- Clean up.
CLOSE db_cursor;
DEALLOCATE db_cursor;
GO
The script will create a new database backup file of every
database on the server each time it's executed.
However, the path may not have unlimited disk space. This is where we need to use xp_delete_file.
Xp_delete_file takes a five parameters:
- File Type = 0 for backup files or 1 for report files.
- Folder Path = The folder to delete files. The path must end with a backslash "\".
- File Extension = This could be 'BAK' or 'TRN' or whatever you normally use.
- Date = The cutoff date for what files need to be deleted.
- Subfolder = 0 to ignore subfolders, 1 to delete files in subfolders.
To incorporate this into our backup script, we just need to
add another parameter to define the cutoff date.
DECLARE @DeleteDate DATETIME = DATEADD(wk,-1,GETDATE());
In this example, we want to purge any backup files that are more than one week old and have a file extension of 'BAK'.
Next we just need to add the xp_delete_file after the backup loop completes.
EXEC master.sys.xp_delete_file 0,@path,'BAK',@DeleteDate,0;
We only need to execute xp_delete_file once with subfolder=0
parameter since all backup files are being stored in the same folder. If there was a separate subfolder for each
database, then we'd change the subfolder parameter to 1 to delete files from
each of those database subfolders.
The updated script would look like this.
DECLARE @name VARCHAR(50); -- Database name
DECLARE @path VARCHAR(256); -- Path for backup files
DECLARE @fileName VARCHAR(256); -- Filename for backup
DECLARE @fileDate VARCHAR(20); -- Used for file name
DECLARE
@DeleteDate DATETIME =
DATEADD(wk,-1,GETDATE()); -- Cutoff date
-- Path to
backups.
SET @path = 'D:\Backup\';
-- Get date to
include in file name.
SELECT @fileDate = CONVERT(VARCHAR(20),GETDATE(),112);
-- Dynamically
get each database on the server.
DECLARE db_cursor CURSOR FOR
SELECT name
FROM master.sys.databases
WHERE name NOT IN ('master','model','msdb','tempdb');
OPEN db_cursor;
FETCH NEXT FROM db_cursor INTO @name;
-- Loop through
the list to backup each database.
WHILE @@FETCH_STATUS = 0
BEGIN
-- Build the path and file name.
SET @fileName = @path + @name + '_' + @fileDate + '.BAK';
-- Backup the database.
BACKUP DATABASE
@name TO DISK = @fileName WITH INIT;
-- Loop to the next database.
FETCH NEXT
FROM db_cursor INTO
@name;
END
-- Purge old
backup files from disk.
EXEC master.sys.xp_delete_file 0,@path,'BAK',@DeleteDate,0;
-- Clean up.
CLOSE db_cursor;
DEALLOCATE db_cursor;
GO
Don't let the stored procedure file name, xp_delete_file,
fool you. Microsoft has some internal
code that only allows for the deletion of database and transaction log backup
files. The stored procedure cannot be
used to delete any other type of file.
Yes, it really is that simple.