Why Gemfury? Push, build, and install  RubyGems npm packages Python packages Maven artifacts PHP packages Go Modules Debian packages RPM packages NuGet packages

Repository URL to install this package:

Details    
ansible / community / windows / plugins / modules / win_wait_for_process.ps1
Size: Mime:
#!powershell

# Copyright: (c) 2017, Ansible Project
# Copyright: (c) 2018, Dag Wieers (@dagwieers) <dag@wieers.com>
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)

#AnsibleRequires -CSharpUtil Ansible.Basic
#Requires -Module Ansible.ModuleUtils.SID

$spec = @{
    options = @{
        process_name_exact = @{ type = 'list'; elements = 'str' }
        process_name_pattern = @{ type = 'str' }
        pid = @{ type = 'int'; default = 0 }
        owner = @{ type = 'str' }
        sleep = @{ type = 'int'; default = 1 }
        pre_wait_delay = @{ type = 'int'; default = 0 }
        post_wait_delay = @{ type = 'int'; default = 0 }
        process_min_count = @{ type = 'int'; default = 1 }
        state = @{ type = 'str'; default = 'present'; choices = @( 'absent', 'present' ) }
        timeout = @{ type = 'int'; default = 300 }
    }
    mutually_exclusive = @(
        @( 'pid', 'process_name_exact' ),
        @( 'pid', 'process_name_pattern' ),
        @( 'process_name_exact', 'process_name_pattern' )
    )
    required_one_of = @(
        , @( 'owner', 'pid', 'process_name_exact', 'process_name_pattern' )
    )
    supports_check_mode = $true
}

$module = [Ansible.Basic.AnsibleModule]::Create($args, $spec)

$process_name_exact = $module.Params.process_name_exact
$process_name_pattern = $module.Params.process_name_pattern
$process_id = $module.Params.pid  # pid is a reserved variable in PowerShell, using process_id instead
$owner = $module.Params.owner
$sleep = $module.Params.sleep
$pre_wait_delay = $module.Params.pre_wait_delay
$post_wait_delay = $module.Params.post_wait_delay
$process_min_count = $module.Params.process_min_count
$state = $module.Params.state
$timeout = $module.Params.timeout

$module.Result.changed = $false
$module.Result.elapsed = 0
$module.Result.matched_processes = @()

# Validate the input
if ($state -eq "absent" -and $sleep -ne 1) {
    $module.Warn("Parameter 'sleep' has no effect when waiting for a process to stop.")
}

if ($state -eq "absent" -and $process_min_count -ne 1) {
    $module.Warn("Parameter 'process_min_count' has no effect when waiting for a process to stop.")
}

if ($owner -and ("IncludeUserName" -notin (Get-Command -Name Get-Process).Parameters.Keys)) {
    $module.FailJson("This version of Powershell does not support filtering processes by 'owner'.")
}

Function Get-FilteredProcess {
    [cmdletbinding()]
    Param(
        [String]
        $Owner,
        $ProcessNameExact,
        $ProcessNamePattern,
        [int]
        $ProcessId
    )

    $FilteredProcesses = @()

    try {
        $Processes = Get-Process -IncludeUserName
        $SupportsUserNames = $true
    }
    catch [System.Management.Automation.ParameterBindingException] {
        $Processes = Get-Process
        $SupportsUserNames = $false
    }

    foreach ($Process in $Processes) {

        # If a process name was specified in the filter, validate that here.
        if ($ProcessNamePattern) {
            if ($Process.ProcessName -notmatch $ProcessNamePattern) {
                continue
            }
        }

        # If a process name was specified in the filter, validate that here.
        if ($ProcessNameExact -and $ProcessNameExact -notcontains $Process.ProcessName) {
            continue
        }

        # If a PID was specified in the filter, validate that here.
        if ($ProcessId -and $ProcessId -ne 0) {
            if ($ProcessId -ne $Process.Id) {
                continue
            }
        }

        # If an owner was specified in the filter, validate that here.
        if ($Owner) {
            if (-not $Process.UserName) {
                continue
            }
            elseif ((Convert-ToSID($Owner)) -ne (Convert-ToSID($Process.UserName))) {
                # NOTE: This is rather expensive
                continue
            }
        }

        if ($SupportsUserNames -eq $true) {
            $FilteredProcesses += @{ name = $Process.ProcessName; pid = $Process.Id; owner = $Process.UserName }
        }
        else {
            $FilteredProcesses += @{ name = $Process.ProcessName; pid = $Process.Id }
        }
    }

    return , $FilteredProcesses
}

$module_start = Get-Date
Start-Sleep -Seconds $pre_wait_delay

if ($state -eq "present" ) {

    # Wait for a process to start
    do {

        $Processes = Get-FilteredProcess -Owner $owner -ProcessNameExact $process_name_exact -ProcessNamePattern $process_name_pattern -ProcessId $process_id
        $module.Result.matched_processes = $Processes

        if ($Processes.count -ge $process_min_count) {
            break
        }

        if (((Get-Date) - $module_start).TotalSeconds -gt $timeout) {
            $module.Result.elapsed = ((Get-Date) - $module_start).TotalSeconds
            $module.FailJson("Timed out while waiting for process(es) to start")
        }

        Start-Sleep -Seconds $sleep

    } while ($true)

}
elseif ($state -eq "absent") {

    # Wait for a process to stop
    $Processes = Get-FilteredProcess -Owner $owner -ProcessNameExact $process_name_exact -ProcessNamePattern $process_name_pattern -ProcessId $process_id
    $module.Result.matched_processes = $Processes

    if ($Processes.count -gt 0 ) {
        try {
            # This may randomly fail when used on specially protected processes (think: svchost)
            Wait-Process -Id $Processes.pid -Timeout $timeout
        }
        catch [System.TimeoutException] {
            $module.Result.elapsed = ((Get-Date) - $module_start).TotalSeconds
            $module.FailJson("Timeout while waiting for process(es) to stop")
        }
    }

}

Start-Sleep -Seconds $post_wait_delay
$module.Result.elapsed = ((Get-Date) - $module_start).TotalSeconds

$module.ExitJson()