Customizing Error Page

HTTP error response

In the NetScaler WAF you have the ability to configure a custom response when blocking traffic. This is valid for all 3 types of traffic - HTML, XML, JSON. For error handling you can use redirect URL or an error object. The error configuration consist of "error content" and response status code. We're going to discuss customization options for local error object, why we have them and should we configure them. This is configurable under "Security > Citrix Web App Firewall > Imports" and inside your AppFW Profile in Profile Settings:

Image

I would use Status Code "403 Forbidden", but you can configure any code you want.

You generally want to configure two error pages.

  • NON-PROD Error page

This error page will contain additional information regarding the blocked request and should be used only during the configuration phase of the WAF. This is because the page will contain the category and the exact rule blocking the request. Such information should not be provided to the threat actors.

  • PROD Error page

This error page will be with limited information like general blocking message and service desk URL.

You can always choose not to provide any error in response to a blocked traffic. This is probably best from protection point of view, because block feedback (error page) is still a feedback. "No feedback" does not disclose any information and hides the operational ways of working. Same difference as to "DROP" and "RESET" (RST packet is still a valuable information). You have to make a trade off, if you want error page for reporting false-positives.

Error Page

Below you can find the error page provided by Citrix. Original citrix documentation about this here

<html>
<head>
<title>Application Firewall Block Page</title>
</head>
<body>
<h1><B>Your request has been blocked by a security policy<B><BR></H1>
<H3>Access has been blocked - if you feel this is in error, please contact the site
administrators quoting the following: </H3> <UL>
<li>NS Transaction ID: ${NS_TRANSACTION_ID}
<li>AppFW Session ID: ${NS_APPFW_SESSION_ID}
<li>Violation Category: ${NS_APPFW_VIOLATION_CATEGORY}
<li>Violation Details: ${NS_APPFW_VIOLATION_LOG} </UL>
</body>
</html>

This response contains the additional information to identify the rule quickly and configure relaxation rule.

  • NS Transaction ID
  • AppFW Session ID
  • Violation Category
  • Violation Details

Depending on SYSLOG config format this would look like this:

Image

You would notice the Category "APPFW_DENYURL" and the Violation Details where the actual rule pattern is displayed "Disallow Deny URL for rule pattern="^.*generic.html$"

Customization

All this content is saved to a file in /var/tmp directory where the rest of your configured HTML responder pages live.

The NetScaler does not provide a build-in way for customization of this error page like including a logo and it doesn't need to as we can quite easily do it. Any formatiing rely on your HTML CSS Style config in that error page.

For our customization purposes we can embedd an image in the HTML document. Images can be emdedded into HTML and JSON files under base64 encoding. Base64 encoding makes file sizes roughly 33% larger than their original binary representations, which means more data down the wire. This isn't really a concern as we're using it only for a simple error page.

Go to base64-image.de and convert your image to base64. After that simply place it in your error page in the img tag.

<img src="" />

We can add simple formating, styling and embedd the image. Nothing too fancy it just feels better.

Image

I've inlcuded a Service Desk button for reporting the issue. Any ticket creation autmation could be included there.

Image

You can see that I have an additional field on the feedback part - Cookie. In this case I've configured the NetScaler's authentication cookie to be logged. The WAF protected resource is also protected by NetScaler AAA. In the past this was perfectly working, for some reason currently it does not. Intrestingly this has been an issue in build 11.0 for NS_APPFW_SESSION_ID:

From Release Notes: When the application firewall redirects a blocked request to a customized error page, the NS_APPFW_SESSION_ID;variable on the error page might not display the session ID accurately. If the request does not contain a session cookie, the variable might display a hyphen (-) instead of the session ID. [From Build 64.34] [# 599052]

Below you can find my error page:

<!DOCTYPE html>
<html class="">

<head>
	<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
		
	<title>alekaf.com</title>
	<meta name="ROBOTS" content="NOINDEX, NOFOLLOW">
		
	<style>			.error-msg,.error-title{display:block;text-align:center}.error-container{left:auto;width:90%;padding:5%;color:#333;margin:0;background-color:#fff}.error-title{font-size:30px;color:#555;margin:0 0 50px}.error-msg{line-height:1.5em;margin:0}@media all and (min-width:750px){.error-msg{text-align:left}}@media all and (min-width:900px){.error-msg,.error-title{text-align:left}.error-container{position:relative;width:840px;margin-left:50%;left:-440px;top:50%;margin-top:2%;padding:20px;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px;-webkit-box-shadow:0 0 5px 0 rgba(0,0,0,.18);-moz-box-shadow:0 0 5px 0 rgba(0,0,0,.18);box-shadow:0 0 5px 0 rgba(0,0,0,.18)}body{background-color:#f1f1f1}.error-msg{margin-bottom:30px;margin-left:40px;margin-right:40px}}.service-desk{text-align:center;padding-top:50px;}.company-logo{text-align:center;padding-top:50px;}.exclam{background-color:#113797;}body{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;font-family:"Trebuchet MS",arial,sans-serif;margin:0;font-family:'Poppins', sans-serif;display: flex;align-items:center;justify-content: center;color:#333C48;}.alert{display: flex;align-items:center;padding: 0.55rem 0.65rem 0.55rem 0.75rem;border-radius:1rem;min-width:400px;justify-content: space-between;margin-bottom: 2rem;box-shadow:0px 3.2px 13.8px rgba(0, 0, 0, 0.02),0px 7.6px 33.3px rgba(0, 0, 0, 0.028),0px 14.4px 62.6px rgba(0, 0, 0, 0.035),0px 25.7px 111.7px rgba(0, 0, 0, 0.042),0px 48px 208.9px rgba(0, 0, 0, 0.05),0px 115px 500px rgba(0, 0, 0, 0.07)}.content{display: flex;  align-items:center;}.icon{padding: 0.5rem;margin-right: 1rem;border-radius:39% 61% 42% 58% / 50% 51% 49% 50%;box-shadow:0px 3.2px 13.8px rgba(0, 0, 0, 0.02),0px 7.6px 33.3px rgba(0, 0, 0, 0.028),0px 14.4px 62.6px rgba(0, 0, 0, 0.035), 0px 25.7px 111.7px rgba(0, 0, 0, 0.042),0px 48px 208.9px rgba(0, 0, 0, 0.05),0px 115px 500px rgba(0, 0, 0, 0.07)}.danger{  background-color: rgba(236, 77, 43, 0.2);border:2px solid #EF9400;}.danger .icon{background-color: #EC4D2B;}table {width:100%;border-collapse:separate;border:solid #EF9400 2px; border-radius:6px;}td {	background-color: #ec4d2b33;border-top: none;border-left: none;}.button-19 { appearance: button; background-color: #ec4d2b33; border: solid transparent; border-radius: 16px; border-width: 0 0 4px;  box-sizing: border-box;color: #FFFFFF;cursor: pointer;display: inline-block;font-family: din-round,sans-serif;font-size: 15px;font-weight: 700;letter-spacing: .8px;line-height: 20px; margin: 0; outline: none;overflow: visible; padding: 13px 16px;text-align: center;text-transform: uppercase;touch-action: manipulation;transform: translateZ(0);transition: filter .2s;user-select: none;  -webkit-user-select: none;vertical-align: middle;white-space: nowrap;width: 25%;}.button-19:after { background-clip: padding-box; background-color: #E9967A; border: solid transparent;  border-radius: 16px; border-width: 0 0 4px; bottom: -4px; content: ""; left: 0; position: absolute; right: 0; top: 0; z-index: -1;}.button-19:main,.button-19:focus { user-select: auto;}.button-19:hover:not(:disabled) {filter: brightness(1.1); -webkit-filter: brightness(1.1);}.button-19:disabled { cursor: auto;}
	</style>
	<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> <meta http-equiv="cleartype" content="on"> <meta name="HandheldFriendly" content="True"> <meta name="MobileOptimized" content="width"> <meta name="viewport" content="width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=2.0">
	
<script>


</script>

	
</head>

<body>

<div class="error-container">
<div class="container">
  <div class="danger alert">
    <div class="content">
      <div class="icon">
        <svg height="50" viewBox="0 0 512 512" width="50" xmlns="http://www.w3.org/2000/svg"><path fill="#fff" d="M449.07,399.08,278.64,82.58c-12.08-22.44-44.26-22.44-56.35,0L51.87,399.08A32,32,0,0,0,80,446.25H420.89A32,32,0,0,0,449.07,399.08Zm-198.6-1.83a20,20,0,1,1,20-20A20,20,0,0,1,250.47,397.25ZM272.19,196.1l-5.74,122a16,16,0,0,1-32,0l-5.74-121.95v0a21.73,21.73,0,0,1,21.5-22.69h.21a21.74,21.74,0,0,1,21.73,22.7Z"/></svg>
    </div>
      <p>Your request has been blocked by a security policy!</p>
    </div>
  </div>
</div>
			
<span class="error-msg">If you feel this is in error, please contact the site administrators quoting the following:</span>

<div>
	<table>
        <tbody>
          <tr >
            <td>
              ID
            </td>
            <td>
              ${NS_TRANSACTION_ID}
            </td>
          </tr>
          <tr >
            <td>
              Session
            </td>
            <td>
              ${NS_APPFW_SESSION_ID}
            </td>
          </tr>
          <tr >
            <td>
              Cookie
            </td>
            <td>
              ${COOKIE("nsc_tmas")}
            </td>
          </tr>
          <tr >
            <td>
              Category
            </td>
            <td>
              ${NS_APPFW_VIOLATION_CATEGORY}
            </td>
          </tr>
          <tr >
            <td>
              Violation
            </td>
            <td>
              ${NS_APPFW_VIOLATION_LOG}
            </td>
          </tr>
        </tbody>
      </table>
</div>

<div class="service-desk">
<a href="https://servicedesk.com" class="button-19" role="button">Report</a>
</div>

<div class="company-logo">
<img src="" />
</div>

</body>

</html>

Even if you decide the provide your clients with an error message. One issue remains for non-browser API clients. What if legit request is blocked? How to report this with API clients and how long it will take you to detect the false-positive block?

I hope this has been informative to you and thank you for reading!