SQL
Server is the database that we use in many of our products. And as you know
there are also many versions of SQL Server out there as well. At the moment our core product is still using
SQL Server 2008 R2 unfortunately.
Quite
a while ago we need to import data from a legacy application into core product,
which is using SQL Server 2008 R2.
however the legacy database is using some very old technology. During that time the guy who worked on the
data migration project decided to migrate the database to SQL 2012 assume that we will have upgrade our product to SQL2012 by the time
when he finish, also there are some nice features in SQL2012 made his life a
little bit easier.
So , here
comes the problem, as time goes by , the
person who worked on data migration is long gone and our core product
fail to upgrade to SQL 2012, cause we never get the time. I was ask to import
some data from a client to our production core system. Which is still using SQL
2008 R2.
That's
where the problem comes in. Update
database from lower version to higher version is nice and easy, SQL Server have
very good support. Not too much need to be done. However if I need to downgrade a database
from Higher version to lower version, the tool doesn't support by default. I
can't restore a database from SQL2012 to SQL2008.
The Data
Compare feature in my Visual Studio 2010 unfortunately doesn’t support database
version higher than SQL 2008 R2 either.
I can also use SQL server's data import/export to do this as well, but since we use Timestamp field in our database, and use that to do concurrency check, which make it is very tedious to change the each single one of table setting.
Thus I did a bit search. And decide to create a script to achieve this goal. Here is what I did, in case someone else need to do similar task. And also for future myself , just in case I need to do similar things later on.
I can also use SQL server's data import/export to do this as well, but since we use Timestamp field in our database, and use that to do concurrency check, which make it is very tedious to change the each single one of table setting.
Thus I did a bit search. And decide to create a script to achieve this goal. Here is what I did, in case someone else need to do similar task. And also for future myself , just in case I need to do similar things later on.
The following few conditions are unique in my situation, which is also the foundation that this solution is based on.
- The database in SQL 2012 and SQL 2008 have exactly the same schema, table structure etc.
- We don't use any fancy features of SQL Server, no triggers, no SSB , no Queue etc. just basic table.
- Our primary key is unique comprise with two part, one part is using the database auto-generated , Identity or Guid, the other part is maintained by our application based on setting. Thus we won't violate any primary key constrain.
First of
all, I need to create a linked server my SQL 2008 R2 , so as I can access the database in 2012
format. You will need to have access to both SQL2008 and SQL2012 databases.
Click
"New Linked Server…" option.
Assume I have the SQL Server 2012 and SQL Server 2008 R2 installed on
the same machine.
And the
instance name of SQL 2012 is "SQL2012"
Also
go to security tab to specify logons that you need to use to access SQL2012
After
these , I can access my SQL2012 database
easily.
A few
things I need to do in my script.
- Need to traverse every table in the database. To check whether I have data there need to move across to SQL2008
- If the table contains Identity column ,then I need to SET IDENTITY_INSERT ON , before I insert data.
- If the table has foreign key constrains , I need to turn off the constrian check , because the order to insert data is undetermined
- After insert data, need to SET IDENTITY_INSERT OFF .
- We are using Timestamp column in our database which is readonly , so I need to specifically exclude Timestamp column from the script.
Here is
the script that I used to generate the data import script.
USE [Your database name]
GOIF OBJECT_ID('SQLStatement') IS NOT NULLBEGIN
DROP TABLE SQLStatementENDGOCREATE TABLE SQLStatement([SQL] NVARCHAR(MAX),Id INT NOT NULL IDENTITY)GOSET NOCOUNT ON
DECLARE @TenantId INT
DECLARE @FromTable NVARCHAR(255)DECLARE @ColumnList NVARCHAR(MAX)DECLARE @SourceTable NVARCHAR(255)DECLARE @SourceTablePreFix NVARCHAR(255)DECLARE @SQLPrepareSource NVARCHAR(MAX)--SET @SourceTablePreFix = '[eVisionRelease].'SET @SourceTablePreFix = '.\sql2012.[Your database name].'SET @TenantId = 3SET @SQLPrepareSource = 'insert #rectables select ''?'',count(*) rec_count from ? 'CREATE TABLE #rectables (tbl_name VARCHAR(100), rec_count INT)EXEC sp_MSforeachtable @SQLPrepareSourceDECLARE cTable CURSOR FOR
SELECT tbl_name FROM #rectables WHERE rec_count > 0
OPEN cTable
FETCH NEXT FROM cTable INTO @FromTableWHILE @@FETCH_STATUS = 0BEGIN
DECLARE @cols AS NVARCHAR(MAX), @query AS NVARCHAR(MAX)
SET @SourceTable = @SourceTablePreFix + @FromTable
--get all columns, except timestamp
SELECT name AS ColumnName,column_id AS ColumnId INTO #ColumnSource FROM sys.columns WHERE OBJECT_ID=OBJECT_ID(@FromTable) AND system_type_id != 189
SELECT @cols = STUFF((SELECT ',' + QUOTENAME(ColumnName)
FROM #ColumnSource
GROUP BY ColumnName, ColumnId
ORDER BY ColumnId
FOR XML PATH(''), TYPE
).value('.', 'NVARCHAR(MAX)')
,1,1,'')
DECLARE @HasIdentityColumn INT
SET @HasIdentityColumn = (SELECT COUNT(*) FROM sys.identity_columns WHERE [object_id] = OBJECT_ID(@FromTable))
INSERT INTO SQLStatement ([SQL]) SELECT 'ALTER TABLE ' + @FromTable + ' NOCHECK CONSTRAINT ALL'
IF @HasIdentityColumn > 0
INSERT INTO SQLStatement ([SQL]) SELECT 'SET IDENTITY_INSERT ' + @FromTable + ' ON'
INSERT INTO SQLStatement ([SQL]) SELECT 'DELETE FROM ' +@FromTable + ' WHERE TenantId=@TenantId'
INSERT INTO SQLStatement ([SQL]) SELECT 'INSERT INTO ' + @FromTable + '(' + @cols + ') SELECT ' + @cols +' FROM ' + @SourceTable + ' WHERE TenantId=@TenantId'
IF @HasIdentityColumn > 0 INSERT INTO SQLStatement ([SQL]) SELECT 'SET IDENTITY_INSERT ' + @FromTable + ' OFF'
DROP TABLE #ColumnSource
FETCH NEXT FROM cTable INTO @FromTable
END
CLOSE cTableDEALLOCATE cTable DROP TABLE #rectablesSELECT [SQL] FROM SQLStatement ORDER BY Id
------------------------------------------------------------
After execute this script, it will generate the final script that we will need to
execute to import data. Run the script in SQL2008 database. The one we need to import data into.
Finger cross!
Final
step, as we turn off the constrain check on foreign key, after we done with the
import we need to turn it back on.
USE [Your databasename]
EXEC sp_MSforeachtable 'ALTER TABLE ? WITH CHECK CHECK CONSTRAINT ALL'
That's
it...
Be aware this solution can be only executed on an isolated and offline environment, if you trying to do this on your live database, DBA will certainly kill you….
Be aware this solution can be only executed on an isolated and offline environment, if you trying to do this on your live database, DBA will certainly kill you….
No comments:
Post a Comment