SQL Server 2019 High Availability (SQL Server Simplified)
SQL Server 2019 High Availability (SQL Server Simplified)
Downloads needed:
https://www.microsoft.com/en-us/evalcenter/evaluate-windows-server-2012-
r2
https://www.microsoft.com/en-us/evalcenter/evaluate-sql-server-2016?i=1
https://www.vmware.com/products/player/faqs/faqs.html
Prerequisites
        You should follow the article, A comprehensive guide to SQL Server
        Always On Availability Groups on Windows Server 2016, and be
        ready with the Powered on virtual machines.
        You should get a basic knowledge of domain controller, active
        directory, IP configurations
Enable Domain Controller and Active Directory in a virtual machine for SQL
Server Always On Availability Groups
Before we enable these features and roles, let’s go over their brief
description:
        Domain Controller: A domain controller servers all security
        authentications requests for a Windows Server domain. In an
        organization, each server is a member of the domain controller. We
        use an FQDN [ServerName].[Domain] to connect with the server
        DNS: You cannot remember the IP addresses of all servers. For
        example, we can easily connect to SQLShack.com, but if you have its
        IP address, it is difficult to remember all URL’s IP address. It is a
        standard method to associate names instead of the IP address
        Active Directory: It is a container that consists of organization units
        for all users, their credentials, groups. All users must authenticate
        themselves to use an organization resource
Click on the Add Roles and Features. It opens the wizard with brief
information. We can skip this step.
In the next step, Select option Role-based or feature-based installation
and click Next.
In the DNS option, skip the configuration and move towards the next page.
It shows the NetBIOS domain name. It is the domain name without .com
suffix.
First, it does the prerequisite check. We can ignore the warning messages
here.
It performs the reboot of the VM.
After reboot, you can verify that the computer is part of the
MyDemoSQL.com domain. At this point, we have only one VM configured
with the domain.
Network configuration for the Static IP and DNS Server
We require a static IP for the domain controller VM along with the SQL
Server Always On Availability Groups. Type ipconfig and it returns the
following output.
In the output, we can note the IPv4 address, subnet mask and default
gateway.
Click OK to save the changes. You can again type ipconfig in the command
prompt to validate these settings.
Open the Reverse Lookup Zones folder and right-click on it to launch a new
zone wizard.
In the next zone wizard, go with the default option – Primary zone.
In the next step, select the zone replication scope- To all DNS servers
running on domain controllers in the domain: MyDemoSQL.com
Select the type of lookup zone as a Reverse lookup zone.
We are using an IPv4 address range, so select the IPv4 reverse lookup zone.
In case you use an IPv6 IP range, select the other option.
In the reverse lookup zone name, we need to enter the network ID portion of
the IP address. It is the digits before the last dot. In my case, I have an IP
address in the range 10.0.2.15. Therefore, the network SID is 10.0.2
To create a new AD user, right-click on the container (in this case, Users) and
create a new user.
Specify the first name, last name, user login name. It should be a unique login
name in an OU.
On the next page, specify the password of this AD user along with
configuration options. You can specify options such as :
       User must change password at next logon
       User cannot change password
       Password never expires
       An account is disabled
Add this user as an administrator in all three VM’s as well. Add this user to
the local administrator of all three VM’s. Open the computer management
from the server manager -> Tools-> Computer management.
Add SQLNode1 and SQLNode2 in the domain for SQL Server Always On
availability groups
In the next step, open the network properties of the SQLNode1 and
SQLNode2. Enter the following values for the IP address.
SQLNode1 network configuration
       IP address: 10.0.2.21
       Subnet mask: 255.255.255.0
       Preferred gateway – blank
       Preferred DNS server: 10.0.2.15 ( it is the IP address of our DNS
       server)
Validate IP configurations
It opens the system properties. Click on the change, and you can specify the
computer name and its domain.
Click on OK, and it joins the VM into the specified domain. You need to
specify the domain admin user name and password to allow it as a member in
the MyDemoSQL.com domain.
You get a welcome message, as shown below, once it adds the server
successfully.
Conclusion
In this article, we configured Domain Controller, Active Directory and DNS
in a virtual machine. Later, we configured Reverse lookup zones, domain
admin account, local admin account and added the servers in the domain for
SQL Server Always On availability group.
SQL Server Installation
Checklist for Domain Controller
       Change network adapter to bridged
       Disable windows firewall
       Enter IP address: 192.168.0.50
       Restart VM
       Change computer name DC
       Create domain (SQL)
       Start server from domain user clusteradmin account
       Disable server manager popup window
       Turn of Internet Explorer (IE) popup
       Set the option for enabling sharing of folders
Checklist for server 1
       Make sure the DC is powered on
       Change network adapter to bridged
       Disable windows firewall
       Enter IP address: 192.168.0.51
       Restart VM
       Change computer name Server1
       Join domain (SQL)
       Start server from domain user clusteradmin account
       Disable server manager popup window
       Turn of Internet Explorer (IE) popup
       Set the option for enabling sharing of folders
Summary of step:
      Created 4 virtual machine on free VMware software
      One Domain Controller (DC)
        Three virtual Servers (Server1, Server2, Server3)
        Configured the virtual servers (changed sever name, IE off, disable
        firewall, IP address, etc.)
        Joined SQL domain
        Installed SQL Servers 2016 on each virtual server – except the DC
        Updated the free VMware workstation to 30 day trial version to
        manage easier with tabs
        Uploaded the four VMs from the path of VMs
        Then scroll down and look for Developer edition, click Download
        Now, and save the file. The developer edition is a full version and
        comes packaged with a fully featured free edition specially designed
        for developers.
