Skip to content

Support HTTP persistent connections in Web Cmdlets #19249

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Conversation

stevenebutler
Copy link
Contributor

@stevenebutler stevenebutler commented Mar 1, 2023

PR Summary

Fis #12764

This PR addresses 12764 by using persistent HTTP connections when Invoke-WebRequest uses the -Session variable. It does this by

  1. Tracking the HttpClient object in the saved WebRequestSession.
  2. Monitoring properties in WebRequestSession and rebuilding the HttpClient if they are changed. This is done to ensure that existing scripts that might tweak parameters between invocations of IWR actually apply the changes so they will behave in a backwards compatible manner. When anything that impacts the HttpClient is changed, the persistent connection is abandoned by disposing the HttpClient and underlying HttpClientHandler and constructing a new one. A warning is logged when this occurs as it has a performance impact (though no worse than the existing behaviour).
  3. To support point 2, many parameters are stored as internal set-only properties in the WebRequestSession so changes between invocations that would require a new connection can be detected and the handler rebuilt appropriately.

PR Context

This PR makes a considerable difference to performance when using Invoke-WebRequest or Invoke-RestMethod with sessions when multiple requests are made to the same host. This is particularly evident for HTTPS requests, as each HTTPS connection can take hundreds of milliseconds to establish.

This PR as written passes existing tests in WebCmdlets.Tests.ps1 and additional WebSession tests that verify the HttpClient is rebuilt when appropriate.

The PR provides as much backwards compatibility as possible at the expense of the complexity of tracking how the HttpClient used in the WebSession was constructed and recreating the HttpClient in the case it would have been constructed differently between invocations in scripts that assume it is okay to change parameters between invocations with the same WebSession.

This PR squashes the commits in the original PR#19173 following multiple round of reviews by @iSazonov and @CarloToso on a previous version of the PR. PR#19173 has been withdrawn due to number of commits and volume of review comments.

Performance Impact

Performance impact of maintaining HTTP connection persistence is measured by comparing PowerShell 7.3.3 with this PR using this simple script:

(Measure-Command {
    $null =Invoke-WebRequest https://postman-echo.com/get?i=1 -SessionVariable Session
    1..10 | ForEach-Object {
            $null =Invoke-WebRequest https://postman-echo.com/get?i=$_ -WebSession $Session
    }
}).ToString()

PowerShell 7.3.3

00:00:08.3019013

PR Branch

00:00:03.4589770

PR Checklist

@stevenebutler
Copy link
Contributor Author

Codefactor is complaining about some tests that were not added by this PR using plain text as a password.

@CarloToso will that impact being able to merge this PR?

@CarloToso
Copy link
Contributor

@stevenebutler Don't worry about those two codefactor issues, I introduced the in #19190

@iSazonov iSazonov requested a review from SteveL-MSFT March 3, 2023 16:47
@iSazonov iSazonov added the CL-General Indicates that a PR should be marked as a general cmdlet change in the Change Log label Mar 3, 2023
Use persistent HTTP connections when Invoke-WebRequest uses the -Session variable.

This is achieved by
* tracking the HttpClient object in the saved WebRequestSession
* monitoring properties in WebRequestSession and rebuilding the
  HttpClient if they are changed. This is done to ensure that
  existing scripts that might change parameters between invocations
  of IWR actually apply the changes so they will behave in a backwards
  compatible manner. When anything that impacts the HttpClient is
  changed, the persistent connection is abandoned by disposing the
  HttpClient and underlying HttpClientHandler and constructing a new
  one. A verbose message is logged when this occurs as it has a
  performance impact (though no worse than the existing behaviour).
* To support the point 2 above, many switches were added as internal
  set-only properties to the WebRequestSession so changes between
  invocations can be detected and the handler rebuilt appropriately.
@stevenebutler stevenebutler force-pushed the support-connection-persistence-in-web-cmdlets branch from e7d0b02 to 2f21034 Compare March 3, 2023 22:56
@pull-request-quantifier-deprecated

