使用文件替换法和文件组离线的方式来使数据库上线抛弃ndf文件

使用文件替换法和文件组离线的方式来使数据库上线抛弃ndf文件

http://www.sqlnotes.info/2013/05/07/attach-database-with-missing-ndf-file/

没有数据库备份,直接分离附加的方式

I wonder if you’ve had the situation that I had before where you have to attach a database with one or few missing .ndf files. It seems pretty common since I’ve just seen the same scenario at one of my clients. The reasons behind it are quite similar – They created a new file group with files to host some temporary data for data fixing or testing. New files were not created under the folder as usual and they thought they would remove them right after tasks complete. But the latter part got forgotten afterwards.

 

Months later, due to various reasons, the database got detached and copied over to another location. Original hard drives were already re-assigned. Backups went to secured place far from the city. DBA then found that one/few small ndf files which contains nothing were not copied over.

It’s not a production database, however, it’s still quite important for other teams, such as development and QA. Retrieving backup will take more than a week. You know that the least important missing parts of the database preventing your to attach the database and let your teams to use the most important parts.

Here is the solution allowing you quickly mount the database with missing NDF files.

01 use master
02 if db_id('TestDB') is not null
03     drop database TestDB
04 go
05 create database TestDB on  PRIMARY ( name = 'TestDB', filename = 'C:\Temp\TestDB.mdf'),
06  filegroup FG1 ( name = 'TestDB_File1', filename = 'C:\Temp\TestDB_File1.ndf')
07  log on ( name = 'TestDB_log', filename = 'C:\Temp\TestDB_log.ldf')
08 GO
09 create table TestDB.dbo.a (id int) on [PRIMARY]
10 insert into TestDB.dbo.a values(1)
11 create table TestDB.dbo.b (id int) on [FG1]
12 insert into TestDB.dbo.b values(2)
13 go
14 select * from TestDB.dbo.a
15 select * from TestDB.dbo.b
16 go
17 use master
18 go
19 exec sp_detach_db 'TestDB'
20 --- delete TestDB_File1.ndf
21 --- Copy TestDB.mdf and TestDB_log.ldf to elsewhere
22  
23 use master
24 if db_id('TestDB') is not null
25     drop database TestDB
26 go
27 create database TestDB on  PRIMARY ( name = 'TestDB', filename = 'C:\Temp\TestDB.mdf'),
28  filegroup FG1 ( name = 'TestDB_File1', filename = 'C:\Temp\TestDB_File1.ndf')
29  log on ( name = 'TestDB_log', filename = 'C:\Temp\TestDB_log.ldf')
30 go
31 alter database TestDB modify file(name = 'TestDB_File1', offline)
32 go
33 alter database TestDB set offline
34 -- remove all database files
35 -- copy TestDB.mdf and TestDB_log.ldf back to C:\Temp
36 go
37 alter database TestDB set online
38 /*
39 you will receive message below. but it's fine
40 The Service Broker in database "TestDB" will be disabled because the Service Broker GUID in the database (7DE06CC2-F709-4353-BC17-30A8D141EEFE) does not match the one in sys.databases (9E1BD254-BE20-483A-9E95-ACA98E9009A2).
41 */
42 select * from TestDB.dbo.a
43 /*
44 id
45 -----------
46 1
47 (1 row(s) affected)
48 */
49 select * from TestDB.dbo.b
50 /*
51 Msg 8653, Level 16, State 1, Line 1
52 The query processor is unable to produce a plan for the table or view 'b' because the table resides in a filegroup which is not online.
53 */

The idea behind is that

  1. Create a empty database with the save file layout, including names, filegroups, location, etc, everything but data
  2. Take the missing files offline
  3. Take the database offline
  4. Use the data files to overwrite the empty data files
  5. Bring the database online

After that, the database will become operational. DBCC checkDB will not return errors. Tables reside in the missing files will not be accessible and drop-able, but you can rename them. Usually such kind of issues happen in none production environment. Removing those unusable tables is not extremely critical in the most of the time.  But if you’d like to make the database clean, SQL Server does allow you to modify the system meta data. I will explain it in my future posts.