Once the download is successful, go to the downloaded folder and
look for the SQLServer2017-SSEI-Dev.exe file. Double click on that
file, and you should see Getting things ready, which means
everything is going smoothly.
Then it will open the SQL Server 2017 setup window in which, by
default, the checkbox with specify a free edition to Developer would
be selected. All you need to do is click next and accept the license
terms and again click on next. It will install the setup files and move
to the next step, which is install rules. You might get a warning
of windows firewall, which you can safely ignore.
Then you will be directed to the Feature Selection setup in which you
will find a lot of options to select from, like instance features,
database engine services, machine learning services, etc. You need to
select the Database engine services and click next.
Disadvantages are:
        Need to manage all the databases separately
        There isn’t possibility for an automatic failover; must be manual
        And secondary database isn’t fully readable while the restore process
        is running
Operating modes:
There are two available modes and they are related to the state in which the
secondary log shipped SQL Server database will be:
        Standby mode – the database is available for querying and users can
        access it, but in read-only mode
        Restore mode – the database is not accessible
STEP BY STEPS:
        Create a domain user account for log shipping
        Create a Sql login for log shipping with sysadmin permission
        Create a domain service account for Sql Server and Sql agent
        Create a network share folder for the backups on primary with
        permission to Sql agent (read)
        Create a network share folder for the copy/restore on secondary with
        permission to Sql agent (read and write)
        Find recovery mode of log shipping database
        Take full back up of the primary database to network share
        Restore full and log backup on secondary server with no recovery
        option (restoring state)
        Set up log shipping via gui on primary database
        Copy log shipping via gui in secondary server
        Restore log shipping via gui in secondary server
        Check the log shipping process with data
Permissions
Must have sysadmin rights on the server
Step 2
On the primary server, right click on the database in SSMS and select
Properties. Then select the Transaction Log Shipping Page. Check
the “Enable this as primary database in a log shipping
configuration” check box.
Step 3
The next step is to configure and schedule a transaction log backup. Click
on Backup Settings… to do this.
If you are creating backups on a network share enter the network path or for
the local machine you can specify the local folder path. The backup
compression feature was introduced in SQL Server 2008 edition. While
configuring log shipping, we can control the backup compression behavior of
log backups by specifying the compression option. When this step is
completed it will create the backup job on the Primary Server.
Step 4
In this step we will configure the secondary instance and database. Click on
the Add… button to configure the Secondary Server instance and database.
You can add multiple servers if you want to setup one to many server log-
shipping.
On the primary server, right click on the database in SSMS and select
Properties. Then select the Transaction Log Shipping Page. Check
the “Enable this as primary database in a log shipping
configuration” check box.
Step 3
The next step is to configure and schedule a transaction log backup. Click
on Backup Settings… to do this.
If you are creating backups on a network share enter the network path or for
the local machine you can specify the local folder path. The backup
compression feature was introduced in SQL Server 2008 edition. While
configuring log shipping, we can control the backup compression behavior of
log backups by specifying the compression option. When this step is
completed it will create the backup job on the Primary Server.
Step 4
In this step we will configure the secondary instance and database. Click on
the Add… button to configure the Secondary Server instance and database.
You can add multiple servers if you want to setup one to many server log-
shipping.https://e85f02d7c3dad118af06f228d566e301.safeframe.googlesyndication.com/s
0-37/html/container.html
When you click the Add… button it will take you to the below screen where
you have to configure the Secondary Server and database. Click on
the Connect… button to connect to the secondary server. Once you connect
to the secondary server you can access the three tabs as shown below.
Click on Settings… button which will take you to the “Log Shipping
Monitor Settings” screen. Click on Connect … button to setup a monitor
server. Monitoring can be done from the source server, target server or a
separate SQL Server instance. We can configure alerts on source / destination
server if respective jobs fail. Lastly we can also configure how long job
history records are retained in the MSDB database. Please note that you
cannot add a monitor instance once log shipping is configured.
Step 6
Click on the OK button to finish the Log Shipping configuration and it will
show you the below screen.
Steps:
Find the last back up, copy and restore files
On primary:
Run the following script to find last backup made:
On secondary:
Run the following script to find last copy and restore made:
   Use msdb
   Go
   Select secondary_server, secondary_database, last_copied_file,
   last_restored_date ,last_restored_file,*
   From log_shipping_monitor_secondary
From log_shipping_monitor_secondary
         run the last tail tlog backup of primary
         run the copy and restore jobs on secondary
         disable all three jobs
         run tlog back up on primary with no recovery
         copy paste the tail tlog backup from primary to secondary folder
         restore secondary database with the tail backup with recovery mode
RESTORE LOG [Test] FROM DISK = N'C:\d\Tail.trn'
WITH NORECOVERY
GO
  --6 start the log shipping process from start making this the primary,
  and the original primary the secondary
USE [master]
RESTORE DATABASE [Orphans]
FROM DISK = N'\\SERVER2\D\ORPHANS.BAK' WITH FILE = 1,
MOVE N'Orphans'
TO N'C:\Program Files\Microsoft SQL
Server\MSSQL11.MSSQLSERVER\MSSQL\DATA\Orphans.mdf',
MOVE N'Orphans_log'
TO N'C:\Program Files\Microsoft SQL
Server\MSSQL11.MSSQLSERVER\MSSQL\DATA\Orphans_log.ldf',
NOUNLOAD,
REPLACE,
STATS = 5
GO
--STEP 7.
--RUN THIS ON SERVER2
--RUN THIS ON SERVER2
--FIND SPECIFICALLY SANDY'S SID
USE ORPHANS
GO
Select LOGINNAME, DBNAME, SID--<< FIND LOGINS SID FOR
(SANDY)
from sys.syslogins WHERE loginname = 'SANDY'
ORDER BY 1 DESC
Select NAME,sid --<< FIND SID FOR SANDY
from sys.sysusers WHERE NAME = 'SANDY'
ORDER BY 1 DESC
USE MASTER
GO
SELECT name as SQL_LogIn,SID as SQL_SID FROM sys.syslogins
WHERE [name] = 'sandy'
GO
USE Orphans
GO
SELECT name DataBase_User,SID as Database_SID FROM sysusers
WHERE [name] = 'sandy'
GO
  --STEP 10
  --To resolve an orphaned user, resync the SID of the user to map to the
  login
  USE Orphans
  GO
  EXEC sp_change_users_login 'update_one', 'SANDY', 'SANDY'
  --STEP 11. VERIFY
  USE ORPHANS
  GO
  Select LOGINNAME, DBNAME, SID--<< FIND LOGINS SID FOR
  (SANDY)
  from sys.syslogins WHERE loginname = 'SANDY'
  ORDER BY 1 DESC
  Select NAME,sid --<< FIND SID FOR SANDY
  from sys.sysusers WHERE NAME = 'SANDY'
  ORDER BY 1 DESC
  Exec sp_change_users_login ‘autofix’, ‘tom’
  Select * from sys.syslogins
  Select * from sys.sysusers
