By: Daniel Farina | Comments (16) | Related: More > Database Administration
Problem
My company bought new storage and we were instructed to move our SQL Server databases to it. The problem arose when we needed to move a 2 TB database with a time frame of 15 minutes. In this tip I will show you how I managed to successfully do it with the native SQL Server options.
Solution
There are several ways to move a database, each one has its pros and cons. Based on this assumption we can state that there isn't a right and wrong method, but there will always be one method that is more suitable for a given scenario.
I will briefly cover the most common methods to move a database and enumerate its pros and cons.
The SQL Server Database Detach and Attach Method to Move a Database
This is the first method that comes to mind when we need to move a database. It consists of executing sp_detach_db system stored procedure, moving the database files and then issuing a CREATE DATABASE FOR ATTACH statement.
Pros
- It is the easiest way to move a database.
- It doesn't produce fragmentation on tables and indexes.
Cons
- Database is inaccessible while files are being moved.
- For a replicated database to be detached, it must be unpublished.
- When a database is detached, all its metadata is dropped. If the database was the default database of any login accounts, master becomes their default database. Also cross database ownership chaining is broken.
- Before detaching a database, you must drop all of its snapshots.
- If database is being mirrored it cannot be detached until the database mirroring session is terminated.
The SQL Server Database Backup and Restore Method to Move a Database
The second easiest method, in my opinion, to move a database is to back it up and restore it by using the WITH MOVE switch to specify a new file location.
Pros
- Anybody who knows how to backup and restore a database can do it.
- It doesn't produce fragmentation on tables and indexes.
Cons
- Database or filegroup is inaccessible until the database restore finishes.
The SQL Server OFFLINE Database Method to Move a Database
By setting the database offline we can then manually move the files and issue an ALTER DATABASE MODIFY FILE command to update the database's metadata and after that set the database back online.
Pros
- It is much like the Detach / Attach method.
- It doesn't produce fragmentation on tables and indexes.
- It preserves database metadata unlike the Detach / Attach method.
- It works with published and mirrored databases.
Cons
- Database is inaccessible while files are being moved.
The SQL Server New Filegroup Method to Move a Database
This method requires a little more effort. It consists of creating a new filegroup and moving objects into the new filegroup by rebuilding tables and indexes. After that, you can delete the old filegroup unless it was the primary filegroup.
Pros
- It doesn't produce fragmentation on tables and indexes.
- It preserves database metadata.
Cons
- This method won't work for the primary filegroup.
- Since you are altering the database design, it may be prudent to speak with someone responsible for the application like the developer or perform a lab test prior to executing this task.
- Requires more work than the earlier mentioned methods.
The SQL Server Log Shipping Method to Move a Database
For this method, you configure your database as a log shipping primary and during a maintenance window you switch databases (remove log shipping and rename databases). Of course if your database is in simple recovery model you need to change it to the full or bulk logged recovery model prior configuring log shipping. Here is a tip that explains Step By Step SQL Server Log Shipping.
Pros
- It doesn't produce fragmentation on tables and indexes.
- It preserves database metadata.
- Database remains online, so users can continue accessing to it.
Cons
- It needs SQL Server Agent service up and running.
- Since Log shipping uses a folder to store backups you may need to review folder permissions if you are running SQL Server Agent and SQL Server services with different accounts.
- Although this method doesn't require much work compared to other options it requires a little bit more knowledge; you need to know how to set up log shipping and failover to the database on the new storage in a successful manner.
Manual SQL Server Log Shipping to Move a Database
Basically log shipping automates the process of having a synchronized secondary database and is great if you need to maintain the synchronization over time, but in this case we can do the same without the automation because we want to have a synchronized database for maybe one day and then switchover. So, just to save steps, we can restore a full backup as a new database without recovery and in the maintenance window take a log backup and restore it to the secondary with recovery and perform the switchover (rename the databases) to our new database.
Pros
- It doesn't produce fragmentation on tables and indexes.
- It preserves database metadata.
- Database remains online, so users can continue accessing to it.
- It requires fewer configurations than Log Shipping Method
Cons
- Remember to use WITH MOVE when restoring the backup.
The DBCC SHRINKFILE Method to Move a Database
This option consists of emptying each file by executing the DBCC SHRINKFILE with the EMPTYFILE argument. You probably heard about shrink procedures and fragmentation, but this is not something you will do every day. You will probably rarely change database file locations. Books Online states that DBCC SHRINKFILE with the EMPTYFILE argument "empties a file by moving the data from the specified file to other files in the same filegroup, allowing the file to be removed from the database".
This is the slowest of all methods. Just to let you know, it took almost 2 weeks to move my 2 TB database. The long running times of shrink operations are mainly because those are single threaded operations unlike table or index rebuilds that can be parallelized. Furthermore, if you have enabled HyperThreading on your CPU it can take even longer because when this feature is enabled the processor cache per core is divided by two. Take for example a CPU which has 4 cores and 8 MB of cache memory. When HyperThreading is disabled each core has a single execution thread, so each execution thread has 2 MB of cache memory (8 MB divided by the number of cores). In opposition, when HyperThreading is enabled, each core hosts two execution threads and that gives us 1 MB of cache memory per thread (8 MB divided by the number of cores, 4 and then divided by the number of threads per core, 2).
If you use this method then you should also consider the impact on performance. Even being that this a single threaded operation it could have a profound effect on performance. Not for the data movement per se, but for data fragmentation. We all know that shrinking a database is the worst thing you can do for index performance due to logical file fragmentation. Of course the impact will depend on your environment and factors like tables size and query types.
Also if you use this method; remember that shrinking a data file generates transaction log records, so be careful if your database is the primary database on a log shipping configuration.
Pros
- Database remains online, so users can continue accessing to it.
Cons
- It may take longer than the other methods.
- It produces a lot of fragmentation on tables and indexes.
- It is a logged operation, so if your database is in FULL recovery model this method won't be suitable.
- Not recommended when using SQL Server Log Shipping.
Sample Code
1. First we create our sample Database
USE [master] GO CREATE DATABASE [TestDB] CONTAINMENT = NONE ON PRIMARY ( NAME = N'TestDB', FILENAME = N'E:\MSSQL\TestDB.mdf' , SIZE = 4096KB , MAXSIZE = UNLIMITED, FILEGROWTH = 1024KB ), ( NAME = N'TestDB_1', FILENAME = N'E:\MSSQL\TestDB_1.ndf' , SIZE = 4096KB , MAXSIZE = UNLIMITED, FILEGROWTH = 1024KB ), ( NAME = N'TestDB_2', FILENAME = N'E:\MSSQL\TestDB_2.ndf' , SIZE = 4096KB , MAXSIZE = UNLIMITED, FILEGROWTH = 1024KB ) LOG ON ( NAME = N'TestDB_log', FILENAME = N'E:\MSSQL\TestDB.ldf' , SIZE = 4096KB , MAXSIZE = 2048GB , FILEGROWTH = 1024KB) GO ALTER DATABASE [TestDB] SET RECOVERY SIMPLE
2. This will be our test table
USE TestDB GO IF OBJECT_ID('dbo.SampleTable', 'U') IS NOT NULL DROP TABLE dbo.SampleTable GO CREATE TABLE dbo.SampleTable ( ID INT IDENTITY(1, 1) , Data DATETIME PRIMARY KEY CLUSTERED ( ID ) ) GO
3. Now we insert some test data.
Change the number after GO for your needs.
USE TestDB GO INSERT INTO dbo.SampleTable ( Data ) VALUES ( GETDATE() ) GO 1000000
4. In this step we create the files on our new drive.
USE [master] GO ALTER DATABASE [TestDB] ADD FILE ( NAME = N'TestDB_1_New', FILENAME = N'G:\MSSQL\TestDB_1_New.ndf' , SIZE = 4096KB , FILEGROWTH = 1024KB ) TO FILEGROUP [PRIMARY] GO ALTER DATABASE [TestDB] ADD FILE ( NAME = N'TestDB_2_New', FILENAME = N'G:\MSSQL\TestDB_2_New.ndf' , SIZE = 4096KB , FILEGROWTH = 1024KB ) TO FILEGROUP [PRIMARY] GO
5. We must change auto grow settings for the old files, so when we move data it goes to the new files and not the old ones.
USE [master] GO ALTER DATABASE [TestDB] MODIFY FILE ( NAME = N'TestDB', FILEGROWTH = 0) GO ALTER DATABASE [TestDB] MODIFY FILE ( NAME = N'TestDB_1', FILEGROWTH = 0) GO ALTER DATABASE [TestDB] MODIFY FILE ( NAME = N'TestDB_2', FILEGROWTH = 0) GO
6. It's time to move the data.
If your primary file is too big you may want to truncate the free space to move the file quickly. Also when empting the primary file you will receive an error message telling you that system's tables data can only reside into primary file and cannot be moved. Since our sample database is configured using the simple recovery model, to empty our transaction log we execute a CHECKPOINT and then proceed to truncate the log.
USE TestDB GO DBCC SHRINKFILE('TestDB_1', EMPTYFILE) GO DBCC SHRINKFILE('TestDB_1', TRUNCATEONLY) GO DBCC SHRINKFILE('TestDB_2', EMPTYFILE) GO DBCC SHRINKFILE('TestDB_2', TRUNCATEONLY) GO DBCC SHRINKFILE('TestDB', EMPTYFILE) GO DBCC SHRINKFILE('TestDB', TRUNCATEONLY) GO CHECKPOINT DBCC SHRINKFILE('TestDB_log', TRUNCATEONLY) GO
7. We put our Database into OFFLINE state to move our Database primary data file and transaction log.
USE master GO ALTER DATABASE TestDB SET OFFLINE WITH ROLLBACK IMMEDIATE
8. Now we update the Primary File Location in our database's Metadata executing an ALTER DATABASE command.
USE master GO ALTER DATABASE TestDB MODIFY FILE ( NAME = N'TestDB', FILENAME = N'G:\MSSQL\TestDB.mdf' ); ALTER DATABASE TestDB MODIFY FILE ( NAME = N'TestDB_log', FILENAME = N'G:\MSSQL\TestDB.ldf' );
9. We put our Database back into ONLINE state.
USE master GO ALTER DATABASE TestDB SET ONLINE WITH ROLLBACK IMMEDIATE
10. In this step we remove the old files.
USE [TestDB] GO ALTER DATABASE [TestDB] REMOVE FILE [TestDB_1] GO USE [TestDB] GO ALTER DATABASE [TestDB] REMOVE FILE [TestDB_2] GO
11. This step is to rename new files. You can omit this if you don't want to use the original filenames.
USE [TestDB] GO ALTER DATABASE [TestDB] MODIFY FILE (NAME=N'TestDB_1_New', NEWNAME=N'TestDB_1') GO USE [TestDB] GO ALTER DATABASE [TestDB] MODIFY FILE (NAME=N'TestDB_2_New', NEWNAME=N'TestDB_2') GO
Next Steps
- After using the SHRINK method you will need to rebuild all indexes, this tip will help - SQL Server script to rebuild all indexes for all tables and all databases.
- If your database is mirrored this tip will help: How to move database files of a Mirrored SQL Server Database.
- In case your database is being replicated read this: Moving database files for a replicated SQL Server database
- In this tip Brian Kelley explains How to move a SQL Server database within the same instance without losing settings like cross database ownership chaining.
- This tip explains the DETACH / ATTACH procedure: Move SQL Server transaction log files to a different location via TSQL and SSMS.
- If you use the DETACH - ATTACH method this tip will help: Limitations when attaching a SQL Server Database.
- Need to execute a TSQL batch multiple times as I done in this tip?: Executing a TSQL batch multiple times using GO.
About the author
This author pledges the content of this article is based on professional experience and not AI generated.
View all my tips