Hide progress of Invoke-WebRequest
Asked Answered
F

3

95

How can I hide the progress display of Invoke-WebRequest? I do a lot of successive requests and have my own Write-Progress display that I use, so I don't need the built-in one popping up underneath it every time.

I use the mshtml results (the IE COM object) that are created from the result of Invoke-WebRequest automatically, so I can't switch to a WebClient or something like that, unless someone provides instructions on how to get an mshtml object from a WebClient request.

Fai answered 12/9, 2013 at 17:30 Comment(1)
Note that performance is order of magnitudes better when hiding progressbar. I was waiting for few minutes until I canceled <300MB download. When turned off progress bar, it completed within few seconds. Related: Progress bar can significantly impact cmdlet performanceScrip
A
139

Use the $progressPreference variable. It should have a value of 'Continue' by default unless you've edited it elsewhere, which tells Powershell to display the progress bar. Since you mentioned that you have your own custom progress displays, I would reset it immediately after the cmdlet is executed. For example:

$ProgressPreference = 'SilentlyContinue'    # Subsequent calls do not display UI.
Invoke-WebRequest ...
$ProgressPreference = 'Continue'            # Subsequent calls do display UI.
Write-Progress ...

More info on preference variables at about_preference_variables. Here's the entry for $ProgressPreference:

$ProgressPreference
-------------------
Determines how Windows PowerShell responds to progress updates 
        generated by a script, cmdlet or provider, such as the progress bars
        generated by the Write-Progress cmdlet. The Write-Progress cmdlet 
        creates progress bars that depict the status of a command.

        Valid values:
          Stop:               Does not display the progress bar. Instead,
                                it displays an error message and stops executing.

          Inquire:            Does not display the progress bar. Prompts
                                for permission to continue. If you reply
                                with Y or A, it displays the progress bar.

          Continue:           Displays the progress bar and continues with
          (Default)             execution.

          SilentlyContinue:   Executes the command, but does not display
                                the progress bar.
Aggie answered 12/9, 2013 at 17:35 Comment(3)
In case anyone else has issues with this... you may need to explicitly specify $global:progressPreference = 'silentlyContinue' if calling in to other script blocks.Algorithm
Using $oldProgressPreference = $progressPreference; $progressPreference = 'SilentlyContinue'; and later $progressPreference = $oldProgressPreference will in addition retain the previous setting which can improve consistency when linking scripts.Valorie
@ChrisBaxter thanks! Without the $global bit it was working fine for me when I had the commands written out in my CI config directly, but it stopped working once I moved it into a powershell script. Setting the global variable fixed it right up!Murrey
M
11

Here is a reusable function to temporarily hide the progress of any script block and automatically restore the progress preference when the script block ends, even if an exception (script-terminating error) is thrown by the script block.

# Create an in-memory module so $ScriptBlock doesn't run in new scope
$null = New-Module {
    function Invoke-WithoutProgress {
        [CmdletBinding()]
        param (
            [Parameter(Mandatory)] [scriptblock] $ScriptBlock
        )

        # Save current progress preference and hide the progress
        $prevProgressPreference = $global:ProgressPreference
        $global:ProgressPreference = 'SilentlyContinue'

        try {
            # Run the script block in the scope of the caller of this module function
            . $ScriptBlock
        }
        finally {
            # Restore the original behavior
            $global:ProgressPreference = $prevProgressPreference
        }
    }
}

Usage example:

Invoke-WithoutProgress {
    # Here $ProgressPreference is set to 'SilentlyContinue'
    Invoke-WebRequest ...
}

# Now $ProgressPreference is restored
Write-Progress ...

Notes:

  • The New-Module call is there so the script block passed to Invoke-WithoutProgress doesn't run in a new scope (allowing it to directly modify surrounding variables, similar to ForEach-Object's script block). See this answer for more information.
Morgenthaler answered 15/8, 2022 at 11:24 Comment(2)
Just wondering, doesn't this technically also swallow exceptions, which may be undesired? I like the approach (have done similar stuff with lambdas in C++ and Python), but swallowing exceptions as a side effect should probably be noted. Thanks for showing the technique, though.Suanne
@Suanne No, it doesn't swallow exceptions as there is no catch block in the function. The finally block just makes sure that the $ProgressPreference variable is restored in any case, but exceptions still bubble up the call stack, after the finally block has been run.Morgenthaler
M
4

PowerShell 7.4 has been released and it has added -ProgressAction as a common parameter to all the cmdlets. This means you can use -ProgressAction SilentlyContinue for Invoke-WebRequest and easily hide its progress.

Mv answered 17/11, 2023 at 13:58 Comment(1)
This is unfortunately currently a bit broken, as per this GitHub issue: The -ProgressAction parameter doesn't work (PowerShell 7.4.1 only)Insecticide

© 2022 - 2024 — McMap. All rights reserved.