John Huang – SQL MCM & MVP, http://www.sqlnotes.info

 

 

 Posted by  at 5:00 am Add comments Tagged with: 

 10 Responses to “Attach Database with Missing NDF File”

  1. Thank you for sharing this valuable article.. Its realy very interesting.

  2. You don’t mention specific SQL Server versions (or I missed it if you did)
    I have a similar situation with a 2012 instance — large MDF file for my database, and I added a second file to the file group, but the second file was never used.
    We lost the drive that housed the second file, and I don’t have a backup at the OS level nor a database backup.

    Do you think this method will work with SQL2012?
    I’ll try to replicate the problem on a test DB today.

    • Yes, this method works for version 2008 and 2012. But never tested on 2005.

      • Works well on my test instance. I was concerned, since my case didn’t have a secondary (FG1) file group, but it appears to work on the PRIMARY file group as well.

        I’ll be trying it on my production DB this afternoon, and I’ll let you know how it goes!

      • Works like a champ! Thank you so much.

        –Rob

  3. work great~especially when ndf only contain index

  4. Thanks for the article..

    it found really helpful when i lost ndf file after reconfiguring new server.. we are hopless and it helped to revert back over database…

    Thanks…

  5. Hi John,

    Thank you for great articles you write on SQL server.
    I have question on moving .ndf files

    Say, I have database(DB1) on server(server1), which has 3 FileGroups(FG1,FG2,FG3) along with .MDF and .LDF files.
    Now I have another database(DB2) on another server(server2), I would like to copy the .NDF files from server1 to server2, and do a simple command like.

    CREATE DATABASE …. FOR ATTACHMENT.

    Do you think this is possible?

    Thank you
    Nik

 
USE master
IF DB_ID('TestDB') IS NOT NULL
    DROP DATABASE TestDB
go

CREATE DATABASE TestDB ON PRIMARY ( name = 'TestDB', filename = 'D:\TestDB.mdf'),
    FILEGROUP FG1 ( name = 'TestDB_File1', filename = 'D:\TestDB_File1.ndf')
    LOG ON ( name = 'TestDB_log', filename = 'D:\TestDB_log.ldf')
GO

USE TestDB
GO

CREATE TABLE TestDB.dbo.a ( id INT )
 
INSERT  INTO TestDB.dbo.a
VALUES  ( 1 )

CREATE TABLE TestDB.dbo.b ( id INT )
ON  [FG1]
INSERT  INTO TestDB.dbo.b
VALUES  ( 2 )
go

SELECT  *
FROM    TestDB.dbo.a
SELECT  *
FROM    TestDB.dbo.b
go

USE master
go
EXEC sp_detach_db 'TestDB'
--- delete TestDB_File1.ndf
--- Copy TestDB.mdf and TestDB_log.ldf to elsewhere

USE master
IF DB_ID('TestDB') IS NOT NULL
    DROP DATABASE TestDB
go


CREATE DATABASE TestDB ON PRIMARY ( name = 'TestDB', filename = 'C:\Temp\TestDB.mdf'),
    FILEGROUP FG1 ( name = 'TestDB_File1', filename = 'C:\Temp\TestDB_File1.ndf')
    LOG ON ( name = 'TestDB_log', filename = 'C:\Temp\TestDB_log.ldf')
go
ALTER DATABASE TestDB MODIFY FILE(name = 'TestDB_File1', offline)
go
ALTER DATABASE TestDB SET OFFLINE
-- remove all database files
-- copy TestDB.mdf and TestDB_log.ldf back to C:\Temp
go
ALTER DATABASE TestDB SET ONLINE
/*
you will receive message below. but it's fine
The Service Broker in database "TestDB" will be disabled because the Service Broker GUID in the database (7DE06CC2-F709-4353-BC17-30A8D141EEFE) does not match the one in sys.databases (9E1BD254-BE20-483A-9E95-ACA98E9009A2).
*/
SELECT  *
FROM    TestDB.dbo.a
/*
id
-----------
1
(1 row(s) affected)
*/
SELECT  *
FROM    TestDB.dbo.b
/*
Msg 8653, Level 16, State 1, Line 1
The query processor is unable to produce a plan for the table or view 'b' because the table resides in a filegroup which is not online.
*/

 

posted @ 2014-10-25 16:13  桦仔  阅读(295)  评论(0编辑  收藏  举报