Sunday, 31 August 2014

How to Backup Database in SQL Server - Create a Full Back up of Database

Introduction

In this article we will take a look at the steps you need to follow to take a full backup of SQL Server Database. The steps mentioned in this article are applicable for SQL Server 2005, SQL Server 2008, SQL Server 2008 R2 and SQL Server 2012.

Permissions Required to Take Database Backup in SQL Server

In order to take backups a user must be a member of DBCREATOR Server Role and DB_OWNER Database Role else you will receive the below mentioned error while performing backups.

Error Message

Msg 262, Level 14, State 1, Line 1
BACKUP DATABASE permission denied in database 'DatabaseName'.
Msg 3013, Level 16, State 1, Line 1
BACKUP DATABASE is terminating abnormally.

Sample TSQL Script to Create Login with DBCREATOR Server Role and DB_OWNER Database Role

CREATE LOGIN BackupRestoreAdmin WITH PASSWORD='$tr0ngP@$$w0rd'
GO
CREATE
 USER BackupRestoreAdmin FOR LOGIN BackupRestoreAdmin
GO
EXEC
 sp_addsrvrolemember 'BackupRestoreAdmin', 'dbcreator'
GO
EXEC
 sp_addrolemember 'db_owner','BackupRestoreAdmin'
GO
Database Administrator or a user who is a member of DBCREATOR Server Role and DB_OWNER Database Role can take a full backup of database using either:-
• Using SQL Server Management Studio
• Using TSQL scripts
Let us take a look at both the above mentioned options to perform a full backup of a database in SQL Server.

Take Full Backup of a Database Using SQL Server Management Studio

1. Open SQL Server Management Studio and connect to the appropriate instance of Microsoft SQL Server Database Engine in Object Explorer.
2. Expand Databases node and then right click the database which you want to take a full backup and point to Tasks, and then click Back Up… option as shown in the below snippet to open up Back Up Database dialog box.
How to backup Database Using SQL Server Management Studio - Perform Full Database Backup
3. In Back Up Database Dialog box, in Database List box, verify the database name. Optionally you can even select a different database to backup. You will be able to perform a Full Backup of a database which is FULL, BULK_LOGGED or Simple recovery model. To know more about different recovery models in SQL Server read the following article titled Understanding SQL Server Recovery Model.
4. In Backup type list box select Full. Once you have taken a full backup of a database then you can perform Differential backup or Transactional log backups. However, if you database is in Simple recovery model you will not be able to take a Transactional log backup. This is by design from Microsoft.
5. If you want to perform a Copy-Only backup of a database then choose Copy-Only Backup option in Back Up Database Dialog box. Using copy-only backup option you can take a full, differential or transactional log backup which is independent of the sequence of convential SQL Server backups. The copy-only backup option was introduced in SQL Server 2005.
6. In Backup Component select Databases and in Backup set leave the default Name and optionally enter Desciption. Leave the default value as 0 days for Backup set will expire after option.
Backup Database Dialog Box in SQL Server Management Studio
7. Click the Add button to select the file or backup device for the backup destination as shown in the below snippet and click OK to return to Backup Database Dialog box.
Select Backup Destination
8. Once you have completed the General settings you will be able to see a screen similar to below snippet.
Backup Database Dialog Box in SQL Server Management Studio

9. Select Options in the Select a page pane to view the advanced options. In Overwrite Media choose Back up to the existing media set and Append to the existing backup set option as shown in the below snippet

10. Under Reliability section, you can select the checkbox Verify backup when finished option. By selecting this option you can make sure the database backup is good.

Back up Database Options Page in SQL Server Management Studio
11. Under Compression section, you can select the Compress backup from the drop down list as shown in the below snippet. Microsoft introduced Database Backup Compression Feature in SQL Server 2008 for Enterprise Edition. To know more about Backup Compression read the article titled Implement Backup Compression Feature In SQL Server 2008.
12. To generate TSQL Script for the database backup click Scripts and choose Script Action to your choice from the different options which are available as shown in the snippet below.
Generate Database Backup Script Using SSMS
13. Once the database is successfully backed up you will get a popup message similar to the one shown in below snippet.
Successful Database Backup Dialog Box in SQL Server Management Studio

How to Create Full Database Backup Using TSQL Script

