Public/ConvertTo-WmiFilter.ps1

function ConvertTo-WmiFilter {
    <#
    .SYNOPSIS
        Converts an Active Directory object to a WMI filter object.
 
    .DESCRIPTION
        This function takes an Active Directory object, retrieves its corresponding WMI filter from the Group Policy domain,
        and returns it as an object of type "Microsoft.GroupPolicy.WmiFilter". The function includes error handling and retry
        logic to address Active Directory replication delays.
 
    .PARAMETER ADObject
        An array of ADObject instances representing the Active Directory objects to convert to WMI filters.
 
    .EXAMPLE
        ConvertTo-WmiFilter -ADObject $adObjects
 
    .INPUTS
        [Microsoft.ActiveDirectory.Management.ADObject[]] - An array of Active Directory objects.
 
    .OUTPUTS
        [Microsoft.GroupPolicy.WmiFilter] - The converted WMI filter object.
 
    .NOTES
        Version: 1.1
        DateModified: 31/May/2024
        LastModifiedBy: Vicente Rodriguez Eguibar
            vicente@eguibar.com
            Eguibar Information Technology S.L.
            http://www.eguibarit.com
    #>

    [CmdletBinding(SupportsShouldProcess = $true, ConfirmImpact = 'Medium')]
    [OutputType([void])]

    Param (
        [Microsoft.ActiveDirectory.Management.ADObject[]] $ADObject
    )

    Begin {
        $error.Clear()
        $txt = ($constants.Header -f
            (Get-Date).ToShortDateString(),
            $MyInvocation.Mycommand,
            (Get-FunctionDisplay $PsBoundParameters -Verbose:$False)
        )
        Write-Verbose -Message $txt

        ##############################
        # Module imports

        Import-Module -Name 'GroupPolicy' -SkipEditionCheck -Force -Verbose:$false | Out-Null

        ##############################
        # Variables Definition

        # The concept of this function has been taken directly from the GPWmiFilter.psm1 module
        # written by Bin Yi from Microsoft. I have modified it to allow for the challenges of
        # Active Directory replication. It will return the WMI filter as an object of type
        # "Microsoft.GroupPolicy.WmiFilter".
        $gpDomain = New-Object -TypeName Microsoft.GroupPolicy.GPDomain

    } #end Begin

    Process {

        # Iterate each ADObject
        foreach ($item in $ADObject) {
            if ($PSCmdlet.ShouldProcess($item.Name, 'Convert to WMI Filter')) {
                $path = 'MSFT_SomFilter.Domain="{0}",ID="{1}"' -f $gpDomain.DomainName, $item.Name
                $filter = $null
                $attempt = 0
                $maxAttempts = 4

                do {
                    try {
                        $filter = $gpDomain.GetWmiFilter($path)
                    } catch {
                        Write-Error -Message 'The WMI filter could not be found.'
                        ###Get-CurrentErrorToDisplay -CurrentError $error[0]
                        throw
                    } #end Try-Catch

                    if ($filter) {
                        [Guid]$Guid = $item.Name.Substring(1, $item.Name.Length - 2)

                        $filter | Add-Member -MemberType NoteProperty -Name Guid -Value $Guid -PassThru
                        $filter | Add-Member -MemberType NoteProperty -Name Content -Value $item.'msWMI-Parm2' -PassThru

                        break
                    } else {
                        $attempt++
                        if ($attempt -lt $maxAttempts) {
                            Write-Warning -Message 'Waiting 5 seconds for Active Directory replication to complete.'
                            Start-Sleep -Seconds 5
                            Write-Warning -Message 'Trying again to retrieve the WMI filter.'
                        } else {
                            Write-Error -Message 'Max attempts reached. Could not retrieve the WMI filter.'
                            break
                        } #end If-Else
                    } #end If-Else
                } while ($attempt -lt $maxAttempts)
            } #end ShouldProcess
        } #end Foreach
    } #end Process

    End {
        $txt = ($Constants.Footer -f $MyInvocation.InvocationName,
            'converting the WMI filter.'
        )
        Write-Verbose -Message $txt
    } #end Function
} #end Function