exec sp_help_revlogin
   --EXECUTE ON PRIMARY
   Use master
   Go
   sp_help_log_shipping_monitor
   Use master
  Go
  sp_help_log_shipping_monitor_primary
  @primary_server = 'server1', --<< PROVIDE THE PARAMETER
  SERVER NAME
  @primary_database = 'test' --<< PROVIDE THE PARAMETER
  DATABASE NAME
  SELECT *
  FROM msdb.dbo.sysjobs
  WHERE category_id = 6
  SELECT *
  FROM [msdb].[dbo].[sysjobhistory]
  WHERE [message] like '%Operating system error%'
  SELECT *
  FROM [msdb].[dbo].[log_shipping_monitor_error_detail]
  WHERE [message] like '%Operating system error%'
  SELECT *
  FROM msdb.dbo.sysjobs
  WHERE category_id = 6
  SELECT *
  FROM [msdb].[dbo].[sysjobhistory]
  WHERE [message] like '%Operating system error%'
  SELECT *
  FROM [msdb].[dbo].[log_shipping_monitor_error_detail]
  WHERE [message] like '%Operating system error%'
Database Mirroring
What is database mirroring?
Database mirroring is process having a redundant copy of a single database at
another location to ensure continuous data availability in case of a disaster on
the principal database.
Database mirroring ensures that one viable copy of a database will always
remain accessible during disaster recovery or down time needed for the
principal server
The principal server is the source server and the mirror is the destination
server
There are two types of operation modes when using database mirroring:
Synchronous operation mode:
This is used when very real time accuracy is required; which means that the
system must immediately copy every change in the principal’s content to the
mirror and vice-versa (this is referred to as a hot standby)
Asynchronous operation mode:
This is used when the content is not fully synchronized, and thus may result
in some data loss (this is referred to as a warm standby)
Advantages:
        Relatively easy to set up
        Database mirroring is an automatic failover process
       All application connection can be redirected automatically with
       proper configuration
       There will not be data transfer latency (synchronous mode)
Disadvantages:
       Can only have one to one relationship with principal and mirror
       Cannot be used for reporting solution (mirror in restoring state)
       Mirroring supports only Full Recovery (not bulk or simple mode)
  --1 primary
  alter database mirror
  set recovery full
  go
  --2 primary
  backup database mirror
  to disk = 'c:\s\full.bak'
  go
  --3 primary
  backup log mirror
  to disk = 'c:\s\tlog.trn'
  go
  --4 primary
  create endpoint endpoint_principal
  state = started
  as tcp (listener_port = 5022)
  for database_mirroring (role = partner)
  go
  --5 primary
