SCOMHelper.psm1

<#
  2022.10.11.1247 - Added Start-SCOMTrace to support mgmt servers. Typically this module will only exist on SCOM mgmt servers.
  2022.06.20 - Added new function: Convert-MAML2HTML
  2022.06.10 - Added Get-SCOMRunningWorkflows. Improved Export-SCOMEffectiveMonitoringConfigurationReport function to eliminate empty Path issue.
  2020.11.19 - Update-SCOMComputerGroup: improved a little bit. Update-OMGroupDiscovery2: added default Mgmtservername value and failure logging.
  2020.08.18 - Added new function: Update-SCOMComputerGroup. Fixed small issue with New-SCOMClassGraph.
  2020.08.11 - Added a Private.ps1 file to include common functions.
  2020.07.22 - Remove 'Import-Module OperationsManager'
#>


# Load private functions
. (Join-Path $PSScriptRoot Private.ps1)


#######################################################################
#######################################################################
#######################################################################
#######################################################################


<#
.Synopsis
  This will initiate SCOM agent tracing, output trace files, then format the trace files into .log for humans.
 
.EXAMPLE
  Start-SCOMTrace -General -Verbose
.EXAMPLE
  Start-SCOMTrace -General -TraceSeconds 600 -GeneralGuidLevel Extended -OutPath C:\TraceLogs -Verbose
.EXAMPLE
  Start-SCOMTrace -Specific -TraceSeconds 600 -SpecificTraceName 'MyCustomTrace' -OutPath C:\TraceLogs -Verbose -MaxLogMB 2048
 
.NOTES
Function: Start-SCOMTrace
Author: Tyson Paul (https://monitoringguys.com/)
Version History:
2021.01.21.11047 - Rewrite into proper function for SCOMAgentHelper module
2020.11.20.1727 - v1
 
#>

Function Start-SCOMTrace
{

  Param (
    # the duration of the trace
    [Parameter(Mandatory = $false, 
        ValueFromPipeline = $false,
        ValueFromPipelineByPropertyName = $false, 
        ValueFromRemainingArguments = $false)]
    [Alias('Duration')]
    [int]$TraceSeconds = 300,


    <#
        'Specific' requires workflow override of 'TraceEnabled' property for specific object instance (not group or class).
 
        ------- Override Example -------
 
        <Monitoring>
        <Overrides>
        <MonitorPropertyOverride ID="WorkflowDebugger.WorkflowTraceOverride" Context="SQLServer!Microsoft.SQLServer.Windows.DBEngine" ContextInstance="2c821943-be80-5733-0b16-59850426eabe" Enforced="false" Monitor="TempDBPerformance!TempDBPerformance.TempDBPercentUsage.Monitor" Property="TraceEnabled">
        <Value>true</Value>
        </MonitorPropertyOverride>
        </Overrides>
        </Monitoring>
 
        ------- End Override Example -------
    #>

    [Parameter(Mandatory = $true, 
        ValueFromPipeline = $false,
        ValueFromPipelineByPropertyName = $false,
                   ParameterSetName='Parameter Set 1')]
    [switch]$Specific,


    # Name of the logfile. Used for a Specific trace scenario only.
    [Parameter(Mandatory = $false, 
        ValueFromPipeline = $false,
        ValueFromPipelineByPropertyName = $false,
                   ParameterSetName='Parameter Set 1')]
    [string]$SpecificTraceName = 'MyCustomWorkflowTrace',


    # General trace for common workflow types
    [Parameter(Mandatory = $true, 
        ValueFromPipeline = $false,
        ValueFromPipelineByPropertyName = $false,
                   ParameterSetName='Parameter Set 2')]
    [switch]$General,


    # Which GUIDs to include in the capture. Used for General trace scenario only.
    [Parameter(Mandatory = $false, 
        ValueFromPipeline = $false,
        ValueFromPipelineByPropertyName = $false,
                   ParameterSetName='Parameter Set 2')]
    [ValidateSet('Basic', 'Extended', 'Full')]
    [string]$GeneralGuidLevel = 'Basic',


    #Max size of log files (circular)
    [Parameter(Mandatory = $false, 
        ValueFromPipeline = $false,
        ValueFromPipelineByPropertyName = $false)]
    [int]$MaxLogMB = 1024,


    # This is the path to the agent Tools folder. Use only if the default path is broken or incorrect in the registry.
    [Parameter(Mandatory = $false, 
        ValueFromPipeline = $false,
        ValueFromPipelineByPropertyName = $false)]
    [string]$AgentToolsPath = 'NONE',


    # Where to output the trace files
    [Parameter(Mandatory = $false, 
        ValueFromPipeline = $false,
        ValueFromPipelineByPropertyName = $false)]
    [string]$OutputPath = 'C:\Windows\Logs\OpsMgrTrace'

  )

  #==============================================================================

  # Basic is most common. It's rare to need/use Extended or Full. This hash is a clever way to control tiers of inclusion.
  $hashGuidFileNames = [ordered]@{}
  Switch ($GeneralGuidLevel) {
    {
      $_ -match 'Basic|Extended|Full'
    } 
    {
      $hashGuidFileNames['TracingGuidsNative'] = 'TracingGuidsNative.txt'
      $hashGuidFileNames['TracingGuidsScript'] = 'TracingGuidsScript.txt'
      $hashGuidFileNames['TracingGuidsManaged'] = 'TracingGuidsManaged.txt'
    }
  
    {
      $_ -match 'Extended|Full'
    } 
    {
      $hashGuidFileNames['TracingGuidsConfigService'] = 'TracingGuidsConfigService.txt'
      $hashGuidFileNames['TracingGuidsDAS'] = 'TracingGuidsDAS.txt'
      $hashGuidFileNames['TracingGuidsFailover'] = 'TracingGuidsFailover.txt'
      $hashGuidFileNames['TracingGuidsUI'] = 'TracingGuidsUI.txt'
    }
  
    'Full' 
    {
      $hashGuidFileNames['TracingGuidsAdvisor'] = 'TracingGuidsAdvisor.txt'
      $hashGuidFileNames['TracingGuidsAPM'] = 'TracingGuidsAPM.txt'
      $hashGuidFileNames['TracingGuidsApmConnector'] = 'TracingGuidsApmConnector.txt'
      $hashGuidFileNames['TracingGuidsBID'] = 'TracingGuidsBID.txt'
      $hashGuidFileNames['TracingGuidsNASM'] = 'TracingGuidsNASM.txt'
    }
  }

  If (-NOT (Test-Path $OutputPath -PathType Container)){
    New-Item -Path $OutputPath -ItemType Directory -ErrorAction SilentlyContinue
  }
  
  If (-NOT(Test-Path -Path $AgentToolsPath -PathType Container -ErrorAction SilentlyContinue) ) 
  {
    # Locate tools directory
    $setupKey = Get-Item -Path 'HKLM:\Software\Microsoft\Microsoft Operations Manager\3.0\Setup'
    $InstallDirectory = $setupKey.GetValue('InstallDirectory') 
    $AgentToolsPath = Join-Path -Path $InstallDirectory -ChildPath 'Tools'
  }

  Set-Location $AgentToolsPath

  Write-Output "`nAny active traces will first be stopped.`n"
  # Stop any previous tracing
  & .\Stoptracing.cmd

  Write-Output "`n`n`n"

  If ($General ) {
    ForEach ($Key in @($hashGuidFileNames.Keys))
    { 
      $outFile = (Join-Path -Path $OutputPath -ChildPath "$($Key).etl")
      Write-Output "Start General trace session for $outFile.`n"
      # Start general trace
      .\TraceLogSM.exe -start $Key -flag 0x1F -level 6 -f $outFile -b 64 -ft 10 -cir $MaxLogMB -guid $($hashGuidFileNames[$Key])
    }
  }
  
  If ($Specific) {
    # This hash variable is used for the stop/format commands at the end.
    $hashGuidFileNames = [ordered]@{
      $SpecificTraceName = $SpecificTraceName
    }
    $MySpecificTraceFile = Join-Path -Path $OutputPath -ChildPath "$($SpecificTraceName).etl"
    Write-Output "Start Specific trace session: $SpecificTraceName.`n"
    # Start trace for specific workflow object instance
    & .\TraceLogSM.exe -start $SpecificTraceName -flag 0xFF -level 5 -ft 1 -rt -GUID '#c85ab4ed-7f0f-42c7-8421-995da9810fdd' -b 1024 -f $MySpecificTraceFile
  }


  Write-Output "`n`nTracing for $TraceSeconds seconds..." 
  $EndTime = (Get-Date).AddSeconds($TraceSeconds)
  $TMFExists = $false

  While ((Get-Date) -le $EndTime) {
    If (((Get-Date).Second)%10 -eq 0) 
    {
      Write-Output "`n$([math]::Max(([int]($EndTime - (Get-Date)).TotalSeconds),0) ) seconds remain until trace completion..."
    }

    If (-NOT $TMFExists){
      # Ensure that 'all.tmf' exists in Tools folder
      $allPath = (Join-Path $AgentToolsPath 'all.tmf')
      If (-NOT (Test-Path -Path $allPath -PathType Leaf)) {
        Write-Output "`nIn the meantime, 'all.tmf' not found at this path: [$($allPath)]. No problem, I ($(whoami.exe)) will fix it now."
        Write-Output "Will extract all .cabs in the TMF folder. Will concatenate all .tmf files from the .\TMF\*.tmf path into a single file: .\all.tmf"
        $CABs = @(Get-ChildItem -Path (Join-Path $AgentToolsPath 'TMF') -Filter '*.cab')
        ForEach ($Cab in $CABs) {
          expand.exe $CABs.FullName -F:*.tmf ".\tmf\$($Cab.BaseName).tmf"
        }
        Write-Output "`nBuilding 'all.tmf' file..."
        Get-Content .\tmf\*.tmf | Out-File .\all.tmf -Encoding utf8

        If (-NOT (Test-Path -Path $allPath -PathType Leaf)) {
          Write-Output "`nFailed to create necessary tmf file at path: [$($allPath)]."
        }
        Else {
          Write-Output "`nGood news! 'all.tmf' now exists at this path: [$($allPath)]."
          $TMFExists = $True
        }
      }
      Else {
        Write-Output "`nGood news! 'all.tmf' exists at this path: [$($allPath)]."
        $TMFExists = $True
      }
    }

    Start-Sleep -Seconds 1
  }
 
  # If script has been stopped manually for some reason, no problem! You can run the below commands to stop the TRACES and format the logs.
  # Stop tracing session and format logs for humans to read
  @($hashGuidFileNames.Keys) | ForEach-Object -Process {
    Write-Output "`n`nStopping trace of session: $_."
    & .\TraceLogSM.exe -stop $_ 
  }
 
  @($hashGuidFileNames.Keys) | ForEach-Object -Process {
    Write-Output "`n`nFormating: $(Join-Path -Path $OutputPath -ChildPath "$($_).etl")..."
    # Format the trace data into plain text for humans
    & .\TraceFmtSM.exe $(Join-Path -Path $OutputPath -ChildPath "$($_).etl") -tmf .\all.tmf -o $(Join-Path -Path $OutputPath -ChildPath "$($_).LOG")
  }
  Write-Output "`nFormat done!`n"
}#End Function


#######################################################################


<#
    .Synopsis
    Will clear SCOM agent cache on local machine
   
    .EXAMPLE
    Clear-SCOMCache -HealthServiceStatePath 'D:\Program Files\Microsoft System Center 2012 R2\Operations Manager\Server\Health Service State'
 
    .EXAMPLE
    Clear-SCOMCache -TimeoutSeconds 60
 
    .INPUTS
    None
   
    .OUTPUTS
    Boolean [true|false]
 
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Version History:
    2020.07.21 - Improved error handling a bit. Added cmdletbinding. Changed outputtype to Bool.
    2020.02.27 - Cleaned up code a bit. Improved logic. Removed WaitSeconds parameter as it was not effective.
    2018.05.25 - Cleaned up. Revised a few things.
     
#>

Function Clear-SCOMCache {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
      SupportsShouldProcess=$false, 
      PositionalBinding=$false,
      HelpUri = 'http://www.microsoft.com/',
  ConfirmImpact='Medium')]
  [OutputType([BOOL])]
  
  Param(
    
    # How many seconds to wait before aborting
    [int]$TimeoutSeconds = 30,
    
    # If the registry does not contain the correct path you can specify a path here.
    [string]$HealthServiceStatePath = (Join-Path (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Microsoft Operations Manager\3.0\Setup" -Name "InstallDirectory").InstallDirectory 'Health Service State' )

  )
 
 
  $ServiceName = 'HealthService'
  Write-Verbose "Stopping service: $ServiceName"
  
  $StartTime = Get-Date
  Try {
    Stop-Service -Name $ServiceName -Verbose -Force -PassThru
    While ( ((Get-Service -Name $ServiceName).Status -ne "Stopped") -AND ((Get-Date) -le ($StartTime.AddSeconds($TimeoutSeconds) )) ) {
      Write-Verbose "Waiting for $ServiceName to stop..."
      Start-Sleep -Seconds 5
    } 
  } Catch {
    Write-Warning "Problem while stopping service: $($_). Continuing..."
  }
  
  If ((Get-Service -Name $ServiceName).Status -ne "Stopped") {
    Write-Warning "Failed to stop service: $ServiceName in the time allowed! Will attempt to leave HealthService in 'Started' state."
    Start-Service $ServiceName -Verbose
    Return $false
  }
  Else {
    Write-Verbose "Attempting to remove cache folder: $HealthServiceStatePath"
    Try {
      Get-Item $HealthServiceStatePath | Remove-Item -Recurse -Force -Verbose -ErrorAction Stop
      Start-Sleep -Seconds 1
    } Catch {
      Write-Warning "Error while deleting cache files. $($_)"  
    }
    If (Test-Path -PathType Container $HealthServiceStatePath) {
      Write-Warning "Failed to remove folder: [$($HealthServiceStatePath)] !"
    }
    Else {
      [bool]$folderRemoved = $true
      Write-Verbose "Health Service State folder removed succesfully: [$($HealthServiceStatePath)]."
    }

    $StartTime = Get-Date
    Try {
      Start-Service -Name $ServiceName -Verbose
      While ( ((Get-Service -Name $ServiceName).Status -ne "Running") -AND ((Get-Date) -le ($StartTime.AddSeconds($TimeoutSeconds) )) ) {
        Start-Sleep -Seconds 3
      }
      If ((Get-Service -Name $ServiceName).Status -ne "Running") {
        Write-Warning "Failed to start service: $ServiceName in the time allowed! Verify service manually."
      }
      Else {
        If ( (Test-Path -PathType Container $HealthServiceStatePath) -AND ([bool]$folderRemoved)){
          Write-Verbose "Cache folder auto-created: $HealthServiceStatePath !"
          Write-Verbose "SCOM agent Cache has been cleared."
          Return $true
        }
      }
    } Catch {
      Throw
      Return $false
    }

    If (-NOT [bool]$folderRemoved){
      Write-Warning "SCOM cache was not fully cleared; not all files/folders were deleted. For best results, try again or reboot and try again. Alternatively you could try clearing cache manually."
      Return $false
    }
    
    #default scenario, assumed something didn't go quite right
    Return $false
  }
} #End Function
#######################################################################

<#
    .Synopsis
    An easy way to compare individual characters of two strings.
     
    .EXAMPLE
    Compare-String -String1 "Captain Jack will get you by tonight" -String2 "Çaptain JAck"
    # Example Output
 
    Index String1/ASCII String2/ASCII Compare Algorithm String1_Hash String2_Hash
    ----- ------------- ------------- ------- --------- ------------ ------------
    0 C:67 Ç:199 Different SHA1 32096c2e0eff33d844ee6d675407ace18289357d 1d866b5fef7cdae2d4deba53c7ae4bdbf76ca671
    1 a:97 a:97 Same SHA1 86f7e437faa5a7fce15d1ddcb9eaeaea377667b8 86f7e437faa5a7fce15d1ddcb9eaeaea377667b8
    2 p:112 p:112 Same SHA1 516b9783fca517eecbd1d064da2d165310b19759 516b9783fca517eecbd1d064da2d165310b19759
    3 t:116 t:116 Same SHA1 8efd86fb78a56a5145ed7739dcb00c78581c5375 8efd86fb78a56a5145ed7739dcb00c78581c5375
    4 a:97 a:97 Same SHA1 86f7e437faa5a7fce15d1ddcb9eaeaea377667b8 86f7e437faa5a7fce15d1ddcb9eaeaea377667b8
    5 i:105 i:105 Same SHA1 042dc4512fa3d391c5170cf3aa61e6a638f84342 042dc4512fa3d391c5170cf3aa61e6a638f84342
    6 n:110 n:110 Same SHA1 d1854cae891ec7b29161ccaf79a24b00c274bdaa d1854cae891ec7b29161ccaf79a24b00c274bdaa
    7 :32 :32 Same SHA1 b858cb282617fb0956d960215c8e84d1ccf909c6 b858cb282617fb0956d960215c8e84d1ccf909c6
    8 J:74 J:74 Same SHA1 58668e7669fd564d99db5d581fcdb6a5618440b5 58668e7669fd564d99db5d581fcdb6a5618440b5
    9 a:97 A:65 Different SHA1 86f7e437faa5a7fce15d1ddcb9eaeaea377667b8 6dcd4ce23d88e2ee9568ba546c007c63d9131c1b
    10 c:99 c:99 Same SHA1 84a516841ba77a5b4648de2cd0dfcb30ea46dbb4 84a516841ba77a5b4648de2cd0dfcb30ea46dbb4
    11 k:107 k:107 Same SHA1 13fbd79c3d390e5d6585a21e11ff5ec1970cff0c 13fbd79c3d390e5d6585a21e11ff5ec1970cff0c
    
    The Example above will compare the individual characters of the shortest parameter to that of the other parameter and display ASCII values for each while indicating any differences between characters of the same index/location.
     
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Version: 1.0
    Version History:
    2020.02.26 - Slight revision. Added commenting, better Help content. Added to SCOMHelper module.
    2018.04.20 - Original
#>

Function Compare-String {
  Param(
    [Parameter(Mandatory=$true, 
        ValueFromPipeline=$false,
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$String1,

    [Parameter(Mandatory=$true, 
        ValueFromPipeline=$false,
        Position=1,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$String2,

    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
        Position=2,
    ParameterSetName='Parameter Set 1')]
    [ValidateSet("SHA", "SHA1", "MD5", "SHA256", "SHA384", "SHA512","RIPEMD160")]
    $Algorithm = "SHA1" #SHA1 is default. Ref: https://docs.microsoft.com/en-us/dotnet/api/system.security.cryptography.hashalgorithm.create?view=netframework-4.8
  )

  ######################### MAIN ################################
  [System.Collections.ArrayList]$arr = @()
  0..([math]::min(($String1.Length -1),($String2.Length -1))) | ForEach-Object {
    $String1_char = [string]$String1[$_]
    $String2_char = [string]$String2[$_]
    $String1_hash = Get-StringHash $String1_char -Algorithm $Algorithm
    $String2_hash = Get-StringHash $String2_char -Algorithm $Algorithm
    $obj = New-Object PSCustomObject
    $obj | Add-Member -Name 'Index' -Value $_ -MemberType NoteProperty
    $obj | Add-Member -Name 'String1/ASCII' -Value ("$($String1_char):$([double][char]$String1_char)") -MemberType NoteProperty
    $obj | Add-Member -Name 'String2/ASCII' -Value ("$($String2_char):$([double][char]$String2_char)") -MemberType NoteProperty

    If ($String1_hash -ne $String2_hash){
      $obj | Add-Member -Name 'Compare' -Value "Different" -MemberType NoteProperty
    }
    Else{
      $obj | Add-Member -Name 'Compare' -Value "Same" -MemberType NoteProperty
    }
    $obj | Add-Member -Name "Algorithm" -Value $Algorithm -MemberType NoteProperty  
    $obj | Add-Member -Name "String1_Hash" -Value $String1_hash -MemberType NoteProperty
    $obj | Add-Member -Name "String2_Hash" -Value $String2_hash -MemberType NoteProperty
        
    $null = $arr.Add($obj)
  }
  Return ($arr | Format-Table -AutoSize)
} #End Function
#######################################################################

<#
  .DESCRIPTION
  Will convert MAML text or file to HTML.
 
    .EXAMPLE
    # Convert knowledge article and write to .html file
    #Example
    PS C:\> $monitor = Get-SCOMMonitor -DisplayName 'M365 Teams - Chat Synthetic Test Performance Monitor'
    PS C:\> $Article = $monitor.GetKnowledgeArticle(([System.Globalization.CultureInfo]'en-US'))
    PS C:\> Convert-MAMLToHTML -XML $Article.MamlContent | Set-Content C:\temp\KnowledgeArticle.html
 
 
  .NOTES
  Author: Tyson Paul (https://monitoringguys.com/)
  References:
  https://devio.wordpress.com/2009/09/15/command-line-xslt-processor-with-powershell/
  https://systemcenter.wiki/
 
  Version History:
  2022.06.20 - v1
#>

Function Convert-MAMLToHTML {
    [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
                  PositionalBinding=$false,
                  HelpUri = 'https://monitoringguys.com/')]
  Param (
    # Path to XMLFile
    [Parameter(Mandatory=$true, 
                ValueFromPipeline=$false,
                ValueFromPipelineByPropertyName=$false, 
                ValueFromRemainingArguments=$false, 
                Position=0,
                ParameterSetName='XMLFileExists')]
    [ValidateNotNullOrEmpty()]
    [string]$XMLFile,
    
    # XML data (raw)
    [Parameter(Mandatory=$true, 
                ValueFromPipeline=$false,
                ValueFromPipelineByPropertyName=$false, 
                ValueFromRemainingArguments=$false, 
                Position=0,
                ParameterSetName='XMLData')]
    [ValidateNotNullOrEmpty()]
    [string]$XML,

    # Path to XSL transform file.
    [Parameter(Mandatory=$false, 
                ValueFromPipeline=$false,
                ValueFromPipelineByPropertyName=$false, 
                ValueFromRemainingArguments=$false, 
                Position=1)]
    [string]$XSLFile
  )

  ################# FUNCTIONS #################
  Function Test-XMLFile {
    <#
        .SYNOPSIS
        Test the validity of an XML file
        .NOTES
        https://stackoverflow.com/questions/14423861/how-to-validate-xml-for-correct-syntax-format
    #>

    [CmdletBinding()]
    param (
        [parameter(mandatory=$true)][ValidateNotNullorEmpty()][string]$xmlFilePath
    )

    # Check the file exists
    if (!(Test-Path -Path $xmlFilePath)){
        throw "$xmlFilePath is not valid. Please provide a valid path to the .xml fileh"
    }
    # Check for Load or Parse errors when loading the XML file
    $xml = New-Object System.Xml.XmlDocument
    try {
        $xml.Load((Get-ChildItem -Path $xmlFilePath).FullName)
        return $true
    }
    catch [System.Xml.XmlException] {
        Write-Verbose "$xmlFilePath : $($_.toString())"
        return $false
    }
  }
  
  ##############################################
  Function Write-DefaultTransformFile {
      # https://systemcenter.wiki/
      $XSLData = @"
<?xml version="1.0" encoding="utf-8"?>
 
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
                xmlns:msxsl="urn:schemas-microsoft-com:xslt"
                xmlns:maml="http://schemas.microsoft.com/maml/2004/10"
                exclude-result-prefixes="msxsl maml">
<xsl:output method="html" indent="no" encoding="utf-8" />
 
 
<xsl:template match="/">
    <xsl:apply-templates />
</xsl:template>
 
 
 
<xsl:template match="maml:section">
<xsl:apply-templates />
</xsl:template>
 
 
 
<xsl:template match="maml:lineBreak">
<br />
</xsl:template>
 
 
 
<xsl:template match="maml:navigationLink">
<a>
    <xsl:attribute name="href">
        <xsl:value-of select="maml:uri/@href"/>
    </xsl:attribute>
    <xsl:choose>
        <xsl:when test="maml:uri/@condition!=''">
            <xsl:attribute name="condition">
                <xsl:value-of select="maml:uri/@condition" />
            </xsl:attribute>
        </xsl:when>
    </xsl:choose>
    <xsl:choose>
        <xsl:when test="maml:uri/@uri!=''">
            <xsl:attribute name="uri">
                <xsl:value-of select="maml:uri/@uri" />
            </xsl:attribute>
        </xsl:when>
    </xsl:choose>
    <xsl:value-of select="maml:linkText"/>
</a>
</xsl:template>
 
 
 
<xsl:template match="maml:list">
<ul>
    <xsl:apply-templates />
</ul>
</xsl:template>
 
<xsl:template match="maml:listItem">
<li>
    <xsl:apply-templates />
</li>
</xsl:template>
 
 
 
<xsl:template match="maml:title">
<h1>
    <xsl:value-of select="."/>
</h1>
</xsl:template>
 
<xsl:template match="maml:subTitle">
<h2>
 <xsl:value-of select="." />
</h2>
</xsl:template>
 
<xsl:template match="text()">
<xsl:value-of select="."/>
</xsl:template>
 
<xsl:template match="maml:example">
    <pre>
        <xsl:apply-templates />
    </pre>
</xsl:template>
 
<xsl:template match="maml:codeInline">
    <code>
        <xsl:apply-templates />
    </code>
</xsl:template>
 
<xsl:template match="maml:computerOutputInline">
    <pre>
        <xsl:apply-templates />
    </pre>
</xsl:template>
 
<xsl:template match="maml:procedure">
    <ol>
        <xsl:apply-templates />
    </ol>
</xsl:template>
 
<xsl:template match="maml:step">
    <li>
        <xsl:apply-templates />
    </li>
</xsl:template>
 
<xsl:template match="maml:para">
    <p>
        <xsl:apply-templates />
    </p>
</xsl:template>
 
<xsl:template match="maml:ui">
    <b>
        <xsl:apply-templates />
    </b>
</xsl:template>
 
<xsl:template match="maml:entry">
    <td>
        <xsl:apply-templates />
    </td>
</xsl:template>
 
<xsl:template match="maml:headerEntry">
    <th>
        <xsl:apply-templates />
    </th>
</xsl:template>
 
<xsl:template match="maml:tableHeader">
  <thead>
     <xsl:apply-templates />
  </thead>
</xsl:template>
 
<xsl:template match="maml:row">
    <tr>
        <xsl:apply-templates />
    </tr>
</xsl:template>
 
<xsl:template match="maml:table">
    <table>
        <xsl:apply-templates />
    </table>
</xsl:template>
 
</xsl:stylesheet>
 
 
"@

    $XSLFile = Join-Path $TempDir 'maml2html.xsl'
    $XSLData | Set-Content -Path $XSLFile -Force -Encoding UTF8
  
  }
  
  ############## END FUNCTIONS #################
  ##############################################
  
  if ((-NOT $XMLFile) -AND (-NOT $XML))
  {
    Get-Help Convert-MAMLToHtml -Examples
    Return
  }

  # Setup temp dir for output file. Using a redirector to an output file is the only way I could find to obtain the property bag data.
  $TempDir = (Join-Path (Join-Path $env:Windir "Temp") (Join-Path 'SCOMHelper' 'Export-SCOMKnowledge') )
  New-Item -Type Directory -Path $TempDir -ErrorAction SilentlyContinue | Out-Null
  If (-NOT(Test-Path -Path $TempDir -PathType Container)) {
    Write-Error "Unable to create/access directory: [$($TempDir)]. " 
    Return $false
  }

  If ($XSLFile.Length) {
    If( -NOT (Test-Path $XSLFile)) { 
      Throw "XSL input file not found: $($_)"
      Exit
    }
  }
  Else {
    #dot-sourced
    . Write-DefaultTransformFile

  }
  $tmpTransformFile = New-TemporaryFile
  
  If ($XML) {
    $XMLFile = (New-TemporaryFile).FullName
    $XML | Set-Content -Path $XMLFile -Force -Encoding UTF8
  }

  If (-NOT (Test-XMLFile -xmlFilePath $XMLFile) ){
    $tmpXMLFile = (New-TemporaryFile).FullName
    # Make sure XML has a root
    "<Root>$(Get-Content -Path $XMLFile)</Root>" | Set-Content -Path $tmpXMLFile -Force -Encoding UTF8
  
    If (-NOT (Test-XMLFile -xmlFilePath $tmpXMLFile) ){
      # Make sure XML has a root
      Throw "Error loading XML data from path: $XMLFile"
      Return
    }
    # Set path to new tmp file which contains root element.
    $XMLFile = $tmpXMLFile
  }

  $xslt = New-Object System.Xml.Xsl.XslCompiledTransform;
  Try {
    $xslt.Load($XSLFile);
  } Catch {
    Write-Warning "Failed to load transform file [$($XSLFile)]. Will use default 'maml2html' transform instead."
    . Write-DefaultTransformFile
    Try {
      $xslt.Load($XSLFile);
    } Catch {
      Write-Error "Fatal Error: Failed to load default transform file [$($XSLFile)]. "
      Return
    }
  }
  $xslt.Transform($XMLFile, $tmpTransformFile.FullName);

  Return (Get-Content -Path $tmpTransformFile.FullName);
}


#######################################################################
<#
    .DESCRIPTION
    Will install MOMAgent.msi on one or more target computers. Agent will need to be manually approved unless security settings allow automatic approval.
 
    .EXAMPLE
    #
    # This example will set two target computer names, prompt for a credential, then deploy the agent to the target computers.
    # The installation scriptblock will wait 120 seconds for the service to install and present a status of "Running" before returning the service status to the output.
    PS C:\> $Computers = "scorch1.contoso.com","devdb01.contoso.com"
    PS C:\> $MyCred = Get-Credential -UserName 'Contoso\tpaul' -Message "Enter Action credential (Local Admin on targets)"
    PS C:\> Deploy-SCOMAgent -USE_SETTINGS_FROM_AD 0 -USE_MANUALLY_SPECIFIED_SETTINGS 1 -MANAGEMENT_GROUP "SCOMLAB" -MANAGEMENT_SERVER_AD_NAME "MS01.CONTOSO.COM" -MANAGEMENT_SERVER_DNS "MS01.CONTOSO.COM" -ACTIONS_USE_COMPUTER_ACCOUNT 1 -NOAPM 1 -InstallCredential $MyCred -TargetComputer $Computers -WaitForServiceSeconds 120
 
    .EXAMPLE
    #
    # This example will pipe the the array of computer names to the function. This is slower than passing the array as a parameter with '-TargetComputer'.
    # The installation scriptblock will wait 120 seconds for the service to install and present a status of "Running" before returning the service status to the output.
    PS C:\> $Computers = "scorch1.contoso.com","devdb01.contoso.com"
    PS C:\> $Computers | Deploy-SCOMAgent -USE_SETTINGS_FROM_AD 0 -USE_MANUALLY_SPECIFIED_SETTINGS 1 -MANAGEMENT_GROUP "SCOMLAB" -MANAGEMENT_SERVER_AD_NAME "MS01.CONTOSO.COM" -MANAGEMENT_SERVER_DNS "MS01.CONTOSO.COM" -ACTIONS_USE_COMPUTER_ACCOUNT 1
 
 
    .EXAMPLE
    #
    # This example demonstrates how you would specify an alternate Default Action Account. This is a very, very rare requirement but here's an example anyway.
    PS C:\> $InstallCred = Get-Credential -UserName 'Contoso\tpaul' -Message "Enter Action credential (Local Admin on targets)"
    PS C:\> $DefaultActionCred = Get-Credential -UserName 'Contoso\svc_LowPrivAccount' -Message "Enter Default Action credential (for very, very rare low-priv scenarios)"
    PS C:\> Deploy-SCOMAgent -USE_SETTINGS_FROM_AD 0 -USE_MANUALLY_SPECIFIED_SETTINGS 1 -MANAGEMENT_GROUP "SCOMLAB" -MANAGEMENT_SERVER_AD_NAME "MS01.CONTOSO.COM" -MANAGEMENT_SERVER_DNS "MS01.CONTOSO.COM" -ACTIONS_USE_COMPUTER_ACCOUNT 0 -NOAPM 1 -TargetComputer "devdb01.contoso.com" -WaitForServiceSeconds 180 -DefaultActionAccountCredential $DefaultActionCred -InstallCredential $InstallCred
 
 
    .INPUTS
    Function accepts an array of computer names as piped input.
    .OUTPUTS
    Outputs objects for each installation attempt.
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/2019/11/12/scomhelper/
    History:
    2020.05.13 - First version
     
    For additional info on manual installation parameters see: https://docs.microsoft.com/en-us/system-center/scom/manage-deploy-windows-agent-manually?view=sc-om-2019
     
#>

Function Deploy-SCOMAgent {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
      SupportsShouldProcess=$true, 
      PositionalBinding=$false,
      HelpUri = 'http://www.microsoft.com/',
  ConfirmImpact='Medium')]
  Param (
    # The account that will remotely execute the installation of the agent. This function relies on WSMAN therefore this credential must have rights to PSRemoting access to the target machines. This is not the Microsoft Monitoring Agent (HealthService) service credentials. Default: <current user creds>
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [Alias("Credential")]
    [pscredential]$InstallCredential,


    # {0|1} Indicates whether the management group settings properties will be set on the command line. Use 0 if you want to set the properties at the command line. Use 1 to use the management group settings from Active Directory.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=1,
    ParameterSetName='Parameter Set 1')]
    [ValidateSet('0','1')]
    [string]$USE_SETTINGS_FROM_AD = 0,

    # {0|1} If USE_SETTINGS_FROM_AD=1, then USE_MANUALLY_SPECIFIED_SETTINGS must equal 0.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=2,
    ParameterSetName='Parameter Set 1')]
    [ValidateSet('0','1')]
    [string]$USE_MANUALLY_SPECIFIED_SETTINGS = 1,

    # Specifies the management group (name) that will manage the computer.
    [Parameter(Mandatory=$true, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=3,
    ParameterSetName='Parameter Set 1')]
    [string]$MANAGEMENT_GROUP,


    # Specifies the fully qualified domain name for the management server. To use a gateway server, enter the gateway server FQDN as MANAGEMENT_SERVER_DNS.
    [Parameter(Mandatory=$true, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=4,
    ParameterSetName='Parameter Set 1')]
    [ValidatePattern('(?=^.{4,253}$)(^((?!-)[a-zA-Z0-9-]{1,63}(?<!-)\.)+[a-zA-Z]{2,63}$)')]
    [string]$MANAGEMENT_SERVER_DNS,


    # Use this parameter if the computer's DNS and Active Directory names differ to set to the fully qualified Active Directory Domain Services name.
    [Parameter(Mandatory=$true, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=5,
    ParameterSetName='Parameter Set 1')]
    [ValidatePattern('(?=^.{4,253}$)(^((?!-)[a-zA-Z0-9-]{1,63}(?<!-)\.)+[a-zA-Z]{2,63}$)')]
    [string]$MANAGEMENT_SERVER_AD_NAME,


    # Sets the health service port number. It is rare to modify this and not advised.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=6,
    ParameterSetName='Parameter Set 1')]
    [ValidateRange(1025,65535)]
    [int]$SECURE_PORT = 5723,


    # Optional parameter. Use this parameter if you want to install the agent to a path other than the default installation path. Note that \Agent will be appended to this value.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=7,
    ParameterSetName='Parameter Set 1')]
    [System.IO.FileInfo]$INSTALLDIR,


    # Indicates whether the service will run as a specified user account (0) or the default 'LocalSystem' account (1). Default=1.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=8,
    ParameterSetName='Parameter Set 1')]
    [ValidateSet('0','1')]
    [string]$ACTIONS_USE_COMPUTER_ACCOUNT=1,


    # This is the Microsoft Monitoring Agent (HealthService) Default Action Account credentials if other than 'LocalSystem'. Use of an alternate credential is very rare and typically unnecessary. Use with caution.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=9,
    ParameterSetName='Parameter Set 1')]
    [pscredential]$DefaultActionAccountCredential,
        
        
    # {0|1} Optional parameter. Installs the Operations Manager agent without .NET Application Performance Monitoring. If you are using AVIcode 5.7, NOAPM=1 leaves the AVIcode agent in place. If you are using AVIcode 5.7 and install the Operations Manager agent by using momagent.msi without NOAPM=1, the AVIcode agent will not work correctly and an alert will be generated.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=10,
    ParameterSetName='Parameter Set 1')]
    [ValidateSet('0','1')]
    [string]$NOAPM=1,


    # One or more target computers FQDN, comma-separated.
    [Parameter(Mandatory=$True, 
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$true, 
        ValueFromRemainingArguments=$false, 
        Position=11,
    ParameterSetName='Parameter Set 1')]
    [string[]]$TargetComputer,


    # The full path to the agent installation file. Default = '<SCOM Mgmt Server Install Directory>\AgentManagement\amd64\MOMAgent.msi'
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=12,
    ParameterSetName='Parameter Set 1')]
    [System.IO.FileInfo]$AgentMSIPath,


    # The number of seconds to wait for the service to appear after installation before returning from the remote PSSession. Default:180.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=13,
    ParameterSetName='Parameter Set 1')]
    [int]$WaitForServiceSeconds = 180
  )

  #region Begin
  Begin {  

    $hash = @{}
    $hash.msi = $null
    $hash.outDir = @((Join-Path $env:Windir "Temp"),'C:\Temp')
    If (-NOT $AgentMSIPath){
      $hash.InstallDirectory =  (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Microsoft Operations Manager\3.0\Setup" -Name "InstallDirectory").InstallDirectory
      $hash.agentPath = (Join-Path $hash.InstallDirectory 'AgentManagement\amd64\MOMAgent.msi' )
    }
    Try{
      $hash.fileObj = Get-Item -Path $hash.agentPath -ErrorAction STOP
      $hash.msi = [System.IO.File]::ReadAllBytes($hash.fileObj.FullName)
    } Catch {
      Throw "Failure getting MOMAgent.msi from path: [$($hash.agentPath)]! Either specify the correct path to the file or put the file at this location. "
      Return 1
    }
    # amount of time to wait for HealthService service to enumerate in Services before returning from Scriptblock
    $hash.WaitForServiceSeconds = $WaitForServiceSeconds
    [bool]$hash.WhatIf = (-Not ($PSCmdlet.ShouldProcess($hash)))
    
    $Params = @{
      USE_SETTINGS_FROM_AD = $USE_SETTINGS_FROM_AD
      USE_MANUALLY_SPECIFIED_SETTINGS = $USE_MANUALLY_SPECIFIED_SETTINGS
      MANAGEMENT_GROUP = $MANAGEMENT_GROUP
      MANAGEMENT_SERVER_DNS = $MANAGEMENT_SERVER_DNS
      MANAGEMENT_SERVER_AD_NAME = $MANAGEMENT_SERVER_AD_NAME
      SECURE_PORT = $SECURE_PORT
      INSTALLDIR = $INSTALLDIR
      ACTIONS_USE_COMPUTER_ACCOUNT = $ACTIONS_USE_COMPUTER_ACCOUNT
      NOAPM  = $NOAPM
    }

    If ($DefaultActionAccountCredential){
      $Params.ACTIONSUSER = $DefaultActionAccountCredential.GetNetworkCredential().UserName
      Try {
        $Params.ACTIONSDOMAIN = $DefaultActionAccountCredential.GetNetworkCredential().Domain
      }Catch{
        $Params.ACTIONSDOMAIN = ''
      }
      $Params.ACTIONSPASSWORD = $DefaultActionAccountCredential.GetNetworkCredential().Password
    }
    #-------------------------------------------------
    #region InstallationBlock
    $scriptBlock = [scriptblock]{
      Param (
        $hash,
        $Params
      )
      # Buid custom object to return. Initially assume it doesn't exist.
      $obj = New-Object -TypeName PSCUSTOMOBJECT
      $obj | Add-Member -MemberType NoteProperty -Name 'ServerName' -Value "$($env:COMPUTERNAME).$($env:USERDNSDOMAIN)"
      $obj | Add-Member -MemberType NoteProperty -Name 'ServiceName' -Value 'HealthService'
      $obj | Add-Member -MemberType NoteProperty -Name 'Status' -Value 'NOT FOUND'
      $obj | Add-Member -MemberType NoteProperty -Name 'Comment' -Value ''

      Try{
        # If service already exists, do nothing. Return.
        $Service = Get-Service -Name HealthService -ErrorAction Stop
        $obj.Status = $Service.Status.ToString()
        $obj.Comment = "Service already exists. No action taken."
        Return $obj
      }Catch {
        # Assume service doesn't appear yet. Proceed to install.
      }

      # Write the .msi to temp dir on remote target.
      Try {
        $exePath = (Join-Path $hash.outDir[0] $hash.fileObj.Name)
        [System.IO.File]::WriteAllBytes($exePath, $hash.msi)
      }Catch {
        Try{
          $exePath = (Join-Path $hash.outDir[1] $hash.fileObj.Name)
          [System.IO.File]::WriteAllBytes($exePath, $hash.msi)
        }
        Catch {
          $obj.Comment = "Remote connection successful but failed to copy bits to path: [$($exePath)]. Service not installed. Error was $_"
          Return $obj
        }
      }

      #region Build CMD string
      $CMD = "msiexec.exe /i $exePath /passive /l*v $(Split-Path $exePath -Parent)\OMAgentinstall.log"
      ForEach ($Key in @($Params.Keys)) {
        switch ($Key)
        {
          # Wrap InstallDir in double quotes
          {($Key -match 'INSTALLDIR') -AND ($Params.$Key.LENGTH -ge 1) } {$CMD += " $($Key)=```"$($Params.$_)```""; Continue; }
        
          {$Params.$_.LENGTH -ge 1 } {$CMD += " $($Key)=$($Params.$_)"}

          Default {Continue}
        }
      }
      
      # it makes sense to hardcode this param=1
      $CMD += " AcceptEndUserLicenseAgreement=1"
      #endregion Build CMD


      #Install Command
      If ($hash.WhatIf){
        $obj.Status = "Running (WHATIF)"
        $obj.Comment = "(WHATIF) Service installed successfully. Bits were actually written to this path successfully: [$($exePath)]"
        Return $obj
      }
      
      Else {
        # Get down to business!
        Invoke-Expression $CMD
          
        #Set timer
        $ServiceTimer = [System.Diagnostics.Stopwatch]::StartNew()
        Do{
          Try{
            $Service = Get-Service -Name HealthService -ErrorAction Stop
          }Catch {
            # Assume service doesn't appear yet
            Start-Sleep -Seconds 3
          }
        }While (($ServiceTimer.Elapsed.TotalSeconds -lt $hash.WaitForServiceSeconds) -AND  ([string]($Service.Status) -ne "Running") )
        $ServiceTimer.Stop()

        If ([bool]$Service ){
          $obj.Status = [string]($Service.Status)
          $obj.Comment = "Service installed successfully."
        }
        Else {
          $obj.Comment = "Service not installed/found. Command line: `n$($CMD) `n`nTimerSeconds:$($ServiceTimer.Elapsed.TotalSeconds), WaitForServiceSeconds:$($hash.WaitForServiceSeconds), ServiceStatus:$($Service.Status), ServiceRetrievedSuccess:$([bool]$Service)."
        }
      }
      Return $obj
    }
    #endregion #region InstallationBlock
    #-------------------------------------------------

  }
  #endregion Begin




  #region Process
  Process {
  
    # ----------- Configure remote session -----------
    #region ConfigureSessionRemotely
    $SessionConfigName = 'MaxDataSizeMB500'
    $ConfigureSessionBlock = [scriptblock]{
      Param(
        $SessionConfigName 
      )
      Try {
        # If session option is already registered, get it
        $var = Get-PSSessionConfiguration -Name $SessionConfigName -ErrorAction Stop
      }Catch {
        $var = Register-PSSessionConfiguration -Name $SessionConfigName
      }
      $var = Set-PSSessionConfiguration -Name $SessionConfigName -MaximumReceivedDataSizePerCommandMB 500 -MaximumReceivedObjectSizeMB 500
    }
    $SessionParams = @{
      ComputerName = $TargetComputer
    }
    If ($InstallCredential) {
      $SessionParams.Credential = $InstallCredential
    }

    $Session = New-PSSession @SessionParams 
    #Set session config on remote
    Try {
      Invoke-Command -Session $Session -ArgumentList $SessionConfigName -ScriptBlock $ConfigureSessionBlock -ErrorAction Stop
    }
    Catch {
      Write-Warning "WinRM Reset will typically cause an 'I/O operation has been aborted...' message." 
    }

    Remove-PSSession -Session $Session
    #endregion ConfigureSessionRemotely
    #-------------------------------------------------



    #-------------------------------------------------
    #region SetMyLocalSessionConfig
    Try {
      # If session option is already registered, get it
      $var = (Get-PSSessionConfiguration -Name $SessionConfigName -ErrorAction Stop)
    }Catch {
      $var = Register-PSSessionConfiguration -Name $SessionConfigName
    }
    $var = Set-PSSessionConfiguration -Name $SessionConfigName -MaximumReceivedDataSizePerCommandMB 500 -MaximumReceivedObjectSizeMB 500
    #endregion SetMyLocalSessionConfig
    #-------------------------------------------------


    #-------------------------------------------------
    #region PrepareInstallSession
    $SessionName = "MySCOMInstallSession"
    $SessionParams = @{
      Name = $SessionName
      ComputerName = $TargetComputer 
      ConfigurationName = $SessionConfigName
    }
    If ($InstallCredential) {
      $SessionParams.Credential = $InstallCredential
    }
    $Session = New-PSSession @SessionParams 
    #endregion PrepareInstallSession
    #-------------------------------------------------
        

    # Execute installation
    $InvokeParams = @{
      Session = $Session
      ScriptBlock = $scriptBlock
      ArgumentList = @($hash, $Params)
      Verbose = $true
    }
    $Result = Invoke-Command @InvokeParams
    Remove-PSSession -Session $Session

    Return $Result
  }
  #region Process
  
  End {
  
  }
  
} #end Function


#######################################################################
<#
    .SYNOPSIS
    Operations Manager Powershell function to output the effective monitoring configuration.
    .DESCRIPTION
    This will output a generous amount of information about monitors, rules, discoveries and the effective config for a given object (and all hosted/contained objects). CSV or HTML output.
    There exists a similar function in the standard OpsMan PowerShell module but AFAIK, even with the most recent UR10(SCOM2016) update at the time of this writing, it is limited in its output and I don’t find it very useful which is why I wrote this function.
 
    .EXAMPLE
    #
    #Example
    PS C:\> $ComputerName = "DB01.CONTOSO.COM"
    PS C:\> $objs = Get-SCOMClass -Name 'Microsoft.Windows.Computer' | Get-SCOMClassInstance | Where-Object Name -match "$ComputerName"
    PS C:\> $ExportPath = 'C:\Temp\EffectiveConfig\'
    PS C:\> Export-SCOMEffectiveMonitoringConfigurationReport -MonitoringObject $objs -ExportCSVPath $ExportPath -IndividualFiles -SortBy AlertPriority -SortOrder Ascending -Verbose
 
    The commands above will export a CSV file of monitoring configuration for the Windows Computer object and everything hosted on it to a CSV file sorted ascending by AlertPriorty, with verbose output to the screen.
    The command will also output individual CSV files for every contained or hosted object.
 
    .EXAMPLE
    #
    #Example
    PS C:\> $obj = Get-SCOMClass -Name 'Microsoft.SQLServer.2016.DBEngine' | Get-SCOMClassInstance | Where DisplayName -eq 'MySQLInstance'
    PS C:\> $ExportPath = 'C:\Temp\EffectiveConfig\'
    PS C:\> Export-SCOMEffectiveMonitoringConfigurationReport -MonitoringObject $obj -ExportHTMLPath $ExportPath -HTMLFormat Basic
 
    The commands above will output a fancy HTML report for the specified SQL instance(s).
    NOTE: The 'fancy' HTML report should only be used for instances with few hosted objects. This report does not perform well with a high row count.
 
    .EXAMPLE
    #
    #EXAMPLE
    PS C:\> $Name = 'DC3SQLDev01'
    PS C:\> $Date = (Get-date -f "yyyyMMdd_hhmmss")
    PS C:\> $OutFolder = (Join-Path -Path 'D:\Temp' -ChildPath $Date)
    PS C:\> $Computer = Get-SCOMClass -Name 'microsoft.windows.computer' | Get-SCOMClassInstance | Where-Object Name -match $Name
    PS C:\> Export-SCOMEffectiveMonitoringConfigurationReport -MonitoringObject $Computer -ExportCSVPath (Join-Path $OutFolder ("$($Name)_EffectiveConfig_$($Date)")) -SortBy WorkflowId
 
 
    .NOTES
    Author: Tyson Paul
    Original Date: 7-25-2014
    History
    2022.05.25: Seeing duplicate workflows for same instance, one has empty Path.
    2020.07.16: Added another example.
    2020.02.19: Found that override property "Enabled" value (true/false) is case-sensitive and therefore must be lower case for this report to run correctly. Added function to fix existing overrides if they are found while running this function/report. Ended up fixing Start-SCOMOverrideTool.ps1 script too.
    Clean-Name function: improved invalid/illegal character replacement strategy.
    2019.08.26: Major revision.
 
    .PARAMETER MonitoringObject
    Accepts one or more SCOM Monitoring Objects (use Get-SCOMClassInstance to retrieve a monitoring object).
    .PARAMETER ExportCSVPath
    Path/folder to export CSV report(s).
    .PARAMETER ExportHTMLPath
    Path/folder to export HTML report(s).
    .PARAMETER HTMLFormat
    There are two options: Basic and Fancy. Both report styles include a text filter powered by Javascript.
    .PARAMETER IndividualFiles
    Will also output individual report files for each object that may be hosted on or contained by the MonitoringObject(s).
    .PARAMETER SortBy
    Report will be sorted by this column/property.
    .PARAMETER SortOrder
    Ascending or Descending. Default is Ascending.
 
#>

Function Export-SCOMEffectiveMonitoringConfigurationReport {
  [CmdletBinding(DefaultParameterSetName='P1',
      SupportsShouldProcess=$true,
  PositionalBinding=$false)]

  Param
  (

    #Any valid monitoring object(s) (class instance or group)
    [Parameter(Mandatory=$true, ParameterSetName = "P1",
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=0)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [Microsoft.EnterpriseManagement.Monitoring.MonitoringObject[]]$MonitoringObject,


    #Directory to export CSV fil
    [Parameter(Mandatory=$false, ParameterSetName = "P1",
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=1)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$ExportCSVPath,


    #Directory to export HTML file
    [Parameter(Mandatory=$false, ParameterSetName='P1',
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=2)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$ExportHTMLPath,


    #Formatting of HTML export
    [Parameter(Mandatory=$false, ParameterSetName='P1',
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=3)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [ValidateSet("Fancy", "Basic")]
    [string]$HTMLFormat = "Basic",

    #Report rows will be sorted by this Workflow property
    [Parameter(Mandatory=$false, ParameterSetName='P1',
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=4)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [ValidateSet("EntityName",
        "EntityId",
        "Path",
        "ClassName",
        "ClassId",
        "WorkflowName",
        "WorkflowDisplayName",
        "WorkflowType",
        "WorkflowId",
        "WorkflowMPName",
        "Enabled",
        "GeneratesAlert",
        "AlertSeverity",
        "AlertPriority",
        "Overridden",
        "Description",
        "Parameters",
    "Overrides")]
    [string]$SortBy = "EntityId",

    #Formatting of HTML export
    [Parameter(Mandatory=$false, ParameterSetName='P1',
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=5)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [ValidateSet("Ascending", "Descending")]
    [string]$SortOrder = 'Ascending',

    # Will output individual csv or html files for all individual objects in addition to the merged/master file
    [switch]$IndividualFiles

  )

  $TESTING = $FALSE
  <# UNCOMMENT FOR TESTING
      # This will avoid loading Mons/Rules every time while testing.
      WRITE-HOST 'TESTING' -F RED -B YELLOW
      $TESTING = $TRUE
  #>


  ###################################################################
  # Will clean up names/strings with special characters (like URLs and Network paths)
  Function Clean-Name {
    Param(
      [string]$uglyString
    )
    # Remove problematic characters and leading/trailing spaces
    #$prettyString = (($uglyString.Replace(':','_')).Replace('/','_')).Replace('\','_').Trim()
    $prettyString = $uglyString.Split([IO.Path]::GetInvalidFileNameChars()) -join '_'

    # If the string has been modified, output that info
    If ($uglyString -ne $prettyString) {
      Write-Verbose "There was a small problem with the characters in this parameter: [$($uglyString)]..."
      Write-Verbose "Original Name:`t`t$($uglyString)"
      Write-Verbose "Modified Name:`t`t$($prettyString)"
    }

    Return $prettyString
    #>
  }
  ########################################################################################################
  
  Function Build-HTMLBasic {
  
    [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
        SupportsShouldProcess=$true, 
    PositionalBinding=$false)]
    Param (
      #array of workflows to be exported to the Path
      [Parameter(Mandatory=$false, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false, 
          Position=0,
      ParameterSetName='Parameter Set 1')]
      [System.Object[]]$arrWFs,

      #title of HTML doc
      [Parameter(Mandatory=$false, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false, 
          Position=2,
      ParameterSetName='Parameter Set 1')]
      [string]$Title = ""
    )


    $HTML = ''
    $HTML += @"
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
    <head>
        <title>$Title</title>
    </head>
    <body>
        <style type="text/css">
 
a { text-decoration: none; }
a:link, a:visited {
    color: blue;
}
a:hover {
    color: red;
}
.trrow {}
.tg {border-collapse:collapse;border-spacing:1;}
.tg td{font-family:Arial, sans-serif;font-size:14px;padding:6px 6px;border-style:solid;border-width:1px;overflow:hidden;word-break:normal;border-color:black;}
.tg th{font-family:Arial, sans-serif;font-size:14px;font-weight:normal;padding:1px 6px;border-style:solid;border-width:1px;overflow:hidden;word-break:normal;border-color:black;}
.tg .defaultstyle{border-color:inherit;vertical-align:center;text-align:left}
.tg .headerstyle{font-weight:bold;font-size:100%;border-color:inherit;vertical-align:center}
.tg .columnNameStyle{border-color:inherit;vertical-align:center;text-align:center}
 
 
.tg .classnamestyle{font-weight:bold;border-color:inherit;vertical-align:center}
.tg .keystyle{font-weight:bold;color:#fe0000;border-color:inherit;text-align:right;vertical-align:center}
.tg .keypropertystyle{font-weight:bold;color:#fe0000;border-color:inherit;vertical-align:center}
.tg .propertystyle{color:#22771a;border-color:inherit;vertical-align:center;horizontal-align:center}
propertystyle
@media screen and (max-width: 767px)
 
{
    .tg {width: auto !important;}
    .tg col {width: auto !important;}
    .tg-wrap {overflow-x: auto;-webkit-overflow-scrolling: touch;}
}
 
#myInput {
    background-image: url(); /* Add a search icon to input */
    background-position: 10px 12px; /* Position the search icon */
    background-repeat: no-repeat; /* Do not repeat the icon image */
    width: 100%; /* Full-width */
    font-size: 16px; /* Increase font-size */
    padding: 12px 20px 12px 40px; /* Add some padding */
    border: 1px solid #ddd; /* Add a grey border */
    margin-bottom: 6px; /* Add some space below the input */
}
 
#myTable tr {
    /* Add a bottom border to all table rows */
    border-bottom: 1px solid #ddd;
     
}
 
#myTable tr.header, #myTable tr:hover {
    /* Add a grey background color to the table header and on hover */
    background-color: #f1f1f1;
}
 
        </style>
 
<script>
function myFunction() {
  // Declare variables
  var input, filter, table, tr, td1,td2,td3,td4,td5,td6,td7,td8,td9, i;
  input = document.getElementById("myInput");
  filter = input.value.toUpperCase();
  table = document.getElementById("myTable");
  //tr = table.getElementsByTagName("tr");
  tr = table.getElementsByClassName("trrow");
 
  // Loop through all table rows, and hide those who don't match the search query
  for (i = 0; i < tr.length; i++) {
   td1 = tr[i].getElementsByClassName("defaultstyle")[1];
     td2 = tr[i].getElementsByClassName("defaultstyle")[2];
     td3 = tr[i].getElementsByClassName("defaultstyle")[3];
     td4 = tr[i].getElementsByClassName("defaultstyle")[4];
     td5 = tr[i].getElementsByClassName("defaultstyle")[5];
     td6 = tr[i].getElementsByClassName("defaultstyle")[6];
     td7 = tr[i].getElementsByClassName("defaultstyle")[7];
     td8 = tr[i].getElementsByClassName("defaultstyle")[8];
     td9 = tr[i].getElementsByClassName("defaultstyle")[9];
    if (td1) {
      if ( (td1.innerHTML.toUpperCase().indexOf(filter) > -1) || (td2.innerHTML.toUpperCase().indexOf(filter) > -1) || (td3.innerHTML.toUpperCase().indexOf(filter) > -1) || (td4.innerHTML.toUpperCase().indexOf(filter) > -1) || (td5.innerHTML.toUpperCase().indexOf(filter) > -1) || (td6.innerHTML.toUpperCase().indexOf(filter) > -1) || (td7.innerHTML.toUpperCase().indexOf(filter) > -1) || (td8.innerHTML.toUpperCase().indexOf(filter) > -1) || (td9.innerHTML.toUpperCase().indexOf(filter) > -1) ) {
        tr[i].style.display = "";
      } else {
        tr[i].style.display = "none";
      }
    }
  }
}
</script>
 
<input type="text" id="myInput" onkeyup="myFunction()" placeholder="Search for text..">
 
<div class="tg-wrap">
 
<table id="myTable" class="tg">
 
"@



    $headerHTML = ''
    $rows = ''
    ForEach ($header in $arrOrderedHeaders ) {
      $headerHTML += @"
    <th class="headerstyle">$($header)</th>
 
"@

    }
    $HTML += @"
  <tr>
$headerHTML
  </tr>
 
"@


    $i =1
    #Build rows
    ForEach ($WF in $arrWFs[0..($arrWFs.Count -1)]) {
      $percentComplete = "{0:N0}" -f (($i/([math]::Max(($arrWFs.Count),1 )) ) *100)
      Write-Progress -Activity "Building HTML report headers" -Status "$($percentComplete) percent complete." -PercentComplete $percentComplete 
      $rows += @"
  <tr class="trrow">
 
"@

      ForEach ($Header in $arrOrderedHeaders) {
        If ($Header -eq 'Index') {
          $rows += @"
    <td class="defaultstyle">$($i)</td>
 
"@
      
        }
        Else{
          $rows += @"
    <td class="defaultstyle">$($WF.$Header)</td>
 
"@

        }
      }#end ForEach Header

      $rows += @"
  </tr>
 
"@

      $i++
    }#end ForEach WF
    Write-Progress -Activity "Building HTML report" -Status "$($percentComplete) percent complete." -PercentComplete 100 -Completed
    $HTML += @"
$rows
</table>
</div>
</body>
</html>
"@


    Return $HTML

  }# End Build-HTMLBasic
  ###################################################################
  Function Build-HTMLFancyResize {
    [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
        SupportsShouldProcess=$true, 
    PositionalBinding=$false)]
    Param (
      #array of workflows to be exported to the Path
      [Parameter(Mandatory=$false, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false, 
          Position=0,
      ParameterSetName='Parameter Set 1')]
      [System.Object[]]$arrWFs,

      #title of HTML doc
      [Parameter(Mandatory=$false, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false, 
          Position=2,
      ParameterSetName='Parameter Set 1')]
      [string]$Title = ""
    )  


    $HTML = ''
    $HTML += @"
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
    <head>
        <title>$Title</title>
    </head>
 
<style type="text/css">
 
.trrow {}
 
td{
    text-indent:5px;
    color:#444;
    border-bottom:1px solid #bbb;
    border-left:1px solid #bbb;
}
"@


  


    $headerHTML = @"
    <tr>
 
"@

    $rows = ''
    $CSSColumns =''
    [int]$pxFactor = 8
    $maxColumnSize = (50 * $pxFactor)
    #Build CSS style for each column and Header HTML
    ForEach ($header in $arrOrderedHeaders ) {
      #CSS
      #header name width in px
      Try{
        #[int]$headerNamePx = [System.Math]::Max(($header.Length * $pxFactor), (($arrAllWorkflows.$header | Measure-Object -Maximum).Maximum.ToString().Length * $pxFactor) )
        [int]$headerNamePx = [System.Math]::Max(($header.Length * $pxFactor), ([System.Math]::Min($maxColumnSize, (($arrWFs.$header | Where-Object Length -gt 0 | % {$_.Length} ) | Measure-Object -Maximum ).Maximum * $pxFactor ))  )
      } Catch {
        [int]$headerNamePx = ($header.length * $pxFactor)
      }

      $CSSColumns += @"
td.$($header), th.$($header) {
    text-align: left;
    width:$($headerNamePx)px;
    min-width:$([math]::Min(30,($headerNamePx /4)))px;
}
"@


      #Header HTML
      $headerHTML += @"
      <th class="$header">$($header)</th>
 
"@


    }#end ForEach $header
    
    $headerHTML += @"
  </tr>
 
"@


    $HTML += @"
  $CSSColumns
 
"@


    $HTML += @'
 
td.Overrides, th.Overrides{
    border-right:1px solid #2e638e;
}
 
td.bottom{
    border-bottom:1px solid #2e638e;
}
 
td.disabled, th.disabled{
    width:25px;
    text-align: center;
    min-width:25px;
}
 
#myInput {
    background-image: url(); /* Add a search icon to input */
    background-position: 10px 12px; /* Position the search icon */
    background-repeat: no-repeat; /* Do not repeat the icon image */
    width: 100%; /* Full-width */
    font-size: 16px; /* Increase font-size */
    padding: 12px 20px 12px 40px; /* Add some padding */
    border: 1px solid #ddd; /* Add a grey border */
    margin-bottom: 6px; /* Add some space below the input */
}
 
#normal tr:hover {
    /* Add a grey background color to the table header and on hover */
    background-color: #f1f1f1;
}
 
 
body{
    background-color: white;
    text-align:center;
    padding:55px;
}
 
 
.center{
    text-align:left;
}
 
#table{
    width:100%;
    max-width:1600px;
}
 
th{
 background-image:url();
   height:30px;
   background-repeat:no-repeat;
   color:white;
   text-shadow: #012b4d 2px 2px 2px;
   text-align: center;
}
 
 
.grip{
    width:20px;
    height:15px;
    margin-top:-3px;
    background-image:url();
    margin-left:-5px;
    position:relative;
    z-index:88;
    cursor:e-resize;
}
 
.grip:hover{
    background-position-x:-20px;
}
.JCLRLastGrip .grip{
    background-position-y:-18px;
    left:-2px;
}
 
.dragging .grip{
    background-position-x:-40px;
}
 
.sampleText{
    position:relative;
    width:100%;
}
 
.dotted{
    background-image:url();
    background-repeat:repeat-y;
}
 
input.check{
     
}
 
#sample2Txt{
    float:right;
}
label{
    color:#0361ae;
}
      .scroll{
          overflow:scroll;
          max-width:100%;
          position:relative;
      }
  </style>
  <script type="text/javascript">
/*!
 * jQuery JavaScript Library v1.6.2
 * http://jquery.com/
 *
 * Copyright 2011, John Resig
 * Dual licensed under the MIT or GPL Version 2 licenses.
 * http://jquery.org/license
 *
 * Includes Sizzle.js
 * http://sizzlejs.com/
 * Copyright 2011, The Dojo Foundation
 * Released under the MIT, BSD, and GPL Licenses.
 *
 * Date: Thu Jun 30 14:16:56 2011 -0400
 */
(function(a,b){function cv(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function cs(a){if(!cg[a]){var b=c.body,d=f("<"+a+">").appendTo(b),e=d.css("display");d.remove();if(e==="none"||e===""){ch||(ch=c.createElement("iframe"),ch.frameBorder=ch.width=ch.height=0),b.appendChild(ch);if(!ci||!ch.createElement)ci=(ch.contentWindow||ch.contentDocument).document,ci.write((c.compatMode==="CSS1Compat"?"<!doctype html>":"")+"<html><body>"),ci.close();d=ci.createElement(a),ci.body.appendChild(d),e=f.css(d,"display"),b.removeChild(ch)}cg[a]=e}return cg[a]}function cr(a,b){var c={};f.each(cm.concat.apply([],cm.slice(0,b)),function(){c[this]=a});return c}function cq(){cn=b}function cp(){setTimeout(cq,0);return cn=f.now()}function cf(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function ce(){try{return new a.XMLHttpRequest}catch(b){}}function b$(a,c){a.dataFilter&&(c=a.dataFilter(c,a.dataType));var d=a.dataTypes,e={},g,h,i=d.length,j,k=d[0],l,m,n,o,p;for(g=1;g<i;g++){if(g===1)for(h in a.converters)typeof h=="string"&&(e[h.toLowerCase()]=a.converters[h]);l=k,k=d[g];if(k==="*")k=l;else if(l!=="*"&&l!==k){m=l+" "+k,n=e[m]||e["* "+k];if(!n){p=b;for(o in e){j=o.split(" ");if(j[0]===l||j[0]==="*"){p=e[j[1]+" "+k];if(p){o=e[o],o===!0?n=p:p===!0&&(n=o);break}}}}!n&&!p&&f.error("No conversion from "+m.replace(" "," to ")),n!==!0&&(c=n?n(c):p(o(c)))}}return c}function bZ(a,c,d){var e=a.contents,f=a.dataTypes,g=a.responseFields,h,i,j,k;for(i in g)i in d&&(c[g[i]]=d[i]);while(f[0]==="*")f.shift(),h===b&&(h=a.mimeType||c.getResponseHeader("content-type"));if(h)for(i in e)if(e[i]&&e[i].test(h)){f.unshift(i);break}if(f[0]in d)j=f[0];else{for(i in d){if(!f[0]||a.converters[i+" "+f[0]]){j=i;break}k||(k=i)}j=j||k}if(j){j!==f[0]&&f.unshift(j);return d[j]}}function bY(a,b,c,d){if(f.isArray(b))f.each(b,function(b,e){c||bC.test(a)?d(a,e):bY(a+"["+(typeof e=="object"||f.isArray(e)?b:"")+"]",e,c,d)});else if(!c&&b!=null&&typeof b=="object")for(var e in b)bY(a+"["+e+"]",b[e],c,d);else d(a,b)}function bX(a,c,d,e,f,g){f=f||c.dataTypes[0],g=g||{},g[f]=!0;var h=a[f],i=0,j=h?h.length:0,k=a===bR,l;for(;i<j&&(k||!l);i++)l=h[i](c,d,e),typeof l=="string"&&(!k||g[l]?l=b:(c.dataTypes.unshift(l),l=bX(a,c,d,e,l,g)));(k||!l)&&!g["*"]&&(l=bX(a,c,d,e,"*",g));return l}function bW(a){return function(b,c){typeof b!="string"&&(c=b,b="*");if(f.isFunction(c)){var d=b.toLowerCase().split(bN),e=0,g=d.length,h,i,j;for(;e<g;e++)h=d[e],j=/^\+/.test(h),j&&(h=h.substr(1)||"*"),i=a[h]=a[h]||[],i[j?"unshift":"push"](c)}}}function bA(a,b,c){var d=b==="width"?a.offsetWidth:a.offsetHeight,e=b==="width"?bv:bw;if(d>0){c!=="border"&&f.each(e,function(){c||(d-=parseFloat(f.css(a,"padding"+this))||0),c==="margin"?d+=parseFloat(f.css(a,c+this))||0:d-=parseFloat(f.css(a,"border"+this+"Width"))||0});return d+"px"}d=bx(a,b,b);if(d<0||d==null)d=a.style[b]||0;d=parseFloat(d)||0,c&&f.each(e,function(){d+=parseFloat(f.css(a,"padding"+this))||0,c!=="padding"&&(d+=parseFloat(f.css(a,"border"+this+"Width"))||0),c==="margin"&&(d+=parseFloat(f.css(a,c+this))||0)});return d+"px"}function bm(a,b){b.src?f.ajax({url:b.src,async:!1,dataType:"script"}):f.globalEval((b.text||b.textContent||b.innerHTML||"").replace(be,"/*$0*/")),b.parentNode&&b.parentNode.removeChild(b)}function bl(a){f.nodeName(a,"input")?bk(a):"getElementsByTagName"in a&&f.grep(a.getElementsByTagName("input"),bk)}function bk(a){if(a.type==="checkbox"||a.type==="radio")a.defaultChecked=a.checked}function bj(a){return"getElementsByTagName"in a?a.getElementsByTagName("*"):"querySelectorAll"in a?a.querySelectorAll("*"):[]}function bi(a,b){var c;if(b.nodeType===1){b.clearAttributes&&b.clearAttributes(),b.mergeAttributes&&b.mergeAttributes(a),c=b.nodeName.toLowerCase();if(c==="object")b.outerHTML=a.outerHTML;else if(c!=="input"||a.type!=="checkbox"&&a.type!=="radio"){if(c==="option")b.selected=a.defaultSelected;else if(c==="input"||c==="textarea")b.defaultValue=a.defaultValue}else a.checked&&(b.defaultChecked=b.checked=a.checked),b.value!==a.value&&(b.value=a.value);b.removeAttribute(f.expando)}}function bh(a,b){if(b.nodeType===1&&!!f.hasData(a)){var c=f.expando,d=f.data(a),e=f.data(b,d);if(d=d[c]){var g=d.events;e=e[c]=f.extend({},d);if(g){delete e.handle,e.events={};for(var h in g)for(var i=0,j=g[h].length;i<j;i++)f.event.add(b,h+(g[h][i].namespace?".":"")+g[h][i].namespace,g[h][i],g[h][i].data)}}}}function bg(a,b){return f.nodeName(a,"table")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function W(a,b,c){b=b||0;if(f.isFunction(b))return f.grep(a,function(a,d){var e=!!b.call(a,d,a);return e===c});if(b.nodeType)return f.grep(a,function(a,d){return a===b===c});if(typeof b=="string"){var d=f.grep(a,function(a){return a.nodeType===1});if(R.test(b))return f.filter(b,d,!c);b=f.filter(b,d)}return f.grep(a,function(a,d){return f.inArray(a,b)>=0===c})}function V(a){return!a||!a.parentNode||a.parentNode.nodeType===11}function N(a,b){return(a&&a!=="*"?a+".":"")+b.replace(z,"`").replace(A,"&")}function M(a){var b,c,d,e,g,h,i,j,k,l,m,n,o,p=[],q=[],r=f._data(this,"events");if(!(a.liveFired===this||!r||!r.live||a.target.disabled||a.button&&a.type==="click")){a.namespace&&(n=new RegExp("(^|\\.)"+a.namespace.split(".").join("\\.(?:.*\\.)?")+"(\\.|$)")),a.liveFired=this;var s=r.live.slice(0);for(i=0;i<s.length;i++)g=s[i],g.origType.replace(x,"")===a.type?q.push(g.selector):s.splice(i--,1);e=f(a.target).closest(q,a.currentTarget);for(j=0,k=e.length;j<k;j++){m=e[j];for(i=0;i<s.length;i++){g=s[i];if(m.selector===g.selector&&(!n||n.test(g.namespace))&&!m.elem.disabled){h=m.elem,d=null;if(g.preType==="mouseenter"||g.preType==="mouseleave")a.type=g.preType,d=f(a.relatedTarget).closest(g.selector)[0],d&&f.contains(h,d)&&(d=h);(!d||d!==h)&&p.push({elem:h,handleObj:g,level:m.level})}}}for(j=0,k=p.length;j<k;j++){e=p[j];if(c&&e.level>c)break;a.currentTarget=e.elem,a.data=e.handleObj.data,a.handleObj=e.handleObj,o=e.handleObj.origHandler.apply(e.elem,arguments);if(o===!1||a.isPropagationStopped()){c=e.level,o===!1&&(b=!1);if(a.isImmediatePropagationStopped())break}}return b}}function K(a,c,d){var e=f.extend({},d[0]);e.type=a,e.originalEvent={},e.liveFired=b,f.event.handle.call(c,e),e.isDefaultPrevented()&&d[0].preventDefault()}function E(){return!0}function D(){return!1}function m(a,c,d){var e=c+"defer",g=c+"queue",h=c+"mark",i=f.data(a,e,b,!0);i&&(d==="queue"||!f.data(a,g,b,!0))&&(d==="mark"||!f.data(a,h,b,!0))&&setTimeout(function(){!f.data(a,g,b,!0)&&!f.data(a,h,b,!0)&&(f.removeData(a,e,!0),i.resolve())},0)}function l(a){for(var b in a)if(b!=="toJSON")return!1;return!0}function k(a,c,d){if(d===b&&a.nodeType===1){var e="data-"+c.replace(j,"$1-$2").toLowerCase();d=a.getAttribute(e);if(typeof d=="string"){try{d=d==="true"?!0:d==="false"?!1:d==="null"?null:f.isNaN(d)?i.test(d)?f.parseJSON(d):d:parseFloat(d)}catch(g){}f.data(a,c,d)}else d=b}return d}var c=a.document,d=a.navigator,e=a.location,f=function(){function J(){if(!e.isReady){try{c.documentElement.doScroll("left")}catch(a){setTimeout(J,1);return}e.ready()}}var e=function(a,b){return new e.fn.init(a,b,h)},f=a.jQuery,g=a.$,h,i=/^(?:[^<]*(<[\w\W]+>)[^>]*$|#([\w\-]*)$)/,j=/\S/,k=/^\s+/,l=/\s+$/,m=/\d/,n=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,o=/^[\],:{}\s]*$/,p=/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g,q=/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,r=/(?:^|:|,)(?:\s*\[)+/g,s=/(webkit)[ \/]([\w.]+)/,t=/(opera)(?:.*version)?[ \/]([\w.]+)/,u=/(msie) ([\w.]+)/,v=/(mozilla)(?:.*? rv:([\w.]+))?/,w=/-([a-z])/ig,x=function(a,b){return b.toUpperCase()},y=d.userAgent,z,A,B,C=Object.prototype.toString,D=Object.prototype.hasOwnProperty,E=Array.prototype.push,F=Array.prototype.slice,G=String.prototype.trim,H=Array.prototype.indexOf,I={};e.fn=e.prototype={constructor:e,init:function(a,d,f){var g,h,j,k;if(!a)return this;if(a.nodeType){this.context=this[0]=a,this.length=1;return this}if(a==="body"&&!d&&c.body){this.context=c,this[0]=c.body,this.selector=a,this.length=1;return this}if(typeof a=="string"){a.charAt(0)!=="<"||a.charAt(a.length-1)!==">"||a.length<3?g=i.exec(a):g=[null,a,null];if(g&&(g[1]||!d)){if(g[1]){d=d instanceof e?d[0]:d,k=d?d.ownerDocument||d:c,j=n.exec(a),j?e.isPlainObject(d)?(a=[c.createElement(j[1])],e.fn.attr.call(a,d,!0)):a=[k.createElement(j[1])]:(j=e.buildFragment([g[1]],[k]),a=(j.cacheable?e.clone(j.fragment):j.fragment).childNodes);return e.merge(this,a)}h=c.getElementById(g[2]);if(h&&h.parentNode){if(h.id!==g[2])return f.find(a);this.length=1,this[0]=h}this.context=c,this.selector=a;return this}return!d||d.jquery?(d||f).find(a):this.constructor(d).find(a)}if(e.isFunction(a))return f.ready(a);a.selector!==b&&(this.selector=a.selector,this.context=a.context);return e.makeArray(a,this)},selector:"",jquery:"1.6.2",length:0,size:function(){return this.length},toArray:function(){return F.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this[this.length+a]:this[a]},pushStack:function(a,b,c){var d=this.constructor();e.isArray(a)?E.apply(d,a):e.merge(d,a),d.prevObject=this,d.context=this.context,b==="find"?d.selector=this.selector+(this.selector?" ":"")+c:b&&(d.selector=this.selector+"."+b+"("+c+")");return d},each:function(a,b){return e.each(this,a,b)},ready:function(a){e.bindReady(),A.done(a);return this},eq:function(a){return a===-1?this.slice(a):this.slice(a,+a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(F.apply(this,arguments),"slice",F.call(arguments).join(","))},map:function(a){return this.pushStack(e.map(this,function(b,c){return a.call(b,c,b)}))},end:function(){return this.prevObject||this.constructor(null)},push:E,sort:[].sort,splice:[].splice},e.fn.init.prototype=e.fn,e.extend=e.fn.extend=function(){var a,c,d,f,g,h,i=arguments[0]||{},j=1,k=arguments.length,l=!1;typeof i=="boolean"&&(l=i,i=arguments[1]||{},j=2),typeof i!="object"&&!e.isFunction(i)&&(i={}),k===j&&(i=this,--j);for(;j<k;j++)if((a=arguments[j])!=null)for(c in a){d=i[c],f=a[c];if(i===f)continue;l&&f&&(e.isPlainObject(f)||(g=e.isArray(f)))?(g?(g=!1,h=d&&e.isArray(d)?d:[]):h=d&&e.isPlainObject(d)?d:{},i[c]=e.extend(l,h,f)):f!==b&&(i[c]=f)}return i},e.extend({noConflict:function(b){a.$===e&&(a.$=g),b&&a.jQuery===e&&(a.jQuery=f);return e},isReady:!1,readyWait:1,holdReady:function(a){a?e.readyWait++:e.ready(!0)},ready:function(a){if(a===!0&&!--e.readyWait||a!==!0&&!e.isReady){if(!c.body)return setTimeout(e.ready,1);e.isReady=!0;if(a!==!0&&--e.readyWait>0)return;A.resolveWith(c,[e]),e.fn.trigger&&e(c).trigger("ready").unbind("ready")}},bindReady:function(){if(!A){A=e._Deferred();if(c.readyState==="complete")return setTimeout(e.ready,1);if(c.addEventListener)c.addEventListener("DOMContentLoaded",B,!1),a.addEventListener("load",e.ready,!1);else if(c.attachEvent){c.attachEvent("onreadystatechange",B),a.attachEvent("onload",e.ready);var b=!1;try{b=a.frameElement==null}catch(d){}c.documentElement.doScroll&&b&&J()}}},isFunction:function(a){return e.type(a)==="function"},isArray:Array.isArray||function(a){return e.type(a)==="array"},isWindow:function(a){return a&&typeof a=="object"&&"setInterval"in a},isNaN:function(a){return a==null||!m.test(a)||isNaN(a)},type:function(a){return a==null?String(a):I[C.call(a)]||"object"},isPlainObject:function(a){if(!a||e.type(a)!=="object"||a.nodeType||e.isWindow(a))return!1;if(a.constructor&&!D.call(a,"constructor")&&!D.call(a.constructor.prototype,"isPrototypeOf"))return!1;var c;for(c in a);return c===b||D.call(a,c)},isEmptyObject:function(a){for(var b in a)return!1;return!0},error:function(a){throw a},parseJSON:function(b){if(typeof b!="string"||!b)return null;b=e.trim(b);if(a.JSON&&a.JSON.parse)return a.JSON.parse(b);if(o.test(b.replace(p,"@").replace(q,"]").replace(r,"")))return(new Function("return "+b))();e.error("Invalid JSON: "+b)},parseXML:function(b,c,d){a.DOMParser?(d=new DOMParser,c=d.parseFromString(b,"text/xml")):(c=new ActiveXObject("Microsoft.XMLDOM"),c.async="false",c.loadXML(b)),d=c.documentElement,(!d||!d.nodeName||d.nodeName==="parsererror")&&e.error("Invalid XML: "+b);return c},noop:function(){},globalEval:function(b){b&&j.test(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(w,x)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toUpperCase()===b.toUpperCase()},each:function(a,c,d){var f,g=0,h=a.length,i=h===b||e.isFunction(a);if(d){if(i){for(f in a)if(c.apply(a[f],d)===!1)break}else for(;g<h;)if(c.apply(a[g++],d)===!1)break}else if(i){for(f in a)if(c.call(a[f],f,a[f])===!1)break}else for(;g<h;)if(c.call(a[g],g,a[g++])===!1)break;return a},trim:G?function(a){return a==null?"":G.call(a)}:function(a){return a==null?"":(a+"").replace(k,"").replace(l,"")},makeArray:function(a,b){var c=b||[];if(a!=null){var d=e.type(a);a.length==null||d==="string"||d==="function"||d==="regexp"||e.isWindow(a)?E.call(c,a):e.merge(c,a)}return c},inArray:function(a,b){if(H)return H.call(b,a);for(var c=0,d=b.length;c<d;c++)if(b[c]===a)return c;return-1},merge:function(a,c){var d=a.length,e=0;if(typeof c.length=="number")for(var f=c.length;e<f;e++)a[d++]=c[e];else while(c[e]!==b)a[d++]=c[e++];a.length=d;return a},grep:function(a,b,c){var d=[],e;c=!!c;for(var f=0,g=a.length;f<g;f++)e=!!b(a[f],f),c!==e&&d.push(a[f]);return d},map:function(a,c,d){var f,g,h=[],i=0,j=a.length,k=a instanceof e||j!==b&&typeof j=="number"&&(j>0&&a[0]&&a[j-1]||j===0||e.isArray(a));if(k)for(;i<j;i++)f=c(a[i],i,d),f!=null&&(h[h.length]=f);else for(g in a)f=c(a[g],g,d),f!=null&&(h[h.length]=f);return h.concat.apply([],h)},guid:1,proxy:function(a,c){if(typeof c=="string"){var d=a[c];c=a,a=d}if(!e.isFunction(a))return b;var f=F.call(arguments,2),g=function(){return a.apply(c,f.concat(F.call(arguments)))};g.guid=a.guid=a.guid||g.guid||e.guid++;return g},access:function(a,c,d,f,g,h){var i=a.length;if(typeof c=="object"){for(var j in c)e.access(a,j,c[j],f,g,d);return a}if(d!==b){f=!h&&f&&e.isFunction(d);for(var k=0;k<i;k++)g(a[k],c,f?d.call(a[k],k,g(a[k],c)):d,h);return a}return i?g(a[0],c):b},now:function(){return(new Date).getTime()},uaMatch:function(a){a=a.toLowerCase();var b=s.exec(a)||t.exec(a)||u.exec(a)||a.indexOf("compatible")<0&&v.exec(a)||[];return{browser:b[1]||"",version:b[2]||"0"}},sub:function(){function a(b,c){return new a.fn.init(b,c)}e.extend(!0,a,this),a.superclass=this,a.fn=a.prototype=this(),a.fn.constructor=a,a.sub=this.sub,a.fn.init=function(d,f){f&&f instanceof e&&!(f instanceof a)&&(f=a(f));return e.fn.init.call(this,d,f,b)},a.fn.init.prototype=a.fn;var b=a(c);return a},browser:{}}),e.each("Boolean Number String Function Array Date RegExp Object".split(" "),function(a,b){I["[object "+b+"]"]=b.toLowerCase()}),z=e.uaMatch(y),z.browser&&(e.browser[z.browser]=!0,e.browser.version=z.version),e.browser.webkit&&(e.browser.safari=!0),j.test(" ")&&(k=/^[\s\xA0]+/,l=/[\s\xA0]+$/),h=e(c),c.addEventListener?B=function(){c.removeEventListener("DOMContentLoaded",B,!1),e.ready()}:c.attachEvent&&(B=function(){c.readyState==="complete"&&(c.detachEvent("onreadystatechange",B),e.ready())});return e}(),g="done fail isResolved isRejected promise then always pipe".split(" "),h=[].slice;f.extend({_Deferred:function(){var a=[],b,c,d,e={done:function(){if(!d){var c=arguments,g,h,i,j,k;b&&(k=b,b=0);for(g=0,h=c.length;g<h;g++)i=c[g],j=f.type(i),j==="array"?e.done.apply(e,i):j==="function"&&a.push(i);k&&e.resolveWith(k[0],k[1])}return this},resolveWith:function(e,f){if(!d&&!b&&!c){f=f||[],c=1;try{while(a[0])a.shift().apply(e,f)}finally{b=[e,f],c=0}}return this},resolve:function(){e.resolveWith(this,arguments);return this},isResolved:function(){return!!c||!!b},cancel:function(){d=1,a=[];return this}};return e},Deferred:function(a){var b=f._Deferred(),c=f._Deferred(),d;f.extend(b,{then:function(a,c){b.done(a).fail(c);return this},always:function(){return b.done.apply(b,arguments).fail.apply(this,arguments)},fail:c.done,rejectWith:c.resolveWith,reject:c.resolve,isRejected:c.isResolved,pipe:function(a,c){return f.Deferred(function(d){f.each({done:[a,"resolve"],fail:[c,"reject"]},function(a,c){var e=c[0],g=c[1],h;f.isFunction(e)?b[a](function(){h=e.apply(this,arguments),h&&f.isFunction(h.promise)?h.promise().then(d.resolve,d.reject):d[g](h)}):b[a](d[g])})}).promise()},promise:function(a){if(a==null){if(d)return d;d=a={}}var c=g.length;while(c--)a[g[c]]=b[g[c]];return a}}),b.done(c.cancel).fail(b.cancel),delete b.cancel,a&&a.call(b,b);return b},when:function(a){function i(a){return function(c){b[a]=arguments.length>1?h.call(arguments,0):c,--e||g.resolveWith(g,h.call(b,0))}}var b=arguments,c=0,d=b.length,e=d,g=d<=1&&a&&f.isFunction(a.promise)?a:f.Deferred();if(d>1){for(;c<d;c++)b[c]&&f.isFunction(b[c].promise)?b[c].promise().then(i(c),g.reject):--e;e||g.resolveWith(g,b)}else g!==a&&g.resolveWith(g,d?[a]:[]);return g.promise()}}),f.support=function(){var a=c.createElement("div"),b=c.documentElement,d,e,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u;a.setAttribute("className","t"),a.innerHTML=" <link/><table></table><a href='/a' style='top:1px;float:left;opacity:.55;'>a</a><input type='checkbox'/>",d=a.getElementsByTagName("*"),e=a.getElementsByTagName("a")[0];if(!d||!d.length||!e)return{};g=c.createElement("select"),h=g.appendChild(c.createElement("option")),i=a.getElementsByTagName("input")[0],k={leadingWhitespace:a.firstChild.nodeType===3,tbody:!a.getElementsByTagName("tbody").length,htmlSerialize:!!a.getElementsByTagName("link").length,style:/top/.test(e.getAttribute("style")),hrefNormalized:e.getAttribute("href")==="/a",opacity:/^0.55$/.test(e.style.opacity),cssFloat:!!e.style.cssFloat,checkOn:i.value==="on",optSelected:h.selected,getSetAttribute:a.className!=="t",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0},i.checked=!0,k.noCloneChecked=i.cloneNode(!0).checked,g.disabled=!0,k.optDisabled=!h.disabled;try{delete a.test}catch(v){k.deleteExpando=!1}!a.addEventListener&&a.attachEvent&&a.fireEvent&&(a.attachEvent("onclick",function(){k.noCloneEvent=!1}),a.cloneNode(!0).fireEvent("onclick")),i=c.createElement("input"),i.value="t",i.setAttribute("type","radio"),k.radioValue=i.value==="t",i.setAttribute("checked","checked"),a.appendChild(i),l=c.createDocumentFragment(),l.appendChild(a.firstChild),k.checkClone=l.cloneNode(!0).cloneNode(!0).lastChild.checked,a.innerHTML="",a.style.width=a.style.paddingLeft="1px",m=c.getElementsByTagName("body")[0],o=c.createElement(m?"div":"body"),p={visibility:"hidden",width:0,height:0,border:0,margin:0},m&&f.extend(p,{position:"absolute",left:-1e3,top:-1e3});for(t in p)o.style[t]=p[t];o.appendChild(a),n=m||b,n.insertBefore(o,n.firstChild),k.appendChecked=i.checked,k.boxModel=a.offsetWidth===2,"zoom"in a.style&&(a.style.display="inline",a.style.zoom=1,k.inlineBlockNeedsLayout=a.offsetWidth===2,a.style.display="",a.innerHTML="<div style='width:4px;'></div>",k.shrinkWrapBlocks=a.offsetWidth!==2),a.innerHTML="<table><tr><td style='padding:0;border:0;display:none'></td><td>t</td></tr></table>",q=a.getElementsByTagName("td"),u=q[0].offsetHeight===0,q[0].style.display="",q[1].style.display="none",k.reliableHiddenOffsets=u&&q[0].offsetHeight===0,a.innerHTML="",c.defaultView&&c.defaultView.getComputedStyle&&(j=c.createElement("div"),j.style.width="0",j.style.marginRight="0",a.appendChild(j),k.reliableMarginRight=(parseInt((c.defaultView.getComputedStyle(j,null)||{marginRight:0}).marginRight,10)||0)===0),o.innerHTML="",n.removeChild(o);if(a.attachEvent)for(t in{submit:1,change:1,focusin:1})s="on"+t,u=s in a,u||(a.setAttribute(s,"return;"),u=typeof a[s]=="function"),k[t+"Bubbles"]=u;o=l=g=h=m=j=a=i=null;return k}(),f.boxModel=f.support.boxModel;var i=/^(?:\{.*\}|\[.*\])$/,j=/([a-z])([A-Z])/g;f.extend({cache:{},uuid:0,expando:"jQuery"+(f.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(a){a=a.nodeType?f.cache[a[f.expando]]:a[f.expando];return!!a&&!l(a)},data:function(a,c,d,e){if(!!f.acceptData(a)){var g=f.expando,h=typeof c=="string",i,j=a.nodeType,k=j?f.cache:a,l=j?a[f.expando]:a[f.expando]&&f.expando;if((!l||e&&l&&!k[l][g])&&h&&d===b)return;l||(j?a[f.expando]=l=++f.uuid:l=f.expando),k[l]||(k[l]={},j||(k[l].toJSON=f.noop));if(typeof c=="object"||typeof c=="function")e?k[l][g]=f.extend(k[l][g],c):k[l]=f.extend(k[l],c);i=k[l],e&&(i[g]||(i[g]={}),i=i[g]),d!==b&&(i[f.camelCase(c)]=d);if(c==="events"&&!i[c])return i[g]&&i[g].events;return h?i[f.camelCase(c)]||i[c]:i}},removeData:function(b,c,d){if(!!f.acceptData(b)){var e=f.expando,g=b.nodeType,h=g?f.cache:b,i=g?b[f.expando]:f.expando;if(!h[i])return;if(c){var j=d?h[i][e]:h[i];if(j){delete j[c];if(!l(j))return}}if(d){delete h[i][e];if(!l(h[i]))return}var k=h[i][e];f.support.deleteExpando||h!=a?delete h[i]:h[i]=null,k?(h[i]={},g||(h[i].toJSON=f.noop),h[i][e]=k):g&&(f.support.deleteExpando?delete b[f.expando]:b.removeAttribute?b.removeAttribute(f.expando):b[f.expando]=null)}},_data:function(a,b,c){return f.data(a,b,c,!0)},acceptData:function(a){if(a.nodeName){var b=f.noData[a.nodeName.toLowerCase()];if(b)return b!==!0&&a.getAttribute("classid")===b}return!0}}),f.fn.extend({data:function(a,c){var d=null;if(typeof a=="undefined"){if(this.length){d=f.data(this[0]);if(this[0].nodeType===1){var e=this[0].attributes,g;for(var h=0,i=e.length;h<i;h++)g=e[h].name,g.indexOf("data-")===0&&(g=f.camelCase(g.substring(5)),k(this[0],g,d[g]))}}return d}if(typeof a=="object")return this.each(function(){f.data(this,a)});var j=a.split(".");j[1]=j[1]?"."+j[1]:"";if(c===b){d=this.triggerHandler("getData"+j[1]+"!",[j[0]]),d===b&&this.length&&(d=f.data(this[0],a),d=k(this[0],a,d));return d===b&&j[1]?this.data(j[0]):d}return this.each(function(){var b=f(this),d=[j[0],c];b.triggerHandler("setData"+j[1]+"!",d),f.data(this,a,c),b.triggerHandler("changeData"+j[1]+"!",d)})},removeData:function(a){return this.each(function(){f.removeData(this,a)})}}),f.extend({_mark:function(a,c){a&&(c=(c||"fx")+"mark",f.data(a,c,(f.data(a,c,b,!0)||0)+1,!0))},_unmark:function(a,c,d){a!==!0&&(d=c,c=a,a=!1);if(c){d=d||"fx";var e=d+"mark",g=a?0:(f.data(c,e,b,!0)||1)-1;g?f.data(c,e,g,!0):(f.removeData(c,e,!0),m(c,d,"mark"))}},queue:function(a,c,d){if(a){c=(c||"fx")+"queue";var e=f.data(a,c,b,!0);d&&(!e||f.isArray(d)?e=f.data(a,c,f.makeArray(d),!0):e.push(d));return e||[]}},dequeue:function(a,b){b=b||"fx";var c=f.queue(a,b),d=c.shift(),e;d==="inprogress"&&(d=c.shift()),d&&(b==="fx"&&c.unshift("inprogress"),d.call(a,function(){f.dequeue(a,b)})),c.length||(f.removeData(a,b+"queue",!0),m(a,b,"queue"))}}),f.fn.extend({queue:function(a,c){typeof a!="string"&&(c=a,a="fx");if(c===b)return f.queue(this[0],a);return this.each(function(){var b=f.queue(this,a,c);a==="fx"&&b[0]!=="inprogress"&&f.dequeue(this,a)})},dequeue:function(a){return this.each(function(){f.dequeue(this,a)})},delay:function(a,b){a=f.fx?f.fx.speeds[a]||a:a,b=b||"fx";return this.queue(b,function(){var c=this;setTimeout(function(){f.dequeue(c,b)},a)})},clearQueue:function(a){return this.queue(a||"fx",[])},promise:function(a,c){function m(){--h||d.resolveWith(e,[e])}typeof a!="string"&&(c=a,a=b),a=a||"fx";var d=f.Deferred(),e=this,g=e.length,h=1,i=a+"defer",j=a+"queue",k=a+"mark",l;while(g--)if(l=f.data(e[g],i,b,!0)||(f.data(e[g],j,b,!0)||f.data(e[g],k,b,!0))&&f.data(e[g],i,f._Deferred(),!0))h++,l.done(m);m();return d.promise()}});var n=/[\n\t\r]/g,o=/\s+/,p=/\r/g,q=/^(?:button|input)$/i,r=/^(?:button|input|object|select|textarea)$/i,s=/^a(?:rea)?$/i,t=/^(?:autofocus|autoplay|async|checked|controls|defer|disabled|hidden|loop|multiple|open|readonly|required|scoped|selected)$/i,u=/\:|^on/,v,w;f.fn.extend({attr:function(a,b){return f.access(this,a,b,!0,f.attr)},removeAttr:function(a){return this.each(function(){f.removeAttr(this,a)})},prop:function(a,b){return f.access(this,a,b,!0,f.prop)},removeProp:function(a){a=f.propFix[a]||a;return this.each(function(){try{this[a]=b,delete this[a]}catch(c){}})},addClass:function(a){var b,c,d,e,g,h,i;if(f.isFunction(a))return this.each(function(b){f(this).addClass(a.call(this,b,this.className))});if(a&&typeof a=="string"){b=a.split(o);for(c=0,d=this.length;c<d;c++){e=this[c];if(e.nodeType===1)if(!e.className&&b.length===1)e.className=a;else{g=" "+e.className+" ";for(h=0,i=b.length;h<i;h++)~g.indexOf(" "+b[h]+" ")||(g+=b[h]+" ");e.className=f.trim(g)}}}return this},removeClass:function(a){var c,d,e,g,h,i,j;if(f.isFunction(a))return this.each(function(b){f(this).removeClass(a.call(this,b,this.className))});if(a&&typeof a=="string"||a===b){c=(a||"").split(o);for(d=0,e=this.length;d<e;d++){g=this[d];if(g.nodeType===1&&g.className)if(a){h=(" "+g.className+" ").replace(n," ");for(i=0,j=c.length;i<j;i++)h=h.replace(" "+c[i]+" "," ");g.className=f.trim(h)}else g.className=""}}return this},toggleClass:function(a,b){var c=typeof a,d=typeof b=="boolean";if(f.isFunction(a))return this.each(function(c){f(this).toggleClass(a.call(this,c,this.className,b),b)});return this.each(function(){if(c==="string"){var e,g=0,h=f(this),i=b,j=a.split(o);while(e=j[g++])i=d?i:!h.hasClass(e),h[i?"addClass":"removeClass"](e)}else if(c==="undefined"||c==="boolean")this.className&&f._data(this,"__className__",this.className),this.className=this.className||a===!1?"":f._data(this,"__className__")||""})},hasClass:function(a){var b=" "+a+" ";for(var c=0,d=this.length;c<d;c++)if((" "+this[c].className+" ").replace(n," ").indexOf(b)>-1)return!0;return!1},val:function(a){var c,d,e=this[0];if(!arguments.length){if(e){c=f.valHooks[e.nodeName.toLowerCase()]||f.valHooks[e.type];if(c&&"get"in c&&(d=c.get(e,"value"))!==b)return d;d=e.value;return typeof d=="string"?d.replace(p,""):d==null?"":d}return b}var g=f.isFunction(a);return this.each(function(d){var e=f(this),h;if(this.nodeType===1){g?h=a.call(this,d,e.val()):h=a,h==null?h="":typeof h=="number"?h+="":f.isArray(h)&&(h=f.map(h,function(a){return a==null?"":a+""})),c=f.valHooks[this.nodeName.toLowerCase()]||f.valHooks[this.type];if(!c||!("set"in c)||c.set(this,h,"value")===b)this.value=h}})}}),f.extend({valHooks:{option:{get:function(a){var b=a.attributes.value;return!b||b.specified?a.value:a.text}},select:{get:function(a){var b,c=a.selectedIndex,d=[],e=a.options,g=a.type==="select-one";if(c<0)return null;for(var h=g?c:0,i=g?c+1:e.length;h<i;h++){var j=e[h];if(j.selected&&(f.support.optDisabled?!j.disabled:j.getAttribute("disabled")===null)&&(!j.parentNode.disabled||!f.nodeName(j.parentNode,"optgroup"))){b=f(j).val();if(g)return b;d.push(b)}}if(g&&!d.length&&e.length)return f(e[c]).val();return d},set:function(a,b){var c=f.makeArray(b);f(a).find("option").each(function(){this.selected=f.inArray(f(this).val(),c)>=0}),c.length||(a.selectedIndex=-1);return c}}},attrFn:{val:!0,css:!0,html:!0,text:!0,data:!0,width:!0,height:!0,offset:!0},attrFix:{tabindex:"tabIndex"},attr:function(a,c,d,e){var g=a.nodeType;if(!a||g===3||g===8||g===2)return b;if(e&&c in f.attrFn)return f(a)[c](d);if(!("getAttribute"in a))return f.prop(a,c,d);var h,i,j=g!==1||!f.isXMLDoc(a);j&&(c=f.attrFix[c]||c,i=f.attrHooks[c],i||(t.test(c)?i=w:v&&c!=="className"&&(f.nodeName(a,"form")||u.test(c))&&(i=v)));if(d!==b){if(d===null){f.removeAttr(a,c);return b}if(i&&"set"in i&&j&&(h=i.set(a,d,c))!==b)return h;a.setAttribute(c,""+d);return d}if(i&&"get"in i&&j&&(h=i.get(a,c))!==null)return h;h=a.getAttribute(c);return h===null?b:h},removeAttr:function(a,b){var c;a.nodeType===1&&(b=f.attrFix[b]||b,f.support.getSetAttribute?a.removeAttribute(b):(f.attr(a,b,""),a.removeAttributeNode(a.getAttributeNode(b))),t.test(b)&&(c=f.propFix[b]||b)in a&&(a[c]=!1))},attrHooks:{type:{set:function(a,b){if(q.test(a.nodeName)&&a.parentNode)f.error("type property can't be changed");else if(!f.support.radioValue&&b==="radio"&&f.nodeName(a,"input")){var c=a.value;a.setAttribute("type",b),c&&(a.value=c);return b}}},tabIndex:{get:function(a){var c=a.getAttributeNode("tabIndex");return c&&c.specified?parseInt(c.value,10):r.test(a.nodeName)||s.test(a.nodeName)&&a.href?0:b}},value:{get:function(a,b){if(v&&f.nodeName(a,"button"))return v.get(a,b);return b in a?a.value:null},set:function(a,b,c){if(v&&f.nodeName(a,"button"))return v.set(a,b,c);a.value=b}}},propFix:{tabindex:"tabIndex",readonly:"readOnly","for":"htmlFor","class":"className",maxlength:"maxLength",cellspacing:"cellSpacing",cellpadding:"cellPadding",rowspan:"rowSpan",colspan:"colSpan",usemap:"useMap",frameborder:"frameBorder",contenteditable:"contentEditable"},prop:function(a,c,d){var e=a.nodeType;if(!a||e===3||e===8||e===2)return b;var g,h,i=e!==1||!f.isXMLDoc(a);i&&(c=f.propFix[c]||c,h=f.propHooks[c]);return d!==b?h&&"set"in h&&(g=h.set(a,d,c))!==b?g:a[c]=d:h&&"get"in h&&(g=h.get(a,c))!==b?g:a[c]},propHooks:{}}),w={get:function(a,c){return f.prop(a,c)?c.toLowerCase():b},set:function(a,b,c){var d;b===!1?f.removeAttr(a,c):(d=f.propFix[c]||c,d in a&&(a[d]=!0),a.setAttribute(c,c.toLowerCase()));return c}},f.support.getSetAttribute||(f.attrFix=f.propFix,v=f.attrHooks.name=f.attrHooks.title=f.valHooks.button={get:function(a,c){var d;d=a.getAttributeNode(c);return d&&d.nodeValue!==""?d.nodeValue:b},set:function(a,b,c){var d=a.getAttributeNode(c);if(d){d.nodeValue=b;return b}}},f.each(["width","height"],function(a,b){f.attrHooks[b]=f.extend(f.attrHooks[b],{set:function(a,c){if(c===""){a.setAttribute(b,"auto");return c}}})})),f.support.hrefNormalized||f.each(["href","src","width","height"],function(a,c){f.attrHooks[c]=f.extend(f.attrHooks[c],{get:function(a){var d=a.getAttribute(c,2);return d===null?b:d}})}),f.support.style||(f.attrHooks.style={get:function(a){return a.style.cssText.toLowerCase()||b},set:function(a,b){return a.style.cssText=""+b}}),f.support.optSelected||(f.propHooks.selected=f.extend(f.propHooks.selected,{get:function(a){var b=a.parentNode;b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex)}})),f.support.checkOn||f.each(["radio","checkbox"],function(){f.valHooks[this]={get:function(a){return a.getAttribute("value")===null?"on":a.value}}}),f.each(["radio","checkbox"],function(){f.valHooks[this]=f.extend(f.valHooks[this],{set:function(a,b){if(f.isArray(b))return a.checked=f.inArray(f(a).val(),b)>=0}})});var x=/\.(.*)$/,y=/^(?:textarea|input|select)$/i,z=/\./g,A=/ /g,B=/[^\w\s.|`]/g,C=function(a){return a.replace(B,"\\$&")};f.event={add:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){if(d===!1)d=D;else if(!d)return;var g,h;d.handler&&(g=d,d=g.handler),d.guid||(d.guid=f.guid++);var i=f._data(a);if(!i)return;var j=i.events,k=i.handle;j||(i.events=j={}),k||(i.handle=k=function(a){return typeof f!="undefined"&&(!a||f.event.triggered!==a.type)?f.event.handle.apply(k.elem,arguments):b}),k.elem=a,c=c.split(" ");var l,m=0,n;while(l=c[m++]){h=g?f.extend({},g):{handler:d,data:e},l.indexOf(".")>-1?(n=l.split("."),l=n.shift(),h.namespace=n.slice(0).sort().join(".")):(n=[],h.namespace=""),h.type=l,h.guid||(h.guid=d.guid);var o=j[l],p=f.event.special[l]||{};if(!o){o=j[l]=[];if(!p.setup||p.setup.call(a,e,n,k)===!1)a.addEventListener?a.addEventListener(l,k,!1):a.attachEvent&&a.attachEvent("on"+l,k)}p.add&&(p.add.call(a,h),h.handler.guid||(h.handler.guid=d.guid)),o.push(h),f.event.global[l]=!0}a=null}},global:{},remove:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){d===!1&&(d=D);var g,h,i,j,k=0,l,m,n,o,p,q,r,s=f.hasData(a)&&f._data(a),t=s&&s.events;if(!s||!t)return;c&&c.type&&(d=c.handler,c=c.type);if(!c||typeof c=="string"&&c.charAt(0)==="."){c=c||"";for(h in t)f.event.remove(a,h+c);return}c=c.split(" ");while(h=c[k++]){r=h,q=null,l=h.indexOf(".")<0,m=[],l||(m=h.split("."),h=m.shift(),n=new RegExp("(^|\\.)"+f.map(m.slice(0).sort(),C).join("\\.(?:.*\\.)?")+"(\\.|$)")),p=t[h];if(!p)continue;if(!d){for(j=0;j<p.length;j++){q=p[j];if(l||n.test(q.namespace))f.event.remove(a,r,q.handler,j),p.splice(j--,1)}continue}o=f.event.special[h]||{};for(j=e||0;j<p.length;j++){q=p[j];if(d.guid===q.guid){if(l||n.test(q.namespace))e==null&&p.splice(j--,1),o.remove&&o.remove.call(a,q);if(e!=null)break}}if(p.length===0||e!=null&&p.length===1)(!o.teardown||o.teardown.call(a,m)===!1)&&f.removeEvent(a,h,s.handle),g=null,delete t[h]}if(f.isEmptyObject(t)){var u=s.handle;u&&(u.elem=null),delete s.events,delete s.handle,f.isEmptyObject(s)&&f.removeData(a,b,!0)}}},customEvent:{getData:!0,setData:!0,changeData:!0},trigger:function(c,d,e,g){var h=c.type||c,i=[],j;h.indexOf("!")>=0&&(h=h.slice(0,-1),j=!0),h.indexOf(".")>=0&&(i=h.split("."),h=i.
shift(),i.sort());if(!!e&&!f.event.customEvent[h]||!!f.event.global[h]){c=typeof c=="object"?c[f.expando]?c:new f.Event(h,c):new f.Event(h),c.type=h,c.exclusive=j,c.namespace=i.join("."),c.namespace_re=new RegExp("(^|\\.)"+i.join("\\.(?:.*\\.)?")+"(\\.|$)");if(g||!e)c.preventDefault(),c.stopPropagation();if(!e){f.each(f.cache,function(){var a=f.expando,b=this[a];b&&b.events&&b.events[h]&&f.event.trigger(c,d,b.handle.elem)});return}if(e.nodeType===3||e.nodeType===8)return;c.result=b,c.target=e,d=d!=null?f.makeArray(d):[],d.unshift(c);var k=e,l=h.indexOf(":")<0?"on"+h:"";do{var m=f._data(k,"handle");c.currentTarget=k,m&&m.apply(k,d),l&&f.acceptData(k)&&k[l]&&k[l].apply(k,d)===!1&&(c.result=!1,c.preventDefault()),k=k.parentNode||k.ownerDocument||k===c.target.ownerDocument&&a}while(k&&!c.isPropagationStopped());if(!c.isDefaultPrevented()){var n,o=f.event.special[h]||{};if((!o._default||o._default.call(e.ownerDocument,c)===!1)&&(h!=="click"||!f.nodeName(e,"a"))&&f.acceptData(e)){try{l&&e[h]&&(n=e[l],n&&(e[l]=null),f.event.triggered=h,e[h]())}catch(p){}n&&(e[l]=n),f.event.triggered=b}}return c.result}},handle:function(c){c=f.event.fix(c||a.event);var d=((f._data(this,"events")||{})[c.type]||[]).slice(0),e=!c.exclusive&&!c.namespace,g=Array.prototype.slice.call(arguments,0);g[0]=c,c.currentTarget=this;for(var h=0,i=d.length;h<i;h++){var j=d[h];if(e||c.namespace_re.test(j.namespace)){c.handler=j.handler,c.data=j.data,c.handleObj=j;var k=j.handler.apply(this,g);k!==b&&(c.result=k,k===!1&&(c.preventDefault(),c.stopPropagation()));if(c.isImmediatePropagationStopped())break}}return c.result},props:"altKey attrChange attrName bubbles button cancelable charCode clientX clientY ctrlKey currentTarget data detail eventPhase fromElement handler keyCode layerX layerY metaKey newValue offsetX offsetY pageX pageY prevValue relatedNode relatedTarget screenX screenY shiftKey srcElement target toElement view wheelDelta which".split(" "),fix:function(a){if(a[f.expando])return a;var d=a;a=f.Event(d);for(var e=this.props.length,g;e;)g=this.props[--e],a[g]=d[g];a.target||(a.target=a.srcElement||c),a.target.nodeType===3&&(a.target=a.target.parentNode),!a.relatedTarget&&a.fromElement&&(a.relatedTarget=a.fromElement===a.target?a.toElement:a.fromElement);if(a.pageX==null&&a.clientX!=null){var h=a.target.ownerDocument||c,i=h.documentElement,j=h.body;a.pageX=a.clientX+(i&&i.scrollLeft||j&&j.scrollLeft||0)-(i&&i.clientLeft||j&&j.clientLeft||0),a.pageY=a.clientY+(i&&i.scrollTop||j&&j.scrollTop||0)-(i&&i.clientTop||j&&j.clientTop||0)}a.which==null&&(a.charCode!=null||a.keyCode!=null)&&(a.which=a.charCode!=null?a.charCode:a.keyCode),!a.metaKey&&a.ctrlKey&&(a.metaKey=a.ctrlKey),!a.which&&a.button!==b&&(a.which=a.button&1?1:a.button&2?3:a.button&4?2:0);return a},guid:1e8,proxy:f.proxy,special:{ready:{setup:f.bindReady,teardown:f.noop},live:{add:function(a){f.event.add(this,N(a.origType,a.selector),f.extend({},a,{handler:M,guid:a.handler.guid}))},remove:function(a){f.event.remove(this,N(a.origType,a.selector),a)}},beforeunload:{setup:function(a,b,c){f.isWindow(this)&&(this.onbeforeunload=c)},teardown:function(a,b){this.onbeforeunload===b&&(this.onbeforeunload=null)}}}},f.removeEvent=c.removeEventListener?function(a,b,c){a.removeEventListener&&a.removeEventListener(b,c,!1)}:function(a,b,c){a.detachEvent&&a.detachEvent("on"+b,c)},f.Event=function(a,b){if(!this.preventDefault)return new f.Event(a,b);a&&a.type?(this.originalEvent=a,this.type=a.type,this.isDefaultPrevented=a.defaultPrevented||a.returnValue===!1||a.getPreventDefault&&a.getPreventDefault()?E:D):this.type=a,b&&f.extend(this,b),this.timeStamp=f.now(),this[f.expando]=!0},f.Event.prototype={preventDefault:function(){this.isDefaultPrevented=E;var a=this.originalEvent;!a||(a.preventDefault?a.preventDefault():a.returnValue=!1)},stopPropagation:function(){this.isPropagationStopped=E;var a=this.originalEvent;!a||(a.stopPropagation&&a.stopPropagation(),a.cancelBubble=!0)},stopImmediatePropagation:function(){this.isImmediatePropagationStopped=E,this.stopPropagation()},isDefaultPrevented:D,isPropagationStopped:D,isImmediatePropagationStopped:D};var F=function(a){var b=a.relatedTarget,c=!1,d=a.type;a.type=a.data,b!==this&&(b&&(c=f.contains(this,b)),c||(f.event.handle.apply(this,arguments),a.type=d))},G=function(a){a.type=a.data,f.event.handle.apply(this,arguments)};f.each({mouseenter:"mouseover",mouseleave:"mouseout"},function(a,b){f.event.special[a]={setup:function(c){f.event.add(this,b,c&&c.selector?G:F,a)},teardown:function(a){f.event.remove(this,b,a&&a.selector?G:F)}}}),f.support.submitBubbles||(f.event.special.submit={setup:function(a,b){if(!f.nodeName(this,"form"))f.event.add(this,"click.specialSubmit",function(a){var b=a.target,c=b.type;(c==="submit"||c==="image")&&f(b).closest("form").length&&K("submit",this,arguments)}),f.event.add(this,"keypress.specialSubmit",function(a){var b=a.target,c=b.type;(c==="text"||c==="password")&&f(b).closest("form").length&&a.keyCode===13&&K("submit",this,arguments)});else return!1},teardown:function(a){f.event.remove(this,".specialSubmit")}});if(!f.support.changeBubbles){var H,I=function(a){var b=a.type,c=a.value;b==="radio"||b==="checkbox"?c=a.checked:b==="select-multiple"?c=a.selectedIndex>-1?f.map(a.options,function(a){return a.selected}).join("-"):"":f.nodeName(a,"select")&&(c=a.selectedIndex);return c},J=function(c){var d=c.target,e,g;if(!!y.test(d.nodeName)&&!d.readOnly){e=f._data(d,"_change_data"),g=I(d),(c.type!=="focusout"||d.type!=="radio")&&f._data(d,"_change_data",g);if(e===b||g===e)return;if(e!=null||g)c.type="change",c.liveFired=b,f.event.trigger(c,arguments[1],d)}};f.event.special.change={filters:{focusout:J,beforedeactivate:J,click:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(c==="radio"||c==="checkbox"||f.nodeName(b,"select"))&&J.call(this,a)},keydown:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(a.keyCode===13&&!f.nodeName(b,"textarea")||a.keyCode===32&&(c==="checkbox"||c==="radio")||c==="select-multiple")&&J.call(this,a)},beforeactivate:function(a){var b=a.target;f._data(b,"_change_data",I(b))}},setup:function(a,b){if(this.type==="file")return!1;for(var c in H)f.event.add(this,c+".specialChange",H[c]);return y.test(this.nodeName)},teardown:function(a){f.event.remove(this,".specialChange");return y.test(this.nodeName)}},H=f.event.special.change.filters,H.focus=H.beforeactivate}f.support.focusinBubbles||f.each({focus:"focusin",blur:"focusout"},function(a,b){function e(a){var c=f.event.fix(a);c.type=b,c.originalEvent={},f.event.trigger(c,null,c.target),c.isDefaultPrevented()&&a.preventDefault()}var d=0;f.event.special[b]={setup:function(){d++===0&&c.addEventListener(a,e,!0)},teardown:function(){--d===0&&c.removeEventListener(a,e,!0)}}}),f.each(["bind","one"],function(a,c){f.fn[c]=function(a,d,e){var g;if(typeof a=="object"){for(var h in a)this[c](h,d,a[h],e);return this}if(arguments.length===2||d===!1)e=d,d=b;c==="one"?(g=function(a){f(this).unbind(a,g);return e.apply(this,arguments)},g.guid=e.guid||f.guid++):g=e;if(a==="unload"&&c!=="one")this.one(a,d,e);else for(var i=0,j=this.length;i<j;i++)f.event.add(this[i],a,g,d);return this}}),f.fn.extend({unbind:function(a,b){if(typeof a=="object"&&!a.preventDefault)for(var c in a)this.unbind(c,a[c]);else for(var d=0,e=this.length;d<e;d++)f.event.remove(this[d],a,b);return this},delegate:function(a,b,c,d){return this.live(b,c,d,a)},undelegate:function(a,b,c){return arguments.length===0?this.unbind("live"):this.die(b,null,c,a)},trigger:function(a,b){return this.each(function(){f.event.trigger(a,b,this)})},triggerHandler:function(a,b){if(this[0])return f.event.trigger(a,b,this[0],!0)},toggle:function(a){var b=arguments,c=a.guid||f.guid++,d=0,e=function(c){var e=(f.data(this,"lastToggle"+a.guid)||0)%d;f.data(this,"lastToggle"+a.guid,e+1),c.preventDefault();return b[e].apply(this,arguments)||!1};e.guid=c;while(d<b.length)b[d++].guid=c;return this.click(e)},hover:function(a,b){return this.mouseenter(a).mouseleave(b||a)}});var L={focus:"focusin",blur:"focusout",mouseenter:"mouseover",mouseleave:"mouseout"};f.each(["live","die"],function(a,c){f.fn[c]=function(a,d,e,g){var h,i=0,j,k,l,m=g||this.selector,n=g?this:f(this.context);if(typeof a=="object"&&!a.preventDefault){for(var o in a)n[c](o,d,a[o],m);return this}if(c==="die"&&!a&&g&&g.charAt(0)==="."){n.unbind(g);return this}if(d===!1||f.isFunction(d))e=d||D,d=b;a=(a||"").split(" ");while((h=a[i++])!=null){j=x.exec(h),k="",j&&(k=j[0],h=h.replace(x,""));if(h==="hover"){a.push("mouseenter"+k,"mouseleave"+k);continue}l=h,L[h]?(a.push(L[h]+k),h=h+k):h=(L[h]||h)+k;if(c==="live")for(var p=0,q=n.length;p<q;p++)f.event.add(n[p],"live."+N(h,m),{data:d,selector:m,handler:e,origType:h,origHandler:e,preType:l});else n.unbind("live."+N(h,m),e)}return this}}),f.each("blur focus focusin focusout load resize scroll unload click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup error".split(" "),function(a,b){f.fn[b]=function(a,c){c==null&&(c=a,a=null);return arguments.length>0?this.bind(b,a,c):this.trigger(b)},f.attrFn&&(f.attrFn[b]=!0)}),function(){function u(a,b,c,d,e,f){for(var g=0,h=d.length;g<h;g++){var i=d[g];if(i){var j=!1;i=i[a];while(i){if(i.sizcache===c){j=d[i.sizset];break}if(i.nodeType===1){f||(i.sizcache=c,i.sizset=g);if(typeof b!="string"){if(i===b){j=!0;break}}else if(k.filter(b,[i]).length>0){j=i;break}}i=i[a]}d[g]=j}}}function t(a,b,c,d,e,f){for(var g=0,h=d.length;g<h;g++){var i=d[g];if(i){var j=!1;i=i[a];while(i){if(i.sizcache===c){j=d[i.sizset];break}i.nodeType===1&&!f&&(i.sizcache=c,i.sizset=g);if(i.nodeName.toLowerCase()===b){j=i;break}i=i[a]}d[g]=j}}}var a=/((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^\[\]]*\]|['"][^'"]*['"]|[^\[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,d=0,e=Object.prototype.toString,g=!1,h=!0,i=/\\/g,j=/\W/;[0,0].sort(function(){h=!1;return 0});var k=function(b,d,f,g){f=f||[],d=d||c;var h=d;if(d.nodeType!==1&&d.nodeType!==9)return[];if(!b||typeof b!="string")return f;var i,j,n,o,q,r,s,t,u=!0,w=k.isXML(d),x=[],y=b;do{a.exec(""),i=a.exec(y);if(i){y=i[3],x.push(i[1]);if(i[2]){o=i[3];break}}}while(i);if(x.length>1&&m.exec(b))if(x.length===2&&l.relative[x[0]])j=v(x[0]+x[1],d);else{j=l.relative[x[0]]?[d]:k(x.shift(),d);while(x.length)b=x.shift(),l.relative[b]&&(b+=x.shift()),j=v(b,j)}else{!g&&x.length>1&&d.nodeType===9&&!w&&l.match.ID.test(x[0])&&!l.match.ID.test(x[x.length-1])&&(q=k.find(x.shift(),d,w),d=q.expr?k.filter(q.expr,q.set)[0]:q.set[0]);if(d){q=g?{expr:x.pop(),set:p(g)}:k.find(x.pop(),x.length===1&&(x[0]==="~"||x[0]==="+")&&d.parentNode?d.parentNode:d,w),j=q.expr?k.filter(q.expr,q.set):q.set,x.length>0?n=p(j):u=!1;while(x.length)r=x.pop(),s=r,l.relative[r]?s=x.pop():r="",s==null&&(s=d),l.relative[r](n,s,w)}else n=x=[]}n||(n=j),n||k.error(r||b);if(e.call(n)==="[object Array]")if(!u)f.push.apply(f,n);else if(d&&d.nodeType===1)for(t=0;n[t]!=null;t++)n[t]&&(n[t]===!0||n[t].nodeType===1&&k.contains(d,n[t]))&&f.push(j[t]);else for(t=0;n[t]!=null;t++)n[t]&&n[t].nodeType===1&&f.push(j[t]);else p(n,f);o&&(k(o,h,f,g),k.uniqueSort(f));return f};k.uniqueSort=function(a){if(r){g=h,a.sort(r);if(g)for(var b=1;b<a.length;b++)a[b]===a[b-1]&&a.splice(b--,1)}return a},k.matches=function(a,b){return k(a,null,null,b)},k.matchesSelector=function(a,b){return k(b,null,null,[a]).length>0},k.find=function(a,b,c){var d;if(!a)return[];for(var e=0,f=l.order.length;e<f;e++){var g,h=l.order[e];if(g=l.leftMatch[h].exec(a)){var j=g[1];g.splice(1,1);if(j.substr(j.length-1)!=="\\"){g[1]=(g[1]||"").replace(i,""),d=l.find[h](g,b,c);if(d!=null){a=a.replace(l.match[h],"");break}}}}d||(d=typeof b.getElementsByTagName!="undefined"?b.getElementsByTagName("*"):[]);return{set:d,expr:a}},k.filter=function(a,c,d,e){var f,g,h=a,i=[],j=c,m=c&&c[0]&&k.isXML(c[0]);while(a&&c.length){for(var n in l.filter)if((f=l.leftMatch[n].exec(a))!=null&&f[2]){var o,p,q=l.filter[n],r=f[1];g=!1,f.splice(1,1);if(r.substr(r.length-1)==="\\")continue;j===i&&(i=[]);if(l.preFilter[n]){f=l.preFilter[n](f,j,d,i,e,m);if(!f)g=o=!0;else if(f===!0)continue}if(f)for(var s=0;(p=j[s])!=null;s++)if(p){o=q(p,f,s,j);var t=e^!!o;d&&o!=null?t?g=!0:j[s]=!1:t&&(i.push(p),g=!0)}if(o!==b){d||(j=i),a=a.replace(l.match[n],"");if(!g)return[];break}}if(a===h)if(g==null)k.error(a);else break;h=a}return j},k.error=function(a){throw"Syntax error, unrecognized expression: "+a};var l=k.selectors={order:["ID","NAME","TAG"],match:{ID:/#((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,CLASS:/\.((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,NAME:/\[name=['"]*((?:[\w\u00c0-\uFFFF\-]|\\.)+)['"]*\]/,ATTR:/\[\s*((?:[\w\u00c0-\uFFFF\-]|\\.)+)\s*(?:(\S?=)\s*(?:(['"])(.*?)\3|(#?(?:[\w\u00c0-\uFFFF\-]|\\.)*)|)|)\s*\]/,TAG:/^((?:[\w\u00c0-\uFFFF\*\-]|\\.)+)/,CHILD:/:(only|nth|last|first)-child(?:\(\s*(even|odd|(?:[+\-]?\d+|(?:[+\-]?\d*)?n\s*(?:[+\-]\s*\d+)?))\s*\))?/,POS:/:(nth|eq|gt|lt|first|last|even|odd)(?:\((\d*)\))?(?=[^\-]|$)/,PSEUDO:/:((?:[\w\u00c0-\uFFFF\-]|\\.)+)(?:\((['"]?)((?:\([^\)]+\)|[^\(\)]*)+)\2\))?/},leftMatch:{},attrMap:{"class":"className","for":"htmlFor"},attrHandle:{href:function(a){return a.getAttribute("href")},type:function(a){return a.getAttribute("type")}},relative:{"+":function(a,b){var c=typeof b=="string",d=c&&!j.test(b),e=c&&!d;d&&(b=b.toLowerCase());for(var f=0,g=a.length,h;f<g;f++)if(h=a[f]){while((h=h.previousSibling)&&h.nodeType!==1);a[f]=e||h&&h.nodeName.toLowerCase()===b?h||!1:h===b}e&&k.filter(b,a,!0)},">":function(a,b){var c,d=typeof b=="string",e=0,f=a.length;if(d&&!j.test(b)){b=b.toLowerCase();for(;e<f;e++){c=a[e];if(c){var g=c.parentNode;a[e]=g.nodeName.toLowerCase()===b?g:!1}}}else{for(;e<f;e++)c=a[e],c&&(a[e]=d?c.parentNode:c.parentNode===b);d&&k.filter(b,a,!0)}},"":function(a,b,c){var e,f=d++,g=u;typeof b=="string"&&!j.test(b)&&(b=b.toLowerCase(),e=b,g=t),g("parentNode",b,f,a,e,c)},"~":function(a,b,c){var e,f=d++,g=u;typeof b=="string"&&!j.test(b)&&(b=b.toLowerCase(),e=b,g=t),g("previousSibling",b,f,a,e,c)}},find:{ID:function(a,b,c){if(typeof b.getElementById!="undefined"&&!c){var d=b.getElementById(a[1]);return d&&d.parentNode?[d]:[]}},NAME:function(a,b){if(typeof b.getElementsByName!="undefined"){var c=[],d=b.getElementsByName(a[1]);for(var e=0,f=d.length;e<f;e++)d[e].getAttribute("name")===a[1]&&c.push(d[e]);return c.length===0?null:c}},TAG:function(a,b){if(typeof b.getElementsByTagName!="undefined")return b.getElementsByTagName(a[1])}},preFilter:{CLASS:function(a,b,c,d,e,f){a=" "+a[1].replace(i,"")+" ";if(f)return a;for(var g=0,h;(h=b[g])!=null;g++)h&&(e^(h.className&&(" "+h.className+" ").replace(/[\t\n\r]/g," ").indexOf(a)>=0)?c||d.push(h):c&&(b[g]=!1));return!1},ID:function(a){return a[1].replace(i,"")},TAG:function(a,b){return a[1].replace(i,"").toLowerCase()},CHILD:function(a){if(a[1]==="nth"){a[2]||k.error(a[0]),a[2]=a[2].replace(/^\+|\s*/g,"");var b=/(-?)(\d*)(?:n([+\-]?\d*))?/.exec(a[2]==="even"&&"2n"||a[2]==="odd"&&"2n+1"||!/\D/.test(a[2])&&"0n+"+a[2]||a[2]);a[2]=b[1]+(b[2]||1)-0,a[3]=b[3]-0}else a[2]&&k.error(a[0]);a[0]=d++;return a},ATTR:function(a,b,c,d,e,f){var g=a[1]=a[1].replace(i,"");!f&&l.attrMap[g]&&(a[1]=l.attrMap[g]),a[4]=(a[4]||a[5]||"").replace(i,""),a[2]==="~="&&(a[4]=" "+a[4]+" ");return a},PSEUDO:function(b,c,d,e,f){if(b[1]==="not")if((a.exec(b[3])||"").length>1||/^\w/.test(b[3]))b[3]=k(b[3],null,null,c);else{var g=k.filter(b[3],c,d,!0^f);d||e.push.apply(e,g);return!1}else if(l.match.POS.test(b[0])||l.match.CHILD.test(b[0]))return!0;return b},POS:function(a){a.unshift(!0);return a}},filters:{enabled:function(a){return a.disabled===!1&&a.type!=="hidden"},disabled:function(a){return a.disabled===!0},checked:function(a){return a.checked===!0},selected:function(a){a.parentNode&&a.parentNode.selectedIndex;return a.selected===!0},parent:function(a){return!!a.firstChild},empty:function(a){return!a.firstChild},has:function(a,b,c){return!!k(c[3],a).length},header:function(a){return/h\d/i.test(a.nodeName)},text:function(a){var b=a.getAttribute("type"),c=a.type;return a.nodeName.toLowerCase()==="input"&&"text"===c&&(b===c||b===null)},radio:function(a){return a.nodeName.toLowerCase()==="input"&&"radio"===a.type},checkbox:function(a){return a.nodeName.toLowerCase()==="input"&&"checkbox"===a.type},file:function(a){return a.nodeName.toLowerCase()==="input"&&"file"===a.type},password:function(a){return a.nodeName.toLowerCase()==="input"&&"password"===a.type},submit:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"submit"===a.type},image:function(a){return a.nodeName.toLowerCase()==="input"&&"image"===a.type},reset:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"reset"===a.type},button:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&"button"===a.type||b==="button"},input:function(a){return/input|select|textarea|button/i.test(a.nodeName)},focus:function(a){return a===a.ownerDocument.activeElement}},setFilters:{first:function(a,b){return b===0},last:function(a,b,c,d){return b===d.length-1},even:function(a,b){return b%2===0},odd:function(a,b){return b%2===1},lt:function(a,b,c){return b<c[3]-0},gt:function(a,b,c){return b>c[3]-0},nth:function(a,b,c){return c[3]-0===b},eq:function(a,b,c){return c[3]-0===b}},filter:{PSEUDO:function(a,b,c,d){var e=b[1],f=l.filters[e];if(f)return f(a,c,b,d);if(e==="contains")return(a.textContent||a.innerText||k.getText([a])||"").indexOf(b[3])>=0;if(e==="not"){var g=b[3];for(var h=0,i=g.length;h<i;h++)if(g[h]===a)return!1;return!0}k.error(e)},CHILD:function(a,b){var c=b[1],d=a;switch(c){case"only":case"first":while(d=d.previousSibling)if(d.nodeType===1)return!1;if(c==="first")return!0;d=a;case"last":while(d=d.nextSibling)if(d.nodeType===1)return!1;return!0;case"nth":var e=b[2],f=b[3];if(e===1&&f===0)return!0;var g=b[0],h=a.parentNode;if(h&&(h.sizcache!==g||!a.nodeIndex)){var i=0;for(d=h.firstChild;d;d=d.nextSibling)d.nodeType===1&&(d.nodeIndex=++i);h.sizcache=g}var j=a.nodeIndex-f;return e===0?j===0:j%e===0&&j/e>=0}},ID:function(a,b){return a.nodeType===1&&a.getAttribute("id")===b},TAG:function(a,b){return b==="*"&&a.nodeType===1||a.nodeName.toLowerCase()===b},CLASS:function(a,b){return(" "+(a.className||a.getAttribute("class"))+" ").indexOf(b)>-1},ATTR:function(a,b){var c=b[1],d=l.attrHandle[c]?l.attrHandle[c](a):a[c]!=null?a[c]:a.getAttribute(c),e=d+"",f=b[2],g=b[4];return d==null?f==="!=":f==="="?e===g:f==="*="?e.indexOf(g)>=0:f==="~="?(" "+e+" ").indexOf(g)>=0:g?f==="!="?e!==g:f==="^="?e.indexOf(g)===0:f==="$="?e.substr(e.length-g.length)===g:f==="|="?e===g||e.substr(0,g.length+1)===g+"-":!1:e&&d!==!1},POS:function(a,b,c,d){var e=b[2],f=l.setFilters[e];if(f)return f(a,c,b,d)}}},m=l.match.POS,n=function(a,b){return"\\"+(b-0+1)};for(var o in l.match)l.match[o]=new RegExp(l.match[o].source+/(?![^\[]*\])(?![^\(]*\))/.source),l.leftMatch[o]=new RegExp(/(^(?:.|\r|\n)*?)/.source+l.match[o].source.replace(/\\(\d+)/g,n));var p=function(a,b){a=Array.prototype.slice.call(a,0);if(b){b.push.apply(b,a);return b}return a};try{Array.prototype.slice.call(c.documentElement.childNodes,0)[0].nodeType}catch(q){p=function(a,b){var c=0,d=b||[];if(e.call(a)==="[object Array]")Array.prototype.push.apply(d,a);else if(typeof a.length=="number")for(var f=a.length;c<f;c++)d.push(a[c]);else for(;a[c];c++)d.push(a[c]);return d}}var r,s;c.documentElement.compareDocumentPosition?r=function(a,b){if(a===b){g=!0;return 0}if(!a.compareDocumentPosition||!b.compareDocumentPosition)return a.compareDocumentPosition?-1:1;return a.compareDocumentPosition(b)&4?-1:1}:(r=function(a,b){if(a===b){g=!0;return 0}if(a.sourceIndex&&b.sourceIndex)return a.sourceIndex-b.sourceIndex;var c,d,e=[],f=[],h=a.parentNode,i=b.parentNode,j=h;if(h===i)return s(a,b);if(!h)return-1;if(!i)return 1;while(j)e.unshift(j),j=j.parentNode;j=i;while(j)f.unshift(j),j=j.parentNode;c=e.length,d=f.length;for(var k=0;k<c&&k<d;k++)if(e[k]!==f[k])return s(e[k],f[k]);return k===c?s(a,f[k],-1):s(e[k],b,1)},s=function(a,b,c){if(a===b)return c;var d=a.nextSibling;while(d){if(d===b)return-1;d=d.nextSibling}return 1}),k.getText=function(a){var b="",c;for(var d=0;a[d];d++)c=a[d],c.nodeType===3||c.nodeType===4?b+=c.nodeValue:c.nodeType!==8&&(b+=k.getText(c.childNodes));return b},function(){var a=c.createElement("div"),d="script"+(new Date).getTime(),e=c.documentElement;a.innerHTML="<a name='"+d+"'/>",e.insertBefore(a,e.firstChild),c.getElementById(d)&&(l.find.ID=function(a,c,d){if(typeof c.getElementById!="undefined"&&!d){var e=c.getElementById(a[1]);return e?e.id===a[1]||typeof e.getAttributeNode!="undefined"&&e.getAttributeNode("id").nodeValue===a[1]?[e]:b:[]}},l.filter.ID=function(a,b){var c=typeof a.getAttributeNode!="undefined"&&a.getAttributeNode("id");return a.nodeType===1&&c&&c.nodeValue===b}),e.removeChild(a),e=a=null}(),function(){var a=c.createElement("div");a.appendChild(c.createComment("")),a.getElementsByTagName("*").length>0&&(l.find.TAG=function(a,b){var c=b.getElementsByTagName(a[1]);if(a[1]==="*"){var d=[];for(var e=0;c[e];e++)c[e].nodeType===1&&d.push(c[e]);c=d}return c}),a.innerHTML="<a href='#'></a>",a.firstChild&&typeof a.firstChild.getAttribute!="undefined"&&a.firstChild.getAttribute("href")!=="#"&&(l.attrHandle.href=function(a){return a.getAttribute("href",2)}),a=null}(),c.querySelectorAll&&function(){var a=k,b=c.createElement("div"),d="__sizzle__";b.innerHTML="<p class='TEST'></p>";if(!b.querySelectorAll||b.querySelectorAll(".TEST").length!==0){k=function(b,e,f,g){e=e||c;if(!g&&!k.isXML(e)){var h=/^(\w+$)|^\.([\w\-]+$)|^#([\w\-]+$)/.exec(b);if(h&&(e.nodeType===1||e.nodeType===9)){if(h[1])return p(e.getElementsByTagName(b),f);if(h[2]&&l.find.CLASS&&e.getElementsByClassName)return p(e.getElementsByClassName(h[2]),f)}if(e.nodeType===9){if(b==="body"&&e.body)return p([e.body],f);if(h&&h[3]){var i=e.getElementById(h[3]);if(!i||!i.parentNode)return p([],f);if(i.id===h[3])return p([i],f)}try{return p(e.querySelectorAll(b),f)}catch(j){}}else if(e.nodeType===1&&e.nodeName.toLowerCase()!=="object"){var m=e,n=e.getAttribute("id"),o=n||d,q=e.parentNode,r=/^\s*[+~]/.test(b);n?o=o.replace(/'/g,"\\$&"):e.setAttribute("id",o),r&&q&&(e=e.parentNode);try{if(!r||q)return p(e.querySelectorAll("[id='"+o+"'] "+b),f)}catch(s){}finally{n||m.removeAttribute("id")}}}return a(b,e,f,g)};for(var e in a)k[e]=a[e];b=null}}(),function(){var a=c.documentElement,b=a.matchesSelector||a.mozMatchesSelector||a.webkitMatchesSelector||a.msMatchesSelector;if(b){var d=!b.call(c.createElement("div"),"div"),e=!1;try{b.call(c.documentElement,"[test!='']:sizzle")}catch(f){e=!0}k.matchesSelector=function(a,c){c=c.replace(/\=\s*([^'"\]]*)\s*\]/g,"='$1']");if(!k.isXML(a))try{if(e||!l.match.PSEUDO.test(c)&&!/!=/.test(c)){var f=b.call(a,c);if(f||!d||a.document&&a.document.nodeType!==11)return f}}catch(g){}return k(c,null,null,[a]).length>0}}}(),function(){var a=c.createElement("div");a.innerHTML="<div class='test e'></div><div class='test'></div>";if(!!a.getElementsByClassName&&a.getElementsByClassName("e").length!==0){a.lastChild.className="e";if(a.getElementsByClassName("e").length===1)return;l.order.splice(1,0,"CLASS"),l.find.CLASS=function(a,b,c){if(typeof b.getElementsByClassName!="undefined"&&!c)return b.getElementsByClassName(a[1])},a=null}}(),c.documentElement.contains?k.contains=function(a,b){return a!==b&&(a.contains?a.contains(b):!0)}:c.documentElement.compareDocumentPosition?k.contains=function(a,b){return!!(a.compareDocumentPosition(b)&16)}:k.contains=function(){return!1},k.isXML=function(a){var b=(a?a.ownerDocument||a:0).documentElement;return b?b.nodeName!=="HTML":!1};var v=function(a,b){var c,d=[],e="",f=b.nodeType?[b]:b;while(c=l.match.PSEUDO.exec(a))e+=c[0],a=a.replace(l.match.PSEUDO,"");a=l.relative[a]?a+"*":a;for(var g=0,h=f.length;g<h;g++)k(a,f[g],d);return k.filter(e,d)};f.find=k,f.expr=k.selectors,f.expr[":"]=f.expr.filters,f.unique=k.uniqueSort,f.text=k.getText,f.isXMLDoc=k.isXML,f.contains=k.contains}();var O=/Until$/,P=/^(?:parents|prevUntil|prevAll)/,Q=/,/,R=/^.[^:#\[\.,]*$/,S=Array.prototype.slice,T=f.expr.match.POS,U={children:!0,contents:!0,next:!0,prev:!0};f.fn.extend({find:function(a){var b=this,c,d;if(typeof a!="string")return f(a).filter(function(){for(c=0,d=b.length;c<d;c++)if(f.contains(b[c],this))return!0});var e=this.pushStack("","find",a),g,h,i;for(c=0,d=this.length;c<d;c++){g=e.length,f.find(a,this[c],e);if(c>0)for(h=g;h<e.length;h++)for(i=0;i<g;i++)if(e[i]===e[h]){e.splice(h--,1);break}}return e},has:function(a){var b=f(a);return this.filter(function(){for(var a=0,c=b.length;a<c;a++)if(f.contains(this,b[a]))return!0})},not:function(a){return this.pushStack(W(this,a,!1),"not",a)},filter:function(a){return this.pushStack(W(this,a,!0),"filter",a)},is:function(a){return!!a&&(typeof a=="string"?f.filter(a,this).length>0:this.filter(a).length>0)},closest:function(a,b){var c=[],d,e,g=this[0];if(f.isArray(a)){var h,i,j={},k=1;if(g&&a.length){for(d=0,e=a.length;d<e;d++)i=a[d],j[i]||(j[i]=T.test(i)?f(i,b||this.context):i);while(g&&g.ownerDocument&&g!==b){for(i in j)h=j[i],(h.jquery?h.index(g)>-1:f(g).is(h))&&c.push({selector:i,elem:g,level:k});g=g.parentNode,k++}}return c}var l=T.test(a)||typeof a!="string"?f(a,b||this.context):0;for(d=0,e=this.length;d<e;d++){g=this[d];while(g){if(l?l.index(g)>-1:f.find.matchesSelector(g,a)){c.push(g);break}g=g.parentNode;if(!g||!g.ownerDocument||g===b||g.nodeType===11)break}}c=c.length>1?f.unique(c):c;return this.pushStack(c,"closest",a)},index:function(a){if(!a||typeof a=="string")return f.inArray(this[0],a?f(a):this.parent().children());return f.inArray(a.jquery?a[0]:a,this)},add:function(a,b){var c=typeof a=="string"?f(a,b):f.makeArray(a&&a.nodeType?[a]:a),d=f.merge(this.get(),c);return this.pushStack(V(c[0])||V(d[0])?d:f.unique(d))},andSelf:function(){return this.add(this.prevObject)}}),f.each({parent:function(a){var b=a.parentNode;return b&&b.nodeType!==11?b:null},parents:function(a){return f.dir(a,"parentNode")},parentsUntil:function(a,b,c){return f.dir(a,"parentNode",c)},next:function(a){return f.nth(a,2,"nextSibling")},prev:function(a){return f.nth(a,2,"previousSibling")},nextAll:function(a){return f.dir(a,"nextSibling")},prevAll:function(a){return f.dir(a,"previousSibling")},nextUntil:function(a,b,c){return f.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return f.dir(a,"previousSibling",c)},siblings:function(a){return f.sibling(a.parentNode.firstChild,a)},children:function(a){return f.sibling(a.firstChild)},contents:function(a){return f.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:f.makeArray(a.childNodes)}},function(a,b){f.fn[a]=function(c,d){var e=f.map(this,b,c),g=S.call(arguments);O.test(a)||(d=c),d&&typeof d=="string"&&(e=f.filter(d,e)),e=this.length>1&&!U[a]?f.unique(e):e,(this.length>1||Q.test(d))&&P.test(a)&&(e=e.reverse());return this.pushStack(e,a,g.join(","))}}),f.extend({filter:function(a,b,c){c&&(a=":not("+a+")");return b.length===1?f.find.matchesSelector(b[0],a)?[b[0]]:[]:f.find.matches(a,b)},dir:function(a,c,d){var e=[],g=a[c];while(g&&g.nodeType!==9&&(d===b||g.nodeType!==1||!f(g).is(d)))g.nodeType===1&&e.push(g),g=g[c];return e},nth:function(a,b,c,d){b=b||1;var e=0;for(;a;a=a[c])if(a.nodeType===1&&++e===b)break;return a},sibling:function(a,b){var c=[];for(;a;a=a.nextSibling)a.nodeType===1&&a!==b&&c.push(a);return c}});var X=/ jQuery\d+="(?:\d+|null)"/g,Y=/^\s+/,Z=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,$=/<([\w:]+)/,_=/<tbody/i,ba=/<|&#?\w+;/,bb=/<(?:script|object|embed|option|style)/i,bc=/checked\s*(?:[^=]|=\s*.checked.)/i,bd=/\/(java|ecma)script/i,be=/^\s*<!(?:\[CDATA\[|\-\-)/,bf={option:[1,"<select multiple='multiple'>","</select>"],legend:[1,"<fieldset>","</fieldset>"],thead:[1,"<table>","</table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],col:[2,"<table><tbody></tbody><colgroup>","</colgroup></table>"],area:[1,"<map>","</map>"],_default:[0,"",""]};bf.optgroup=bf.option,bf.tbody=bf.tfoot=bf.colgroup=bf.caption=bf.thead,bf.th=bf.td,f.support.htmlSerialize||(bf._default=[1,"div<div>","</div>"]),f.fn.extend({text:function(a){if(f.isFunction(a))return this.each(function(b){var c=f(this);c.text(a.call(this,b,c.text()))});if(typeof a!="object"&&a!==b)return this.empty().append((this[0]&&this[0].ownerDocument||c).createTextNode(a));return f.text(this)},wrapAll:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapAll(a.call(this,b))});if(this[0]){var b=f(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&a.firstChild.nodeType===1)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapInner(a.call(this,b))});return this.each(function(){var b=f(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){return this.each(function(){f(this).wrapAll(a)})},unwrap:function(){return this.parent().each(function(){f.nodeName(this,"body")||f(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.appendChild(a)})},prepend:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this)});if(arguments.length){var a=f(arguments[0]);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,f(arguments[0]).toArray());return a}},remove:function(a,b){for(var c=0,d;(d=this[c])!=null;c++)if(!a||f.filter(a,[d]).length)!b&&d.nodeType===1&&(f.cleanData(d.getElementsByTagName("*")),f.cleanData([d])),d.parentNode&&d.parentNode.removeChild(d);return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++){b.nodeType===1&&f.cleanData(b.getElementsByTagName("*"));while(b.firstChild)b.removeChild(b.firstChild)}return this},clone:function(a,b){a=a==null?!1:a,b=b==null?a:b;return this.map(function(){return f.clone(this,a,b)})},html:function(a){if(a===b)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(X,""):null;if(typeof a=="string"&&!bb.test(a)&&(f.support.leadingWhitespace||!Y.test(a))&&!bf[($.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Z,"<$1></$2>");try{for(var c=0,d=this.length;c<d;c++)this[c].nodeType===1&&(f.cleanData(this[c].getElementsByTagName("*")),this[c].innerHTML=a)}catch(e){this.empty().append(a)}}else f.isFunction(a)?this.each(function(b){var c=f(this);c.html(a.call(this,b,c.html()))}):this.empty().append(a);return this},replaceWith:function(a){if(this[0]&&this[0].parentNode){if(f.isFunction(a))return this.each(function(b){var c=f(this),d=c.html();c.replaceWith(a.call(this,b,d))});typeof a!="string"&&(a=f(a).detach());return this.each(function(){var b=this.nextSibling,c=this.parentNode;f(this).remove(),b?f(b).before(a):f(c).append(a)})}return this.length?this.pushStack(f(f.isFunction(a)?a():a),"replaceWith",a):this},detach:function(a){return this.remove(a,!0)},domManip:function(a,c,d){var e,g,h,i,j=a[0],k=[];if(!f.support.checkClone&&arguments.length===3&&typeof j=="string"&&bc.test(j))return this.each(function(){f(this).domManip(a,c,d,!0)});if(f.isFunction(j))return this.each(function(e){var g=f(this);a[0]=j.call(this,e,c?g.html():b),g.domManip(a,c,d)});if(this[0]){i=j&&j.parentNode,f.support.parentNode&&i&&i.nodeType===11&&i.childNodes.length===this.length?e={fragment:i}:e=f.buildFragment(a,this,k),h=e.fragment,h.childNodes.length===1?g=h=h.firstChild:g=h.firstChild;if(g){c=c&&f.nodeName(g,"tr");for(var l=0,m=this.length,n=m-1;l<m;l++)d.call(c?bg(this[l],g):this[l],e.cacheable||m>1&&l<n?f.clone(h,!0,!0):h)}k.length&&f.each(k,bm)}return this}}),f.buildFragment=function(a,b,d){var e,g,h,i;b&&b[0]&&(i=b[0].ownerDocument||b[0]),i.createDocumentFragment||(i=c),a.length===1&&typeof a[0]=="string"&&a[0].length<512&&i===c&&a[0].charAt(0)==="<"&&!bb.test(a[0])&&(f.support.checkClone||!bc.test(a[0]))&&(g=!0,h=f.fragments[a[0]],h&&h!==1&&(e=h)),e||(e=i.createDocumentFragment(),f.clean(a,i,e,d)),g&&(f.fragments[a[0]]=h?e:1);return{fragment:e,cacheable:g}},f.fragments={},f.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){f.fn[a]=function(c){var d=[],e=f(c),g=this.length===1&&this[0].parentNode;if(g&&g.nodeType===11&&g.childNodes.length===1&&e.length===1){e[b](this[0]);return this}for(var h=0,i=e.length;h<i;h++){var j=(h>0?this.clone(!0):this).get();f(e[h])[b](j),d=d.concat(j
)}return this.pushStack(d,a,e.selector)}}),f.extend({clone:function(a,b,c){var d=a.cloneNode(!0),e,g,h;if((!f.support.noCloneEvent||!f.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!f.isXMLDoc(a)){bi(a,d),e=bj(a),g=bj(d);for(h=0;e[h];++h)bi(e[h],g[h])}if(b){bh(a,d);if(c){e=bj(a),g=bj(d);for(h=0;e[h];++h)bh(e[h],g[h])}}e=g=null;return d},clean:function(a,b,d,e){var g;b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||b[0]&&b[0].ownerDocument||c);var h=[],i;for(var j=0,k;(k=a[j])!=null;j++){typeof k=="number"&&(k+="");if(!k)continue;if(typeof k=="string")if(!ba.test(k))k=b.createTextNode(k);else{k=k.replace(Z,"<$1></$2>");var l=($.exec(k)||["",""])[1].toLowerCase(),m=bf[l]||bf._default,n=m[0],o=b.createElement("div");o.innerHTML=m[1]+k+m[2];while(n--)o=o.lastChild;if(!f.support.tbody){var p=_.test(k),q=l==="table"&&!p?o.firstChild&&o.firstChild.childNodes:m[1]==="<table>"&&!p?o.childNodes:[];for(i=q.length-1;i>=0;--i)f.nodeName(q[i],"tbody")&&!q[i].childNodes.length&&q[i].parentNode.removeChild(q[i])}!f.support.leadingWhitespace&&Y.test(k)&&o.insertBefore(b.createTextNode(Y.exec(k)[0]),o.firstChild),k=o.childNodes}var r;if(!f.support.appendChecked)if(k[0]&&typeof (r=k.length)=="number")for(i=0;i<r;i++)bl(k[i]);else bl(k);k.nodeType?h.push(k):h=f.merge(h,k)}if(d){g=function(a){return!a.type||bd.test(a.type)};for(j=0;h[j];j++)if(e&&f.nodeName(h[j],"script")&&(!h[j].type||h[j].type.toLowerCase()==="text/javascript"))e.push(h[j].parentNode?h[j].parentNode.removeChild(h[j]):h[j]);else{if(h[j].nodeType===1){var s=f.grep(h[j].getElementsByTagName("script"),g);h.splice.apply(h,[j+1,0].concat(s))}d.appendChild(h[j])}}return h},cleanData:function(a){var b,c,d=f.cache,e=f.expando,g=f.event.special,h=f.support.deleteExpando;for(var i=0,j;(j=a[i])!=null;i++){if(j.nodeName&&f.noData[j.nodeName.toLowerCase()])continue;c=j[f.expando];if(c){b=d[c]&&d[c][e];if(b&&b.events){for(var k in b.events)g[k]?f.event.remove(j,k):f.removeEvent(j,k,b.handle);b.handle&&(b.handle.elem=null)}h?delete j[f.expando]:j.removeAttribute&&j.removeAttribute(f.expando),delete d[c]}}}});var bn=/alpha\([^)]*\)/i,bo=/opacity=([^)]*)/,bp=/([A-Z]|^ms)/g,bq=/^-?\d+(?:px)?$/i,br=/^-?\d/,bs=/^[+\-]=/,bt=/[^+\-\.\de]+/g,bu={position:"absolute",visibility:"hidden",display:"block"},bv=["Left","Right"],bw=["Top","Bottom"],bx,by,bz;f.fn.css=function(a,c){if(arguments.length===2&&c===b)return this;return f.access(this,a,c,!0,function(a,c,d){return d!==b?f.style(a,c,d):f.css(a,c)})},f.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=bx(a,"opacity","opacity");return c===""?"1":c}return a.style.opacity}}},cssNumber:{fillOpacity:!0,fontWeight:!0,lineHeight:!0,opacity:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":f.support.cssFloat?"cssFloat":"styleFloat"},style:function(a,c,d,e){if(!!a&&a.nodeType!==3&&a.nodeType!==8&&!!a.style){var g,h,i=f.camelCase(c),j=a.style,k=f.cssHooks[i];c=f.cssProps[i]||i;if(d===b){if(k&&"get"in k&&(g=k.get(a,!1,e))!==b)return g;return j[c]}h=typeof d;if(h==="number"&&isNaN(d)||d==null)return;h==="string"&&bs.test(d)&&(d=+d.replace(bt,"")+parseFloat(f.css(a,c)),h="number"),h==="number"&&!f.cssNumber[i]&&(d+="px");if(!k||!("set"in k)||(d=k.set(a,d))!==b)try{j[c]=d}catch(l){}}},css:function(a,c,d){var e,g;c=f.camelCase(c),g=f.cssHooks[c],c=f.cssProps[c]||c,c==="cssFloat"&&(c="float");if(g&&"get"in g&&(e=g.get(a,!0,d))!==b)return e;if(bx)return bx(a,c)},swap:function(a,b,c){var d={};for(var e in b)d[e]=a.style[e],a.style[e]=b[e];c.call(a);for(e in b)a.style[e]=d[e]}}),f.curCSS=f.css,f.each(["height","width"],function(a,b){f.cssHooks[b]={get:function(a,c,d){var e;if(c){if(a.offsetWidth!==0)return bA(a,b,d);f.swap(a,bu,function(){e=bA(a,b,d)});return e}},set:function(a,b){if(!bq.test(b))return b;b=parseFloat(b);if(b>=0)return b+"px"}}}),f.support.opacity||(f.cssHooks.opacity={get:function(a,b){return bo.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?parseFloat(RegExp.$1)/100+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle;c.zoom=1;var e=f.isNaN(b)?"":"alpha(opacity="+b*100+")",g=d&&d.filter||c.filter||"";c.filter=bn.test(g)?g.replace(bn,e):g+" "+e}}),f(function(){f.support.reliableMarginRight||(f.cssHooks.marginRight={get:function(a,b){var c;f.swap(a,{display:"inline-block"},function(){b?c=bx(a,"margin-right","marginRight"):c=a.style.marginRight});return c}})}),c.defaultView&&c.defaultView.getComputedStyle&&(by=function(a,c){var d,e,g;c=c.replace(bp,"-$1").toLowerCase();if(!(e=a.ownerDocument.defaultView))return b;if(g=e.getComputedStyle(a,null))d=g.getPropertyValue(c),d===""&&!f.contains(a.ownerDocument.documentElement,a)&&(d=f.style(a,c));return d}),c.documentElement.currentStyle&&(bz=function(a,b){var c,d=a.currentStyle&&a.currentStyle[b],e=a.runtimeStyle&&a.runtimeStyle[b],f=a.style;!bq.test(d)&&br.test(d)&&(c=f.left,e&&(a.runtimeStyle.left=a.currentStyle.left),f.left=b==="fontSize"?"1em":d||0,d=f.pixelLeft+"px",f.left=c,e&&(a.runtimeStyle.left=e));return d===""?"auto":d}),bx=by||bz,f.expr&&f.expr.filters&&(f.expr.filters.hidden=function(a){var b=a.offsetWidth,c=a.offsetHeight;return b===0&&c===0||!f.support.reliableHiddenOffsets&&(a.style.display||f.css(a,"display"))==="none"},f.expr.filters.visible=function(a){return!f.expr.filters.hidden(a)});var bB=/%20/g,bC=/\[\]$/,bD=/\r?\n/g,bE=/#.*$/,bF=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,bG=/^(?:color|date|datetime|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,bH=/^(?:about|app|app\-storage|.+\-extension|file|widget):$/,bI=/^(?:GET|HEAD)$/,bJ=/^\/\//,bK=/\?/,bL=/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi,bM=/^(?:select|textarea)/i,bN=/\s+/,bO=/([?&])_=[^&]*/,bP=/^([\w\+\.\-]+:)(?:\/\/([^\/?#:]*)(?::(\d+))?)?/,bQ=f.fn.load,bR={},bS={},bT,bU;try{bT=e.href}catch(bV){bT=c.createElement("a"),bT.href="",bT=bT.href}bU=bP.exec(bT.toLowerCase())||[],f.fn.extend({load:function(a,c,d){if(typeof a!="string"&&bQ)return bQ.apply(this,arguments);if(!this.length)return this;var e=a.indexOf(" ");if(e>=0){var g=a.slice(e,a.length);a=a.slice(0,e)}var h="GET";c&&(f.isFunction(c)?(d=c,c=b):typeof c=="object"&&(c=f.param(c,f.ajaxSettings.traditional),h="POST"));var i=this;f.ajax({url:a,type:h,dataType:"html",data:c,complete:function(a,b,c){c=a.responseText,a.isResolved()&&(a.done(function(a){c=a}),i.html(g?f("<div>").append(c.replace(bL,"")).find(g):c)),d&&i.each(d,[c,b,a])}});return this},serialize:function(){return f.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?f.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||bM.test(this.nodeName)||bG.test(this.type))}).map(function(a,b){var c=f(this).val();return c==null?null:f.isArray(c)?f.map(c,function(a,c){return{name:b.name,value:a.replace(bD,"\r\n")}}):{name:b.name,value:c.replace(bD,"\r\n")}}).get()}}),f.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "),function(a,b){f.fn[b]=function(a){return this.bind(b,a)}}),f.each(["get","post"],function(a,c){f[c]=function(a,d,e,g){f.isFunction(d)&&(g=g||e,e=d,d=b);return f.ajax({type:c,url:a,data:d,success:e,dataType:g})}}),f.extend({getScript:function(a,c){return f.get(a,b,c,"script")},getJSON:function(a,b,c){return f.get(a,b,c,"json")},ajaxSetup:function(a,b){b?f.extend(!0,a,f.ajaxSettings,b):(b=a,a=f.extend(!0,f.ajaxSettings,b));for(var c in{context:1,url:1})c in b?a[c]=b[c]:c in f.ajaxSettings&&(a[c]=f.ajaxSettings[c]);return a},ajaxSettings:{url:bT,isLocal:bH.test(bU[1]),global:!0,type:"GET",contentType:"application/x-www-form-urlencoded",processData:!0,async:!0,accepts:{xml:"application/xml, text/xml",html:"text/html",text:"text/plain",json:"application/json, text/javascript","*":"*/*"},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText"},converters:{"* text":a.String,"text html":!0,"text json":f.parseJSON,"text xml":f.parseXML}},ajaxPrefilter:bW(bR),ajaxTransport:bW(bS),ajax:function(a,c){function w(a,c,l,m){if(s!==2){s=2,q&&clearTimeout(q),p=b,n=m||"",v.readyState=a?4:0;var o,r,u,w=l?bZ(d,v,l):b,x,y;if(a>=200&&a<300||a===304){if(d.ifModified){if(x=v.getResponseHeader("Last-Modified"))f.lastModified[k]=x;if(y=v.getResponseHeader("Etag"))f.etag[k]=y}if(a===304)c="notmodified",o=!0;else try{r=b$(d,w),c="success",o=!0}catch(z){c="parsererror",u=z}}else{u=c;if(!c||a)c="error",a<0&&(a=0)}v.status=a,v.statusText=c,o?h.resolveWith(e,[r,c,v]):h.rejectWith(e,[v,c,u]),v.statusCode(j),j=b,t&&g.trigger("ajax"+(o?"Success":"Error"),[v,d,o?r:u]),i.resolveWith(e,[v,c]),t&&(g.trigger("ajaxComplete",[v,d]),--f.active||f.event.trigger("ajaxStop"))}}typeof a=="object"&&(c=a,a=b),c=c||{};var d=f.ajaxSetup({},c),e=d.context||d,g=e!==d&&(e.nodeType||e instanceof f)?f(e):f.event,h=f.Deferred(),i=f._Deferred(),j=d.statusCode||{},k,l={},m={},n,o,p,q,r,s=0,t,u,v={readyState:0,setRequestHeader:function(a,b){if(!s){var c=a.toLowerCase();a=m[c]=m[c]||a,l[a]=b}return this},getAllResponseHeaders:function(){return s===2?n:null},getResponseHeader:function(a){var c;if(s===2){if(!o){o={};while(c=bF.exec(n))o[c[1].toLowerCase()]=c[2]}c=o[a.toLowerCase()]}return c===b?null:c},overrideMimeType:function(a){s||(d.mimeType=a);return this},abort:function(a){a=a||"abort",p&&p.abort(a),w(0,a);return this}};h.promise(v),v.success=v.done,v.error=v.fail,v.complete=i.done,v.statusCode=function(a){if(a){var b;if(s<2)for(b in a)j[b]=[j[b],a[b]];else b=a[v.status],v.then(b,b)}return this},d.url=((a||d.url)+"").replace(bE,"").replace(bJ,bU[1]+"//"),d.dataTypes=f.trim(d.dataType||"*").toLowerCase().split(bN),d.crossDomain==null&&(r=bP.exec(d.url.toLowerCase()),d.crossDomain=!(!r||r[1]==bU[1]&&r[2]==bU[2]&&(r[3]||(r[1]==="http:"?80:443))==(bU[3]||(bU[1]==="http:"?80:443)))),d.data&&d.processData&&typeof d.data!="string"&&(d.data=f.param(d.data,d.traditional)),bX(bR,d,c,v);if(s===2)return!1;t=d.global,d.type=d.type.toUpperCase(),d.hasContent=!bI.test(d.type),t&&f.active++===0&&f.event.trigger("ajaxStart");if(!d.hasContent){d.data&&(d.url+=(bK.test(d.url)?"&":"?")+d.data),k=d.url;if(d.cache===!1){var x=f.now(),y=d.url.replace(bO,"$1_="+x);d.url=y+(y===d.url?(bK.test(d.url)?"&":"?")+"_="+x:"")}}(d.data&&d.hasContent&&d.contentType!==!1||c.contentType)&&v.setRequestHeader("Content-Type",d.contentType),d.ifModified&&(k=k||d.url,f.lastModified[k]&&v.setRequestHeader("If-Modified-Since",f.lastModified[k]),f.etag[k]&&v.setRequestHeader("If-None-Match",f.etag[k])),v.setRequestHeader("Accept",d.dataTypes[0]&&d.accepts[d.dataTypes[0]]?d.accepts[d.dataTypes[0]]+(d.dataTypes[0]!=="*"?", */*; q=0.01":""):d.accepts["*"]);for(u in d.headers)v.setRequestHeader(u,d.headers[u]);if(d.beforeSend&&(d.beforeSend.call(e,v,d)===!1||s===2)){v.abort();return!1}for(u in{success:1,error:1,complete:1})v[u](d[u]);p=bX(bS,d,c,v);if(!p)w(-1,"No Transport");else{v.readyState=1,t&&g.trigger("ajaxSend",[v,d]),d.async&&d.timeout>0&&(q=setTimeout(function(){v.abort("timeout")},d.timeout));try{s=1,p.send(l,w)}catch(z){status<2?w(-1,z):f.error(z)}}return v},param:function(a,c){var d=[],e=function(a,b){b=f.isFunction(b)?b():b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};c===b&&(c=f.ajaxSettings.traditional);if(f.isArray(a)||a.jquery&&!f.isPlainObject(a))f.each(a,function(){e(this.name,this.value)});else for(var g in a)bY(g,a[g],c,e);return d.join("&").replace(bB,"+")}}),f.extend({active:0,lastModified:{},etag:{}});var b_=f.now(),ca=/(\=)\?(&|$)|\?\?/i;f.ajaxSetup({jsonp:"callback",jsonpCallback:function(){return f.expando+"_"+b_++}}),f.ajaxPrefilter("json jsonp",function(b,c,d){var e=b.contentType==="application/x-www-form-urlencoded"&&typeof b.data=="string";if(b.dataTypes[0]==="jsonp"||b.jsonp!==!1&&(ca.test(b.url)||e&&ca.test(b.data))){var g,h=b.jsonpCallback=f.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,i=a[h],j=b.url,k=b.data,l="$1"+h+"$2";b.jsonp!==!1&&(j=j.replace(ca,l),b.url===j&&(e&&(k=k.replace(ca,l)),b.data===k&&(j+=(/\?/.test(j)?"&":"?")+b.jsonp+"="+h))),b.url=j,b.data=k,a[h]=function(a){g=[a]},d.always(function(){a[h]=i,g&&f.isFunction(i)&&a[h](g[0])}),b.converters["script json"]=function(){g||f.error(h+" was not called");return g[0]},b.dataTypes[0]="json";return"script"}}),f.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/javascript|ecmascript/},converters:{"text script":function(a){f.globalEval(a);return a}}}),f.ajaxPrefilter("script",function(a){a.cache===b&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),f.ajaxTransport("script",function(a){if(a.crossDomain){var d,e=c.head||c.getElementsByTagName("head")[0]||c.documentElement;return{send:function(f,g){d=c.createElement("script"),d.async="async",a.scriptCharset&&(d.charset=a.scriptCharset),d.src=a.url,d.onload=d.onreadystatechange=function(a,c){if(c||!d.readyState||/loaded|complete/.test(d.readyState))d.onload=d.onreadystatechange=null,e&&d.parentNode&&e.removeChild(d),d=b,c||g(200,"success")},e.insertBefore(d,e.firstChild)},abort:function(){d&&d.onload(0,1)}}}});var cb=a.ActiveXObject?function(){for(var a in cd)cd[a](0,1)}:!1,cc=0,cd;f.ajaxSettings.xhr=a.ActiveXObject?function(){return!this.isLocal&&ce()||cf()}:ce,function(a){f.extend(f.support,{ajax:!!a,cors:!!a&&"withCredentials"in a})}(f.ajaxSettings.xhr()),f.support.ajax&&f.ajaxTransport(function(c){if(!c.crossDomain||f.support.cors){var d;return{send:function(e,g){var h=c.xhr(),i,j;c.username?h.open(c.type,c.url,c.async,c.username,c.password):h.open(c.type,c.url,c.async);if(c.xhrFields)for(j in c.xhrFields)h[j]=c.xhrFields[j];c.mimeType&&h.overrideMimeType&&h.overrideMimeType(c.mimeType),!c.crossDomain&&!e["X-Requested-With"]&&(e["X-Requested-With"]="XMLHttpRequest");try{for(j in e)h.setRequestHeader(j,e[j])}catch(k){}h.send(c.hasContent&&c.data||null),d=function(a,e){var j,k,l,m,n;try{if(d&&(e||h.readyState===4)){d=b,i&&(h.onreadystatechange=f.noop,cb&&delete cd[i]);if(e)h.readyState!==4&&h.abort();else{j=h.status,l=h.getAllResponseHeaders(),m={},n=h.responseXML,n&&n.documentElement&&(m.xml=n),m.text=h.responseText;try{k=h.statusText}catch(o){k=""}!j&&c.isLocal&&!c.crossDomain?j=m.text?200:404:j===1223&&(j=204)}}}catch(p){e||g(-1,p)}m&&g(j,k,m,l)},!c.async||h.readyState===4?d():(i=++cc,cb&&(cd||(cd={},f(a).unload(cb)),cd[i]=d),h.onreadystatechange=d)},abort:function(){d&&d(0,1)}}}});var cg={},ch,ci,cj=/^(?:toggle|show|hide)$/,ck=/^([+\-]=)?([\d+.\-]+)([a-z%]*)$/i,cl,cm=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]],cn,co=a.webkitRequestAnimationFrame||a.mozRequestAnimationFrame||a.oRequestAnimationFrame;f.fn.extend({show:function(a,b,c){var d,e;if(a||a===0)return this.animate(cr("show",3),a,b,c);for(var g=0,h=this.length;g<h;g++)d=this[g],d.style&&(e=d.style.display,!f._data(d,"olddisplay")&&e==="none"&&(e=d.style.display=""),e===""&&f.css(d,"display")==="none"&&f._data(d,"olddisplay",cs(d.nodeName)));for(g=0;g<h;g++){d=this[g];if(d.style){e=d.style.display;if(e===""||e==="none")d.style.display=f._data(d,"olddisplay")||""}}return this},hide:function(a,b,c){if(a||a===0)return this.animate(cr("hide",3),a,b,c);for(var d=0,e=this.length;d<e;d++)if(this[d].style){var g=f.css(this[d],"display");g!=="none"&&!f._data(this[d],"olddisplay")&&f._data(this[d],"olddisplay",g)}for(d=0;d<e;d++)this[d].style&&(this[d].style.display="none");return this},_toggle:f.fn.toggle,toggle:function(a,b,c){var d=typeof a=="boolean";f.isFunction(a)&&f.isFunction(b)?this._toggle.apply(this,arguments):a==null||d?this.each(function(){var b=d?a:f(this).is(":hidden");f(this)[b?"show":"hide"]()}):this.animate(cr("toggle",3),a,b,c);return this},fadeTo:function(a,b,c,d){return this.filter(":hidden").css("opacity",0).show().end().animate({opacity:b},a,c,d)},animate:function(a,b,c,d){var e=f.speed(b,c,d);if(f.isEmptyObject(a))return this.each(e.complete,[!1]);a=f.extend({},a);return this[e.queue===!1?"each":"queue"](function(){e.queue===!1&&f._mark(this);var b=f.extend({},e),c=this.nodeType===1,d=c&&f(this).is(":hidden"),g,h,i,j,k,l,m,n,o;b.animatedProperties={};for(i in a){g=f.camelCase(i),i!==g&&(a[g]=a[i],delete a[i]),h=a[g],f.isArray(h)?(b.animatedProperties[g]=h[1],h=a[g]=h[0]):b.animatedProperties[g]=b.specialEasing&&b.specialEasing[g]||b.easing||"swing";if(h==="hide"&&d||h==="show"&&!d)return b.complete.call(this);c&&(g==="height"||g==="width")&&(b.overflow=[this.style.overflow,this.style.overflowX,this.style.overflowY],f.css(this,"display")==="inline"&&f.css(this,"float")==="none"&&(f.support.inlineBlockNeedsLayout?(j=cs(this.nodeName),j==="inline"?this.style.display="inline-block":(this.style.display="inline",this.style.zoom=1)):this.style.display="inline-block"))}b.overflow!=null&&(this.style.overflow="hidden");for(i in a)k=new f.fx(this,b,i),h=a[i],cj.test(h)?k[h==="toggle"?d?"show":"hide":h]():(l=ck.exec(h),m=k.cur(),l?(n=parseFloat(l[2]),o=l[3]||(f.cssNumber[i]?"":"px"),o!=="px"&&(f.style(this,i,(n||1)+o),m=(n||1)/k.cur()*m,f.style(this,i,m+o)),l[1]&&(n=(l[1]==="-="?-1:1)*n+m),k.custom(m,n,o)):k.custom(m,h,""));return!0})},stop:function(a,b){a&&this.queue([]),this.each(function(){var a=f.timers,c=a.length;b||f._unmark(!0,this);while(c--)a[c].elem===this&&(b&&a[c](!0),a.splice(c,1))}),b||this.dequeue();return this}}),f.each({slideDown:cr("show",1),slideUp:cr("hide",1),slideToggle:cr("toggle",1),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(a,b){f.fn[a]=function(a,c,d){return this.animate(b,a,c,d)}}),f.extend({speed:function(a,b,c){var d=a&&typeof a=="object"?f.extend({},a):{complete:c||!c&&b||f.isFunction(a)&&a,duration:a,easing:c&&b||b&&!f.isFunction(b)&&b};d.duration=f.fx.off?0:typeof d.duration=="number"?d.duration:d.duration in f.fx.speeds?f.fx.speeds[d.duration]:f.fx.speeds._default,d.old=d.complete,d.complete=function(a){f.isFunction(d.old)&&d.old.call(this),d.queue!==!1?f.dequeue(this):a!==!1&&f._unmark(this)};return d},easing:{linear:function(a,b,c,d){return c+d*a},swing:function(a,b,c,d){return(-Math.cos(a*Math.PI)/2+.5)*d+c}},timers:[],fx:function(a,b,c){this.options=b,this.elem=a,this.prop=c,b.orig=b.orig||{}}}),f.fx.prototype={update:function(){this.options.step&&this.options.step.call(this.elem,this.now,this),(f.fx.step[this.prop]||f.fx.step._default)(this)},cur:function(){if(this.elem[this.prop]!=null&&(!this.elem.style||this.elem.style[this.prop]==null))return this.elem[this.prop];var a,b=f.css(this.elem,this.prop);return isNaN(a=parseFloat(b))?!b||b==="auto"?0:b:a},custom:function(a,b,c){function h(a){return d.step(a)}var d=this,e=f.fx,g;this.startTime=cn||cp(),this.start=a,this.end=b,this.unit=c||this.unit||(f.cssNumber[this.prop]?"":"px"),this.now=this.start,this.pos=this.state=0,h.elem=this.elem,h()&&f.timers.push(h)&&!cl&&(co?(cl=!0,g=function(){cl&&(co(g),e.tick())},co(g)):cl=setInterval(e.tick,e.interval))},show:function(){this.options.orig[this.prop]=f.style(this.elem,this.prop),this.options.show=!0,this.custom(this.prop==="width"||this.prop==="height"?1:0,this.cur()),f(this.elem).show()},hide:function(){this.options.orig[this.prop]=f.style(this.elem,this.prop),this.options.hide=!0,this.custom(this.cur(),0)},step:function(a){var b=cn||cp(),c=!0,d=this.elem,e=this.options,g,h;if(a||b>=e.duration+this.startTime){this.now=this.end,this.pos=this.state=1,this.update(),e.animatedProperties[this.prop]=!0;for(g in e.animatedProperties)e.animatedProperties[g]!==!0&&(c=!1);if(c){e.overflow!=null&&!f.support.shrinkWrapBlocks&&f.each(["","X","Y"],function(a,b){d.style["overflow"+b]=e.overflow[a]}),e.hide&&f(d).hide();if(e.hide||e.show)for(var i in e.animatedProperties)f.style(d,i,e.orig[i]);e.complete.call(d)}return!1}e.duration==Infinity?this.now=b:(h=b-this.startTime,this.state=h/e.duration,this.pos=f.easing[e.animatedProperties[this.prop]](this.state,h,0,1,e.duration),this.now=this.start+(this.end-this.start)*this.pos),this.update();return!0}},f.extend(f.fx,{tick:function(){for(var a=f.timers,b=0;b<a.length;++b)a[b]()||a.splice(b--,1);a.length||f.fx.stop()},interval:13,stop:function(){clearInterval(cl),cl=null},speeds:{slow:600,fast:200,_default:400},step:{opacity:function(a){f.style(a.elem,"opacity",a.now)},_default:function(a){a.elem.style&&a.elem.style[a.prop]!=null?a.elem.style[a.prop]=(a.prop==="width"||a.prop==="height"?Math.max(0,a.now):a.now)+a.unit:a.elem[a.prop]=a.now}}}),f.expr&&f.expr.filters&&(f.expr.filters.animated=function(a){return f.grep(f.timers,function(b){return a===b.elem}).length});var ct=/^t(?:able|d|h)$/i,cu=/^(?:body|html)$/i;"getBoundingClientRect"in c.documentElement?f.fn.offset=function(a){var b=this[0],c;if(a)return this.each(function(b){f.offset.setOffset(this,a,b)});if(!b||!b.ownerDocument)return null;if(b===b.ownerDocument.body)return f.offset.bodyOffset(b);try{c=b.getBoundingClientRect()}catch(d){}var e=b.ownerDocument,g=e.documentElement;if(!c||!f.contains(g,b))return c?{top:c.top,left:c.left}:{top:0,left:0};var h=e.body,i=cv(e),j=g.clientTop||h.clientTop||0,k=g.clientLeft||h.clientLeft||0,l=i.pageYOffset||f.support.boxModel&&g.scrollTop||h.scrollTop,m=i.pageXOffset||f.support.boxModel&&g.scrollLeft||h.scrollLeft,n=c.top+l-j,o=c.left+m-k;return{top:n,left:o}}:f.fn.offset=function(a){var b=this[0];if(a)return this.each(function(b){f.offset.setOffset(this,a,b)});if(!b||!b.ownerDocument)return null;if(b===b.ownerDocument.body)return f.offset.bodyOffset(b);f.offset.initialize();var c,d=b.offsetParent,e=b,g=b.ownerDocument,h=g.documentElement,i=g.body,j=g.defaultView,k=j?j.getComputedStyle(b,null):b.currentStyle,l=b.offsetTop,m=b.offsetLeft;while((b=b.parentNode)&&b!==i&&b!==h){if(f.offset.supportsFixedPosition&&k.position==="fixed")break;c=j?j.getComputedStyle(b,null):b.currentStyle,l-=b.scrollTop,m-=b.scrollLeft,b===d&&(l+=b.offsetTop,m+=b.offsetLeft,f.offset.doesNotAddBorder&&(!f.offset.doesAddBorderForTableAndCells||!ct.test(b.nodeName))&&(l+=parseFloat(c.borderTopWidth)||0,m+=parseFloat(c.borderLeftWidth)||0),e=d,d=b.offsetParent),f.offset.subtractsBorderForOverflowNotVisible&&c.overflow!=="visible"&&(l+=parseFloat(c.borderTopWidth)||0,m+=parseFloat(c.borderLeftWidth)||0),k=c}if(k.position==="relative"||k.position==="static")l+=i.offsetTop,m+=i.offsetLeft;f.offset.supportsFixedPosition&&k.position==="fixed"&&(l+=Math.max(h.scrollTop,i.scrollTop),m+=Math.max(h.scrollLeft,i.scrollLeft));return{top:l,left:m}},f.offset={initialize:function(){var a=c.body,b=c.createElement("div"),d,e,g,h,i=parseFloat(f.css(a,"marginTop"))||0,j="<div style='position:absolute;top:0;left:0;margin:0;border:5px solid #000;padding:0;width:1px;height:1px;'><div></div></div><table style='position:absolute;top:0;left:0;margin:0;border:5px solid #000;padding:0;width:1px;height:1px;' cellpadding='0' cellspacing='0'><tr><td></td></tr></table>";f.extend(b.style,{position:"absolute",top:0,left:0,margin:0,border:0,width:"1px",height:"1px",visibility:"hidden"}),b.innerHTML=j,a.insertBefore(b,a.firstChild),d=b.firstChild,e=d.firstChild,h=d.nextSibling.firstChild.firstChild,this.doesNotAddBorder=e.offsetTop!==5,this.doesAddBorderForTableAndCells=h.offsetTop===5,e.style.position="fixed",e.style.top="20px",this.supportsFixedPosition=e.offsetTop===20||e.offsetTop===15,e.style.position=e.style.top="",d.style.overflow="hidden",d.style.position="relative",this.subtractsBorderForOverflowNotVisible=e.offsetTop===-5,this.doesNotIncludeMarginInBodyOffset=a.offsetTop!==i,a.removeChild(b),f.offset.initialize=f.noop},bodyOffset:function(a){var b=a.offsetTop,c=a.offsetLeft;f.offset.initialize(),f.offset.doesNotIncludeMarginInBodyOffset&&(b+=parseFloat(f.css(a,"marginTop"))||0,c+=parseFloat(f.css(a,"marginLeft"))||0);return{top:b,left:c}},setOffset:function(a,b,c){var d=f.css(a,"position");d==="static"&&(a.style.position="relative");var e=f(a),g=e.offset(),h=f.css(a,"top"),i=f.css(a,"left"),j=(d==="absolute"||d==="fixed")&&f.inArray("auto",[h,i])>-1,k={},l={},m,n;j?(l=e.position(),m=l.top,n=l.left):(m=parseFloat(h)||0,n=parseFloat(i)||0),f.isFunction(b)&&(b=b.call(a,c,g)),b.top!=null&&(k.top=b.top-g.top+m),b.left!=null&&(k.left=b.left-g.left+n),"using"in b?b.using.call(a,k):e.css(k)}},f.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),c=this.offset(),d=cu.test(b[0].nodeName)?{top:0,left:0}:b.offset();c.top-=parseFloat(f.css(a,"marginTop"))||0,c.left-=parseFloat(f.css(a,"marginLeft"))||0,d.top+=parseFloat(f.css(b[0],"borderTopWidth"))||0,d.left+=parseFloat(f.css(b[0],"borderLeftWidth"))||0;return{top:c.top-d.top,left:c.left-d.left}},offsetParent:function(){return this.map(function(){var a=this.offsetParent||c.body;while(a&&!cu.test(a.nodeName)&&f.css(a,"position")==="static")a=a.offsetParent;return a})}}),f.each(["Left","Top"],function(a,c){var d="scroll"+c;f.fn[d]=function(c){var e,g;if(c===b){e=this[0];if(!e)return null;g=cv(e);return g?"pageXOffset"in g?g[a?"pageYOffset":"pageXOffset"]:f.support.boxModel&&g.document.documentElement[d]||g.document.body[d]:e[d]}return this.each(function(){g=cv(this),g?g.scrollTo(a?f(g).scrollLeft():c,a?c:f(g).scrollTop()):this[d]=c})}}),f.each(["Height","Width"],function(a,c){var d=c.toLowerCase();f.fn["inner"+c]=function(){var a=this[0];return a&&a.style?parseFloat(f.css(a,d,"padding")):null},f.fn["outer"+c]=function(a){var b=this[0];return b&&b.style?parseFloat(f.css(b,d,a?"margin":"border")):null},f.fn[d]=function(a){var e=this[0];if(!e)return a==null?null:this;if(f.isFunction(a))return this.each(function(b){var c=f(this);c[d](a.call(this,b,c[d]()))});if(f.isWindow(e)){var g=e.document.documentElement["client"+c];return e.document.compatMode==="CSS1Compat"&&g||e.document.body["client"+c]||g}if(e.nodeType===9)return Math.max(e.documentElement["client"+c],e.body["scroll"+c],e.documentElement["scroll"+c],e.body["offset"+c],e.documentElement["offset"+c]);if(a===b){var h=f.css(e,d),i=parseFloat(h);return f.isNaN(i)?h:i}return this.css(d,typeof a=="string"?a:a+"px")}}),a.jQuery=a.$=f})(window);
  </script>
   
   
  <script type="text/javascript">
/**
               _ _____ _ _ _
              | | __ \ (_) | | | |
      ___ ___ | | |__) |___ ___ _ ______ _| |__ | | ___
     / __/ _ \| | _ // _ \/ __| |_ / _` | '_ \| |/ _ \
    | (_| (_) | | | \ \ __/\__ \ |/ / (_| | |_) | | __/
     \___\___/|_|_| \_\___||___/_/___\__,_|_.__/|_|\___|
      
    v1.6 - jQuery plugin created by Alvaro Prieto Lauroba
     
    Licences: MIT & GPL
    Feel free to use or modify this plugin as far as my full name is kept
     
    If you are going to use this plug-in in production environments it is
    strongly recommended to use its minified version: colResizable.min.js
 
*/
 
(function($){
     
    var d = $(document); //window object
    var h = $("head"); //head object
    var drag = null; //reference to the current grip that is being dragged
    var tables = {}; //object of the already processed tables (table.id as key)
    var count = 0; //internal count to create unique IDs when needed.
     
    //common strings for packing
    var ID = "id";
    var PX = "px";
    var SIGNATURE ="JColResizer";
    var FLEX = "JCLRFlex";
     
    //short-cuts
    var I = parseInt;
    var M = Math;
    var ie = navigator.userAgent.indexOf('Trident/4.0')>0;
    var S;
    try{S = sessionStorage;}catch(e){} //Firefox crashes when executed as local file system
     
    //append required CSS rules
    h.append("<style type='text/css'> .JColResizer{table-layout:fixed;} .JColResizer > tbody > tr > td, .JColResizer > tbody > tr > th{overflow:hidden;padding-left:0!important; padding-right:0!important;} .JCLRgrips{ height:0px; position:relative;} .JCLRgrip{margin-left:-5px; position:absolute; z-index:5; } .JCLRgrip .JColResizer{position:absolute;background-color:red;filter:alpha(opacity=1);opacity:0;width:10px;height:100%;cursor: e-resize;top:0px} .JCLRLastGrip{position:absolute; width:1px; } .JCLRgripDrag{ border-left:1px dotted black; } .JCLRFlex{width:auto!important;} .JCLRgrip.JCLRdisabledGrip .JColResizer{cursor:default; display:none;}</style>");
 
     
    /**
     * Function to allow column resizing for table objects. It is the starting point to apply the plugin.
     * @param {DOM node} tb - reference to the DOM table object to be enhanced
     * @param {Object} options - some customization values
     */
    var init = function( tb, options){
        var t = $(tb); //the table object is wrapped
        t.opt = options; //each table has its own options available at anytime
        t.mode = options.resizeMode; //shortcuts
        t.dc = t.opt.disabledColumns;
        if(t.opt.disable) return destroy(t); //the user is asking to destroy a previously colResized table
        var id = t.id = t.attr(ID) || SIGNATURE+count++; //its id is obtained, if null new one is generated
        t.p = t.opt.postbackSafe; //short-cut to detect postback safe
        if(!t.is("table") || tables[id] && !t.opt.partialRefresh) return; //if the object is not a table or if it was already processed then it is ignored.
        if (t.opt.hoverCursor !== 'e-resize') h.append("<style type='text/css'>.JCLRgrip .JColResizer:hover{cursor:"+ t.opt.hoverCursor +"!important}</style>"); //if hoverCursor has been set, append the style
        t.addClass(SIGNATURE).attr(ID, id).before('<div class="JCLRgrips"/>'); //the grips container object is added. Signature class forces table rendering in fixed-layout mode to prevent column's min-width
        t.g = []; t.c = []; t.w = t.width(); t.gc = t.prev(); t.f=t.opt.fixed; //t.c and t.g are arrays of columns and grips respectively
        if(options.marginLeft) t.gc.css("marginLeft", options.marginLeft); //if the table contains margins, it must be specified
        if(options.marginRight) t.gc.css("marginRight", options.marginRight); //since there is no (direct) way to obtain margin values in its original units (%, em, ...)
        t.cs = I(ie? tb.cellSpacing || tb.currentStyle.borderSpacing :t.css('border-spacing'))||2; //table cellspacing (not even jQuery is fully cross-browser)
        t.b = I(ie? tb.border || tb.currentStyle.borderLeftWidth :t.css('border-left-width'))||1; //outer border width (again cross-browser issues)
        // if(!(tb.style.width || tb.width)) t.width(t.width()); //I am not an IE fan at all, but it is a pity that only IE has the currentStyle attribute working as expected. For this reason I can not check easily if the table has an explicit width or if it is rendered as "auto"
        tables[id] = t; //the table object is stored using its id as key
        createGrips(t); //grips are created
     
    };
 
 
    /**
     * This function allows to remove any enhancements performed by this plugin on a previously processed table.
     * @param {jQuery ref} t - table object
     */
    var destroy = function(t){
        var id=t.attr(ID), t=tables[id]; //its table object is found
        if(!t||!t.is("table")) return; //if none, then it wasn't processed
        t.removeClass(SIGNATURE+" "+FLEX).gc.remove(); //class and grips are removed
        delete tables[id]; //clean up data
    };
 
 
    /**
     * Function to create all the grips associated with the table given by parameters
     * @param {jQuery ref} t - table object
     */
    var createGrips = function(t){
     
        var th = t.find(">thead>tr:first>th,>thead>tr:first>td"); //table headers are obtained
        if(!th.length) th = t.find(">tbody>tr:first>th,>tr:first>th,>tbody>tr:first>td, >tr:first>td"); //but headers can also be included in different ways
        th = th.filter(":visible"); //filter invisible columns
        t.cg = t.find("col"); //a table can also contain a colgroup with col elements
        t.ln = th.length; //table length is stored
        if(t.p && S && S[t.id])memento(t,th); //if 'postbackSafe' is enabled and there is data for the current table, its coloumn layout is restored
        th.each(function(i){ //iterate through the table column headers
            var c = $(this); //jquery wrap for the current column
            var dc = t.dc.indexOf(i)!=-1; //is this a disabled column?
            var g = $(t.gc.append('<div class="JCLRgrip"></div>')[0].lastChild); //add the visual node to be used as grip
            g.append(dc ? "": t.opt.gripInnerHtml).append('<div class="'+SIGNATURE+'"></div>');
            if(i == t.ln-1){ //if the current grip is the las one
                g.addClass("JCLRLastGrip"); //add a different css class to stlye it in a different way if needed
                if(t.f) g.html(""); //if the table resizing mode is set to fixed, the last grip is removed since table with can not change
            }
            g.bind('touchstart mousedown', onGripMouseDown); //bind the mousedown event to start dragging
             
            if (!dc){
                //if normal column bind the mousedown event to start dragging, if disabled then apply its css class
                g.removeClass('JCLRdisabledGrip').bind('touchstart mousedown', onGripMouseDown);
            }else{
                g.addClass('JCLRdisabledGrip');
            }
 
            g.t = t; g.i = i; g.c = c; c.w =c.width(); //some values are stored in the grip's node data as shortcut
            t.g.push(g); t.c.push(c); //the current grip and column are added to its table object
            c.width(c.w).removeAttr("width"); //the width of the column is converted into pixel-based measurements
            g.data(SIGNATURE, {i:i, t:t.attr(ID), last: i == t.ln-1}); //grip index and its table name are stored in the HTML
        });
        t.cg.removeAttr("width"); //remove the width attribute from elements in the colgroup
 
        t.find('td, th').not(th).not('table th, table td').each(function(){
            $(this).removeAttr('width'); //the width attribute is removed from all table cells which are not nested in other tables and dont belong to the header
        });
        if(!t.f){
            t.removeAttr('width').addClass(FLEX); //if not fixed, let the table grow as needed
        }
        syncGrips(t); //the grips are positioned according to the current table layout
        //there is a small problem, some cells in the table could contain dimension values interfering with the
        //width value set by this plugin. Those values are removed
         
    };
     
     
    /**
     * Function to allow the persistence of columns dimensions after a browser postback. It is based in
     * the HTML5 sessionStorage object, which can be emulated for older browsers using sessionstorage.js
     * @param {jQuery ref} t - table object
     * @param {jQuery ref} th - reference to the first row elements (only set in deserialization)
     */
    var memento = function(t, th){
        var w,m=0,i=0,aux =[],tw;
        if(th){ //in deserialization mode (after a postback)
            t.cg.removeAttr("width");
            if(t.opt.flush){ S[t.id] =""; return;} //if flush is activated, stored data is removed
            w = S[t.id].split(";"); //column widths is obtained
            tw = w[t.ln+1];
            if(!t.f && tw){ //if not fixed and table width data available its size is restored
                t.width(tw*=1);
                if(t.opt.overflow) { //if overfolw flag is set, restore table width also as table min-width
                    t.css('min-width', tw + PX);
                    t.w = tw;
                }
            }
            for(;i<t.ln;i++){ //for each column
                aux.push(100*w[i]/w[t.ln]+"%"); //width is stored in an array since it will be required again a couple of lines ahead
                th.eq(i).css("width", aux[i] ); //each column width in % is restored
            }
            for(i=0;i<t.ln;i++)
                t.cg.eq(i).css("width", aux[i]); //this code is required in order to create an inline CSS rule with higher precedence than an existing CSS class in the "col" elements
        }else{ //in serialization mode (after resizing a column)
            S[t.id] =""; //clean up previous data
            for(;i < t.c.length; i++){ //iterate through columns
                w = t.c[i].width(); //width is obtained
                S[t.id] += w+";"; //width is appended to the sessionStorage object using ID as key
                m+=w; //carriage is updated to obtain the full size used by columns
            }
            S[t.id]+=m; //the last item of the serialized string is the table's active area (width),
                                                //to be able to obtain % width value of each columns while deserializing
            if(!t.f) S[t.id] += ";"+t.width(); //if not fixed, table width is stored
        }
    };
     
     
    /**
     * Function that places each grip in the correct position according to the current table layout
     * @param {jQuery ref} t - table object
     */
    var syncGrips = function (t){
        t.gc.width(t.w); //the grip's container width is updated
        for(var i=0; i<t.ln; i++){ //for each column
            var c = t.c[i];
            t.g[i].css({ //height and position of the grip is updated according to the table layout
                left: c.offset().left - t.offset().left + c.outerWidth(false) + t.cs / 2 + PX,
                height: t.opt.headerOnly? t.c[0].outerHeight(false) : t.outerHeight(false)
            });
        }
    };
     
     
     
    /**
    * This function updates column's width according to the horizontal position increment of the grip being
    * dragged. The function can be called while dragging if liveDragging is enabled and also from the onGripDragOver
    * event handler to synchronize grip's position with their related columns.
    * @param {jQuery ref} t - table object
    * @param {number} i - index of the grip being dragged
    * @param {bool} isOver - to identify when the function is being called from the onGripDragOver event
    */
    var syncCols = function(t,i,isOver){
        var inc = drag.x-drag.l, c = t.c[i], c2 = t.c[i+1];
        var w = c.w + inc; var w2= c2.w- inc; //their new width is obtained
        c.width( w + PX);
        t.cg.eq(i).width( w + PX);
        if(t.f){ //if fixed mode
            c2.width(w2 + PX);
            t.cg.eq(i+1).width( w2 + PX);
        }else if(t.opt.overflow) { //if overflow is set, incriment min-width to force overflow
            t.css('min-width', t.w + inc);
        }
        if(isOver){
            c.w=w;
            c2.w= t.f ? w2 : c2.w;
        }
    };
 
     
    /**
    * This function updates all columns width according to its real width. It must be taken into account that the
    * sum of all columns can exceed the table width in some cases (if fixed is set to false and table has some kind
    * of max-width).
    * @param {jQuery ref} t - table object
    */
    var applyBounds = function(t){
        var w = $.map(t.c, function(c){ //obtain real widths
            return c.width();
        });
        t.width(t.w = t.width()).removeClass(FLEX); //prevent table width changes
        $.each(t.c, function(i,c){
            c.width(w[i]).w = w[i]; //set column widths applying bounds (table's max-width)
        });
        t.addClass(FLEX); //allow table width changes
    };
     
     
    /**
     * Event handler used while dragging a grip. It checks if the next grip's position is valid and updates it.
     * @param {event} e - mousemove event binded to the window object
     */
    var onGripDrag = function(e){
        if(!drag) return;
        var t = drag.t; //table object reference
        var oe = e.originalEvent.touches;
        var ox = oe ? oe[0].pageX : e.pageX; //original position (touch or mouse)
        var x = ox - drag.ox + drag.l; //next position according to horizontal mouse position increment
        var mw = t.opt.minWidth, i = drag.i ; //cell's min width
        var l = t.cs*1.5 + mw + t.b;
        var last = i == t.ln-1; //check if it is the last column's grip (usually hidden)
        var min = i? t.g[i-1].position().left+t.cs+mw: l; //min position according to the contiguous cells
        var max = t.f ? //fixed mode?
            i == t.ln-1?
                t.w-l:
                t.g[i+1].position().left-t.cs-mw:
            Infinity; //max position according to the contiguous cells
        x = M.max(min, M.min(max, x)); //apply bounding
        drag.x = x; drag.css("left", x + PX); //apply position increment
        if(last){ //if it is the last grip
            var c = t.c[drag.i]; //width of the last column is obtained
            drag.w = c.w + x- drag.l;
        }
        if(t.opt.liveDrag){ //if liveDrag is enabled
            if(last){
                c.width(drag.w);
                if(!t.f && t.opt.overflow){ //if overflow is set, incriment min-width to force overflow
                   t.css('min-width', t.w + x - drag.l);
                }else {
                    t.w = t.width();
                }
            }else{
                syncCols(t,i); //columns are synchronized
            }
            syncGrips(t);
            var cb = t.opt.onDrag; //check if there is an onDrag callback
            if (cb) { e.currentTarget = t[0]; cb(e); } //if any, it is fired
        }
        return false; //prevent text selection while dragging
    };
     
 
    /**
     * Event handler fired when the dragging is over, updating table layout
     * @param {event} e - grip's drag over event
     */
    var onGripDragOver = function(e){
         
        d.unbind('touchend.'+SIGNATURE+' mouseup.'+SIGNATURE).unbind('touchmove.'+SIGNATURE+' mousemove.'+SIGNATURE);
        $("head :last-child").remove(); //remove the dragging cursor style
        if(!drag) return;
        drag.removeClass(drag.t.opt.draggingClass); //remove the grip's dragging css-class
        if (!(drag.x - drag.l == 0)) {
            var t = drag.t;
            var cb = t.opt.onResize; //get some values
            var i = drag.i; //column index
            var last = i == t.ln-1; //check if it is the last column's grip (usually hidden)
            var c = t.g[i].c; //the column being dragged
            if(last){
                c.width(drag.w);
                c.w = drag.w;
            }else{
                syncCols(t, i, true); //the columns are updated
            }
            if(!t.f) applyBounds(t); //if not fixed mode, then apply bounds to obtain real width values
            syncGrips(t); //the grips are updated
            if (cb) { e.currentTarget = t[0]; cb(e); } //if there is a callback function, it is fired
            if(t.p && S) memento(t); //if postbackSafe is enabled and there is sessionStorage support, the new layout is serialized and stored
        }
        drag = null; //since the grip's dragging is over
    };
     
     
    /**
     * Event handler fired when the grip's dragging is about to start. Its main goal is to set up events
     * and store some values used while dragging.
     * @param {event} e - grip's mousedown event
     */
    var onGripMouseDown = function(e){
        var o = $(this).data(SIGNATURE); //retrieve grip's data
        var t = tables[o.t], g = t.g[o.i]; //shortcuts for the table and grip objects
        var oe = e.originalEvent.touches; //touch or mouse event?
        g.ox = oe? oe[0].pageX: e.pageX; //the initial position is kept
        g.l = g.position().left;
        g.x = g.l;
         
        d.bind('touchmove.'+SIGNATURE+' mousemove.'+SIGNATURE, onGripDrag ).bind('touchend.'+SIGNATURE+' mouseup.'+SIGNATURE, onGripDragOver); //mousemove and mouseup events are bound
        h.append("<style type='text/css'>*{cursor:"+ t.opt.dragCursor +"!important}</style>"); //change the mouse cursor
        g.addClass(t.opt.draggingClass); //add the dragging class (to allow some visual feedback)
        drag = g; //the current grip is stored as the current dragging object
        if(t.c[o.i].l) for(var i=0,c; i<t.ln; i++){ c=t.c[i]; c.l = false; c.w= c.width(); } //if the colum is locked (after browser resize), then c.w must be updated
        return false; //prevent text selection
    };
     
     
    /**
     * Event handler fired when the browser is resized. The main purpose of this function is to update
     * table layout according to the browser's size synchronizing related grips
     */
    var onResize = function(){
        for(var t in tables){
            if( tables.hasOwnProperty( t ) ) {
                t = tables[t];
                var i, mw=0;
                t.removeClass(SIGNATURE); //firefox doesn't like layout-fixed in some cases
                if (t.f) { //in fixed mode
                    t.w = t.width(); //its new width is kept
                    for(i=0; i<t.ln; i++) mw+= t.c[i].w;
                    //cell rendering is not as trivial as it might seem, and it is slightly different for
                    //each browser. In the beginning i had a big switch for each browser, but since the code
                    //was extremely ugly now I use a different approach with several re-flows. This works
                    //pretty well but it's a bit slower. For now, lets keep things simple...
                    for(i=0; i<t.ln; i++) t.c[i].css("width", M.round(1000*t.c[i].w/mw)/10 + "%").l=true;
                    //c.l locks the column, telling us that its c.w is outdated
                }else{ //in non fixed-sized tables
                    applyBounds(t); //apply the new bounds
                    if(t.mode == 'flex' && t.p && S){ //if postbackSafe is enabled and there is sessionStorage support,
                        memento(t); //the new layout is serialized and stored for 'flex' tables
                    }
                }
                syncGrips(t.addClass(SIGNATURE));
            }
        }
         
    };
 
 
    //bind resize event, to update grips position
    $(window).bind('resize.'+SIGNATURE, onResize);
 
 
    /**
     * The plugin is added to the jQuery library
     * @param {Object} options - an object that holds some basic customization values
     */
    $.fn.extend({
        colResizable: function(options) {
            var defaults = {
             
                //attributes:
                 
                resizeMode: 'fit', //mode can be 'fit', 'flex' or 'overflow'
                draggingClass: 'JCLRgripDrag', //css-class used when a grip is being dragged (for visual feedback purposes)
                gripInnerHtml: '', //if it is required to use a custom grip it can be done using some custom HTML
                liveDrag: false, //enables table-layout updating while dragging
                minWidth: 15, //minimum width value in pixels allowed for a column
                headerOnly: false, //specifies that the size of the the column resizing anchors will be bounded to the size of the first row
                hoverCursor: "e-resize", //cursor to be used on grip hover
                dragCursor: "e-resize", //cursor to be used while dragging
                postbackSafe: false, //when it is enabled, table layout can persist after postback or page refresh. It requires browsers with sessionStorage support (it can be emulated with sessionStorage.js).
                flush: false, //when postbakSafe is enabled, and it is required to prevent layout restoration after postback, 'flush' will remove its associated layout data
                marginLeft: null, //in case the table contains any margins, colResizable needs to know the values used, e.g. "10%", "15em", "5px" ...
                marginRight: null, //in case the table contains any margins, colResizable needs to know the values used, e.g. "10%", "15em", "5px" ...
                disable: false, //disables all the enhancements performed in a previously colResized table
                partialRefresh: false, //can be used in combination with postbackSafe when the table is inside of an updatePanel,
                disabledColumns: [], //column indexes to be excluded
 
                //events:
                onDrag: null, //callback function to be fired during the column resizing process if liveDrag is enabled
                onResize: null //callback function fired when the dragging process is over
            }
            var options = $.extend(defaults, options);
             
            //since now there are 3 different ways of resizing columns, I changed the external interface to make it clear
            //calling it 'resizeMode' but also to remove the "fixed" attribute which was confusing for many people
            options.fixed = true;
            options.overflow = false;
            switch(options.resizeMode){
                case 'flex': options.fixed = false; break;
                case 'overflow': options.fixed = false; options.overflow = true; break;
            }
 
            return this.each(function() {
                 init( this, options);
            });
        }
    });
})(jQuery);
 
   
</script>
   
  <script type="text/javascript">
    $(function(){
 
     
        $("#normal").colResizable({
            liveDrag:true,
            gripInnerHtml:"<div class='grip'></div>",
            draggingClass:"dragging",
            resizeMode:'fit'
        });
         
        $("#flex").colResizable({
            liveDrag:true,
            gripInnerHtml:"<div class='grip'></div>",
            draggingClass:"dragging",
            resizeMode:'flex'
        });
 
 
      $("#overflow").colResizable({
          liveDrag:true,
          gripInnerHtml:"<div class='grip'></div>",
          draggingClass:"dragging",
          resizeMode:'overflow'
      });
         
 
      $("#disabled").colResizable({
          liveDrag:true,
          gripInnerHtml:"<div class='grip'></div>",
          draggingClass:"dragging",
          resizeMode:'overflow',
          disabledColumns: [2]
      });
 
         
    });
  </script>
  
 <script>
function myFunction() {
  // Declare variables
  var input, filter, table, tr, td1,td2,td3,td4,td5,td6,td7,td8,td9, i;
  input = document.getElementById("myInput");
  filter = input.value.toUpperCase();
  table = document.getElementById("normal");
  //tr = table.getElementsByTagName("tr");
  tr = table.getElementsByClassName("trrow");
 
  // Loop through all table rows, and hide those who don't match the search query
  for (i = 0; i < tr.length; i++) {
   td1 = tr[i].getElementsByClassName("EntityName")[0];
     td2 = tr[i].getElementsByClassName("EntityId")[0];
     td3 = tr[i].getElementsByClassName("Path")[0];
     td4 = tr[i].getElementsByClassName("ClassName")[0];
     td5 = tr[i].getElementsByClassName("ClassId")[0];
     td6 = tr[i].getElementsByClassName("WorkflowName")[0];
     td7 = tr[i].getElementsByClassName("WorkflowDisplayName")[0];
     td8 = tr[i].getElementsByClassName("WorkflowType")[0];
     td9 = tr[i].getElementsByClassName("WorkflowId")[0];
 
    if (td1) {
      if ( (td1.innerHTML.toUpperCase().indexOf(filter) > -1) || (td2.innerHTML.toUpperCase().indexOf(filter) > -1) || (td3.innerHTML.toUpperCase().indexOf(filter) > -1) || (td4.innerHTML.toUpperCase().indexOf(filter) > -1) || (td5.innerHTML.toUpperCase().indexOf(filter) > -1) || (td6.innerHTML.toUpperCase().indexOf(filter) > -1) || (td7.innerHTML.toUpperCase().indexOf(filter) > -1) || (td8.innerHTML.toUpperCase().indexOf(filter) > -1) || (td9.innerHTML.toUpperCase().indexOf(filter) > -1) ) {
        tr[i].style.display = "";
      } else {
        tr[i].style.display = "none";
      }
    }
  }
}
</script>
  
    </head>
<body>
<input type="text" id="myInput" onkeyup="myFunction()" placeholder="Search for text..">
<div class="center" >
     
<table id="normal" width="400%" border="0" cellpadding="5" cellspacing="0">
 
 
'@



    $HTML += @"
$headerHTML
"@


    $i =1

    #Build rows
    ForEach ($WF in $arrWFs) {
      $rows += @"
  <tr class="trrow">
 
"@


      ForEach ($header in $arrOrderedHeaders) {
        If ($header -eq 'Index') { 
          $rows += @"
      <td class="$($Header)">$($i)</td>
 
"@
    
        }
        Else {
          $rows += @"
      <td class="$($Header)">$($WF.$Header)</td>
 
"@

        }
      } #end ForEach Header
      $i++
    } #end ForEach WF


    $rows += @"
  </tr>
 
"@



    $HTML += @"
$rows
</table>
</div>
</body>
</html>
"@


    Return $HTML
  
  } #End Function Build-HTMLFancyResize
  ###################################################################
  
  Function Export-HTMLFile {
    [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
        SupportsShouldProcess=$true, 
    PositionalBinding=$false)]
    Param (
      #array of workflows to be exported to the Path
      [Parameter(Mandatory=$false, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false, 
          Position=0,
      ParameterSetName='Parameter Set 1')]
      [System.Object[]]$arrWFs,

      #export file path
      [Parameter(Mandatory=$false, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false, 
          Position=1,
      ParameterSetName='Parameter Set 1')]
      [string]$Path,

      #title of HTML doc
      [Parameter(Mandatory=$false, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false, 
          Position=2,
      ParameterSetName='Parameter Set 1')]
      [string]$Title = "",

      #which formatting type
      [Parameter(Mandatory=$false, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false, 
          Position=3,
      ParameterSetName='Parameter Set 1')]
      [ValidateSet("Fancy", "Basic")]
      [string]$Format = 'Fancy'
    )
    Write-Host "Building HTML document..." -ForegroundColor Gray
    Switch ($Format){
      "Fancy" {
        $HTML = Build-HTMLFancyResize -arrWFs $arrWFs -Title $Title
      }
      "Basic" {
        $HTML = Build-HTMLBasic -arrWFs $arrWFs -Title $Title
      }

      Default {
        Throw "Problem with Export-HTMLFile. Exiting..."
      }

    }

    Try {
      $HTML | Set-Content -Path $Path -Force
      Write-Verbose "`nFiles written to directory [$(Split-Path $Path -Parent)]"
      Return $Path
    }Catch {
      Write-Error "Unable to write content to HTML file: [$Path]. `n$($Error[0].Message)"
    }
  }
  ###################################################################
  Function Export-CSVFile {
    Param (
      #export file path
      [string]$Path,

      #array of workflows to be exported to the Path
      [System.Object[]]$arrWFs
    )

    Try {
      $arrWFs | Export-Csv -Path $Path -NoTypeInformation -Force
      Write-Verbose "`nFiles written to directory [$(Split-Path $Path -Parent)]"
      Return $Path
    }Catch {
      Write-Error "Unable to write content to CSV file: [$Path]. `n$($Error[0].Message)"
    }
  }

  ###################################################################
  Function FormatObj-ForCSVExport {
    Param (
      # New custom workflow object being built
      [System.Object]$obj,

      # Workflow (from effective config method)
      [System.Object]$WF
    )

    $Params = ''

    Try {
      Switch ($WorkFlow.Type){
        "Rule" {
          $Obj.WorkflowDisplayName = ($hashAllRules[$WorkFlow.WorkflowId.Guid].DisplayName)
          If ($hashAllRules[$WorkFlow.WorkflowId.Guid].DataSourceCollection.Configuration) {
            $Params = (([xml]"<Parameters>$($hashAllRules[$WorkFlow.WorkflowId.Guid].DataSourceCollection.Configuration)</Parameters>").ChildNodes )
          }
          $Obj.WorkflowMPName = ($hashAllRules[$WorkFlow.WorkflowId.Guid].Identifier.Domain[0])
        }

        "Monitor" {
          $Obj.WorkflowDisplayName = ($hashAllMonitors[$WorkFlow.WorkflowId.Guid].DisplayName)
          If ($hashAllMonitors[$WorkFlow.WorkflowId.Guid].Configuration) {
            $Params = (([xml]"<Parameters>$($hashAllMonitors[$WorkFlow.WorkflowId.Guid].Configuration)</Parameters>").ChildNodes )
          }
          $Obj.WorkflowMPName = ($hashAllMonitors[$WorkFlow.WorkflowId.Guid].Identifier.Domain[0])
        }
        default {
          $Obj.WorkflowDisplayName = ($WorkFlow.WorkflowName)
          Write-Host "WorkFlow.Type: '$($WorkFlow.Type)' not found!" -F Yellow
        }
      }
    } Catch { 
      $Obj.WorkflowDisplayName = ($WorkFlow.WorkflowName)
    }

    If ($Params) {
      $propertyNames = ''
      $propertyNames = $Params | Get-Member -MemberType Property | Select-Object Name -ExpandProperty Name
      $ParamsFormattedCSV =''
      ForEach ($Name in $propertyNames) {
        $ParamsFormattedCSV += @"
$($Name): $($Params.$Name)
 
"@
    
      }

      $Obj.Parameters = $ParamsFormattedCSV
    }

    $OverridesFormattedCSV = ''
    If ($WorkFlow.Overridden) {
      ForEach ($Override in $WorkFlow.OverrideableParameters) {
        $propertyNames = $Override | Get-Member -MemberType Property | Select-Object Name -ExpandProperty Name
        $OverridesFormattedCSV += @"
ParameterName: $($Override.ParameterName)
DefaultValue: $($Override.DefaultValue)
EffectiveValue: $($Override.EffectiveValue)
<><><><><><><><><><><><><><><><><>
 
"@
    
      } #end ForEach Override

      $Obj | Add-Member -MemberType NoteProperty -Name "Overrides" -Value $OverridesFormattedCSV
    }
    Else {
      #No Overrides
      $Obj | Add-Member -MemberType NoteProperty -Name "Overrides" -Value ''
    }

    Return $Obj

  }
  ###################################################################
  <# This function is for formatting the Parameters and Overrides fields specifically for HTML.
  #>

  Function FormatObj-ForHTMLExport {
    Param (
      # New custom workflow object being built
      [System.Object]$Obj
    )
    $Params = ''

    Try {
      Switch ($WorkFlow.Type){
        "Rule" {
          $Obj.WorkflowDisplayName = ($hashAllRules[$WorkFlow.WorkflowId.Guid].DisplayName)
          If ($hashAllRules[$WorkFlow.WorkflowId.Guid].DataSourceCollection.Configuration) {
            $Params = (([xml]"<Parameters>$($hashAllRules[$WorkFlow.WorkflowId.Guid].DataSourceCollection.Configuration)</Parameters>").ChildNodes )
          }
          $Obj.WorkflowMPName = ($hashAllRules[$WorkFlow.WorkflowId.Guid].Identifier.Domain[0])
        }

        "Monitor" {
          $Obj.WorkflowDisplayName = ($hashAllMonitors[$WorkFlow.WorkflowId.Guid].DisplayName)
          If ($hashAllMonitors[$WorkFlow.WorkflowId.Guid].Configuration) {
            $Params = (([xml]"<Parameters>$($hashAllMonitors[$WorkFlow.WorkflowId.Guid].Configuration)</Parameters>").ChildNodes )
          }
          $Obj.WorkflowMPName = ($hashAllMonitors[$WorkFlow.WorkflowId.Guid].Identifier.Domain[0])
        }
        default {$Obj.WorkflowDisplayName = ($WorkFlow.WorkflowName)}
      }
    } Catch { 
      $Obj.WorkflowDisplayName = ($WorkFlow.WorkflowName)
    }

    If ($Params) {
      $propertyNames = ''
      $propertyNames = $Params | Get-Member -MemberType Property | Select-Object Name -ExpandProperty Name
      $embedTable = @"
    <table border=1 cellpadding="5">
        <tr style="font-weight:bold">
        <td>Name</td>
        <td>Value</td>
        </tr>
     
"@

      ForEach ($Name in $propertyNames) {
        $value = ( ( (($Params.$Name -Replace '&','&amp;') -Replace '<', '&lt;' ) -Replace '>', '&gt;' ) -Replace '\n', $BR_Token)
        $embedTable += @"
        <tr>
        <td>$($Name)</td>
        <td>$($value)</td>
        </tr>
     
"@
    
      }
      $embedTable += @"
    </table>
     
"@

      $Obj.Parameters = $embedTable
    }

    If ($WorkFlow.Overridden) {
      $embedTable = @"
          <table border=1 cellpadding="5">
              <tr style="font-weight:bold">
                <td>ParameterName</td>
                <td>DefaultValue</td>
              <td>EffectiveValue</td>
              </tr>
     
"@

      ForEach ($Override in $WorkFlow.OverrideableParameters) {
        $propertyNames = $Override | Get-Member -MemberType Property | Select-Object Name -ExpandProperty Name
        $embedTable += @"
              <tr>
                <td>$($Override.ParameterName)</td>
                <td>$($Override.DefaultValue)</td>
                <td>$($Override.EffectiveValue)</td>
            </tr>
     
"@
    
      } #end ForEach Override

      $embedTable += @"
          </table>
     
"@

      $Obj | Add-Member -MemberType NoteProperty -Name "Overrides" -Value $embedTable
    }
    Else {
      #No Overrides
      $Obj | Add-Member -MemberType NoteProperty -Name "Overrides" -Value ''
    }

    Return $Obj
  }
  ###################################################################

  Function Get-CleanNames {
    Param (
      # Should be a MonitoringObject
      [Microsoft.EnterpriseManagement.Monitoring.MonitoringObject]$MonitoringObject
    )
  
    $objCleanNames = New-Object -TypeName pscustomobject
    $objCleanNames | Add-Member -MemberType NoteProperty -Name "Name" -Value (Clean-Name $MonitoringObject.Name)
    If ($MonitoringObject.DisplayName) {
      $objCleanNames | Add-Member -MemberType NoteProperty -Name "DN" -Value (Clean-Name $MonitoringObject.DisplayName)
    }
    Else {
      $objCleanNames | Add-Member -MemberType NoteProperty -Name "DN" -Value (Clean-Name $MonitoringObject.Name)
    }
    $objCleanNames | Add-Member -MemberType NoteProperty -Name "Path" -Value (Clean-Name $MonitoringObject.Path)

    Return $objCleanNames
  }
  ###################################################################

  Function Get-SCOMRelatedObjects {
    [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
    SupportsShouldProcess=$true)]
                    
    Param (
  
      # This should be a SCOM MonitoringObject
      [Parameter(Mandatory=$true, 
          ValueFromPipeline=$true,
          ValueFromPipelineByPropertyName=$true, 
          ValueFromRemainingArguments=$true, 
          Position=0,
      ParameterSetName='Parameter Set 1')]
      [ValidateNotNull()]
      [ValidateNotNullOrEmpty()]
      [Microsoft.EnterpriseManagement.Monitoring.MonitoringObject[]]$object
    )
  
    #region Begin
    Begin {
      [System.Collections.ArrayList]$arr = @()

      Function Dig-Object {
        [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
        SupportsShouldProcess=$true)]
        Param (
  
          # Param1 help description
          [Parameter(Mandatory=$true, 
              ValueFromPipeline=$true,
              ValueFromPipelineByPropertyName=$true, 
              ValueFromRemainingArguments=$true, 
              Position=0,
          ParameterSetName='Parameter Set 1')]
          [ValidateNotNull()]
          [ValidateNotNullOrEmpty()]
          [system.object]$object,
          [string]$indent
        )

      
        Begin{
          [System.Collections.ArrayList]$arr = @()
          #[System.Collections.ArrayList]$related = @()
        }
        Process {
          Write-Verbose "$indent $($_.DisplayName) , $($_.GetLeastDerivedNonAbstractClass())"
          $NULL = $arr.Add($object)
          $tmpRelated = $object.GetRelatedMonitoringObjects()
          If ($tmpRelated.Count -ge 1) { 
            ForEach ($item in ($tmpRelated | Dig-Object -indent ($indent + "--"))) {
              $NULL = $arr.Add($item)
            }
          }
        }
        End {
          Return ($arr )
        }
      }

    }#endregion Begin
    Process{
      ForEach ($item in $object){
        #$arr += $item | Dig-Object -indent "--"
        $item | Dig-Object -indent "--" | ForEach-Object {$NULL = $arr.Add($_)}
      }
    }  
    End{
      Return ($arr | Select-Object -Unique)
    }
    
  }#end Get-SCOMRelatedObjects

  ###################################################################
  Function Load-Cache {
    # Load all Monitors
    Write-Host "Getting monitors..." -F Gray
    $AllMonitors = Get-SCOMMonitor
    $hashAllMonitors = @{}
    $i =1
    ForEach ($Mon in $AllMonitors) {
      $percent = [math]::Round((($i / $AllMonitors.Count) *100),0)
      Write-Progress -Activity "** What's happening? **" -status "Loading monitors. Be patient! [Percent: $($percent)]" -PercentComplete $percent
      $hashAllMonitors.Add($Mon.Id.Guid, $Mon)
      $i++
    }
    Write-Progress -Activity "** What's happening? **" -status "Complete" -Completed
    # Load all rules
    Write-Host "Getting rules..." -F Gray
    $AllRules = Get-SCOMRule
    $hashAllRules = @{}
    $i=1
    ForEach ($Rule in $AllRules) {
      $percent = [math]::Round((($i / $AllRules.Count) *100),0)
      Write-Progress -Activity "** What's happening? **" -status "Loading Rules. Be patient! [Percent: $($percent)]" -PercentComplete $percent
      $hashAllRules.Add($Rule.Id.Guid, $Rule)
      $i++
    }
    Write-Progress -Activity "** What's happening? **" -status "Complete" -Completed
  }
  ###################################################################
  

  #---------------------------------------------------------------
  #--------------------------- MAIN ------------------------------

  #Build headers
  # 2022.06.20 removed "Index" as it was not meaningful when recursively enumerating workflows
  $arrOrderedHeaders = @"
EntityName
EntityId
Path
ClassName
ClassId
WorkflowName
WorkflowDisplayName
WorkflowType
WorkflowId
WorkflowMPName
Enabled
GeneratesAlert
AlertSeverity
AlertPriority
Overridden
Description
Parameters
Overrides
"@
 -split '\r\n'

  Switch ($SortOrder ) {
    'Ascending' {
      [BOOL]$SortOrder = $false
    }
    'Descending' {
      [BOOL]$SortOrder = $true
    }
    Default {
      [BOOL]$SortOrder = $false
    }
  }

  $BR_Token = '<BR />'


  #Validate arguments/paths
  If ( (-NOT($ExportCSVPath)) -AND (-NOT($ExportHTMLPath)) ) {
    Throw 'You must provide at least one valid export path/directory with -ExportHTMLPath or -ExportCSVPath. Exiting...'
  }
  If ($ExportCSVPath) {
    Try{
      If (-NOT(Test-Path -Path $ExportCSVPath -PathType Container)){
        Write-Host "Path: [$($ExportCSVPath)] does not exist. Creating folder now..."
        New-Item -Path $ExportCSVPath -ItemType Directory -ErrorAction Stop
      }
    }
    Catch{
      Write-Error "Invalid path: [$($ExportCSVPath)]. Must be valid directory. Unable to create directory."
      Return 1
    }
  }
  If ($ExportHTMLPath) {
    Try{
      If (-NOT(Test-Path -Path $ExportHTMLPath -PathType Container)){
        Write-Host "Path: [$($ExportHTMLPath)] does not exist. Creating folder now..."
        New-Item -Path $ExportHTMLPath -ItemType Directory -ErrorAction Stop
      }
      Else {
        $SearchIconPath = Join-Path -Path $ExportHTMLPath -ChildPath 'searchicon.png'
      }
    }
    Catch{
      Write-Error "Invalid path: [$($ExportHTMLPath)]. Must be valid directory. Unable to create directory."
      Return 1
    }
  }

  If (-NOT $TESTING){
    Write-Output "Loading all rules and monitors..."
    . Load-Cache
  }

  #---------------------------------------------------------------


  #Thanks to Craig Pero for pointing me in the right direction here!
  $omMg = Get-SCOMManagementGroup | Select-Object -First 1

  #array to contain master list of all workflows of all related objects
  [System.Collections.ArrayList]$arrAllWorkflowsCSV = @()
  [System.Collections.ArrayList]$arrAllWorkflowsHTML = @()

  $timer = [System.Diagnostics.Stopwatch]::StartNew()
  #array to contain all related objects
  $arrObjects = Get-SCOMRelatedObjects -object $MonitoringObject
  Write-Host "$($timer.Elapsed.TotalSeconds) second to retrieve all related objects" -ForegroundColor Gray
  Write-Host "`n`n`n`n"   #Start below Write-Progress bar
  
  $hashSCI = @{} #Keep a collection of each class instance retrieved. Retrieve each class instance only once.
  $i = 1 #counter for the progress bar
  $timer = [System.Diagnostics.Stopwatch]::StartNew()
  [bool]$SkipFixOR = $false
  
  #region
  ForEach ($obj in ($arrObjects |Sort-Object -Property DisplayName) ) {
    $CleanNames = ''
    $CleanName = Get-CleanNames -MonitoringObject $obj

    Write-Host "$($i): " -ForegroundColor Cyan -NoNewline; `
    Write-Host "[" -ForegroundColor Red -NoNewline; `
    Write-Host "$($obj.Path)" -ForegroundColor Yellow -NoNewline; `
    Write-Host "]" -ForegroundColor Red -NoNewline; `
    Write-Host " $($obj.FullName)"  -ForegroundColor Green
  
    $percent = [math]::Round((($i / $arrObjects.Count) *100),0)
    Write-Progress -Activity "** What's happening? **" -status "Getting your data. Be patient! [Percent: $($percent)]" -PercentComplete $percent
    Try{
      $Workflows = $omMg.EffectiveMonitoring.GetAllEffectiveMonitoringWorkflows($obj.Id,$true)
    }
    Catch{
      # It's possible to have "not lowercase" true|false values for the Enabled property which causes this report/method to puke
      Write-Verbose ('Unable to execute: $Workflows = $omMg.EffectiveMonitoring.GetAllEffectiveMonitoringWorkflows($obj.id,$true)' + " for instance of class: Microsoft.Windows.Computer. $($obj.FullName) , ID: $($obj.Id)")
      Write-Error $Error[0].Exception
      Write-Host "Skipping item: $($obj.FullName) , ID: $($obj.Id)" -F Gray
      
      #<#
      If ( ($Error[0].Exception -match 'true|false') -AND ($Error[0].Exception -cmatch '[^true|^false]') -AND (-NOT $SkipFixOR) ) { 
        $choice = ''
        $choice = $null
        While ($choice -notmatch '^n$|^a$') {
          Write-Host "Problem detected with `"Enabled`" property value. This value must be all lowercase. Would you like to attempt to fix overrides?"  -ForegroundColor Yellow -BackgroundColor Red
          $choice = (Read-Host "fix [a]ll or [n]o ?").ToLower()
        }
        # Fix overrides
        If ($choice -eq 'a') {
          If ($ExportCSVPath){ $tmpPath = $ExportCSVPath }
          Else { $tmpPath = $ExportHTMLPath}
          Set-OverrideEnabledCase -ExportPath $tmpPath
          Write-Host "`nOverride Values should be fixed. Try to run this report again. `n`nExiting..." -F Yellow
          Exit
        }
           
        #Do not fix overrides. Don't prompt again.
        Else {
          Write-Host "Will not prompt again to repair overrides. " -F Gray
          $SkipFixOR = $true #user will not be prompted again to fix
        }
      }
      
      Continue;
    } #end Catch
    #>
    

    [System.Collections.ArrayList]$arrThisObjWorkflowsCSV = @()
    [System.Collections.ArrayList]$arrThisObjWorkflowsHTML = @()
    #[int]$w = 1
    #region
    ForEach ($WorkFlow in ($Workflows | Sort-Object -Property EntityId -Descending:$SortOrder )) {
      $tmpObj = New-Object -TypeName PSCUSTOMOBJECT
      # $tmpObj | Add-Member -MemberType NoteProperty -Name "NAME" -Value "VALUE"
      #$tmpObj | Add-Member -MemberType NoteProperty -Name "Index" -Value $w
      $tmpObj | Add-Member -MemberType NoteProperty -Name "EntityName" -Value ($WorkFlow.EntityName)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "EntityId" -Value ($WorkFlow.EntityId.Guid)

      # Retrieve each class instance only once.
      If (-NOT $hashSCI[($WorkFlow.EntityId.Guid)] ) {
        $hashSCI[($WorkFlow.EntityId.Guid)] = (Get-SCOMClassInstance -Id $WorkFlow.EntityId.Guid)
      }
      $tmpObj | Add-Member -MemberType NoteProperty -Name "Path" -Value (($hashSCI[($WorkFlow.EntityId.Guid)]).Path )
     
       
      $tmpObj | Add-Member -MemberType NoteProperty -Name "ClassName" -Value ($WorkFlow.EntityTypeName)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "ClassId" -Value ($WorkFlow.EntityTypeId.Guid)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "WorkflowName" -Value ($WorkFlow.WorkflowName)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "WorkflowDisplayName" -Value "PLACEHOLDER"
      $tmpObj | Add-Member -MemberType NoteProperty -Name "WorkflowType" -Value ($WorkFlow.Type)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "WorkflowId" -Value ($WorkFlow.WorkflowId.Guid)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "WorkflowMPName" -Value "PLACEHOLDER" 
      $tmpObj | Add-Member -MemberType NoteProperty -Name "Enabled" -Value ($WorkFlow.Enabled.ToString())
      $tmpObj | Add-Member -MemberType NoteProperty -Name "GeneratesAlert" -Value ($WorkFlow.GeneratesAlert.ToString())
      $tmpObj | Add-Member -MemberType NoteProperty -Name "AlertSeverity" -Value ($WorkFlow.AlertSeverity)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "AlertPriority" -Value ($WorkFlow.AlertPriority)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "Overridden" -Value ($WorkFlow.Overridden.ToString())
      $tmpObj | Add-Member -MemberType NoteProperty -Name "Description" -Value ($WorkFlow.Description)
      $tmpObj | Add-Member -MemberType NoteProperty -Name "Parameters" -Value ''

      If ($ExportCSVPath){
        $tmpObjCSV = FormatObj-ForCSVExport -Obj $tmpObj.PsObject.Copy()
        $arrAllWorkflowsCSV.Add($tmpObjCSV) | Out-Null
        $arrThisObjWorkflowsCSV.Add($tmpObjCSV) | Out-Null
      }
      If ($ExportHTMLPath){
        $tmpObjHTML = FormatObj-ForHTMLExport -Obj $tmpObj.PsObject.Copy()
        $arrAllWorkflowsHTML.Add($tmpObjHTML) | Out-Null
        $arrThisObjWorkflowsHTML.Add($tmpObjHTML) | Out-Null
      }
      $w++
    }
    #endregion ForEach Workflow (of the Obj)
  
    If ($ExportCSVPath -AND $IndividualFiles){
      Try {
        $ObjExportFilePath = (Join-Path $ExportCSVPath "($($CleanName.Path))_$($CleanName.DN).csv" )
        Export-CSVFile -arrWFs $arrThisObjWorkflowsCSV -Path $ObjExportFilePath -NoTypeInformation -Force
      }Catch {
        Write-Error "Problem exporting to CSV [$ObjExportFilePath]. $($Error[0].Message)"
      }
    }
  
    If ($ExportHTMLPath -AND $IndividualFiles) {
      Try {
        $ObjExportFilePath = (Join-Path $ExportHTMLPath "($($CleanName.Path))_$($CleanName.DN).html" )
        Export-HTMLFile -arrWFs $arrThisObjWorkflowsHTML -Path $ObjExportFilePath -Title $obj.Name -Format $HTMLFormat
      }Catch {
        Write-Error "Problem exporting to HTML [$ObjExportFilePath]. $($Error[0].Message)"
      }
    }
    $i++
  }
  #endregion ForEach Obj

  Write-Progress -Activity "** What's happening? **" -status "Completed" -Completed
  $arrAllWorkflowsCSV  = ($arrAllWorkflowsCSV | Sort-Object -Property $SortBy -Descending:$SortOrder)
  $arrAllWorkflowsHTML  = ($arrAllWorkflowsHTML | Sort-Object -Property $SortBy -Descending:$SortOrder)
  $CleanNames = ''
  # Output master files if appropriate
  If ($ExportCSVPath){
    Try {
      If ($MonitoringObject.Count -eq 1){
        $CleanName = Get-CleanNames -MonitoringObject $MonitoringObject[0]
        $ObjExportFilePath = (Join-Path $ExportCSVPath "($($CleanName.Path))_$($CleanName.DN).csv"  )
      }
      Else {
        $ObjExportFilePath = (Join-Path $ExportCSVPath "MERGED.CSV" )
      }
      Export-CSVFile -arrWFs $arrAllWorkflowsCSV -Path $ObjExportFilePath -NoTypeInformation -Force
    }Catch {
      Write-Error "Problem exporting to CSV [$ObjExportFilePath]. $($Error[0].Message)"
    }
  }
  
  If ($ExportHTMLPath) {
    Try {
      If ($MonitoringObject.Count -eq 1){
        $CleanName = Get-CleanNames -MonitoringObject $MonitoringObject[0]
        $ObjExportFilePath = (Join-Path $ExportHTMLPath "($($CleanName.Path))_$($CleanName.DN).html" )
      }
      Else {
        $ObjExportFilePath = (Join-Path $ExportHTMLPath "MERGED.HTML" )
      }
      Export-HTMLFile -arrWFs $arrAllWorkflowsHTML -Path $ObjExportFilePath -Title 'All Workflows' -Format $HTMLFormat
    }Catch {
      Write-Error "Problem exporting to HTML [$ObjExportFilePath]. $($Error[0].Message)"
    }
  }

  Write-Host "$($timer.Elapsed.TotalSeconds) seconds to complete workflow export for $($i) objects." -ForegroundColor Gray

  #---------------------------------------------------------------
} #End Function

<#
    .SYNOPSIS
    Will export Operations Manager event log events to CSV
 
    .EXAMPLE
    Export-SCOMEventsToCSV
 
    The above example will output the newest 1000 SCOM events (default 1000) to a .CSV at C:\<computername>_OpsManEvents.csv
 
    .EXAMPLE
    Export-SCOMEventsToCSV -Newest 1500 -Path c:\Temp\SCOMlog.csv
 
 
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Original Date: 2018.03.22
    History:
#>

Function Export-SCOMEventsToCSV {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
      SupportsShouldProcess=$true, 
      PositionalBinding=$false,
  ConfirmImpact='Medium')]
  Param (
    [int]$Newest = 1000,

    # Param1 help description
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=1,
    ParameterSetName='Parameter Set 1')]
    [Alias("Path")] 
    [string]$OutFileCSV = "C:\$($env:COMPUTERNAME)_OpsManEvents.csv"
  )
  Get-EventLog -LogName 'Operations Manager' -Newest $Newest | Export-Csv -Path $OutFileCSV -NoTypeInformation -Force
  Return (Get-Item $OutFileCSV)
}
#######################################################################

<#
    .SYNOPSIS
    This script will get all rule and monitor knowledge article content and output the information to separate files (Rules.html and Monitors.html) in the output folder path specified.
 
    .PARAMETER OutFolder
    The output folder path where output files will be created. Must be a container/directory, not a file.
 
    .PARAMETER ManagementPack
    A collection/array of one or more management pack objects.
 
    .PARAMETER MgmtServerFQDN
    Alias: ManagementServer
    Fully Qualified Domain Name of the SCOM management server.
 
    .PARAMETER NoKnowledgeExclude
    By default ALL workflows will be included in the output files. Enable this switch to exclude workflows which have no Knowledge Article content.
 
    .PARAMETER ExportCSV
    Export results to CSV files instead of default format (.html)
 
    .PARAMETER Topic
    This will customize the name of the output files. Useful when dumping multiple sets to the same output folder.
 
    .PARAMETER ShowResult
    Will open the Windows file Explorer to show output files.
 
    .EXAMPLE
    (Get-SCOMManagementPack -Name *.AD.*) | Export-SCOMKnowledge -OutFolder 'C:\MyReports' -Topic "AD_" -ShowResult
    In the example above the variable will be assigned a collection of all management pack objects with ".AD." in the name.
    That subset/collection of management pack objects will be passed into the script. The script will output workflow details for all Rules and Monitors contained in ALL management packs within that set.
    The output file names will be: "AD_Rules.html" and "AD_Monitors.html". Finally the script will open Windows File Explorer to the location of the output directory.
 
    .Example
    PS C:\> Export-SCOMKnowledge -OutFolder 'C:\Export' -ManagementPack (Get-SCOMManagementPack -Name "*ad.*") -Filter '201[0-6]'
    The command above will output all rules/monitors from management packs which contain 'ad.' in the Name and which contain '201x' in the Name or DisplayName of the workflow where 'x' represents any single digit 0-6 .
 
    .EXAMPLE
    PS C:\> Export-SCOMKnowledge -OutFolder "C:\Temp" -ManagementPack (Get-SCOMManagementPack -Name *SQL*) -Topic "SQL_Packs_"
    The command above will output workflow details for all Rules and Monitors contained in ALL management packs with "SQL" in the management pack Name.
    The output file names will be: "SQL_Packs_Rules.html" and "SQL_Packs_Monitors.html"
 
    .EXAMPLE
    PS C:\> Export-SCOMKnowledge -OutFolder "C:\MyReports" -ManagementServer "ms01.contoso.com" -NoKnowledgeExclude
    In the example above, the command will connect to management server: "ms01.contoso.com", will output workflow details for all Rules and Monitors (only if they contain a Knowledge Article) to two separate files in the specified folder. The output file names will be: "Rules.html" and "Monitors.html"
 
    .Example
    PS C:\> Export-SCOMKnowledge -OutFolder 'C:\Export' -ManagementPack (Get-SCOMManagementPack -Name "*ad.*") -Filter '(?=200[0-8])((?!Monitoring).)*$'
    The command above will output all rules/monitors from management packs which contain 'ad' in the Name and which contain '200x' in the Name or DisplayName of the workflow where 'x' is numbers 1-8 but excluding workflows that contain 'monitoring'.
 
    .Example
    #Export SCOM Knowledge for rules/monitors
    Get-SCOMManagementGroupConnection | Remove-SCOMManagementGroupConnection -Verbose
    New-SCOMManagementGroupConnection -ComputerName YOUR_SERVERNAME
    $SQLMPs = Get-SCOMManagementPack -Name *sql*
    $Outfolder = 'D:\SCOMReports'
 
    # Workflows to separate files
    ForEach ($MP in $SQLMPs){
    $rules = $MP.GetRules().Count
    $mons = $MP.GetMonitors().Count
    If (($rules + $mons) -gt 0){
    Export-SCOMKnowledge -OutFolder $Outfolder -ManagementPack $MP -Topic "$($MP.Name)_"
    }
    Else{
    Write-Host "No monitor/rule workflows found: " -NoNewline;
    Write-Host "$($MP.DisplayName)" -f Yellow
    }
    }
 
    #All MPs combined
    Export-SCOMKnowledge -OutFolder $Outfolder -ManagementPack $SQLMPs -Topic "ALL_SQL_"
 
 
    .LINK
    https://monitoringguys.com/
 
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
 
    History:
    2022.06.20: Used arraylist to improve speed when gathering rules/mons. Leveraged new Convert-MAML2HTML function for article conversion.
    2017/12/05: Added Topic parameter to customize output file names. Added ShowResult switch.
    2017/10/16: Added support for regex filtering on DisplayName or Name of monitors/rules.
    2016/05/03: Added option to export to CSV. Although embedded tables in the KnowledgeArticle don't convert very well.
    2016/04/26: Improved formatting of output files. Now includes embedded parameter tables and restored article href links.
    2016/04/26: Fixed script to accept ManagementPack pipeline input
 
#>

Function Export-SCOMKnowledge {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
      SupportsShouldProcess=$false,
      SupportsPaging = $true,
  PositionalBinding=$false)]

  Param(
    #1
    [Parameter(Mandatory=$true,
        ValueFromPipeline=$false,
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$OutFolder,

    #2
    [Parameter(Mandatory=$false,
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$true,
        ValueFromRemainingArguments=$false,
        Position=1,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [System.Object[]]$ManagementPack,

    #3
    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        Position=2,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [Alias("ManagementServer")]
    [string]$MgmtServerFQDN,

    #4
    [Parameter(Mandatory=$false,
        Position=3,
    ParameterSetName='Parameter Set 1')]
    [Switch]$NoKnowledgeExclude,

    #5
    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        Position=4,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$OpsDBServer,

    #6
    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        Position=5,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$OpsDBName,

    #7
    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        Position=6,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$Topic,

    #8
    [Parameter(Mandatory=$false,
        Position=7,
    ParameterSetName='Parameter Set 1')]
    [Alias("WorkflowFilter")] 
    [string[]]$Filter,

    #9
    [Parameter(Mandatory=$false,
        Position=8,
    ParameterSetName='Parameter Set 1')]
    [switch]$ExportCSV,

    #10
    [Parameter(Mandatory=$false,
        Position=9,
    ParameterSetName='Parameter Set 1')]
    [switch]$ShowResult=$false
  )

  #### UNCOMMENT FOR TESTING ####
  #$OutFolder = "c:\Test"

  #region Functions
  Begin {
    #------------------------------------------------------------------------------------
    Function ProcessArticle {
      Param(
        $article
      )
      If ($article -ne "None")    #some rules don't have any knowledge articles
      {
        #Retrieve and format article content
        $MamlText = $null
        $HtmlText = $null

        If ($null -ne $article.MamlContent)
        {
          $MamlText = $article.MamlContent
          #$articleContent = fnMamlToHtml($MamlText)
          $articleContent = ($MamlText | ForEach-Object { Convert-MAMLToHtml -XML $_ })
        }

        If ($null -ne $article.HtmlContent)
        {
          $HtmlText = $article.HtmlContent
          $articleContent = CleanHTML($HtmlText)
        }
      }

      If ($null -eq $articleContent)
      {
        $articleContent = "No resolutions were found for this alert."
      }

      Return $articleContent
    }

    #------------------------------------------------------------------------------------
    Function ProcessWorkflows {
      Param(
        $Workflows,
        [string]$WFType
      )
      $thisWFType = $WFType
      $myWFCollectionObj = @()
      [int]$row=1
      ForEach ($thisWF in $Workflows) {
        Write-Progress -Activity "Processing $WFType" -status "Getting Alert [$($row)]: $($thisWF.DisplayName) " -percentComplete ($row / $($Workflows.count) * 100)
        #$ErrorActionPreference = 'SilentlyContinue'
        $article = $NULL
        Try {
          $article = $thisWF.GetKnowledgeArticle($cultureInfo)
        } Catch {
          #Empty catch is fine here
        }

        If (-NOT $article){
          $error.Remove($Error[0])
          $articlecontent = "None"
          If ($NoKnowledgeExclude){ Continue; }
        }
        Else{
          $articleContent = ProcessArticle $article
        }

        If ($ExportCSV) {
          $WFName = $($thisWF.Name)
          $WFDisplayName = $($thisWF.DisplayName)
        }
        Else {
          $WFName = '<name>' + $($thisWF.Name) + '</name>'
          $WFDisplayName = '<displayname>' + $($thisWF.DisplayName) + '</displayname>'
        }
        $WFDescription = $thisWF.Description
        If ($WFDescription.Length -lt 1) {$WFDescription = "None"}
        #region Get_alert_name
        If ($WFType -like "Rule") {
          $thisWFType = "$($WFType): " + "$($thisWF.WriteActionCollection.Name)"
          # Proceed only if rule is an "alert" rule.
          If ($thisWF.WriteActionCollection.Name -Like "GenerateAlert"){
            $AlertMessageID = $($thisWF.WriteActionCollection.Configuration).Split('"',3)[1]
            $Query = @"
Select [LTValue]
FROM [OperationsManager].[dbo].[LocalizedText]
WHERE ElementName like '$($AlertMessageID)'
AND LTStringType = '1'
 
"@


            $AlertDisplayName = Invoke-CLSqlCmd -Query $Query -Server $OpsDBServer -Database $OpsDBName
            If ($ExportCSV) { $AlertName = $AlertDisplayName }
            Else { $AlertName = '<alertname>' + $AlertDisplayName + '</alertname>' }
          }
          Else {
            If ($ExportCSV) { $AlertName = 'N/A' }
            Else { $AlertName = '<noalertname>N/A</noalertname>' }
          }
        }
        Else {
          # Workflow is not a rule, therefore it is a monitor.
          $Query = @"
Select LocalizedText.LTValue
From [OperationsManager].[dbo].[LocalizedText]
INNER JOIN [OperationsManager].[dbo].[Monitor]
on LocalizedText.LTStringId=Monitor.AlertMessage
Where Monitor.MonitorID like '$($thisWF.ID)'
AND LocalizedText.LTStringType = '1'
 
"@

          $AlertDisplayName = Invoke-CLSqlCmd -Query $Query -Server $OpsDBServer -Database $OpsDBName
          # Not all monitors generate an alert.
          If ($AlertDisplayName -like "EMPTYRESULT") {
            If ($ExportCSV) { $AlertName = 'N/A' }
            Else { $AlertName = '<noalertname>N/A</noalertname>' }
          }
          Else {
            If ($ExportCSV) { $AlertName = $AlertDisplayName }
            Else { $AlertName = '<alertname>' + $AlertDisplayName + '</alertname>' }
          }
        } #endregion Get_alert_name


        $WFID = $thisWF.ID
        $WFMgmtPackID = $thisWF.GetManagementPack().Name
        # Build the custom object to represent the workflow, add properties/values.
        $myWFObj = New-Object -TypeName System.Management.Automation.PSObject
        $myWFObj | Add-Member -MemberType NoteProperty -Name "Row" -Value $Row
        $myWFObj | Add-Member -MemberType NoteProperty -Name "DisplayName" -Value $WFDisplayName
        $myWFObj | Add-Member -MemberType NoteProperty -Name "AlertName" -Value $AlertName
        $myWFObj | Add-Member -MemberType NoteProperty -Name "KnowledgeArticle" -Value $articleContent
        $myWFObj | Add-Member -MemberType NoteProperty -Name "Description" -Value $WFDescription
        $myWFObj | Add-Member -MemberType NoteProperty -Name "ManagementPackID" -Value $WFMgmtPackID
        $myWFObj | Add-Member -MemberType NoteProperty -Name "Name" -Value $WFName
        $myWFObj | Add-Member -MemberType NoteProperty -Name "WorkflowType" -Value $thisWFType
        $myWFObj | Add-Member -MemberType NoteProperty -Name "ID" -Value $WFID

        $myWFCollectionObj += $myWFObj
        $Row++
      }

      Return $myWFCollectionObj
    }

    #------------------------------------------------------------------------------------
    Function fnMamlToHTML{
      
      param
      (
        $MAMLText
      )
      $HTMLText = "";
      $HTMLText = $MAMLText -replace ('xmlns:maml="http://schemas.microsoft.com/maml/2004/10"');

      $HTMLText = $HTMLText -replace ("<maml:section>");
      $HTMLText = $HTMLText -replace ("<maml:section >");
      $HTMLText = $HTMLText -replace ("</maml:section>");
      $HTMLText = $HTMLText -replace ("<section >");

      #ui = underline. Not going to bother with this html conversion
      $HTMLText = $HTMLText -replace ("<maml:ui>", "");
      $HTMLText = $HTMLText -replace ("</maml:ui>", "");

      #Only convert the maml tables if not exporting to CSV
      IF ($ExportCSV) {
        $HTMLText = $HTMLText -replace ("<maml:para>", " ");
        $HTMLText = $HTMLText -replace ("<maml:para />", " ");
        $HTMLText = $HTMLText -replace ("</maml:para>", " ");
        $HTMLText = $HTMLText -replace ("<maml:title>", "");
        $HTMLText = $HTMLText -replace ("</maml:title>", "");
        $HTMLText = $HTMLText -replace ("<maml:list>", "");
        $HTMLText = $HTMLText -replace ("</maml:list>", "");
        $HTMLText = $HTMLText -replace ("<maml:listitem>", "");
        $HTMLText = $HTMLText -replace ("</maml:listitem>", "");
        $HTMLText = $HTMLText -replace ("<maml:table>", "");
        $HTMLText = $HTMLText -replace ("</maml:table>", "");
        $HTMLText = $HTMLText -replace ("<maml:row>", "");
        $HTMLText = $HTMLText -replace ("</maml:row>", "");
        $HTMLText = $HTMLText -replace ("<maml:entry>", "");
        $HTMLText = $HTMLText -replace ("</maml:entry>", "");
        $HTMLText = $HTMLText -replace ('<tr><td><p>Name</p></td><td><p>Description</p></td><td><p>Default Value</p></td></tr>', 'Name Description DefaultValue');
      }
      Else {
        $HTMLText = $HTMLText -replace ("maml:para", "p");
        $HTMLText = $HTMLText -replace ("<maml:table>", "<table>");
        $HTMLText = $HTMLText -replace ("</maml:table>", "</table>");
        $HTMLText = $HTMLText -replace ("<maml:row>", "<tr>");
        $HTMLText = $HTMLText -replace ("</maml:row>", "</tr>");
        $HTMLText = $HTMLText -replace ("<maml:entry>", "<td>");
        $HTMLText = $HTMLText -replace ("</maml:entry>", "</td>");
        $HTMLText = $HTMLText -replace ("<maml:title>", "<h3>");
        $HTMLText = $HTMLText -replace ("</maml:title>", "</h3>");
        $HTMLText = $HTMLText -replace ("<maml:list>", "<ul>");
        $HTMLText = $HTMLText -replace ("</maml:list>", "</ul>");
        $HTMLText = $HTMLText -replace ("<maml:listitem>", "<li>");
        $HTMLText = $HTMLText -replace ("</maml:listitem>", "</li>");
        $HTMLText = $HTMLText -replace ('<tr><td><p>Name</p></td><td><p>Description</p></td><td><p>Default Value</p></td></tr>', '<th>Name</th><th>Description</th><th>Default Value</th>');
      }
      # Replace all maml links with html href formatted links
      while ($HTMLText -like "*<maml:navigationLink>*"){
        If ($HtmlText -like "*uri condition*" ) {
          $HTMLText = Fix-HREF -mystring $HTMLText -irregular
        }
        Else {
          $HTMLText = Fix-HREF -mystring $HTMLText
        }
      }
      Return $HTMLText;
    }
    #------------------------------------------------------------------------------------
    Function CleanHTML{
      
      param
      (
        $HTMLText
      )
      $TrimedText = "";
      $TrimedText = $HTMLText -replace ("&lt;", "<")
      $TrimedText = $TrimedText -replace ("&gt;", ">")
      $TrimedText = $TrimedText -replace ("&quot;", '"')
      $TrimedText = $TrimedText -replace ("&amp;", '&')
      $TrimedText;
    }

    #------------------------------------------------------------------------------------
    #Remove maml link formatting, replace with HTML
    Function Fix-HREF {
      Param(
        [string]$mystring,
        [switch]$irregular
      )
      If ($irregular){
        $href_link_tag_begin = '<maml:uri condition'
      }
      Else {
        $href_link_tag_begin = '<maml:uri href="'

        $href_name_tag_begin = '<maml:navigationLink><maml:linkText>'
        $href_name_tag_end = '</maml:linkText>'
        $href_link_tag_end = '" /></maml:navigationLink>'

        $href_name_length = ($mystring.IndexOf($href_name_tag_end)) - ( $mystring.IndexOf($href_name_tag_begin) + $href_name_tag_begin.Length)
        $href_name = $mystring.Substring( ($mystring.IndexOf($href_name_tag_begin) + $href_name_tag_begin.Length ), $href_name_length)

        $href_link_length = ($mystring.IndexOf($href_link_tag_end)) - ( $mystring.IndexOf($href_link_tag_begin) + $href_link_tag_begin.Length -1)
        $href_link = $mystring.Substring( ($mystring.IndexOf($href_link_tag_begin) + $href_link_tag_begin.Length ), $href_link_length -1)
      }
      $Chunk_Name = $mystring.Substring( $mystring.IndexOf($href_name_tag_begin), (($mystring.IndexOf($href_name_tag_end) + $href_name_tag_end.Length - 1) - $mystring.IndexOf($href_name_tag_begin) +1 ) )
      $Chunk_HREF = $mystring.Substring( $mystring.IndexOf($href_link_tag_begin), (($mystring.IndexOf($href_link_tag_end) + $href_link_tag_end.Length - 1) - $mystring.IndexOf($href_link_tag_begin) +1 ) )

      If ($irregular){
        $newstring = $mystring.Replace(("$Chunk_Name" + "$Chunk_HREF"), '' )
      }
      Else {
        #Example: <a href="http://www.bing.com">here</a> to go to Bing.
        $newstring = $mystring.Replace(("$Chunk_Name" + "$Chunk_HREF"), ('<a href="' + $href_link + '">' + "$href_name" + '</a>') )
      }
      Return $newstring
    }

    #------------------------------------------------------------------------------------
    Function Invoke-CLSqlCmd {

      param(
        [string]$Server,
        [string]$Database,
        [string]$Query,
        [int]$QueryTimeout = 30, #The time in seconds to wait for the command to execute. The default is 30 seconds.
        [int]$ConnectionTimeout = 15  #The time (in seconds) to wait for a connection to open. The default value is 15 seconds.
      )
      BEGIN {
      }
      PROCESS {

        try {
          $sqlConnection = New-Object System.Data.SqlClient.SqlConnection
          $sqlConnection.ConnectionString = "Server=$Server;Database=$Database;Trusted_Connection=True;Connection Timeout=$ConnectionTimeout;"
          $sqlConnection.Open()
          try {
            $sqlCmd = New-Object System.Data.SqlClient.SqlCommand
            $sqlCmd.CommandText = $Query
            $sqlCmd.CommandTimeout = $QueryTimeout
            $sqlCmd.Connection = $SqlConnection
            try {
              $Value = @()
              $sqlReader = $sqlCmd.ExecuteReader()
              #The default position of the SqlDataReader is before the first record.
              #Therefore, you must call Read to begin accessing any data.
              #Return Value: true if there are more rows; otherwise false.
              If ($sqlReader.Read()) { # $NUL #function returns true. pipe to nowhere.
                $Value = ($sqlReader.GetValue(0), $sqlReader.GetName(0))
                If ( ($Value[1].ToString().Length -eq 0) -and ($sqlReader.Fieldcount -eq 2) ) {
                  #Write-Host "NoName!" -foreground Yellow -background Red
                  try{
                    $ResultName = $sqlReader.GetValue(1)
                  } finally {
                  }
                  If ($ResultName) {
                    $Value[1]=$ResultName
                  }
                  Else{
                    $Value[1]='Name'
                  }
                }
                #Write-Host "Value2: $Value2"
              }
              Else{
                $Value = ("EMPTYRESULT",'Result')
              }
            }
            finally {
              $sqlReader.Close()
            }
          } finally {
            $sqlCmd.Dispose()
          }
        } finally {
          $sqlConnection.Close()
        }
      }
      END {
        Return $Value[0]
      }
    } #endregion Invoke-CLSqlCmd
    #------------------------------------------------------------------------------------


    #####################################################################################
    $ThisScript = $MyInvocation.MyCommand.Path
    [System.Collections.ArrayList]$Rules=@()
    [System.Collections.ArrayList]$Monitors=@()
    $InstallDirectory =  (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Microsoft Operations Manager\3.0\Setup" -Name "InstallDirectory").InstallDirectory
    $PoshModulePath = Join-Path (Split-Path $InstallDirectory) PowerShell
    $env:PSModulePath += (";" + "$PoshModulePath")

    # If Topic is provided, make sure it contains an underscore.
    If (($Topic.Length -ge 2) -and (($Topic.IndexOf('_')+1) -ne $Topic.Length) ) {
      $Topic = "$Topic"+"_"
    }

    #Get OpsDB Server name if not provided
    If (!($OpsDBServer)){
      $OpsDBServer = (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Microsoft Operations Manager\3.0\Setup" -Name "DatabaseServerName").DatabaseServerName
    }
    #Get OpsDB name if not provided
    If (!($OpsDBName)){
      $OpsDBName = (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Microsoft Operations Manager\3.0\Setup" -Name "DatabaseName").DatabaseName
    }
    # Add 2012 Functionality/cmdlets
    . Import-SCOMPowerShellModule

    If ($Error) {
      $modulepaths=$env:PSModulePath.split(';')
      $Error.Clear()
    }

    # If no mgmt server name has been set, assume that local host is the mgmt server. It's worth a shot.
    If ($MgmtServerFQDN -eq "") {
      #Get FQDN of local host executing the script (could be any mgmt server when using SCOM2012 Resource Pools)
      $objIPProperties = [System.Net.NetworkInformation.IPGlobalProperties]::GetIPGlobalProperties()
      If ($null -eq $objIPProperties.DomainName) {
        $MgmtServerFQDN = $objIPProperties.HostName
      }
      Else {
        $MgmtServerFQDN = $objIPProperties.HostName + "." +$objIPProperties.DomainName
      }
    }
    $Error.Clear()
    #Connect to Localhost Note: perhaps this can be done differently/cleaner/faster if connecting to self?
    Write-Host "Connecting to: $MgmtServerFQDN ..." -ForegroundColor Gray -BackgroundColor Black
    New-SCManagementGroupConnection -Computer $MgmtServerFQDN # | Out-Null
    If ($Error) {
      Write-Host "Failed to connect to: $MgmtServerFQDN ! Exiting. " -ForegroundColor Red -BackgroundColor Yellow
      Exit
    }
    Else {
      Write-Host "Connected to: $MgmtServerFQDN " -ForegroundColor Magenta -BackgroundColor Black
    }

    #Set Culture Info
    $cultureInfo = [System.Globalization.CultureInfo]'en-US'

    $cssHead = @"
<style>
displayname {
    color: blue;
    font: 16px Arial, sans-serif;
}
alertname {
    color: red;
    font: 16px Arial, sans-serif;
}
noalertname {
    color: grey;
    font: 16px Arial, sans-serif;
}
 
name {
    font: 10px Arial, sans-serif;
}
body {
    font: normal 14px Verdana, Arial, sans-serif;
}
 
table, th, td {
    border-collapse: collapse;
    border: 1px solid black;
}
 
th, td {
    padding: 10px;
    text-align: left;
}
tr:hover {background-color: #ccffcc}
 
th {
    background-color: #4CAF50;
    color: white;
}
 
 
</style>
 
"@


    # Make sure output folder exists
    If (-not (Test-Path -Path $OutFolder -PathType Container )) {
      New-Item -Path $OutFolder -ItemType Directory -Force -ErrorAction Stop
    }

    # If output files already exist, remove them
    "Rules.html","Monitors.html" | ForEach-Object {
      If (Test-Path -PathType Leaf (Join-Path $OutFolder $_) ) {
        Remove-Item -Path (Join-Path $OutFolder $_) -Force
      }
    }

  } #endregion Begin

  #region
  Process {

    # If a set of MPs is specified, only process workflows contained in the MPs.
    If ($ManagementPack){

      $ManagementPack | Select-Object DisplayName,Name,ID,Version | Format-Table -AutoSize

      # If filter keyword(s) exist, then filter the results according to the regex patterns submitted in the parameter value.
      If ($Filter) {
        Foreach ($ThisRegex in $Filter) {
          Write-Host "Getting all filtered Rules..." -ForegroundColor Yellow
          ( (Get-SCOMRule -ManagementPack $ManagementPack | Where-Object {( $_.Name -match "$ThisRegex") -OR ( $_.DisplayName -match "$ThisRegex")} ) | ForEach-Object {$NULL = $Rules.Add($_) })
          Write-Host "Getting all filtered Monitors..." -ForegroundColor Yellow
          ( (Get-SCOMMonitor -ManagementPack $ManagementPack | Where-Object {( $_.Name -match "$ThisRegex") -OR ( $_.DisplayName -match "$ThisRegex")} )  | ForEach-Object {$NULL = $Monitors.Add($_) })
        }
      }
      # If no filter(s) exist, then return all rules/mons from the designated MP.
      Else {
        Write-Host "Getting ALL Rules..." -ForegroundColor Yellow
        ( (Get-SCOMRule -ManagementPack $ManagementPack ) | ForEach-Object {$NULL = $Rules.Add($_) })
        Write-Host "Getting ALL Monitors..." -ForegroundColor Yellow
        ( (Get-SCOMMonitor -ManagementPack $ManagementPack )  | ForEach-Object {$NULL = $Monitors.Add($_) })
      }
    }
    # Else, get ALL workflows in ALL MPs
    Else {
      # If filter(s) exist, then filter the results according to the regex patterns submitted in the parameter value.
      If ($Filter) {
        Foreach ($ThisRegex in $Filter) {
          Write-Host "Getting all filtered Rules..." -ForegroundColor Yellow
          ( (Get-SCOMRule | Where-Object {( $_.Name -match "$ThisRegex") -OR ( $_.DisplayName -match "$ThisRegex")}) | ForEach-Object {$NULL = $Rules.Add($_) })
          Write-Host "Getting all filtered Monitors..." -ForegroundColor Yellow
          ( (Get-SCOMMonitor | Where-Object {( $_.Name -match "$ThisRegex") -OR ( $_.DisplayName -match "$ThisRegex")} ) | ForEach-Object {$NULL = $Monitors.Add($_) })
        }
      }
      Else{
        Write-Host "Getting ALL Rules..." -ForegroundColor Yellow
        ( (Get-SCOMRule) | ForEach-Object {$NULL = $Rules.Add($_) })
        Write-Host "Getting ALL Monitors..." -ForegroundColor Yellow
        ( (Get-SCOMMonitor) | ForEach-Object {$NULL = $Monitors.Add($_) })
      }
    }

  } #endregion

  #region
  End {

    Write-Host "`nTotal Rules Found: $($Rules.Count)" -BackgroundColor Black -ForegroundColor Green
    $myRulesObj = ProcessWorkflows -Workflows $Rules -WFType "Rule"
    If (($ExportCSV)) {
      $myRulesObjTemp = $myRulesObj | ConvertTo-Csv
      $myRulesObj = CleanHTML $myRulesObjTemp | ConvertFrom-CSV
      Write-Host "Exporting rules to CSV: "$(Join-Path $OutFolder ($Topic +"Rules.csv")) -F Cyan
      #Export to CSV
      $myRulesObj | Export-Csv -Path $(Join-Path $OutFolder ($Topic +"Rules.csv")) -NoTypeInformation -Encoding UTF8
    }
    Else {
      $RulesTempContent = $myRulesObj | ConvertTo-HTML -Title "SCOM Rules" -Head $cssHead
      $RulesTempContent =  CleanHTML $RulesTempContent
      $RulesTempContent = $RulesTempContent -Replace '<table>', '<table border="1" cellpadding="20">'
      $RulesTempContent | Set-Content (Join-Path $OutFolder ($Topic +"Rules.html")) -Encoding UTF8
    }
    Write-Host "Total Monitors Found: $($Monitors.Count)" -BackgroundColor Black -ForegroundColor Green
    $myMonitorObj = ProcessWorkflows -Workflows $Monitors -WFType "Monitor"
    If (($ExportCSV)) {
      $myMonitorObjTemp = $myMonitorObj | ConvertTo-CSV
      $myMonitorObj = CleanHTML $myMonitorObjTemp | ConvertFrom-CSV
      Write-Host "Exporting monitors to CSV: "$(Join-Path $OutFolder ($Topic + "Monitors.csv")) -F Cyan
      #Export to CSV
      $myMonitorObj | Export-Csv -Path $(Join-Path $OutFolder ($Topic + "Monitors.csv")) -NoTypeInformation -Encoding UTF8
    }
    Else {
      $MonitorTempContent = $myMonitorObj | ConvertTo-HTML  -Title "SCOM Monitors" -Head $cssHead
      $MonitorTempContent =  CleanHTML $MonitorTempContent
      $MonitorTempContent = $MonitorTempContent -Replace '<table>', '<table border="1" cellpadding="20">'
      $MonitorTempContent | Set-Content (Join-Path $OutFolder ($Topic + "Monitors.html")) -Encoding UTF8
    }
    Write-host "Output folder: $OutFolder" -BackgroundColor Black -ForegroundColor Yellow

    If ($ShowResult) {
      Explorer.exe $OutFolder
    }
  } #endregion
}
#######################################################################


<#
    .Synopsis
    This script will export all SCOM overrides to various file formats.
 
    .DESCRIPTION
    Will export a generous amount of information about overrides. Output may be filtered by management pack and exported to html, csv, xml, and json file formats.
 
 
    .PARAMETER OutDir
    Directory where output files will be created. Folder will get created if it doesn't already exist.
 
    .PARAMETER ExportFormats
    File format. TypeInformation is excluded. "CSV","JSON","HTML","XML"
 
    .PARAMETER CSS
    Any custom CSS for the HTML report. This will be inserted into the <Head> tag of the document. See examples.
 
    .PARAMETER FilterMPNames
    This can be a comma-separated string of management pack names or it can be an array of single names or a combination of both. See examples. The report(s) will only include overrides that exist in these MPs.
 
    .PARAMETER SortBy
    Sort rows by column name.
 
    .PARAMETER SortOrder
    Ascending or descending based on SortBy parameter.
 
    .PARAMETER DateStampFile
    By default the output files will include a datestamp in the name. Specify $false to produce a consistent filename. See examples.
 
 
    .PARAMETER WriteToEventLog
    Will write debugging info to Opsman event log.
 
    .EXAMPLE
    Export-SCOMOverrides -OutDir C:\Reports -ExportFormats HTML -FilterMPNames 'URLGenie.OVERRIDES,OpsMgr.Self.Maintenance.Overrides','PortGenie.OVERRIDES','WindowsService.OVERRIDES' -Verbose
 
    This example demonstrates that you can supply a comma-separated list of MP Names or an array of names or both.
 
    .EXAMPLE
    Export-SCOMOverrides -SortBy ORMPName -SortOrder Ascending -DateStampFile:$false -Verbose
 
 
    .EXAMPLE
    #Example
    PS C:\> [string]$CSS = @'
    <style>
    table, th, td {
    border: 1px solid black;
    border-collapse: collapse;
    }
 
    th, td {
    padding: 5px;
    }
 
    td {
    text-align: center;
    }
    </style>
    '@
     
    PS C:\> Export-SCOMOverrides -SortBy ORMPName -ExportFormats HTML -CSS $CSS -OutDir "C:\Reports"
 
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/2019/11/12/scomhelper/
    History:
    2020.05.13 - First version
     
    .INPUTS
    Inputs to this cmdlet (if any)
 
    .OUTPUTS
    Will output to std output or up to 4 different output file formats.
 
#>

Function Export-SCOMOverrides { 

  [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
      SupportsShouldProcess=$true, 
      PositionalBinding=$false,
      HelpUri = 'https://monitoringguys.com/',
  ConfirmImpact='Medium')]

  Param (

    # Folder for exported files (HTML and CSV)
    [Parameter(Mandatory=$true, ParameterSetName='FileExport',
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=0)]
    [string]$OutDir,



    # Export Formats: <html | csv | xml | json | all>
    [Parameter(Mandatory=$false, ParameterSetName='FileExport',
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=1)]
    [ValidateSet(
        "All",
        "CSV",
        "JSON",
        "HTML",
    "XML")]
    $ExportFormats ,


    # Optional CSS for HTML file export. Will appear in <Head> section of output file.
    [Parameter(Mandatory=$false, ParameterSetName='FileExport',
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=2)]
    [string]$CSS = @'
<style>
table, th, td {
  border: 1px solid black;
  border-collapse: collapse;
}
 
th, td {
  padding: 3px;
}
 
td {
  text-align: left;
}
</style>
'@

    ,

    # Comma-separated list of management pack Names or array of names (or both) for which to include overrides from only THESE MPs. Otherwise ALL ORs from ALL MPs will be exported.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=3)]
    [string[]]$FilterMPNames = '',


    #Report rows will be sorted by this Workflow property
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=4)]
    [ValidateSet("ContextDN",
        "ContextInstanceDN",
        "ContextIsGroup",
        "Enforced",
        "ORMPName",
        "ORName",
        "Parameter",
        "Property",
        "Value",
        "Workflow",
        "WorkflowDN",
        "WorkflowMPName",
    "WorkflowType")]
    [string]$SortBy = "WorkflowDN",

    #Formatting of HTML export
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=5)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [ValidateSet("Ascending", "Descending")]
    [string]$SortOrder = 'Ascending',


    # Will export files with unique datestamp name. <true|false>
    [Parameter(Mandatory=$false, ParameterSetName='FileExport',
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=6)]
    [switch]$DateStampFile = $true,


    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
    Position=7)]
    [switch]$WriteToEventLog
  )


  [int]$maxLogLength = 31000 #max chars to allow for event log messages
  [string]$ScriptName = 'Export-Overrides.ps1'
  $ScriptNameInstanceGUID = (New-Guid).Guid.Substring(((New-Guid).Guid.Length ) -6).ToUpper()
  [string]$whoami = whoami.exe

  ########################################################
  Function LogIt {
    Param(
      [switch]$Display,
      [int]$EventID,
      [int]$Line,
      $maxLogLength=31000,
      [string]$msg = '<No message provided>',
      [bool]$Proceed,
      [int]$Type = 2
    ) 

    $output = @"
 
 
Message: $msg
 
ThisScriptInstanceGUID: $ScriptNameInstanceGUID
ScriptLine: $Line
Running As: $whoami
This script: $ScriptName
 
 
Any Errors: $Error
 
"@


    If ($Proceed) 
    {
      $oEvent = New-Object -ComObject 'MOM.ScriptAPI'
      If ($output.Length -gt $maxLogLength){
        $output = ($output.Substring(0,([math]::Min($output.Length,$maxLogLength) )) + '...TRUNCATED...')
      }
      $oEvent.LogScriptEvent($ScriptName,$EventID,$Type,$output )
      #Display output, appropriate for an agent task.
    
    }
    If ($Display ) {
      Write-Verbose $msg
    }
  }
  ###############################################################################
  Function Load-Cache {

    LogIt -msg "Building cache..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display

    # Cache all classes
    LogIt -msg "Getting all Classes..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $AllClasses = Get-SCOMClass
    LogIt -msg "$($AllClasses.Count) found." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $hashAllClasses = @{}
    $hashAllClassesID = @{}
    ForEach ($Class in $AllClasses) {
      $hashAllClasses.Add($Class.Name, $Class)
      $hashAllClassesID.Add($Class.ID.GUID,$Class)
    }

    # Cache all monitors
    LogIt -msg "Getting all Monitors..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $AllMonitors = Get-SCOMMonitor
    LogIt -msg "$($AllMonitors.Count) Monitors found." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $hashAllMonitors = @{}
    ForEach ($Mon in $AllMonitors) {
      $hashAllMonitors.Add($Mon.Id.Guid, $Mon)
    }

    # Cache all rules
    LogIt -msg "Getting all Rules..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $AllRules = Get-SCOMRule
    LogIt -msg "$($AllRules.Count) Rules found." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $hashAllRules = @{}

    ForEach ($Rule in $AllRules) {
      $hashAllRules.Add($Rule.Id.Guid, $Rule)
    }

    # Cache all discoveries
    LogIt -msg "Getting all Discoveries..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $AllDiscoveries = Get-SCOMDiscovery
    LogIt -msg "$($AllDiscoveries.Count) Discoveries found." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $hashAllDiscoveries = @{}
    ForEach ($Disc in $AllDiscoveries) {
      $hashAllDiscoveries.Add($Disc.Id.Guid, $Disc)
    }

    # Cache all groups
    LogIt -msg "Getting all Groups..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $AllGroups = Get-SCOMGroup
    LogIt -msg "$($AllGroups.Count) Groups found." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    $hashAllGroups = @{}
    ForEach ($Group in $AllGroups) {
      $hashAllGroups.Add($Group.FullName, $Group)
    }
  }

  ########################################################
  Function Load-MPs {
    Param(
      [ValidateSet("Sealed","Unsealed","All")]
      [string]$type="All"
    )
  
    $stopwatch = [System.Diagnostics.Stopwatch]::StartNew()
    If ($Reload -eq $true ){ $type = 'All'}
    switch ($type)
    {
      'All' {
        LogIt -msg "Getting ALL MPs. This may take a minute..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        $AllMPs = Get-SCOMManagementPack | Where-Object {$_.Name -notin $ExcludedMPs.Values}
        $hashAllMPs = @{}
        $AllMPs | ForEach-Object {
          $hashAllMPs.Add($_.Name,$_)
        }
        $tmpmsg = "$($AllMPs.Count ) MPs found..."
      }

      {$_ -in 'All','Sealed' } {
        LogIt -msg "Getting Sealed MPs. This may take a minute..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        $SealedMPs = $AllMPs | Where-Object -FilterScript {$_.Sealed -eq $True } | Where-Object {$_.Name -notin $ExcludedMPs.Values}
        $hashSealedMPs = @{}
        $SealedMPs | ForEach-Object {
          $hashSealedMPs.Add($_.Name,$_)
        }
        $tmpmsg += " $($SealedMPs.Count ) Sealed MPs found..."
      }

      {$_ -in 'All','Unsealed' } {
        LogIt -msg "Getting unsealed MPs. This may take a minute..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        $UnsealedMPs = $AllMPs | Where-Object -FilterScript {$_.Sealed -eq $False } | Where-Object {$_.Name -notin $ExcludedMPs.Values}
        $hashUnsealedMPs = @{}
        $UnsealedMPs | ForEach-Object {
          $hashUnsealedMPs.Add($_.Name,$_)
        }      
        $tmpmsg += " $($UnsealedMPs.Count ) unsealed MPs found..."
      }

      Default {Write-Host "Default switch. Problem." -F Yellow}
    }

    [double]$elapsed = $stopwatch.Elapsed.TotalSeconds
    $stopwatch.Stop()
    LogIt -msg "$tmpmsg Operation took $($elapsed) seconds." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display

  }
  ########################################################
  Function Select-Overrides {
    Param(
      [Parameter( Mandatory=$false, 
      ParameterSetName='1')]
      [System.Object[]]$MPs,
    
      [System.Object[]]$Override,
      [Switch]$silent
    )
  
    [System.Collections.ArrayList]$arrayOR = @()

    If ([bool]$MPs) {
      LogIt -msg "`nGetting overrides from $($MPs.Count) unsealed MP(s). This may take a minute..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
      # Return only the overrides from the previously selected MPs
      ForEach ($MP in $MPs) {
        Try{
          $arrayOR += ( ($AllMPs | Where-Object { $_.Name -eq $MP.Name}).GetOverrides() ) 
        }Catch{
          Write-Host "No valid overrides found in MP: $($MP.Name)"
        }
      }
    }
    Else {
      LogIt -msg "`nGetting overrides from $($AllMPs.Count) Management Packs. This may take a minute..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
      $arrayOR = $AllMPs | ForEach-Object {$_.GetOverrides()}
    }

    If ($arrayOR.Count -eq 0){
      LogIt -msg "No valid overrides found in MP(s). To be valid for selection, ORs must not reference locally defined targets or workflows (in the same unsealed pack)." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    }
  
    # Just set variable here. Function should be dot-sourced when called.
    $ORs = $arrayOR
  }
  ########################################################
  Function Format-OR {
    Param(
      $ORs
    )
    [System.Collections.ArrayList]$arrObject = @()
    [int]$i=0

    LogIt -msg "Begin formatting ORs..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    # 'MonitorPropertyOverride', 'RulePropertyOverride', 'DiscoveryPropertyOverride'
    # 'MonitorConfigurationOverride', 'RuleConfigurationOverride', 'DiscoveryConfigurationOverride'
    # I'm going to have to do this the old-fashioned way
    ForEach ($item in $ORs) {
      $i++
      #Write-Progress -Activity "Formatting Data" -status "$i of ($ORs.Count)" -percentComplete ($i / ($ORs.Count*100))
      $Object = New-Object PSObject

      # Will address Diagnostic and Recovery overrides at a later time. Maybe.
      If ($item.XMLTag -match 'Diagnostic|Recovery' ) { Continue; }

      $Object | add-member Noteproperty ORName ([string]$item.Name)
      $Object | add-member Noteproperty ORMPName ([string]$item.Identifier.Domain[0])
    
      # is Monitor
      If ([bool]$item.Monitor.Identifier.Path.Count) {
        $Object | add-member Noteproperty WorkflowMPName ($item.Monitor.Identifier.Domain[0].ToString())
        $Object | add-member Noteproperty WorkflowType "Monitor"
        $Object | add-member Noteproperty Workflow ($item.Monitor.Identifier.Path[0].ToString())
        #$Object | add-member Noteproperty WorkflowDN ((Get-SCOMMonitor -Id $item.Monitor.Id.Guid).DisplayName )
        $Object | add-member Noteproperty WorkflowDN (($hashAllMonitors[$item.Monitor.Id.Guid]).DisplayName )
        $Object | add-member Noteproperty WorkflowId ([string]$item.Monitor.Id.Guid)
      }
      # is Rule
      ElseIf (($item.Rule.Identifier.Path.Count)) { 
        $Object | add-member Noteproperty WorkflowMPName ($item.Rule.Identifier.Domain[0].ToString())
        $Object | add-member Noteproperty WorkflowType "Rule"
        $Object | add-member Noteproperty Workflow ($item.Rule.Identifier.Path[0].ToString() )
        #$Object | add-member Noteproperty WorkflowDN ((Get-SCOMRule -Id $item.Rule.Id.Guid).DisplayName )
        $Object | add-member Noteproperty WorkflowDN (($hashAllRules[$item.Rule.Id.Guid]).DisplayName )
        $Object | add-member Noteproperty WorkflowId ([string]$item.Rule.Id.Guid)
      } 
      # is Discovery
      ElseIf ([bool]($item.Discovery.Identifier.Path.Count)) { 
        $Object | add-member Noteproperty WorkflowMPName ($item.Discovery.Identifier.Domain[0].ToString())
        $Object | add-member Noteproperty WorkflowType "Discovery"
        $Object | add-member Noteproperty Workflow ($item.Discovery.Identifier.Path[0].ToString()) 
        #$Object | add-member Noteproperty WorkflowDN ((Get-SCOMDiscovery -Id $item.Discovery.Id.Guid).DisplayName )
        $Object | add-member Noteproperty WorkflowDN (($hashAllDiscoveries[$item.Discovery.Id.Guid]).DisplayName )
      
        $Object | add-member Noteproperty WorkflowId ([string]$item.Discovery.Id.Guid)
      } 
      Else {
        $Object | add-member Noteproperty Workflow "ERROR"
        $Object | add-member Noteproperty WorkflowDN "ERROR"
        $Object | add-member Noteproperty WorkflowType "ERROR"
      }

      
      # To be added at some point
      # Diagnostic Task
      # Recovery Task
      # Note: $item.Target.Identifier.Path[0]
      

      If ([bool]($item.Property.Count)) { 
        $Object | add-member Noteproperty Property ($item.Property.ToString() )
        $Object | add-member Noteproperty Parameter ""
      }
      Else { 
        $Object | add-member Noteproperty Property ""
        $Object | add-member Noteproperty Parameter $item.Parameter 
      }
        
      $Object | add-member Noteproperty Value $item.Value

      If ([bool]($hashAllClassesID[$item.Context.Id.Guid])){
        $Object | add-member Noteproperty Context ( $hashAllClassesID[$item.Context.Id.Guid].Name )
        $Object | add-member Noteproperty ContextID ($item.Context.Id.Guid.ToString())
        $Object | add-member Noteproperty ContextDN ( $hashAllClassesID[$item.Context.Id.Guid].DisplayName ) 
      
        If ( $hashAllGroups.ContainsKey($hashAllClassesID[$item.Context.Id.Guid].Name) ) {
          $Object | add-member Noteproperty ContextIsGroup 'True'
        }
        Else {
          $Object | add-member Noteproperty ContextIsGroup 'False'
        }
      }
      #If the Context/Target does not exist, then add the ORName to a dirty list
      Else { 
        Try{
          # Need to figure out what to do with this collection of duds.
          $StaleORNames.Add($item.Name,'Stale')
        } 
        Catch { #typically it's a bad idea to have an empty Catch but it's fine in this situation
        }
        $Object | add-member Noteproperty Context "DOES NOT EXIST - STALE REFERENCE"
        $Object | add-member Noteproperty ContextID ""
        $Object | add-member Noteproperty ContextDN ""
      }

      If ([Bool]($item.ContextInstance.Count)) { 
        $Object | add-member Noteproperty ContextInstanceID ($item.ContextInstance.Guid.ToString() )
        If ([bool]($DN = (Get-SCOMClassInstance -Id $item.ContextInstance.Guid).DisplayName) ) {
          # This might prove to be very time consuming
          $Object | add-member Noteproperty ContextInstanceDN $DN
        } Else{
          $Object | add-member Noteproperty ContextInstanceDN ""
        }

      } 
      Else {
        $Object | add-member Noteproperty ContextInstanceID ""
        $Object | add-member Noteproperty ContextInstanceDN ""
      }

      $Object | add-member Noteproperty Enforced $item.Enforced
      $Object | add-member Noteproperty OR_MPVersion $item.Identifier.Version.ToString()
      $arrObject.Add($Object) | Out-Null
    }
    If ($Object) {
      Remove-Variable -Name Object -ErrorAction SilentlyContinue
    }
    LogIt -msg "Done formatting ORs." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
  
    # Just set variable here. Function should be dot-sourced when called.
    $ORs_F = $arrObject
  }
  ########################################################

  Function Verify-OutDir {
    Param (
      [string]$OutDir
    )
  
    If (-NOT (Test-Path -Path $OutDir -PathType Container )) { 
      Try {
        New-Item -Path $OutDir -ItemType Directory -Force -ErrorAction Stop
        Return $true
      }Catch{
        Write-Error "Unable to verify output directory."
        Return $false
      }
    }
    Else {
      Return (Test-Path -Path $OutDir -PathType Container )
    }
  }

  ########################################################
  Function __LINE__ { 
    $MyInvocation.ScriptLineNumber 
  }
  ########################################################


  #====================================================================================
  #====================================== MAIN ======================================

  [int]$info=0
  [int]$critical=1
  [int]$warn=2

  # This will help control output sort order command
  Switch ($SortOrder ) {
    'Ascending' {
      [BOOL]$SortOrder = $false
    }
    'Descending' {
      [BOOL]$SortOrder = $true
    }
    Default {
      [BOOL]$SortOrder = $false
    }
  }

  # These are MPs that shouldn't be tampered with or consulted.
  $ExcludedMPs =  @{            
    1 = 'Microsoft.SystemCenter.SecureReferenceOverride'
    2 = 'Microsoft.SystemCenter.GlobalServiceMonitor.SecureReferenceOverride'
    3 = 'Microsoft.SystemCenter.Notifications.Internal'
    4 = 'Microsoft.SystemCenter.NetworkDiscovery.Internal'
    5 = 'Microsoft.SystemCenter.Advisor.SecureReferenceOverride'
  } 


  If ($DateStampFile){
    [string]$FileName = "overrides_$(Get-Date -F yyyyMMdd_HHmmss)"
  }
  Else {
    [string]$FileName = "overrides"
  }

  If (-NOT (Verify-OutDir -OutDir $OutDir)) {
    LogIt -msg "Unable to verify export directory [$($OutDir)]. Please run this task with permissions to create the directory or specify a valid path. Exiting." -Type $critical -EventID 9995 -Proceed $true -Line $(__LINE__) -Display
    Exit
  }

  # Functions are dot-sourced (which is perfectly fine) because output statements in the functions are meant to be seen in agent task.
  # Output statements in functions will contaminate function return values.
  . Load-Cache
  
  . Load-MPs -type 'All'

  [System.Collections.ArrayList]$FilteredMPs = @()
  ForEach ($Item in $FilterMPNames) {
    ForEach ($Name in @($Item.Split(',').Split(';').Split(':')) )
    {
      $null = $FilteredMPs.Add($hashAllMPs[$Name])
    }
  }

  #. Load-MPs -type 'All'
  . Select-Overrides -MPs $FilteredMPs

  # Result will populate '$ORs_F'
  . Format-OR -ORs $ORs

 
  If ($ORs_F.Count) {
    $ORs_F = $ORs_F | Sort-Object -Property $SortBy -Descending:$SortOrder 
  }

  If (-NOT $ExportFormats){
    LogIt -msg "Returning overrides to std out." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
    Return $ORs_F
  }
  Else {
    Switch ($ExportFormats) {
      {$_ -match 'all|csv'} { 
        Try{
          $ORs_F | Export-Csv -Path (Join-Path $OutDir "$($FileName).csv") -NoTypeInformation -Force -Encoding UTF8
          LogIt -msg "Exporting overrides to: [$((Join-Path $OutDir "$($FileName).csv"))] ..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        } Catch {
          LogIt -msg "FAILED Exporting overrides to: [$((Join-Path $OutDir "$($FileName).csv"))] ..." -Type $warn -EventID 9995 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        }
      }

      {$_ -match 'all|html'} { 
        Try{
          $ORs_F | ConvertTo-Html -Head $css | Set-Content -Path (Join-Path $OutDir "$($FileName).html") -Encoding UTF8 -Force
          LogIt -msg "Exporting overrides to: [$((Join-Path $OutDir "$($FileName).html"))] ..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        } Catch {
          LogIt -msg "FAILED Exporting overrides to: [$((Join-Path $OutDir "$($FileName).html"))] ..." -Type $warn -EventID 9995 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        }
      }
  
      {$_ -match 'all|json'} { 
        Try{
          $ORs_F | ConvertTo-Json | Set-Content -Path (Join-Path $OutDir "$($FileName).json") -Force -Encoding UTF8
          LogIt -msg "Exporting overrides to: [$((Join-Path $OutDir "$($FileName).json"))] ..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        } Catch {
          LogIt -msg "FAILED Exporting overrides to: [$((Join-Path $OutDir "$($FileName).json"))] ..." -Type $warn -EventID 9995 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        }
      }

      {$_ -match 'all|xml'} { 
        Try{
          $ORs_F | Export-Clixml -Path (Join-Path $OutDir "$($FileName).xml") -Force -Encoding UTF8
          LogIt -msg "Exporting overrides to: [$((Join-Path $OutDir "$($FileName).xml"))] ..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        } Catch {
          LogIt -msg "FAILED Exporting overrides to: [$((Join-Path $OutDir "$($FileName).xml"))] ..." -Type $warn -EventID 9995 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
        }
      }
    
      default      { 
        LogIt -msg "FAILED Exporting overrides! Something is wrong with `$ExportFormats [$($ExportFormats)]" -Type $critical -EventID 9995 -Proceed $true -Line $(__LINE__) -Display
      }
    }#end switch
  }
  LogIt -msg "Script end ..." -Type $info -EventID 9990 -Proceed $WriteToEventLog -Line $(__LINE__) -Display
}


#######################################################################
<#
    .DESCRIPTION
    Super fast ping results for large quantities of hosts.
    .EXAMPLE
    Fast-Ping "myserver.domain.com","yourserver.domain.com"
    .EXAMPLE
    Get-Content C:\test\RandomHosts.txt | Select-Object -First 100 | Fast-Ping
    .NOTES
    Author: Tyson Paul
    Original Author: unknown
    Original script: https://community.idera.com/database-tools/powershell/powertips/b/tips/posts/final-super-fast-ping-command
    I modifed this a bit. I added the ability to determine the max number of hostnames to process without
    choking the Get-WmiObject cmdlet, then recursively probe the chunks of the input hosts array.
#>

Function Fast-Ping
{
  param
  (
    # make parameter pipeline-aware
    [Parameter(Mandatory,ValueFromPipeline)]
    
    # one or more computer names (array)
    [string[]]$ComputerName,
        
    $TimeoutMillisec = 4000
  )

  begin
  {
    [int]$maxParam = 16384
    # use this to collect computer names that were sent via pipeline
    [Collections.ArrayList]$bucket = @()
    
    # hash table with error code to text translation
    $StatusCode_ReturnValue = 
    @{
      0 = 'Success'
      11001 = 'Buffer Too Small'
      11002 = 'Destination Net Unreachable'
      11003 = 'Destination Host Unreachable'
      11004 = 'Destination Protocol Unreachable'
      11005 = 'Destination Port Unreachable'
      11006 = 'No Resources'
      11007 = 'Bad Option'
      11008 = 'Hardware Error'
      11009 = 'Packet Too Big'
      11010 = 'Request Timed Out'
      11011 = 'Bad Request'
      11012 = 'Bad Route'
      11013 = 'TimeToLive Expired Transit'
      11014 = 'TimeToLive Expired Reassembly'
      11015 = 'Parameter Problem'
      11016 = 'Source Quench'
      11017 = 'Option Too Big'
      11018 = 'Bad Destination'
      11032 = 'Negotiating IPSEC'
      11050 = 'General Failure'
    }
    
    
    # hash table with calculated property that translates
    # numeric return value into friendly text

    $statusFriendlyText = @{
      Name       = 'Status'
      Expression = { 
        # take status code and use it as index into
        # the hash table with friendly names
        # make sure the key is of same data type (int)
        $StatusCode_ReturnValue[([int]$_.StatusCode)]
      }
    }

    # calculated property that returns $true when status -eq 0
    $IsOnline = @{
      Name       = 'Online'
      Expression = {
        $_.StatusCode -eq 0 
      }
    }

    # do DNS resolution when system responds to ping
    $DNSName = @{
      Name       = 'DNSName'
      Expression = {
        if ($_.StatusCode -eq 0) 
        { 
          if ($_.Address -like '*.*.*.*') 
          {
            [Net.DNS]::GetHostByAddress($_.Address).HostName  
          } 
          else  
          {
            [Net.DNS]::GetHostByName($_.Address).HostName  
          } 
        }
      }
    }
  }
    
  process
  {
    # add each computer name to the bucket
    # we either receive a string array via parameter, or
    # the process block runs multiple times when computer
    # names are piped
    $ComputerName | ForEach-Object -Process {
      $name = $_ -replace ' ',''
      $null = $bucket.Add($name)
    }
  }
    
  end
  {
    # Recursively break down query strings that are too long.
    # Assemble the query with all addresses
    $query = $bucket -join "' or Address='"

        
    If ($query.length -gt $maxParam)
    {
      $tmpQuery = $query.Clone()
      $div = 1
      # If the query is too large, figure out what size chunk of the addresses will be under threshold/max size
      While ($tmpQuery.Length -gt $maxParam)
      {
        $div++
        $tmpQuery = $bucket[0..([math]::Floor(($bucket.Count / $div)))] -join "' or Address='"
      }
        
      # This size chunk of addresses should be small enough, acceptable as a parameter.
      $start = [math]::Floor(($bucket.Count / $div))
      Write-Verbose -Message "Starting size: $start"        
        
      $index = 0
      For($i = $start; $i -le ($bucket.Count -1); $i = ([math]::Min(($i + $start),($bucket.Count -1))) ) 
      {
        Write-Verbose -Message "Range:$($index)-$($i)" 
        $Result += Fast-Ping -ComputerName $bucket[($index)..$i] -TimeoutMillisec $TimeoutMillisec
        If ($i -eq ($bucket.Count -1)) 
        {
          break
        }
        $index = $i+1
      }
    }
        
    Else 
    {
      $Result += Get-WmiObject -Class Win32_PingStatus -Filter "(Address='$query') and timeout=$TimeoutMillisec" | Select-Object -Property Address, $DNSName, $IsOnline, $statusFriendlyText
    }

    Return $Result
  }
}

#######################################################################
<#
    .SYNOPSIS
    This script will get workflow details for one or more alerts, including the Knowledge Article content (neatly formatted in HTML).
 
    .DESCRIPTION
    Accepts one parameter, the alert object array of one or more SCOM alert objects: Microsoft.EnterpriseManagement.Monitoring.MonitoringAlert[]. This function can be useful with an automation/ticketing solution.
 
    .PARAMETER Alert
    The alert object(s) to be queried for the workflow details including the Knowledge article (if one exists).
 
    .PARAMETER MgmtServerFQDN
    Fully Qualified Domain Name of the SCOM management server.
    Alias: ManagementServer
 
    .PARAMETER OutputArticleOnly
    Will output only the Knowledge Article(s) content (neatly formatted in HTML). This may abe useful for situations where a service/ticketing connector is used to get alert properties.
 
    .EXAMPLE
    Get-SCOMAlert | Select-Object -First 1 | Get-SCOMAlertKnowledge
    Will display alert info (including Knowledge Article in HTML format) for the first alert object returned.
 
    .EXAMPLE
    PS C:\> Get-SCOMAlertKnowledge -Alert (Get-SCOMAlert | Select-Object -First 3) -OutputArticleOnly
    Will output only the HTML Knowledge Articles for the first 3 alert objects returned.
 
    .EXAMPLE
    PS C:\> Get-SCOMAlertKnowledge -Alert (Get-SCOMAlert -Name *sql*)
    Will output alert info (including Knowledge Article in HTML format) for any alerts with "sql" in the alert name.
 
    .NOTES
    Author: Tyson Paul
    Date: 2016/3/31
    Blog: https://monitoringguys.com/
#>

Function Get-SCOMAlertKnowledge {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
      SupportsShouldProcess=$true,
      SupportsPaging = $true,
  PositionalBinding=$true)]
  Param(

    [Parameter(Mandatory=$true,
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [Microsoft.EnterpriseManagement.Monitoring.MonitoringAlert[]]$Alert,

    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
        Position=1,
    ParameterSetName='Parameter Set 1')]
    [Alias("ManagementServer")]
    [string]$MgmtServerFQDN = "",

    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
        Position=2,
    ParameterSetName='Parameter Set 1')]
    [Switch]$OutputArticleOnly = $false

  )


  Begin {

    #------------------------------------------------------------------------------------
    Function ProcessArticle {
      Param(
        $article
      )
      If ($article -ne "None")    #some rules don't have any knowledge articles
      {
        #Retrieve and format article content
        $MamlText = $null
        $HtmlText = $null

        If ($null -ne $article.MamlContent)
        {
          $MamlText = $article.MamlContent
          $articleContent = fnMamlToHtml($MamlText)

        }

        If ($null -ne $article.HtmlContent)
        {
          $HtmlText = $article.HtmlContent
          $articleContent = fnTrimHTML($HtmlText)
        }
      }

      If ($null -eq $articleContent)
      {
        $articleContent = "No resolutions were found for this alert."
      }

      Return $articleContent
    }

    #------------------------------------------------------------------------------------
    Function ProcessWorkflow {
      Param(
        $Workflows,
        [string]$WFType
      )
      $myWFCollectionObj = @()

      ForEach ($thisWF in $Workflows) {
        $ErrorActionPreference = 'SilentlyContinue'
        $article = $thisWF.GetKnowledgeArticle($cultureInfo)
        If ($? -eq $false){
          $error.Remove($Error[0])
          $article = "None"
        }
        Else{
          $articleContent = ProcessArticle $article
        }

        $WFName = $thisWF.Name
        $WFDisplayName = $thisWF.DisplayName
        $WFDescription = $thisWF.Description
        $WFID = $thisWF.ID
        If ($WFDescription.Length -lt 1) {$WFDescription = "None"}

        #Note: Only a small subset of alert properties are gathered here. Add additional Note Properties as needed using the format below.
        $myWFObj = New-Object -TypeName System.Management.Automation.PSObject
        $myWFObj | Add-Member -MemberType NoteProperty -Name "WorkflowType" -Value $WFType
        $myWFObj | Add-Member -MemberType NoteProperty -Name "Name" -Value $WFName
        $myWFObj | Add-Member -MemberType NoteProperty -Name "DisplayName" -Value $WFDisplayName
        $myWFObj | Add-Member -MemberType NoteProperty -Name "Description" -Value $WFDescription
        $myWFObj | Add-Member -MemberType NoteProperty -Name "ID" -Value $WFID
        $myWFObj | Add-Member -MemberType NoteProperty -Name "KnowledgeArticle" -Value $articleContent
        $myWFCollectionObj += $myWFObj
      }

      Return $myWFCollectionObj
    }

    #------------------------------------------------------------------------------------
    Function fnMamlToHTML{
      
      param
      (
        $MAMLText
      )
      $HTMLText = "";
      $HTMLText = $MAMLText -replace ('xmlns:maml="http://schemas.microsoft.com/maml/2004/10"');
      $HTMLText = $HTMLText -replace ("maml:para", "p");
      $HTMLText = $HTMLText -replace ("maml:");
      $HTMLText = $HTMLText -replace ("</section>");
      $HTMLText = $HTMLText -replace ("<section>");
      $HTMLText = $HTMLText -replace ("<section >");
      $HTMLText = $HTMLText -replace ("<title>", "<h3>");
      $HTMLText = $HTMLText -replace ("</title>", "</h3>");
      $HTMLText = $HTMLText -replace ("<listitem>", "<li>");
      $HTMLText = $HTMLText -replace ("</listitem>", "</li>");
      $HTMLText;
    }
    #------------------------------------------------------------------------------------
    Function fnTrimHTML($HTMLText){
      $TrimedText = "";
      $TrimedText = $HTMLText -replace ("&lt;", "<")
      $TrimedText = $TrimedText -replace ("&gt;", ">")
      <# $TrimedText = $TrimedText -replace ("<html>")
          $TrimedText = $TrimedText -replace ("<HTML>")
          $TrimedText = $TrimedText -replace ("</html>")
          $TrimedText = $TrimedText -replace ("</HTML>")
          $TrimedText = $TrimedText -replace ("<body>")
          $TrimedText = $TrimedText -replace ("<BODY>")
          $TrimedText = $TrimedText -replace ("</body>")
          $TrimedText = $TrimedText -replace ("</BODY>")
          $TrimedText = $TrimedText -replace ("<h1>", "<h3>")
          $TrimedText = $TrimedText -replace ("</h1>", "</h3>")
          $TrimedText = $TrimedText -replace ("<h2>", "<h3>")
          $TrimedText = $TrimedText -replace ("</h2>", "</h3>")
          $TrimedText = $TrimedText -replace ("<H1>", "<h3>")
          $TrimedText = $TrimedText -replace ("</H1>", "</h3>")
          $TrimedText = $TrimedText -replace ("<H2>", "<h3>")
          $TrimedText = $TrimedText -replace ("</H2>", "</h3>")
      #>

      $TrimedText;
    }
    #------------------------------------------------------------------------------------

    ###########################################################################################


    $ThisScript = $MyInvocation.MyCommand.Path
    $InstallDirectory =  (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Microsoft Operations Manager\3.0\Setup" -Name "InstallDirectory").InstallDirectory
    $PoshModulePath = Join-Path (Split-Path $InstallDirectory) PowerShell
    $env:PSModulePath += (";" + "$PoshModulePath")

    # Add 2012 Functionality/cmdlets
    . Import-SCOMPowerShellModule

    If ($Error) {
      $modulepaths=$env:PSModulePath.split(';')
      $Error.Clear()
    }

    # If no mgmt server name has been set, assume that local host is the mgmt server. It's worth a shot.
    If ($MgmtServerFQDN -eq "") {
      #Get FQDN of local host executing the script (could be any mgmt server when using SCOM2012 Resource Pools)
      $objIPProperties = [System.Net.NetworkInformation.IPGlobalProperties]::GetIPGlobalProperties()
      If ($null -eq $objIPProperties.DomainName) {
        $MgmtServerFQDN = $objIPProperties.HostName
      }
      Else {
        $MgmtServerFQDN = $objIPProperties.HostName + "." +$objIPProperties.DomainName
      }
    }

    #Connect to Localhost Note: perhaps this can be done differently/cleaner/faster if connecting to self?
    New-SCManagementGroupConnection -Computer $MgmtServerFQDN | Out-Null
    # If ($Error) { LogIt -EventID 9995 -Type $warn -Message "Log any Errors..." ; $Error.Clear() }

    #Set Culture Info
    $cultureInfo = [System.Globalization.CultureInfo]'en-US'
    $objWFCollection = @()
  }

  Process{
    #Depending on how the alert object(s) are passed in, the ForEach may be needed. (parameter vs. piped)
    ForEach ($objAlert in $Alert)
    {
      $workflowID = $objAlert.MonitoringRuleId
      $bIsMonitorAlert = $objAlert.IsMonitorAlert
      If ($bIsMonitorAlert -eq $false) {
        $WFType = "Rule"
        $workflow = Get-SCOMRule -Id $workflowID
      }
      ElseIf ($bIsMonitorAlert -eq $true) {
        $WFType = "Monitor"
        $workflow = Get-SCOMMonitor -Id $workflowID
      }

      # The funciton being called is designed to accept one or more workflows.
      # It will return one or more custom objects with workfow details, including (most importantly) the KnowlegeArticle.
      $objWFCollection += ProcessWorkflow -Workflows $Workflow -WFType $WFType
    }
  } #End Process

  End{
    If ($OutputArticleOnly){
      Return $objWFCollection.KnowledgeArticle
    }
    Else{
      Return $objWFCollection
    }
  }
}#End Function
#######################################################################

<#
    .Synopsis
    Will display statistics about the total number of SCOM class instances and the management packs from which they originate.
    .EXAMPLE
    Get-SCOMClassInfo -Top 10
    .EXAMPLE
    Get-SCOMClassInfo -Top 30 -MPStats
    .EXAMPLE
    Get-SCOMClassInfo -MPStats -ShowGraph
    .INPUTS
    None
    .OUTPUTS
    Custom Object: {InstanceCount,ClassName,MPName}, {MPName,TotalInstances}
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Version: 1.0
    Original Date: 2018.05.25
    .FUNCTIONALITY
    Statistical reporting.
    .LINK
    Get-SCOMMPFileInfo
    New-SCOMClassGraph
    Unseal-SCOMMP
#>

Function Get-SCOMClassInfo {
  Param (
    #Will return top N results by InstanceCount
    [int]$Top=30,
    [switch]$MPStats,
    [switch]$ShowGraph
  )

  [System.Collections.ArrayList]$arrClasses = @()
  # Get all MPs that were not installed OotB.
  $MPs = Get-SCManagementPack
  $MPs = $MPs | Sort-Object TimeCreated | Select-Object -Last ($MPs.Count - 98)
  $Classes = Get-SCClass -ManagementPack $MPs | Where-Object {($_.Singleton -eq $false) -AND ($_.Abstract -eq $false) }
  Class ClassStats{
    [int]$Instances
    [string]$ClassName
    [string]$ClassDisplayName
    [string]$MPName    
  }
  
  ForEach ($Class in $Classes) {
    $Instances = $Class | Get-SCOMClassInstance
    [ClassStats]$obj = @{
      Instances = $Instances.Count
      ClassName = $Class.Name
      ClassDisplayName = $Class.DisplayName
      MPName = ($Class.Identifier.Domain[0])
    }
    $Null = $arrClasses.Add($obj)
  }
  If ($MPStats){
    $Results = (($arrClasses | Group-Object -Property MPName ) | Select-Object @{Name='MPName';E={$_.Name} },@{Name='TotalInstances';E={($_.Group.Instances | Measure-Object -Sum).Sum}} |Sort-Object TotalInstances -Descending | Select-Object -First $Top)
    If ([bool]$ShowGraph) {
      $Results | Out-ConsoleGraph -Property TotalInstances -Columns TotalInstances,MPName
    }
    Else {
      Return $Results
    }
  }
  Else{
    $Results = ($arrClasses | Sort-Object Instances -Descending | Select-Object -First $Top)
    If ([bool]$ShowGraph) {
      $Results | Out-ConsoleGraph -Property Instances -Columns Instances,ClassName
    }
    Else {
      Return $Results
    }
  }
} #End Function

#######################################################################

<#
    .Synopsis
 
    .DESCRIPTION
    Tim Culham of www.culham.net wrote an awesome lightweight Daily Health Check Script
    which can be found here:
    http://www.culham.net/powershell/scom-2012-scom-2012-r2-daily-check-powershell-script-html-report/
 
    I have been meaning to write something similiar for awhile so decided to take his wonderfully written
    script/structure and extend it by adding in a number of the more in depth Database Health Checks/KH Useful
    SQL Queries that I frequently find myself asking customers to run when they are experiencing performance
    issues with the Ops & DW DB's.
 
    I will cleanup my code in a later version and add additional functionality but for now Kudo's to Tim!
 
    MBullwin 11/3/2014
    www.OpsConfig.com
 
    [ https://gallery.technet.microsoft.com/SCOM-Health-Check-fd2272ec ]
    As with all scripts I post, this is provided AS-IS without warrenty so please test first and use at your own risk.
 
    What this version of the script will give you:(Some of these are just features which are carried over from the original, many are added)
 
    01. Version/Service Pack/Edition of SQL for each SCOM DB Server
 
    02. Disk Space Info for Ops DB, DW DB, and associated Temp DB's
 
    03. Database Backup Status for all DB's except Temp.
 
    04. Top 25 Largest Tables for Ops DB and DW DB
 
    05. Number of Events Generated Per Day (Ops DB)
 
    06. Top 10 Event Generating Computers (Ops DB)
 
    07. Top 25 Events by Publisher (Ops DB)
 
    08. Number of Perf Insertions Per Day (Ops DB)
 
    09. Top 25 Perf Insertions by Object/Counter Name (Ops DB)
 
    10. Top 25 Alerts by Alert Count
 
    11. Alerts with a Repeat Count higher than 200
 
    12. Stale State Change Data
 
    13. Top 25 Monitors Changing State in the last 7 Days
 
    14. Top 25 Monitors Changing State By Object
 
    15. Ops DB Grooming History
 
    16. Snapshot of DW Staging Tables
 
    17. DW Grooming Retention
 
    18. Management Server checks (Works well on prem, seems to have some issues with gateways due to remote calls-if you see some errors flash by have no fear though I wouldn't necessarily trust the results coming back from a Gateway server in the report depending on firewall settings)
 
    19. Daily KPI
 
    20. MP's Modified in the Last 24 hours
 
    21. Overrides in Default MP Check
 
    22. Unintialized Agents
 
    23. Agent Stats (Healthy, Warning, Critical, Unintialized, Total)
 
    24. Agent Pending Management Summary
 
    25. Alert Summary
 
    26. Servers in Maintenance Mode
 
    For more details on this version checkout: www.OpsConfig.com
    .NOTES
    Author: MBullwin
    Blog: www.OpsConfig.com
    Version: v1.1
    Date: 11/3/2014
    Original Script: https://gallery.technet.microsoft.com/SCOM-Health-Check-fd2272ec
 
    Modifications by Tyson:
    23020.08.11 - Added ManagementServer param. It's not fancy, but better than hardcoded localhost.
 
#>

Function Get-SCOMHealthCheckOpsConfig {

  Param (
    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        Position=2,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$ManagementServer='Localhost'
  )
  ###################################################################################################################
  #
  # Tim Culham of www.culham.net wrote an awesome lightweight Daily Health Check Script
  # which can be found here:
  #
  # http://www.culham.net/powershell/scom-2012-scom-2012-r2-daily-check-powershell-script-html-report/
  #
  # I have been meaning to write something similiar for awhile so decided to take his wonderfully written
  # script/structure and extend it by adding in a number of the more in depth Database Health Checks/KH Useful
  # SQL Queries that I frequently find myself asking customers to run when they are experiencing performance
  # issues with the Ops & DW DB's.
  #
  # I will cleanup my code in a later version and add additional functionality but for now Kudo's to Tim!
  #
  #
  # MBullwin 11/3/2014
  # www.OpsConfig.com
  #
  # As with all scripts I post, this is provided AS-IS without warrenty so please test first and use at your own risk.
  ######################################################################################################################
  $StartTime=Get-Date

  . Import-SCOMPowerShellModule

  # Connect to localhost when running on the management server or define a server to connect to.
  $connect = New-SCManagementGroupConnection -ComputerName $ManagementServer

  # The Name and Location of are we going to save this Report
  $CreateReportLocation = [System.IO.Directory]::CreateDirectory("c:\SCOM_Health_Check")
  $ReportLocation = "c:\SCOM_Health_Check"
  $ReportName = "$(get-date -format "yyyy-M-dd")-SCOM-HealthCheck.html"
  $ReportPath = "$ReportLocation\$ReportName"

  # Create header for HTML Report
  $Head = "<style>"
  $Head +="BODY{background-color:#CCCCCC;font-family:Calibri,sans-serif; font-size: small;}"
  $Head +="TABLE{border-width: 1px;border-style: solid;border-color: black;border-collapse: collapse; width: 98%;}"
  $Head +="TH{border-width: 1px;padding: 0px;border-style: solid;border-color: black;background-color:#293956;color:white;padding: 5px; font-weight: bold;text-align:left;}"
  $Head +="TD{border-width: 1px;padding: 0px;border-style: solid;border-color: black;background-color:#F0F0F0; padding: 2px;}"
  $Head +="</style>"


  # Retrieve the name of the Operational Database and Data WareHouse Servers from the registry.
  $reg = Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Microsoft Operations Manager\3.0\Setup\"
  $OperationsManagerDBServer = $reg.DatabaseServerName
  $OperationsManagerDWServer = $reg.DataWarehouseDBServerName
  # If the value is empty in this key, then we'll use the Get-SCOMDataWarehouseSetting cmdlet.
  If (!($OperationsManagerDWServer))
  {$OperationsManagerDWServer = Get-SCOMDataWarehouseSetting | Select-Object -expandProperty DataWarehouseServerName}

  $OperationsManagerDBServer = $OperationsManagerDBServer.ToUpper()
  $OperationsManagerDWServer = $OperationsManagerDWServer.ToUpper()

  $ReportingURL = Get-SCOMReportingSetting | Select-Object -ExpandProperty ReportingServerUrl
  $WebConsoleURL = Get-SCOMWebAddressSetting | Select-Object -ExpandProperty WebConsoleUrl
  <#
      # The number of days before Database Grooming
      # These are my settings, I use this to determine if someone has changed something
      # Feel free to comment this part out if you aren't interested
      $AlertDaysToKeep = 2
      $AvailabilityHistoryDaysToKeep = 2
      $EventDaysToKeep = 1
      $JobStatusDaysToKeep = 1
      $MaintenanceModeHistoryDaysToKeep = 2
      $MonitoringJobDaysToKeep = 2
      $PerformanceDataDaysToKeep = 2
      $StateChangeEventDaysToKeep = 2
 
      # SCOM Agent Heartbeat Settings
      $AgentHeartbeatInterval = 180
      $MissingHeartbeatThreshold = 3
  #>


  # SQL Server Function to query the Operational Database Server
  function Run-OpDBSQLQuery
  {
    Param($sqlquery)

    $SqlConnection = New-Object System.Data.SqlClient.SqlConnection
    $SqlConnection.ConnectionString = "Server=$OperationsManagerDBServer;Database=OperationsManager;Integrated Security=True"
    $SqlCmd = New-Object System.Data.SqlClient.SqlCommand
    $SqlCmd.CommandText = $sqlquery
    $SqlCmd.Connection = $SqlConnection
    $SqlAdapter = New-Object System.Data.SqlClient.SqlDataAdapter
    $SqlAdapter.SelectCommand = $SqlCmd
    $DataSet = New-Object System.Data.DataSet
    $SqlAdapter.Fill($DataSet) | Out-Null
    $SqlConnection.Close()
    $DataSet.Tables[0]
  }


  # SQL Server Function to query the Data Warehouse Database Server
  function Run-OpDWSQLQuery
  {
    Param($sqlquery)

    $SqlConnection = New-Object System.Data.SqlClient.SqlConnection
    $SqlConnection.ConnectionString = "Server=$OperationsManagerDWServer;Database=OperationsManagerDW;Integrated Security=True"
    $SqlCmd = New-Object System.Data.SqlClient.SqlCommand
    $SqlCmd.CommandText = $sqlquery
    $SqlCmd.Connection = $SqlConnection
    $SqlAdapter = New-Object System.Data.SqlClient.SqlDataAdapter
    $SqlAdapter.SelectCommand = $SqlCmd
    $DataSet = New-Object System.Data.DataSet
    $SqlAdapter.Fill($DataSet) | Out-Null
    $SqlConnection.Close()
    $DataSet.Tables[0]
  }


  # Retrieve the Data for the Majority of the Report
  # Truth is we probably don't need all of this data, but even on a busy environment it only takes a couple of mins to run.
  Write-Host "Retrieving Agents"
  $Agents = Get-SCOMAgent
  Write-Host "Retrieving Alerts"
  $Alerts = Get-SCOMAlert
  Write-Host "Retrieving Groups"
  $Groups = Get-SCOMGroup
  Write-Host "Retrieving Management Group"
  $ManagementGroup = Get-SCOMManagementGroup
  Write-Host "Retrieving Management Packs"
  $ManagementPacks = Get-SCManagementPack
  Write-Host "Retrieving Management Servers"
  $ManagementServers = Get-SCOMManagementServer
  Write-Host "Retrieving Monitors"
  $Monitors = Get-SCOMMonitor
  Write-Host "Retrieving Rules"
  $Rules = Get-SCOMRule

  # Check to see if the Reporting Server Site is OK
  $ReportingServerSite = New-Object System.Net.WebClient
  $ReportingServerSite = [net.WebRequest]::Create($ReportingURL)
  $ReportingServerSite.UseDefaultCredentials = $true
  $ReportingServerStatus = $ReportingServerSite.GetResponse() | Select-Object -expandProperty statusCode
  # This code can convert the "OK" Result to an Integer, like 200
  # (($web.GetResponse()).Statuscode) -as [int]

  # Check to see if the Web Server Site is OK
  $WebConsoleSite = New-Object System.Net.WebClient
  $WebConsoleSite = [net.WebRequest]::Create($WebConsoleURL)
  $WebConsoleSite.UseDefaultCredentials = $true
  $WebConsoleStatus = $WebConsoleSite.GetResponse() | Select-Object -expandProperty statusCode

  # SQL Server Function to query Size of the Database Server
  $DatabaseSize = @"
select a.FILEID,
[FILE_SIZE_MB]=convert(decimal(12,2),round(a.size/128.000,2)),
[SPACE_USED_MB]=convert(decimal(12,2),round(fileproperty(a.name,'SpaceUsed')/128.000,2)),
[FREE_SPACE_MB]=convert(decimal(12,2),round((a.size-fileproperty(a.name,'SpaceUsed'))/128.000,2)) ,
[GROWTH_MB]=convert(decimal(12,2),round(a.growth/128.000,2)),
NAME=left(a.NAME,15),
FILENAME=left(a.FILENAME,60)
from dbo.sysfiles a
"@


  #SQL Server Function to query Size of the TempDB
  $TempDBSize =@"
USE tempdb
select a.FILEID,
[FILE_SIZE_MB]=convert(decimal(12,2),round(a.size/128.000,2)),
[SPACE_USED_MB]=convert(decimal(12,2),round(fileproperty(a.name,'SpaceUsed')/128.000,2)),
[FREE_SPACE_MB]=convert(decimal(12,2),round((a.size-fileproperty(a.name,'SpaceUsed'))/128.000,2)) ,
[GROWTH_MB]=convert(decimal(12,2),round(a.growth/128.000,2)),
NAME=left(a.NAME,15),
FILENAME=left(a.FILENAME,60)
from dbo.sysfiles a
"@


  #SQL Server Function to query the version of SQL
  $SQLVersion =@"
SELECT SERVERPROPERTY('productversion') AS "Product Version", SERVERPROPERTY('productlevel') AS "Service Pack", SERVERPROPERTY ('edition') AS "Edition"
"@


  # Run the Size Query against the Operational Database and Data Warehouse Database Servers
  $OPDBSize = Run-OpDBSQLQuery $DatabaseSize
  $DWDBSize = Run-OpDWSQLQuery $DatabaseSize
  $OPTPSize = Run-OpDBSQLQuery $TempDBSize
  $DWTPSize = Run-OpDWSQLQuery $TempDBSize
  $OPSQLVER = Run-OpDBSQLQuery $SQLVersion
  $DWSQLVER = Run-OpDWSQLQuery $SQLVersion

  # Insert the Database Server details into the Report
  $ReportOutput += "<h2>Database Servers</h2>"
  $ReportOutput += "<p>Operational Database Server : $OperationsManagerDBServer</p>"
  $ReportOutput += $OPSQLVER | Select-Object "Product Version", "Service Pack", Edition | ConvertTo-Html -Fragment
  $ReportOutput += "<p>Data Warehouse Database Server : $OperationsManagerDWServer</p>"
  $ReportOutput += $DWSQLVER | Select-Object "Product Version", "Service Pack", Edition | ConvertTo-Html -Fragment

  # Insert the Size Results for the Operational Database into the Report
  $ReportOutput += "<h3>$OperationsManagerDBServer Operations Manager DB</h4>"
  $ReportOutput += $OPDBSize | Select-Object Name, FILE_SIZE_MB, SPACE_USED_MB, FREE_SPACE_MB, FILENAME | ConvertTo-HTML -fragment
  $ReportOutput += "<br></br>"
  $ReportOutput += "<h3>Operations Temp DB</h4>"
  $ReportOutput += $OPTPSize | Select-Object Name, FILE_SIZE_MB, SPACE_USED_MB, FREE_SPACE_MB, FILENAME | ConvertTo-HTML -fragment

  # Insert the Size Results for the Data Warehouse Database and TempDB into the Report
  $ReportOutput += "<br>"
  $ReportOutput += "<h3>$OperationsManagerDWServer Data Warehouse DB</h4>"
  $ReportOutput += $DWDBSize | Select-Object Name, FILE_SIZE_MB, SPACE_USED_MB, FREE_SPACE_MB, FILENAME | ConvertTo-HTML -fragment
  $ReportOutput += "<br></br>"
  $ReportOutput += "<h3>Data Warehouse Temp DB</h4>"
  $ReportOutput += $DWTPSize | Select-Object Name, FILE_SIZE_MB, SPACE_USED_MB, FREE_SPACE_MB, FILENAME | ConvertTo-HTML -fragment

  # SQL Query to find out how many State Changes there were yesterday
  $StateChangesYesterday = @"
-- How Many State Changes Yesterday?:
select count(*) from StateChangeEvent
where cast(TimeGenerated as date) = cast(getdate()-1 as date)
"@


  $StateChanges = Run-OpDBSQLQuery $StateChangesYesterday | Select-Object -ExpandProperty Column1 | Out-String

  $AllStats = @()

  $StatSummary = New-Object psobject
  $StatSummary | Add-Member noteproperty "Open Alerts" (($Alerts | Where-Object {$_.ResolutionState -ne 255}).count)
  $StatSummary | Add-Member noteproperty "Groups" ($Groups.Count)
  $StatSummary | Add-Member noteproperty "Monitors" ($Monitors.Count)
  $StatSummary | Add-Member noteproperty "Rules" ($Rules.Count)
  $StatSummary | Add-Member noteproperty "State Changes Yesterday" ($StateChanges | ForEach-Object {$_.TrimStart()} | ForEach-Object {$_.TrimEnd()})

  $AllStats += $StatSummary


  #SQL Query Top 10 Event generating computers
  $TopEventGeneratingComputers = @"
SELECT top 10 LoggingComputer, COUNT(*) AS TotalEvents
FROM EventallView
GROUP BY LoggingComputer
ORDER BY TotalEvents DESC
"@


  #SQL Query number of Events Generated per day
  $NumberOfEventsPerDay = @"
SELECT CASE WHEN(GROUPING(CONVERT(VARCHAR(20), TimeAdded, 101)) = 1)
THEN 'All Days'
ELSE CONVERT(VARCHAR(20), TimeAdded, 101) END AS DayAdded,
COUNT(*) AS NumEventsPerDay
FROM EventAllView
GROUP BY CONVERT(VARCHAR(20), TimeAdded, 101) WITH ROLLUP
ORDER BY DayAdded DESC
"@


  #SQL Query Most Common Events by Publishername
  $MostCommonEventsByPub = @"
SELECT top 25 Number AS "Event Number", Publishername, COUNT(*) AS TotalEvents
FROM EventAllView
GROUP BY Number, Publishername
ORDER BY TotalEvents DESC
"@


  #SQL Query the Number of Performance Insertions per Day
  $NumberofPerInsertsPerDay = @"
SELECT CASE WHEN(GROUPING(CONVERT(VARCHAR(20), TimeSampled, 101)) = 1)
THEN 'All Days' ELSE CONVERT(VARCHAR(20), TimeSampled, 101)
END AS DaySampled, COUNT(*) AS NumPerfPerDay
FROM PerformanceDataAllView
GROUP BY CONVERT(VARCHAR(20), TimeSampled, 101) WITH ROLLUP
ORDER BY DaySampled DESC
"@


  #SQL Query the Most common perf insertions by perf counter name
  $MostCommonPerfByN = @"
select top 25 pcv.objectname, pcv.countername, count (pcv.countername) as total from
performancedataallview as pdv, performancecounterview as pcv
where (pdv.performancesourceinternalid = pcv.performancesourceinternalid)
group by pcv.objectname, pcv.countername
order by count (pcv.countername) desc
"@


  #SQL Query the Top 25 Alerts by Alert Count
  $MostCommonAByAC = @"
SELECT Top 25 AlertStringName, Name, SUM(1) AS
AlertCount, SUM(RepeatCount+1) AS AlertCountWithRepeatCount
FROM Alertview WITH (NOLOCK)
GROUP BY AlertStringName, Name
ORDER BY AlertCount DESC
"@


  #SQL Query for Stale State Change Data
  $StaleStateChangeData = @"
declare @statedaystokeep INT
SELECT @statedaystokeep = DaysToKeep from PartitionAndGroomingSettings WHERE ObjectName = 'StateChangeEvent'
SELECT COUNT(*) as 'Total StateChanges',
count(CASE WHEN sce.TimeGenerated > dateadd(dd,-@statedaystokeep,getutcdate()) THEN sce.TimeGenerated ELSE NULL END) as 'within grooming retention',
count(CASE WHEN sce.TimeGenerated < dateadd(dd,-@statedaystokeep,getutcdate()) THEN sce.TimeGenerated ELSE NULL END) as '> grooming retention',
count(CASE WHEN sce.TimeGenerated < dateadd(dd,-30,getutcdate()) THEN sce.TimeGenerated ELSE NULL END) as '> 30 days',
count(CASE WHEN sce.TimeGenerated < dateadd(dd,-90,getutcdate()) THEN sce.TimeGenerated ELSE NULL END) as '> 90 days',
count(CASE WHEN sce.TimeGenerated < dateadd(dd,-365,getutcdate()) THEN sce.TimeGenerated ELSE NULL END) as '> 365 days'
from StateChangeEvent sce
"@


  #SQL Query Noisest monitors changing state in the past 7 days
  $NoisyMonitorData = @"
select distinct top 25
m.DisplayName as MonitorDisplayName,
m.Name as MonitorIdName,
mt.typename AS TargetClass,
count(sce.StateId) as NumStateChanges
from StateChangeEvent sce with (nolock)
join state s with (nolock) on sce.StateId = s.StateId
join monitorview m with (nolock) on s.MonitorId = m.Id
join managedtype mt with (nolock) on m.TargetMonitoringClassId = mt.ManagedTypeId
where m.IsUnitMonitor = 1
  -- Scoped to within last 7 days
AND sce.TimeGenerated > dateadd(dd,-7,getutcdate())
group by m.DisplayName, m.Name,mt.typename
order by NumStateChanges desc
"@


  #SQL Query Top 25 Monitors changing state by Object
  $NoisyMonitorByObject =@"
select distinct top 25
bme.DisplayName AS ObjectName,
bme.Path,
m.DisplayName as MonitorDisplayName,
m.Name as MonitorIdName,
mt.typename AS TargetClass,
count(sce.StateId) as NumStateChanges
from StateChangeEvent sce with (nolock)
join state s with (nolock) on sce.StateId = s.StateId
join BaseManagedEntity bme with (nolock) on s.BasemanagedEntityId = bme.BasemanagedEntityId
join MonitorView m with (nolock) on s.MonitorId = m.Id
join managedtype mt with (nolock) on m.TargetMonitoringClassId = mt.ManagedTypeId
where m.IsUnitMonitor = 1
   -- Scoped to specific Monitor (remove the "--" below):
   -- AND m.MonitorName like ('%HealthService%')
   -- Scoped to specific Computer (remove the "--" below):
   -- AND bme.Path like ('%sql%')
   -- Scoped to within last 7 days
AND sce.TimeGenerated > dateadd(dd,-7,getutcdate())
group by s.BasemanagedEntityId,bme.DisplayName,bme.Path,m.DisplayName,m.Name,mt.typename
order by NumStateChanges desc
"@


  #SQL Query Grooming Settings for the Operational Database
  $OpsDBGrooming =@"
SELECT
ObjectName,
GroomingSproc,
DaysToKeep,
GroomingRunTime,
DataGroomedMaxTime
FROM PartitionAndGroomingSettings WITH (NOLOCK)
"@



  #SQL Query DW DB Staging Tables
  $DWDBStagingTables = @"
select count(*) AS "Alert Staging Table" from Alert.AlertStage
"@


  $DWDBStagingTablesEvent =@"
select count (*) AS "Event Staging Table" from Event.eventstage
"@


  $DWDBStagingTablesPerf =@"
select count (*) AS "Perf Staging Table" from Perf.PerformanceStage
"@


  $DWDBStagingTablesState =@"
select count (*) AS "State Staging Table" from state.statestage
"@


  #SQL Query DW Grooming Retention
  $DWDBGroomingRetention =@"
select ds.datasetDefaultName AS 'Dataset Name', sda.AggregationTypeId AS 'Agg Type 0=raw, 20=Hourly, 30=Daily', sda.MaxDataAgeDays AS 'Retention Time in Days'
from dataset ds, StandardDatasetAggregation sda
WHERE ds.datasetid = sda.datasetid ORDER by "Retention Time in Days" desc
"@


  #SQL function to Query the Top 25 largest tables in a database
  $DWDBLargestTables =@"
SELECT TOP 25
a2.name AS [tablename], (a1.reserved + ISNULL(a4.reserved,0))* 8 AS reserved,
a1.rows as row_count, a1.data * 8 AS data,
(CASE WHEN (a1.used + ISNULL(a4.used,0)) > a1.data THEN (a1.used + ISNULL(a4.used,0)) - a1.data ELSE 0 END) * 8 AS index_size,
(CASE WHEN (a1.reserved + ISNULL(a4.reserved,0)) > a1.used THEN (a1.reserved + ISNULL(a4.reserved,0)) - a1.used ELSE 0 END) * 8 AS unused,
(row_number() over(order by (a1.reserved + ISNULL(a4.reserved,0)) desc))%2 as l1,
a3.name AS [schemaname]
FROM (SELECT ps.object_id, SUM (CASE WHEN (ps.index_id < 2) THEN row_count ELSE 0 END) AS [rows],
SUM (ps.reserved_page_count) AS reserved,
SUM (CASE WHEN (ps.index_id < 2) THEN (ps.in_row_data_page_count + ps.lob_used_page_count + ps.row_overflow_used_page_count)
ELSE (ps.lob_used_page_count + ps.row_overflow_used_page_count) END ) AS data,
SUM (ps.used_page_count) AS used
FROM sys.dm_db_partition_stats ps
GROUP BY ps.object_id) AS a1
LEFT OUTER JOIN (SELECT it.parent_id,
SUM(ps.reserved_page_count) AS reserved,
SUM(ps.used_page_count) AS used
FROM sys.dm_db_partition_stats ps
INNER JOIN sys.internal_tables it ON (it.object_id = ps.object_id)
WHERE it.internal_type IN (202,204)
GROUP BY it.parent_id) AS a4 ON (a4.parent_id = a1.object_id)
INNER JOIN sys.all_objects a2 ON ( a1.object_id = a2.object_id )
INNER JOIN sys.schemas a3 ON (a2.schema_id = a3.schema_id)
WHERE a2.type <> N'S' and a2.type <> N'IT'
"@


  #SQL Function to query and check backup status of SQL Databases
  $SQLBackupStatus =@"
SELECT
d.name,
DATEDIFF(Day, COALESCE(MAX(b.backup_finish_date), d.create_date), GETDATE()) AS [DaysSinceBackup]
FROM
sys.databases d
LEFT OUTER JOIN msdb.dbo.backupset b
ON d.name = b.database_name
WHERE
d.is_in_standby = 0
AND source_database_id is null
AND d.name NOT LIKE 'tempdb'
AND (b.[type] IN ('D', 'I') OR b.[type] IS NULL)
GROUP BY
d.name, d.create_date
"@


  # Run additional SQL Queries against the Operational Database
  $OPTOPALERT = Run-OpDBSQLQuery $TopEventGeneratingComputers
  $OPNUMEPERDAY = Run-OpDBSQLQuery $NumberOfEventsPerDay
  $OPMOSTCOMEVENT = Run-OpDBSQLQuery $MostCommonEventsByPub
  $OPPERFIPERD = Run-OpDBSQLQuery $NumberofPerInsertsPerDay
  $OPPERFIBYN = Run-OpDBSQLQuery $MostCommonPerfByN
  $OPTOPALERTC = Run-OpDBSQLQuery $MostCommonAByAC
  $OPSTALESTD = Run-OpDBSQLQuery $StaleStateChangeData
  $OPNOISYMON = Run-OpDBSQLQuery $NoisyMonitorData
  $OPNOISYMONOBJ = Run-OpDBSQLQuery $NoisyMonitorByObject
  $OPDBGROOM = Run-OpDBSQLQuery $OpsDBGrooming
  $OPLARGTAB = Run-OpDBSQLQuery $DWDBLargestTables
  $OPDBBACKUP = Run-OpDBSQLQuery $SQLBackupStatus

  #Run additional SQL Queries against the DW DB
  $DWDBSGTB = Run-OpDWSQLQuery $DWDBStagingTables
  $DWDBSGTBEV = Run-OpDWSQLQuery $DWDBStagingTablesEvent
  $DWDBSGTBPE = Run-OpDWSQLQuery $DWDBStagingTablesPerf
  $DWDBSGTBST = Run-OpDWSQLQuery $DWDBStagingTablesState
  $DWDBGRET = Run-OpDWSQLQuery $DWDBGroomingRetention
  $DWDBLARGETAB = Run-OpDWSQLQuery $DWDBLargestTables
  $DWDBBACKUP = Run-OpDWSQLQuery $SQLBackupStatus

  #Output to HTML Report

  $ReportOutput += "<h2>Operational Database Health</h2>"
  $ReportOutput += "<h3>Operations Database Backup Status</h3>"
  $ReportOutput += $OPDBBACKUP | Select-Object name, DaysSinceBackup | ConvertTo-HTML -Fragment
  $ReportOutput += "<h3>Operations Database Top 25 Largest Tables</h3>"
  $ReportOutput += $OPLARGTAB | Select-Object tablename, reserved, row_count, data, index_size, unused |ConvertTo-Html -Fragment
  $ReportOutput += "<h3>Number of Events Generated Per Day</h3>"
  $ReportOutput += $OPNUMEPERDAY | Select-Object NumEventsPerDay, DayAdded | ConvertTo-HTML -Fragment
  $ReportOutput += "<h3>Top 10 Event Generating Computers</h3>"
  $ReportOutput += $OPTOPALERT | Select-Object LoggingComputer, TotalEvents | ConvertTo-HTML -Fragment
  $ReportOutput += "<h3>Top 25 Events By Publisher</h3>"
  $ReportOutput += $OPMOSTCOMEVENT | Select-Object "Event Number", Publishername, TotalEvents | ConvertTo-Html -Fragment
  $ReportOutput += "<h3>Number of Perf Insertions Per Day</h3>"
  $ReportOutput += $OPPERFIPERD | Select-Object DaySampled, NumPerfPerDay | ConvertTo-Html -Fragment
  $ReportOutput += "<h3>Top 25 Perf Insertions by Object/Counter Name</h3>"
  $ReportOutput += $OPPERFIBYN | Select-Object objectname, countername, total | ConvertTo-Html -Fragment
  $ReportOutput += "<h3>Top 25 Alerts by Alert Count</h3>"
  $ReportOutput += $OPTOPALERTC | Select-Object AlertStringName, Name, AlertCount, AlertCountWithRepeatCount | ConvertTo-Html -Fragment



  # Get the alerts with a repeat count higher than the variable $RepeatCount
  $RepeatCount = 200

  $ReportOutput += "<br>"
  $ReportOutput += "<h3>Alerts with a Repeat Count higher than $RepeatCount</h3>"


  # Produce a table of all Open Alerts above the repeatcount and add it to the Report
  $ReportOutput += $Alerts | Where-Object {$_.RepeatCount -ge $RepeatCount -and $_.ResolutionState -ne 255} | Select-Object Name, Category, NetBIOSComputerName, RepeatCount | Sort-Object repeatcount -desc | ConvertTo-HTML -fragment

  #Output to HTML report
  $ReportOutput += "<h3>Stale State Change Data</h3>"
  $ReportOutput += $OPSTALESTD | Select-Object "Total StateChanges", "within grooming retention", "> grooming retention","> 30 days","> 90 days","> 365 days"| ConvertTo-Html -Fragment
  $ReportOutput += "<h3>Top 25 Monitors Changing State in the last 7 Days</h3>"
  $ReportOutput += $OPNOISYMON | Select-Object MonitorDisplayName, MonitorIdName, TargetClass, NumStateChanges | ConvertTo-Html -Fragment
  $ReportOutput += "<h3>Top 25 Monitors Changing State By Object</h3>"
  $ReportOutput += $OPNOISYMONOBJ | Select-Object ObjectName, Path, MonitorDisplayName, MonitorIdName,TargetClass, NumStateChanges | ConvertTo-Html -Fragment
  $ReportOutput += "<h3>Operations Database Grooming History</h3>"
  $ReportOutput += $OPDBGROOM | Select-Object ObjectName, GroomingSproc, DaysToKeep, GroomingRunTime,DataGroomedMaxTime | ConvertTo-HTML -Fragment

  # SQL Query to find out what Grooming Jobs have run in the last 24 hours
  $DidGroomingRun = @"
-- Did Grooming Run?:
SELECT [InternalJobHistoryId]
      ,[TimeStarted]
      ,[TimeFinished]
      ,[StatusCode]
      ,[Command]
      ,[Comment]
FROM [dbo].[InternalJobHistory]
WHERE [TimeStarted] >= DATEADD(day, -2, GETDATE())
Order by [TimeStarted]
"@


  # Produce a table of Grooming History and add it to the Report
  $ReportOutput += "<h3>Grooming History From The Past 48 Hours</h3>"
  $ReportOutput += Run-OpDBSQLQuery $DidGroomingRun InternalJobHistoryId, TimeStarted, TimeFinished, StatusCode, Command, Comment | Select-Object | ConvertTo-HTML -fragment

  #Produce Table of DW DB Health
  $ReportOutput +="<h2>Data Warehouse Database Health</h2>"
  $ReportOutput +="<h3>Data Warehouse DB Backup Status</h3>"
  $ReportOutput +=$DWDBBACKUP | Select-Object name, DaysSinceBackup | ConvertTo-Html -Fragment
  $ReportOutput +="<h3>Data Warehouse Top 25 Largest Tables</h3>"
  $ReportOutput +=$DWDBLARGETAB | Select-Object tablename, reserved, row_count, data, index_size, unused |ConvertTo-Html -fragment
  $ReportOutput +="<h3>Data Warehouse Staging Tables</h3>"
  $ReportOutput +=$DWDBSGTB | Select-Object "Alert Staging Table", Table | ConvertTo-Html -Fragment
  $ReportOutput +=$DWDBSGTBEV | Select-Object "Event Staging Table", Table | ConvertTo-Html -Fragment
  $ReportOutput +=$DWDBSGTBPE | Select-Object "Perf Staging Table", Table | ConvertTo-Html -Fragment
  $ReportOutput +=$DWDBSGTBST | Select-Object "State Staging Table", Table| ConvertTo-Html -Fragment
  $ReportOutput +="<h3>Data Warhouse Grooming Retention</h3>"
  $ReportOutput +=$DWDBGRET | Select-Object "Dataset Name", "Agg Type 0=raw, 20=Hourly, 30=Daily","Retention Time in Days"| ConvertTo-Html -Fragment

  # Insert the Results for the Number of Management Servers into the Report
  $ReportOutput += "<p>Number of Management Servers : $($ManagementServers.count)</p>"

  # Retrieve the data for the Management Servers
  $ReportOutput += "<br>"
  $ReportOutput += "<h2>Management Servers</h2>"

  $AllManagementServers = @()

  ForEach ($ManagementServer in $ManagementServers)
  {
    # Find out the Server Uptime for each of the Management Servers
    #Original query referenced -computer $ManagementServer.Name this was an error I modified to .Displayname to fix
    #TPaul: updated this to use CIM instead of WMI
    $lastboottime = (Get-CimInstance -ClassName win32_operatingsystem ).LastBootUpTime
    $sysuptime =New-TimeSpan $lastboottime (Get-Date)  
    $totaluptime = "" + $sysuptime.days + " days " + $sysuptime.hours + " hours " + $sysuptime.minutes + " minutes " + $sysuptime.seconds + " seconds"

    # Find out the Number of WorkFlows Running on each of the Management Servers
    $perfWorkflows = Get-Counter -ComputerName $ManagementServer.DisplayName -Counter "\Health Service\Workflow Count" -SampleInterval 1 -MaxSamples 1

    # The Performance Counter seems to return a lot of other characters and spaces...I only want the number of workflows, let's dump the rest
    [int]$totalWorkflows = $perfWorkflows.readings.Split(':')[-1] | ForEach-Object {$_.TrimStart()} | ForEach-Object {$_.TrimEnd()}

    $ManagementServerProperty = New-Object psobject
    $ManagementServerProperty | Add-Member noteproperty "Management Server" ($ManagementServer.DisplayName)
    $ManagementServerProperty | Add-Member noteproperty "Health State" ($ManagementServer.HealthState)
    $ManagementServerProperty | Add-Member noteproperty "Version" ($ManagementServer.Version)
    $ManagementServerProperty | Add-Member noteproperty "Action Account" ($ManagementServer.ActionAccountIdentity)
    $ManagementServerProperty | Add-Member noteproperty "System Uptime" ($totaluptime)
    $ManagementServerProperty | Add-Member noteproperty "Workflows" ($totalWorkflows)
    $AllManagementServers += $ManagementServerProperty
  }

  # Insert the Results for the Management Servers into the Report
  $ReportOutput += $AllManagementServers | Select-Object "Management Server", "Health State", "Version", "Action Account", "System Uptime", "Workflows" | Sort-Object "Management Server" | ConvertTo-HTML -fragment

  # Insert the Results for the Stats and State Changes into the Report
  $ReportOutput += "<br>"
  $ReportOutput += "<h2>Daily KPI</h2>"
  $ReportOutput += $AllStats | Select-Object "Open Alerts", "Groups", "Monitors", "Rules", "State Changes Yesterday" | ConvertTo-HTML -fragment

  # Retrieve and Insert the Results for the Management Packs that have been modified into the Report
  Write-Host "Checking for Management Packs that have been modified in the last 24 hours"

  $ReportOutput += "<br>"
  $ReportOutput += "<h2>Management Packs Modified in the Last 24 Hours</h2>"
  If (!($ManagementPacks | Where-Object {$_.LastModified -ge (Get-Date).addhours(-24)}))
  {
    $ReportOutput += "<p>No Management Packs have been updated in the last 24 hours</p>"
  }
  Else
  {
    $ReportOutput += $ManagementPacks | Where-Object {$_.LastModified -ge (Get-Date).addhours(-24)} | Select-Object Name, LastModified | ConvertTo-HTML -fragment
  }


  # Retrieve and Insert the Results for the Default Management Pack into the Report
  # This 'should be empty'...don't store stuff here!
  Write-Host "Checking for the Default Management Pack for Overrides"
  $ReportOutput += "<br>"
  $ReportOutput += "<h2>The Default Management Pack</h2>"

  # Excluding these 2 ID's as they are part of the default MP for DefaultUser and Language Code Overrides
  $excludedID = "5a67584f-6f63-99fc-0d7a-55587d47619d", "e358a914-c851-efaf-dda9-6ca5ef1b3eb7"
  $defaultMP = $ManagementPacks | Where-Object {$_.Name -match "Microsoft.SystemCenter.OperationsManager.DefaultUser"}
  ##Changed below line for compat with PowerShell 2.0
  ##If (!($defaultMP.GetOverrides() | ? {$_.ID -notin $excludedID}))
  If (!($defaultMP.GetOverrides() | Where-Object {$excludedID -NotContains $_.ID}))
  {
    $ReportOutput += "<p>There are no Overrides being Stored in the Default Management Pack</p>"
  }
  Else
  {

    ##Changed below line for compat with PowerShell 2.0
    #$foundOverride = Get-SCOMClassInstance -id ($defaultMP.GetOverrides() | ? {$_.ID -notin $excludedID -AND $_.ContextInstance -ne $guid} | select -expandproperty ContextInstance -ea SilentlyContinue)
    $foundOverride = Get-SCOMClassInstance -id ($defaultMP.GetOverrides() | Where-Object {$excludedID -NotContains $_.ID -AND $_.ContextInstance -ne $guid} | Select-Object -expandproperty ContextInstance -ea SilentlyContinue)


    $ReportOutput += "<p>Found overrides against the following targets: $foundOverride.Displayname</p>"
    ##PowerShell 2.0 Compat
    ##$ReportOutput += $($defaultMP.GetOverrides() | ? {$_.ID -notin $excludedID} | Select Name, Property, Value, LastModified, TimeAdded) | ConvertTo-HTML -fragment
    $ReportOutput += $($defaultMP.GetOverrides() | Where-Object {$excludedID -NotContains $_.ID} | Select-Object -Property Name, Property, Value, LastModified, TimeAdded) | ConvertTo-HTML -fragment

  }



  # Show all Agents that are in an Uninitialized State
  Write-Host "Checking for Uninitialized Agents"

  $ReportOutput += "<br>"
  $ReportOutput += "<h2>Uninitialized Agents</h2>"
  If (!($Agents | Where-Object -FilterScript {$_.HealthState -eq "Uninitialized"} | Select-Object -Property Name))
  {
    $ReportOutput += "<p>No Agents are in the Uninitialized State</p>"
  }
  Else
  {
    $ReportOutput += $Agents | Where-Object -FilterScript {$_.HealthState -eq "Uninitialized"} | Select-Object -Property Name | ConvertTo-HTML -fragment
  }


  # Show a Summary of all Agents States
  $healthy = $uninitialized = $warning = $critical = 0

  Write-Host "Checking Agent States"

  $ReportOutput += "<br>"
  $ReportOutput += "<h3>Agent Stats</h3>"

  switch ($Agents | Select-Object HealthState ) {
    {$_.HealthState -like "Success"} {$healthy++}
    {$_.HealthState -like "Uninitialized"} {$uninitialized++}
    {$_.HealthState -like "Warning"}  {$warning++}
    {$_.HealthState -like "Error"} {$critical++}
  }
  $totalagents = ($healthy + $warning + $critical + $uninitialized)

  $AllAgents = @()

  $iAgent = New-Object psobject
  $iAgent | Add-Member noteproperty -Name "Agents Healthy" -Value $healthy
  $iAgent | Add-Member noteproperty -Name "Agents Warning" -Value $warning
  $iAgent | Add-Member noteproperty -Name "Agents Critical" -Value $critical
  $iAgent | Add-Member noteproperty -Name "Agents Uninitialized" -Value $uninitialized
  $iAgent | Add-Member noteproperty -Name "Total Agents" -Value $totalagents

  $AllAgents += $iAgent

  $ReportOutput += $AllAgents | Select-Object -Property "Agents Healthy", "Agents Warning", "Agents Critical", "Agents Uninitialized", "Total Agents" | ConvertTo-HTML -fragment

  # Agent Pending Management States
  Write-Host "Checking Agent Pending Management States"

  $ReportOutput += "<br>"
  $ReportOutput += "<h3>Agent Pending Management Summary</h3>"

  $pushInstall = $PushInstallFailed = $ManualApproval = $RepairAgent = $RepairFailed = $UpdateFailed = 0

  $agentpending = Get-SCOMPendingManagement
  switch ($agentpending | Select-Object AgentPendingActionType ) {
    {$_.AgentPendingActionType -like "PushInstall"} {$pushInstall++}
    {$_.AgentPendingActionType -like "PushInstallFailed"} {$PushInstallFailed++}
    {$_.AgentPendingActionType -like "ManualApproval"}  {$ManualApproval++}
    {$_.AgentPendingActionType -like "RepairAgent"} {$RepairAgent++}
    {$_.AgentPendingActionType -like "RepairFailed"} {$RepairFailed++}
    {$_.AgentPendingActionType -like "UpdateFailed"} {$UpdateFailed++}

  }

  $AgentsPending = @()

  $AgentSummary = New-Object psobject
  $AgentSummary | Add-Member noteproperty "Push Install" ($pushInstall)
  $AgentSummary | Add-Member noteproperty "Push Install Failed" ($PushInstallFailed)
  $AgentSummary | Add-Member noteproperty "Manual Approval" ($ManualApproval)
  $AgentSummary | Add-Member noteproperty "Repair Agent" ($RepairAgent)
  $AgentSummary | Add-Member noteproperty "Repair Failed" ($RepairFailed)
  $AgentSummary | Add-Member noteproperty "Update Failed" ($UpdateFailed)

  $AgentsPending += $AgentSummary

  # Produce a table of all Agent Pending Management States and add it to the Report
  $ReportOutput += $AgentsPending | Select-Object "Push Install", "Push Install Failed", "Manual Approval", "Repair Agent", "Repair Failed", "Update Failed" | ConvertTo-HTML -fragment

  $ReportOutput += "<br>"
  $ReportOutput += "<h2>Alerts</h2>"

  $AlertsAll = ($Alerts | Where-Object {$_.ResolutionState -ne 255}).Count
  $AlertsWarning = ($Alerts | Where-Object {$_.Severity -eq "Warning" -AND $_.ResolutionState -ne 255}).Count
  $AlertsError = ($Alerts | Where-Object {$_.Severity -eq "Error" -AND $_.ResolutionState -ne 255}).Count
  $AlertsInformation = ($Alerts | Where-Object {$_.Severity -eq "Information" -AND $_.ResolutionState -ne 255}).Count
  $Alerts24hours = ($Alerts | Where-Object {$_.TimeRaised -ge (Get-Date).addhours(-24) -AND $_.ResolutionState -ne 255}).count

  $AllAlerts = @()


  $AlertSeverity = New-Object psobject
  $AlertSeverity | Add-Member noteproperty "Warning" ($AlertsWarning)
  $AlertSeverity | Add-Member noteproperty "Error" ($AlertsError)
  $AlertSeverity | Add-Member noteproperty "Information" ($AlertsInformation)
  $AlertSeverity | Add-Member noteproperty "Last 24 Hours" ($Alerts24hours)
  $AlertSeverity | Add-Member noteproperty "Total Open Alerts" ($AlertsAll)
  $AllAlerts += $AlertSeverity


  # Produce a table of all alert counts for warning, error, information, Last 24 hours and Total Alerts and add it to the Report
  $ReportOutput += $AllAlerts | Select-Object "Warning", "Error", "Information", "Last 24 Hours", "Total Open Alerts" | ConvertTo-HTML -fragment

  <#
      # Check if the Action Account is a Local Administrator on Each Management Server
      # This will only work if the account is a member of the Local Administrators Group directly.
      # If it has access by Group Membership, you can look for that Group instead
      # $ActionAccount = "YourGrouptoCheck"
      # Then replace all 5 occurrences below of $ManagementServer.ActionAccountIdentity with $ActionAccount
 
      Write-Host "Checking if the Action Account is a member of the Local Administrators Group for each Management Server"
 
      $ReportOutput += "<br>"
      $ReportOutput += "<h2>SCOM Action Account</h2>"
 
      ForEach ($ms in $ManagementServers.DisplayName | sort DisplayName)
      {
      $admins = @()
      $group =[ADSI]"WinNT://$ms/Administrators"
      $members = @($group.psbase.Invoke("Members"))
      $members | foreach {
      $obj = new-object psobject -Property @{
      Server = $Server
      Admin = $_.GetType().InvokeMember("Name", 'GetProperty', $null, $_, $null)
      }
      $admins += $obj
      }
 
      If ($admins.admin -match $ManagementServer.ActionAccountIdentity)
      {
      # Write-Host "The user $($ManagementServer.ActionAccountIdentity) is a Local Administrator on $ms"
      $ReportOutput += "<p>The user $($ManagementServer.ActionAccountIdentity) is a Local Administrator on $ms</p>"
      }
      Else
      {
      # Write-Host "The user $($ManagementServer.ActionAccountIdentity) is NOT a Local Administrator on $ms"
      $ReportOutput += "<p><span style=`"color: `#CC0000;`">The user $($ManagementServer.ActionAccountIdentity) is NOT a Local Administrator on $ms</span></p>"
      }
      }
  #>




  # Objects in Maintenance Mode

  #SQL Query Servers in MMode
  $ServersInMM =@"
select DisplayName from dbo.MaintenanceMode mm
join dbo.BaseManagedEntity bm on mm.BaseManagedEntityId = bm.BaseManagedEntityId
where Path is NULL and IsInMaintenanceMode = 1
"@


  $OpsDBSIMM = Run-OpDBSQLQuery $ServersInMM

  $ReportOutput += "<br>"
  $ReportOutput += "<h2>Servers in Maintenance Mode</h2>"

  If (!($OpsDBSIMM))
  {
    $ReportOutput += "<p>There are no objects in Maintenance Mode</p>"
  }
  Else
  {
    $ReportOutput += $OpsDBSIMM | Select-Object DisplayName | ConvertTo-HTML -fragment
  }

  <#
      # Global Grooming Settings
      # Simple comparisons against the values set at the beginning of this script
      # I use this to see if anyone has changed the settings. So set the values at the top of this script to match the values that your environment 'should' be set to.
 
      $ReportOutput += "<br>"
      $ReportOutput += "<h2>SCOM Global Settings</h2>"
 
 
      $SCOMDatabaseGroomingSettings = Get-SCOMDatabaseGroomingSetting
 
 
      If ($SCOMDatabaseGroomingSettings.AlertDaysToKeep -ne $AlertDaysToKeep)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">Alert Days to Keep has been changed! Reset back to $AlertDaysToKeep</span></p>"}
      Else {$ReportOutput += "<p>Alert Days is correctly set to $AlertDaysToKeep</p>"}
 
      If ($SCOMDatabaseGroomingSettings.AvailabilityHistoryDaysToKeep -ne $AvailabilityHistoryDaysToKeep)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">Availability History Days has been changed! Reset back to $AvailabilityHistoryDaysToKeep</span></p>"}
      Else {$ReportOutput += "<p>Availability History Days is correctly set to $AvailabilityHistoryDaysToKeep</p>"}
 
      If ($SCOMDatabaseGroomingSettings.EventDaysToKeep -ne $EventDaysToKeep)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">Event Days has been changed! Reset back to $EventDaysToKeep</span></p>"}
      Else {$ReportOutput += "<p>Event Days is correctly set to $EventDaysToKeep</p>"}
 
      If ($SCOMDatabaseGroomingSettings.JobStatusDaysToKeep -ne $JobStatusDaysToKeep)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">Job Days (Task History) has been changed! Reset back to $JobStatusDaysToKeep</span></p>"}
      Else {$ReportOutput += "<p>Job Days (Task History) is correctly set to $JobStatusDaysToKeep</p>"}
 
      If ($SCOMDatabaseGroomingSettings.MaintenanceModeHistoryDaysToKeep -ne $MaintenanceModeHistoryDaysToKeep)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">Maintenance Mode History has been changed! Reset back to $MaintenanceModeHistoryDaysToKeep</span></p>"}
      Else {$ReportOutput += "<p>Maintenance Mode History is correctly set to $MaintenanceModeHistoryDaysToKeep</p>"}
 
      If ($SCOMDatabaseGroomingSettings.MonitoringJobDaysToKeep -ne $MonitoringJobDaysToKeep)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">Monitoring Job Data has been changed! Reset back to $MonitoringJobDaysToKeep</span></p>"}
      Else {$ReportOutput += "<p>Monitoring Job Data is correctly set to $MonitoringJobDaysToKeep</p>"}
 
      If ($SCOMDatabaseGroomingSettings.PerformanceDataDaysToKeep -ne $PerformanceDataDaysToKeep)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">Performance Data has been changed! Reset back to $PerformanceDataDaysToKeep</span></p>"}
      Else {$ReportOutput += "<p>Performance Data is correctly set to $PerformanceDataDaysToKeep</p>"}
 
      If ($SCOMDatabaseGroomingSettings.StateChangeEventDaysToKeep -ne $StateChangeEventDaysToKeep)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">State Change Data has been changed! Reset back to $StateChangeEventDaysToKeep</span></p>"}
      Else {$ReportOutput += "<p>State Change Data is correctly set to $StateChangeEventDaysToKeep</p>"}
 
 
      # SCOM Agent Heartbeat Settings
      $HeartBeatSetting = Get-SCOMHeartbeatSetting
 
      If ($HeartBeatSetting.AgentHeartbeatInterval -ne 180 -AND $HeartBeatSetting.MissingHeartbeatThreshold -ne 3)
      {$ReportOutput += "<p><span style=`"color: `#CC0000;`">The HeartBeat Settings have been changed! Reset back to $AgentHeartbeatInterval and $MissingHeartbeatThreshold</span></p>"}
      Else {$ReportOutput += "<p>The HeartBeat Settings are correctly set to Interval: $AgentHeartbeatInterval and Missing Threshold: $MissingHeartbeatThreshold</p>"}
  #>

  # How long did this script take to run?
  $EndTime=Get-Date
  $TotalRunTime=$EndTime-$StartTime

  # Add the time to the Report
  $ReportOutput += "<br>"
  $ReportOutput += "<p>Total Script Run Time: $($TotalRunTime.hours) hrs $($TotalRunTime.minutes) min $($TotalRunTime.seconds) sec</p>"

  # Close the Body of the Report
  $ReportOutput += "</body>"

  Write-Host "Saving HTML Report to $ReportPath"

  # Save the Final Report to a File
  ConvertTo-HTML -head $Head -body "$ReportOutput" | Out-File $ReportPath

  Invoke-Item "$ReportPath"

  <#
      # Send Final Report by email...
 
      Write-Host "Emailing Report"
      $SMTPServer ="your.smtpserver.com"
      $Body = ConvertTo-HTML -head $Head -body "$ReportOutput"
      $SmtpClient = New-Object Net.Mail.SmtpClient($smtpServer);
      $mailmessage = New-Object system.net.mail.mailmessage
      $mailmessage.from = "sender@company.com"
      $mailmessage.To.add("recipient@company.com")
      # Want more recipient's? Just add a new line
      $mailmessage.To.add("anotherrecipient@company.com")
      $mailmessage.Subject = "Tims SCOM Healthcheck Report"
      $MailMessage.IsBodyHtml = $true
      $mailmessage.Body = $Body
      $smtpclient.Send($mailmessage)
  #>


  <#
      # Insert the Results for the Reporting Server URL into the Report
      $ReportOutput += "<h2>Reporting Server</h2>"
      $ReportOutput += "<p>Reporting Server URL : <a href=`"$ReportingURL/`">$ReportingURL</a></p>"
      $ReportOutput += "<p>The Reporting Server URL Status : $ReportingServerStatus</p>"
 
      # Insert the Results for the Web Console URL into the Report
      $ReportOutput += "<h2>Web Console Servers</h2>"
      $ReportOutput += "<p>Web Console URL : <a href=`"$WebConsoleURL/`">$WebConsoleURL</a></p>"
      $ReportOutput += "<p>The Web Console URL Status : $WebConsoleStatus</p>"
  #>


  # A bit of cleanup
  #Clear-Variable Agents, Alerts, Groups, ManagementGroup, ManagementPacks, ManagementServer, Monitors, Rules, ReportOutput, StartTime, EndTime, TotalRunTime, SCOMDatabaseGroomingSettings

}
#######################################################################

<#
    .Synopsis
    List version and file path information for all management pack files (*.mp, *.mpb, *.xml).
    .DESCRIPTION
    Will display MP file and version information.
 
    .EXAMPLE
    Get-SCOMMPFileInfo -inDir 'C:\Program Files (x86)\System Center Management Packs' -Verbose | Select Name,DisplayName,Version,FullName | Format-Table
 
    .EXAMPLE
    Get-SCOMMPFileInfo -inDir 'C:\Program Files (x86)\System Center Management Packs' | Select-Object Name,DisplayName,Version,FullName,KeyToken, @{Name="References"; Expression={($_.References.GetEnumerator() | Out-String) } } | Out-GridView -PassThru
 
    The above example will display all of the MP properties in Gridview with the References column displayed in a more friendly format.
 
    .EXAMPLE
    Get-SCOMMPFileInfo -inDir 'C:\Program Files (x86)\System Center Management Packs' | Out-GridView -PassThru | % {& explorer.exe (Split-Path -Parent $_.FullName) }
 
    The above example will present a list of your management packs in Gridview. Once you select one or more items-OK, the folder(s) containing the management pack(s) will be opened with File Explorer.
 
    .EXAMPLE
    Get-SCOMMPFileInfo -inDir 'C:\MySealedMPs' | Format-Table | ConvertTo-Html | Set-Content C:\mpfiles.html; C:\mpfiles.html
 
    The above example will create an HTML report of your MP files then open the HTML file with the default application.
 
    .INPUTS
    System.String, Switch
    .OUTPUTS
    Custom object
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    History:
    2020.02.27 - Improved efficiency. Added verbose output.
    2020.02.21 - Added more examples. Removed Gridview and Passthru switches.
    2019.07.29 - Added a few more MP properties to the returned object(s)
    2018.05.30 - Added GridView option as well as some other cleanup.
    .FUNCTIONALITY
    Useful for identifying specific versions of management packs in your local archive.
    .LINK
    Get-SCOMClassInfo
    New-SCOMClassGraph
    Unseal-SCOMMP
#>

Function Get-SCOMMPFileInfo {

  [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
      SupportsShouldProcess=$true,
  PositionalBinding=$false)]
      
  Param (
    [Parameter(Mandatory=$true,
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [Alias("Path")]
    [string]$inDir

    # deprecated
    # [switch]$GridView=$true,
    
    # deprecated
    #[switch]$passthru=$true
  )
 
  [System.Collections.ArrayList]$arrMaster = @()

  $MPFiles = Get-ChildItem -Path $inDir -Recurse -Include *.mp,*.mpb,*.xml
  ForEach ($File in $MPFiles.FullName) {
    Write-Verbose $File
    
    Switch ($File) {
      {$_ -Match '\.mp$|\.xml$'} {
        $MP = Get-SCManagementPack -ManagementPackFile $File
        break;
      }

      # Get MPB (bundles)
      {$_ -Match '\.mpb$'} {
        #Sometimes bundles cannot be retrieved
        Try {
          $MP = Get-SCManagementPack -BundleFile $File
        }Catch {
          Write-Host "Unable to retrieve file: [ $($File) ]. FullName.Length:[$($File.Length)]" -F Red
          break;
        }
      }
    }#end Switch
    $obj = New-Object PSCustomObject
    $obj | Add-Member -Type NoteProperty -Name Name -Value $MP.Name
    $obj | Add-Member -Type NoteProperty -Name DisplayName -Value $MP.DisplayName
    $obj | Add-Member -Type NoteProperty -Name Version -Value $MP.Version
    $obj | Add-Member -Type NoteProperty -Name FullName -Value $File
    $obj | Add-Member -Type NoteProperty -Name Description -Value $MP.Description
    $obj | Add-Member -Type NoteProperty -Name ID -Value $MP.ID.GUID
    $obj | Add-Member -Type NoteProperty -Name KeyToken -Value $MP.KeyToken
    $obj | Add-Member -Type NoteProperty -Name DefaultLanguageCode -Value $MP.DefaultLanguageCode
    $obj | Add-Member -Type NoteProperty -Name References -Value $MP.References

    Write-Verbose "$File"
    $Null = $arrMaster.Add($obj)
  }#end ForEach
  
  Return $arrMaster
}#End Function
#######################################################################

<#
    .Synopsis
    Will return the friendly name of a RunAs account.
    .DESCRIPTION
    Often times RunAs account SSIDs will appear in Operations Manager event logs which makes it difficult to determine which account is involved. This will correlate the friendly name with the SSID provided.
    .EXAMPLE
    PS C:\> Get-SCOMRunAsAccountName -SSID '0000F61CD9E515695ED4A018518C053E3CD87251D500000000000000000000000000000000000000'
    .Parameter -SSID
    The SSID as it is presented in the Operations Manager event log; an 80 character string of Hex digits [0-9A-F]
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Original Date: 2013.09.23
    History: I think I originally got most of this from a Technet blog: https://social.technet.microsoft.com/Forums/systemcenter/en-US/0b9bd679-a712-435e-9a27-8b3041cddac8/how-to-find-the-runasaccount-from-the-ssid?forum=operationsmanagergeneral
#>

Function Get-SCOMRunAsAccountName{
  param (
    [Parameter(
        Mandatory=$true,
    HelpMessage="Please enter the SSID")]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$SSID
  )

  Get-SCOMRunAsAccount | Sort-Object Name | ForEach-Object {
    $string = $null;$_.SecureStorageId | ForEach-Object {
      $string = $string + "{0:X2}" -f $_
    }

    $RunAsAccountName = $_.Name
    [string]$RunAsAccountSSID = $string
    If ($SSID -match $RunAsAccountSSID) {
      Write-Host "The Run As Account Name is: $RunAsAccountName"
    }
  }
}#end Function
#######################################################################


<#
    .DESCRIPTION
    Will output all RunAs accounts and any associated Security Profiles.
 
    .EXAMPLE
    Get-SCOMRunAsProfilesAccounts
 
    The above example will connect to localhost and output RunAs account and Seucrity Profile information.
 
    .EXAMPLE
    Get-SCOMRunAsProfilesAccounts -ManagementServer 'ms01.contoso.com'
 
    The above example will connect to the designated server and output RunAs account and Seucrity Profile information.
 
    .EXAMPLE
    Get-SCOMRunAsProfilesAccounts | Out-Gridview -Passthru
 
    The above example will display results in Gridview. Any selected rows can be passed through to the PowerShell Console for further review or manipulation.
 
    .INPUTS
    System.String
    .OUTPUTS
    Custom object
    .NOTES
    Author: Tyson Paul, derived from the works of Kevin Holman, Dirk Brinkman, Mihai (Sarbulescu?)
    Blog: https://monitoringguys.com/
    History:
    2020.02.25 - Revision of previous version (v1.0, Kevin Holman); improved efficiency, modified formatting.
    Ref: Kevin Holman: https://kevinholman.com/2017/07/27/document-your-scom-runas-account-and-profiles-script/
     
    .FUNCTIONALITY
    This function works best when piped to Out-Gridview or exported to CSV.
    .LINK
    Get-SCOMRunAsAccountName
#>

Function Get-SCOMRunAsProfilesAccounts {

  Param(
    [string]$ManagementServer = 'LOCALHOST'
  )


  ############# FUNCTIONS ###################
  Function Get-ClassfromInstance{
    [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
        SupportsShouldProcess=$true, 
        PositionalBinding=$false,
    ConfirmImpact='Medium')]
    [OutputType([tinyClass])]

    Param (
      $Instance
    )
    $class = (Get-SCOMClass -Id $instance.LeastDerivedNonAbstractMonitoringClassId)
    $tmpClass = [tinyClass]::new()
    $tmpClass.Id = $class.Id
    $tmpClass.Name = $class.Name
    If ($class.DisplayName.Length -ne 0) {
      $tmpClass.DisplayName = $class.DisplayName
    }
  
    Return $tmpClass
  }
  ########### END FUNCTIONS #################


  ########### BEGIN MAIN #################

  # My custom object
  Class Account {
    [string]$RunAsAccountName = ''
    [string]$Domain = ''
    [string]$Username = ''
    [string]$AccountType = ''
    [string]$ProfileName = ''
    [string]$ProfileDisplayName = ''
    [string]$TargetClassID = ''
    [string]$TargetClassName = ''
    [string]$TargetClassDisplayName = ''
    [string]$TargetID = ''
    [string]$TargetName = ''
    [string]$TargetDisplayName = ''
  }

  Class tinyClass {
    [string]$Id = ''
    [string]$Name = ''
    [string]$DisplayName = ''
  }

  # Main array for account collection
  [System.Collections.ArrayList]$AccountDataArray = @()

  # Load Modules and Connect to SCOM
  . Import-SCOMPowerShellModule

  # Remove any existing/default mgmt group connection
  Get-SCOMManagementGroupConnection | Remove-SCOMManagementGroupConnection

  # Connect to designated mgmt server (explicit)
  $MGConnection = New-SCManagementGroupConnection -ComputerName $ManagementServer
  
  $MG = Connect-OMManagementGroup -SDK $ManagementServer
  
  # Load Assembly and define ManagementGroup Object
  $CoreDLL = 'Microsoft.EnterpriseManagement.Core' 
  $null = [reflection.assembly]::LoadWithPartialName($CoreDLL) 
  $EMG = New-Object -TypeName Microsoft.EnterpriseManagement.EnterpriseManagementGroup -ArgumentList ($ManagementServer)

  #Process HealthService based Action Accounts Section
  #=======================================================
  ForEach ($RunAsProfile in (Get-SCOMRunAsProfile))
  {        
    # Get Health Service array associated with the profile
    $HSRef = $MG.GetMonitoringSecureDataHealthServiceReferenceBySecureReferenceId($RunAsProfile.ID)
    ForEach ($HS in $HSRef)
    {
      $TargetName = (Get-SCOMClassInstance -Id $HS.HealthServiceId).Displayname
      $MonitoringData = $HS.GetMonitoringSecureData()
      $tempAccount = [Account]::new()
      $tempAccount.RunAsAccountName = $MonitoringData.Name
      $tempAccount.Domain = $MonitoringData.Domain
      $tempAccount.Username = $MonitoringData.UserName
      $tempAccount.AccountType = $MonitoringData.SecureDataType
      $tempAccount.ProfileName = $RunAsProfile.Name
      $tempAccount.TargetID = $HS.HealthServiceId.Guid.ToString()
      $tempAccount.TargetName = $TargetName
      $tempAccount.TargetDisplayName = ''
      $tmpClass = Get-ClassfromInstance -Instance (Get-SCOMClassInstance -Id $HS.HealthServiceId)
      $tempAccount.TargetClassId = $tmpClass.Id
      $tempAccount.TargetClassName = $tmpClass.Name
      $tempAccount.TargetClassDisplayName = $tmpClass.DisplayName

      If ($null -ne $RunAsProfile.DisplayName ) 
      {
        $tempAccount.ProfileDisplayName = $RunAsProfile.DisplayName
      }    

      #$AccountDataArray += $tempAccount
      $null = $AccountDataArray.Add($tempAccount)
    }
  }
  #=======================================================
  # End ForEach RunAsProfile

  # Process all RunAsAccounts targeted at other targets
  #=======================================================
  #Get all RunAsAccounts
  $colAccounts = $EMG.Security.GetSecureData()

  #Loop through each RunAs account
  ForEach ($account in $colAccounts)
  {
    $secStorId = $account.SecureStorageId
    $stringBuilder = New-Object -TypeName System.Text.StringBuilder
    $secStorId | ForEach-Object {$null = $stringBuilder.Append($_.ToString('X2'))}
    $MPCriteria = "Value='{0}'" -f $stringBuilder.ToString()
    $moc = New-Object -TypeName Microsoft.EnterpriseManagement.Configuration.ManagementPackOverrideCriteria -ArgumentList ($MPCriteria)
    $overrides = $EMG.Overrides.GetOverrides($moc)

    If ($overrides.Count -eq 0)
    {
      $tempAccount = [Account]::new()
      $tempAccount.RunAsAccountName = $account.Name
      $tempAccount.Domain = $account.Domain
      $tempAccount.Username = $account.UserName
      $tempAccount.AccountType = $account.SecureDataType
      $tempAccount.ProfileName = 'No Profile Assigned'
    
      $null = $AccountDataArray.Add($tempAccount)
    }
    # Customizations/overrides exist for the account, process them all.
    Else
    {
      ForEach ($override in $overrides) 
      {
        # Every account will typically have these values
        $tempAccount = [Account]::new()
        $tempAccount.RunAsAccountName = $account.Name
        $tempAccount.Domain = $account.Domain
        $tempAccount.Username = $account.UserName
        $tempAccount.AccountType = $account.SecureDataType
        $tempAccount.TargetClassId = $override.Context.Id.Guid.ToString()
        $tmpClass = Get-SCClass -Id $tempAccount.TargetClassId
        $tempAccount.TargetClassName = $tmpClass.Name

        If ($null -ne $tmpClass.DisplayName) 
        {
          $tempAccount.TargetClassDisplayName = $tmpClass.DisplayName
        } 

      
        # If a ContextInstance (specific object) exists, add instance-specific data.
        If ($null -ne $override.ContextInstance)
        {
          $tempAccount.TargetID = $override.ContextInstance.Guid.ToString()
          $TargetClassInstance = Get-SCOMClassInstance -Id $tempAccount.TargetID
          $tempAccount.TargetName = $TargetClassInstance.Name
        
          If ($null -ne $TargetClassInstance.DisplayName) 
          {
            $tempAccount.TargetDisplayName = $TargetClassInstance.DisplayName
          }
        }
      
        $secRef = $EMG.Security.GetSecureReference($override.SecureReference.Id)
        $tempAccount.ProfileName = $secRef.Name
        If ($null -ne $secRef.DisplayName) 
        {
          $tempAccount.ProfileDisplayName = $secRef.DisplayName
        }
        
        # Add item to array
        $null = $AccountDataArray.Add($tempAccount)
      }
    } #End ForEach Override
  }
  #=======================================================
  # End ForEach account


  # Sort by RunAsAccountName. This is a nice touch for the user.
  Return ($AccountDataArray | Sort-Object -Property RunAsAccountName)

} #End Function
#######################################################################

<#
    .NOTES
    Name: Get-SCOMAgentWorkflows
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Version History:
    2022.06.09.1749 - v1
    Previously Get-SCOMRunningWorkflows was a standalone function for "running" workflows only.
 
    .DESCRIPTION
    Will output an array of objects containing helpful information on instances and the workflows running/failed for those instances.
 
    .EXAMPLE
    #
 
    #Example 1
    #All workflows (Running and Failed) are retrieved for the agent name provided. The results are saved in a variable, $r, which is piped to GridView.
    The function call is dot-sourced so that any subsequent calls to this function will execute faster.
 
    PS C:\> $r = . Get-SCOMAgentWorkflows -Status All -AgentName 'db01.contoso.com' -Verbose
    PS C:\> $r | Out-GridView
 
 
    #Example 2
    # Agent names matching the mask provided are piped to the function.
 
    PS C:\> (Get-SCOMAgent -DNSHostName *db*).DisplayName | Get-SCOMAgentWorkflows
 
 
    #Example 3
    # Results are sumarized by agents with failed workflows
    PS C:\> $result = Get-SCOMAgentWorkflows -AgentName 'ms02.contoso.com','db01.contoso.com','devdb01.contoso.com'
    PS C:\> $result | Where-Object Status -eq 'Failed' | Group-Object Agent | Select-Object @{Name="Failed";Expression={$_.Count}},Name
 
    Failed Name
    ------ ----
    14 devdb01.CONTOSO.COM
    9 DB01.CONTOSO.COM
 
 
 
    #Advanced examples for analyzing the workflow data
 
    # Storing results to XML, then consuming results from XML.
    PS C:\> Get-SCOMAgentWorkflows -AgentName 'ad01.contoso.com' | Export-Clixml -Path 'C:\Test\ad01.contoso.com.XML'
    $WFs = Import-Clixml -Path 'C:\Test\ad01.contoso.com.XML'
 
    # Total Instances
    $TOI = $WFs.instance.count
    Write-Host "Total object instances: $TOI"
 
    # Worflows types:
    $WFAT = $WFs.workflows.workflow | Group-Object Name | Measure-Object | Select-Object count -ExpandProperty count
    Write-Host "Worflows types: $WFAT"
 
    # Total WF instances running, all types:
    $TWFAT = $WFs.wfcount | Measure-Object -Sum | Select-Object sum -ExpandProperty sum
    Write-Host "Total WF instances running, all types: $TWFAT"
 
    # Workflows of category type
    $Category = 'EventCollection'
    $WFoCT = $WFs.workflows.workflow | ? {($_.Category -match $Category) -AND ($_.WriteActionCollection -notmatch 'Alert')}
    $WFoCT_sum = $WFoCT | Group-Object Name | Measure-Object | Select-Object count -ExpandProperty Count
    Write-Host "Workflows of category type [$($Category), Not Alert]: $WFoCT_sum"
 
    # Total WF instances of category type
    $TWFIoCT = $WFoCT | Group-Object Name | Measure-Object -sum count | Select-Object sum -ExpandProperty sum
    Write-Host "Total WF instances of category type [$($Category)]: $TWFIoCT"
 
    #Lists
    ###############################################
 
    # Classes
    $classes = Get-SCOMClass -ID (($wfs.InstanceName).LeastDerivedNonAbstractManagementPackClassId | Group-Object).Name
    Write-Host "$($Classes.count) class target types"
 
    # Class Names
    $Classes = ($wfs.InstanceName).LeastDerivedNonAbstractManagementPackClassId`
    | Group-Object `
    | Select-Object count `
    , @{Name="Class"; Expression = {(Get-SCOMClass -id $_.Name).Name}} `
    , @{Name="Id";Expression = {$_.Name}}, @{Name="Identifier"; Expression = {(Get-SCOMClass -id $_.Name).Identifier}} `
    | Sort-Object count `
    | Format-Table
 
    $Classes
    $allWorkflows = @{}
    # List of workflows, instance count, category
    $WFs.workflows.workflow | % {$allWorkflows[($_.Name)] = $_}
    $WFs.workflows.workflow | ? {($_.Category -match $Category) -AND ($_.WriteActionCollection -notmatch 'Alert')} | Group-Object Name | Select-Object count,name,@{Name='DisplayName';Expression = {$allWorkflows[($_.Name)].DisplayName}} ,@{Name='Category';Expression = {$Category}} | Sort-Object count | Format-Table
 
    # All workflow types with counts
    $WFs.workflows.workflow | Group-Object Name | Select-Object count,name,@{Name='DisplayName';Expression = {$allWorkflows[($_.Name)].DisplayName}} ,@{Name='Category';Expression = {$allWorkflows[($_.Name)].Category}} | Sort-Object count | Format-Table
 
#>


Function Get-SCOMAgentWorkflows {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
      SupportsShouldProcess=$false, 
      PositionalBinding=$false,
      HelpUri = 'https://monitoringguys.com/',
  ConfirmImpact='Medium')]
  Param(

    # FQDN of one or more agents
    [Parameter(Mandatory=$true, 
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
        Position=0,
    ParameterSetName='AgentName')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string[]]$AgentName,
    

    # Param1 help description
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
    ValueFromRemainingArguments=$false)]
    [ValidateSet('TasktResult', 'Detailed')]
    $OutputType = 'Detailed',
    
    
    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
    ValueFromRemainingArguments=$false)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [ValidateSet('Failed','Running','All')]
    $Status = 'All',


    # Seconds before function will abandon waiting for task to complete
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
    ValueFromRemainingArguments=$false)]
    [int]$TimeoutSeconds=600
  )
  
  BEGIN {
  
    $HSClass = Get-SCOMClass -Name "Microsoft.SystemCenter.HealthService"
    $AllHS = (Get-SCOMClassInstance -Class $HSClass)
    [System.Collections.ArrayList]$AllResults = @()
    [System.Collections.ArrayList]$TaskNames = @()
    [System.Collections.ArrayList]$Task = @()
    $thisStatus = @{
      "Microsoft.SystemCenter.GetAllRunningWorkflows" = "Running"
      "Microsoft.SystemCenter.GetAllFailedWorkflows" = "Failed"
    }
  
    $hashTasks = @{
      "Microsoft.SystemCenter.GetAllRunningWorkflows" = [System.Collections.ArrayList]@()
      "Microsoft.SystemCenter.GetAllFailedWorkflows" = [System.Collections.ArrayList]@()
    }
      
    Switch ($Status) {
      'All' {
        $NULL = $TaskNames.Add('Microsoft.SystemCenter.GetAllRunningWorkflows')
        $NULL = $TaskNames.Add('Microsoft.SystemCenter.GetAllFailedWorkflows')
      }
      'Failed' {
        $NULL = $TaskNames.Add('Microsoft.SystemCenter.GetAllFailedWorkflows')
    
      }
      'Running' {
        $NULL = $TaskNames.Add('Microsoft.SystemCenter.GetAllRunningWorkflows')
      }
    }
  
    $MinWait = 2
    $WaitSeconds = [math]::Min($MinWait,$TimeoutSeconds)
      
      
    ############################ FUNCTIONS ################################
  
    <#
        .NOTES
        Name: Get-SCOMAgentWorkflowsReport
        Author: Tyson Paul
        Blog: https://monitoringguys.com/
        Version History:
        2020.09.22.1433 - v1
 
        .DESCRIPTION
        Will output an array of objects containing helpful information on instances and the workflows running for those instances.
 
        .EXAMPLE
        #
        #Example
        PS C:\> $TaskResult = Get-SCOMAgentWorkflows -AgentName 'ad01.contoso.com','db01.contoso.com' -OutputType 'TasktResult'
        PS C:\> $TaskResult | Get-SCOMAgentWorkflowsReport
 
        .EXAMPLE
        #
        #Example
        PS C:\> Get-SCOMAgentWorkflowsReport -TaskResult (Get-SCOMAgentWorkflows -AgentName 'ad01.contoso.com' -OutputType TasktResult)
 
        .EXAMPLE
        #
        #Example
        PS C:\> Get-SCOMAgentWorkflowsReport -TaskResult $TaskResult
 
        .INPUTS
        Task result from 'Microsoft.SystemCenter.GetAllRunningWorkflows' agent task.
 
        .OUTPUTS
        [System.Object[]]
    #>

    Function Get-SCOMAgentWorkflowsReport {
      Param (
        [Parameter(Mandatory=$true, 
            ValueFromPipeline=$true,
            ValueFromPipelineByPropertyName=$false, 
            ValueFromRemainingArguments=$false, 
        ParameterSetName='Parameter Set 1')]
        [ValidateNotNull()]
        [ValidateNotNullOrEmpty()]  
        [System.Object[]]$TaskResult,
      
        [Parameter(Mandatory=$true,
            ValueFromPipeline=$false,
            ValueFromPipelineByPropertyName=$false, 
            ValueFromRemainingArguments=$false, 
        ParameterSetName='Parameter Set 1')]
        [ValidateNotNull()]
        [ValidateNotNullOrEmpty()]
        [ValidateSet('Failed','Running')]
        $Status 
      )
  
      BEGIN {
  
        ############################ FUNCTIONS ################################
        Function Build-Obj {
          Param (
            $WF,
            [string]$Type ='NotDefined'
          )
          $tmp = New-Object PSCustomObject
          $tmp | Add-Member -MemberType NoteProperty -Name 'Type' -Value $type
          $tmp | Add-Member -MemberType NoteProperty -Name 'Workflow' -Value $wf
          Return $tmp
        }
        ######################## END FUNCTIONS ################################
    
        Remove-Variable -Name 'Instances','rpt1' -ErrorAction Ignore
        [System.Collections.ArrayList]$arrReports = @()

        # Load all workflow types into one hash. dot-source this function to speed up multiple runs.
        If (-NOT $allWFTypes) {
          $allWFTypes=@{}
          $allClasses=@{}
          Write-Warning "Loading all workflows into cache object. This may require a few minutes. dot-source this function next time to speed up consecutive uses."
          ForEach ($item in @(Get-SCOMMonitor)) { $allWFTypes[($item.Name)] = ( Build-Obj -WF $item -Type Monitor) }
          ForEach ($item in @(Get-SCOMRule)) { $allWFTypes[($item.Name)] = ( Build-Obj -WF $item -Type Rule) }
          ForEach ($item in @(Get-SCOMDiscovery)) { $allWFTypes[($item.Name)] = ( Build-Obj -WF $item -Type Discovery) }
        
          # Build hash of all class types
          ForEach ($Class in (Get-SCOMClass) ){ $allClasses[$Class.Id.Guid] = $Class}        
        }
        Else {
          Write-Verbose "'allWFTypes' object [$($allWFTypes.Count)] already cached. 'allClasses' object already cached [$($allClasses.Count)]. Will not rebuild; use existing. This will save time."
        }
      }#end BEGIN
  
      #----------------------------------------------------------------
  
      PROCESS {
  
        ForEach ($Result in $TaskResult) {
          $ThisAgent = (Get-SCOMClassInstance -Id $Result.TargetObjectId)
          # Get the instance objects involved
          $Instances = (($Result.Output | ForEach-Object {[xml]$_}).DataItem)
          $hashInstances = @{}
          $Instances.Details.Instance | ForEach-Object {$hashInstances[($_.ID -Replace '{|}' ,'' )] = $_ }
  
          # Get basic workflow data: Instance ID (of target object), Workflow count, Workflow list/array
          $InstanceDetails = ([xml]($Result.Output)).DataItem.Details.Instance | Select-Object @{Name='Id';Expression={(($_.ID -Replace '{|}','').ToString())}}, @{Name='WFCount';Expression={$_.Workflow.Count} }, @{Name='Workflow';Expression={$_.Workflow | Sort-Object} } | Sort-Object WFCount,Workflow -Descending 

          $InstanceInfo = [ordered]@{}
          # Build an object with lots of data about each Instance, and the workflows running on its behalf
          ForEach ($Obj in $InstanceDetails) {
            $thisInstance = (Get-SCOMClassInstance -Id $Obj.ID )
            $objHash = [Ordered]@{
              WFCount = $obj.WFCount
              Status = $Status
              Agent = $ThisAgent
              InstanceID = $Obj.ID
              InstanceFullName = ($thisInstance.FullName)
              InstanceName = $thisInstance
              Class = $allClasses[($ThisInstance.LeastDerivedNonAbstractManagementPackClassId.Guid)]
              Workflows = [array]($obj.Workflow | ForEach-Object {$allWFTypes[$_] })
            }
            Try {
              $InstanceInfo[$objHash.InstanceFullName] = $objHash
            } Catch {
              Write-Warning "$_"
            }
          }

          # Report #1
          $rpt1 = ForEach ($key in @($InstanceInfo.Keys)) {
            $Key | Select-Object `
            -Property @{Name='WFCount';Expression={$InstanceInfo[$key].WFCount}} `
            ,@{Name='Status';Expression={$InstanceInfo[$key].Status}} `
            ,@{Name='Agent';Expression={$InstanceInfo[$key].Agent}} `
            ,@{Name='InstanceFullName';Expression={$InstanceInfo[$key].InstanceFullName}} `
            ,@{Name='InstanceName';Expression={$InstanceInfo[$key].InstanceName}} `
            ,@{Name='InstanceID';Expression={$InstanceInfo[$key].InstanceID}} `
            ,@{Name='Class';Expression={$InstanceInfo[$key].Class}} `
            ,@{Name='Workflows';Expression={$InstanceInfo[$key].Workflows}} `
          }
          $Null = $arrReports.Add($rpt1)
        }#end FOREACH
      }#end PROCESS
  
      #----------------------------------------------------------------
  
      END {
        Return $arrReports
      }#end END
    } #End Function Get-SCOMAgentWorkflowsReport
    ######################## END FUNCTIONS ################################
  
  } #End BEGIN



  PROCESS {
    $Instance = $NULL
  
    Try {
      Write-Verbose "Get instances of Health Service class with name(s): $AgentName"
        
      $NotAvailable = $AllHS | Where-Object {($_.Path -in $AgentName) -AND ($_.IsAvailable -eq $false )}
      ForEach ($UnavailableAgent in $NotAvailable) {
        Write-Warning "UNAVAILABLE: $($UnavailableAgent.DisplayName)"
      }
      $Instance = $AllHS | Where-Object {($_.Path -in $AgentName ) -AND ($_.IsAvailable -eq $true )}
      If (-NOT $Instance) {
        Throw "Agents not found or unavailable:`n$($AgentName)`n"
      }
    } Catch {
      Write-Warning $_
      Continue;
    }
  
    ForEach ($TaskName in $TaskNames) {
      $TaskWF = $NULL
      $TaskResult = $NULL
  
      Write-Verbose "Get SCOM task: $TaskName"
      $TaskWF = Get-SCOMTask -Name $TaskName
    
      Try {
        Write-Verbose "Starting the task now $(Get-Date) for [$($Instance.Count)] agents:"
        #$Instance.DisplayName | ForEach-Object {Write-Verbose $_}
        $Error.Clear()
        $thisTask = Start-SCOMTask -Instance $Instance -Task $TaskWF -Verbose:$([bool]($VerbosePreference -eq 'Continue')) -ErrorAction Stop
        $ThisTask | Where-Object {$_} | ForEach-Object {  $NULL = $hashTasks[$TaskName].Add($_) }
      } Catch {
        Write-Error "Failed to start task: $TaskName. $_`nExiting. "
        Continue;
      }
    } #End ForEach TaskName

  } #end PROCESS
    
  END {
    $ScriptTimer = [System.Diagnostics.Stopwatch]::StartNew()
      
    ForEach ($TaskName in @($hashTasks.Keys)) {
      # Keep looping while tasks are still running (Scheduled, Queued, etc. (other than Failed/Succeeded))
      Do {
        $secondsRemain = "{0:N0}" -F $($TimeoutSeconds - $ScriptTimer.Elapsed.TotalSeconds)
        Write-Verbose "`n$(Get-Date): Waiting for task(s) to complete. [$($secondsRemain)] seconds remain until timeout. Sleeping for $($WaitSeconds) seconds."
        Start-Sleep -Seconds $WaitSeconds
        If (-NOT $hashTasks[$TaskName].Id.Count) {
          Write-Warning "No tasks exist for workflow: $($TaskName)"
          Continue;
        }
        $Results = (Get-SCOMTaskResult -Id $hashTasks[$TaskName].Id)
      } While ((([regex]::matches(($Results.Status),'Succeeded').Count + [regex]::matches(($Results.Status),'Failed').Count) -LT ($Instance.Count)) -AND ($ScriptTimer.Elapsed.TotalSeconds -le $TimeoutSeconds) )
      $ScriptTimer.Stop()
  
      # If some task is not Status=Succeeded, show report
      ForEach ($TaskResult in ($Results | Where-Object {$_.Status -notmatch "Succeeded" }) ) {
        $ThisInstance = $Instance | Where-object {$_.Id.Guid -match $TaskResult.TargetObjectId.Guid} 
        Write-Warning "$($TaskName):$($ThisInstance.Path):$($TaskResult.Status)"
      } #end ForEach TaskResult
    
      If ($ScriptTimer.Elapsed.TotalSeconds -gt $TimeoutSeconds){
        Write-Warning "Timer expired. Check task status manually. TaskID: [$($Task.Id.Guid)]. Exiting"
      }
      Else {
        Switch ($OutputType) {
          'TasktResult' {
            $Results | Where-Object {$_} | ForEach-Object {$NULL = $AllResults.Add($_)}
          }
          'Detailed' {
            # dot-sourced to potentially reduce multiple loading of the workflow cache in this function
            (. Get-SCOMAgentWorkflowsReport -Status $thisStatus[$TaskName] -TaskResult $Results -Verbose:$VerbosePreference ) | Where-Object {$_} | ForEach-Object {$_ | ForEach-Object {$NULL = $AllResults.Add($_)} }
          }
        }
      }
      
    }
    Return $AllResults
  }
    
}#End Get-SCOMAgentWorkflows
  


#######################################################################

<#
    .SYNOPSIS
    This is used to generate a unique hash string from an ordinary string value.
 
    .NOTES
    Author: Tyson Paul
    Date: 2018.05.30
    Blog: https://monitoringguys.com/
    History:
    Adapted from http://jongurgul.com/blog/get-stringhash-get-filehash/
#>

Function Get-StringHash{
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
      SupportsShouldProcess=$true,
      PositionalBinding=$false,
      HelpUri = 'https://monitoringguys.com/',
  ConfirmImpact='Medium')]

  Param (
    [Parameter(Mandatory=$true,
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [String] $String,

    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false,
        ValueFromRemainingArguments=$false,
        Position=1,
    ParameterSetName='Parameter Set 1')]
    [ValidateSet("SHA", "SHA1", "MD5", "SHA256", "SHA384", "SHA512","MACTripleDES","RIPEMD160")]
    $Algorithm = "MACTripleDES" #SHA1 is default. https://docs.microsoft.com/en-us/dotnet/api/system.security.cryptography.hashalgorithm.create?view=netframework-4.8
  )

  $StringBuilder = New-Object System.Text.StringBuilder
  [System.Security.Cryptography.HashAlgorithm]::Create($Algorithm).ComputeHash([System.Text.Encoding]::UTF8.GetBytes($String))|ForEach-Object{
    [Void]$StringBuilder.Append($_.ToString("x2"))
  }
  $StringBuilder.ToString()
} #End Function
#######################################################################


<#
    .Synopsis
    Will create a graphical structure (.png file) that represents SCOM class taxonomy; all SCOM class attributes, properties, hosting relationships, and discovery relationships for a SCOM class.
     
    .DESCRIPTION
    This function will use the GraphViz package to produce a graph-like structure (.png file) that will represent one or more SCOM
    classes and it's full hierarchy which includes parents, attributes, properties, hosting relationships (both hosted and hosting), and any discoveries which are capable of discovering the related classes.
    Any number of SCOM class objects or names may be piped to the function.
 
    This function relies on the following modules. User will be prompted for permission to install these:
    GraphViz: from the Chocolatey repo. (https://www.graphviz.org/)
    PSGraph: https://github.com/KevinMarquette/PSGraph
 
    .EXAMPLE
    PS C:\> New-SCOMClassGraph -ClassName 'Microsoft.SQLServer.2012.DBEngine'
    .EXAMPLE
    PS C:\> New-SCOMClassGraph -ClassName 'Microsoft.SQLServer.2014.AlwaysOn.DatabaseReplica' -Caching:$False
 
    The above example will generate a new graph even if one already exists in the default storage directory.
    .EXAMPLE
    PS C:\> New-SCOMClassGraph -ClassName 'Microsoft.Windows.Server.6.2.LogicalDisk' -ShowDiscoveries:$false
 
    The above example will create a graph but will not include Discovery relationships.
    NOTE: If you are not seeing discovery data in your graphs it may be because the graphs are cached without discovery data.
    Try using the caching switch to force new graph files to be created. Discovery data will be included by default.
     
    Example: -Caching:$false
    .EXAMPLE
    PS C:\> Get-SCOMClass -Name *sql* | New-SCOMClassGraph -ShowGraph:$false
 
    The above example will create graph files for ALL SQL classes but will not open the graph files in the default application.
    Typically there are a tremendous number of SQL classes in the SQL management packs.
     
    .EXAMPLE
    PS C:\> New-SCOMClassGraph -ID 'ea99500d-8d52-fc52-b5a5-10dcd1e9d2bd'
     
    .EXAMPLE
    PS C:\> (Get-SCOMClassInstance -DisplayName "http://ms01.contoso.com:80/" ).getclasses().Name | New-SCOMClassGraph
 
    The above example will show the class graph for a specific instance of a class type.
     
    .EXAMPLE
    PS C:\> (Get-SCOMClassInstance -DisplayName "SQLEXPRESS" ).getclasses().Name | New-SCOMClassGraph
 
    The above example will show the class graph for a specific instance of a class type.
 
    .EXAMPLE
    PS C:\> New-SCOMClassGraph -ShowIndex
 
    The example above will display an html index page for all of the graph files. The html file will open with the default application.
 
    .EXAMPLE
    PS C:\> Get-SCOMClass | Select Name,Displayname, @{N='ManagementPackName';E={$_.Identifier.Domain[0]} }| Out-GridView -PassThru | New-SCOMClassGraph
 
    This example above is a much fancier way to select one or more class names with the help of GridView. This command will get all SCOM classes in the management group and present Name, DisplayName, and ManagementPackName in GridView for easy browsing, filtering, and selection by the user. Select one or more classes from the Grid View, click OK. The selected class name(s) gets piped into the function: New-SCOMClassGraph
 
    .EXAMPLE
    #Example Script
 
    $int=0
    $arr =@()
    ForEach ($class in (Get-SCOMClass)) {
    $obj = New-Object pscustomobject
    $obj | add-member -name 'Index' -Value $int -MemberType NoteProperty
    $obj | add-member -name 'Name' -Value ($Class.Name) -MemberType NoteProperty
    $obj | add-member -name 'DisplayName' -Value ($class.DisplayName) -MemberType NoteProperty
    $arr+=$obj
    $int++
    }
    $arr | Out-GridView -PassThru | New-SCOMClassGraph -Caching:$false -Combine
 
    This example will present a gridview list of classes for you to select. Multi-select supported to combine numerous classes on a single graph. Be careful, don't overdo it.
 
    .Parameter ClassName
    This is the name of a class. (not the DisplayName)
     
    .Parameter Class
    An Operations Manager class object.
 
    .Parameter ID
    ID of a class.
 
    .Parameter Caching
    This will enable/disable the use of preexisting graphs. If $false, a new graph will be created. If $true, the script will look for an existing graph for the class. If no graph exists, a new one will be created.
     
    .Parameter ShowGraph
    This allows the user to generate the graphs in the designated directory without opening them (with the default application associated to the .png file type)
     
    .Parameter ShowDiscoveries
    This will allow the user to omit the discovery relationships.
     
    .Parameter ShowIndex
    This will display an html index page for all of the graph files. The html file will open with the default application.
     
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
 
    History
    2020.12.30 - Improved GraphViz package detection.(Thanks, Mike N.)
    2020.12.08 - Added ability to sanitize class "Description" text; remove unwanted characters which can cause graphviz to puke.
    2020.06.19 - Added property types and additional info to classes and discovery nodes. Index will now get updated every run.
    2020.03.12 - Removed prompts to install GraphViz and Chocolatey packages. Instead provided suggestions on how to install them. Defeated Coronavirus with cigars and whiskey.
    2020.02.28 - Improved temp directory creation/verification
    2019.10.02 - Fixed Discovery nodes bug. Discoveries appear correctly now.
    2019.09.06 - Updated output dir to $env:Windir instead of $env:Temp
    2019.06.25 - Fixed -Combine flaw
    2019.06.24 - Added input field and search feature to index html
    2019.06.06 - Improved Combine functionality.
    Added -ShowIndex switch
    2019.04.29 - Added colored connected arrows to better identify relationships.
    2018.06.14 - Initial release.
 
    .LINK
    Get-SCOMMPFileInfo
    Get-SCOMClassInfo
#>

Function New-SCOMClassGraph {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
      SupportsShouldProcess=$false,
      PositionalBinding=$false,
      HelpUri = 'https://monitoringguys.com/',
  ConfirmImpact='Medium')]
  Param (

    [Parameter(Mandatory=$true,
        ValueFromPipeline=$true,
        ValueFromRemainingArguments=$false,
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [Microsoft.EnterpriseManagement.Configuration.ManagementPackClass[]]$Class,

    [Parameter(Mandatory=$true,
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$true,
        ValueFromRemainingArguments=$false,
        Position=0,
    ParameterSetName='Parameter Set 2')]
    [alias('Name', 'DisplayName')]
    [string[]]$ClassName,

    [Parameter(Mandatory=$true,
        ValueFromPipeline=$false,
        ValueFromRemainingArguments=$false,
        Position=0,
    ParameterSetName='Parameter Set 3')]
    [string[]]$ID,

    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 1')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 2')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 3')]
    [string]$outDir,

    # This will enable/disable the use of a previous graph file. Disable: will create a new/fresh graph file.
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 1')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 2')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 3')]
    [switch]$Caching=$false,

    # When combining multiple classes on the same graph the host node color might not appear correctly. However, any arrows will indicate the relationship type based on color. This will depend entirely on which classes get enumerated first.
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 1')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 2')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 3')]
    [switch]$Combine,

    # This will open the graph file with the default application (.png)
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 1')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 2')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 3')]
    [switch]$ShowGraph=$true,

    # This will include discovery nodes in the graph(s)
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 1')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 2')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 3')]
    [switch]$ShowDiscoveries=$true,

    # This will create/update, then open the index file with the default application (.html)
    [Parameter(Mandatory=$false,
        Position=0,
    ParameterSetName='Parameter Set 4')]
    [switch]$ShowIndex,

    # If foreign/old graphs exist in the output directory (but class no longer exists in this mgmt group), delete graph files.
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 1')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 2')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 3')]
    [Parameter(Mandatory=$false,
    ParameterSetName='Parameter Set 4')]
    [switch]$RemoveForeignGraphs=$false

  )

  Begin{
    #######################################################################

    [Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12
    . Import-SCOMPowerShellModule
    Import-Module PSGraph,PowerShellGet -ErrorAction SilentlyContinue

    If (-NOT ($PSVersionTable.PSVersion.Major -ge 5)){
      Write-Error "PowerShell version: $($PSVersionTable.PSVersion.Major) detected! "
      Write-Error "Upgrade to PowerShell version 5 or greater to use this function."
      Break
    }

    'OperationsManager' | ForEach-Object {
      If (-Not [bool](Get-Module -Name $_ -ErrorAction SilentlyContinue )) {
        Write-Error "Required module '$_' does not exist. Please install the module or run this function from a machine where the Operations Manager Console exists. Typically the required module will exists wherever the Console has been installed. Exiting."
        Break
      }
    }

    # Verify if GraphViz is present.
    If (-NOT ([bool](Get-Package -Name "GraphViz*" -ErrorAction SilentlyContinue))) {
      Write-Host "Required package 'GraphViz' does not exist." -F Red
      Try{
        Write-Host "`nIs 'Chocolately' Package Manager installed? ( https://chocolatey.org )" -F Gray
        (choco)[0]
        Write-Host "Yes!"

      }Catch {
        Write-Host "`n'Chocolatey' (The Package Manager for Windows at https://chocolatey.org) does not exist OR is outdated." -F Red
        Write-Host "Please install/update 'Chocolatey' to v0.10.15 or later from https://chocolatey.org ."  
        Write-host "`nOne way to install Chocolatey is with the following PowerShell commands:"
        Write-Host -F Cyan @'
Set-ExecutionPolicy Bypass -Scope Process -Force;
[System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072;
iex ((New-Object System.Net.WebClient).DownloadString('https://chocolatey.org/install.ps1'))
'@

      } #end Catch
      
      Write-Host "`nUse Chocolatey to install GraphViz with the following PowerShell command:" 
      Write-Host 'choco install graphviz -y' -F Cyan
      Write-Error "`nExiting." -ErrorAction Stop
    }
    
    # Make sure PSGraph is available/loaded
    # https://github.com/KevinMarquette/PSGraph
    If (-Not [bool](Get-Module -Name 'PSGraph' -ErrorAction SilentlyContinue  )) {
      Write-Error "Required module 'PSGraph' does not exist. Please install module."
      $choice = Read-Host "Install module: PSGraph (Y/N) ?"
      While ($choice -notmatch '[y]|[n]'){
        $choice = Read-Host "Y/N?"
      }
      Switch ($choice){
        'y' {
          # Install PSGraph from the Powershell Gallery
          Write-Host "Find-Module PSGraph | Install-Module -Verbose"
          Find-Module PSGraph -Verbose | Install-Module -Verbose
          If ($?){
            Write-Host "Module installed!"
            Import-Module PSGraph -Verbose
          }
          Else {
            Write-Error "Problem installing module.
              You may manually download from this location: 'https://github.com/KevinMarquette/PSGraph'. Additional information here: 'https://kevinmarquette.github.io/2017-01-30-Powershell-PSGraph/'.
            For information on how to install a PowerShell module, see this article: 'https://msdn.microsoft.com/en-us/library/dd878350(v=vs.85).aspx'.`nExiting.`n"
;
            Break
          }
        }
        'n' {
          Write-Host "Module will not be installed.`nExiting.`n"
          Break
        }
      }
    }

    ###############################################################
    Function Load-hashBasicMPInfo {
      # Load all MPs into hash. Will need the MP names below
      $MPs = Get-SCOMManagementPack
      $hashMPBasicInfo = @{}
      
      ForEach ($MP in $MPs) {
        $obj = New-Object PSCUSTOMOBJECT
        $obj | Add-Member -MemberType NoteProperty -Name IsSealed -Value $($MP.Sealed.ToString())
        If ($MP.DisplayName.Length -gt 0){
          $obj | Add-Member -MemberType NoteProperty -Name DisplayName -Value $($MP.DisplayName)
        }
        Else {
          $obj | Add-Member -MemberType NoteProperty -Name DisplayName -Value $($MP.Name)
        }
        $obj | Add-Member -MemberType NoteProperty -Name Version -Value $($MP.Version.ToString() )

        $hashMPBasicInfo.Add($MP.Name,$obj)
      }
      
      Return $hashMPBasicInfo
    }
    
    ###############################################################
    # This is a customized version of the original function located in the PSGraph module here: https://github.com/KevinMarquette/PSGraph
    Function Record {
      <#
          .SYNOPSIS
          Creates a record object for GraphViz.
          This is a customized version of the "Record" function that exists in the PSGraph module from Kevin Marquette.
          I added some additional parameters to control font styles.
      #>

      [OutputType('System.String')]
      [cmdletbinding(DefaultParameterSetName = 'Script')]
      param(
        [Parameter(
            Mandatory=$true,
            Position = 0
        )]
        [alias('ID', 'Node')]
        [string]
        $Name,

        [Parameter(
            Position = 1,
            ValueFromPipeline=$true,
            ParameterSetName = 'Strings'
        )]
        [alias('Rows')]
        [Object[]]
        $Row,

        [Parameter(
            Position = 1,
            ParameterSetName = 'Script'
        )]
        [ScriptBlock]
        $ScriptBlock,

        [Parameter(
            Position = 2
        )]
        [ScriptBlock]
        $RowScript,

        [string]
        $Label,

        # Added for color customization of header
        [string]$FONTCOLOR = 'white',
        [string]$BGCOLOR = 'black',

        # Added for color customization of table
        [string]$FILLCOLOR = 'white',
        [string]$STYLE = 'filled',
        [string]$TABLECOLOR = 'black'

      )
      begin
      {
        $tableData = [System.Collections.ArrayList]::new()
        if ( [string]::IsNullOrEmpty($Label) )
        {
          $Label = $Name
        }
      }
      process
      {
        if ( $null -ne $ScriptBlock )
        {
          $Row = $ScriptBlock.Invoke()
        }

        if ( $null -ne $RowScript )
        {
          $Row = foreach ( $node in $Row )
          {
            @($node).ForEach($RowScript)
          }
        }

        $results = foreach ( $node in $Row )
        {
          Row -Label $node
        }

        foreach ( $node in $results )
        {
          [void]$tableData.Add($node)
        }
      }
      end
      {
        #$html = '<TABLE CELLBORDER="1" BORDER="0" CELLSPACING="0"><TR><TD bgcolor="black" align="center"><font color="white"><B>{0}</B></font></TD></TR>{1}</TABLE>' -f $Label, ($tableData -join '')
        $html = '<TABLE CELLBORDER="1" COLOR="'+$TABLECOLOR+'" BORDER="0" CELLSPACING="0"><TR><TD bgcolor="'+$BGCOLOR+'" align="center"><font color="'+$FONTCOLOR+'"><B>{0}</B></font></TD></TR>{1}</TABLE>' -f $Label, ($tableData -join '')
        #Node $Name @{label = $html; shape = 'none'; fontname = "Courier New"; style = "filled"; penwidth = 1; fillcolor = "white"}
        Node $Name @{label = $html; shape = 'none'; fontname = "Courier New"; style = $STYLE; penwidth = 1; fillcolor = $FILLCOLOR}
      }
    }#End Function
    ###############################################################

    Function Dig-Class {
      [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
          SupportsShouldProcess=$true,
          PositionalBinding=$false,
      ConfirmImpact='Medium')]
      [Alias()]
      [OutputType([System.Object[]])]

      Param(
        [Parameter(Mandatory=$true,
            ValueFromPipeline=$true,
            ValueFromPipelineByPropertyName=$false,
            ValueFromRemainingArguments=$false,
            Position=0,
        ParameterSetName='Parameter Set 1')]
        [ValidateNotNull()]
        [ValidateNotNullOrEmpty()]
        [Microsoft.EnterpriseManagement.Configuration.ManagementPackClass]$Class,

        # Determines if this Class object is hosting another class. Will affect the shape/color
        [Parameter(Mandatory=$false,
            ValueFromPipeline=$false,
            Position=1,
        ParameterSetName='Parameter Set 1')]
        [switch]$IsHosting=$false,

        [System.Collections.Hashtable]$hashCollection=@{}
      )
      Begin {}
      Process{
        Write-Host $Class.Name -ForegroundColor Yellow
            
        $hashThisClass = @{}
        # If a base class exists, dig it
        If ([bool]$Class.Base.ID.Guid){
          $BaseClass = ($hashAllClassesID[$Class.Base.ID.Guid] )
          If (-NOT ($hashCollection[$BaseClass.Name])) {
            $hashCollection = (Dig-Class $BaseClass -hashCollection $hashCollection) 
          }
          Try{
            $hashCollection[$BaseClass.Name]["Edges"] += @{($Class.Name)=('child')} 
          }Catch{
            #Empty Catch is fine here
          }
        }

        # Make sure that if a class is hosting that its 'Hosting' flag gets set. This is important when multiple class targets are combined on a single graph.
        If (($IsHosting) -AND ($hashCollection[$Class.Name]) ) {
          $hashCollection[$Class.Name].Hosting = $True
        }
        
        # If a hosting class exists, dig it.
        If ($Class.Hosted){
          $hashGraph2= @{}
          $hostClass = $Class.FindHostClass()
          If (-NOT ($hashCollection[$hostClass.Name])) {
            $hashHosting = (Dig-Class -Class $hostClass -IsHosting -hashCollection $hashCollection)
            $hashGraph2 = (Merge-HashTables -hmaster $hashCollection -htnew $hashHosting)
            $hashCollection = $hashGraph2
          }
          Else {
            $hashCollection[$hostClass.Name].Hosting = $True
          }

          Try{
            $hashCollection[$hostClass.Name]["Edges"] += @{($Class.Name)=('hosted')}
          }Catch{
            #Empty Catch is fine here
          }
        }
        #endregion


        [System.Collections.ArrayList]$PropNames = @()
        If ([bool]$Class.DisplayName) {
          $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $s_ClasAttributeBold -thisString "DisplayName: $($Class.DisplayName.ToString())"))
        }

        $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $true -thisString "ID/GUID: $($Class.ID.Guid)"))
        $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $true -thisString "MP: $($Class.GetManagementPack().Name)"))
        $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $true -thisString "MPDisplayName: $(($Class.GetManagementPack()).DisplayName)"))
        $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $true -thisString "MPVersion: $(($Class.GetManagementPack()).Version)"))

        If ($Class.Hosted) {
          $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $true -thisString "Hosted: $($Class.Hosted.ToString())"))
        }
        Else {
          $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $s_ClasAttributeBold -thisString "Hosted: $($Class.Hosted.ToString())"))
        }

        If ([bool]$Class.Abstract) {
          $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $TRUE -thisString "Abstract: $($Class.Abstract.ToString())"))
        }
        Else{
          $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $s_ClasAttributeBold -thisString "Abstract: $($Class.Abstract.ToString())"))
        }

        If ([bool]$Class.Extension) {
          $NULL = $PropNames.Add((Stylize-Row -Color $c_ExtensionFont -Bold $s_ClasAttributeBold -thisString "Extension: $($Class.Extension.ToString())"))
        }
        Else {
          $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $s_ClasAttributeBold -thisString "Extension: $($Class.Extension.ToString())"))
        }
        
        $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $s_ClasAttributeBold -thisString "Singleton: $($Class.Singleton.ToString())"))
        $NULL = $PropNames.Add((Stylize-Row -Color $c_classAttribute -Bold $s_ClasAttributeBold -thisString "Description: $(Customize-StringLength -String (Sanitize-String -String $Class.Description) -WordCount $LineBreak)"))

        # Get all properties of the class
        $classProps = $Class.GetProperties() | Sort-Object
        $tmpProps = @()
        ForEach ($Prop in $classProps) {
          $n = $Prop | Select-Object -Property Name -ExpandProperty Name
          If ($Prop.Key -eq $true){
            $tmpProps += "<Font Color=`"Red`"><B>[$($Prop.Type)] $($n)</B></Font>"
          }
          Else {
            $tmpProps += "[$($Prop.Type)] $($n)"
          }
        }
        $PropNames += $tmpProps
        
        $hashStyling = Get-DefaultStyling
        $hashThisClass = [Ordered]@{
          "PropNames" = $PropNames
          "Abstract" = $Class.Abstract
          "Singleton" = $Class.Singleton
          "Hosted" = $Class.Hosted
          "Hosting" = $IsHosting
          "Styling" = $hashStyling
        }

        # Keep any previous dig results, specifically Edges already identified.
        If ([bool]($hashCollection[$Class.Name].Edges)){
          $hashThisClass.Edges = $hashCollection[$Class.Name].Edges
        }

        $hashCollection[$Class.Name] = $hashThisClass
        Return $hashCollection
      }#end Process
      End {}
    }
    ###############################################################

    Function Merge-HashTables {
      param(
        $hmaster,

        $htnew
      )
      $hmaster.keys | ForEach-Object {
        $key = $_
        If (-NOT $htnew.containskey($key)) {
          $htnew.Add($key,$hmaster.$key)
        }
        Else {
          ForEach ($E in @($hmaster[$_].Edges.Keys)) {
            Try{
              If (-NOT [System.Object]$htnew[$_].Edges.ContainsKey($E) ){
                [System.Object]$htnew[$_].Edges.Add($E,$hmaster[$_].Edges.$E) 
              } 
            }Catch {
              # Empty catch is fine here
            }
          }
        }
      }
      #$htnew = $htold + $htnew
      return $htnew
    }
    ###############################################################

    Function Stylize-Record {
      # Modify colors for any hosted/hosting records
      Param(
        $hashtable
      )
      $hashtable.Keys | ForEach-Object{
        If ($hashtable[$_].Abstract -eq $true){
          # Table Header
          $hashtable[$_].Styling.FONTCOLOR = $c_AbstHdrFont
          $hashtable[$_].Styling.BGCOLOR = $c_AbstHdrBG
          # Table/Rows
          $hashtable[$_].Styling.FILLCOLOR = $c_AbstFill
          $hashtable[$_].Styling.Style = $c_AbstStyle
          $hashtable[$_].Styling.TABLECOLOR = $c_AbstTable
        }
        
        If ($hashtable[$_].Singleton -eq $true){
          # Table Header
          $hashtable[$_].Styling.FONTCOLOR = $c_SingletonHdrFont
          $hashtable[$_].Styling.BGCOLOR = $c_SingletonHdrBG
          # Table/Rows
          $hashtable[$_].Styling.FILLCOLOR = $c_SingletonFill
          $hashtable[$_].Styling.Style = $c_SingletonStyle
          $hashtable[$_].Styling.TABLECOLOR = $c_SingletonTable
        }
        
        
        If ($hashtable[$_].Hosted -eq $true) {
          # Table Header
          $hashtable[$_].Styling.FONTCOLOR = $c_hostedHdrFont
          $hashtable[$_].Styling.BGCOLOR = $c_hostedHdrBG
          # Table/Rows
          $hashtable[$_].Styling.FILLCOLOR = $c_hostedFill
          $hashtable[$_].Styling.Style = $c_hostedStyling
          $hashtable[$_].Styling.TABLECOLOR = $c_hostedTable
        }

        If ($hashtable[$_].Hosting -eq $true){
          # Table Header
          $hashtable[$_].Styling.FONTCOLOR = $c_hostingHdrFont
          $hashtable[$_].Styling.BGCOLOR = $c_hostingHdrBG
          # Table/Rows
          $hashtable[$_].Styling.FILLCOLOR = $c_hostingFill
          $hashtable[$_].Styling.Style = $c_hostingStyle #"filled"
          #$hashtable[$_].Styling.TABLECOLOR = $c_hostingTable
        }

        # Make sure the Abstract classes retain their header bg color
        If ($hashtable[$_].Abstract -eq $true){
          # Table Header
          $hashtable[$_].Styling.BGCOLOR = $c_AbstHdrBG
        }

        If ($hashtable[$_].Discovery -eq $true){
          # Table Header
          $hashtable[$_].Styling.FONTCOLOR = $c_DiscHdrFont
          $hashtable[$_].Styling.BGCOLOR = $c_DiscHdrBG
          # Table/Rows
          $hashtable[$_].Styling.FILLCOLOR = $c_DiscFill
          $hashtable[$_].Styling.Style = $c_DiscStyle
          $hashtable[$_].Styling.TABLECOLOR = $c_DiscTable
        }
      }
      Return $hashtable
    }
    ###############################################################

    Function Stylize-Row {
      [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
          SupportsShouldProcess=$true,
          PositionalBinding=$false,
      ConfirmImpact='Medium')]
      [Alias()]
      [OutputType([System.Object[]])]
      Param(
        [Parameter(Mandatory=$true,
            ValueFromPipeline=$true,
            ValueFromPipelineByPropertyName=$false,
            ValueFromRemainingArguments=$false,
            Position=0,
        ParameterSetName='Parameter Set 1')]
        [ValidateNotNull()]
        [ValidateNotNullOrEmpty()]
        [System.String[]]$thisString,

        [string]$Color = 'black',
        [bool]$Bold = $false

      )
      Begin{}
      Process{
        ForEach ($string in $thisString){
          # Remove any existing styling
          $newString = ($String -replace '(<Font)|(Color="[a-zA-Z]*">)|(<B>)|(<\/B>)|(<\/Font>)','' )
          # Add styling
          $newString = $newString | Where-Object { (-not ([string]::IsNullOrEmpty($_))) } | ForEach-Object -Process {"<Font Color=`"$($Color)`"><B>$($_)</B></Font>"}
          If (-NOT $Bold){
            $newString = ($newString -replace '(<B>)|(</B>)','' )
          }
          Return $newString
        }
      }
      End{}
    }
    ###############################################################

    #Remove undesireable characters
    Function Sanitize-String {
      Param (
        [string]$String
      )

      $string = ($string -Replace '“|”', '"' )
      Return $string

    }
    ###############################################################


    ###############################################################
    # A simple function to add a newline at every Nth word
    Function Customize-StringLength {
      Param (
        [string]$String,
        [int]$WordCount = 5
      )
      
      $nthWord = 0
      $finalString = ""
      $spaceRegex = [Regex] " "
      ForEach ($item in ($spaceRegex.split($String))) {
        $nthWord++
        if (($nthWord % $WordCount) -eq 0){
          $item = "<br/>$item"
        }
        $finalString += $item + " "
      }
      $finalString.TrimEnd().TrimStart()
    }
    ###############################################################

    Function Get-DefaultStyling {
      # Default Record style
      $hashStyling = [Ordered]@{
        # Table Header
        FONTCOLOR = $c_defaultHdrFont
        BGCOLOR = $c_defaultHdrBG

        # Table/Rows
        FILLCOLOR = $c_defaultFill
        Style = $c_defaultStyle
        TABLECOLOR = $c_defaultTable
      }
      Return $hashStyling
    }
    ###############################################################

    Function Get-Discoveries {
      Param(
        [hashtable]$hashMaster,
        [hashtable]$hashDiscoveries = @{}
      )

      #$discHash = @{}
      $discHash = $hashDiscoveries.Clone()
      # Get only discoveries which potentially create classes (not relationships)
      $discs = $hashAllDiscoveries.Values | Where-Object { [bool]$_.DiscoveryClassCollection}

      ForEach ($disc in $discs) {
        If ($disc.DiscoveryClassCollection.Count -gt 1){
          $c=1
        }
        Else {
          $c=$null
        }

        ForEach ($discCollection in $disc.DiscoveryClassCollection) {
          $thisHash = @{}
          If ($hashMaster.containsKey([string]$discCollection.typeid.Identifier.path)) {
            $PropNames = [System.Collections.ArrayList]@()
            $Null = $PropNames.Add( (Stylize-Row -thisString "DisplayName: $($Disc.DisplayName)" -Color $c_DiscDisplayName -Bold $true) )
            $Null = $PropNames.Add( (Stylize-Row -thisString "TargetName: $([string]$disc.target.Identifier.Path)" -Color $c_DiscTarget -Bold $true) )
            
            If ($hashAllClasses[$disc.target.Identifier.Path].DisplayName) {
              $Null = $PropNames.Add( (Stylize-Row -thisString "TargetDisplayName: $([string]$hashAllClasses[$disc.target.Identifier.Path].DisplayName)" -Color $c_DiscTarget -Bold $true) )
            }
            Else {
              $Null = $PropNames.Add( (Stylize-Row -thisString "TargetDisplayName: " -Color $c_DiscTarget -Bold $true) )
            }

            $Null = $PropNames.Add( (Stylize-Row -thisString "EnabledByDefault: $([string]$Disc.Enabled)" -Color $c_DiscTarget -Bold $true) )
            $Null = $PropNames.Add( (Stylize-Row -thisString "ID/GUID: $([string]$Disc.ID.Guid)" -Color $c_DiscTarget -Bold $true) )

            $Null = $PropNames.Add( (Stylize-Row -thisString "MP: $([string]$Disc.Identifier.Domain[0])" -Color $c_DiscTarget -Bold $true) )
            
            If ($hashMPBasicInfo[$Disc.Identifier.Domain[0]].DisplayName) {
              $Null = $PropNames.Add( (Stylize-Row -thisString "MPDisplayName: $([string]$hashMPBasicInfo[$Disc.Identifier.Domain[0]].DisplayName)" -Color $c_DiscTarget -Bold $true) )
            }
            
            $Null = $PropNames.Add( (Stylize-Row -thisString "MPVersion: $([string]$hashMPBasicInfo[$Disc.Identifier.Domain[0]].Version)" -Color $c_DiscTarget -Bold $true) )
            $NULL = $PropNames.Add((Stylize-Row -thisString "Description: $(Customize-StringLength -String (Sanitize-String -String $Disc.Description) -WordCount $LineBreak)" -Color $c_DiscTarget -Bold $true ))

            ForEach ($Item in ($discCollection.propertycollection | Sort-Object PropertyID) ) {
              $tmpClassName = [string]$Item.TypeID.Identifier.Path[0]
              $PropertyName = [string]$Item.PropertyID
              $PropType = (($hashAllClasses[$tmpClassName]).PropertyCollection | Where-Object {$_.Name -eq $PropertyName}).Type.ToString()
              $Null = $PropNames.Add( (Stylize-Row -thisString "[$($PropType)] $PropertyName" -Color $c_DiscRow) )
            }


            $thisHash.Add('PropNames',$PropNames)
            $thisHash.'Edges' += @([string]$discCollection.typeid.Identifier.path)
            $thisHash.'Label' = $disc.Name
            $thisHash.'Discovery' = $true
            $thisHash.'Styling' = Get-DefaultStyling
            $c++
            Try{
              $discHash.Add(([string]$disc.Name+$c),$thisHash) 
            }Catch{ 
              #empty Catch is fine here
            }
          }
        }
      }
      Return $discHash
    }
    ###############################################################

    Function New-Graph {

      # If all classes are meant to be combined into a single graph, then the filename will be calculated with a hash function.
      If ($Combine) {
        $hashstring = Get-StringHash -String ($hashGraph.GetEnumerator() | Out-String)
        $outPath = (Join-Path $OutDir ("MD5Hash_$($hashstring)" +".png" ))
      }

      Graph {
        #region Classes
        ForEach ($ClassKey in @($hashGraph.Keys)) {
          # These will determine if Hosted/Hosting nodes appear in the Legend at the top of the .png
          If ([bool]$hashGraph[$ClassKey].Hosted) {$HostedExists = $true}
          If ([bool]$hashGraph[$ClassKey].Hosting) {$HostingExists = $true}
          If ([bool]$hashGraph[$ClassKey].Singleton) {$SingletonExists = $true}

          $params = @{
            Name = $ClassKey
            Row = $hashGraph[$ClassKey].PropNames
          }
          $params += $hashGraph[$ClassKey].Styling
          Record @params
          ForEach ($E in @($hashGraph[$ClassKey].Edges.Keys | Where-Object {$_.Length -gt 0}) ) {
            If ([bool]$hashGraph.$E.Hosted) {
              $EdgeStyle = "bold"
            }
            Else {
              $EdgeStyle = "bold"
            }          
            [string]$edgeColor = ($hashEdgeColor[$hashGraph[$ClassKey].Edges.$E])
            Edge -From $ClassKey -To $E @{color=$edgeColor;style=$EdgeStyle}
          }
        }
        #endregion Classes

        If ($ShowDiscoveries) {
          #region Discoveries
          $hashDiscoveries.Keys | ForEach-Object {
            $Params = @{
              Name = $_
              Label = $hashDiscoveries[$_].Label
              Row = $hashDiscoveries[$_].PropNames
            }
            $Params += $hashDiscoveries[$_].Styling
            Record @Params
            [string]$edgeColor = ($hashEdgeColor['Discovery'])
            Edge -From $_ -To ($hashDiscoveries[$_].Edges | Select-Object -Unique) @{color=$edgeColor}
          }
          #endregion Discoveries
        }

        #region Legend
        SubGraph -Attributes @{label='Legend'} -ScriptBlock {

          If ($HostedExists) {
            # Output Legend nodes
            Record -Name "Hosted Class" `
            -Row `
            @((Stylize-Row -thisString 'Class Attribute' -Color $c_classAttribute -Bold $s_ClasAttributeBold),`
              (Stylize-Row -thisString 'Key Property' -Color $c_Key -Bold $s_KeyBold),`
            'Property') `
            -BGCOLOR $c_hostedHdrBG `
            -FONTCOLOR $c_hostedHdrFont `
            -FILLCOLOR $c_hostedFill `
            -STYLE $c_hostedStyling `
            -TABLECOLOR $c_hostedTable
          }

          If ($HostingExists) {
            Record -Name "Hosting Class" `
            -Row `
            @((Stylize-Row -thisString 'Class Attribute' -Color $c_classAttribute -Bold $s_ClasAttributeBold),`
              (Stylize-Row -thisString 'Key Property' -Color $c_Key -Bold $s_KeyBold),`
            'Property') `
            -BGCOLOR $c_hostingHdrBG `
            -FONTCOLOR $c_hostingHdrFont `
            -FILLCOLOR $c_hostingFill `
            -STYLE $c_hostingStyle `
            -TABLECOLOR $c_hostingTable
          }

          If ($SingletonExists) {
            Record -Name "Group (Class)" `
            -Row `
            (Stylize-Row -thisString 'Class Attribute' -Color $c_classAttribute -Bold $s_ClasAttributeBold)`
            -BGCOLOR $c_SingletonHdrBG `
            -FONTCOLOR $c_SingletonHdrFont `
            -FILLCOLOR $c_SingletonFill `
            -STYLE $c_SingletonStyle `
            -TABLECOLOR $c_SingletonTable
          }

          Record -Name "Abstract Class" `
          -Row `
          @((Stylize-Row -thisString 'Class Attribute' -Color $c_classAttribute -Bold $s_ClasAttributeBold),`
            (Stylize-Row -thisString 'Key Property' -Color $c_Key -Bold $s_KeyBold),`
          'Property') `
          -BGCOLOR $c_AbstHdrBG `
          -FONTCOLOR $c_AbstHdrFont `
          -FILLCOLOR $c_AbstFill `
          -STYLE $c_AbstStyle `
          -TABLECOLOR $c_AbstTable

          Record -Name "Class (ordinary)" `
          -Row `
          @((Stylize-Row -thisString 'Class Attribute' -Color $c_classAttribute -Bold $s_ClasAttributeBold),`
            (Stylize-Row -thisString 'Key Property' -Color $c_Key -Bold $s_KeyBold),`
          'Property') `
          -BGCOLOR $c_defaultHdrBG `
          -FONTCOLOR $c_defaultHdrFont `
          -FILLCOLOR $c_defaultFill `
          -STYLE $c_defaultStyle `
          -TABLECOLOR $c_defaultTable

          If ($ShowDiscoveries -and ([bool]$hashDiscoveries.Count)) {
            Record -Name "Discovery" `
            -Row `
            @((Stylize-Row -thisString 'Basic Discovery Info' -Color $c_DiscTarget -Bold $true ),`
            (Stylize-Row -thisString 'Discovered Property' -Color $c_DiscRow ))`
            -BGCOLOR $c_DiscHdrBG `
            -FONTCOLOR $c_DiscHdrFont `
            -FILLCOLOR $c_DiscFill `
            -STYLE $c_DiscStyle `
            -TABLECOLOR $c_DiscTable
          }
        } #end SubGraph
        #endregion Legend

      } | Export-PSGraph -DestinationPath $outPath -ShowGraph:$ShowGraph
      #--------------------------------

    }
    #######################################################################
    <#
        .Synopsis
        Will create an html index file for all of the graphs present in the default directory.
        .DESCRIPTION
        This function will enumerate all of the .png files in the default ClassGraph directory and build an html file with links to all of the graphs.
 
        .EXAMPLE
        PS C:\> New-SCOMClassGraphIndex -indexFileName 'C:\SCOMgraphs\MySCOMClassGraphs.html'
        The above example will generate the index file at the location specified.
 
        .EXAMPLE
        PS C:\> New-SCOMClassGraphIndex
        The above example will simply generate a new index at the default location.
 
        .EXAMPLE
        PS C:\> New-SCOMClassGraphIndex -ShowIndex
        The above example will generate a new index at the default location and then open it with the default program.
 
 
        .Parameter Dir
        The output directory where the .png files should be located. This is also where the html index file will be stored.
 
        .Parameter filterExt
        The type of files to locate in the output directory. The index will be built from files of this type.
 
        .Parameter indexFileName
        The name of the index file to create.
 
        .Parameter ShowIndex
        This will open the index file with the default program.
 
        .NOTES
        Author: Tyson Paul
        Blog: https://monitoringguys.com/
 
        History
        2021.05.13.1034 - Updated New-SCOMClassGraphIndex; better handling of foreign graphs (graphs for which no class exists in the mgmt group)
        2019.06.03 - Published to SCOMHelper module
 
        .LINK
        New-SCOMClassGraph
 
    #>

    Function New-SCOMClassGraphIndex {
      Param (
        [string]$outDir,
        [string]$filterExt='png',
        [string]$indexFileName='ClassGraphIndex.html',
        [bool]$RemoveForeignGraphs,
        [switch]$ShowIndex
      )

      If (-Not $outDir ) {
        Write-Verbose "No directory provided for index file. "
        Return
      }
      If (-NOT(Test-Path -Path $outDir -PathType Container)) {
        Write-Verbose "No directory found for index file. "
        Return
      }

      $SearchIconFileName = 'searchicon.png'
      [System.Byte[]]$SearchIcon = @(137,80,78,71,13,10,26,10,0,0,0,13,73,72,68,82,0,0,0,21,0,0,0,21,8,6,0,0,0,169,23,165,150,0,0,0,1,115,82,71,66,0,174,206,28,233,0,0,0,4,103,65,77,65,0,0,177,143,11,252,97,5,0,0,0,9,112,72,89,115,0,0,18,116,0,0,18,116,1,222,102,31,120,0,0,0,2,98,75,71,68,0,255,135,143,204,191,0,0,0,9,118,112,65,103,0,0,1,42,0,0,1,41,0,80,22,101,49,0,0,0,37,116,69,88,116,100,97,116,101,58,99,114,101,97,116,101,0,50,48,49,51,45,48,52,45,49,48,84,48,54,58,53,57,58,48,55,45,48,55,58,48,48,142,65,137,81,0,0,0,37,116,69,88,116,100,97,116,101,58,109,111,100,105,102,121,0,50,48,49,51,45,48,52,45,49,48,84,48,54,58,53,57,58,48,55,45,48,55,58,48,48,255,28,49,237,0,0,0,25,116,69,88,116,83,111,102,116,119,97,114,101,0,119,119,119,46,105,110,107,115,99,97,112,101,46,111,114,103,155,238,60,26,0,0,0,17,116,69,88,116,84,105,116,108,101,0,115,101,97,114,99,104,45,105,99,111,110,194,131,236,125,0,0,2,42,73,68,65,84,56,79,165,148,73,171,234,64,16,133,43,237,172,32,142,224,74,17,87,46,149,44,4,17,197,127,237,74,112,4,5,193,165,162,162,91,39,156,16,231,251,238,41,210,33,209,168,240,238,7,33,73,119,245,169,234,170,234,86,126,126,33,3,247,251,157,166,211,41,173,86,43,58,30,143,60,230,118,187,41,24,12,82,34,145,224,239,111,152,68,7,131,1,141,199,99,114,58,157,36,132,32,69,81,120,28,38,143,199,131,174,215,43,197,98,49,202,100,50,60,254,14,93,180,211,233,208,102,179,33,135,195,193,19,136,24,66,0,226,118,187,157,191,49,14,155,98,177,200,255,86,176,40,34,156,205,102,28,161,20,75,165,82,20,14,135,57,98,56,27,141,70,60,7,241,219,237,70,129,64,128,84,85,213,100,204,40,191,6,63,149,74,133,60,30,15,139,33,103,249,124,94,155,54,211,235,245,104,189,94,179,240,233,116,162,92,46,199,226,207,8,20,5,17,2,136,190,19,4,217,108,150,109,97,135,20,32,255,86,8,84,25,91,196,214,176,229,111,164,211,105,182,197,154,229,114,169,141,154,17,104,27,20,2,222,35,145,136,54,252,158,80,40,196,182,88,35,187,226,25,161,21,159,145,45,244,9,68,104,196,82,84,54,51,4,183,219,45,127,127,2,54,50,74,32,91,205,136,192,73,129,55,155,205,198,109,243,141,225,112,200,66,16,181,170,60,16,241,120,156,46,151,11,111,11,253,215,239,247,181,169,87,224,20,61,43,11,155,76,38,181,25,51,194,235,245,82,52,26,101,65,68,48,159,207,169,217,108,114,63,74,118,187,29,117,187,93,154,76,38,220,82,136,18,15,214,89,161,31,211,106,181,170,159,24,164,3,78,100,17,16,25,210,131,7,230,50,167,216,97,185,92,214,143,182,196,116,161,180,219,109,46,4,162,121,238,4,152,193,41,222,16,193,60,156,226,146,41,149,74,228,114,185,52,203,39,81,176,88,44,248,164,32,119,82,24,139,253,126,63,31,14,116,11,156,227,45,35,62,159,207,124,193,224,168,131,23,81,35,48,6,198,40,0,118,211,106,181,94,132,11,133,2,249,124,190,207,162,159,216,239,247,212,104,52,76,194,184,100,16,241,127,139,130,195,225,64,245,122,93,23,70,154,80,232,63,137,2,220,29,181,90,77,191,233,208,41,127,22,5,200,39,250,24,157,160,170,42,253,3,11,167,101,180,126,138,179,206,0,0,0,0,73,69,78,68,174,66,96,130)

      # This is the cute, little 'search' icon that appears on the master class list search field
      $SearchIconPath = Join-Path $OutDir "CSS\$($SearchIconFileName)"
      If (-NOT(Test-Path -Path ($SearchIconPath) -PathType Leaf )) {
        New-Item -Path (Join-Path $OutDir "CSS") -ItemType Container -Force -ErrorAction SilentlyContinue | Out-Null
        $SearchIcon | Set-Content -Path $SearchIconPath -Encoding BYTE
      }

      # Load all classes into hash
      $SCOMClasses = Get-SCOMClass
      $hashClasses = @{}
      ForEach ($Class in $SCOMClasses) {
        $hashClasses.Add($Class.Name, $Class)
      }

      $head = @'
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
    <head>
        <title>Class Catalog</title>
    </head>
    <body>
 
<style type="text/css">
 
a { text-decoration: none; }
a:link, a:visited {
    color: blue;
}
a:hover {
    color: red;
}
 
.tg {border-collapse:collapse;border-spacing:0;}
.tg td{font-family:Arial, sans-serif;font-size:14px;padding:1px 6px;border-style:solid;border-width:1px;overflow:hidden;word-break:normal;border-color:black;}
.tg th{font-family:Arial, sans-serif;font-size:14px;font-weight:normal;padding:1px 6px;border-style:solid;border-width:1px;overflow:hidden;word-break:normal;border-color:black;}
.tg .defaultstyle{border-color:inherit;vertical-align:center;text-align:left}
.tg .headerstyle{font-weight:bold;font-size:100%;border-color:inherit;vertical-align:center}
.tg .columnNameStyle{border-color:inherit;vertical-align:center;text-align:center}
 
 
.tg .classnamestyle{font-weight:bold;border-color:inherit;vertical-align:center}
.tg .keystyle{font-weight:bold;color:#fe0000;border-color:inherit;text-align:right;vertical-align:center}
.tg .keypropertystyle{font-weight:bold;color:#fe0000;border-color:inherit;vertical-align:center}
.tg .propertystyle{color:#22771a;border-color:inherit;vertical-align:center;horizontal-align:center}
propertystyle
@media screen and (max-width: 767px)
 
{
    .tg {width: auto !important;}
    .tg col {width: auto !important;}
    .tg-wrap {overflow-x: auto;-webkit-overflow-scrolling: touch;}
}
 
#myInput {
    background-image: url('./CSS/searchicon.png'); /* Add a search icon to input */
    background-position: 10px 12px; /* Position the search icon */
    background-repeat: no-repeat; /* Do not repeat the icon image */
    width: 100%; /* Full-width */
    font-size: 16px; /* Increase font-size */
    padding: 12px 20px 12px 40px; /* Add some padding */
    border: 1px solid #ddd; /* Add a grey border */
    margin-bottom: 6px; /* Add some space below the input */
}
 
#myTable tr {
    /* Add a bottom border to all table rows */
    border-bottom: 1px solid #ddd;
}
 
#myTable tr.header, #myTable tr:hover {
    /* Add a grey background color to the table header and on hover */
    background-color: #f1f1f1;
}
 
</style>
 
<script>
function myFunction() {
  // Declare variables
  var input, filter, table, tr, td1,td2,td4,td5,td6, i;
  input = document.getElementById("myInput");
  filter = input.value.toUpperCase();
  table = document.getElementById("myTable");
  tr = table.getElementsByTagName("tr");
 
  // Loop through all table rows, and hide those who don't match the search query
  for (i = 0; i < tr.length; i++) {
    td1 = tr[i].getElementsByTagName("td")[1];
     td2 = tr[i].getElementsByTagName("td")[2];
     td4 = tr[i].getElementsByTagName("td")[4];
     td5 = tr[i].getElementsByTagName("td")[5];
     td6 = tr[i].getElementsByTagName("td")[6];
    if (td1) {
      if ( (td1.innerHTML.toUpperCase().indexOf(filter) > -1) || (td2.innerHTML.toUpperCase().indexOf(filter) > -1) || (td4.innerHTML.toUpperCase().indexOf(filter) > -1) || (td5.innerHTML.toUpperCase().indexOf(filter) > -1) || (td6.innerHTML.toUpperCase().indexOf(filter) > -1)) {
        tr[i].style.display = "";
      } else {
        tr[i].style.display = "none";
      }
    }
  }
}
</script>
 
 
<input type="text" id="myInput" onkeyup="myFunction()" placeholder="Search for names..">
 
<div class="tg-wrap">
    <table id="myTable" class="tg">
 
      <tr>
        <th class="headerstyle">#</th>
        <th class="headerstyle">Class Name</th>
        <th class="headerstyle">DisplayName</th>
        <th class="headerstyle">IsAbstract</th>
        <th class="headerstyle">MP DisplayName</th>
        <th class="headerstyle">MP Name</th>
        <th class="headerstyle">MP Version</th>
        <th class="headerstyle">MP IsSealed</th>
      </tr>
 
'@


      $Rows = ''
      $thisRow = ''
      [int]$i = 0
      $graphFiles = (Get-ChildItem -Path $outDir -Filter *.$($filterExt) -Exclude "*MD5Hash_*","searchicon.png" -Recurse )
      ForEach ($file in $graphFiles) {
        $className = $file.Name.Replace(".$($filterExt)",'')

        # increase row count
        $i++

        #region ForeignGraphDetected
        If ($SCOMClasses.Name -Notcontains $ClassName) {
          
          # If foreign/old graph exists (but class no longer exists in this mgmt group), delete graph file
          If ($RemoveForeignGraphs) {
            $file | Remove-Item -Force
            # Skip to next class
            Continue
          }

          # Keep foreign graph, add it to the list of recognized graphs.
          Else {
            $thisRow = @"
            <tr>
              <td class="defaultstyle">$($i)</td>
              <td class="defaultstyle"><a href="./$($file.Name)">$($className)</a></td>
              <td class="defaultstyle">N/A</td>
              <td class="defaultstyle">N/A</td>
              <td class="defaultstyle">N/A</td>
              <td class="defaultstyle">N/A</td>
              <td class="defaultstyle">N/A</td>
              <td class="defaultstyle">N/A</td>
            </tr>
 
"@

          }
        } #endregion ForeignGraphDetected

        # Class exists in mgmt group so add all related info to list.
        Else {
          Try{
            $thisRow = @"
            <tr>
              <td class="defaultstyle">$($i)</td>
              <td class="defaultstyle"><a href="./$($file.Name)">$($className)</a></td>
              <td class="defaultstyle">$($hashClasses[$className].DisplayName)</td>
              <td class="defaultstyle">$($hashClasses[$className].Abstract)</td>
              <td class="defaultstyle">$($hashMPBasicInfo.($hashClasses[$className].Identifier.Domain[0]).DisplayName)</td>
              <td class="defaultstyle">$($hashClasses[$className].Identifier.Domain[0])</td>
              <td class="defaultstyle">$($hashMPBasicInfo.($hashClasses[$className].Identifier.Domain[0]).Version)</td>
              <td class="defaultstyle">$($hashMPBasicInfo.($hashClasses[$className].Identifier.Domain[0]).IsSealed)</td>
            </tr>
 
"@

          }Catch {
            Write-Error $Error[0].Exception
          }
        }

        $Rows += $thisRow
      } #ForEach

      $end += @'
</table></div>
</body></html>
'@


      $indexFilePath = (Join-Path $outDir $indexFileName)
      Write-Verbose "Index File: $($indexFilePath)"
      ($head + $Rows + $end) | Set-Content $indexFilePath
      
      If ($ShowIndex) { 
        # Open the index file with default program
        & $indexFilePath
      }
    }

    #######################################################################

    $LineBreak = 5
    #region Styling/Colors
    $c_Key = 'Red'
    $s_KeyBold = $true

    $c_classAttribute = '#7503a5'#'#8e8a86'
    $s_ClasAttributeBold = $false #$true

    $c_ExtensionFont = 'green'

    $c_defaultHdrBG = 'black'
    $c_defaultHdrFont = 'white'
    $c_defaultFill = 'white'
    $c_defaultStyle = ''
    $c_defaultTable = 'black'

    $c_DiscHdrFont = 'white'
    $c_DiscHdrBG = '#206dea' #Blue
    $c_DiscFill = ''
    $c_DiscStyle = ''
    $c_DiscTable = '#002560' #DarkBlue

    $c_DiscRow = 'blue'
    $c_DiscDisplayName = '#0646ad'
    $c_DiscTarget = '#0646ad'

    $c_hostedHdrFont = '#d87d22' #'orange' #'#e0bc60'
    $c_hostedHdrBG = 'white' #gray
    $c_hostedFill = '' #'#e0bc60'
    $c_hostedStyling = ''
    $c_hostedTable = 'green' #'orange' #'#c66d00'

    $c_hostingHdrFont = '#cc6602'
    $c_hostingHdrBG = 'white'
    $c_hostingFill = '#ffd089'#'#f7be6a'
    $c_hostingStyle = 'filled'
    $c_hostingTable = 'black' #$c_hostingFill #'#c96d12'
    $hashEdgeColor = [ordered]@{
      Child = 'black'
      Discovery = $c_DiscDisplayName
      Hosted = $c_hostingHdrFont
    }

    $c_AbstHdrFont = '#706d72' #'gray'
    $c_AbstHdrBG = '#e9ccf9' 
    $c_AbstFill = ''
    $c_AbstStyle = ''
    $c_AbstTable = '#e9ccf9'
    
    $c_SingletonHdrFont = 'black' #'gray'
    $c_SingletonHdrBG = '#706d72' 
    $c_SingletonFill = ''
    $c_SingletonStyle = ''
    $c_SingletonTable = '#706d72'
    
    #endregion Styling/Colors

    # If multiple classes are meant to be combined on a single graph, the existence this variable will enable that functionality.
    If ($Combine){
      $hashGraph = @{}
    }

    # Ensure an output directory has been specified
    If (-Not [Bool]$outDir ) {
      $outDir = (Join-Path (Join-Path $env:Windir "Temp") (Join-Path 'SCOMHelper' 'New-SCOMClassGraph' ) )
    }

    # Attempt to create temp folder if needed
    If (-NOT (Test-Path -Path $OutDir -PathType Container)){
      Write-Host "Attempting to create temp folder: $OutDir" -F Gray
      Try{
        New-Item -ItemType Directory -Path $OutDir -Force -ErrorAction Stop
      }Catch{
        Write-Host "Failed to create temp directory. " -F Yellow -B Red
        Write-Host "Run this tool as administrator." -F Yellow -B Red
        Write-Host "If that doesn't work, create this folder path manually: [$($OutDir)]" -F Yellow -B Red
        $null = Read-Host "Press any key to exit." 
        Exit
      }
    }

    If ($ShowIndex) { 
      # Create a new index file, then open it
      New-SCOMClassGraphIndex -outDir $outDir -ShowIndex -RemoveForeignGraphs $RemoveForeignGraphs
      Return
    }

    # Build basic MP info hash
    $hashMPBasicInfo = Load-hashBasicMPInfo
    If ($ShowDiscoveries){
      . Load-SCOMCache -LoadDiscoveries
    }
    . Load-SCOMCache -LoadClasses
  }#end Begin


  Process{
    #$hashGraph = @{}
    $hashDiscoveries = @{}
    If ([Bool]$ID){
      #[System.Collections.ArrayList]$Class = @()
      $Class = New-Object -TypeName System.Collections.ArrayList
      #Get any matching keys(ID) from $hashAllClassesID
      ForEach ($thisID in $ID) {
        $thisID = $_ -Replace '{|}',''
        If ($hashAllClassesID[$thisID]) {
          $NULL = $Class.Add(($hashAllClassesID[$thisID]))
        }
      }
    }
    ElseIf ([Bool]$ClassName){
      [System.Collections.ArrayList]$Class = @()
      
      ForEach ($thisClassName in $ClassName) {
        If (-NOT ($hashAllClasses.ContainsKey($thisClassName)) ){
          Write-Host "WARNING: Class with name: [$($thisClassName)] does not exist!" -F Red -BackgroundColor Yellow
          continue
        }

        $NULL = $class.Add(($hashAllClasses[$thisClassName]))
      }
    }
    
    ForEach ($thisClass in $Class) {
      $HostedExists = $false
      $HostingExists = $false
      $SingletonExists = $false
      $fileNameBase = $thisClass.Name

      If (-NOT $Combine) {
        $outPath = (Join-Path $OutDir ("$fileNameBase" +".png" ))
      }
      # If the graph has already been created, no sense creating it again. Open exising file.
      If ($caching) {
        If ($Combine) {
          Write-Error "Cannot use -Combine:true -Caching:true together"
          Break
        }
        ElseIf (Test-Path -Path $outPath -PathType Leaf) {
          Invoke-Item $outPath
          Break
        }
      }


      If (-NOT $Combine){
        # This will clear the variable for every iteration, start a fresh graph of class
        $hashGraph = @{}

        #This will dig into the Class taxonomy and identify the entire family tree, storing all parents/properties/relationships into a hash table.
        $hashGraph = (Dig-Class -Class $thisClass )

        If ($ShowDiscoveries) {
          # This will clear the variable for every iteration, start a fresh graph of discoveries
          $hashDiscoveries = @{}
          #This will get any discoveries related to any of the classes
          $hashDiscoveries = Get-Discoveries -hashMaster $hashGraph 
        
          # This will stylize the Discovery nodes.
          $hashDiscoveries = Stylize-Record -hashtable $hashDiscoveries
        } 

        # This will stylize the class nodes (names, properties) based on their type (abstract, hosted, hosting, etc.)
        $hashGraph = Stylize-Record -hashtable $hashGraph
        # Will output individual graphs
        New-Graph 
      }
      Else {
        #This will dig into the Class taxonomy and identify the entire family tree, storing all parents/properties/relationships into a hash table.
        $hashGraph = (Dig-Class -Class $thisClass -hashCollection $hashGraph)
      }
    } #end ForEach Class

  }#end Process

  End{
    
    # Will output one graph to display all classes
    If ($Combine){ 
      If ($ShowDiscoveries) {
        #This will get any discoveries related to any of the classes.
        $hashDiscoveries = Get-Discoveries -hashMaster $hashGraph -hashDiscoveries $hashDiscoveries
        
        # This will stylize the Discovery nodes.
        $hashDiscoveries = Stylize-Record -hashtable $hashDiscoveries
      }
      # This will stylize the class nodes (names, properties) based on their type (abstract, hosted, hosting, etc.)
      $hashGraph = Stylize-Record -hashtable $hashGraph
      New-Graph 
    }

    If (-NOT $ShowIndex) { 
      # Create a new index file, then open it
      New-SCOMClassGraphIndex -outDir $outDir -RemoveForeignGraphs $RemoveForeignGraphs
    }
  }#end End

}#end New-SCOMClassGraph

#######################################################################

<#
    .DESCRIPTION
    Will create an instance group containing Windows Computers [Microsoft.Windows.Computer] objects and optionally include related Health Service Watchers [Microsoft.SystemCenter.HealthServiceWatcher].
 
    .EXAMPLE
    "ms01.contoso.com","db01.contoso.com" | New-SCOMComputerGroup -MPName "Accounting Team Computer/HSW Group 2020" -NameSpace "Accounting" -Verbose
 
    .EXAMPLE
    $FQDNs = (Get-SCOMAgent | Where-Object Name -like "*DB*").Name
    New-SCOMComputerGroup -MPName "My Database Computers-HSW Group" -ComputerName $FQDNs -NameSpace "DBSERVERS" -AddHealthServiceWatchers -Verbose -PassThru
 
    The above example will get all agent names which match the pattern and add the Computer objects as well as the corresponding Health Service Watcher objects.
 
    .INPUTS
    Accepts array of computer FQDN names (fully qualified domain name)
 
    .OUTPUTS
    SCOM Management Pack [Microsoft.EnterpriseManagement.Configuration.ManagementPack]
 
    .NOTES
    Script: New-SCOMComputerGroup
    Author: Tyson Paul (https://monitoringguys.com/2019/11/12/scomhelper/)
    Version History:
    2020.08.18.1333 - Added MP and Group display name parameters
    2020.08.11.1309 - Polished a bit. HSW objects are optional.
    2020.06.17 - v1.0
#>

Function New-SCOMComputerGroup {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
      SupportsShouldProcess=$false, 
      PositionalBinding=$false,
      HelpUri = 'http://www.microsoft.com/',
  ConfirmImpact='Medium')]
  [Alias()]
  [OutputType([Microsoft.EnterpriseManagement.Configuration.ManagementPack])]

  Param (
    # New MP custom name
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 1')]
    [string]$MPName='',

    # New MP custom DISPLAY name
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 1')]
    [string]$MPDisplayName='',


    # Name of computer to be added to the group (and associated Health Service Watcher instance)
    [Parameter(Mandatory=$true, 
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string[]]$ComputerName,

    # Name of mgmt server to use for SDK connection
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 1')]
    [string]$MgmtServerName,
    
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
    ParameterSetName='Parameter Set 1')]
    [string]$GroupName,

    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
    ParameterSetName='Parameter Set 1')]
    [string]$GroupDisplayName,

    # Namespace for group name. Example: "Lab" or "Accounting". This will affect only the hidden/ugly element "ID" in the MP.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 1')]
    [string]$NameSpace = "GROUP",

    # This will include corresponding Health Service Watcher objects in the group.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 1')]
    [switch]$AddHealthServiceWatchers,
    
    # This will return the new [Microsoft.EnterpriseManagement.Configuration.ManagementPack] object.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false, 
    ParameterSetName='Parameter Set 1')]
    [switch]$PassThru

  )

  Begin {

    ###################################### FUNCTIONS #######################################################
    ########################################################################################################
    Function Build-RefString {
      Param (
        $MP
      )
      Return ("$($MP.Name)$($MP.version.tostring())".Replace('.',''))
    }
    ########################################################################################################
    Function Clean-DisplayName {
      [OutputType([String])]
      Param(
        # Param1 help description
        [Parameter(Mandatory=$false, 
            ValueFromPipeline=$true,
            ValueFromPipelineByPropertyName=$false, 
            ValueFromRemainingArguments=$false, 
            Position=0,
        ParameterSetName='Parameter Set 1')]
        [string]$Name=''
      )
      Begin{}
      Process{
        Return ($Name -replace '[^a-zA-Z0-9. \-()]', '')
      }
      End {}
    }
    ########################################################################################################
    Function Clean-ID {
      [OutputType([String])]
      Param(
        # Param1 help description
        [Parameter(Mandatory=$false, 
            ValueFromPipeline=$true,
            ValueFromPipelineByPropertyName=$false, 
            ValueFromRemainingArguments=$false, 
            Position=0,
        ParameterSetName='Parameter Set 1')]
        [string]$Name=''
      )
      Begin{}
      Process{
        $tmp = $Name.Replace(' ','.').Replace('..','.').Replace('..','.')
        $tmp2 = ( ($tmp -replace '[^a-zA-Z0-9.]', '').Replace('..','.').Replace('..','.') )
        Return $tmp2
      }
      End{}
    }
    ########################################################################################################
    Function _LINE_ { 
      $MyInvocation.ScriptLineNumber 
    }
    ########################################################################################################
    ########################################################################################################
    Function Connect-MgmtGroup {
      # Connect to SCOM mgmt group
      Try{
        $MG = Get-SCOMManagementGroup -ErrorAction Stop -Verbose:$VerbosePreference
        If (-NOT $MG ) {
          Write-Verbose "No SCOMManagementGroupConnection"
          Throw
        }
        ElseIf($MgmtServerName) {
          If ($MG.ConnectionSettings.ServerName -ne $MgmtServerName){
            Write-Verbose "Existing MG connection, wrong server: [$($MG.ConnectionSettings.ServerName)]. Establish connection to new server: [$($MgmtServerName)]..."
            Throw
          }
        }
      } Catch {
        Try {
          $MG = Connect-OMManagementGroup -SDK $MgmtServerName
          If (-NOT $MG) {
            Throw "No SCOMManagementGroupConnection"
          }
        }Catch {
          Write-Error "$(_LINE_): Unable to establish connection to designated mgmt server: [$($MgmtServerName)]. Exiting."
          Exit
        }
      }
    }
    ########################################################################################################

    $DateStamp = (Get-Date -f "yyyyMMdd.hhmmss")
 
    # Clean names to remove invalid characters
    $CleanMPDisplayName = Clean-DisplayName -Name $MPDisplayName
    $CleanMPID = Clean-ID -Name $MPName
    If (-NOT $CleanMPID) {
      $CleanMPID = Clean-ID -Name $CleanMPDisplayName
    }

    If (-NOT $CleanMPID) {
      Write-Error "No valid MPName provided. Exiting."
      Return 
    }
    
    # Will try to accomodate both group Name and DisplayName
    If ($GroupDisplayName) {
      Write-Verbose "Group DisplayName provided. Will attempt to clean it..."
      $GroupDisplayName = $GroupDisplayName | Clean-DisplayName
    }
    $CleanGroupID = $GroupName | Clean-ID
    If (-NOT $CleanGroupID) {
      Write-Verbose "No valid GroupName. Attempt to use cleaned group DisplayName (if provided)..."
      $CleanGroupID = $GroupDisplayName | Clean-ID
    }
    If (-NOT $CleanGroupID) {
      $CleanGroupID = "$($CleanMPID).$($DateStamp).Group"
      Write-Verbose "No valid GroupName. Will use default group naming: [$($CleanGroupID)]"
    }

    Write-Verbose @"
New MP DisplayName provided by user:[$MPName]
DisplayName after being sanitized:[$($CleanMPDisplayName)]
MPID after being sanitized:[$($CleanMPID)]]
 
"@


    Write-Verbose "Attempt import of OperationsManager module"
    . Import-SCOMPowerShellModule
    
    Write-Verbose "Verify mgmt group connection..."
    . Connect-MgmtGroup

    Write-Verbose "Get necessary MP objects for building group definitions..."
    $Formula = ''
    # Get the necessary reference MPs
    $LibMp = Get-SCOMManagementPack -Name Microsoft.Windows.Library
    $InsMp = Get-SCOMManagementPack -Name Microsoft.SystemCenter.InstanceGroup.Library
    $SCmp = Get-SCOMManagementPack -Name Microsoft.SystemCenter.Library

    # Build the alias Strings
    $Libalias = Build-RefString -MP $LibMp
    $InsAlias = Build-RefString -MP $InsMp
    $SCAlias = Build-RefString -MP $SCmp
   
    # Check if the MP already exists if not we will create it
    $MPAlreadyExists = Get-SCOMManagementPack -Name $CleanMPID
    If($MPAlreadyExists)
    {
      Throw @"
 
Error: Management pack with name [$($MPName)] already exists.
New MP DisplayName provided by user:[$MPName]
DisplayName after being sanitized:[$($CleanMPDisplayName)]
MPID after being sanitized:[$($CleanMPID)]]
 
"@

      Try {$MG.Dispose()}Catch{<#Tidy up#>}
      Exit
    }

    [System.Collections.ArrayList]$arrWinComp = @()
    [System.Collections.ArrayList]$arrHSW = @()
  
    $WinCompClass = Get-SCOMClass -Name "Microsoft.Windows.Computer"
    $HSWClass = Get-SCOMClass -Name "Microsoft.SystemCenter.HealthServiceWatcher"
  }#end Begin

  Process {
    # Create collection of Computers and HSW objects
    ForEach ($CName in $ComputerName) {
      $WinComp = $NULL
      $Criteria = "DisplayName = '$CName'"
    
      $objCriteria = New-Object Microsoft.EnterpriseManagement.Monitoring.MonitoringObjectCriteria($Criteria ,$WinCompClass)
      $WinComp = $MG.GetMonitoringObjects($objCriteria)
      If (-NOT $WinComp) {
        Write-Warning "Computer: [$($CName)] not found! Verify that the fully qualified domain name exists."
        Continue
      }
      $CompID = $WinComp.Id.Guid
      Write-Verbose "Found Computer:`t[$($CName)], [$($CompID)]"
      $Null = $arrWinComp.Add(" <MonitoringObjectId>$CompID</MonitoringObjectId>")
    
      If ($AddHealthServiceWatchers) {
        $objCriteria = New-Object Microsoft.EnterpriseManagement.Monitoring.MonitoringObjectCriteria($Criteria ,$HSWClass)
        $HSW = $MG.GetMonitoringObjects($objCriteria)  
        $WatchID = $HSW.Id.Guid
        Write-Verbose "Found HSW:`t`t[$($CName)], [$($WatchID)]"
        $Null = $arrHSW.Add(" <MonitoringObjectId>$WatchID</MonitoringObjectId>")
      }
    }
  }#end Process

  End {
    # Add Windows Computer GUIDs to explicit membership rule
    $Formula += @"
  <MembershipRule>
  <MonitoringClass>`$MPElement[Name="$($Libalias)!Microsoft.Windows.Server.Computer"]`$</MonitoringClass>
  <RelationshipClass>`$MPElement[Name="$($InsAlias)!Microsoft.SystemCenter.InstanceGroupContainsEntities"]`$</RelationshipClass>
  <IncludeList>
  $arrWinComp
  </IncludeList>
  </MembershipRule>
"@

    # Add HSW GUIDs to explicit membership rule, if appropriate
    If ($AddHealthServiceWatchers) {
      $Formula += @"
 
  <MembershipRule>
  <MonitoringClass>`$MPElement[Name="$($SCAlias)!Microsoft.SystemCenter.HealthServiceWatcher"]`$</MonitoringClass>
  <RelationshipClass>`$MPElement[Name="$($InsAlias)!Microsoft.SystemCenter.InstanceGroupContainsEntities"]`$</RelationshipClass>
  <IncludeList>
  $arrHSW
  </IncludeList>
  </MembershipRule>
"@

    }

    $store = New-Object Microsoft.EnterpriseManagement.Configuration.IO.ManagementPackFileStore
  
    # Create the MP Object
    Write-Verbose "Create new MP object (virtual)..."
    $MP = New-Object Microsoft.EnterpriseManagement.Configuration.ManagementPack($CleanMPID, $CleanMPID, (New-Object Version(1, 0, 0)), $store)

    # Set the MP Display Name
    $MP.DisplayName = $CleanMPDisplayName

    # Add MP references
    Write-Verbose "Add reverences to MP object..."
    $LibRef = New-object Microsoft.EnterpriseManagement.Configuration.ManagementPackReference($MP, $LibMp.Name, $LibMp.KeyToken, $LibMp.Version)
    $InsRef = New-object Microsoft.EnterpriseManagement.Configuration.ManagementPackReference($MP, $InsMp.Name, $InsMp.KeyToken, $InsMp.Version)
    $SCRef = New-object Microsoft.EnterpriseManagement.Configuration.ManagementPackReference($MP, $SCmp.Name, $SCmp.KeyToken, $SCmp.Version)
    $MP.References.Add($Libalias,$LibRef)
    $MP.References.Add($InsAlias,$InsRef)
    $MP.References.Add($SCAlias,$SCRef)

    Try {
      # Save Changes
      $MP.AcceptChanges()
      Write-Verbose "MP object was verified successfully."
    } Catch {
      Throw "Failed to verify new MP content.`nError:$($_)`nExiting."
      Try {$MG.Dispose()}Catch{<#Tidy up#>}
      Exit
    }
  
    Try {
      # Import MP
      Write-Verbose "Importing new empty MP into management group: [$($MG.Name)]..."
      $MG.ImportManagementPack($mp)
    } Catch {
      Throw "Failed to import new MP.`nError:$($_)`nExiting."
      Try {$MG.Dispose()}Catch{<#Tidy up#>}
      Exit
    }  

    #TYSON $GroupID = "$($CleanMPID).Group"
    $GroupID = "$($CleanGroupID)"
    If ($GroupID -notmatch 'group$') {
      $GroupID = "$($GroupID).Group"
    }

    # Create Group Object and try to insert it into the MP.
    Write-Verbose "Create new group object: [$($GroupID)]..."
    $group = New-Object Microsoft.EnterpriseManagement.Monitoring.CustomMonitoringObjectGroup($NameSpace, $GroupID, $CleanMPID, $Formula)
    If ($GroupDisplayName) {
      $group.DisplayName = $GroupDisplayName
    }
    Try{
      Write-Verbose "Inserting new Group formula into MP..."
      $MP.InsertCustomMonitoringObjectGroup($Group)
      Write-Verbose "Success!"
    }
    Catch{
      Throw "Failed to add group formula to MP.`nError:$($_)`nExiting."
      Try {$MG.Dispose()}Catch{<#Try to tidy up#>}
      Exit
    }
  
    # Finally, get the new pack
    If ($PassThru) {
      Get-SCOMManagementPack -Name $CleanMPID 
    }

  }#end End
}#End Function

#######################################################################

<#
    .SYNOPSIS
    Will ping all entries in your HOSTS file (C:\WINDOWS\System32\drivers\etc\hosts) very quickly.
 
    .EXAMPLE
    PS C:\> Ping-AllHosts -Count 30
 
    Will ping all hosts a total of 30 times, pausing for the default 10 seconds inbetween iterations.
    .EXAMPLE
    PS C:\> Ping-AllHosts -t -DelayMS 30000
 
    Will continue to ping hosts waiting 30 seconds inbetween iterations.
    .EXAMPLE
    PS C:\> Ping-AllHosts -Count 10
 
    Will ping all hosts a total of 10 times.
    .NOTES
    Name: Ping-AllHosts
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Date: 2018.04.30
    History: v1.0
 
#>

Function Ping-AllHosts {
  Param (
    [long]$Count=1,
    [long]$DelayMS=10000,
    [switch]$t
  )

  $HOSTSfile = (Join-Path $env:SystemRoot '\System32\drivers\etc\hosts')
  If (-NOT (Test-Path $HOSTSfile -PathType Leaf) ) {
    Write-Error "Cannot find 'hosts' file at: $HOSTSFile . Exiting."
    Return
  }
  If ($t) {
    $Count = 9999999999999
    Write-Host "`$Count = $Count" -ForegroundColor Yellow

  }
  [long]$i=1
  While ($i -le $Count) {
    Write-Host "Attempt: $i. Remaining attempts: $($Count - $i)" -F Green
    # Parse the Hosts file, extract only the lines that do NOT begin with hash (#). Discard any leading tabs or spaces on all lines. Ignore blank lines
    $lines = (( (Get-Content $HOSTSfile).Replace("`t",'#').Replace(" ",'#').Replace("##",'#') | Where-Object {$_ -match '^[^#]'}))  | Where-Object {$_.Length -gt 1}

    $hash = @{}
    # Assign names and IPs to hash table
    $lines | ForEach-Object { $hash[$_.Split('#')[1]] = $_.Split('#')[0] }
    If (-Not [bool]$lines) {
      Write-Host "No valid hosts found in file. Exiting."
      Return
    }
    # Perform simultaneous address connection testing, as job.
    $job = Test-Connection -ComputerName ($hash.Keys | Out-String -stream) -Count 1 -AsJob

    Do{
      Write-Host 'Waiting for ping test to complete...' -F Yellow
      Start-Sleep 3

    }While($job.JobStateInfo.State -eq "Running")
    Receive-Job $job |
    Select-Object @{N="Source"; E={$_.PSComputerName}},
    @{N="Target"; E={$_.Address}},
    IPV4Address,
    @{N='Time(ms)';E={($_.'ResponseTime')} },
    @{N='Alive?';E={([bool]$_.'ResponseTime')} } |
    Sort-Object Target |
    Format-Table -AutoSize

    If ($i -eq $Count) {Return}
    $i++
    Write-Host "Sleeping $($DelayMS / 1000) Seconds before next ping test..."
    Start-Sleep -Milliseconds $DelayMS
  }
}#end Function
#######################################################################

<#
    .Synopsis
    Will remove obsolete aliases from unsealed management packs.
    .DESCRIPTION
    This will not alter the original file but rather will output modified versions to the designated output folder.
 
    .EXAMPLE
    PS C:\> Remove-SCOMObsoleteReferenceFromMPFile -inFile 'C:\Unsealed MPs\*.xml' -outDir 'C:\Usealed MPs\Modified MPs'
 
    .EXAMPLE
    PS C:\> Remove-SCOMObsoleteReferenceFromMPFile -inFile 'C:\Unsealed MPs\MyOverrides.xml' -outDir 'C:\Usealed MPs\Modified MPs'
    .EXAMPLE
    PS C:\> Remove-SCOMObsoleteReferenceFromMPFile -inFile (GetChildItem 'C:\Unsealed MPs\*.xml').FullName
 
    .EXAMPLE
    PS C:\> Get-ChildItem -Path 'C:\Unsealed MPs\*.xml' | Remove-SCOMObsoleteReferenceFromMPFile
 
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Version: 1.0
    Date: 20018.05.09
#>

Function Remove-SCOMObsoleteReferenceFromMPFile {
  [CmdletBinding()]

  Param (
    # Path to input file(s) (YourUnsealedPack.xml)
    [Parameter(Mandatory=$true,
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$true,
    Position=0)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [Alias("FullName")]
    [string[]]$inFile,

    # Path to output folder, to save modified .xml file(s)
    [Parameter(Mandatory=$false,
        ValueFromPipeline=$false,
    Position=1)]
    [string]$outDir
  )

  Begin{}
  Process{
    [System.Object[]]$paths = (Get-ChildItem -Path $inFile )
    ForEach ($FilePath in $paths){
      If (-not [bool]$outDir){
        $outDir = (Join-Path (Split-Path $FilePath -Parent) "MODIFIED")
      }
      If (-NOT (Test-Path -Path $outDir) ){
        New-Item -Path $outDir -ItemType Directory -Force -ErrorAction SilentlyContinue | Out-Null
      }
      Write-Host "Input File: " -NoNewline; Write-Host $FilePath -F Cyan
      Try{
        [xml]$xml = Get-Content -Path $FilePath
        $References = $xml.ManagementPack.Manifest.References
      }Catch{
        Continue
      }
      If (-NOT [bool]$References) {Continue}
      $References.SelectNodes("*") | ForEach-Object {
        # If neither the Alias or ID of the referenced MP is found within the body of the MP, remove the reference.
        If (-NOT (($xml.InnerXml -match ("$($_.Alias)!")) -or ($xml.InnerXml -match ("$($_.ID)!"))) ) {
          Write-Host "Obsolete Reference Removed: $($_.Alias) : $($_.ID)" -F Green
          $References.RemoveChild($_) | Out-Null
        }
      }
      $outFile = (Join-Path $outDir (Split-Path $FilePath -Leaf))
      Write-Host "Saving to file: " -NoNewline; Write-Host $outFile -F Yellow
      $Xml.Save($outFile)
    }
  }
  End{}
}# END FUNCTION
#######################################################################

<#
    .Synopsis
    Will standardize all of the aliases in one or more unsealed (.xml) management pack files.
    .DESCRIPTION
    This script will inventory all of the MP reference IDs that can be found in the single file or set of .XML files specified by the InputPath. It will create a customized set of condensed alias acronyms to be standardized across all of the unsealed management packs. It will then replace the aliases in the Manifest as well as throughout the elements where the aliases are used.
    .Parameter InputPath
    This can be a directory or full path to an unsealed management pack .xml file.
    .EXAMPLE
    PS C:\> Set-SCOMMPAliases -InputPath 'C:\UnSealedMPs\'
    .EXAMPLE
    PS C:\> Set-SCOMMPAliases -InputPath 'C:\UnSealedMPs\MyCustom.Monitoring.xml'
 
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Date: 2017/12/13
    History:
    2017/12/13: First version
#>

Function Set-SCOMMPAliases {
  Param (
    [Parameter(Mandatory=$true)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [ValidateScript({Test-Path -Path $_})]
    [string]$InputPath #Either a folder or full file path
  )

  ##### TESTING ######
  #$InputPath = 'C:\Test\Test\TempTest'
  ##### TESTING ######

  # This will be used to seed the reference catalog with some hardcoded values for some common aliases
  $seed =  @'
Microsoft.SystemCenter = SC
Microsoft.SystemCenter.Apm.Infrastructure = APMInf
Microsoft.SystemCenter.Apm.Infrastructure.Monitoring = APMInfMon
Microsoft.SystemCenter.Apm.Library = APMLib
Microsoft.SystemCenter.Apm.NTServices = APMNT
Microsoft.SystemCenter.Apm.Wcf = APMWcf
Microsoft.SystemCenter.Apm.Web = APMWeb
Microsoft.SystemCenter.DataWarehouse.ApmReports.Library = APMReports
Microsoft.SystemCenter.InstanceGroup.Library = SCIGL
Microsoft.SystemCenter.Internal = SCInt
Microsoft.SystemCenter.Library =SCLib
Microsoft.SystemCenter.WebApplication.Library = WebAppLib
Microsoft.SystemCenter.WebApplicationSolutions.Library = WebAppSolLib
Microsoft.SystemCenter.WebApplicationSolutions.Library.Resources.ENU = WebAppLibRes
Microsoft.SystemCenter.WebApplicationTest.External.Library = WebAppTestExtLib
Microsoft.SystemCenter.WebApplicationTest.Library = WebAppTestLib
Microsoft.Windows.InternetInformationServices.2003 = IIS2003
Microsoft.Windows.Library = Windows
Microsoft.Windows.Server.Library = WinSrvLib
System.Health.Library = Health
System.Library = System
System.NetworkManagement.Library = NetManLib
System.NetworkManagement.Monitoring = NetManMon
System.NetworkManagement.Reports = NetManRpt
System.NetworkManagement.Templates = NetManTmp
System.Performance.Library = Performance
'@

  # $seed -split "`r`n" | Sort-Object

  ##############################################################
  # Will replace common/popular namespaces with smaller, friendly acronym
  Function AcronymCommonNames {
    Param (
      [Parameter(Mandatory=$true)]
      [ValidateNotNull()]
      [ValidateNotNullOrEmpty()]
      [string]$thisString
    )
    $acronyms = ConvertFrom-StringData -StringData @"
Microsoft.SQLServer = MSQL
Microsoft.Windows.Server = MWS
Microsoft.Windows.InternetInformationServices = IIS
"@


    If ( $acronyms.ContainsKey($thisString) ) {
      $arrNames = $acronyms.GetEnumerator().Name
      $arrNames | ForEach-Object {
        #Write-Host $_.Name $_.value
        $thisString = $thisString.Replace(($_),($acronyms.$_) )
      }
    }
    Return $thisString
  }


  ##############################################################
  Function Create-Alias {
    Param(
      [Parameter(Mandatory=$true)]
      [ValidateNotNull()]
      [ValidateNotNullOrEmpty()]
      [string]$Seed,

      [Parameter(Mandatory=$true)]
      [ValidateNotNull()]
      [ValidateNotNullOrEmpty()]
      # array of existing values, to prevent duplicates
      [system.Object[]]$Existing
    )

    $array = AcronymCommonNames -thisString $Seed
    $array = $array.Split('.')
    $newAlias = $array[0].Replace('Microsoft','').Replace('Windows','Win').Replace('HewlettPackard','HP')
    For($i = 1; $i -lt $array.Count; $i++) {
      $fragment = $array[$i]
      If ( $fragment -match '\d+' ) {
        $newAlias += $fragment
      }
      ElseIf ( $fragment -match '^ApplicationMonitoring.?' ) {
        $newAlias += 'AppMon'
      }
      ElseIf ( $fragment -match '^Dashboard.?' ) {
        $newAlias += 'Dash'
      }
      ElseIf ( $fragment -match '^Discovery.?' ) {
        $newAlias += 'Disc'
      }
      ElseIf ( $fragment -match '^Library.?' ) {
        $newAlias += 'Lib'
      }
      ElseIf ( $fragment -match '^Linux.?' ) {
        $newAlias += 'Linux'
      }
      ElseIf ( $fragment -match '^Monitor.?' ) {
        $newAlias += 'Mon'
      }
      ElseIf ( $fragment -match '^NetworkManagement.?' ) {
        $newAlias += 'NetMan'
      }
      ElseIf ( $fragment -match '^Network.?' ) {
        $newAlias += 'Net'
      }
      ElseIf ( $fragment -match '^Report.?' ) {
        $newAlias += 'Rpt'
      }
      ElseIf ( $fragment -match '^Server.?' ) {
        $newAlias += 'Ser'
      }
      ElseIf ( $fragment -match '^System.?' ) {
        $newAlias += 'Sys'
      }
      ElseIf ( $fragment -match '^Template.?' ) {
        $newAlias += 'Tmp'
      }
      ElseIf ( $fragment -match '^Unix.?' ) {
        $newAlias += 'Unix'
      }
      ElseIf ( $fragment -match '^Visual.?' ) {
        $newAlias += 'Vis'
      }
      ElseIf ( $fragment -match '^Windows.?' ) {
        $newAlias += 'Win'
      }
      Else {
        # Use capitalized letters in MP name to build the acronym
        $tempchar = ''
        [char[]]$fragment | ForEach-Object {
          If ([char]::IsUpper($_) ){$tempchar+=$_ }
        }
        # $newAlias += $fragment[0]
        $newAlias += $tempchar
      }
    }
    $i = 2
    $tempAlias = $newAlias
    While ($Existing -contains ($tempAlias)) {
      $tempAlias = $newAlias +"$i"
      $i++
    }

    Return $tempAlias
  }

  ##############################################################
  # This function will replace all instances of existing aliases with the newly customized aliases.
  Function Standardize-AllReferenceAliases {
    Param(
      [Parameter(Mandatory=$true)]
      [ValidateNotNull()]
      [ValidateNotNullOrEmpty()]
      [System.Object[]]$MPPaths,

      [Parameter(Mandatory=$true)]
      [ValidateNotNull()]
      [ValidateNotNullOrEmpty()]
      [Hashtable]$Catalog
    )
    ForEach ($MPPath in $MPPaths){

      $mpxml = [xml](Get-Content $MPPath)
      $arrReferences = $mpxml.GetEnumerator().Manifest.references.reference
      [int]$i = 0
      $content = (Get-Content $MPPath)
      # Replace aliases wherever used in xml
      ForEach ($ref in ($arrReferences) ) {
        <# Aliases typically appear in two scenarios...
            1) Examples: referring to some element property:
            <Property>$MPElement[Name="Windows!Microsoft.Windows.Computer"]/PrincipalName$</Property>
            <MonitoringClass>$MPElement[Name="SCLib!Microsoft.SystemCenter.ManagedComputer"]$</MonitoringClass>
            <RelationshipClass>$MPElement[Name="SCIGL!Microsoft.SystemCenter.InstanceGroupContainsEntities"]$
        #>

        $aliasrefbang = '="'+"$($ref.Alias)"+"!"
        $newAlias = ('="'+($Catalog.($ref.ID))+'!')
        $content = ($content -Replace ([regex]::Escape($aliasrefbang)), ([regex]::Escape($newAlias)) )

        <#
            2) Examples: referring to a workflow, type, datasource, etc.:
            <DataSource ID="GroupPopulationDataSource" TypeID="SCLib!Microsoft.SystemCenter.GroupPopulator">
            <DiscoveryRelationship TypeID="SCIGL!Microsoft.SystemCenter.InstanceGroupContainsEntities" />
        #>

        $aliasrefbang = '>'+"$($ref.Alias)"+"!"
        $newAlias = ('>'+($Catalog.($ref.ID))+'!')
        $content = ($content -Replace ([regex]::Escape($aliasrefbang)), ([regex]::Escape($newAlias)) )
        #$content | Set-Content $MPPath -Force -Encoding UTF8
      }

      # Reload the XML because it was just rewritten
      #$mpxml = [xml](Get-Content $MPPath)
      $mpxml = [xml]($content)
      $arrReferences = $mpxml.GetEnumerator().Manifest.references.reference
      [int]$i = 0
      # Replace aliases in Manifest
      ForEach ($ref in ($arrReferences) ) {
        $ref.Alias = $Catalog.($ref.ID)
        $i++
      }

      Write-Host "`t$i references updated! [ " -NoNewline -b Black -f Cyan
      Write-Host $(Split-Path -Path $MPPath -Leaf) -NoNewline
      Write-Host " ]" -b Black -f Cyan
      $mpxml.Save($MPPath)
    }
  } #endFunction

  ##############################################################
  <# This function is designed to append (or trim) a unique hash value to/from the aliases
      (in the Manifest as well as elsewhere in the management pack). This is necessary in the rare cases where existing
      aliases already match my customized new condensed acronyms in the $cat (catalog). "Windows!" is a good example
      of this. I've encountered random MPs which have the alias, "Windows" already assigned to a reference. This becomes
      problematic when I attempt to change references to the 'Microsoft.Windows.Library' after which the file ends up with
      multiple identical alias references (to different sealed MPs) using the exact same alias, "Windows". This is a clever way
      to make sure that all aliases are unique (no conflicts). This function should be run twice on the catalog; Once to
      append the unique string to aliases in the catalog (after the cat has been generated, of course), then modify the files.
      Then run again to set aliases to their final, desired values, then modify the files again to their final desired values.
  #>

  Function Modify-Aliases {
    Param(
      [Hashtable]$cat,
      [Switch]$Localized
    )
    $hash = Get-StringHash $env:COMPUTERNAME
    If ($Localized){
      @($cat.Keys) | ForEach-Object {
        $Cat[$_] = $cat[$_]+$hash
      }
    }
    Else{
      @($cat.Keys) | ForEach-Object {
        $Cat[$_] = [string]$cat[$_] -Replace $hash, ""
      }
    }
    Return $cat
  }

  ##############################################################
  #http://jongurgul.com/blog/get-stringhash-get-filehash/
  Function Get-StringHash
  {
    
    param
    (
      [string]
      $String,

      [string]
      $HashName = "MD5"
    )
    $StringBuilder = New-Object System.Text.StringBuilder
    [System.Security.Cryptography.HashAlgorithm]::Create($HashName).ComputeHash([System.Text.Encoding]::UTF8.GetBytes($String))|ForEach-Object{
      [Void]$StringBuilder.Append($_.ToString("x2"))
    }
    $StringBuilder.ToString()
  }

  #-------------------------------------------------------------
  #################### MAIN ################################


  ##############################################################
  # Create catalog of all existing references/aliases
  # The idea here is to seed the catalog with common pack names.
  # For other MPs dynamically create aliases based on doted name structure
  ##############################################################
  $cat = ConvertFrom-StringData -StringData $seed

  If (Test-Path -Path $InputPath -PathType Container -ErrorAction SilentlyContinue) {
    $MPPaths = (Get-ChildItem $InputPath -Recurse -Include *.xml -File ).FullName | Sort-Object
    Write-Host "$($MPPaths.Count) files found..." -F Yellow -B Black
  }
  ElseIf ((Test-Path -Path $InputPath -PathType Leaf -ErrorAction SilentlyContinue) -and ($InputPath -like "*.xml") ) {
    $MPPaths = $InputPath
  }
  Else { Write-Host "Something is wrong with `$InputPath: [$($InputPath)]."; Exit}

  $refIDs = @()
  [int]$i=0
  ForEach ($MPPath in $MPPaths) {
    Write-Host $MPPath -F Green
    $mpxml = [xml](Get-Content $MPPath)
    #$mpxml.SelectNodes("//Reference") | % {
    $mpxml.ManagementPack.Manifest.References.Reference | ForEach-Object {
      $refIDs += $_.ID
      $_
      Write-Progress -Activity "Creating master catalog of known References" -Status $_.ID -PercentComplete ($i / $MPPaths.Count*100)
    }
    $i++
  }
  # Create array of all unique IDs
  $AllRefIDs = ($refIDs | Group-Object ).Name | Sort-Object
  $AllRefIDs | ForEach-Object {
    #If ID not in catalog yet, add it with customized alias
    If ( -not($cat.ContainsKey($_.ToString())) ){
      $cat.Add($_,(Create-Alias -Seed $_ -Existing $cat.Values) )
    }
  }

  # Update all aliases to unique values to prevent naming conflicts when modifying files
  $cat = Modify-Aliases -Localized:$true -cat $cat
  Write-Host "Updating references...(first pass, randomized unique values)" -b Black -f Yellow
  Standardize-AllReferenceAliases -MPPaths $MPPaths -Catalog $cat

  # Update all aliases to final, correct values, and modify files
  $cat = Modify-Aliases -Localized:$false -cat $cat
  Write-Host "Updating references...(final pass)" -b Black -f Green
  Standardize-AllReferenceAliases -MPPaths $MPPaths -Catalog $cat

  # Display the reference/alias Catalog
  $arrCat = @()
  ForEach ($key in $cat.Keys ){
    #Write-Host $key " = " $cat[$key]
    $arrCat += ("$key = $($cat[$key])" )
  }
  Return ($arrCat | Sort-Object)

} #End Function
#######################################################################

<#
    .Synopsis
    Will display all known modules contained in all sealed management packs as well as basic schema information.
    Based on the original script found here: http://sc.scomurr.com/scom-2012-r2-mp-authoring-getting-modules-and-their-configurations/
    .EXAMPLE
    PS C:\> Show-SCOMModules
 
    .EXAMPLE
    PS C:\> Show-SCOMModules | Out-GridView
 
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Original Date: 2017.12.13
 
    History: 2018.01.22: Refined.
    2017/12/13: First version
#>

Function Show-SCOMModules {

  #############################################
  Function Get-ModuleRunasProfileName {
    Param(
      $Module,
      $Profiles
    )
    $ModuleRunAs = $Profiles | Where-Object { $_.Id.Guid -eq $module.RunAs.Id.GUID }
    Return ($ModuleRunAs.Name+";"+$ModuleRunAs.DisplayName)
  }
  #############################################
  
  $collection=@()
  $RAAP = Get-SCOMRunAsProfile
  ForEach ($MP in (Get-SCManagementPack | Where-Object{$_.Sealed -eq $True})) {
    ForEach($Module in ($MP.GetModuleTypes() | Sort-Object -Property Name | Sort-Object -Property XMLTag)) {
      $RunAs = ''
      $Object = New-Object PSObject
      $Object | Add-Member Noteproperty -Name Module -Value $Module.Name
      $Object | Add-Member Noteproperty -Name ManagementPackName -Value $MP.Name
      $Object | Add-Member Noteproperty -Name Accessibility -Value $Module.Accessibility

      If ([bool]$Module.RunAs){
        $RunAs = Get-ModuleRunasProfileName -Module $Module -Profiles $RAAP
      }
      $Object | Add-Member Noteproperty -Name RunAs -Value $RunAs
      $Object | Add-Member Noteproperty -Name ID -Value $Module.ID

      if($null -ne $Module.Configuration.Schema) {
        #
        $set = ($Module.Configuration.schema.split("<") | Where-Object {$_ -match "^xsd:element.*"} | `
        ForEach-Object -Process {$_.substring($_.indexof("name")).split("=")[1].split()[0]})
        $Object | Add-Member Noteproperty Schema $set
      }
      $Object | Add-Member Noteproperty -Name Managed -Value $Module.Managed

      $collection += $object
    }
  }

  Return $collection
}
#######################################################################

<#
    .Synopsis
    This function is designed to launch a SCOM Property Bag PowerShell script or SCOM PowerShell discovery. It will output a formatted table to the screen. It can also output a hashtable or write the DataItem to an xml file..
    .DESCRIPTION
    When authoring and troubleshooting a SCOM Property Bag or discovery PowerShell script it is helpful to be able to analyze the resulting DataItem. This script will make this process easier by outputing the DataItem(s) to the screen and optionally an XML file. The default output file will have the same path/name as the input file (+.xml). Script must propertly use 'MOM.ScriptAPI' and output the proerty bag.
    Why is this useful? It's easy to output the DataItem to the screen but then it's difficult to read/analyze. If you open the .xml file with a fancy text editor like Notepad++ you can leverage the XMLTools plugin to format the data nicely. This is especially helpful for large property bags.
    Example :
    $api = New-Object -ComObject 'MOM.ScriptAPI'
    $bag = $api.CreatePropertyBag()
    $bag.AddValue('Message',"Some message")
    $api.Return($bag) #This method will output the DataItem text for analysis and is only used for script testing/analysis.
 
    .EXAMPLE
    #
    # This will execute the example script and output the property bag data of each bag ($Bag assumed to be in the script) to <$env:windir\Temp\><script file name>_<uniquetimestamp>.xml. Individual bags will be returned in a formatted table.
    PS C:\> Show-SCOMPropertyBag 'C:\Program Files\WindowsPowerShell\Modules\SCOMHelper\SCOMPowerShellPropertyBagTypes.ps1'
 
    .EXAMPLE
    #
    # This will execute MyScript.ps1 and output the property bag data of each bag ($Bag assumed to be in the script) to <$env:windir\Temp\><script file name>_<uniquetimestamp>.xml and then open the .xml file with the default application.
    PS C:\> Show-SCOMPropertyBag -FilePath C:\test\MyScript.ps1 -ShowXML
 
    .EXAMPLE
    #
    # This will execute C:\test\MyScript.ps1 and output the property bag data of each bag ($Bag assumed to be in the script) to <$env:windir\Temp\><script file name>_<uniquetimestamp>.xml.
    # All property bag data will be displayed to the screen in one formatted table. Use -Verbose switch to display output file location.
    PS C:\> Show-SCOMPropertyBag -FilePath C:\test\MyScript.ps1 -Format Combined -Verbose
 
    .EXAMPLE
    #
    # This will store the discovery dataitem to a hash table variable.
    PS C:\> $hash = Show-SCOMPropertyBag -FilePath 'C:\Test\MyDiscoveryScript.ps1' -Format HashTable
    PS C:\> $hash.1 | Out-GridView
 
 
    .NOTES
    Name: Show-SCOMPropertyBag
    Author: Tyson Paul
    History:
    2022.06.14 - Added ability to digest PB collections as well as discovery dataitems.
    2019.04.19 - Initial release.
#>

Function Show-SCOMPropertyBag {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1', 
      SupportsShouldProcess=$true, 
  PositionalBinding=$false)]

  Param (
    # The full path to the input script (PowerShell .ps1) file.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$true, 
        ValueFromRemainingArguments=$false, 
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string]$FilePath,

    # Powershell script file object. Typically this type of object would be obtained with Get-ChildItem.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$true,
        ValueFromPipelineByPropertyName=$true, 
        ValueFromRemainingArguments=$false, 
        Position=0,
    ParameterSetName='Parameter Set 2')]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [System.IO.FileInfo]$File,

    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$true, 
    Position=1)]
    [ValidateSet("Combined", "Separate","HashTable")]
    # Output formated objects (property bags) instead of default hash table. Not applicable to discovery bags.
    # Separate: Individual bags
    # Combined: One large table containing all data from all bags
    # HashTable: Hash table containing all bag data. Useful for spying on data values that don't otherwise display in the table format due to large size.
    [string]$Format='Separate',

        
    # This will open the .XML output file type with default application.
    [switch]$ShowXML
  )
  #region BEGIN
  Begin{
    #----------------- FUNCTIONS -----------------------
    Function FormatEach {
      Param(
        $hash 
      )

      Return $hash.Value
      
    }
    
    #--------------- END FUNCTIONS ---------------------

    $hashVariantTypes = [ordered]@{
      0 = "Empty"
      1 = "Null (unconfirmed)"
      2 = "Short (unconfirmed)"
      3 = "Integer"
      4 = "Float"
      5 = "Double"
      6 = "Currency (unconfirmed)"
      7 = "Date"
      8 = "String"
      9 = "Object (unconfirmed)"
      10 = "Error (unconfirmed)"
      11 = "Boolean"
      12 = "Variant (unconfirmed)"
      13 = "DataObject (unconfirmed)"
      14 = "Decimal"
      15 = "Byte (unconfirmed)"
      16 = "Char (unconfirmed)"
      17 = "Byte"
      18 = "Char"
      20 = "Long"
    }

    # Setup temp dir for output file. Using a redirector to an output file is the only way I could find to obtain the property bag data.
    $TempDir = (Join-Path (Join-Path $env:Windir "Temp") (Join-Path 'SCOMHelper' 'Show-SCOMPropertyBag') )
    New-Item -Type Directory -Path $TempDir -ErrorAction SilentlyContinue | Out-Null
    If (-not(Test-Path -Path $TempDir -PathType Container)) {
      Write-Error "Unable to create/access directory: [$($TempDir)]. " 
      Return $false
    }
    $hash = @{}
    $int=1
  }
  #endregion BEGIN
  
  
  Process {
    $batchstamp = Get-Date -Format o | ForEach-Object {$_ -replace ":", "."}
    If ([bool]$File){
      If (Test-Path -PathType Leaf -Path $File.FullName -ErrorAction SilentlyContinue){
        $OutFile = (Join-Path $TempDir "$($File.Name)_$($batchstamp)_.xml" )
        cmd /c PowerShell.exe -file $File.FullName > $OutFile
      }
    }

    Else {
      If (-NOT [bool]$FilePath) {
        #$FilePath = 'C:\Program Files\WindowsPowerShell\Modules\SCOMHelper\1.1\SCOMPowerShellPropertyBagTypes.ps1'
        $FilePath = Join-Path $psScriptRoot SCOMPowerShellPropertyBagTypes.ps1
      }
      If (Test-Path -PathType Leaf -Path $FilePath -ErrorAction SilentlyContinue){
        $OutFile = (Join-Path $TempDir "$(Split-Path $FilePath -Leaf)_$($batchstamp)_.xml")
        cmd /c PowerShell.exe -file $FilePath > $OutFile
      }
      Else {
        Throw "Problem with FilePath: [ $($FilePath) ]."
      }
    }
    # Remove generic typename that often appears as a result of standard '$bag' statement
    $content = ((Get-Content $OutFile -Raw ).Replace('System.__ComObject','').Replace('ÿ',''))  -Replace '\<Collection\>|\<\/Collection\>'
    $content | Set-Content $OutFile
    
    $token = (Get-Date -f fffffff)
    # Add a delimiter after each DataItem, then split on it.
    $array = (($content -replace '\<\/DataItem\>',"</DataItem>$($token)" ) -split $token)

    ForEach ($DI in $array[0..($array.Count -2)]){
          # If Discovery bag
          If (([xml]$DI).DataItem.DiscoveryType -match '0') {
            #ForEach () { }
            ForEach ($Item in ([xml]$DI).DataItem) { 
              [System.Collections.ArrayList]$arrayItems = @()
              'type','time','sourceHealthServiceId','DiscoveryType','DiscoverySourceType','DiscoverySourceObjectId','DiscoverySourceManagedEntity' | ForEach-Object {
                $obj = New-Object PSCUSTOMOBJECT
                $obj | Add-Member -MemberType NoteProperty -Name "ClassInstance" -Value 'N/A'
                $obj | Add-Member -MemberType NoteProperty -Name 'Name' -Value $_
                $obj | Add-Member -MemberType NoteProperty -Name 'Value' -Value ($Item.$_)
                $NULL = $arrayItems.Add($obj)
              }

              ForEach ($ClassInstance in ($Item.ClassInstances).ClassInstance){
                ForEach ($Settings in $ClassInstance.Settings){
                  ForEach ($Setting in $Settings.Setting){
                    $obj = New-Object PSCUSTOMOBJECT
                    $obj | Add-Member -MemberType NoteProperty -Name "ClassInstance" -Value ([string]$ClassInstance.TypeId)
                    $obj | Add-Member -MemberType NoteProperty -Name "Name" -Value ([string]$Setting.Name)
                    $obj | Add-Member -MemberType NoteProperty -Name "Value" -Value ([string]$Setting.Value)
                    $NULL = $arrayItems.Add($obj)
                  }
                }
              }
              
            }#end ForEach Item
          } #end Discovery bag type
      
      # Assume property bag
      Else {
        #ForEach ($DI in $array[0..($array.Count -2)]){
        [System.Collections.ArrayList]$arrayItems = @()
        'type','time','sourceHealthServiceId' | ForEach-Object {
          $obj = New-Object PSCUSTOMOBJECT

          $obj | Add-Member -MemberType NoteProperty -Name 'Name' -Value $_
          $obj | Add-Member -MemberType NoteProperty -Name 'VariantType' -Value ''
          $obj | Add-Member -MemberType NoteProperty -Name 'Value' -Value (([xml]$DI).DataItem.$_)
          $NULL = $arrayItems.Add($obj)
        }
 
        ForEach ($Property in ([xml]$DI).DataItem.Property){
          $obj = New-Object PSCUSTOMOBJECT
          $obj | Add-Member -MemberType NoteProperty -Name "Name" -Value ([string]$Property.Name)
          $obj | Add-Member -MemberType NoteProperty -Name "VariantType" -Value ("{0,2},{1}" -F $([int]$Property.VariantType), $($hashVariantTypes.[int]$Property.VariantType))
          $obj | Add-Member -MemberType NoteProperty -Name "Value" -Value ([string]$Property.'#text')
          $NULL = $arrayItems.Add($obj)
        }
      }
      
      
      $hash.Add($int,$arrayItems)
      $int++
    }
    
    If ($ShowXML) {
      # Open with default application
      & $OutFile
    }

    Write-Verbose "Output file: $OutFile"
  }

  End {
    If ($Format -eq 'Separate'){
      ForEach ($element in @($hash.GetEnumerator() )) { 
        FormatEach -hash $element | Out-Host
      }
    }
    
    ElseIf ($Format -eq 'Combined'){
      ForEach ($element in @($hash.GetEnumerator() )) { 
        FormatEach -hash $element 
      }
    }
    
    Else{
      Return $hash
    }
  }
} #end Function
#######################################################################



#######################################################################

<#
    .Synopsis
    Will test any number of TCP ports on any number of hosts.
    .DESCRIPTION
    Will accept piped Computers/IPs or Ports and then test those ports on the targets for TCP connectivity.
    .Parameter Computer
    Computer NetBIOS name, FQDN, or IP
    .Parameter Port
    TCP Port number to test.
    .Parameter TimeoutMS
    The amount of time to wait (in milliseconds) before abandoning a connection attempt on a given port.
    .EXAMPLE
    PS C:\> Test-Port -Computer '8.8.8.8' -Port 53
    .EXAMPLE
    PS C:\> 443,80,53,135,137,5723 | Test-Port -Computer 'MS01.contoso.com','DB01' | Sort-Object Computer,Port
    .EXAMPLE
    PS C:\> 'MS01.contoso.com','DB01' | Test-Port -Port 5723
    .NOTES
    Author: Tyson Paul
    Blog: https://monitoringguys.com/
    Version: 1.3
    Date: 2018.02.07
    History:
    2018.05.31: Improved error handling.
 
    Adapted from Boe Prox (https://gallery.technet.microsoft.com/scriptcenter/97119ed6-6fb2-446d-98d8-32d823867131)
#>

Function Test-Port {
  [CmdletBinding(DefaultParameterSetName='Parameter Set 1',
      SupportsShouldProcess=$true,
      PositionalBinding=$false,
      HelpUri = 'https://monitoringguys.com/',
  ConfirmImpact='Medium')]
  Param (
    [Parameter(Mandatory=$true,
        ValueFromPipeline=$true,
        ValueFromRemainingArguments=$false,
        Position=0,
    ParameterSetName='Parameter Set 1')]
    [string[]]$Computer,

    [Parameter(Mandatory=$true,
        ValueFromPipeline=$true,
        ValueFromRemainingArguments=$false,
        Position=1,
    ParameterSetName='Parameter Set 1')]
    [int[]]$Port,

    [int]$TimeoutMS=5000
  )
  Begin {
    $arrResults = @()
    $Error.Clear()
  }
  Process {
    If (-not $Port) { $Port = Read-host "Enter the port number to access" }

    ForEach ($thisComputer in $Computer) {
      ForEach ($thisPort in $Port) {
        $Status = "Failure"
        $objResult = New-Object -TypeName PSCustomObject
        $tcpobject = New-Object System.Net.Sockets.TcpClient
        #Connect to remote machine's port
        $connect = $tcpobject.BeginConnect($thisComputer,$thisPort,$null,$null)
        #Configure a timeout before quitting - time in milliseconds
        $wait = $connect.AsyncWaitHandle.WaitOne($TimeoutMS,$false)
        If (-Not $Wait) {
          $Message = "Connection Failure. Address:[$($thisComputer)], Port:[$($thisPort)] connection timed out [$($TimeoutMS) milliseconds].`n"
        }
        Else {
          Try{
            $tcpobject.EndConnect($connect)  #| out-Null
            $Status = "Success"
            $Message = "Connection Success. Address:[$($thisComputer)], Port:[$($thisPort)] connection successful.`n"
          }Catch{
            #If ([bool]$Error[0]) {
            $Message = ("{0}" -f $error[0].Exception.InnerException)
            #}
            #Else {
            # $Status = "Success"
            # $Message = "Connection Success. Address:[$($thisComputer)], Port:[$($thisPort)] connection successful.`n"
            #}
          }
        }
        $objResult | Add-Member -MemberType NoteProperty -Name Computer -Value $thisComputer
        $objResult | Add-Member -MemberType NoteProperty -Name Port -Value $thisPort
        $objResult | Add-Member -MemberType NoteProperty -Name Status -Value $Status
        $objResult | Add-Member -MemberType NoteProperty -Name Result -Value $Message
        $arrResults += $objResult
        $Error.Clear()
      }#End ForEach Port
    }#End ForEach Computer
  }#End Process

  End {
    Return $arrResults
  }

} #End Function
#######################################################################

<#
    .Synopsis
    Will unseal MP and MPB files along with any additional resources contained therein and place into version-coded folders.
    .EXAMPLE
    PS C:\> Unseal-SCOMMP -inDir C:\MyMPs -outDir C:\MyMPs\UnsealedMPs
    .EXAMPLE
    PS C:\> Unseal-SCOMMP -inDir 'C:\Program Files (x86)\System Center Management Packs' -outDir 'C:\Temp\Unsealtest_OUT'
    .EXAMPLE
    PS C:\> Unseal-SCOMMP -inDir 'C:\en_system_center_2012_r2_operations_manager_x86_and_x64_dvd_2920299\ManagementPacks' -outDir 'C:\Temp\Unsealtest_OUT'
    .INPUTS
    Strings (directory paths)
    .OUTPUTS
    XML files (+ any resource files: .dll, .bmp, .ico etc.)
 
    .NOTES
    Author (Revised by): Tyson Paul (I modified this heavily but the original that I used as a framework was not written by me.)
    Original Author: I can't seem to find where I originally got this script from. Jon Almquist maybe?
    Blog: https://monitoringguys.com/
    History:
    2019.05.01 - Fixed issue where output path would display extra backslash
    2018.05.25 - Initial.
    .LINK
    Get-SCOMMPFileInfo
    Get-SCOMClassInfo
#>

Function Unseal-SCOMMP {
  Param(
    $inDir="C:\Temp\Sealed",
    $outDir="C:\Temp\AllUnsealed"
  )

  [Reflection.Assembly]::LoadWithPartialName("Microsoft.EnterpriseManagement.Core")
  [Reflection.Assembly]::LoadWithPartialName("Microsoft.EnterpriseManagement.Packaging")

  $mpStore = New-Object Microsoft.EnterpriseManagement.Configuration.IO.ManagementPackFileStore($inDir)
  $mps = Get-ChildItem $inDir *.mp -Recurse
  If ($null -ne $mps) {
    ForEach ($file in $mps) {
      $MPFilePath = $file.FullName
      $mp = New-Object Microsoft.EnterpriseManagement.Configuration.ManagementPack($MPFilePath)
      $newOutDirPath = (Join-Path $outDir  ($file.Name + "($($mp.Version.ToString()))") )
      New-Item -Path $newOutDirPath -ItemType Directory -ErrorAction SilentlyContinue
      Write-Host $file
      $mpWriter = New-Object Microsoft.EnterpriseManagement.Configuration.IO.ManagementPackXmlWriter( $newOutDirPath )
      $mpWriter.WriteManagementPack($mp)
    }
  }

  #now dump MPB files
  $mps = Get-ChildItem $inDir *.mpb -Recurse
  $mpbReader = [Microsoft.EnterpriseManagement.Packaging.ManagementPackBundleFactory]::CreateBundleReader()
  If ($null -ne $mps) {
    ForEach ($mp in $mps) {
      $bundle = $mpbReader.Read($mp.FullName, $mpStore)
      $newOutDirPath = (Join-Path $outDir ($mp.Name + "($($bundle.ManagementPacks.Version.ToString()))" ))
      New-Item -Path $newOutDirPath -ItemType Directory -ErrorAction SilentlyContinue
      Write-Host $mp
      $mpWriter = New-Object Microsoft.EnterpriseManagement.Configuration.IO.ManagementPackXmlWriter($newOutDirPath)
      ForEach($bundleItem in $bundle.ManagementPacks) {
        #write the xml
        $mpWriter.WriteManagementPack($bundleItem)
        $streams = $bundle.GetStreams($bundleItem)
        ForEach($stream in $streams.Keys) {
          $mpElement = $bundleItem.FindManagementPackElementByName($stream)
          $fileName = $mpElement.FileName
          If ($null -eq $fileName) {
            $outpath = (Join-Path $newOutDirPath ('.' + $stream + '.bin'))
          }
          Else {
            If ($fileName.IndexOf('\') -gt 0) {
              #$outpath = Split-Path $fileName -Leaf
              $outpath = (Join-Path $newOutDirPath (Split-Path $fileName -Leaf))
            }
            Else {
              $outpath = (Join-Path $newOutDirPath $fileName)
            }
          }
          Write-Host "`t$outpath"
          $fs = [system.io.file]::Create($outpath)
          $streams[$stream].WriteTo($fs)
          $fs.Close()
        }
      }
    }
  }

}#End Function
#######################################################################
<#
    .DESCRIPTION
    Will overwrite an existing instance group or computer group containing Windows Computers [Microsoft.Windows.Computer] objects and optionally include related Health Service Watchers [Microsoft.SystemCenter.HealthServiceWatcher] if the original group type is Microsoft.SystemCenter.InstanceGroup.
    WARNING: This will replace existing group members with the objects for the provided computer names. This will technically not "add" to the exiting group.
 
    .EXAMPLE
    Update-SCOMComputerGroup -ComputerName 'ms01.contoso.com','ms02.contoso.com' -MgmtServerName ms01.contoso.com -GroupId '4f04c0c7-e3c3-8ef0-a710-1414dd320c5b' -PassThru -AddHealthServiceWatchers -Verbose
 
    The above example will update the designated instance group with the two Computers and associated Health Service Watchers. It will prompt for confirmation before
    updating the existing group membership. It will return the group object.
 
    .EXAMPLE
    #
    $FQDNs = (Get-SCOMAgent | Where-Object Name -like "*ProdDB*").Name
    Update-SCOMComputerGroup -ComputerName $FQDNs -MgmtServerName ms01.contoso.com -GroupName 'SQL.Production.SQLServerComputers.Group' -PassThru -Verbose -Force
 
    The above example will update the the designated group (instance group or Computer group) with all Computer objects that match the name criteria. . It will not prompt for confirmation before
    updating the existing group membership. It will return the group object.
 
    .INPUTS
    Accepts array of computer FQDN names (fully qualified domain name)
 
    .OUTPUTS
    SCOM Management Pack [Microsoft.EnterpriseManagement.Monitoring.MonitoringObjectGroup]
 
    .NOTES
    Script: Update-SCOMComputerGroup
    Author: Tyson Paul (https://monitoringguys.com/2019/11/12/scomhelper/)
    Version History:
    2020.11.19.1613 - Added ability to add core references if they are missing.
    2020.08.18.1703 - initial
 
    https://docs.microsoft.com/en-us/system-center/scom/manage-create-manage-groups?view=sc-om-2019
#>

Function Update-SCOMComputerGroup {
  [CmdletBinding(SupportsShouldProcess=$false, 
      PositionalBinding=$false,
      HelpUri = 'http://www.microsoft.com/',
  ConfirmImpact='Medium')]
  [Alias()]
  [OutputType([Microsoft.EnterpriseManagement.Monitoring.MonitoringObjectGroup])]

  Param (

    # Name of computer to be added to the group (and associated Health Service Watcher instance if used with -AddHealthServiceWatchers parameter)
    [Parameter(Mandatory=$true, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
    ValueFromRemainingArguments=$false)]
    [ValidateNotNull()]
    [ValidateNotNullOrEmpty()]
    [string[]]$ComputerName,

    # Name of mgmt server to use for SDK connection
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
    ValueFromRemainingArguments=$false)]
    [string]$MgmtServerName = 'LOCALHOST',
    
    # Name of the group (not DisplayName)
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 1')]
    [string]$GroupName,

    # DisplayName of the group (not Name )
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 2')]
    [string]$GroupDisplayName,

    # ID/GUID of the group
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
        ValueFromRemainingArguments=$false,
    ParameterSetName='Parameter Set 3')]
    [string]$GroupId,

    # This will include corresponding Health Service Watcher objects in the group if the group is an 'instance group'.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
    ValueFromRemainingArguments=$false)]
    [switch]$AddHealthServiceWatchers,
    
    # This will return the new group [Microsoft.EnterpriseManagement.Monitoring.MonitoringObjectGroup] object.
    [Parameter(Mandatory=$false, 
        ValueFromPipeline=$false,
        ValueFromPipelineByPropertyName=$false, 
    ValueFromRemainingArguments=$false)]
    [switch]$PassThru,
    
    # Will not prompt for confirmation
    [switch]$Force

  )


  ########################################################################################################
  ##################################### FUNCTIONS #######################################################
  Function Get-GroupType {
    Param (
      [string]$GroupName
    )
    #Get the group class
    Write-Verbose "Getting the group '$GroupName'."
    $GroupClassCriteria = New-Object Microsoft.EnterpriseManagement.Configuration.MonitoringClassCriteria("Name='$GroupName'")
    $GroupClass = $MG.GetMonitoringClasses($GroupClassCriteria)[0]
    If ($GroupClass -eq $null)
    {
      Write-Error "$GroupName is not found."
      Return $false
    }
    Else 
    {
      #Check if this monitoring class is actually an instance group or computer group
      Write-Verbose "Check if the group '$GroupName' is an instance group or a computer group."
      $GroupBaseTypes = $GroupClass.GetBaseTypes()
      Foreach ($item in $GroupBaseTypes)
      {
        If ($item.Id.Tostring() -eq '4ce499f1-0298-83fe-7740-7a0fbc8e2449')
        {
          Write-Verbose "'$GroupName' is an instance group."
          Return 'Microsoft.SystemCenter.InstanceGroup'
        }
        If ($item.Id.Tostring() -eq '0c363342-717b-5471-3aa5-9de3df073f2a')
        {
          Write-Verbose "'$GroupName' is a computer group."
          Return 'Microsoft.SystemCenter.ComputerGroup'
        }
      }

      # Should never reach this if group is valid
      Return $false
    
    }
  }#End Function Get-GroupType
  ########################################################################################################
  <#
      Will return the MP which contains the Discovery for a group.
      Only if there exists a single discovery datasource for the group.
  #>

  Function Get-GroupDiscoveryMP {
    Param (
      [string]$GroupName
    )
  
    #Get the group class
    Write-Verbose "Getting the group '$GroupName'."
    $GroupClassCriteria = New-Object Microsoft.EnterpriseManagement.Configuration.MonitoringClassCriteria("Name='$GroupName'")
    $GroupClass = $MG.GetMonitoringClasses($GroupClassCriteria)[0]
    If ($GroupClass -eq $null)
    {
      Write-Error "$GroupName is not found."
      Return $false
    }

    #Check if this monitoring class is actually an instance group or computer group
    Write-Verbose "Check if the group '$GroupName' is an instance group or a computer group."
    $GroupBaseTypes = $GroupClass.GetBaseTypes()
    $bIsGroup = $false
    Foreach ($item in $GroupBaseTypes)
    {
      If ($item.Id.Tostring() -eq '4ce499f1-0298-83fe-7740-7a0fbc8e2449')
      {
        Write-Verbose "'$GroupName' is an instance group."
        $bIsGroup = $true
      }
      If ($item.Id.Tostring() -eq '0c363342-717b-5471-3aa5-9de3df073f2a')
      {
        Write-Verbose "'$GroupName' is a computer group."
        $bIsGroup = $true
      }
    }
    If ($bIsGroup -eq $false)
    {
      Write-Error "$GroupName is not an instance group or a computer group."
      Return $false
    }

    #Get Group object
    $GroupObject = $MG.GetMonitoringObjects($GroupClass)[0]

    $GroupDiscoveries = $GroupObject.GetMonitoringDiscoveries()
    $iGroupPopDiscoveryCount = 0
    $GroupPopDiscovery = $null
    Foreach ($Discovery in $GroupDiscoveries)
    {
      $DiscoveryDS = $Discovery.DataSource
      #Microsft.SystemCenter.GroupPopulator ID is 488000ef-e20b-1ac4-d3b1-9d679435e1d7
      If ($DiscoveryDS.TypeID.Id.ToString() -eq '488000ef-e20b-1ac4-d3b1-9d679435e1d7')
      {
        #This data source module is using Microsft.SystemCenter.GroupPopulator
        $iGroupPopDiscoveryCount = $iGroupPopDiscoveryCount + 1
        $GroupPopDiscovery = $Discovery
        Write-Verbose "Group Populator discovery found: '$($GroupPopDiscovery.Name)'"
      }
    }
    If ($iGroupPopDiscoveryCount.count -eq 0)
    {
      Write-Error "No group populator discovery found for $GroupName."
      Return $false
    }

    If ($iGroupPopDiscoveryCount.count -gt 1)
    {
      Write-Error "$GroupName has multiple discoveries using Microsft.SystemCenter.GroupPopulator Module type. Unable to continue."
      Return $false
    }

    #Get the MP of where the group populator discovery is defined
    $GroupPopDiscoveryMP = $GroupPopDiscovery.GetManagementPack()
    $GroupPopDiscoveryMPName = $GroupPopDiscoveryMP.Name
    Write-Verbose "The group populator discovery '$($GroupPopDiscovery.Name)' is defined in management pack '$GroupPopDiscoveryMPName'."

    #Write Error and exit if the group discovery MP is sealed
    Write-Verbose "Checking if '$GroupPopDiscoveryMPName' MP is sealed."
    If ($GroupPopDiscoveryMP.sealed -eq $true)
    {
      Write-Error "Unable to update the group discovery because it is defined in a sealed MP: '$($GroupPopDiscoveryMP.DisplayName)'."
      Return $false
    } else {
      Write-Verbose "'$GroupPopDiscoveryMPName' MP is unsealed. OK to continue."
    }

    Return $GroupPopDiscoveryMP
  }
  ########################################################################################################
  Function Create-ExplicitMembershipConfig {
    Param (
      [Parameter(Mandatory=$true, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false,
      ParameterSetName='Parameter Set 1')]
      [ValidateNotNull()]
      [ValidateNotNullOrEmpty()]
      [array]$ComputerName,
    
      [Parameter(Mandatory=$true, 
          ValueFromPipeline=$false,
          ValueFromPipelineByPropertyName=$false, 
          ValueFromRemainingArguments=$false,
      ParameterSetName='Parameter Set 1')]
      [ValidateNotNull()]
      [ValidateNotNullOrEmpty()]
      [string]$GroupName,
      
      [string]$RelationshipClass,
    
      [string]$WindowsAlias='Windows',
      [string]$SCAlias='SystemCenter',
      [string]$InstanceGroupAlias='InstanceGroupLibrary'
  
    )
  
    $WinCompClass = Get-SCOMClass -Name "Microsoft.Windows.Computer"
    $HSWClass = Get-SCOMClass -Name "Microsoft.SystemCenter.HealthServiceWatcher"
    
    [System.Collections.ArrayList]$arrWinComp = @()
    [System.Collections.ArrayList]$arrHSW = @()
    
    # Find appropriate Computer/HSW objects, build collections
    ForEach ($CName in $ComputerName) {
      $WinComp = $NULL
      $Criteria = "DisplayName = '$CName'"
    
      $objCriteria = New-Object Microsoft.EnterpriseManagement.Monitoring.MonitoringObjectCriteria($Criteria ,$WinCompClass)
      $WinComp = $MG.GetMonitoringObjects($objCriteria)
      If (-NOT $WinComp) {
        Write-Warning "Computer: [$($CName)] not found! Verify that the fully qualified domain name exists."
        Continue
      }
      $CompID = $WinComp.Id.Guid
      Write-Verbose "Found Computer:`t[$($CName)], [$($CompID)]"
      $Null = $arrWinComp.Add(" <MonitoringObjectId>$CompID</MonitoringObjectId>`n")

      If ($AddHealthServiceWatchers) {
        $objCriteria = New-Object Microsoft.EnterpriseManagement.Monitoring.MonitoringObjectCriteria($Criteria ,$HSWClass)
        $HSW = $MG.GetMonitoringObjects($objCriteria)  
        $WatchID = $HSW.Id.Guid
        Write-Verbose "Found HSW:`t`t[$($CName)], [$($WatchID)]"
        $Null = $arrHSW.Add(" <MonitoringObjectId>$WatchID</MonitoringObjectId>`n")
      }
    }
 
    # Add Windows Computer GUIDs to explicit membership rule
    $Rules += @"
  <MembershipRule>
  <MonitoringClass>`$MPElement[Name="$($WindowsAlias)!Microsoft.Windows.Server.Computer"]`$</MonitoringClass>
  $($RelationshipClass)
  <IncludeList>
  $arrWinComp
  </IncludeList>
  </MembershipRule>
"@
 
 
    # If appropriate, add HSW rules to membership rules
    If ($arrHSW.Count) {
      $Rules += @"
 
  <MembershipRule>
  <MonitoringClass>`$MPElement[Name="$($SCAlias)!Microsoft.SystemCenter.HealthServiceWatcher"]`$</MonitoringClass>
  $($RelationshipClass)
  <IncludeList>
  $arrHSW
  </IncludeList>
  </MembershipRule>
"@
    
    }
 
    # Assemble the new configuration
    $NewConfiguration = @"
    <RuleId>`$MPElement`$</RuleId>
    <GroupInstanceId>`$MPElement[Name="$($GroupName)"]`$</GroupInstanceId>
    <MembershipRules>
    $($Rules)
    </MembershipRules>
"@

    Return $NewConfiguration
  

  }#End Funtion Create-ExplicitMembershipConfig

  ########################################################################################################
  

  Write-Verbose "Attempt import of OperationsManager module"
  . Import-SCOMPowerShellModule

  Try {
    Write-Verbose "Verify mgmt group connection..."
    If ($MgmtServerName) {
      $MG = Connect-OMManagementGroup -SDK $MgmtServerName -Verbose:$VerbosePreference
    }
    ElseIf (-NOT $MG) {
      $MG = Connect-OMManagementGroup -SDK 'Localhost' -Verbose:$VerbosePreference
    }
  } Catch {
    Write-Error "Failed to connect to mgmt group. $_"
    Exit
  }
  
  [string]$WindowsAlias='Windows'
  [string]$SCAlias='SystemCenter'
  [string]$InstanceGroupAlias='InstanceGroupLibrary'

  $DateStamp = (Get-Date -f "yyyyMMdd.hhmmss")
  $BackupDir = (Join-Path (Join-Path $env:Windir "Temp") (Join-Path 'SCOMHelper' "Update-SCOMComputerGroup_$($DateStamp)" ) )
  
  If ($GroupId){
    $objGroup = (Get-SCOMGroup -Id $GroupId)
  }
  If ($GroupName) {
    $Class = Get-SCOMClass -Name $GroupName
    $objGroup = (Get-SCOMGroup -DisplayName ($Class.DisplayName))
  }
  If ($GroupDisplayName) {
    $objGroup = (Get-SCOMGroup -DisplayName $GroupDisplayName)
  }
  $GroupName = $objGroup.FullName
  $GroupPopDiscoveryMP = Get-GroupDiscoveryMP -GroupName $GroupName

  If (-NOT $GroupPopDiscoveryMP) {
    Write-Error "Unable to retrieve management pack for group [$GroupName]. Exiting."
    Return
  }
  Try {
    New-Item -Path $BackupDir -ItemType Directory -Verbose -Force
    $GroupPopDiscoveryMP | Export-SCOMManagementPack -Path $BackupDir -Verbose -ErrorAction Stop
    Write-Verbose "Backup Directory: [$($BackupDir)]"
  } Catch {
    Throw "Unable to backup existing MP to path: [$($BackupDir)]. No action taken. Exiting. Error:`n$_"
    Return 
  }

  <#
      When defining the new discovery configuration we will need to use a specific alias depending on the group type.
      Aliases tend to be unique in unsealed MPs. Get unique aliases from existing group MP so they can be used in the new/updated member Configuration rules.
      This approach is a little sloppy and I may update it someday if I expand the group population automation abilities.
  #>

  #ForEach ($Key in @($GroupPopDiscoveryMP.References.Keys)) {
    # Locate "Windows" alias
    If ($GroupPopDiscoveryMP.References.Value.Name -match '^Microsoft.Windows.Library$') {
      $thisAlias = ($GroupPopDiscoveryMP.References | Where-Object {$_.Value.Name -match '^Microsoft.Windows.Library$'} ).Key
      $WindowsAlias = $thisAlias
      #Continue;
    }
    #Add the reference if it does not exist
    Else {
      Add-MPReference -UnsealedMPName $GroupPopDiscoveryMP.Name -ReferenceMPName 'Microsoft.Windows.Library' -ReferenceAlias $WindowsAlias
    }

    # Locate "SystemCenter" alias
    If ( ($GroupPopDiscoveryMP.References.Value.Name -match '^Microsoft.SystemCenter.Library$' )) {
      $thisAlias = ($GroupPopDiscoveryMP.References | Where-Object {$_.Value.Name -match '^Microsoft.SystemCenter.Library$'} ).Key
      $SCAlias = $thisAlias
      #Continue;
    }
    #Add the reference if it does not exist
    Else {
      Add-MPReference -UnsealedMPName $GroupPopDiscoveryMP.Name -ReferenceMPName 'Microsoft.Windows.Library' -ReferenceAlias $SCAlias
    }
     
    # Locate "InstanceGroup" alias
    If ( ($GroupPopDiscoveryMP.References.Value.Name -match '^Microsoft.SystemCenter.InstanceGroup.Library$' )) {
      $thisAlias = ($GroupPopDiscoveryMP.References | Where-Object {$_.Value.Name -match '^Microsoft.SystemCenter.InstanceGroup.Library$'} ).Key
      $InstanceGroupAlias = $thisAlias
      #Continue;
    }
    #Add the reference if it does not exist
    Else {
      Add-MPReference -UnsealedMPName $GroupPopDiscoveryMP.Name -ReferenceMPName 'Microsoft.Windows.Library' -ReferenceAlias $InstanceGroupAlias
    }

  #}
  
  
  # Determine what type of group this is: InstanceGroup or ComputerGroup
  $GroupClassType = Get-GroupType -GroupName $GroupName
  
  # Construct the appropriate RelationshipClass string. This must agree with the exiting discovery/group type
  Switch ($GroupClassType){
    'Microsoft.SystemCenter.InstanceGroup' {
      $RelationshipClass = @"
<RelationshipClass>`$MPElement[Name="$($InstanceGroupAlias)!Microsoft.SystemCenter.InstanceGroupContainsEntities"]`$</RelationshipClass>
"@

    }

    'Microsoft.SystemCenter.ComputerGroup' {
      If ($AddHealthServiceWatchers) {
        Write-Error "Existing group type is [$($GroupClassType)]. Cannot add Health Service Watchers to exiting group. No Action taken. (Must be 'Microsoft.SstemCenter.InstanceGroup' to add other instance types. Exiting.)"
        Return
      }
      $RelationshipClass = @"
<RelationshipClass>`$MPElement[Name="$($SCAlias)!Microsoft.SystemCenter.ComputerGroupContainsComputer"]`$</RelationshipClass>
"@

    }
    
    default {
      Write-Error "Unknown base class type [$($GroupClassType)] for group [$($GroupName)]. Exiting."
      Return
    }
  }
  
  Write-Verbose "Building new configuration..."
  $NewConfiguration = Create-ExplicitMembershipConfig -ComputerName $ComputerName -GroupName $GroupName -RelationshipClass $RelationshipClass -WindowsAlias $WindowsAlias -SCAlias $SCAlias -InstanceGroupAlias $InstanceGroupAlias
  Write-Verbose "New Configuration:`n$($NewConfiguration)"
  $Proceed = $false
  $r = ''
  If (-NOT $Force) {
    While ($r -notmatch 'y|n') {
      Write-Warning "Warning: This will overwrite existing group."
      $r = Read-Host 'Proceed? (Y/N)'
    }
    If ($r -match 'y') {
      $Proceed = $true
    }
    Else {
      Write-Warning "No action taken. Exiting."
      Return
    }
  }
  
  If ($Force -OR $Proceed) {
    $result = Update-OMGroupDiscovery2 -SDK $MgmtServerName -GroupName $GroupName -NewConfiguration $NewConfiguration -Verbose:$VerbosePreference
    Write-Verbose "Group Update succesful?: $result"
  }
  # Finally, get the new pack
  If ($PassThru) {
    Get-SCOMGroup -Id ($objGroup.Id) -Verbose:$VerbosePreference
  }


}#End Function Update-SCOMComputerGroup

#######################################################################



$TempFolder = (Join-Path (Join-Path $env:Windir "Temp") SCOMHelper )
If (-NOT (Test-Path -Path $TempFolder -PathType Container)){
  Write-Host "Attempting to create temp folder: $TempFolder" -F DarkCyan
  Try{
    New-Item -ItemType Directory -Path $TempFolder -Force -ErrorAction Stop
  }Catch{
    Write-Host "Failed to create temp directory. Please run this tool as administrator." -F Red
  }
}

. $PSScriptRoot\Start-SCOMOverrideTool.ps1
New-Alias -Name 'Export-EffectiveMonitoringConfiguration' -Value 'Export-SCOMEffectiveMonitoringConfigurationReport' -ErrorAction Ignore
New-Alias -Name 'Disable-SCOMAllEventRules' -Value 'Start-SCOMOverrideTool' -ErrorAction Ignore
New-Alias -Name  'Get-SCOMRunningWorkflows' -Value 'Get-SCOMAgentWorkflows' -ErrorAction Ignore
Export-ModuleMember -Alias *

$Functions = @'
Clear-SCOMCache
Compare-String
Convert-MAMLToHtml
Deploy-SCOMAgent
Disable-SCOMAllEventRules
Export-SCOMEffectiveMonitoringConfigurationReport
Export-SCOMEventsToCSV
Export-SCOMKnowledge
Export-SCOMOverrides
Fast-Ping
Get-SCOMAlertKnowledge
Get-SCOMClassInfo
Get-SCOMHealthCheckOpsConfig
Get-SCOMMPFileInfo
Get-SCOMRunAsAccountName
Get-SCOMRunAsProfilesAccounts
Get-SCOMAgentWorkflows
Get-StringHash
New-SCOMClassGraph
New-SCOMComputerGroup
Ping-AllHosts
Remove-SCOMObsoleteReferenceFromMPFile
Set-SCOMMPAliases
Show-SCOMModules
Show-SCOMPropertyBag
Start-SCOMOverrideTool
Start-SCOMTrace
Test-Port
Unseal-SCOMMP
Update-SCOMComputerGroup
'@
 -split  "\n" -replace "`r","" | Export-ModuleMember