heat-templates/hot/Windows/MSSQLServer/heat-powershell-utils.psm1
Adrian Vladu d22d8b2587 HOT for MSSQL Server with unit tests
It deploys an MSSQL Server instance.
The MSSQL ISO file is copied from an SMB share.

The unit tests for the powershell module(user data scripts) are written
using Pester 3.0

Change-Id: If2b024d7717e2846bb199e950c314ce9c9d778e7
Partially-Implements: blueprint windows-instances
2014-09-18 03:55:24 +03:00

370 lines
10 KiB
PowerShell
Executable File

#ps1_sysnative
# Copyright 2014 Cloudbase Solutions Srl
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
$rebotCode = 1001
$reexecuteCode = 1002
$rebootAndReexecuteCode = 1003
function Log-HeatMessage {
param(
[string]$Message
)
Write-Host $Message
}
function ExitFrom-Script {
param(
[int]$ExitCode
)
exit $ExitCode
}
function ExecuteWith-Retry {
param(
[ScriptBlock]$Command,
[int]$MaxRetryCount=10,
[int]$RetryInterval=3,
[array]$Arguments=@()
)
$currentErrorActionPreference = $ErrorActionPreference
$ErrorActionPreference = "Continue"
$retryCount = 0
while ($true) {
try {
$res = Invoke-Command -ScriptBlock $Command `
-ArgumentList $Arguments
$ErrorActionPreference = $currentErrorActionPreference
return $res
} catch [System.Exception] {
$retryCount++
if ($retryCount -gt $MaxRetryCount) {
$ErrorActionPreference = $currentErrorActionPreference
throw $_.Exception
} else {
Write-Error $_.Exception
Start-Sleep $RetryInterval
}
}
}
}
function Execute-Command {
param(
[ScriptBlock]$Command,
[array]$Arguments=@(),
[string]$ErrorMessage
)
$res = Invoke-Command -ScriptBlock $Command -ArgumentList $Arguments
if ($LASTEXITCODE -ne 0) {
throw $ErrorMessage
}
return $res
}
function Install-WindowsFeatures {
param(
[Parameter(Mandatory=$true)]
[array]$Features,
[int]$RebootCode=$rebootAndReexecuteCode
)
$winVer = (Get-WmiObject -class Win32_OperatingSystem).Version.Split('.')
$isWinServer2008R2 = (($winVer[0] -eq 6) -and ($winVer[1] -eq 1))
if ($isWinServer2008R2 -eq $true) {
Import-Module ServerManager
}
$rebootNeeded = $false
foreach ($feature in $Features) {
if ($isWinServer2008R2 -eq $true) {
$state = ExecuteWith-Retry -Command {
Add-WindowsFeature -Name $feature -ErrorAction Stop
}
} else {
$state = ExecuteWith-Retry -Command {
Install-WindowsFeature -Name $feature -ErrorAction Stop
}
}
if ($state.Success -eq $true) {
if ($state.RestartNeeded -eq 'Yes') {
$rebootNeeded = $true
}
} else {
throw "Install failed for feature $feature"
}
}
if ($rebootNeeded -eq $true) {
ExitFrom-Script $RebootCode
}
}
function CopyFrom-SambaShare {
param(
[Parameter(Mandatory=$true)]
[string]$SambaDrive,
[Parameter(Mandatory=$true)]
[string]$SambaShare,
[Parameter(Mandatory=$true)]
[string]$SambaFolder,
[Parameter(Mandatory=$true)]
[string]$FileName,
[Parameter(Mandatory=$true)]
[string]$Destination
)
$p = New-PSDrive -Name $SambaDrive -Root $SambaShare -PSProvider FileSystem
$samba = ($SambaDrive + ":\\" + $SambaFolder)
$localPath = "$Destination\$FileName"
if (!(Test-Path $localPath)) {
Copy-Item "$samba\$FileName" $Destination -Recurse
}
return $localPath
}
function Unzip-File {
param(
[Parameter(Mandatory=$true)]
[string]$ZipFile,
[Parameter(Mandatory=$true)]
[string]$Destination
)
$shellApp = New-Object -ComObject Shell.Application
$zipFileNs = $shellApp.NameSpace($ZipFile)
$destinationNs = $shellApp.NameSpace($Destination)
$destinationNs.CopyHere($zipFileNs.Items(), 0x4)
}
function Download-File {
param(
[Parameter(Mandatory=$true)]
[string]$DownloadLink,
[Parameter(Mandatory=$true)]
[string]$DestinationFile
)
$webclient = New-Object System.Net.WebClient
ExecuteWith-Retry -Command {
$webclient.DownloadFile($DownloadLink, $DestinationFile)
}
}
# Get-FileHash for Powershell versions less than 4.0 (SHA1 algorithm only)
function Get-FileSHA1Hash {
[CmdletBinding()]
param(
[parameter(Mandatory=$true)]
[string]$Path,
[string]$Algorithm = "SHA1"
)
process
{
if ($Algorithm -ne "SHA1") {
throw "Unsupported algorithm: $Algorithm"
}
$fullPath = Resolve-Path $Path
$f = [System.IO.File]::OpenRead($fullPath)
$sham = $null
try {
$sham = New-Object System.Security.Cryptography.SHA1Managed
$hash = $sham.ComputeHash($f)
$hashSB = New-Object System.Text.StringBuilder `
-ArgumentList ($hash.Length * 2)
foreach ($b in $hash) {
$sb = $hashSB.AppendFormat("{0:x2}", $b)
}
return [PSCustomObject]@{ Algorithm="SHA1";
Hash=$hashSB.ToString().ToUpper();
Path=$fullPath }
}
finally {
$f.Close()
if($sham) {
$sham.Clear()
}
}
}
}
function Check-FileIntegrityWithSHA1 {
param(
[Parameter(Mandatory=$true)]
[string]$File,
[Parameter(Mandatory=$true)]
[string]$ExpectedSHA1Hash
)
if ($PSVersionTable.PSVersion.Major -lt 4) {
$hash = (Get-FileSHA1Hash -Path $File).Hash
} else {
$hash = (Get-FileHash -Path $File -Algorithm "SHA1").Hash
}
if ($hash -ne $ExpectedSHA1Hash) {
throw ("SHA1 hash not valid for file: $filename. " +
"Expected: $ExpectedSHA1Hash Current: $hash")
}
}
function Install-Program {
param(
[Parameter(Mandatory=$true)]
[string]$DownloadLink,
[Parameter(Mandatory=$true)]
[string]$DestinationFile,
[Parameter(Mandatory=$true)]
[string]$ExpectedSHA1Hash,
[Parameter(Mandatory=$true)]
[string]$Arguments,
[Parameter(Mandatory=$true)]
[string]$ErrorMessage
)
Download-File $DownloadLink $DestinationFile
Check-FileIntegrityWithSHA1 $DestinationFile $ExpectedSHA1Hash
$p = Start-Process -FilePath $DestinationFile `
-ArgumentList $Arguments `
-PassThru `
-Wait
if ($p.ExitCode -ne 0) {
throw $ErrorMessage
}
Remove-Item $DestinationFile
}
function Set-IniFileValue {
[CmdletBinding()]
param(
[parameter(Mandatory=$true, ValueFromPipeline=$true)]
[string]$Key,
[parameter()]
[string]$Section = "DEFAULT",
[parameter(Mandatory=$true)]
[string]$Value,
[parameter(Mandatory=$true)]
[string]$Path
)
process
{
$Source = @"
using System;
using System.Text;
using System.Runtime.InteropServices;
namespace PSCloudbase
{
public sealed class Win32IniApi
{
[DllImport("kernel32.dll", CharSet=CharSet.Unicode, SetLastError=true)]
public static extern uint GetPrivateProfileString(
string lpAppName,
string lpKeyName,
string lpDefault,
StringBuilder lpReturnedString,
uint nSize,
string lpFileName);
[DllImport("kernel32.dll", CharSet=CharSet.Unicode, SetLastError=true)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool WritePrivateProfileString(
string lpAppName,
string lpKeyName,
StringBuilder lpString, // Don't use string, as Powershell replaces $null with an empty string
string lpFileName);
[DllImport("Kernel32.dll")]
public static extern uint GetLastError();
}
}
"@
Add-Type -TypeDefinition $Source -Language CSharp
$retVal = [PSCloudbase.Win32IniApi]::WritePrivateProfileString($Section, $Key, $Value, $Path)
if (!$retVal -and [PSCloudbase.Win32IniApi]::GetLastError()) {
throw "Cannot set value in ini file: " + [PSCloudbase.Win32IniApi]::GetLastError()
}
}
}
function LogTo-File {
param(
$LogMessage,
$LogFile="C:\cfn\userdata.log",
$Topic="general"
)
$date = Get-Date
$fullMessage = "$Date | $Topic | $LogMessage"
Add-Content -Path $logFile -Value $fullMessage
}
function Open-Port($port, $protocol, $name) {
netsh.exe advfirewall firewall add rule name=$name dir=in action=allow `
protocol=$protocol localport=$port
}
function Add-WindowsUser {
param(
[parameter(Mandatory=$true)]
[string]$Username,
[parameter(Mandatory=$true)]
[string]$Password
)
NET.EXE USER $Username $Password '/ADD'
}
function Send-HeatWaitSignal {
param(
[parameter(Mandatory=$true)]
[string]$Endpoint,
[parameter(Mandatory=$true)]
[string]$Token,
$Message,
$Success=$true
)
$statusMap = @{
$true="SUCCESS";
$false="FAILURE"
}
$heatMessage = @{
"status"=$statusMap[$Success];
"reason"="Configuration script has been executed.";
"data"=$Message;
}
$headers = @{
"X-Auth-Token"=$Token;
"Accept"="application/json";
"Content-Type"= "application/json";
}
$heatMessageJSON = $heatMessage | ConvertTo-JSON
$result = Invoke-RestMethod -Method POST -Uri $Endpoint `
-Body $heatMessageJSON -Headers $headers
}
Export-ModuleMember -Function *