The automated steps looks like this...
- Scheduled job "Create RG Snaps" runs...
- PWS Script "Run-RGSnaps.ps1" which make 4 parallel calls to...
- PWS Script "Create-RGSnaps.ps1" which save the snaps to... \\server\FOLDER\RedGate_Snaps\ in...
- SERVER$INSTANCE.DB naming format
There are 3 methods by which the PWS Script "Create-RGSnaps.ps1" can be run manually.
- One Database
- Open Powershell
- Navigate to scripts folder, issue a "sl C:\
\Script" (no quotes) command - To run against yourdb on SERVER1\INSTANCE1, for example, issue a "./Create-RGSnaps.ps1 SERVER1\INSTANCE1 yourdb" (no quotes) command. Output will look like the below...
SQL Compare Command Line V8.0.0.965==============================================================================
Copyright c Red Gate Software Ltd 1999-2009 Serial Number: 012-345-678901-2345 Creating database snapshot file '\\SERVER\FOLDER\RedGate_Snaps\SERVER1$INSTANCE1.yourdb .snp' from database 'yourdb ' on 'server1\instance1'...
OK
- One Instance
- Open Powershell
- Navigate to scripts folder, issue a "sl C:\
\Script" (no quotes) command - To run against SERVER1\INSTANCE1, for example, issue a "./Create-RGSnaps.ps1 SERVER1\INSTANCE1" (no quotes) command. Output will look the same as above...
- Multiple Instances
- Open powershell
- Navigate to scripts folder, issue a "sl C:\
\Script" (no quotes) command - To run against SERVER1\INSTANCE1 & SERVER2\INSTANCE2, for example, create a file named Instances.txt (or a name of your choice) in "C:\
\Script" that contains entries for SERVER1\INSTANCE1 & SERVER2\INSTANCE2 (one line per instance) - Issue a "Get-Content Instances.txt ForEach {./Create-RGSnaps $_}" (no quotes) command. Output will look the same as above...
The script that does the work "Create-RGSnaps.ps1" takes a required parameter of Instance & an optional parameter of Database. If a value for Database is passed the script calls the command line interface for SQL Compare with no further processing. If no value for Database is passed the script parses all DBs on the Instance that are active/accessable & calls SQL Compare for each DB. The code is below...
- param($Instance = $(throw 'Instance is required.'), $Database)
- $ErrorActionPreference = "SilentlyContinue"
- $Path = "\\SERVER\FOLDER\RedGate_Snaps\"
- $Query = "SELECT name FROM dbo.sysdatabases (NOLOCK) WHERE dbid > 4 AND name NOT IN ('distribution','DBADatabase') AND has_dbaccess(name) = 1"
- #########################
- function Get-SQLData
- {
- param($Instance = $(throw 'Instance is required.'), $DBName = $(throw 'Database is required.'), $Query = $(throw 'Query is required.'))
- Write-Verbose "Get-SqlData Instance:$Instance Database:$DBName Query:$Query"
- $ConnString = "Server=$Instance;Database=$DBName;Integrated Security=SSPI;"
- $DataAdapter = New-Object "System.Data.SqlClient.SqlDataAdapter" ($Query,$ConnString)
- $DataTable = New-Object "System.Data.DataTable"
- [void]$DataAdapter.fill($DataTable)
- $DataTable
- } #Get-SQLData
- #########################
- function Get-Databases
- {
- param($Instance = $(throw 'Instance is required.'))
- Get-SqlData $Instance 'master' $Query
- } #Get-Databases
- #########################
- function Invoke-SqlCompare
- {
- param($Instance = $(throw 'Instance is required.'), $Database = $(throw 'Database is required.'), $Path = $(throw 'Path for Snap files required.'))
- $InstanceStrng = ($Instance -replace('\\','`$')).ToUpper()
- $Command = '&"C:\Program Files (x86)\Red Gate\SQL Compare 8\sqlcompare.exe" /Server1:' + $Instance + ' /database1:"' + $Database + '" /makesnapshot:"' + $Path + $InstanceStrng + '.' + $Database + '.snp" /force'
- Invoke-Expression $Command
- } #Invoke-SqlCompare
- #########################
- #Main
- If ($Database)
- {Invoke-SQLCompare $Instance $Database $Path}
- Else
- {Get-Databases $Instance | ForEach {Invoke-SQLCompare $Instance $_.name $Path}}
The script that runs the Create script in parallel "Run-RGSnaps.ps1" was written by my boss, his blog is here http://chadwickmiller.spaces.live.com/default.aspx & was only modified by me to meet my needs. Since in Powershell 1.0 there's no way to spawn background processes the Run script launches new PWS threads unitil it reaches the number defined as it's max & then checks every 15 seconds for less than the max threads (excluding itself) & if less than the defined max, spawns a new thread. I take no credit for his excellent work, just want to include for completeness. The script takes no parameters, it has a built-in function to get a list of Prod Instances we have defined in a DB & pulls back the ones marked active & passes them to the create script to run in parallel.
I tested running serially and with 4 & 8 iterations in parallel with nothing else running on the host server. The best balance between performance & resource utilization was at 4 parallel iterations (your results will probably vary). Serially resource usage was minimal we saw 8% CPU, 447 DBs across 26 Instances completed in 56 minutes average; at 4 parallel iteration we saw 33% CPU & completion of the same DBs/Instances took 15 minutes average; at 4 parallel iteration we saw 85% CPU & completion of the same DBs/Instances took 12 minutes average. the server we're running this on has 4 single core procs, my suspicion is that you can run as many threads as you have processor cores. Code is below...
- $ScriptRoot = "C:\<yourdir>\Script\"
- $SQLPSXServer = "SERVER3\INSTANCE3"
- $SQLPSXDatabase = "DB3"
- $Query = "SELECT <column> FROM dbo.<table> (NOLOCK) WHERE IsEnabled = 'true'"
- $MaxThread = 4
- $ServerList = New-Object System.Collections.ArrayList
- #########################
- function Get-SQLData
- {
- param($Instance = $(throw 'Instance is required.'), $DBName = $(throw 'Database is required.'), $Query = $(throw 'Query is required.'))
- Write-Verbose "Get-SqlData Instance:$Instance Database:$DBName Query:$Query"
- $ConnString = "Server=$Instance;Database=$DBName;Integrated Security=SSPI;"
- $DataAdapter = New-Object "System.Data.SqlClient.SqlDataAdapter" ($Query,$ConnString)
- $DataTable = New-Object "System.Data.DataTable"
- [void]$DataAdapter.fill($DataTable)
- $DataTable
- } #Get-SQLData
- ########################
- function Get-SqlList
- {
- param ($SQLPSXServer = $(throw 'SQLPSX Instance is required.'), $SQLPSXDatabase = $(throw 'SQLPSX Database is required.'), $Query = $(throw 'Query is required.'))
- Get-SqlData $SQLPSXServer $SQLPSXDatabase $Query | ForEach {$ServerList.Add($_.Server) > $null}
- }# Get-SqlList
- ########################
- function LaunchThread
- {
- param($SQLServer)
- $StartInfo = new-object System.Diagnostics.ProcessStartInfo
- $StartInfo.FileName = "$pshome\powershell.exe"
- $StartCommand = $ScriptRoot + "Create-RGSnaps.ps1 " + $SQLServer
- $StartInfo.Arguments = " -NoProfile -Command " + $StartCommand
- $StartInfo.WorkingDirectory = $ScriptRoot
- $StartInfo.LoadUserProfile = $true
- $StartInfo.UseShellExecute = $true
- [System.Diagnostics.Process]::Start($StartInfo) > $null
- }# LaunchThread
- ########################
- #Main
- Get-SqlList $SQLPSXServer $SQLPSXDatabase $Query
- While ($ServerList.Count -gt 0)
- {
- If ($(Get-Process | where {$_.ProcessName -eq 'Powershell' -and $_.Id -ne $PID} | Measure-Object).count -lt $MaxThread)
- {
- $Server = $ServerList[0]
- LaunchThread $Server
- $ServerList.Remove("$server")
- }
- Else
- {
- [System.Threading.Thread]::Sleep(15000)
- }
- }
No comments:
Post a Comment