$EVENT_ID_SUCCESS = 1000 #Use IDs in the range 1 - 1000
$EVENT_ID_SCRIPTERROR = 999 #Then you can use eventcreate.exe to test the MP
#Scripting.FileSystemObject constants
$FOR_READING = 1 #Open a file for reading only. You can#t write to this file.
$FOR_WRITING = 2 #Open a file for writing.
$FOR_APPENDING = 8 #Open a file and write to the end of the file.
$WIN_SRV_2012_OSVer = "6.2"
#******************************************************************************
# FUNCTION: CheckByOSCurrentVersion
# DESCRIPTION: Returns True if the Registry Key for CurrentVersion
# is equal the target OS Versions Number.
# RETURNS: Boolean: True, if build is greater or equal than the given number
#******************************************************************************
function CheckByOSCurrentVersion()
{
$strCurrentOSVer = Get-ItemProperty "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\"
$strCurrentOSVer = $strCurrentOSVer.CurrentVersion
$CheckByOSCurrentVersion = $false
if($strCurrentOSVer -ge $WIN_SRV_2012_OSVer){
$CheckByOSCurrentVersion = $true
}else{
$CheckByOSCurrentVersion = $false
}
return $CheckByOSCurrentVersion;
}
$useCIM = CheckByOSCurrentVersion
#Global vars
#$mblnLogSuccessEvent #As String(ToBoolean) (passed in as parameter of script)
#$mstrIdentifier #As String (passed in as parameter of script)
#*********************************************************************************************
# PROCEDURE: Main
# DESCRIPTION: Reads the script parameters and creates the cluster disk monitoring data.
# PARAMETERS: void
#*********************************************************************************************
Function Main()
{
$objMomScriptAPI = New-Object -comObject MOM.ScriptAPI
#Monitor instances
$boolNoMonitoredClusterDiskAvailable = $false
$list = MonitorClusterDisks $strTargetComputer $strClusterName $objMomScriptAPI ([ref]$boolNoMonitoredClusterDiskAvailable)
$blnSuccess = $false
if($list -ne $null){
$blnSuccess = $true
}
foreach($data in $list)
{
#Add the values for the perf counter#s object name, counter name, instance name and the value)
if($data.$PERFORMANCE_OBJECT_NAME -eq $null)
{
continue
}
$blnSuccess = $true
$objTypedPropertyBag = $objMomScriptAPI.CreateTypedPropertyBag($PERFORMANCE_DATA_TYPE)
$objTypedPropertyBag.AddValue($PERFORMANCE_OBJECT_NAME, $data.$PERFORMANCE_OBJECT_NAME)
$objTypedPropertyBag.AddValue($PERFORMANCE_COUNTER_NAME, $data.$PERFORMANCE_COUNTER_NAME)
$objTypedPropertyBag.AddValue($PERFORMANCE_INSTANCE_NAME, $data.$PERFORMANCE_INSTANCE_NAME)
if(($data.$PERFORMANCE_VALUE_NAME -eq $null) -or ($data.$PERFORMANCE_VALUE_NAME -eq "")){
$objTypedPropertyBag.AddValue($PERFORMANCE_VALUE_NAME, 0)
}else{
$objTypedPropertyBag.AddValue($PERFORMANCE_VALUE_NAME, $data.$PERFORMANCE_VALUE_NAME)
}
#Necessary information for the property bag data source filter condition
#to enable Agent cook down
$objTypedPropertyBag.AddValue($INSTANCE_NAME, $data.$INSTANCE_NAME)
$objTypedPropertyBag
}
if ($blnSuccess)
{
if ([System.Convert]::ToBoolean($mblnLogSuccessEvent) -eq $true)
{
$objMomScriptAPI.LogScriptEvent((Split-Path $MyInvocation.ScriptName -Leaf), $EVENT_ID_SUCCESS, $EVENT_TYPE_INFORMATION, "-- Cluster Disks Monitoring Script -- Script executed successfully.")
}
}
else
{
$objMomScriptAPI.LogScriptEvent((Split-Path $MyInvocation.ScriptName -Leaf), $EVENT_ID_SCRIPTERROR, $EVENT_TYPE_WARNING, "-- Cluster Disks Monitoring Script -- An error occurred while running the script.")
}
if ($blnSuccess -and $boolNoMonitoredClusterDiskAvailable)
{
#Create an Empty typed property bag of the performance data type
$objTypedPropertyBag = $objMomScriptAPI.CreateTypedPropertyBag($PERFORMANCE_DATA_TYPE)
#Add the property bag to the script#s XML output
$objTypedPropertyBag
}
}
#****************************************************************************************************************
# FUNCTION: MonitorClusterDisks
# DESCRIPTION: Creates monitoring data for all instances of the class #Microsoft.Windows.Server.ClusterDisksMonitoring.ClusterDisk# using WMI.
# PARAMETERS: IN String strTargetComputer: principal name of the targeted #Microsoft.Windows.Cluster.VirtualServer# instance.
# IN String strClusterName: the cluster containing the cluster shared volume
# RETURNS: Boolean: True if successful
#****************************************************************************************************************
Function MonitorClusterDisks($strTargetComputer, $strClusterName, $objMomScriptAPI, [ref]$boolNoMonitoredClusterDiskAvailable) #As Boolean
{
$InstancesCount = 0
#Connect to WMI NS \\.\root\MSCluster
$error.Clear()
# !!! Refactoring comment:
# Original VBScript only tries to connect to the namespace. Piping to get only the first one saves time.
if($useCIM){
$oWMI = Get-CimClass -Namespace $WMI_MSCLUSTER_NAMESPACE -ErrorAction SilentlyContinue | select -First 1
}else{
$oWMI = Get-WMIObject -Namespace $WMI_MSCLUSTER_NAMESPACE -Class $WMI_MSCLUSTER_CLUSTER_RESOURCE -ErrorAction SilentlyContinue | select -First 1
}
if ($error.Count -eq 0)
{
$blnSuccess = $true
$strOSVersion = (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion").CurrentVersion
#Adding OS Validation to use propert WMI query to get correct instances
#ResourceClass and IsClusterSharedVolume properties are not available in Windows 2008 OS, only 2008 R2 and beyond.
if ($strOSVersion -eq "6.0")
{
$strWMIQuery = "select * from " + $WMI_MSCLUSTER_CLUSTER_RESOURCE + " where CoreResource = FALSE"
}
else
{
$strWMIQuery = "select * from " + $WMI_MSCLUSTER_CLUSTER_RESOURCE + " where ResourceClass = 1 and IsClusterSharedVolume = FALSE and CoreResource = FALSE"
}
if($useCIM){
$objClusterResources = Get-CimInstance -Namespace $WMI_MSCLUSTER_NAMESPACE -Query $strWMIQuery
}else{
$objClusterResources = Get-WMIObject -Namespace $WMI_MSCLUSTER_NAMESPACE -Query $strWMIQuery
}
#Loop through all returned cluster shared volumes
foreach ($objClusterResource in $objClusterResources)
{
#Get the associated disk
if($useCIM){
$objClusterDisks = (Get-CimAssociatedInstance -InputObject $objClusterResource -Association $WMI_CLUSTER_RESOURCE_TO_DISK_ASSOCIATOR_CLASS)
}else{
$objClusterDisks = $objClusterResource.GetRelated("MSCluster_Disk")
}
foreach ($objClusterDisk in $objClusterDisks)
{
#Get the associated disk partition
if($useCIM){
$objClusterDiskPartitions = (Get-CimAssociatedInstance -InputObject $objClusterDisk -Association $WMI_CLUSTER_DISKPARTITION_TO_DISK_ASSOCIATOR_CLASS)
}else{
$objClusterDiskPartitions = $objClusterDisk.GetRelated("MSCluster_DiskPartition")
}
foreach ($objClusterDiskPartition in $objClusterDiskPartitions)
{
$InstancesCount += 1
$list = CreateMonitoringData $objClusterDisk $objClusterDiskPartition $objClusterResource $strTargetComputer $strClusterName $objMomScriptAPI
}
}
}
$boolNoMonitoredClusterDiskAvailable.Value = ($InstancesCount -le 0)
}
else
{
$boolNoMonitoredClusterDiskAvailable.Value = $true
}
return $list
}
#****************************************************************************************************************
# FUNCTION: CreateMonitoringData
# DESCRIPTION: Create monitoring data for a cluster shared volume.
#
# PARAMETERS: IN Object objClusterResource: the cluster disks as SWbemObject of type MSCluster_Disk.
# IN Object objClusterDiskPartition: the associated disk partition as SWbemObject of type MSCluster_DiskPartition.
# IN Object objClusterResource: the associated cluster resource as SWbemObject of type MSCluster_Resource.
# IN String strTargetComputer: the principal name of the targeted cluster virtual server hosting the cluster shared volume,
# i.e. the cluster name and not the current owner!
# IN String strClusterName: the cluster containing the cluster shared volume
# RETURNS: Boolean: True if successful
#****************************************************************************************************************
Function CreateMonitoringData($objClusterDisk, $objClusterDiskPartition, $objClusterResource,
$strTargetComputer, $strClusterName, $objMomScriptAPI) #As Boolean
{
$list = New-Object System.Collections.ArrayList($null)
#Create performance data
$strPartitionName = $objClusterResource.$WMI_NAME_PROPERTY_NAME + "_" + $objClusterDiskPartition.$WMI_PATH_PROPERTY_NAME
#***************************************************************************************
# PROCEDURE: CreatePerformanceTypedPropertyBag
# DESCRIPTION: Generates a performance typed property bag and adds it to the script#s XML output.
# PARAMETERS: IN String strObjectName: name of the performance object
# IN String strCounterName: name of the performance counter
# IN String strInstanceName: name of the performance counter in instance
# IN Variant varValue: value of the performance counter
# IN String strCSVInstanceName: Friendly volume name of the cluster shared volume
# IN Object objMomScriptAPI: initialised MOM.ScriptAPI object
#***************************************************************************************
Function CreatePerformanceTypedPropertyBag($strObjectName, $strCounterName, $strInstanceName,
$varValue, $strCSVInstanceName, $objMomScriptAPI)
{