This PR has 305 quantified lines of changes. In general, a change size of upto 200 lines is ideal for the best PR experience!


Quantification details

Label      : Large
Size       : +255 -50
Percentile : 70.5%

Total files changed: 5

Change summary by file extension:
.cs : +141 -50
.resx : +3 -0
.ps1 : +111 -0

Change counts above are quantified counts, based on the PullRequestQuantifier customizations.

Why proper sizing of changes matters

Optimal pull request sizes drive a better predictable PR flow as they strike a
balance between between PR complexity and PR review overhead. PRs within the
optimal size (typical small, or medium sized PRs) mean:

  • Fast and predictable releases to production:
    • Optimal size changes are more likely to be reviewed faster with fewer
      iterations.
    • Similarity in low PR complexity drives similar review times.
  • Review quality is likely higher as complexity is lower:
    • Bugs are more likely to be detected.
    • Code inconsistencies are more likely to be detected.
  • Knowledge sharing is improved within the participants:
    • Small portions can be assimilated better.
  • Better engineering practices are exercised:
    • Solving big problems by dividing them in well contained, smaller problems.
    • Exercising separation of concerns within the code changes.

What can I do to optimize my changes

  • Use the PullRequestQuantifier to quantify your PR accurately
    • Create a context profile for your repo using the context generator
    • Exclude files that are not necessary to be reviewed or do not increase the review complexity. Example: Autogenerated code, docs, project IDE setting files, binaries, etc. Check out the Excluded section from your prquantifier.yaml context profile.
    • Understand your typical change complexity, drive towards the desired complexity by adjusting the label mapping in your prquantifier.yaml context profile.
    • Only use the labels that matter to you, see context specification to customize your prquantifier.yaml context profile.
  • Change your engineering behaviors
    • For PRs that fall outside of the desired spectrum, review the details and check if:
      • Your PR could be split in smaller, self-contained PRs instead
      • Your PR only solves one particular issue. (For example, don't refactor and code new features in the same PR).

How to interpret the change counts in git diff output

  • One line was added: +1 -0
  • One line was deleted: +0 -1
  • One line was modified: +1 -1 (git diff doesn't know about modified, it will
    interpret that line like one addition plus one deletion)
  • Change percentiles: Change characteristics (addition, deletion, modification)
    of this PR in relation to all other PRs within the repository.


Was this comment helpful? 👍  :ok_hand:  :thumbsdown: (Email)
Customize PullRequestQuantifier for this repository.

@stevenebutler
Copy link
Contributor Author

I have squashed all commits into a single commit.

@adityapatwardhan adityapatwardhan merged commit 9e34114 into PowerShell:master Mar 7, 2023
@adityapatwardhan
Copy link
Member

@stevenebutler Thank you for your contribution!

@God-damnit-all
Copy link
Contributor

@stevenebutler Thank you for your contribution!

I'm happy to see this merged. You can safely close #12764 now.

@ghost
Copy link

ghost commented Mar 14, 2023

🎉v7.4.0-preview.2 has been released which incorporates this pull request.:tada:

Handy links:

@tonyg00
Copy link

tonyg00 commented Mar 15, 2023

I tested Preview 2 and back to back invocations of Invoke-WebRequest and Invoke-RestMethod are not re-using ports, new ones are created.

@CarloToso
Copy link
Contributor

@tonyg00 did you use the -SessionVariable and then -WebSession parameter?

@tonyg00
Copy link

tonyg00 commented Mar 15, 2023

I did not, I just executed, Invoke-WebRequest -uri which is the most common use case for me of these cmdlets.
What is the correct syntax for this?

@CarloToso
Copy link
Contributor

Don't worry, it's a new feature, it's not yet documented.
I think this syntax should work:

Invoke-WebRequest -Uri -SessionVariable session
Invoke-WebRequest -Uri -WebSession $session

@tonyg00
Copy link

tonyg00 commented Mar 15, 2023

That is working, Thanks!!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
CL-General Indicates that a PR should be marked as a general cmdlet change in the Change Log Large
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants