Skip to content

Latest commit

 

History

History
874 lines (627 loc) · 41.5 KB

File metadata and controls

874 lines (627 loc) · 41.5 KB

Azure SDK Tools Test Proxy

To users writing SDK tests: This document is not intended to be a test preparation reference. You should instead refer to documentation in your specific language repository in order to configure recorded tests.

Test documentation by language:

Table of contents

This test proxy is intended to provide out-of-process record/playback capabilities compatible with any language. It offers session and recording level customization during both record and playback of a test.

All that is required to start recording is to make minor updates to the requests made within a given test. A standard request looks something like this:

request_changes

Modified to be recordable by the test proxy, it should look like this:

request_changes_after

There is a walkthrough through the process below in the how do I use the test proxy to get a recording.

For a more detailed explanation of how the test proxy works, along with links to demo projects for various languages, refer to the following post on the Azure SDK blog:

Level up your cloud testing game with the Azure SDK test proxy

Installation

Via Local Compile or .NET

  1. Install .NET 5.0 or 6.0
  2. Install test-proxy
dotnet tool update azure.sdk.tools.testproxy --global --add-source https://pkgs.dev.azure.com/azure-sdk/public/_packaging/azure-sdk-for-net/nuget/v3/index.json --version "1.0.0-dev*" --ignore-failed-sources

The test-proxy is also available from the azure-sdk-for-net public feed

Note: if you're not authorized to access these feeds, make sure you have Azure Artifacts Credential Provider installed. You can also download executable.

To uninstall an existing test-proxy

dotnet tool uninstall --global azure.sdk.tools.testproxy

After successful installation, run the tool:

test-proxy --storage-location <location>

If you've already installed the tool, you can always check the installed version by invoking:

test-proxy --version

Via Standalone Executable

Standalone executable versions of the test-proxy are published to github releases on this repository.

These executables are produced for multiple platforms and are available attached as assets:

image

Which should I install?

The version suffix for each release is based on the date. In most cases, the latest version should be totally stable.

For safety, the "official target" version that the azure-sdk team uses is present within eng/common/testproxy/target_version.txt. New "official" versions are tested by all consumers prior to updating the target version within this file.

Command line arguments

This is the help information for test-proxy. It uses the nuget package System.CommandLine to parse arguments.

The test-proxy executable fulfills one of two primary purposes:

  1. The test-proxy server (the only option up to this point)
  2. asset-sync verbs
    • push
    • restore
    • reset
    • config

This is surfaced by only showing options for the default commands. Each individual command has its own argument set that can be detailed by invoking test-proxy <command> --help.

/>Azure.Sdk.Tools.TestProxy --help
Description:
  This tool is used by the Azure SDK team in two primary ways:

    - Run as a http record/playback server. ("start" / default verb)
    - Invoke a CLI Tool to interact with recordings in an external store. ("push", "restore", "reset", "config")

Usage:
  Azure.Sdk.Tools.TestProxy [command] [options]

Options:
  -l, --storage-location <storage-location>  The path to the target local git repo. If not provided as an argument, Environment
                                             variable TEST_PROXY_FOLDER will be consumed. Lacking both, the current working
                                             directory will be utilized.
  -p, --storage-plugin <storage-plugin>      The plugin for the selected storage, default is Git storage is GitStore. (Currently
                                             the only option) [default: GitStore]
  --version                                  Show version information
  -?, -h, --help                             Show help and usage information

Commands:
  start <args>  Start the TestProxy.
  push          Push the assets, referenced by assets.json, into git.
  restore       Restore the assets, referenced by assets.json, from git.
  reset         Reset the assets, referenced by assets.json, from git to their original files referenced by the tag. Will prompt if
                there are pending changes unless indicated by -y/--yes.
  config        Interact with an assets.json.

For the config verb, there are subverbs!

/>Azure.Sdk.Tools.TestProxy config --help
Description:
  Interact with an assets.json.

Usage:
  Azure.Sdk.Tools.TestProxy config [command] [options]

Options:
  -l, --storage-location <storage-location>  The path to the target local git repo. If not provided as an argument, Environment
                                             variable TEST_PROXY_FOLDER will be consumed. Lacking both, the current working
                                             directory will be utilized.
  -p, --storage-plugin <storage-plugin>      The plugin for the selected storage, default is Git storage is GitStore. (Currently
                                             the only option) [default: GitStore]
  -?, -h, --help                             Show help and usage information

Commands:
  create  Enter a prompt and create an assets.json.
  show    Show the content of a given assets.json.
  locate  Get the assets repo root for a given assets.json path.

Storage Location

The test-proxy resolves the storage location via:

  1. Command Line argument --storage-location or the abbreviation -l
  2. Environment variable TEST_PROXY_FOLDER
  3. If neither are present, the working directory from which the server is invoked.

Port Assignation

By default, the server will listen on the following port mappings:

protocol port
http 5000
https 5001

Warning

MacOS users should be aware that airplay by default utilizes port 5000. Ensure the port is free or use a non-default port as described below.

Environment Variable

Set ASPNETCORE_URLS to define a custom port for either http or https (or both). Here are some examples:

$env:ASPNETCORE_URLS="http://*:3331"  // Set custom port for http only
$env:ASPNETCORE_URLS="http://*3331;https://*:8881"  // set custom ports for both http and https

Input arguments

When starting the test proxy there are Verbs, as shown in the test-proxy --help output above. For the start verb, there can be additional command arguments that need to get passed to Host.CreateDefaultBuilder 'as is'. The way this is done is through the use of -- or dashdash as it's referred to. If -- is on the command line, everything after will be treated as arguments that will be passed into this Host call.

For example, you can use command line argument --urls to bind to a non-default host and port. This configuration will override the environment configuration. To bind to localhost http 9000, provide the argument -- --urls http://localhost:9000. Note that --, space, then the --urls http://localhost:9000.

test-proxy -- --urls "http://localhost:9000;https://localhost:9001"

Auto port binding

Because the test-proxy honors the ASP.NET server configuration environment variable and CLI arguments, users can allow the test-proxy to automatically choose what port it binds to.

The trick is to provide override the default ASP.NET url to http://0.0.0.0:0;https://0.0.0.0:0 via ASPNETCORE_URLS env variable or -- --urls CLI argument.

Then, the user must listen to the stdout from the test-proxy process and parse out the following:

|>test-proxy start -- --urls "http://0.0.0.0:0;https://0.0.0.0:0"
Running proxy version is Azure.Sdk.Tools.TestProxy 20241209.1
...
[11:40:31] info: Microsoft.Hosting.Lifetime[14]
      Now listening on: http://0.0.0.0:55552
[11:40:31] info: Microsoft.Hosting.Lifetime[14]
      Now listening on: https://0.0.0.0:55553
...

To discover the mapped ports.

Environment Variables

The test-proxy is integrated with the following environment variables.

Variable Usage
TEST_PROXY_FOLDER if command-line argument storage-location is not provided when invoking the proxy, this environment variable is also checked for a valid directory to use as test-proxy context.
Logging__LogLevel__Default Defaults to Information. Possible valid values are

Debug, Information, Warning, Error, Critical.

Do not set for .NET test runs as it would cause the tests themselves to start emitting logs.
Logging__LogLevel__Azure.Sdk.Tools.TestProxy Set to Debug to see request level logs emitted by the Test Proxy.

How do I use the test-proxy to get a recording?

Before we begin a technical explanation, lets first discuss the general use of the test-proxy.

  1. User starts test-proxy, default url the proxy will be available on is http://localhost:5000.
  2. User initiates a POST to the proxy url http://localhost:5000/Record/Start with a body (which we will describe later) describing what we are recording.
  3. User sends requests through the test-proxy by modifying their actual test requests as described below.
  4. User initates a POST to the proxy url http://localhost:5000/Record/Stop.
  5. Repeat steps 2 through 4 as necessary for each test.

There is a lot of detail omitted from the above, but this is the general process.

The test-proxy is really meant to be integrated with a test framework. One that allows steps 2 and 4 to be handled by setup and teardown of individual record/playback sessions as well as updating the test requests so that they flow to the test-proxy instead of their actual target.

Please reference the clients under sample-clients/ folder for examples of what the process of modifying the requests actually looks like.

Installation and initial run

Reference the Installation section of this readme. SSL Configuration is discussed below in SSL Support.

A couple notes before running the test-proxy:

  • Reference command line arguments and understand the options.
  • The test-proxy runs in a "context" which is just a directory on your disk. When initializing a recording, all paths should be relative to this context directory.

Where will my recordings end up?

Recall in the previous section that the test-proxy started in a context. Again, this is just a directory on your disk. For the azure-sdk repositories, the test-proxy is normally started in the root of the repo. All recording files are then referenced relative from the root of that repo.

For example, let's invoke the test-proxy:

test-proxy --storage-location "C:/repo/sdk-for-net/"

We start a test run by POST-ing to either /Record/Start or /Playback/Start. Within the body of the post request we provide a file location within JSON body key x-recording-file.

Note: The test-proxy supports external storage of recordings. In this context there is an additional body key x-recording-assets-file that must be provided. Read more about this in the asset-sync documentation..

This key is used in combination with the context directory to either store or load an existing recording.

For example, lets say that:

  • The test-proxy has been started in directory C:/repo/sdk-for-net/.
  • The user POSTS to /Record/Start
  • The key passed in x-recording-file is a valid path: sdk/anomalydetector/Azure.AI.AnomalyDetector/tests/SessionRecords/GetresultForChangePoint.json

The test-proxy effectively joins the context directory and the incoming path to come up with a final location on disk.

context = C:/repo/sdk-for-net/
key = sdk/anomalydetector/Azure.AI.AnomalyDetector/tests/SessionRecords/GetresultForChangePoint.json
final path = C:/repo/sdk-for-net/sdk/anomalydetector/Azure.AI.AnomalyDetector/tests/SessionRecords/GetresultForChangePoint.json

When the user POSTS to /Record/Stop the recording will be written to the file as described directly above.

During a playback start, the value for x-recording-file is used to load an existing recording into memory and serve requests from it!

Please note that if a absolute path is presented in header x-recording-file. The test-proxy will write directly to that file, wherever it is. If the parent folders do not exist, they will be created at run-time during the write operation.

Start the test run

To start an individual test recording or playback, users will POST to a route on the running test-proxy. In real-world cases, this initial POST should be taken care of by the setup function of an individual test.

// Targeted URI: https://localhost:5001/record/start
// request body
{
    "x-recording-file": "<path-to-test>/<testfile>.<testname>"
}
// if the x-recording-file doesn't end with ".json", that will be appended prior to writing.

You will receive a recordingId in the reponse under header x-recording-id. This value should be included under header x-recording-id in all further requests.

For playback, you will receive the location of the recording file (if it is on disk) in the header x-base64-recording-file-location. This will be a base64 encoded string, as it can contain characters that are not allowed in header values.

Run your tests

The implicit assumption about this proxy is that you as a dev have some way to reroute your existing requests (with some header additions) to the test-proxy.

The implementation is language specific, but what you want to do is:

  1. Prevent each request originating from the test test from hitting their original endpoint.
  2. Make the following changes to every outgoing request
    1. Place original request scheme + hostname in header x-recording-upstream-base-uri
      • Example header setting: x-recording-upstream-base-uri: "https://fakeazsdktestaccount.table.core.windows.net"
    2. Replace request scheme:hostname with Proxy Server scheme:hostname. (currently https://localhost:5001 or http://localhost:5000)
      • Example transformation: https://fakeazsdktestaccount.table.core.windows.net/Tables -> http://localhost:5001/Tables.
    3. Add header "x-recording-id": <x-recording-id> from startup step
    4. Add header "x-recording-mode": "record"
  3. As each request hits the test proxy (due to the fact the target hostname has been updated), the test-proxy will invoke the requests and store their results prior to returning to the test code.
    • The request/response results are saved into memory based on the header x-recording-id. Forgetting this means your recordings will always be empty!
    • Recordings are saved after STOPPING your test.

A custom transport or request policy are both examples of implementing the above logic.

When finished running test

After your test has finished and there are no additional requests to be recorded, a test must be stopped to save the recording to disk.

POST to the proxy server:

// Targeted URI: https://localhost:5001/record/start

// header dictionary
{
    "x-recording-id": "<x-recording-id>",
    "Content-Type": "application/json"
}

// optional body storing VARIABLE values. See section below for additional detail.
{
    "key1": "value1",
    "key2": "value2"
}

The Content-Type must be set ONLY if a body with values is also sent. Read section directly below storing variables for a description.

This will finalize your recording by:

  • Removing from active sessions.
  • Applying session/recording sanitizers.
  • Saving to disk.

In test frameworks integrating with the test-proxy, this function should be invoked in the teardown function or the language equivalent.

Storing variables

In the above example notice that there is an optional body. This is extremely useful when your tests have an element of "randomness" to them. A great example of non-secret randomness would be a tablename provided during tables storage tests. It is extremely common for a given azure-sdk test framework to generate a name like u324bca. Unfortunately, randomness can lead to difficulties during playback. The URI, headers, and body of a request must match exactly with a recording entry.

Given that recordings are not traditionally accessible to the client code, there is no way to "retrieve" what those random non-secret values WERE. Without that capability, one must sanitize everything that could be possibly random.

An alternative is this variable concept. During a final POST to /Record/Stop, set the Content-Type header and make the body of the request a simple JSON map. The test-proxy will pass back these values in the body of /Playback/Start.

Customizing what gets recorded

Some tests send large request bodies that are not meaningful and should not be stored in the session records. In order to disable storing the request body for a specific request, add the request header "x-recording-skip" and set the value to "request-body". This header can also be used to skip an entire request/response pair from being included in the recording - this is useful for cleanup code that you might have as part of your test. To skip the request/response pair, set the "x-recording-skip" header value to "request-response". Note that the "x-recording-skip" should only be specified when in Record mode. As a result, any request that would use the "request-response" value when in Record mode should not be sent when in Playback mode. For requests that use "request-body" in Record mode, you should either null out the body of the request before sending to the test proxy when in Playback mode, or you can set a CustomDefaultMatcher with compareBodies = false.

One can also prevent update of a recording file on disk by providing a header of value x-recording-skip: "request-response" along with your POST to /Record/Stop.

How do I use the test-proxy to play a recording back?

Start playback

Extremely similar to recording start.

POST to the proxy server:

// Targeted URI: https://localhost:5001/playback/start
// request body
{
    "x-recording-file": "<path-to-test>/recordings/<testfile>.<testname>"
}

As with /record/start, check response header x-recording-id to get the recording-id for usage later on.

Optional variables returned

In the case where a user set variables in /Record/Stop, those values will also be returned in the body of the result from /Playback/Start. There is no manipulation of these values, whatever is saved during /Record/Stop should be identical to what is available from /Playback/Start.

As one could guess based on the routes, these variables are saved and returned on a per-recording basis.

During your test run

The configuration here is practically identical to what we lay out in Run your tests. The only difference is that x-recording-mode header should be set to playback.

Stop playback

This really only allows the server to free up a few bits, but:

POST to the proxy server:

// targeted URI: https://localhost:5001/playback/stop
// header dictionary
{
    "x-recording-id": "<x-recording-id>"
}

An important note about perf testing

If a user does not provide a fileId via body key x-recording-file, the recording will be saved in-memory only. If a recording is saved into memory, the only way to retrieve it is to access the playback by passing along the original recordingId that you recorded it with.

Start the recording without a x-recording-file body value.

// targeted URI: https://localhost:5001/record/start
// the request body will be NULL / unset

The POST will return a valid recordingId value which we will call X.

To load this recording for playback...

// Targeted URI https://localhost:5001/playback/start
// header dictionary
{
    "x-recording-id": "X"
}

When attempting to use in-memory recording, you cannot submit empty body {} to /Record/Start. Doing so will result in an error, as it is expecting a recording file if a body provided.

See example implementations

Of course, feel free to check any of the examples to see actual test code and invocations.

Session and Test Level Transforms, Sanitizers, and Matchers

The test-proxy is a record/playback solution. As a result, there a few concepts that devs will likely recognize from other record/playback solutions:

  • A Sanitizer is used to remove sensitive information prior to storage. Sanitizers are applied...
    • To the request matching against the recording entries during playback mode.
    • To the recording entries loaded from disk when starting a playback session.
    • To the recording entries before they are saved to disk when stopping a record session.
  • Matchers are used to retrieve a request/response pair from a previous recording. By default, it functions by comparing URI, Headers, and Body. As of now, only a single matcher can be used when retrieving an entry during playback.
  • A Transform is used when a user needs to "transform" a matched recording response with some value from the incoming request. This action is specific to playback mode. For instance, the test-proxy has two default transforms:
    • x-ms-client-id is copied from request and applied to response prior to return.
    • x-ms-client-request-id is copied from request and applied to response prior to return.

Default sets of matcher, transforms, and sanitizers are applied during recording and playback. These default settings are all set at the session level. Customization is allowed for these default sets by accessing the Admin controller.

When creating a custom sanitizer/matcher/transform, a single constructor must be provided.

This is due to the fact that if there are arguments to the constructor, the body attributes will be mapped by name.

Add Sanitizer

Add a simple Uri Sanitizer that leverages lookahead to ensure it's not overly aggressive:

// POST TO Targeted URI: <proxyURL>/Admin/AddSanitizer
// header dictionary
{
    "x-abstraction-identifier": "UriRegexSanitizer"
}
// request body
{
    "value": "fakeaccount",
    "regex": "[a-z]+(?=\\.(?:table|blob|queue)\\.core\\.windows\\.net)"
}

Add a more expansive Header sanitizer that uses a target group instead of filtering by lookahead:

// POST to URI <proxyURL>/Admin/AddSanitizer
// headers
{
    "x-abstraction-identifier": "HeaderRegexSanitizer"
}
// request body
{
    "key": "Location",
    "value": "fakeaccount",
    "regex": "https\\:\\/\\/(?<account>[a-z]+)\\.(?:table|blob|queue)\\.core\\.windows\\.net",
    "groupForReplace": "account"
}

Note: Regex strings are being passed as a member of a JSON object. As visible in the above example, escape characters must be double escaped. \n -> \\n. \\ -> \\\\. Read the "string" description on json.org for more detail.

A note about where sanitizers apply

Each sanitizer is optionally prefaced with the specific part of the request/response pair that it applies to. These prefixes are

  • Uri
  • Header
  • Body

A sanitizer that does not include this prefix is something different, and probably applies at the session level instead on an individual request/response pair.

Passing sanitizers in bulk

In some cases, users need to register a lot (10+) of sanitizers. In this case, going back and forth with the proxy server individually is not very efficient. To ameliorate this, the proxy honors multiple sanitizers in the same request if the user utilizes /Admin/AddSanitizers.

// POST to URI <proxyURL>/Admin/AddSanitizers
// note the request body is simply an array of objects
[
    {
        "Name": "GeneralRegexSanitizer",
        "Body": {
            "regex": "[a-zA-Z]?",
            "value": "hello_there",
            "condition": {
                "UriRegex": ".+/Tables"
            }
        }
    },
    {
        "Name": "HeaderRegexSanitizer",
        "Body": { // <-- the contents of this property mirror what would be passed in the request body for individual AddSanitizer()
            "key": "Location",
            "value": "fakeaccount",
            "regex": "https\\:\\/\\/(?<account>[a-z]+)\\.(?:table|blob|queue)\\.core\\.windows\\.net",
            "groupForReplace": "account"
        }
    }
]

Knowing what was added

When AddSanitizer or AddSanitizers is called, check the response body for an array containing the ids of sanitizers that have been registered.

Example response body:

// POSTS to Admin/AddSanitizer has individual result
{
  "Sanitizer": "3"
}
// POSTS to Admin/AddSanitizers has multiple results
{
  "Sanitizers": ["3", "4", "9"]
}

Removing a sanitizer

Following #8120, sanitizers were given identifiers so that they can be removed.

  • The default session sanitizers list can be found in code here.
  • Visiting http://localhost:5000/Info/Active on your browser when the proxy is running on your machine will present you with an easy summary of these available sanitizers as well.

When a recording or playback session is begun (Playback/Start or Record/Start), all the current session sanitizers are applied to the recording. Session sanitizers added after record or playback has begun do not apply to the prior-started session.

To remove a session-level sanitizer, one only must...

// request method = `POST`
// request URI = <proxyUrl>/Admin/RemoveSanitizers
// request headers =
{
  "Content-Type": "application/json",
  "Content-Length": 36
}
// request body =
{
  Sanitizers: ["AZSDK002", "AZSDK003"]
}

On successful request, users will receive a list of the removed identifiers.

// response body
{
  Removed: ["ID1"]
}

To remove a sanitizer from a specific recording...

// request method = `POST`
// request URI = <proxyUrl>/Admin/RemoveSanitizers
// request headers =
{
  "Content-Type": "application/json",
  "Content-Length": 36,
  "x-recording-id": "your-recording-id-here"
}
// request body =
{
  Sanitizers: ["AZSDK002", "AZSDK003"]
}

Set a Matcher

Setting a matcher is just like adding a sanitizer. Set the x-abstraction-identifier value to the name of the matcher you want to instantiate, and provide the proper constructor arguments in the body! Check Info/Available/ for available matchers.

// POST to URI <proxyURL>/Admin/SetMatcher
// headers
{
    "x-abstraction-identifier": "BodilessMatcher"
}
// request body is just empty JSON for a BodilessMatcher
{}

The Custom Default Matcher

The default matcher for test-proxy is usable in most situations. Sanitizers can be used to clear non-available information prior to saving to disk, and playback tests should ensure that they match the expected values in the recordings. Sanization can usually eliminate the need for custom matching, but in some circumstances this is just not feasible.

If a dev must only change a couple elements of the default matcher, the CustomDefault matcher is the way to go!

// POST to URI <proxyURL>/Admin/SetMatcher
// headers
{
    "x-abstraction-identifier": "CustomDefaultMatcher"
}
// request body
{
    // Should the body value be compared during lookup operations?
    "compareBodies": true,
    // A comma separated list of additional headers that should be excluded during matching. "Excluded" headers are entirely ignored.
    "excludedHeaders": "traceparent",
    // A comma separated list of additional headers that should be ignored during matching. Any headers that are "ignored" will not do value comparison when matching.
    // The "presence" of the headers will still be checked however.
    "ignoredHeaders": "User-Agent, Origin, Content-Length",
    // By default, the test-proxy does not sort query params before matching. Setting true will sort query params alphabetically before comparing URI.
    "ignoredQueryOrdering": false,
    // A comma separated list of query parameterse that should be ignored during matching.
    "ignoredQueryParameters" "token"
}

Add a Transform

Just like sanitizers and matchers, select an abstraction-id in header, provide a json body with any arguments/details.

// POST to URI <proxyURL>/Admin/AddTransform
// headers
{
    "x-abstraction-identifier": "ApiVersionTransform"
}
// empty request body for ApiVersionTransform
{}

This will add a transform that copies api-version header from request onto the matched response during playback.

For Sanitizers, Matchers, or Transforms in general

When invoked as basic requests to the Admin controller, these settings will be applied to all further requests and responses. Both Playback and Recording. Where applicable.

  • sanitizers are applied before the recording is saved to disk AND on an incoming request prior to the match operation.
  • A custom matcher can be set for a session or individual recording and is applied when retrieving an entry from a loaded recording. Only a single matcher can be honored when matching requests to recorded entries. Registering two matchers one after another will simply result in matching using the last matcher provided.
  • transforms are applied when returning a request during playback.

Currently, the configured set of transforms/playback/sanitizers are NOT propogated onto disk alongside the recording.

Viewing available/active Sanitizers, Matchers, and Transforms

Launch the test-proxy through your chosen method, then visit:

  • <proxyUrl>/Info/Available to see all available.
  • <proxyUrl>/Info/Active to see all currently active for all sessions.

Note that the constructor arguments that are documented must be present (where documented as such) in the body of the POST sent to the Admin Interface.

To see customizations on a specific recording

This only works while a session is available. A specific session is only available before it has been stopped. Once that happens it has been written to disk or evacuated from server memory.

Example flow

  • Start playback for "hello_world.json"
  • Receive a recordingId back
  • Place a breakpoint before the part of your code that calls /Record/Stop or Playback/Stop.
  • Visit the url <proxyUrl>/Info/Active?id=<your-recording-id>

Resetting active Sanitizers, Matchers, and Transforms

Given that the test-proxy offers the ability to set up customizations for an entire session or a single recording, it also must provide the ability to reset these settings without entirely restarting the server.

This is allowed through the use of the /Admin/Reset API. A reset operation "returns to default".

Reset the session

To reset a session to default customization, POST to Admin/Reset.

This API operates exclusively on the Session level if no recordingId is provided in the header. Any customizations on individual recordings are left untouched.

Reset for a specific recordingId

However, if a recordingId is provided in the header dictionary, the reset operation applies to only that individual recording.

// POST TO url: <proxyURL>/Admin/Reset
// header dictionary
{
    "x-recording-id": "<guid>"
}

The session level updates will remain unchanged.

Recording Options

The test-proxy offers further customization beyond that offered by sanitizers, matchers, and transforms. To access this additional functionality, one must POST to /Admin/SetRecordingOptions with a json body that contains the options.

// below is an object representing all valid inputs for SetRecordingOptions body
// POST to /Admin/SetRecordingOptions
{
   // boolean value accepted. string or raw.
   "HandleRedirects": "true/false"
   // setting context directory will change the "root" path that the test proxy uses when loading a recording
   "ContextDirectory": "<valid path on local disk>",
   // as yet unused. will allow to swap away from git-backed recording storage
   "AssetsStore": "<NullStore/GitStore>",
   // customizes some transport settings all or a single recording
   "Transport": {
      // any number of certificates will be allowed here
      "Certificates": [
        {
          "PemValue": "<string content>",
          "PemKey": "<string content>",
        }
      ],
      // used specifically so that an SSL connection presenting a non-standard certificate can still be validated
      "TLSValidationCert": "<public key portion of TLS cert>",
      // if not provided, the TLS Validation validation callbacks will be used for all targeted hosts
      // if provided, only requests to the hostname contained herein will be validated against the cert present in TLSValidationCert
      "TLSValidationCertHost": "<hostname for the targeted resource associated with TLS Cert>",
   }
}

Redirection Settings

The test-proxy follows redirects by default. That means that if the initial request sent by the test-proxy results in some 3XX redirect status, it will follow the redirect.

In certain cases, a user will want to utilize both behaviors in their tests. For instance, the javascript browser tests run with redirect enabled, however for their node versions, the redirect functionality is explicitly disabled. The test-proxy exposes this as a setting HandleRedirects within the settings object POST-ed to /Admin/SetRecordingOptions.

Example:

// POST to URI: https://localhost:5001/Admin/SetRecordingOptions
// body is a json dictionary, the value of HandleRedirects can be multiple representation of "true"
{
    "HandleRedirects": true
}
// ...or
{
    "HandleRedirects": 1
}

// to disable, it's just the opposite, with similar alternative support
{
    "HandleRedirects": false
}
// ...or
{
    "HandleRedirects": "false"
}

Providing your own Host header

In normal running, the test-proxy actually sets the Host header automatically. If one wants to provide a specific header that gets used during the request, the request must be accompanied by header:

"x-recording-upstream-host-header": "<host value>"

Testing

This project uses Xunit as the test framework. This is the most popular .NET test solution according to this twitter poll and it's also what the majority of the test projects in the Azure/azure-sdk-tools repo utilize as well.

SSL Support

The test-proxy server supports SSL, but due to its local-hosted nature, SSL validation will always be an issue without some manual intervention. Real SSL certificates are not available for localhost! As a result of this, if attempting to leverage SSL when recording, some additional setup will be required.

Within this repository there is a single certificate.

  • eng/common/testproxy/dotnet-devcert.pfx: generated on a Ubuntu distribution using openssl.

Troubleshooting

Visual studio

If you get the message dialog The project doesn't know how to run the profile Azure.Sdk.Tools.TestProxy, you can fix it by reviewing the next two things:

ASP.NET and web development

Run Visual Studio installer and make sure ASP.NET and web development is installed.

image

Then, confirm in the right panel that Development time IIS support is not checked:

image

Windows IIS

Add Internet Information Services to your Windows installation. Here is the list of features to enable:

image

Asset Sync (Retrieve External Test Recordings)

The test-proxy optionally offers integration with other git repositories for storing and retrieving recordings. This enables the proxy to work against repositories that do not emplace their test recordings directly alongside their test implementations.

For further reading about this feature, please refer to the asset-sync documentation folder.