alter database mirror
set partner = 'tcp://server1:5022'
go
--6 mirror
USE [master]
RESTORE DATABASE [mirror]
FROM DISK = N'C:\d\full.bak'
WITH FILE = 1,
MOVE N'mirror'
TO N'C:\Program Files\Microsoft SQL
Server\MSSQL11.MSSQLSERVER\MSSQL\DATA\mirror.mdf',
MOVE N'mirror_log'
TO N'C:\Program Files\Microsoft SQL
Server\MSSQL11.MSSQLSERVER\MSSQL\DATA\mirror_log.ldf',
NORECOVERY,
NOUNLOAD,
STATS = 5
GO
--log mirror
RESTORE LOG [mirror]
FROM DISK = N'C:\d\tlog.trn'
WITH FILE = 1,
NORECOVERY,
NOUNLOAD,
STATS = 10
GO
--7 mirror
create endpoint endpoint_mirror
state = started
as tcp (listener_port = 5023)
for database_mirroring (role = partner)
go
--8 mirror
alter database mirror
set partner = 'tcp://server2:5023'
go
--ENABLE SQLCMD MODE TO SWITCH BETWEEN DIFFERENT
SQL SERVERS WTIH A A QUERY PANE
--QUERY - SQLMCMD MODE
--:CONNECT SQLSERVER1
--Or for a non-default instance
--:CONNECT DESKTOP-QMOOH4U\DEV
--DROP DATABASE MIRROR
USE MASTER
GO
CREATE DATABASE MIRROR
GO
USE MIRROR
GO
CREATE TABLE CHOCOLATES
(NAME VARCHAR (25))
INSERT INTO CHOCOLATES
VALUES ('GODIVA'),('MARS'),('HERSHYS'),('DOVE'),('KITKAT')
SELECT * FROM CHOCOLATES
BACKUP DATABASE [MIRROR] TO DISK =
N'C:\Mirror\CHOC.BAK' WITH iNIT
BACKUP LOG [MIRROR] TO DISK = N'C:\Mirror\CHOC.TRN'
WITH iNIT
USE [master]
GO
--CHANGE CONNECTIONS TO DEV SERVER: THEN RUN
:CONNECT DESKTOP-QMOOH4U\DEV
RESTORE DATABASE [MIRROR]
FROM DISK = N'C:\Mirror\CHOC.BAK'
WITH FILE = 1,
MOVE N'MIRROR'
TO N'C:\Program Files\Microsoft SQL
Server\MSSQL12.DEV\MSSQL\DATA\MIRROR.mdf',
MOVE N'MIRROR_log'
TO N'C:\Program Files\Microsoft SQL
Server\MSSQL12.DEV\MSSQL\DATA\MIRROR_log.ldf',
NORECOVERY
GO
RESTORE LOG [MIRROR]
FROM DISK = N'C:\Mirror\CHOC.TRN'
WITH FILE = 1,
NORECOVERY
GO
--9 primary
alter database mirror
set partner = 'tcp://server1:5022'
go
--SQL SCRIPTS FOR MONITORING AND INVESTIGATING
DATABASE MIRRORING:
--INFORMATION ABOUT DATABASE MIRRORING
select DB_NAME(database_id)
dbname,mirroring_state_desc,mirroring_role_desc,
mirroring_safety_level_desc,mirroring_safety_sequence
mirroring_partner_name,mirroring_partner_instance,
mirroring_witness_state,mirroring_witness_state_desc,
mirroring_failover_lsn,mirroring_connection_timeout,mirroring_redo_queue,
mirroring_end_of_log_lsn,mirroring_replication_lsn,*
from master.sys.database_mirroring
where mirroring_state is not null
--INFORMATION ABOUT DATABASE MIRRORING
CONNECTIONS
select
state_desc,connect_time,login_time,authentication_method,principal_name,
remote_user_name,last_activity_time,is_accept,login_state_desc,
receives_posted,sends_posted,total_bytes_sent,total_bytes_received,
total_sends,total_receives,*
from sys.dm_db_mirroring_connections
--INFORMATION ABOUT DATABASE MIRRORING
ENDPOINTS
select name,endpoint_id,protocol_desc,type_desc,state_desc,role_desc,
connection_auth_desc,*
from sys.database_mirroring_endpoints
--SCRIPT TO INDICATE WHICH DATABASE HAS BEEN
MIRRORED:
SELECT DB_NAME(database_id) AS mirrored
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
ORDER BY DB_NAME(database_id);
--SCRIPT TO INDICATE WHICH DATABASE IS IN SYNC MODE
SELECT DB_NAME(database_id) AS synchronous_mode
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
--AND mirroring_role_desc = 'PRINCIPAL'
--AND mirroring_role_desc = 'MIRROR'
AND mirroring_safety_level_desc = 'FULL'
ORDER BY DB_NAME(database_id);
--SCRIPT TO INDICATE WHICH DATABASE IS IN ASYNC
MODE
SELECT DB_NAME(database_id) AS asynchronous_mode
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
--AND mirroring_role_desc = 'PRINCIPAL'
--AND mirroring_role_desc = 'MIRROR'
AND mirroring_safety_level_desc = 'OFF'
ORDER BY DB_NAME(database_id);
--SCRIPT TO INDICATE WHICH DATABASE HAS FULLT
SYNCHRONIZED OR NOT
SELECT
DB_NAME(database_id) AS fully_synchronized
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
--AND mirroring_role_desc = 'PRINCIPAL'
--AND mirroring_role_desc = 'MIRROR'
AND mirroring_state_desc = 'SYNCHRONIZED'
ORDER BY DB_NAME(database_id);
--SETTING THE ASYNCHRONOUS MODE OFF AND ON
SELECT
 'ALTER DATABASE [' + DB_NAME(database_id) + '] SET
PARTNER SAFETY OFF;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has been set to
asynchronous mirroring mode.'';'
  AS
command_to_set_mirrored_database_to_use_synchronous_mirroring_mode
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_safety_level_desc = 'FULL'
ORDER BY DB_NAME(database_id);
SELECT
  'ALTER DATABASE [' + DB_NAME(database_id) + '] SET
PARTNER SAFETY FULL;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has been set to
synchronous mirroring mode.'';'
  AS
command_to_set_mirrored_database_to_use_synchronous_mirroring_mode
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_safety_level_desc = 'OFF'
ORDER BY DB_NAME(database_id);
--TO SUSPEND THE DATABASE MIRRORING
SELECT
'ALTER DATABASE [' + DB_NAME( database_id ) + '] SET
PARTNER SUSPEND;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has had mirroring
paused.'';'
  AS command_to_pause_mirroring_for_the_mirrored_database
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_state_desc <> 'SUSPENDED'
ORDER BY DB_NAME(database_id);
--TO RESUME THE DATABASE MIRRORING
SELECT
'ALTER DATABASE [' + DB_NAME( database_id ) + '] SET
PARTNER RESUME;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has had mirroring
resumed.'';'
  AS command_to_resume_mirroring_for_the_mirrored_database
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_state_desc = 'SUSPENDED'
ORDER BY DB_NAME(database_id);
--SETTING TO FAILOVER DATABASE MIRRORING
SELECT
'ALTER DATABASE [' + DB_NAME(database_id) + '] SET
PARTNER FAILOVER;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has been been manually
failed over.'';'
  AS command_to_manually_failover_the_mirrored_database
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_safety_level_desc = 'FULL'
AND mirroring_state_desc = 'SYNCHRONIZED'
ORDER BY DB_NAME(database_id);
-- HISTORY OF RESULTS
Exec msdb..sp_dbmmonitorresults 'db',1,0
--use on mirror db to stop miroring and drop db
alter database mirror set partner off
  restore database mirror
  drop database MIRROR
 There will be data transfer latency.   There will not be data transfer
 >1min.                                  latency.
11. If all is configured correctly, the database mirroring session is now active.
SQL Server Database Mirroring Monitoring
The GUI monitoring tool for database monitoring
While I recommend you pay attention to all properties of the GUI, a few we
should take in consideration are the following:
       An increase in the unsent log indicates that the size of the transaction
       log has been generated on the principal server but has not yet been
       sent to the mirror server. This increase in number will indicate that
       during a failover, this about of data will be ‘lost’
       The unrestored log size shows how much transaction log has not
       been replayed on the mirror database. The larger the unrestored
       log, the longer a failover will take
SSMS : Connect to Principle Server > Select Database > Right Click , Go to
Task & Select Launch Database Mirroring Monitor
SELECT
 'ALTER DATABASE [' + DB_NAME(database_id) + '] SET
PARTNER SAFETY OFF;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has been set to
asynchronous mirroring mode.'';'
  AS
command_to_set_mirrored_database_to_use_synchronous_mirroring_mode
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_safety_level_desc = 'FULL'
ORDER BY DB_NAME(database_id);
SELECT
  'ALTER DATABASE [' + DB_NAME(database_id) + '] SET
PARTNER SAFETY FULL;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has been set to
synchronous mirroring mode.'';'
  AS
command_to_set_mirrored_database_to_use_synchronous_mirroring_mode
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_safety_level_desc = 'OFF'
ORDER BY DB_NAME(database_id);
--TO SUSPEND THE DATABASE MIRRORING
SELECT
'ALTER DATABASE [' + DB_NAME( database_id ) + '] SET
PARTNER SUSPEND;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has had mirroring
paused.'';'
  AS command_to_pause_mirroring_for_the_mirrored_database
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_state_desc <> 'SUSPENDED'
ORDER BY DB_NAME(database_id);
--TO RESUME THE DATABASE MIRRORING
SELECT
'ALTER DATABASE [' + DB_NAME( database_id ) + '] SET
PARTNER RESUME;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has had mirroring
resumed.'';'
  AS command_to_resume_mirroring_for_the_mirrored_database
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_state_desc = 'SUSPENDED'
ORDER BY DB_NAME(database_id);
SELECT
'ALTER DATABASE [' + DB_NAME(database_id) + '] SET
PARTNER FAILOVER;'
+ ' PRINT ''[' + DB_NAME(database_id) + '] has been been manually
failed over.'';'
  AS command_to_manually_failover_the_mirrored_database
FROM master.sys.database_mirroring
WHERE 1=1
AND mirroring_guid IS NOT NULL
AND mirroring_role_desc = 'PRINCIPAL'
AND mirroring_safety_level_desc = 'FULL'
AND mirroring_state_desc = 'SYNCHRONIZED'
ORDER BY DB_NAME(database_id);
-- HISTORY OF RESULTS
Exec msdb..sp_dbmmonitorresults 'prod',1,0
--use on mirror db to stop miroring and drop db
alter database mirror set partner off
restore database mirror
drop database MIRROR
USE [master]
GO
DROP ENDPOINT [endpoint_mirror]
GO
SQL Server Replication
What is replication?
Replication is a set of technologies for copying and distributing data and
database objects from one database to another and then synchronizing
between databases to maintain consistency. Using replication, you can
distribute data to different locations. Unlike other methods of high
availability, it doesn’t distribute entire database, but only distributes
some part of database like tables, views or stored procedures
There are four types of replication
       Transactional replication
       Peer to peer replication
       Snapshot replication
       Merge replication
    1. Transactional replication
             It replicates each transaction for the article being published
             When a transaction is written to the transaction log, the Log
             Reader Agent reads it from the transaction log and writes it to
             the distribution database and then to the subscriber
             Only committed transactions are replicated to ensure data
             consistency
             Transactional replication is widely applied where high latency
             is not allowed, such as an OLTP system for a bank or a stock
             trading firm, because you always need real-time updates of
             cash or stocks
Distribution Agent
       The Distribution Agent is used with snapshot replication and
       transactional replication.
       It applies the initial snapshot to the Subscriber and moves
       transactions held in the distribution database to Subscribers.
       The Distribution Agent runs at either the Distributor for push
       subscriptions or at the Subscriber for pull subscriptions
2. Peer-to-Peer Replication
       Peer to Peer replication is the process of having multiple servers that
       act as both publisher and subscriber with its own distributor. As
       such, when you update, insert or delete records, they
       are synchronized in real time with each other. The topology
       (architecture) of the design involves a master server from which
       other servers connect to the master database.
       These server are referred to as nodes.
       If any one of the locations is down, the other locations can still stay
       synchronized, because each node acts as a publisher and a subscriber.
       Backup and restore database to each server in replication is needed
       Enable distributor in each server that is in replication
  --sp_removedbreplication
  use master
  go
  create database PTP
  go
  use ptp
  go
  create table Computers
  (Computerid int primary key,
  Name varchar (20))
  use PTP
  go
  insert into Computers
  values (1,'Sony'),(2,'HP'),(3,'Dell'),(4,'Apple')
  select * from Computers
  use PTP
  go
  insert into Computers
  values (6,'Lenova')
3. Snapshot replication:
Snapshot Agent
       Used with all types of replication
       It prepares the schema of published tables and objects
       Stores the snapshot files
       Records information about synchronization in the distribution
       database
       Snapshot Agent runs at the Distributor
Push or Pull
       Push – a push subscription pushes data from publisher to the
       subscriber
       Pull – a pull subscription requests changes from the Publisher. This
       allows the subscriber to pull data as needed. This is useful for
       disconnected machines such as notebook computers that are not
       always connected
Distribution Agent
       The Distribution Agent is used with snapshot replication and
       transactional replication.
       It applies the initial snapshot to the Subscriber and moves
       transactions held in the distribution database to Subscribers.
       The Distribution Agent runs at either the Distributor for push
       subscriptions or at the Subscriber for pull subscriptions
