2010-01-19 11 views
11

मैं एक SQL स्क्रिप्ट लिखना चाहता हूं जो एक ही सर्वर पर डेटाबेस कॉपी करेगा। मैं बैकअप/पुनर्स्थापना कर सकता था, लेकिन मुझे लगता है कि किसी भी तरह से "कॉपी" करने के लिए यह तेज़ हो सकता है। यदि संभव है तो किस प्रकार, क्या कोई जानता है? क्या एक स्क्रिप्ट लिखने का कोई तरीका है जो सिर्फ अलग हो जाएगा, फ़ाइल को एचडी पर कॉपी करें, और फिर दोनों प्रतियों को दोबारा दोहराएं?SQL स्क्रिप्ट डेटाबेस को "कॉपी" करने के लिए

+1

कॉपी डेटाबेस विज़ार्ड का उपयोग क्यों न करें? 2005+ –

+1

मानते हैं क्या आप डेटा को प्रतिलिपि बनाना चाहते हैं या केवल ऑब्जेक्ट्स? –

+0

मैं बस बैकअप/रीस्टोर करूँगा, लेकिन इसमें घंटों लग रहे हैं क्योंकि डेटा बहुत बड़ा है। मैं सोच रहा था कि चीजों को गति देने के लिए केवल डेटा/कॉपी/ATTACHX2 चाल करने के लिए यह तेज़ हो सकता है। मैंने सोचा कि फ़ाइल कॉपी बैकअप/रीस्टोर – skb

उत्तर

0

मुझे यकीन है कि नहीं कर रहा हूँ, लेकिन मुझे लगता है कि आप इस के लिए देख रहे हैं:

BACKUP DATABASE MyDB TO DISK='D:\MyDB.bak' 
2

आज़माएं:

USE master 
GO 
-- the original database (use 'SET @DB = NULL' to disable backup) 
DECLARE @DB varchar(200) 
SET @DB = 'PcTopp' 
-- the backup filename 
DECLARE @BackupFile varchar(2000) 
SET @BackupFile = 'c:\pctopp\sqlserver\backup.dat' 
-- the new database name 
DECLARE @TestDB varchar(200) 
SET @TestDB = 'TestDB' 
-- the new database files without .mdf/.ldf 
DECLARE @RestoreFile varchar(2000) 
SET @RestoreFile = 'c:\pctopp\sqlserver\backup' 
-- **************************************************************** 
-- no change below this line 
-- **************************************************************** 

DECLARE @query varchar(2000) 
DECLARE @DataFile varchar(2000) 
SET @DataFile = @RestoreFile + '.mdf' 
DECLARE @LogFile varchar(2000) 
SET @LogFile = @RestoreFile + '.ldf' 
IF @DB IS NOT NULL 
BEGIN 
SET @query = 'BACKUP DATABASE ' + @DB + ' TO DISK = ' + QUOTENAME(@BackupFile, '''') 
EXEC (@query) 
END 
-- RESTORE FILELISTONLY FROM DISK = 'C:\temp\backup.dat' 
-- RESTORE HEADERONLY FROM DISK = 'C:\temp\backup.dat' 
-- RESTORE LABELONLY FROM DISK = 'C:\temp\backup.dat' 
-- RESTORE VERIFYONLY FROM DISK = 'C:\temp\backup.dat' 
IF EXISTS(SELECT * FROM sysdatabases WHERE name = @TestDB) 
BEGIN 
SET @query = 'DROP DATABASE ' + @TestDB 
EXEC (@query) 
END 
RESTORE HEADERONLY FROM DISK = @BackupFile 
DECLARE @File int 
SET @File = @@ROWCOUNT 
DECLARE @Data varchar(500) 
DECLARE @Log varchar(500) 
SET @query = 'RESTORE FILELISTONLY FROM DISK = ' + QUOTENAME(@BackupFile , '''') 
CREATE TABLE #restoretemp 
(
LogicalName varchar(500), 
PhysicalName varchar(500), 
type varchar(10), 
FilegroupName varchar(200), 
size int, 
maxsize bigint 
) 
INSERT #restoretemp EXEC (@query) 
SELECT @Data = LogicalName FROM #restoretemp WHERE type = 'D' 
SELECT @Log = LogicalName FROM #restoretemp WHERE type = 'L' 
PRINT @Data 
PRINT @Log 
TRUNCATE TABLE #restoretemp 
DROP TABLE #restoretemp 
IF @File > 0 
BEGIN 
SET @query = 'RESTORE DATABASE ' + @TestDB + ' FROM DISK = ' + QUOTENAME(@BackupFile, '''') + 
' WITH MOVE ' + QUOTENAME(@Data, '''') + ' TO ' + QUOTENAME(@DataFile, '''') + ', MOVE ' + 
QUOTENAME(@Log, '''') + ' TO ' + QUOTENAME(@LogFile, '''') + ', FILE = ' + CONVERT(varchar, @File) 
EXEC (@query) 
END 
GO 

here

1

से मिल गया वहाँ एक रास्ता है एक स्क्रिप्ट लिखने के लिए जो सिर्फ अलग हो जाएगी, फ़ाइल को एचडी पर कॉपी करें, और फिर दोनों प्रतियों को दोबारा दोहराएं?

हां। अलग करने और संलग्न करने के लिए आप sp_detach_db और sp_attach_db का उपयोग कर सकते हैं। फ़ाइलों की प्रतिलिपि बनाने के लिए, आप xp_cmdshell और xcopy का उपयोग कर सकते हैं।

फिर भी, मुझे लगता है कि बैकअप-और-पुनर्स्थापना दृष्टिकोण आसान है, क्योंकि आपको फ़ाइलों की प्रतिलिपि बनाने की आवश्यकता नहीं है।

1

यहाँ कोड टोनी पोस्ट जो SQL सर्वर 2005

USE master 
GO 
-- the original database (use 'SET @DB = NULL' to disable backup) 
DECLARE @DB varchar(200) 
SET @DB = 'GMSSDB' 
-- the backup filename 
DECLARE @BackupFile varchar(2000) 
SET @BackupFile = 'c:\temp\backup.dat' 
-- the new database name 
DECLARE @TestDB varchar(200) 
SET @TestDB = 'GMSSDBArchive' 
-- the new database files without .mdf/.ldf 
DECLARE @RestoreFile varchar(2000) 
SET @RestoreFile = 'c:\temp\backup' 
-- **************************************************************** 
-- no change below this line 
-- **************************************************************** 

DECLARE @query varchar(2000) 
DECLARE @DataFile varchar(2000) 
SET @DataFile = @RestoreFile + '.mdf' 
DECLARE @LogFile varchar(2000) 
SET @LogFile = @RestoreFile + '.ldf' 
IF @DB IS NOT NULL 
BEGIN 
SET @query = 'BACKUP DATABASE ' + @DB + ' TO DISK = ' + QUOTENAME(@BackupFile, '''') 
EXEC (@query) 
END 
-- RESTORE FILELISTONLY FROM DISK = 'C:\temp\backup.dat' 
-- RESTORE HEADERONLY FROM DISK = 'C:\temp\backup.dat' 
-- RESTORE LABELONLY FROM DISK = 'C:\temp\backup.dat' 
-- RESTORE VERIFYONLY FROM DISK = 'C:\temp\backup.dat' 
IF EXISTS(SELECT * FROM sysdatabases WHERE name = @TestDB) 
BEGIN 
SET @query = 'DROP DATABASE ' + @TestDB 
EXEC (@query) 
END 

CREATE TABLE #headeronly 
(
BackupName nvarchar(128) null, 
BackupDescription nvarchar(255) null, 
BackupType smallint, 
ExpirationDate datetime null, 
Compressed bit, 
Position smallint, 
DeviceType tinyint, 
UserName nvarchar(128), 
ServerName nvarchar(128), 
DatabaseName nvarchar(128), 
DatabaseVersion int, 
DatabaseCreationDate datetime, 
BackupSize numeric(20,0), 
FirstLSN numeric(25,0), 
LastLSN numeric(25,0), 
CheckpointLSN numeric(25,0), 
DatabaseBackupLSN numeric(25,0), 
BackupStartDate datetime, 
BackupFinishDate datetime, 
SortOrder smallint, 
CodePage smallint, 
UnicodeLocaleId int, 
UnicodeComparisonStyle int, 
CompatibilityLevel tinyint, 
SoftwareVendorId int, 
SoftwareVersionMajor int, 
SoftwareVersionMinor int, 
SoftwareVersionBuild int, 
MachineName nvarchar(128), 
Flags int, 
BindingID uniqueidentifier, 
RecoveryForkID uniqueidentifier, 
Collation nvarchar(128), 
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(25,0) NULL, 
RecoveryModel nvarchar(60), 
DifferentialBaseLSN numeric(25,0) NULL, 
DifferentialBaseGUID uniqueidentifier, 
BackupTypeDescription nvarchar(60), 
BackupSetGUID uniqueidentifier NULL 
) 
--RESTORE HEADERONLY FROM DISK = @BackupFile 
SET @query = 'RESTORE HEADERONLY FROM DISK = ' + QUOTENAME(@BackupFile, '''') 
INSERT #headeronly exec(@query) 


DECLARE @File int 
select @File = count(1) from #headeronly 
print CONVERT(varchar, @File) 
DROP TABLE #headeronly 


DECLARE @Data varchar(500) 
DECLARE @Log varchar(500) 
SET @query = 'RESTORE FILELISTONLY FROM DISK = ' + QUOTENAME(@BackupFile , '''') 

--RESTORE FILELISTONLY FROM DISK = 'c:\temp\backup.dat' 

CREATE TABLE #restoretemp 
(
LogicalName nvarchar(128), 
PhysicalName nvarchar(260), 
type char(1), 
FilegroupName nvarchar(128), 
size numeric(20,0), 
maxsize numeric(20,0), 
FileID bigint, 
CreateLSN numeric(25,0), 
DropLSN numeric(25,0)NULL, 
UniqueID uniqueidentifier, 
ReadOnlyLSN numeric(25,0) NULL, 
ReadWriteLSN numeric(25,0) NULL, 
BackupSizeInBytes bigint, 
SourceBlockSize int, 
FileGroupID int, 
LogGroupGUID uniqueidentifier NULL, 
DifferentialBaseLSN numeric(25,0) NULL, 
DifferentialBaseGUID uniqueidentifier, 
IsReadOnly bit, 

IsPresent bit 

) 
--select * from EXEC (@query) 
INSERT #restoretemp EXEC (@query) 
SELECT @Data = LogicalName FROM #restoretemp WHERE type = 'D' 
SELECT @Log = LogicalName FROM #restoretemp WHERE type = 'L' 
PRINT @Data 
PRINT @Log 
TRUNCATE TABLE #restoretemp 
DROP TABLE #restoretemp 
print CONVERT(varchar, @File) 
IF @File > 0 
BEGIN 

SET @query = 'RESTORE DATABASE ' + @TestDB + ' FROM DISK = ' + QUOTENAME(@BackupFile, '''') + 
' WITH MOVE ' + QUOTENAME(@Data, '''') + ' TO ' + QUOTENAME(@DataFile, '''') + ', MOVE ' + 
QUOTENAME(@Log, '''') + ' TO ' + QUOTENAME(@LogFile, '''') + ', FILE = ' + CONVERT(varchar, @File) 
print 'starting restore' 
EXEC (@query) 
print 'finished restore' 
END 
GO 
18

में काम करता है शेर @Tony का एक संस्करण है: हाय - मैं अपनी स्क्रिप्ट का उपयोग कर कुछ समस्याओं का अनुभव है, इसलिए मैं का एक संकर के साथ आया था अपनी स्क्रिप्ट और इस पोस्ट: script that copies a database के लिए link

USE master; 
GO 
-- the original database (use 'SET @DB = NULL' to disable backup) 
DECLARE @SourceDatabaseName varchar(200) 
DECLARE @SourceDatabaseLogicalName varchar(200) 
DECLARE @SourceDatabaseLogicalNameForLog varchar(200) 
DECLARE @query varchar(2000) 
DECLARE @DataFile varchar(2000) 
DECLARE @LogFile varchar(2000) 
DECLARE @BackupFile varchar(2000) 
DECLARE @TargetDatabaseName varchar(200) 
DECLARE @TargetDatbaseFolder varchar(2000) 

-- **************************************************************** 

SET @SourceDatabaseName = '[Source.DB]'     -- Name of the source database 
SET @SourceDatabaseLogicalName = 'Source_DB'    -- Logical name of the DB (check DB properties/Files tab) 
SET @SourceDatabaseLogicalNameForLog = 'Source_DB_log' -- Logical name of the DB (check DB properties/Files tab) 
SET @BackupFile = 'C:\Temp\backup.dat'         -- FileName of the backup file 
SET @TargetDatabaseName = 'TargetDBName'      -- Name of the target database 
SET @TargetDatbaseFolder = 'C:\Temp\' 

-- **************************************************************** 

SET @DataFile = @TargetDatbaseFolder + @TargetDatabaseName + '.mdf'; 
SET @LogFile = @TargetDatbaseFolder + @TargetDatabaseName + '.ldf'; 

-- Backup the @SourceDatabase to @BackupFile location 
IF @SourceDatabaseName IS NOT NULL 
BEGIN 
SET @query = 'BACKUP DATABASE ' + @SourceDatabaseName + ' TO DISK = ' + QUOTENAME(@BackupFile,'''') 
PRINT 'Executing query : ' + @query; 
EXEC (@query) 
END 
PRINT 'OK!'; 

-- Drop @TargetDatabaseName if exists 
IF EXISTS(SELECT * FROM sysdatabases WHERE name = @TargetDatabaseName) 
BEGIN 
SET @query = 'DROP DATABASE ' + @TargetDatabaseName 
PRINT 'Executing query : ' + @query; 
EXEC (@query) 
END 
PRINT 'OK!' 

-- Restore database from @BackupFile into @DataFile and @LogFile 
SET @query = 'RESTORE DATABASE ' + @TargetDatabaseName + ' FROM DISK = ' + QUOTENAME(@BackupFile,'''') 
SET @query = @query + ' WITH MOVE ' + QUOTENAME(@SourceDatabaseLogicalName,'''') + ' TO ' + QUOTENAME(@DataFile ,'''') 
SET @query = @query + ' , MOVE ' + QUOTENAME(@SourceDatabaseLogicalNameForLog,'''') + ' TO ' + QUOTENAME(@LogFile,'''') 
PRINT 'Executing query : ' + @query 
EXEC (@query) 
PRINT 'OK!' 
+0

से तेज हो सकती है देखें कि क्या आप [इस नए प्रश्न] का उत्तर दे सकते हैं (http://stackoverflow.com/questions/2095910/sql-script-to-copy-a-डेटाबेस) मैंने पूछा है। यह पूछ रहा है कि आपने दो अलग-अलग सर्वरों में क्या किया है। – Adamantish

+0

@Adamantish - क्या लिंक सही है? इस सवाल पर इंगित करता है। –

+0

पता नहीं कैसे मैं ऐसा करने में कामयाब रहा, स्पाइक। [यह] (http://stackoverflow.com/questions/21482205/script-to-copy-डेटाबेस- बीच- सर्वर) यह था लेकिन मुझे लगता है कि मैंने इसे समझ लिया है। बस निष्पादित करें [LinkedServerName]। [DBName] .dbo.sp_executesql "स्पाइक का रिमोट डीडीएल कोड"। अभी तक यह कोशिश नहीं की है क्योंकि मुझे इस समस्या के लिए एक अलग दृष्टिकोण मिला है लेकिन भविष्य में आपकी स्क्रिप्ट का उपयोग करके कल्पना कर सकते हैं। – Adamantish

3

स्रोत।

USE master; 

DECLARE 
    @SourceDatabaseName AS SYSNAME = '<SourceDB>', 
    @TargetDatabaseName AS SYSNAME = '<TargetDB>' 



-- ============================================ 
-- Define path where backup will be saved 
-- ============================================ 
IF NOT EXISTS (SELECT 1 FROM sys.databases WHERE name = @SourceDatabaseName) 
    RAISERROR ('Variable @SourceDatabaseName is not set correctly !', 20, 1) WITH LOG  

DECLARE @SourceBackupFilePath varchar(2000) 
SELECT @SourceBackupFilePath = BMF.physical_device_name 
FROM 
    msdb.dbo.backupset B 
    JOIN msdb.dbo.backupmediafamily BMF ON B.media_set_id = BMF.media_set_id 
WHERE B.database_name = @SourceDatabaseName 
ORDER BY B.backup_finish_date DESC 

SET @SourceBackupFilePath = REPLACE(@SourceBackupFilePath, '.bak', '_clone.bak') 



-- ============================================ 
-- Backup source database 
-- ============================================ 
DECLARE @Sql NVARCHAR(MAX) 
SET @Sql = 'BACKUP DATABASE @SourceDatabaseName TO DISK = ''@SourceBackupFilePath''' 
SET @Sql = REPLACE(@Sql, '@SourceDatabaseName', @SourceDatabaseName) 
SET @Sql = REPLACE(@Sql, '@SourceBackupFilePath', @SourceBackupFilePath) 
SELECT 'Performing backup...', @Sql as ExecutedSql 
EXEC (@Sql) 



-- ============================================ 
-- Automatically compose database files (.mdf and .ldf) paths 
-- ============================================ 
DECLARE 
      @LogicalDataFileName as NVARCHAR(MAX) 
     , @LogicalLogFileName as NVARCHAR(MAX) 
     , @TargetDataFilePath as NVARCHAR(MAX) 
     , @TargetLogFilePath as NVARCHAR(MAX) 

SELECT 
    @LogicalDataFileName = name, 
    @TargetDataFilePath = SUBSTRING(physical_name,1,LEN(physical_name)-CHARINDEX('\',REVERSE(physical_name))) + '\' + @TargetDatabaseName + '.mdf' 
FROM sys.master_files 
WHERE 
    database_id = DB_ID(@SourceDatabaseName)   
    AND type = 0   -- datafile file 

SELECT 
    @LogicalLogFileName = name, 
    @TargetLogFilePath = SUBSTRING(physical_name,1,LEN(physical_name)-CHARINDEX('\',REVERSE(physical_name))) + '\' + @TargetDatabaseName + '.ldf' 
FROM sys.master_files 
WHERE 
    database_id = DB_ID(@SourceDatabaseName)   
    AND type = 1   -- log file  



-- ============================================ 
-- Restore target database 
-- ============================================ 
IF EXISTS (SELECT 1 FROM sys.databases WHERE name = @TargetDatabaseName) 
    RAISERROR ('A database with the same name already exists!', 20, 1) WITH LOG   

SET @Sql = 'RESTORE DATABASE @TargetDatabaseName 
FROM DISK = ''@SourceBackupFilePath'' 
WITH MOVE ''@LogicalDataFileName'' TO ''@TargetDataFilePath'', 
MOVE ''@LogicalLogFileName'' TO ''@TargetLogFilePath''' 
SET @Sql = REPLACE(@Sql, '@TargetDatabaseName', @TargetDatabaseName) 
SET @Sql = REPLACE(@Sql, '@SourceBackupFilePath', @SourceBackupFilePath) 
SET @Sql = REPLACE(@Sql, '@LogicalDataFileName', @LogicalDataFileName) 
SET @Sql = REPLACE(@Sql, '@TargetDataFilePath', @TargetDataFilePath) 
SET @Sql = REPLACE(@Sql, '@LogicalLogFileName', @LogicalLogFileName) 
SET @Sql = REPLACE(@Sql, '@TargetLogFilePath', @TargetLogFilePath) 
SELECT 'Restoring...', @Sql as ExecutedSql 
EXEC (@Sql) 
संबंधित मुद्दे