Web Application Firewall exclusion lists
The Azure Application Gateway Web Application Firewall (WAF) provides protection for web applications. This article describes the configuration for WAF exclusion lists. These settings are located in the WAF policy associated to your Application Gateway. To learn more about WAF policies, see Azure Web Application Firewall on Azure Application Gateway and Create Web Application Firewall policies for Application Gateway.
Sometimes WAF might block a request that you want to allow for your application. WAF exclusion lists allow you to omit certain request attributes from a WAF evaluation. The rest of the request is evaluated as normal.
For example, Active Directory inserts tokens that are used for authentication. When used in a request header, these tokens can contain special characters that might trigger a false positive detection from the WAF rules. By adding the header to an exclusion list, you can configure WAF to ignore the header, but WAF still evaluates the rest of the request.
You can configure exclusions to apply when specific WAF rules are evaluated, or to apply globally to the evaluation of all WAF rules. Exclusion rules apply to your whole web application.
Identify request attributes to exclude
When you configure a WAF exclusion, you must specify the attributes of the request that should be excluded from the WAF evaluation. You can configure a WAF exclusion for the following request attributes:
- Request headers
- Request cookies
- Request attribute name (args) can be added as an exclusion element, such as:
- Form field name
- JSON entity
- URL query string args
You can specify an exact request header, body, cookie, or query string attribute match. Or, you can specify partial matches. Use the following operators to configure the exclusion:
- Equals: This operator is used for an exact match. As an example, for selecting a header named bearerToken, use the equals operator with the selector set as bearerToken.
- Starts with: This operator matches all fields that start with the specified selector value.
- Ends with: This operator matches all request fields that end with the specified selector value.
- Contains: This operator matches all request fields that contain the specified selector value.
- Equals any: This operator matches all request fields. * is the selector value. For example, you would use this operator when you don't know the exact values for a given match variable but want to make sure that the request traffic still gets excluded from rules evaluation.
When processing exclusions the WAF engine performs a case sensitive/insensitive match based on the following table. Additionally, regular expressions aren't allowed as selectors and XML request bodies aren't supported.
Request Body Part | CRS 3.1 and Earlier | CRS 3.2 and Later |
---|---|---|
Header* | Case Insensitive | Case Insensitive |
Cookie* | Case Insensitive | Case Sensitive |
Query String* | Case Insensitive | Case Sensitive |
URL-Encoded Body | Case Insensitive | Case Sensitive |
JSON Body | Case Insensitive | Case Sensitive |
XML Body | Not Supported | Not Supported |
Multipart Body | Case Insensitive | Case Sensitive |
*Depending on your application, the names, and values, of your headers, cookies and query args can be case sensitive or insensitive.
Note
For more information and troubleshooting help, see WAF troubleshooting.
Request attributes by keys and values
When you configure an exclusion, you need to determine whether you want to exclude the key or the value from WAF evaluation.
For example, suppose your requests include this header:
My-Header: 1=1
The value of the header (1=1
) might be detected as an attack by the WAF. But if you know this is a legitimate value for your scenario, you can configure an exclusion for the value of the header. To do so, you use the RequestHeaderValues match variable, the operator contains, and the selector (My-Header
). This configuration stops evaluation of all values for the header My-Header
.
Note
Request attributes by key and values are only available in CRS 3.2 or newer and Bot Manager 1.0 or newer.
Request attributes by names work the same way as request attributes by values, and are included for backward compatibility with CRS 3.1 and earlier versions. We recommend you use request attributes by values instead of attributes by names. For example, use RequestHeaderValues instead of RequestHeaderNames.
In contrast, if your WAF detects the header's name (My-Header
) as an attack, you could configure an exclusion for the header key by using the RequestHeaderKeys request attribute. The RequestHeaderKeys attribute is only available in CRS 3.2 or newer and Bot Manager 1.0 or newer.
Request attribute examples
The following table shows some examples of how you might structure your exclusion for a given match variable.
Attribute to Exclude | matchVariable | selectorMatchOperator | Example selector | Example request | What gets excluded |
---|---|---|---|---|---|
Query string | RequestArgKeys | Equals | /etc/passwd |
Uri: http://localhost:8080/?/etc/passwd=test |
/etc/passwd |
Query string | RequestArgKeys | EqualsAny | N/A | Uri: http://localhost:8080/?/etc/passwd=test&.htaccess=test2 |
/etc/passwd and .htaccess |
Query string | RequestArgNames | Equals | text |
Uri: http://localhost:8080/?text=/etc/passwd |
/etc/passwd |
Query string | RequestArgNames | EqualsAny | N/A | Uri: http://localhost:8080/?text=/etc/passwd&text2=.cshrc |
/etc/passwd and .cshrc |
Query string | RequestArgValues | Equals | text |
Uri: http://localhost:8080/?text=/etc/passwd |
/etc/passwd |
Query string | RequestArgValues | EqualsAny | N/A | Uri: http://localhost:8080/?text=/etc/passwd&text2=.cshrc |
/etc/passwd and .cshrc |
Request body | RequestArgKeys | Contains | sleep |
Request body: {"sleep(5)": "test"} |
sleep(5) |
Request body | RequestArgKeys | EqualsAny | N/A | Request body: {".zshrc": "value", "sleep(5)":"value2"} |
.zshrc and sleep(5) |
Request body | RequestArgNames | Equals | test |
Request body: {"test": ".zshrc"} |
.zshrc |
Request body | RequestArgNames | EqualsAny | N/A | Request body: {"key1": ".zshrc", "key2":"sleep(5)"} |
.zshrc and sleep(5) |
Request body | RequestArgValues | Equals | test |
Request body: {"test": ".zshrc"} |
.zshrc |
Request body | RequestArgValues | EqualsAny | N/A | Request body: {"key1": ".zshrc", "key2":"sleep(5)"} |
.zshrc and sleep(5) |
Header | RequestHeaderKeys | Equals | X-Scanner |
Header: {"X-Scanner": "test"} |
X-scanner |
Header | RequestHeaderKeys | EqualsAny | N/A | Header: {"X-Scanner": "test", "x-ratproxy-loop": "value"} |
X-Scanner and x-ratproxy-loop |
Header | RequestHeaderNames | Equals | head1 |
Header: {"head1": "X-Scanner"} |
X-scanner |
Header | RequestHeaderNames | EqualsAny | N/A | Header: {"head1": "myvar=1234", "User-Agent": "(hydra)"} |
myvar=1234 and (hydra) |
Header | RequestHeaderValues | Equals | head1 |
Header: {"head1": "X-Scanner"} |
X-scanner |
Header | RequestHeaderValues | EqualsAny | N/A | Header: {"head1": "myvar=1234", "User-Agent": "(hydra)"} |
myvar=1234 and (hydra) |
Cookie | RequestCookieKeys | Contains | /etc/passwd |
Header: {"Cookie": "/etc/passwdtest=hello1"} |
/etc/passwdtest |
Cookie | RequestCookieKeys | EqualsAny | N/A | Header: {"Cookie": "/etc/passwdtest=hello1", "Cookie": ".htaccess=test1"} |
/etc/passwdtest and .htaccess |
Cookie | RequestCookieNames | Equals | arg1 |
Header: {"Cookie": "arg1=/etc/passwd"} |
/etc/passwd |
Cookie | RequestCookieNames | EqualsAny | N/A | Header: {"Cookie": "arg1=/etc/passwd", "Cookie": "arg1=.cshrc"} |
/etc/passwd and .cshrc |
Cookie | RequestCookieValues | Equals | arg1 |
Header: {"Cookie": "arg1=/etc/passwd"} |
/etc/passwd |
Cookie | RequestCookieValues | EqualsAny | N/A | Header: {"Cookie": "arg1=/etc/passwd", "Cookie": "arg1=.cshrc"} |
/etc/passwd and .cshrc |
Note
If you create an exclusion using the selectorMatchOperator EqualsAny
, anything you put in the selector field gets converted to "*" by the backend when the exclusion is created.
Exclusion scopes
Exclusions can be configured to apply to a specific set of WAF rules, to rulesets, or globally across all rules.
Tip
It's a good practice to make exclusions as narrow and specific as possible, to avoid accidentally leaving room for attackers to exploit your system. When you need to add an exclusion rule, use per-rule exclusions wherever possible.
Per-rule exclusions
You can configure an exclusion for a specific rule, group of rules, or rule set. You must specify the rule or rules that the exclusion applies to. You also need to specify the request attribute that should be excluded from the WAF evaluation. To exclude a complete group of rules, only provide the ruleGroupName
parameter, the rules
parameter is only useful when you want to limit the exclusion to specific rules of a group.
Per-rule exclusions are available when you use the OWASP (CRS) ruleset version 3.2 or later or Bot Manager ruleset version 1.0 or later.
Example
Suppose you want the WAF to ignore the value of the User-Agent
request header. The User-Agent
header contains a characteristic string that allows the network protocol peers to identify the application type, operating system, software vendor, or software version of the requesting software user agent. For more information, see User-Agent.
There can be any number of reasons to disable evaluating this header. There could be a string that the WAF detects and assumes it’s malicious. For example, the User-Agent
header might include the classic SQL injection attack x=x
in a string. In some cases, this can be legitimate traffic. So you might need to exclude this header from WAF evaluation.
You can use the following approaches to exclude the User-Agent
header from evaluation by all of the SQL injection rules:
To configure a per-rule exclusion by using the Azure portal, follow these steps:
Navigate to the WAF policy, and select Managed rules.
Select Add exclusions.
In Applies to, select the CRS ruleset to apply the exclusion to, such as OWASP_3.2.
Select Add rules, and select the rules you want to apply exclusions to.
Configure the match variable, operator, and selector. Then select Save.
You can configure multiple exclusions.
You can also exclude the User-Agent
header from evaluation just by rule 942270:
Follow the steps described in the preceding example, and select rule 942270 in step 4.
Global exclusions
You can configure an exclusion to apply across all WAF rules.
Example
Suppose you want to exclude the value in the user parameter that is passed in the request via the URL. For example, say it’s common in your environment for the user
query string argument to contain a string that the WAF views as malicious content, so it blocks it. You can exclude all query string arguments where the name begins with the word user
, so that the WAF doesn't evaluate the field's value.
The following example shows how you can exclude the user
query string argument from evaluation:
To configure a global exclusion by using the Azure portal, follow these steps:
Navigate to the WAF policy, and select Managed rules.
Select Add exclusions.
In Applies to, select Global
Configure the match variable, operator, and selector. Then select Save.
You can configure multiple exclusions.
So if the URL http://www.contoso.com/?user%3c%3e=joe
is scanned by the WAF, it doesn't evaluate the string joe, but it still evaluates the parameter name user%3c%3e.
Next steps
- After you configure your WAF settings, you can learn how to view your WAF logs. For more information, see Application Gateway diagnostics.
- Learn more about Azure network security