4. Merge replication
       This is the most complex types of replication which allows changes
       to happen at both the publisher and subscriber.
       As the name implies, changes are merged to keep data consistency
       and a uniform set of data.
       Just like transactional replication, an initial synchronization is done
       by applying snapshot.
       When a transaction occurs at the Publisher or Subscriber, the change
       is written to change tracking tables. The Merge Agent checks these
       tracking tables and sends the transaction to the distribution database
       where it gets propagated.
       The merge agent has the capability of resolving conflicts that occur
       during data synchronization.
       An example of using merge replication can be a store with many
       branches where products may be centrally stored in inventory.
       As the overall inventory is reduced it is propagated to the other stores
       to keep the databases synchronized.
Some applications also require that changes flow from the Subscriber back to
the Publisher. merge replication provide options for these types of
applications.
       Merge Agent – The Merge Agent is used with merge replication. It
       applies the initial snapshot to the Subscriber and moves and
       reconciles incremental data changes that occur. Each merge
       subscription has its own Merge Agent that connects to both the
       Publisher and the Subscriber and updates both. The Merge Agent
       runs at either the Distributor for push subscriptions or the Subscriber
       for pull subscriptions.
       Queue Reader Agent – The Queue Reader Agent is used with
       transactional replication with the queued updating option. The agent
       runs at the Distributor and moves changes made at the Subscriber
       back to the Publisher. Unlike the Distribution Agent and the Merge
       Agent, only one instance of the Queue Reader Agent exists to service
       all Publishers and publications for a given distribution database.
Pre-requisites
    1. The account must at minimum be a member of the db_owner fixed
       database role in the SQL replication Publisher, Distributor and
       Subscriber databases
    2. For securing the replication snapshot folder using a Snapshot Agent,
       the account must have read and write or modify permission on the
       replication snapshot share
    3. At least one database should have an article and must possess
       Primary Key; a basic rule that every article should have a Primary
       Key is considered as best candidate for Transactional SQL
       Replication. The primary key is used to maintain uniqueness of
       records.
    4. Scheduling the agent and jobs
    5. Sufficient network bandwidth
    6. Enough disk space for the databases being published; we need to
       make sure that we have enough space available for the SQL
       transaction log for the published database
Initial setup
The exercises demonstrate how to configure SQL Replication to copy a few
tables, stored procedures and views from AdventureWorks2016 sample
database to another database. To keep this setup simple, we use SQL Server
Management Studio. The transactional replication setup has a single SQL
Server instance to play the roles of SQL replication Publisher, Distributor,
and another SQL Server instance play the role of the SQL replication
Subscriber.
Getting Started
To first set up transaction SQL replication, you must configure the SQL
replication Distributor and create a SQL replication Publication, replication
snapshot folder and a SQL replication Subscription.
Configure Distributor
The following steps walk you through the process of creating the SQL
replication Distributor:
    1. Open SSMS and connect to the SQL Server instance
   10.         In the complete the wizard page, Review the settings and
   configuration options, and then click Finish to enable the Distributor
Configure Publisher
Once you’ve configured the Distributor, you can create a publication. Let’s
follow the steps:
    1. In Object Explorer, locate the Replication folder, right-click Local
       Publication, and then click New Publication
    2. Next, the New Publication Wizard appears and outlines the general
       information about creating Publication.
3. In the Publication Database page, select the Adventureworks2016
   database and click Next.
6. Once you’re done with the object selection, Check the Show only
   checked articles in the list optionto list candidates of Publication. In
   this case, 2 tables, 1 Stored Procedures are selected.
    7. Next, in the Filter Table Rows page, define filters that should be
       applied to your articles. Let us
    8. In the Snapshot Agent page specify when to run the Snapshot Agent.
       It can be run immediately or it can schedule to run at a later time. In
       this case, Create a snapshot immediately is used.
9. Now, in the Agent Security page, specify the account to use to run
   the Snapshot Agent using Security settings…
   10.        Click Ok
11.         In the complete Wizard Actions page, you’ve two
options. You can create the Publication immediately or save the
configuration in the script file to run at a later time.
14.         Now, you can see that Publication is created under the
local publication folder
Configure Subscriber The final step in setting up replication is to create
the subscription. Let us walk-through the steps:
    1. In Object Explorer, expand the Replication folder, right-click Local
       Subscriptions, and then click New Subscriptions
.
8. On the Distribution Agent Security page, select the ellipsis (…)
   button. Type in the process account details and Click Ok.
9. Select Finish accepting the default values on the remaining pages and
   completing the wizard.
   10.                On the complete the wizard page, you can see the
   detailed summary of the newly created subscription. Click Finish
11.                 In the Creating Subscription(s)…page should
show that the process has been successful or not.
Prerequisites:
       Must install Window Failover cluster Services on each node
       Must have a shared folder for each server to access for backups
       Must take a full backup of the primary database(s)
       All server must be in full recovery mode
       Set up cluster
Maximum
number of
               unlimited   1 Mirror        4 Secondaries
secondary
databases
Requires
                                           Yes, however the SQL Servers
Windows        No          No
                                           can be stand-alone
Clustering
                           Yes,
                           requires
Automatic                  witness
               Manual                      Yes, one
failover                   server and
                           high-safety
                           mode
                           No, each
Groups of                  database is     Multiple databases can be
               No
databases                  configured      grouped to failover together
                           separately
                           Only
Can be used    Stand By    against a
for            mode        database        Yes
reporting     (limited)   snapshot of
                          the mirror
Offload
              No          No            Yes
backups
Active
database is   Primary     Principal     Primary
called
                          Requires
                                        Redirection handled by
Connection                special
                                        Windows Clustering with a
to active     manual      connection
                                        virtual name. Special
database                  string for
                                        connection string not required
                          failover