The following example TSQL script can be used to take the full back up of AdventureWorks database.
DECLARE @BackupDatabaseName          SYSNAME 
DECLARE
 @DatabaseBackupLocation      NVARCHAR(120) 
SET @BackupDatabaseName = 'AdventureWorks'
SET
 @DatabaseBackupLocation = 'C:\DBBackups\AdventureWorks.BAK'
/* Database Backup Script Using Compression Feature of SQL Server 2008 */
BACKUP
 DATABASE  @BackupDatabaseName
TO
  DISK = @DatabaseBackupLocation
WITH
 NOFORMAT, NOINIT,  
SKIP
, NOREWIND, NOUNLOAD, COMPRESSION,  STATS = 25 
/* Verify the Database Backup File Once the Backup has completed successfully */
RESTORE
 VERIFYONLY
FROM
 DISK = @DatabaseBackupLocation
GO
TSQL Script to Restore Database Backup in SQL Server

Conclusion

In this article you have seen how easily you can take a Full Database Back of an SQL Server Database.


Best Practice for Store Procedure

Best Practice 1:

Name of store procedure should not start with SP_

Cause:

SQL server uses prefix SP_ for system store procedures which are in sys schema of Master database.
E.g:- SP_WHO, SP_HELP etc. 
If we create any user defined store procedure sql server will first search in Master database then in our current working directory. 

Let's assume our current database is Exact_Help. Now we are creating store procedure named SP_WHO'

USE Exact_Help

CREATE PROCEDURE SP_WHO
AS
BEGIN
   
    SELECT 'Exact Help'
   
END

Now we are executing this procedure:

EXECUTE SP_WHO

Sample output:

spid
ecid
status
loginame
hostname
blk
dbname
cmd
request_id
1
0
background                   
sa
                                                                                                                               
0   
NULL
RESOURCE MONITOR
0
2
0
background                   
sa
                                                                                                                               
0   
NULL
XE DISPATCHER  
0
3
0
background                   
sa
                                                                                                                               
0   
NULL
XE TIMER       
0
4
0
background                   
sa
                                                                                                                               
0   
NULL
LAZY WRITER    
0
5
0
background                   
sa
                                                                                                                                
0   
NULL
LOG WRITER     
0
6
0
background                   
sa
                                                                                                                               
0   
NULL
LOCK MONITOR   
0
7
0
background                   
sa
                                                                                                                               
0   
master
SIGNAL HANDLER 
0

It is not executing our procedure!!! It is executing Master.sys.SP_WHO

If we will execute:

EXECUTE EXACT_HELP.dbo.SP_WHO

Event that it will execute Master.sys.SP_WHO

Problems if user defines procedures begin with prefix sp_:
a. It will take extra execution time to search the stored procedure in sys schema of Master database.
b. If this procedure is already a system stored procedure we will get unexpected result.

Best Practice 2

We should keep the NOCOUNT setting on in the stored procedure.

Cause: 

In any application stored is mostly executed by program where there is no need to know the total numbers of affected rows by SELECT, INSERT, UPDATE, DELETE, MARGE etc statements. I am explaining it by an example. We are creating a stored procedure named usp_GetMessage:

CREATE PROCEDURE usp_GetMessage
AS
BEGIN
   
    SELECT * FROM sys.messages
        
END

Now we are executing this procedure:

EXECUTE usp_GetMessage

If we will click on the Messages tab of query result grid will find a message something like:

(97526 row(s) affected)

Sql server has to pay extra cost to calculate total numbers of affected rows by this procedure. But if we will set the nocount on that is:

ALTER PROCEDURE usp_GetMessage
AS
BEGIN
   
    SET NOCOUNT ON
   
    SELECT * FROM sys.messages
   
END

Now if will execute the procedure sql server will no count the total affected rows by this procedure and we can decrease the total execution time of procedure in each execution.

Note:

There is no relation of NOCOUNT setting and @@ROWCOUNT function.

Best Practice 3:

Never update the value of parameters of a store procedure before using in sql queries.

Cause

I am explaining it by an example. Creating following objects in sql server

--Creating a table
CREATE TABLE tblMessage (
    ntMessageID BIGINT IDENTITY PRIMARY KEY,
    ntSeverity INT,
    vcMessage VARCHAR(500)
)

--Inserting some records
INSERT INTO tblMessage
SELECT
    Severity,
    LEFT ([TEXT], 500)
FROM Sys.messages

--Creating a non clustered index on it
CREATE NONCLUSTERED INDEX NCI_Severity
ON tblMessage(ntSeverity)
INCLUDE(vcMessage)

GO

--Creating stored procedure
CREATE PROC uspGetMsg1(
     @ntSeverity AS INT
)
AS
    
     SET @ntSeverity = @ntSeverity - 4
         
     SELECT vcMessage
     FROM tblMessage
     WHERE ntSeverity = @ntSeverity
GO
    
--Creating similar stored procedure
CREATE PROC uspGetMsg2(
     @ntSeverity AS INT
)
AS
             
     SELECT vcMessage
     FROM tblMessage
     WHERE ntSeverity = @ntSeverity

Store procedure uspGetMsg1 and uspGetMsg2 is exactly same expect strore procedure usoGetMsg1 modified the parameters. Now we are going to excecute the store procedure uspGetMsg1 and uspGetMsg2:

EXECUTE uspGetMsg1 16  
EXECUTE uspGetMsg2 12

In the procedure uspGetMsg1:
@ntSeverity = @ntSeverity - 4
ntSeverity = 16-4 = 2

So both stored procedure will get the data from tblMessage of same ntSeverity that is 12. Now we are comparing the execution plan:


For same ntSeverity we are getting different execution cost. Why?
Stored procedure uspGetMsg1 has got compiled for ntSeverity = 16 not for the updated value of ntSeverity. You can check it by right clicking on SELECT operator in the execution plan checking parameter list in the properties:

 
It shows the procedure has compiled for @ntSeverity = 16 but while actual value is 12. So execution plan is displaying the query cost for ntSeverity = 16 not for 12. If we will observe the tblMessage we will find this table has around 69839 records for ntSeverity = 16 while three are only around 11 records for ntSeverity = 12 that why query 1 is costlier than query 2

Conclusion: 

Sql server is not able to complie the query with updated value of procedure parameters which may lead to wrong execution plan. So do not update the procures parameters before using the queries.

Best Practice 4

Don't write such stored procedure which has big difference in execution plan due to change of value of parameters.

Best Practice 5

Use TRY CATCH block to handle the errors.
I am explaining it one example. Let's assume in our current database has three tables of following schema:

CREATE TABLE tblEmp(
     ntEmpID  BIGINT PRIMARY KEY IDENTITY,
     vcName  VARCHAR(10)
)

CREATE TABLE tblDep(
     ntDepID  BIGINT PRIMARY KEY IDENTITY,
     vcName  VARCHAR(20)
)

CREATE TABLE tblOrder(
     ntOrderID  BIGINT PRIMARY KEY IDENTITY,
     vcName  VARCHAR(10)
)

Now we are creating a store procedure:

CREATE PROC uspInserData
AS
BEGIN
     BEGIN TRY
    
          INSERT INTO tblEmp VALUES('Scott'),('Greg')
          INSERT INTO tblDep VALUES('Information Thechnoloy'),('Electrical')
          INSERT INTO tblOrder VALUES('Book'),('Copy')
         
     END TRY
     BEGIN CATCH
         
          SELECT
              ERROR_MESSAGE() AS ErrMsg,
              ERROR_LINE() AS ErrLine,
              ERROR_PROCEDURE() AS ErrProc
                
     END CATCH
END

Now we are going to execute this store procedure:

 EXECUTE dbo.uspInserData

We will get error message like:

ErrMsg
ErrLine
ErrProc
String or binary data would be truncated.
7
uspInserData

In the catch block we can handle message according to our requirements. This error message doesn't tell which table has thrown this error message. With help of Error line number we can find it. Line number:
1. Always start from CREATE or ALTER PROC
2. Error line number also includes empty line
3. Since dynamic sql executes in different scope so error line in case of error in dynamic query start from beginning of the dynamic query.

Best Practice 7: 

Follow coding convention.

Best Practice 8: 

Deallocate the temporary tables, sql cursors if it has no use.

Cause: 

Temporary tables and sql cursors are create in the tempdb database. Disk space etc of system database tempdb is very crucial for query performance so we should use resources of tempdb when it is really necessary. For example:

CREATE uspTempdbManagemet   
AS
BEGIN

     DECLARE @ntSeverity AS INT = 0
     DECLARE @ntCount AS INT
    
     SELECT severity,COUNT(*) AS ntCount
     INTO #Tempdb
     FROM sys.messages
     GROUP BY severity
    
     UPDATE tblEmployee SET ntAge = ntAge + 1
     SELECT * FROM tblEmployee
    
    
     WHILE(EXISTS(SELECT * FROM #Tempdb WHERE severity >@ntSeverity)) BEGIN
    
          SELECT
              @ntSeverity = severity,
              @ntCount = ntCount
          FROM #Tempdb
          WHERE severity <> @ntSeverity
          ORDER BY severity
    
          EXECUTE dbo.uspMsg @ntSeverity,@ntCount
         
          SET @ntSeverity = @ntSeverity + 1
         
     END
    
     SELECT * FROM tblOrder
     SELECT * FROM tblOrderHistory
    
END

Above stored procedure has few problems from use of temporary table point of view. We should:

a. Create temporary table when it is necessary
b. Drop temporary table when it is not necessary

Now we are writing same stored procedure in better way:

CREATE uspTempdbManagemet   
AS
BEGIN

     DECLARE @ntSeverity AS INT = 0
     DECLARE @ntCount AS INT
    
     UPDATE tblEmployee SET ntAge = ntAge + 1
     SELECT * FROM tblEmployee
    
    
     SELECT severity,COUNT(*) AS ntCount
     INTO #Tempdb
     FROM sys.messages
     GROUP BY severity
    
     WHILE(EXISTS(SELECT * FROM #Tempdb WHERE severity >@ntSeverity)) BEGIN
    
          SELECT
              @ntSeverity = severity,
              @ntCount = ntCount
          FROM #Tempdb
          WHERE severity <> @ntSeverity
          ORDER BY severity
    
          EXECUTE dbo.uspMsg @ntSeverity,@ntCount
         
          SET @ntSeverity = @ntSeverity + 1
         
     END
    
     DROP TABLE #Tempdb
    
     SELECT * FROM tblOrder
     SELECT * FROM tblOrderHistory
    
END

Same is true for other temporary objects like sql cursor etc.

Best Practice 9: 

Avoid using global temporary table in the stored procedure.

Cause: 

Scope of global temporary tables is connection. For example:

CREATE PROC uspGlobalTempTable
AS
     SELECT *
     INTO ##tblGlobal
     FROM sys.messages
    
     WAITFOR DELAY '0:00:30';
     DROP TABLE ##tblGlobal 

Now execute this store procedure in two query window of Sql server management studio (ssms).

Query window 1:
EXECUTE dbo.uspGlobalTempTable

Query window 2:
EXECUTE dbo.uspGlobalTempTable

We may get error message like:
There is already an object named '##tblGlobal' in the database.

So try to avoid using global temporary tables in the store procedures.

Best Practice 10

Use schema name while executing any stored procedure.

Cause: 

If we will not specify the schema name in the stored procedure first it will search in sys schema then default schema of user. For example:

Let's assume any user has dbo as default schema. He has created stored procedure in his default schema:
CREATE PROC XP_LOGININFO
AS
     SELECT DB_NAME()

Now he is executing this stored procedure:

EXECUTE XP_LOGININFO    

He is expecting that result set will be current database name while he got something like this:

account name
type
privilege
mapped login name
permission path
NT AUTHORITY\SYSTEM
user
admin
NT AUTHORITY\SYSTEM
NULL
NT SERVICE\MSSQL$SQLEXPRESS
group
admin
NT SERVICE\MSSQL$SQLEXPRESS
NULL
Ritesh-PC\Ritesh
user
admin
Ritesh-PC\Ritesh
NULL
BUILTIN\Users
group
user
BUILTIN\Users
NULL
RITESH-PC\ExactHelp
user
user
RITESH-PC\ExactHelp
NULL

Sql server has first searched this procedure in the sys schema and it found this procedure at there and executed it and it didn't executed the stored procedure created by him. 

So correct and efficient way is to execute any stored procedure is specify the schema name. For example:

EXECUTE dbo.XP_LOGININFO     

SQL Server Query Execution Plans for beginners – Clustered Index Operators

In this article we will continue discussing the various execution plan operators related to clustered indexes and what they do, when do the...