While you are still in SQL Server Configuration Manager, right click on
SQL Server Services to open the Properties dialog box. Navigate to the
AlwaysOn High Availability tab, and select Enable AlwaysOn Availability
Groups checkbox.
Restart the SQL Server Service after making these changes.
Do these steps on all of your replicas.
4. Configure Logins and Replicas
If it isn’t there already, add your SQL Service account (which should be a
domain account – not the local machine service account) as a SQL login
through SQL Management Studio (SSMS).
Add your SQL Service account to the Administrators group on each replica
(via Computer Management)
Give connect permissions to the SQL Service account through SSMS: Right
click on the SQL Service login to open the Properties dialog box. Navigate
to the Securables page, and make sure the Connect SQL Grant box is
checked. You will do this on every replica.
Make sure all your replicas Allow Remote Connections. You can do this
through SSMS in the instance Properties, or by using sp_configure.
First, you will specify your AlwaysOn group name. Name it something
descriptive and unambiguous:
Next, you will select the databases you want to include in your AlwaysOn
group. All of the databases in your instance will show up in this list…you
don’t have to include all of them in your group… select only the ones to be
included in the AlwaysOn group.
Also, next to each database is a blue link that signifies whether your database
is ready to be included into your group or not. If the link does not say ‘Meets
prerequisites’, then you can click on the link to get a more in-depth
explanation of what you need to do.
Correct any discrepancies, and then select the databases to include in the
AlwaysOn group:
Next, is the Specify Replicas page where you will add the replicas to be
included in your AlwaysOn group. Add and connect the replicas by clicking
the Add Replica… button.
For each replica, you will need to specify whether you want Automatic or
Manual Failover, Synchronous or Asynchronous Data Replication, and what
type of Connections you will allow your end users to connect with.
On this Specify Replicas page, there are several tabs at the top. The second
tab is the Endpoints tab. On this tab verify that the port number is 5022. If
you have more than one instance on your server, you might need to create
another endpoint.
Next tab is the Backup Preferences tab. This is where you will choose
where you want your backups to occur, and how you prioritize which replica
will run your backups.
The last tab in the page is the Listener tab. Here you will select the Create
an availability group listener button.
Enter the DNS name, which is the name that will be used in your application
connection string.
Enter port number 1433.
And enter the IP address for your listener. This should be an unused IP
address on your network.
Next page in the wizard is the Select Initial Data Synchronization page.
Here is where you will join your databases to the AlwaysOn group.
The Full option is the Microsoft default option, and is the one that uses the
File Share. The other two options (Join and Skip) are fine too, especially if
you have large databases. With these other two options, you will restore the
databases yourself, to each secondary replica. But this example uses the Full
option, so you will browse to and select the File Share created earlier. And
remember the SQL Service account and all replicas must have read/write
permission to the File Share.
Next, ensure that your Validation checks return successful results. If you get
any errors, you need to stop and correct these before proceeding.
In the Summary page, verify that all your configuration settings are correct,
and click Finish.
The Results page will show the progress of the installation. Verify that all
tasks have completed successfully. Because there was no quorum set up
while creating the WSFC earlier, we are seeing a warning message
here….this will not cause the installation to fail.
After the results are complete, and everything has finished successfully, you
can now see the AlwaysOn Availability Group created in SSMS:
The Availability Group Name: AGroup_Dev
All the Replicas, and whether they are primary or secondary
All the Databases included in the AlwaysOn group
And the Listener created for the group.
This AlwaysOn Group will also be visible on all of the replicas as well.
SQL Server Backup Database on Secondary Replicas
Backup Types Supported on Secondary Replicas
        BACKUP DATABASE supports only copy-only full backups of
        databases, files, or filegroups when it is executed on secondary
        replicas. Note that copy-only backups do not impact the log chain or
        clear the differential bitmap.
        Differential backups are not supported on secondary replicas.
        BACKUP LOG supports only regular log backups (the
        COPY_ONLY option is not supported for log backups on
        secondary replicas).
A consistent log chain is ensured across log backups taken on any of the
replicas (primary or secondary), irrespective of their availability mode
(synchronous-commit or asynchronous-commit).
        To back up a secondary database, a secondary replica must be able to
        communicate with the primary replica and must be
        SYNCHRONIZED or SYNCHRONIZING.
   use rep7
   go
   create table numbers
   (num int)
   insert into numbers
   values (1000)
   go 1000
   --on secondary replica. Run this command or create a job
   BACKUP DATABASE [rep7]
   TO DISK = N'C:\backup 777\fullbackup777.bak'
   WITH COPY_ONLY, --<< backups on secondary replicas occurs
   only if you set this option (must be copy_only)
   NOFORMAT,
   NOINIT,
   NAME = N'rep7-Full Database Backup'
   GO
Going through the Availability Groups options, we can notice that we have
distinct backup options:
The available options can be confusing, if you never tried them before. So
let’s take a look on each one…
Prefer Secondary
This option is very conceptual. Basically, you can run the backup command
from any replica!
“Automated backups for the availability groups should occur on secondary
replica (…)”
We can easily test this by running a backup command from the Primary
replica:
The result:
10 percent processed.
20 percent processed.
30 percent processed.
40 percent processed.
50 percent processed.
60 percent processed.
70 percent processed.
80 percent processed.
90 percent processed.
Processed 24320 pages for database ‘AdventureWorks2016’, file
‘AdventureWorks2014_Data’ on file 2.
100 percent processed.
Processed 25 pages for database ‘AdventureWorks2016’, file
‘AdventureWorks2014_Log’ on file 2.
BACKUP DATABASE successfully processed 24345 pages in 39.265
seconds (4.843 MB/sec).
Yes, it worked! As it would work from any other replica, being a secondary
or not! But, what is the reason behind this? As I said before, the keywords are
“Automated” and “should”.
Translating this, we can understand that “Should” would mean “yes, this
would be good to run in a secondary. If not, that’s ok…”.
What about the “Automated” keyword? This case is different. SQL Server
assumes that when you run a BACKUP DATABASE command, like we did,
you know what you are doing. Basically, you set the backup rules, so if you
are doing the backup on the primary replica, you assume the risk. In other
hand, an automated backup would be done by using the beloved SQL Server
Maintenance Plan. Let’s analyse this:
I created a simple Maintenance Plan, with a backup task including all the
databases in the instance. You can notice that there’s a warning in the bottom
of the window.
“This backup type is not supported on a secondary replica and this task will
fail if the task runs on a secondary replica.”
Why this? Because we are including databases that are part of an Availability
Group in the plan. The problem here is simple: SQL Server does not support
a regular backup to be made on Secondary replicas. In this case we are
dealing with the primary one, but in case of a failover, the backup job is
going to fail!
How can we solve this? We need to go to the tab “Options” and tick the
“Copy-only backup” option:
Going back to the “Automated” keyword, if we press the “View T-SQL”
button, we will be able to explore the code generated to execute the backups.
Based on my lab, I’ll get the following, for the AdventureWorks2016
database:
   DECLARE
  @preferredReplica INT SET
  @preferredReplica = (SELECT [master].sys.Fn_hadr_backup_is_preferred_replica(‘Ad
  IF ( @preferredReplica = 1 ) BEGIN    BACKUP DATABASE [AdventureWorks20
  Files\Microsoft SQL
  Server\MSSQL13.MSSQLSERVER\MSSQL\Backup\AdventureWorks2016_backup_2
  copy_only, noformat, noinit, NAME = N’AdventureWorks2016_backup_2015_11_2
  rewind, nounload, stats = 10
  END
Code starts with the declaration and value attribution of a variable called
“preferredReplica”:
  DECLARE
 @preferredReplica INT SET
 @preferredReplica = (SELECT
 [master].sys.Fn_hadr_backup_is_preferred_replica(‘AdventureWorks2016’))
By running the select, present in code, I will get the following :
[master].sys.Fn_hadr_backup_is_preferred_replica(‘<DATABASE_NAME>’)
Here is the point where we prove that the “Automation” is just conceptual.
Basically, the SQL Server Maintenance Plan, is AG aware, and it has the
backup task ready to interpret the backup options of the Availability Group.
Knowing this, what happed if we run a simple (without IFs) BACKUP
DATABASE command in the primary? It’s going to work! So, don’t be
“eluded” thinking that set the backup preferences is enough… There are more
things to do!
One of my suggestions is use the heavily tested and community approved
scripts from Ola Hallegren, which are Availability Groups aware, and very
smart and flexible!
Secondary Only
Continuing with the analysis, we can now check the second option
“Secondary Only”, which the description states that the backups MUST occur
on a Secondary Replica. Let’s test it:
The output:
10 percent processed.
20 percent processed.
30 percent processed.
40 percent processed.
50 percent processed.
60 percent processed.
70 percent processed.
80 percent processed.
90 percent processed.
Processed 24320 pages for database ‘AdventureWorks2016’, file
‘AdventureWorks2014_Data’ on file 3.
100 percent processed.
Processed 27 pages for database ‘AdventureWorks2016’, file
‘AdventureWorks2014_Log’ on file 3.
BACKUP DATABASE successfully processed 24347 pages in 12.018
seconds (15.827 MB/sec).
Implementing Backup on Secondary Replicas (AlwaysOn Availability
Groups)
In this article I’d like to show how to set up the backup of this database on
servers holding the secondary database replicas.
We can configure the Backup Preferences setting as “Prefer secondary”
(right-click Availability Groups\RESOURCES-Group – Properties):
This setting assumes that a backup can be taken on any server holding a
secondary role at some point in time and given that any server can be
Primary – let’s say – on Monday and Secondary on Tuesday, I prefer having
my database backed up to a network share – it allows me to have an unbroken
chain of backup files regardles of the server wich currently executes the
backup job.
I start by running the Maintenance Plan Wizard on SQL1:
I’m not going to set a schedule in this test cause I will execute Maintenance
Plans manually, but in a production environment execution of the backup
tasks should be certainly scheduled.
I always use “Maintenance Cleanup Task” as the final step in such
maintenance plans but I failed trying to make it work on a network shared
folder, that’s why I don’t check it in here.
The account being used for running the Maintenance Plan (SQL Server Agent
service account by default as we’ve seen on the Select Plan
Properties window) must have the appropriate permissions for the shared
folder to create backup files in it. In this test I grant Test\sqlagent Full
Control permission (both for share and ntfs).
Now I must create exactly the same maintenance plan on the other server –
SQL2.
Now let’s execute the ALWAYS ON Group Backup maintenance plan on the
primary replica (SQL1):
Look! The plan completed successfully but there’re no .bak or .trn files in the
folder – that’s because according to the AlwaysON Backup
Preferences setting backups should be taken on secondary servers and only in
the absence of any secondary server backup can be taken on a primary server.
As both SQL1 (Primary) and SQL2(Secondary) are up and running the
backup job should not produce any backup files when run on SQL1.
Now I run the same plan on the secondary server – SQL2:
The result:
– both .bak and .trn files have been created.*
* Only .bak file should be created in this case since we haven’t
check “Backup Database(Transaction Log)”. I find difficulty in interpreting
this behaviour.
After recreating Maintenance Plans trn files get created only in case the
checkbox “Backup Database(Transaction Log)” is explicitly checked.
Then let’s make a test failover and repeat running the maintenance plan on
the new secondary server – SQL1:
As you see the new .bak and .trn files have been created in the
\\sql1\Netbackup folder – thus you can have all backup files in the single
location, regradles of which server the maintenance plan currently runs on.
                                                                                 THE END
Thank You for your time. I hope this book helped you, and gave you an in depth knowledge of SQL
Server 2019 High Availability. If you enjoyed reading this book do leave a good review and let others
benefit too!
Thanks Once Again….